2014-12-02 06:50:07 +00:00
|
|
|
|
/* mbed Microcontroller Library
|
|
|
|
|
*******************************************************************************
|
2018-03-28 13:00:53 +00:00
|
|
|
|
* Copyright (c) 2018, STMicroelectronics
|
2014-12-02 06:50:07 +00:00
|
|
|
|
* All rights reserved.
|
|
|
|
|
*
|
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
|
*
|
|
|
|
|
* 1. Redistributions of source code must retain the above copyright notice,
|
|
|
|
|
* this list of conditions and the following disclaimer.
|
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
|
|
|
* and/or other materials provided with the distribution.
|
|
|
|
|
* 3. Neither the name of STMicroelectronics nor the names of its contributors
|
|
|
|
|
* may be used to endorse or promote products derived from this software
|
|
|
|
|
* without specific prior written permission.
|
|
|
|
|
*
|
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
|
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
|
|
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
|
|
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
*******************************************************************************
|
|
|
|
|
*/
|
|
|
|
|
#if DEVICE_SLEEP
|
|
|
|
|
|
2017-02-21 16:43:22 +00:00
|
|
|
|
#include "sleep_api.h"
|
2018-03-28 13:00:53 +00:00
|
|
|
|
#include "us_ticker_api.h"
|
2018-06-20 12:58:57 +00:00
|
|
|
|
#include "us_ticker_data.h"
|
2018-03-28 13:00:53 +00:00
|
|
|
|
#include "mbed_critical.h"
|
|
|
|
|
#include "mbed_error.h"
|
2014-12-02 06:50:07 +00:00
|
|
|
|
|
2018-07-04 14:44:00 +00:00
|
|
|
|
extern void save_timer_ctx(void);
|
|
|
|
|
extern void restore_timer_ctx(void);
|
2014-12-02 06:50:07 +00:00
|
|
|
|
|
2018-04-06 13:18:16 +00:00
|
|
|
|
/* Wait loop - assuming tick is 1 us */
|
|
|
|
|
static void wait_loop(uint32_t timeout)
|
|
|
|
|
{
|
|
|
|
|
uint32_t t1, t2, elapsed = 0;
|
|
|
|
|
t1 = us_ticker_read();
|
|
|
|
|
do {
|
|
|
|
|
t2 = us_ticker_read();
|
|
|
|
|
elapsed = (t2 > t1) ? (t2 - t1) : ((uint64_t)t2 + 0xFFFFFFFF - t1 + 1);
|
|
|
|
|
} while (elapsed < timeout);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-18 14:35:42 +00:00
|
|
|
|
|
2018-03-28 13:37:45 +00:00
|
|
|
|
// On L4 platforms we've seen unstable PLL CLK configuraiton
|
|
|
|
|
// when DEEP SLEEP exits just few <20>s after being entered
|
|
|
|
|
// So we need to force MSI usage before setting clocks again
|
2018-06-18 14:35:42 +00:00
|
|
|
|
static void ForcePeriphOutofDeepSleep(void)
|
2018-04-06 15:03:53 +00:00
|
|
|
|
{
|
2018-03-28 13:37:45 +00:00
|
|
|
|
uint32_t pFLatency = 0;
|
2018-06-18 14:35:42 +00:00
|
|
|
|
RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
|
2018-03-28 13:37:45 +00:00
|
|
|
|
|
|
|
|
|
#if (TARGET_STM32L4 || TARGET_STM32L1) /* MSI used for L4 */
|
|
|
|
|
/* Get the Clocks configuration according to the internal RCC registers */
|
|
|
|
|
HAL_RCC_GetClockConfig(&RCC_ClkInitStruct, &pFLatency);
|
|
|
|
|
|
|
|
|
|
// Select HSI ss system clock source as a first step
|
|
|
|
|
#ifdef RCC_CLOCKTYPE_PCLK2
|
2018-06-27 12:21:07 +00:00
|
|
|
|
RCC_ClkInitStruct.ClockType = (RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_HCLK
|
|
|
|
|
| RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2);
|
2018-03-28 13:37:45 +00:00
|
|
|
|
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
|
|
|
|
|
#else
|
2018-06-27 12:21:07 +00:00
|
|
|
|
RCC_ClkInitStruct.ClockType = (RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_HCLK
|
|
|
|
|
| RCC_CLOCKTYPE_PCLK1);
|
2018-03-28 13:37:45 +00:00
|
|
|
|
#endif
|
|
|
|
|
RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_MSI;
|
|
|
|
|
RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
|
|
|
|
|
RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1;
|
|
|
|
|
if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, pFLatency) != HAL_OK) {
|
|
|
|
|
error("clock issue\r\n");
|
|
|
|
|
}
|
|
|
|
|
#else /* HSI used on others */
|
|
|
|
|
/* Get the Clocks configuration according to the internal RCC registers */
|
|
|
|
|
HAL_RCC_GetClockConfig(&RCC_ClkInitStruct, &pFLatency);
|
|
|
|
|
|
|
|
|
|
/**Initializes the CPU, AHB and APB busses clocks
|
|
|
|
|
*/
|
|
|
|
|
#ifdef RCC_CLOCKTYPE_PCLK2
|
2018-06-27 12:21:07 +00:00
|
|
|
|
RCC_ClkInitStruct.ClockType = (RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_SYSCLK
|
|
|
|
|
| RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2);
|
2018-03-28 13:37:45 +00:00
|
|
|
|
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
|
|
|
|
|
#else
|
2018-06-27 12:21:07 +00:00
|
|
|
|
RCC_ClkInitStruct.ClockType = (RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_SYSCLK
|
|
|
|
|
| RCC_CLOCKTYPE_PCLK1);
|
2018-03-28 13:37:45 +00:00
|
|
|
|
#endif
|
|
|
|
|
RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_HSI;
|
|
|
|
|
RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
|
|
|
|
|
RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
|
|
|
|
|
if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, pFLatency) != HAL_OK) {
|
|
|
|
|
error("clock issue");
|
|
|
|
|
}
|
|
|
|
|
#endif // TARGET_STM32L4
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-18 14:35:42 +00:00
|
|
|
|
static void ForceOscOutofDeepSleep(void)
|
|
|
|
|
{
|
|
|
|
|
RCC_OscInitTypeDef RCC_OscInitStruct = {0};
|
|
|
|
|
|
|
|
|
|
/* Enable Power Control clock */
|
|
|
|
|
__HAL_RCC_PWR_CLK_ENABLE();
|
|
|
|
|
|
|
|
|
|
/* Get the Oscillators configuration according to the internal RCC registers */
|
|
|
|
|
HAL_RCC_GetOscConfig(&RCC_OscInitStruct);
|
|
|
|
|
|
|
|
|
|
#if (TARGET_STM32L4 || TARGET_STM32L1) /* MSI used for L4 */
|
|
|
|
|
/**Initializes the CPU, AHB and APB busses clocks
|
|
|
|
|
*/
|
|
|
|
|
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_MSI;
|
|
|
|
|
RCC_OscInitStruct.MSIState = RCC_MSI_ON;
|
|
|
|
|
RCC_OscInitStruct.MSICalibrationValue = RCC_MSICALIBRATION_DEFAULT;
|
|
|
|
|
RCC_OscInitStruct.MSIClockRange = RCC_MSIRANGE_4; // Intermediate freq, 1MHz range
|
|
|
|
|
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;
|
|
|
|
|
if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) {
|
|
|
|
|
error("clock issue\r\n");
|
|
|
|
|
}
|
|
|
|
|
#else /* HSI used on others */
|
|
|
|
|
/**Initializes the CPU, AHB and APB busses clocks
|
|
|
|
|
*/
|
|
|
|
|
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI;
|
|
|
|
|
RCC_OscInitStruct.HSIState = RCC_HSI_ON;
|
|
|
|
|
RCC_OscInitStruct.HSICalibrationValue = 16;
|
|
|
|
|
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE;
|
|
|
|
|
if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) {
|
|
|
|
|
error("clock issue");
|
|
|
|
|
}
|
|
|
|
|
#endif // TARGET_STM32L4
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* The content of this function has been split into 2 separate functions
|
|
|
|
|
so that the involved structures are not allocated on the stack in parallel.
|
|
|
|
|
This will reduce the maximum stack usage in case on non-optimized / debug
|
|
|
|
|
compilers settings */
|
|
|
|
|
static void ForceClockOutofDeepSleep(void)
|
|
|
|
|
{
|
|
|
|
|
ForceOscOutofDeepSleep();
|
|
|
|
|
ForcePeriphOutofDeepSleep();
|
|
|
|
|
}
|
|
|
|
|
|
2017-01-18 13:56:36 +00:00
|
|
|
|
void hal_sleep(void)
|
2016-11-23 15:41:32 +00:00
|
|
|
|
{
|
2017-07-25 09:30:05 +00:00
|
|
|
|
// Disable IRQs
|
|
|
|
|
core_util_critical_section_enter();
|
|
|
|
|
|
2014-12-02 06:50:07 +00:00
|
|
|
|
// Request to enter SLEEP mode
|
|
|
|
|
HAL_PWR_EnterSLEEPMode(PWR_MAINREGULATOR_ON, PWR_SLEEPENTRY_WFI);
|
2017-07-25 09:30:05 +00:00
|
|
|
|
|
|
|
|
|
// Enable IRQs
|
|
|
|
|
core_util_critical_section_exit();
|
2014-12-02 06:50:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-08-06 15:57:24 +00:00
|
|
|
|
extern int serial_IsTxOngoing(void);
|
|
|
|
|
|
2017-01-18 13:56:36 +00:00
|
|
|
|
void hal_deepsleep(void)
|
2014-12-02 06:50:07 +00:00
|
|
|
|
{
|
2018-08-06 15:57:24 +00:00
|
|
|
|
/* WORKAROUND:
|
|
|
|
|
* MBED serial driver does not handle deepsleep lock
|
|
|
|
|
* to prevent entering deepsleep until HW serial FIFO is empty.
|
|
|
|
|
* This is tracked in mbed issue 4408.
|
|
|
|
|
* For now, we're checking all Serial HW FIFO. If any transfer is ongoing
|
|
|
|
|
* we're not entering deep sleep and returning immediately. */
|
|
|
|
|
if(serial_IsTxOngoing()) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-25 09:30:05 +00:00
|
|
|
|
// Disable IRQs
|
|
|
|
|
core_util_critical_section_enter();
|
|
|
|
|
|
2018-07-04 14:44:00 +00:00
|
|
|
|
save_timer_ctx();
|
2016-11-23 15:41:32 +00:00
|
|
|
|
|
2014-12-02 06:50:07 +00:00
|
|
|
|
// Request to enter STOP mode with regulator in low power mode
|
2016-11-23 15:41:32 +00:00
|
|
|
|
#if TARGET_STM32L4
|
2017-01-31 10:44:41 +00:00
|
|
|
|
int pwrClockEnabled = __HAL_RCC_PWR_IS_CLK_ENABLED();
|
|
|
|
|
int lowPowerModeEnabled = PWR->CR1 & PWR_CR1_LPR;
|
2017-11-07 16:50:56 +00:00
|
|
|
|
|
2017-02-07 21:24:20 +00:00
|
|
|
|
if (!pwrClockEnabled) {
|
2017-01-09 11:21:24 +00:00
|
|
|
|
__HAL_RCC_PWR_CLK_ENABLE();
|
2017-02-07 21:24:20 +00:00
|
|
|
|
}
|
|
|
|
|
if (lowPowerModeEnabled) {
|
2017-01-09 11:21:24 +00:00
|
|
|
|
HAL_PWREx_DisableLowPowerRunMode();
|
2017-02-07 21:24:20 +00:00
|
|
|
|
}
|
2017-11-07 16:50:56 +00:00
|
|
|
|
|
2017-01-31 10:44:41 +00:00
|
|
|
|
HAL_PWREx_EnterSTOP2Mode(PWR_STOPENTRY_WFI);
|
2017-11-07 16:50:56 +00:00
|
|
|
|
|
2017-02-07 21:24:20 +00:00
|
|
|
|
if (lowPowerModeEnabled) {
|
2017-01-31 10:44:41 +00:00
|
|
|
|
HAL_PWREx_EnableLowPowerRunMode();
|
2017-02-07 21:24:20 +00:00
|
|
|
|
}
|
|
|
|
|
if (!pwrClockEnabled) {
|
2017-01-09 11:21:24 +00:00
|
|
|
|
__HAL_RCC_PWR_CLK_DISABLE();
|
2017-02-07 21:24:20 +00:00
|
|
|
|
}
|
2016-11-23 15:41:32 +00:00
|
|
|
|
#else /* TARGET_STM32L4 */
|
2014-12-02 06:50:07 +00:00
|
|
|
|
HAL_PWR_EnterSTOPMode(PWR_LOWPOWERREGULATOR_ON, PWR_STOPENTRY_WFI);
|
2016-11-23 15:41:32 +00:00
|
|
|
|
#endif /* TARGET_STM32L4 */
|
2018-07-04 14:44:00 +00:00
|
|
|
|
|
2018-03-28 13:37:45 +00:00
|
|
|
|
// Verify Clock Out of Deep Sleep
|
|
|
|
|
ForceClockOutofDeepSleep();
|
2016-11-23 15:41:32 +00:00
|
|
|
|
|
2014-12-02 06:50:07 +00:00
|
|
|
|
// After wake-up from STOP reconfigure the PLL
|
|
|
|
|
SetSysClock();
|
2016-11-23 15:41:32 +00:00
|
|
|
|
|
2018-04-06 13:18:16 +00:00
|
|
|
|
/* Wait for clock to be stabilized.
|
|
|
|
|
* TO DO: a better way of doing this, would be to rely on
|
|
|
|
|
* HW Flag. At least this ensures proper operation out of
|
|
|
|
|
* deep sleep */
|
|
|
|
|
wait_loop(500);
|
|
|
|
|
|
2018-07-04 14:44:00 +00:00
|
|
|
|
restore_timer_ctx();
|
2017-02-10 16:01:36 +00:00
|
|
|
|
|
2018-03-28 13:37:45 +00:00
|
|
|
|
// Enable IRQs
|
2018-06-27 12:21:07 +00:00
|
|
|
|
core_util_critical_section_exit();
|
2014-12-02 06:50:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif
|