1. SysTick介绍

1.1 什么是SysTick

SysTick,既滴答定时器,是内核中的一个特殊定时器,用于提供系统级的定时服务。该定时器是一个24的递减计数器,具有自动重装载值的功能。当计数器到达自动重装载值的时候,它会自动重新加载并开始新的计数周期。

SysTick定时器的主要功能包括实现简单的延时,生成定时中断以及进行精准定时和周期定时等。此外,SysTick定时器还可以被用作其他目的,例如,作为操作系统的时基(如FreeRTOS),软件看门狗等系统调度操作,在STM32中SysTick通常以HCLK(APB时钟)或者HCLK/8作为运行时钟。

1.2 SysTick工作原理

• 在使用Systick定时器进行延时操作时,可以设定初值并使能滴答定时器后,每经过一个系统时钟周期,计数值就减1。当计数到0时,Systick计数器会自动重装初值并继续计数,同时内部的COUNTFLAG标志会置位,触发中断(如果中断使能)。这样,可以在中断处理函数中实现特定的延时逻辑。

2. SysTick寄存器

2.1 SysTick控制及状态寄存器(CTRL)

这个寄存器主要用于使能时钟和中断,选择时钟来源,以及提供定时器当前状态查询功能(COUNTFLAG)。

控制位有COUNTFLAG,CLKSOURCE,TICKINT,ENABLE等。

2.2 SysTick重装载值寄存器(LOAD)

这个寄存器主要用于设置SysTick定时器的重装载值,即定时器计数值的初始化

当计数器递减到0的时候,如果使能自动重装载值的功能,则计数器的值会被自动设置成这个寄存器的值,从而进行下一个计数周期。

控制位有RELOAD,如图:

2.3 SysTick当前数值寄存器(VAL)

这个寄存器的主要作用是读取或者写入SysTick定时器的值

当读取的时候,返回当前计数器的剩余值,当写入时,则清零计数器的值,并且还会清除CTRL寄存器里面COUNTFLAG标志。

控制位是CURRENT,如图:

3. 手写延时函数

• 在delay.c文件中,只要会了us级别延时,后面的ms和s级延时,按照单位换算就行了,HAL库已经有了HAL_Delay这个延时函数,为什么还要重写一个?因为HAL库这个只能提供ms级别的延时,而重写这个能提供us级别延时。

void delay_us(uint32_t nus)
{
    uint32_t temp;
    SysTick->LOAD = 72 * nus;                           /* 设置定时器重装值 */
    SysTick->VAL = 0x00;                                /* 清空当前计数值 */
    SysTick->CTRL |= 1 << 2;                            /* 设置分频系数为1分频 */
    SysTick->CTRL |= 1 << 0;                            /* 启动定时器 */
    do
    {
        temp = SysTick->CTRL;
    } while ((temp & 0x01) && !(temp & (1 << 16)));     /* 等待计数到0 */
    SysTick->CTRL &= ~(1 << 0);                         /* 关闭定时器 */
}

4. 手写带操作系统延时函数,上述已经有了一个us级别延时,为什么还要写带操作系统的延时,因为上述的延时是需要频繁打开关闭计数器

4.1 具体流程,如下图:

4.2 代码

/*void delay_us(uint32_t nus){//这个不用关闭计数器,
    uint32_t ticks = nus * 72;//一共要计的多少个数
    uint32_t tcnt = 0;//已经计了多少个数
    uint32_t told = SysTick->VAL;    //上次循环VAL的值
    uint32_t tnow = SysTick->VAL;//当前VAL的值
    while(1){//递减计数器
        tnow = SysTick->VAL;//获得当前VAL值
        if(tnow != told){
           if(tnow < told){
              tcnt += (told - tnow);
           }else {
              tcnt += SysTick->LOAD - (tnow - told);
           }
        }
        told = tnow;
        if(tcnt >= ticks){
            break;
        }
    }

}*/

void delay_us(uint32_t nus){//所有的配置都在HAL_InitTick这里配置好了
    uint32_t ticks = nus * 72;//一共要计的多少个数
    uint32_t tcnt = 0;//已经计了多少个数
    uint32_t told = SysTick->VAL;    //上次循环VAL的值
    uint32_t tnow ;//当前VAL的值
    uint32_t reload = SysTick->LOAD;//重装载值
    while(1){
        tnow = SysTick->VAL;//获得当前VAL值
        if(tnow != told){
           if(tnow < told){
              tcnt += (told - tnow);
           }else {
              tcnt += reload - (tnow - told);
           }
           
           told = tnow;
           if(tcnt >= ticks){
                 break;
           }   
        }
        
    }

}

Logo

openvela 操作系统专为 AIoT 领域量身定制,以轻量化、标准兼容、安全性和高度可扩展性为核心特点。openvela 以其卓越的技术优势,已成为众多物联网设备和 AI 硬件的技术首选,涵盖了智能手表、运动手环、智能音箱、耳机、智能家居设备以及机器人等多个领域。

更多推荐