STM32在FreeRTOS下的us延时

STM32在FreeRTOS下的us延时

前言

freeRTOS下跑SPI时需要微秒级别的延时,但是freeRTOS只提供了毫秒级的,记录一下实现us延时的方法。

前期分析

最简单的方式就是开个定时器或者干脆直接计算一下用nop做都可以实现us延时,但是显然还是使用滴答定时器更为优雅。

滴答定时器(SysTick)挂在NVIC上,是一个24位的减数定时器,通过STK_LOAD寄存器控制其重装载值,STK_VAL寄存器存储了当前的计数值,因此实现思路如下:

1、计算延时器件要计的计数值

2、关闭任务调度(防止延时的时候被任务调度器打断)

3、获取当前重装载值(当触发了重装载以后计算需要使用)

4、获取开始延时时寄存器里面的计数值

5、不断获取当前的计数值

6、当当前计数值大于需要计的计数值时,退出延时

步骤1

滴答定时器(SysTick)的时钟输入为HCLK的八分频,则其最小频率也大于1MHZ,因此直接拿其时钟频率除1M,则可以得到计时1us所需要的计数值,通过与目标延时时间(单位为us)相乘,即可以得到总计数时间。

步骤2

任务调度的开关需要使用到调度锁,其概念如下:
调度锁就是 RTOS 提供的调度器开关函数,如果某个任务调用了调度锁开关函数,处于调度锁开和调度锁关之间的代码在执行期间是不会被高优先级的任务抢占的,即任务调度被禁止。这一点要跟临界段的作用区分开,调度锁只是禁止了任务调度,并没有关闭任何中断,中断还是正常执行的。而临界段进行了开关中断操作。
打开调度锁(禁止任务调度)

c 复制代码
vTaskSuspendAll()

关闭调度锁(允许任务调度)

c 复制代码
xTaskResumeAll(void)

步骤3、4、5

获取当前滴答定时器的重装载值可以通过读取LOAD寄存器实现

c 复制代码
told=SysTick->VAL; 

读取当前计数值通过寄存器VAL实现

c 复制代码
tnow=SysTick->VAL;	

实现代码

c 复制代码
//nus为要延时的us数.		    								   
void delay_us(u32 nus)
{		
	u32 ticks;
	u32 told,tnow,tcnt=0;
	u32 reload=SysTick->LOAD;						//LOAD的值	    	 
	ticks=nus*(SystemCoreClock/1000000); 	//需要的节拍数	  		 
	tcnt=0;
	vTaskSuspendAll();									//阻止OS调度,防止打断us延时
	told=SysTick->VAL;        					//刚进入时的计数器值
	while(1)
	{
		tnow=SysTick->VAL;	
		if(tnow!=told)
		{	    
			if(tnow<told)tcnt+=told-tnow;		//这里注意一下SYSTICK是一个递减的计数器就可以了.
			else tcnt+=reload-tnow+told;	    
			told=tnow;
			if(tcnt>=ticks)break;						//时间超过/等于要延迟的时间,则退出.
		}  
	};
	xTaskResumeAll();										//恢复OS调度									    
}
相关推荐
xyx-3v17 小时前
ESP32-C3开发板简单介绍
单片机·嵌入式硬件
Dillon Dong17 小时前
从C到Simulink: 使用STM32硬件支持包后为什么还不支持PC仿真ARM建模程序
c语言·stm32·simulink
刻BITTER17 小时前
在TRAE 上安装PlatformIO
c++·单片机·嵌入式硬件·arduino
m0_5557629018 小时前
I.MX8 Plus —— 系统总线与互连
嵌入式硬件
NULL指向我21 小时前
STM32F407VET6学习笔记14:Bootloader程序笔记
笔记·stm32·学习
别了,李亚普诺夫21 小时前
定时器与输出比较-学习笔记
笔记·stm32·学习
mastercoder--21 小时前
速通51单片机————按键控制LED流水灯与定时器闹钟
单片机·嵌入式硬件·51单片机
西城微科方案开发21 小时前
智能打气泵方案开发设计
单片机·嵌入式硬件·方案公司推荐
硅农深芯1 天前
ARM Cortex-M系列的内核型号有什么区别,该怎么选择
arm开发·单片机·芯片
Y unes1 天前
《uboot基础命令记录①》
linux·驱动开发·嵌入式硬件·mcu·ubuntu·uboot