STM32外设系列—TB6612FNG

本文介绍了TB6612FNG电机驱动IC的使用方法,包括引脚连接、控制逻辑和电机调速原理。通过STM32初始化GPIO和PWM,实现对12V减速电机的正反转与调速控制。此外,还展示了如何通过串口接收占空比数据来动态调整电机转速。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


🎀 文章作者:二土电子

🌸 关注文末公众号获取其他资料和工程文件!

🐸 期待大家一起学习交流!


本文涉及到定时器和串口的知识,详细内容可见博主STM32速成笔记专栏

一、TB6612简介

TB6612FNG是东芝半导体的一款驱动电机的IC。一个TB6612FNG可以驱动两个电机,每一个驱动都有两个逻辑输入引脚,一个输出引脚和一个PWM引脚。可以通过给两个逻辑输入引脚不同的电平来控制电机的运行状态,通过PWM输入引脚实现电机调速。TB6612FNG还具有以下特点

  • 电源电压最大可到15V
  • 输出电流最大可达3.2A
  • 内置热停机电路和低压检测电路
  • 有正转,反转,短制动和停止四种模式

TB6612FNG

二、TB6612使用方法

2.1 TB6612引脚连接

引脚连接
PWMAA通道的PWM输入
AIN2A通道逻辑输入2引脚
AIN1A通道逻辑输入1引脚
STBY待机引脚,接低电平处于待机模式,接高电平开始工作
BIN1B通道逻辑输入1引脚
BIN2B通道逻辑输入2引脚
PWMBB通道PWM输入引脚
GND
VM电源输入正极,最大接15V
VCC逻辑电源正极,接3.3V
AO1A通道输出1引脚
AO2A通道输出2引脚
BO2B通道输出2引脚
BO1B通道输出1引脚

使用时VM接电机电源的正极,GND接电机电源的负极。IN1和IN2接逻辑输入,PWM接PWM输出引脚。O1和O2接电机的正负极。

2.2 控制逻辑

IN1和IN2的高低电平状态对应不同的电机运行状态,二者的对应关系如下

IN1001
IN2010
电机运行状态停止正转反转

上述的正反转是AO1接电机正极,AO2接电机负极的对应关系。

2.3 电机调速

电机调速的远离比较简单,只需要给TB6612FNG的PWM输入引脚输入10KHz的PWM波。调节占空比即可调节转速。需要注意的是如果PWM配置的极性是低电平,那么设置占空比时的值越大,电机转速越低。相反,如果PWM配置的极性是高电平,那么设置占空比时的值越大,电机转速越高。

三、实战项目

3.1 项目简介

本项目比较简单,使用TB6612驱动一个12V减速电机。利用串口发送占空比,实现电机的调速。

3.2 初始化GPIO

初始化GPIO完成的工作是初始化逻辑控制引脚,程序如下

/*
 *==============================================================================
 *函数名称:Drv_MotorGpio_Init
 *函数功能:初始化Motor的GPIO
 *输入参数:无
 *返回值:无
 *备  注:这里只初始化了逻辑控制IO,PWM的IO在定时器配置PWM时初始化
 *==============================================================================
 */
void Drv_MotorGpio_Init (void)
{
	GPIO_InitTypeDef GPIO_InitStructure;   // 定义结构体
	// 开启时钟
	RCC_APB2PeriphClockCmd(MOTOR_GPIO_TIM,ENABLE);

	// 配置结构体
	GPIO_InitStructure.GPIO_Pin = MOTOR_GPIO_PIN;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;   // 推挽式输出
	GPIO_Init(MOTOR_GPIO, &GPIO_InitStructure);
}

宏定义如下

// 电机逻辑控制GPIO
#define MOTOR_GPIO_TIM   RCC_APB2Periph_GPIOC
#define MOTOR_GPIO   GPIOC
#define MOTOR_GPIO_PIN   GPIO_Pin_7 | GPIO_Pin_8

3.3 PWM初始化

PWM初始化程序如下

/*
 *==============================================================================
 *函数名称:TIM2_CH1_PWM_Init
 *函数功能:初始化定时器2的PWM通道1
 *输入参数:per:自动重装载值;psc:预分频系数
 *返回值:无
 *备  注:无
 *==============================================================================
 */
void TIM2_CH1_PWM_Init (u16 per,u16 psc)
{
	// 结构体定义
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;
	TIM_OCInitTypeDef TIM_OCInitStructure;
	GPIO_InitTypeDef GPIO_InitStructure;
	
	// 开启时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);
	
	// 初始化GPIO
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;   // 复用推挽输出
	GPIO_Init(GPIOA,&GPIO_InitStructure);
	
	// 初始化定时器参数
	TIM_TimeBaseInitStructure.TIM_Period = per;   // 自动装载值
	TIM_TimeBaseInitStructure.TIM_Prescaler = psc;   // 分频系数
	TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;
	TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;   // 设置向上计数模式
	TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStructure);	
	
	// 初始化PWM参数
	TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;   // 比较输出模式
	TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;   // 输出极性
	TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;   // 输出使能
	TIM_OC1Init(TIM2,&TIM_OCInitStructure);   // 输出比较通道1初始化
	
	TIM_OC1PreloadConfig(TIM2,TIM_OCPreload_Enable);   // 使能TIMx在 CCR1 上的预装载寄存器
	TIM_ARRPreloadConfig(TIM2,ENABLE);   // 使能预装载寄存器
	
	TIM_Cmd(TIM2,ENABLE);   // 使能定时器	
}

初始化时配置如下,配置为10KHz

	TIM2_CH1_PWM_Init(1000,71);   // 初始化PWM

3.4 电机控制程序

电机正反转和停止控制程序如下

/*
 *==============================================================================
 *函数名称:Med_Motor_Go
 *函数功能:电机正转
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
 */
void Med_Motor_Go (void)
{
	MOTOR_IN1 = 1;
	MOTOR_IN2 = 0;
}
/*
 *==============================================================================
 *函数名称:Med_Motor_Stop
 *函数功能:电机停转
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
 */
void Med_Motor_Stop (void)
{
	MOTOR_IN1 = 0;
	MOTOR_IN2 = 0;
}
/*
 *==============================================================================
 *函数名称:Med_Motor_Reverse
 *函数功能:电机反转
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
 */
void Med_Motor_Reverse (void)
{
	MOTOR_IN1 = 0;
	MOTOR_IN2 = 1;
}

宏定义如下

// 电机逻辑控制引脚
#define MOTOR_IN1   PCout(7)
#define MOTOR_IN2   PCout(8)

3.5 串口接收处理函数

串口需要根据接收到的占空比来配置输出PWM的占空比,配置占空比使用的库函数是

void TIM_SetCompare1(TIM_TypeDef* TIMx, uint16_t Compare1)

串口只支持输入大于100小于1000的占空比,串口接收处理程序如下

/*
 *==============================================================================
 *函数名称:USART1_IRQHandler
 *函数功能:USART1中断服务函数
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
 */
u32 gReceCount = 0;   // 接收计数变量
u32 gClearCount = 0;   // 清空接收数组计数变量
u8 gReceFifo[1500];   // 接收数组
u8 gReceEndFlag = 0;   // 接收完成标志位 

void USART1_IRQHandler(void)  
{
	if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)   //接收到一个字节  
	{
		gReceFifo[gReceCount++] = USART_ReceiveData(USART1);
	}
	else if(USART_GetITStatus(USART1,USART_IT_IDLE) != RESET)   //接收到一帧数据
	{
		USART1->SR;   // 先读SR
		USART1->DR;   // 再读DR
		
		gReceEndFlag = 1;   // 接收完成标志置1 
	} 
}
/*
 *==============================================================================
 *函数名称:Uart_Rece_Pares
 *函数功能:解析串口接收内容
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
 */
void Uart_Rece_Pares(void)   // 串口接收内容解析函数
{
	u16 pwmDuty = 0;   // 接收串口发送来的占空比
	
	if (gReceEndFlag  == 1)   // 如果接收完成
	{
		// 解析接收内容
		
		// 一位数
		if (gReceCount == 3)
		{
			pwmDuty = (gReceFifo[0] - 48);
		}
		else if (gReceCount == 4)
		{
			pwmDuty = (gReceFifo[0] - 48) * 10;
			pwmDuty = pwmDuty + (gReceFifo[1] - 48);
		}
		else if (gReceCount == 5)
		{
			pwmDuty = (gReceFifo[0] - 48) * 100;
			pwmDuty = pwmDuty + (gReceFifo[1] - 48) * 10;
			pwmDuty = pwmDuty + (gReceFifo[2] - 48);
		}
		
		printf ("duty=%d\r\n",pwmDuty);
		TIM_SetCompare1(TIM2,pwmDuty);
		
		// 清空接收数组
		for (gClearCount = 0;gClearCount < gReceCount;gClearCount ++)
		{
			gReceFifo[gClearCount] = ' ';
		}
			
		gReceEndFlag = 0;   // 清除接收完成标志位
		gReceCount = 0;   // 清零接收计数变量
	}
}
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

二土电子

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值