打印
[CW32F030系列]

Cortex-M内核中的精确延时方法

[复制链接]
392|11
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
claretttt|  楼主 | 2025-4-18 12:02 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
为什么要学习这种延时的方法?
  • 很多时候我们跑操作系统,就一般会占用一个硬件定时器——SysTick,而我们一般操作系统的时钟节拍一般是设置100-1000HZ,也就是1ms——10ms产生一次中断。很多裸机教程使用延时函数又是基于SysTick的,这样一来又难免产生冲突。
  • 很多人会说,不是还有定时器吗,定时器的计时是超级精确的。这点我不否认,但是假设,如果一个系统,总是进入定时器中断(10us一次/1us一次/0.5us一次),那整个系统就会经常被打断,线程的进行就没办法很好运行啊。此外还消耗一个硬件定时器资源,一个硬件定时器可能做其他事情呢!
  • 对应ST HAL库的修改,其实杰杰个人觉得吧,ST的东西什么都好,就是出的HAL库太恶心了,没办法,而HAL库中有一个HAL_Delay(),他也是采用SysTick延时的,在移植操作系统的时候,会有诸多不便,不过好在,HAL_Delay()是一个弱定义的,我们可以重写这个函数的实现,那么,采用内核延时当然是最好的办法啦(个人是这么觉得的)当然你有能力完全用for循环写个简单的延时还是可以的。
  • 可能我说的话没啥权威,那我就引用Cortex-M3权威指南中的一句话——“DWT 中有剩余的计数器,它们典型地用于程序代码的“性能速写”(profiling)。通过编程它们,就可以让它们在计数器溢出时发出事件(以跟踪数据包的形式)。最典型地,就是使用 CYCCNT寄存器来测量执行某个任务所花的周期数,这也可以用作时间基准相关的目的(操作系统中统计 CPU使用率可以用到它)。”
Cortex-M中的DWT
它有一个32位的寄存器叫CYCCNT,它是一个向上的计数器,记录的是内核时钟运行的个数,内核时钟跳动一次,该计数器就加1,精度非常高,决定内核的频率是多少,如果是F103系列,内核时钟是72M,那精度就是1/72M = 14ns,而程序的运行时间都是微秒级别的,所以14ns的精度是远远够的。最长能记录的时间为:60s=2的32次方/72000000(假设内核频率为72M,内核跳一次的时间大概为1/72M=14ns),而如果是H7这种400M主频的芯片,那它的计时精度高达2.5ns(1/400000000 = 2.5),而如果是 i.MX RT1052这种比较牛X的处理器,最长能记录的时间为:8.13s=2的32次方/528000000 (假设内核频率为528M,内核跳一次的时间大概为1/528M=1.9ns) 。当CYCCNT溢出之后,会清0重新开始向上计数。

m3、m4、m7杰杰实测可用(m0不可用)。精度:1/内核频率(s)。
要实现延时的功能,总共涉及到三个寄存器:DEMCR 、DWT_CTRL、DWT_CYCCNT,分别用于开启DWT功能、开启CYCCNT及获得系统时钟计数值。
DEMCR
使能DWT_CYCCNT寄存器之前,先清0。让我们看看DWT_CYCCNT的基地址,从ARM-Cortex-M手册中可以看到其基地址是0xE000 1004,复位默认值是0,而且它的类型是可读可写的,我们往0xE000 1004这个地址写0就将DWT_CYCCNT清0了。



关于CYCCNTENACYCCNTENA Enable the CYCCNT counter. If not enabled, the counter does not count and no event isgenerated for PS sampling or CYCCNTENA. In normal use, the debugger must initializethe CYCCNT counter to 0.
它是DWT控制寄存器的第一位,写1使能,则启用CYCCNT计数器,否则CYCCNT计数器将不会工作。



在这里插入图片描述
综上所述想要使用DWT的CYCCNT步骤:

  • 先使能DWT外设,这个由另外内核调试寄存器DEMCR的位24控制,写1使能
  • 使能CYCCNT寄存器之前,先清0。
  • 使能CYCCNT寄存器,这个由DWT的CYCCNTENA 控制,也就是DWT控制寄存器的位0控制,写1使能
代码实现
/**
  ******************************************************************
  * [url=home.php?mod=space&uid=288409]@file[/url]    core_delay.c
  * [url=home.php?mod=space&uid=187600]@author[/url]  fire
  * [url=home.php?mod=space&uid=895143]@version[/url] V1.0
  * [url=home.php?mod=space&uid=212281]@date[/url]    2018-xx-xx
  * [url=home.php?mod=space&uid=247401]@brief[/url]   使用内核寄存器精确延时
  ******************************************************************
  * @attention
  *
  * 实验平台:野火 STM32开发板  
  * 论坛    :http://www.firebbs.cn
  * 淘宝    :https://fire-stm32.taobao.com
  *
  ******************************************************************
  */

#include "./delay/core_delay.h"   

/*
**********************************************************************
*         时间戳相关寄存器定义
**********************************************************************
*/
/*
在Cortex-M里面有一个外设叫DWT(Data Watchpoint and Trace),
该外设有一个32位的寄存器叫CYCCNT,它是一个向上的计数器,
记录的是内核时钟运行的个数,最长能记录的时间为:
10.74s=2的32次方/400000000
(假设内核频率为400M,内核跳一次的时间大概为1/400M=2.5ns)
当CYCCNT溢出之后,会清0重新开始向上计数。
使能CYCCNT计数的操作步骤:
1、先使能DWT外设,这个由另外内核调试寄存器DEMCR的位24控制,写1使能
2、使能CYCCNT寄存器之前,先清0
3、使能CYCCNT寄存器,这个由DWT_CTRL(代码上宏定义为DWT_CR)的位0控制,写1使能
*/


#define  DWT_CR      *(__IO uint32_t *)0xE0001000
#define  DWT_CYCCNT  *(__IO uint32_t *)0xE0001004
#define  DEM_CR      *(__IO uint32_t *)0xE000EDFC


#define  DEM_CR_TRCENA                   (1 << 24)
#define  DWT_CR_CYCCNTENA                (1 <<  0)


/**
  * @brief  初始化时间戳
  * @param  无
  * @retval 无
  * [url=home.php?mod=space&uid=536309]@NOTE[/url]   使用延时函数前,必须调用本函数
  */
HAL_StatusTypeDef HAL_InitTick(uint32_t TickPriority)
{
    /* 使能DWT外设 */
    DEM_CR |= (uint32_t)DEM_CR_TRCENA;               

    /* DWT CYCCNT寄存器计数清0 */
    DWT_CYCCNT = (uint32_t)0u;

    /* 使能Cortex-M DWT CYCCNT寄存器 */
    DWT_CR |= (uint32_t)DWT_CR_CYCCNTENA;

    return HAL_OK;
}

/**
  * @brief  读取当前时间戳
  * @param  无
  * @retval 当前时间戳,即DWT_CYCCNT寄存器的值
  */
uint32_t CPU_TS_TmrRd(void)
{        
  return ((uint32_t)DWT_CYCCNT);
}

/**
  * @brief  读取当前时间戳
  * @param  无
  * @retval 当前时间戳,即DWT_CYCCNT寄存器的值
  */
uint32_t HAL_GetTick(void)
{        
  return ((uint32_t)DWT_CYCCNT/SysClockFreq*1000);
}


/**
  * @brief  采用CPU的内部计数实现精确延时,32位计数器
  * @param  us : 延迟长度,单位1 us
  * @retval 无
  * @note   使用本函数前必须先调用CPU_TS_TmrInit函数使能计数器,
            或使能宏CPU_TS_INIT_IN_DELAY_FUNCTION
            最大延时值为8秒,即8*1000*1000
  */
void CPU_TS_Tmr_Delay_US(uint32_t us)
{
  uint32_t ticks;
  uint32_t told,tnow,tcnt=0;

  /* 在函数内部初始化时间戳寄存器, */  
#if (CPU_TS_INIT_IN_DELAY_FUNCTION)  
  /* 初始化时间戳并清零 */
  HAL_InitTick(5);
#endif

  ticks = us * (GET_CPU_ClkFreq() / 1000000);  /* 需要的节拍数 */      
  tcnt = 0;
  told = (uint32_t)CPU_TS_TmrRd();         /* 刚进入时的计数器值 */

  while(1)
  {
    tnow = (uint32_t)CPU_TS_TmrRd();  
    if(tnow != told)
    {
        /* 32位计数器是递增计数器 */   
      if(tnow > told)
      {
        tcnt += tnow - told;  
      }
      /* 重新装载 */
      else
      {
        tcnt += UINT32_MAX - told + tnow;
      }

      told = tnow;

      /*时间超过/等于要延迟的时间,则退出 */
      if(tcnt >= ticks)break;
    }  
  }
}



/*********************************************END OF FILE**********************/



使用特权

评论回复
沙发
LOVEEVER| | 2025-4-20 14:51 | 只看该作者
它有一个32位的寄存器叫CYCCNT,它是一个向上的计数器,记录的是内核时钟运行的个数

使用特权

评论回复
板凳
twinkhahale| | 2025-5-8 14:20 | 只看该作者
其实估计还是用定时器的延时方式比较靠谱

使用特权

评论回复
地板
tiakon| | 2025-5-8 15:30 | 只看该作者
一般可以用内部的滴答定时器可以的

使用特权

评论回复
5
yuliangren| | 2025-5-8 16:36 | 只看该作者
这种内核延时是不是就只能靠滴答定时器了啊

使用特权

评论回复
6
nqty| | 2025-5-8 18:09 | 只看该作者
在Cortex-M内核中实现精确延时需结合硬件特性(如SysTick定时器、DWT周期计数器)与软件优化才可以

使用特权

评论回复
7
一切D都好| | 2025-5-8 19:15 | 只看该作者
基础延时方法(低精度/易实现)空循环(忙等待)原理:通过CPU空转消耗固定指令周期实现延时。

使用特权

评论回复
8
wamed| | 2025-5-8 20:34 | 只看该作者
利用Cortex-M内核的24位SysTick定时器,基于系统时钟周期性触发中断。

使用特权

评论回复
9
teaccch| | 2025-5-8 22:28 | 只看该作者
1us级延时(假设SystemCoreClock=72MHz,1计数=13.89ns)。受SysTick优先级限制(可能被更高优先级中断打断)

使用特权

评论回复
10
星星点点didi| | 2025-5-9 09:16 | 只看该作者
高精度延时方案(纳秒级/低抖动)DWT周期计数器,利用Cortex-M3/M4/M7内核的DWT单元中的CYCCNT寄存器,直接读取CPU周期数。

使用特权

评论回复
11
suiziq| | 2025-5-9 11:26 | 只看该作者
硬件定时器,利用外设定时器(如STM32的TIM2-TIM7)实现PWM输出或输入捕获,精度更高。

使用特权

评论回复
12
ewyu| | 2025-5-9 12:50 | 只看该作者
建议禁用优化(如GCC的-O0)或使用volatile避免延时函数被优化掉。

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

69

主题

1534

帖子

0

粉丝