STM32快速入门(定时器之输出PWM波形)

news2025/1/11 6:55:52

STM32快速入门(定时器之输出PWM波形)

前言

本节主要讲解STM32利用通用定时器,利用CCR和CNT寄存器,输出指定占空比和频率的PWM波形。其功能的应用有:实现LED呼吸灯的效果、控制步进电机、控制直流电机转速等。

导航

图98 通用定时器框图:

TimerPWM

图片引自STM32 F1XX系列的中文参考手册。在通用定时器章节的定时器架构图中,本章讲解的定时器输出功能位于右下角的红色矩形中。

定时器实现PWM输出的实现细节

参考中文手册,实现细节图125如下:

PWM

它内部实现是:输出模式控制器通过比较TIMx_CCR1(比较捕获寄存器)和TIMx_CNT(计数器)的值,由输出模式控制器来确定输出高(有效)电平,还是低(无效)电平,用户可以通过改变TIMx_CCR1寄存器的值来改变PWM的占空比。这通常会将输出模式控制器配置成PWM模式1或PWM模式2,两种模式就是互为取反的关系,同时这两种模式也是输出模式控制器最常用的配置。

对于原理图125左侧的输出模式控制器:

该部分作用是:控制输出模式控制器的输出行为。这里将输出模式控制器的输出标记为OC1REF。

输出模式控制器有7种配置,这些配置是通过操作 TIMx_CCMR1.OC1M[6:4] 实现,7种配置在中文手册中的描述如下,手册中的描述可能太晦涩,配合江科的表格会更友好:

TIMx_CCMR1__OC1M2

TIMx_CCMR1__OC1M1

对于原理图125右侧:

右侧包括一个极性选择器和输出使能电路。

该部分作用是:1、对输出电压的极性进行控制。2、控制输出电路的使能。

我们可以通过配置 TIMx_CCER.CC1P[1] 控制选择器是直接选择OC1REF波形(输出模式控制器的输出)还是选择OC1REF的反相波形。也就是说,输出模式控制器配置成PWM1/WPM2可以实现对OC1REF的反相,配置 TIMx_CCER.CC1P[1] 间接配置选择器也能实现对OC1REF的反相。通过配置 TIMx_CCER.CC1E[0] 可以实现对输出电路的使能。

由定时器输出PWM的原理可以得出调节占空比的公式,PWM频率就是定时器溢出的周期、占空比就是TIMx_CCR1的值,其计算公式如下:

PWMFrequency

关于定时器的PWM模块还需提一句的是,有三个寄存器存在缓冲寄存器/影子寄存器的概念的,这三个寄存器分别是:ARR、PSC、CCRx。影子寄存器的存在延续旧值的生命周期,这样让旧值继续该时期的使命。如果用户提供的新值立马生效,系统就会出于一种未定义的状态。有了缓冲寄存器/影子寄存器的概念,在一个更新周期中真正起作用的是影子寄存器,而用户想要修改预分频控制寄存器,会先将值写到缓冲器中,待这个更新周期过去,才会将缓冲器的值给到影子寄存器

下面中文手册的的两张时序图可以很好的说明了:

Shadow

缓冲寄存器 ----> 预分频控制寄存器

影子寄存器 ----> 预分频缓冲器

还需注意的是:TIMx_CR1.ARPE[7]TIMx_CCMR1.OC1PE[3]寄存器可以让用户选择ARR、CCRx是否启用影子寄存器的功能,而PSC寄存器默认必须使用影子寄存器的功能,但是用户可以通过TIM_PrescalerConfig函数动态配置计数器的预分频系数,它的第三个参数可以选择TIM_PSCReloadMode_Immediate,这会让定时器立即产生一个更新事件,间接实现了立即更新的效果。

定时器实现PWM输出的步骤

综上,可以总结出配置定时器输出部分的套路:

  1. 我们需要把 TIMx_CCMR1.CC1S[1:0] 配置为00,这样CC1通道就被配置为输出

  2. 通过配置 TIMx_CCMR1.OC1M[6:4] ,这里将输出模式控制寄存器配置成PWM1模式。即110。

  3. 配置原理图右部分是否开启反相 TIMx_CCER.CC1P[1] ,这里配置为0不反相。

  4. 最后使能 TIMx_CCER.CC1E[0] 位,来使能原理图右边的输出使能电路。

一般的话,我门还会配置 TIMx_CCMR1.OC1PE[3]TIMx_CR1.ARPE[7],分别启用TIMx_CCR1、TIMx_ARR寄存器的影子功能。

定时器实现PWM输出的库函数实现

因为我的开发板LED0被焊在了PB5所以,所以需要将定时器的PWM波形输出到PB5上。经过查表需要对TIM3_CH2输出进行一个重映射;此外还要将PB5配置成复用推挽输出的状态。

GPIOCfg

IORemap

定时器PWM输出配置,实例代码如下:

void LunarInitTIM3() {
	TIM_TimeBaseInitTypeDef TIM3_Cfg;
	GPIO_InitTypeDef GPIOB5_Cfg;
	TIM_OCInitTypeDef TIM3_OCCfg;

    // 定时器时基配置   BEGIN

	// 打开TIM3所需要的时钟 APB1
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
	TIM_TimeBaseStructInit(&TIM3_Cfg);
	// 配置使用内部时钟 72M Hz
	TIM_InternalClockConfig(TIM3);

    // 这里配置定时器更新频率是1000HZ
	TIM3_Cfg.TIM_CounterMode = TIM_CounterMode_Up;
	TIM3_Cfg.TIM_Period = 100 - 1;
	TIM3_Cfg.TIM_Prescaler = 720 - 1;

	TIM_TimeBaseInit(TIM3, &TIM3_Cfg);
	// 因为TIM_TimeBaseInit会置TIMx_EGR.UG[0]为1,手动产生一个更新事件,
	// 同时会同步影子寄存器的值,而该更新事件又会产生一个多余的中断,所以,
	// 我们需要在开启中断之前,手动清楚更新事件标志位
	TIM_ClearFlag(TIM3, TIM_FLAG_Update);

    // 定时器时基配置   END


	// 配置GPIO 	BEGIN

	// 开启复用时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);
	// 部分重映射
	GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3, ENABLE);
	// 初始化GPIOB5为推挽复用输出
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);

 	GPIOB5_Cfg.GPIO_Mode = GPIO_Mode_AF_PP;
 	GPIOB5_Cfg.GPIO_Pin = GPIO_Pin_5;
 	GPIOB5_Cfg.GPIO_Speed = GPIO_Speed_2MHz;

 	GPIO_Init(GPIOB, &GPIOB5_Cfg);

	// 配置GPIO 	END


	// 配置TIM3的PWM输出	BEGIN

	TIM_OCStructInit(&TIM3_OCCfg);

	TIM3_OCCfg.TIM_OCMode = TIM_OCMode_PWM1;
	TIM3_OCCfg.TIM_OCPolarity = TIM_OCPolarity_High;
	TIM3_OCCfg.TIM_OutputState = TIM_OutputState_Enable;
	TIM3_OCCfg.TIM_Pulse = 0;

	TIM_OC2Init(TIM3, &TIM3_OCCfg);

	// 配置TIM3的PWM输出	END

	// 使能arr和ccr寄存器的影子功能
	TIM_OC2PreloadConfig(TIM3, TIM_OCPreload_Enable);
	TIM_ARRPreloadConfig(TIM3, ENABLE);

	// 使能更新中断
	// TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE);
	// 开启定时器
	TIM_Cmd(TIM3, ENABLE);
}

int main() {

	// 初始化定时器
	LunarInitTIM3();
    // 初始化系统定时器
	SYSTick_Init();

	int dir = 0, cr = 0;

	while(1) {
		TIM_SetCompare2(TIM3, cr);
		Delay_Ms(20);

		if (dir == 0) {
			cr++;
			if (cr > 99) {
				dir = 1;
				cr = 99;
			}
		} else {
			cr--;
			if (cr < 0) {
				dir = 0;
				cr = 0;
			}
		}
	}

	return 0;
}

实验结果就是PB5处的LED灯实现了呼吸的效果。


本章完结

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1669536.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

GUI Pro - Survival Clean

通过开发生存游戏的经验,我们制作了一个带有科幻概念的GUI包。我们希望这个包对你的项目有所帮助。 主要特征 - 2560x1440分辨率图形 - GUI皮肤,包含布局演示场景和预制件 - 提供各种象形图标和项目图标 - 切片元素和白色元素,可定制尺寸和颜色 - 不包括编码和动画 资产 - 1…

栈和队列OJ练习题及解答

前言 上一篇博客已经讲到了栈和队列的数据结构&#xff0c;概括一下&#xff1a;栈后进先出&#xff08;Last In First Out&#xff09;、队列先进先出&#xff08;First In First Out&#xff09;。那么&#xff0c;接下来就来讲讲&#xff0c;关于栈和队列的相关练习题&#…

蓝桥杯-线性动态规划问题背包问题进阶策略详解-

题目&#xff1a;蓝桥云课-青蛙吃虫 解题代码&#xff1a; #include <iostream> #include<cstring> #include<algorithm> using namespace std;const int N106;int f[N][N]; int a[N]; int t,l,r,k,n;int main() {cin>>t;while(t--){scanf("%d%…

FANUC机器人初始化系统的基本方法和步骤

FANUC机器人初始化系统的基本方法和步骤 首先,在做系统初始化之前,必须做好系统的备份,这里做个镜像备份,更详细的镜像备份步骤可参考以下链接中的内容: FANUC机器人进行全部备份和镜像备份以及加载备份文件的具体操作(图文) 如下图所示,在示教器右边的USB接口上插个…

Android finishInputEvent 流程分析

InputDispatcher将事件分发给应用程序后&#xff0c;将该event放入connection的wq队列中&#xff0c;APP处理完之后&#xff0c;需要发送一个完成的信号给InputDispatcher&#xff0c;以供InputDispatcher来将该event从wq队列中移除。我们来分析这个过程。 我们知道&#xff0…

图文详解:synchronized关键字 及其底层原理

目录 一.线程安全问题 二.synchronized关键字 ▐ synchronized图解 ▐ 可重入锁及图解 ▐ synchronized用于方法上 三.Java标准库中synchronized的使用 四.synchronized的底层实现原理 一.线程安全问题 线程安全是指在多线程环境下&#xff0c;对共享资源的访问不会导致…

【数轮】数论、质数、最大公约数、菲蜀定理

数学 唯一分解定理 n>2都可以表示为质因数的乘方。 令 n a1b1a2b2 … \dots … a1,b1 … \dots …都是质因数&#xff0c;b1,b2 … \dots …是对应质因数的数量。 调和级数 11/2 1/3 1/4 ⋯ \cdots ⋯ 1/ n 约等于 logn。 证明过程&#xff1a; 1/3 1/4 < (1/2) …

程序员最趁手的SVM算法,学完你会哭着感谢努力的自己!下篇.

支持向量机上篇内容更重要。 上篇地址&#xff1a;程序员最趁手的SVM算法&#xff0c;学完你会哭着感谢努力的自己&#xff01;上篇。-CSDN博客 废话不说直接进入主题&#xff1a; 6核贝叶斯支持向量机 核贝叶斯支持向量机通过学习一些已知的例子&#xff0c;并找到一个特殊…

【0DAY】湖南建研工程质量检测系统Download接口处存在任意文件读取漏洞

免责声明&#xff1a;文章来源互联网收集整理&#xff0c;请勿利用文章内的相关技术从事非法测试&#xff0c;由于传播、利用此文所提供的信息或者工具而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者本人负责&#xff0c;所产生的一切不良后果与文章作者无关。该…

赋能数据库智能托管,Akamai 发布首款云计算业务线产品!

为了尽可能地简化数据库管理的复杂性&#xff0c;降低数据库成本&#xff0c;Akamai 在近期推出了首款 DBaaS&#xff08;数据库即服务&#xff09;产品——Linode Managed Database。这一数据库产品是 Akamai 自3月份收购 Linode 后发布的首款计算业务线产品。 一、更易用的数…

ipa 分区算法分析,图解

参考 Room Segmentation: Survey, Implementation, and Analysis. 分区算法调查&#xff0c;实现以及评估对比 相关论文 分区算法 New Brooms Sweep Clean - An Autonomous Robotic Cleaning Assistant for Professional Office Cleaning 形态分割 Interactive SLAM using …

EPICS database练习

给定一个以下的数据库&#xff1a; # 指定Limit的上限&#xff0c;初始为10&#xff0c;可以通过通道访问进行设置&#xff0c;上限为100 record(ao, "$(P)Limit") { field(DRVH, "100") field(DOL, "10") field(PINI, "YES") }# 一个…

vue3中的computed

一.computed用法 computed 计算属性就是当依赖的属性的值发生变化的时候&#xff0c;才会触发他的更改&#xff1b;如果依赖的值&#xff0c;不发生变化的时候&#xff0c;使用的是缓存中的属性值。 computed 属性是 Vue3 中的一个响应式计算属性&#xff0c;它可以根据其他响应…

谷歌Flank潜藏3年的Github Action供应链攻击

01 简 介 Flank [1] 是谷歌 Firebase Test lab 开源在 Github 的一个项目&#xff0c;用于同时对多个安卓和IOS设备进行测试。2024年4月15号 AWS 安全工程师 Adnan Khan 公布了关于该项目代码仓库 Github Action CI/CD 存在漏洞的细节[2]&#xff0c;漏洞在2020年于此 代码合…

LabVIEW二维码生成与识别

LabVIEW二维码生成与识别 随着数字化时代的快速发展&#xff0c;QR二维码作为一种高效的信息传递和识别手段&#xff0c;已广泛应用于各行各业。利用LabVIEW软件及其NI视觉开发模块(VDM)来实现一个高效的QR二维码生成与识别系统。该系统不仅能够快速生成带有自定义信息的二维码…

未授权访问:VNC未授权访问

目录 1、漏洞原理 2、环境搭建 3、未授权访问 防御手段 今天继续学习各种未授权访问的知识和相关的实操实验&#xff0c;一共有好多篇&#xff0c;内容主要是参考先知社区的一位大佬的关于未授权访问的好文章&#xff0c;还有其他大佬总结好的文章&#xff1a; 这里附上大…

C++的数据结构(五):树和存储结构及示例

在计算机科学中&#xff0c;树是一种抽象数据类型&#xff08;ADT&#xff09;或是实现这种抽象数据类型的数据结构&#xff0c;用来模拟具有树状结构性质的数据集合。这种数据结构以一系列连接的节点来形成树形结构。在C中&#xff0c;树的概念和存储结构是实现各种复杂算法和…

如何利用AI提高内容生产效率与AIGC典型案例分析

❤️❤️❤️ 欢迎来到我的博客。希望您能在这里找到既有价值又有趣的内容&#xff0c;和我一起探索、学习和成长。欢迎评论区畅所欲言、享受知识的乐趣&#xff01; 推荐&#xff1a;数据分析螺丝钉的首页 格物致知 终身学习 期待您的关注 导航&#xff1a; LeetCode解锁100…

DI-engine强化学习入门(八)如何高效构建决策环境

本文章将介绍 DI-engine 中用于高效构建和标准化不同类型决策环境的系列工具&#xff0c;从而方便我们将各式各样的原始决策问题转化为适合使用强化学习方法解决的形式。 前言 对于“决策环境复杂性”的问题&#xff0c;这里描述的是在强化学习&#xff08;RL&#xff09;领域…

torch 单机 多卡 训练(二)

pytorch.distributed.launch和torchrun的对比 多卡训练 真的烦 并行训练最大的好处&#xff0c;在于GPU内存变大&#xff0c;不是变快 torch.distributed.launch CUDA_VISIBLE_DEVICES0,1,2,3 python -m torch.distributed.launch --nproc_per_node4 --nnodes1 --use_env k…