STM32 PWM波比较输出

本文详细介绍了如何在STM32上配置GPIO、NVIC中断和定时器,特别是针对TIM_OCMode_Toggle模式下的定时器输出模式初始化。通过计算计数频率和占空比,展示了如何生成特定频率和占空比的PWM波。示例代码中,PA6输出10KHz、80%占空比的PWM,而PA7输出1KHz、20%占空比的PWM。在中断服务函数中,利用捕获值动态调整CCR来实现PWM的频率和占空比控制。

关键配置

  1. 对应GPIO
  2. NVIC中断管理
  3. 定时器初始化
  4. 定时器输出模式TIM_OC初始化(模式为TIM_OCMode_Toggle

频率与占空比的计算

  1. 计数频率的计算
    若使用比较输出,则ARR表示计数的上限,基本无用。计数频率由TIM_Prescaler成员变量配置。若配置TIM_Prescaler=71,输入时钟为72MHz72MHz72MHz,则计数频率为:
    f=72M72+1=1Mf=\frac{72M}{72+1}=1Mf=72+172M=1M
  2. 计数规则
    当计数器数到CCR所储存的值时,输出电平就会翻转。
  3. 频率和占空比的配置
      由计数规律我们可以通过修改CCR值的方法来得到我们想要的频率和占空比。
      比如:计数频率为 1M1M1M,想要得到10K10K10K,占空比为80%80\%80%的PWM波。
      令空闲输出电平为0,关闭影子寄存器以便立即赋值。
      当程序开始,立即进入中断,为CCR赋值为20,此时电平为低电平;
      当计数器数过20,翻转电平,变为高电平,再次进入中断,为CCR赋值为20+80;
      当计数器再数80,翻转电平,变为低电平,再次进入中断,为CCR赋值为20+80+20;

    于是可见,数100为一个周期,其中20为低电平,80为高电平。输出PWM波频率1M1000=10KHz\frac{1M}{1000}=10KHz10001M=10KHz,占空比80%80\%80%.

示例代码

从PA6和PA7输出两路PWM;PA6频率为10KHz,占空比为80% ;PA7频率为1KHz,占空比为20%

PWM驱动配置:

void GPIO_Configuration(void)
{
  GPIO_InitTypeDef GPIO_InitStructure;

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_AFIO,ENABLE);
  // PA1 PA2
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

  GPIO_Init(GPIOA, &GPIO_InitStructure);
}


void NVIC_Configuration(void)
{
  NVIC_InitTypeDef NVIC_InitStructure;

  /* Enable the TIM3 global Interrupt */
  NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  NVIC_Init(&NVIC_InitStructure);
}


void PWM_Config()
{
	TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
	TIM_OCInitTypeDef  TIM_OCInitStructure;
	
	
	GPIO_Configuration();
	NVIC_Configuration();
	
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
  /* Time base configuration */
  TIM_TimeBaseStructure.TIM_Period = 65535;                // 在TIM_OCMode_Toggle模式下ARR几乎无用
  TIM_TimeBaseStructure.TIM_Prescaler = 71;                // 72分频,计数频率为72M/72=1M
  TIM_TimeBaseStructure.TIM_ClockDivision = 0;
  TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;//计数方式为向上计数

  TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);

  /* Output Compare Toggle Mode configuration: Channel1 */
  TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_Toggle;
  TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
  TIM_OCInitStructure.TIM_Pulse = CCR1_Val;
  TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low;   // 空闲状态为低电平
  TIM_OC1Init(TIM3, &TIM_OCInitStructure);

  TIM_OC1PreloadConfig(TIM3, TIM_OCPreload_Disable);         // 关闭影子寄存器,当给CCR赋值时,即刻赋值

  /* Output Compare Toggle Mode configuration: Channel2 */
  TIM_OCInitStructure.TIM_Pulse = CCR2_Val;

  TIM_OC2Init(TIM3, &TIM_OCInitStructure);

  TIM_OC2PreloadConfig(TIM3, TIM_OCPreload_Disable);

	TIM_Cmd(TIM3, ENABLE);
	TIM_ITConfig(TIM3, TIM_IT_CC1 | TIM_IT_CC2, ENABLE);
}

中断代码如下:

extern __IO uint16_t CCR1_Val;
extern __IO uint16_t CCR2_Val;
uint16_t capture = 0;
u8 pa6_state=0,pa7_state=0;

float zhankong1=0.2;
float zhankong2=0.2;
void TIM3_IRQHandler(void)
{
  /* TIM3_CH1 toggling with frequency = 10K Hz */
  if (TIM_GetITStatus(TIM3, TIM_IT_CC1) != RESET)
  {
    TIM_ClearITPendingBit(TIM3, TIM_IT_CC1 );
    capture = TIM_GetCapture1(TIM3);
		if(pa6_state ==0)
		{
			TIM_SetCompare1(TIM3, capture + CCR1_Val*zhankong1 );
			pa6_state = 1;
		}
		else
		{
			TIM_SetCompare1(TIM3, capture + CCR1_Val*(1-zhankong1) );
			pa6_state = 0;			
		}
    
  }

  /* TIM3_CH2 toggling with frequency = 1k Hz */
  if (TIM_GetITStatus(TIM3, TIM_IT_CC2) != RESET)
  {
    TIM_ClearITPendingBit(TIM3, TIM_IT_CC2);
    capture = TIM_GetCapture2(TIM3);
		if(pa7_state ==0)
		{
			TIM_SetCompare2(TIM3, capture + CCR2_Val*zhankong2 );
			pa6_state = 1;
		}
		else
		{
			TIM_SetCompare2(TIM3, capture + CCR2_Val*(1-zhankong2) );
			pa6_state = 0;			
		}
  }
}
### STM32 PWM 输出示例代码及教程 #### 定义 GPIO 引脚并初始化 为了使能PWM功能,需先定义用于PWM输出的GPIO引脚,并将其设置为复用推挽输出模式。这一步骤确保了所选引脚能够正确传输来自定时器模块产生的PWM信号。 ```c GPIO_InitTypeDef GPIO_InitStructure; // 初始化GPIO GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1; // 设置使用的具体引脚编号 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; // 复用推挽输出模式 GPIO_Init(GPIOA, &GPIO📐⚗⚗ 📐⚗⚗ 📐⚗⚗ InitStruct); // 对应端口初始化[^3] ``` #### 配置定时器基础参数 接下来配置定时器的基础参数,这些参数决定了PWM形的关键特性,比如频率和周期长度。这里以通用定时器为例说明如何设定预分频系数以及自动重装载寄存器值。 ```c TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; // 填充定时器结构体成员变量 TIM_TimeBaseStructure.TIM_Period = 999; // 自动重载值决定PWM周期 TIM_TimeBaseStructure.TIM_Prescaler = 71; // 预分频值影响实际工作频率 TIM_TimeBaseStructure.TIM_ClockDivision = 0; // 不启用时钟分割 TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; // 向上计数方式 // 应用上述配置到指定定时器实例 TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure); ``` #### 设定比较单元与开启中断(可选) 对于每个参与PWM生成的通道来说,还需要单独配置其对应的捕获/比较单元。这部分涉及到占空比控制,即通过调整CCRx寄存器中的数值来改变高低电平的时间比例。 ```c TIM_OCInitTypeDef TIM_OCInitStructure; // 准备好OCx通道初始状态描述符 TIM_OCStructInit(&TIM_OCInitStructure); // 进一步定制化该通道的行为特征 TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; // 选择PWM模式1 TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; // 开启相应通道输出 TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; // 正常极性 TIM_OCInitStructure.TIM_Pulse = 500; // 初始脉冲宽度(对应于50%占空比) // 将以上配置应用至目标定时器的一个特定通道之上 TIM_OC1Init(TIM2, &TIM_OCInitStructure); ``` #### 启动PWM输出 完成前面几步之后就可以正式开始PWM形的发送过程了。通常情况下只需要简单调用一次函数即可激活整个流程,在某些应用场景下可能还会涉及额外的状态机逻辑处理。 ```c // 主循环内或其他适当位置执行下面语句触发PWM运作 TIM_Cmd(TIM2, ENABLE); // 启用选定定时器 TIM_CtrlPWMOutputs(TIM2, ENABLE); // 如果有必要的话还要打开高级控制部分开关 ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值