基于STM32F429和Cube的主从定时器多通道输出固定个数的PWM波形
主从定时器的原理已在上篇博文:
基于STM32F429+HAL库编写的定时器主从门控模式级联输出固定个数PWM脉冲的程序
讲解了,这篇重点就讲如何实现多通道的PWM级联输出。
1.软件环境
Keil5,Cube5.21
2.Cube配置
选择定时器3,打开通道1和通道2的PWM输出,然后开启主从模式,触发方式为上升沿触发。
频率和占空比的设置请看上篇博文。
生成的代码 如下
void MX_TIM3_Init(void)
{
TIM_MasterConfigTypeDef sMasterConfig = {};
TIM_OC_InitTypeDef sConfigOC = {}; htim3.Instance = TIM3;
htim3.Init.Prescaler = ;
htim3.Init.CounterMode = TIM_COUNTERMODE_UP;
htim3.Init.Period = -;
htim3.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
htim3.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;
if (HAL_TIM_PWM_Init(&htim3) != HAL_OK)
{
Error_Handler();
}
sMasterConfig.MasterOutputTrigger = TIM_TRGO_UPDATE;
sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_ENABLE;
if (HAL_TIMEx_MasterConfigSynchronization(&htim3, &sMasterConfig) != HAL_OK)
{
Error_Handler();
}
sConfigOC.OCMode = TIM_OCMODE_PWM1;
sConfigOC.Pulse = ;
sConfigOC.OCPolarity = TIM_OCPOLARITY_LOW;
sConfigOC.OCFastMode = TIM_OCFAST_DISABLE;
if (HAL_TIM_PWM_ConfigChannel(&htim3, &sConfigOC, TIM_CHANNEL_1) != HAL_OK)
{
Error_Handler();
}
if (HAL_TIM_PWM_ConfigChannel(&htim3, &sConfigOC, TIM_CHANNEL_2) != HAL_OK)
{
Error_Handler();
}
HAL_TIM_MspPostInit(&htim3); }
选择定时器4,从模式为门控模式,触发时钟为TIM3,即ITR2,内部时钟触发
生成的代码如下:
/* TIM4 init function */
void MX_TIM4_Init(void)
{
TIM_ClockConfigTypeDef sClockSourceConfig = {};
TIM_SlaveConfigTypeDef sSlaveConfig = {};
TIM_MasterConfigTypeDef sMasterConfig = {}; htim4.Instance = TIM4;
htim4.Init.Prescaler = ;
htim4.Init.CounterMode = TIM_COUNTERMODE_UP;
htim4.Init.Period = 0xffff;
htim4.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
htim4.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;
if (HAL_TIM_Base_Init(&htim4) != HAL_OK)
{
Error_Handler();
}
sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;
if (HAL_TIM_ConfigClockSource(&htim4, &sClockSourceConfig) != HAL_OK)
{
Error_Handler();
}
sSlaveConfig.SlaveMode = TIM_SLAVEMODE_GATED;
sSlaveConfig.InputTrigger = TIM_TS_ITR2;
if (HAL_TIM_SlaveConfigSynchro(&htim4, &sSlaveConfig) != HAL_OK)
{
Error_Handler();
}
sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
if (HAL_TIMEx_MasterConfigSynchronization(&htim4, &sMasterConfig) != HAL_OK)
{
Error_Handler();
} }
STM32F429的定时器3通道1,2的PWM复用口
PA6 ------> TIM3_CH1
PA7 ------> TIM3_CH2
3,程序介绍
(1)在主函数里开启定时器的中断功能和PWM输出
TIM_SET_CAPTUREPOLARITY(&htim3,TIM_CHANNEL_1,TIM_ICPOLARITY_RISING) ; // 捕获比较1中断使能
TIM_SET_CAPTUREPOLARITY(&htim3,TIM_CHANNEL_2,TIM_ICPOLARITY_RISING) ; // 捕获比较2中断使能 __HAL_TIM_SET_COMPARE(&htim4,TIM_CHANNEL_1,) ; // 输入通道1的捕获比较值CCR1 ,PWM个数为6400个
__HAL_TIM_SET_COMPARE(&htim4,TIM_CHANNEL_2,) ; // 输入通道2的捕获比较值CCR2 HAL_TIM_OC_Start_IT(&htim4,TIM_CHANNEL_1) ; //开启定时器4通道1的输入捕获中断
HAL_TIM_OC_Start_IT(&htim4,TIM_CHANNEL_2) ; //开启定时器4通道2的输入捕获中断 PostInit(); //这个是为了控制通道2的复用GPIO口PA7,可忽略 HAL_TIM_PWM_Start_IT(&htim3, TIM_CHANNEL_1); //开启定时器3通道1的PWM输出中断
HAL_TIM_PWM_Start_IT(&htim3, TIM_CHANNEL_2); //开启定时器3通道2的PWM输出中断
(2)在PWM中断轮询函数关闭定时器的中断功能和PWM输出
void HAL_TIM_PWM_PulseFinishedCallback(TIM_HandleTypeDef *htim) //重写PWM中断轮询弱函数
{ if(__HAL_TIM_GET_FLAG(htim, TIM_FLAG_CC1) != RESET) //判断是否生成中断标志位SR
{
if(__HAL_TIM_GET_IT_SOURCE(htim, TIM_IT_CC1) !=RESET) //定时器中断使能是否开启
{ __HAL_TIM_CLEAR_FLAG(&htim4, TIM_FLAG_CC1); //清除中断标志位SR
if(HAL_TIM_PWM_Stop_IT(&htim3, TIM_CHANNEL_1)==HAL_OK) //关闭定时器3的通道1的PWM输出
{
HAL_TIM_OC_Stop_IT(&htim4,TIM_CHANNEL_1) ; //关闭定时器4的通道1的输入中断捕获
FLAG1_OK = ; //关闭标志置1 } }
} //下面的通道2同理如此
if(__HAL_TIM_GET_FLAG(htim, TIM_FLAG_CC2) != RESET)
{
if(__HAL_TIM_GET_IT_SOURCE(htim, TIM_IT_CC2) !=RESET)
{ __HAL_TIM_CLEAR_FLAG(&htim4, TIM_FLAG_CC2); //清除标志位 PostDeInit(); //这个是为了控制通道2的复用GPIO口PA7,可忽略 if(HAL_TIM_PWM_Stop_IT(&htim3, TIM_CHANNEL_2)==HAL_OK)
{
HAL_TIM_OC_Stop_IT(&htim4,TIM_CHANNEL_2) ; FLAG2_OK = ;
} } } if( FLAG1_OK == &&FLAG2_OK == )
{
FLAG1_OK = ;
FLAG2_OK = ;
FLAG1_Static =; __HAL_TIM_SET_COUNTER(&htim4,); //如果两个通道都关闭好了,就把计数装载值CNT清零 TIM_CCxChannelCmd(TIM3,TIM_CHANNEL_2,TIM_CCx_ENABLE); //这个是为了控制通道2的复用GPIO口PA7,可忽略 //
//Delay100ms();
} }
重头戏来了!!!如果你发现以上代码,通道1,2的GPIO口只能输出通道1的波形,说明你的中断只是进了通道1的,但通道2也触发了
原因就是两者的中断标志位都被清零了,而且是HAL库自动清零的。
两个被注释掉的函数就是HAL库帮你清零的语句,你手动注释掉就行了。
4.注意事项
(1)小心使用延迟函数,定时器可能和延迟函数有冲突;
(2)如果你发现你不能输出超过255个波形,那就在初始化哪里加个__HAL_TIM_SET_AUTORELOAD(&htim4, 0xffff) ,
既往ARR装载0xffff。
(3)下图的3个函数是用来调控PA7这个PWM复用口的,如果你发现你的PWM波形神奇的自动下降沿计数的话,就用来试下吧。
void PostDeInit(void)
{ GPIO_InitTypeDef GPIO_InitStruct = {}; __HAL_RCC_GPIOA_CLK_ENABLE(); GPIO_InitStruct.Pin =GPIO_PIN_7;
GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
GPIO_InitStruct.Pull = GPIO_PULLDOWN;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;
// GPIO_InitStruct.Alternate = 0x00;
HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); } void PostInit(void)
{ GPIO_InitTypeDef GPIO_InitStruct = {}; __HAL_RCC_GPIOA_CLK_ENABLE(); GPIO_InitStruct.Pin =GPIO_PIN_7;
GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
GPIO_InitStruct.Pull = GPIO_NOPULL;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
GPIO_InitStruct.Alternate = 0x02; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); } TIM_CCxChannelCmd(TIM3,TIM_CHANNEL_2,TIM_CCx_ENABLE);
基于STM32F429和Cube的主从定时器多通道输出固定个数的PWM波形的更多相关文章
- 基于STM32F429+HAL库编写的定时器主从门控模式级联输出固定个数PWM脉冲的程序
硬件设备 42步进电机,步进电机驱动器,正点原子F429开发板 开发软件 keil5,Cube 综述 一般要精准的控制电机,就要控制单片机的引脚输出指定个数的PWM波,有多种可实现的方法 ...
- 基于STM32F429和Cube的ov2640程序
1.ov2640和DCMI介绍 OV2640 是 OV(OmniVision)公司生产的一颗 1/4 寸的 CMOS UXGA(1632*1232)图 像传感器.该传感器体积小.工作电压低,提供单片 ...
- 基于Camera Link和PCIe DMA的多通道视频采集和显示系统
基于Camera Link和PCIe DMA的多通道视频采集和显示系统 在主机端PCIe驱动的控制和调度下,视频采集与显示系统可以同时完成对多个Camera Link接口视频采集以及Camera Li ...
- STM32F103ZET6 用定时器级联方式输出特定数目的PWM(转载)
STM32F103ZET6里共有8个定时器,其中高级定时器有TIM1-TIM5.TIM8,共6个.这里需要使用定时器的级联功能,ST的RM0008 REV12的P388和P399页上有说明对于特定的定 ...
- PHP 定时器 边输出边刷新网页
使用定时器的时候当然想网页能够看到输出,不希望网页直接卡住,定时器结束输出一片. 要做到定时器不卡住输出,只需要两个函数就行了,看下面代码 <?php //定时器测试代码 demo //跟踪定时 ...
- STM32F103ZET6 用定时器级联方式输出特定数目的PWM
STM32F103ZET6 用定时器级联方式输出特定数目的PWM STM32F103ZET6里共有8个定时器,其中高级定时器有TIM1-TIM5.TIM8,共6个. 这里需要使用定时器的级联功能,ST ...
- 基于STM32F030F4P9和STM32 CUBEMX 输出PWM波形
STM32F030F4P9定时器功能比较丰富,在此记录项目中使用其自动输出PWM波形(频率:50HZ).CubeMX配置定时器如下图说明. 在此定时器基础时钟为48MHZ,配置中不做分频处理,预分频系 ...
- STM32 TIM 多通道互补PWM波形输出配置快速入门
platform:stm32f10xxx lib:STM32F10x_StdPeriph_Lib_V3.5.0 前言 在做三相逆变的时候,需要软件生成SVPWM波形,具体的算法需要产生三对互补的PWM ...
- AMQ学习笔记 - 14. 实践方案:基于ZooKeeper + ActiveMQ + replicatedLevelDB的主从部署
概述 基于ZooKeeper + ActiveMQ + replicatedLevelDB,在Windows平台的主从部署方案. 主从部署可以提供数据备份.容错[1]的功能,但是不能提供负载均衡的功能 ...
随机推荐
- 移动端--web开展
近期看到群里对关于 移动端 web开发非常是感兴趣.决定写一个关于 移动端的web开发 概念或框架(宝庆对此非常是纠结).也是由于自己一直从事pc 浏览器 web一直对 移动端的不是非常重视,所以趁此 ...
- linux 下Eclipse for C/C++的不常见设置
设置1:build project的时候,让编译器支持 三字母词. 项目文件右击--> Properties-->C/C++ Build--> Settings 如图设置: 再 C ...
- 不积跬步无以至千里(C语言笔记)
第一章 初始C程序 1.C程序结构 简单来说,一个C程序就是由头文件和函数组成 头文件 一条编译预处理命令:作用是在对C程序进行正式编译 ...
- cross-compile-openssl-windows.sh,cross-compile-curl-windows.sh,cross-compile-zlib-windows.sh,build-zlib-visual-studio-2015-cli.bat
https://gist.github.com/artynet build zlib with Visual Studio CLI toolhttps://gist.github.com/artyne ...
- linux_无秘登录问题(不生效)
1 . 登录1,执行命令 ssh-keygen -t rsa 之后一路回 车,查看刚生成的无密码钥对: cd .ssh 后 执行 ll 2 .把 id_rsa.pub 追加到授权的 key 里面去. ...
- 许多其他C++的class样本
class A{ public: A(){}//构造函数,作用分配类所需的空间 }; int main() { A a; } a它是类A示例! 版权声明:本文博客原创文章.博客,未经同意,不得转 ...
- 关于JSP
JSP的本质 JSP本质上就是Servlet, 正常情况下, 它会在第一次被访问的时候被容器转化成Java代码, 然后再从Java代码编译成.class文件, 之后实际就和Servlet没区别了, 也 ...
- jquery 标签页
<!DOCTYPE html><html lang="en" xmlns="http://www.w3.org/1999/xhtml"> ...
- [WPF]有Focus(), 那Unfocus()呢?
原文:[WPF]有Focus(), 那Unfocus()呢? [WPF]有Focus(), 那Unfocus()呢? 周银辉 我们可以调用Focus()方法,让WPF控件获得焦点, 那我现在不想要焦点 ...
- C++ Boost库简介(一些自己的感受)
boost是一个准标准库,相当于STL的延续和扩充,它的设计理念和STL比较接近,都是利用泛型让复用达到最大化.不过对比STL,boost更加实用.STL集中在算法部分,而boost包含了不少工具类, ...