blob: adfa087117c2b57054a8f670008ec849999d9e21 [file] [log] [blame]
Eya8a83a682020-01-29 12:42:18 +01001/**
2 ******************************************************************************
3 * @file stm32f7xx_ll_lptim.c
4 * @author MCD Application Team
5 * @brief LPTIM LL module driver.
6 ******************************************************************************
7 * @attention
8 *
rihab kouki27458ea2021-12-14 11:01:06 +01009 * Copyright (c) 2017 STMicroelectronics.
10 * All rights reserved.
Eya8a83a682020-01-29 12:42:18 +010011 *
rihab kouki27458ea2021-12-14 11:01:06 +010012 * This software is licensed under terms that can be found in the LICENSE file
13 * in the root directory of this software component.
14 * If no LICENSE file comes with this software, it is provided AS-IS.
Eya8a83a682020-01-29 12:42:18 +010015 *
16 ******************************************************************************
17 */
18#if defined(USE_FULL_LL_DRIVER)
19
20/* Includes ------------------------------------------------------------------*/
21#include "stm32f7xx_ll_lptim.h"
22#include "stm32f7xx_ll_bus.h"
Ali Labbene243e61a2020-03-12 16:07:28 +010023#include "stm32f7xx_ll_rcc.h"
24
Eya8a83a682020-01-29 12:42:18 +010025
26#ifdef USE_FULL_ASSERT
Ali Labbene243e61a2020-03-12 16:07:28 +010027#include "stm32_assert.h"
Eya8a83a682020-01-29 12:42:18 +010028#else
Ali Labbene243e61a2020-03-12 16:07:28 +010029#define assert_param(expr) ((void)0U)
rihab kouki27458ea2021-12-14 11:01:06 +010030#endif /* USE_FULL_ASSERT */
Eya8a83a682020-01-29 12:42:18 +010031
32/** @addtogroup STM32F7xx_LL_Driver
33 * @{
34 */
35
Ali Labbene243e61a2020-03-12 16:07:28 +010036#if defined (LPTIM1)
Eya8a83a682020-01-29 12:42:18 +010037
38/** @addtogroup LPTIM_LL
39 * @{
40 */
41
42/* Private types -------------------------------------------------------------*/
43/* Private variables ---------------------------------------------------------*/
44/* Private constants ---------------------------------------------------------*/
45/* Private macros ------------------------------------------------------------*/
46/** @addtogroup LPTIM_LL_Private_Macros
47 * @{
48 */
49#define IS_LL_LPTIM_CLOCK_SOURCE(__VALUE__) (((__VALUE__) == LL_LPTIM_CLK_SOURCE_INTERNAL) \
Eyabcfd32d2021-03-03 16:11:08 +010050 || ((__VALUE__) == LL_LPTIM_CLK_SOURCE_EXTERNAL))
Eya8a83a682020-01-29 12:42:18 +010051
52#define IS_LL_LPTIM_CLOCK_PRESCALER(__VALUE__) (((__VALUE__) == LL_LPTIM_PRESCALER_DIV1) \
Eyabcfd32d2021-03-03 16:11:08 +010053 || ((__VALUE__) == LL_LPTIM_PRESCALER_DIV2) \
54 || ((__VALUE__) == LL_LPTIM_PRESCALER_DIV4) \
55 || ((__VALUE__) == LL_LPTIM_PRESCALER_DIV8) \
56 || ((__VALUE__) == LL_LPTIM_PRESCALER_DIV16) \
57 || ((__VALUE__) == LL_LPTIM_PRESCALER_DIV32) \
58 || ((__VALUE__) == LL_LPTIM_PRESCALER_DIV64) \
59 || ((__VALUE__) == LL_LPTIM_PRESCALER_DIV128))
Eya8a83a682020-01-29 12:42:18 +010060
61#define IS_LL_LPTIM_WAVEFORM(__VALUE__) (((__VALUE__) == LL_LPTIM_OUTPUT_WAVEFORM_PWM) \
Eyabcfd32d2021-03-03 16:11:08 +010062 || ((__VALUE__) == LL_LPTIM_OUTPUT_WAVEFORM_SETONCE))
Eya8a83a682020-01-29 12:42:18 +010063
64#define IS_LL_LPTIM_OUTPUT_POLARITY(__VALUE__) (((__VALUE__) == LL_LPTIM_OUTPUT_POLARITY_REGULAR) \
Eyabcfd32d2021-03-03 16:11:08 +010065 || ((__VALUE__) == LL_LPTIM_OUTPUT_POLARITY_INVERSE))
Eya8a83a682020-01-29 12:42:18 +010066/**
67 * @}
68 */
69
70
71/* Private function prototypes -----------------------------------------------*/
Ali Labbene243e61a2020-03-12 16:07:28 +010072/* Private functions ---------------------------------------------------------*/
73/** @defgroup LPTIM_Private_Functions LPTIM Private Functions
74 * @{
75 */
76/**
77 * @}
78 */
Eya8a83a682020-01-29 12:42:18 +010079/* Exported functions --------------------------------------------------------*/
80/** @addtogroup LPTIM_LL_Exported_Functions
81 * @{
82 */
83
84/** @addtogroup LPTIM_LL_EF_Init
85 * @{
86 */
87
88/**
89 * @brief Set LPTIMx registers to their reset values.
90 * @param LPTIMx LP Timer instance
91 * @retval An ErrorStatus enumeration value:
92 * - SUCCESS: LPTIMx registers are de-initialized
93 * - ERROR: invalid LPTIMx instance
94 */
Tasnim2e3aac42024-05-27 13:47:18 +010095ErrorStatus LL_LPTIM_DeInit(const LPTIM_TypeDef *LPTIMx)
Eya8a83a682020-01-29 12:42:18 +010096{
97 ErrorStatus result = SUCCESS;
98
99 /* Check the parameters */
Ali Labbene243e61a2020-03-12 16:07:28 +0100100 assert_param(IS_LPTIM_INSTANCE(LPTIMx));
101
Eya8a83a682020-01-29 12:42:18 +0100102 if (LPTIMx == LPTIM1)
103 {
104 LL_APB1_GRP1_ForceReset(LL_APB1_GRP1_PERIPH_LPTIM1);
Ali Labbene243e61a2020-03-12 16:07:28 +0100105 LL_APB1_GRP1_ReleaseReset(LL_APB1_GRP1_PERIPH_LPTIM1);
Eya8a83a682020-01-29 12:42:18 +0100106 }
Eya8a83a682020-01-29 12:42:18 +0100107 else
108 {
109 result = ERROR;
110 }
Ali Labbene243e61a2020-03-12 16:07:28 +0100111
Eya8a83a682020-01-29 12:42:18 +0100112 return result;
113}
114
115/**
116 * @brief Set each fields of the LPTIM_InitStruct structure to its default
117 * value.
118 * @param LPTIM_InitStruct pointer to a @ref LL_LPTIM_InitTypeDef structure
119 * @retval None
120 */
Ali Labbene243e61a2020-03-12 16:07:28 +0100121void LL_LPTIM_StructInit(LL_LPTIM_InitTypeDef *LPTIM_InitStruct)
Eya8a83a682020-01-29 12:42:18 +0100122{
123 /* Set the default configuration */
124 LPTIM_InitStruct->ClockSource = LL_LPTIM_CLK_SOURCE_INTERNAL;
125 LPTIM_InitStruct->Prescaler = LL_LPTIM_PRESCALER_DIV1;
126 LPTIM_InitStruct->Waveform = LL_LPTIM_OUTPUT_WAVEFORM_PWM;
127 LPTIM_InitStruct->Polarity = LL_LPTIM_OUTPUT_POLARITY_REGULAR;
128}
129
130/**
131 * @brief Configure the LPTIMx peripheral according to the specified parameters.
132 * @note LL_LPTIM_Init can only be called when the LPTIM instance is disabled.
133 * @note LPTIMx can be disabled using unitary function @ref LL_LPTIM_Disable().
134 * @param LPTIMx LP Timer Instance
135 * @param LPTIM_InitStruct pointer to a @ref LL_LPTIM_InitTypeDef structure
136 * @retval An ErrorStatus enumeration value:
137 * - SUCCESS: LPTIMx instance has been initialized
138 * - ERROR: LPTIMx instance hasn't been initialized
139 */
Ali Labbene33773052022-04-01 12:01:25 +0100140ErrorStatus LL_LPTIM_Init(LPTIM_TypeDef *LPTIMx, const LL_LPTIM_InitTypeDef *LPTIM_InitStruct)
Eya8a83a682020-01-29 12:42:18 +0100141{
142 ErrorStatus result = SUCCESS;
Ali Labbene243e61a2020-03-12 16:07:28 +0100143 /* Check the parameters */
144 assert_param(IS_LPTIM_INSTANCE(LPTIMx));
145 assert_param(IS_LL_LPTIM_CLOCK_SOURCE(LPTIM_InitStruct->ClockSource));
146 assert_param(IS_LL_LPTIM_CLOCK_PRESCALER(LPTIM_InitStruct->Prescaler));
147 assert_param(IS_LL_LPTIM_WAVEFORM(LPTIM_InitStruct->Waveform));
148 assert_param(IS_LL_LPTIM_OUTPUT_POLARITY(LPTIM_InitStruct->Polarity));
149
150 /* The LPTIMx_CFGR register must only be modified when the LPTIM is disabled
Eya8a83a682020-01-29 12:42:18 +0100151 (ENABLE bit is reset to 0).
152 */
Ali Labbene243e61a2020-03-12 16:07:28 +0100153 if (LL_LPTIM_IsEnabled(LPTIMx) == 1UL)
Eya8a83a682020-01-29 12:42:18 +0100154 {
155 result = ERROR;
156 }
157 else
158 {
Ali Labbene243e61a2020-03-12 16:07:28 +0100159 /* Set CKSEL bitfield according to ClockSource value */
160 /* Set PRESC bitfield according to Prescaler value */
161 /* Set WAVE bitfield according to Waveform value */
162 /* Set WAVEPOL bitfield according to Polarity value */
163 MODIFY_REG(LPTIMx->CFGR,
164 (LPTIM_CFGR_CKSEL | LPTIM_CFGR_PRESC | LPTIM_CFGR_WAVE | LPTIM_CFGR_WAVPOL),
165 LPTIM_InitStruct->ClockSource | \
166 LPTIM_InitStruct->Prescaler | \
167 LPTIM_InitStruct->Waveform | \
168 LPTIM_InitStruct->Polarity);
Eya8a83a682020-01-29 12:42:18 +0100169 }
170
171 return result;
172}
173
174/**
Ali Labbene243e61a2020-03-12 16:07:28 +0100175 * @brief Disable the LPTIM instance
176 * @rmtoll CR ENABLE LL_LPTIM_Disable
177 * @param LPTIMx Low-Power Timer instance
178 * @note The following sequence is required to solve LPTIM disable HW limitation.
179 * Please check Errata Sheet ES0335 for more details under "MCU may remain
180 * stuck in LPTIM interrupt when entering Stop mode" section.
181 * @retval None
182 */
183void LL_LPTIM_Disable(LPTIM_TypeDef *LPTIMx)
184{
185 LL_RCC_ClocksTypeDef rcc_clock;
186 uint32_t tmpclksource = 0;
187 uint32_t tmpIER;
188 uint32_t tmpCFGR;
189 uint32_t tmpCMP;
190 uint32_t tmpARR;
rihab kouki27458ea2021-12-14 11:01:06 +0100191 uint32_t primask_bit;
Ali Labbene243e61a2020-03-12 16:07:28 +0100192
193 /* Check the parameters */
194 assert_param(IS_LPTIM_INSTANCE(LPTIMx));
195
rihab kouki27458ea2021-12-14 11:01:06 +0100196 /* Enter critical section */
197 primask_bit = __get_PRIMASK();
198 __set_PRIMASK(1) ;
Ali Labbene243e61a2020-03-12 16:07:28 +0100199
200 /********** Save LPTIM Config *********/
201 /* Save LPTIM source clock */
202 switch ((uint32_t)LPTIMx)
203 {
204 case LPTIM1_BASE:
205 tmpclksource = LL_RCC_GetLPTIMClockSource(LL_RCC_LPTIM1_CLKSOURCE);
206 break;
207 default:
208 break;
209 }
210
211 /* Save LPTIM configuration registers */
212 tmpIER = LPTIMx->IER;
213 tmpCFGR = LPTIMx->CFGR;
214 tmpCMP = LPTIMx->CMP;
215 tmpARR = LPTIMx->ARR;
216
217 /************* Reset LPTIM ************/
218 (void)LL_LPTIM_DeInit(LPTIMx);
219
220 /********* Restore LPTIM Config *******/
221 LL_RCC_GetSystemClocksFreq(&rcc_clock);
222
223 if ((tmpCMP != 0UL) || (tmpARR != 0UL))
224 {
225 /* Force LPTIM source kernel clock from APB */
226 switch ((uint32_t)LPTIMx)
227 {
228 case LPTIM1_BASE:
229 LL_RCC_SetLPTIMClockSource(LL_RCC_LPTIM1_CLKSOURCE_PCLK1);
230 break;
231 default:
232 break;
233 }
234
235 if (tmpCMP != 0UL)
236 {
237 /* Restore CMP and ARR registers (LPTIM should be enabled first) */
238 LPTIMx->CR |= LPTIM_CR_ENABLE;
239 LPTIMx->CMP = tmpCMP;
240
241 /* Polling on CMP write ok status after above restore operation */
242 do
243 {
244 rcc_clock.SYSCLK_Frequency--; /* Used for timeout */
Eyabcfd32d2021-03-03 16:11:08 +0100245 } while (((LL_LPTIM_IsActiveFlag_CMPOK(LPTIMx) != 1UL)) && ((rcc_clock.SYSCLK_Frequency) > 0UL));
Ali Labbene243e61a2020-03-12 16:07:28 +0100246
247 LL_LPTIM_ClearFlag_CMPOK(LPTIMx);
248 }
249
250 if (tmpARR != 0UL)
251 {
252 LPTIMx->CR |= LPTIM_CR_ENABLE;
253 LPTIMx->ARR = tmpARR;
254
255 LL_RCC_GetSystemClocksFreq(&rcc_clock);
256 /* Polling on ARR write ok status after above restore operation */
257 do
258 {
259 rcc_clock.SYSCLK_Frequency--; /* Used for timeout */
Tasnim2e3aac42024-05-27 13:47:18 +0100260 } while (((LL_LPTIM_IsActiveFlag_ARROK(LPTIMx) != 1UL)) && ((rcc_clock.SYSCLK_Frequency) > 0UL));
Ali Labbene243e61a2020-03-12 16:07:28 +0100261
262 LL_LPTIM_ClearFlag_ARROK(LPTIMx);
263 }
264
265
266 /* Restore LPTIM source kernel clock */
267 LL_RCC_SetLPTIMClockSource(tmpclksource);
268 }
269
270 /* Restore configuration registers (LPTIM should be disabled first) */
271 LPTIMx->CR &= ~(LPTIM_CR_ENABLE);
272 LPTIMx->IER = tmpIER;
273 LPTIMx->CFGR = tmpCFGR;
274
rihab kouki27458ea2021-12-14 11:01:06 +0100275 /* Exit critical section: restore previous priority mask */
276 __set_PRIMASK(primask_bit);
Ali Labbene243e61a2020-03-12 16:07:28 +0100277}
278
279/**
Eya8a83a682020-01-29 12:42:18 +0100280 * @}
281 */
282
283/**
284 * @}
285 */
286
287/**
288 * @}
289 */
290
Ali Labbene243e61a2020-03-12 16:07:28 +0100291#endif /* LPTIM1 */
Eya8a83a682020-01-29 12:42:18 +0100292
293/**
294 * @}
295 */
Ali Labbene243e61a2020-03-12 16:07:28 +0100296
Eya8a83a682020-01-29 12:42:18 +0100297#endif /* USE_FULL_LL_DRIVER */