STM32完全学习——使用SysTick精确延时(阻塞式)

一、SysTick相关寄存器

首先关于SysTick寄存器的数据在下面这个文件里面可以找到,平时那个数据手册是没有的。其次我这边使用的开发板是F407的开发板,关于这个寄存器的数据都是来自下面这个文件的截图,一般只会用到这3个寄存器。

二、使用标准库对SysTick进行初始化

首先我们知道当你使用标准库的时候,他已经默认帮你将系统时钟那设置成了168MHz,因此SysTick时钟为168MHz,在初始化里面我选择了8分频,这个样延时的时间可以变得长一点。这里的SCLK的值为168,也就是系统时钟。

cpp 复制代码
void systick_init(uint8_t SCLK)
{
    SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);
    f_us = (SCLK/8);      //定时1微秒所需要的初始值  这个是两个全局变量
    f_ms = f_us * 1000;   //定时1毫秒所需要的初始值
}

三、编写精确延时函数

cpp 复制代码
//定时时间范围最大798915个微秒
void delayus(uint32_t us)
{
    uint32_t temp = 0;
    SysTick->LOAD = f_us*us;   
    SysTick->VAL   = 0x00;   //清除计数器
    SysTick->CTRL |= 0x01;   //打开计数器
    do 
    {
        temp = (SysTick->CTRL) & (1 << 16);  //判断计数是否到了0
    }while((!temp));
    SysTick->VAL   = 0x00;  //清除计数器
    SysTick->CTRL &= ~0x01; //关闭计数器
}
//定时时间范围最大798个毫秒秒
void delaynms(uint32_t ms)
{
    uint32_t temp = 0;
    SysTick->LOAD = f_ms*ms;
    SysTick->VAL   = 0x00; 
    SysTick->CTRL |= 0x01;
    do 
    {
        temp = (SysTick->CTRL) & (1 << 16);
    }while((!temp));
    SysTick->VAL   = 0x00; 
    SysTick->CTRL &= ~0x01;

}


void delayms(uint32_t ms)
{
    uint32_t i = 0;
    for (i=0; i<=ms; i++)
    {
        delaynms(1);
    }
}

四、关于延时不准确的原因

刚开始我使用这个方法进行延时,确实能达到效果,但是我通过示波器来查看时,发现延时并不准确,而且误差有点大。那我就在想是不是系统时钟的问题,一看还真是,因为F407的标准库里面默认你外部HSE接的是25MHz的晶振,但是我发现我的开发板上面接的是8MHz的晶振,应该就是系统里面帮我初始化时钟的时候,没有真正的初始化为168MHz的时钟。通过查看标准库的代码,并作出了下面的修改就可以了。

相关推荐
Ronin-Lotus11 分钟前
嵌入式硬件篇---电感串并联
嵌入式硬件
rannn_11128 分钟前
【MySQL学习|黑马笔记|Day7】触发器和锁(全局锁、表级锁、行级锁、)
笔记·后端·学习·mysql
Wallace Zhang1 小时前
STM32 - Embedded IDE - GCC - 显著减少固件的体积
stm32·单片机·嵌入式硬件
喜欢吃燃面1 小时前
C++算法竞赛:位运算
开发语言·c++·学习·算法
传奇开心果编程1 小时前
【传奇开心果系列】Flet框架实现的家庭记账本示例自定义模板
python·学习·ui·前端框架·自动化
_Kayo_7 小时前
node.js 学习笔记3 HTTP
笔记·学习
CCCC131016310 小时前
嵌入式学习(day 28)线程
jvm·学习
星星火柴93611 小时前
关于“双指针法“的总结
数据结构·c++·笔记·学习·算法
小狗爱吃黄桃罐头11 小时前
正点原子【第四期】Linux之驱动开发篇学习笔记-1.1 Linux驱动开发与裸机开发的区别
linux·驱动开发·学习
fengfuyao98511 小时前
STM32如何定位HardFault错误,一种实用方法
stm32·单片机·嵌入式硬件