大致流程:首先用 RT-Thread 的 icm20608 軟件包讀取 陀螺儀 (Gyroscope) 和 加速度計(jì) (Accelerometer) 的數(shù)據(jù),分別計(jì)算出估計(jì)的角度,再用互補(bǔ)濾波器 (Complementary Filter) 融合兩個角度估計(jì)、進(jìn)行校正,其實(shí)核心算法的代碼就 7 行。最后串口把數(shù)據(jù)發(fā)到電腦上,用 Python + OpenGL 可視化。
Github - STM32 IMU 互補(bǔ)濾波器 (RT-Thread):https://github.com/wuhanstudio/stm32-imu-filter
IMU 傳感器 (Inertial Measurement Unit)
我們先介紹下從 I2C 總線讀取出傳感器原始數(shù)值后,如何處理得到加速度和旋轉(zhuǎn)角速度。一個六軸的 IMU 可以測量 x, y, z 三個方向的重力加速度,和繞三個軸的旋轉(zhuǎn)角速度。比如,開發(fā)板如果靜止放置在桌面上,會測量到 z 方向的重力加速度。
當(dāng)然,如果開發(fā)板靜止不動,繞三個軸的旋轉(zhuǎn)速度都是 0。
由于傳感器的輸出實(shí)際上是來自 ADC 的 16 位數(shù)字信號,我們需要把它的單位轉(zhuǎn)換成重力加速度 g。例如,我們可以選擇測量范圍
,默認(rèn)是
,也就是把傳感器的 16 位輸出
映射到 [-2g, 2g),于是
也就是下面 icm20608 芯片手冊的 Sensitivity Scale Factor。

于是在代碼里面,將原始的 int16 加速度數(shù)據(jù)除以 16384。
double aSensitivity = 16384;
accel_x = accel_x / aSensitivity;
accel_y = accel_y / aSensitivity;
accel_z = accel_z / aSensitivity;
同樣,我們可以換算出角速度

于是在代碼里面,將原始的 int16 角速度數(shù)據(jù)除以 131。
double gSensitivity = 131;
gyrX = gyro_x / gSensitivity;
gyrY = gyro_y / gSensitivity;
gyrZ = gyro_z / gSensitivity;
這樣我們就把 ADC 輸出的 int16 原始數(shù)據(jù)分布轉(zhuǎn)換成了加速度單位 g,和旋轉(zhuǎn)角速度單位 °/s.
互補(bǔ)濾波器 (Complementary Filter)
我們可以用 互補(bǔ)濾波器 結(jié)合 加速度 和 旋轉(zhuǎn)速度 的測量值,得到更準(zhǔn)確的姿態(tài)預(yù)測。
我們使用下面的圖中的坐標(biāo)系,繞 x 軸旋轉(zhuǎn)的角度為 roll,繞 y 軸的旋轉(zhuǎn)方向?yàn)?pitch,繞 z 軸旋轉(zhuǎn)方向?yàn)?yaw。逆時針旋轉(zhuǎn)為正,順時針旋轉(zhuǎn)為負(fù)。
陀螺儀估計(jì)姿態(tài)
陀螺儀測量的是瞬間的旋轉(zhuǎn)角速度,所以位置的估計(jì)其實(shí)就是時間的積分。例如,每過 100ms 測量一次旋轉(zhuǎn)速度,旋轉(zhuǎn)速度 x 時間 = 旋轉(zhuǎn)角度。
// angles based on gyro (deg/s)
gx = gx + gyrX * TIME_STEP_MS / 1000;
gy = gy + gyrY * TIME_STEP_MS / 1000;
gz = gz + gyrZ * TIME_STEP_MS / 1000;
當(dāng)然,由于環(huán)境存在大量噪聲,陀螺儀測量數(shù)據(jù)會存在隨機(jī)的波動,這些噪聲經(jīng)過積分累積,最后會造成位置的漂移。比如下面這張圖,過了很長時間后,雖然開發(fā)板是靜止的,但是右邊的陀螺儀估計(jì)的位置,就無法回到原點(diǎn),這就是長時間的累計(jì)誤差造成的。
加速度計(jì)估計(jì)姿態(tài)
加速度計(jì)不需要積分,我們可以直接對當(dāng)前加速度角度求 arctan 得到角度:

// angles based on accelerometer
ax = atan2(accelY, accelZ) * 180 / M_PI; // roll
ay = atan2(-accelX, sqrt( pow(accelY, 2) + pow(accelZ, 2))) * 180 / M_PI; // pitch
不管我們的開發(fā)板繞 z 軸旋轉(zhuǎn)多少度,重力加速度始終朝向地面。因此開發(fā)板靜止?fàn)顟B(tài),我們無法利用重力加速度知道 z 軸的旋轉(zhuǎn)角度 (yaw),所以上面只計(jì)算 roll 和 pitch,最終 z 軸的旋轉(zhuǎn)角度 yaw 會出現(xiàn)累計(jì)積分誤差。
互補(bǔ)濾波器
我們需要結(jié)合2個測量值是因?yàn)椋盒D(zhuǎn)速度短時間內(nèi)比較準(zhǔn)確,但是由于環(huán)境的噪聲會產(chǎn)生一些隨機(jī)運(yùn)動,時間長了就會漂移,而加速度短時間內(nèi)不一定準(zhǔn)確,但是最終會維持穩(wěn)定。
于是我們就可以取長補(bǔ)短,線性疊加2個測量值的估計(jì),給出更準(zhǔn)確的估計(jì)。
// complementary filter
gx = gx * 0.96 + ax * 0.04;
gy = gy * 0.96 + ay * 0.04;
短時間內(nèi),我們相信陀螺儀測量的旋轉(zhuǎn)角速度 (權(quán)值: 0.96);長時間內(nèi),環(huán)境噪聲逐漸造成的漂移,由加速度計(jì)慢慢進(jìn)行矯正 (權(quán)值: 0.04)。
總結(jié)
最后總結(jié)一下,其實(shí)核心代碼一共就 7 行。我們先利用加速度求解姿態(tài),再利用旋轉(zhuǎn)角速度求解姿態(tài),最后用互補(bǔ)濾波器進(jìn)行一個線性疊加。
// angles based on gyro (deg/s)
gx = gx + gyrX * TIME_STEP_MS / 1000;
gy = gy + gyrY * TIME_STEP_MS / 1000;
gz = gz + gyrZ * TIME_STEP_MS / 1000;
// angles based on accelerometer
ax = atan2(accelY, accelZ) * 180 / M_PI; // roll
ay = atan2(-accelX, sqrt( pow(accelY, 2) + pow(accelZ, 2))) * 180 / M_PI; // pitch
// complementary filter
gx = gx * 0.96 + ax * 0.04;
gy = gy * 0.96 + ay * 0.04;
References
-
https://github.com/mattzzw/Arduino-mpu6050
-
https://github.com/RT-Thread-pa
點(diǎn)擊閱讀原文查看近期賽事
原文標(biāo)題:RT-Thread 互補(bǔ)濾波器 (STM32 + 6 軸 IMU)
文章出處:【微信公眾號:RTThread物聯(lián)網(wǎng)操作系統(tǒng)】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
-
RT-Thread
+關(guān)注
關(guān)注
32文章
1370瀏覽量
41509
原文標(biāo)題:RT-Thread 互補(bǔ)濾波器 (STM32 + 6 軸 IMU)
文章出處:【微信號:RTThread,微信公眾號:RTThread物聯(lián)網(wǎng)操作系統(tǒng)】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
2025 RT-Thread全球技術(shù)大會議程正式發(fā)布!

RT-Thread 上如何實(shí)現(xiàn) SLAAC?
RT-Thread審核團(tuán)招募: 深度參與開源RTOS社區(qū)治理與演進(jìn)

如何將RT-Thread移植到NXP MCUXPressoIDE上

從開源中來到開源中去-RT-Thread北京社區(qū)篇

RT-Thread上CAN實(shí)踐

開源共生 商業(yè)共贏 | RT-Thread 2024開發(fā)者大會報名啟動!

【成都】9月21日RT-Thread巡回線下培訓(xùn)-OpenMV機(jī)器視覺

【大連】9月7日RT-Thread巡回線下培訓(xùn)-OpenMV機(jī)器視覺

2024 RT-Thread全球巡回 線下培訓(xùn)火熱來襲!

【好書推薦】RT-Thread設(shè)備驅(qū)動開發(fā)指南

RT-Thread 新里程碑達(dá)成——GitHub Star 破萬!

基于 RT-Thread專業(yè)版的EtherCAT主站方案

杭州站RT-Thread混合部署Workshop圓滿落幕!

開發(fā)者分享 | 基于先楫RT-Thread BSP,使用CLion開發(fā)應(yīng)用

評論