当前位置: 首页 > news >正文

返回链接 网站惩罚检查 错误检查外国炫酷网站网址

返回链接 网站惩罚检查 错误检查,外国炫酷网站网址,成都公司注册费用,免费网站下载直播软件前言#xff1a;本文为手把手教学飞控核心知识点之一的姿态解算——MPU6050 姿态解算#xff08;飞控专栏第2篇#xff09;。项目中飞行器使用 MPU6050 传感器对飞行器的姿态进行解算#xff08;四元数方法#xff09;#xff0c;搭配设计的卡尔曼滤波器与一阶低通滤波器…前言本文为手把手教学飞控核心知识点之一的姿态解算——MPU6050 姿态解算飞控专栏第2篇。项目中飞行器使用 MPU6050 传感器对飞行器的姿态进行解算四元数方法搭配设计的卡尔曼滤波器与一阶低通滤波器进行数据滤波。当然本篇博客也将为读者朋友教学业内匿名者上位机的代码移植和使用方法。为了方便读者朋友学习本博客将使用传感器模块替代整机进行教学方便读者朋友后续根据自己实际情况移植文末有代码开源 实验硬件 STM32F103C8T6MPU6050USB转TTL 硬件实物图 效果图 一、飞行器姿态解算 1.1 MPU6050概述  飞行器通常搭载一款姿态传感器不管是六轴还是九轴姿态传感器本项目中以最常见的 MPU6050 为例。MPU6050 传感器其实并不能直接输出我们飞行器飞行过程中的欧拉角(Euler-angles)通过读取它的传感器我们可以得到3轴角速度3轴角加速度。得到的角速度和角加速度信息我们是无法直接使用的这个时候我们可以选择使用 DMP 去解算此时飞行器的欧拉角(Euler-angles)情况。当然作者在项目中并没有使用 DMP 去解算飞行器的欧拉角(Euler-angles)而是使用了四元数解算的方法 DMPDigital Motion Processor是一种数字运动处理器它可以从MPU6050等传感器中读取数据并进行解算以获取姿态信息。下面是DMP解算MPU6050的优缺点 优点 DMP使用简单可以直接套用官方库进行开发无需自己编写解算算法。DMP不会占用太多的资源因为它只需要读取传感器数据并进行简单的解算。DMP的输出数据经过处理可以直接用于姿态控制等应用无需再进行复杂的计算。 缺点 DMP的输出数据精度可能不够高特别是在高精度传感场景下。DMP的输出数据不稳定可能会受到噪声等因素的影响。DMP无法测量偏航角只能获取滚动角和俯仰角的信息。 1.2 四元数姿态解算 本小节将以下方思维导图进行分析讲解 初次接触的读者朋友可能对四元数较为陌生这里作者建议大家直接去阅读秦永元的《惯性导航》里面有非常好的讲解大家可以直接看绪论和第九章就可以。 《惯性导航》PDF地址惯性导航第三版 (sciencereading.cn) 下面我们根据思维导图用程序来一步一步实现如何求解欧拉角 1、定义初始四元数的值为q01,q10,q20,q30。 2、读取加速度计值、角速度值程序定义变量分别为ax、ay、azgx、gy、gz将陀螺仪值转为弧度转换如下 gx gx * 0.0174f; //1度0.0174弧度 gy gy * 0.0174f; gz gz * 0.0174f; 3、对加速度值进行归一化 //提取等效旋转矩阵中的重力分量 Gravity.x 2*(NumQ.q1 * NumQ.q3 - NumQ.q0 * NumQ.q2); Gravity.y 2*(NumQ.q0 * NumQ.q1 NumQ.q2 * NumQ.q3); Gravity.z 1-2*(NumQ.q1 * NumQ.q1 NumQ.q2 * NumQ.q2); //加速度归一化 NormAcc 1/sqrt(squa(MPU6050.accX) squa(MPU6050.accY) squa(MPU6050.accZ));//归一化计算 Acc.x pMpu-accX * NormAcc; Acc.y pMpu-accY * NormAcc; Acc.z pMpu-accZ * NormAcc; 4、提取姿态矩阵中的重力分量我们已经得到数学计算公式为 //提取等效旋转矩阵中的重力分量 Gravity.x 2*(NumQ.q1 * NumQ.q3 - NumQ.q0 * NumQ.q2); Gravity.y 2*(NumQ.q0 * NumQ.q1 NumQ.q2 * NumQ.q3); Gravity.z 1-2*(NumQ.q1 * NumQ.q1 NumQ.q2 * NumQ.q2); 5、求姿态误差对两向量进行叉乘(定义ex、ey、ez为三个轴误差元素)数学计算为  //向量差乘得出的值 AccGravity.x (Acc.y * Gravity.z - Acc.z * Gravity.y); AccGravity.y (Acc.z * Gravity.x - Acc.x * Gravity.z); AccGravity.z (Acc.x * Gravity.y - Acc.y * Gravity.x); 6、互补滤波将误差输入PID控制器后与陀螺仪测得的角速度相加修正角速度值程序实现如下(Kp为互补滤波系数这里取Kp0.5实际值根据需要进行调整)  //角速度融合加速度积分补偿值 Gyro.x pMpu-gyroX * Gyro_Gr KpDef * AccGravity.x GyroIntegError.x;//弧度制 Gyro.y pMpu-gyroY * Gyro_Gr KpDef * AccGravity.y GyroIntegError.y; Gyro.z pMpu-gyroZ * Gyro_Gr KpDef * AccGravity.z GyroIntegError.z; 7、解四元数微分方程其数学计算如下初始值q0 1,q1 0,q2 0,q3 0,为角速度为周期时间 // 一阶龙格库塔法, 更新四元数 q0_t (-NumQ.q1*Gyro.x - NumQ.q2*Gyro.y - NumQ.q3*Gyro.z) * HalfTime; q1_t ( NumQ.q0*Gyro.x - NumQ.q3*Gyro.y NumQ.q2*Gyro.z) * HalfTime; q2_t ( NumQ.q3*Gyro.x NumQ.q0*Gyro.y - NumQ.q1*Gyro.z) * HalfTime; q3_t (-NumQ.q2*Gyro.x NumQ.q1*Gyro.y NumQ.q0*Gyro.z) * HalfTime;NumQ.q0 q0_t; NumQ.q1 q1_t; NumQ.q2 q2_t; NumQ.q3 q3_t; 8、四元数归一化归一化方法与加速度归一化方法一样 // 四元数归一化 NormQuat 1/sqrt(squa(NumQ.q0) squa(NumQ.q1) squa(NumQ.q2) squa(NumQ.q3)); NumQ.q0 * NormQuat; NumQ.q1 * NormQuat; NumQ.q2 * NormQuat; NumQ.q3 * NormQuat; 9、计算姿态角数学公式为 #ifdef YAW_GYRO*(float *)pAngE atan2f(2 * NumQ.q1 *NumQ.q2 2 * NumQ.q0 * NumQ.q3, 1 - 2 * NumQ.q2 *NumQ.q2 - 2 * NumQ.q3 * NumQ.q3) * RtA; //yaw#elsefloat yaw_G pMpu-gyroZ * Gyro_G;if((yaw_G 1.0f) || (yaw_G -1.0f)) //数据太小可以认为是干扰不是偏航动作{pAngE-yaw yaw_G * dt; }#endif pAngE-pitch asin(2 * NumQ.q0 *NumQ.q2 - 2 * NumQ.q1 * NumQ.q3) * RtA; pAngE-roll atan2(2 * NumQ.q2 *NumQ.q3 2 * NumQ.q0 * NumQ.q1, 1 - 2 * NumQ.q1 *NumQ.q1 - 2 * NumQ.q2 * NumQ.q2) * RtA; //PITCH 二、卡尔曼滤波详解 卡尔曼的本质递归式最优评估。 卡尔曼的好处是①效率最高甚至是最有用的在系统中能够快速的消除高速白噪声②不会产生严重滞后③所需数据存储量较小便于进行实时处理 在飞行器中卡尔曼滤波的高效率性是十分优秀的但是卡尔曼不能抵抗突变干扰这点在飞行器中一般不会出现数据突变跳变所以卡尔曼很适合运用于四轴飞行器。 飞行器项目中卡尔曼滤波的目标对象是加速度三轴加速度都是独立变量可以分别独立用一维线性卡尔曼进行滤波。由于加速度容易受震动干扰它就是一个很典型的高频高斯白噪声噪声随机对称符合高斯分布的噪声所以加速度用卡尔曼滤波。 角速度不容易受到干扰就用简单的一阶低通互补滤波。 飞控中的滤波算法角加速度卡尔曼滤波角速度一阶互补滤波 将上述数学公式代码化后可以得到卡尔曼滤波代码 #include kalman.h//一维卡尔曼滤波 void kalmanfiter(struct KalmanFilter *EKF,float input) {EKF-NewP EKF-LastP EKF-Q;EKF-Kg EKF-NewP / (EKF-NewP EKF-R);EKF-Out EKF-Out EKF-Kg * (input - EKF-Out);EKF-LastP (1 - EKF-Kg) * EKF-NewP; } 三、CubeMX配置 1、RCC配置外部高速晶振精度更高——HSE 2、SYS配置Debug设置成Serial Wire否则可能导致芯片自锁 3、TIM1配置在TIM1的中断回调函数中发生MPU6050姿态解算与控制都是中断周期3ms 4、I2C1配置配置MCU与MPU6050之间的通讯协议 5、UART配置通过UART1与匿名上位机进行通讯显示飞行器3D姿态 6、时钟树配置 7、工程配置 四、代码与解析 4.1 MPU6050代码 mpu6050.h代码 mpu6050代码中核心是通过I2C通讯读取寄存器地址为0X3B 和 0x43 的数值分别为角加速度和角速度。 #ifndef __MPU6050_H #define __MPU6050_H#include stm32f1xx_hal.h//用什么系列就是什么 //#define MPU_ACCEL_OFFS_REG 0X06 //accel_offs寄存器,可读取版本号,寄存器手册未提到 //#define MPU_PROD_ID_REG 0X0C //prod id寄存器,在寄存器手册未提到 #define MPU_SELF_TESTX_REG 0X0D //自检寄存器X #define MPU_SELF_TESTY_REG 0X0E //自检寄存器Y #define MPU_SELF_TESTZ_REG 0X0F //自检寄存器Z #define MPU_SELF_TESTA_REG 0X10 //自检寄存器A #define MPU_SAMPLE_RATE_REG 0X19 //采样频率分频器 #define MPU_CFG_REG 0X1A //配置寄存器 #define MPU_GYRO_CFG_REG 0X1B //陀螺仪配置寄存器 #define MPU_ACCEL_CFG_REG 0X1C //加速度计配置寄存器 #define MPU_MOTION_DET_REG 0X1F //运动检测阀值设置寄存器 #define MPU_FIFO_EN_REG 0X23 //FIFO使能寄存器 #define MPU_I2CMST_CTRL_REG 0X24 //IIC主机控制寄存器 #define MPU_I2CSLV0_ADDR_REG 0X25 //IIC从机0器件地址寄存器 #define MPU_I2CSLV0_REG 0X26 //IIC从机0数据地址寄存器 #define MPU_I2CSLV0_CTRL_REG 0X27 //IIC从机0控制寄存器 #define MPU_I2CSLV1_ADDR_REG 0X28 //IIC从机1器件地址寄存器 #define MPU_I2CSLV1_REG 0X29 //IIC从机1数据地址寄存器 #define MPU_I2CSLV1_CTRL_REG 0X2A //IIC从机1控制寄存器 #define MPU_I2CSLV2_ADDR_REG 0X2B //IIC从机2器件地址寄存器 #define MPU_I2CSLV2_REG 0X2C //IIC从机2数据地址寄存器 #define MPU_I2CSLV2_CTRL_REG 0X2D //IIC从机2控制寄存器 #define MPU_I2CSLV3_ADDR_REG 0X2E //IIC从机3器件地址寄存器 #define MPU_I2CSLV3_REG 0X2F //IIC从机3数据地址寄存器 #define MPU_I2CSLV3_CTRL_REG 0X30 //IIC从机3控制寄存器 #define MPU_I2CSLV4_ADDR_REG 0X31 //IIC从机4器件地址寄存器 #define MPU_I2CSLV4_REG 0X32 //IIC从机4数据地址寄存器 #define MPU_I2CSLV4_DO_REG 0X33 //IIC从机4写数据寄存器 #define MPU_I2CSLV4_CTRL_REG 0X34 //IIC从机4控制寄存器 #define MPU_I2CSLV4_DI_REG 0X35 //IIC从机4读数据寄存器#define MPU_I2CMST_STA_REG 0X36 //IIC主机状态寄存器 #define MPU_INTBP_CFG_REG 0X37 //中断/旁路设置寄存器 #define MPU_INT_EN_REG 0X38 //中断使能寄存器 #define MPU_INT_STA_REG 0X3A //中断状态寄存器#define MPU_ACCEL_XOUTH_REG 0X3B //加速度值,X轴高8位寄存器 #define MPU_ACCEL_XOUTL_REG 0X3C //加速度值,X轴低8位寄存器 #define MPU_ACCEL_YOUTH_REG 0X3D //加速度值,Y轴高8位寄存器 #define MPU_ACCEL_YOUTL_REG 0X3E //加速度值,Y轴低8位寄存器 #define MPU_ACCEL_ZOUTH_REG 0X3F //加速度值,Z轴高8位寄存器 #define MPU_ACCEL_ZOUTL_REG 0X40 //加速度值,Z轴低8位寄存器#define MPU_TEMP_OUTH_REG 0X41 //温度值高八位寄存器 #define MPU_TEMP_OUTL_REG 0X42 //温度值低8位寄存器#define MPU_GYRO_XOUTH_REG 0X43 //陀螺仪值,X轴高8位寄存器 #define MPU_GYRO_XOUTL_REG 0X44 //陀螺仪值,X轴低8位寄存器 #define MPU_GYRO_YOUTH_REG 0X45 //陀螺仪值,Y轴高8位寄存器 #define MPU_GYRO_YOUTL_REG 0X46 //陀螺仪值,Y轴低8位寄存器 #define MPU_GYRO_ZOUTH_REG 0X47 //陀螺仪值,Z轴高8位寄存器 #define MPU_GYRO_ZOUTL_REG 0X48 //陀螺仪值,Z轴低8位寄存器#define MPU_I2CSLV0_DO_REG 0X63 //IIC从机0数据寄存器 #define MPU_I2CSLV1_DO_REG 0X64 //IIC从机1数据寄存器 #define MPU_I2CSLV2_DO_REG 0X65 //IIC从机2数据寄存器 #define MPU_I2CSLV3_DO_REG 0X66 //IIC从机3数据寄存器#define MPU_I2CMST_DELAY_REG 0X67 //IIC主机延时管理寄存器 #define MPU_SIGPATH_RST_REG 0X68 //信号通道复位寄存器 #define MPU_MDETECT_CTRL_REG 0X69 //运动检测控制寄存器 #define MPU_USER_CTRL_REG 0X6A //用户控制寄存器 #define MPU_PWR_MGMT1_REG 0X6B //电源管理寄存器1 #define MPU_PWR_MGMT2_REG 0X6C //电源管理寄存器2 #define MPU_FIFO_CNTH_REG 0X72 //FIFO计数寄存器高八位 #define MPU_FIFO_CNTL_REG 0X73 //FIFO计数寄存器低八位 #define MPU_FIFO_RW_REG 0X74 //FIFO读写寄存器 #define MPU_DEVICE_ID_REG 0X75 //器件ID寄存器,who am i寄存器//如果AD0脚(9脚)接地,IIC地址为0X68(不包含最低位). //如果接V3.3,则IIC地址为0X69(不包含最低位). #define MPU_ADDR 0X68//因为MPU6050的AD0接GND,所以则读写地址分别为0XD1和0XD0 // (如果AD0接VCC,则读写地址分别为0XD3和0XD2) #define MPU_READ 0XD1 #define MPU_WRITE 0XD0uint8_t MPU_Init(void); //初始化MPU6050 uint8_t MPU_Write_Len(uint8_t reg,uint8_t len,uint8_t *buf); //IIC连续写 uint8_t MPU_Read_Len(uint8_t reg,uint8_t len,uint8_t *buf); //IIC连续读 uint8_t MPU_Write_Byte(uint8_t reg,uint8_t data); //IIC写一个字节 uint8_t MPU_Read_Byte(uint8_t reg); //IIC读一个字节uint8_t MPU_Set_Gyro_Fsr(uint8_t fsr); uint8_t MPU_Set_Accel_Fsr(uint8_t fsr); uint8_t MPU_Set_LPF(uint16_t lpf); uint8_t MPU_Set_Rate(uint16_t rate); uint8_t MPU_Set_Fifo(uint8_t sens);float MPU_Get_Temperature(void); uint8_t MPU_Get_Gyroscope(short *gx,short *gy,short *gz); uint8_t MPU_Get_Accelerometer(short *ax,short *ay,short *az);void MpuGetData(void);#endif mpu6050.c代码 #include mpu6050.h #include alldata.h #include kalman.h #include stdio.h #include i2c.hstatic volatile int16_t *pMpu (int16_t *)MPU6050; int16_t MpuOffset[6] {0}; //MPU6050补偿数值//初始化MPU6050 //返回值:0,成功 // 其他,错误代码 uint8_t MPU_Init(void) { uint8_t res;extern I2C_HandleTypeDef hi2c1;HAL_I2C_Init(hi2c1);MPU_Write_Byte(MPU_PWR_MGMT1_REG,0X80); //复位MPU6050MPU_Write_Byte(MPU_PWR_MGMT1_REG,0X00); //唤醒MPU6050 MPU_Set_Gyro_Fsr(3); //陀螺仪传感器,±2000dpsMPU_Set_Accel_Fsr(0); //加速度传感器,±2gMPU_Set_Rate(50); //设置采样率50HzMPU_Write_Byte(MPU_INT_EN_REG,0X00); //关闭所有中断MPU_Write_Byte(MPU_USER_CTRL_REG,0X00); //I2C主模式关闭MPU_Write_Byte(MPU_FIFO_EN_REG,0X00); //关闭FIFOMPU_Write_Byte(MPU_INTBP_CFG_REG,0X80); //INT引脚低电平有效resMPU_Read_Byte(MPU_DEVICE_ID_REG);printf(\r\nMPU6050:0x%2x\r\n,res);if(resMPU_ADDR)//器件ID正确{MPU_Write_Byte(MPU_PWR_MGMT1_REG,0X01); //设置CLKSEL,PLL X轴为参考MPU_Write_Byte(MPU_PWR_MGMT2_REG,0X00); //加速度与陀螺仪都工作MPU_Set_Rate(50); //设置采样率为50Hz}else return 1;return 0; } //设置MPU6050陀螺仪传感器满量程范围 //fsr:0,±250dps;1,±500dps;2,±1000dps;3,±2000dps //返回值:0,设置成功 // 其他,设置失败 uint8_t MPU_Set_Gyro_Fsr(uint8_t fsr) {return MPU_Write_Byte(MPU_GYRO_CFG_REG,fsr3);//设置陀螺仪满量程范围 } //设置MPU6050加速度传感器满量程范围 //fsr:0,±2g;1,±4g;2,±8g;3,±16g //返回值:0,设置成功 // 其他,设置失败 uint8_t MPU_Set_Accel_Fsr(uint8_t fsr) {return MPU_Write_Byte(MPU_ACCEL_CFG_REG,fsr3);//设置加速度传感器满量程范围 } //设置MPU6050的数字低通滤波器 //lpf:数字低通滤波频率(Hz) //返回值:0,设置成功 // 其他,设置失败 uint8_t MPU_Set_LPF(uint16_t lpf) {uint8_t data0;if(lpf188)data1;else if(lpf98)data2;else if(lpf42)data3;else if(lpf20)data4;else if(lpf10)data5;else data6; return MPU_Write_Byte(MPU_CFG_REG,data);//设置数字低通滤波器 } //设置MPU6050的采样率(假定Fs1KHz) //rate:4~1000(Hz) //返回值:0,设置成功 // 其他,设置失败 uint8_t MPU_Set_Rate(uint16_t rate) {uint8_t data;if(rate1000)rate1000;if(rate4)rate4;data1000/rate-1;dataMPU_Write_Byte(MPU_SAMPLE_RATE_REG,data); //设置数字低通滤波器return MPU_Set_LPF(rate/2); //自动设置LPF为采样率的一半 }//得到温度值 //返回值:温度值(扩大了100倍) float MPU_Get_Temperature(void) {unsigned char buf[2]; short raw;float temp;MPU_Read_Len(MPU_TEMP_OUTH_REG,2,buf); raw(buf[0]8)| buf[1]; temp(36.53((double)raw)/340)*100; // temp (long)((35 (raw / 340)) * 65536L);return temp/100.0f; } //得到陀螺仪值(原始值) //gx,gy,gz:陀螺仪x,y,z轴的原始读数(带符号) //返回值:0,成功 // 其他,错误代码 uint8_t MPU_Get_Gyroscope(short *gx,short *gy,short *gz) {uint8_t buf[6],res; resMPU_Read_Len(MPU_GYRO_XOUTH_REG,6,buf);if(res0){*gx((uint16_t)buf[0]8)|buf[1]; *gy((uint16_t)buf[2]8)|buf[3]; *gz((uint16_t)buf[4]8)|buf[5];} return res; } //得到加速度值(原始值) //gx,gy,gz:陀螺仪x,y,z轴的原始读数(带符号) //返回值:0,成功 // 其他,错误代码 uint8_t MPU_Get_Accelerometer(short *ax,short *ay,short *az) {uint8_t buf[6],res; resMPU_Read_Len(MPU_ACCEL_XOUTH_REG,6,buf);if(res0){*ax((uint16_t)buf[0]8)|buf[1]; *ay((uint16_t)buf[2]8)|buf[3]; *az((uint16_t)buf[4]8)|buf[5];} return res;; }//IIC连续写 uint8_t MPU_Write_Len(uint8_t reg,uint8_t len,uint8_t *buf) {extern I2C_HandleTypeDef hi2c1;HAL_I2C_Mem_Write(hi2c1, MPU_WRITE, reg, I2C_MEMADD_SIZE_8BIT, buf, len, 0xfff);HAL_Delay(100);return 0; } //IIC连续读 //addr:器件地址 //reg:要读取的寄存器地址 //len:要读取的长度 //buf:读取到的数据存储区 //返回值:0,正常 // 其他,错误代码 uint8_t MPU_Read_Len(uint8_t reg,uint8_t len,uint8_t *buf) { extern I2C_HandleTypeDef hi2c1;HAL_I2C_Mem_Read(hi2c1, MPU_READ, reg, I2C_MEMADD_SIZE_8BIT, buf, len, 0xfff);HAL_Delay(100);return 0; } //IIC写一个字节 //reg:寄存器地址 //data:数据 //返回值:0,正常 // 其他,错误代码 uint8_t MPU_Write_Byte(uint8_t reg,uint8_t data) { extern I2C_HandleTypeDef hi2c1;unsigned char W_Data0;W_Data data;HAL_I2C_Mem_Write(hi2c1, MPU_WRITE, reg, I2C_MEMADD_SIZE_8BIT, W_Data, 1, 0xfff);HAL_Delay(100);return 0; } //IIC读一个字节 //reg:寄存器地址 //返回值:读到的数据 uint8_t MPU_Read_Byte(uint8_t reg) {extern I2C_HandleTypeDef hi2c1;unsigned char R_Data0;HAL_I2C_Mem_Read(hi2c1, MPU_READ, reg, I2C_MEMADD_SIZE_8BIT, R_Data, 1, 0xfff);HAL_Delay(100);return R_Data; }/* 读取MPU6050数据并加滤波 */ void MpuGetData(void) {uint8_t i;uint8_t buffer[12];HAL_I2C_Mem_Read(hi2c1, MPU_READ, 0X3B, I2C_MEMADD_SIZE_8BIT, buffer, 6, 0xfff); /* 读取角加速度 */HAL_I2C_Mem_Read(hi2c1, MPU_READ, 0x43, I2C_MEMADD_SIZE_8BIT, buffer[6], 6, 0xfff); /* 读取角速度 */for(i0;i6;i){pMpu[i] (((int16_t)buffer[i1] 8) | buffer[(i1)1])-MpuOffset[i]; /* 将数据整为16bit并减去水平校准值 */if(i 3) /* 角加速度卡尔曼滤波 */{{static struct KalmanFilter EKF[3] {{0.02,0,0,0,0.001,0.543},{0.02,0,0,0,0.001,0.543},{0.02,0,0,0,0.001,0.543}}; kalmanfiter(EKF[i],(float)pMpu[i]); pMpu[i] (int16_t)EKF[i].Out;}}if(i 2) /* 角速度一阶互补滤波 */{ uint8_t ki-3;const float factor 0.15f; static float tBuff[3]; pMpu[i] tBuff[k] tBuff[k] * (1 - factor) pMpu[i] * factor; }} } 4.2 Kalman和一阶互补滤波滤波代码 kalman.c #include kalman.h//一维卡尔曼滤波 void kalmanfiter(struct KalmanFilter *EKF,float input) {EKF-NewP EKF-LastP EKF-Q;EKF-Kg EKF-NewP / (EKF-NewP EKF-R);EKF-Out EKF-Out EKF-Kg * (input - EKF-Out);EKF-LastP (1 - EKF-Kg) * EKF-NewP; } 一阶互补滤波 if(i 2) /* 角速度一阶互补滤波 */ { uint8_t ki-3;const float factor 0.15f; /* 互补滤波的影响因子 */static float tBuff[3]; pMpu[i] tBuff[k] tBuff[k] * (1 - factor) pMpu[i] * factor; } 以上 2 种滤波都存在 mpu6050.c 代码中进行调用 这个适当取 P 0.02 默认初始第一比数据的协方差不为 0也就是不承认第一笔数据完全准确。取 Q0.001就是认为测量方差的过程噪声比较小实际上多小也没人知道随便给个值优化输出。而 R 比较大默认噪声都是信号采集就已经产生的了(XYZ 三轴加速度都是一个样所以参数相同 )。 Q/R 决定着滤波的强度就是收敛速度。收敛速度越快越容易对正确的信号不信任导致正常的信号也被衰减所以滤波有好处有坏处适当就好。  现在来介绍一下角速度的滤波。由于角速度不容易受干扰又需要保持最快速度反应。只要稍微适当给个滤波去除白噪声。本次采用一阶低通滤波。 一阶低通数字滤波器的公式为 y(n) q*x(n) (1-q)*y(n-1) 4.3 IMU代码姿态解算 这部分代码其实就是第一章节的四元数姿态求解具体推导过程和代码思路参考上文还是建议大家多花点时间消化消化。 imu.h: #ifndef __IMU_H #define __IMU_H#include alldata.h#define squa( Sq ) (((float)Sq)*((float)Sq))extern void GetAngle(const _st_Mpu *pMpu,_st_AngE *pAngE, float dt); extern void imu_rest(void);#endif imu.c: #include imu.h #include mpu6050.h #include math.hconst float M_PI 3.1415926535; const float RtA 57.2957795f; const float AtR 0.0174532925f; const float Gyro_G 0.03051756f*2; //陀螺仪初始化量程-2000度每秒于1 / (65536 / 4000) 0.03051756*2 const float Gyro_Gr 0.0005326f*2; //面计算度每秒,转换弧度每秒则 2*0.03051756 * 0.0174533f 0.0005326*2static float NormAcc;/* 四元数系数 */ typedef volatile struct {float q0;float q1;float q2;float q3; } Quaternion; Quaternion NumQ {1, 0, 0, 0};/* 陀螺仪积分误差 */ struct V{float x;float y;float z; }; volatile struct V GyroIntegError {0};/* 四元数解法初始化 */ void imu_rest(void) {NumQ.q0 1;NumQ.q1 0;NumQ.q2 0;NumQ.q3 0; GyroIntegError.x 0;GyroIntegError.y 0;GyroIntegError.z 0;Angle.pitch 0;Angle.roll 0; }void GetAngle(const _st_Mpu *pMpu,_st_AngE *pAngE, float dt) { volatile struct V Gravity,Acc,Gyro,AccGravity;static float KpDef 0.5f ;static float KiDef 0.0001f; //static float KiDef 0.00001f;float q0_t,q1_t,q2_t,q3_t;//float NormAcc; float NormQuat; float HalfTime dt * 0.5f;//提取等效旋转矩阵中的重力分量 Gravity.x 2*(NumQ.q1 * NumQ.q3 - NumQ.q0 * NumQ.q2); Gravity.y 2*(NumQ.q0 * NumQ.q1 NumQ.q2 * NumQ.q3); Gravity.z 1-2*(NumQ.q1 * NumQ.q1 NumQ.q2 * NumQ.q2); // 加速度归一化//printf(accX:%d\r\n,MPU6050.accX);NormAcc 1/sqrt(squa(MPU6050.accX) squa(MPU6050.accY) squa(MPU6050.accZ));//printf(NorAcc%f\r\n,NormAcc);// NormAcc Q_rsqrt(squa(MPU6050.accX) squa(MPU6050.accY) squa(MPU6050.accZ));Acc.x pMpu-accX * NormAcc;Acc.y pMpu-accY * NormAcc;Acc.z pMpu-accZ * NormAcc;//向量差乘得出的值AccGravity.x (Acc.y * Gravity.z - Acc.z * Gravity.y);AccGravity.y (Acc.z * Gravity.x - Acc.x * Gravity.z);AccGravity.z (Acc.x * Gravity.y - Acc.y * Gravity.x);//再做加速度积分补偿角速度的补偿值GyroIntegError.x AccGravity.x * KiDef;GyroIntegError.y AccGravity.y * KiDef;GyroIntegError.z AccGravity.z * KiDef;//角速度融合加速度积分补偿值Gyro.x pMpu-gyroX * Gyro_Gr KpDef * AccGravity.x GyroIntegError.x;//弧度制Gyro.y pMpu-gyroY * Gyro_Gr KpDef * AccGravity.y GyroIntegError.y;Gyro.z pMpu-gyroZ * Gyro_Gr KpDef * AccGravity.z GyroIntegError.z;// 一阶龙格库塔法, 更新四元数q0_t (-NumQ.q1*Gyro.x - NumQ.q2*Gyro.y - NumQ.q3*Gyro.z) * HalfTime;q1_t ( NumQ.q0*Gyro.x - NumQ.q3*Gyro.y NumQ.q2*Gyro.z) * HalfTime;q2_t ( NumQ.q3*Gyro.x NumQ.q0*Gyro.y - NumQ.q1*Gyro.z) * HalfTime;q3_t (-NumQ.q2*Gyro.x NumQ.q1*Gyro.y NumQ.q0*Gyro.z) * HalfTime;NumQ.q0 q0_t;NumQ.q1 q1_t;NumQ.q2 q2_t;NumQ.q3 q3_t;// 四元数归一化NormQuat 1/sqrt(squa(NumQ.q0) squa(NumQ.q1) squa(NumQ.q2) squa(NumQ.q3));NumQ.q0 * NormQuat;NumQ.q1 * NormQuat;NumQ.q2 * NormQuat;NumQ.q3 * NormQuat; // 四元数转欧拉角{#ifdef YAW_GYRO*(float *)pAngE atan2f(2 * NumQ.q1 *NumQ.q2 2 * NumQ.q0 * NumQ.q3, 1 - 2 * NumQ.q2 *NumQ.q2 - 2 * NumQ.q3 * NumQ.q3) * RtA; //yaw#elsefloat yaw_G pMpu-gyroZ * Gyro_G;if((yaw_G 1.0f) || (yaw_G -1.0f)) //数据太小可以认为是干扰不是偏航动作{pAngE-yaw yaw_G * dt;printf(Yaw:%f\r\n,pAngE-yaw);}#endifpAngE-pitch asin(2 * NumQ.q0 *NumQ.q2 - 2 * NumQ.q1 * NumQ.q3) * RtA; pAngE-roll atan2(2 * NumQ.q2 *NumQ.q3 2 * NumQ.q0 * NumQ.q1, 1 - 2 * NumQ.q1 *NumQ.q1 - 2 * NumQ.q2 * NumQ.q2) * RtA; //PITCH printf(Pitch:%f;\r\n,pAngE-pitch);printf(Roll:%f;\r\n,pAngE-roll);} } 4.4 main函数 该阶段下的 main 函数为 while 空循环MPU6050 读取和解算欧拉角都在 TIM1 的中断回调函数中进行代码如下 void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim) {if (htim (htim1)) //如果定时器TIM1中断发生{MpuGetData();GetAngle(MPU6050,Angle,0.003f);Send_ANOTC();} } 五、匿名上位机3D姿态显示 5.1 匿名上位机概述 匿名上位机为匿名科创团队自主设计的知名上位机通常用于与无人机或飞行器进行通信和数据传输。它可以通过串口、蓝牙、Wi-Fi和USB等通信方式与无人机连接并可以实时接收和显示来自无人机的数据如传感器数据、GPS定位信息、飞行控制指令等。此外匿名上位机还可以进行数据存储、分析和可视化3D模型方便用户对无人机数据进行处理和分析作者本次仅为大家提供串口协议下的可视化3D姿态代码。 匿名科创官网welcome [匿名科创] (anotc.com) 5.2 匿名上位机通讯 补充说明匿名科创团队制作的匿名上位机一直在迭代升级不同版本之间的通讯协议和功能使用时可能存在一定差异。作者使用的匿名上位机V5.0版本上位机软件在文末资源包内 匿名科创上位机为使用者提供了详细的通讯协议资料点击左边程序设置图标点击红框所示的通信协议。 匿名上位机提供了很多数据传输协议模板此时飞控项目我们仅考虑飞机姿态的基本信息从WPS的表格中找到如下信息 通过上图可以清晰的发现匿名上位机的通讯是使用数据帧格式的即存在规定格式的数据格式。接下来作者帮大家解析一下STATUS帧下的数据格式飞机姿态信息帧。 为了方便读者朋友对数据格式的解读作者这里将匿名上位机的数据帧分为4部分 第一部分帧头“AAAA”匿名上位机每次接受数据都需要“AAAA”格式的帧头打前阵 第二部分功能字长度部分STATUS帧下的功能字为0x01▲协议中长度字节LEN表示该数据帧内包含数据的字节总长度不包括帧头、功能字、长度字节和最后的校验位只是数据的字节长度和。比如该帧数据内容为3个int16型数据那么LEN等于6一个字节8位算一个长度。故STATUS帧下的数据长度为22241112 第三部分核心数据部分该部分为通讯协议中的核心部分。定义一个数组写入翻滚角roll*100俯仰角pitch*100偏航角yaw*1000为使用气压计0飞行模式0加锁 第四部分SUM即校验位▲SUM等于从该数据帧第一字节开始也就是帧头开始至该帧数据的最后一字节所有字节的和只保留低八位高位舍去。当然校验位SUM并非必须的你可以不写 贴心寄语在实际嵌入式工程项目中对于数据帧的处理是非常常见的。很多时候往往需要工程师自己制定和编写收发数据帧格式作者建议大家可以多磨练这方面的能力。 作者通过串口usart1与匿名上位机进行通讯通讯协议代码如下供大家参考 comm.c代码 #include comm.h #include alldata.h #include usart.h/* ANOTC匿名站通讯机制串口版本 */ void Send_ANOTC() {uint8_t i;uint8_t len0; int16_t Anto[12];int8_t *pt (int8_t*)(Anto); Anto[2] (int16_t)(Angle.roll*100);Anto[3] (int16_t)(Angle.pitch*100);Anto[4] -(int16_t)(Angle.yaw*100);Anto[5] 0;//没有高度数据Anto[6] 0;//飞行模式Anto[7] ALL_flag.unlock8;//解锁信息len 12; //数据长度Anto[0] 0XAAAA; //帧头Anto[1] len | 0x018; //功能字长度pt[len4] (int8_t)(0xAA0xAA);pt[len4] (int8_t)(0xAA0xAA);for(i2;ilen4;i2) //a swap with b;{pt[i] ^ pt[i1];pt[i1] ^ pt[i];pt[i] ^ pt[i1];pt[len4] pt[i] pt[i1];}HAL_UART_Transmit(huart1,pt,len5,0xFFFF); //采用串口发送到匿名上位机 } 5.3 匿名上位机使用 匿名上位机V5.0版本提供了3种类型的通讯方式我们这里选择串口模式波特率设置为115200具体如下图所示 第一步点击右下角打开连接第二步点击飞控状态图标作者没有写数据校验位故右下角警告可能出现警告无视即可 我们仅需要关注ROL、RIT、YAW和飞控状态即可可以看出来匿名上位机的使用是非常简单 六、项目效果 无人机3D姿态显示 七、项目代码 代码地址 基于STM32的四旋翼无人机项目二MPU6050姿态解算含匿名上位机串口通讯版本代码-嵌入式文档类资源-CSDN文库https://download.csdn.net/download/black_sneak/87934739 如果积分不够的朋友点波关注评论区留下邮箱作者无偿提供源码和后续问题解答。求求啦关注一波吧
http://www.w-s-a.com/news/689754/

相关文章:

  • 做空压机网站的公司有哪些wordpress 外部链接
  • 网站建设管理成本估计小红书推广平台
  • 一级a做爰片免费观看网站焦作建设企业网站公司
  • 欧阳网站建设2022华为云营销季
  • 快速学做网站高水平的大连网站建设
  • 专业做房地产网站建设wordpress侧面小工具
  • 旅游网站开发的重要性wordpress添加广告插件
  • 关于网站建设管理工作的报告婚纱网站php
  • 东莞市建设培训中心网站那个网站可以看高速的建设情况
  • 网站开发工具安卓版专业小程序商城开发
  • 网站不备案影响收录吗深圳住房网站app
  • 交网站建设域名计入什么科目开发平台教程
  • 个人网站定制北京快速建站模板
  • 河南海华工程建设监理公司网站高端论坛网站建设
  • 网站建设网络推广方案图片编辑器免费
  • 如何用dw做网站设计设计头条
  • 网站建设基础及流程北京商场购物中心排名
  • 青州市城乡建设局网站自建网站步骤
  • wordpress文章延迟加载优化设计答案四年级上册语文
  • 做网站源码要给客户嘛怎么在运行打开wordpress
  • 北海住房和城乡建设局网站wordpress标题去掉私密
  • 织梦网站安装视频做网站都有那些步骤
  • 网站空间大小选择沈阳微信网站制作
  • 网站分享对联广告网站结构的类型
  • 三维家是在网站上做还是在app上北京网站建设 专业10年
  • 模拟网站建设网站建设认准猪八戒网
  • godaddy网站建设教程微信手机网站搭建
  • 网站建设 商城淄博网站制作哪家好
  • 廊坊手机网站团队国际贸易进口代理公司
  • 运用django做网站网站建设问题及解决办法