你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32单片机:四旋翼飞行器的飞控实现

[复制链接]
STMCU小助手 发布时间:2022-12-30 18:48

232328047951590.jpg

尝试制作这个四旋翼飞控的过程,感触颇多,整理了思绪之后,把重要的点一一记下来;
这个飞控是基于STM32,整合了MPU6050,即陀螺仪和重力加速计,但没有融合电子罗盘;

另外,四旋翼飞行器的运动方式请百度百科,不太复杂,具体不再赘述;

这是飞控程序的控制流程(一个执行周期):

232148407167682.jpg

比较重要的地方:

1.i2c通信方式;
  因为我不是学电类专业,最开始对i2c这些是没有一点概念,最后通过Google了解了一些原理,然后发现STM32的开发库是带有i2c通信的相关函数的,但是我最后还是没有用这些函数。
我通过GPIO模拟i2c,这样也能获得mpu6050的数据,虽然代码多了一些,但是比较好的理解i2c的原理。
  STM32库实现的模拟i2c代码(注释好像因为编码问题跪了):
  1. /*******************************************************************************
  2. // file                :    i2c_conf.h
  3. // MCU                : STM32F103VET6
  4. // IDE                : Keil uVision4
  5. // date                £º2014.2.28
  6. *******************************************************************************/
  7. #include "stm32f10x.h"

  8. #define   uchar unsigned char
  9. #define   uint unsigned int

  10. #define      FALSE 0  
  11. #define   TRUE  1

  12. void I2C_GPIO_Config(void);
  13. void I2C_delay(void);
  14. void delay5ms(void);
  15. int I2C_Start(void);
  16. void I2C_Stop(void);
  17. void I2C_Ack(void);
  18. void I2C_NoAck(void);
  19. int I2C_WaitAck(void);
  20. void I2C_SendByte(u8 SendByte);
  21. unsigned char I2C_RadeByte(void);
  22. int Single_Write(uchar SlaveAddress,uchar REG_Address,uchar REG_data);
  23. unsigned char Single_Read(unsigned char SlaveAddress,unsigned char REG_Address);
复制代码

  1. /*******************************************************************************
  2. // file                :  i2c_conf.c
  3. // MCU                : STM32F103VET6
  4. // IDE                : Keil uVision4
  5. // date                £º2014.2.28
  6. *******************************************************************************/

  7. #include "i2c_conf.h"

  8. #define SCL_H         GPIOB->BSRR = GPIO_Pin_6      
  9. #define SCL_L         GPIOB->BRR  = GPIO_Pin_6         
  10. #define SDA_H         GPIOB->BSRR = GPIO_Pin_7
  11. #define SDA_L         GPIOB->BRR  = GPIO_Pin_7

  12. #define SCL_read      GPIOB->IDR  & GPIO_Pin_6        //IDR:¶Ë¿ÚÊäÈë¼Ä´æÆ÷¡£
  13. #define SDA_read      GPIOB->IDR  & GPIO_Pin_7

  14.                      


  15. void I2C_GPIO_Config(void)
  16. {
  17.   GPIO_InitTypeDef  GPIO_InitStructure;

  18.   GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_6;
  19.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  20.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;   //¿ªÂ©Êä³öģʽ
  21.   GPIO_Init(GPIOB, &GPIO_InitStructure);

  22.   GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_7;
  23.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  24.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;
  25.   GPIO_Init(GPIOB, &GPIO_InitStructure);
  26. }


  27. void I2C_delay(void)
  28. {
  29.         
  30.    int i=6; //ÕâÀï¿ÉÒÔÓÅ»¯ËÙ¶È    £¬¾­²âÊÔ×îµÍµ½5»¹ÄÜдÈë
  31.    while(i)
  32.    {
  33.      i--;
  34.    }  
  35. }

  36. void delay5ms(void)
  37. {
  38.         
  39.    int i=5000;  
  40.    while(i)
  41.    {
  42.      i--;
  43.    }  
  44. }


  45. int I2C_Start(void)
  46. {
  47.     SDA_H;                                            //II2ЭÒ鹿¶¨±ØÐëÔÚʱÖÓÏßΪµÍµçƽµÄǰÌáÏ£¬²Å¿ÉÒÔÈà Êý¾ÝÏßÐźŸıä
  48.     SCL_H;
  49.     I2C_delay();
  50.    
  51.     if(!SDA_read)
  52.         return FALSE;                //SDAÏßΪµÍµçƽÔò×ÜÏßæ,Í˳ö
  53.     SDA_L;                                
  54.    
  55.     I2C_delay();
  56.    
  57.     if(SDA_read)
  58.         return FALSE;                //SDAÏßΪ¸ßµçƽÔò×ÜÏß³ö´í,Í˳ö
  59.     SDA_L;                                
  60.    
  61.     I2C_delay();                    
  62.    
  63.     return TRUE;
  64. }



  65. void I2C_Stop(void)
  66. {
  67.     SCL_L;
  68.     I2C_delay();
  69.     SDA_L;
  70.     I2C_delay();
  71.    
  72.     SCL_H;
  73.     I2C_delay();
  74.     SDA_H;
  75.     I2C_delay();
  76. }


  77. void I2C_Ack(void)
  78. {   
  79.     SCL_L;
  80.     I2C_delay();
  81.     SDA_L;
  82.     I2C_delay();
  83.     SCL_H;
  84.     I2C_delay();
  85.     SCL_L;
  86.     I2C_delay();
  87. }   


  88. void I2C_NoAck(void)
  89. {   
  90.     SCL_L;
  91.     I2C_delay();
  92.     SDA_H;
  93.     I2C_delay();
  94.     SCL_H;
  95.     I2C_delay();
  96.     SCL_L;
  97.     I2C_delay();
  98. }



  99. int I2C_WaitAck(void)      //·µ»ØÎª:=1ÓÐACK,  =0ÎÞACK
  100. {
  101.     SCL_L;
  102.     I2C_delay();
  103.     SDA_H;            
  104.     I2C_delay();
  105.     SCL_H;
  106.     I2C_delay();
  107.     if(SDA_read)
  108.     {
  109.       SCL_L;
  110.       I2C_delay();
  111.       return FALSE;
  112.     }
  113.     SCL_L;
  114.     I2C_delay();
  115.     return TRUE;
  116. }




  117. void I2C_SendByte(u8 SendByte) //Êý¾Ý´Ó¸ßλµ½µÍλ//
  118. {
  119.     u8 i=8;
  120.     while(i--)
  121.     {
  122.         SCL_L;
  123.         I2C_delay();
  124.             
  125.       if(SendByte&0x80)        // 0x80 = 1000 0000;
  126.         SDA_H;  
  127.       else
  128.         SDA_L;   
  129.             
  130.         SendByte<<=1;   // SendByte×óÒÆÒ»Î»¡£
  131.         I2C_delay();
  132.             
  133.                 SCL_H;
  134.         I2C_delay();
  135.     }
  136.     SCL_L;
  137. }  




  138. unsigned char I2C_RadeByte(void)  //Êý¾Ý´Ó¸ßλµ½µÍλ//
  139. {
  140.     u8 i=8;
  141.     u8 ReceiveByte=0;

  142.     SDA_H;               
  143.     while(i--)
  144.     {
  145.       ReceiveByte<<=1;      //×óÒÆÒ»Î»£¬
  146.             
  147.       SCL_L;
  148.       I2C_delay();
  149.             
  150.             SCL_H;
  151.       I2C_delay();   
  152.             
  153.       if(SDA_read)
  154.       {
  155.         ReceiveByte|=0x01; //дÈë
  156.       }
  157.     }
  158.     SCL_L;
  159.     return ReceiveByte;
  160. }



  161. int Single_Write(uchar SlaveAddress,uchar REG_Address,uchar REG_data)            
  162. {
  163.       if(!I2C_Start())
  164.             return FALSE;
  165.     I2C_SendByte(SlaveAddress);   //·¢ËÍÉ豸µØÖ·+дÐźŠ   //I2C_SendByte(((REG_Address & 0x0700) >>7) | SlaveAddress & 0xFFFE);    //ÉèÖÃ¸ßÆðʼµØÖ·+Æ÷¼þµØÖ·
  166.    
  167.         if(!I2C_WaitAck())
  168.         {
  169.             I2C_Stop();
  170.             return FALSE;
  171.         }
  172.    
  173.         I2C_SendByte(REG_Address );   //ÉèÖÃµÍÆðʼµØÖ·      
  174.     I2C_WaitAck();   
  175.    
  176.         I2C_SendByte(REG_data);
  177.     I2C_WaitAck();   
  178.    
  179.         I2C_Stop();
  180.     delay5ms();
  181.     return TRUE;
  182. }



  183. unsigned char Single_Read(unsigned char SlaveAddress,unsigned char REG_Address)
  184. {   
  185.     unsigned char REG_data;

  186.    
  187.         if(!I2C_Start())
  188.             return FALSE;
  189.     I2C_SendByte(SlaveAddress); //I2C_SendByte(((REG_Address & 0x0700) >>7) | REG_Address & 0xFFFE);//ÉèÖÃ¸ßÆðʼµØÖ·+Æ÷¼þµØÖ·
  190.     if(!I2C_WaitAck())
  191.         {
  192.             I2C_Stop();

  193.             return FALSE;
  194.         }
  195.     I2C_SendByte((u8) REG_Address);   //ÉèÖÃµÍÆðʼµØÖ·      
  196.     I2C_WaitAck();
  197.         I2C_Start();
  198.    
  199.         I2C_SendByte(SlaveAddress+1);
  200.     I2C_WaitAck();

  201.         REG_data= I2C_RadeByte();
  202.     I2C_NoAck();
  203.     I2C_Stop();

  204.     return REG_data;

  205. }
复制代码

2.mpu6050;
  然后用写好的模拟i2c函数读取mpu6050,根据mpu6050手册的各寄存器地址,读取到了重力加速计和陀螺仪的各分量;

传感器采样率设置为200Hz,这个值是因为我电调频率为200Hz,也就是说,我的程序循环一次0.005s,一般来说,采样率高点没问题,别比执行一次闭环控制的周期长就行了;

陀螺仪量程±2000°/s,加速计量程±2g, 量程越大,取值越不精确;

这里注意,由于我们没有采用磁力计,而陀螺仪存在零偏,所以最终在yaw方向上没有绝对的参考系,不能建立绝对的地理坐标系,这样最好的结果也仅仅是在yaw上存在缓慢漂移。

3.互补滤波;
  融合时,陀螺仪的积分运算很大程度上决定了飞行器的瞬时运动情况,而重力加速计通过长时间的累积不断矫正陀螺仪产生的误差,最终得到准确的机体姿态。
  这里我们采用Madgwick提供的UpdateIMU算法来得到姿态角所对应的四元数,之后只需要经过简单运算便可转换为实时欧拉角。感谢Madgwick大大为开源做出的贡献。
  1. #define sampleFreq    512.0f        // sample frequency in Hz
  2. #define betaDef        0.1f        // 2 * proportional gain


  3. volatile float beta = betaDef;
  4. volatile float q0 = 1.0f, q1 = 0.0f, q2 = 0.0f, q3 = 0.0f;

  5. float invSqrt(float x);

  6. void MadgwickAHRSupdateIMU(float gx, float gy, float gz, float ax, float ay, float az) {
  7.     float recipNorm;
  8.     float s0, s1, s2, s3;
  9.     float qDot1, qDot2, qDot3, qDot4;
  10.     float _2q0, _2q1, _2q2, _2q3, _4q0, _4q1, _4q2 ,_8q1, _8q2, q0q0, q1q1, q2q2, q3q3;

  11.     // Rate of change of quaternion from gyroscope
  12.     qDot1 = 0.5f * (-q1 * gx - q2 * gy - q3 * gz);
  13.     qDot2 = 0.5f * (q0 * gx + q2 * gz - q3 * gy);
  14.     qDot3 = 0.5f * (q0 * gy - q1 * gz + q3 * gx);
  15.     qDot4 = 0.5f * (q0 * gz + q1 * gy - q2 * gx);

  16.     // Compute feedback only if accelerometer measurement valid (avoids NaN in accelerometer normalisation)
  17.     if(!((ax == 0.0f) && (ay == 0.0f) && (az == 0.0f))) {

  18.         // Normalise accelerometer measurement
  19.         recipNorm = invSqrt(ax * ax + ay * ay + az * az);
  20.         ax *= recipNorm;
  21.         ay *= recipNorm;
  22.         az *= recipNorm;

  23.         // Auxiliary variables to avoid repeated arithmetic
  24.         _2q0 = 2.0f * q0;
  25.         _2q1 = 2.0f * q1;
  26.         _2q2 = 2.0f * q2;
  27.         _2q3 = 2.0f * q3;
  28.         _4q0 = 4.0f * q0;
  29.         _4q1 = 4.0f * q1;
  30.         _4q2 = 4.0f * q2;
  31.         _8q1 = 8.0f * q1;
  32.         _8q2 = 8.0f * q2;
  33.         q0q0 = q0 * q0;
  34.         q1q1 = q1 * q1;
  35.         q2q2 = q2 * q2;
  36.         q3q3 = q3 * q3;

  37.         // Gradient decent algorithm corrective step
  38.         s0 = _4q0 * q2q2 + _2q2 * ax + _4q0 * q1q1 - _2q1 * ay;
  39.         s1 = _4q1 * q3q3 - _2q3 * ax + 4.0f * q0q0 * q1 - _2q0 * ay - _4q1 + _8q1 * q1q1 + _8q1 * q2q2 + _4q1 * az;
  40.         s2 = 4.0f * q0q0 * q2 + _2q0 * ax + _4q2 * q3q3 - _2q3 * ay - _4q2 + _8q2 * q1q1 + _8q2 * q2q2 + _4q2 * az;
  41.         s3 = 4.0f * q1q1 * q3 - _2q1 * ax + 4.0f * q2q2 * q3 - _2q2 * ay;
  42.         recipNorm = invSqrt(s0 * s0 + s1 * s1 + s2 * s2 + s3 * s3); // normalise step magnitude
  43.         s0 *= recipNorm;
  44.         s1 *= recipNorm;
  45.         s2 *= recipNorm;
  46.         s3 *= recipNorm;

  47.         // Apply feedback step
  48.         qDot1 -= beta * s0;
  49.         qDot2 -= beta * s1;
  50.         qDot3 -= beta * s2;
  51.         qDot4 -= beta * s3;
  52.     }

  53.     // Integrate rate of change of quaternion to yield quaternion
  54.     q0 += qDot1 * (1.0f / sampleFreq);
  55.     q1 += qDot2 * (1.0f / sampleFreq);
  56.     q2 += qDot3 * (1.0f / sampleFreq);
  57.     q3 += qDot4 * (1.0f / sampleFreq);

  58.     // Normalise quaternion
  59.     recipNorm = invSqrt(q0 * q0 + q1 * q1 + q2 * q2 + q3 * q3);
  60.     q0 *= recipNorm;
  61.     q1 *= recipNorm;
  62.     q2 *= recipNorm;
  63.     q3 *= recipNorm;
  64. }


  65. float invSqrt(float x) {
  66.     float halfx = 0.5f * x;
  67.     float y = x;
  68.     long i = *(long*)&y;
  69.     i = 0x5f3759df - (i>>1);
  70.     y = *(float*)&i;
  71.     y = y * (1.5f - (halfx * y * y));
  72.     return y;
  73. }
复制代码

4.获取期望姿态;
  也就是遥控部分了,让用户介入控制。
  本着拿来主义的原则,用上”圆点博士开源项目”提供的安卓的开源蓝牙控制端。

232318220769179.jpg

  圆点博士给出了数据包格式,同过HC-06蓝牙模块接连到STM32串口1,再无线连接到控制端,这样我们就可以获得控制端不断发送的数据包了,并实时更新期望姿态角,这里只需要注意输出的姿态角和实时姿态角方向一致以及数据包的校验就行了。

5.PID控制算法;
  由于简单的线性控制不可能满足四轴飞行器这个灵敏的系统,引入PID控制器来更好的纠正系统。
  简介:PID实指“比例proportional”、“积分integral”、“微分derivative”,这三项构成PID基本要素。每一项完成不同任务,对系统功能产生不同的影响。

232323437163784.jpg

以Pitch为例:
  
232324081702580.png

error为期望角减去实时角度得到的误差;
iState为积分i参数对应累积过去时间里的误差总和;
if语句限定iState范围,繁殖修正过度;
微分d参数为当前姿态减去上次姿态,估算当前速度(瞬间速度);
总调整量为p,i,d三者之和;

这样,P代表控制系统的响应速度,越大,响应越快。
I,用来累积过去时间内的误差,修正P无法达到的期望姿态值(静差);
D,加强对机体变化的快速响应,对P有抑制作用。

PID各参数的整定需要综合考虑控制系统的各个方面,才能达到最佳效果。

输出PWM信号:
PID计算完成之后,便可以通过STM32自带的定时资源很容易的调制出四路pwm信号,采用的电调pwm格式为50Hz,高电平持续时间0.5ms-2.5ms;
我以1.0ms-2.0ms为每个电机的油门行程,这样,1ms的宽度均匀的对应电调的从最低到最高转速。
至此,一个用stm32和mpu6050搭建的飞控系统就算实现了。
剩下的调试PID参数了。

转载自:Moran_
收藏 评论0 发布时间:2022-12-30 18:48

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版