
这里使用STM32G4系列Nucleo开发板演示如下需求的实现过程。 TIMER2使用3个通道的PWM输出,占空比可能被不时修改。修改的数据通过UART传输过来,UART接收的数据通过DMA传输到内存。新的数据接收后,基于TIMER更新事件触发DMA,利用TIMER的更新事件触发DMA Burst 而一次性用新数据修改3个通道的CCR值以调整PWM输出。每次新数据的接收允许以产生按键动作为准,即每次按键动作允许一次数据更新。
这里有两点要注意,TIM2是32位定时器,3个CCR数据使用32位格式。另外,3个数据的修改基于更新事件一次性修改,不能出现混乱,即不可以某一个时刻3个CCR寄存器的数据不是来自同一批的。为了避免数据混乱这种情况,我这里平常禁止TIMER2的更新事件的产生,只有每次收到新数据后才临时允许更新事件的产生,并在TIMER触发的DMA完成中断里再次禁止更新事件的产生。 使用LPUART从外部接收新数据,选用LPUART主要原因是它的TX/RX刚好跟G4Nucleo开发板的虚拟串口接在一起的,并无其它特别原因。至于按键就选用板载按键PC13,并开启了对应的外部中断。 大致的数据传输流程如下图所示:
关于定时器更新事件的能否产生的控制由TIMER的控制寄存器里的UDIS位决定:
我在代码里对该位的操作,写成宏的方式,便于阅读。
该位默认为0。若该位被软件置1,定时器的更新事件将不能产生,意味着开启预装功能时,影子寄存器内容不能被更新,同时也不能因溢出操作、计数器复位操作而触发相应中断或DMA请求 。 测试代码的里几个主要的基本操作: 一、响应按键事件,启动UART的DMA接收。 HAL_UART_Receive_DMA(&hlpuart1,(uint8_t *)CCR_rx, 12); 二、基于UART事件的DMA接收完成中断,允许TIMER更新事件的产生,并启动基于TIMER更新事件的DMA Burst 传输。 Permit_UpdateEvt; //Updated event permitted HAL_TIM_DMABurst_MultiWriteStart(&htim2, TIM_DMABASE_CCR1,TIM_DMA_UPDATE,(uint32_t *)CCR_rx,TIM_DMABURSTLENGTH_3TRANSFERS,3); 三、基于TIMER事件的DMA完成中断,禁止更新事件的产生,并基于串口通信提示可以接受下次数据更新。 __HAL_TIM_CLEAR_FLAG(&htim2,TIM_FLAG_UPDATE); Forbid_UpdateEvt;//update event forbidden Indicating_CCR_Updated(); 再看看基本的CubeMx配置,配置比较简单。下面是LPUART的配置贴图。
我开启LPUART 的DMA接收,其发送功能使用查询阻塞模式,主要用来输出一些提示信息。 下面是TIM2的一些基本配置截图。开启3个通道的PWM输出和基于TIMER更新事件的DMA传输。
最后,我们来简单验证下。下面截图就是通过串口助手键入新数据后TIMER的输出结果。
整体上,操作流程就是每次按键操作提示可以修改占空比了;串口终端键入新的3个字的数据,基于UART接收事件的DMA传输完成后提示数据收到;基于TIMER事件的DMA完成完成后提示数据更新结束,提示等待下次按键动作。 好,今天的分享就到这里,下次再聊。如果有人想要完整的测试工程代码的话,可以私下留言,只要时间不是过去太久且我这边有保存的话,都可以分享供参考。
|
经验分享 | 基于STM32G4芯片不同程序区的跳转话题
经验分享 | 利用TIMER测量信号周期及占空比的精度话题
经验分享 | Flash 全片自检过程中巧用 Linker 自定义变量
经验分享 | 一个 STM32U5 SBSFU SecureFault 问题定位与分析
经验分享 | 高精度定时器如何实现 PWM 零满占空比的连续切换
经验分享 | STM32G474 勘误手册中 SRAM Write Error 详解
经验分享 | 定时器触发ADC时可能遇到的几种情形
STM32总结概述
实战经验 | 1小时在STM32MPU上运行YOLOv8——训练篇
经验分享 | STM32CubeMX 生成时钟获取函数的分析