Merge pull request #11198 from SteveCEvans/sce_rc2
[betaflight.git] / lib / main / STM32F4 / Drivers / STM32F4xx_HAL_Driver / Src / stm32f4xx_hal_pwr.c
blobec27001bb861525b9f307e996812983116661199
1 /**
2 ******************************************************************************
3 * @file stm32f4xx_hal_pwr.c
4 * @author MCD Application Team
5 * @version V1.7.1
6 * @date 14-April-2017
7 * @brief PWR HAL module driver.
8 * This file provides firmware functions to manage the following
9 * functionalities of the Power Controller (PWR) peripheral:
10 * + Initialization and de-initialization functions
11 * + Peripheral Control functions
13 ******************************************************************************
14 * @attention
16 * <h2><center>&copy; COPYRIGHT(c) 2017 STMicroelectronics</center></h2>
18 * Redistribution and use in source and binary forms, with or without modification,
19 * are permitted provided that the following conditions are met:
20 * 1. Redistributions of source code must retain the above copyright notice,
21 * this list of conditions and the following disclaimer.
22 * 2. Redistributions in binary form must reproduce the above copyright notice,
23 * this list of conditions and the following disclaimer in the documentation
24 * and/or other materials provided with the distribution.
25 * 3. Neither the name of STMicroelectronics nor the names of its contributors
26 * may be used to endorse or promote products derived from this software
27 * without specific prior written permission.
29 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
30 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
31 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
32 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
33 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
34 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
35 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
36 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
37 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
38 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
40 ******************************************************************************
41 */
43 /* Includes ------------------------------------------------------------------*/
44 #include "stm32f4xx_hal.h"
46 /** @addtogroup STM32F4xx_HAL_Driver
47 * @{
50 /** @defgroup PWR PWR
51 * @brief PWR HAL module driver
52 * @{
55 #ifdef HAL_PWR_MODULE_ENABLED
57 /* Private typedef -----------------------------------------------------------*/
58 /* Private define ------------------------------------------------------------*/
59 /** @addtogroup PWR_Private_Constants
60 * @{
63 /** @defgroup PWR_PVD_Mode_Mask PWR PVD Mode Mask
64 * @{
65 */
66 #define PVD_MODE_IT 0x00010000U
67 #define PVD_MODE_EVT 0x00020000U
68 #define PVD_RISING_EDGE 0x00000001U
69 #define PVD_FALLING_EDGE 0x00000002U
70 /**
71 * @}
74 /**
75 * @}
76 */
77 /* Private macro -------------------------------------------------------------*/
78 /* Private variables ---------------------------------------------------------*/
79 /* Private function prototypes -----------------------------------------------*/
80 /* Private functions ---------------------------------------------------------*/
82 /** @defgroup PWR_Exported_Functions PWR Exported Functions
83 * @{
86 /** @defgroup PWR_Exported_Functions_Group1 Initialization and de-initialization functions
87 * @brief Initialization and de-initialization functions
89 @verbatim
90 ===============================================================================
91 ##### Initialization and de-initialization functions #####
92 ===============================================================================
93 [..]
94 After reset, the backup domain (RTC registers, RTC backup data
95 registers and backup SRAM) is protected against possible unwanted
96 write accesses.
97 To enable access to the RTC Domain and RTC registers, proceed as follows:
98 (+) Enable the Power Controller (PWR) APB1 interface clock using the
99 __HAL_RCC_PWR_CLK_ENABLE() macro.
100 (+) Enable access to RTC domain using the HAL_PWR_EnableBkUpAccess() function.
102 @endverbatim
103 * @{
107 * @brief Deinitializes the HAL PWR peripheral registers to their default reset values.
108 * @retval None
110 void HAL_PWR_DeInit(void)
112 __HAL_RCC_PWR_FORCE_RESET();
113 __HAL_RCC_PWR_RELEASE_RESET();
117 * @brief Enables access to the backup domain (RTC registers, RTC
118 * backup data registers and backup SRAM).
119 * @note If the HSE divided by 2, 3, ..31 is used as the RTC clock, the
120 * Backup Domain Access should be kept enabled.
121 * @retval None
123 void HAL_PWR_EnableBkUpAccess(void)
125 *(__IO uint32_t *) CR_DBP_BB = (uint32_t)ENABLE;
129 * @brief Disables access to the backup domain (RTC registers, RTC
130 * backup data registers and backup SRAM).
131 * @note If the HSE divided by 2, 3, ..31 is used as the RTC clock, the
132 * Backup Domain Access should be kept enabled.
133 * @retval None
135 void HAL_PWR_DisableBkUpAccess(void)
137 *(__IO uint32_t *) CR_DBP_BB = (uint32_t)DISABLE;
141 * @}
144 /** @defgroup PWR_Exported_Functions_Group2 Peripheral Control functions
145 * @brief Low Power modes configuration functions
147 @verbatim
149 ===============================================================================
150 ##### Peripheral Control functions #####
151 ===============================================================================
153 *** PVD configuration ***
154 =========================
155 [..]
156 (+) The PVD is used to monitor the VDD power supply by comparing it to a
157 threshold selected by the PVD Level (PLS[2:0] bits in the PWR_CR).
158 (+) A PVDO flag is available to indicate if VDD/VDDA is higher or lower
159 than the PVD threshold. This event is internally connected to the EXTI
160 line16 and can generate an interrupt if enabled. This is done through
161 __HAL_PWR_PVD_EXTI_ENABLE_IT() macro.
162 (+) The PVD is stopped in Standby mode.
164 *** Wake-up pin configuration ***
165 ================================
166 [..]
167 (+) Wake-up pin is used to wake up the system from Standby mode. This pin is
168 forced in input pull-down configuration and is active on rising edges.
169 (+) There is one Wake-up pin: Wake-up Pin 1 on PA.00.
170 (++) For STM32F446xx there are two Wake-Up pins: Pin1 on PA.00 and Pin2 on PC.13
171 (++) For STM32F410xx/STM32F412Zx/STM32F412Rx/STM32F412Vx/STM32F412Cx there are three Wake-Up pins: Pin1 on PA.00, Pin2 on PC.00 and Pin3 on PC.01
173 *** Low Power modes configuration ***
174 =====================================
175 [..]
176 The devices feature 3 low-power modes:
177 (+) Sleep mode: Cortex-M4 core stopped, peripherals kept running.
178 (+) Stop mode: all clocks are stopped, regulator running, regulator
179 in low power mode
180 (+) Standby mode: 1.2V domain powered off.
182 *** Sleep mode ***
183 ==================
184 [..]
185 (+) Entry:
186 The Sleep mode is entered by using the HAL_PWR_EnterSLEEPMode(PWR_MAINREGULATOR_ON, PWR_SLEEPENTRY_WFI)
187 functions with
188 (++) PWR_SLEEPENTRY_WFI: enter SLEEP mode with WFI instruction
189 (++) PWR_SLEEPENTRY_WFE: enter SLEEP mode with WFE instruction
191 -@@- The Regulator parameter is not used for the STM32F4 family
192 and is kept as parameter just to maintain compatibility with the
193 lower power families (STM32L).
194 (+) Exit:
195 Any peripheral interrupt acknowledged by the nested vectored interrupt
196 controller (NVIC) can wake up the device from Sleep mode.
198 *** Stop mode ***
199 =================
200 [..]
201 In Stop mode, all clocks in the 1.2V domain are stopped, the PLL, the HSI,
202 and the HSE RC oscillators are disabled. Internal SRAM and register contents
203 are preserved.
204 The voltage regulator can be configured either in normal or low-power mode.
205 To minimize the consumption In Stop mode, FLASH can be powered off before
206 entering the Stop mode using the HAL_PWREx_EnableFlashPowerDown() function.
207 It can be switched on again by software after exiting the Stop mode using
208 the HAL_PWREx_DisableFlashPowerDown() function.
210 (+) Entry:
211 The Stop mode is entered using the HAL_PWR_EnterSTOPMode(PWR_MAINREGULATOR_ON)
212 function with:
213 (++) Main regulator ON.
214 (++) Low Power regulator ON.
215 (+) Exit:
216 Any EXTI Line (Internal or External) configured in Interrupt/Event mode.
218 *** Standby mode ***
219 ====================
220 [..]
222 The Standby mode allows to achieve the lowest power consumption. It is based
223 on the Cortex-M4 deep sleep mode, with the voltage regulator disabled.
224 The 1.2V domain is consequently powered off. The PLL, the HSI oscillator and
225 the HSE oscillator are also switched off. SRAM and register contents are lost
226 except for the RTC registers, RTC backup registers, backup SRAM and Standby
227 circuitry.
229 The voltage regulator is OFF.
231 (++) Entry:
232 (+++) The Standby mode is entered using the HAL_PWR_EnterSTANDBYMode() function.
233 (++) Exit:
234 (+++) WKUP pin rising edge, RTC alarm (Alarm A and Alarm B), RTC wake-up,
235 tamper event, time-stamp event, external reset in NRST pin, IWDG reset.
237 *** Auto-wake-up (AWU) from low-power mode ***
238 =============================================
239 [..]
241 (+) The MCU can be woken up from low-power mode by an RTC Alarm event, an RTC
242 Wake-up event, a tamper event or a time-stamp event, without depending on
243 an external interrupt (Auto-wake-up mode).
245 (+) RTC auto-wake-up (AWU) from the Stop and Standby modes
247 (++) To wake up from the Stop mode with an RTC alarm event, it is necessary to
248 configure the RTC to generate the RTC alarm using the HAL_RTC_SetAlarm_IT() function.
250 (++) To wake up from the Stop mode with an RTC Tamper or time stamp event, it
251 is necessary to configure the RTC to detect the tamper or time stamp event using the
252 HAL_RTCEx_SetTimeStamp_IT() or HAL_RTCEx_SetTamper_IT() functions.
254 (++) To wake up from the Stop mode with an RTC Wake-up event, it is necessary to
255 configure the RTC to generate the RTC Wake-up event using the HAL_RTCEx_SetWakeUpTimer_IT() function.
257 @endverbatim
258 * @{
262 * @brief Configures the voltage threshold detected by the Power Voltage Detector(PVD).
263 * @param sConfigPVD: pointer to an PWR_PVDTypeDef structure that contains the configuration
264 * information for the PVD.
265 * @note Refer to the electrical characteristics of your device datasheet for
266 * more details about the voltage threshold corresponding to each
267 * detection level.
268 * @retval None
270 void HAL_PWR_ConfigPVD(PWR_PVDTypeDef *sConfigPVD)
272 /* Check the parameters */
273 assert_param(IS_PWR_PVD_LEVEL(sConfigPVD->PVDLevel));
274 assert_param(IS_PWR_PVD_MODE(sConfigPVD->Mode));
276 /* Set PLS[7:5] bits according to PVDLevel value */
277 MODIFY_REG(PWR->CR, PWR_CR_PLS, sConfigPVD->PVDLevel);
279 /* Clear any previous config. Keep it clear if no event or IT mode is selected */
280 __HAL_PWR_PVD_EXTI_DISABLE_EVENT();
281 __HAL_PWR_PVD_EXTI_DISABLE_IT();
282 __HAL_PWR_PVD_EXTI_DISABLE_RISING_EDGE();
283 __HAL_PWR_PVD_EXTI_DISABLE_FALLING_EDGE();
285 /* Configure interrupt mode */
286 if((sConfigPVD->Mode & PVD_MODE_IT) == PVD_MODE_IT)
288 __HAL_PWR_PVD_EXTI_ENABLE_IT();
291 /* Configure event mode */
292 if((sConfigPVD->Mode & PVD_MODE_EVT) == PVD_MODE_EVT)
294 __HAL_PWR_PVD_EXTI_ENABLE_EVENT();
297 /* Configure the edge */
298 if((sConfigPVD->Mode & PVD_RISING_EDGE) == PVD_RISING_EDGE)
300 __HAL_PWR_PVD_EXTI_ENABLE_RISING_EDGE();
303 if((sConfigPVD->Mode & PVD_FALLING_EDGE) == PVD_FALLING_EDGE)
305 __HAL_PWR_PVD_EXTI_ENABLE_FALLING_EDGE();
310 * @brief Enables the Power Voltage Detector(PVD).
311 * @retval None
313 void HAL_PWR_EnablePVD(void)
315 *(__IO uint32_t *) CR_PVDE_BB = (uint32_t)ENABLE;
319 * @brief Disables the Power Voltage Detector(PVD).
320 * @retval None
322 void HAL_PWR_DisablePVD(void)
324 *(__IO uint32_t *) CR_PVDE_BB = (uint32_t)DISABLE;
328 * @brief Enables the Wake-up PINx functionality.
329 * @param WakeUpPinx: Specifies the Power Wake-Up pin to enable.
330 * This parameter can be one of the following values:
331 * @arg PWR_WAKEUP_PIN1
332 * @arg PWR_WAKEUP_PIN2 available only on STM32F410xx/STM32F446xx/STM32F412Zx/STM32F412Rx/STM32F412Vx/STM32F412Cx devices
333 * @arg PWR_WAKEUP_PIN3 available only on STM32F410xx/STM32F412xx devices
334 * @retval None
336 void HAL_PWR_EnableWakeUpPin(uint32_t WakeUpPinx)
338 /* Check the parameter */
339 assert_param(IS_PWR_WAKEUP_PIN(WakeUpPinx));
341 /* Enable the wake up pin */
342 SET_BIT(PWR->CSR, WakeUpPinx);
346 * @brief Disables the Wake-up PINx functionality.
347 * @param WakeUpPinx: Specifies the Power Wake-Up pin to disable.
348 * This parameter can be one of the following values:
349 * @arg PWR_WAKEUP_PIN1
350 * @arg PWR_WAKEUP_PIN2 available only on STM32F410xx/STM32F446xx/STM32F412Zx/STM32F412Rx/STM32F412Vx/STM32F412Cx devices
351 * @arg PWR_WAKEUP_PIN3 available only on STM32F410xx/STM32F412Zx/STM32F412Rx/STM32F412Vx/STM32F412Cx devices
352 * @retval None
354 void HAL_PWR_DisableWakeUpPin(uint32_t WakeUpPinx)
356 /* Check the parameter */
357 assert_param(IS_PWR_WAKEUP_PIN(WakeUpPinx));
359 /* Disable the wake up pin */
360 CLEAR_BIT(PWR->CSR, WakeUpPinx);
364 * @brief Enters Sleep mode.
366 * @note In Sleep mode, all I/O pins keep the same state as in Run mode.
368 * @note In Sleep mode, the systick is stopped to avoid exit from this mode with
369 * systick interrupt when used as time base for Timeout
371 * @param Regulator: Specifies the regulator state in SLEEP mode.
372 * This parameter can be one of the following values:
373 * @arg PWR_MAINREGULATOR_ON: SLEEP mode with regulator ON
374 * @arg PWR_LOWPOWERREGULATOR_ON: SLEEP mode with low power regulator ON
375 * @note This parameter is not used for the STM32F4 family and is kept as parameter
376 * just to maintain compatibility with the lower power families.
377 * @param SLEEPEntry: Specifies if SLEEP mode in entered with WFI or WFE instruction.
378 * This parameter can be one of the following values:
379 * @arg PWR_SLEEPENTRY_WFI: enter SLEEP mode with WFI instruction
380 * @arg PWR_SLEEPENTRY_WFE: enter SLEEP mode with WFE instruction
381 * @retval None
383 void HAL_PWR_EnterSLEEPMode(uint32_t Regulator, uint8_t SLEEPEntry)
385 /* Check the parameters */
386 assert_param(IS_PWR_REGULATOR(Regulator));
387 assert_param(IS_PWR_SLEEP_ENTRY(SLEEPEntry));
389 /* Clear SLEEPDEEP bit of Cortex System Control Register */
390 CLEAR_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SLEEPDEEP_Msk));
392 /* Select SLEEP mode entry -------------------------------------------------*/
393 if(SLEEPEntry == PWR_SLEEPENTRY_WFI)
395 /* Request Wait For Interrupt */
396 __WFI();
398 else
400 /* Request Wait For Event */
401 __SEV();
402 __WFE();
403 __WFE();
408 * @brief Enters Stop mode.
409 * @note In Stop mode, all I/O pins keep the same state as in Run mode.
410 * @note When exiting Stop mode by issuing an interrupt or a wake-up event,
411 * the HSI RC oscillator is selected as system clock.
412 * @note When the voltage regulator operates in low power mode, an additional
413 * startup delay is incurred when waking up from Stop mode.
414 * By keeping the internal regulator ON during Stop mode, the consumption
415 * is higher although the startup time is reduced.
416 * @param Regulator: Specifies the regulator state in Stop mode.
417 * This parameter can be one of the following values:
418 * @arg PWR_MAINREGULATOR_ON: Stop mode with regulator ON
419 * @arg PWR_LOWPOWERREGULATOR_ON: Stop mode with low power regulator ON
420 * @param STOPEntry: Specifies if Stop mode in entered with WFI or WFE instruction.
421 * This parameter can be one of the following values:
422 * @arg PWR_STOPENTRY_WFI: Enter Stop mode with WFI instruction
423 * @arg PWR_STOPENTRY_WFE: Enter Stop mode with WFE instruction
424 * @retval None
426 void HAL_PWR_EnterSTOPMode(uint32_t Regulator, uint8_t STOPEntry)
428 /* Check the parameters */
429 assert_param(IS_PWR_REGULATOR(Regulator));
430 assert_param(IS_PWR_STOP_ENTRY(STOPEntry));
432 /* Select the regulator state in Stop mode: Set PDDS and LPDS bits according to PWR_Regulator value */
433 MODIFY_REG(PWR->CR, (PWR_CR_PDDS | PWR_CR_LPDS), Regulator);
435 /* Set SLEEPDEEP bit of Cortex System Control Register */
436 SET_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SLEEPDEEP_Msk));
438 /* Select Stop mode entry --------------------------------------------------*/
439 if(STOPEntry == PWR_STOPENTRY_WFI)
441 /* Request Wait For Interrupt */
442 __WFI();
444 else
446 /* Request Wait For Event */
447 __SEV();
448 __WFE();
449 __WFE();
451 /* Reset SLEEPDEEP bit of Cortex System Control Register */
452 CLEAR_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SLEEPDEEP_Msk));
456 * @brief Enters Standby mode.
457 * @note In Standby mode, all I/O pins are high impedance except for:
458 * - Reset pad (still available)
459 * - RTC_AF1 pin (PC13) if configured for tamper, time-stamp, RTC
460 * Alarm out, or RTC clock calibration out.
461 * - RTC_AF2 pin (PI8) if configured for tamper or time-stamp.
462 * - WKUP pin 1 (PA0) if enabled.
463 * @retval None
465 void HAL_PWR_EnterSTANDBYMode(void)
467 /* Select Standby mode */
468 SET_BIT(PWR->CR, PWR_CR_PDDS);
470 /* Set SLEEPDEEP bit of Cortex System Control Register */
471 SET_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SLEEPDEEP_Msk));
473 /* This option is used to ensure that store operations are completed */
474 #if defined ( __CC_ARM)
475 __force_stores();
476 #endif
477 /* Request Wait For Interrupt */
478 __WFI();
482 * @brief This function handles the PWR PVD interrupt request.
483 * @note This API should be called under the PVD_IRQHandler().
484 * @retval None
486 void HAL_PWR_PVD_IRQHandler(void)
488 /* Check PWR Exti flag */
489 if(__HAL_PWR_PVD_EXTI_GET_FLAG() != RESET)
491 /* PWR PVD interrupt user callback */
492 HAL_PWR_PVDCallback();
494 /* Clear PWR Exti pending bit */
495 __HAL_PWR_PVD_EXTI_CLEAR_FLAG();
500 * @brief PWR PVD interrupt callback
501 * @retval None
503 __weak void HAL_PWR_PVDCallback(void)
505 /* NOTE : This function Should not be modified, when the callback is needed,
506 the HAL_PWR_PVDCallback could be implemented in the user file
511 * @brief Indicates Sleep-On-Exit when returning from Handler mode to Thread mode.
512 * @note Set SLEEPONEXIT bit of SCR register. When this bit is set, the processor
513 * re-enters SLEEP mode when an interruption handling is over.
514 * Setting this bit is useful when the processor is expected to run only on
515 * interruptions handling.
516 * @retval None
518 void HAL_PWR_EnableSleepOnExit(void)
520 /* Set SLEEPONEXIT bit of Cortex System Control Register */
521 SET_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SLEEPONEXIT_Msk));
525 * @brief Disables Sleep-On-Exit feature when returning from Handler mode to Thread mode.
526 * @note Clears SLEEPONEXIT bit of SCR register. When this bit is set, the processor
527 * re-enters SLEEP mode when an interruption handling is over.
528 * @retval None
530 void HAL_PWR_DisableSleepOnExit(void)
532 /* Clear SLEEPONEXIT bit of Cortex System Control Register */
533 CLEAR_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SLEEPONEXIT_Msk));
537 * @brief Enables CORTEX M4 SEVONPEND bit.
538 * @note Sets SEVONPEND bit of SCR register. When this bit is set, this causes
539 * WFE to wake up when an interrupt moves from inactive to pended.
540 * @retval None
542 void HAL_PWR_EnableSEVOnPend(void)
544 /* Set SEVONPEND bit of Cortex System Control Register */
545 SET_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SEVONPEND_Msk));
549 * @brief Disables CORTEX M4 SEVONPEND bit.
550 * @note Clears SEVONPEND bit of SCR register. When this bit is set, this causes
551 * WFE to wake up when an interrupt moves from inactive to pended.
552 * @retval None
554 void HAL_PWR_DisableSEVOnPend(void)
556 /* Clear SEVONPEND bit of Cortex System Control Register */
557 CLEAR_BIT(SCB->SCR, ((uint32_t)SCB_SCR_SEVONPEND_Msk));
561 * @}
565 * @}
568 #endif /* HAL_PWR_MODULE_ENABLED */
570 * @}
574 * @}
577 /************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/