stm32实现毫秒ms微秒us级延时

stm32实现毫秒ms微秒us级延时

 

上一篇文章简单捋了一下32时钟初始化的过程,对systick嘀嗒定时器有了一定的了解吧

实现方法有很多种,推荐一个博客:https://blog.csdn.net/u011878611/article/details/107304203/

我自己在此基础上实现的贴一下,就当存档了(主要参考上面这个博客完成的,向大佬致敬)

【有一点需要注意:使用systick自定义了延时函数,HAL库定义的HAL_Delay就不能用了,因为已经改变了systick的配置,HAL_Delay也是依靠systick来工作的】

delay.c

#include "delay.h"


void delay_ms(uint32_t nms)
{
	
	//法2:没有最大时延限制
	uint32_t i = 0;
	
	SysTick_Config(SystemCoreClock / 1000);//1ms置位一次
	for(i = 0; i < nms; i++)
	{
		while(!( SysTick->CTRL&(1<<16) ));
	}
	SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk;		//关闭计数器

	
	//法1:最大延时为:nms<=1864ms		
	uint32_t ms_ = SystemCoreClock / 1000U; 	//ms级延时基数
	uint32_t us_ = ms_ / 1000U;								//us级延时基数	
	
	SysTick->LOAD=(uint32_t)nms*ms_ - 1; 				//时间加载,SysTick->LOAD为24位寄存器,所以,最大延时为:nms<=1864ms			 
	SysTick->VAL=0x00;        									//清空计数器
	SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk ;		//开始倒数	 
	
	while(!( SysTick->CTRL&(1<<16) )); 					//等待时间到达  	 
	SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk;		//关闭计数器
	SysTick->VAL =0X00;      					 					//清空计数器
	
}


void delay_us(uint32_t nus)
{
	//法2:没有最大时延限制,但是SystemCoreClock不能太低,32M不行,64M或更高可以(具体临界值多少没有测量)
	//具体原因应该是频率太低,导致while循环里面还没有识别到标志位置位,该标志位就被systick清除了
	uint32_t i = 0;

  SysTick_Config(SystemCoreClock / 1000000);//1us置位一次
	for(i = 0; i < nus; i++)
	{
		while(!( SysTick->CTRL&(1<<16) ));
	}
	SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk;		//关闭计数器
	
	
	//法1:最大延时为:nus<=1864135us,低频率也行
	uint32_t ms_ = SystemCoreClock / 1000U; 	//ms级延时基数
	uint32_t us_ = ms_ / 1000U;							//us级延时基数
	
	SysTick->LOAD=(uint32_t)nus*us_-1; 					//时间加载,SysTick->LOAD为24位寄存器,所以,最大延时为:nus<=1864135us	  		 
	SysTick->VAL=0x00;        									//清空计数器
	SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk ;		//开始倒数	  
	
	while(!( SysTick->CTRL&(1<<16) )); 					//等待时间到达 
	SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk;		//关闭计数器
	SysTick->VAL =0X00;      					 					//清空计数器	

}

总结:

  • 知识呀,不用真的会忘的。就像这个systick,我之前读过的《Cortex-M3权威指南》里面是有介绍的,但是要实现ms、us延时的时候一点印象都没有,如果不是在网上找到了别人用systick,我还想着傻傻的用定时器去做延时呢哈哈。暂且原谅自己吧,谁不会忘呢?用到了再学呗!冲冲冲!

 


使用过程中遇到的问题:

  • 自定义延时改变了systick的配置,甚至用过之后关闭了计时器,那么系统时钟会不会受到影响呢?比如函数超时检测,用的就是以systick为1ms为时基的
  • systick用作时基的方式:配置了中断,每次中断就调用HAL_IncTick()来为一个全局变量+1,需要用到时间时,读取这个值两次即可得到时间差
  • 现在把systick定时器关了,HAL_GetTick()函数获取的uwTick数值一直是不变的了,所以永远不会超时!!经测试确实如此!

为了实现定时函数,把其他函数需要使用的时基关闭了??

  • 一种解决方法:延时使用完毕后把systick配置为原始状态,只是在使用延时的时候占用systick。经测试,可行!
  • 更改后的配置如下:
void delay_ms(uint32_t nms)
{
	
	//法2:没有最大时延限制
	uint32_t i = 0;
	
	SysTick_Config(SystemCoreClock / 1000);//1ms置位一次
	for(i = 0; i < nms; i++)
	{
		while(!( SysTick->CTRL&(1<<16) ));
	}
	//SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk;		//关闭计数器
	HAL_InitTick(uwTickPrio);//将systick配置为系统最初配置的状态(这个函数在HAL_RCC_ClockConfig()函数的最下面拷过来的)

}

 

  • 1
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
STM32实现毫秒微秒延时可以使用 SysTick 定时器和内部时钟校准的方式。 首先,需要配置 SysTick 定时器作为中断源,使其每隔一定时间(例如 1 毫秒)产生一次中断。然后,在中断服务函数中计数器自增,用于记录经过的时间。最后,在需要延时的地方,可以通过判断计数器是否达到指定值来进行延时。 以下是一个基于 SysTick 定时器毫秒延时函数: ```c void delay_ms(uint32_t ms) { uint32_t start_time = HAL_GetTick(); while ((HAL_GetTick() - start_time) < ms); } ``` 这个函数使用了 HAL 库提供的 HAL_GetTick() 函数来获取当前的毫秒时间戳,然后在循环中等待时间达到指定值。 对于微秒延时,可以使用内部时钟校准方式。在 STM32 中,内部时钟提供了一个精确的时钟源,可以用于生成微秒延时。以下是一个基于内部时钟的微秒延时函数: ```c void delay_us(uint32_t us) { uint32_t start_time = DWT->CYCCNT; uint32_t delay_cycles = us * (SystemCoreClock / 1000000); while ((DWT->CYCCNT - start_time) < delay_cycles); } ``` 这个函数使用了 DWT 寄存器提供的 CYCCNT 寄存器来记录 CPU 周期数,然后在循环中等待 CPU 周期数达到指定值。需要注意的是,使用 DWT 寄存器需要先打开 DWT 和 CPU 周期计数器。 需要注意的是,以上两个函数的延时时间是近似值,可能会因为硬件和软件因素而有所偏差。对于需要更加精确的延时,可以使用硬件定时器或者外部时钟源。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值