9. 基于TI MSPM0L1306的PWM输出详解与呼吸灯实战
9. 基于TI MSPM0L1306的PWM输出详解与呼吸灯实战最近在玩TI的MSPM0L1306这块板子发现它的PWM功能配置起来挺有意思的尤其是配合官方的SysConfig图形化工具比直接怼寄存器方便多了。很多刚开始接触这块板子的朋友可能会觉得PWM配置有点绕今天我就以最经典的“呼吸灯”实验为例手把手带你从原理到代码把MSPM0L的PWM功能彻底搞明白。咱们这篇教程的目标很明确让你能独立配置MSPM0L的PWM外设并写出一个流畅的呼吸灯效果。过程中我会穿插一些实际项目里容易踩的坑帮你避过去。1. PWM到底是什么一个开关的故事PWM中文叫脉宽调制听起来挺唬人其实原理特别简单。咱们用一个最生活的例子来理解它。想象一下你面前有一个LED灯和一个电灯开关。如果你以非常快的速度比如一秒钟几百次去反复按动这个开关开一下马上关关一下马上开。由于人眼有视觉暂留效应你根本看不到它在闪烁只会觉得这个灯一直亮着但亮度好像比一直开着要暗一些。这就是PWM最核心的思想用数字信号只有开和关来模拟一个中间状态的“模拟量”比如亮度。频率就是你一秒钟内按动开关的次数。次数太少比如一秒一次你就能看到灯在闪烁次数足够多比如一秒几百次你就感觉不到闪烁了。在PWM里这就是PWM波的频率。占空比就是一个周期内开关“打开”的时间占总时间的比例。比如一个周期是1秒你开了0.8秒关了0.2秒那占空比就是80%。占空比越大平均下来灯就越亮占空比越小灯就越暗。所以PWM本质上就是通过快速、精确地控制一个引脚输出高电平开和低电平关的时间比例来实现对电压、电流、功率等模拟量的“无级”调节。在MSPM0L上这个“快速开关”的动作就是由定时器这个外设来精准控制的。2. MSPM0L1306的PWM家底与配置核心在动手写代码前得先摸清咱们手头这块芯片的“家底”。MSPM0L1306内部有4个定时器分为两种通用定时器 (TIMG)功能比较基础适合做定时、PWM输出、输入捕获这些常规操作。高级控制定时器 (TIMA)功能更强大通常用在电机控制等复杂场景。重点来了PWM功能就是基于这些定时器实现的。每个定时器都配有2个独立的PWM输出通道比如TIMG0有通道0和通道1。每个通道可以映射到芯片的特定引脚上而且一个通道可能对应多个备选引脚。举个例子手册里说TIMG0的通道0TIMG_C0可以映射到PA5或者PA12脚。这意味着你可以根据电路板布线自由选择用PA5还是PA12来输出PWM波非常灵活。咱们这次呼吸灯实验用的就是板载LED所在的PA14引脚它属于某个定时器的某个通道具体是哪个需要在SysConfig里查看和配置。PWM有两种对齐模式理解它们对后续配置有帮助边沿对齐模式定时器向下计数。周期开始时输出高电平当计数值降到与咱们设置的“比较值”相等时输出翻转为低电平直到本周期结束。这是最常用、最直观的模式。中心对齐模式定时器先向上再向下计数。输出波形在比较值点会翻转两次产生的PWM波关于中心对称常用于电机驱动可以减少谐波。对于呼吸灯我们用边沿对齐模式就足够了。3. 手把手配置用SysConfig工具生成PWM代码TI的SDK提供了一个强大的图形化配置工具——SysConfig。它能帮我们自动生成底层的外设初始化代码省去了查手册、算寄存器值的麻烦。咱们的配置就靠它了。3.1 打开工程与SysConfig首先在Keil MDK里打开SDK提供的empty例程工程。这个工程可以理解为一个干净的“模板”。在Keil的工程文件列表里找到并双击打开empty.syscfg文件。这时SysConfig的图形化配置界面就会自动弹出来或者从Window菜单里也能打开。提示这个empty工程已经预先配置好了一些基础外设比如1ms的滴答定时器SysTick、1秒中断的通用定时器、串口0、一个按键和一个LED的GPIO。这些配置都是从前面章节的例程继承过来的咱们可以直接用。3.2 配置PWM参数现在开始配置呼吸灯的核心——PWM。添加PWM外设在SysConfig界面左侧的“Peripherals”外设列表中找到并点击TIMER-PWM选项卡。然后点击ADD按钮为工程添加一个PWM外设实例。系统可能会自动分配一个定时器比如TIMG0给它。设置基本参数点击新添加的PWM实例例如TIMER_0在右侧的属性面板中进行设置。咱们的目标是做一个呼吸灯需要关注这几个参数频率 (Frequency)设为100 Hz。这个频率远高于人眼能分辨的闪烁频率约80Hz所以看起来就是连续变亮变暗不会有闪烁感。占空比 (Duty Cycle)初始值可以先设为0%也就是灯全灭开始。周期计数值 (Period)这个值决定了PWM的分辨率。我们设为1000。这意味着一个PWM周期会被分成1000份我们可以通过设置0-999之间的比较值来精细调节占空比。比较值越大高电平时间越长灯越亮。比较值 (Compare Value)初始也设为1000与周期值相同。在边沿对齐模式下这会导致输出一直为低因为计数值从1000开始向下一开始就匹配了比较值立刻翻转为低。我们后面会在代码里动态修改这个值。配置输出引脚关键一步我们需要把PWM信号输出到正确的引脚。在PWM配置中找到通道0Channel 0的引脚配置选项。从下拉列表中选择我们板载LED所连接的引脚——PA14。重要因为PA14现在被分配给了PWM功能它就不能再被当作普通的GPIO来控制了。所以我们需要回到左侧的GPIO配置选项卡下找到之前可能配置为LED输出的PA14设置把它删除或禁用。现在整个工程里应该只剩下一个按键的GPIO配置了。保存并生成代码点击SysConfig顶部的保存按钮然后回到Keil MDK点击编译按钮或按F7。SysConfig会自动根据你的图形化配置生成对应的C语言初始化代码在ti_msp_dl_config.c/.h文件中。这一步非常重要它把我们点的“按钮”变成了实实在在的寄存器配置。4. 编写呼吸灯核心代码配置工具已经把硬件底层的工作都做好了接下来就是写应用程序逻辑让LED“呼吸”起来。我们打开包含main函数的empty.c文件。呼吸灯的原理很简单在一个PWM周期内让比较值决定高电平时间从0慢慢增加到最大值灯渐亮然后再从最大值慢慢减小到0灯渐暗如此循环。这里有个细节我们之前把PWM周期设为了1000所以比较值的有效范围就是0到999。当比较值为0时占空比0%灯全灭为999时占空比约99.9%灯最亮。下面是完整的main函数代码我加了详细注释#include ti_msp_dl_config.h // 定义一个全局变量用于实现精准的毫秒级延时 volatile unsigned int delay_times 0; // 利用SysTick滴答定时器中断实现的毫秒延时函数 void delay_ms(unsigned int ms) { delay_times ms; while(delay_times ! 0); // 等待中断服务函数将delay_times减为0 } int main(void) { int i 0; // 循环变量用于控制PWM比较值 // 初始化函数由SysConfig工具生成包含了PWM、GPIO等所有外设的初始化 SYSCFG_DL_init(); while (1) // 主循环 { // 呼吸灯“吸气”过程灯逐渐变亮 for (i 0; i 999; i) { // 核心函数设置PWM通道的比较值 // 参数1: PWM实例由SysConfig定义例如 TIMER_0_INST // 参数2: 要设置的比较值 (i) // 参数3: 通道索引 (0 表示通道0) DL_TimerG_setCaptureCompareValue(PWM_LED_INST, i, GPIO_PWM_LED_C0_IDX); // 延时1ms控制亮度变化的速度。这个值越小呼吸越快。 delay_ms(1); } // 呼吸灯“呼气”过程灯逐渐变暗 for (i 999; i 0; i--) { DL_TimerG_setCaptureCompareValue(PWM_LED_INST, i, GPIO_PWM_LED_C0_IDX); delay_ms(1); } // 循环回到while(1)开头开始下一次呼吸 } } // SysTick中断服务函数每1ms进入一次 void SysTick_Handler(void) { if(delay_times ! 0) { delay_times--; // 递减延时计数 } }代码关键点解析DL_TimerG_setCaptureCompareValue函数这是TI驱动库提供的API专门用来动态修改PWM的比较值。调用它就能实时改变PWM的占空比从而改变LED亮度。函数里用到的PWM_LED_INST和GPIO_PWM_LED_C0_IDX这些宏都是SysConfig工具根据你的配置自动生成的直接使用即可非常方便。delay_ms函数这里没有用简单的for循环空转来实现延时而是利用了SysTick定时器中断。这样做的好处是“阻塞式”延时更精准且不浪费CPU资源在等待期间CPU可以休眠。务必确保你的工程里已经使能了SysTick定时器empty例程默认是配置好的。循环范围 0-999为什么是999因为我们的PWM周期设置的是1000。在边沿对齐、向下计数的模式下计数值从1000开始向下。当比较值设置为1000时一上来就匹配输出低电平设置为0时永远不匹配因为计数值到不了0这里需要根据具体模式确认通常比较值设为周期值会导致0%占空比。为了安全地获得0%到近100%的占空比我们使用0-999的范围。你可以尝试把i改成1000看看效果。将代码编译、下载到开发板后你应该就能看到板载的LED柔和地渐亮渐灭实现呼吸灯效果了。5. 调试心得与扩展思考第一次跑通这个实验你可能会觉得“哦PWM也不过如此嘛”但这里面有几个点在以后做实际项目时非常有用频率选择驱动LED100Hz完全足够。但如果用来驱动电机频率可能需要在几千Hz到几十kHz以避免电机产生可闻噪音。驱动舵机则需要标准的50Hz周期20msPWM信号。频率的选择需要根据负载特性来定。分辨率我们把周期设为1000就有了1000级的亮度调节分辨率。如果你需要更精细的控制可以增大周期值比如10000但要注意在定时器时钟源固定的情况下周期值越大PWM频率就越低。这是一个需要权衡的地方。引脚复用就像前面说的一个PWM通道可能对应多个引脚。如果你的硬件设计已经固定了一定要在SysConfig里选对引脚。如果发现PWM输出不对首先检查引脚配置和电路连接。从呼吸灯到电机调速这个实验的代码框架完全可以移植到直流电机调速上。你只需要把DL_TimerG_setCaptureCompareValue函数中的i值换成由按键、传感器或者PID算法计算出来的值就能实现闭环速度控制了。PWM是嵌入式控制中最基础也最强大的工具之一。通过这个呼吸灯实验你不仅掌握了MSPM0L上PWM的配置方法更重要的是理解了“数字信号模拟模拟量”的核心思想。下次遇到需要调光、调速、调功率的场景你就能自信地说交给我用PWM搞定。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2419372.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!