/** ****************************************************************************** * @addtogroup OpenPilotModules OpenPilot Modules * @{ * @addtogroup StabilizationModule Stabilization Module * @brief Stabilization PID loops in an airframe type independent manner * @note This object updates the @ref ActuatorDesired "Actuator Desired" based on the * PID loops on the @ref AttitudeDesired "Attitude Desired" and @ref AttitudeState "Attitude State" * @{ * * @file outerloop.c * @author The LibrePilot Project, http://www.librepilot.org Copyright (C) 2015. * The OpenPilot Team, http://www.openpilot.org Copyright (C) 2014. * @brief Attitude stabilization module. * * @see The GNU Public License (GPL) Version 3 * *****************************************************************************/ /* * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include // Private constants #define CALLBACK_PRIORITY CALLBACK_PRIORITY_REGULAR #define UPDATE_EXPECTED (1.0f / PIOS_SENSOR_RATE) #define UPDATE_MIN 1.0e-6f #define UPDATE_MAX 1.0f #define UPDATE_ALPHA 1.0e-2f // Private variables static DelayedCallbackInfo *callbackHandle; static AttitudeStateData attitude; static uint8_t previous_mode[AXES] = { 255, 255, 255, 255 }; static float gyro_filtered[3] = { 0, 0, 0 }; static PiOSDeltatimeConfig timeval; static bool pitchMin = false; static bool pitchMax = false; static bool rollMin = false; static bool rollMax = false; // Private functions static void stabilizationOuterloopTask(); static void GyroStateUpdatedCb(__attribute__((unused)) UAVObjEvent *ev); static void AttitudeStateUpdatedCb(__attribute__((unused)) UAVObjEvent *ev); void stabilizationOuterloopInit() { RateDesiredInitialize(); StabilizationDesiredInitialize(); AttitudeStateInitialize(); GyroStateInitialize(); StabilizationStatusInitialize(); FlightStatusInitialize(); ManualControlCommandInitialize(); PIOS_DELTATIME_Init(&timeval, UPDATE_EXPECTED, UPDATE_MIN, UPDATE_MAX, UPDATE_ALPHA); callbackHandle = PIOS_CALLBACKSCHEDULER_Create(&stabilizationOuterloopTask, CALLBACK_PRIORITY, CBTASK_PRIORITY, CALLBACKINFO_RUNNING_STABILIZATION0, STACK_SIZE_BYTES); GyroStateConnectCallback(GyroStateUpdatedCb); AttitudeStateConnectCallback(AttitudeStateUpdatedCb); } /** * WARNING! This callback executes with critical flight control priority every * time a gyroscope update happens do NOT put any time consuming calculations * in this loop unless they really have to execute with every gyro update */ static void stabilizationOuterloopTask() { AttitudeStateData attitudeState; RateDesiredData rateDesired; StabilizationDesiredData stabilizationDesired; StabilizationStatusOuterLoopData enabled; AttitudeStateGet(&attitudeState); StabilizationDesiredGet(&stabilizationDesired); RateDesiredGet(&rateDesired); StabilizationStatusOuterLoopGet(&enabled); float *stabilizationDesiredAxis = &stabilizationDesired.Roll; float *rateDesiredAxis = &rateDesired.Roll; int t; float dT = PIOS_DELTATIME_GetAverageSeconds(&timeval); StabilizationStatusOuterLoopOptions newThrustMode = StabilizationStatusOuterLoopToArray(enabled)[STABILIZATIONSTATUS_OUTERLOOP_THRUST]; bool reinit = (newThrustMode != previous_mode[STABILIZATIONSTATUS_OUTERLOOP_THRUST]); #ifndef PIOS_EXCLUDE_ADVANCED_FEATURES // Trigger a disable message to the alt hold on reinit to prevent that loop from running when not in use. if (reinit) { if (previous_mode[STABILIZATIONSTATUS_OUTERLOOP_THRUST] == STABILIZATIONSTATUS_OUTERLOOP_ALTITUDE || previous_mode[STABILIZATIONSTATUS_OUTERLOOP_THRUST] == STABILIZATIONSTATUS_OUTERLOOP_ALTITUDEVARIO) { if (newThrustMode != STABILIZATIONSTATUS_OUTERLOOP_ALTITUDE && newThrustMode != STABILIZATIONSTATUS_OUTERLOOP_ALTITUDEVARIO) { // disable the altvario velocity control loop stabilizationDisableAltitudeHold(); } } } #endif // update previous mode previous_mode[STABILIZATIONSTATUS_OUTERLOOP_THRUST] = newThrustMode; // calculate the thrust desired switch (newThrustMode) { #ifndef PIOS_EXCLUDE_ADVANCED_FEATURES case STABILIZATIONSTATUS_OUTERLOOP_ALTITUDE: rateDesiredAxis[STABILIZATIONSTATUS_OUTERLOOP_THRUST] = stabilizationAltitudeHold(stabilizationDesiredAxis[STABILIZATIONSTATUS_OUTERLOOP_THRUST], ALTITUDEHOLD, reinit); break; case STABILIZATIONSTATUS_OUTERLOOP_ALTITUDEVARIO: rateDesiredAxis[STABILIZATIONSTATUS_OUTERLOOP_THRUST] = stabilizationAltitudeHold(stabilizationDesiredAxis[STABILIZATIONSTATUS_OUTERLOOP_THRUST], ALTITUDEVARIO, reinit); break; #endif case STABILIZATIONSTATUS_OUTERLOOP_DIRECT: case STABILIZATIONSTATUS_OUTERLOOP_DIRECTWITHLIMITS: default: rateDesiredAxis[STABILIZATIONSTATUS_OUTERLOOP_THRUST] = stabilizationDesiredAxis[STABILIZATIONSTATUS_OUTERLOOP_THRUST]; break; } float local_error[3]; #if defined(PIOS_QUATERNION_STABILIZATION) if (stabSettings.settings.ForceRollPitchDuringYawTransition == STABILIZATIONSETTINGS_FORCEROLLPITCHDURINGYAWTRANSITION_FALSE) { // Quaternion calculation of error in each axis. Uses more memory. float rpy_desired[3]; float q_desired[4]; float q_error[4]; for (t = 0; t < 3; t++) { switch (StabilizationStatusOuterLoopToArray(enabled)[t]) { case STABILIZATIONSTATUS_OUTERLOOP_ATTITUDE: case STABILIZATIONSTATUS_OUTERLOOP_RATTITUDE: case STABILIZATIONSTATUS_OUTERLOOP_WEAKLEVELING: rpy_desired[t] = stabilizationDesiredAxis[t]; break; case STABILIZATIONSTATUS_OUTERLOOP_DIRECTWITHLIMITS: case STABILIZATIONSTATUS_OUTERLOOP_DIRECT: default: rpy_desired[t] = ((float *)&attitudeState.Roll)[t]; break; } } RPY2Quaternion(rpy_desired, q_desired); quat_inverse(q_desired); quat_mult(q_desired, &attitudeState.q1, q_error); quat_inverse(q_error); Quaternion2RPY(q_error, local_error); } else { #else /* if defined(PIOS_QUATERNION_STABILIZATION) */ { #endif /* if defined(PIOS_QUATERNION_STABILIZATION) */ // Simpler algorithm for CC, less memory local_error[0] = stabilizationDesiredAxis[0] - attitudeState.Roll; local_error[1] = stabilizationDesiredAxis[1] - attitudeState.Pitch; local_error[2] = stabilizationDesiredAxis[2] - attitudeState.Yaw; // find shortest way float modulo = fmodf(local_error[2] + 180.0f, 360.0f); if (modulo < 0) { local_error[2] = modulo + 180.0f; } else { local_error[2] = modulo - 180.0f; } } // Feed forward: Assume things always get worse before they get better local_error[0] = local_error[0] - (gyro_filtered[0] * stabSettings.stabBank.AttitudeFeedForward.Roll); local_error[1] = local_error[1] - (gyro_filtered[1] * stabSettings.stabBank.AttitudeFeedForward.Pitch); local_error[2] = local_error[2] - (gyro_filtered[2] * stabSettings.stabBank.AttitudeFeedForward.Yaw); for (t = STABILIZATIONSTATUS_OUTERLOOP_ROLL; t < STABILIZATIONSTATUS_OUTERLOOP_THRUST; t++) { reinit = (StabilizationStatusOuterLoopToArray(enabled)[t] != previous_mode[t]); previous_mode[t] = StabilizationStatusOuterLoopToArray(enabled)[t]; if (reinit) { stabSettings.outerPids[t].iAccumulator = 0; } switch (StabilizationStatusOuterLoopToArray(enabled)[t]) { case STABILIZATIONSTATUS_OUTERLOOP_ATTITUDE: rateDesiredAxis[t] = pid_apply(&stabSettings.outerPids[t], local_error[t], dT); break; case STABILIZATIONSTATUS_OUTERLOOP_RATTITUDE: { float stickinput[3]; stickinput[0] = boundf(stabilizationDesiredAxis[0] / stabSettings.stabBank.RollMax, -1.0f, 1.0f); stickinput[1] = boundf(stabilizationDesiredAxis[1] / stabSettings.stabBank.PitchMax, -1.0f, 1.0f); stickinput[2] = boundf(stabilizationDesiredAxis[2] / stabSettings.stabBank.YawMax, -1.0f, 1.0f); float rateDesiredAxisRate = stickinput[t] * StabilizationBankManualRateToArray(stabSettings.stabBank.ManualRate)[t]; // limit corrective rate to maximum rates to not give it overly large impact over manual rate when joined together rateDesiredAxis[t] = boundf(pid_apply(&stabSettings.outerPids[t], local_error[t], dT), -StabilizationBankManualRateToArray(stabSettings.stabBank.ManualRate)[t], StabilizationBankManualRateToArray(stabSettings.stabBank.ManualRate)[t] ); // Compute the weighted average rate desired // Using max() rather than sqrt() for cpu speed; // - this makes the stick region into a square; // - this is a feature! // - hold a roll angle and add just pitch without the stick sensitivity changing float magnitude = fabsf(stickinput[t]); if (t < 2) { magnitude = fmaxf(fabsf(stickinput[0]), fabsf(stickinput[1])); } // modify magnitude to move the Att to Rate transition to the place // specified by the user // we are looking for where the stick angle == transition angle // and the Att rate equals the Rate rate // that's where Rate x (1-StickAngle) [Attitude pulling down max X Ratt proportion] // == Rate x StickAngle [Rate pulling up according to stick angle] // * StickAngle [X Ratt proportion] // so 1-x == x*x or x*x+x-1=0 where xE(0,1) // (-1+-sqrt(1+4))/2 = (-1+sqrt(5))/2 // and quadratic formula says that is 0.618033989f // I tested 14.01 and came up with .61 without even remembering this number // I thought that moving the P,I, and maxangle terms around would change this value // and that I would have to take these into account, but varying // all P's and I's by factors of 1/2 to 2 didn't change it noticeably // and varying maxangle from 4 to 120 didn't either. // so for now I'm not taking these into account // while working with this, it occurred to me that Attitude mode, // set up with maxangle=190 would be similar to Ratt, and it is. #define STICK_VALUE_AT_MODE_TRANSITION 0.618033989f // the following assumes the transition would otherwise be at 0.618033989f // and THAT assumes that Att ramps up to max roll rate // when a small number of degrees off of where it should be // if below the transition angle (still in attitude mode) // '<=' instead of '<' keeps rattitude_mode_transition_stick_position==1.0 from causing DZ if (magnitude <= stabSettings.rattitude_mode_transition_stick_position) { magnitude *= STICK_VALUE_AT_MODE_TRANSITION / stabSettings.rattitude_mode_transition_stick_position; } else { magnitude = (magnitude - stabSettings.rattitude_mode_transition_stick_position) * (1.0f - STICK_VALUE_AT_MODE_TRANSITION) / (1.0f - stabSettings.rattitude_mode_transition_stick_position) + STICK_VALUE_AT_MODE_TRANSITION; } rateDesiredAxis[t] = (1.0f - magnitude) * rateDesiredAxis[t] + magnitude * rateDesiredAxisRate; } break; case STABILIZATIONSTATUS_OUTERLOOP_WEAKLEVELING: // FIXME: local_error[] is rate - attitude for Weak Leveling // The only ramifications are: // Weak Leveling Kp is off by a factor of 3 to 12 and may need a different default in GCS // Changing Rate mode max rate currently requires a change to Kp // That would be changed to Attitude mode max angle affecting Kp // Also does not take dT into account { float stickinput[3]; stickinput[0] = boundf(stabilizationDesiredAxis[0] / stabSettings.stabBank.RollMax, -1.0f, 1.0f); stickinput[1] = boundf(stabilizationDesiredAxis[1] / stabSettings.stabBank.PitchMax, -1.0f, 1.0f); stickinput[2] = boundf(stabilizationDesiredAxis[2] / stabSettings.stabBank.YawMax, -1.0f, 1.0f); float rate_input = stickinput[t] * StabilizationBankManualRateToArray(stabSettings.stabBank.ManualRate)[t]; float weak_leveling = local_error[t] * stabSettings.settings.WeakLevelingKp; weak_leveling = boundf(weak_leveling, -stabSettings.settings.MaxWeakLevelingRate, stabSettings.settings.MaxWeakLevelingRate); // Compute desired rate as input biased towards leveling rateDesiredAxis[t] = rate_input + weak_leveling; } break; case STABILIZATIONSTATUS_OUTERLOOP_DIRECTWITHLIMITS: rateDesiredAxis[t] = stabilizationDesiredAxis[t]; // default for all axes // now test limits for pitch and/or roll if (t == 1) { // pitch if ((attitudeState.Pitch < -stabSettings.stabBank.PitchMax) || pitchMin) { pitchMin = true; // Attitude exceeds pitch min, // Do Attitude stabilisation at min pitch angle while user still maintain negative pitch if (stabilizationDesiredAxis[t] < 0.0f) { local_error[t] = -stabSettings.stabBank.PitchMax - attitudeState.Pitch; rateDesiredAxis[t] = pid_apply(&stabSettings.outerPids[t], local_error[t], dT); } else { // Stop Attitude stabilization and return to Rate pitchMin = false; } } else if ((attitudeState.Pitch > stabSettings.stabBank.PitchMax) || pitchMax) { pitchMax = true; // Attitude exceeds pitch max // Do Attitude stabilisation at max pitch angle while user still maintain positive pitch if (stabilizationDesiredAxis[t] > 0.0f) { local_error[t] = stabSettings.stabBank.PitchMax - attitudeState.Pitch; rateDesiredAxis[t] = pid_apply(&stabSettings.outerPids[t], local_error[t], dT); } else { // Stop Attitude stabilization and return to Rate pitchMax = false; } } } else if (t == 0) { // roll if ((attitudeState.Roll < -stabSettings.stabBank.RollMax) || rollMin) { rollMin = true; // Attitude exceeds roll min, // Do Attitude stabilisation at min roll angle while user still maintain negative roll if (stabilizationDesiredAxis[t] < 0.0f) { local_error[t] = -stabSettings.stabBank.RollMax - attitudeState.Roll; rateDesiredAxis[t] = pid_apply(&stabSettings.outerPids[t], local_error[t], dT); } else { // Stop Attitude stabilization and return to Rate rollMin = false; } } else if ((attitudeState.Roll > stabSettings.stabBank.RollMax) || rollMax) { rollMax = true; // Attitude exceeds roll max // Do Attitude stabilisation at max roll angle while user still maintain positive roll if (stabilizationDesiredAxis[t] > 0.0f) { local_error[t] = stabSettings.stabBank.RollMax - attitudeState.Roll; rateDesiredAxis[t] = pid_apply(&stabSettings.outerPids[t], local_error[t], dT); } else { // Stop Attitude stabilization and return to Rate rollMax = false; } } } break; case STABILIZATIONSTATUS_OUTERLOOP_DIRECT: default: rateDesiredAxis[t] = stabilizationDesiredAxis[t]; break; } } RateDesiredSet(&rateDesired); { FlightStatusArmedOptions armed; FlightStatusArmedGet(&armed); float throttleDesired; ManualControlCommandThrottleGet(&throttleDesired); if (armed != FLIGHTSTATUS_ARMED_ARMED || ((stabSettings.settings.LowThrottleZeroIntegral == STABILIZATIONSETTINGS_LOWTHROTTLEZEROINTEGRAL_TRUE) && throttleDesired < 0)) { // Force all axes to reinitialize when engaged for (t = 0; t < AXES; t++) { previous_mode[t] = 255; } } } // update cruisecontrol based on attitude cruisecontrol_compute_factor(&attitudeState, rateDesired.Thrust); stabSettings.monitor.rateupdates = 0; } static void AttitudeStateUpdatedCb(__attribute__((unused)) UAVObjEvent *ev) { #ifndef STABILIZATION_ATTITUDE_DOWNSAMPLED // to reduce CPU utilization, outer loop is not executed on every state update static uint8_t cpusaver = 0; if ((cpusaver++ % OUTERLOOP_SKIPCOUNT) == 0) { #endif // this does not need mutex protection as both eventdispatcher and stabi run in same callback task! AttitudeStateGet(&attitude); PIOS_CALLBACKSCHEDULER_Dispatch(callbackHandle); #ifndef STABILIZATION_ATTITUDE_DOWNSAMPLED } #endif } static void GyroStateUpdatedCb(__attribute__((unused)) UAVObjEvent *ev) { GyroStateData gyroState; GyroStateGet(&gyroState); gyro_filtered[0] = gyro_filtered[0] * stabSettings.feedForward_alpha[0] + gyroState.x * (1 - stabSettings.feedForward_alpha[0]); gyro_filtered[1] = gyro_filtered[1] * stabSettings.feedForward_alpha[1] + gyroState.y * (1 - stabSettings.feedForward_alpha[1]); gyro_filtered[2] = gyro_filtered[2] * stabSettings.feedForward_alpha[2] + gyroState.z * (1 - stabSettings.feedForward_alpha[2]); } /** * @} * @} */