STM32F1定时器-PWM输出

这篇具有很好参考价值的文章主要介绍了STM32F1定时器-PWM输出。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

定时器-PWM输出

STM32 PWM工作过程

STM32F1定时器-PWM输出

ARR寄存器决定PWM周期,CCR寄存器决定占空比

通道1为例的PWM输出电路图

STM32F1定时器-PWM输出

CCR1:捕获比较(值)寄存器(x =1,2,3,4):设置比较值。

CCMR1:OC1M[2:0]位:对于PWM方式下,用于设置PWM模式1【110】 或者PWM模式2【111】

CCER:CC1P位:输入/捕获1输出极性。0:高电平有效,1:低电平有效。

CCER:CC1E位:输入/捕获1输出使能。0:关闭,1:打开。

PWM模式1和模式2的区别

捕获/比较模式寄存器 1(TIMx_CCMR1)

STM32F1定时器-PWM输出

有效电平并不是指高电平或者低电平,设置高电平有效还是低电平有效要看CCER寄存器的CC1P位

CCER:CC1P位:输入/捕获1输出极性。0:高电平有效,1:低电平有效。

如果设置为高电平有效,那么在PWM模式1的向上计数时,TIMx_CNT < TIMx_CCR1时通道1输出的就是高电平

如果设置为低电平有效,那么在PWM模式1的向上计数时,TIMx_CNT < TIMx_CCR1时通道1输出的就是低电平

PWM模式

脉冲宽度调制模式可以产生一个由TIMx_ARR寄存器确定频率、由TIMx_CCRx寄存器确定占空比的信号。

在TIMx_CCMRx寄存器中的OCxM位写入’110’(PWM模式1)或’111’(PWM模式2),能够独立地设置每个OCx输出通道产生一路PWM。

必须设置TIMx_CCMRx寄存器OCxPE位以使能相应的预装载寄存器,对应的库函数是:

void TIM_OC2PreloadConfig(TIM_TypeDef*TIMx, uint16_t TIM_OCPreload);

最后还要设置TIMx_CR1寄存器的ARPE位,(在向上计数或中心对称模式中)使能自动重装载的预装载寄存器,对应的库函数是:

void TIM_ARRPreloadConfig(TIM_TypeDef* TIMx,FunctionalState NewState);

自动重载的预装载寄存器

ARPE = 1时的更新事件,意思就是CNT计数器在计数时,在到达ARR寄存器值溢出之前,人为的又往ARR寄存器里写入了新值,那这个新值会立即生效,计数到这个新值时产生溢出

STM32F1定时器-PWM输出

ARPE = 0时更新事件,意思是在CNT计数器计数过程中,在计数溢出之前,人为改变了ARR寄存器的值,写入了新值,那这个新值会在下一个计数周期才生效,在写入新值的当前周期里,还是计数到旧值就产生溢出

STM32F1定时器-PWM输出

void TIM_ARRPreloadConfig(TIM_TypeDe* TIMx, FunctionalState NewStale);

简单的说, ARPE=1,ARR立即生效;APRE=0,ARR下个比较周期生效。

PWM输出库函数

void TIM_OCxInit(TIM_TypeDef* TIMx, TIM_OClnitTypeDef* TIM_OCInitStruct);

结构体类型

typedef struct
{
	uint16_t TM_OCMode;			//PWM模式1或者模式2
    uint16_t TIM_OutputState;	//输出使能/失能
    uint16_t TIM_OutputNState;
	uint16_t TIM_Pulse;			//比较值,写CCRx
    uint16_t TIM_OCPolarity; 	//比较输出极性
    uint16_t TIM_OCNPolarity;
	uint16_t TIM_OCldleState;
    uint16_t TIM_OCNldleState;
} TIM_OCInitTypeDef;

初始化示例

TIM_OCInitStructure.TlM_OCMode = TIM_OCMode_PWM2;		//PWM模式2
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;	//比较输出使能
TIM_OCInitStructure.TIM_Pulse=100;
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;	//输出极性:TIM输出比较极性高
TIM_OC2Init(TIM3,&TIM_OCInitStructure);			//根据T指定的参数初始化外设TIM3 OC2

设置比较值函数:

void TIM_SetCompareX(TIM_TypeDef* TIMx, uint16_t Compare2);

使能输出比较预装载:

void TIM_OC2PreloadConfig(TIM_TypeDef* TIMx,uint16_t TIM_OCPreload);

使能自动重装载的预装载寄存器允许位:

void TIM_ARRPreloadConfig(TIM_TypeDef* TIMx, FunctionalState NewState);

实验

使用定时器3的PWM功能,输出占空比可变的PWM波,用来驱动LED灯,从而达到LED亮度由暗变亮,又从亮变暗,如此循环。

PWM输出配置步骤:

1、使能定时器3和相关IO口时钟。

​ 使能定时器3时钟:RCC_APB1PeriphClockCmd();
​ 使能GPIOB时钟:RCC_ APB2PeriphClockCmd();

2、初始化IO口为复用功能输出。函数: GPIO_Init();

​ GPIO_InitStructure.GPIO Mode = GPIO_Mode_AF_PP;

3、这里我们是要把PB5用作定时器的PWM输出引脚,所以要重映射配置,所以需要开启AFIO时钟。同时设置重映射。

​ RCC_APB2PeriphClockCmd(RCC_APB2Periph_ AFIO,ENABLE);
​ GPIO_PinRemapConfig(GPIO_ PartialRemap_TIM3, ENABLE); //定时器3重映射到相应的引脚

4、初始化定时器: ARR,PSC等: TIM_TimeBaselnit();

5、初始化输出比较参数: TIM_OC2Init();

6、使能预装载寄存器TIM_OC2PreloadConfig(TIM3,TIM_OCPreload_Enable);

7、使能定时器。TIM_Cmd();

8、不断改变比较值CCRx,达到不同的占空比效果: TIM_SetCompare2();

定时器PWM初始化示例,C8T6的板

因为开发板的LED灯引脚PA1同时也是TIM2_CH2引脚,这里使用了引脚复用功能,没有重映射

/**
 * @name   TIM_PWM_Init
 * @brief  定时器PWM初始化
 * @param  arr:自动重装值
 *         psc:时钟预分频系数
 * @retval None
 */
void TIM_PWM_Init(u16 arr,u16 psc)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
    TIM_OCInitTypeDef TIM_OCInitTypeDefStructure;

    //使能定时器2时钟
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);

    //使能GPIOA时钟
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);

    //PA1配置
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;     //设置为复用推挽输出
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    
    //PA1初始化
    GPIO_Init(GPIOA,&GPIO_InitStructure);

    //把PA1用作定时器的PWM输出引脚。初始化定时器
    TIM_TimeBaseInitStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值
    TIM_TimeBaseInitStructure.TIM_Prescaler = psc;//设置用来作为TIMx时钟频率除数的预分频值
    TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;   //TIM向上计数模式
    TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割:TDTS = Tck_tim

    //定时器2初始化
    TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStructure);

    //通道2配置
    TIM_OCInitTypeDefStructure.TIM_OCMode = TIM_OCMode_PWM2;    //PWM模式2
    TIM_OCInitTypeDefStructure.TIM_OutputState = TIM_OutputState_Enable;    //比较输出使能
    TIM_OCInitTypeDefStructure.TIM_Pulse = 100;
    TIM_OCInitTypeDefStructure.TIM_OCPolarity = TIM_OCPolarity_High;    //输出极性:TIM输出比较极性高

    //初始化输出比较参数
    TIM_OC2Init(TIM2,&TIM_OCInitTypeDefStructure);

    //使能预装载寄存器
    TIM_OC2PreloadConfig(TIM2,TIM_OCPreload_Enable);

    //使能定时器
    TIM_Cmd(TIM2,ENABLE);
}

main函数

因为dir初始化为1,所以进入while循环后,led0pwmval++,等led0pwmval增加到大于300时,dir 被置为 0,led0pwmval又开始减少,减到0后,dir置为1,led0pwmval再增加,如此反复

调用TIM_SetCompare2函数往CCR2寄存器写入不断变化的led0pwmval值,输出PWM,LED灯呈呼吸灯的效果文章来源地址https://www.toymoban.com/news/detail-489730.html

#include "PWM.h"
int main()
{
	u16 led0pwmval = 0;
	u8 dir = 1;
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);	//NVIC初始化分组
	LED_Init();		//LED初始化
	delay_init();	//延时初始化
	TIM_PWM_Init(899,0);	//不分频。PWM频率=72000000/900=80Khz
	while(1)
	{
		delay_ms(10);
		if(dir)
		{
			led0pwmval++;
		}
		else
		{
			led0pwmval--;
		}

		if(led0pwmval > 300){dir = 0;}
		if(led0pwmval == 0){dir = 1;}

		//往CCR2寄存器写入不断变化的led0pwmval值,调整比较值,就能输出不断变化的占空比了
		TIM_SetCompare2(TIM2,led0pwmval);
	}
}

到了这里,关于STM32F1定时器-PWM输出的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • STM32F4使用高级定时器(TIM1和TIM8)输出PWM问题

    STM32F4使用高级定时器(TIM1和TIM8)输出PWM时要使用TIM_CtrlPWMOutputs使能PWM输出,否则不会输出PWM波形,这一点是和通用定时器输出PWM不一样的地方,通用定时器是不用配置TIM_CtrlPWMOutputs函数的。、

    2024年02月15日
    浏览(22)
  • 基于STM32F103C8T6的端口重映射及定时器输出PWM控制

    #一、端口重映射原理及部分/完全重映射 #二、端口重映射的配置 #三、PWM控制 ##1、通用定时器输出PWM ##2、PWM的工作原理 ##3、PWM的内部运作机制 ##4、PWM的模式 ##5、自动加载的预载寄存器 #四、定时器输出PWM结构体及库函数的配置 #五、项目硬件 #六、项目代码 一、端口重映射

    2024年02月13日
    浏览(23)
  • 定时器详解 -- 定时器中断、PWM输出 --stm32

    STM32F103系列芯片拥有多种定时器,包括基本定时器、通用定时器和高级定时器,每种定时器都具有一些特定的功能。 向上计数:计数器从0计数到自动重装载值(ARR),然后重新从0开始计数并且产生一个计数器溢出事件。 向下计数:计数器从自动重装载值(ARR)开始向下计数

    2024年02月11日
    浏览(27)
  • GD32F303高级定时器输出互补PWM-开发笔记

    ◼ 总通道数:4; ◼ 计数器宽度:16位; ◼ 时钟源可选:内部时钟,内部触发,外部输入,外部触发; ◼ 多种计数模式:向上计数,向下计数和中央计数; ◼ 正交编码器接口:被用来追踪运动和分辨旋转方向和位置; ◼ 霍尔传感器接口:用来做三相电机控制; ◼ 可编程

    2024年02月09日
    浏览(26)
  • STM32F1定时器(TIM1~TIM8)

    STM32共11个定时器,2个高级控制定时器TIM1和TIM8,4个通用定时器TIM2~TIM5,两个基本定时器TIM6和TIM7,两个看门狗定时器和一个系统滴答定时器Systick. 高级定时器TIM1和TIM8的时钟由APB1产生,其它六个通用定时器的时钟由APB2产生。它们的最大频率都可以配置成系统时钟的频率。 定

    2024年02月15日
    浏览(27)
  • STM32定时器输出PWM脉冲

    一、什么是PWM脉冲? PWM (Pulse Width Modulation) 脉冲宽度调制,通常应用于惯性系统中,( 不是不能即停的设备,因为如果设备不具有关心,那么运动是断断续续的,不具有连续性 ),通过脉宽调节输出不同的波形作用域受控对象。 二、 PWM中三个重要的参数 1.频率=1/Ts ;这个很

    2024年02月15日
    浏览(25)
  • STM32TIM定时器PWM输出比较(适用于通用,高级定时器)

    在定时器中我们最常用的功能就是输出PWM,大多是用在电机控制方面,目前网络上相关资料也有很多,但是,很多不利于我们“现搜现用”我这里不是说我写的有多好,而是你搜索到此类文章时大部分是急于解决目前的问题,一段相关代码和讲解就行,当然不是学习背后的原

    2024年01月25日
    浏览(27)
  • STM32(7)-定时器输出PWM的原理分析

    概念+代码 OC(Output Compare)输出比较 输出比较可以通过比较CNT与CCR寄存器值的关系,来对输出电平进行置1、置0或翻转的操作,用于输出一定频率和占空比的PWM波形 每个高级定时器和通用定时器都拥有4个输出比较通道 高级定时器的前3个通道额外拥有死区生成和互补输出的功

    2024年02月04日
    浏览(27)
  • STM32使用CubeMx配置定时器输出PWM

            项目中经常使用到STM32来输出PWM,每次配置过后过不了多久就会忘记,稍微需要对配置做出修改时都要翻很久的手册,所以决定结合实例把PWM配置的详细步骤记录下来,这样在下次配置时可以很快的捡起来。         本文档的行文结构如下,首先,说明实际需

    2024年02月03日
    浏览(34)
  • STM32使用高级定时器输出互补pwm波

    最近做的一个项目用到stm32,网上查了很多资料,也踩了很多坑,这里记录一下配置的步骤和说明 硬件使用的是stm32h750vbt6; 软件用到了stm32cubemx和keil5; 打开Debug模式方便调试,可以忽略,不影响代码运行 在用cube配置时钟源时,有下面三个选项 Disable(禁用) BYPASS Clock Sou

    2024年02月16日
    浏览(22)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包