/** ****************************************************************************** * @file stm32h7rsxx_hal_timebase_tim_template.c * @author MCD Application Team * @brief Template for HAL time base based on the peripheral hardware TIM6. * * This file override the native HAL time base functions (defined as weak) * the TIM time base: * + Initializes the TIM6 peripheral to generate a Period elapsed Event each 1ms * + HAL_IncTick is called inside HAL_TIM_PeriodElapsedCallback ie each 1ms * ****************************************************************************** * @attention * * Copyright (c) 2022 STMicroelectronics. * All rights reserved. * * This software is licensed under terms that can be found in the LICENSE file * in the root directory of this software component. * If no LICENSE file comes with this software, it is provided AS-IS. * ****************************************************************************** @verbatim ============================================================================== ##### How to use this driver ##### ============================================================================== [..] This file must be copied to the application folder and modified as follows: (#) Rename it to 'stm32h7rsxx_hal_timebase_tim.c' (#) Add this file and the TIM HAL driver files to your project and make sure HAL_TIM_MODULE_ENABLED is defined in stm32h7rsxx_hal_conf.h [..] (@) The application needs to ensure that the time base is always set to 1 millisecond to have correct HAL operation. @endverbatim */ /* Includes ------------------------------------------------------------------*/ #include "stm32h7rsxx_hal.h" /** @addtogroup STM32H7RSxx_HAL_Driver * @{ */ /** @addtogroup HAL_TimeBase * @{ */ /* Private typedef -----------------------------------------------------------*/ /* Private define ------------------------------------------------------------*/ #define TIM_CNT_FREQ 500000U /* Timer counter frequency : 500 kHz */ #define TIM_FREQ 1000U /* Timer frequency : 1 kHz => to have 1 ms interrupt */ /* Private macro -------------------------------------------------------------*/ /* Private variables ---------------------------------------------------------*/ static TIM_HandleTypeDef TimHandle; /* Private function prototypes -----------------------------------------------*/ void TIM6_IRQHandler(void); #if (USE_HAL_TIM_REGISTER_CALLBACKS == 1U) void TimeBase_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim); #endif /* USE_HAL_TIM_REGISTER_CALLBACKS */ /* Private functions ---------------------------------------------------------*/ /** * @brief This function configures the TIM6 as a time base source. * The time source is configured to have 1ms time base with a dedicated * Tick interrupt priority. * @note This function is called automatically at the beginning of program after * reset by HAL_Init() or at any time when clock is configured, by HAL_RCC_ClockConfig(). * @param TickPriority Tick interrupt priority. * @retval HAL status */ HAL_StatusTypeDef HAL_InitTick(uint32_t TickPriority) { RCC_ClkInitTypeDef clkconfig; uint32_t uwTimclock; uint32_t uwAPB1Prescaler; uint32_t uwPrescalerValue; uint32_t pFLatency; HAL_StatusTypeDef Status; /* Enable TIM6 clock */ __HAL_RCC_TIM6_CLK_ENABLE(); /* Get clock configuration */ HAL_RCC_GetClockConfig(&clkconfig, &pFLatency); /* Get APB1 prescaler */ uwAPB1Prescaler = clkconfig.APB1CLKDivider; /* Compute TIM6 clock */ if (uwAPB1Prescaler == RCC_APB1_DIV1) { uwTimclock = HAL_RCC_GetPCLK1Freq(); } else if (uwAPB1Prescaler == RCC_APB1_DIV2) { uwTimclock = 2UL * HAL_RCC_GetPCLK1Freq(); } else { if (__HAL_RCC_GET_TIMCLKPRESCALER() == RCC_TIMPRES_DISABLE) { uwTimclock = 2UL * HAL_RCC_GetPCLK1Freq(); } else { uwTimclock = 4UL * HAL_RCC_GetPCLK1Freq(); } } /* Compute the prescaler value to have TIM6 counter clock equal to TIM_CNT_FREQ */ uwPrescalerValue = (uint32_t)((uwTimclock / TIM_CNT_FREQ) - 1U); /* Initialize TIM6 */ TimHandle.Instance = TIM6; /* Initialize TIMx peripheral as follow: + Period = [uwTickFreq * (TIM_CNT_FREQ/TIM_FREQ) - 1]. to have a (uwTickFreq/TIM_FREQ) s time base. + Prescaler = (uwTimclock/TIM_CNT_FREQ - 1) to have a TIM_CNT_FREQ counter clock. + ClockDivision = 0 + Counter direction = Up */ TimHandle.Init.Period = ((uint32_t)uwTickFreq * (TIM_CNT_FREQ / TIM_FREQ)) - 1U; TimHandle.Init.Prescaler = uwPrescalerValue; TimHandle.Init.ClockDivision = 0; TimHandle.Init.CounterMode = TIM_COUNTERMODE_UP; TimHandle.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE; Status = HAL_TIM_Base_Init(&TimHandle); if (Status == HAL_OK) { /* Start the TIM time Base generation in interrupt mode */ Status = HAL_TIM_Base_Start_IT(&TimHandle); if (Status == HAL_OK) { if (TickPriority < (1UL << __NVIC_PRIO_BITS)) { /* Configure the TIM6 global Interrupt priority */ HAL_NVIC_SetPriority(TIM6_IRQn, TickPriority, 0); /* Enable the TIM6 global Interrupt */ HAL_NVIC_EnableIRQ(TIM6_IRQn); uwTickPrio = TickPriority; } else { Status = HAL_ERROR; } } } #if (USE_HAL_TIM_REGISTER_CALLBACKS == 1U) HAL_TIM_RegisterCallback(&TimHandle, HAL_TIM_PERIOD_ELAPSED_CB_ID, TimeBase_TIM_PeriodElapsedCallback); #endif /* USE_HAL_TIM_REGISTER_CALLBACKS */ /* Return function status */ return Status; } /** * @brief Suspend Tick increment. * @note Disable the tick increment by disabling TIM6 update interrupt. * @retval None */ void HAL_SuspendTick(void) { /* Disable TIM6 update interrupt */ __HAL_TIM_DISABLE_IT(&TimHandle, TIM_IT_UPDATE); } /** * @brief Resume Tick increment. * @note Enable the tick increment by enabling TIM6 update interrupt. * @retval None */ void HAL_ResumeTick(void) { /* Enable TIM6 update interrupt */ __HAL_TIM_ENABLE_IT(&TimHandle, TIM_IT_UPDATE); } /** * @brief Period elapsed callback in non blocking mode * @note This function is called when TIM6 interrupt took place, inside * HAL_TIM6_IRQHandler(). It makes a direct call to HAL_IncTick() to increment * a global variable "uwTick" used as application time base. * @param htim TIM handle * @retval None */ #if (USE_HAL_TIM_REGISTER_CALLBACKS == 1U) void TimeBase_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim) #else void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim) #endif /* USE_HAL_TIM_REGISTER_CALLBACKS */ { /* Prevent unused argument(s) compilation warning */ UNUSED(htim); HAL_IncTick(); } /** * @brief This function handles TIM6 interrupt request. * @retval None */ void TIM6_IRQHandler(void) { HAL_TIM_IRQHandler(&TimHandle); } /** * @} */ /** * @} */