STM32学习笔记——PWM基础知识与720电机驱动

前言:

为了方便查看博客,特意申请了一个公众号,附上二维码,有兴趣的朋友可以关注,和我一起讨论学习,一起享受技术,一起成长。

在这里插入图片描述


1. 定义

脉冲宽度调制:是利用微处理器的数字输出来对模拟电路进行控制的一种非常有效的技术,广泛应用在从测量、通信到功率控制与变换的许多领域中。(百度百科)它是一种对模拟信号电平进行数字编码的方法,通过对一系列脉冲的宽度进行调制,来等效地获得所需要波形(含形状和幅值)。

SPWM波形: 脉冲宽度按正弦规律变化而和正弦波等效的 PWM 波形。

在这里插入图片描述

把正弦半波波形分成N等份,就可把正弦半波看成由N个彼此相连的脉冲所组成的波形。如果把上述脉冲序列用同样数量的等幅而不等宽的矩形脉冲序列代替,使矩形脉冲的中点和相应正弦等分的中点重合,且使矩形脉冲和相应正弦部分面积(即冲量)相等,就得到一组脉冲序列,这就是PWM波形。根据冲量相等,效果相同的原理,PWM波形和正弦半波是等效的,如上图。

pwm的频率:指每秒钟信号从高电平到低电平再回到高电平的次数。

占空比: 输出的PWM中,高电平保持的时间与该 PWM 的时钟周期的时间之比。

在这里插入图片描述

分辨率: 是占空比最小能达到多少,如8位的PWM:理论的分辨率就是1:255(单斜率), 16位的的PWM理论就是1:65535(单斜率)。

PWM 是一种对模拟信号电平进行数字编码的方法。通过高分辨率计数器的使用,方波的占空比被调制用来对一个具体模拟信号的电平进行编码。PWM信号仍然是数字的,因为在给定的任何时刻,满幅值的直流供电要么完全有(ON),要么完全无(OFF)。电压或电流源是以一种通(ON)或断(OFF)的重复脉冲序列被加到模拟负载上去的。通的时候即是直流供电被加到负载上的时候,断的时候即是供电被断开的时候。只要带宽足够,任何模拟值都可以使用PWM进行编码。

PWM 调制出的交流电压值主要取决于我们所控制的占空比(电压 = 占空比 X 幅值),调制出来的交流电压的波形精度取决于所设定的载波频率。

2. 电机驱动

对于 PWM 波的输出配置,可以参考STM32学习笔记一一PWM 输出

2.1 电路连接:

在这里插入图片描述

2.2 软件实现:
头文件:
#ifndef __MOTOR__H_
#define __MOTOR__H_

#include "system.h"

#define MOTOR1_PWM	GPIO_Pin_6

#define MOTOR_PWMMAX	1000

void MOTOR_GPIO_Init(void);
void TIM3_PWM_Init(void);
void MOTOR_Control(int16_t motor1_pwm);

#endif

/******************************************************************************************
* 函  数:void MOTOR_GPIO_Init(void)
* 功  能:电机引脚初始化
* 参  数:无
* 返回值:无
* 备  注:TIM3 CH1(PWM1) -> PA6
*******************************************************************************************/
void MOTOR_GPIO_Init(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;

	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);

	GPIO_InitStructure.GPIO_Pin = MOTOR1_PWM;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;	//	GPIO复用推挽输出
	GPIO_Init(GPIOA,&GPIO_InitStructure);
	//GPIO_SetBits(GPIOA,MOTOR1_PWM);
}

/******************************************************************************************
* 函  数:void TIM3_PWM_Init(void)
* 功  能:定时器输出和PWM配置
* 参  数:无
* 返回值:无
* 备  注:TIM3 CH1
*******************************************************************************************/
void TIM3_PWM_Init(void)
{
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;	//定时器变量
	TIM_OCInitTypeDef TIM_OCInitStructure;	//输出比较结构体变量
	
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
	
	TIM_TimeBaseInitStructure.TIM_Period = 1000-1;	//设置自动重装载的周期值;f=72M/1000=72KHz
	TIM_TimeBaseInitStructure.TIM_Prescaler = 100;	//设置预分频值
	TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;	//向上计数模式
	TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;	//设置时钟分割
	TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStructure);	
	
	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;	//PWM1模式
	TIM_OCInitStructure.TIM_Pulse = 0;	//初始化占空比为0
	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;//输出极性高
	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能
	TIM_OC1Init(TIM3,&TIM_OCInitStructure);
	TIM_OC1PreloadConfig(TIM3,TIM_OCPreload_Enable);	//使能TIMx在ARR上的预装载寄存器
	
	TIM_Cmd(TIM3,ENABLE);
}

测试:

#include "stm32f10x.h"
#include "led.h"
#include "systick.h"
#include "motor.h"

int main(void)
{
	uint8_t dir=1;
	uint16_t motor1_pwmval=0;
	SysTick_Init();
	LED_Init();
	MOTOR_GPIO_Init();
	TIM3_PWM_Init();
	
	while(1)
	{
		USER_LED_ON();
		delay_ms(500);
		USER_LED_OFF();
		delay_ms(500);
		
		
		if(dir)
			motor1_pwmval++;
		else 
			motor1_pwmval--;	 
 		if(motor1_pwmval>999)
			dir=0;
		if(motor1_pwmval==0)
			dir=1;	   					 
		TIM_SetCompare1(TIM3,motor1_pwmval);	//设置占空比0-999
	}
}

连接好电路,下载程序,可观察到小电机停止–转动–停止循环。


参考:

1.PWM波的介绍

2.一文读懂PWM原理及其在电源中的应用

  • 2
    点赞
  • 50
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
您好!下面是一个基于PID算法控制STM32直流减速电机PWM驱动的代码示例: ```C #include "stm32f4xx.h" // PID参数 float Kp = 1.0; float Ki = 0.5; float Kd = 0.2; // 电机速度设定值 int targetSpeed = 1000; // PID变量 float error = 0; float integral = 0; float derivative = 0; float lastError = 0; // PWM输出频率和分辨率 int pwmFrequency = 20000; // 20kHz int pwmResolution = 1000; // 10-bit分辨率 // 初始化PWM void PWM_init(void) { RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOC, ENABLE); GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_Init(GPIOC, &GPIO_InitStructure); GPIO_PinAFConfig(GPIOC, GPIO_PinSource6, GPIO_AF_TIM3); GPIO_PinAFConfig(GPIOC, GPIO_PinSource7, GPIO_AF_TIM3); TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_OCInitTypeDef TIM_OCInitStructure; TIM_TimeBaseStructure.TIM_Period = pwmResolution - 1; TIM_TimeBaseStructure.TIM_Prescaler = (SystemCoreClock / pwmFrequency) - 1; TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OCInitStructure.TIM_Pulse = 0; TIM_OC1Init(TIM3, &TIM_OCInitStructure); TIM_OC2Init(TIM3, &TIM_OCInitStructure); TIM_Cmd(TIM3, ENABLE); } // 初始化编码器 void Encoder_init(void) { RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE); GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_PinAFConfig(GPIOA, GPIO_PinSource0, GPIO_AF_TIM5); GPIO_PinAFConfig(GPIOA, GPIO_PinSource1, GPIO_AF_TIM5); RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM5, ENABLE); TIM_EncoderInterfaceConfig(TIM5, TIM_EncoderMode_TI12, TIM_ICPolarity_Rising, TIM_ICPolarity_Rising); TIM_Cmd(TIM5, ENABLE); } // 获取编码器计数器值 int Encoder_getCount(void) { return (int)TIM_GetCounter(TIM5); } // PID控制 void PID_control(void) { int currentSpeed = Encoder_getCount(); // 计算误差 error = targetSpeed - currentSpeed; // 计算积分项 integral += error; // 计算微分项 derivative = error - lastError; lastError = error; // 计算PID输出 float output = Kp * error + Ki * integral + Kd * derivative; // 限制输出范围 if (output > pwmResolution) { output = pwmResolution; } else if (output < -pwmResolution) { output = -pwmResolution; } // 设置PWM占空比 TIM_SetCompare1(TIM3, (uint16_t)(pwmResolution - fabs(output))); } int main(void) { PWM_init(); Encoder_init(); while (1) { PID_control(); } } ``` 这段代码使用了STM32的TIM3模块作为PWM输出,使用TIM5模块作为编码器输入。其中,`PWM_init()`函数用于初始化PWM输出,`Encoder_init()`函数用于初始化编码器。`PID_control()`函数用于进行PID控制,计算PID输出,并设置PWM占空比。 请根据您的具体硬件和需求进行适当的修改和调整。希望对您有所帮助!如果还有其他问题,请随时提问。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值