/**************************************************************************** * * (c) 2009-2016 QGROUNDCONTROL PROJECT * * QGroundControl is licensed according to the terms in the file * COPYING.md in the root of the source code directory. * ****************************************************************************/ #include "MockLink.h" #include "QGCLoggingCategory.h" #include "QGCApplication.h" #ifdef UNITTEST_BUILD #include "UnitTest.h" #endif #include #include #include #include // FIXME: Hack to work around clean headers #include "FirmwarePlugin/PX4/px4_custom_mode.h" QGC_LOGGING_CATEGORY(MockLinkLog, "MockLinkLog") QGC_LOGGING_CATEGORY(MockLinkVerboseLog, "MockLinkVerboseLog") /// @file /// @brief Mock implementation of a Link. /// /// @author Don Gagne float MockLink::_vehicleLatitude = 47.633033f; float MockLink::_vehicleLongitude = -122.08794f; float MockLink::_vehicleAltitude = 3.5f; int MockLink::_nextVehicleSystemId = 128; const char* MockLink::_failParam = "COM_FLTMODE6"; const char* MockConfiguration::_firmwareTypeKey = "FirmwareType"; const char* MockConfiguration::_vehicleTypeKey = "VehicleType"; const char* MockConfiguration::_sendStatusTextKey = "SendStatusText"; const char* MockConfiguration::_failureModeKey = "FailureMode"; MockLink::MockLink(MockConfiguration* config) : _missionItemHandler(this, qgcApp()->toolbox()->mavlinkProtocol()) , _name("MockLink") , _connected(false) , _vehicleSystemId(_nextVehicleSystemId++) , _vehicleComponentId(200) , _inNSH(false) , _mavlinkStarted(true) , _mavBaseMode(MAV_MODE_FLAG_MANUAL_INPUT_ENABLED | MAV_MODE_FLAG_CUSTOM_MODE_ENABLED) , _mavState(MAV_STATE_STANDBY) , _firmwareType(MAV_AUTOPILOT_PX4) , _vehicleType(MAV_TYPE_QUADROTOR) , _fileServer(NULL) , _sendStatusText(false) , _apmSendHomePositionOnEmptyList(false) , _failureMode(MockConfiguration::FailNone) , _sendHomePositionDelayCount(10) // No home position for 4 seconds , _sendGPSPositionDelayCount(100) // No gps lock for 5 seconds , _currentParamRequestListComponentIndex(-1) , _currentParamRequestListParamIndex(-1) , _logDownloadCurrentOffset(0) , _logDownloadBytesRemaining(0) { _config = config; if (_config) { _firmwareType = config->firmwareType(); _vehicleType = config->vehicleType(); _sendStatusText = config->sendStatusText(); _failureMode = config->failureMode(); _config->setLink(this); } union px4_custom_mode px4_cm; px4_cm.data = 0; px4_cm.main_mode = PX4_CUSTOM_MAIN_MODE_MANUAL; _mavCustomMode = px4_cm.data; _fileServer = new MockLinkFileServer(_vehicleSystemId, _vehicleComponentId, this); Q_CHECK_PTR(_fileServer); moveToThread(this); _loadParams(); } MockLink::~MockLink(void) { _disconnect(); if (!_logDownloadFilename.isEmpty()) { QFile::remove(_logDownloadFilename); } } bool MockLink::_connect(void) { if (!_connected) { _connected = true; start(); emit connected(); } return true; } void MockLink::_disconnect(void) { if (_connected) { _connected = false; quit(); wait(); emit disconnected(); } } void MockLink::run(void) { QTimer timer1HzTasks; QTimer timer10HzTasks; QTimer timer500HzTasks; QObject::connect(&timer1HzTasks, &QTimer::timeout, this, &MockLink::_run1HzTasks); QObject::connect(&timer10HzTasks, &QTimer::timeout, this, &MockLink::_run10HzTasks); QObject::connect(&timer500HzTasks, &QTimer::timeout, this, &MockLink::_run500HzTasks); timer1HzTasks.start(1000); timer10HzTasks.start(100); timer500HzTasks.start(2); exec(); QObject::disconnect(&timer1HzTasks, &QTimer::timeout, this, &MockLink::_run1HzTasks); QObject::disconnect(&timer10HzTasks, &QTimer::timeout, this, &MockLink::_run10HzTasks); QObject::disconnect(&timer500HzTasks, &QTimer::timeout, this, &MockLink::_run500HzTasks); _missionItemHandler.shutdown(); } void MockLink::_run1HzTasks(void) { if (_mavlinkStarted && _connected) { _sendVibration(); if (!qgcApp()->runningUnitTests()) { // Sending RC Channels during unit test breaks RC tests which does it's own RC simulation _sendRCChannels(); } if (_sendHomePositionDelayCount > 0) { // We delay home position for better testing _sendHomePositionDelayCount--; } else { _sendHomePosition(); } if (_sendStatusText) { _sendStatusText = false; _sendStatusTextMessages(); } } } void MockLink::_run10HzTasks(void) { if (_mavlinkStarted && _connected) { _sendHeartBeat(); if (_sendGPSPositionDelayCount > 0) { // We delay gps position for better testing _sendGPSPositionDelayCount--; } else { _sendGpsRawInt(); } } } void MockLink::_run500HzTasks(void) { if (_mavlinkStarted && _connected) { _paramRequestListWorker(); _logDownloadWorker(); } } void MockLink::_loadParams(void) { QFile paramFile; if (_firmwareType == MAV_AUTOPILOT_ARDUPILOTMEGA) { if (_vehicleType == MAV_TYPE_FIXED_WING) { paramFile.setFileName(":/unittest/APMArduPlaneMockLink.params"); } else if (_vehicleType == MAV_TYPE_SUBMARINE ) { paramFile.setFileName(":/unittest/APMArduSubMockLink.params"); } else { paramFile.setFileName(":/unittest/APMArduCopterMockLink.params"); } } else { paramFile.setFileName(":/unittest/PX4MockLink.params"); } bool success = paramFile.open(QFile::ReadOnly); Q_UNUSED(success); Q_ASSERT(success); QTextStream paramStream(¶mFile); while (!paramStream.atEnd()) { QString line = paramStream.readLine(); if (line.startsWith("#")) { continue; } QStringList paramData = line.split("\t"); Q_ASSERT(paramData.count() == 5); int componentId = paramData.at(1).toInt(); QString paramName = paramData.at(2); QString valStr = paramData.at(3); uint paramType = paramData.at(4).toUInt(); QVariant paramValue; switch (paramType) { case MAV_PARAM_TYPE_REAL32: paramValue = QVariant(valStr.toFloat()); break; case MAV_PARAM_TYPE_UINT32: paramValue = QVariant(valStr.toUInt()); break; case MAV_PARAM_TYPE_INT32: paramValue = QVariant(valStr.toInt()); break; case MAV_PARAM_TYPE_UINT16: paramValue = QVariant((quint16)valStr.toUInt()); break; case MAV_PARAM_TYPE_INT16: paramValue = QVariant((qint16)valStr.toInt()); break; case MAV_PARAM_TYPE_UINT8: paramValue = QVariant((quint8)valStr.toUInt()); break; case MAV_PARAM_TYPE_INT8: paramValue = QVariant((qint8)valStr.toUInt()); break; default: qCritical() << "Unknown type" << paramType; paramValue = QVariant(valStr.toInt()); break; } qCDebug(MockLinkVerboseLog) << "Loading param" << paramName << paramValue; _mapParamName2Value[componentId][paramName] = paramValue; _mapParamName2MavParamType[paramName] = static_cast(paramType); } } void MockLink::_sendHeartBeat(void) { mavlink_message_t msg; mavlink_msg_heartbeat_pack_chan(_vehicleSystemId, _vehicleComponentId, mavlinkChannel(), &msg, _vehicleType, // MAV_TYPE _firmwareType, // MAV_AUTOPILOT _mavBaseMode, // MAV_MODE _mavCustomMode, // custom mode _mavState); // MAV_STATE respondWithMavlinkMessage(msg); } void MockLink::_sendVibration(void) { mavlink_message_t msg; mavlink_msg_vibration_pack_chan(_vehicleSystemId, _vehicleComponentId, mavlinkChannel(), &msg, 0, // time_usec 50.5, // vibration_x, 10.5, // vibration_y, 60.0, // vibration_z, 1, // clipping_0 2, // clipping_0 3); // clipping_0 respondWithMavlinkMessage(msg); } void MockLink::respondWithMavlinkMessage(const mavlink_message_t& msg) { uint8_t buffer[MAVLINK_MAX_PACKET_LEN]; int cBuffer = mavlink_msg_to_send_buffer(buffer, &msg); QByteArray bytes((char *)buffer, cBuffer); emit bytesReceived(this, bytes); } /// @brief Called when QGC wants to write bytes to the MAV void MockLink::_writeBytes(const QByteArray bytes) { if (_inNSH) { _handleIncomingNSHBytes(bytes.constData(), bytes.count()); } else { if (bytes.startsWith(QByteArray("\r\r\r"))) { _inNSH = true; _handleIncomingNSHBytes(&bytes.constData()[3], bytes.count() - 3); } _handleIncomingMavlinkBytes((uint8_t *)bytes.constData(), bytes.count()); } } /// @brief Handle incoming bytes which are meant to be interpreted by the NuttX shell void MockLink::_handleIncomingNSHBytes(const char* bytes, int cBytes) { Q_UNUSED(cBytes); // Drop back out of NSH if (cBytes == 4 && bytes[0] == '\r' && bytes[1] == '\r' && bytes[2] == '\r') { _inNSH = false; return; } if (cBytes > 0) { qDebug() << "NSH:" << (const char*)bytes; #if 0 // MockLink not quite ready to handle this correctly yet if (strncmp(bytes, "sh /etc/init.d/rc.usb\n", cBytes) == 0) { // This is the mavlink start command _mavlinkStarted = true; } #endif } } /// @brief Handle incoming bytes which are meant to be handled by the mavlink protocol void MockLink::_handleIncomingMavlinkBytes(const uint8_t* bytes, int cBytes) { mavlink_message_t msg; mavlink_status_t comm; for (qint64 i=0; i= cParameters) { // We've sent the last parameter for this component, move to next component if (++_currentParamRequestListComponentIndex >= _mapParamName2Value.keys().count()) { // We've finished sending the last parameter for the last component, request is complete _currentParamRequestListComponentIndex = -1; } else { _currentParamRequestListParamIndex = 0; } } } void MockLink::_handleParamSet(const mavlink_message_t& msg) { mavlink_param_set_t request; mavlink_msg_param_set_decode(&msg, &request); Q_ASSERT(request.target_system == _vehicleSystemId); int componentId = request.target_component; // Param may not be null terminated if exactly fits char paramId[MAVLINK_MSG_PARAM_SET_FIELD_PARAM_ID_LEN + 1]; paramId[MAVLINK_MSG_PARAM_SET_FIELD_PARAM_ID_LEN] = 0; strncpy(paramId, request.param_id, MAVLINK_MSG_PARAM_SET_FIELD_PARAM_ID_LEN); qCDebug(MockLinkLog) << "_handleParamSet" << componentId << paramId << request.param_type; Q_ASSERT(_mapParamName2Value.contains(componentId)); Q_ASSERT(_mapParamName2Value[componentId].contains(paramId)); Q_ASSERT(request.param_type == _mapParamName2MavParamType[paramId]); // Save the new value _setParamFloatUnionIntoMap(componentId, paramId, request.param_value); // Respond with a param_value to ack mavlink_message_t responseMsg; mavlink_msg_param_value_pack_chan(_vehicleSystemId, componentId, // component id mavlinkChannel(), &responseMsg, // Outgoing message paramId, // Parameter name request.param_value, // Send same value back request.param_type, // Send same type back _mapParamName2Value[componentId].count(), // Total number of parameters _mapParamName2Value[componentId].keys().indexOf(paramId)); // Index of this parameter respondWithMavlinkMessage(responseMsg); } void MockLink::_handleParamRequestRead(const mavlink_message_t& msg) { mavlink_message_t responseMsg; mavlink_param_request_read_t request; mavlink_msg_param_request_read_decode(&msg, &request); const QString paramName(QString::fromLocal8Bit(request.param_id, strnlen(request.param_id, MAVLINK_MSG_PARAM_REQUEST_READ_FIELD_PARAM_ID_LEN))); int componentId = request.target_component; // special case for magic _HASH_CHECK value if (request.target_component == MAV_COMP_ID_ALL && paramName == "_HASH_CHECK") { mavlink_param_union_t valueUnion; valueUnion.type = MAV_PARAM_TYPE_UINT32; valueUnion.param_uint32 = 0; // Special case of magic hash check value mavlink_msg_param_value_pack_chan(_vehicleSystemId, componentId, mavlinkChannel(), &responseMsg, request.param_id, valueUnion.param_float, MAV_PARAM_TYPE_UINT32, 0, -1); respondWithMavlinkMessage(responseMsg); return; } Q_ASSERT(_mapParamName2Value.contains(componentId)); char paramId[MAVLINK_MSG_PARAM_REQUEST_READ_FIELD_PARAM_ID_LEN + 1]; paramId[0] = 0; Q_ASSERT(request.target_system == _vehicleSystemId); if (request.param_index == -1) { // Request is by param name. Param may not be null terminated if exactly fits strncpy(paramId, request.param_id, MAVLINK_MSG_PARAM_REQUEST_READ_FIELD_PARAM_ID_LEN); } else { // Request is by index Q_ASSERT(request.param_index >= 0 && request.param_index < _mapParamName2Value[componentId].count()); QString key = _mapParamName2Value[componentId].keys().at(request.param_index); Q_ASSERT(key.length() <= MAVLINK_MSG_PARAM_REQUEST_READ_FIELD_PARAM_ID_LEN); strcpy(paramId, key.toLocal8Bit().constData()); } Q_ASSERT(_mapParamName2Value[componentId].contains(paramId)); Q_ASSERT(_mapParamName2MavParamType.contains(paramId)); if (_failureMode == MockConfiguration::FailMissingParamOnAllRequests && strcmp(paramId, _failParam) == 0) { qCDebug(MockLinkLog) << "Ignoring request read for " << _failParam; // Fail to send this param no matter what return; } mavlink_msg_param_value_pack_chan(_vehicleSystemId, componentId, // component id mavlinkChannel(), &responseMsg, // Outgoing message paramId, // Parameter name _floatUnionForParam(componentId, paramId), // Parameter value _mapParamName2MavParamType[paramId], // Parameter type _mapParamName2Value[componentId].count(), // Total number of parameters _mapParamName2Value[componentId].keys().indexOf(paramId)); // Index of this parameter respondWithMavlinkMessage(responseMsg); } void MockLink::emitRemoteControlChannelRawChanged(int channel, uint16_t raw) { uint16_t chanRaw[18]; for (int i=0; i<18; i++) { chanRaw[i] = UINT16_MAX; } chanRaw[channel] = raw; mavlink_message_t responseMsg; mavlink_msg_rc_channels_pack_chan(_vehicleSystemId, _vehicleComponentId, mavlinkChannel(), &responseMsg, // Outgoing message 0, // time since boot, ignored 18, // channel count chanRaw[0], // channel raw value chanRaw[1], // channel raw value chanRaw[2], // channel raw value chanRaw[3], // channel raw value chanRaw[4], // channel raw value chanRaw[5], // channel raw value chanRaw[6], // channel raw value chanRaw[7], // channel raw value chanRaw[8], // channel raw value chanRaw[9], // channel raw value chanRaw[10], // channel raw value chanRaw[11], // channel raw value chanRaw[12], // channel raw value chanRaw[13], // channel raw value chanRaw[14], // channel raw value chanRaw[15], // channel raw value chanRaw[16], // channel raw value chanRaw[17], // channel raw value 0); // rss respondWithMavlinkMessage(responseMsg); } void MockLink::_handleFTP(const mavlink_message_t& msg) { Q_ASSERT(_fileServer); _fileServer->handleFTPMessage(msg); } void MockLink::_handleCommandLong(const mavlink_message_t& msg) { mavlink_command_long_t request; uint8_t commandResult = MAV_RESULT_UNSUPPORTED; mavlink_msg_command_long_decode(&msg, &request); switch (request.command) { case MAV_CMD_COMPONENT_ARM_DISARM: if (request.param1 == 0.0f) { _mavBaseMode &= ~MAV_MODE_FLAG_SAFETY_ARMED; } else { _mavBaseMode |= MAV_MODE_FLAG_SAFETY_ARMED; } commandResult = MAV_RESULT_ACCEPTED; break; case MAV_CMD_PREFLIGHT_CALIBRATION: _handlePreFlightCalibration(request); commandResult = MAV_RESULT_ACCEPTED; break; case MAV_CMD_PREFLIGHT_STORAGE: commandResult = MAV_RESULT_ACCEPTED; break; case MAV_CMD_REQUEST_AUTOPILOT_CAPABILITIES: commandResult = MAV_RESULT_ACCEPTED; _respondWithAutopilotVersion(); break; case MAV_CMD_USER_1: // Test command which always returns MAV_RESULT_ACCEPTED commandResult = MAV_RESULT_ACCEPTED; break; case MAV_CMD_USER_2: // Test command which always returns MAV_RESULT_FAILED commandResult = MAV_RESULT_FAILED; break; case MAV_CMD_USER_3: // Test command which returns MAV_RESULT_ACCEPTED on second attempt static bool firstCmdUser3 = true; if (firstCmdUser3) { firstCmdUser3 = false; return; } else { firstCmdUser3 = true; commandResult = MAV_RESULT_ACCEPTED; } break; case MAV_CMD_USER_4: // Test command which returns MAV_RESULT_FAILED on second attempt static bool firstCmdUser4 = true; if (firstCmdUser4) { firstCmdUser4 = false; return; } else { firstCmdUser4 = true; commandResult = MAV_RESULT_FAILED; } break; case MAV_CMD_USER_5: // No response return; break; } mavlink_message_t commandAck; mavlink_msg_command_ack_pack_chan(_vehicleSystemId, _vehicleComponentId, mavlinkChannel(), &commandAck, request.command, commandResult); respondWithMavlinkMessage(commandAck); } void MockLink::_respondWithAutopilotVersion(void) { mavlink_message_t msg; uint8_t customVersion[8] = { }; uint32_t flightVersion = 0; if (_firmwareType == MAV_AUTOPILOT_ARDUPILOTMEGA) { flightVersion |= 3 << (8*3); flightVersion |= 3 << (8*2); flightVersion |= 0 << (8*1); flightVersion |= FIRMWARE_VERSION_TYPE_DEV << (8*0); } else if (_firmwareType == MAV_AUTOPILOT_PX4) { flightVersion |= 1 << (8*3); flightVersion |= 4 << (8*2); flightVersion |= 1 << (8*1); flightVersion |= FIRMWARE_VERSION_TYPE_DEV << (8*0); } mavlink_msg_autopilot_version_pack_chan(_vehicleSystemId, _vehicleComponentId, mavlinkChannel(), &msg, 0, // capabilities, flightVersion, // flight_sw_version, 0, // middleware_sw_version, 0, // os_sw_version, 0, // board_version, (uint8_t *)&customVersion, // flight_custom_version, (uint8_t *)&customVersion, // middleware_custom_version, (uint8_t *)&customVersion, // os_custom_version, 0, // vendor_id, 0, // product_id, 0); // uid respondWithMavlinkMessage(msg); } void MockLink::setMissionItemFailureMode(MockLinkMissionItemHandler::FailureMode_t failureMode) { _missionItemHandler.setMissionItemFailureMode(failureMode); } void MockLink::_sendHomePosition(void) { mavlink_message_t msg; float bogus[4]; bogus[0] = 0.0f; bogus[1] = 0.0f; bogus[2] = 0.0f; bogus[3] = 0.0f; mavlink_msg_home_position_pack_chan(_vehicleSystemId, _vehicleComponentId, mavlinkChannel(), &msg, (int32_t)(_vehicleLatitude * 1E7), (int32_t)(_vehicleLongitude * 1E7), (int32_t)(_vehicleAltitude * 1000), 0.0f, 0.0f, 0.0f, &bogus[0], 0.0f, 0.0f, 0.0f); respondWithMavlinkMessage(msg); } void MockLink::_sendGpsRawInt(void) { static uint64_t timeTick = 0; mavlink_message_t msg; mavlink_msg_gps_raw_int_pack_chan(_vehicleSystemId, _vehicleComponentId, mavlinkChannel(), &msg, timeTick++, // time since boot 3, // 3D fix (int32_t)(_vehicleLatitude * 1E7), (int32_t)(_vehicleLongitude * 1E7), (int32_t)(_vehicleAltitude * 1000), UINT16_MAX, UINT16_MAX, // HDOP/VDOP not known UINT16_MAX, // velocity not known UINT16_MAX, // course over ground not known 8); // satellite count respondWithMavlinkMessage(msg); } void MockLink::_sendStatusTextMessages(void) { struct StatusMessage { MAV_SEVERITY severity; const char* msg; }; static const struct StatusMessage rgMessages[] = { { MAV_SEVERITY_INFO, "#Testing audio output" }, { MAV_SEVERITY_EMERGENCY, "Status text emergency" }, { MAV_SEVERITY_ALERT, "Status text alert" }, { MAV_SEVERITY_CRITICAL, "Status text critical" }, { MAV_SEVERITY_ERROR, "Status text error" }, { MAV_SEVERITY_WARNING, "Status text warning" }, { MAV_SEVERITY_NOTICE, "Status text notice" }, { MAV_SEVERITY_INFO, "Status text info" }, { MAV_SEVERITY_DEBUG, "Status text debug" }, }; for (size_t i=0; iseverity, status->msg); respondWithMavlinkMessage(msg); } } MockConfiguration::MockConfiguration(const QString& name) : LinkConfiguration(name) , _firmwareType(MAV_AUTOPILOT_PX4) , _vehicleType(MAV_TYPE_QUADROTOR) , _sendStatusText(false) , _failureMode(FailNone) { } MockConfiguration::MockConfiguration(MockConfiguration* source) : LinkConfiguration(source) { _firmwareType = source->_firmwareType; _vehicleType = source->_vehicleType; _sendStatusText = source->_sendStatusText; _failureMode = source->_failureMode; } void MockConfiguration::copyFrom(LinkConfiguration *source) { LinkConfiguration::copyFrom(source); MockConfiguration* usource = dynamic_cast(source); if (!usource) { qWarning() << "dynamic_cast failed" << source << usource; return; } _firmwareType = usource->_firmwareType; _vehicleType = usource->_vehicleType; _sendStatusText = usource->_sendStatusText; _failureMode = usource->_failureMode; } void MockConfiguration::saveSettings(QSettings& settings, const QString& root) { settings.beginGroup(root); settings.setValue(_firmwareTypeKey, (int)_firmwareType); settings.setValue(_vehicleTypeKey, (int)_vehicleType); settings.setValue(_sendStatusTextKey, _sendStatusText); settings.setValue(_failureModeKey, (int)_failureMode); settings.sync(); settings.endGroup(); } void MockConfiguration::loadSettings(QSettings& settings, const QString& root) { settings.beginGroup(root); _firmwareType = (MAV_AUTOPILOT)settings.value(_firmwareTypeKey, (int)MAV_AUTOPILOT_PX4).toInt(); _vehicleType = (MAV_TYPE)settings.value(_vehicleTypeKey, (int)MAV_TYPE_QUADROTOR).toInt(); _sendStatusText = settings.value(_sendStatusTextKey, false).toBool(); _failureMode = (FailureMode_t)settings.value(_failureModeKey, (int)FailNone).toInt(); settings.endGroup(); } void MockConfiguration::updateSettings() { if (_link) { MockLink* ulink = dynamic_cast(_link); if (ulink) { // Restart connect not supported qWarning() << "updateSettings not supported"; //ulink->_restartConnection(); } } } MockLink* MockLink::_startMockLink(MockConfiguration* mockConfig) { LinkManager* linkManager = qgcApp()->toolbox()->linkManager(); mockConfig->setDynamic(true); linkManager->linkConfigurations()->append(mockConfig); return qobject_cast(linkManager->createConnectedLink(mockConfig)); } MockLink* MockLink::startPX4MockLink(bool sendStatusText, MockConfiguration::FailureMode_t failureMode) { MockConfiguration* mockConfig = new MockConfiguration("PX4 MockLink"); mockConfig->setFirmwareType(MAV_AUTOPILOT_PX4); mockConfig->setVehicleType(MAV_TYPE_QUADROTOR); mockConfig->setSendStatusText(sendStatusText); mockConfig->setFailureMode(failureMode); return _startMockLink(mockConfig); } MockLink* MockLink::startGenericMockLink(bool sendStatusText, MockConfiguration::FailureMode_t failureMode) { MockConfiguration* mockConfig = new MockConfiguration("Generic MockLink"); mockConfig->setFirmwareType(MAV_AUTOPILOT_GENERIC); mockConfig->setVehicleType(MAV_TYPE_QUADROTOR); mockConfig->setSendStatusText(sendStatusText); mockConfig->setFailureMode(failureMode); return _startMockLink(mockConfig); } MockLink* MockLink::startAPMArduCopterMockLink(bool sendStatusText, MockConfiguration::FailureMode_t failureMode) { MockConfiguration* mockConfig = new MockConfiguration("APM ArduCopter MockLink"); mockConfig->setFirmwareType(MAV_AUTOPILOT_ARDUPILOTMEGA); mockConfig->setVehicleType(MAV_TYPE_QUADROTOR); mockConfig->setSendStatusText(sendStatusText); mockConfig->setFailureMode(failureMode); return _startMockLink(mockConfig); } MockLink* MockLink::startAPMArduPlaneMockLink(bool sendStatusText, MockConfiguration::FailureMode_t failureMode) { MockConfiguration* mockConfig = new MockConfiguration("APM ArduPlane MockLink"); mockConfig->setFirmwareType(MAV_AUTOPILOT_ARDUPILOTMEGA); mockConfig->setVehicleType(MAV_TYPE_FIXED_WING); mockConfig->setSendStatusText(sendStatusText); mockConfig->setFailureMode(failureMode); return _startMockLink(mockConfig); } MockLink* MockLink::startAPMArduSubMockLink(bool sendStatusText, MockConfiguration::FailureMode_t failureMode) { MockConfiguration* mockConfig = new MockConfiguration("APM ArduSub MockLink"); mockConfig->setFirmwareType(MAV_AUTOPILOT_ARDUPILOTMEGA); mockConfig->setVehicleType(MAV_TYPE_SUBMARINE); mockConfig->setSendStatusText(sendStatusText); mockConfig->setFailureMode(failureMode); return _startMockLink(mockConfig); } void MockLink::_sendRCChannels(void) { mavlink_message_t msg; mavlink_msg_rc_channels_pack_chan(_vehicleSystemId, _vehicleComponentId, mavlinkChannel(), &msg, 0, // time_boot_ms 8, // chancount 1500, // chan1_raw 1500, // chan2_raw 1500, // chan3_raw 1500, // chan4_raw 1500, // chan5_raw 1500, // chan6_raw 1500, // chan7_raw 1500, // chan8_raw UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, UINT16_MAX, 0); // rssi respondWithMavlinkMessage(msg); } void MockLink::_handlePreFlightCalibration(const mavlink_command_long_t& request) { const char* pCalMessage; static const char* gyroCalResponse = "[cal] calibration started: 2 gyro"; static const char* magCalResponse = "[cal] calibration started: 2 mag"; static const char* accelCalResponse = "[cal] calibration started: 2 accel"; if (request.param1 == 1) { // Gyro cal pCalMessage = gyroCalResponse; } else if (request.param2 == 1) { // Mag cal pCalMessage = magCalResponse; } else if (request.param5 == 1) { // Accel cal pCalMessage = accelCalResponse; } else { return; } mavlink_message_t msg; mavlink_msg_statustext_pack_chan(_vehicleSystemId, _vehicleComponentId, mavlinkChannel(), &msg, MAV_SEVERITY_INFO, pCalMessage); respondWithMavlinkMessage(msg); } void MockLink::_handleLogRequestList(const mavlink_message_t& msg) { mavlink_log_request_list_t request; mavlink_msg_log_request_list_decode(&msg, &request); if (request.start != 0 && request.end != 0xffff) { qWarning() << "MockLink::_handleLogRequestList cannot handle partial requests"; return; } mavlink_message_t responseMsg; mavlink_msg_log_entry_pack_chan(_vehicleSystemId, _vehicleComponentId, mavlinkChannel(), &responseMsg, _logDownloadLogId, // log id 1, // num_logs 1, // last_log_num 0, // time_utc _logDownloadFileSize); // size respondWithMavlinkMessage(responseMsg); } void MockLink::_handleLogRequestData(const mavlink_message_t& msg) { mavlink_log_request_data_t request; mavlink_msg_log_request_data_decode(&msg, &request); if (_logDownloadFilename.isEmpty()) { #ifdef UNITTEST_BUILD _logDownloadFilename = UnitTest::createRandomFile(_logDownloadFileSize); #endif } if (request.id != 0) { qWarning() << "MockLink::_handleLogRequestData id must be 0"; return; } if (request.ofs > _logDownloadFileSize - 1) { qWarning() << "MockLink::_handleLogRequestData offset past end of file request.ofs:size" << request.ofs << _logDownloadFileSize; return; } // This will trigger _logDownloadWorker to send data _logDownloadCurrentOffset = request.ofs; if (request.ofs + request.count > _logDownloadFileSize) { request.count = _logDownloadFileSize - request.ofs; } _logDownloadBytesRemaining = request.count; } void MockLink::_logDownloadWorker(void) { if (_logDownloadBytesRemaining != 0) { QFile file(_logDownloadFilename); if (file.open(QIODevice::ReadOnly)) { uint8_t buffer[MAVLINK_MSG_LOG_DATA_FIELD_DATA_LEN]; qint64 bytesToRead = qMin(_logDownloadBytesRemaining, (uint32_t)MAVLINK_MSG_LOG_DATA_FIELD_DATA_LEN); Q_ASSERT(file.seek(_logDownloadCurrentOffset)); Q_ASSERT(file.read((char *)buffer, bytesToRead) == bytesToRead); qDebug() << "MockLink::_logDownloadWorker" << _logDownloadCurrentOffset << _logDownloadBytesRemaining; mavlink_message_t responseMsg; mavlink_msg_log_data_pack_chan(_vehicleSystemId, _vehicleComponentId, mavlinkChannel(), &responseMsg, _logDownloadLogId, _logDownloadCurrentOffset, bytesToRead, &buffer[0]); respondWithMavlinkMessage(responseMsg); _logDownloadCurrentOffset += bytesToRead; _logDownloadBytesRemaining -= bytesToRead; file.close(); } else { qWarning() << "MockLink::_logDownloadWorker open failed" << file.errorString(); } } }