STM32系统Tick定时器的初始化和配置

本文详细介绍了如何在系统时钟为72MHz的情况下配置SysTick定时器,以实现1ms周期中断,并通过计算和示例代码展示了SysTick_Config函数的参数设置。适合深入理解Cortex-M0定时器配置。
摘要由CSDN通过智能技术生成

1.首先确定SystemCoreClock系统时钟

 系统时钟需要根据具体的系统时钟配置来确定,这里采用SystemCoreClock = 72MHZ来说明。至于系统时钟的计算和配置,后面单独介绍说明一下吧。

2. 配置SysTick定时器

SysTick_Config(SystemCoreClock/OS_TICKS_PER_SEC);//初始化并使能SysTick定时器

假设我们想配置SysTick定时器中断间隔周期为1ms,即每间隔1ms定时中断来一次。

可定义一个宏定义

#define OS_TICKS_PER_SEC       1000    /* Set the number of ticks in one second  

即每1S SysTick定时器中断触发1000次,下面就需要计算一个每次中断触发间隔多少个时钟周期了。

SystemCoreClock 时钟频率是70000000Hz(72Mhz),那边一个时钟周期T=1/F = 1/72000000(S),  1(S)/T=F=72000000,即1S有72000000个时钟周期

(这里说的可能比较绕,方便部分群体理解吧~~)

下面来说一下SysTick_Config的入参,入参的定义可以见下面代码,@param   ticks   number of ticks between two interrupts   即SysTick每两次中断的间隔时间->间隔多少个ticks系统时钟周期数量, 可由SystemCoreClock /OS_TICKS_PER_SEC计算得到。至于SysTick_Config接口内部实现可见下面代码注释。  

上述说明如有不正确的地方,望指正~

/**
 * @brief  Initialize and start the SysTick counter and its interrupt.
 *
 * @param   ticks   number of ticks between two interrupts
 * @return  1 = failed, 0 = successful
 *
 * Initialise the system tick timer and its interrupt and start the
 * system tick timer / counter in free running mode to generate 
 * periodical interrupts.
 */
static __INLINE uint32_t SysTick_Config(uint32_t ticks)
{ 
  if (ticks > SysTick_LOAD_RELOAD_Msk)  return (1);            /* Reload value impossible */
                                                               
  SysTick->LOAD  = (ticks & SysTick_LOAD_RELOAD_Msk) - 1;      /* set reload register */
  NVIC_SetPriority (SysTick_IRQn, (1<<__NVIC_PRIO_BITS) - 1);  /* set Priority for Cortex-M0 System Interrupts */
  SysTick->VAL   = 0;                                          /* Load the SysTick Counter Value */
  SysTick->CTRL  = SysTick_CTRL_CLKSOURCE_Msk | 
                   SysTick_CTRL_TICKINT_Msk   | 
                   SysTick_CTRL_ENABLE_Msk;                    /* Enable SysTick IRQ and SysTick Timer */
  return (0);                                                  /* Function successful */
}

 

以下是一个简单的软件定时器封装代码示例,适用于STM32: 首先,定义一个定时器结构体,其中包含定时器 ID、定时器周期、定时器回调函数等信息: ``` typedef struct { uint8_t id; // 定时器 ID uint32_t period; // 定时器周期,单位为毫秒 void (*callback)(void); // 定时器回调函数 uint32_t tick; // 定时器计数器 } Timer_t; ``` 接下来,定义一个定时器数组,并初始化为全 0: ``` #define MAX_TIMER_NUM 10 // 最大定时器数量 Timer_t Timer_List[MAX_TIMER_NUM] = {0}; ``` 然后,实现一个定时器初始化函数,用于配置定时器时钟和定时器中断: ``` void Timer_Init(void) { // 配置定时器时钟为 1ms SysTick_Config(SystemCoreClock / 1000); } ``` 接着,实现一个定时器注册函数,用于向定时器数组中添加定时器: ``` uint8_t Timer_Register(uint32_t period, void (*callback)(void)) { uint8_t i; for (i = 0; i < MAX_TIMER_NUM; i++) { if (Timer_List[i].id == 0) { // 如果该定时器未被占用 Timer_List[i].id = i + 1; Timer_List[i].period = period; Timer_List[i].callback = callback; Timer_List[i].tick = 0; return Timer_List[i].id; // 返回定时器 ID } } return 0; // 返回 0 表示注册失败 } ``` 接着,实现一个定时器删除函数,用于从定时器数组中移除定时器: ``` void Timer_Delete(uint8_t id) { uint8_t i; for (i = 0; i < MAX_TIMER_NUM; i++) { if (Timer_List[i].id == id) { // 如果找到该定时器 Timer_List[i].id = 0; // 将该定时器 ID 置为 0,表示未被占用 Timer_List[i].period = 0; Timer_List[i].callback = NULL; Timer_List[i].tick = 0; break; } } } ``` 最后,实现一个定时器服务函数,在 SysTick 中断中调用该函数,用于判断定时器是否到期,如果到期则执行定时器回调函数: ``` void Timer_Service(void) { uint8_t i; for (i = 0; i < MAX_TIMER_NUM; i++) { if (Timer_List[i].id != 0) { // 如果该定时器已被占用 Timer_List[i].tick++; // 定时器计数器加 1 if (Timer_List[i].tick >= Timer_List[i].period) { // 如果定时器到期 Timer_List[i].tick = 0; // 重置定时器计数器 Timer_List[i].callback(); // 执行定时器回调函数 } } } } ``` 在主函数中调用定时器初始化函数,然后在 SysTick 中断中调用定时器服务函数即可。 以上是一个简单的软件定时器封装代码示例,可以根据实际需求进行修和扩展。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值