嵌入式学习笔记——SysTick(系统滴答)

news/2024/11/30 14:33:54/

系统滴答

  • 前言
  • SysTick概述
    • SysTick是个啥
    • SysTick结构框图
      • 1. 时钟选择
      • 2.计数器部分
      • 3.中断部分
      • 工作一个计数周期(从重装载值减到0)的最大延时时间
      • 工作流程
    • SysTick寄存器
      • 1.控制和状态寄存器SysTick->CTRL
      • 2.重装载值寄存器SysTick->LOAD
      • 3.当前值寄存器SysTick->VAL
      • 4.校准值寄存器
    • 配置流程
  • 代码
  • 利用系统滴答实现时间片轮询
    • 效果
  • 总结

前言

上一篇中,介绍了关于STM32F407的时钟系统,在了解了系统的时钟后,最重要的内容就是搞定定时器的操作,本文从最基本的定时器,也是内核里面自带的一个定时器——SysTick(系统滴答)来进行介绍。旨在搞清楚什么是系统滴答,系统滴答有什么用,系统滴答怎么用。

SysTick概述

SysTick这个词其实之前出现过,在介绍中断的时候,就是下面这个图,SysTick就出现了,看他的位置,在图中阴影部分内,也就是说,SysTick是内核里面的属于NVIC的一部分;不是类似USART、GPIO的片上外设,而是一个内核内的外设;看图中有个箭头指向了NVIC,说明它是可以像前面用过的EXTI、USART来产生中断的。
在这里插入图片描述

SysTick是个啥

关于是个啥这种问题,实在是不好表述,咱还是让官方来作答吧。
在这里插入图片描述
看了上面的描述,会有一个大致的概念,首先,它是一个可编程的系统定时器,其次,它被用来做延时和计时的操作,然后还可以触发中断。有一点需要纠正,上面说它是一个32位的自动递减计数器,这点有误,在STM32F407中,它是一个24位的自动递减计数器。
这里一直在说系统滴答是个定时器,那么定时器是个啥,直白点说,定时器就是一个按照时间规律递增或者递减的计数器,在STM32中这个时间规律就是时钟,例如,我们假设系统滴答的时钟是168MHZ;那么系统滴答这个定时器就会在一秒钟内,从0自增到168 000 000;同样的换个方向来理解,就是说计数器计满168000 000就是1s钟的时间。至于递减和递增,递减就是说计数器的初始有我们给定,然后计数器就从这个值开始做自减;而自增则是,我们给定值,然后计数器从0开始自增,一直增加到这个数。

好了,在有了一个大致的映像后,下面就来具体分析它的结构和功能。

SysTick结构框图

由于系统滴答是内部定时器,所以在ST公司的中文参考手册是找不到的,只有在ARM的权威指南中才可以找到相关描述,具体位置在M3和M4权威指南的第九章第五节。
在这里插入图片描述
下拉就可以看见系统框图:
还是按照老套路,把能够省略部分先噶了,这里可以很明显的看见最下面红框与上面的东西都没有联系,所以它是可以噶了的,他的作用就是校准SysTick的,一般来说,SysTick就是使用的系统时钟,如果这个不准了,那么多半这个单片机也命不久矣,所以这个东西可以直接不看。
在这里插入图片描述
去掉不需要看的,接下来就分模块一个部分一个部分的来介绍。

1. 时钟选择

如下图,左侧的红框代表的就是系统滴答的时钟输入选部分;绿色框内是一个二选一数据选择器,两个输入分别是处理器时钟以及经过上升沿检测的参考时钟;执行选择的是下方的“控制和状态寄存器的第2位”,具体的选择流程在寄存器部分会详细介绍。然后时钟就给到了计数器。
在这里插入图片描述
既然有两个输入的时钟,那么这两个时钟具体是指什么呢?
其一是处理器时钟,也就是我们说的主频,对于STM32F407来说对应168MHZ;那么另外一个参考时钟是什么呢?其实这个时钟在昨天的时钟树介绍中也出现了。如下图所示,橙色框中的到Cortex系统定时器的就是这里的参考时钟,可以发现,它经过了一个8分频的分频器,也就是说这个时钟的频率应该是168/8=21Mhz。

在这里插入图片描述

2.计数器部分

计数器简化后如下图所示,这是一个计数器的最基本结构,首先有三部分输入:
1.时钟基准:这个时钟直接决定了这个计数器多少时间执行一次计数;
2.重装载值:上方的重装载值直接决定了计数器的最大计数值;
3.控制部分:控制部分直接决定了计数器什么时刻开始计数,什么时候关闭计数,这里的第0位就是用来控制计数器是否计数的。
在这里插入图片描述
然后是输出部分,输出只有一个方向,就是4的位置,注意描述:当计数器从1减到0的时候会触发,而且这个触发是指向了“控制和状态寄存器”的,这就说明,当计数完成的时候,在“控制和状态寄存器”中会有对应的位,让我么来判断计时是否完成。
最后,最主要的部分,就是橙色框的24位向下计数器,它的作用就是隔一段时间将数值减一。当然,这里的明子就叫向下计数器,那么肯定还有对应的向上计数器,以及中心对齐的计数器,这个在后面基本通用和高级定时器中会碰到,遇到了再说。

3.中断部分

然后这个图还剩最后一部分,就是有关中断的了,这里有一个与门,与门的输入一个来自计数器技术完成后的标志,另一个来自“控制与状态寄存器”的第1位,也就是中断使能,说明在需要使用到中断的过程中,需要使能这个位才能开启中断。

在这里插入图片描述

工作一个计数周期(从重装载值减到0)的最大延时时间

弄清楚了上面的结构后,就可以计算出两个频率下,计数器工作一个周期,最长所需要花费的时间。
最大的重装载值:2^24=16777216
系统滴答具备两个时钟源:
内核时钟:主频提供时钟 168MHZ
最大的延时时长:1S16777216/168 000 000=0.09986S
0.09986s---->99.8ms
外部时钟:由AHB线提供 21MHZ
最大的延时时长:1S
16777216/21 000 000=0.7989 S
0.7989s-----》798.9ms

工作流程

根据框图的分析,可以大致总结出系统滴答的初始化流程:

{①选择时钟;②根据自己所需时间计算出重装载值;③使能计数器;④判断对应的标志位是否到了,到了说明计时到了,没到说明计时还没到
}

SysTick寄存器

其实根据框图,寄存器也已经猜的七七八八了,还是具体的看一眼,关于系统滴答一共有四个寄存器。
在这里插入图片描述

1.控制和状态寄存器SysTick->CTRL

在这里插入图片描述
写法:SysTick->CTRL
功能:对系统滴答定时器做控制,以及读取对应的状态
第0位:ENALEB
置1:使能计数器 一直重复工作
置0:失能计数器

第1位:中断使能位 计数标志一定会置1/中断标志
置1:使能中断
置0:失能中断

第2位:选择时钟源 默认1
置1:选择内核时钟 168MHZ
置0:外部参考时钟 21MHZ

第16位:标志位 只读
为1:计数器到0则返回1
为0:读取时清零
读取时的具体写法:

while(! (SysTick->CTRL & (1<<16)) );

2.重装载值寄存器SysTick->LOAD

在这里插入图片描述
写法: SysTick->LOAD
功能:提供计数器的最大值
用法:直接写入需要写入的最大计数值
不能超过最大的重装载值范围(0-1667216)
SysTick->LOAD=arr-1;
这个值具体写入多少,要结合需求,计算出大小

3.当前值寄存器SysTick->VAL

写法:SysTick->VAL
功能:存储计数器的当前值
读取这个寄存器:能够获取到计数器的当前值
写入这个寄存器:任意值都能清除计数标志位

4.校准值寄存器

在分析框图的时候提到过,这个一般不用。

配置流程

这里的配置流程分为两类:
其一是实现一个延时功能,延时功能只需要定时器工作一个周期,也就是从重装载值减到一的一个过程,执行一次后需要关闭定时器,不让他还会不停的从重装载值减到0然后又从重装载值减到0无限循环。
伪代码:

实现系统的us延时(参数)
{//选择时钟 建议选择外部时钟//写入重装载值  21*参数//当前值清零//打开计数器//等待标志位置1//关闭计数器
}

其二就是利用中断,一定时间进一次中断,以此来实现一个时间片轮询的操作方式。这时候,就需要定时器一直计数了,所以不能计数完成后就关闭计数器了。伪代码如下:

系统滴答的初始化代码
{//选择系统滴答的时钟//配置系统抵达的重装载值//当前值清零//打开中断使能//NVIC控制器//开启定时器   
}
中断服务函数
{判断标志;清楚标志;执行操作。
}

代码

#include "SysTick.h"
u16 SysTick_us;
u16 SysTick_ms;/*******************************
函数名:SysTick_Init
函数功能:初始化系统滴答,选择外部时钟
函数形参:u32 sysclk 系统时钟168(MHZ)
函数返回值:void
备注:开启1ms中断********************************/
void SysTick_Init(u32 sysclk) //168MHZ
{u32 pri;//存储优先级合成函数返回的优先级SysTick->CTRL &=~(1<<2); //选择外部时钟,必须清零默认是1内核时钟SysTick_us=sysclk/8;           //21     1us//外部时钟8分频SysTick_ms=SysTick_us*1000;            //21 000  1msSysTick->LOAD = SysTick_ms-1;//重装载值21000-1SysTick->VAL=0;    //清空计数器,清标志位SysTick->CTRL |=1<<1;   //使能中断 /*-----------------------配置NVIC---------------------------------------------*/	pri=NVIC_EncodePriority(7-2,1,2);NVIC_SetPriority(SysTick_IRQn,pri);NVIC_EnableIRQ(SysTick_IRQn);SysTick->CTRL |=1<<0;   //使能计数器  
}/*******************************
函数名:SysTick_Delay_us
函数功能:系统滴答实现us延时
函数形参:u32 nus
函数返回值:void
备注:
//因为LOAD为24位,所以最大重装载值16,777,216
最长时间:形参最大值,798,915us********************************/
void SysTick_Delay_us(u32 nus)//1us
{SysTick->LOAD =nus*SysTick_us;//传进来的参数*21  nus 传多少就是多少微秒SysTick->VAL=0;    //清空计数器,清标志位SysTick->CTRL |=1<<0;   //使能  while(!(SysTick->CTRL & 1<<16));//等待计数完成SysTick->CTRL &=~(1<<0);  //关闭计数器SysTick->VAL=0;     //清空计数器,清标志位
}
/*******************************
函数名:SysTick_Delay_ms
函数功能:系统滴答实现ms延时
函数形参:u32 nms
函数返回值:void
备注:
形参最大值798ms
********************************/
void SysTick_Delay_ms(u32 nms)
{SysTick->LOAD =nms*SysTick_ms;//传进来的参数*21  nms 传多少就是多少毫秒SysTick->VAL=0;    //清空计数器,清标志位SysTick->CTRL |=1<<0;   //使能  while(!(SysTick->CTRL & 1<<16));//等待标志位到SysTick->CTRL &=~(1<<0);  //关闭计数器SysTick->VAL=0;     //清空计数器,清标志位
}
//中断服务函数:/*******************************
函数名:SysTick_Handler
函数功能:系统滴答的中断服务函数函数
函数形参:无
函数返回值:void
备注:1ms进一次中断
********************************/
void SysTick_Handler(void)
{static u8 i=100;while(SysTick->CTRL &(1<<16))//检测中断标志,同时也是清除标志位mtime--;Led_cnt++;_TIMER_1MS++;i--;if(i==0){i = 100;_TIMER_100MS ++;}
}

利用系统滴答实现时间片轮询

使用时间片轮询的方式编程,可以很好地解决之前遇见的阻塞问题,在系统滴答里面定义好对应的计时变量,然后根据这个计时变量来执行所需要的操作。
如下图所示:这里笔者一共选取了三个时间变量分别计时1S、100ms、200ms,其中一秒钟的时序对应一次串口打印输出;100ms与200ms分别对应LED1和LED2的闪烁;除此之外还有一个轮询为0的情况用来存放不需要严格时序刷新的任务。
在这里插入图片描述

效果

最终效果如下:通过时间戳可以看出来SysTick的计时还是比较准准确的。
在这里插入图片描述

总结

系统滴答就是一个系统内的定时器,其主要作用就是提供精确延时以及计时的功能,可以借此实现时间片轮询的代码框架。


http://www.ppmy.cn/news/34556.html

相关文章

【CodeForces】Codeforces Round 859 (Div. 4) D

嘿嘿嘿&#xff0c;CF虐我千百遍&#xff0c;我待CF如初见&#xff01; &#xff08;doge&#xff09; 目录 题目含义&#xff1a; 前缀和&#xff1a; 代码 &#x1f386;音乐分享&#xff08;点击链接可以听哦&#xff09; A Hundred Miles&#xff08;一百英里&#xff09;…

HAL库 STM32 串口通信

一、实验条件将STM32的PA9复用为串口1的TX&#xff0c;PA10复用为串口1的RX。STM32芯片的输出TX和接收RX与CH340的接收RX和发送TX相连&#xff08;收发交叉且PCB上默认没有相连&#xff0c;所以需要用P3跳线帽进行手动连接&#xff09;&#xff0c;CH340的另一端通过USB口引出与…

用嘴写代码?继ChatGPT和NewBing之后,微软又开始整活了,Github Copilot X!

用嘴写代码&#xff1f;继ChatGPT和NewBing之后&#xff0c;微软又开始整活了&#xff0c;Github Copilot X&#xff01; AI盛行的时代来临了&#xff0c;在这段时间&#xff0c;除了爆火的GPT3.5后&#xff0c;OpenAI发布了GPT4版本&#xff0c;同时微软也在Bing上开始加入了A…

Mysql数据库如何调优

MYSQL数据库调优 索引 1、对于常用的查询字段加索引&#xff0c;但如果常用字段只有几个常量值就不需要加索引&#xff0c;或者使用索引会失效的情况&#xff1b; 2、索引失效的情况&#xff1a; ​ 1、索引列使用函数&#xff0c;计算&#xff08;加减乘除等&#xff09; …

【Linux】进程优先级 环境变量

进程优先级 环境变量 一、进程优先级1、基本概念2、查看以及修改系统进程的优先级3、一些其他的关于进程优先级的指令和函数调用4、与进程优先级有关的一些进程性质二、环境变量1、基本概念2、和环境变量相关的命令3、Linux中的常见环境变量介绍4、环境变量的组织方式以及在C代…

基于XML的自动装配~

基于XML的自动装配之场景模拟&#xff1a; 自动装配&#xff1a;根据指定的策略&#xff0c;在IOC容器中匹配某一个bean&#xff0c;自动为指定的bean中所依赖的类类型或者接口类型赋值 之前我们学过的依赖注入&#xff0c;我们在为不同属性赋值时&#xff0c;例如类类型的属性…

【C++】内联函数inline

文章目录概念使用特性原理概念 C中内联函数的出现解决了C语言宏函数的不足&#xff0c;类似于宏展开&#xff0c;这种在函数调用处直接嵌入函数体的函数称为内联函数&#xff0c;又称内嵌函数或内置函数。 以inline修饰的函数叫做内联函数&#xff0c;编译时C编译器会在调用内…

AJAX,Axios,JSON简单了解

一. AJAX简介概念: AJAX(Asynchronous JavaScript And XML): 异步的JavaScript 和XMLAJAX作用:1.与服务器进行数据交换: 通过AJAX可以给服务器发送请求&#xff0c;并获取服务器响应的数据使用了AJAX和服务器进行通信&#xff0c;就可以使用 HTMLAJAX来替换JSP页面了2.异步交互…