/*===================================================================== QGroundControl Open Source Ground Control Station (c) 2009 - 2014 QGROUNDCONTROL PROJECT This file is part of the QGROUNDCONTROL project QGROUNDCONTROL 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. QGROUNDCONTROL 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 QGROUNDCONTROL. If not, see . ======================================================================*/ #include "PX4AutoPilotPlugin.h" #include "AutoPilotPluginManager.h" #include "UASManager.h" #include "PX4ParameterLoader.h" #include "PX4AirframeLoader.h" #include "FlightModesComponentController.h" #include "AirframeComponentController.h" #include "QGCMessageBox.h" /// @file /// @brief This is the AutoPilotPlugin implementatin for the MAV_AUTOPILOT_PX4 type. /// @author Don Gagne enum PX4_CUSTOM_MAIN_MODE { PX4_CUSTOM_MAIN_MODE_MANUAL = 1, PX4_CUSTOM_MAIN_MODE_ALTCTL, PX4_CUSTOM_MAIN_MODE_POSCTL, PX4_CUSTOM_MAIN_MODE_AUTO, PX4_CUSTOM_MAIN_MODE_ACRO, PX4_CUSTOM_MAIN_MODE_OFFBOARD, PX4_CUSTOM_MAIN_MODE_STABILIZED, }; enum PX4_CUSTOM_SUB_MODE_AUTO { PX4_CUSTOM_SUB_MODE_AUTO_READY = 1, PX4_CUSTOM_SUB_MODE_AUTO_TAKEOFF, PX4_CUSTOM_SUB_MODE_AUTO_LOITER, PX4_CUSTOM_SUB_MODE_AUTO_MISSION, PX4_CUSTOM_SUB_MODE_AUTO_RTL, PX4_CUSTOM_SUB_MODE_AUTO_LAND, PX4_CUSTOM_SUB_MODE_AUTO_RTGS }; union px4_custom_mode { struct { uint16_t reserved; uint8_t main_mode; uint8_t sub_mode; }; uint32_t data; float data_float; }; PX4AutoPilotPlugin::PX4AutoPilotPlugin(UASInterface* uas, QObject* parent) : AutoPilotPlugin(uas, parent), _parameterFacts(NULL), _airframeComponent(NULL), _radioComponent(NULL), _flightModesComponent(NULL), _sensorsComponent(NULL), _safetyComponent(NULL), _powerComponent(NULL), _incorrectParameterVersion(false) { Q_ASSERT(uas); _parameterFacts = new PX4ParameterLoader(this, uas, this); Q_CHECK_PTR(_parameterFacts); connect(_parameterFacts, &PX4ParameterLoader::parametersReady, this, &PX4AutoPilotPlugin::_pluginReadyPreChecks); connect(_parameterFacts, &PX4ParameterLoader::parameterListProgress, this, &PX4AutoPilotPlugin::parameterListProgress); _airframeFacts = new PX4AirframeLoader(this, uas, this); Q_CHECK_PTR(_airframeFacts); PX4ParameterLoader::loadParameterFactMetaData(); PX4AirframeLoader::loadAirframeFactMetaData(); } PX4AutoPilotPlugin::~PX4AutoPilotPlugin() { delete _parameterFacts; delete _airframeFacts; } QList PX4AutoPilotPlugin::getModes(void) { union px4_custom_mode px4_cm; AutoPilotPluginManager::FullMode_t fullMode; QList modeList; px4_cm.data = 0; px4_cm.main_mode = PX4_CUSTOM_MAIN_MODE_MANUAL; fullMode.baseMode = MAV_MODE_FLAG_CUSTOM_MODE_ENABLED | MAV_MODE_FLAG_MANUAL_INPUT_ENABLED; fullMode.customMode = px4_cm.data; modeList << fullMode; px4_cm.data = 0; px4_cm.main_mode = PX4_CUSTOM_MAIN_MODE_ACRO; fullMode.baseMode = MAV_MODE_FLAG_CUSTOM_MODE_ENABLED | MAV_MODE_FLAG_MANUAL_INPUT_ENABLED; fullMode.customMode = px4_cm.data; modeList << fullMode; px4_cm.data = 0; px4_cm.main_mode = PX4_CUSTOM_MAIN_MODE_STABILIZED; fullMode.baseMode = MAV_MODE_FLAG_CUSTOM_MODE_ENABLED | MAV_MODE_FLAG_MANUAL_INPUT_ENABLED | MAV_MODE_FLAG_STABILIZE_ENABLED; fullMode.customMode = px4_cm.data; modeList << fullMode; px4_cm.data = 0; px4_cm.main_mode = PX4_CUSTOM_MAIN_MODE_ALTCTL; fullMode.baseMode = MAV_MODE_FLAG_CUSTOM_MODE_ENABLED | MAV_MODE_FLAG_MANUAL_INPUT_ENABLED | MAV_MODE_FLAG_STABILIZE_ENABLED; fullMode.customMode = px4_cm.data; modeList << fullMode; px4_cm.data = 0; px4_cm.main_mode = PX4_CUSTOM_MAIN_MODE_POSCTL; fullMode.baseMode = MAV_MODE_FLAG_CUSTOM_MODE_ENABLED | MAV_MODE_FLAG_MANUAL_INPUT_ENABLED | MAV_MODE_FLAG_STABILIZE_ENABLED | MAV_MODE_FLAG_GUIDED_ENABLED; fullMode.customMode = px4_cm.data; modeList << fullMode; px4_cm.data = 0; px4_cm.main_mode = PX4_CUSTOM_MAIN_MODE_AUTO; fullMode.baseMode = MAV_MODE_FLAG_CUSTOM_MODE_ENABLED | MAV_MODE_FLAG_AUTO_ENABLED | MAV_MODE_FLAG_STABILIZE_ENABLED | MAV_MODE_FLAG_GUIDED_ENABLED; fullMode.customMode = px4_cm.data; modeList << fullMode; px4_cm.data = 0; px4_cm.main_mode = PX4_CUSTOM_MAIN_MODE_OFFBOARD; fullMode.baseMode = MAV_MODE_FLAG_CUSTOM_MODE_ENABLED | MAV_MODE_FLAG_AUTO_ENABLED | MAV_MODE_FLAG_STABILIZE_ENABLED | MAV_MODE_FLAG_GUIDED_ENABLED; fullMode.customMode = px4_cm.data; modeList << fullMode; return modeList; } QString PX4AutoPilotPlugin::getAudioModeText(uint8_t baseMode, uint32_t customMode) { QString mode; Q_ASSERT(baseMode & MAV_MODE_FLAG_CUSTOM_MODE_ENABLED); if (baseMode & MAV_MODE_FLAG_CUSTOM_MODE_ENABLED) { union px4_custom_mode px4_mode; px4_mode.data = customMode; if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_MANUAL) { mode = "manual"; } else if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_ACRO) { mode = "caro"; } else if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_STABILIZED) { mode = "stabilized"; } else if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_ALTCTL) { mode = "altitude control"; } else if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_POSCTL) { mode = "position control"; } else if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_AUTO) { mode = "auto and "; if (px4_mode.sub_mode == PX4_CUSTOM_SUB_MODE_AUTO_READY) { mode += "ready"; } else if (px4_mode.sub_mode == PX4_CUSTOM_SUB_MODE_AUTO_TAKEOFF) { mode += "taking off"; } else if (px4_mode.sub_mode == PX4_CUSTOM_SUB_MODE_AUTO_LOITER) { mode += "loitering"; } else if (px4_mode.sub_mode == PX4_CUSTOM_SUB_MODE_AUTO_MISSION) { mode += "following waypoints"; } else if (px4_mode.sub_mode == PX4_CUSTOM_SUB_MODE_AUTO_RTL) { mode += "returning to land"; } else if (px4_mode.sub_mode == PX4_CUSTOM_SUB_MODE_AUTO_LAND) { mode += "landing"; } } else if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_OFFBOARD) { mode = "offboard controlled"; } if (baseMode != 0) { mode += " mode"; } // ARMED STATE DECODING if (baseMode & (uint8_t)MAV_MODE_FLAG_DECODE_POSITION_SAFETY) { mode.append(" and armed"); } } else { mode = ""; } return mode; } QString PX4AutoPilotPlugin::getShortModeText(uint8_t baseMode, uint32_t customMode) { QString mode; Q_ASSERT(baseMode & MAV_MODE_FLAG_CUSTOM_MODE_ENABLED); if (baseMode & MAV_MODE_FLAG_CUSTOM_MODE_ENABLED) { union px4_custom_mode px4_mode; px4_mode.data = customMode; if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_MANUAL) { mode = "|MANUAL"; } else if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_ACRO) { mode = "|ACRO"; } else if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_STABILIZED) { mode = "|STAB"; } else if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_ALTCTL) { mode = "|ALTCTL"; } else if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_POSCTL) { mode = "|POSCTL"; } else if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_AUTO) { mode = "|AUTO"; if (px4_mode.sub_mode == PX4_CUSTOM_SUB_MODE_AUTO_READY) { mode += "|READY"; } else if (px4_mode.sub_mode == PX4_CUSTOM_SUB_MODE_AUTO_TAKEOFF) { mode += "|TAKEOFF"; } else if (px4_mode.sub_mode == PX4_CUSTOM_SUB_MODE_AUTO_LOITER) { mode += "|LOITER"; } else if (px4_mode.sub_mode == PX4_CUSTOM_SUB_MODE_AUTO_MISSION) { mode += "|MISSION"; } else if (px4_mode.sub_mode == PX4_CUSTOM_SUB_MODE_AUTO_RTL) { mode += "|RTL"; } else if (px4_mode.sub_mode == PX4_CUSTOM_SUB_MODE_AUTO_LAND) { mode += "|LAND"; } } else if (px4_mode.main_mode == PX4_CUSTOM_MAIN_MODE_OFFBOARD) { mode = "|OFFBOARD"; } } else { // use base_mode - not autopilot-specific if (baseMode == 0) { mode = "|PREFLIGHT"; } else if (baseMode & MAV_MODE_FLAG_DECODE_POSITION_AUTO) { mode = "|AUTO"; } else if (baseMode & MAV_MODE_FLAG_DECODE_POSITION_MANUAL) { mode = "|MANUAL"; if (baseMode & MAV_MODE_FLAG_DECODE_POSITION_GUIDED) { mode += "|GUIDED"; } else if (baseMode & MAV_MODE_FLAG_DECODE_POSITION_STABILIZE) { mode += "|STABILIZED"; } } } return mode; } void PX4AutoPilotPlugin::clearStaticData(void) { PX4ParameterLoader::clearStaticData(); PX4AirframeLoader::clearStaticData(); } const QVariantList& PX4AutoPilotPlugin::vehicleComponents(void) { if (_components.count() == 0 && !_incorrectParameterVersion) { Q_ASSERT(_uas); _airframeComponent = new AirframeComponent(_uas, this); Q_CHECK_PTR(_airframeComponent); _airframeComponent->setupTriggerSignals(); _components.append(QVariant::fromValue((VehicleComponent*)_airframeComponent)); _radioComponent = new RadioComponent(_uas, this); Q_CHECK_PTR(_radioComponent); _radioComponent->setupTriggerSignals(); _components.append(QVariant::fromValue((VehicleComponent*)_radioComponent)); _flightModesComponent = new FlightModesComponent(_uas, this); Q_CHECK_PTR(_flightModesComponent); _flightModesComponent->setupTriggerSignals(); _components.append(QVariant::fromValue((VehicleComponent*)_flightModesComponent)); _sensorsComponent = new SensorsComponent(_uas, this); Q_CHECK_PTR(_sensorsComponent); _sensorsComponent->setupTriggerSignals(); _components.append(QVariant::fromValue((VehicleComponent*)_sensorsComponent)); _powerComponent = new PowerComponent(_uas, this); Q_CHECK_PTR(_powerComponent); _powerComponent->setupTriggerSignals(); _components.append(QVariant::fromValue((VehicleComponent*)_powerComponent)); _safetyComponent = new SafetyComponent(_uas, this); Q_CHECK_PTR(_safetyComponent); _safetyComponent->setupTriggerSignals(); _components.append(QVariant::fromValue((VehicleComponent*)_safetyComponent)); } return _components; } /// This will perform various checks prior to signalling that the plug in ready void PX4AutoPilotPlugin::_pluginReadyPreChecks(void) { // Check for older parameter version set // FIXME: Firmware is moving to version stamp parameter set. Once that is complete the version stamp // should be used instead. if (parameterExists(FactSystem::defaultComponentId, "SENS_GYRO_XOFF")) { _incorrectParameterVersion = true; QGCMessageBox::warning("Setup", "This version of GroundControl can only perform vehicle setup on a newer version of firmware. " "Please perform a Firmware Upgrade if you wish to use Vehicle Setup."); } _pluginReady = true; emit pluginReadyChanged(_pluginReady); }