平衡小车项目解读日志
2016/3/31
1. 关于6050陀螺仪模块问题
2. 今天開始着手平衡小车项目,蓝牙模块不用关心。仅仅要知道能够连接,再推断串口发来的指令就可以。
3. 好像比較简单的是使用DMP。通过结合DMP,能够将我们的原始数据转换成四元素输出,再通过四元素算出欧拉角。从而得到yawroll 和 pitch。
4. 由于做的是平衡小车。那么就不须要roll 和 yaw,直接通过pitch就可以。
2016/4/1
1. 昨天卡在一个比較愚蠢的问题上,就是我非常少自己配置project什么的,所以当測试卖家发来的例程的时候。编译不通过,最后发现是要在keil4下打开才干够用。期中涉及的问题我也不想深究,毕竟配置project什么的不是我学习的重点。
2. 今天总结了一条经验:解决这个问题,一定要善于使用最简单的方法。不做无用功,也不能浪费宝贵的时间。比方我发现他们的例程下进去,小车根本就不能动,排查问题,肯定是资料问题。我问了卖家以后。他们发了新的资料。成了!project编译错误,网上百度一下,好像keil4下能够正常使用,那就装一个keil4,成了!这就是最简单的解决这个问题方法,不要自己死磕。不能转牛角尖。
3. MPU6050:首先我要做的就是可以使用例程,将MPU获取的数据打印出来,这样便于调试,也便于我接下来的学习;然后深入理解6050的工作方式和通信方式(IIC通信)。再学习6050在DMP方法下,输出的数据怎么可以整定成小车的姿态,那么平衡小车中比較关键的内容就结束了!
由于还是写成项目日志比較清楚。所以准备写项目日志,暂停写直至平衡小车项目结束,又一次開始学战舰开发板的其它模块。
2016/4/8
1. 最终度过了学校里的期中考试,可以又一次開始项目了。小车的项目尽管比較难理解。可是我还是有非常大的兴趣的,由于这是在做实实在在可以拿得出手的项目啊!
2. 继续6050:出现故障,就是平衡小车上没有串口输出。下载都是用stlink,那么这样就不能直观的看到6050读出的数据了。
在这之前,我还用了我买的战舰开发板,准备将里面的数据通过串口打印,可是,板子上留出了接口,并且和我买的6050接口有差别,即使一根一根的把排线插上去,也没办法读出来,一直显示错误,这样的问题我并不想深究,由于有可能根本就是两个模块的引脚定义不同,在这里,我就当STM32已经通过IIC总线。将想要的数据读取了出来了。
3. IIC:平衡小车上的原理图说明,PB10为IIC的SCL。PB11为IIC的SDA信号线。
4. 电机的初始化:使用的电机驱动模块是LV8731驱动芯片
5. 平衡小车的电机:如今的PCB上画的相应的引脚例如以下
|
FR |
AT1 |
AT2 |
STEP |
左边电机 |
L_FR -> PB15 |
L_AT1 -> PC6 |
L_AT2 -> PB0 |
L_STEP -> PB9 |
右边电机 |
R_FR -> PB1 |
R_AT1 -> PA5 |
R_AT2 -> PA4 |
R_STEP -> PA1 |
2016/4/10
1. 关于步进电机:昨天晚上把卖家提供的例程进行了改动。因为提供的资料非常少,所以仅仅能先从例程開始,我发现,电机的正转反转是通过FR控制,那么电机动起来就要通过STEP的取正再取反的过程,使得电机活动起来。至于AT1和AT2两个引脚是什么我还没有搞明确,等今天晚上回去再研究。观察小车电机的驱动方法,以及PWM控制,这样电机部分就结束了。
2. 我们如今用的电机应该是混合式两相四线步进电机。
3. LV8731驱动芯片介绍:这个芯片能够用来驱动平衡小车上的两个4线两相步进电机。并且使用这个芯片使得电机的驱动变得更加简单,手冊中说明特长为,增加STEP信号、励磁STEP就能够行进。
4. LV8731端子说明:ATT1(保持通电电流切换端子),ATT2(保持通电电流切换端子),STEP(STEP信号输入端子),FR(CW/CCW信号输入端子)。
5. ATT1。ATT2:也就是程序中的AT1和AT2两个port,这两个端子是用来设置VREF输入电压的衰减功能。原来这两个port仅仅是为了在电机保持通电时,设定较低的输出电流,能够省电,主要是在使用充电电池让小车工作的时候。能够保持较长时间的工作,而不至于电池消耗过快。
6. 那么除了以上两个引脚,我如今还缺FR和STEP两个引脚问题。
从我之前的总结,FR用来控制方向,STEP信号用来变换产生电机转动的动力(通俗的讲)。
7. FR在手冊上说明。就是用来控制正转反转的。也就是说。通过这个驱动芯片。我能够直接通过设置FR这个端子(引脚)的电平就能够控制电机的正反转,简直方便!
8. STEP:由此能够看出。STEP的向上跳变能够使得电机励磁STEP移动,不断的跳变,电机不断的变换相位,假设此时的FR为恒定。那么在STEP不断向上跳变的过程中。电机就会沿着一个方向,以固定的角度(术语为步距角,此电机为1.8度)。
当中的ST端子是CHIP有效端子,也就是芯片使能信号。这里的STP信号才是STEP。
9. 发现一个问题,就是两个电机的FR在某一时刻为相反的电平时。连个电机的转动方向才是同样的。这个原因事实上非常easy,由于两个电机是对称的。那么当一个电机正转时。还有一个电机就必须在它对称的方向反转才干都往一个方向转动,就像镜子往一个方向梳头道理一样。
10. 第二个问题,电机驱动的过程中,假设STEP信号跳变的频率过高的话。也就是STEP频繁的高低电平跳变。那么电机会难以正常转动,并发出轰鸣。
这个问题的解决办法:步进电机存在一个空载启动频率,也就是要在这个频率之下,电机才干正常工作。要是想要加速。那么就要有一个从低速到快速变化的过程,才干达到一定的快速运动的目的,一般设置的启动频率 = 2 * (360 / 1.8),也就是转动一周的脉冲数的两倍,针对这个电机是400,单位应该就是Hz。很多其它的问题能够看看这个博客,我感觉总结的非常好!http://blog.sina.com.cn/s/blog_5e1491780100ucid.html
11. 第三个问题,这个问题比較麻烦。两个电机并不能在比較高或者比較低频率的时候,保持两个电机同样的状态,当频率较低的时候,右边电机居然直接不动了,而左边的电机运作比較正常,这个问题是硬件物理上,到时候能够通过调节PWM波修正,应该也不算大问题。
12. 还遗留了一个问题,就是如今使用驱动芯片,所以驱动步进电机变得非常easy非常简单,可是关键的。步进电机究竟是怎样实现转动的?
13. 步进电机的原理:步进电机就是将电脉冲转化为角位移的运行机构。通俗一点讲:当步进驱动器接收到一个脉冲信号,它就驱动步进电机按设定的方向转动一个固定的角度(及步进角)。如今能够确定的是。我使用的电机为四线两相电机。也就是说,有AB两相。四线就是分为A、 A-、B、 B-这四根线,此时也就是,仅仅要将这四根线轮流的,依照上图的顺序轮流供电,就能够使得电机进行依照步距角转动动作。具体的能够參照此博客http://blog.csdn.net/djimon/article/details/7290297
14. 至此。电机以及电机驱动部分结束。
总结一下,如今项目中使用驱动芯片。这个芯片极大的方便了我们编程。仅仅要我们设置了电机在通电状态下的电流大小。然后设置FR,也就是电机转动方向,通过STEP的向上脉冲驱动电机依照步距角转动就可以,电机上的四根线依照时序变化已经不是我们须要关心的东西了,芯片已经帮我们完毕,那么我们能够有很多其它的精力去进行电机的灵活控制,增加PWM波就可以设置电机转动的速度。
2016/4/11
1. 今天再次尝试读出6050的数据,由于这是小车的一个重点部分,不能稀里糊涂的就觉得已经读出了数据了。
2. 由于我的战舰开发板上,预留了还有一种port模式的6050port。所以我也不知道这样是不是能用,所以我要是实在不可以通过战舰板子读取数据的话。最不济可以用51单片机去模拟,反正也就是个IIC协议。
3. MPU6050:首先。通过模块的原理图。能够看到AD0接地,也就是这个模块的物理地址为0x68。假设AD0接高电平,那么地址就是0x69。
4. 如今使用我自己的战舰开发板上的历程,驱动6050,发现我曾经买的6050模块居然是坏的,换了个模块立刻就读出来了,明明型号都一样,我自己的就是不能用,没办法仅仅能从小车上拆下来进行调试,也幸好这个能够,也第一次看到了数据。
5. 数据总结:Pitch(绕x轴转动的角度)。Roll(绕y轴转动的角度),Yaw(绕z轴转动的角度)这三个是从6050中。通过DMP(数字运动处理)直接读取出来的数据。
注意:。!
!!这里对于角度的理解有问题。!
!!
首先战舰上使用的程序是针对还有一种封装的6050,所以读出来的数据顺势的向我觉得的方向。让我误解了,通过陀螺仪上的丝印(白色的字或图案)。能够看到。我们读出来的数据都是陀螺仪绕着一个轴转动时产生的。就像一根棍子上穿了一个环。那么这个环的转动才是这个轴上变化的数据。那么依据如今小车上使用的6050能够看出。小车前后倒,是绕着X轴。而非Y轴!一開始我以为是Y轴箭头方向才是正方向。是错误的。
6. 小车使用到的角度:依据小车PCB上画的port方向,非常明显基本仅仅用到了X轴的角度变换。详细是怎样,还要继续研究。
7. 如今获得了小车的三个方向的角度数据,接下来就是怎样获得欧拉角(由章动角θ、进动角ψ和自转角φ组成)。
8. 事实上MPU6050输出的四元数。根本不须要我们关心,由于如今我使用了6050中自带的DMP进行获取数据。所以读出来的就是已经被换算好的欧拉角。这样CPU就有很多其它的精力做其它的事情。
2016/4/12
1. 今天配置两个定时器,分别用来为两个电机提供PWM信号,也就是STEP引脚的信号。
2. 当中用到了定时器PWM波输出模式,这个模式,能够通俗的描写叙述出来。就是。在1数到100之间,设置了一个值。在小于这个值得时候一直站着。在大于这个值得时候一直坐着,如此周期往复就出现了PWM波。这个波形也就是给STEP使用。在上面也提到过这个信号,而这个信号也是最好保持50%的占空比使得电机的运动和保持合理分配。
3. 首先看一下我用逻辑分析仪採集到的PWM波
4. 上面一个为左边电机,以下一个为右边电机的STEP引脚,也就是定时器的输出引脚上面的PWM波。
5. 怎样将定时器4和2的PWM波输出映射到左边电机的引脚PB9和右边电机的引脚PA1的?简单的说,在《STM32參考手冊》中,有有关各个定时器以及其它各个中断与引脚的映射关系,仅仅要对比就能够。
须要注意的是,对应的引脚要设置为推挽输出,须要使用对应的引脚就须要初始化对应的通道,使得通道输出到GPIO上,实现PWM的输出。
6. IIC部分我临时不准备细细研究这里面的库函数的用法。反正这是固定的格式,到后面开发。非常多时候都是拿过来用封装得非常好的函数。所以如今我仅仅当它能够这样使用,仅仅要理解它的协议以及它的功能就可以。
7. 关于6050的寄存器配置:到了一定程度的时候。我就不想再自己琢磨着去一个一个寄存器的配置了,而是怎么能够使用别人提供的例程。进行改动,然后能够把模块使用起来,由于这些寄存器的配置和学习STM32的时候一样,仅仅是以某种方式将每一个位代表的值写入到相应的寄存器,那么硬件就能够工作了。
8. 注:对于一些硬件的操作以及芯片内部的配置使用,都是通过配置寄存器实现的,这是我玩了那么多模块以后的感觉,甚至有时候模块现成到根本不用我们关心寄存器的配置。直接能够使用。关键仅仅是怎么用起来,在用起来的基础上再做其它操作。
9. UART2:用于接收蓝牙模块发送来的命令。
10. 关于中断的优先级问题:依据NVIC_IRQChannelPreemptionPriority 和NVIC_IRQChannelSubPriority能够推断他们的优先级,这是有规则的。详细能够见手冊,优先级号越小优先级越高。
11. 中断优先级:定时器2和4作为电机PWM波输出并没实用到中断,所以不存在中断优先级问题,而定时器3的中断优先级要高于串口中断2。
|
TIM2 |
TIM3 |
TIM4 |
UART2 |
PreemptionPriority |
- |
0 |
- |
2 |
SubPriority |
- |
1 |
- |
0 |
12. TIM3: (Period + 1) * (Prescaler+ 1) / 72M = 1ms,定时器3用于产生定时採集MPU6050数据的周期信号,这个信号在20次中断进入后有效,也就是这个周期信号为20ms产生一次,然后让轮询while循环检測到这个标志位,做对应的操作,比方採集数据等。
2016/4/13
1. 6050读出的原始数据问题:通过IIC接口读出6050中的七个数据,那么读出来的数据是如何的呢?首先设置的角速度量程是+-2000度每秒为 +/- 2000,加速度量程是+-2000度每秒为 +/- 4g。所以灵敏度为:65536/4000=16.4LSB/(°/S) 和灵敏度为:65536/8=8192LSB/g。并且,通过手冊,我们能够看到,无论是角度速还是角加速度,都是以两个八位的字节表示一个量,且高位在低字节,比方Gx就是获得了Gx =(data[0] << 8) | data[1]; 这样就获得了一个十六位组成的数据(ADC的精度就是16位)。
2. 关于加速度滤波算法:这里用到的加速度滤波算法是简单的均值滤波。即存储了之前十五个加速度值,然后加上新值,进行平均。就获得新的加速度。可是我觉得这种算法十分简陋,最起码也要加权平均算法,对于较新的值分配较大的权值。这样才比較客观。出现这种情况另一个原因。就是我们根本无法预知下一个加速度的大小,就无法选择更好的算法了。
3. 注:记得在消化完以后,改进这个小算法。
2016/4/14
1. 注:关于6050的角速度偏差矫正,还没有非常好的掌握,这个偏差好像能够为非常多值,可是没有搞懂他们是怎样整定的。所以先默认使用0x01。
2. 关于角速度:我们在上面做的角速度都是没有转化为真正的角速度的值。仅仅是在6050中读出来的原值。那么怎样转化为我们须要的真实角速度的值呢?陀螺仪选的量程是+-2000度每秒。他们每一个数值的寄存器都是16位的。相应的满量程数值为65536,65536/4000 = 16.4,我们仅仅要将读到的数据除以16.4,就能够的到实际的角速度值了。可是做乘法比做除法快,所以我们不除16.4。而是乘以16.4的倒数0.0609756097560976。
能够写个公式 :真实角速度= (设置量程 / 16位满量程) * 原值。
3. 通过角加速度计算出当前位置与某一个平面的夹角。这个是一个数学上的运算,事实上非常easy。
4. 在以上获得当前获得的实际陀螺仪显示出来的真实角度和角速度以后。我们假设想把数据直接用的话,是不行的。这终究是不准确的。存在很多误差,这时必须使用滤波算法,将这些不稳定的因素去除。去除融合完的数据,这样运用到小车上才更加稳定可靠。
5. 卡尔曼滤波算法:这个算法的话,我们仅仅要大体的了解他的原理就能够,然后套用他的五条公式。就能够。原理的话。大体上理解是。通过预測。然后通过“可信度”。来计算出当前的比較正确的值。我也难以对这样的比較复杂的算法做出分析,时间和能力有限。
6. 老实说,我如今根本就不是在做项目,根本就是阅读理解。解析凝视别人现成的程序,这样的不是自己写程序。而是在别人代码上进行理解的工作,个人感觉好像别人在开party,我仅仅能在外面听听动静。我个人还是比較崇尚项目从头開始做的,由于这样知根知底。知道我做了什么。下一步又要做什么。
如今我拿着现成的PCB,现成的代码,就是不断的为这个程序加凝视,而且删除无用部分,同一时候对一些简单的算法进行改进。尽管理解了一个模块的工作原理和一个平衡小车的平衡原理。可是往细了研究就不行了,我根本不能读懂别人的代码中关于算法的部分。可读性太差,全然没凝视。我要推測这是什么东西。为什么要用1000000 / sl。头疼。那么我如今就不准备再深入理解他的PID算法。由于算法都是一样的。正好我们也有一门《计算机控制系统》的课在学,到时候再研究研究理论吧。http://www.arduino.cn/thread-12813-1-1.html,这位仁兄的解说还是非常不错的。
7. 关于P和D的參数整定:我如今用的是这种方法来对參数进行整定,这个过程比較消耗时间,所以须要不断的调试这两个參数,直至比較稳定的站立。
项目思考总结:有点没有善始善终的感觉。也可能是今天有点负面情绪,也可能今天该结束了,毕竟平衡小车的平衡原理明确了,就是算法方面的解读出现故障,一个原因是我自己对计算机控制原理的欠缺,虽说学校正在开这门课。可是我还是学的非常表面。非常理论。并不能非常好的运用到实际来。本来还想用Matlab仿真试试,可是可是,我借口还是挺多的。还有一个原因就是程序可读性非常差。我也第一次有这么深刻的体会。寻常强调的代码规范是多么重要。以后我要引起几倍的重视,我也在一些方面做到了代码规范了。
总结一下平衡小车实现的思路:
1)对6050用IIC读取原始数据;
2)同一时候驱动电机启动;
3)定时開始採样6050中的数据,将数据拿出来滤波、矫正、转化为真实的角度(通过角加速度求得)和角加速度;
4)将角度和角速度通过卡尔曼算法进行融合。获得比較稳定和准确的角度和角加速度;
5)还不算完,要将计算完的角度和角加速度变成对应的PWM波频率,而且要设置速度的上限还有方向,同一时候还要通过PID算法做负反馈,不断的调整这个PWM波频率,将这个频率给定时器初始化,让定时器的周期频率发生改变。从而达到电机转动速度尾随6050的数据变化而变化的目的;
6)最后在PID算法中,须要不断的调整P和D的參数,调整小车的反应速度和稳定性,这是一个耗时的调试操作。
遗憾的是,我如今不能像曾经一样钓鱼能够钓一天,也不能长时间的把精力放在一件事上,这是坏习惯。静不下心,就潜不到深海。