MockLink.cc 52.8 KB
Newer Older
1
2
3
4
5
6
7
8
/****************************************************************************
 *
 *   (c) 2009-2016 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>
 *
 * QGroundControl is licensed according to the terms in the file
 * COPYING.md in the root of the source code directory.
 *
 ****************************************************************************/
9

Don Gagne's avatar
Don Gagne committed
10
11

#include "MockLink.h"
12
#include "QGCLoggingCategory.h"
13
#include "QGCApplication.h"
14
15

#ifdef UNITTEST_BUILD
DonLakeFlyer's avatar
DonLakeFlyer committed
16
#include "UnitTest.h"
17
#endif
Don Gagne's avatar
Don Gagne committed
18
19
20
21
22
23
24

#include <QTimer>
#include <QDebug>
#include <QFile>

#include <string.h>

25
26
// FIXME: Hack to work around clean headers
#include "FirmwarePlugin/PX4/px4_custom_mode.h"
Daniel Agar's avatar
Daniel Agar committed
27

28
QGC_LOGGING_CATEGORY(MockLinkLog, "MockLinkLog")
Don Gagne's avatar
Don Gagne committed
29
QGC_LOGGING_CATEGORY(MockLinkVerboseLog, "MockLinkVerboseLog")
30

Don Gagne's avatar
Don Gagne committed
31
32
33
34
35
/// @file
///     @brief Mock implementation of a Link.
///
///     @author Don Gagne <don@thegagnes.com>

36
37
// Vehicle position is set close to default Gazebo vehicle location. This allows for multi-vehicle
// testing of a gazebo vehicle and a mocklink vehicle
DonLakeFlyer's avatar
DonLakeFlyer committed
38
39
40
41
42
43
44
45
46
#if 1
double      MockLink::_defaultVehicleLatitude =     47.397;
double      MockLink::_defaultVehicleLongitude =    8.5455;
double      MockLink::_defaultVehicleAltitude =     488.056;
#else
double      MockLink::_defaultVehicleLatitude =     47.6333022928789;
double      MockLink::_defaultVehicleLongitude =    -122.08833157994995;
double      MockLink::_defaultVehicleAltitude =     19.0;
#endif
47
48
int         MockLink::_nextVehicleSystemId =        128;
const char* MockLink::_failParam =                  "COM_FLTMODE6";
Don Gagne's avatar
Don Gagne committed
49

50
const char* MockConfiguration::_firmwareTypeKey =   "FirmwareType";
51
const char* MockConfiguration::_vehicleTypeKey =    "VehicleType";
52
const char* MockConfiguration::_sendStatusTextKey = "SendStatusText";
Don Gagne's avatar
Don Gagne committed
53
const char* MockConfiguration::_highLatencyKey =    "HighLatency";
Don Gagne's avatar
Don Gagne committed
54
const char* MockConfiguration::_failureModeKey =    "FailureMode";
Don Gagne's avatar
Don Gagne committed
55

56
MockLink::MockLink(SharedLinkConfigurationPointer& config)
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
    : LinkInterface                         (config)
    , _missionItemHandler                   (this, qgcApp()->toolbox()->mavlinkProtocol())
    , _name                                 ("MockLink")
    , _connected                            (false)
    , _mavlinkChannel                       (0)
    , _vehicleSystemId                      (_nextVehicleSystemId++)
    , _vehicleComponentId                   (MAV_COMP_ID_AUTOPILOT1)
    , _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)
    , _vehicleLatitude                      (_defaultVehicleLatitude + ((_vehicleSystemId - 128) * 0.0001))     // Slight offset for each vehicle
    , _vehicleLongitude                     (_defaultVehicleLongitude + ((_vehicleSystemId - 128) * 0.0001))
    , _vehicleAltitude                      (_defaultVehicleAltitude)
    , _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
Don Gagne's avatar
Don Gagne committed
79
    , _currentParamRequestListComponentIndex(-1)
80
81
82
    , _currentParamRequestListParamIndex    (-1)
    , _logDownloadCurrentOffset             (0)
    , _logDownloadBytesRemaining            (0)
83
    , _adsbAngle                            (0)
Don Gagne's avatar
Don Gagne committed
84
{
85
86
87
88
    MockConfiguration* mockConfig = qobject_cast<MockConfiguration*>(_config.data());
    _firmwareType = mockConfig->firmwareType();
    _vehicleType = mockConfig->vehicleType();
    _sendStatusText = mockConfig->sendStatusText();
Don Gagne's avatar
Don Gagne committed
89
    _highLatency = mockConfig->highLatency();
90
    _failureMode = mockConfig->failureMode();
Don Gagne's avatar
Don Gagne committed
91

Don Gagne's avatar
Don Gagne committed
92
93
94
95
96
97
    union px4_custom_mode   px4_cm;

    px4_cm.data = 0;
    px4_cm.main_mode = PX4_CUSTOM_MAIN_MODE_MANUAL;
    _mavCustomMode = px4_cm.data;

Don Gagne's avatar
Don Gagne committed
98
99
    _fileServer = new MockLinkFileServer(_vehicleSystemId, _vehicleComponentId, this);
    Q_CHECK_PTR(_fileServer);
dogmaphobic's avatar
dogmaphobic committed
100

Don Gagne's avatar
Don Gagne committed
101
    moveToThread(this);
dogmaphobic's avatar
dogmaphobic committed
102

Don Gagne's avatar
Don Gagne committed
103
    _loadParams();
104
105
106

    _adsbVehicleCoordinate = QGeoCoordinate(_vehicleLatitude, _vehicleLongitude).atDistanceAndAzimuth(1000, _adsbAngle);
    _adsbVehicleCoordinate.setAltitude(100);
Don Gagne's avatar
Don Gagne committed
107
108
109
110
}

MockLink::~MockLink(void)
{
111
    _disconnect();
112
113
114
    if (!_logDownloadFilename.isEmpty()) {
        QFile::remove(_logDownloadFilename);
    }
Don Gagne's avatar
Don Gagne committed
115
116
}

117
bool MockLink::_connect(void)
Don Gagne's avatar
Don Gagne committed
118
{
119
120
    if (!_connected) {
        _connected = true;
Don Gagne's avatar
Don Gagne committed
121
122
123
124
125
126
127
128
        _mavlinkChannel = qgcApp()->toolbox()->linkManager()->_reserveMavlinkChannel();
        if (_mavlinkChannel == 0) {
            qWarning() << "No mavlink channels available";
            return false;
        }
        // MockLinks use Mavlink 2.0
        mavlink_status_t* mavlinkStatus = mavlink_get_channel_status(_mavlinkChannel);
        mavlinkStatus->flags &= ~MAVLINK_STATUS_FLAG_OUT_MAVLINK1;
129
130
131
        start();
        emit connected();
    }
132

Don Gagne's avatar
Don Gagne committed
133
134
135
    return true;
}

Don Gagne's avatar
Don Gagne committed
136
void MockLink::_disconnect(void)
Don Gagne's avatar
Don Gagne committed
137
{
138
    if (_connected) {
Don Gagne's avatar
Don Gagne committed
139
140
141
        if (_mavlinkChannel != 0) {
            qgcApp()->toolbox()->linkManager()->_freeMavlinkChannel(_mavlinkChannel);
        }
142
        _connected = false;
Daniel Agar's avatar
Daniel Agar committed
143
144
        quit();
        wait();
145
146
        emit disconnected();
    }
Don Gagne's avatar
Don Gagne committed
147
148
149
150
}

void MockLink::run(void)
{
Don Gagne's avatar
Don Gagne committed
151
152
153
    QTimer  timer1HzTasks;
    QTimer  timer10HzTasks;
    QTimer  timer500HzTasks;
154

Don Gagne's avatar
Don Gagne committed
155
156
157
    QObject::connect(&timer1HzTasks,  &QTimer::timeout, this, &MockLink::_run1HzTasks);
    QObject::connect(&timer10HzTasks, &QTimer::timeout, this, &MockLink::_run10HzTasks);
    QObject::connect(&timer500HzTasks, &QTimer::timeout, this, &MockLink::_run500HzTasks);
158

Don Gagne's avatar
Don Gagne committed
159
160
161
    timer1HzTasks.start(1000);
    timer10HzTasks.start(100);
    timer500HzTasks.start(2);
162

Don Gagne's avatar
Don Gagne committed
163
    exec();
164

Don Gagne's avatar
Don Gagne committed
165
166
167
    QObject::disconnect(&timer1HzTasks,  &QTimer::timeout, this, &MockLink::_run1HzTasks);
    QObject::disconnect(&timer10HzTasks, &QTimer::timeout, this, &MockLink::_run10HzTasks);
    QObject::disconnect(&timer500HzTasks, &QTimer::timeout, this, &MockLink::_run500HzTasks);
dogmaphobic's avatar
dogmaphobic committed
168

169
    _missionItemHandler.shutdown();
Don Gagne's avatar
Don Gagne committed
170
171
172
173
}

void MockLink::_run1HzTasks(void)
{
174
    if (_mavlinkStarted && _connected) {
DonLakeFlyer's avatar
DonLakeFlyer committed
175
176
        if (_highLatency) {
            _sendHighLatency2();
177
        } else {
DonLakeFlyer's avatar
DonLakeFlyer committed
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
            _sendVibration();
            _sendADSBVehicles();
            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();
            }
194
        }
Don Gagne's avatar
Don Gagne committed
195
196
197
198
199
    }
}

void MockLink::_run10HzTasks(void)
{
DonLakeFlyer's avatar
DonLakeFlyer committed
200
201
202
203
    if (_highLatency) {
        return;
    }

204
    if (_mavlinkStarted && _connected) {
Don Gagne's avatar
Don Gagne committed
205
        _sendHeartBeat();
Don Gagne's avatar
Don Gagne committed
206
207
208
209
210
211
        if (_sendGPSPositionDelayCount > 0) {
            // We delay gps position for better testing
            _sendGPSPositionDelayCount--;
        } else {
            _sendGpsRawInt();
        }
Don Gagne's avatar
Don Gagne committed
212
213
214
    }
}

Don Gagne's avatar
Don Gagne committed
215
void MockLink::_run500HzTasks(void)
Don Gagne's avatar
Don Gagne committed
216
{
DonLakeFlyer's avatar
DonLakeFlyer committed
217
218
219
220
    if (_highLatency) {
        return;
    }

221
    if (_mavlinkStarted && _connected) {
Don Gagne's avatar
Don Gagne committed
222
        _paramRequestListWorker();
223
        _logDownloadWorker();
Don Gagne's avatar
Don Gagne committed
224
225
226
227
228
    }
}

void MockLink::_loadParams(void)
{
229
230
231
232
    QFile paramFile;

    if (_firmwareType == MAV_AUTOPILOT_ARDUPILOTMEGA) {
        if (_vehicleType == MAV_TYPE_FIXED_WING) {
233
            paramFile.setFileName(":/MockLink/APMArduPlaneMockLink.params");
Rustom Jehangir's avatar
Rustom Jehangir committed
234
        } else if (_vehicleType == MAV_TYPE_SUBMARINE ) {
235
            paramFile.setFileName(":/MockLink/APMArduSubMockLink.params");
236
        } else {
237
            paramFile.setFileName(":/MockLink/APMArduCopterMockLink.params");
238
239
        }
    } else {
240
        paramFile.setFileName(":/MockLink/PX4MockLink.params");
241
242
    }

243

Don Gagne's avatar
Don Gagne committed
244
245
246
    bool success = paramFile.open(QFile::ReadOnly);
    Q_UNUSED(success);
    Q_ASSERT(success);
247

Don Gagne's avatar
Don Gagne committed
248
    QTextStream paramStream(&paramFile);
249

Don Gagne's avatar
Don Gagne committed
250
251
    while (!paramStream.atEnd()) {
        QString line = paramStream.readLine();
252

Don Gagne's avatar
Don Gagne committed
253
254
255
        if (line.startsWith("#")) {
            continue;
        }
256

Don Gagne's avatar
Don Gagne committed
257
258
        QStringList paramData = line.split("\t");
        Q_ASSERT(paramData.count() == 5);
259

Don Gagne's avatar
Don Gagne committed
260
261
262
        QString paramName = paramData.at(2);
        QString valStr = paramData.at(3);
        uint paramType = paramData.at(4).toUInt();
263

Don Gagne's avatar
Don Gagne committed
264
265
        QVariant paramValue;
        switch (paramType) {
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
        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;
Don Gagne's avatar
Don Gagne committed
291
        }
292

Don Gagne's avatar
Don Gagne committed
293
        qCDebug(MockLinkVerboseLog) << "Loading param" << paramName << paramValue;
294

295
        _mapParamName2Value[_vehicleComponentId][paramName] = paramValue;
296
        _mapParamName2MavParamType[paramName] = static_cast<MAV_PARAM_TYPE>(paramType);
Don Gagne's avatar
Don Gagne committed
297
298
299
300
301
302
303
    }
}

void MockLink::_sendHeartBeat(void)
{
    mavlink_message_t   msg;

Don Gagne's avatar
Don Gagne committed
304
305
    mavlink_msg_heartbeat_pack_chan(_vehicleSystemId,
                                    _vehicleComponentId,
Don Gagne's avatar
Don Gagne committed
306
                                    _mavlinkChannel,
Don Gagne's avatar
Don Gagne committed
307
308
309
310
311
312
                                    &msg,
                                    _vehicleType,        // MAV_TYPE
                                    _firmwareType,      // MAV_AUTOPILOT
                                    _mavBaseMode,        // MAV_MODE
                                    _mavCustomMode,      // custom mode
                                    _mavState);          // MAV_STATE
dogmaphobic's avatar
dogmaphobic committed
313

Don Gagne's avatar
Don Gagne committed
314
315
    respondWithMavlinkMessage(msg);
}
Don Gagne's avatar
Don Gagne committed
316

DonLakeFlyer's avatar
DonLakeFlyer committed
317
318
319
320
void MockLink::_sendHighLatency2(void)
{
    mavlink_message_t   msg;

321
322
323
324
    union px4_custom_mode   px4_cm;
    px4_cm.data = _mavCustomMode;

    qDebug() << "Sending" << _mavCustomMode;
DonLakeFlyer's avatar
DonLakeFlyer committed
325
326
327
328
329
330
331
    mavlink_msg_high_latency2_pack_chan(_vehicleSystemId,
                                        _vehicleComponentId,
                                        _mavlinkChannel,
                                        &msg,
                                        0,                          // timestamp
                                        _vehicleType,               // MAV_TYPE
                                        _firmwareType,              // MAV_AUTOPILOT
332
                                        px4_cm.custom_mode_hl,      // custom_mode
DonLakeFlyer's avatar
DonLakeFlyer committed
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
                                        (int32_t)(_vehicleLatitude  * 1E7),
                                        (int32_t)(_vehicleLongitude * 1E7),
                                        (int16_t)_vehicleAltitude,
                                        (int16_t)_vehicleAltitude,  // target_altitude,
                                        0,                          // heading
                                        0,                          // target_heading
                                        0,                          // target_distance
                                        0,                          // throttle
                                        0,                          // airspeed
                                        0,                          // airspeed_sp
                                        0,                          // groundspeed
                                        0,                          // windspeed,
                                        0,                          // wind_heading
                                        UINT8_MAX,                  // eph not known
                                        UINT8_MAX,                  // epv not known
                                        0,                          // temperature_air
                                        0,                          // climb_rate
                                        -1,                         // battery, do not use?
                                        0,                          // wp_num
                                        0,                          // failure_flags
                                        0, 0, 0);                   // custom0, custom1, custom2
    respondWithMavlinkMessage(msg);
}

Don Gagne's avatar
Don Gagne committed
357
358
359
360
void MockLink::_sendVibration(void)
{
    mavlink_message_t   msg;

Don Gagne's avatar
Don Gagne committed
361
362
    mavlink_msg_vibration_pack_chan(_vehicleSystemId,
                                    _vehicleComponentId,
Don Gagne's avatar
Don Gagne committed
363
                                    _mavlinkChannel,
Don Gagne's avatar
Don Gagne committed
364
365
366
367
368
369
370
371
                                    &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
Don Gagne's avatar
Don Gagne committed
372
373
374
375

    respondWithMavlinkMessage(msg);
}

Don Gagne's avatar
Don Gagne committed
376
377
378
void MockLink::respondWithMavlinkMessage(const mavlink_message_t& msg)
{
    uint8_t buffer[MAVLINK_MAX_PACKET_LEN];
dogmaphobic's avatar
dogmaphobic committed
379

Don Gagne's avatar
Don Gagne committed
380
381
382
383
384
385
    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
386
void MockLink::_writeBytes(const QByteArray bytes)
Don Gagne's avatar
Don Gagne committed
387
388
389
390
391
392
393
394
{
    if (_inNSH) {
        _handleIncomingNSHBytes(bytes.constData(), bytes.count());
    } else {
        if (bytes.startsWith(QByteArray("\r\r\r"))) {
            _inNSH  = true;
            _handleIncomingNSHBytes(&bytes.constData()[3], bytes.count() - 3);
        }
395

Don Gagne's avatar
Don Gagne committed
396
397
398
399
400
401
402
403
        _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);
404

Don Gagne's avatar
Don Gagne committed
405
406
407
408
409
410
411
412
    // 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;
413

Don Gagne's avatar
Don Gagne committed
414
415
#if 0
        // MockLink not quite ready to handle this correctly yet
Don Gagne's avatar
Don Gagne committed
416
417
418
419
        if (strncmp(bytes, "sh /etc/init.d/rc.usb\n", cBytes) == 0) {
            // This is the mavlink start command
            _mavlinkStarted = true;
        }
Don Gagne's avatar
Don Gagne committed
420
#endif
Don Gagne's avatar
Don Gagne committed
421
422
423
424
425
426
427
428
    }
}

/// @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;
429

Don Gagne's avatar
Don Gagne committed
430
431
    for (qint64 i=0; i<cBytes; i++)
    {
Don Gagne's avatar
Don Gagne committed
432
        if (!mavlink_parse_char(_mavlinkChannel, bytes[i], &msg, &comm)) {
Don Gagne's avatar
Don Gagne committed
433
434
            continue;
        }
dogmaphobic's avatar
dogmaphobic committed
435

436
        if (_missionItemHandler.handleMessage(msg)) {
Don Gagne's avatar
Don Gagne committed
437
438
            continue;
        }
439

Don Gagne's avatar
Don Gagne committed
440
        switch (msg.msgid) {
441
442
443
        case MAVLINK_MSG_ID_HEARTBEAT:
            _handleHeartBeat(msg);
            break;
444

445
446
447
        case MAVLINK_MSG_ID_PARAM_REQUEST_LIST:
            _handleParamRequestList(msg);
            break;
448

449
450
451
        case MAVLINK_MSG_ID_SET_MODE:
            _handleSetMode(msg);
            break;
452

453
454
455
        case MAVLINK_MSG_ID_PARAM_SET:
            _handleParamSet(msg);
            break;
456

457
458
459
        case MAVLINK_MSG_ID_PARAM_REQUEST_READ:
            _handleParamRequestRead(msg);
            break;
dogmaphobic's avatar
dogmaphobic committed
460

461
462
463
        case MAVLINK_MSG_ID_FILE_TRANSFER_PROTOCOL:
            _handleFTP(msg);
            break;
dogmaphobic's avatar
dogmaphobic committed
464

465
466
467
        case MAVLINK_MSG_ID_COMMAND_LONG:
            _handleCommandLong(msg);
            break;
468

469
470
471
        case MAVLINK_MSG_ID_MANUAL_CONTROL:
            _handleManualControl(msg);
            break;
Don Gagne's avatar
Don Gagne committed
472

473
474
475
476
477
478
479
480
        case MAVLINK_MSG_ID_LOG_REQUEST_LIST:
            _handleLogRequestList(msg);
            break;

        case MAVLINK_MSG_ID_LOG_REQUEST_DATA:
            _handleLogRequestData(msg);
            break;

481
482
        default:
            break;
Don Gagne's avatar
Don Gagne committed
483
484
485
486
487
488
489
        }
    }
}

void MockLink::_handleHeartBeat(const mavlink_message_t& msg)
{
    Q_UNUSED(msg);
490
    qCDebug(MockLinkLog) << "Heartbeat";
Don Gagne's avatar
Don Gagne committed
491
492
493
494
495
496
}

void MockLink::_handleSetMode(const mavlink_message_t& msg)
{
    mavlink_set_mode_t request;
    mavlink_msg_set_mode_decode(&msg, &request);
497

498
    Q_ASSERT(request.target_system == _vehicleSystemId);
499

500
501
502
503
    _mavBaseMode = request.base_mode;
    _mavCustomMode = request.custom_mode;
}

Don Gagne's avatar
Don Gagne committed
504
505
506
507
508
509
510
511
void MockLink::_handleManualControl(const mavlink_message_t& msg)
{
    mavlink_manual_control_t manualControl;
    mavlink_msg_manual_control_decode(&msg, &manualControl);

    qDebug() << "MANUAL_CONTROL" << manualControl.x << manualControl.y << manualControl.z << manualControl.r;
}

512
void MockLink::_setParamFloatUnionIntoMap(int componentId, const QString& paramName, float paramFloat)
513
514
{
    mavlink_param_union_t   valueUnion;
515

516
517
    Q_ASSERT(_mapParamName2Value.contains(componentId));
    Q_ASSERT(_mapParamName2Value[componentId].contains(paramName));
518
    Q_ASSERT(_mapParamName2MavParamType.contains(paramName));
519

520
    valueUnion.param_float = paramFloat;
521

522
    MAV_PARAM_TYPE paramType = _mapParamName2MavParamType[paramName];
523

524
    QVariant paramVariant;
525

526
    switch (paramType) {
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
    case MAV_PARAM_TYPE_REAL32:
        paramVariant = QVariant::fromValue(valueUnion.param_float);
        break;

    case MAV_PARAM_TYPE_UINT32:
        paramVariant = QVariant::fromValue(valueUnion.param_uint32);
        break;

    case MAV_PARAM_TYPE_INT32:
        paramVariant = QVariant::fromValue(valueUnion.param_int32);
        break;

    case MAV_PARAM_TYPE_UINT16:
        paramVariant = QVariant::fromValue(valueUnion.param_uint16);
        break;

    case MAV_PARAM_TYPE_INT16:
        paramVariant = QVariant::fromValue(valueUnion.param_int16);
        break;

    case MAV_PARAM_TYPE_UINT8:
        paramVariant = QVariant::fromValue(valueUnion.param_uint8);
        break;

    case MAV_PARAM_TYPE_INT8:
        paramVariant = QVariant::fromValue(valueUnion.param_int8);
        break;

    default:
        qCritical() << "Invalid parameter type" << paramType;
        paramVariant = QVariant::fromValue(valueUnion.param_int32);
        break;
Don Gagne's avatar
Don Gagne committed
559
    }
560

561
    qCDebug(MockLinkLog) << "_setParamFloatUnionIntoMap" << paramName << paramVariant;
562
    _mapParamName2Value[componentId][paramName] = paramVariant;
Don Gagne's avatar
Don Gagne committed
563
564
}

565
/// Convert from a parameter variant to the float value from mavlink_param_union_t
566
float MockLink::_floatUnionForParam(int componentId, const QString& paramName)
Don Gagne's avatar
Don Gagne committed
567
{
568
    mavlink_param_union_t   valueUnion;
569

570
571
    Q_ASSERT(_mapParamName2Value.contains(componentId));
    Q_ASSERT(_mapParamName2Value[componentId].contains(paramName));
572
    Q_ASSERT(_mapParamName2MavParamType.contains(paramName));
573

574
    MAV_PARAM_TYPE paramType = _mapParamName2MavParamType[paramName];
575
    QVariant paramVar = _mapParamName2Value[componentId][paramName];
576

577
    switch (paramType) {
578
    case MAV_PARAM_TYPE_REAL32:
579
        valueUnion.param_float = paramVar.toFloat();
580
        break;
581

582
583
584
585
586
587
588
    case MAV_PARAM_TYPE_UINT32:
        if (_firmwareType == MAV_AUTOPILOT_ARDUPILOTMEGA) {
            valueUnion.param_float = paramVar.toUInt();
        } else {
            valueUnion.param_uint32 = paramVar.toUInt();
        }
        break;
589

590
591
592
593
594
595
596
    case MAV_PARAM_TYPE_INT32:
        if (_firmwareType == MAV_AUTOPILOT_ARDUPILOTMEGA) {
            valueUnion.param_float = paramVar.toInt();
        } else {
            valueUnion.param_int32 = paramVar.toInt();
        }
        break;
597

598
599
600
601
602
603
604
    case MAV_PARAM_TYPE_UINT16:
        if (_firmwareType == MAV_AUTOPILOT_ARDUPILOTMEGA) {
            valueUnion.param_float = paramVar.toUInt();
        } else {
            valueUnion.param_uint16 = paramVar.toUInt();
        }
        break;
605

606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
    case MAV_PARAM_TYPE_INT16:
        if (_firmwareType == MAV_AUTOPILOT_ARDUPILOTMEGA) {
            valueUnion.param_float = paramVar.toInt();
        } else {
            valueUnion.param_int16 = paramVar.toInt();
        }
        break;

    case MAV_PARAM_TYPE_UINT8:
        if (_firmwareType == MAV_AUTOPILOT_ARDUPILOTMEGA) {
            valueUnion.param_float = paramVar.toUInt();
        } else {
            valueUnion.param_uint8 = paramVar.toUInt();
        }
        break;

    case MAV_PARAM_TYPE_INT8:
        if (_firmwareType == MAV_AUTOPILOT_ARDUPILOTMEGA) {
            valueUnion.param_float = (unsigned char)paramVar.toChar().toLatin1();
        } else {
            valueUnion.param_int8 = (unsigned char)paramVar.toChar().toLatin1();
        }
        break;

    default:
        if (_firmwareType == MAV_AUTOPILOT_ARDUPILOTMEGA) {
            valueUnion.param_float = paramVar.toInt();
        } else {
            valueUnion.param_int32 = paramVar.toInt();
        }
        qCritical() << "Invalid parameter type" << paramType;
637
    }
638

639
    return valueUnion.param_float;
Don Gagne's avatar
Don Gagne committed
640
641
642
643
}

void MockLink::_handleParamRequestList(const mavlink_message_t& msg)
{
Don Gagne's avatar
Don Gagne committed
644
645
646
647
    if (_failureMode == MockConfiguration::FailParamNoReponseToRequestList) {
        return;
    }

Don Gagne's avatar
Don Gagne committed
648
    mavlink_param_request_list_t request;
649

Don Gagne's avatar
Don Gagne committed
650
    mavlink_msg_param_request_list_decode(&msg, &request);
651

652
    Q_ASSERT(request.target_system == _vehicleSystemId);
653
    Q_ASSERT(request.target_component == MAV_COMP_ID_ALL);
dogmaphobic's avatar
dogmaphobic committed
654

Don Gagne's avatar
Don Gagne committed
655
656
657
658
    // Start the worker routine
    _currentParamRequestListComponentIndex = 0;
    _currentParamRequestListParamIndex = 0;
}
659

Don Gagne's avatar
Don Gagne committed
660
661
662
663
664
665
666
/// Sends the next parameter to the vehicle
void MockLink::_paramRequestListWorker(void)
{
    if (_currentParamRequestListComponentIndex == -1) {
        // Initial request complete
        return;
    }
dogmaphobic's avatar
dogmaphobic committed
667

Don Gagne's avatar
Don Gagne committed
668
669
670
    int componentId = _mapParamName2Value.keys()[_currentParamRequestListComponentIndex];
    int cParameters = _mapParamName2Value[componentId].count();
    QString paramName = _mapParamName2Value[componentId].keys()[_currentParamRequestListParamIndex];
671

Don Gagne's avatar
Don Gagne committed
672
673
674
    if ((_failureMode == MockConfiguration::FailMissingParamOnInitialReqest || _failureMode == MockConfiguration::FailMissingParamOnAllRequests) && paramName == _failParam) {
        qCDebug(MockLinkLog) << "Skipping param send:" << paramName;
    } else {
675

Don Gagne's avatar
Don Gagne committed
676
677
        char paramId[MAVLINK_MSG_ID_PARAM_VALUE_LEN];
        mavlink_message_t responseMsg;
678

Don Gagne's avatar
Don Gagne committed
679
680
        Q_ASSERT(_mapParamName2Value[componentId].contains(paramName));
        Q_ASSERT(_mapParamName2MavParamType.contains(paramName));
681

Don Gagne's avatar
Don Gagne committed
682
        MAV_PARAM_TYPE paramType = _mapParamName2MavParamType[paramName];
683

Don Gagne's avatar
Don Gagne committed
684
685
        Q_ASSERT(paramName.length() <= MAVLINK_MSG_ID_PARAM_VALUE_LEN);
        strncpy(paramId, paramName.toLocal8Bit().constData(), MAVLINK_MSG_ID_PARAM_VALUE_LEN);
dogmaphobic's avatar
dogmaphobic committed
686

Don Gagne's avatar
Don Gagne committed
687
688
        qCDebug(MockLinkLog) << "Sending msg_param_value" << componentId << paramId << paramType << _mapParamName2Value[componentId][paramId];

Don Gagne's avatar
Don Gagne committed
689
690
        mavlink_msg_param_value_pack_chan(_vehicleSystemId,
                                          componentId,                                   // component id
Don Gagne's avatar
Don Gagne committed
691
                                          _mavlinkChannel,
Don Gagne's avatar
Don Gagne committed
692
693
694
695
696
697
                                          &responseMsg,                                  // Outgoing message
                                          paramId,                                       // Parameter name
                                          _floatUnionForParam(componentId, paramName),   // Parameter value
                                          paramType,                                     // MAV_PARAM_TYPE
                                          cParameters,                                   // Total number of parameters
                                          _currentParamRequestListParamIndex);           // Index of this parameter
Don Gagne's avatar
Don Gagne committed
698
        respondWithMavlinkMessage(responseMsg);
699
    }
dogmaphobic's avatar
dogmaphobic committed
700

Don Gagne's avatar
Don Gagne committed
701
702
703
704
705
706
707
708
    // Move to next param index
    if (++_currentParamRequestListParamIndex >= 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;
709
        }
Don Gagne's avatar
Don Gagne committed
710
711
712
713
714
715
716
    }
}

void MockLink::_handleParamSet(const mavlink_message_t& msg)
{
    mavlink_param_set_t request;
    mavlink_msg_param_set_decode(&msg, &request);
717

718
    Q_ASSERT(request.target_system == _vehicleSystemId);
719
    int componentId = request.target_component;
dogmaphobic's avatar
dogmaphobic committed
720

721
722
    // Param may not be null terminated if exactly fits
    char paramId[MAVLINK_MSG_PARAM_SET_FIELD_PARAM_ID_LEN + 1];
Don Gagne's avatar
Don Gagne committed
723
    paramId[MAVLINK_MSG_PARAM_SET_FIELD_PARAM_ID_LEN] = 0;
724
    strncpy(paramId, request.param_id, MAVLINK_MSG_PARAM_SET_FIELD_PARAM_ID_LEN);
725

Don Gagne's avatar
Don Gagne committed
726
    qCDebug(MockLinkLog) << "_handleParamSet" << componentId << paramId << request.param_type;
dogmaphobic's avatar
dogmaphobic committed
727

728
729
    Q_ASSERT(_mapParamName2Value.contains(componentId));
    Q_ASSERT(_mapParamName2Value[componentId].contains(paramId));
730
    Q_ASSERT(request.param_type == _mapParamName2MavParamType[paramId]);
731

732
    // Save the new value
733
    _setParamFloatUnionIntoMap(componentId, paramId, request.param_value);
734

735
736
    // Respond with a param_value to ack
    mavlink_message_t responseMsg;
Don Gagne's avatar
Don Gagne committed
737
738
    mavlink_msg_param_value_pack_chan(_vehicleSystemId,
                                      componentId,                                               // component id
Don Gagne's avatar
Don Gagne committed
739
                                      _mavlinkChannel,
Don Gagne's avatar
Don Gagne committed
740
741
742
743
744
745
                                      &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
Don Gagne's avatar
Don Gagne committed
746
    respondWithMavlinkMessage(responseMsg);
Don Gagne's avatar
Don Gagne committed
747
748
749
750
}

void MockLink::_handleParamRequestRead(const mavlink_message_t& msg)
{
751
    mavlink_message_t   responseMsg;
Don Gagne's avatar
Don Gagne committed
752
753
    mavlink_param_request_read_t request;
    mavlink_msg_param_request_read_decode(&msg, &request);
754

Don Gagne's avatar
Don Gagne committed
755
    const QString paramName(QString::fromLocal8Bit(request.param_id, strnlen(request.param_id, MAVLINK_MSG_PARAM_REQUEST_READ_FIELD_PARAM_ID_LEN)));
756
    int componentId = request.target_component;
757
758

    // special case for magic _HASH_CHECK value
Don Gagne's avatar
Don Gagne committed
759
    if (request.target_component == MAV_COMP_ID_ALL && paramName == "_HASH_CHECK") {
760
761
762
763
        mavlink_param_union_t   valueUnion;
        valueUnion.type = MAV_PARAM_TYPE_UINT32;
        valueUnion.param_uint32 = 0;
        // Special case of magic hash check value
Don Gagne's avatar
Don Gagne committed
764
765
        mavlink_msg_param_value_pack_chan(_vehicleSystemId,
                                          componentId,
Don Gagne's avatar
Don Gagne committed
766
                                          _mavlinkChannel,
Don Gagne's avatar
Don Gagne committed
767
768
769
770
771
772
                                          &responseMsg,
                                          request.param_id,
                                          valueUnion.param_float,
                                          MAV_PARAM_TYPE_UINT32,
                                          0,
                                          -1);
773
774
775
776
        respondWithMavlinkMessage(responseMsg);
        return;
    }

777
    Q_ASSERT(_mapParamName2Value.contains(componentId));
778

Don Gagne's avatar
Don Gagne committed
779
780
    char paramId[MAVLINK_MSG_PARAM_REQUEST_READ_FIELD_PARAM_ID_LEN + 1];
    paramId[0] = 0;
781

782
    Q_ASSERT(request.target_system == _vehicleSystemId);
783

784
785
786
    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);
Don Gagne's avatar
Don Gagne committed
787
    } else {
788
        // Request is by index
789

Don Gagne's avatar
Don Gagne committed
790
        Q_ASSERT(request.param_index >= 0 && request.param_index < _mapParamName2Value[componentId].count());
791

792
        QString key = _mapParamName2Value[componentId].keys().at(request.param_index);
793
794
        Q_ASSERT(key.length() <= MAVLINK_MSG_PARAM_REQUEST_READ_FIELD_PARAM_ID_LEN);
        strcpy(paramId, key.toLocal8Bit().constData());
Don Gagne's avatar
Don Gagne committed
795
    }
796

797
    Q_ASSERT(_mapParamName2Value[componentId].contains(paramId));
798
    Q_ASSERT(_mapParamName2MavParamType.contains(paramId));
799

Don Gagne's avatar
Don Gagne committed
800
801
802
803
804
805
    if (_failureMode == MockConfiguration::FailMissingParamOnAllRequests && strcmp(paramId, _failParam) == 0) {
        qCDebug(MockLinkLog) << "Ignoring request read for " << _failParam;
        // Fail to send this param no matter what
        return;
    }

Don Gagne's avatar
Don Gagne committed
806
807
    mavlink_msg_param_value_pack_chan(_vehicleSystemId,
                                      componentId,                                               // component id
Don Gagne's avatar
Don Gagne committed
808
                                      _mavlinkChannel,
Don Gagne's avatar
Don Gagne committed
809
810
811
812
813
814
                                      &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
Don Gagne's avatar
Don Gagne committed
815
    respondWithMavlinkMessage(responseMsg);
Don Gagne's avatar
Don Gagne committed
816
817
}

Don Gagne's avatar
Don Gagne committed
818
819
820
void MockLink::emitRemoteControlChannelRawChanged(int channel, uint16_t raw)
{
    uint16_t chanRaw[18];
dogmaphobic's avatar
dogmaphobic committed
821

Don Gagne's avatar
Don Gagne committed
822
823
824
825
    for (int i=0; i<18; i++) {
        chanRaw[i] = UINT16_MAX;
    }
    chanRaw[channel] = raw;
dogmaphobic's avatar
dogmaphobic committed
826

Don Gagne's avatar
Don Gagne committed
827
    mavlink_message_t responseMsg;
Don Gagne's avatar
Don Gagne committed
828
829
    mavlink_msg_rc_channels_pack_chan(_vehicleSystemId,
                                      _vehicleComponentId,
Don Gagne's avatar
Don Gagne committed
830
                                      _mavlinkChannel,
Don Gagne's avatar
Don Gagne committed
831
832
833
834
                                      &responseMsg,          // Outgoing message
                                      0,                     // time since boot, ignored
                                      18,                    // channel count
                                      chanRaw[0],            // channel raw value
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
            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
Don Gagne's avatar
Don Gagne committed
853
854
855
856
857
858
859
    respondWithMavlinkMessage(responseMsg);
}

void MockLink::_handleFTP(const mavlink_message_t& msg)
{
    Q_ASSERT(_fileServer);
    _fileServer->handleFTPMessage(msg);
Don Gagne's avatar
Don Gagne committed
860
}
861
862
863

void MockLink::_handleCommandLong(const mavlink_message_t& msg)
{
Don Gagne's avatar
Don Gagne committed
864
865
866
    static bool firstCmdUser3 = true;
    static bool firstCmdUser4 = true;

867
    mavlink_command_long_t request;
868
    uint8_t commandResult = MAV_RESULT_UNSUPPORTED;
dogmaphobic's avatar
dogmaphobic committed
869

870
871
    mavlink_msg_command_long_decode(&msg, &request);

872
873
    switch (request.command) {
    case MAV_CMD_COMPONENT_ARM_DISARM:
874
875
876
877
878
        if (request.param1 == 0.0f) {
            _mavBaseMode &= ~MAV_MODE_FLAG_SAFETY_ARMED;
        } else {
            _mavBaseMode |= MAV_MODE_FLAG_SAFETY_ARMED;
        }
879
        commandResult = MAV_RESULT_ACCEPTED;
880
881
        break;
    case MAV_CMD_PREFLIGHT_CALIBRATION:
882
883
884
        _handlePreFlightCalibration(request);
        commandResult = MAV_RESULT_ACCEPTED;
        break;
885
886
887
    case MAV_CMD_PREFLIGHT_STORAGE:
        commandResult = MAV_RESULT_ACCEPTED;
        break;
888
889
890
891
    case MAV_CMD_REQUEST_AUTOPILOT_CAPABILITIES:
        commandResult = MAV_RESULT_ACCEPTED;
        _respondWithAutopilotVersion();
        break;
Don Gagne's avatar
Don Gagne committed
892
893
894
895
896
897
898
899
900
901
902
    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
        if (firstCmdUser3) {
DonLakeFlyer's avatar
DonLakeFlyer committed
903
904
            firstCmdUser3 = false;
            return;
Don Gagne's avatar
Don Gagne committed
905
906
907
908
909
910
911
912
        } else {
            firstCmdUser3 = true;
            commandResult = MAV_RESULT_ACCEPTED;
        }
        break;
    case MAV_CMD_USER_4:
        // Test command which returns MAV_RESULT_FAILED on second attempt
        if (firstCmdUser4) {
DonLakeFlyer's avatar
DonLakeFlyer committed
913
914
            firstCmdUser4 = false;
            return;
Don Gagne's avatar
Don Gagne committed
915
916
917
918
919
920
921
922
923
        } else {
            firstCmdUser4 = true;
            commandResult = MAV_RESULT_FAILED;
        }
        break;
    case MAV_CMD_USER_5:
        // No response
        return;
        break;
924
    }
925
926

    mavlink_message_t commandAck;
Don Gagne's avatar
Don Gagne committed
927
928
    mavlink_msg_command_ack_pack_chan(_vehicleSystemId,
                                      _vehicleComponentId,
Don Gagne's avatar
Don Gagne committed
929
                                      _mavlinkChannel,
Don Gagne's avatar
Don Gagne committed
930
931
                                      &commandAck,
                                      request.command,
DonLakeFlyer's avatar
DonLakeFlyer committed
932
                                      commandResult,
Gus Grubba's avatar
Gus Grubba committed
933
934
935
936
                                      0,    // progress
                                      0,    // result_param2
                                      0,    // target_system
                                      0);   // target_component
937
    respondWithMavlinkMessage(commandAck);
938
939
}

940
941
942
943
void MockLink::_respondWithAutopilotVersion(void)
{
    mavlink_message_t msg;

Don Gagne's avatar
Don Gagne committed
944
945
    uint8_t customVersion[8] = { };
    uint32_t flightVersion = 0;
946
#if !defined(NO_ARDUPILOT_DIALECT)
Don Gagne's avatar
Don Gagne committed
947
948
    if (_firmwareType == MAV_AUTOPILOT_ARDUPILOTMEGA) {
        flightVersion |= 3 << (8*3);
949
        flightVersion |= 5 << (8*2);
Don Gagne's avatar
Don Gagne committed
950
951
952
        flightVersion |= 0 << (8*1);
        flightVersion |= FIRMWARE_VERSION_TYPE_DEV << (8*0);
    } else if (_firmwareType == MAV_AUTOPILOT_PX4) {
953
#endif
Don Gagne's avatar
Don Gagne committed
954
955
956
957
        flightVersion |= 1 << (8*3);
        flightVersion |= 4 << (8*2);
        flightVersion |= 1 << (8*1);
        flightVersion |= FIRMWARE_VERSION_TYPE_DEV << (8*0);
958
#if !defined(NO_ARDUPILOT_DIALECT)
Don Gagne's avatar
Don Gagne committed
959
    }
960
#endif
Don Gagne's avatar
Don Gagne committed
961
962
    mavlink_msg_autopilot_version_pack_chan(_vehicleSystemId,
                                            _vehicleComponentId,
Don Gagne's avatar
Don Gagne committed
963
                                            _mavlinkChannel,
Don Gagne's avatar
Don Gagne committed
964
                                            &msg,
965
                                            MAV_PROTOCOL_CAPABILITY_MAVLINK2 | (_firmwareType == MAV_AUTOPILOT_ARDUPILOTMEGA ? 0 : MAV_PROTOCOL_CAPABILITY_MISSION_FENCE | MAV_PROTOCOL_CAPABILITY_MISSION_RALLY),
Don Gagne's avatar
Don Gagne committed
966
967
968
969
970
971
972
973
974
                                            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,
Don Gagne's avatar
Don Gagne committed
975
976
                                            0,                               // uid
                                            0);                              // uid2
977
978
979
    respondWithMavlinkMessage(msg);
}

Don Gagne's avatar
Don Gagne committed
980
void MockLink::setMissionItemFailureMode(MockLinkMissionItemHandler::FailureMode_t failureMode)
981
{
Don Gagne's avatar
Don Gagne committed
982
    _missionItemHandler.setMissionItemFailureMode(failureMode);
983
}
984
985
986

void MockLink::_sendHomePosition(void)
{
Don Gagne's avatar
Don Gagne committed
987
988
989
990
991
992
993
994
    mavlink_message_t msg;

    float bogus[4];
    bogus[0] = 0.0f;
    bogus[1] = 0.0f;
    bogus[2] = 0.0f;
    bogus[3] = 0.0f;

Don Gagne's avatar
Don Gagne committed
995
996
    mavlink_msg_home_position_pack_chan(_vehicleSystemId,
                                        _vehicleComponentId,
Don Gagne's avatar
Don Gagne committed
997
                                        _mavlinkChannel,
Don Gagne's avatar
Don Gagne committed
998
999
1000
                                        &msg,
                                        (int32_t)(_vehicleLatitude * 1E7),
                                        (int32_t)(_vehicleLongitude * 1E7),
For faster browsing, not all history is shown. View entire blame