Merge pull request #11297 from SteveCEvans/baro_state
[betaflight.git] / src / main / fc / rc_adjustments.c
blob51b7a0e4396bea474a2240222da5b133cfd886f6
1 /*
2 * This file is part of Cleanflight and Betaflight.
4 * Cleanflight and Betaflight are free software. You can redistribute
5 * this software and/or modify this software under the terms of the
6 * GNU General Public License as published by the Free Software
7 * Foundation, either version 3 of the License, or (at your option)
8 * any later version.
10 * Cleanflight and Betaflight are distributed in the hope that they
11 * will be useful, but WITHOUT ANY WARRANTY; without even the implied
12 * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13 * See the GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this software.
18 * If not, see <http://www.gnu.org/licenses/>.
21 #include <stdbool.h>
22 #include <stdint.h>
23 #include <string.h>
25 #include <math.h>
27 #include "platform.h"
29 #include "blackbox/blackbox.h"
30 #include "blackbox/blackbox_fielddefs.h"
32 #include "build/build_config.h"
34 #include "common/axis.h"
35 #include "common/maths.h"
36 #include "common/utils.h"
38 #include "config/feature.h"
40 #include "drivers/time.h"
42 #include "config/config.h"
43 #include "fc/controlrate_profile.h"
44 #include "fc/rc_controls.h"
45 #include "fc/rc.h"
47 #include "flight/pid.h"
48 #include "flight/pid_init.h"
50 #include "io/beeper.h"
51 #include "io/ledstrip.h"
52 #include "io/pidaudio.h"
54 #include "osd/osd.h"
56 #include "pg/pg.h"
57 #include "pg/pg_ids.h"
58 #include "pg/rx.h"
60 #include "rx/rx.h"
62 #include "rc_adjustments.h"
64 #include "scheduler/scheduler.h"
66 #define ADJUSTMENT_RANGE_COUNT_INVALID -1
68 PG_REGISTER_ARRAY(adjustmentRange_t, MAX_ADJUSTMENT_RANGE_COUNT, adjustmentRanges, PG_ADJUSTMENT_RANGE_CONFIG, 2);
70 uint8_t pidAudioPositionToModeMap[7] = {
71 // on a pot with a center detent, it's easy to have center area for off/default, then three positions to the left and three to the right.
72 // current implementation yields RC values as below.
74 PID_AUDIO_PIDSUM_X, // 900 - ~1071 - Min
75 PID_AUDIO_PIDSUM_Y, // ~1071 - ~1242
76 PID_AUDIO_PIDSUM_XY, // ~1242 - ~1414
77 PID_AUDIO_OFF, // ~1414 - ~1585 - Center
78 PID_AUDIO_OFF, // ~1585 - ~1757
79 PID_AUDIO_OFF, // ~1757 - ~1928
80 PID_AUDIO_OFF, // ~1928 - 2100 - Max
82 // Note: Last 3 positions are currently pending implementations and use PID_AUDIO_OFF for now.
85 STATIC_UNIT_TESTED int stepwiseAdjustmentCount = ADJUSTMENT_RANGE_COUNT_INVALID;
86 STATIC_UNIT_TESTED timedAdjustmentState_t stepwiseAdjustments[MAX_ADJUSTMENT_RANGE_COUNT];
88 STATIC_UNIT_TESTED int continuosAdjustmentCount;
89 STATIC_UNIT_TESTED continuosAdjustmentState_t continuosAdjustments[MAX_ADJUSTMENT_RANGE_COUNT];
91 static void blackboxLogInflightAdjustmentEvent(adjustmentFunction_e adjustmentFunction, int32_t newValue)
93 #ifndef USE_BLACKBOX
94 UNUSED(adjustmentFunction);
95 UNUSED(newValue);
96 #else
97 if (blackboxConfig()->device) {
98 flightLogEvent_inflightAdjustment_t eventData;
99 eventData.adjustmentFunction = adjustmentFunction;
100 eventData.newValue = newValue;
101 eventData.floatFlag = false;
102 blackboxLogEvent(FLIGHT_LOG_EVENT_INFLIGHT_ADJUSTMENT, (flightLogEventData_t*)&eventData);
104 #endif
107 // sync with adjustmentFunction_e
108 static const adjustmentConfig_t defaultAdjustmentConfigs[ADJUSTMENT_FUNCTION_COUNT - 1] = {
110 .adjustmentFunction = ADJUSTMENT_RC_RATE,
111 .mode = ADJUSTMENT_MODE_STEP,
112 .data = { .step = 1 }
113 }, {
114 .adjustmentFunction = ADJUSTMENT_RC_EXPO,
115 .mode = ADJUSTMENT_MODE_STEP,
116 .data = { .step = 1 }
117 }, {
118 .adjustmentFunction = ADJUSTMENT_THROTTLE_EXPO,
119 .mode = ADJUSTMENT_MODE_STEP,
120 .data = { .step = 1 }
121 }, {
122 .adjustmentFunction = ADJUSTMENT_PITCH_ROLL_RATE,
123 .mode = ADJUSTMENT_MODE_STEP,
124 .data = { .step = 1 }
125 }, {
126 .adjustmentFunction = ADJUSTMENT_YAW_RATE,
127 .mode = ADJUSTMENT_MODE_STEP,
128 .data = { .step = 1 }
129 }, {
130 .adjustmentFunction = ADJUSTMENT_PITCH_ROLL_P,
131 .mode = ADJUSTMENT_MODE_STEP,
132 .data = { .step = 1 }
133 }, {
134 .adjustmentFunction = ADJUSTMENT_PITCH_ROLL_I,
135 .mode = ADJUSTMENT_MODE_STEP,
136 .data = { .step = 1 }
137 }, {
138 .adjustmentFunction = ADJUSTMENT_PITCH_ROLL_D,
139 .mode = ADJUSTMENT_MODE_STEP,
140 .data = { .step = 1 }
141 }, {
142 .adjustmentFunction = ADJUSTMENT_YAW_P,
143 .mode = ADJUSTMENT_MODE_STEP,
144 .data = { .step = 1 }
145 }, {
146 .adjustmentFunction = ADJUSTMENT_YAW_I,
147 .mode = ADJUSTMENT_MODE_STEP,
148 .data = { .step = 1 }
149 }, {
150 .adjustmentFunction = ADJUSTMENT_YAW_D,
151 .mode = ADJUSTMENT_MODE_STEP,
152 .data = { .step = 1 }
153 }, {
154 .adjustmentFunction = ADJUSTMENT_RATE_PROFILE,
155 .mode = ADJUSTMENT_MODE_SELECT,
156 .data = { .switchPositions = 3 }
157 }, {
158 .adjustmentFunction = ADJUSTMENT_PITCH_RATE,
159 .mode = ADJUSTMENT_MODE_STEP,
160 .data = { .step = 1 }
161 }, {
162 .adjustmentFunction = ADJUSTMENT_ROLL_RATE,
163 .mode = ADJUSTMENT_MODE_STEP,
164 .data = { .step = 1 }
165 }, {
166 .adjustmentFunction = ADJUSTMENT_PITCH_P,
167 .mode = ADJUSTMENT_MODE_STEP,
168 .data = { .step = 1 }
169 }, {
170 .adjustmentFunction = ADJUSTMENT_PITCH_I,
171 .mode = ADJUSTMENT_MODE_STEP,
172 .data = { .step = 1 }
173 }, {
174 .adjustmentFunction = ADJUSTMENT_PITCH_D,
175 .mode = ADJUSTMENT_MODE_STEP,
176 .data = { .step = 1 }
177 }, {
178 .adjustmentFunction = ADJUSTMENT_ROLL_P,
179 .mode = ADJUSTMENT_MODE_STEP,
180 .data = { .step = 1 }
181 }, {
182 .adjustmentFunction = ADJUSTMENT_ROLL_I,
183 .mode = ADJUSTMENT_MODE_STEP,
184 .data = { .step = 1 }
185 }, {
186 .adjustmentFunction = ADJUSTMENT_ROLL_D,
187 .mode = ADJUSTMENT_MODE_STEP,
188 .data = { .step = 1 }
189 }, {
190 .adjustmentFunction = ADJUSTMENT_RC_RATE_YAW,
191 .mode = ADJUSTMENT_MODE_STEP,
192 .data = { .step = 1 }
193 }, {
194 .adjustmentFunction = ADJUSTMENT_PITCH_ROLL_F,
195 .mode = ADJUSTMENT_MODE_STEP,
196 .data = { .step = 1 }
197 }, {
198 .adjustmentFunction = ADJUSTMENT_FEEDFORWARD_TRANSITION,
199 .mode = ADJUSTMENT_MODE_STEP,
200 .data = { .step = 1 }
201 }, {
202 .adjustmentFunction = ADJUSTMENT_HORIZON_STRENGTH,
203 .mode = ADJUSTMENT_MODE_SELECT,
204 .data = { .switchPositions = 255 }
205 }, {
206 .adjustmentFunction = ADJUSTMENT_PID_AUDIO,
207 .mode = ADJUSTMENT_MODE_SELECT,
208 .data = { .switchPositions = ARRAYLEN(pidAudioPositionToModeMap) }
209 }, {
210 .adjustmentFunction = ADJUSTMENT_PITCH_F,
211 .mode = ADJUSTMENT_MODE_STEP,
212 .data = { .step = 1 }
213 }, {
214 .adjustmentFunction = ADJUSTMENT_ROLL_F,
215 .mode = ADJUSTMENT_MODE_STEP,
216 .data = { .step = 1 }
217 }, {
218 .adjustmentFunction = ADJUSTMENT_YAW_F,
219 .mode = ADJUSTMENT_MODE_STEP,
220 .data = { .step = 1 }
221 }, {
222 .adjustmentFunction = ADJUSTMENT_OSD_PROFILE,
223 .mode = ADJUSTMENT_MODE_SELECT,
224 .data = { .switchPositions = 3 }
225 }, {
226 .adjustmentFunction = ADJUSTMENT_LED_PROFILE,
227 .mode = ADJUSTMENT_MODE_SELECT,
228 .data = { .switchPositions = 3 }
232 #if defined(USE_OSD) && defined(USE_OSD_ADJUSTMENTS)
233 static const char * const adjustmentLabels[] = {
234 "RC RATE",
235 "RC EXPO",
236 "THROTTLE EXPO",
237 "ROLL RATE",
238 "YAW RATE",
239 "PITCH/ROLL P",
240 "PITCH/ROLL I",
241 "PITCH/ROLL D",
242 "YAW P",
243 "YAW I",
244 "YAW D",
245 "RATE PROFILE",
246 "PITCH RATE",
247 "ROLL RATE",
248 "PITCH P",
249 "PITCH I",
250 "PITCH D",
251 "ROLL P",
252 "ROLL I",
253 "ROLL D",
254 "RC RATE YAW",
255 "PITCH/ROLL F",
256 "FF TRANSITION",
257 "HORIZON STRENGTH",
258 "ROLL RC RATE",
259 "PITCH RC RATE",
260 "ROLL RC EXPO",
261 "PITCH RC EXPO",
262 "PID AUDIO",
263 "PITCH F",
264 "ROLL F",
265 "YAW F",
266 "OSD PROFILE",
269 static int adjustmentRangeNameIndex = 0;
270 static int adjustmentRangeValue = -1;
271 #endif
273 static int applyStepAdjustment(controlRateConfig_t *controlRateConfig, uint8_t adjustmentFunction, int delta)
275 beeperConfirmationBeeps(delta > 0 ? 2 : 1);
276 int newValue;
277 switch (adjustmentFunction) {
278 case ADJUSTMENT_RC_RATE:
279 case ADJUSTMENT_ROLL_RC_RATE:
280 newValue = constrain((int)controlRateConfig->rcRates[FD_ROLL] + delta, 1, CONTROL_RATE_CONFIG_RC_RATES_MAX);
281 controlRateConfig->rcRates[FD_ROLL] = newValue;
282 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_RC_RATE, newValue);
283 if (adjustmentFunction == ADJUSTMENT_ROLL_RC_RATE) {
284 break;
286 // fall through for combined ADJUSTMENT_RC_EXPO
287 FALLTHROUGH;
288 case ADJUSTMENT_PITCH_RC_RATE:
289 newValue = constrain((int)controlRateConfig->rcRates[FD_PITCH] + delta, 1, CONTROL_RATE_CONFIG_RC_RATES_MAX);
290 controlRateConfig->rcRates[FD_PITCH] = newValue;
291 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_RC_RATE, newValue);
292 break;
293 case ADJUSTMENT_RC_EXPO:
294 case ADJUSTMENT_ROLL_RC_EXPO:
295 newValue = constrain((int)controlRateConfig->rcExpo[FD_ROLL] + delta, 0, CONTROL_RATE_CONFIG_RC_EXPO_MAX);
296 controlRateConfig->rcExpo[FD_ROLL] = newValue;
297 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_RC_EXPO, newValue);
298 if (adjustmentFunction == ADJUSTMENT_ROLL_RC_EXPO) {
299 break;
301 // fall through for combined ADJUSTMENT_RC_EXPO
302 FALLTHROUGH;
303 case ADJUSTMENT_PITCH_RC_EXPO:
304 newValue = constrain((int)controlRateConfig->rcExpo[FD_PITCH] + delta, 0, CONTROL_RATE_CONFIG_RC_EXPO_MAX);
305 controlRateConfig->rcExpo[FD_PITCH] = newValue;
306 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_RC_EXPO, newValue);
307 break;
308 case ADJUSTMENT_THROTTLE_EXPO:
309 newValue = constrain((int)controlRateConfig->thrExpo8 + delta, 0, 100); // FIXME magic numbers repeated in cli.c
310 controlRateConfig->thrExpo8 = newValue;
311 initRcProcessing();
312 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_THROTTLE_EXPO, newValue);
313 break;
314 case ADJUSTMENT_PITCH_ROLL_RATE:
315 case ADJUSTMENT_PITCH_RATE:
316 newValue = constrain((int)controlRateConfig->rates[FD_PITCH] + delta, 0, CONTROL_RATE_CONFIG_RATE_MAX);
317 controlRateConfig->rates[FD_PITCH] = newValue;
318 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_RATE, newValue);
319 if (adjustmentFunction == ADJUSTMENT_PITCH_RATE) {
320 break;
322 // fall through for combined ADJUSTMENT_PITCH_ROLL_RATE
323 FALLTHROUGH;
324 case ADJUSTMENT_ROLL_RATE:
325 newValue = constrain((int)controlRateConfig->rates[FD_ROLL] + delta, 0, CONTROL_RATE_CONFIG_RATE_MAX);
326 controlRateConfig->rates[FD_ROLL] = newValue;
327 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_RATE, newValue);
328 break;
329 case ADJUSTMENT_YAW_RATE:
330 newValue = constrain((int)controlRateConfig->rates[FD_YAW] + delta, 0, CONTROL_RATE_CONFIG_RATE_MAX);
331 controlRateConfig->rates[FD_YAW] = newValue;
332 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_YAW_RATE, newValue);
333 break;
334 case ADJUSTMENT_PITCH_ROLL_P:
335 case ADJUSTMENT_PITCH_P:
336 newValue = constrain((int)currentPidProfile->pid[PID_PITCH].P + delta, 0, 200); // FIXME magic numbers repeated in cli.c
337 currentPidProfile->pid[PID_PITCH].P = newValue;
338 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_P, newValue);
340 if (adjustmentFunction == ADJUSTMENT_PITCH_P) {
341 break;
343 // fall through for combined ADJUSTMENT_PITCH_ROLL_P
344 FALLTHROUGH;
345 case ADJUSTMENT_ROLL_P:
346 newValue = constrain((int)currentPidProfile->pid[PID_ROLL].P + delta, 0, 200); // FIXME magic numbers repeated in cli.c
347 currentPidProfile->pid[PID_ROLL].P = newValue;
348 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_P, newValue);
349 break;
350 case ADJUSTMENT_PITCH_ROLL_I:
351 case ADJUSTMENT_PITCH_I:
352 newValue = constrain((int)currentPidProfile->pid[PID_PITCH].I + delta, 0, 200); // FIXME magic numbers repeated in cli.c
353 currentPidProfile->pid[PID_PITCH].I = newValue;
354 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_I, newValue);
355 if (adjustmentFunction == ADJUSTMENT_PITCH_I) {
356 break;
358 // fall through for combined ADJUSTMENT_PITCH_ROLL_I
359 FALLTHROUGH;
360 case ADJUSTMENT_ROLL_I:
361 newValue = constrain((int)currentPidProfile->pid[PID_ROLL].I + delta, 0, 200); // FIXME magic numbers repeated in cli.c
362 currentPidProfile->pid[PID_ROLL].I = newValue;
363 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_I, newValue);
364 break;
365 case ADJUSTMENT_PITCH_ROLL_D:
366 case ADJUSTMENT_PITCH_D:
367 newValue = constrain((int)currentPidProfile->pid[PID_PITCH].D + delta, 0, 200); // FIXME magic numbers repeated in cli.c
368 currentPidProfile->pid[PID_PITCH].D = newValue;
369 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_D, newValue);
370 if (adjustmentFunction == ADJUSTMENT_PITCH_D) {
371 break;
373 // fall through for combined ADJUSTMENT_PITCH_ROLL_D
374 FALLTHROUGH;
375 case ADJUSTMENT_ROLL_D:
376 newValue = constrain((int)currentPidProfile->pid[PID_ROLL].D + delta, 0, 200); // FIXME magic numbers repeated in cli.c
377 currentPidProfile->pid[PID_ROLL].D = newValue;
378 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_D, newValue);
379 break;
380 case ADJUSTMENT_YAW_P:
381 newValue = constrain((int)currentPidProfile->pid[PID_YAW].P + delta, 0, 200); // FIXME magic numbers repeated in cli.c
382 currentPidProfile->pid[PID_YAW].P = newValue;
383 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_YAW_P, newValue);
384 break;
385 case ADJUSTMENT_YAW_I:
386 newValue = constrain((int)currentPidProfile->pid[PID_YAW].I + delta, 0, 200); // FIXME magic numbers repeated in cli.c
387 currentPidProfile->pid[PID_YAW].I = newValue;
388 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_YAW_I, newValue);
389 break;
390 case ADJUSTMENT_YAW_D:
391 newValue = constrain((int)currentPidProfile->pid[PID_YAW].D + delta, 0, 200); // FIXME magic numbers repeated in cli.c
392 currentPidProfile->pid[PID_YAW].D = newValue;
393 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_YAW_D, newValue);
394 break;
395 case ADJUSTMENT_RC_RATE_YAW:
396 newValue = constrain((int)controlRateConfig->rcRates[FD_YAW] + delta, 1, CONTROL_RATE_CONFIG_RC_RATES_MAX);
397 controlRateConfig->rcRates[FD_YAW] = newValue;
398 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_RC_RATE_YAW, newValue);
399 break;
400 case ADJUSTMENT_PITCH_ROLL_F:
401 case ADJUSTMENT_PITCH_F:
402 newValue = constrain(currentPidProfile->pid[PID_PITCH].F + delta, 0, 2000);
403 currentPidProfile->pid[PID_PITCH].F = newValue;
404 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_F, newValue);
406 if (adjustmentFunction == ADJUSTMENT_PITCH_F) {
407 break;
409 // fall through for combined ADJUSTMENT_PITCH_ROLL_F
410 FALLTHROUGH;
411 case ADJUSTMENT_ROLL_F:
412 newValue = constrain(currentPidProfile->pid[PID_ROLL].F + delta, 0, 2000);
413 currentPidProfile->pid[PID_ROLL].F = newValue;
414 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_F, newValue);
415 break;
416 case ADJUSTMENT_YAW_F:
417 newValue = constrain(currentPidProfile->pid[PID_YAW].F + delta, 0, 2000);
418 currentPidProfile->pid[PID_YAW].F = newValue;
419 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_YAW_F, newValue);
420 break;
421 #if defined(USE_FEEDFORWARD)
422 case ADJUSTMENT_FEEDFORWARD_TRANSITION:
423 newValue = constrain(currentPidProfile->feedforward_transition + delta, 1, 100); // FIXME magic numbers repeated in cli.c
424 currentPidProfile->feedforward_transition = newValue;
425 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_FEEDFORWARD_TRANSITION, newValue);
426 break;
427 #endif
428 default:
429 newValue = -1;
430 break;
433 return newValue;
436 static int applyAbsoluteAdjustment(controlRateConfig_t *controlRateConfig, adjustmentFunction_e adjustmentFunction, int value)
438 int newValue;
440 switch (adjustmentFunction) {
441 case ADJUSTMENT_RC_RATE:
442 case ADJUSTMENT_ROLL_RC_RATE:
443 newValue = constrain(value, 1, CONTROL_RATE_CONFIG_RC_RATES_MAX);
444 controlRateConfig->rcRates[FD_ROLL] = newValue;
445 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_RC_RATE, newValue);
446 if (adjustmentFunction == ADJUSTMENT_ROLL_RC_RATE) {
447 break;
449 // fall through for combined ADJUSTMENT_RC_EXPO
450 FALLTHROUGH;
451 case ADJUSTMENT_PITCH_RC_RATE:
452 newValue = constrain(value, 1, CONTROL_RATE_CONFIG_RC_RATES_MAX);
453 controlRateConfig->rcRates[FD_PITCH] = newValue;
454 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_RC_RATE, newValue);
455 break;
456 case ADJUSTMENT_RC_EXPO:
457 case ADJUSTMENT_ROLL_RC_EXPO:
458 newValue = constrain(value, 1, CONTROL_RATE_CONFIG_RC_EXPO_MAX);
459 controlRateConfig->rcExpo[FD_ROLL] = newValue;
460 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_RC_EXPO, newValue);
461 if (adjustmentFunction == ADJUSTMENT_ROLL_RC_EXPO) {
462 break;
464 // fall through for combined ADJUSTMENT_RC_EXPO
465 FALLTHROUGH;
466 case ADJUSTMENT_PITCH_RC_EXPO:
467 newValue = constrain(value, 0, CONTROL_RATE_CONFIG_RC_EXPO_MAX);
468 controlRateConfig->rcExpo[FD_PITCH] = newValue;
469 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_RC_EXPO, newValue);
470 break;
471 case ADJUSTMENT_THROTTLE_EXPO:
472 newValue = constrain(value, 0, 100); // FIXME magic numbers repeated in cli.c
473 controlRateConfig->thrExpo8 = newValue;
474 initRcProcessing();
475 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_THROTTLE_EXPO, newValue);
476 break;
477 case ADJUSTMENT_PITCH_ROLL_RATE:
478 case ADJUSTMENT_PITCH_RATE:
479 newValue = constrain(value, 0, CONTROL_RATE_CONFIG_RATE_MAX);
480 controlRateConfig->rates[FD_PITCH] = newValue;
481 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_RATE, newValue);
482 if (adjustmentFunction == ADJUSTMENT_PITCH_RATE) {
483 break;
485 // fall through for combined ADJUSTMENT_PITCH_ROLL_RATE
486 FALLTHROUGH;
487 case ADJUSTMENT_ROLL_RATE:
488 newValue = constrain(value, 0, CONTROL_RATE_CONFIG_RATE_MAX);
489 controlRateConfig->rates[FD_ROLL] = newValue;
490 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_RATE, newValue);
491 break;
492 case ADJUSTMENT_YAW_RATE:
493 newValue = constrain(value, 0, CONTROL_RATE_CONFIG_RATE_MAX);
494 controlRateConfig->rates[FD_YAW] = newValue;
495 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_YAW_RATE, newValue);
496 break;
497 case ADJUSTMENT_PITCH_ROLL_P:
498 case ADJUSTMENT_PITCH_P:
499 newValue = constrain(value, 0, 200); // FIXME magic numbers repeated in cli.c
500 currentPidProfile->pid[PID_PITCH].P = newValue;
501 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_P, newValue);
503 if (adjustmentFunction == ADJUSTMENT_PITCH_P) {
504 break;
506 // fall through for combined ADJUSTMENT_PITCH_ROLL_P
507 FALLTHROUGH;
508 case ADJUSTMENT_ROLL_P:
509 newValue = constrain(value, 0, 200); // FIXME magic numbers repeated in cli.c
510 currentPidProfile->pid[PID_ROLL].P = newValue;
511 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_P, newValue);
512 break;
513 case ADJUSTMENT_PITCH_ROLL_I:
514 case ADJUSTMENT_PITCH_I:
515 newValue = constrain(value, 0, 200); // FIXME magic numbers repeated in cli.c
516 currentPidProfile->pid[PID_PITCH].I = newValue;
517 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_I, newValue);
518 if (adjustmentFunction == ADJUSTMENT_PITCH_I) {
519 break;
521 // fall through for combined ADJUSTMENT_PITCH_ROLL_I
522 FALLTHROUGH;
523 case ADJUSTMENT_ROLL_I:
524 newValue = constrain(value, 0, 200); // FIXME magic numbers repeated in cli.c
525 currentPidProfile->pid[PID_ROLL].I = newValue;
526 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_I, newValue);
527 break;
528 case ADJUSTMENT_PITCH_ROLL_D:
529 case ADJUSTMENT_PITCH_D:
530 newValue = constrain(value, 0, 200); // FIXME magic numbers repeated in cli.c
531 currentPidProfile->pid[PID_PITCH].D = newValue;
532 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_D, newValue);
533 if (adjustmentFunction == ADJUSTMENT_PITCH_D) {
534 break;
536 // fall through for combined ADJUSTMENT_PITCH_ROLL_D
537 FALLTHROUGH;
538 case ADJUSTMENT_ROLL_D:
539 newValue = constrain(value, 0, 200); // FIXME magic numbers repeated in cli.c
540 currentPidProfile->pid[PID_ROLL].D = newValue;
541 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_D, newValue);
542 break;
543 case ADJUSTMENT_YAW_P:
544 newValue = constrain(value, 0, 200); // FIXME magic numbers repeated in cli.c
545 currentPidProfile->pid[PID_YAW].P = newValue;
546 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_YAW_P, newValue);
547 break;
548 case ADJUSTMENT_YAW_I:
549 newValue = constrain(value, 0, 200); // FIXME magic numbers repeated in cli.c
550 currentPidProfile->pid[PID_YAW].I = newValue;
551 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_YAW_I, newValue);
552 break;
553 case ADJUSTMENT_YAW_D:
554 newValue = constrain(value, 0, 200); // FIXME magic numbers repeated in cli.c
555 currentPidProfile->pid[PID_YAW].D = newValue;
556 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_YAW_D, newValue);
557 break;
558 case ADJUSTMENT_RC_RATE_YAW:
559 newValue = constrain(value, 1, CONTROL_RATE_CONFIG_RC_RATES_MAX);
560 controlRateConfig->rcRates[FD_YAW] = newValue;
561 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_RC_RATE_YAW, newValue);
562 break;
563 case ADJUSTMENT_PITCH_ROLL_F:
564 case ADJUSTMENT_PITCH_F:
565 newValue = constrain(value, 0, 2000);
566 currentPidProfile->pid[PID_PITCH].F = newValue;
567 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_PITCH_F, newValue);
569 if (adjustmentFunction == ADJUSTMENT_PITCH_F) {
570 break;
572 // fall through for combined ADJUSTMENT_PITCH_ROLL_F
573 FALLTHROUGH;
574 case ADJUSTMENT_ROLL_F:
575 newValue = constrain(value, 0, 2000);
576 currentPidProfile->pid[PID_ROLL].F = newValue;
577 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_ROLL_F, newValue);
578 break;
579 case ADJUSTMENT_YAW_F:
580 newValue = constrain(value, 0, 2000);
581 currentPidProfile->pid[PID_YAW].F = newValue;
582 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_YAW_F, newValue);
583 break;
584 #if defined(USE_FEEDFORWARD)
585 case ADJUSTMENT_FEEDFORWARD_TRANSITION:
586 newValue = constrain(value, 1, 100); // FIXME magic numbers repeated in cli.c
587 currentPidProfile->feedforward_transition = newValue;
588 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_FEEDFORWARD_TRANSITION, newValue);
589 break;
590 #endif
591 default:
592 newValue = -1;
593 break;
596 return newValue;
599 static uint8_t applySelectAdjustment(adjustmentFunction_e adjustmentFunction, uint8_t position)
601 uint8_t beeps = 0;
603 switch (adjustmentFunction) {
604 case ADJUSTMENT_RATE_PROFILE:
605 if (getCurrentControlRateProfileIndex() != position) {
606 changeControlRateProfile(position);
607 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_RATE_PROFILE, position);
609 beeps = position + 1;
611 break;
612 case ADJUSTMENT_HORIZON_STRENGTH:
614 uint8_t newValue = constrain(position, 0, 200); // FIXME magic numbers repeated in serial_cli.c
615 if (currentPidProfile->pid[PID_LEVEL].D != newValue) {
616 beeps = ((newValue - currentPidProfile->pid[PID_LEVEL].D) / 8) + 1;
617 currentPidProfile->pid[PID_LEVEL].D = newValue;
618 blackboxLogInflightAdjustmentEvent(ADJUSTMENT_HORIZON_STRENGTH, position);
621 break;
622 case ADJUSTMENT_PID_AUDIO:
623 #ifdef USE_PID_AUDIO
625 pidAudioModes_e newMode = pidAudioPositionToModeMap[position];
626 if (newMode != pidAudioGetMode()) {
627 pidAudioSetMode(newMode);
630 #endif
631 break;
632 case ADJUSTMENT_OSD_PROFILE:
633 #ifdef USE_OSD_PROFILES
634 if (getCurrentOsdProfileIndex() != (position + 1)) {
635 changeOsdProfileIndex(position+1);
637 #endif
638 break;
639 case ADJUSTMENT_LED_PROFILE:
640 #ifdef USE_LED_STRIP
641 if (getLedProfile() != position) {
642 setLedProfile(position);
644 #endif
645 break;
647 default:
648 break;
651 if (beeps) {
652 beeperConfirmationBeeps(beeps);
655 return position;
658 #define ADJUSTMENT_FUNCTION_CONFIG_INDEX_OFFSET 1
660 static void calcActiveAdjustmentRanges(void)
662 adjustmentRange_t defaultAdjustmentRange;
663 memset(&defaultAdjustmentRange, 0, sizeof(defaultAdjustmentRange));
665 stepwiseAdjustmentCount = 0;
666 continuosAdjustmentCount = 0;
667 for (int i = 0; i < MAX_ADJUSTMENT_RANGE_COUNT; i++) {
668 const adjustmentRange_t * const adjustmentRange = adjustmentRanges(i);
669 if (memcmp(adjustmentRange, &defaultAdjustmentRange, sizeof(defaultAdjustmentRange)) != 0) {
670 const adjustmentConfig_t *adjustmentConfig = &defaultAdjustmentConfigs[adjustmentRange->adjustmentConfig - ADJUSTMENT_FUNCTION_CONFIG_INDEX_OFFSET];
671 if (adjustmentRange->adjustmentCenter == 0 && adjustmentConfig->mode != ADJUSTMENT_MODE_SELECT) {
672 timedAdjustmentState_t *adjustmentState = &stepwiseAdjustments[stepwiseAdjustmentCount++];
673 adjustmentState->adjustmentRangeIndex = i;
674 adjustmentState->timeoutAt = 0;
675 adjustmentState->ready = true;
676 } else {
677 continuosAdjustmentState_t *adjustmentState = &continuosAdjustments[continuosAdjustmentCount++];
678 adjustmentState->adjustmentRangeIndex = i;
679 adjustmentState->lastRcData = 0;
685 #define VALUE_DISPLAY_LATENCY_MS 2000
687 #if defined(USE_OSD) && defined(USE_OSD_ADJUSTMENTS)
688 static void updateOsdAdjustmentData(int newValue, adjustmentFunction_e adjustmentFunction)
690 static timeMs_t lastValueChangeMs;
692 timeMs_t currentTimeMs = millis();
693 if (newValue != -1
694 && adjustmentFunction != ADJUSTMENT_RATE_PROFILE // Rate profile already has an OSD element
695 #ifdef USE_OSD_PROFILES
696 && adjustmentFunction != ADJUSTMENT_OSD_PROFILE
697 #endif
699 adjustmentRangeNameIndex = adjustmentFunction;
700 adjustmentRangeValue = newValue;
702 lastValueChangeMs = currentTimeMs;
705 if (cmp32(currentTimeMs, lastValueChangeMs + VALUE_DISPLAY_LATENCY_MS) >= 0) {
706 adjustmentRangeNameIndex = 0;
709 #endif
711 #define RESET_FREQUENCY_2HZ (1000 / 2)
713 static void processStepwiseAdjustments(controlRateConfig_t *controlRateConfig, const bool canUseRxData)
715 const timeMs_t now = millis();
717 for (int index = 0; index < stepwiseAdjustmentCount; index++) {
718 timedAdjustmentState_t *adjustmentState = &stepwiseAdjustments[index];
719 const adjustmentRange_t *const adjustmentRange = adjustmentRanges(adjustmentState->adjustmentRangeIndex);
720 const adjustmentConfig_t *adjustmentConfig = &defaultAdjustmentConfigs[adjustmentRange->adjustmentConfig - ADJUSTMENT_FUNCTION_CONFIG_INDEX_OFFSET];
721 const adjustmentFunction_e adjustmentFunction = adjustmentConfig->adjustmentFunction;
723 if (!isRangeActive(adjustmentRange->auxChannelIndex, &adjustmentRange->range) ||
724 adjustmentFunction == ADJUSTMENT_NONE) {
725 adjustmentState->timeoutAt = 0;
727 continue;
730 if (cmp32(now, adjustmentState->timeoutAt) >= 0) {
731 adjustmentState->timeoutAt = now + RESET_FREQUENCY_2HZ;
732 adjustmentState->ready = true;
735 if (!canUseRxData) {
736 continue;
739 const uint8_t channelIndex = NON_AUX_CHANNEL_COUNT + adjustmentRange->auxSwitchChannelIndex;
741 if (adjustmentConfig->mode == ADJUSTMENT_MODE_STEP) {
742 int delta;
743 if (rcData[channelIndex] > rxConfig()->midrc + 200) {
744 delta = adjustmentConfig->data.step;
745 } else if (rcData[channelIndex] < rxConfig()->midrc - 200) {
746 delta = -adjustmentConfig->data.step;
747 } else {
748 // returning the switch to the middle immediately resets the ready state
749 adjustmentState->ready = true;
750 adjustmentState->timeoutAt = now + RESET_FREQUENCY_2HZ;
751 continue;
753 if (!adjustmentState->ready) {
754 continue;
757 int newValue = applyStepAdjustment(controlRateConfig, adjustmentFunction, delta);
759 setConfigDirty();
761 pidInitConfig(currentPidProfile);
763 adjustmentState->ready = false;
765 #if defined(USE_OSD) && defined(USE_OSD_ADJUSTMENTS)
766 updateOsdAdjustmentData(newValue, adjustmentConfig->adjustmentFunction);
767 #else
768 UNUSED(newValue);
769 #endif
774 static void setConfigDirtyIfNotPermanent(const channelRange_t *range)
776 if (!(range->startStep == MIN_MODE_RANGE_STEP && range->endStep == MAX_MODE_RANGE_STEP)) {
777 // Only set the configuration dirty if this range is NOT permanently enabled (and the config thus never used).
778 setConfigDirty();
782 static void processContinuosAdjustments(controlRateConfig_t *controlRateConfig)
784 for (int i = 0; i < continuosAdjustmentCount; i++) {
785 continuosAdjustmentState_t *adjustmentState = &continuosAdjustments[i];
786 const adjustmentRange_t * const adjustmentRange = adjustmentRanges(adjustmentState->adjustmentRangeIndex);
787 const uint8_t channelIndex = NON_AUX_CHANNEL_COUNT + adjustmentRange->auxSwitchChannelIndex;
788 const adjustmentConfig_t *adjustmentConfig = &defaultAdjustmentConfigs[adjustmentRange->adjustmentConfig - ADJUSTMENT_FUNCTION_CONFIG_INDEX_OFFSET];
789 const adjustmentFunction_e adjustmentFunction = adjustmentConfig->adjustmentFunction;
791 if (isRangeActive(adjustmentRange->auxChannelIndex, &adjustmentRange->range) &&
792 adjustmentFunction != ADJUSTMENT_NONE) {
794 if (rcData[channelIndex] != adjustmentState->lastRcData) {
795 int newValue = -1;
797 if (adjustmentConfig->mode == ADJUSTMENT_MODE_SELECT) {
798 int switchPositions = adjustmentConfig->data.switchPositions;
799 if (adjustmentFunction == ADJUSTMENT_RATE_PROFILE && systemConfig()->rateProfile6PosSwitch) {
800 switchPositions = 6;
802 const uint16_t rangeWidth = (2100 - 900) / switchPositions;
803 const uint8_t position = (constrain(rcData[channelIndex], 900, 2100 - 1) - 900) / rangeWidth;
804 newValue = applySelectAdjustment(adjustmentFunction, position);
806 setConfigDirtyIfNotPermanent(&adjustmentRange->range);
807 } else {
808 // If setting is defined for step adjustment and center value has been specified, apply values directly (scaled) from aux channel
809 if (adjustmentRange->adjustmentCenter &&
810 (adjustmentConfig->mode == ADJUSTMENT_MODE_STEP)) {
811 int value = (((rcData[channelIndex] - PWM_RANGE_MIDDLE) * adjustmentRange->adjustmentScale) / (PWM_RANGE_MIDDLE - PWM_RANGE_MIN)) + adjustmentRange->adjustmentCenter;
813 newValue = applyAbsoluteAdjustment(controlRateConfig, adjustmentFunction, value);
815 setConfigDirtyIfNotPermanent(&adjustmentRange->range);
817 pidInitConfig(currentPidProfile);
820 #if defined(USE_OSD) && defined(USE_OSD_ADJUSTMENTS)
821 updateOsdAdjustmentData(newValue, adjustmentConfig->adjustmentFunction);
822 #else
823 UNUSED(newValue);
824 #endif
825 adjustmentState->lastRcData = rcData[channelIndex];
827 } else {
828 adjustmentState->lastRcData = 0;
833 void processRcAdjustments(controlRateConfig_t *controlRateConfig)
835 const bool canUseRxData = rxIsReceivingSignal();
837 // Recalculate the new active adjustments if required
838 if (stepwiseAdjustmentCount == ADJUSTMENT_RANGE_COUNT_INVALID) {
839 // This can take up to 30us and is only call when not armed so ignore this timing as it doesn't impact flight
840 schedulerIgnoreTaskExecTime();
841 calcActiveAdjustmentRanges();
844 processStepwiseAdjustments(controlRateConfig, canUseRxData);
846 if (canUseRxData) {
847 processContinuosAdjustments(controlRateConfig);
850 #if defined(USE_OSD) && defined(USE_OSD_ADJUSTMENTS)
851 // Hide the element if there is no change
852 updateOsdAdjustmentData(-1, 0);
853 #endif
856 #if defined(USE_OSD) && defined(USE_OSD_ADJUSTMENTS)
857 const char *getAdjustmentsRangeName(void)
859 if (adjustmentRangeNameIndex > 0) {
860 return &adjustmentLabels[adjustmentRangeNameIndex - 1][0];
861 } else {
862 return NULL;
866 int getAdjustmentsRangeValue(void)
868 return adjustmentRangeValue;
870 #endif
872 void activeAdjustmentRangeReset(void)
874 stepwiseAdjustmentCount = ADJUSTMENT_RANGE_COUNT_INVALID;