文章目录
- 输出比较
- PWM
- 输出比较通道
- 参数计算
- 舵机简介
- 直流电机简介
- TB6612
- PWM基本结构
- PWM驱动呼吸灯
- PWM驱动舵机
- PWM控制电机
输出比较
输出比较,简称OC(Output Compare)。
输出比较的原理是,当定时器计数值与比较值相等或者满足某种特定条件时,比较通道会产生一个输出信号,这个输出信号可以用来触发外部事件,如控制其他外设的操作,或者驱动外部电路。
在每个高级定时器和通用定时器都拥有4个输出比较通道。
高级定时器的前3个通道额外拥有死区生成和互补输出的功能。
PWM
我们可以利用输出比较来对外产生一个PWM频率。
PWM(Pulse Width Modulation)脉冲宽度调制,是一种常用的控制信号技术。通过改变信号的脉冲宽度来控制电力开关装置的平均功率。在PWM中,周期保持不变,而脉冲的宽度可以根据需要进行调整。
PWM技术广泛应用于电力电子领域,特别是在电机控制和电源调节方面。通过调整PWM信号的占空比(脉冲宽度与周期之比),可以精确地控制输出信号的平均电压或电流。这种控制方式可以实现对电机速度、亮度、电压等参数的精确控制,具有高效率、高精度和低成本的优点。
频率=1/Ts;占空比=Ton/Ts;分辨率=占空比的变化步距。
我们可以利用输出比较,对输出电平进行一定程度的控制,就能输出PWM频率。
输出比较通道
通用定时器总框图。
放大效果:
在比较通道左边,CNT计数器与捕获/比较寄存器中的值进行比较大小,再根据控制器,就会输出一定的电平。
ETRF是定时器的小功能,一般不用到。
REF(rreference)实际上就是指这里信号的高低电平。REF可以映射到主模式的TRGO输出上去;REF的主要去向是去到输出使能电路,它会先走向一个寄存器,如果寄存器输出为0,那么电平将不翻转,保持原样;如果信号为1,REF会通向一个非门取反,也就是高低电平翻转的信号;接着通过使能电路,将有一个寄存器(CC1E)控制;最后到OC1引脚,接到CH1通道上。
输出模式控制器,可以根据自己需求来选择模式:
参数计算
红色线表示CCR,也就是比较值;黄色线表示自动装载寄存器中的值(ARR);蓝色线表示计数值CNT;
当CNT<CCR时,输出高电平;当CNT>=CCR时,输出低电平。
这里对应的是PWM模式1的向上计数。
PWM频率: Freq = CK_PSC / (PSC + 1) / (ARR + 1)
PWM占空比: Duty = CCR / (ARR + 1)
PWM分辨率: Reso = 1 / (ARR + 1)
舵机简介
舵机(Servo)是一种常用的电动执行器,通常用于控制机械运动和定位定位。它由一个直流电机、减速装置、位置反馈装置和控制电路组成。
舵机的工作原理是控制电路根据输入信号生成特定的PWM信号,并驱动直流电机和减速装置运转,使输出轴转动到所需的位置。位置反馈装置(常用的是旋转式电位器)会实时监测输出轴的位置,并将信息反馈给控制电路,以便进行修正和精确控制。
SG90使用要求:输入PWM信号要求:周期为20ms,高电平宽度为0.5ms~2.5ms
硬件电路:
直流电机简介
直流电机是一种将电能转换为机械能的装置,有两个电极,当电极正接时,电机正转,当电极反接时,电机反转。
直流电机属于大功率器件,GPIO口无法直接驱动,需要配合电机驱动电路来操作。
TB6612
TB6612是一款双路H桥型的直流电机驱动芯片,常用于控制直流电机的转动方向和速度。它具有高效率、低功耗和高输出等特点,适用于各种电机驱动应用。
TB6612芯片内部集成了H桥驱动电路,可通过控制引脚实现正转、反转、制动和浮动等操作。它可以工作于3.3V或5V逻辑电平,支持PWM输入控制电机速度,并提供过流保护功能,防止电机过载。
硬件电路:
PWM基本结构
通过配置时基单元,,让计时器驱动与CCR比较,在PWM模式1下输出电平最后通向GPIO口。
PWM驱动呼吸灯
接线模式:
PWM.h
#ifndef __PWM_H__
#define __PWM_H__
void PWM_Init();
void PWM_SetCompare(uint16_t Compare);
#endif
PWM.c
#include "stm32f10x.h" // Device header
void PWM_Init()
{
//开启APB1外设开关
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP; //复用推挽输出
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
GPIO_Init(GPIOA,&GPIO_InitStructure);
//配置内部时钟TIM2
TIM_InternalClockConfig(TIM2);
//时钟结构体初始化
TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
TIM_TimeBaseInitStructure.TIM_ClockDivision=TIM_CKD_DIV1; //不分频
TIM_TimeBaseInitStructure.TIM_CounterMode=TIM_CounterMode_Up; //计时器模式
TIM_TimeBaseInitStructure.TIM_Period=100-1; //自动加载寄存器周期值
TIM_TimeBaseInitStructure.TIM_Prescaler=1-1; //预分频值
TIM_TimeBaseInitStructure.TIM_RepetitionCounter=0; //指定重复计时器的值,这里不用到
TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStructure);
//配置输出比较结构体
TIM_OCInitTypeDef TIM_OCInitStructure;
TIM_OCStructInit(&TIM_OCInitStructure);
TIM_OCInitStructure.TIM_OCMode=TIM_OCMode_PWM1; //配置输出比较模式
TIM_OCInitStructure.TIM_OCPolarity=TIM_OCPolarity_High; //指定输出极性
TIM_OCInitStructure.TIM_OutputState=TIM_OutputState_Enable;//输出比较状态
TIM_OCInitStructure.TIM_Pulse=0; //指定要捕获的脉冲值CCR
TIM_OC1Init(TIM2,&TIM_OCInitStructure);
//启用TIM2外设控制
TIM_Cmd(TIM2,ENABLE);
}
//设置CCR比较值
void PWM_SetCompare(uint16_t Compare)
{
TIM_SetCompare1(TIM2,Compare);
}
对于GPIO口来说,不止是接通了外设,还需要将PWM频率传输给外设,所以使用了复用推挽输出。
输出比较结构体有多个成员,我们这里一些成员不用到,所以先进行结构体初始化,再进行对一些成员的赋值。
main.c
#include "stm32f10x.h" // Device header
#include "Delay.h"
#include "PWM.h"
uint16_t i;
int main()
{
OLED_Init();
PWM_Init();
while(1)
{
for(i=0;i<=100;i++)
{
PWM_SetCompare(i);
Delay_ms(10);
}
for(i=0;i<=100;i++)
{
PWM_SetCompare(100-i);
Delay_ms(10);
}
}
}
PWM驱动舵机
OLED函数所取地址
连接方式:
SYTM32驱动电压只有3.3V,舵机需要5V的驱动电压;
Servo.c
#include "stm32f10x.h" // Device header
#include "PWM.h"
void Servo_Init()
{
PWM_Init();
}
void Servo_SetAngle(float Angle)
{
PWM_SetCompare(Angle/180*200+50);
}
Servo.h
#ifndef __SERVO_H__
#define __SERVO_H__
void Servo_Init();
void Servo_SetAngle(float Angle);
#endif
这里的PWM频率是有要求的,所以需要将PWM.c的CNT和ARR进行修改:频率=72M/720/2000=50Hz。
这里的转动度数需要根据占空比来进行计算:
Key.c
#include "stm32f10x.h" // Device header
#include "Delay.h"
void Key_Init(void)
{
//设置APB2外设时钟开关
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
//对结构体成员的选择
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 | GPIO_Pin_11;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
//结构体初始化
GPIO_Init(GPIOB, &GPIO_InitStructure);
}
//获取键码
uint8_t Key_GetNum(void)
{
uint8_t KeyNum = 0;
if (GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_1) == 0)
{
Delay_ms(20);
while (GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_1) == 0);
Delay_ms(20);
KeyNum = 1;
}
if (GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_11) == 0)
{
Delay_ms(20);
while (GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_11) == 0);
Delay_ms(20);
KeyNum = 2;
}
return KeyNum;
}
Key.h
#ifndef __KEY_H_
#define __KEY_H_
void Key_Init(void);
uint8_t Key_GetNum(void);
#endif
main.c
#include "stm32f10x.h" // Device header
#include "Delay.h"
#include "OLED.h"
#include "Servo.h"
#include "Key.h"
uint16_t angle;
int main()
{
OLED_Init();
Servo_Init();
Key_Init();
OLED_ShowString(1,1,"Angle:");
while(1)
{
if(Key_GetNum()==1)
{
angle+=30;
if(angle>180)
{
angle=0;
}
}
Servo_SetAngle(angle);
OLED_ShowNum(1,7,angle,3);
}
}
PWM控制电机
接线方式:
Motor.h
#ifndef __MOTOR_H__
#define __MOTOR_H__
void Motor_init();
void Motor_GetSpeed(int8_t Speed);
#endif
Motor.c
#include "stm32f10x.h" // Device header
#include "PWM.h"
void Motor_init()
{
//设置APB2外设时钟开关
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
//对结构体成员的选择
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4 | GPIO_Pin_5;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
//结构体初始化
GPIO_Init(GPIOA, &GPIO_InitStructure);
PWM_Init();
}
//速度输入函数
void Motor_GetSpeed(int8_t Speed)
{
if(Speed>=0)
{
GPIO_SetBits(GPIOA,GPIO_Pin_5);
GPIO_ResetBits(GPIOA,GPIO_Pin_4);
PWM_SetCompare(Speed);
}
else
{
GPIO_SetBits(GPIOA,GPIO_Pin_4);
GPIO_ResetBits(GPIOA,GPIO_Pin_5);
PWM_SetCompare(-Speed);
}
}
PWM的占空比作为速度的调节,两个接口的正反接作为方向的控制。
对于电机驱动电路所接引脚,需要进行GPIO口的初始化;
当输入速度为负时,将接口上引脚进行电平翻转,读者可以进行尝试,怎么接为正,怎么接为负;对于小于0的速度,需要加上符号变成正数。
main.c
#include "stm32f10x.h" // Device header
#include "Delay.h"
#include "OLED.h"
#include "Motor.h"
#include "Key.h"
int8_t Speed;
int main()
{
OLED_Init();
Motor_init();
Key_Init();
OLED_ShowString(1,1,"Speed:");
while(1)
{
if(Key_GetNum()==1)
{
Speed+=20;
if(Speed>100)
{
Speed=-100;
}
}
Motor_GetSpeed(Speed);
OLED_ShowSignedNum(1,7,Speed,3);
}
}