STM32 HAL 库 uS 延时的 3 种实现方式

lSTM32 HAL 库 uS 延时的 3 种实现方式 - 知乎 (zhihu.com)

来源知乎,文章底部保留了原作者联系方式.

侵删.

CUBEMX 可视化初始化配置,结合 HAL 库,给我们开发带来了很多便利,但 HAL 库封装的延时函数目前仅支持 ms 级别的延时,日常很多情况下会用到 us 延时,特别是一些传感器的数据读取过程,对时序要求比较严格,us 延时必不可少,基于此项需求,此次给大家介绍 3 种 uS 延时的实现方式,方法同样适用标准库,不足之处,还请大佬指出。

实验目标

  • 使用普通定时器实现 us 延时
  • 使用 Systick 功能实现 us 延时
  • 使用 for 循环实现 us 延时

1、普通定时器实现 us 延时

使用定时器 TIM2 来实现 us 延时,采用 cubemx 对工程进行配置,时钟是 MCU 的心脏,先对时钟进行配置。

1.1、外部时钟选择

也可以使用内部 RC 高速时钟,本次主要介绍使用外部高速时钟,上图:

我板子上焊接的是 8M 的晶体,如果小伙伴们的板子上不是 8M,根据自己的晶振频率配置即可,左侧圈 1 中,可以根据自己的晶体频率,输入相应的频率,经过分频、倍频后,系统时钟频率设置为最大,168MHZ,APB1 的时钟频率为 84MHZ,也是后面用到的 TIM2 挂载的时钟源的频率。

1.2、TIM2 基础配置

这个就比较简单了,分频系数 83,计数单位为 84MHZ/84 = 1uS,向上计数方式,周期 65535,由于没有使用到中断,不需要开启中断。

时钟及定时器的配置就完成了,下面是 cubemx 生成工程时的几项设置,建议大家勾选。 首先是 HAL 库是否需要包含所有的文件,我们选择只需要用到的文件,这样可以缩短工程编译时间,只编译我们用到的库文件,接着是勾选为每个外设生成单独的.c .h 文件,这个建议一定要勾选,会使代码结构非常清晰,第三点就非常的重要了,用过 cubemx 的小伙伴是否遇到过每次重新生成工程后,之前添加的文件都不见了,这一项勾选之后,会保留用户文件。

然后是编译器选择,可以根据自己喜欢的 IDE 选择,我选择的是 KEIL5。

至此,配置工作就完成了,生成工程就可以了。

1.3、代码实现

/*
    普通定时器实现us延时
*/
void user_delaynus_tim(uint32_t nus)
{

    uint16_t  differ = 0xffff-nus-5;
    //设置定时器2的技术初始值
  __HAL_TIM_SetCounter(&htim2,differ);
  //开启定时器
  HAL_TIM_Base_Start(&htim2);

  while( differ<0xffff-5)
    {
        differ = __HAL_TIM_GetCounter(&htim2);
    };
 //关闭定时器
  HAL_TIM_Base_Stop(&htim2);
}
/*
    普通定时器实现ms延时,可直接使用HAL库函数HAL_delay()
*/
void delay_ms_tim(uint16_t nms)
{
    uint32_t i;
    for(i=0;i<nms;i++) user_delaynus_tim(1000);
}

1.4、实现效果

通过延时翻转 IO,逻辑分析仪测试延时时间,分别测试了延时 20us,500ms,下面是测量图:

2、Systick 功能实现 us 延时

2.1、Systick介绍

CM3与CM4包含一个系统计数器SysTick,是一个24位倒计数定时器,当计数到0 时,将从RELOAD寄存器中自动重装载定时初值,只要把它在SysTick->CTRL中的使能位清除,则一直存在。 寄存器介绍:

相应代码在core_cm4.h中

/**
  \brief  Structure type to access the System Timer (SysTick).
 */
typedef struct
{
  __IOM uint32_t CTRL;                   /*!< Offset: 0x000 (R/W)  SysTick Control and Status Register */
  __IOM uint32_t LOAD;                   /*!< Offset: 0x004 (R/W)  SysTick Reload Value Register */
  __IOM uint32_t VAL;                    /*!< Offset: 0x008 (R/W)  SysTick Current Value Register */
  __IM  uint32_t CALIB;                  /*!< Offset: 0x00C (R/ )  SysTick Calibration Register */
} SysTick_Type;

SysTick控制及状态寄存器(0xE000_E010): 该寄存器第0位:表示SysTick使能位,0表示关;1表示开;SysTick_CTRL_ENABLE_Mask; 第1位:表示SysTIck中断使能位,0,表示关闭断;1表示打开中断;SysTick_CTRL_TICKINT_Mask

第2位:表示时钟源选择位,0,表示HCLK/8;1表示HCLK作为时钟源;SysTick_CTRL_SOURCE_Mask;

第16位:表示计数比较标志,如果上次读取本寄存器,计算到了0,则该位置1,如果读取该位,该位将自动清零。 SysTick的LOAD寄存器: 为递减计数,是24位寄存器,最大值为0xFFFFFF; SysTick的VAL寄存器: 24位寄存器,读取时返回当前计数值,写它则使其清零,同时会清零CTRL寄存器中的COUNTFLAG标志。

2.2、代码实现

/*
Systick功能实现us延时,参数SYSCLK为系统时钟
*/
uint32_t fac_us;

void HAL_Delay_us_init(uint8_t SYSCLK)
{
     fac_us=SYSCLK; 
}

void HAL_Delay_us(uint32_t nus)
{
    uint32_t ticks;
    uint32_t told,tnow,tcnt=0;
    uint32_t reload=SysTick->LOAD;
    ticks=nus*fac_us; 
    told=SysTick->VAL; 
    while(1)
    {
        tnow=SysTick->VAL;
        if(tnow!=told)
        {
            if(tnow<told)tcnt+=told-tnow;
            else tcnt+=reload-tnow+told;
            told=tnow;
            if(tcnt>=ticks)break; 
        }
    };
}

2.3 实现效果

通过延时翻转IO,逻辑分析仪测试延时时间,测试了延时 20us,下面是测量图:

for循环实现us延时

这个方法比较接地气,采用NOP空语句实现,具体实现起来最好是看汇编代码,有兴趣的小伙伴可以研究研究,直接上代码:

/*
for循环实现延时us
*/
void for_delay_us(uint32_t nus)
{
    uint32_t Delay = nus * 168/4;
    do
    {
        __NOP();
    }
    while (Delay --);
}

实现效果: 通过延时翻转IO,逻辑分析仪测试延时时间,测试了延时 20us,下面是测量图:

本次要分享的内容就要结束啦,希望对us延时有疑惑的小伙伴有帮助,实现方法不止这些,暂时就分享这3种。

如果你觉得对自己有帮助的话,给个赞,点个关注,点个在看,感谢前进的道路上有你的陪伴! 图片欢迎大家关注Embeded小飞哥,让我快点遇到优秀的你,然后一起变得更加优秀,加油!!!小飞哥微信号:w974762670,加好友进群一起交流呀!

公众号:

<think>好的,我现在需要帮助用户解决关于STM32微控制器实现微秒级延时的问题。首先,用户的需求很明确,他们想要具体的实现方法或代码示例。我需要回忆一下STM32中常用的延时方法,以及如何达到微秒级别的精度。 首先,常见的延时方法有几种:使用SysTick定时器、使用通用定时器(如TIM2、TIM3等)、或者使用DWT(Data Watchpoint and Trace)单元中的周期计数器。用户提到的微秒级延时,可能需要较高的精度,所以普通的循环延时可能不够准确,尤其是在不同的时钟频率下。 接下来,我需要考虑用户可能的开发环境,比如使用的是HAL还是标准外设HAL中虽然有HAL_Delay函数,但那是毫秒级的,基于SysTick。要实现微秒级,可能需要自己配置定时器或者使用其他方法。 想到DWT单元,它有一个周期计数器,可以用来测量精确的时间。这种方法不需要额外的定时器配置,只需要启用DWT功能,然后读取计数器值。这可能是一个高效的方式,因为DWT是内核的一部分,访问速度快,精度高。不过需要确认DWT在目标芯片上是否可用,比如STM32F4系列是支持的。 另外,通用定时器也是一个选择。比如配置一个定时器为微秒级的分频,然后在需要延时时启动定时器,等待计数完成。这种方法需要占用一个定时器资源,但可以更灵活地控制延时时间。 还有可能用户需要的是基于NOP指令的循环延时,这种方法通过计算指令周期来估算时间。不过这种方法受编译器优化和时钟频率影响较大,移植性较差,但实现简单。 现在需要考虑如何结合用户提供的引用内容。查看用户给出的引用,引用3提到了STM32F0xx移植EasyFlash的配置,其中涉及SPI和Flash设备。这可能与延时函数的需求关系不大,但可能用户在使用某些外设时需要精确的延时,比如SPI通信中的时序要求。所以需要确保延时函数的准确性,尤其是在与外设交互时。 接下来,需要组织回答的结构,先介绍不同的方法,然后给出代码示例。同时要提到每种方法的优缺点,适用场景,以及注意事项,比如时钟配置、中断影响等。 可能的问题点:如何确保延时的准确性,尤其是在有中断发生的情况下。比如使用DWT或定时器时,如果发生中断,延时可能会被延长,需要用户根据实际应用场景考虑是否需要禁用中断。 另外,需要提醒用户,不同型号的STM32时钟频率可能不同,代码中的分频系数需要根据实际系统时钟调整。例如,如果系统时钟是72MHz,那么每个微秒对应的时钟周期数就是72。所以在使用定时器或DWT时,需要根据系统时钟计算正确的计数值。 最后,生成相关问题,帮助用户进一步了解相关主题,比如如何优化延时精度、不同延时方法的比较、或者如何处理中断对延时的影响等。</think>### STM32微秒级延时实现方法 在STM32开发中,实现微秒级延时的常用方法主要有三种:**DWT周期计数器法**、**硬件定时器法**和**指令循环法**。以下为具体实现及代码示例: --- #### 一、DWT周期计数器法(推荐) **原理**:利用Cortex-M内核的DWT单元中的CYCCNT寄存器记录CPU时钟周期数,通过计算周期数差值实现精确延时。 **优势**:无需占用硬件资源,精度可达1个时钟周期(例如72MHz时钟对应13.89ns分辨率)[^1]。 ```c #include "core_cm4.h" void DWT_Init(void) { CoreDebug->DEMCR |= CoreDebug_DEMCR_TRCENA_Msk; DWT->CYCCNT = 0; DWT->CTRL |= DWT_CTRL_CYCCNTENA_Msk; } void delay_us(uint32_t us) { uint32_t start = DWT->CYCCNT; uint32_t cycles = us * (SystemCoreClock / 1000000); while ((DWT->CYCCNT - start) < cycles); } ``` **注意事项**: 1. 需在`SystemCoreClock`变量中正确配置系统时钟频率(例如72MHz) 2. 部分低端型号(如STM32F0)可能不支持DWT功能 --- #### 二、硬件定时器法 **原理**:配置通用定时器(如TIM2)以微秒为计数单位,通过计数器溢出实现延时。 **优势**:稳定性高,不受CPU负载影响。 ```c // 以TIM2为例(72MHz时钟) void TIM_Delay_Init(void) { RCC->APB1ENR |= RCC_APB1ENR_TIM2EN; TIM2->PSC = 71; // 分频至1MHz(72MHz/(71+1)) TIM2->ARR = 0xFFFF; TIM2->CR1 |= TIM_CR1_CEN; } void delay_us(uint16_t us) { TIM2->CNT = 0; while (TIM2->CNT < us); } ``` **注意事项**: 1. 需根据实际时钟调整预分频值(PSC) 2. 避免与其他功能共用同一定时器 --- #### 三、指令循环法 **原理**:通过计算NOP指令执行时间实现粗略延时。 **优势**:无需硬件资源,适用于简单场景。 ```c void delay_us(uint16_t us) { uint32_t delay = us * (SystemCoreClock / 1000000) / 4; while (delay--) { __NOP(); } } ``` **注意事项**: 1. 实际延时受编译器优化影响 2. 需通过示波器校准 --- ### 性能对比 | 方法 | 精度 | CPU占用 | 适用场景 | |--------------|---------|---------|------------------------| | DWT计数器 | &plusmn;1周期 | 无 | 高精度实时控制[^2] | | 硬件定时器 | &plusmn;1us | 低 | 多任务系统 | | 指令循环 | &plusmn;5us | 100% | 非关键延时 | ---
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值