vtx: fix VTX_SETTINGS_POWER_COUNT and add dummy entries to saPowerNames
[inav.git] / src / main / flight / servos.c
blob40711706e78ddf2f58e802be157ad15ff18841a5
1 /*
2 * This file is part of Cleanflight.
4 * Cleanflight is free software: you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation, either version 3 of the License, or
7 * (at your option) any later version.
9 * Cleanflight is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with Cleanflight. If not, see <http://www.gnu.org/licenses/>.
18 #include <stdbool.h>
19 #include <stdint.h>
20 #include <stdlib.h>
21 #include <string.h>
23 #include "platform.h"
25 #include "build/debug.h"
26 #include "build/build_config.h"
28 #include "common/axis.h"
29 #include "common/filter.h"
30 #include "common/maths.h"
31 #include "programming/global_variables.h"
33 #include "config/config_reset.h"
34 #include "config/feature.h"
35 #include "config/parameter_group.h"
36 #include "config/parameter_group_ids.h"
38 #include "drivers/pwm_output.h"
39 #include "drivers/pwm_mapping.h"
40 #include "drivers/time.h"
41 #include "drivers/gimbal_common.h"
42 #include "drivers/headtracker_common.h"
44 #include "fc/config.h"
45 #include "fc/fc_core.h"
46 #include "fc/rc_controls.h"
47 #include "fc/rc_modes.h"
48 #include "fc/runtime_config.h"
49 #include "fc/controlrate_profile.h"
50 #include "fc/settings.h"
52 #include "flight/imu.h"
53 #include "flight/mixer.h"
54 #include "flight/pid.h"
55 #include "flight/servos.h"
57 #include "io/gps.h"
59 #include "rx/rx.h"
61 #include "sensors/gyro.h"
63 PG_REGISTER_WITH_RESET_TEMPLATE(servoConfig_t, servoConfig, PG_SERVO_CONFIG, 3);
65 PG_RESET_TEMPLATE(servoConfig_t, servoConfig,
66 .servoCenterPulse = SETTING_SERVO_CENTER_PULSE_DEFAULT,
67 .servoPwmRate = SETTING_SERVO_PWM_RATE_DEFAULT, // Default for analog servos
68 .servo_lowpass_freq = SETTING_SERVO_LPF_HZ_DEFAULT, // Default servo update rate is 50Hz, everything above Nyquist frequency (25Hz) is going to fold and cause distortions
69 .servo_protocol = SETTING_SERVO_PROTOCOL_DEFAULT,
70 .flaperon_throw_offset = SETTING_FLAPERON_THROW_OFFSET_DEFAULT,
71 .tri_unarmed_servo = SETTING_TRI_UNARMED_SERVO_DEFAULT,
72 .servo_autotrim_rotation_limit = SETTING_SERVO_AUTOTRIM_ROTATION_LIMIT_DEFAULT
76 void Reset_servoMixers(servoMixer_t *instance)
78 for (int i = 0; i < MAX_SERVO_RULES; i++){
79 RESET_CONFIG(servoMixer_t, &instance[i],
80 .targetChannel = 0,
81 .inputSource = 0,
82 .rate = 0,
83 .speed = 0
84 #ifdef USE_PROGRAMMING_FRAMEWORK
85 ,.conditionId = -1
86 #endif
91 PG_REGISTER_ARRAY_WITH_RESET_FN(servoParam_t, MAX_SUPPORTED_SERVOS, servoParams, PG_SERVO_PARAMS, 3);
93 void pgResetFn_servoParams(servoParam_t *instance)
95 for (int i = 0; i < MAX_SUPPORTED_SERVOS; i++) {
96 RESET_CONFIG(servoParam_t, &instance[i],
97 .min = DEFAULT_SERVO_MIN,
98 .max = DEFAULT_SERVO_MAX,
99 .middle = DEFAULT_SERVO_MIDDLE,
100 .rate = 100
105 int16_t servo[MAX_SUPPORTED_SERVOS];
107 static uint8_t servoRuleCount = 0;
108 static servoMixer_t currentServoMixer[MAX_SERVO_RULES];
111 //Was used to keep track of servo rules in all mixer_profile, In order to Apply mixer speed limit when rules turn off
112 static servoMixer_t currentServoMixer[MAX_SERVO_RULES*MAX_MIXER_PROFILE_COUNT];
113 static bool currentServoMixerActivative[MAX_SERVO_RULES*MAX_MIXER_PROFILE_COUNT]; // if true, the rule is used by current servo mixer
116 static bool servoOutputEnabled;
118 static bool mixerUsesServos;
119 static uint8_t minServoIndex;
120 static uint8_t maxServoIndex;
122 static biquadFilter_t servoFilter[MAX_SUPPORTED_SERVOS];
123 static bool servoFilterIsSet;
125 static servoMetadata_t servoMetadata[MAX_SUPPORTED_SERVOS];
126 static rateLimitFilter_t servoSpeedLimitFilter[MAX_SERVO_RULES];
128 STATIC_FASTRAM pt1Filter_t rotRateFilter;
129 STATIC_FASTRAM pt1Filter_t targetRateFilter;
131 int16_t getFlaperonDirection(uint8_t servoPin)
133 if (servoPin == SERVO_FLAPPERON_2) {
134 return -1;
135 } else {
136 return 1;
141 * Compute scaling factor for upper and lower servo throw
143 void servoComputeScalingFactors(uint8_t servoIndex) {
144 servoMetadata[servoIndex].scaleMax = (servoParams(servoIndex)->max - servoParams(servoIndex)->middle) / 500.0f;
145 servoMetadata[servoIndex].scaleMin = (servoParams(servoIndex)->middle - servoParams(servoIndex)->min) / 500.0f;
148 void computeServoCount(void)
150 static bool firstRun = true;
151 if (!firstRun) {
152 return;
154 minServoIndex = 255;
155 maxServoIndex = 0;
156 for (int j = 0; j < MAX_MIXER_PROFILE_COUNT; j++) {
157 for (int i = 0; i < MAX_SERVO_RULES; i++) {
158 // check if done
159 if (mixerServoMixersByIndex(j)[i].rate == 0){
160 break;
162 if (mixerServoMixersByIndex(j)[i].targetChannel < minServoIndex) {
163 minServoIndex = mixerServoMixersByIndex(j)[i].targetChannel;
166 if (mixerServoMixersByIndex(j)[i].targetChannel > maxServoIndex) {
167 maxServoIndex = mixerServoMixersByIndex(j)[i].targetChannel;
169 mixerUsesServos = true;
172 firstRun = false;
175 void servosInit(void)
177 // give all servos a default command
178 for (int i = 0; i < MAX_SUPPORTED_SERVOS; i++) {
179 servo[i] = servoParams(i)->middle;
183 * load mixer
185 computeServoCount();
186 loadCustomServoMixer();
188 // If there are servo rules after all, update variables
189 if (mixerUsesServos) {
190 servoOutputEnabled = true;
193 for (uint8_t i = 0; i < MAX_SUPPORTED_SERVOS; i++) {
194 servoComputeScalingFactors(i);
198 int getServoCount(void)
200 if (mixerUsesServos) {
201 return 1 + maxServoIndex - minServoIndex;
203 else {
204 return 0;
208 void loadCustomServoMixer(void)
210 servoRuleCount = 0;
211 memset(currentServoMixer, 0, sizeof(currentServoMixer));
213 // load custom mixer into currentServoMixer
214 for (int i = 0; i < MAX_SERVO_RULES; i++) {
215 // check if done
216 if (customServoMixers(i)->rate == 0){
217 break;
219 currentServoMixer[servoRuleCount] = *customServoMixers(i);
220 servoSpeedLimitFilter[servoRuleCount].state = 0;
221 servoRuleCount++;
225 static void filterServos(void)
227 if (servoConfig()->servo_lowpass_freq) {
228 // Initialize servo lowpass filter (servos are calculated at looptime rate)
229 if (!servoFilterIsSet) {
230 for (int i = 0; i < MAX_SUPPORTED_SERVOS; i++) {
231 biquadFilterInitLPF(&servoFilter[i], servoConfig()->servo_lowpass_freq, getLooptime());
232 biquadFilterReset(&servoFilter[i], servo[i]);
234 servoFilterIsSet = true;
237 for (int i = 0; i < MAX_SUPPORTED_SERVOS; i++) {
238 // Apply servo lowpass filter and do sanity cheching
239 servo[i] = (int16_t)lrintf(biquadFilterApply(&servoFilter[i], (float)servo[i]));
243 for (int i = 0; i < MAX_SUPPORTED_SERVOS; i++) {
244 servo[i] = constrain(servo[i], servoParams(i)->min, servoParams(i)->max);
248 void writeServos(void)
250 filterServos();
252 #if !defined(SITL_BUILD)
253 int servoIndex = 0;
254 bool zeroServoValue = false;
257 * in case of tricopters, there might me a need to zero servo output when unarmed
259 if (currentMixerConfig.platformType == PLATFORM_TRICOPTER && !ARMING_FLAG(ARMED) && !servoConfig()->tri_unarmed_servo) {
260 zeroServoValue = true;
263 for (int i = minServoIndex; i <= maxServoIndex; i++) {
264 if (zeroServoValue) {
265 pwmWriteServo(servoIndex++, 0);
266 } else {
267 pwmWriteServo(servoIndex++, servo[i]);
270 #endif
273 void servoMixer(float dT)
275 int16_t input[INPUT_SOURCE_COUNT]; // Range [-500:+500]
277 if (FLIGHT_MODE(MANUAL_MODE)) {
278 input[INPUT_STABILIZED_ROLL] = rcCommand[ROLL];
279 input[INPUT_STABILIZED_PITCH] = rcCommand[PITCH];
280 input[INPUT_STABILIZED_YAW] = rcCommand[YAW];
281 } else {
282 // Assisted modes (gyro only or gyro+acc according to AUX configuration in Gui
283 input[INPUT_STABILIZED_ROLL] = axisPID[ROLL];
284 input[INPUT_STABILIZED_PITCH] = axisPID[PITCH];
285 input[INPUT_STABILIZED_YAW] = axisPID[YAW];
287 // Reverse yaw servo when inverted in 3D mode only for multirotor and tricopter
288 if (feature(FEATURE_REVERSIBLE_MOTORS) && (rxGetChannelValue(THROTTLE) < PWM_RANGE_MIDDLE) &&
289 (currentMixerConfig.platformType == PLATFORM_MULTIROTOR || currentMixerConfig.platformType == PLATFORM_TRICOPTER)) {
290 input[INPUT_STABILIZED_YAW] *= -1;
294 input[INPUT_STABILIZED_ROLL_PLUS] = constrain(input[INPUT_STABILIZED_ROLL], 0, 1000);
295 input[INPUT_STABILIZED_ROLL_MINUS] = constrain(input[INPUT_STABILIZED_ROLL], -1000, 0);
296 input[INPUT_STABILIZED_PITCH_PLUS] = constrain(input[INPUT_STABILIZED_PITCH], 0, 1000);
297 input[INPUT_STABILIZED_PITCH_MINUS] = constrain(input[INPUT_STABILIZED_PITCH], -1000, 0);
298 input[INPUT_STABILIZED_YAW_PLUS] = constrain(input[INPUT_STABILIZED_YAW], 0, 1000);
299 input[INPUT_STABILIZED_YAW_MINUS] = constrain(input[INPUT_STABILIZED_YAW], -1000, 0);
301 input[INPUT_FEATURE_FLAPS] = FLIGHT_MODE(FLAPERON) ? servoConfig()->flaperon_throw_offset : 0;
303 input[INPUT_MAX] = 500;
304 #ifdef USE_PROGRAMMING_FRAMEWORK
305 input[INPUT_GVAR_0] = constrain(gvGet(0), -1000, 1000);
306 input[INPUT_GVAR_1] = constrain(gvGet(1), -1000, 1000);
307 input[INPUT_GVAR_2] = constrain(gvGet(2), -1000, 1000);
308 input[INPUT_GVAR_3] = constrain(gvGet(3), -1000, 1000);
309 input[INPUT_GVAR_4] = constrain(gvGet(4), -1000, 1000);
310 input[INPUT_GVAR_5] = constrain(gvGet(5), -1000, 1000);
311 input[INPUT_GVAR_6] = constrain(gvGet(6), -1000, 1000);
312 input[INPUT_GVAR_7] = constrain(gvGet(7), -1000, 1000);
313 #endif
315 if (IS_RC_MODE_ACTIVE(BOXCAMSTAB)) {
316 input[INPUT_GIMBAL_PITCH] = scaleRange(attitude.values.pitch, -900, 900, -500, +500);
317 input[INPUT_GIMBAL_ROLL] = scaleRange(attitude.values.roll, -1800, 1800, -500, +500);
318 } else {
319 input[INPUT_GIMBAL_PITCH] = 0;
320 input[INPUT_GIMBAL_ROLL] = 0;
323 input[INPUT_STABILIZED_THROTTLE] = mixerThrottleCommand - 1000 - 500; // Since it derives from rcCommand or mincommand and must be [-500:+500]
325 input[INPUT_MIXER_TRANSITION] = isMixerTransitionMixing * 500; //fixed value
327 // center the RC input value around the RC middle value
328 // by subtracting the RC middle value from the RC input value, we get:
329 // data - middle = input
330 // 2000 - 1500 = +500
331 // 1500 - 1500 = 0
332 // 1000 - 1500 = -500
333 #define GET_RX_CHANNEL_INPUT(x) (rxGetChannelValue(x) - PWM_RANGE_MIDDLE)
334 input[INPUT_RC_ROLL] = GET_RX_CHANNEL_INPUT(ROLL);
335 input[INPUT_RC_PITCH] = GET_RX_CHANNEL_INPUT(PITCH);
336 input[INPUT_RC_YAW] = GET_RX_CHANNEL_INPUT(YAW);
337 input[INPUT_RC_THROTTLE] = GET_RX_CHANNEL_INPUT(THROTTLE);
338 input[INPUT_RC_CH5] = GET_RX_CHANNEL_INPUT(AUX1);
339 input[INPUT_RC_CH6] = GET_RX_CHANNEL_INPUT(AUX2);
340 input[INPUT_RC_CH7] = GET_RX_CHANNEL_INPUT(AUX3);
341 input[INPUT_RC_CH8] = GET_RX_CHANNEL_INPUT(AUX4);
342 input[INPUT_RC_CH9] = GET_RX_CHANNEL_INPUT(AUX5);
343 input[INPUT_RC_CH10] = GET_RX_CHANNEL_INPUT(AUX6);
344 input[INPUT_RC_CH11] = GET_RX_CHANNEL_INPUT(AUX7);
345 input[INPUT_RC_CH12] = GET_RX_CHANNEL_INPUT(AUX8);
346 input[INPUT_RC_CH13] = GET_RX_CHANNEL_INPUT(AUX9);
347 input[INPUT_RC_CH14] = GET_RX_CHANNEL_INPUT(AUX10);
348 input[INPUT_RC_CH15] = GET_RX_CHANNEL_INPUT(AUX11);
349 input[INPUT_RC_CH16] = GET_RX_CHANNEL_INPUT(AUX12);
350 input[INPUT_RC_CH17] = GET_RX_CHANNEL_INPUT(AUX13);
351 input[INPUT_RC_CH18] = GET_RX_CHANNEL_INPUT(AUX14);
352 #ifdef USE_34CHANNELS
353 input[INPUT_RC_CH19] = GET_RX_CHANNEL_INPUT(AUX15);
354 input[INPUT_RC_CH20] = GET_RX_CHANNEL_INPUT(AUX16);
355 input[INPUT_RC_CH21] = GET_RX_CHANNEL_INPUT(AUX17);
356 input[INPUT_RC_CH22] = GET_RX_CHANNEL_INPUT(AUX18);
357 input[INPUT_RC_CH23] = GET_RX_CHANNEL_INPUT(AUX19);
358 input[INPUT_RC_CH24] = GET_RX_CHANNEL_INPUT(AUX20);
359 input[INPUT_RC_CH25] = GET_RX_CHANNEL_INPUT(AUX21);
360 input[INPUT_RC_CH26] = GET_RX_CHANNEL_INPUT(AUX22);
361 input[INPUT_RC_CH27] = GET_RX_CHANNEL_INPUT(AUX23);
362 input[INPUT_RC_CH28] = GET_RX_CHANNEL_INPUT(AUX24);
363 input[INPUT_RC_CH29] = GET_RX_CHANNEL_INPUT(AUX25);
364 input[INPUT_RC_CH30] = GET_RX_CHANNEL_INPUT(AUX26);
365 input[INPUT_RC_CH31] = GET_RX_CHANNEL_INPUT(AUX27);
366 input[INPUT_RC_CH32] = GET_RX_CHANNEL_INPUT(AUX28);
367 input[INPUT_RC_CH33] = GET_RX_CHANNEL_INPUT(AUX29);
368 input[INPUT_RC_CH34] = GET_RX_CHANNEL_INPUT(AUX30);
369 #endif
370 #undef GET_RX_CHANNEL_INPUT
372 #ifdef USE_HEADTRACKER
373 headTrackerDevice_t *dev = headTrackerCommonDevice();
374 if(dev && headTrackerCommonIsValid(dev) && !IS_RC_MODE_ACTIVE(BOXGIMBALCENTER)) {
375 input[INPUT_HEADTRACKER_PAN] = headTrackerCommonGetPanPWM(dev) - PWM_RANGE_MIDDLE;
376 input[INPUT_HEADTRACKER_TILT] = headTrackerCommonGetTiltPWM(dev) - PWM_RANGE_MIDDLE;
377 input[INPUT_HEADTRACKER_ROLL] = headTrackerCommonGetRollPWM(dev) - PWM_RANGE_MIDDLE;
378 } else {
379 input[INPUT_HEADTRACKER_PAN] = 0;
380 input[INPUT_HEADTRACKER_TILT] = 0;
381 input[INPUT_HEADTRACKER_ROLL] = 0;
383 #else
384 input[INPUT_HEADTRACKER_PAN] = 0;
385 input[INPUT_HEADTRACKER_TILT] = 0;
386 input[INPUT_HEADTRACKER_ROLL] = 0;
387 #endif
389 #ifdef USE_SIMULATOR
390 simulatorData.input[INPUT_STABILIZED_ROLL] = input[INPUT_STABILIZED_ROLL];
391 simulatorData.input[INPUT_STABILIZED_PITCH] = input[INPUT_STABILIZED_PITCH];
392 simulatorData.input[INPUT_STABILIZED_YAW] = input[INPUT_STABILIZED_YAW];
393 simulatorData.input[INPUT_STABILIZED_THROTTLE] = input[INPUT_STABILIZED_THROTTLE];
394 #endif
396 for (int i = 0; i < MAX_SUPPORTED_SERVOS; i++) {
397 servo[i] = 0;
400 // mix servos according to rules
401 for (int i = 0; i < servoRuleCount; i++) {
402 const uint8_t target = currentServoMixer[i].targetChannel;
403 const uint8_t from = currentServoMixer[i].inputSource;
405 int16_t inputRaw = input[from];
408 * Check if conditions for a rule are met, not all conditions apply all the time
410 #ifdef USE_PROGRAMMING_FRAMEWORK
411 if (!logicConditionGetValue(currentServoMixer[i].conditionId)) {
412 inputRaw = 0;
414 #endif
416 * Apply mixer speed limit. 1 [one] speed unit is defined as 10us/s:
417 * 0 = no limiting
418 * 1 = 10us/s -> full servo sweep (from 1000 to 2000) is performed in 100s
419 * 10 = 100us/s -> full sweep (from 1000 to 2000) is performed in 10s
420 * 100 = 1000us/s -> full sweep in 1s
422 int16_t inputLimited = (int16_t) rateLimitFilterApply4(&servoSpeedLimitFilter[i], inputRaw, currentServoMixer[i].speed * 10, dT);
424 servo[target] += ((int32_t)inputLimited * currentServoMixer[i].rate) / 100;
427 for (int i = 0; i < MAX_SUPPORTED_SERVOS; i++) {
430 * Apply servo rate
432 servo[i] = ((int32_t)servoParams(i)->rate * servo[i]) / 100L;
435 * Perform acumulated servo output scaling to match servo min and max values
436 * Important: is servo rate is > 100%, total servo output might be bigger than
437 * min/max
439 if (servo[i] > 0) {
440 servo[i] = (int16_t) (servo[i] * servoMetadata[i].scaleMax);
441 } else {
442 servo[i] = (int16_t) (servo[i] * servoMetadata[i].scaleMin);
446 * Add a servo midpoint to the calculation
448 servo[i] += servoParams(i)->middle;
451 * Constrain servo position to min/max to prevent servo damage
452 * If servo was saturated above min/max, that means that user most probably
453 * allowed the situation when smix weight sum for an output was above 100
455 servo[i] = constrain(servo[i], servoParams(i)->min, servoParams(i)->max);
459 * When not armed, apply servo low position to all outputs that include a throttle or stabilizet throttle in the mix
461 if (!ARMING_FLAG(ARMED)) {
462 for (int i = 0; i < servoRuleCount; i++) {
463 const uint8_t target = currentServoMixer[i].targetChannel;
464 const uint8_t from = currentServoMixer[i].inputSource;
466 if (from == INPUT_STABILIZED_THROTTLE || from == INPUT_RC_THROTTLE) {
467 servo[target] = motorConfig()->mincommand;
473 #define SERVO_AUTOTRIM_TIMER_MS 2000
475 typedef enum {
476 AUTOTRIM_IDLE,
477 AUTOTRIM_COLLECTING,
478 AUTOTRIM_SAVE_PENDING,
479 AUTOTRIM_DONE,
480 } servoAutotrimState_e;
482 void processServoAutotrimMode(void)
484 static servoAutotrimState_e trimState = AUTOTRIM_IDLE;
485 static timeMs_t trimStartedAt;
487 static int16_t servoMiddleBackup[MAX_SUPPORTED_SERVOS];
488 static int32_t servoMiddleAccum[MAX_SUPPORTED_SERVOS];
489 static int32_t servoMiddleAccumCount[MAX_SUPPORTED_SERVOS];
491 if (isFwAutoModeActive(BOXAUTOTRIM)) {
492 switch (trimState) {
493 case AUTOTRIM_IDLE:
494 if (ARMING_FLAG(ARMED)) {
495 for (int axis = FD_ROLL; axis <= FD_YAW; axis++) {
496 for (int i = 0; i < servoRuleCount; i++) {
497 // Reset servo middle accumulator
498 const uint8_t target = currentServoMixer[i].targetChannel;
499 const uint8_t source = currentServoMixer[i].inputSource;
500 if (source == axis) {
501 servoMiddleBackup[target] = servoParams(target)->middle;
502 servoMiddleAccum[target] = 0;
503 servoMiddleAccumCount[target] = 0;
507 trimStartedAt = millis();
508 trimState = AUTOTRIM_COLLECTING;
510 else {
511 break;
513 // Fallthru
515 case AUTOTRIM_COLLECTING:
516 if (ARMING_FLAG(ARMED)) {
517 for (int axis = FD_ROLL; axis <= FD_YAW; axis++) {
518 for (int i = 0; i < servoRuleCount; i++) {
519 const uint8_t target = currentServoMixer[i].targetChannel;
520 const uint8_t source = currentServoMixer[i].inputSource;
521 if (source == axis) {
522 servoMiddleAccum[target] += servo[target];
523 servoMiddleAccumCount[target]++;
528 if ((millis() - trimStartedAt) > SERVO_AUTOTRIM_TIMER_MS) {
529 for (int axis = FD_ROLL; axis <= FD_YAW; axis++) {
530 for (int i = 0; i < servoRuleCount; i++) {
531 const uint8_t target = currentServoMixer[i].targetChannel;
532 const uint8_t source = currentServoMixer[i].inputSource;
533 if (source == axis) {
534 servoParamsMutable(target)->middle = servoMiddleAccum[target] / servoMiddleAccumCount[target];
538 trimState = AUTOTRIM_SAVE_PENDING;
539 pidResetErrorAccumulators(); //Reset Iterm since new midpoints override previously acumulated errors
542 else {
543 trimState = AUTOTRIM_IDLE;
545 break;
547 case AUTOTRIM_SAVE_PENDING:
548 // Wait for disarm and save to EEPROM
549 if (!ARMING_FLAG(ARMED)) {
550 saveConfigAndNotify();
551 trimState = AUTOTRIM_DONE;
553 break;
555 case AUTOTRIM_DONE:
556 break;
559 else {
560 // We are deactivating servo trim - restore servo midpoints
561 if (trimState == AUTOTRIM_SAVE_PENDING) {
562 for (int axis = FD_ROLL; axis <= FD_YAW; axis++) {
563 for (int i = 0; i < servoRuleCount; i++) {
564 const uint8_t target = currentServoMixer[i].targetChannel;
565 const uint8_t source = currentServoMixer[i].inputSource;
566 if (source == axis) {
567 servoParamsMutable(target)->middle = servoMiddleBackup[target];
573 trimState = AUTOTRIM_IDLE;
577 #define SERVO_AUTOTRIM_FILTER_CUTOFF 1 // LPF cutoff frequency
578 #define SERVO_AUTOTRIM_CENTER_MIN 1300
579 #define SERVO_AUTOTRIM_CENTER_MAX 1700
580 #define SERVO_AUTOTRIM_UPDATE_SIZE 5
581 #define SERVO_AUTOTRIM_ATTITUDE_LIMIT 50 // 5 degrees
583 void processContinuousServoAutotrim(const float dT)
585 static timeMs_t lastUpdateTimeMs;
586 static servoAutotrimState_e trimState = AUTOTRIM_IDLE;
587 static uint32_t servoMiddleUpdateCount;
589 const float rotRateMagnitudeFiltered = pt1FilterApply4(&rotRateFilter, fast_fsqrtf(vectorNormSquared(&imuMeasuredRotationBF)), SERVO_AUTOTRIM_FILTER_CUTOFF, dT);
590 const float targetRateMagnitudeFiltered = pt1FilterApply4(&targetRateFilter, getTotalRateTarget(), SERVO_AUTOTRIM_FILTER_CUTOFF, dT);
592 if (ARMING_FLAG(ARMED)) {
593 trimState = AUTOTRIM_COLLECTING;
594 if ((millis() - lastUpdateTimeMs) > 500) {
595 const bool planeIsFlyingStraight = rotRateMagnitudeFiltered <= DEGREES_TO_RADIANS(servoConfig()->servo_autotrim_rotation_limit);
596 const bool noRotationCommanded = targetRateMagnitudeFiltered <= servoConfig()->servo_autotrim_rotation_limit;
597 const bool sticksAreCentered = !areSticksDeflected();
598 const bool planeIsFlyingLevel = ABS(attitude.values.pitch + DEGREES_TO_DECIDEGREES(getFixedWingLevelTrim())) <= SERVO_AUTOTRIM_ATTITUDE_LIMIT
599 && ABS(attitude.values.roll) <= SERVO_AUTOTRIM_ATTITUDE_LIMIT;
600 if (
601 planeIsFlyingStraight &&
602 noRotationCommanded &&
603 planeIsFlyingLevel &&
604 sticksAreCentered &&
605 !FLIGHT_MODE(MANUAL_MODE) &&
606 isGPSHeadingValid() // TODO: proper flying detection
608 // Plane is flying straight and level: trim servos
609 for (int axis = FD_ROLL; axis <= FD_PITCH; axis++) {
610 // For each stabilized axis, add 5 units of I-term to all associated servo midpoints
611 const float axisIterm = getAxisIterm(axis);
612 if (fabsf(axisIterm) > SERVO_AUTOTRIM_UPDATE_SIZE) {
613 const int8_t ItermUpdate = axisIterm > 0.0f ? SERVO_AUTOTRIM_UPDATE_SIZE : -SERVO_AUTOTRIM_UPDATE_SIZE;
614 for (int i = 0; i < servoRuleCount; i++) {
615 #ifdef USE_PROGRAMMING_FRAMEWORK
616 if (!logicConditionGetValue(currentServoMixer[i].conditionId)) {
617 continue;
619 #endif
620 const uint8_t target = currentServoMixer[i].targetChannel;
621 const uint8_t source = currentServoMixer[i].inputSource;
622 if (source == axis) {
623 // Convert axis I-term to servo PWM and add to midpoint
624 const float mixerRate = currentServoMixer[i].rate / 100.0f;
625 const float servoRate = servoParams(target)->rate / 100.0f;
626 servoParamsMutable(target)->middle += (int16_t)(ItermUpdate * mixerRate * servoRate);
627 servoParamsMutable(target)->middle = constrain(servoParamsMutable(target)->middle, SERVO_AUTOTRIM_CENTER_MIN, SERVO_AUTOTRIM_CENTER_MAX);
630 pidReduceErrorAccumulators(ItermUpdate, axis);
633 servoMiddleUpdateCount++;
635 // Reset timer
636 lastUpdateTimeMs = millis();
638 } else if (trimState == AUTOTRIM_COLLECTING) {
639 // We have disarmed, save midpoints to EEPROM
640 saveConfigAndNotify();
641 trimState = AUTOTRIM_IDLE;
644 // Debug
645 DEBUG_SET(DEBUG_AUTOTRIM, 0, servoParams(2)->middle);
646 DEBUG_SET(DEBUG_AUTOTRIM, 2, servoParams(3)->middle);
647 DEBUG_SET(DEBUG_AUTOTRIM, 4, servoParams(4)->middle);
648 DEBUG_SET(DEBUG_AUTOTRIM, 6, servoParams(5)->middle);
649 DEBUG_SET(DEBUG_AUTOTRIM, 1, servoMiddleUpdateCount);
650 DEBUG_SET(DEBUG_AUTOTRIM, 3, MAX(RADIANS_TO_DEGREES(rotRateMagnitudeFiltered), targetRateMagnitudeFiltered));
651 DEBUG_SET(DEBUG_AUTOTRIM, 5, axisPID_I[FD_ROLL]);
652 DEBUG_SET(DEBUG_AUTOTRIM, 7, axisPID_I[FD_PITCH]);
655 void processServoAutotrim(const float dT) {
656 #ifdef USE_SIMULATOR
657 if (ARMING_FLAG(SIMULATOR_MODE_HITL)) {
658 return;
660 #endif
661 if(!STATE(AIRPLANE))
663 return;
665 if (feature(FEATURE_FW_AUTOTRIM)) {
666 processContinuousServoAutotrim(dT);
667 } else {
668 processServoAutotrimMode();
672 bool isServoOutputEnabled(void)
674 return servoOutputEnabled;
677 void setServoOutputEnabled(bool flag)
679 servoOutputEnabled = flag;
682 bool isMixerUsingServos(void)
684 return mixerUsesServos;