当前位置: 首页 > article >正文

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实现毫秒级延时的方法就给大家介绍到这里。这个技能是嵌入式开发的基石之一务必亲手实践一遍。以后遇到需要“定时”或“延时”的场景就不用再写那些不靠谱的空循环了。

相关文章:

4. MSPM0 SysTick滴答定时器实现毫秒级精确延时与LED闪烁实战

4. MSPM0 SysTick滴答定时器实现毫秒级精确延时与LED闪烁实战 大家好,我是老李,一个在嵌入式行业摸爬滚打了十几年的工程师。最近在带几个学生做电赛项目,用的正好是TI的MSPM0系列开发板。我发现很多初学者在实现“延时”这个看似简单的功能时…...

Arduino 入门手册:基于ESP32-S3R8N8的智能硬件开发实战指南

Arduino 入门手册:基于ESP32-S3R8N8的智能硬件开发实战指南 很多刚接触硬件的朋友问我,想做个智能小车或者遥控灯,但看到复杂的电路和寄存器就头疼,有没有什么简单好上手的办法?当然有,那就是Arduino。今天…...

gte-base-zh保姆级教程:从启动到调用,小白也能玩转文本嵌入

gte-base-zh保姆级教程:从启动到调用,小白也能玩转文本嵌入 1. 引言:让机器读懂你的文字 你有没有想过,怎么让电脑理解“苹果”这个词,指的是水果公司还是可以吃的水果?或者,怎么让一个系统知…...

手把手教你用STM32 HAL库实现IIC通信(以AT24C02为例)

STM32 HAL库实战:I2C驱动AT24C02全流程解析与避坑指南 I2C总线在嵌入式系统中就像一条隐形的数据高速公路,连接着各种传感器、存储器和外设。作为STM32开发者,掌握HAL库的I2C操作不仅能提升开发效率,更能避免许多底层调试的"…...

YOLOv5小目标检测实战:手把手教你集成NWD Loss提升模型精度(附完整代码)

YOLOv5小目标检测实战:集成NWD Loss的完整指南与调优策略 在无人机巡检、卫星图像分析等场景中,小目标检测一直是计算机视觉领域的难点。传统IoU(Intersection over Union)指标在面对像素级小目标时,往往因为轻微的定位…...

Obsidian+TeraCloud+WebDAV:零基础搭建25G免费同步网盘(附推荐码)

ObsidianTeraCloudWebDAV:零基础搭建25G免费同步网盘 在信息爆炸的时代,知识管理工具已成为现代人提升效率的刚需。Obsidian作为一款基于Markdown的本地优先笔记应用,凭借其强大的双向链接和知识图谱功能,迅速成为知识工作者的首…...

Ubuntu Server 下 Docker 的快速安装与优化配置指南

1. 环境准备与旧版本清理 在开始安装Docker之前,我们需要确保系统环境干净且具备必要的依赖项。我遇到过不少因为旧版本残留导致的问题,所以建议先执行清理操作。Ubuntu Server默认可能包含旧版docker.io或lxc-docker,这些可能与新版Docker C…...

手眼标定太复杂?试试这款超简单易用的开源标定工具!

1. 为什么你需要这款手眼标定工具? 第一次接触机器人手眼标定时,我被各种数学公式和复杂的标定流程搞得晕头转向。传统的标定方法往往需要手动计算坐标变换矩阵,还要处理各种误差补偿,光是理解原理就要花上好几天。直到发现这个开…...

VMware虚拟机安装Ubuntu部署DeepSeek-OCR-2:完整教程

VMware虚拟机安装Ubuntu部署DeepSeek-OCR-2:完整教程 1. 引言 如果你正在寻找一种在隔离环境中测试OCR模型的方法,那么使用VMware虚拟机安装Ubuntu并部署DeepSeek-OCR-2可能是个不错的选择。这个教程将带你一步步完成整个过程,从VMware配置…...

微表情数据集获取全攻略:从申请到使用的完整指南

1. 微表情数据集的价值与应用场景 微表情研究在心理学、人机交互和安全检测等领域越来越重要。这些短暂的面部表情变化通常只持续1/25到1/5秒,却能揭示人真实的情绪状态。要开展相关研究,高质量的数据集是基础。我刚开始做微表情分析时,花了大…...

PyTorch-2.x-Universal-Dev使用体验:国内源加速的深度学习环境

PyTorch-2.x-Universal-Dev使用体验:国内源加速的深度学习环境 1. 为什么你需要一个“开箱即用”的PyTorch环境? 如果你尝试过从零开始搭建一个PyTorch深度学习环境,大概率经历过这些“痛苦时刻”:花半小时下载几个G的CUDA驱动&…...

解锁Deepin Boot Maker的4大实战价值:打造安全高效的启动盘制作流程

解锁Deepin Boot Maker的4大实战价值:打造安全高效的启动盘制作流程 【免费下载链接】deepin-boot-maker 项目地址: https://gitcode.com/gh_mirrors/de/deepin-boot-maker 当你需要为新电脑安装系统、修复受损操作系统,或测试不同Linux发行版时…...

摄影小白必看:如何用MTF曲线挑选最适合你的镜头(附实战对比)

摄影小白必看:如何用MTF曲线挑选最适合你的镜头(附实战对比) 当你第一次听说"MTF曲线"这个词时,可能会觉得这是专业摄影师才需要了解的复杂概念。但实际上,理解MTF曲线就像掌握了一把钥匙,能帮你…...

BLDC电机控制避坑指南:从霍尔信号处理到PWM调制的5个常见问题

BLDC电机控制避坑指南:从霍尔信号处理到PWM调制的5个常见问题 在无刷直流电机(BLDC)控制系统的开发过程中,工程师们常常会遇到各种棘手的技术挑战。这些问题不仅影响系统性能,还可能导致硬件损坏或控制失效。本文将深入…...

从数据获取到分析应用:ERA5-Land月尺度降水、气温与辐射数据的全流程处理指南

1. ERA5-Land数据基础认知 第一次接触气象数据的朋友可能会被各种专业术语吓到,但ERA5-Land其实就像一台记录地球表面天气变化的超级摄像机。这个由欧洲中期天气预报中心(ECMWF)维护的数据集,以9公里9公里的网格精度,持…...

ARMA模型调参避坑指南:当ACF/PACF都拖尾时如何确定p,q阶数?

ARMA模型调参实战:当ACF/PACF拖尾时的阶数选择策略 引言 在电商销量预测、股票价格分析、能源需求预估等场景中,时间序列建模都是核心工具。许多数据分析师在掌握了ARIMA模型基础后,往往会在ARMA(p,q)阶数选择这个关键步骤上陷入困境——当自…...

如何用乒乓缓存机制优化你的嵌入式系统性能(附代码示例)

如何用乒乓缓存机制优化你的嵌入式系统性能(附代码示例) 在嵌入式系统开发中,实时性和高效性往往是核心诉求。想象一下,当你正在处理高频率的传感器数据或实时音视频流时,任何I/O延迟都可能导致数据丢失或系统卡顿。这…...

Godot 4实战:如何绕过工程目录限制实现动态图片导入(附完整代码)

Godot 4实战:突破工程目录限制的动态图片加载方案 在开发电子书阅读器这类需要动态加载外部资源的应用时,Godot引擎默认的资源管理机制往往会成为绊脚石。最近在开发一个支持多格式电子书的阅读器时,我遇到了一个典型问题:如何在不…...

优化FF14游戏体验:自动过场动画跳过技术解析与实践指南

优化FF14游戏体验:自动过场动画跳过技术解析与实践指南 【免费下载链接】FFXIV_ACT_CutsceneSkip 项目地址: https://gitcode.com/gh_mirrors/ff/FFXIV_ACT_CutsceneSkip 游戏流程中断问题:过场动画的隐性成本分析 在大型多人在线角色扮演游戏中…...

如何利用阿里云镜像加速Deeplearning4j的Maven依赖下载(附完整POM.xml配置)

阿里云镜像加速Deeplearning4j依赖下载实战指南 如果你曾经被Maven依赖下载速度折磨得怀疑人生,那么这篇文章就是为你准备的。作为Java生态中最流行的深度学习框架之一,Deeplearning4j的强大功能背后是一系列复杂的依赖关系,而这些依赖默认从…...

OpenCV轮廓检测实战:5种mode参数效果对比与选型指南(附代码)

OpenCV轮廓检测实战:5种mode参数效果对比与选型指南(附代码) 在计算机视觉项目中,轮廓检测往往是物体识别、形状分析和工业检测的第一步。OpenCV提供的cv2.findContours()函数虽然功能强大,但其mode参数的四种不同选项…...

ChatGPT内容生成指令与范例大全:从零构建高效提示词工程

ChatGPT内容生成指令与范例大全:从零构建高效提示词工程 刚开始接触ChatGPT这类大语言模型时,你是不是也遇到过这样的烦恼?明明想让AI写一篇产品介绍,结果它给你生成了一篇抒情散文;想让它总结技术文档,它…...

Qwen3-14b_int4_awq多场景应用:跨境电商独立站商品页文案AI批量生成

Qwen3-14b_int4_awq多场景应用:跨境电商独立站商品页文案AI批量生成 1. 模型简介与部署验证 Qwen3-14b_int4_awq是基于Qwen3-14b模型的量化版本,采用int4精度和AWQ(Activation-aware Weight Quantization)技术进行优化。这个版本…...

3步打造数据恢复利器:Deepin Boot Maker应急救援指南

3步打造数据恢复利器:Deepin Boot Maker应急救援指南 【免费下载链接】deepin-boot-maker 项目地址: https://gitcode.com/gh_mirrors/de/deepin-boot-maker 当系统崩溃时,你是否遇到过重要文件无法访问的困境?当硬盘分区损坏时&…...

深入解析Redis持久化:RDB与AOF的实战对比与选型指南

1. Redis持久化的重要性与基本概念 想象一下你正在运营一个电商平台,突然服务器断电重启,所有用户购物车里的商品、秒杀活动的库存数据全部消失——这种灾难性场景正是Redis持久化要解决的核心问题。作为内存数据库,Redis的数据默认只存在于R…...

服务器为什么会被攻击?服务器遭受攻击后,如何进行防护

常见的网络攻击 DDoS攻击是最常见的攻击方式,全称为分布式拒绝服务攻击,又称为“洪水式攻击”。DDoS是一种基于DoS的特殊形式的拒绝服务攻击,DoS攻击就是利用合理的服务请求来占用过多的服务资源,从而使合法用户无法得到服务的响…...

再见 MCP

好吧,代理 AI 社区正在经历一次重大的现实检验。 如果你关注开发者动态已有一段时间,模型上下文协议(MCP) 本应是通用标准。 我不会说它没有帮助。我构建了 Splitwise MCP,因为付费升级 pro 不是一个好主意。 它是将…...

CMake项目构建必知:CMAKE_CURRENT_SOURCE_DIR和CMAKE_SOURCE_DIR的实战区别与常见坑点

CMake路径变量深度解析:如何精准掌控CMAKE_CURRENT_SOURCE_DIR与CMAKE_SOURCE_DIR 当你第一次在CMake项目中看到CMAKE_CURRENT_SOURCE_DIR和CMAKE_SOURCE_DIR这两个变量时,可能会觉得它们看起来非常相似——毕竟都包含"SOURCE_DIR"这个部分。但…...

Qwen3-TTS-Tokenizer-12Hz实战教程:语音克隆流程中tokens中间表示

Qwen3-TTS-Tokenizer-12Hz实战教程:语音克隆流程中tokens中间表示 你是不是也好奇,那些能模仿任何人声音的AI语音克隆技术,到底是怎么把一段声音“记住”并“复刻”出来的?秘密就藏在声音的“数字密码”——tokens中间表示里。 …...

SolidWorks到Unity全流程:如何将自定义模型完美导入Unity(含FBX转换避坑指南)

SolidWorks到Unity全流程:工业级模型的高保真迁移指南 当工业设计师与游戏开发者相遇,最大的挑战往往不是创意碰撞,而是技术壁垒的打破。SolidWorks作为工业设计领域的标杆工具,其生成的精密模型如何无损进入Unity的实时渲染世界&…...