欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

【STM32】PWM程序

程序员文章站 2024-02-22 18:36:23
...

00. 目录

01. PWM简介

脉冲宽度调制(PWM),是英文“Pulse Width Modulation”的缩写,简称脉宽调制,是利用微处理器的数字输出来对模拟电路进行控制的一种非常有效的技术。简单一点,就是对脉冲宽度的控制,PWM 原理如图 14.1.1 所示:
【STM32】PWM程序

图 14.1.1 就是一个简单的 PWM 原理示意图。图中,我们假定定时器工作在向上计数 PWM模式,且当 CNT<CCRx 时,输出 0,当 CNT>=CCRx 时输出 1。那么就可以得到如上的 PWM示意图:当 CNT 值小于 CCRx 的时候,IO 输出低电平(0),当 CNT 值大于等于 CCRx 的时候,IO 输出高电平(1),当 CNT 达到 ARR 值的时候,重新归零,然后重新向上计数,依次循环。改变 CCRx 的值,就可以改变 PWM 输出的占空比,改变 ARR 的值,就可以改变 PWM 输出的频率,这就是 PWM 输出的原理。

STM32F4 的定时器除了 TIM6 和 7。其他的定时器都可以用来产生 PWM 输出。其中高级定时器 TIM1 和 TIM8 可以同时产生多达 7 路的 PWM 输出。而通用定时器也能同时产生多达 4路的 PWM 输出!这里我们仅使用 TIM14 的 CH1 产生一路 PWM 输出。

02. 硬件设计

用到的硬件资源有:

1) 指示灯 DS0
2) 定时器 TIM14
这两个我们前面都已经介绍了,因为 TIM14_CH1 可以通过 PF9 输出 PWM,而 DS0 就是直接在 PF9 上面的,所以电路上并没有任何变化。

03. 配置步骤

3.1 开启 TIM14 和 和 GPIO 时钟,配置 PF9 选择复用功能 AF9 (TIM14)输出。

要使用 TIM14,我们必须先开启 TIM14 的时钟。这里我们还要配置 PF9 为复用(AF9)输出,才可以实现 TIM14_CH1 的 PWM 经过 PF9输出。 库函数使能 TIM14 时钟的方法是:

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM14,ENABLE); //TIM14 时钟使能

我们还要使能 GPIOF 的时钟。然后我们要配置 PF9 引脚映射至 AF9,复用为定时器 14,调用的函数为:

GPIO_PinAFConfig(GPIOF,GPIO_PinSource9,GPIO_AF_TIM14); //GPIOF9 复用为定时器 14

这里还需要说明一下,对于定时器通道的引脚关系,大家可以查看 STM32F4 对应的数据手册,比如我们 PWM 实验,我们使用的是定时器 14 的通道 1,对应的引脚 PF9 可以从数据手册表中查看:
【STM32】PWM程序

3.2 初始化 TIM14, 设置 TIM14 的 ARR 和 和 PSC 等参数。

在开启了 TIM14 的时钟之后,我们要设置 ARR 和 PSC 两个寄存器的值来控制输出 PWM的周期。当 PWM 周期太慢(低于 50Hz)的时候,我们就会明显感觉到闪烁了。因此,PWM周期在这里不宜设置的太小。这在库函数是通过 TIM_TimeBaseInit 函数实现的,

TIM_TimeBaseStructure.TIM_Period = arr; //设置自动重装载值
TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置预分频值
TIM_TimeBaseStructure.TIM_ClockDivision = 0; //设置时钟分割:TDTS = Tck_tim
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //向上计数模式
TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据指定的参数初始化 TIMx 的

3.3 设置 TIM14_CH1 的 PWM 模式 ,能 使能 TIM14 的 CH1 输出。

我们要设置 TIM14_CH1 为 PWM 模式(默认是冻结的),因为我们的 DS0 是低电平亮,而我们希望当 CCR1 的值小的时候,DS0 就暗,CCR1 值大的时候,DS0 就亮,所以我们要通过配置 TIM14_CCMR1 的相关位来控制 TIM14_CH1 的模式。在库函数中,PWM 通道设置是通过函数 TIM_OC1Init()~TIM_OC4Init()来设置的,不同的通道的设置函数不一样,这里我们使用的是通道 1,所以使用的函数是 TIM_OC1Init()。

void TIM_OC1Init(TIM_TypeDef* TIMx, TIM_OCInitTypeDef* TIM_OCInitStruct)

TIM_OCInitTypeDef类型结构体

typedef struct
{
    uint16_t TIM_OCMode;
    uint16_t TIM_OutputState;
    uint16_t TIM_OutputNState; */
    uint16_t TIM_Pulse;
    uint16_t TIM_OCPolarity;
    uint16_t TIM_OCNPolarity;
    uint16_t TIM_OCIdleState;
    uint16_t TIM_OCNIdleState;
} TIM_OCInitTypeDef;

这里我们讲解一下与我们要求相关的几个成员变量:
参数 TIM_OCMode 设置模式是 PWM 还是输出比较,这里我们是 PWM 模式。
参数 TIM_OutputState 用来设置比较输出使能,也就是使能 PWM 输出到端口。
参数 TIM_OCPolarity 用来设置极性是高还是低。
其他的参数 TIM_OutputNState,TIM_OCNPolarity,TIM_OCIdleState 和 TIM_OCNIdleState 是高级定时器才用到的。

TIM_OCInitTypeDef TIM_OCInitStructure; 
TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; //选择模式 PWM
TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能
TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low; //输出极性低
TIM_OC1Init(TIM14, &TIM_OCInitStructure); //根据T指定的参数初始化外设TIM1 4OC1

3.4 使能 TIM14 。

在完成以上设置了之后,我们需要使能 TIM14。使能 TIM14 的方法前面已经讲解过:

TIM_Cmd(TIM14, ENABLE); //使能 TIM14

3.5 修改 TIM14_CCR1 来控制占空比。

在经过以上设置之后,PWM 其实已经开始输出了,只是其占空比和频率都是固定的,而我们通过修改 TIM14_CCR1 则可以控制 CH1 的输出占空比。继而控制 DS0 的亮度。在库函数中,修改 TIM14_CCR1 占空比的函数是:

void TIM_SetCompare1(TIM_TypeDef* TIMx, uint16_t Compare2)

04. 程序示例

pwm.h

#ifndef __PWM_H__
#define __PWM_H__

#include "sys.h"

//定时器初始化
void TIM14_PWM_Init(u32 arr, u16 psc);

#endif /*__PWM_H__*/

pwm.c

#include "pwm.h"


//定时器初始化
void TIM14_PWM_Init(u32 arr, u16 psc)
{
	
	GPIO_InitTypeDef GPIO_Init_Struct;
	
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
	
	TIM_OCInitTypeDef TIM_OCIInitStruct;
	
	//使能定时器TIM14时钟
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM14, ENABLE);

	//使能GPIO时钟 GPIOF
	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOF, ENABLE);
	
	//GPIOF选择位复用功能 GPIOF  PF9
	GPIO_PinAFConfig(GPIOF, GPIO_PinSource9, GPIO_AF_TIM14);
	
	//初始化GPIO
	GPIO_Init_Struct.GPIO_Pin = GPIO_Pin_9;
	GPIO_Init_Struct.GPIO_Mode = GPIO_Mode_AF;
	GPIO_Init_Struct.GPIO_OType = GPIO_OType_PP;
	GPIO_Init_Struct.GPIO_PuPd = GPIO_PuPd_UP;
	GPIO_Init_Struct.GPIO_Speed = GPIO_Speed_50MHz;
	
	GPIO_Init(GPIOF, &GPIO_Init_Struct);
	
	//初始化定时器14
	TIM_TimeBaseInitStruct.TIM_Period = arr;
	TIM_TimeBaseInitStruct.TIM_Prescaler = psc;
	TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;
	TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;
	
	TIM_TimeBaseInit(TIM14, &TIM_TimeBaseInitStruct);
	
	//设置TIM14_CH1为PWM模式
	
	TIM_OCIInitStruct.TIM_OCMode = TIM_OCMode_PWM1;
	TIM_OCIInitStruct.TIM_OutputState = TIM_OutputState_Enable;
	TIM_OCIInitStruct.TIM_OCPolarity = TIM_OCPolarity_Low;
	TIM_OC1Init(TIM14, &TIM_OCIInitStruct);
	
	//使能TIM14在CCR1上的预装载寄存器
	TIM_OC1PreloadConfig(TIM14, TIM_OCPreload_Enable);
	
	//ARPE使能
	TIM_ARRPreloadConfig(TIM14, ENABLE);
	
	//使能TIM14
	TIM_Cmd(TIM14, ENABLE);
	
}


main.c

#include "sys.h"
#include "delay.h"
#include "usart.h"
#include "led.h"
#include "beep.h"
#include "key.h"
#include "exti.h"
#include "iwdg.h"
#include "wwdg.h"
#include "timer.h"
#include "pwm.h"

int main(void)
{ 
	u16 led0Value = 0;
	u8 dir = 1;
	
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	
	delay_init(168);
	
	TIM14_PWM_Init(500 - 1, 84 - 1);

	
	while(1)
	{
		delay_ms(10);
		if (dir)
		{
			led0Value++;
		}
		else
		{
			led0Value--;
		}
		
		if (led0Value > 300)
		{
			dir = 0;
		}
		
		if (0 == led0Value)
		{
			dir = 1;
		}
		
		TIM_SetCompare1(TIM14, led0Value);
	}
	
}


05. 结果验证

我们将看 DS0 不停的由暗变到亮,然后又从亮变到暗。

06. 附录

6.1 【STM32】STM32系列教程汇总

网址:【STM32】STM32系列教程汇总

07. 声明

该教程参考了正点原子的《STM32 F4 开发指南》