定时器用来产生PWM输出:

STM32 的定时器除了 TIM6 和 7。其他的定时器都可以用来产生 PWM 输出。其中高级定时器 TIM1 和 TIM8 可以同时产生多达 7 路的 PWM 输出。而通用定时器也能同时产生多达 4路的 PWM 输出,这样,STM32 最多可以同时产生 30 路 PWM 输出!

STM32 PWM工作过程

STMCUBEMX PWM输出Pulse_初始化


STMCUBEMX PWM输出Pulse_重映射应用在PWM_02


STM32 PWM工作过程(通道1为例:

STMCUBEMX PWM输出Pulse_寄存器_03

  • 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 & PWM模式2

寄存器TIMx_CCMR1的OC1M[2:0]位来分析:

STMCUBEMX PWM输出Pulse_重映射应用在PWM_04


STM32 PWM工作过程

STMCUBEMX PWM输出Pulse_初始化_05


STM32 PWM

STMCUBEMX PWM输出Pulse_寄存器_06

void TIM_OC2PreloadConfig(TIM_TypeDef* TIMx, uint16_t TIM_OCPreload);

void TIM_ARRPreloadConfig(TIM_TypeDef* TIMx, FunctionalState NewState);

自动重载的预装载寄存器

STMCUBEMX PWM输出Pulse_初始化_07


STMCUBEMX PWM输出Pulse_PWM_08

  • 捕获/比较模块由一个预装载寄存器和一个影子寄存器组成。读写过程仅操作预装载寄存器。在捕获模式下,捕获发生在影子寄存器上,然后再复制到预装载寄存器中。在比较模式下,预装载寄存器的内容被复制到影子寄存器中,然后影子寄存器的内容和计数器进行比较。
  • 根据TIMx_CR1寄存器中APRE位的设置:APRE=0时,预装载寄存器的内容可以随时传送到影子寄存器,此时二者是连通的;而APRE=1时,在每一次更新事件(UEV)时,才把预装在寄存器的内容传送到影子寄存器。
void TIM_ARRPreloadConfig(TIM_TypeDef* TIMx, FunctionalState NewState);

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

STM32 定时器3输出通道引脚

STMCUBEMX PWM输出Pulse_寄存器_09


PWM输出库函数

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

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

TIM_OCInitStructure.TIM_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);

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);

4、 初始化定时器:ARR,PSC等:

TIM_TimeBaseInit();

5、 初始化输出比较参数:

TIM_OC2Init();

6、 使能预装载寄存器:

TIM_OC2PreloadConfig(TIM3, TIM_OCPreload_Enable);

7、 使能定时器:

TIM_Cmd();

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

TIM_SetCompare2();

PWM 初始化代码:

#include "pwm.h"
#include "sys.h"

void PWM_Init(u16 arr,u16 psc)
{
	GPIO_InitTypeDef GPIO_InitStrue;
	TIM_TimeBaseInitTypeDef TIM_TimStrue;
	TIM_OCInitTypeDef TIM_OCInitStrue;
	
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);//使能相应定时器的时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);//使能相应GPIO的时钟
	
	GPIO_InitStrue.GPIO_Mode=GPIO_Mode_AF_PP;//设置模式为复用输出
	GPIO_InitStrue.GPIO_Pin=GPIO_Pin_5;
	GPIO_InitStrue.GPIO_Speed=GPIO_Speed_50MHz;
	GPIO_Init(GPIOB,&GPIO_InitStrue);//初始化GPIOB.5
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);//开启AFIO时钟
	GPIO_PinRemapConfig(GPIO_PartialRemap_TIM3,ENABLE);//设置重映射
	
	TIM_TimStrue.TIM_ClockDivision=TIM_CKD_DIV1;//设置时钟分割:TDTS = Tck_tim
	TIM_TimStrue.TIM_CounterMode=TIM_CounterMode_Up;
	TIM_TimStrue.TIM_Period=arr;//自动装载值
	TIM_TimStrue.TIM_Prescaler=psc;//预分频系数的设置
	TIM_TimeBaseInit(TIM3,&TIM_TimStrue);//初始化定时器
	
	TIM_OCInitStrue.TIM_OCMode=TIM_OCMode_PWM2;//PWM模式2
	TIM_OCInitStrue.TIM_OCPolarity=TIM_OCPolarity_High;//比较输出极性,TIM输出比较极性高
	TIM_OCInitStrue.TIM_OutputState=TIM_OutputState_Enable;//输出使能 
	TIM_OCInitStrue.TIM_Pulse=100;//比较值,写CCRx
	TIM_OC2Init(TIM3,&TIM_OCInitStrue);//根据指定的参数初始化外设TIM3 OC2(因为是通道2,所以是OC2)
	
	TIM_OC2PreloadConfig(TIM3,TIM_OCPreload_Enable);//使能预装载寄存器
	
	TIM_Cmd(TIM3,ENABLE);//使能定时器
}

主函数代码:

#include "sys.h"
#include "delay.h"
#include "led.h"
#include "pwm.h"
 
 int main(void)
 {	
	u16 led0pwmval=0;
	u8 dir=1;	
	delay_init();	    	 //延时函数初始化	  
 	LED_Init();			     //LED端口初始化
 	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;										 
		TIM_SetCompare2(TIM3,led0pwmval);//不断改变比较值CCRx,达到不同的占空比效果		   
	}	 
 }

以上代码可以实现呼吸灯的效果