ArduPilot姿态控制:从抖动发散到稳如磐石的实战解剖
你有没有遇到过这样的场景?
刚调完ATC_RAT_PIT_P,飞机悬停时横滚轴开始低频“呼吸式”晃动;
把ATC_RAT_YAW_D加大一点想压住偏航振荡,结果一打方向就“甩头”;
GPS信号弱时定点突然漂移,地面站显示EKF健康度掉到0,但飞机居然没炸——它只是悄悄切到了另一套姿态解算逻辑……
这些不是玄学,也不是飞控“不听话”,而是ArduPilot姿态控制在真实嵌入式约束下,与物理世界激烈博弈留下的可读痕迹。它不藏在Mission Planner的滑块背后,而藏在AC_AttitudeControl::rate_bf_to_accel()那几十行C++里,在Quaternion::integrate()的四阶龙格-库塔迭代中,在AP_AHRS_NavEKF::check_healthy()返回false的一瞬间。
本文不讲“什么是PID”,也不堆砌公式推导。我们直接钻进v4.4主干代码的血管里,看姿态控制如何在一帧5ms的IMU中断中完成一次生死判断——从传感器原始采样、坐标系撕裂与缝合、误差量化、增益动态缩放,到最终PWM脉冲落进电调引脚。这不是理论复述,而是一份带血丝的手术记录。
三环不是串联,是分层责任制
很多初学者误以为ArduPilot的“外环→中环→内环”是教科书式的线性级联:角度误差进P,输出角速率;角速率误差进PI,输出倾角;倾角再进混控……
错。这三环之间没有固定的数据管道,它们共享同一套状态,但各自对“失控”的定义完全不同。
外环(角度环)真正干的活,是“定调”:它不关心你现在转得多快,只问“你离目标姿态还差多少?”
它用的是四元数误差的矢量部分(q_err.q1,q2,q3),经P增益后直接映射为期望角速率_des_rate。注意:这里没有I项——因为角度误差的积分会直接导致“越调越远”,ArduPilot把它交给中环去消化。中环(角速率环)才是真正的“执行法官”:它盯着陀螺仪读出的实时角速率
gyro.x/y/z,和外环给的_des_rate做减法,得到角速率误差。这个误差走的是带抗饱和的PI控制器:cpp // libraries/AC_PID/AC_PID.cpp: L276 float AC_PID::get_i() { if (_flags._i_enabled && _integrator < _imax && _integrator > -_imax) { _integrator += _ki * error * dt; } return _integrator