三轴磁力计解算姿态(四元数)

原理

根据地磁场向量在水平面上的投影来计算载体的偏航角,类似于加速度计解算姿态,不同在于磁场易受干扰,且只能得到偏航角。

方法

假设导航坐标系为东北天,载体坐标系为右前上。

初始载体坐标系和导航坐标系重合,对应的四元数为q=[1,0,0,0],使用此四元数表示载体在导航坐标系下的旋转

总体步骤参考三轴加速度计解算姿态(四元数),主要根据磁力计测得的磁场向量在水平方向上的投影与磁北向量的夹角来旋转载体。

当载体水平放置时,磁力计磁场向量在水平方向上的投影很好计算,直接取x,y分量即可。但是当载体不处于水平时,需要将测量的向量先转换到导航坐标系,然后再取其在水平面上的投影,这样磁力计解算可以不受载体姿态影响。

以下是Eigen3的示例代码,先根据加速度计的数据得到载体坐标系到导航坐标系的旋转矩阵,然后再对磁力计的数据进行旋转,将其旋转至导航坐标系,最后取水平面上的投影来计算偏航角。

#include "Eigen/Core"
#include "Eigen/Geometry"
#include <cmath>

// 初始四元数
Eigen::Quaternionf quaternion = Eigen::Quaternionf(1.0f, 0.0f, 0.0f, 0.0f);

while(1) {
	// 读取加速度计数据,单位mg
	float acc[3];
	read_acc(acc);
	// 加速度计测量的加速度向量,单位转为g
	Eigen::Vector3f acceleration = Eigen::Vector3f(acc[0] / 1000.0f, acc[1] / 1000.0f, acc[2] / 1000.0f).normalized();
	// 导航坐标系下的标准重力向量
	Eigen::Vector3f gravity = Eigen::Vector3f(0.0f, 0.0f, 1.0f);
	// 测量的加速度从载体系转换到导航系
	acceleration = (quaternion.toRotationMatrix() * acceleration).normalized();
	// 计算旋转轴
	Eigen::Vector3f rotate_vector = acceleration.cross(gravity);
	// 计算旋转角度
	float theta = acos(acceleration.dot(gravity));
	// 得到旋转对应的四元数
	Eigen::Vector3f v = rotate_vector.normalized() * std::sin(theta / 2.0f);
	Eigen::Quaternionf q = Eigen::Quaternionf(std::cos(theta / 2.0f), v.x(), v.y(), v.z()).normalized();
	// 应用旋转
	quaternion = (q * quaternion).normalized();



	// 磁力计数据,任意单位
	float mag[3];
    read_mag(mag);

    // 磁北方向,导航坐标系(东北天)
    Eigen::Vector3f north = Eigen::Vector3f(0, 1, 0);

    // 磁力计测量的磁场向量,载体坐标系(右前上)
    Eigen::Vector3f magnetometer = Eigen::Vector3f(mag[0], mag[1], mag[2]);

    // 转换到导航坐标系
    magnetometer = quaternion.toRotationMatrix() * magnetometer;

    // xy平面投影
    magnetometer.z() = 0;
    magnetometer.normalize();

    // 计算旋转轴
    rotate_vector = magnetometer.cross(north);
    // 计算旋转角度
    theta = acos(magnetometer.dot(north));
    // 获取旋转四元数
    v = rotate_vector.normalized() * std::sin(theta / 2.0f);
    q = Eigen::Quaternionf(std::cos(theta / 2.0f), v.x(), v.y(), v.z()).normalized();
    // 应用旋转
    quaternion = (q * quaternion).normalized();
}
  • 0
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
本文档的主要内容详细介绍的是MPU9250芯片的相关资料合集,主要内容包括了:使用MPU9250芯片进行的运动驱动器程序,英文森运动传感器通用评估委员会(UEVB)用户指南,MPU-9250产品中文说明书,MPU-9250产品英文数据手册,MPU-9250寄存器和说明,叉积法融合陀螺和加速度核心程序详解,姿态解算说明   MPU9250是一-个QFN封装的复合芯片(MCM), 它由2部分组成。一组是3轴加速度还有3轴陀螺仪,另一组则是AKM公司的AK8963 3轴磁力计。所以,MPU9250是一款9轴运动跟踪装置,他在小小的3X3X1mm的封装中融合了3轴加速度,3 轴陀螺仪以及数字运动处理器(DMP) 并且兼容MPU6515。其完美的I2C方案,可直接输出9轴的全部数据。一体化的设,运动性的融合,时钟校准功能,让开发者避开了繁琐复杂的芯片选择和外设成本,保证最佳的性能。本芯片也为兼容其它传感器开放了辅助12C接口,比如连接压力传感器。   MPU9250的具有三个16位加速度AD输出,三个16位陀螺仪AD输出,三个6位磁力计AD输出。精密的慢速和快速运动跟踪,提供给客户全量程的可编程陀螺仪参数选择(土250,土500,士1000, and土2000 。/秒(dps)),可编程的加速度参数选择土2g,土4g,土8g, 土16g, 以及最大磁力计可达到土4800uT。   其他业界领先的功能还有可编程的数字滤波器, 40-85 C时带高精度的1%的时钟漂移,嵌入了温度传感器,并且带有可编程中断。该装置提供12C和SPI的接口,2. 4-3. 6V的供电电压,还有单独的数字10口,支持1.71V到VDD。   通信采用400KHz的12C和1MHz的SPI,若需要更快的速度,可以用SPI在20MHz的RDDLER RIDLER模式下直接读取传感器和中断寄存器。   采用CMOS-MEMS的制作平台,让传感器以低成本的高性能集成在- -个3x3x1mm的芯片内,并且能承受住10, 000g 的震动冲击。 应用领域 无需触碰操作的技术 手势控制 体感游戏控制器 位置查找服务 手机等便携式游戏设备 PS4或XBOX等游戏手柄控制器 3D电视遥控器或机顶盒 3D鼠标 可穿戴的健康智能设备
ICM-20948是一款集成了三轴加速度三轴陀螺仪和三轴地磁的惯性测量单元(IMU)。它可以通过I2C或SPI接口与主控制器通信。 在ICM-20948中,磁力计是由三轴地磁组成的。地磁可以测量地球的磁场,因此可以用于导航和方向控制。 磁力计的读数可以通过I2C或SPI接口读取。在I2C模式下,可以通过以下代码获取磁力计读数: ``` Wire.beginTransmission(0x68); // ICM-20948的I2C地址为0x68 Wire.write(0x49); // 寄存器地址为0x49,对应磁力计X轴低8位 Wire.endTransmission(false); Wire.requestFrom(0x68, 6, true); // 读取磁力计X、Y、Z轴低8位和高8位,共6个字节 int16_t mx = Wire.read() | (Wire.read() << 8); // 拼接低8位和高8位,得到磁力计X轴数值 int16_t my = Wire.read() | (Wire.read() << 8); // 拼接低8位和高8位,得到磁力计Y轴数值 int16_t mz = Wire.read() | (Wire.read() << 8); // 拼接低8位和高8位,得到磁力计Z轴数值 ``` 在SPI模式下,可以通过以下代码获取磁力计读数: ``` digitalWrite(CS_PIN, LOW); // 使能SPI从设备 SPI.transfer(0x49); // 寄存器地址为0x49,对应磁力计X轴低8位 int16_t mx = SPI.transfer(0) | (SPI.transfer(0) << 8); // 拼接低8位和高8位,得到磁力计X轴数值 int16_t my = SPI.transfer(0) | (SPI.transfer(0) << 8); // 拼接低8位和高8位,得到磁力计Y轴数值 int16_t mz = SPI.transfer(0) | (SPI.transfer(0) << 8); // 拼接低8位和高8位,得到磁力计Z轴数值 digitalWrite(CS_PIN, HIGH); // 禁用SPI从设备 ``` 需要注意的是,ICM-20948的磁力计读数需要进行校准才能得到准确的结果。一般的校准方法是在不受干扰的环境下,将IMU在六个方向上旋转并记录磁力计读数,然后通过线性变换得到校准参数。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值