完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
为什么要学习这种延时的方法?
在Cortex-M里面有一个外设叫DWT(Data Watchpoint and Trace),是用于系统调试及跟踪, 它有一个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这种比较牛逼的处理器,最长能记录的时间为: 8.13s=2的32次方/528000000 (假设内核频率为528M,内核跳一次的时间大概为1/528M=1.9ns) 。当CYCCNT溢出之后,会清0重新开始向上计数。 m3、m4、m7杰杰实测可用(m0不可用)。要实现延时的功能,总共涉及到三个寄存器:DEMCR 、DWT_CTRL、DWT_CYCCNT,分别用于开启DWT功能、开启CYCCNT及获得系统时钟计数值。 DEMCR 想要使能DWT外设,需要由另外的内核调试寄存器DEMCR的位24控制,写1使能(划重点啦,要考试!!)。 DEMCR的地址是0xE000 EDFC 关于DWT_CYCCNT 使能DWT_CYCCNT寄存器之前,先清0。 让我们看看DWT_CYCCNT的基地址,从ARM-Cortex-M手册中可以看到其基地址是0xE000 1004,复位默认值是0,而且它的类型是可读可写的,我们往0xE000 1004这个地址写0就将DWT_CYCCNT清0了。 关于CYCCNTENA CYCCNTENA Enable the CYCCNT counter. If not enabled, the counter does not count and no event is generated for PS sampling or CYCCNTENA. In normal use, the debugger must initialize the CYCCNT counter to 0. 它是DWT控制寄存器的第一位,写1使能,则启用CYCCNT计数器,否则CYCCNT计数器将不会工作。 综上所述 想要使用DWT的CYCCNT步骤: 代码实现 /** ****************************************************************** * @file core_delay.c * @author fire * @version V1.0 * @date 2018-xx-xx * @brief 使用内核寄存器精确延时 ****************************************************************** * @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 无 * @note 使用延时函数前,必须调用本函数 */ 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**********************/ #ifndef __CORE_DELAY_H #define __CORE_DELAY_H #include "stm32h7xx.h" /* 获取内核时钟频率 */ #define GET_CPU_ClkFreq() HAL_RCC_GetSysClockFreq() #define SysClockFreq (218000000) /* 为方便使用,在延时函数内部调用CPU_TS_TmrInit函数初始化时间戳寄存器, 这样每次调用函数都会初始化一遍。 把本宏值设置为0,然后在main函数刚运行时调用CPU_TS_TmrInit可避免每次都初始化 */ #define CPU_TS_INIT_IN_DELAY_FUNCTION 0 /******************************************************************************* * 函数声明 ******************************************************************************/ uint32_t CPU_TS_TmrRd(void); HAL_StatusTypeDef HAL_InitTick(uint32_t TickPriority); //使用以下函数前必须先调用CPU_TS_TmrInit函数使能计数器,或使能宏CPU_TS_INIT_IN_DELAY_FUNCTION //最大延时值为8秒 void CPU_TS_Tmr_Delay_US(uint32_t us); #define HAL_Delay(ms) CPU_TS_Tmr_Delay_US(ms*1000) #define CPU_TS_Tmr_Delay_S(s) CPU_TS_Tmr_Delay_MS(s*1000) #endif /* __CORE_DELAY_H */ 注意事项: 使用者如果不是在HAL库中使用,注释掉: uint32_t HAL_GetTick(void) { return ((uint32_t)DWT_CYCCNT/SysClockFreq*1000); } 同时建议重新命名HAL_InitTick()函数。 按照自己的平台重写以下宏定义: /* 获取内核时钟频率 */ #define GET_CPU_ClkFreq() HAL_RCC_GetSysClockFreq() #define SysClockFreq (218000000) 后记 其实在ucos-iii 源码中,有一个功能是测量关中断时间的功能,就是使用STM32的时间戳,即记录程序运行的某个时刻,如果记录下程序前后的两个时刻点,即可以算出这段程序的运行时间。 |
|
|
|
只有小组成员才能发言,加入小组>>
调试STM32H750的FMC总线读写PSRAM遇到的问题求解?
1771 浏览 1 评论
X-NUCLEO-IHM08M1板文档中输出电流为15Arms,15Arms是怎么得出来的呢?
1619 浏览 1 评论
1070 浏览 2 评论
STM32F030F4 HSI时钟温度测试过不去是怎么回事?
724 浏览 2 评论
ST25R3916能否对ISO15693的标签芯片进行分区域写密码?
1673 浏览 2 评论
1936浏览 9评论
STM32仿真器是选择ST-LINK还是选择J-LINK?各有什么优势啊?
729浏览 4评论
STM32F0_TIM2输出pwm2后OLED变暗或者系统重启是怎么回事?
569浏览 3评论
594浏览 3评论
stm32cubemx生成mdk-arm v4项目文件无法打开是什么原因导致的?
552浏览 3评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2024-12-22 18:49 , Processed in 1.005586 second(s), Total 76, Slave 60 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号