Cortex-M0+处理器中内置有一个SysTick定时器,通常用于嵌入式操作系统的多任务切换,在不使用操作系统的应用中,亦可作为其它用途,如定时、计时或者为需要周期性执行的任务提供中断源。
1.SysTick工作原理
SysTick定时器内部含有一个24位的递减计数器,当计数减至0时,会从SysTick的重装载寄存器中取值作为计数器的初始值,同时可以选择在这个时候产生中断(异常号:15)。例如设置重装载寄存器为100,那么当计数减为0时,就会重新复位为100继续递减计数。
它的特点是:
• 24 位递减计数器
• 自动重装载能力
• 当计数器达到 0 时产生可屏蔽的系统中断
2.SysTick寄存器介绍
在core_cm0plus.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;
IOM uint32_t CTRL控制寄存器:
第0位:ENABLE,SysTick使能位(0:关闭SysTick功能,1:开启SysTick功能);
第1位:TICKINT,SysTick中断使能位(0:关闭SysTick中断,1:开启SysTick中断);
第2位:CLKSOURCE,SysTick时钟选择(1:使用HCLK,0:使用参考时钟频率);
第3为:COUNTFLAG,SysTick计数比较标志,如果在上次读取本寄存器后,SysTick已经数到0了,则该位为1,如果读取该位,该位自动清零。
__IOM uint32_t LOAD重载寄存器:
24位的寄存器,最大计数0xFFFFFF。当SysTick计数器递减至0时,重载寄存器中的值就会被重装载,继续开始递减。
__IOM uint32_t VAL当前值寄存器:
24位的寄存器,读取时返回当前计数器的计数值,写任何值都会使之清零,同时还会清除SysTick 控制寄存器中的COUNTFLAG 标志。
__IM uint32_t CALIB校准值寄存器:
只读寄存器,主要存放10mS校准值,该值和MCU相关。
3.操控SysTick定时器
在MDK开发环境中,我们不必要非得去操作每一个寄存器,可以通过调用CW函数库中的函数来进行相关的操作。
void InitTick(uint32_t HclkFreq)
初始化SysTick滴答定时器,带入的参数为HCLK的频率,如HCLK为24MHz,则带入参数为24000000。该函数会调用uint32_t SysTick_Config(uint32_t ticks)函数完成SysTick定时器的相关配置并启动。SysTick默认为1mS定时器,如果需要修改定时周期,则需要修改uint32_t SysTick_Config(uint32_t ticks)函数中重装载值寄存器配置。
__STATIC_INLINE uint32_t SysTick_Config(uint32_t ticks)
初始化系统计时器及其中断,并启动。
(1) SysTick->LOAD = (uint32_t)(ticks - 1UL);
设置SysTick重装载值
(2)NVIC_SetPriority(SysTick_IRQn,
(1UL << __NVIC_PRIO_BITS) - 1UL);
设置SysTick定时器中断优先级
(3) Tick->VAL = 0UL;
加载SysTick计数器值
(4) SysTick->CTRL = SysTick_CTRL_CLKSOURCE_Msk |
SysTick_CTRL_TICKINT_Msk |
SysTick_CTRL_ENABLE_Msk;
设置SysTick IRQ中断使能,并开启SysTick定时器
4.Systick延时功能实现
对于mS级及以上的延时,可在完成SysTick定时器初始化后,通过SysTickDelay(uint32_t Delay)函数来实现,该函数的形参为需要延时的mS数。
对于uS级延时,一般通过调整__NOP 空指令数量来实现,不建议用SysTick定时器来实现,主要原因是M0+系统中固有的中断响应时间(压栈和出栈)、中断处理时间等会影响uS延时精度。
int main(void)
{
__RCC_GPIOC_CLK_ENABLE(); //设置HCLK为24MHz
InitTick( 24000000 ); //初始化SysTick为1mS定时器
GPIO_InitTypeDef GPIO_InitStructure1 = {0} ; //初始化对应GPIO口
GPIO_InitStructure1.Pins = GPIO_PIN_3 ;
GPIO_InitStructure1.IT = GPIO_IT_NONE;
GPIO_InitStructure1.Mode = GPIO_MODE_OUTPUT_PP;
GPIO_Init(CW_GPIOC, &GPIO_InitStructure1);
PC03_SETHIGH(); // LED灯置高电平,低电平有效
while(1)
{
PC03_TOG(); //PC03口电平反转
SysTickDelay (100);//延时100mS
PC03_TOG();//PC03口电平再次反转
SysTickDelay (100);//延时100mS
}
}