jjzjj

STM32蓝牙控制循迹避障小车源代码——1.电机驱动,变速

灵风_Brend 2023-07-17 原文

STM32蓝牙控制循迹避障小车源代码——1.电机驱动,变速

注意-所需模块:

主控:STM32F103C8T6(F1系列板子均可以)



驱动芯片:L298N(1个)

接线:L298N:
A6–IN1
A7–IN2
B0–IN3
B1–IN4
ENA:5V
ENB:5V

代码

所有的代码都是直接从工程里面复制的,实测是没有问题的。

1.电机控制:moter.c

参考文章: 基于STM32F103C8T6的循迹避障小车完整制作过程

这篇文章详细讲解了PWM调速原理以及定时器通道的选择

我这里再简单总结一下:
IN1,IN2控制电机原理:
首先我们将OUT1和OUT2分别接电机的两极,这时我们控制单片机使输入端IN1接入高电平,则相应的OUT1端也就变为高电平;将IN2接入低电平后相应的OUT2也就变为低电平,OUT1和OUT2间有了电位差,这样电机就能转起来了。

  • PWM输出到IN1~IN4四个端子,来实现调速。
    开启定时器3的4个通道
    定时器3的通道1-4分别对应PA6, PA7, PB0, PB1,将这四个引脚分别接到 IN1-IN4,实现调速。

  • 调用函数:TIM_SetCompare1(TIM_TypeDef* TIMx, uint16_t Compare1);
    这个函数中的数字1代表的是TIMx的通道1,TIMx中的x可以取1-17但除了6,7。两个参数 TIMx , Compare1。
    Compare1是用于与TIMx比较的数,相当于TIMx的一个周期时间减去Compare1。
    差值越小(Compare1值越大),输出PWM的高电平时间越短,转速越低。

moter.c:

#include "moter.h"

//注意:声明不能出现在块中的可执行语句之后!
//在main函数里直接调用TIM3_PWM_Init()这个函数就完成了引脚和定时器的初始化
//然后在后面直接用控制运动函数
//IN1-IN4分别接PA6,PA7,PB0,PB1

void TIM3_PWM_Init(void)
{
	
	TIM_OCInitTypeDef TIM_OCInitStruct;
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
	GPIO_InitTypeDef GPIO_InitStruct;


	//使能定时器3时钟,使能IO口时钟GPIOA,GPIOB,开启AFIO
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB|RCC_APB2Periph_AFIO,ENABLE);
	
	//定时器3初始化
	TIM_TimeBaseInitStruct.TIM_ClockDivision=0;
	TIM_TimeBaseInitStruct.TIM_CounterMode=TIM_CounterMode_Up;
	TIM_TimeBaseInitStruct.TIM_Period=899;
	TIM_TimeBaseInitStruct.TIM_Prescaler=0;
	TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStruct);
	
	//初始化端口复用的A6 A7 B0 B1
	GPIO_InitStruct.GPIO_Mode=GPIO_Mode_AF_PP;
	GPIO_InitStruct.GPIO_Pin=GPIO_Pin_6|GPIO_Pin_7;
	GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStruct);
	
	GPIO_InitStruct.GPIO_Mode=GPIO_Mode_AF_PP;
	GPIO_InitStruct.GPIO_Pin=GPIO_Pin_0|GPIO_Pin_1;
	GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;
	GPIO_Init(GPIOB,&GPIO_InitStruct);
	
	
	//初始化并配置定时器3的4个通道
	
	//通道1
	TIM_OCInitStruct.TIM_OCMode=TIM_OCMode_PWM2;
	TIM_OCInitStruct.TIM_OutputState=ENABLE;
	TIM_OCInitStruct.TIM_OCPolarity=TIM_OCPolarity_High;
	TIM_OCInitStruct.TIM_Pulse=900;
	TIM_OC1Init(TIM3,&TIM_OCInitStruct);//初始化通道1
	
	TIM_OC1PreloadConfig(TIM3,TIM_OCPreload_Enable);
	
	//通道2
	TIM_OCInitStruct.TIM_OCMode=TIM_OCMode_PWM2;
	TIM_OCInitStruct.TIM_OutputState=ENABLE;
	TIM_OCInitStruct.TIM_OCPolarity=TIM_OCPolarity_High;
	TIM_OCInitStruct.TIM_Pulse=900;
	TIM_OC2Init(TIM3,&TIM_OCInitStruct);//初始化通道2
	
	TIM_OC2PreloadConfig(TIM3,TIM_OCPreload_Enable);

	
	//通道3
	TIM_OCInitStruct.TIM_OCMode=TIM_OCMode_PWM2;
	TIM_OCInitStruct.TIM_OutputState=ENABLE;
	TIM_OCInitStruct.TIM_OCPolarity=TIM_OCPolarity_High;
	TIM_OCInitStruct.TIM_Pulse=900;
	TIM_OC3Init(TIM3,&TIM_OCInitStruct);//初始化通道3
	
	TIM_OC3PreloadConfig(TIM3,TIM_OCPreload_Enable);

	//通道4
	TIM_OCInitStruct.TIM_OCMode=TIM_OCMode_PWM2;
	TIM_OCInitStruct.TIM_OutputState=ENABLE;
	TIM_OCInitStruct.TIM_OCPolarity=TIM_OCPolarity_High;
	TIM_OCInitStruct.TIM_Pulse=900;
	TIM_OC4Init(TIM3,&TIM_OCInitStruct);//初始化通道4
	
	TIM_OC4PreloadConfig(TIM3,TIM_OCPreload_Enable);

	//使能定时器3
	TIM_Cmd(TIM3,ENABLE);
	
}

//Compare1是用于与TIMx比较的数,相当于TIMx的一个周期时间减去Compare1。
//差值越小(Compare值越大),输出PWM的高电平时间越短,转速越低。
void Forward(int a)
{
	TIM_SetCompare1(TIM3,a);//差值越大,速度越慢
	TIM_SetCompare2(TIM3,900);
	TIM_SetCompare3(TIM3,a);
	TIM_SetCompare4(TIM3,900);
}

void Backward(int a)
{
	TIM_SetCompare1(TIM3,900);
	TIM_SetCompare2(TIM3,a);
	TIM_SetCompare3(TIM3,900);
	TIM_SetCompare4(TIM3,a);
}

void STOP()
{
	TIM_SetCompare1(TIM3,900);
	TIM_SetCompare2(TIM3,900);
	TIM_SetCompare3(TIM3,900);
	TIM_SetCompare4(TIM3,900);
}

void Turn_left(int a)
{
	TIM_SetCompare1(TIM3,900);
	TIM_SetCompare2(TIM3,a);
	TIM_SetCompare3(TIM3,a);
	TIM_SetCompare4(TIM3,900);
}

void Turn_right(int a)
{
	TIM_SetCompare1(TIM3,a);
	TIM_SetCompare2(TIM3,900);
	TIM_SetCompare3(TIM3,900);
	TIM_SetCompare4(TIM3,a);
}

void Big_Turn_left(int a)
{
	TIM_SetCompare1(TIM3,900);
	TIM_SetCompare2(TIM3,a);
	TIM_SetCompare3(TIM3,a);
	TIM_SetCompare4(TIM3,900);
}

void Big_Turn_right(int a)
{
	TIM_SetCompare1(TIM3,a);
	TIM_SetCompare2(TIM3,900);
	TIM_SetCompare3(TIM3,900);
	TIM_SetCompare4(TIM3,a);
}




  • 在moter.c代码中,我定义了小车的7种运动方式,参数a用来可用来调控速度。a取值在[0,900],我一般取[200,600],且a的值越大,电机转动速度越慢(注意当a的值过大时,可能会驱动不了)。

    正在学习STM32的同学可以跟着下面的步骤自己配置一下

    TIM3_PWM_Init()函数配置步骤

    1.使能定时器3时钟:
        调用函数:RCC_APB1PeriphClockCmd();
    使能IO口时钟:
        调用函数:RCC_APB2PeriphClockCmd();//GPIOA,GPIOB,AFIO
    2.定时器3初始化:
        调用函数:TIM_TimeBaseInit();//周期899,预分频系数0,计数器模式up,始终分割0
    3.初始化端口复用的A6,A7,B0,B1:
        调用函数:GPIO_Init();//设置为复用推挽输出 AF_PP
    4.初始化并配置定时器34个通道:
        例如通道1TIM_OC1Init();//模式:PWM2,极性:high,脉宽:900,输出状态:使能
    			  TIM_OC1PreloadConfig();
    5.最后使能定时器3:
        调用函数:TIM_Cmd();
    6.Forward()等运动函数中,调用
       			 TIM_SetCompare1(TIM3,X);
       			 TIM_SetCompare2(TIM3,X);
       			 TIM_SetCompare3(TIM3,X);
       			 TIM_SetCompare4(TIM3,X);
    

moter.h:

#ifndef __MOTER_H
#define __MOTER_H
#include "stm32f10x.h"

void TIM3_PWM_Init(void);

void Forward(int a);
void Backward(int a);
void Turn_left(int a);
void Turn_right(int a);
void Big_Turn_left(int a);
void Big_Turn_right(int a);

void STOP(void);

#endif

测试

将moter.h和moter.c两个代码写进工程里,如果想要测试代码是否正确,电机接线是否接反了,可以在主函数里写进以下代码。

#include "stm32f10x.h"
#include "moter.h"
#include "delay.h"

int main(void)
{
  delay_init();    //延时初始化		
	TIM3_PWM_Init();	//电机pwm   TIM3

	while(1)
	{
		Forward(500);//前进,也可以换成其他运动。如Turn_left(400);
        			//参数越大,电机转速越慢
	}
	
}

工程下载

测试:输STM32蓝牙控制循迹避障小车源代码——1.电机驱动,变速zip-嵌入式文档类资源-CSDN文库


下一节我会写循迹模块

有关STM32蓝牙控制循迹避障小车源代码——1.电机驱动,变速的更多相关文章

  1. Ruby Readline 在向上箭头上使控制台崩溃 - 2

    当我在Rails控制台中按向上或向左箭头时,出现此错误:irb(main):001:0>/Users/me/.rvm/gems/ruby-2.0.0-p247/gems/rb-readline-0.4.2/lib/rbreadline.rb:4269:in`blockin_rl_dispatch_subseq':invalidbytesequenceinUTF-8(ArgumentError)我使用rvm来管理我的ruby​​安装。我正在使用=>ruby-2.0.0-p247[x86_64]我使用bundle来管理我的gem,并且我有rb-readline(0.4.2)(人们推荐的最少

  2. ruby-on-rails - 带 Spring 锁的 Rails 4 控制台 - 2

    我正在使用Ruby2.1.1和Rails4.1.0.rc1。当执行railsc时,它被锁定了。使用Ctrl-C停止,我得到以下错误日志:~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.2/lib/spring/client/run.rb:47:in`gets':Interruptfrom~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.2/lib/spring/client/run.rb:47:in`verify_server_version'from~/.rvm/gems/ruby-2.1.1/gems/spring-1.1.

  3. ruby-on-rails - openshift 上的 rails 控制台 - 2

    我将我的Rails应用程序部署到OpenShift,它运行良好,但我无法在生产服务器上运行“Rails控制台”。它给了我这个错误。我该如何解决这个问题?我尝试更新ruby​​gems,但它也给出了权限被拒绝的错误,我也无法做到。railsc错误:Warning:You'reusingRubygems1.8.24withSpring.UpgradetoatleastRubygems2.1.0andrun`gempristine--all`forbetterstartupperformance./opt/rh/ruby193/root/usr/share/rubygems/rubygems

  4. ruby-on-rails - 如何在 Ruby on Rails 中实现由 JSF 2.0 (Primefaces) 驱动的 UI 魔法 - 2

    按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visitthehelpcenter指导。关闭10年前。问题1)我想知道ruby​​onrails是否有功能类似于primefaces的gem。我问的原因是如果您使用primefaces(http://www.primefaces.org/showcase-labs/ui/home.jsf),开发人员无需担心javascript或jquery的东西。据我所知,JSF是一个规范,基于规范的各种可用实现,prim

  5. FOHEART H1数据手套驱动Optitrack光学动捕双手运动(Unity3D) - 2

    本教程将在Unity3D中混合Optitrack与数据手套的数据流,在人体运动的基础上,添加双手手指部分的运动。双手手背的角度仍由Optitrack提供,数据手套提供双手手指的角度。 01  客户端软件分别安装MotiveBody与MotionVenus并校准人体与数据手套。MotiveBodyMotionVenus数据手套使用、校准流程参照:https://gitee.com/foheart_1/foheart-h1-data-summary.git02  数据转发打开MotiveBody软件的Streaming,开始向Unity3D广播数据;MotionVenus中设置->选项选择Unit

  6. STM32读取串口传感器数据(颗粒物传感器,主动上传) - 2

    文章目录1.开发板选择*用到的资源2.串口通信(个人理解)3.代码分析(注释比较详细)1.主函数2.串口1配置3.串口2配置以及中断函数4.注意问题5.源码链接1.开发板选择我用的是STM32F103RCT6的板子,不过代码大概在F103系列的板子上都可以运行,我试过在野火103的霸道板上也可以,主要看一下串口对应的引脚一不一样就行了,不一样的就更改一下。*用到的资源keil5软件这里用到了两个串口资源,采集数据一个,串口通信一个,板子对应引脚如下:串口1,TX:PA9,RX:PA10串口2,TX:PA2,RX:PA32.串口通信(个人理解)我就从串口采集传感器数据这个过程说一下我自己的理解,

  7. C51单片机——实现用独立按键控制LED亮灭(调用函数篇) - 2

    说在前面这部分我本来是合为一篇来写的,因为目的是一样的,都是通过独立按键来控制LED闪灭本质上是起到开关的作用,即调用函数和中断函数。但是写一篇太累了,我还是决定分为两篇写,这篇是调用函数篇。在本篇中你主要看到这些东西!!!1.调用函数的方法(主要讲语法和格式)2.独立按键如何控制LED亮灭3.程序中的一些细节(软件消抖等)1.调用函数的方法思路还是比较清晰地,就是通过按下按键来控制LED闪灭,即每按下一次,LED取反一次。重要的是,把按键与LED联系在一起。我打算用K1来作为开关,看了一下开发板原理图,K1连接的是单片机的P31口,当按下K1时,P31是与GND相连的,也就是说,当我按下去时

  8. ruby-on-rails - 在 Rails 控制台中使用 asset_path - 2

    在我的Character模型中,我添加了:字符.rbbefore_savedoself.profile_picture_url=asset_path('icon.png')end但是,对于数据库中已存在的所有角色,它们的profile_picture_url为nil。因此,我想进入控制台并遍历所有这些并进行设置。在我试过的控制台中:Character.find_eachdo|c|c.profile_picture_url=asset_path('icon.png')end但这给出了错误:NoMethodError:undefinedmethod`asset_path'formain:O

  9. ruby-on-rails - 带有 Pry 的 Rails 控制台 - 2

    当我进入Rails控制台时,我已将pry设置为加载代替irb。我找不到该页面或不记得如何将其恢复为默认行为,因为它似乎干扰了我的Rubymine调试器。有什么建议吗? 最佳答案 我刚发现问题,pry-railsgem。忘记了它的目的是让“railsconsole”打开pry。 关于ruby-on-rails-带有Pry的Rails控制台,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/question

  10. ruby - 将全局 $stdout 重新分配给控制台 - ruby - 2

    我正在尝试将$stdout设置为临时写入一个文件,然后返回到一个文件。test.rb:old_stdout=$stdout$stdout.reopen("mytestfile.out",'w+')puts"thisgoesinmytestfile"$stdout=old_stdoutputs"thisshouldbeontheconsole"$stdout.reopen("mytestfile1.out",'w+')puts"thisgoesinmytestfile1:"$stdout=old_stdoutputs"thisshouldbebackontheconsole"这是输出。r

随机推荐