STM32智能小车------PWM驱动直流电机


  • 大家好哇!我是小光,嵌入式爱好者,一个想要成为系统架构师的普通大学生。
  • 进入正题,最近在做小车,目前已经可以跑起来了,然后就是加入各种各样的传感器,使得她的功能更加丰富。
  • 本次就谈谈做小车中的第一步,也是最重要的一步:PWM驱动直流减速电机,也是就让小车跑起来。
  • 非常感谢大家的阅读,如果有不对的地方欢迎指正

一、硬件连接

1.器件清单:

1.STM32F103C8T6最小系统板
2.直流减速电机
3.TB6612FNG驱动模块

2.接线:

在这里插入图片描述
接线方式1(PWM驱动电机):
PWMA----PA11:接板子可设置PWM的IO口,
PWMA----PA2:接板子可设置PWM的IO口;
STBY------3.3v:使能;
AIN1-------PB12:普通IO口;
AIN2-------PB13:普通IO口;
BIN1-------PB14:普通IO口;
BIN2-------PB15:普通IO口;
VM---------12V电源正极:供电;
GND-------电源负极;
VCC--------板子5V;
GND--------板子GND;
A1(AO1)————电机1(无正负)
A2(AO2)————电机1(无正负,和正传反转有关)
B1(BO1)————电机2(无正负)
B2(BO2)————电机2(无正负,和正传反转有关)
注意:TB6612一定要和板子共地;
接线方式2(不空速):
不用接PWMA和PWMB就可以了。

3.驱动电机

现在就说说为什么这样接线吧:
在这里插入图片描述
我们只需要让四个IO输出相应的高低电平,然后通过PWM调速,就可以让小车动起来了,是不是很简单(狗头保命),好了,下面开始讲解软件部分。

二、软件驱动代码编写

1.初始化代码

代码示例:

void Motor_Init(u16 arr,u16 psc)
{
	GPIO_InitTypeDef GPIO_InitStructure;
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //使能PB端口时钟
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15;	//端口配置
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;      //推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;     //50M
	GPIO_Init(GPIOB, &GPIO_InitStructure);				  //根据设定参数初始化GPIOB 
	
	PWM_Init(arr,psc);
}

这就是很简单的GPIO初始化啦,然后就是PWM初始化:

/**************************************************************************
函数功能:控制电机的两路PWM初始化
入口参数:无
返回  值:无
**************************************************************************/
void PWM_Init(u16 arr,u16 psc)
{		 		
	GPIO_InitTypeDef GPIO_InitStructure;
	TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
	TIM_OCInitTypeDef  TIM_OCInitStructure;
	//使能GPIO和定时器时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1, ENABLE);    //
  RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA , ENABLE);  //使能GPIO外设时钟使能
	//设置该引脚为复用输出功能,输出TIM1 CH4和TIM2 CH3的PWM脉冲波形
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_11;   //TIM_CH1 //TIM_CH4
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;         //复用推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA, &GPIO_InitStructure);
  //初始化定时器arr、psc
	TIM_TimeBaseStructure.TIM_Period = arr;                //设置在下一个更新事件装入活动的自动重装载寄存器周期的值	 
	TIM_TimeBaseStructure.TIM_Prescaler =psc;              //设置用来作为TIMx时钟频率除数的预分频值  不分频
	TIM_TimeBaseStructure.TIM_ClockDivision = 0;           //设置时钟分割:TDTS = Tck_tim
	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式
	TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure);        //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位
  //初始化比较参数
	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;      //选择定时器模式:TIM脉冲宽度调制模式1
	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能
	TIM_OCInitStructure.TIM_Pulse = 0;                            //设置待装入捕获比较寄存器的脉冲值
	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;     //输出极性:TIM输出比较极性高
	TIM_OC4Init(TIM1, &TIM_OCInitStructure);               //根据TIM_OCInitStruct中指定的参数初始化外设TIMx
  //MOE 主输出使能	
	TIM_CtrlPWMOutputs(TIM1,ENABLE);	                  
  //CH4预装载使能	 
	TIM_OC4PreloadConfig(TIM1, TIM_OCPreload_Enable);     
  //使能TIMx在ARR上的预装载寄存器
	TIM_ARRPreloadConfig(TIM1, ENABLE);                   
  //使能TIM1
	TIM_Cmd(TIM1, ENABLE);                                
	
	//TIM2_CH3
	TIM_TimeBaseStructure.TIM_Period = arr;                //设置在下一个更新事件装入活动的自动重装载寄存器周期的值	 
	TIM_TimeBaseStructure.TIM_Prescaler =psc;              //设置用来作为TIMx时钟频率除数的预分频值  不分频
	TIM_TimeBaseStructure.TIM_ClockDivision = 0;           //设置时钟分割:TDTS = Tck_tim
	TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  //TIM向上计数模式
	TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);        //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位


	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;      //选择定时器模式:TIM脉冲宽度调制模式1
	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //比较输出使能
	TIM_OCInitStructure.TIM_Pulse = 0;                            //设置待装入捕获比较寄存器的脉冲值
	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;     //输出极性:TIM输出比较极性高
	TIM_OC3Init(TIM2, &TIM_OCInitStructure);               //根据TIM_OCInitStruct中指定的参数初始化外设TIMx

	TIM_CtrlPWMOutputs(TIM2,ENABLE);	                  //MOE 主输出使能	

	TIM_OC3PreloadConfig(TIM2, TIM_OCPreload_Enable);     //CH3预装载使能	 

	TIM_ARRPreloadConfig(TIM2, ENABLE);                   //使能TIMx在ARR上的预装载寄存器

	TIM_Cmd(TIM2, ENABLE);                                //使能TIM2
} 

大家可以仔细看看注释,这里我用的是PA11(TIM1 CH4)和 PA2(TIM2 CH3),这样我们就初始化完了。

2.电机驱动代码

.h文件:

//位带操作,实现51类似的GPIO控制功能
//具体实现思想,参考<<CM3权威指南>>第五章(87页~92页).
//IO口操作宏定义
#define BITBAND(addr, bitnum) ((addr & 0xF0000000)+0x2000000+((addr &0xFFFFF)<<5)+(bitnum<<2)) 
#define MEM_ADDR(addr)  *((volatile unsigned long  *)(addr)) 
#define BIT_ADDR(addr, bitnum)   MEM_ADDR(BITBAND(addr, bitnum)) 

//IO口地址映射
#define GPIOA_ODR_Addr    (GPIOA_BASE+12) //0x4001080C 
#define GPIOB_ODR_Addr    (GPIOB_BASE+12) //0x40010C0C    

#define GPIOA_IDR_Addr    (GPIOA_BASE+8) //0x40010808 
#define GPIOB_IDR_Addr    (GPIOB_BASE+8) //0x40010C08 
//IO口操作,只对单一的IO口!
//确保n的值小于16!
#define PAout(n)   BIT_ADDR(GPIOA_ODR_Addr,n)  //输出 
#define PAin(n)    BIT_ADDR(GPIOA_IDR_Addr,n)  //输入 

#define PBout(n)   BIT_ADDR(GPIOB_ODR_Addr,n)  //输出 
#define PBin(n)    BIT_ADDR(GPIOB_IDR_Addr,n)  //输入 


#define PWMA   TIM2->CCR3    //PA2
#define AIN2   PBout(15) 
#define AIN1   PBout(14)
#define BIN1   PBout(13)
#define BIN2   PBout(12)
#define PWMB   TIM1->CCR4    //PA11

void Set_Pwm(int moto1,int moto2);
int myabs(int a);

这里我用了PB12-15作为输入的引脚,大家也可以用其他的引脚,只是需要把宏定义改一下就可以了
.c文件:

/**************************************************************************
函数功能:赋值给PWM寄存器
入口参数:左轮PWM、右轮PWM
返回  值:无
**************************************************************************/
void Set_Pwm(int motorLeft,int motorRight)
{
	if(motorLeft>0)     AIN2=0, AIN1=1;
	else if(motorLeft == 0) AIN2=0, AIN1=0;
	else 	            AIN2=1,	AIN1=0;
	PWMA=myabs(motorLeft);
	if(motorRight>0)	BIN1=0,	BIN2=1;
	else if(motorRight == 0) BIN1=0,	BIN2=0;
	else                BIN1=1,	BIN2=0;
	PWMB=myabs(motorRight);	
}

/**************************************************************************
函数功能:绝对值函数
入口参数:int
返回  值:unsigned int
**************************************************************************/
int myabs(int a)
{ 		   
	int temp;
	if(a<0)  
	  temp=-a;  
	else 
	  temp=a;
	return temp;
}

这个函数void Set_Pwm(int motorLeft,int motorRight)就是根据TB6612驱动电机的原理编写的代码,方便我们控制电机。
我们调用时直接SetPwm(LeftSpeed,RightSpeed);就可以了,是不是很简单呢?(狗头保命);

总结

奉上视频:

你见过会说话的小车吗

本文仅仅简单介绍了TB6612控制直流电机的原理和驱动代码的编写,这样就完成了智能小车的第一步,后面会加上更多传感器如:超声波,红外避障,红外遥控,超声波,手势传感器等等,去实现更多的功能,感谢阅读,如果觉得我的文章对你有帮助的话,就点个赞吧!爱了爱了

版权声明:本文为CSDN博主「小光学嵌入式」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/qq_52608074/article/details/121958486

生成海报
点赞 0

小光学嵌入式

我还没有学会写个人说明!

暂无评论

发表评论

相关推荐

STM32F2————配置时钟延迟不准的问题

STM32F2配置时钟问题 笔者在本科毕业设计使用STM32F207芯片,但是在配置时钟时出现了问题。 问题 我按照F1写代码的延时函数放在F2竟然不准了 换个办法 使用Systick时钟也是不准,原因是笔者代

为什么重写printf函数没有用?

以前在网上找了无数方法去重写printf函数,但发现都没效果,今天偶然发现重写printf函数可以了,原因是以前没有勾选微库(Use MicroLlB)! 这里