Systick滴答定时器寄存器、delay()延时函数、SysTick_Config函数

SysTick定时器
SysTick定时器,是一个简单的定时器,对于CM3、CM4内核的芯片都有SysTick定时器。SysTick 是一个 24 位的倒计数定时器,当计数到 0 时,将从RELOAD 寄存器中自动重装载定时初值,开始新一轮计数。只要不把它在 SysTick 控制及状态寄存器中的使能位清除(不把Systick定时器关掉),就永不停息。SysTick定时器被捆绑在NVIC中,用于产生SYSTICK异常(异常号:15)。在以前,大多操作系统需要一个硬件定时器来产生操作系统需要的滴答中断,作为整个系统的时基。例如,为多个任务许以不同数目的时间片,确保没有一个任务能霸占系统;或者把每个定时器周期的某个时间范围赐予特定的任务等,还有操作系统提供的各种定时功能,都与这个滴答定时器有关。因此,需要一个定时器来产生周期性的中断,而且最好还让用户程序不能随意访问它的寄存器,以维持操作系统“心跳”的节律。

作用:

Systick定时器常用来做延时,或者实时系统的心跳时钟。这样可以节省MCU资源,不用浪费一个定时器。比如:UCOS中,分时复用,需要一个最小的时间戳,一般在STM32+UCOS系统中,都采用Systick做UCOS的心跳时钟。

Systick寄存器:

  • CLRL寄存器:Systick控制和状态寄存器
  • LOAD寄存器:Systick自动重装载寄存器
  • VAL寄存器:Systick当前值寄存器
  • CALIB寄存器:Systick校准值寄存器

Systick控制和状态寄存器-CTRL
在这里插入图片描述对于STM32,外部时钟源是HCLK(AHB总线时钟的1/8)内核时钟是HCLK时钟。
配置函数:Systick_CLKSourceConfig();

Systick重装载数值寄存器-LOAD(24位)
在这里插入图片描述
Systick当前值定时器-VAL
在这里插入图片描述
Systick校准寄存器-CAL
在这里插入图片描述
固件库中的Systick相关函数:
SysTick_CLCSourConfig()//Systick时钟源选择,在misc.c文件中。
SysTick_config(uint32_t ticks)//初始化systick时钟为HCLK,并开启中断,在core_cm3/core_cm4文件中。

Systick中断服务函数:
void SysTick_Handler(void);

用中断的方式实现delay延时:(比较耗费资源)

static __IO uint32_t TimingDelay;
void Delay(__IO uint32_t nTime)
{ TimingDelay = nTime;while(TimingDelay != 0);
}
void SysTick_Handler(void)
{if (TimingDelay != 0x00) { TimingDelay--;}
}int main(void){if (SysTick_Config(SystemCoreClock / 1000)) //systick时钟为HCLK(系统时钟),中断时间间隔1ms,SysTick_Config()这个函数是设置两次中断之间的时间。{while (1);}while(1){ Delay(200);//200ms}
}

SysTick_Config函数:

SysTick_Config的参数,其实就是一个时钟次数,叫systick重装定时器的值。意思就是我要多少个1/fosc 时间后中断一下。根据学过的物理中的时间与频率的公式:fosc=1/T T=1/fosc ,fosc为Systick的频率。如果STSystick时钟频率为:72MHz,每次的时间为:T=1/72MHz。1秒钟为:1/(每次的时间)=1/(1/72MHz)=72 000 000次。1MHz是:1000 000。反过来讲。SysTick_Config(72000)代表:72000*(1/72MHz)=1/1000=1(ms)。即定时为1ms。如果需要1S则可以通一设置一个全局变量,然后定初值得为1000,这样每个systick中断一次,这个全局变量减1,减到0,即systick中断1000次,时间为:1ms1000=1S。从而实现1S的定时。 因为SysTick定时器是:24位的,最大定时时间为:2的24次方(1/72MHz)的时间,这里Systick频率为:72MHz的情况下。

delay_init()函数:

static u8  fac_us=0;//us延时倍乘数			   
static u16 fac_ms=0;//ms延时倍乘数,在ucos下,代表每个节拍的ms数
void delay_init()
{
#if SYSTEM_SUPPORT_OS  							//如果需要支持OS.u32 reload;
#endifSysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);//选择外部时钟  HCLK/8fac_us=SystemCoreClock/8000000;//为系统时钟的1/8  相当于72000000/8000000等于九//一次是1/systick的时钟频率就是1/9 us,fac_us相当于1us,这里的将9赋值给他是给寄存器LOAD计数用的,从9到1就是1us
#if SYSTEM_SUPPORT_OS  //如果需要支持OS.reload=SystemCoreClock/8000000;//每秒钟的计数次数 单位为M  reload*=1000000/delay_ostickspersec;//根据delay_ostickspersec设定溢出时间//reload为24位寄存器,最大值:16777216,在72M下,约合1.86s左右	fac_ms=1000/delay_ostickspersec;//代表OS可以延时的最少单位	   SysTick->CTRL|=SysTick_CTRL_TICKINT_Msk;//开启SYSTICK中断SysTick->LOAD=reload; //每1/delay_ostickspersec秒中断一次	SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk;	//开启SYSTICK    #elsefac_ms=(u16)fac_us*1000;//非OS下,代表每个ms需要的systick时钟数   
#endif
}								    

void delay_us()函数:

void delay_us(u32 nus)
{		u32 temp;	    	 SysTick->LOAD=nus*fac_us; 					//时间加载	  		 SysTick->VAL=0x00;        					//清空计数器SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk ;	//开始倒数	  do{temp=SysTick->CTRL;}while((temp&0x01)&&!(temp&(1<<16)));		//等待时间到达   SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk;	//关闭计数器SysTick->VAL =0X00;      					 //清空计数器	 
}

void delay_ms()函数:

void delay_ms(u16 nms)
{	 		  	  u32 temp;		   SysTick->LOAD=(u32)nms*fac_ms;				//时间加载(SysTick->LOAD为24bit)SysTick->VAL =0x00;							//清空计数器SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk ;	//开始倒数  do{temp=SysTick->CTRL;}while((temp&0x01)&&!(temp&(1<<16)));		//等待时间到达   SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk;	//关闭计数器SysTick->VAL =0X00;       					//清空计数器	  	    
} 
  • T=1/1KHZ=1ms=1微秒=1/1000秒
  • T=1/1MHZ=1us=1微秒=1/1000000秒
  • 10MHz=1/10us=0.1us
  • 20MHz=1/20us=0.05us
  • 50MHz=1/50us=0.02us
  • 1GHz=1ns=1纳秒=1/1000000000秒
  • 1MHZ=1000KHZ=1000000HZ
  • 1GHZ=1000MHZ

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/272338.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

查看docker容器日志

1&#xff1a;实时查看docker容器id为 02c5ac132ee5 的最后10行日志 docker logs -f -t --tail 10 02c5ac132ee5 2:查看指定时间后的日志&#xff0c;只显示最后100行&#xff1a; docker logs -f -t --since"2020-02-14" --tail100 d7db22166a0a 3:查看最近20分钟的…

Web UI 自动化测试环境搭建 (转载自51测试天地第三十九期上)

1. 安装 Python 2.7 并设置系统环境变量 2. 下载并安装 python setuptools Easily download, build, install, upgrade, and uninstall Python packages https://pypi.python.org/pypi/setuptools#installation-instructions 2.1 找到ez_setup.py&#xff0c;点击右键--目标另存…

STM32F1 端口复用、端口(部分和完全)重映射

端口复用功能 STM32 有很多的内置外设&#xff08;比如&#xff1a;串口、ADC、DAC等是独立的模块和内核连接在一起&#xff09;&#xff0c;这些外设的外部引脚都是与 GPIO 复用的。也就是说&#xff0c;一个 GPIO如果可以复用为内置外设的功能引脚&#xff0c;那么当…

docker启动容器后容器状态为Exited (137) 5 seconds ago

1&#xff1a;因为容器里的运行的代码报错了&#xff0c;然后容器 Exited (1) 3 seconds ago 了&#xff0c;通过 docker logs -f container_id 能看到哪里错了 容器桩体为exited&#xff0c;说明容器已经退出停止 先查看查看镜像id ps images 在后台运行一个容器 为了保证提…

STM32中断优先级的管理(NVIC)

STM32 NVIC 中断优先级管理 CM3 内核支持 256 个中断&#xff0c;其中包含了 16 个内核中断和 240 个外部中断&#xff0c;并且具有 256级的可编程中断设置。STM32 并没有使用 CM3 内核的全部东西&#xff0c;而是只用了它的一部分。STM32 有 84 个中断&#xff0c;包括 16 个…

docker修改容器名字

查看一下容器的名字 这个laughing_elion是下载es时候默认的名字 修改容器名字 docker rename 容器原来名 要改为的名字 最后可以看到容器名已经修改成功

STM32 串行通信原理

处理器与外部设备通信的两种方式&#xff1a; 并行通信 传输原理&#xff1a;数据各个位同时传输。优点&#xff1a;速度快缺点&#xff1a;占用引脚资源多 串行通信 传输原理&#xff1a;数据按位顺序传输。优点&#xff1a;占用引脚资源少缺点&#xff1a;速度相对较慢 …

linus下centos7防火墙设置

CentOS7 默认使用firewalld防火墙&#xff0c;如果想换回iptables防火墙&#xff0c;可关闭firewalld并安装iptables。 1:安装firewalld服务 yum install firewalld 2、firewalld的基本使用 启动&#xff1a; systemctl start firewalld &#xff08;关闭后显示notrunning&a…

串口通信寄存器/库函数配置、实例编写

常用的串口相关寄存器 USART_SR状态寄存器USART_DR数据寄存器USART_BRR波特率寄存器 串口操作相关库函数&#xff08;省略入口参数&#xff09;&#xff1a; void USART_Init(); //串口初始化&#xff1a;波特率&#xff0c;数据字长&#xff0c;奇偶校验&#xff0c;硬件流…

使用docker安装Mongodb

下载mongo3.2的docker镜像&#xff1a; docker pull mongo:3.2 使用docker命令启动&#xff1a; docker run -p 27017:27017 --name mongo \ -v /mydata/mongo/db:/data/db \ -d mongo:3.2 查看已经成功启动

什么是真正的程序员?

什么是真正的程序员 这篇文章的原文来自&#xff1a;A Little Printf Story作者仿照《小王子》中的情节&#xff0c;通过小printf遇见的不同类型的程序员&#xff0c;最后悟出什么才是真正的程序员&#xff01;第一次翻译有很多不妥&#xff0c;欢迎留言指正。 文章略长&#x…

为什么会出现docker

docker镜像&#xff0c;一次构建&#xff0c;到处运行

docker和虚拟机的区别

虚拟机缺点&#xff1a; docker和虚拟机的不同之处&#xff1a;

独立看门狗实验

为什么要看门狗 在由单片机构成的微型计算机系统中&#xff0c;由于单片机的工作常常会受到来自外界电磁场的干扰&#xff0c;造成程序的跑飞&#xff0c;而陷入死循环&#xff0c;程序的正常运行被打断&#xff0c;由单片机控制的系统无法继续工作&#xff0c;会造成整个系统…

Manthan, Codefest 16

暴力 A - Ebony and Ivory import java.util.*; import java.io.*;public class Main {public static void main(String[] args) {Scanner cin new Scanner (new BufferedInputStream (System.in));int a cin.nextInt ();int b cin.nextInt ();int c cin.nextInt ();for…

docker资源

Docker资源 Docker官方英文资源&#xff1a; docker官网&#xff1a;http://www.docker.com Docker windows入门&#xff1a;https://docs.docker.com/windows/ Docker Linux 入门&#xff1a;https://docs.docker.com/linux/ Docker mac 入门&#xff1a;https://docs.do…

STM32 通用定时器基本原理

STM32F10x系列总共最多有8个定时器&#xff1a; 三种STM32定时器区别&#xff1a; 通用定时器功能特点描述&#xff1a; ①、 STM32 的通用 TIMx (TIM2、TIM3、TIM4 和 TIM5)定时器功能特点包括&#xff1a; 位于低速的APB1总线上(时钟来源可以是APB1的时钟) 16 位向上、向…

初识-Android之智能短信项目相关技术整理

标签页切换采用传统的TabHost&#xff1a; 采用TabActivty实现TabHost。 效果图-后补&#xff1a; 相关技术详解推荐&#xff1a; http://blog.csdn.net/zhouli_05/article/details/7696054 这里我解决了一个TabActivity和子Activity共享TabActivity的OptionMenu的问题&#xf…