嵌入式开发十九:SysTick—系统定时器

2024-06-20 14:28

本文主要是介绍嵌入式开发十九:SysTick—系统定时器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

         在前面实验中我们使用到的延时都是通过SysTick进行延时的。 我们知道,延时有两种方式:软件延时,即CPU 循环等待产生的,这个延时是不精确的。第二种就是滴答定时器延时,本篇博客就来介绍 STM32F4 内部 SysTick 系统定时器,通过一个简单的 LED 流水灯程序来讲述如何配置 SysTick 系统定时器实现精确延时。学习可以参考《STM32F3 与 F4 系列 Cortex M4 内核编程手册》 4.5 SysTick timer (STK) 章节或者参考库函数中 core_cm4.h 文件 。

目录

一、SysTick 定时器介绍

二、SysTick 定时器操作

2.1 SysTick 定时器寄存器

2.1.1 控制和状态寄存器:CTRL

2.1.2  重装载寄存器:LOAD

2.1.3 当前数值寄存器:VAL

2.1.4  校准数值寄存器:CALIB 

2.2 系统节拍定时器的工作原理

2.3 SysTick 定时器操作步骤

2.4  使用SysTick 定时器实现精准延时

2.4.1 实现1微秒延时

2.4.2 实现1毫秒延时

2.4.3 实现1秒延时

三、SysTick 定时实验

一、SysTick 定时器介绍

       SysTick 定时器也叫 SysTick 滴答定时器,它是 Cortex-M4 内核的一个外设, 被嵌入在 NVIC 中,用来产生SYSTICK异常(异常号:15)。它是一个 24 位向下递减的定时器,每计数一次所需时间为 1/SYSTICK,SYSTICK 是系统定时器时钟,它可以直接取自系统时钟,还可以通过系统时钟 8 分频后获取,本套程序中我们采用后者,即每计数一次所需时间为 1/(168/8)us,换句话说在 1us 的时间内会计数 21 次。当定时器计数到 0 时,将 从 LOAD 寄存器中自动重装定时器初值,重新向下递减计数,如此循环往复。如果开启 SysTick 中断的话,当定时器计数到 0,将产生一个中断信号。如下图所示,因此只要知道计数的次数就可以准确得到它的延时时间。 因为 SysTick 是属于 CM4 内核的外设,所以所有基于 CM4 内核的单片机都具有这个系统定时器,使得软件在 CM4 单片机中可以很容易的移植。系统定时器一般用于操作系统, 用于产生时基,维持操作系统的心跳。

如何计算延时时间?

       如果时钟源选择8分频后的即21MHZ,那么,1秒钟就会计数21000000次,(计数一次的时间就是:1/21000000),如此:如果想要定时1毫秒,就要计数21000次,定时1微秒,就要计数21次!

二、SysTick 定时器操作

        在 STM32F4 库函数中,并没有提供相应的 SysTick 定时器配置函数,我们要操作 SysTick 定时器就需要了解它的寄存器功能。其实 SysTick 定时器寄存器很 简单,只有 4 个,分别是 CTRL、LOAD、VAL、CALIB,在使用 SysTick 产生定时的时候, 只需要配置前三个寄存器,最后一个校准寄存器不需要使用。对应如下图所示:

2.1 SysTick 定时器寄存器

2.1.1 控制和状态寄存器:CTRL

CTRL 是 SysTick 定时器的控制及状态寄存器。其相应位功能如下:

注:CLKSOUTCE 位是用于选择 SysTick 定时器时钟来源:

  1. 如果该位为 1,表示其时钟是由系统时钟直接提供即 168M。
  2. 如果该位为 0,表示其时钟是由系统时钟八分频后提供即 168/8=21M。

2.1.2  重装载寄存器:LOAD

LOAD 是 SysTick 定时器的重装载数值寄存器。其相应位功能如下:

因为 STM32F4 的 SysTick 定时器是一个 24 位递减计数器,因此重装载寄存器中只使用到了低 24 位,即 bit0-bit23。当系统复位时,其值为 0。

2.1.3 当前数值寄存器:VAL

VAL 是 SysTick 定时器的当前数值寄存器。其相应位功能如下:

同样只有 bit0-bit24 有效,复位时值为 0。

2.1.4  校准数值寄存器:CALIB 

CALIB 是 SysTick 定时器的校准数值寄存器。其相应位功能如下:

此寄存器在定时实验中不需要使用,可以不用了解。

2.2 系统节拍定时器的工作原理

        当系统节拍定时器⼯作时,该定时器⾸先会从寄存器LOAD存储的值开始递减计数。当递减为0 后,寄存器CTRL的COUNTFLAG状态位会置1,同时会重装载寄存器LOAD预置的值。 当计数到0时,通过设置寄存器CTRL的TICKINT的值来产⽣异常(中断),或是⽆动作。

2.3 SysTick 定时器操作步骤

SysTick 定时器的操作可以分为 4 步:

  1. 设置 SysTick 定时器的时钟源。
  2. 设置 SysTick 定时器的重装初始值(如果要使用中断的话,就将中断使能打开)。
  3. 清零 SysTick 定时器当前计数器的值。
  4. 打开 SysTick 定时器。

2.4  使用SysTick 定时器实现精准延时

2.4.1 实现1微秒延时

void Sleep_us(uint32_t us)
{while(us--){SysTick ->CTRL = (1 << 0);   //定时器使能第0位置1SysTick ->CTRL &= ~(1<<2);   //选择时钟源:第2位置0,选择外部时钟源,由系统时钟八分频后提供即 168/8=21MSysTick ->CTRL &= ~(1<<1);   //延时时间到无动作:第1位置0SysTick ->VAL = 0x0;        //当前数值寄存器初值赋0SysTick ->LOAD = 21;       //重装载数值寄存器的值,定时1微秒,所以是21while(!(SysTick ->CTRL & (1<<16)));  //死循环等待计数值减到0SysTick ->CTRL = ~(1<<0);    //关闭定时器,第0位置0}}//复用上述函数实现延时1秒
void Sleep_s(uint32_t s)
{while(s--){Sleep_ms(1000);}
}

2.4.2 实现1毫秒延时

void Sleep_ms(uint32_t ms)
{while(ms--){SysTick ->CTRL = (1 << 0);   //定时器使能第0位置1SysTick ->CTRL &= ~(1<<2);   //选择时钟源:第2位置0,选择外部时钟源,由系统时钟八分频后提供即 168/8=21MSysTick ->CTRL &= ~(1<<1);   //延时时间到无动作:第1位置0SysTick ->VAL = 0x0;        //当前数值寄存器初值赋0SysTick ->LOAD = 21000;       //重装载数值寄存器的值,定时1毫秒,所以是21000while(!(SysTick ->CTRL & (1<<16)));  //死循环等待计数值减到0SysTick ->CTRL = ~(1<<0);    //关闭定时器,第0位置0}}

2.4.3 实现1秒延时

void Sleep_s(uint32_t s)
{while(s--){SysTick ->CTRL = (1 << 0);   //定时器使能第0位置1SysTick ->CTRL &= ~(1<<2);   //选择时钟源:第2位置0,选择外部时钟源,由系统时钟八分频后提供即 168/8=21MSysTick ->CTRL &= ~(1<<1);   //延时时间到无动作:第1位置0SysTick ->VAL = 0x0;        //当前数值寄存器初值赋0SysTick ->LOAD = 21000000;       //重装载数值寄存器的值,定时1秒,所以是21000000while(!(SysTick ->CTRL & (1<<16)));  //死循环等待计数值减到0SysTick ->CTRL = ~(1<<0);    //关闭定时器,第0位置0}}

1秒=1000毫秒=1000微秒。

三、SysTick 定时实验

利用 SysTick 产生 1s 的时基,LED 以 1s 的频率闪烁。

led.h文件

#ifndef __MYLED_H
#define __MYLED_Hvoid LED_Init(void);#endif

led.c 文件

#include "stm32f4xx.h"                  // Device header
#include "myled.h"/*开时钟  打开外设对应的时钟(查看参考手册,该外设挂在哪个数据总线上),对应GPIO在哪条总线开哪条GPIOF外设 挂在AHB1总线上,所以要打开AHB1的时钟,双击函数,右键->go to definition*/void LED_Init(void)
{//第一步:使能GPIOF的时钟 RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOF, ENABLE);//使能 GPIOF 时钟//第二步:GPIOF9,F10 初始化设置GPIO_InitTypeDef GPIO_InitStructure;GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10;//LED0 和 LED1 对应 IO 口GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽输出GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;//100MHzGPIO_Init(GPIOF, &GPIO_InitStructure);//初始化 GPIO//第三步:设置灯的初始状态GPIO_SetBits(GPIOF,GPIO_Pin_9 | GPIO_Pin_10);//GPIOF9,F10 设置高电平,灯灭
}

mydelay.h

#ifndef __MYLED_H
#define __MYLED_Hvoid LED_Init(void);#endif

mydelay.c

#include "stm32f4xx.h"                  // Device header
#include "mydelay.h"void My_Delay_us(uint32_t num)
{while(num--){SysTick ->CTRL = (1 << 0);SysTick ->CTRL &= ~(1<<2);SysTick ->CTRL &= ~(1<<1);SysTick ->VAL = 0x0;SysTick ->LOAD = 21;while(!(SysTick ->CTRL & (1<<16)));SysTick ->CTRL = ~(1<<0);}
}void My_Delay_ms(uint32_t num)
{while(num--){My_Delay_us(1000);}
}void My_Delay_s(uint32_t num)
{while(num--){My_Delay_ms(1000);}
}

main.c文件

#include "stm32f4xx.h"                  // Device header
#include "stdio.h"
#include "mydelay.h"
#include "myled.h"int main(void)
{LED_Init();while(1){My_Delay_ms(1000);           //延时1秒GPIO_ToggleBits(GPIOF,GPIO_Pin_9 | GPIO_Pin_10);}
}

实验现象:

     两个灯每隔一秒闪烁一次。

至此,我们的本次的学习就结束了。通过以上几个实验,相信对串口通信有了深入的理解,这一节我们就讲解到这里,希望能对大家的开发有帮助。 如有兴趣,感谢点赞、关注、收藏,若有不正地方,还请各位大佬多多指教!

这篇关于嵌入式开发十九:SysTick—系统定时器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1078388

相关文章

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

在不同系统间迁移Python程序的方法与教程

《在不同系统间迁移Python程序的方法与教程》本文介绍了几种将Windows上编写的Python程序迁移到Linux服务器上的方法,包括使用虚拟环境和依赖冻结、容器化技术(如Docker)、使用An... 目录使用虚拟环境和依赖冻结1. 创建虚拟环境2. 冻结依赖使用容器化技术(如 docker)1. 创

CentOS系统Maven安装教程分享

《CentOS系统Maven安装教程分享》本文介绍了如何在CentOS系统中安装Maven,并提供了一个简单的实际应用案例,安装Maven需要先安装Java和设置环境变量,Maven可以自动管理项目的... 目录准备工作下载并安装Maven常见问题及解决方法实际应用案例总结Maven是一个流行的项目管理工具

基于Python开发PPTX压缩工具

《基于Python开发PPTX压缩工具》在日常办公中,PPT文件往往因为图片过大而导致文件体积过大,不便于传输和存储,所以本文将使用Python开发一个PPTX压缩工具,需要的可以了解下... 目录引言全部代码环境准备代码结构代码实现运行结果引言在日常办公中,PPT文件往往因为图片过大而导致文件体积过大,

使用DeepSeek API 结合VSCode提升开发效率

《使用DeepSeekAPI结合VSCode提升开发效率》:本文主要介绍DeepSeekAPI与VisualStudioCode(VSCode)结合使用,以提升软件开发效率,具有一定的参考价值... 目录引言准备工作安装必要的 VSCode 扩展配置 DeepSeek API1. 创建 API 请求文件2.

C#实现系统信息监控与获取功能

《C#实现系统信息监控与获取功能》在C#开发的众多应用场景中,获取系统信息以及监控用户操作有着广泛的用途,比如在系统性能优化工具中,需要实时读取CPU、GPU资源信息,本文将详细介绍如何使用C#来实现... 目录前言一、C# 监控键盘1. 原理与实现思路2. 代码实现二、读取 CPU、GPU 资源信息1.

在C#中获取端口号与系统信息的高效实践

《在C#中获取端口号与系统信息的高效实践》在现代软件开发中,尤其是系统管理、运维、监控和性能优化等场景中,了解计算机硬件和网络的状态至关重要,C#作为一种广泛应用的编程语言,提供了丰富的API来帮助开... 目录引言1. 获取端口号信息1.1 获取活动的 TCP 和 UDP 连接说明:应用场景:2. 获取硬

JAVA系统中Spring Boot应用程序的配置文件application.yml使用详解

《JAVA系统中SpringBoot应用程序的配置文件application.yml使用详解》:本文主要介绍JAVA系统中SpringBoot应用程序的配置文件application.yml的... 目录文件路径文件内容解释1. Server 配置2. Spring 配置3. Logging 配置4. Ma

2.1/5.1和7.1声道系统有什么区别? 音频声道的专业知识科普

《2.1/5.1和7.1声道系统有什么区别?音频声道的专业知识科普》当设置环绕声系统时,会遇到2.1、5.1、7.1、7.1.2、9.1等数字,当一遍又一遍地看到它们时,可能想知道它们是什... 想要把智能电视自带的音响升级成专业级的家庭影院系统吗?那么你将面临一个重要的选择——使用 2.1、5.1 还是

基于Python开发电脑定时关机工具

《基于Python开发电脑定时关机工具》这篇文章主要为大家详细介绍了如何基于Python开发一个电脑定时关机工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 简介2. 运行效果3. 相关源码1. 简介这个程序就像一个“忠实的管家”,帮你按时关掉电脑,而且全程不需要你多做