/* 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 "rtos.h" #include "lp_ticker_api_tests.h" #include "hal/lp_ticker_api.h" #include "hal/mbed_lp_ticker_wrapper.h" #if !DEVICE_LPTICKER #error [NOT_SUPPORTED] Low power timer not supported for this target #endif using namespace utest::v1; volatile int intFlag = 0; ticker_irq_handler_type prev_handler; #define US_PER_MS 1000 #define TICKER_GLITCH_TEST_TICKS 1000 #define TICKER_INT_VAL 500 #define TICKER_DELTA 10 #define LP_TICKER_OV_LIMIT 4000 /* Flush serial buffer before deep sleep * * Since deepsleep() may shut down the UART peripheral, we wait for some time * to allow for hardware serial buffers to completely flush. * * Take NUMAKER_PFM_NUC472 as an example: * Its UART peripheral has 16-byte Tx FIFO. With baud rate set to 9600, flush * Tx FIFO would take: 16 * 8 * 1000 / 9600 = 13.3 (ms). So set wait time to * 20ms here for safe. * * This should be replaced with a better function that checks if the * hardware buffers are empty. However, such an API does not exist now, * so we'll use the busy_wait_ms() function for now. */ #define SERIAL_FLUSH_TIME_MS 20 void busy_wait_ms(int ms) { const ticker_data_t *const ticker = get_us_ticker_data(); uint32_t start = ticker_read(ticker); while ((ticker_read(ticker) - start) < (uint32_t)(ms * US_PER_MS)); } /* Since according to the ticker requirements min acceptable counter size is * - 12 bits for low power timer - max count = 4095, * then all test cases must be executed in this time windows. * HAL ticker layer handles counter overflow and it is not handled in the target * ticker drivers. Ensure we have enough time to execute test case without overflow. */ void overflow_protect() { uint32_t time_window; time_window = LP_TICKER_OV_LIMIT; const uint32_t ticks_now = lp_ticker_read(); const ticker_info_t *p_ticker_info = lp_ticker_get_info(); const uint32_t max_count = ((1 << p_ticker_info->bits) - 1); if ((max_count - ticks_now) > time_window) { return; } while (lp_ticker_read() > ticks_now); } void ticker_event_handler_stub(const ticker_data_t *const ticker) { /* Indicate that ISR has been executed in interrupt context. */ if (core_util_is_isr_active()) { intFlag++; } /* Clear and disable ticker interrupt. */ lp_ticker_clear_interrupt(); lp_ticker_disable_interrupt(); } /* Test that the ticker has the correct frequency and number of bits. */ void lp_ticker_info_test() { const ticker_info_t *p_ticker_info = lp_ticker_get_info(); TEST_ASSERT(p_ticker_info->frequency >= 4000); TEST_ASSERT(p_ticker_info->frequency <= 64000); TEST_ASSERT(p_ticker_info->bits >= 12); } #if DEVICE_SLEEP /* Test that the ticker continues operating in deep sleep mode. */ void lp_ticker_deepsleep_test() { intFlag = 0; lp_ticker_init(); /* Give some time Green Tea to finish UART transmission before entering * deep-sleep mode. */ busy_wait_ms(SERIAL_FLUSH_TIME_MS); overflow_protect(); const uint32_t tick_count = lp_ticker_read(); /* Set interrupt. Interrupt should be fired when tick count is equal to: * tick_count + TICKER_INT_VAL. */ lp_ticker_set_interrupt(tick_count + TICKER_INT_VAL); TEST_ASSERT_TRUE(sleep_manager_can_deep_sleep_test_check()); while (!intFlag) { sleep(); } TEST_ASSERT_EQUAL(1, intFlag); } #endif /* Test that the ticker does not glitch backwards due to an incorrectly implemented ripple counter driver. */ void lp_ticker_glitch_test() { lp_ticker_init(); overflow_protect(); uint32_t last = lp_ticker_read(); const uint32_t start = last; while (last < (start + TICKER_GLITCH_TEST_TICKS)) { const uint32_t cur = lp_ticker_read(); TEST_ASSERT(cur >= last); last = cur; } } #if DEVICE_LPTICKER utest::v1::status_t lp_ticker_deepsleep_test_setup_handler(const Case *const source, const size_t index_of_case) { /* disable everything using the lp ticker for this test */ osKernelSuspend(); ticker_suspend(get_lp_ticker_data()); #if DEVICE_LPTICKER && (LPTICKER_DELAY_TICKS > 0) lp_ticker_wrapper_suspend(); #endif prev_handler = set_lp_ticker_irq_handler(ticker_event_handler_stub); return greentea_case_setup_handler(source, index_of_case); } utest::v1::status_t lp_ticker_deepsleep_test_teardown_handler(const Case *const source, const size_t passed, const size_t failed, const failure_t reason) { set_lp_ticker_irq_handler(prev_handler); #if DEVICE_LPTICKER && (LPTICKER_DELAY_TICKS > 0) lp_ticker_wrapper_resume(); #endif ticker_resume(get_lp_ticker_data()); osKernelResume(0); return greentea_case_teardown_handler(source, passed, failed, reason); } #endif utest::v1::status_t test_setup(const size_t number_of_cases) { GREENTEA_SETUP(20, "default_auto"); return verbose_test_setup_handler(number_of_cases); } Case cases[] = { Case("lp ticker info test", lp_ticker_info_test), #if DEVICE_SLEEP Case("lp ticker sleep test", lp_ticker_deepsleep_test_setup_handler, lp_ticker_deepsleep_test, lp_ticker_deepsleep_test_teardown_handler), #endif Case("lp ticker glitch test", lp_ticker_glitch_test) }; Specification specification(test_setup, cases); int main() { return !Harness::run(specification); }