4. MSPM0 SysTick滴答定时器实现毫秒级精确延时与LED闪烁实战
4. MSPM0 SysTick滴答定时器实现毫秒级精确延时与LED闪烁实战大家好我是老李一个在嵌入式行业摸爬滚打了十几年的工程师。最近在带几个学生做电赛项目用的正好是TI的MSPM0系列开发板。我发现很多初学者在实现“延时”这个看似简单的功能时要么用for循环空转精度差还浪费CPU要么对系统自带的定时器望而却步。今天我就手把手带大家用MSPM0内核自带的SysTick定时器实现一个精准、好用的毫秒级延时函数并最终用它来让LED灯规规矩矩地闪烁。学完这个你就能为后续驱动传感器、通信等复杂任务打下扎实的时间控制基础。1. 为什么我们需要“精确延时”在单片机编程里“等一会儿”这个需求太常见了。比如让LED闪烁、给传感器一个稳定时间、或者让蜂鸣器发出不同长度的“嘀嘀”声。新手最常用的方法是写个for或while循环让CPU空转数数这也就是常说的“阻塞延时”。注意什么是阻塞延时想象一下你在等一壶水烧开在这期间你啥也干不了只能盯着水壶。阻塞延时就是让整个程序“傻等”CPU被完全占用无法响应其他事件比如按键按下。这在简单任务中勉强能用但在复杂系统中是致命的。所以我们的目标是实现一个不“阻塞”CPU主循环的精确延时。这样CPU在“等待”期间还能抽空去做点别的事情比如扫描键盘、刷新显示。SysTick定时器就是帮我们实现这个目标的得力助手。2. 认识我们的助手SysTick滴答定时器SysTick是ARM Cortex-M系列处理器包括MSPM0使用的Cortex-M0内核内部自带的一个简易定时器。你可以把它想象成单片机内部的“心脏起搏器”或“节拍器”。它有几个关键特点对咱们初学者特别友好24位向下计数器像一个倒计时的秒表从你设定的值开始减减到0就触发一个事件。自动重载减到0后它会自动把最初设定的值重新装进去开始下一轮倒计时周而复始。与内核时钟绑定它的计时基准是内核的主时钟MCLK。在MSPM0上默认是32MHz。这意味着它的计时非常精准不受你程序里其他代码的影响。标准统一所有用Cortex-M内核的芯片都有它所以今天学的代码以后换别的M系列芯片也几乎能直接用移植性超好。它的工作原理很简单我们告诉SysTick“请你每数N个时钟周期就提醒我一次”。因为时钟周期是固定的比如32MHz下每个周期是1/32,000,000秒所以我们就能得到非常精确的时间间隔。3. 实战第一步用SYSCONFIG工具配置SysTickTI为MSPM0提供了非常方便的图形化配置工具——SYSCONFIG。咱们不用死记硬背复杂的寄存器点点鼠标就能完成大部分设置。3.1 打开配置工程首先确保你安装了TI的SDK并在Keil MDK中打开一个空白工程例如empty工程。在Keil的工程文件列表里找到并双击empty.syscfg文件。这个文件就是图形化配置的入口。文件打开后SYSCONFIG的配置界面通常会随之打开。如果没有你也可以在文件上右键选择打开方式。3.2 添加并配置SysTick在SYSCONFIG界面左侧你会看到一系列外设列表。找到“SYSTICK”并点击它然后点击“ADD”按钮这样就把SysTick定时器添加到我们的项目配置中了。接下来是关键的计算和配置时钟源默认使用MCLK频率为32MHz。计算重载值我们想要SysTick每1毫秒ms提醒我们一次。时钟周期 1 / 32,000,000 Hz ≈ 0.00000003125 秒 (31.25纳秒)1毫秒需要的周期数 0.001秒 / (1/32,000,000)秒 32,000配置参数在右侧的配置面板中将Reload Value重载值设置为32000。使能中断勾选Enable Interrupt使能中断。这样当计数器减到0时就会触发中断跳转到我们指定的函数里执行代码。配置完成后记得点击保存。然后回到Keil点击编译按钮。SYSCONFIG工具会根据你的图形化设置自动生成对应的底层初始化代码在ti_msp_dl_config.c/h中非常省心。4. 编写非阻塞的精确延时函数配置好了硬件接下来就是写代码逻辑了。我们的目标是调用delay_ms(1000)程序就能精确等待1秒钟同时又不卡住CPU。这里需要一个关键变量和两个函数配合4.1 核心变量delay_times这是一个全局变量用来记录还需要延时多少毫秒。它必须用volatile关键字修饰。volatile unsigned int delay_times 0;提示volatile是告诉编译器“这个变量可能被程序以外的力量改变比如中断函数你优化代码的时候别动它每次都要老老实实地从内存里读取最新值。” 在中断和主程序共享变量时这是必须的。4.2 延时函数delay_ms()这个函数是给主程序调用的接口。// 搭配滴答定时器实现的精确ms延时 void delay_ms(unsigned int ms) { delay_times ms; // 1. 设置需要延时的毫秒数 while(delay_times ! 0); // 2. 等待直到中断函数把delay_times减为0 }函数逻辑很清晰你告诉我要等多少毫秒我就把这个数存起来然后原地等待直到这个数被别人变成0。4.3 中断服务函数SysTick_Handler()这个函数是SysTick定时器的“中断服务程序”。每当1毫秒时间到CPU就会暂停手头的工作立刻跳转过来执行它。// 滴答定时器中断服务函数 void SysTick_Handler(void) { if(delay_times ! 0) { delay_times--; // 每过1ms就把需要等待的时间减1 } }它的工作就是“偷偷地”帮主程序倒计时。每过1毫秒它就进来把delay_times这个变量减1。三者如何协作主程序调用delay_ms(1000)将delay_times设为1000然后进入while循环等待。SysTick定时器独立运行每过1毫秒就触发一次中断执行SysTick_Handler。中断函数将delay_times从1000减到999、998……直到0。当delay_times变为0时主程序里的while循环条件不成立跳出循环延时结束。整个过程主程序只是在“检查标志”而实际的计时工作由SysTick这个“后台助手”精准完成。这就是非阻塞延时的核心思想。5. 终极实验让LED灯精准闪烁理论说得再多不如动手试一下。现在我们把上面学到的延时函数用起来控制一个LED灯以1秒的间隔闪烁。假设你的开发板上LED连接在某个GPIO引脚上并且在SYSCONFIG里已经配置好这个引脚为输出并定义了宏LED1_PORT和LED1_PIN_14_PIN。下面是完整的main.c示例代码#include ti_msp_dl_config.h volatile unsigned int delay_times 0; // 精确毫秒延时函数 void delay_ms(unsigned int ms) { delay_times ms; while(delay_times ! 0); } int main(void) { // 初始化系统和外设SYSCONFIG生成的代码 SYSCFG_DL_init(); while(1) { // 点亮LED DL_GPIO_setPins(LED1_PORT, LED1_PIN_14_PIN); delay_ms(1000); // 等待1000毫秒 // 熄灭LED DL_GPIO_clearPins(LED1_PORT, LED1_PIN_14_PIN); delay_ms(1000); // 再等待1000毫秒 // 如此循环LED便以1秒为周期闪烁 } } // SysTick中断服务函数必须用这个名字 void SysTick_Handler(void) { if(delay_times ! 0) { delay_times--; } }烧录代码到开发板你就能看到LED灯以非常精确的1秒亮、1秒灭的节奏开始闪烁了。你可以尝试修改delay_ms(1000)中的参数比如改成500LED就会闪烁得更快。6. 常见问题与进阶思考中断函数名必须正确SysTick_Handler这个名字是ARM Cortex-M架构规定的不能随意更改否则编译器找不到中断入口你的延时函数就无法工作。精度验证如果你有逻辑分析仪或者示波器可以测量一下GPIO引脚输出的高低电平时间会发现它非常接近你设定的1000毫秒远比用循环空转实现的延时要精准得多。进阶应用这个delay_times机制其实是一个简单的“软件定时器”雏形。你可以扩展它维护多个定时变量来实现多个不同时间的定时任务让你的程序在等待多个延时的同时还能高效地处理其他事务。好了关于MSPM0上使用SysTick实现毫秒级延时的方法就给大家介绍到这里。这个技能是嵌入式开发的基石之一务必亲手实践一遍。以后遇到需要“定时”或“延时”的场景就不用再写那些不靠谱的空循环了。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2419142.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!