1.STM32-Systick滴答定时器
-
Systick定时器,是一个简单的定时器,对于ST的CM3,CM4,CM7内核芯片,都有Systick定时器。 -
Systick定时器常用来做延时,或者实时系统的心跳时钟。这样可以节省MCU资源,不用浪费一个定时器。比如UCOS中,分时复用,需要一个最小的时间戳,一般在STM32+UCOS系统中,都采用Systick做UCOS心跳时钟。 -
Systick定时器就是系统滴答定时器,一个24 位的倒计数定时器,计到0 时,将从RELOAD 寄存器中自动重装载定时初值。只要不把它在SysTick 控制及状态寄存器中的使能位清除,就永不停息,即使在睡眠模式下也能工作。 -
SysTick定时器被捆绑在NVIC中,用于产生SYSTICK异常(异常号:15)。 -
Systick中断的优先级也可以设置。 -
4个Systick寄存器 CTRL SysTick 控制和状态寄存器
LOAD SysTick 自动重装载除值寄存器
VAL SysTick 当前值寄存器
CALIB SysTick 校准值寄存器
可在core-core_cm7.h文件中找到
typedef struct
{
__IOM uint32_t CTRL; /*!< Offset: 0x000 (R/W) SysTick Control and Status Register */
__IOM uint32_t LOAD; /*!< Offset: 0x004 (R/W) SysTick Reload Value Register */
__IOM uint32_t VAL; /*!< Offset: 0x008 (R/W) SysTick Current Value Register */
__IM uint32_t CALIB; /*!< Offset: 0x00C (R/ ) SysTick Calibration Register */
} SysTick_Type;
SysTick 控制和状态寄存器- CTRL:
对于STM32,外部时钟源是 HCLK(AHB总线时钟)的1/8。内核时钟是 HCLK时钟。配置函数:HAL_SYSTICK_CLKSourceConfig();
SysTick重装载数值寄存器-LOAD
SysTick当前数值寄存器-VAL
HAL库中的Systick相关函数:
stm32f7xx_hal_cortex.c文件中:HAL_SYSTICK_CLKSourceConfig () ; //Systick时钟源选择
如果SysTick的时钟源自HCLK,假设外部晶振为25M,倍频到216MHZ,那么SysTick的时钟即为216MHZ,也就是SysTick的计数器VAL每减1,就代表时间过了1/216us。
void HAL_SYSTICK_CLKSourceConfig(uint32_t CLKSource)
{
assert_param(IS_SYSTICK_CLK_SOURCE(CLKSource));
if (CLKSource == SYSTICK_CLKSOURCE_HCLK)
{
SysTick->CTRL |= SYSTICK_CLKSOURCE_HCLK;
}
else
{
SysTick->CTRL &= ~SYSTICK_CLKSOURCE_HCLK;
}
}
可以看一下CLKSource可以有哪几种:找到IS_SYSTICK_CLK_SOURCE的定义,发现可以为SYSTICK_CLKSOURCE_HCLK和SYSTICK_CLKSOURCE_HCLK_DIV8,也就是不分频或者8分频。
#define IS_SYSTICK_CLK_SOURCE(SOURCE) (((SOURCE) == SYSTICK_CLKSOURCE_HCLK) || \
((SOURCE) == SYSTICK_CLKSOURCE_HCLK_DIV8))
core_cm7.h文件中:SysTick_Config (uint32_t ticks) //初始化systick,时钟为HCLK,并开启中断。
ticks:经过多少个systick周期发生一次中断。用来配置SysTick定时器经过多少个ticks发生一次中断。
__STATIC_INLINE uint32_t SysTick_Config(uint32_t ticks)
{
if ((ticks - 1UL) > SysTick_LOAD_RELOAD_Msk)
{
return (1UL);
}
SysTick->LOAD = (uint32_t)(ticks - 1UL);
NVIC_SetPriority (SysTick_IRQn, (1UL << __NVIC_PRIO_BITS) - 1UL);
SysTick->VAL = 0UL;
SysTick->CTRL = SysTick_CTRL_CLKSOURCE_Msk |
SysTick_CTRL_TICKINT_Msk |
SysTick_CTRL_ENABLE_Msk;
return (0UL);
}
首先看到SysTick_LOAD_RELOAD_Msk,由于Systick是一个24 位的倒计数定时器,所以值不能太大。
#define SysTick_LOAD_RELOAD_Msk (0xFFFFFFUL )
Systick中断服务函数:void SysTick_Handler (void);
2.delay函数
void delay_init(u8 SYSCLK)
{
#if SYSTEM_SUPPORT_OS
u32 reload;
#endif
HAL_SYSTICK_CLKSourceConfig(SYSTICK_CLKSOURCE_HCLK);
fac_us=SYSCLK;
#if SYSTEM_SUPPORT_OS
reload=SYSCLK;
reload*=1000000/delay_ostickspersec;
fac_ms=1000/delay_ostickspersec;
SysTick->CTRL|=SysTick_CTRL_TICKINT_Msk;
SysTick->LOAD=reload;
SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk;
#endif
}
在main中调用delay_init(216);//延时初始化。1微秒等于10的负6次方秒,1MHZ等于10^6赫兹。
fac_us=SYSCLK;系统时钟是216MHZ,fac_us=216意思是:systick运行1us,需要多少个systick周期。由于systick是216MHZ,所以需要216个周期。后面如果需要延时n个微秒,只需要n*fac_us即可。由于设置的SysTick频率为HCLK,所以调用delay_init函数中的参数SYSCLK就是216 。
delay_us函数思路:
在循环里检测当前值,如果当前值小于前一次的值,说明没有减到0,通过told-tnow就可以知道当前跑了几个周期。否则,说明已经溢出了,也就是说当前跑了reload-tnow+told个周期。最后如果时间超过/等于要延迟的时间,则退出。
void delay_us(u32 nus)
{
u32 ticks;
u32 told,tnow,tcnt=0;
u32 reload=SysTick->LOAD;
ticks=nus*fac_us;
delay_osschedlock();
told=SysTick->VAL;
while(1)
{
tnow=SysTick->VAL;
if(tnow!=told)
{
if(tnow<told)tcnt+=told-tnow;
else tcnt+=reload-tnow+told;
told=tnow;
if(tcnt>=ticks)break;
}
};
delay_osschedunlock();
}
|