/** ****************************************************************************** * * @file pathfollower.c * @author The LibrePilot Project, http://www.librepilot.org Copyright (C) 2016. * The OpenPilot Team, http://www.openpilot.org Copyright (C) 2015. * @brief This module compared @ref PositionActuatl to @ref ActiveWaypoint * and sets @ref AttitudeDesired. It only does this when the FlightMode field * of @ref ManualControlCommand is Auto. * * @see The GNU Public License (GPL) Version 3 * @addtogroup LibrePilot LibrePilotModules Modules PathFollower Navigation * *****************************************************************************/ /* * 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 */ /** * Input object: PathDesired * Input object: PositionState * Input object: ManualControlCommand * Output object: StabilizationDesired * * This module acts as "autopilot" - it controls the setpoints of stabilization * based on current flight situation and desired flight path (PathDesired) as * directed by flightmode selection or pathplanner * This is a periodic delayed callback module * * Modules have no API, all communication to other modules is done through UAVObjects. * However modules may use the API exposed by shared libraries. * See the OpenPilot wiki for more details. * http://www.openpilot.org/OpenPilot_Application_Architecture * */ extern "C" { #include #include #include #include #include #include #include #include #include "plans.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include } #include "pathfollowercontrol.h" #include "vtollandcontroller.h" #include "vtolautotakeoffcontroller.h" #include "vtolvelocitycontroller.h" #include "vtolbrakecontroller.h" #include "vtolflycontroller.h" #include "fixedwingflycontroller.h" #include "fixedwingautotakeoffcontroller.h" #include "fixedwinglandcontroller.h" #include "grounddrivecontroller.h" // Private constants #define CALLBACK_PRIORITY CALLBACK_PRIORITY_LOW #define CBTASK_PRIORITY CALLBACK_TASK_FLIGHTCONTROL #define PF_IDLE_UPDATE_RATE_MS 100 #define STACK_SIZE_BYTES 2048 // Private types // Private variables static DelayedCallbackInfo *pathFollowerCBInfo; static uint32_t updatePeriod = PF_IDLE_UPDATE_RATE_MS; static FrameType_t frameType = FRAME_TYPE_MULTIROTOR; static PathStatusData pathStatus; static PathDesiredData pathDesired; static FixedWingPathFollowerSettingsData fixedWingPathFollowerSettings; static GroundPathFollowerSettingsData groundPathFollowerSettings; static VtolPathFollowerSettingsData vtolPathFollowerSettings; static FlightStatusData flightStatus; static PathFollowerControl *activeController = 0; // Private functions static void pathFollowerTask(void); static void SettingsUpdatedCb(UAVObjEvent *ev); static void airspeedStateUpdatedCb(UAVObjEvent *ev); static void pathFollowerObjectiveUpdatedCb(UAVObjEvent *ev); static void flightStatusUpdatedCb(UAVObjEvent *ev); static void updateFixedAttitude(float *attitude); static void pathFollowerInitializeControllersForFrameType(); /** * Initialise the module, called on startup * \returns 0 on success or -1 if initialisation failed */ extern "C" int32_t PathFollowerStart() { // Start main task PathStatusGet(&pathStatus); SettingsUpdatedCb(NULL); PIOS_CALLBACKSCHEDULER_Dispatch(pathFollowerCBInfo); return 0; } /** * Initialise the module, called on startup * \returns 0 on success or -1 if initialisation failed */ extern "C" int32_t PathFollowerInitialize() { // initialize objects FixedWingPathFollowerStatusInitialize(); FlightStatusInitialize(); PathStatusInitialize(); PathSummaryInitialize(); PathDesiredInitialize(); PositionStateInitialize(); VelocityStateInitialize(); VelocityDesiredInitialize(); StabilizationDesiredInitialize(); AirspeedStateInitialize(); AttitudeStateInitialize(); PoiLocationInitialize(); ManualControlCommandInitialize(); StabilizationBankInitialize(); VtolSelfTuningStatsInitialize(); PIDStatusInitialize(); StatusVtolLandInitialize(); StatusGroundDriveInitialize(); StatusVtolAutoTakeoffInitialize(); // VtolLandFSM additional objects AccelStateInitialize(); // Init references to controllers PathFollowerControl::Initialize(&pathDesired, &flightStatus, &pathStatus); // Create object queue pathFollowerCBInfo = PIOS_CALLBACKSCHEDULER_Create(&pathFollowerTask, CALLBACK_PRIORITY, CBTASK_PRIORITY, CALLBACKINFO_RUNNING_PATHFOLLOWER, STACK_SIZE_BYTES); FixedWingPathFollowerSettingsConnectCallback(&SettingsUpdatedCb); GroundPathFollowerSettingsConnectCallback(&SettingsUpdatedCb); VtolPathFollowerSettingsConnectCallback(&SettingsUpdatedCb); PathDesiredConnectCallback(&pathFollowerObjectiveUpdatedCb); FlightStatusConnectCallback(&flightStatusUpdatedCb); SystemSettingsConnectCallback(&SettingsUpdatedCb); AirspeedStateConnectCallback(&airspeedStateUpdatedCb); VtolSelfTuningStatsConnectCallback(&SettingsUpdatedCb); return 0; } MODULE_INITCALL(PathFollowerInitialize, PathFollowerStart); void pathFollowerInitializeControllersForFrameType() { static uint8_t multirotor_initialised = 0; static uint8_t fixedwing_initialised = 0; static uint8_t ground_initialised = 0; switch (frameType) { case FRAME_TYPE_MULTIROTOR: case FRAME_TYPE_HELI: if (!multirotor_initialised) { VtolLandController::instance()->Initialize(&vtolPathFollowerSettings); VtolAutoTakeoffController::instance()->Initialize(&vtolPathFollowerSettings); VtolVelocityController::instance()->Initialize(&vtolPathFollowerSettings); VtolFlyController::instance()->Initialize(&vtolPathFollowerSettings); VtolBrakeController::instance()->Initialize(&vtolPathFollowerSettings); multirotor_initialised = 1; } break; case FRAME_TYPE_FIXED_WING: if (!fixedwing_initialised) { FixedWingFlyController::instance()->Initialize(&fixedWingPathFollowerSettings); FixedWingAutoTakeoffController::instance()->Initialize(&fixedWingPathFollowerSettings); FixedWingLandController::instance()->Initialize(&fixedWingPathFollowerSettings); fixedwing_initialised = 1; } break; case FRAME_TYPE_GROUND: if (!ground_initialised) { GroundDriveController::instance()->Initialize(&groundPathFollowerSettings); ground_initialised = 1; } break; default: break; } } static void pathFollowerSetActiveController(void) { // Init controllers for the frame type if (activeController == 0) { // Initialise pathFollowerInitializeControllersForFrameType(); AlarmsSet(SYSTEMALARMS_ALARM_GUIDANCE, SYSTEMALARMS_ALARM_OK); switch (frameType) { case FRAME_TYPE_MULTIROTOR: case FRAME_TYPE_HELI: switch (pathDesired.Mode) { case PATHDESIRED_MODE_BRAKE: // brake then hold sequence controller activeController = VtolBrakeController::instance(); activeController->Activate(); break; case PATHDESIRED_MODE_VELOCITY: // velocity roam controller activeController = VtolVelocityController::instance(); activeController->Activate(); break; case PATHDESIRED_MODE_GOTOENDPOINT: case PATHDESIRED_MODE_FOLLOWVECTOR: case PATHDESIRED_MODE_CIRCLERIGHT: case PATHDESIRED_MODE_CIRCLELEFT: activeController = VtolFlyController::instance(); activeController->Activate(); break; case PATHDESIRED_MODE_LAND: // land with optional velocity roam option activeController = VtolLandController::instance(); activeController->Activate(); break; case PATHDESIRED_MODE_AUTOTAKEOFF: activeController = VtolAutoTakeoffController::instance(); activeController->Activate(); break; default: activeController = 0; AlarmsSet(SYSTEMALARMS_ALARM_GUIDANCE, SYSTEMALARMS_ALARM_UNINITIALISED); break; } break; case FRAME_TYPE_FIXED_WING: switch (pathDesired.Mode) { case PATHDESIRED_MODE_GOTOENDPOINT: case PATHDESIRED_MODE_FOLLOWVECTOR: case PATHDESIRED_MODE_CIRCLERIGHT: case PATHDESIRED_MODE_CIRCLELEFT: activeController = FixedWingFlyController::instance(); activeController->Activate(); break; case PATHDESIRED_MODE_LAND: // land with optional velocity roam option activeController = FixedWingLandController::instance(); activeController->Activate(); break; case PATHDESIRED_MODE_AUTOTAKEOFF: activeController = FixedWingAutoTakeoffController::instance(); activeController->Activate(); break; default: activeController = 0; AlarmsSet(SYSTEMALARMS_ALARM_GUIDANCE, SYSTEMALARMS_ALARM_UNINITIALISED); break; } break; case FRAME_TYPE_GROUND: switch (pathDesired.Mode) { case PATHDESIRED_MODE_GOTOENDPOINT: case PATHDESIRED_MODE_FOLLOWVECTOR: case PATHDESIRED_MODE_CIRCLERIGHT: case PATHDESIRED_MODE_CIRCLELEFT: activeController = GroundDriveController::instance(); activeController->Activate(); break; default: activeController = 0; AlarmsSet(SYSTEMALARMS_ALARM_GUIDANCE, SYSTEMALARMS_ALARM_UNINITIALISED); break; } break; default: activeController = 0; AlarmsSet(SYSTEMALARMS_ALARM_GUIDANCE, SYSTEMALARMS_ALARM_UNINITIALISED); break; } } } /** * Module thread, should not return. */ static void pathFollowerTask(void) { FlightStatusGet(&flightStatus); if (flightStatus.ControlChain.PathFollower != FLIGHTSTATUS_CONTROLCHAIN_TRUE) { AlarmsSet(SYSTEMALARMS_ALARM_GUIDANCE, SYSTEMALARMS_ALARM_UNINITIALISED); PIOS_CALLBACKSCHEDULER_Schedule(pathFollowerCBInfo, PF_IDLE_UPDATE_RATE_MS, CALLBACK_UPDATEMODE_SOONER); return; } pathStatus.UID = pathDesired.UID; pathStatus.Status = PATHSTATUS_STATUS_INPROGRESS; pathFollowerSetActiveController(); if (activeController) { activeController->UpdateAutoPilot(); PIOS_CALLBACKSCHEDULER_Schedule(pathFollowerCBInfo, updatePeriod, CALLBACK_UPDATEMODE_SOONER); return; } switch (pathDesired.Mode) { case PATHDESIRED_MODE_FIXEDATTITUDE: updateFixedAttitude(pathDesired.ModeParameters); AlarmsSet(SYSTEMALARMS_ALARM_GUIDANCE, SYSTEMALARMS_ALARM_OK); break; case PATHDESIRED_MODE_DISARMALARM: AlarmsSet(SYSTEMALARMS_ALARM_GUIDANCE, SYSTEMALARMS_ALARM_CRITICAL); break; default: pathStatus.Status = PATHSTATUS_STATUS_CRITICAL; AlarmsSet(SYSTEMALARMS_ALARM_GUIDANCE, SYSTEMALARMS_ALARM_ERROR); break; } PathStatusSet(&pathStatus); PIOS_CALLBACKSCHEDULER_Schedule(pathFollowerCBInfo, updatePeriod, CALLBACK_UPDATEMODE_SOONER); } static void pathFollowerObjectiveUpdatedCb(__attribute__((unused)) UAVObjEvent *ev) { PathDesiredGet(&pathDesired); if (activeController && pathDesired.Mode != activeController->Mode()) { activeController->Deactivate(); activeController = 0; } pathFollowerSetActiveController(); if (activeController) { activeController->ObjectiveUpdated(); } } // we use this to deactivate active controllers as the pathdesired // objective never gets updated with switching to a non-pathfollower flight mode static void flightStatusUpdatedCb(__attribute__((unused)) UAVObjEvent *ev) { if (!activeController) { return; } FlightStatusControlChainData controlChain; FlightStatusControlChainGet(&controlChain); if (controlChain.PathFollower != FLIGHTSTATUS_CONTROLCHAIN_TRUE) { activeController->Deactivate(); activeController = 0; } } static void SettingsUpdatedCb(__attribute__((unused)) UAVObjEvent *ev) { VtolPathFollowerSettingsGet(&vtolPathFollowerSettings); FrameType_t previous_frameType = frameType; frameType = GetCurrentFrameType(); if (frameType == FRAME_TYPE_CUSTOM) { switch (vtolPathFollowerSettings.TreatCustomCraftAs) { case VTOLPATHFOLLOWERSETTINGS_TREATCUSTOMCRAFTAS_FIXEDWING: frameType = FRAME_TYPE_FIXED_WING; break; case VTOLPATHFOLLOWERSETTINGS_TREATCUSTOMCRAFTAS_VTOL: frameType = FRAME_TYPE_MULTIROTOR; break; case VTOLPATHFOLLOWERSETTINGS_TREATCUSTOMCRAFTAS_GROUND: frameType = FRAME_TYPE_GROUND; break; } } // If frame type changes, initialise the frame type controllers if (frameType != previous_frameType) { pathFollowerInitializeControllersForFrameType(); } switch (frameType) { case FRAME_TYPE_MULTIROTOR: case FRAME_TYPE_HELI: updatePeriod = vtolPathFollowerSettings.UpdatePeriod; break; case FRAME_TYPE_FIXED_WING: FixedWingPathFollowerSettingsGet(&fixedWingPathFollowerSettings); updatePeriod = fixedWingPathFollowerSettings.UpdatePeriod; break; case FRAME_TYPE_GROUND: GroundPathFollowerSettingsGet(&groundPathFollowerSettings); updatePeriod = groundPathFollowerSettings.UpdatePeriod; break; default: updatePeriod = vtolPathFollowerSettings.UpdatePeriod; break; } if (activeController) { activeController->SettingsUpdated(); } } static void airspeedStateUpdatedCb(__attribute__((unused)) UAVObjEvent *ev) { FixedWingFlyController::instance()->AirspeedStateUpdatedCb(ev); FixedWingAutoTakeoffController::instance()->AirspeedStateUpdatedCb(ev); } /** * Compute desired attitude from a fixed preset * */ static void updateFixedAttitude(float *attitude) { StabilizationDesiredData stabDesired; StabilizationDesiredGet(&stabDesired); stabDesired.Roll = attitude[0]; stabDesired.Pitch = attitude[1]; stabDesired.Yaw = attitude[2]; stabDesired.Thrust = attitude[3]; stabDesired.StabilizationMode.Roll = STABILIZATIONDESIRED_STABILIZATIONMODE_ATTITUDE; stabDesired.StabilizationMode.Pitch = STABILIZATIONDESIRED_STABILIZATIONMODE_ATTITUDE; stabDesired.StabilizationMode.Yaw = STABILIZATIONDESIRED_STABILIZATIONMODE_AXISLOCK; stabDesired.StabilizationMode.Thrust = STABILIZATIONDESIRED_STABILIZATIONMODE_MANUAL; StabilizationDesiredSet(&stabDesired); }