Tests: Drivers: TimerEvent: Add tests

pull/5046/head
Filip Jagodzinski 2017-09-05 15:54:21 +02:00
parent 7f73785b9a
commit 44c4583f2a
1 changed files with 219 additions and 0 deletions

View File

@ -0,0 +1,219 @@
/* mbed Microcontroller Library
* Copyright (c) 2017 ARM Limited
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "mbed.h"
#include "greentea-client/test_env.h"
#include "unity.h"
#include "utest.h"
#include "drivers/TimerEvent.h"
#include "hal/ticker_api.h"
#include "rtos.h"
using namespace utest::v1;
#define TEST_DELAY_US 50000ULL
class TestTimerEvent: public TimerEvent {
private:
Semaphore sem;
virtual void handler() {
sem.release();
}
public:
TestTimerEvent() :
TimerEvent(), sem(0, 1) {
}
TestTimerEvent(const ticker_data_t *data) :
TimerEvent(data), sem(0, 1) {
}
virtual ~TestTimerEvent() {
}
// Make these methods publicly accessible
using TimerEvent::insert;
using TimerEvent::insert_absolute;
using TimerEvent::remove;
int32_t sem_wait(uint32_t millisec) {
return sem.wait(millisec);
}
};
class TestTimerEventRelative: public TestTimerEvent {
public:
static const int32_t SEM_SLOTS_AFTER_PAST_TS_INSERTED = 0;
TestTimerEventRelative() :
TestTimerEvent() {
}
TestTimerEventRelative(const ticker_data_t *data) :
TestTimerEvent(data) {
}
// Set relative timestamp of internal event to present_time + ts
void set_future_timestamp(timestamp_t ts) {
insert(::ticker_read(_ticker_data) + ts);
}
void set_past_timestamp(void) {
insert(::ticker_read(_ticker_data) - 1UL);
}
};
class TestTimerEventAbsolute: public TestTimerEvent {
public:
static const int32_t SEM_SLOTS_AFTER_PAST_TS_INSERTED = 1;
TestTimerEventAbsolute() :
TestTimerEvent() {
}
TestTimerEventAbsolute(const ticker_data_t *data) :
TestTimerEvent(data) {
}
// Set absolute timestamp of internal event to present_time + ts
void set_future_timestamp(us_timestamp_t ts) {
insert_absolute(::ticker_read_us(_ticker_data) + ts);
}
void set_past_timestamp(void) {
insert_absolute(::ticker_read_us(_ticker_data) - 1ULL);
}
};
/** Template for tests: insert, insert_absolute
*
* Test insert
* Given an instance of @a TimerEvent subclass
* When a tiestamp is set with @a insert()
* and given time elapses
* Then an event handler is called
*
* Test insert_absolute
* Given an instance of @a TimerEvent subclass
* When a tiestamp is set with @a insert_absolute()
* and given time elapses
* Then an event handler is called
*/
template<typename T>
void test_insert(void) {
T tte;
tte.set_future_timestamp(TEST_DELAY_US);
int32_t sem_slots = tte.sem_wait(0);
TEST_ASSERT_EQUAL(0, sem_slots);
sem_slots = tte.sem_wait(TEST_DELAY_US / 1000 + 1);
TEST_ASSERT_EQUAL(1, sem_slots);
tte.remove();
}
/** Template for tests: remove
*
* Test remove after insert
* Given an instance of @a TimerEvent subclass
* When a tiestamp is set with @a insert()
* and timestamp is removed before being reached
* Then the event handler is never called
*
* Test remove after insert_absolute
* Given an instance of @a TimerEvent subclass
* When a tiestamp is set with @a insert_absolute()
* and timestamp is removed before being reached
* Then the event handler is never called
*/
template<typename T>
void test_remove(void) {
T tte;
tte.set_future_timestamp(TEST_DELAY_US * 2);
int32_t sem_slots = tte.sem_wait(TEST_DELAY_US / 1000);
TEST_ASSERT_EQUAL(0, sem_slots);
tte.remove();
sem_slots = tte.sem_wait(TEST_DELAY_US * 2 / 1000 + 1);
TEST_ASSERT_EQUAL(0, sem_slots);
}
/** Test insert_absolute zero
* Given an instance of @a TimerEvent subclass
* When a timestamp of 0 us is set with @a insert_absolute()
* Then an event handler is called instantly
*/
void test_insert_zero(void) {
TestTimerEvent tte;
tte.insert_absolute(0ULL);
int32_t sem_slots = tte.sem_wait(0);
TEST_ASSERT_EQUAL(1, sem_slots);
tte.remove();
}
/** Template for tests: insert, insert_absolute past timestamp
*
* Test insert timestamp from the past
* Given an instance of @a TimerEvent subclass
* When a timestamp of X us is set with @a insert()
* and X is less than present_time
* Then an event handler is **not** called instantly
* (the event is scheduled after the ticker's overflow)
*
* Test insert_absolute timestamp from the past
* Given an instance of @a TimerEvent subclass
* When a timestamp of X us is set with @a insert_absolute()
* and X is less than present_time
* Then an event handler is called instantly
*/
template<typename T>
void test_insert_past(void) {
T tte;
tte.set_past_timestamp();
int32_t sem_slots = tte.sem_wait(0);
TEST_ASSERT_EQUAL(tte.SEM_SLOTS_AFTER_PAST_TS_INSERTED, sem_slots);
tte.remove();
}
utest::v1::status_t test_setup(const size_t number_of_cases)
{
GREENTEA_SETUP(5, "default_auto");
return verbose_test_setup_handler(number_of_cases);
}
Case cases[] = {
Case("Test insert", test_insert<TestTimerEventRelative>),
Case("Test insert_absolute", test_insert<TestTimerEventAbsolute>),
Case("Test remove after insert", test_remove<TestTimerEventRelative>),
Case("Test remove after insert_absolute", test_remove<TestTimerEventAbsolute>),
Case("Test insert_absolute zero", test_insert_zero),
Case("Test insert timestamp from the past", test_insert_past<TestTimerEventRelative>),
Case("Test insert_absolute timestamp from the past", test_insert_past<TestTimerEventAbsolute>),
};
Specification specification(test_setup, cases);
int main()
{
return !Harness::run(specification);
}