Skip to content

Commit 1437651

Browse files
test-mbed_drivers-ticker: improve two ticker test accuracy
test_case_2x_callbacks test was redesigned to eliminate ticker rescheduling and improve time mesure accuracy. Constant ticker rescheduling (detach()/attach_us() calls) was causing the gap between consecutive callback calls was not exact 1ms but 1ms + time needed to call the callback and attach new one. New design just uses two tickers to update counter alternatively every 1ms without rescheduling them
1 parent dd84c0b commit 1437651

File tree

1 file changed

+38
-71
lines changed

1 file changed

+38
-71
lines changed

TESTS/mbed_drivers/ticker/main.cpp

Lines changed: 38 additions & 71 deletions
Original file line numberDiff line numberDiff line change
@@ -38,54 +38,36 @@ volatile uint32_t multi_counter;
3838
DigitalOut led1(LED1);
3939
DigitalOut led2(LED2);
4040

41-
Ticker *volatile ticker1;
42-
Ticker *volatile ticker2;
4341
Timer gtimer;
44-
4542
volatile int ticker_count = 0;
46-
volatile bool print_tick = false;
47-
48-
void ticker_callback_1_switch_to_2(void);
49-
void ticker_callback_2_switch_to_1(void);
5043

51-
void increment_ticker_counter(void)
52-
{
53-
++callback_trigger_count;
54-
}
5544

5645
void switch_led1_state(void)
5746
{
58-
led1 = !led1;
47+
// blink 3 times per second
48+
if((callback_trigger_count % 333) == 0) {
49+
led1 = !led1;
50+
}
5951
}
6052

6153
void switch_led2_state(void)
6254
{
63-
led2 = !led2;
55+
// blink 3 times per second
56+
// make led2 blink at the same callback_trigger_count value as led1
57+
if(((callback_trigger_count - 1) % 333) == 0) {
58+
led2 = !led2;
59+
}
6460
}
6561

66-
void ticker_callback_1_switch_to_2(void)
62+
void ticker_callback_1(void)
6763
{
6864
++callback_trigger_count;
69-
// If ticker is NULL then it is being or has been deleted
70-
if (ticker1) {
71-
ticker1->detach();
72-
}
73-
if (ticker2) {
74-
ticker2->attach_us(ticker_callback_2_switch_to_1, ONE_MILLI_SEC);
75-
}
7665
switch_led1_state();
7766
}
7867

79-
void ticker_callback_2_switch_to_1(void)
68+
void ticker_callback_2(void)
8069
{
8170
++callback_trigger_count;
82-
// If ticker is NULL then it is being or has been deleted
83-
if (ticker2) {
84-
ticker2->detach();
85-
}
86-
if (ticker1) {
87-
ticker1->attach_us(ticker_callback_1_switch_to_2, ONE_MILLI_SEC);
88-
}
8971
switch_led2_state();
9072
}
9173

@@ -110,9 +92,8 @@ void increment_multi_counter(void)
11092

11193
/* Tests is to measure the accuracy of Ticker over a period of time
11294
*
113-
* 1) DUT would start to update callback_trigger_count every milli sec, in 2x callback we use 2 tickers
114-
* to update the count alternatively.
115-
* 2) Host would query what is current count base_time, Device responds by the callback_trigger_count
95+
* 1) DUT would start to update callback_trigger_count every milli sec
96+
* 2) Host would query what is current count base_time, Device responds by the callback_trigger_count.
11697
* 3) Host after waiting for measurement stretch. It will query for device time again final_time.
11798
* 4) Host computes the drift considering base_time, final_time, transport delay and measurement stretch
11899
* 5) Finally host send the results back to device pass/fail based on tolerance.
@@ -123,9 +104,14 @@ void test_case_1x_ticker()
123104
char _key[11] = { };
124105
char _value[128] = { };
125106
int expected_key = 1;
107+
Ticker ticker;
108+
109+
led1 = 1;
110+
led2 = 1;
111+
callback_trigger_count = 0;
126112

127113
greentea_send_kv("timing_drift_check_start", 0);
128-
ticker1->attach_us(&increment_ticker_counter, ONE_MILLI_SEC);
114+
ticker.attach_us(&ticker_callback_1, ONE_MILLI_SEC);
129115

130116
// wait for 1st signal from host
131117
do {
@@ -144,18 +130,32 @@ void test_case_1x_ticker()
144130
TEST_ASSERT_EQUAL_STRING_MESSAGE("pass", _key,"Host side script reported a fail...");
145131
}
146132

147-
void test_case_2x_callbacks()
133+
/* Tests is to measure the accuracy of Ticker over a period of time
134+
*
135+
* 1) DUT would start to update callback_trigger_count every milli sec, we use 2 tickers
136+
* to update the count alternatively.
137+
* 2) Host would query what is current count base_time, Device responds by the callback_trigger_count
138+
* 3) Host after waiting for measurement stretch. It will query for device time again final_time.
139+
* 4) Host computes the drift considering base_time, final_time, transport delay and measurement stretch
140+
* 5) Finally host send the results back to device pass/fail based on tolerance.
141+
* 6) More details on tests can be found in timing_drift_auto.py
142+
*/
143+
void test_case_2x_ticker()
148144
{
149145
char _key[11] = { };
150146
char _value[128] = { };
151147
int expected_key = 1;
148+
Ticker ticker1, ticker2;
152149

153150
led1 = 0;
154-
led2 = 0;
151+
led2 = 1;
155152
callback_trigger_count = 0;
156153

154+
ticker1.attach_us(ticker_callback_1, 2 * ONE_MILLI_SEC);
155+
// delay second ticker to have a pair of tickers tick every one millisecond
156+
wait_us(ONE_MILLI_SEC);
157157
greentea_send_kv("timing_drift_check_start", 0);
158-
ticker1->attach_us(ticker_callback_1_switch_to_2, ONE_MILLI_SEC);
158+
ticker2.attach_us(ticker_callback_2, 2 * ONE_MILLI_SEC);
159159

160160
// wait for 1st signal from host
161161
do {
@@ -307,39 +307,6 @@ void test_attach_us_time(void)
307307
}
308308

309309

310-
utest::v1::status_t one_ticker_case_setup_handler_t(const Case *const source, const size_t index_of_case)
311-
{
312-
ticker1 = new Ticker();
313-
return greentea_case_setup_handler(source, index_of_case);
314-
}
315-
316-
utest::v1::status_t two_ticker_case_setup_handler_t(const Case *const source, const size_t index_of_case)
317-
{
318-
ticker1 = new Ticker();
319-
ticker2 = new Ticker();
320-
return utest::v1::greentea_case_setup_handler(source, index_of_case);
321-
}
322-
323-
utest::v1::status_t one_ticker_case_teardown_handler_t(const Case *const source, const size_t passed, const size_t failed, const utest::v1::failure_t reason)
324-
{
325-
Ticker *temp1 = ticker1;
326-
ticker1 = NULL;
327-
delete temp1;
328-
return utest::v1::greentea_case_teardown_handler(source, passed, failed, reason);
329-
}
330-
331-
utest::v1::status_t two_ticker_case_teardown_handler_t(const Case *const source, const size_t passed, const size_t failed, const utest::v1::failure_t reason)
332-
{
333-
Ticker *temp1 = ticker1;
334-
Ticker *temp2 = ticker2;
335-
ticker1 = NULL;
336-
ticker2 = NULL;
337-
delete temp1;
338-
delete temp2;
339-
return utest::v1::greentea_case_teardown_handler(source, passed, failed, reason);
340-
}
341-
342-
343310
// Test cases
344311
Case cases[] = {
345312
Case("Test attach for 0.01s and time measure", test_attach_time<10000>),
@@ -351,8 +318,8 @@ Case cases[] = {
351318
Case("Test detach", test_detach),
352319
Case("Test multi call and time measure", test_multi_call_time),
353320
Case("Test multi ticker", test_multi_ticker),
354-
Case("Test timers: 1x ticker", one_ticker_case_setup_handler_t,test_case_1x_ticker, one_ticker_case_teardown_handler_t),
355-
Case("Test timers: 2x callbacks", two_ticker_case_setup_handler_t,test_case_2x_callbacks, two_ticker_case_teardown_handler_t)
321+
Case("Test timers: 1x ticker", test_case_1x_ticker),
322+
Case("Test timers: 2x ticker", test_case_2x_ticker)
356323
};
357324

358325
utest::v1::status_t greentea_test_setup(const size_t number_of_cases)

0 commit comments

Comments
 (0)