运动底盘是移动机器人的重要组成部分,不像激光雷达、IMU、麦克风、音响、摄像头这些通用部件能够直接买到,很难买到通用的底盘。一方面是由于底盘的尺寸结构和参数是要与具体机器人匹配的;另外一方面是由于底盘包含软硬件整套解决方案,是不少机器人公司的核心技术,通常不会随便公开。出于强烈的求知欲与学习热情,我想本身DIY一整套两轮差分底盘,而且将完整的设计过程公开出去供你们学习。说干就干,本章节主要内容:html
1.stm32主控硬件设计python
2.stm32主控软件设计nginx
3.底盘通讯协议算法
4.底盘ROS驱动开发django
6.底盘里程计标ubuntu
上一节搭建好了底盘的stm32主控硬件,如今就来讲说怎么开发配套的stm32软件。关于创建stm32工程、使用stm32开发库、stm32软件调试方法等基础知识就很少说了,有须要的能够查阅相关资料学习,我以为http://www.openedv.com《正点原子》的开发资料写的还能够。我就直接从底盘控制的项目入手,直接进行项目中各个功能需求开始分析讲解,如图11,是个人底盘控制stm32工程项目。centos
(图11)底盘控制stm32工程项目网络
电机控制分为两个部分(电机转向控制、电机转速控制),这些都集成在了电机驱动芯片TB6612FNG里面,因此只须要用单片机的IO口产生控制转向的高低电平和控制转速的PWM波就能实现。架构
首先,初始化IO口做为输出脚,用于产生高低电平输出来控制转向,实例代码如图12。
(图12)电机转向控制IO口初始化
而后,用通用定时器TIM4的通道CH1和CH2分别产生两路PWM输出用于两个电机的转速控制,定时器默认引脚分配如图13。
(图13)stm32定时器通道默认引脚分配
初始化通用定时器TIM4的通道CH1和CH2为PWM输出,实例代码如14。
(图14)电机转速控制IO口初始化
最后,将电机转向和速度控制的操做封装在一个函数中,便于其它地方调用,实例代码如图15。
(图15)电机转向和速度控制封装
编码器对底盘来讲相当重要,一方面底盘经过编码器的反馈进行PID闭环速度控制,另外一方面底盘经过编码器进行航迹推演获得里程计用于后续的定位与导航等高级算法中。这里用到的编码器是正交编码器,因此直接使用通用定时器的输入捕获中的编码器模式来读取编码器。采用通用定时器TIM2的通道CH1和CH2捕获encoder1的A相和B相脉冲,采用通用定时器TIM3的通道CH1和CH2捕获encoder2的A相和B相脉冲。
先初始化TIM2做为编码器encoder1的捕获,实例代码如图16。
(图16)初始化TIM2做为编码器encoder1的捕获
而后,将读取编码器计数值的操做封装在一个函数中,便于其它地方调用,实例代码如图17。
(图17)读取编码器encoder1计数值封装
最后,编写TIM2计数溢出时的中断处理函数,实例代码如图18。
(图18)TIM2计数溢出中断处理函数
同理可得TIM3捕获encoder2的代码实现,这里就不在赘述了。
串口2是数据接口,负责接收上位机发送过来的控制指令,同时将编码器值返回给上位机;串口1是debug接口,负责接收上位机发送过来的版本信息请求、PIDm默认值恢复、PID值设定等调试指令,同时将程序中的debug打印信息返回给上位机。可是在底盘正常工做时,只须要链接串口2;串口1是预留出来给有须要本身动手修改PID参数使用的。
首先,配置串口1,先对串口1的输出进行printf函数打印支持,实例代码如图19。
(图19)串口1的输出进行printf函数打印支持
而后,初始化串口1,实例代码如图20。
(图20)初始化串口1
最后,编写串口1接收中断处理函数,此函数主要进行对上位机发过来的数据进行协议解析,实例代码如图21。
(图21)串口1接收中断处理函数
接下来,介绍串口2,初始化串口2,实例代码如图22。
(图22)初始化串口2
而后,将串口2发送数据的操做封装到函数中,便于其它地方调用,实例代码如图23。
(图23)串口2发送数据封装
最后,编写串口2接收中断处理函数,此函数主要进行对上位机发过来的数据进行协议解析,实例代码如图24。
(图24)串口2接收中断处理函数
到这里,串口有1和串口2的数据发送与接收都编写好了,依据咱们定义的usart2数据通讯协议和usart1调试通讯协议,上位机就能够编写对应的程序来跟底盘的串口2和串口1进行通讯了。关于通讯协议的具体内容,将在后续作展开。
我在底盘中采用的是增量型PID算法,编程涉及到的数学表达式有3个,分别是:
e(k) = target_value - current_value
delta_u(k) = Kp*[e(k)-e(k-1)] + Ki*e(k) + Kd*[e(k)-2*e(k-1)+e(k-2)]
u(k) = u(k-1) + delta_u(k)
将这3个数学表达式封装到函数中,便于其它地方调用,实例代码如图25。
(图25)串口2接收中断处理函数
电机1与电机2采用一样的PID算法,因此电机2的PID算法代码实现就不赘述了。关于PID参数的整定方法,将在后续作展开。
经过上面的讲解,各个模块的驱动代码都准备就绪了,如今须要产生一个周期性的过程,在里面实现编码器计数值采样、PID控制等具体实现。这里采用定时器TIM1产生一个周期性的中断,在中断处理函数中实现各模块的具体操做。
首先,配置定时器TIM1,实例代码如图26。
(图26)配置定时器TIM1
而后,编写中断处理函数,实例代码如图27。
(图27)TIM1中断处理函数
2.6.stm32主控软件总体框图
经过上面的讲解,对底盘控制的stm32程序实现有了必定的了解,接下来就来作一个总结。
先来看看main()函数实现,如图28。
(图28)main()函数实现
结合上面TIM1中断处理函数,不难发现,整个stm32程序的执行过程:
a.在main()函数中初始化各个模块;
b.TIM1中断处理函数周期性的读取编码器值、反馈获取的编码值、PID控制;
c.剩下的就是串口1和串口2的通讯交互。
具体stm32主控软件总体框图如图29。
(图29)stm32主控软件总体框图
须要说明的是,在周期性循环体中,要首先读取编码器的值,来保证严格的等间隔采样。
------SLAM+语音机器人DIY系列【目录】快速导览------
第1章:Linux基础
第2章:ROS入门
第3章:感知与大脑
第4章:差分底盘设计
第5章:树莓派3开发环境搭建
第6章:SLAM建图与自主避障导航
2.google-cartographer机器人SLAM建图
第7章:语音交互与天然语言处理
第8章:高阶拓展
2.centos7下部署Django(nginx+uwsgi+django+python3)
----------------文章将持续更新,敬请关注-----------------
若是你们对博文的相关类容感兴趣,或有什么技术疑问,欢迎加入下面的《SLAM+语音机器人DIY》QQ技术交流群,一块儿讨论学习^_^
关于咱们:
视频教程: