mirror of https://github.com/ARMmbed/mbed-os.git
Merge pull request #10124 from jamesbeyond/fm_sleep
Enable low-power ticker and Sleep for FastModelspull/10354/head
commit
3a4f591a76
|
|
@ -74,9 +74,10 @@ Case cases[] = {
|
||||||
Case("1 s delay during deepsleep (attach_us)", test_deepsleep<AttachUSTester<LowPowerTimeout>, 1000000, LONG_DELTA_US>,
|
Case("1 s delay during deepsleep (attach_us)", test_deepsleep<AttachUSTester<LowPowerTimeout>, 1000000, LONG_DELTA_US>,
|
||||||
greentea_failure_handler),
|
greentea_failure_handler),
|
||||||
#endif
|
#endif
|
||||||
|
#if !defined(__ARM_FM) //FastModels not support time drifting test
|
||||||
Case("Timing drift (attach)", test_drift<AttachTester<LowPowerTimeout> >),
|
Case("Timing drift (attach)", test_drift<AttachTester<LowPowerTimeout> >),
|
||||||
Case("Timing drift (attach_us)", test_drift<AttachUSTester<LowPowerTimeout> >),
|
Case("Timing drift (attach_us)", test_drift<AttachUSTester<LowPowerTimeout> >),
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
utest::v1::status_t greentea_test_setup(const size_t number_of_cases)
|
utest::v1::status_t greentea_test_setup(const size_t number_of_cases)
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,121 @@
|
||||||
|
/* mbed Microcontroller Library
|
||||||
|
* Copyright (c) 2006-2019 ARM Limited
|
||||||
|
* SPDX-License-Identifier: Apache-2.0
|
||||||
|
*
|
||||||
|
* 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 <stddef.h>
|
||||||
|
#include "lp_ticker_api.h"
|
||||||
|
#include "PeripheralNames.h"
|
||||||
|
|
||||||
|
#define LP_TICKER_INTERRUPT CMSDK_TIMER0
|
||||||
|
#define LP_TICKER_COUNTER CMSDK_TIMER1
|
||||||
|
#define LP_TICKER_TIMER_IRQn TIMER0_IRQn
|
||||||
|
|
||||||
|
|
||||||
|
/* mbed OS HAL API defined lp_ticker as an increment ticker
|
||||||
|
* MPS2 platform provided in SSE-200 are decrement tickers
|
||||||
|
* with interrupt fired counter reaches 0.
|
||||||
|
*
|
||||||
|
* So 2 Timers are used to construct mbed OS HAL low power ticker.
|
||||||
|
*
|
||||||
|
* TIMER0 is for generating interrupts
|
||||||
|
* and TIMER0 will turned off when it is generating interrupts
|
||||||
|
*
|
||||||
|
* TIMER1 is for counting, and returns inverted binary when read from it
|
||||||
|
* Because TIMER1 is running at the speed of 25Mhz, it need to be shift by 10,
|
||||||
|
* in order to meet mbed HAL lp_ticker definitions
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
static int lp_ticker_inited = 0;
|
||||||
|
|
||||||
|
void lp_ticker_internal_handler(void)
|
||||||
|
{
|
||||||
|
LP_TICKER_INTERRUPT->CTRL &= ~CMSDK_TIMER_CTRL_IRQEN_Msk;
|
||||||
|
LP_TICKER_INTERRUPT->CTRL &= ~CMSDK_TIMER_CTRL_EN_Msk;
|
||||||
|
lp_ticker_irq_handler();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void lp_ticker_init(void)
|
||||||
|
{
|
||||||
|
if (lp_ticker_inited) {
|
||||||
|
lp_ticker_disable_interrupt();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
LP_TICKER_COUNTER->CTRL = 0x0ul;
|
||||||
|
LP_TICKER_INTERRUPT->CTRL = 0x0ul;
|
||||||
|
|
||||||
|
LP_TICKER_COUNTER->RELOAD = 0xFFFFFFFFul;
|
||||||
|
LP_TICKER_INTERRUPT->RELOAD = 0xFFFFFFFFul;
|
||||||
|
|
||||||
|
LP_TICKER_COUNTER->CTRL |= CMSDK_TIMER_CTRL_EN_Msk;
|
||||||
|
|
||||||
|
NVIC_SetVector(LP_TICKER_TIMER_IRQn, (uint32_t)lp_ticker_internal_handler);
|
||||||
|
lp_ticker_inited = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void lp_ticker_free(void)
|
||||||
|
{
|
||||||
|
LP_TICKER_COUNTER->CTRL &= ~CMSDK_TIMER_CTRL_EN_Msk;
|
||||||
|
lp_ticker_disable_interrupt();
|
||||||
|
lp_ticker_inited = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t lp_ticker_read()
|
||||||
|
{
|
||||||
|
return (~LP_TICKER_COUNTER->VALUE) >> 10;
|
||||||
|
}
|
||||||
|
|
||||||
|
void lp_ticker_set_interrupt(timestamp_t timestamp)
|
||||||
|
{
|
||||||
|
uint32_t delta = (timestamp << 10) - (~LP_TICKER_COUNTER->VALUE);
|
||||||
|
|
||||||
|
LP_TICKER_INTERRUPT->CTRL &= ~CMSDK_TIMER_CTRL_EN_Msk;
|
||||||
|
LP_TICKER_INTERRUPT->RELOAD = delta;
|
||||||
|
LP_TICKER_INTERRUPT->CTRL |= CMSDK_TIMER_CTRL_IRQEN_Msk;
|
||||||
|
LP_TICKER_INTERRUPT->CTRL |= CMSDK_TIMER_CTRL_EN_Msk;
|
||||||
|
|
||||||
|
NVIC_EnableIRQ(LP_TICKER_TIMER_IRQn);
|
||||||
|
}
|
||||||
|
|
||||||
|
void lp_ticker_fire_interrupt(void)
|
||||||
|
{
|
||||||
|
NVIC_EnableIRQ(LP_TICKER_TIMER_IRQn);
|
||||||
|
NVIC_SetPendingIRQ(LP_TICKER_TIMER_IRQn);
|
||||||
|
}
|
||||||
|
|
||||||
|
void lp_ticker_disable_interrupt(void)
|
||||||
|
{
|
||||||
|
LP_TICKER_INTERRUPT->CTRL &= ~CMSDK_TIMER_CTRL_IRQEN_Msk;
|
||||||
|
LP_TICKER_INTERRUPT->CTRL &= ~CMSDK_TIMER_CTRL_EN_Msk;
|
||||||
|
NVIC_DisableIRQ(LP_TICKER_TIMER_IRQn);
|
||||||
|
}
|
||||||
|
|
||||||
|
void lp_ticker_clear_interrupt(void)
|
||||||
|
{
|
||||||
|
LP_TICKER_INTERRUPT->INTCLEAR = CMSDK_TIMER_INTCLEAR_Msk;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ticker_info_t *lp_ticker_get_info(void)
|
||||||
|
{
|
||||||
|
static const ticker_info_t info = {
|
||||||
|
24414, // 10 stages scaled from 25MHz (dived by 1024)
|
||||||
|
22 // 22 bit counter
|
||||||
|
};
|
||||||
|
return &info;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -0,0 +1,65 @@
|
||||||
|
|
||||||
|
/** \addtogroup hal */
|
||||||
|
/** @{*/
|
||||||
|
/* mbed Microcontroller Library
|
||||||
|
* Copyright (c) 2018-2019 Arm Limited
|
||||||
|
* SPDX-License-Identifier: Apache-2.0
|
||||||
|
*
|
||||||
|
* 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 "sleep_api.h"
|
||||||
|
#include "us_ticker_api.h"
|
||||||
|
|
||||||
|
#include "PeripheralNames.h"
|
||||||
|
#define US_TICKER_COUNTER CMSDK_DUALTIMER1
|
||||||
|
#define US_TICKER_INTERRUPT CMSDK_DUALTIMER2
|
||||||
|
#define US_TICKER_TIMER_IRQn DUALTIMER_IRQn
|
||||||
|
|
||||||
|
|
||||||
|
#if DEVICE_SLEEP
|
||||||
|
|
||||||
|
void hal_sleep(void)
|
||||||
|
{
|
||||||
|
__WFI();
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Since there is no power management function implemented in MPS2,
|
||||||
|
* Also Deep Sleep mode mean to save power which is not practical on a simulator.
|
||||||
|
* So mbed HAL Deep sleep is mocked by Sleep,
|
||||||
|
* representing a "Waiting For Interrupt" state,
|
||||||
|
* but disabling the Microsec ticker in addition */
|
||||||
|
|
||||||
|
void hal_deepsleep(void)
|
||||||
|
{
|
||||||
|
#if DEVICE_USTICKER
|
||||||
|
uint32_t val = US_TICKER_COUNTER->TimerValue;
|
||||||
|
US_TICKER_COUNTER->TimerControl &= ~CMSDK_DUALTIMER1_CTRL_EN_Msk;
|
||||||
|
US_TICKER_INTERRUPT->TimerControl &= ~CMSDK_DUALTIMER2_CTRL_EN_Msk;
|
||||||
|
US_TICKER_INTERRUPT->TimerControl &= ~CMSDK_DUALTIMER2_CTRL_INTEN_Msk;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
__WFI();
|
||||||
|
|
||||||
|
#if DEVICE_USTICKER
|
||||||
|
US_TICKER_COUNTER->TimerLoad = val;
|
||||||
|
US_TICKER_COUNTER->TimerControl |= CMSDK_DUALTIMER1_CTRL_EN_Msk;
|
||||||
|
US_TICKER_INTERRUPT->TimerControl |= CMSDK_DUALTIMER1_CTRL_INTEN_Msk;
|
||||||
|
US_TICKER_INTERRUPT->TimerControl |= CMSDK_DUALTIMER2_CTRL_EN_Msk;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/**@}*/
|
||||||
|
|
@ -1,5 +1,6 @@
|
||||||
/* mbed Microcontroller Library
|
/* mbed Microcontroller Library
|
||||||
* Copyright (c) 2006-2018 ARM Limited
|
* Copyright (c) 2006-2019 ARM Limited
|
||||||
|
* SPDX-License-Identifier: Apache-2.0
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
|
@ -16,8 +17,8 @@
|
||||||
#include <stddef.h>
|
#include <stddef.h>
|
||||||
#include "us_ticker_api.h"
|
#include "us_ticker_api.h"
|
||||||
#include "PeripheralNames.h"
|
#include "PeripheralNames.h"
|
||||||
#define US_TICKER_TIMER1 CMSDK_DUALTIMER1
|
#define US_TICKER_COUNTER CMSDK_DUALTIMER1
|
||||||
#define US_TICKER_TIMER2 CMSDK_DUALTIMER2
|
#define US_TICKER_INTERRUPT CMSDK_DUALTIMER2
|
||||||
#define US_TICKER_TIMER_IRQn DUALTIMER_IRQn
|
#define US_TICKER_TIMER_IRQn DUALTIMER_IRQn
|
||||||
|
|
||||||
/** mbed OS HAL API defined us_ticker as an increment ticker
|
/** mbed OS HAL API defined us_ticker as an increment ticker
|
||||||
|
|
@ -43,22 +44,22 @@ void us_ticker_init(void)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
US_TICKER_TIMER1->TimerControl = 0x0ul; // disable TIMER1 and reset all control
|
US_TICKER_COUNTER->TimerControl = 0x0ul; // disable TIMER1 and reset all control
|
||||||
US_TICKER_TIMER2->TimerControl = 0x0ul; // disable TIMER2 and reset all control
|
US_TICKER_INTERRUPT->TimerControl = 0x0ul; // disable TIMER2 and reset all control
|
||||||
|
|
||||||
US_TICKER_TIMER1->TimerLoad = 0xFFFFFFFFul;
|
US_TICKER_COUNTER->TimerLoad = 0xFFFFFFFFul;
|
||||||
US_TICKER_TIMER2->TimerLoad = 0xFFFFFFFFul;
|
US_TICKER_INTERRUPT->TimerLoad = 0xFFFFFFFFul;
|
||||||
|
|
||||||
US_TICKER_TIMER1->TimerControl |= CMSDK_DUALTIMER1_CTRL_SIZE_Msk; // set TIMER1 to 32 bit counter
|
US_TICKER_COUNTER->TimerControl |= CMSDK_DUALTIMER1_CTRL_SIZE_Msk; // set TIMER1 to 32 bit counter
|
||||||
US_TICKER_TIMER2->TimerControl |= CMSDK_DUALTIMER2_CTRL_SIZE_Msk; // set TIMER2 to 32 bit counter
|
US_TICKER_INTERRUPT->TimerControl |= CMSDK_DUALTIMER2_CTRL_SIZE_Msk; // set TIMER2 to 32 bit counter
|
||||||
|
|
||||||
US_TICKER_TIMER1->TimerControl |= 0x1 << CMSDK_DUALTIMER1_CTRL_PRESCALE_Pos; // set TIMER1 with 4 stages prescale
|
US_TICKER_COUNTER->TimerControl |= 0x1 << CMSDK_DUALTIMER1_CTRL_PRESCALE_Pos; // set TIMER1 with 4 stages prescale
|
||||||
US_TICKER_TIMER2->TimerControl |= 0x1 << CMSDK_DUALTIMER2_CTRL_PRESCALE_Pos; // set TIMER2 with 4 stages prescale
|
US_TICKER_INTERRUPT->TimerControl |= 0x1 << CMSDK_DUALTIMER2_CTRL_PRESCALE_Pos; // set TIMER2 with 4 stages prescale
|
||||||
|
|
||||||
US_TICKER_TIMER2->TimerControl |= CMSDK_DUALTIMER2_CTRL_MODE_Msk; // set TIMER2 periodic mode
|
US_TICKER_INTERRUPT->TimerControl |= CMSDK_DUALTIMER2_CTRL_MODE_Msk; // set TIMER2 periodic mode
|
||||||
US_TICKER_TIMER2->TimerControl |= CMSDK_DUALTIMER2_CTRL_ONESHOOT_Msk; // set TIMER2 one-shot mode
|
US_TICKER_INTERRUPT->TimerControl |= CMSDK_DUALTIMER2_CTRL_ONESHOOT_Msk; // set TIMER2 one-shot mode
|
||||||
|
|
||||||
US_TICKER_TIMER1->TimerControl |= CMSDK_DUALTIMER1_CTRL_EN_Msk; // enable TIMER1 counter
|
US_TICKER_COUNTER->TimerControl |= CMSDK_DUALTIMER1_CTRL_EN_Msk; // enable TIMER1 counter
|
||||||
|
|
||||||
NVIC_SetVector(US_TICKER_TIMER_IRQn, (uint32_t)us_ticker_irq_handler);
|
NVIC_SetVector(US_TICKER_TIMER_IRQn, (uint32_t)us_ticker_irq_handler);
|
||||||
us_ticker_inited = 1;
|
us_ticker_inited = 1;
|
||||||
|
|
@ -66,24 +67,24 @@ void us_ticker_init(void)
|
||||||
|
|
||||||
void us_ticker_free(void)
|
void us_ticker_free(void)
|
||||||
{
|
{
|
||||||
US_TICKER_TIMER1->TimerControl &= ~CMSDK_DUALTIMER1_CTRL_EN_Msk; // disable TIMER1
|
US_TICKER_COUNTER->TimerControl &= ~CMSDK_DUALTIMER1_CTRL_EN_Msk; // disable TIMER1
|
||||||
US_TICKER_TIMER2->TimerControl &= ~CMSDK_DUALTIMER2_CTRL_EN_Msk; // disable TIMER2
|
US_TICKER_INTERRUPT->TimerControl &= ~CMSDK_DUALTIMER2_CTRL_EN_Msk; // disable TIMER2
|
||||||
us_ticker_disable_interrupt();
|
us_ticker_disable_interrupt();
|
||||||
us_ticker_inited = 0;
|
us_ticker_inited = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t us_ticker_read()
|
uint32_t us_ticker_read()
|
||||||
{
|
{
|
||||||
return ~US_TICKER_TIMER1->TimerValue;
|
return ~US_TICKER_COUNTER->TimerValue;
|
||||||
}
|
}
|
||||||
|
|
||||||
void us_ticker_set_interrupt(timestamp_t timestamp)
|
void us_ticker_set_interrupt(timestamp_t timestamp)
|
||||||
{
|
{
|
||||||
uint32_t delta = timestamp - us_ticker_read();
|
uint32_t delta = timestamp - us_ticker_read();
|
||||||
US_TICKER_TIMER2->TimerControl &= ~CMSDK_DUALTIMER2_CTRL_EN_Msk; // disable TIMER2
|
US_TICKER_INTERRUPT->TimerControl &= ~CMSDK_DUALTIMER2_CTRL_EN_Msk; // disable TIMER2
|
||||||
US_TICKER_TIMER2->TimerLoad = delta; // Set TIMER2 load value
|
US_TICKER_INTERRUPT->TimerLoad = delta; // Set TIMER2 load value
|
||||||
US_TICKER_TIMER2->TimerControl |= CMSDK_DUALTIMER2_CTRL_INTEN_Msk; // enable TIMER2 interrupt
|
US_TICKER_INTERRUPT->TimerControl |= CMSDK_DUALTIMER2_CTRL_INTEN_Msk; // enable TIMER2 interrupt
|
||||||
US_TICKER_TIMER2->TimerControl |= CMSDK_DUALTIMER2_CTRL_EN_Msk; // enable TIMER2 counter
|
US_TICKER_INTERRUPT->TimerControl |= CMSDK_DUALTIMER2_CTRL_EN_Msk; // enable TIMER2 counter
|
||||||
NVIC_EnableIRQ(US_TICKER_TIMER_IRQn);
|
NVIC_EnableIRQ(US_TICKER_TIMER_IRQn);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -96,14 +97,14 @@ void us_ticker_fire_interrupt(void)
|
||||||
|
|
||||||
void us_ticker_disable_interrupt(void)
|
void us_ticker_disable_interrupt(void)
|
||||||
{
|
{
|
||||||
US_TICKER_TIMER2->TimerControl &= ~CMSDK_DUALTIMER2_CTRL_INTEN_Msk;
|
US_TICKER_INTERRUPT->TimerControl &= ~CMSDK_DUALTIMER2_CTRL_INTEN_Msk;
|
||||||
US_TICKER_TIMER2->TimerControl &= ~CMSDK_DUALTIMER2_CTRL_EN_Msk; // disable TIMER2
|
US_TICKER_INTERRUPT->TimerControl &= ~CMSDK_DUALTIMER2_CTRL_EN_Msk; // disable TIMER2
|
||||||
NVIC_DisableIRQ(US_TICKER_TIMER_IRQn);
|
NVIC_DisableIRQ(US_TICKER_TIMER_IRQn);
|
||||||
}
|
}
|
||||||
|
|
||||||
void us_ticker_clear_interrupt(void)
|
void us_ticker_clear_interrupt(void)
|
||||||
{
|
{
|
||||||
US_TICKER_TIMER2->TimerIntClr = CMSDK_DUALTIMER2_INTCLR_Msk;
|
US_TICKER_INTERRUPT->TimerIntClr = CMSDK_DUALTIMER2_INTCLR_Msk;
|
||||||
}
|
}
|
||||||
|
|
||||||
const ticker_info_t *us_ticker_get_info(void)
|
const ticker_info_t *us_ticker_get_info(void)
|
||||||
|
|
|
||||||
|
|
@ -7958,10 +7958,12 @@
|
||||||
"FLASH",
|
"FLASH",
|
||||||
"I2C",
|
"I2C",
|
||||||
"INTERRUPTIN",
|
"INTERRUPTIN",
|
||||||
|
"LPTICKER",
|
||||||
"PORTIN",
|
"PORTIN",
|
||||||
"PORTINOUT",
|
"PORTINOUT",
|
||||||
"PORTOUT",
|
"PORTOUT",
|
||||||
"SERIAL",
|
"SERIAL",
|
||||||
|
"SLEEP",
|
||||||
"SPI",
|
"SPI",
|
||||||
"SPISLAVE",
|
"SPISLAVE",
|
||||||
"TSC",
|
"TSC",
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue