01、前言
在之前的文章在《STM32延时函数的四种方法》使用定时器延时,在《如何测量代码运行时间》中提到使用定时器外设计算代码运行时间。文中提到这种方法的明显缺点就是需要占用一个定时器,一些MCU在特定应用场景下定时器外设资源是十分稀缺的。在留言区有位大佬提到可以使用DWT,我就研究了一番。
02、DWT
在Cortex-M里面有一个外设叫DWT(DataWatchpoint andTrace),是用于系统调试及跟踪,DWT的中文名字应该是:数据观察点触发。在STM32用户手册的第32章节Debugsupport (DBG)有如下框图。
明显DWT属于DBG部分的功能,从上图的标题可以看出DWT属于CortexM3内核的,理论上M3内核的MCU都支持的,这个下文会说明。在这里我将其称之为“隐藏的定时器”,因为他可以代替定时器外设实现上文提到延时功能和测量代码运行时间的功能,DWT不能代替定时器的其他功能。
之所以DWT可以实现延时功能,因为它有一个32的计数器CYCCNT,这是一个向上计数的计数器,当它溢出时会自动清零并重新开始向上计数,它的频率就是内核的主频。简单点说,就是内核时钟跳动一下,CYCCNT计数器就加1。
很明显DWT计数器的精度和系统主频有关,我们常用的STM32F103主频一般为72Mhz,STM32F207一般为120Mhz,STM32H7主频一般为400Mhz。以为主频最低为72Mhz的STM32F103为例,精度是1/72M= 14ns,这个精度足以满足大部分延时函数的需求,同样程序的运行时间都是微秒级别的,远远满足测量代码运行时间的要求。
03、DWT的配置
首选使用DWT前必须使能DBG的系统跟踪,控制使能位在DEMCR寄存器的bit24。注意该寄存器详细说明在STM32的用户手册上查不到,需要在CortexM3内核手册查到,在《Cortex-M3权威指南》书中也可以查到。
在使能CYCCNT计数器前,必须先将其清零。下图是从ARM的官方手册《Cortex-M3Technical Reference Manual》中查到的。
使能CYCCNT计数器,其控制位是DWT控制寄存器的第一位,写1使能,则启用CYCCNT计数器,否则CYCCNT计数器将不会工作。
总结一下:
a.先使能DWT外设,由内核调试寄存器DEM_CR的位24控制,写1使能。
b.使能CYCCNT寄存器之前,先清0。
c.使能CYCCNT寄存器,由DWT_CTRL的位0控制,写1使能。
代码如下
//寄存器基地址 #define DWT_CR *(uint32_t*)0xE0001000 #define DWT_CYCCNT *(uint32_t*)0xE0001004 #define DEM_CR *(uint32_t*)0xE000EDFC //定义需使能位 #define DEM_CR_TRCENA (1<<24) #define DWT_CR_CYCCNTENA (1<<0) //DWT init void DWT_init(void) { DEM_CR |= (uint32_t)DEM_CR_TRCENA; DWT_CYCCNT = (uint32_t)0u; DWT_CR |= (uint32_t)DWT_CR_CYCCNTENA; } //get DWT count uint32_t DWT_TS_GET(void) { return((uint32_t)DWT_CYCCNT); }
04、代码
从上文我们得知,我们已经获得了一个32位向上累加的计数器,溢出会自动清零并累加,频率是系统主频。那么我们简单封装下,就可以实现延时函数。以下代码在120Mhz的STM32F207测试。
//使用DWT延时time_ms毫秒 void DWT_Delay_Ms(uint32_t time_ms) { uint32_t old_counter,current_counter; uint32_t delay_ms; old_counter = DWT_TS_GET(); current_counter = DWT_TS_GET(); delay_ms = 0; while(delay_ms<time_ms) { current_counter = DWT_TS_GET(); if(current_counter > old_counter) delay_ms = (current_counter - old_counter)/(SystemCoreClock/1000); else delay_ms = (current_counter + 0XFFFFFFFF - old_counter)/(SystemCoreClock/1000); } }
使用之前的文章《如何测量代码运行时间》测量延时函数是否准确。
DWT_Delay_Ms(100);//延时100ms time_ms=Time_Difference_ms();
如下图,延时函数精确延时,没有问题
实现测量代码运行时长的函数接口
//使用DWT测量函数运行时间 float DTW_Time_Difference_ms(void) { static uint32_t old_counter; uint32_t counter,couter_current; couter_current = DWT_TS_GET(); if(couter_current > old_counter) counter = couter_current - old_counter; else counter = couter_current + 0XFFFFFFFF - old_counter; old_counter = couter_current; return (counter / (SystemCoreClock/1000)); }
使用之前的文章《STM32延时函数的四种方法》精确延时,然后使用DWT测量延时时间。
delay_ms(300);//延时300ms time_ms=DTW_Time_Difference_ms();
如下图,可以精确测量代码运行时间,没有问题。
05、后记
本文使用DWT代替了定时器部分功能,它的优缺点如下:
1、优点是:方便移植,经过测试在M3、M4、M7内核的MCU上都可以使用。
2、缺点是:和定时器一样,都有一个延时的最大时间,测量代码运行时间的最大值。
如果项目使用MCU有空闲的定时器,且不考虑换MCU的话,我个人建议还是使用通用的定时器外设,不要使用DWT,虽然DWT方便移植,但通用定时器外设简单易懂,对于没有了解过这部分知识的小白,看到DWT的延时函数,还需要学习下。
Keil和IAR的工程文件下载地址:
PCB和工程代码开源地址:
https://github.com/strongercjd/STM32F207VCT6
点击查看本文所在的专辑,STM32F207教程