基于52单片机读取MPU6050 3轴加速度和角速度 并且用四元素法进行姿势解算 发现Pitch角和Roll显示不对
Pitch角显示的是ROLL角,roll角显示也不对,请问我的代码有什么问题
// 晶振:11.0592M
// 显示:串口
// 编译环境 Keil uVision2
//****************************************
#include <REG52.H>
#include <math.h> //Keil library
#include <stdio.h> //Keil library
#include <INTRINS.H>
#define Kp 100.0f//比例增益支配率收敛到加速度计/磁强计
#define Ki 0.002f//积分增益支配率的陀螺仪偏见的衔接
#define halfT 0.001f//采样周期的一半
typedef unsigned char uchar;
typedef unsigned short ushort;
typedef unsigned int uint;
float q0=1,q1=0,q2=0,q3=0;//四元数的元素,代表估计方向
float exInt=0,eyInt=0,ezInt=0;//按比例缩小积分误差
float Yaw,Pitch,Roll;
//****************************************
// 定义51单片机端口
//****************************************
sbit SCL=P1^6; //IIC时钟引脚定义
sbit SDA=P1^7; //IIC数据引脚定义
//****************************************
// 定义MPU6050内部地址
//****************************************
#define SMPLRT_DIV 0x19 //陀螺仪采样率,典型值:0x07(125Hz)
#define CONFIG 0x1A //低通滤波频率,典型值:0x06(5Hz)
#define GYRO_CONFIG 0x1B //陀螺仪自检及测量范围,典型值:0x18(不自检,2000deg/s)
#define ACCEL_CONFIG 0x1C //加速计自检、测量范围及高通滤波频率,典型值:0x01(不自检,2G,5Hz)
#define ACCEL_XOUT_H 0x3B
#define ACCEL_XOUT_L 0x3C
#define ACCEL_YOUT_H 0x3D
#define ACCEL_YOUT_L 0x3E
#define ACCEL_ZOUT_H 0x3F
#define ACCEL_ZOUT_L 0x40
#define TEMP_OUT_H 0x41
#define TEMP_OUT_L 0x42
#define GYRO_XOUT_H 0x43
#define GYRO_XOUT_L 0x44
#define GYRO_YOUT_H 0x45
#define GYRO_YOUT_L 0x46
#define GYRO_ZOUT_H 0x47
#define GYRO_ZOUT_L 0x48
#define PWR_MGMT_1 0x6B //电源管理,典型值:0x00(正常启用)
#define WHO_AM_I 0x75 //IIC地址寄存器(默认数值0x68,只读)
#define SlaveAddress 0xD0 //IIC写入时的地址字节数据,+1为读取
//**************************************************************************************************
//定义类型及变量
//**************************************************************************************************
uchar dis[6]; //显示数字(-511至512)的字符数组
//int dis_data; //变量
float dis_data;
//**************************************************************************************************
//函数声明
//**************************************************************************************************
void Delay5us();
void delay(unsigned int k); //延时
void lcd_printf(uchar *s,int temp_data);
//********************************MPU6050操作函数***************************************************
void InitMPU6050(); //初始化MPU6050
void I2C_Start();
void I2C_Stop();
void I2C_SendACK(bit ack);
bit I2C_RecvACK();
void I2C_SendByte(uchar dat);
uchar I2C_RecvByte();
void I2C_ReadPage();
void I2C_WritePage();
void display_ACCEL_x();
void display_ACCEL_y();
void display_ACCEL_z();
uchar Single_ReadI2C(uchar REG_Address); //读取I2C数据
void Single_WriteI2C(uchar REG_Address,uchar REG_data); //向I2C写入数据
//********************************************************************************
//整数转字符串
//********************************************************************************
void lcd_printf(uchar *s,int temp_data) //
{
if(temp_data<0)
{
temp_data=-temp_data;
*s='-';
}
else *s=' ';
*++s =temp_data/10000+0x30;
temp_data=temp_data%10000; //取余运算
*++s =temp_data/1000+0x30;
temp_data=temp_data%1000; //取余运算
*++s =temp_data/100+0x30;
temp_data=temp_data%100; //取余运算
*++s =temp_data/10+0x30;
temp_data=temp_data%10; //取余运算
*++s =temp_data+0x30;
}
//******************************************************************************************************
//串口初始化
//*******************************************************************************************************
void init_uart()
{
// TMOD=0x21;
TMOD=0x20; //定时器1工作于8位自动重载模式, 用于产生波特率
// TH1=0xf3; //实现波特率4800(系统时钟12MHZ)
// TL1=0xf3;
TL1=0xfd;
TH1=0xfd; //波特率9600
// PCON=0X80;
// SCON=0x50;
SCON=0x50; //设定串口工作方式
PCON=0x00; //波特率不倍增
PS=1; //串口中断设为高优先级别
TR0=1; //启动定时器
TR1=1;
ET0=1; //打开定时器0中断
ES=1;
EA=1;
}
//*************************************************************************************************
//串口发送函数
//*************************************************************************************************
void SeriPushSend(uchar send_data)
{
SBUF=send_data;
while(!TI);TI=0;
}
//*************************************************************************************************
//************************************延时*********************************************************
//*************************************************************************************************
void delay(unsigned int k)
{
unsigned int i,j;
for(i=0;i<k;i++)
{
for(j=0;j<121;j++);
}
}
//************************************************************************************************
//延时5微秒(STC90C52RC@12M)
//不同的工作环境,需要调整此函数
//注意当改用1T的MCU时,请调整此延时函数
//************************************************************************************************
void Delay5us()
{
_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();
_nop_();_nop_();_nop_();_nop_();
}
//*************************************************************************************************
//I2C起始信号
//*************************************************************************************************
void I2C_Start()
{
SDA=1; //拉高数据线
SCL=1; //拉高时钟线
Delay5us(); //延时
SDA=0; //产生下降沿
Delay5us(); //延时
SCL=0; //拉低时钟线
}
//*************************************************************************************************
//I2C停止信号
//*************************************************************************************************
void I2C_Stop()
{
SDA=0; //拉低数据线
SCL=1; //拉高时钟线
Delay5us(); //延时
SDA=1; //产生上升沿
Delay5us(); //延时
}
//**************************************************************************************************
//I2C发送应答信号
//入口参数:ack (0:ACK 1:NAK)
//**************************************************************************************************
void I2C_SendACK(bit ack)
{
SDA=ack; //写应答信号
SCL=1; //拉高时钟线
Delay5us(); //延时
SCL=0; //拉低时钟线
Delay5us(); //延时
}
//****************************************************************************************************
//I2C接收应答信号
//****************************************************************************************************
bit I2C_RecvACK()
{
SCL=1; //拉高时钟线
Delay5us(); //延时
CY=SDA; //读应答信号
SCL=0; //拉低时钟线
Delay5us(); //延时
return CY;
}
//*****************************************************************************************************
//向I2C总线发送一个字节数据
//*****************************************************************************************************
void I2C_SendByte(uchar dat)
{
uchar i;
for (i=0; i<8; i++) //8位计数器
{
dat<<=1; //移出数据的最高位
SDA=CY; //送数据口
SCL=1; //拉高时钟线
Delay5us(); //延时
SCL=0; //拉低时钟线
Delay5us(); //延时
}
I2C_RecvACK();
}
//*****************************************************************************************************
//从I2C总线接收一个字节数据
//******************************************************************************************************
uchar I2C_RecvByte()
{
uchar i;
uchar dat=0;
SDA=1; //使能内部上拉,准备读取数据,
for (i=0; i<8; i++) //8位计数器
{
dat<<=1;
SCL=1; //拉高时钟线
Delay5us(); //延时
dat|=SDA; //读数据
SCL=0; //拉低时钟线
Delay5us(); //延时
}
return dat;
}
//*****************************************************************************************************
//向I2C设备写入一个字节数据
//*****************************************************************************************************
void Single_WriteI2C(uchar REG_Address,uchar REG_data)
{
I2C_Start(); //起始信号
I2C_SendByte(SlaveAddress); //发送设备地址+写信号
I2C_SendByte(REG_Address); //内部寄存器地址,
I2C_SendByte(REG_data); //内部寄存器数据,
I2C_Stop(); //发送停止信号
}
//*******************************************************************************************************
//从I2C设备读取一个字节数据
//*******************************************************************************************************
uchar Single_ReadI2C(uchar REG_Address)
{
uchar REG_data;
I2C_Start(); //起始信号
I2C_SendByte(SlaveAddress); //发送设备地址+写信号
I2C_SendByte(REG_Address); //发送存储单元地址,从0开始
I2C_Start(); //起始信号
I2C_SendByte(SlaveAddress+1); //发送设备地址+读信号
REG_data=I2C_RecvByte(); //读出寄存器数据
I2C_SendACK(1); //接收应答信号
I2C_Stop(); //停止信号
return REG_data;
}
//******************************************************************************************************
//初始化MPU6050
//******************************************************************************************************
void InitMPU6050()
{
Single_WriteI2C(PWR_MGMT_1,0x00); //解除休眠状态
Single_WriteI2C(SMPLRT_DIV,0x07);
Single_WriteI2C(CONFIG,0x06);
Single_WriteI2C(GYRO_CONFIG,0x18);
Single_WriteI2C(ACCEL_CONFIG,0x01);
}
//******************************************************************************************************
//合成数据
//******************************************************************************************************
int GetData(uchar REG_Address)
{
uchar H,L;
H=Single_ReadI2C(REG_Address);
L=Single_ReadI2C(REG_Address+1);
return ((H<<8)+L); //合成数据
}
//******************************************************************************************************
//超级终端(串口调试助手)上显示10位数据
//******************************************************************************************************
void Display10BitData(int value)
{
uchar i;
// value/=64; //转换为10位数据
lcd_printf(dis,value); //转换数据显示
for(i=0;i<6;i++)
{
SeriPushSend(dis[i]);
}
// DisplayListChar(x,y,dis,4); //启始列,行,显示数组,显示长度
}
void IMUupdate()
{
float norm;
float vx,vy,vz;
float ex,ey,ez;
float ax,ay,az;
float gx,gy,gz;
ax=GetData(ACCEL_XOUT_H);
ay=GetData(ACCEL_YOUT_H);
az=GetData(ACCEL_ZOUT_H);
gx=GetData(GYRO_XOUT_H);
gy=GetData(GYRO_YOUT_H);
gz=GetData(GYRO_ZOUT_H);
if(ax*ay*az==0)
return;
norm=sqrt(ax*ax+ay*ay+az*az);
ax=ax/norm;
ay=ay/norm;
az=az/norm;
vx=2*(q1*q3-q0*q2);
vy=2*(q0*q1+q2*q3);
vz=q0*q0-q1*q1-q2*q2+q3*q3;
ex=(ay*vz-az*vy);
ey=(az*vx-ax*vz);
ez=(ax*vy-ay*vx);
exInt=exInt+ex*Ki;
eyInt=eyInt+ey*Ki;
ezInt=ezInt+ez*Ki;
gx=gx+Kp*ex+exInt;
gy=gy+Kp*ey+eyInt;
gz=gz+Kp*ez+ezInt;
q0=q0+(-q1*gx-q2*gy-q3*gz)*halfT;
q1=q1+(q0*gx+q2*gz-q3*gy)*halfT;
q2=q2+(q0*gy-q1*gz+q3*gx)*halfT;
q3=q3+(q0*gz+q1*gy-q2*gx)*halfT;
norm=sqrt(q0*q0+q1*q1+q2*q2+q3*q3);
q0=q0/norm;
q1=q1/norm;
q2=q2/norm;
q3=q3/norm;
Yaw=atan2(2*(q1*q2+q0*q3),q0*q0+q1*q1-q2*q2-q3*q3)*57.3;
Pitch=asin(-2*q1*q3+2*q0*q2)*57.3;
Roll=atan2(2*q2*q3+2*q0*q1,-2*q1*q1-2*q2*q2+1)*57.3;
// Yaw=atan2(2*q1*q2+2*q0*q3,-2*q2*q2-2*q3*q3+1)*57.3;//yaw
// Pitch=asin(-2*q1*q3+2*q0*q2)*57.3;//pitch
// Roll=atan2(2*q2*q3+2*q0*q1,-2*q1*q1-2*q2*q2+1)*57.3;//roll
}
//*******************************************************************************************************
//主程序
//*******************************************************************************************************
void main()
{
delay(500); //上电延时
init_uart();
InitMPU6050();//初始化MPU6050
delay(150);
while(1)
{
IMUupdate();
SeriPushSend('Y');SeriPushSend(':');Display10BitData(Yaw);SeriPushSend(0x20);SeriPushSend(0x20);
SeriPushSend('P');SeriPushSend(':');Display10BitData(Pitch);SeriPushSend(0x20);SeriPushSend(0x20);
SeriPushSend('R');SeriPushSend(':');Display10BitData(Roll);SeriPushSend(0x20);SeriPushSend(0x20);
SeriPushSend(0x0d);
SeriPushSend(0x0a);//换行,回车
delay(2000);
}
关注你的MPU6050的放置方向,结合电路板的方向,对比芯片手册中的基准方向。应该由放置方向引起的坐标系问题。