PWM的应用可以说非常广泛,控制电机速度、灯光亮度、通信调制等众多领域。

PWM的问题小伙伴问的比较多,最近也在用PWM,这里就分享一下关于PWM的一些内容。

什么是PWM?

PWM:Pulse Width Modulation,脉冲宽度调制。

网上的解释很多,通过下图,你就能直观的理解PWM,其实就是高低电平组成的脉冲信号。

通过改变其中频率(脉冲周期)、占空比,就能应用在很多场合。

PWM常见输出方式

通过上面描述,PWM就是一个IO口以不同的时间周期输出高、低电平。

1.新手(菜鸟)级别

while循环中,阻塞延时,控制IO口高低输出:

while(1)
{IO口高电平Delay阻塞延时IO口低电平Delay阻塞延时
}

阻塞延时可以是:软件模拟延时,定时器阻塞延时等。

2.入门(初级)级别

while循环中,非阻赛延时,控制IO口高低输出:

while(1)
{IO口高电平Delay非阻塞延时IO口低电平Delay非阻塞延时
}

非阻赛延时可以是:定时器标识检测、RTOS(系统)延时等。

3.熟悉(中级)级别

定时器中断控制IO高低电平输出:

定时器中断配置 ——> 启动定时器 ——> 响应中断,控制IO高低电平···

4.熟练(中级+)级别

定时器PWM硬件控制输出:

配置PWM对应的IO,以及定时器PWM输出 ——> 启动PWM自动输出···

void AppTask(void *p_arg)
{PWM_TIM_Configuration();PWM_Output(频率, 占空比);while(1){//自己的应用代码}
}

比较:

上面几种PWM输出方式,前面三种都会CPU干预PWM的输出,也就是会占用CPU资源,特别是前面两种方式,不仅占用CPU,误差还比较大。

使用第三种中断方式,如果频率比较高,CPU消耗的也比较严重。这种情况适合于没有硬件PWM输出的单片机。

第四种就是单片机自带硬件PWM输出功能,只需要简单配置就可以自动输出PWM波形,无需CPU干预。

硬件输出PWM例子

这里以大家熟悉的STM32F1为例:为大家简单分享一下硬件定时器输出PWM波形。

PWM定时器相关宏定义:

//定时器计数时钟(1M次/秒)
#define PWM_COUNTER_CLOCK         1000000//预分频值(与系统时钟、计数值有关)
#define PWM_PRESCALER_VALUE       (SystemCoreClock/PWM_COUNTER_CLOCK - 1)

PWM配置:

/*** @brief  定时器PWM输出配置* @param  无* @retval 无*/
void PWM_TIM_Configuration(void)
{GPIO_InitTypeDef        GPIO_InitStructure;TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;TIM_OCInitTypeDef       TIM_OCInitStructure;/* 时钟配置 */RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);/* 引脚配置 */GPIO_InitStructure.GPIO_Pin   = GPIO_Pin_0;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode  = GPIO_Mode_AF_PP;GPIO_Init(GPIOA, &GPIO_InitStructure);/* 时基配置 */TIM_TimeBaseStructure.TIM_Prescaler = PWM_PRESCALER_VALUE;         //预分频值TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;        //向上计数TIM_TimeBaseStructure.TIM_Period = 0xFFFF;                         //定时周期(暂定值)TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;            //分频因子TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);/* PWM模式配置 */TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;                  //输出PWM1模式TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;      //使能输出TIM_OCInitStructure.TIM_Pulse = 0;                                 //脉宽值(暂定值)TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;          //输出极性(TIM_OC1对应通道1)TIM_OC1Init(TIM2, &TIM_OCInitStructure);
}

PWM输出函数接口:

/*** @brief  输出PWM* @param  Frequency:频率Dutycycle:占空比* @retval 无*/
void PWM_Output(uint32_t Frequency, uint32_t Dutycycle)
{uint32_t tim_period;uint32_t tim_pulse;tim_period = PWM_COUNTER_CLOCK/Frequency - 1;                      //计算出计数周期(决定输出的频率)tim_pulse  = (tim_period + 1)*Dutycycle / 100;                     //计算出脉宽值(决定PWM占空比)TIM_Cmd(TIM2, DISABLE);                                            //失能TIMTIM_SetCounter(TIM2, 0);                                           //计数清零TIM_SetAutoreload(TIM2, tim_period);                               //更改频率TIM_SetCompare1(TIM2, tim_pulse);                                  //更改占空比(TIM_SetCompare1对应通道1)TIM_Cmd(TIM2, ENABLE);                                             //使能TIM
}

初始化配置,调用函数接口,直接就输出PWM波形了:

void AppTask(void *p_arg)
{PWM_TIM_Configuration();PWM_Output(1000, 20);while(1){//自己的应用代码}
}

输出PWM波形:

说明:

本例使用的是STM32标准外设库,如果要深入理解其中原理,还是建议使用标准外设库。

当然,如果想要快速使用PWM这个功能,不想理解其原理,可以直接使用STM32CubeMX配置生成代码:

配置注意事项

想要更加精确控制,并更加满足应用层的需求,就需要自己一步一步深入了解原理。

下面说几点常见的问题吧。

1.引脚映射

如果你使用的引脚需要映射,就需要配置对应的参数。

比如:STM32F1使用PB11(需要查看数据手册):

需要增加对应的“映射”代码:

//复用功能
RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);//定时器(PWM)引脚映射
GPIO_PinRemapConfig(GPIO_FullRemap_TIM2, ENABLE);

2.频率和占空比精度

如果使用32位定时器的话,频率范围更宽、精度也可以达到更高。比如:频率:0.01Hz、 占空比0.01%等。

如果是16位的话,其中的参数都不能超过16位(65535):

#define PWM_COUNTER_CLOCK         1000000
#define PWM_PRESCALER_VALUE       (SystemCoreClock/PWM_COUNTER_CLOCK - 1)tim_period = PWM_COUNTER_CLOCK/Frequency - 1;                      //计算出计数周期(决定输出的频率)
tim_pulse  = (tim_period + 1)*Dutycycle / 100;                     //计算出脉宽值(决定PWM占空比)

具体可根据自己情况进行配置,比如PWM(定时器)计数时钟、分频值等。

实际应用代码,建议增加各个参数的判断,以防越界(这里为了方便理解,就写的比较简单)。

3.移植代码注意通道

很多人移植别人代码,总说有问题、不能运行等。其实,有一些细节地方根本就没有修改过来,比如这里的想说的通道。

比如:你从这里的PA0(TIM2通道1),更换为PB11(TIM2通道4),与之对应的“通道”函数也需要修改。

如:

TIM_OC1Init(TIM2, &TIM_OCInitStructure);
TIM_SetCompare1(TIM2, tim_pulse);

改为:

TIM_OC4Init(TIM2, &TIM_OCInitStructure);
TIM_SetCompare4(TIM2, tim_pulse);

写的好的代码,应该宏定义函数接口,比如:

#define PWM_TIMx               TIM2
#define PWM_TIM_CLK            RCC_APB1Periph_TIM2         //源代码须对应APB1、APB2
#define PWM_TIM_GPIO_CLK       RCC_APB2Periph_GPIOB
#define PWM_TIM_PIN            GPIO_Pin_11
#define PWM_TIM_GPIO_PORT      GPIOB
#define PWM_TIM_OCxInit        TIM_OC4Init                 //比较输出通道(与引脚相关)
#define PWM_TIM_SetComparex    TIM_SetCompare4             //设置比较值(占空比)

4.更多

STM32都有硬件PWM输出功能,但不同的系列,其配置可能略有一些差异,简单参考官方例程以及手册。

现在大部分单片机都自带有硬件PWM输出功能,硬件的好处就是不用CPU干预。如果没有,可以尝试上面说的定时器中断的方式。

------------ END ------------

后台回复『嵌入式软件设计与开发』『单片机』『STM32』相关文章。

欢迎关注我的公众号回复“加群”按规则加入技术交流群,回复“1024”查看更多内容。

欢迎关注我的视频号:

点击“阅读原文”查看更多分享,欢迎点分享、收藏、点赞、在看。

PWM常见输出方法及避坑指南相关推荐

  1. 机器视觉行业实践技巧 -- OpenCV技巧与方法:避坑指南

    文章大纲 光源的使用 镜头与焦距 OpenCV 代码脚手架 OpenCV 摄像头 分辨率,帧率等 Yolov5 加载摄像头 stream 样例 OpenCV 多摄像头视频处理 OpenCV 同时显示多 ...

  2. 2022年Matlab毕设避坑指南及选题推荐

    大家好,我是你的matlab大师. 2021学年,给众多的matlab方向童鞋们做了许多的课题,其中很多人加我,发现有大部分的同学因为是第一次没经验,由于自己基础不太行,在做大四的课题设计时,往往会找 ...

  3. 2023年Matlab毕设避坑指南及选题推荐

    大家好! 2021-2022学年,接了很多matlab数字图像处理/语音信号处理的单子,其中碰到了很多同学们的遭遇,给大家分享下.可以说因为都是第一次,没经验,往往会贪图便宜而踩坑被骗.基础不好不会做 ...

  4. 怎么把原来的墙拆掉_电视墙避坑指南要收好!拆掉重装太心累...

    电视墙是家里装修最为重要的一个地方. 相信很多人都想要把电视墙装修得简单又大气,而且还是容易搞卫生的整洁类型~但是,电视墙贴砖过程中,有很多坑需要注意,一不小心就要像下面的业主一样,拆掉重装. 业主反 ...

  5. ext列表禁止滑动_后台列表设计避坑指南(下)

    编辑导语:列表页是后台界面的重要组成之一,上篇说了后台列表设计的"搜索"设计(详情见:后台列表设计避坑指南 上):本篇继续讲剩下的两个部分的"坑"和必坑指南,我 ...

  6. 转行产品经理,必看的避坑指南!

    写代码的你,是否感觉加班成狗,产品经理好样很少加班,title还很高大上! 画图的你,是否感觉工作太局限,上升太慢.而产品经理全生命周期管理,职业更具竞争力! 想转行的你,岗位淘汰快,而互联网还有发展 ...

  7. Android避坑指南,Gson与Kotlin碰撞出一个不安全的操作

    本文已经授权「鸿洋」公众号原创首发. 最近发现微信多了个专辑功能,可以把一系列的原创文章聚合,刚好我每周都会遇到很多同学问我各种各样的问题,部分问题还是比较有意义的,我会在周末详细的写demo验证,简 ...

  8. mac下编译android源码避坑指南(新)

    截至目前mac环境下android源码编译最新避坑指南 避坑方法 配置(不说配置的都是耍流氓) 下载 编译 烧录 注意事项 避坑方法 源码.SDK.机型版本一定要清楚,有些特殊的版本需要特殊的方法,官 ...

  9. Python+Selenium 网页自动化 exe 程序编程实现(最全避坑指南)

    前言 在我的日常工作中,经常需要在内网(不连接互联网)的网页版办公系统中进行抓取网页数据.修改表单等大量重复性的操作.我就想是否可以编写出自动化的工具,将这些日常琐碎的操作变得轻松而高效.虽然本人非计 ...

  10. 百度程序员开发避坑指南(3)

    前两期我们分享了日常工作中前端.移动端开发的相关问题,感兴趣的同学可以在文末推荐阅读跳转查看.本期我们分享三个议题:golang对象池减少gc压力.FFmpeg中的并发控制.paddle的静态图和动态 ...

最新文章

  1. Windows程序的基本结构(转)
  2. 程序员职场第二次课笔记 9.9号
  3. [转载]ios简单sqlite使用
  4. 直播平台虚拟币与人民币的关系
  5. 【转】掀起Azure AD的盖头来——深入理解Microsoft Graph应用程序和服务权限声明
  6. Java自动计算迷宫正确路线算法源码
  7. 镜头的分类及选购指南
  8. Linux开机自动启动Tomcat
  9. 如何使用QXDM 的1477项 转化utc时间
  10. 使用HBuilderX软件快速搭建Vue项目
  11. python中fact_python中fact函数是什么及如何使用?
  12. 【精读笔记】JavaScript高级程序设计 第3章 语言基础
  13. 通过“控制面板-程序和附件“查看程序安装位置(终极篇)
  14. html怎么随机设置颜色,设置随机颜色值
  15. NeuralPS2021下载~论文总结~NeurlPS2021论文pdf
  16. poi设置word表格单元格宽度_java poi 设置word 格式如表格一类的
  17. unity AudioToolkit 音频工具包的简介+使用方法
  18. mount指令之-o参数解析
  19. grpc-gateway
  20. Jenkins 构建CI/CD(一看就会)

热门文章

  1. 安装程序将在重启您的计算机后黑屏,解决Windows 10登陆后黑屏问题
  2. DoIP节点连接状态管理
  3. 3.3.2 WTL应用向导安装和项目建立
  4. PostgreSQL安装异常:Problem running post-install step。
  5. perl安装的详细步骤
  6. CTF竞赛进阶 (一) 密码学
  7. Element is missing end tag
  8. 如何使用 Apache 设置反向代理
  9. Echarts 如何实现一张图现切换不同的X轴
  10. VirusTotal 为 Chrome 和 Firefox 发布 VT4Browsers 扩展