Skip to content

Commit

Permalink
Merge pull request mavlink#6658 from acfloria/fix/comm_lost
Browse files Browse the repository at this point in the history
Comm lost fixes
  • Loading branch information
DonLakeFlyer authored Jun 26, 2018
2 parents 348091b + b0df009 commit 3a83977
Show file tree
Hide file tree
Showing 8 changed files with 56 additions and 57 deletions.
4 changes: 2 additions & 2 deletions qgroundcontrol.pro
Original file line number Diff line number Diff line change
Expand Up @@ -393,14 +393,14 @@ HEADERS += \
src/api/QGCOptions.h \
src/api/QGCSettings.h \
src/api/QmlComponentInfo.h \
src/comm/HeartbeatTimer.h
src/comm/MavlinkMessagesTimer.h

SOURCES += \
src/api/QGCCorePlugin.cc \
src/api/QGCOptions.cc \
src/api/QGCSettings.cc \
src/api/QmlComponentInfo.cc \
src/comm/HeartbeatTimer.cc
src/comm/MavlinkMessagesTimer.cc

#
# Unit Test specific configuration goes here (requires full debug build with all plugins)
Expand Down
5 changes: 4 additions & 1 deletion src/Vehicle/Vehicle.cc
Original file line number Diff line number Diff line change
Expand Up @@ -2508,7 +2508,10 @@ void Vehicle::_linkActiveChanged(LinkInterface *link, bool active, int vehicleID
1); // Request protocol version
}
} else if (!active && !_connectionLost) {
if (_connectionLostEnabled) {
_updatePriorityLink(false /* updateActive */, false /* sendCommand */);

// check if another active link has been found
if (link == _priorityLink) {
_connectionLost = true;
communicationLost = true;
_heardFrom = false;
Expand Down
30 changes: 15 additions & 15 deletions src/comm/LinkInterface.cc
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@

bool LinkInterface::active() const
{
QMapIterator<int /* vehicle id */, HeartbeatTimer*> iter(_heartbeatTimers);
QMapIterator<int /* vehicle id */, MavlinkMessagesTimer*> iter(_mavlinkMessagesTimers);
while (iter.hasNext()) {
iter.next();
if (iter.value()->getActive()) {
Expand All @@ -25,8 +25,8 @@ bool LinkInterface::active() const

bool LinkInterface::link_active(int vehicle_id) const
{
if (_heartbeatTimers.contains(vehicle_id)) {
return _heartbeatTimers.value(vehicle_id)->getActive();
if (_mavlinkMessagesTimers.contains(vehicle_id)) {
return _mavlinkMessagesTimers.value(vehicle_id)->getActive();
} else {
return false;
}
Expand Down Expand Up @@ -189,24 +189,24 @@ void LinkInterface::_activeChanged(bool active, int vehicle_id)
emit activeChanged(this, active, vehicle_id);
}

void LinkInterface::startHeartbeatTimer(int vehicle_id) {
if (_heartbeatTimers.contains(vehicle_id)) {
_heartbeatTimers.value(vehicle_id)->restartTimer();
void LinkInterface::startMavlinkMessagesTimer(int vehicle_id) {
if (_mavlinkMessagesTimers.contains(vehicle_id)) {
_mavlinkMessagesTimers.value(vehicle_id)->restartTimer();
} else {
_heartbeatTimers.insert(vehicle_id, new HeartbeatTimer(vehicle_id, _highLatency));
QObject::connect(_heartbeatTimers.value(vehicle_id), &HeartbeatTimer::activeChanged, this, &LinkInterface::_activeChanged);
_heartbeatTimers.value(vehicle_id)->init();
_mavlinkMessagesTimers.insert(vehicle_id, new MavlinkMessagesTimer(vehicle_id, _highLatency));
QObject::connect(_mavlinkMessagesTimers.value(vehicle_id), &MavlinkMessagesTimer::activeChanged, this, &LinkInterface::_activeChanged);
_mavlinkMessagesTimers.value(vehicle_id)->init();
}
}

void LinkInterface::stopHeartbeatTimer() {
QMapIterator<int /* vehicle id */, HeartbeatTimer*> iter(_heartbeatTimers);
void LinkInterface::stopMavlinkMessagesTimer() {
QMapIterator<int /* vehicle id */, MavlinkMessagesTimer*> iter(_mavlinkMessagesTimers);
while (iter.hasNext()) {
iter.next();
QObject::disconnect(iter.value(), &HeartbeatTimer::activeChanged, this, &LinkInterface::_activeChanged);
_heartbeatTimers[iter.key()]->deleteLater();
_heartbeatTimers[iter.key()] = nullptr;
QObject::disconnect(iter.value(), &MavlinkMessagesTimer::activeChanged, this, &LinkInterface::_activeChanged);
_mavlinkMessagesTimers[iter.key()]->deleteLater();
_mavlinkMessagesTimers[iter.key()] = nullptr;
}

_heartbeatTimers.clear();
_mavlinkMessagesTimers.clear();
}
18 changes: 9 additions & 9 deletions src/comm/LinkInterface.h
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@

#include "QGCMAVLink.h"
#include "LinkConfiguration.h"
#include "HeartbeatTimer.h"
#include "MavlinkMessagesTimer.h"

class LinkManager;

Expand All @@ -39,7 +39,7 @@ class LinkInterface : public QThread

public:
virtual ~LinkInterface() {
stopHeartbeatTimer();
stopMavlinkMessagesTimer();
_config->setLink(NULL);
}

Expand Down Expand Up @@ -264,19 +264,19 @@ private slots:
void _setMavlinkChannel(uint8_t channel);

/**
* @brief startHeartbeatTimer
* @brief startMavlinkMessagesTimer
*
* Start/restart the heartbeat timer for the specific vehicle.
* Start/restart the mavlink messages timer for the specific vehicle.
* If no timer exists an instance is allocated.
*/
void startHeartbeatTimer(int vehicle_id);
void startMavlinkMessagesTimer(int vehicle_id);

/**
* @brief stopHeartbeatTimer
* @brief stopMavlinkMessagesTimer
*
* Stop and deallocate the heartbeat timers for all vehicles if any exists.
* Stop and deallocate the mavlink messages timers for all vehicles if any exists.
*/
void stopHeartbeatTimer();
void stopMavlinkMessagesTimer();

bool _mavlinkChannelSet; ///< true: _mavlinkChannel has been set
uint8_t _mavlinkChannel; ///< mavlink channel to use for this link, as used by mavlink_parse_char
Expand All @@ -303,7 +303,7 @@ private slots:
bool _decodedFirstMavlinkPacket; ///< true: link has correctly decoded it's first mavlink packet
bool _isPX4Flow;

QMap<int /* vehicle id */, HeartbeatTimer*> _heartbeatTimers;
QMap<int /* vehicle id */, MavlinkMessagesTimer*> _mavlinkMessagesTimers;
};

typedef QSharedPointer<LinkInterface> SharedLinkInterfacePointer;
Expand Down
10 changes: 3 additions & 7 deletions src/comm/LinkManager.cc
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,7 @@ void LinkManager::setToolbox(QGCToolbox *toolbox)
_autoConnectSettings = toolbox->settingsManager()->autoConnectSettings();
_mavlinkProtocol = _toolbox->mavlinkProtocol();

connect(_mavlinkProtocol, &MAVLinkProtocol::vehicleHeartbeatInfo, this, &LinkManager::_heartbeatReceived);
connect(_mavlinkProtocol, &MAVLinkProtocol::messageReceived, this, &LinkManager::_mavlinkMessageReceived);

connect(&_portListTimer, &QTimer::timeout, this, &LinkManager::_updateAutoConnectLinks);
_portListTimer.start(_autoconnectUpdateTimerMSecs); // timeout must be long enough to get past bootloader on second pass
Expand Down Expand Up @@ -1008,10 +1008,6 @@ void LinkManager::_freeMavlinkChannel(int channel)
_mavlinkChannelsUsedBitMask &= ~(1 << channel);
}

void LinkManager::_heartbeatReceived(LinkInterface* link, int vehicleId, int componentId, int vehicleFirmwareType, int vehicleType) {
Q_UNUSED(componentId);
Q_UNUSED(vehicleFirmwareType);
Q_UNUSED(vehicleType);

link->startHeartbeatTimer(vehicleId);
void LinkManager::_mavlinkMessageReceived(LinkInterface* link, mavlink_message_t message) {
link->startMavlinkMessagesTimer(message.sysid);
}
2 changes: 1 addition & 1 deletion src/comm/LinkManager.h
Original file line number Diff line number Diff line change
Expand Up @@ -204,7 +204,7 @@ private slots:
SerialConfiguration* _autoconnectConfigurationsContainsPort(const QString& portName);
#endif

void _heartbeatReceived(LinkInterface* link, int vehicleId, int componentId, int vehicleFirmwareType, int vehicleType);
void _mavlinkMessageReceived(LinkInterface* link, mavlink_message_t message);

bool _configUpdateSuspended; ///< true: stop updating configuration list
bool _configurationsLoaded; ///< true: Link configurations have been loaded
Expand Down
20 changes: 10 additions & 10 deletions src/comm/HeartbeatTimer.cc → src/comm/MavlinkMessagesTimer.cc
Original file line number Diff line number Diff line change
Expand Up @@ -7,34 +7,34 @@
*
****************************************************************************/

#include "HeartbeatTimer.h"
#include "MavlinkMessagesTimer.h"

#include <QDebug>

HeartbeatTimer::HeartbeatTimer(int vehicle_id, bool high_latency) :
MavlinkMessagesTimer::MavlinkMessagesTimer(int vehicle_id, bool high_latency) :
_active(true),
_timer(new QTimer),
_vehicleID(vehicle_id),
_high_latency(high_latency)
{
}

void HeartbeatTimer::init()
void MavlinkMessagesTimer::init()
{
if (!_high_latency) {
_timer->setInterval(_heartbeatReceivedTimeoutMSecs);
_timer->setSingleShot(true);
_timer->setInterval(_messageReceivedTimeoutMSecs);
_timer->setSingleShot(false);
_timer->start();
}
emit activeChanged(true, _vehicleID);
QObject::connect(_timer, &QTimer::timeout, this, &HeartbeatTimer::timerTimeout);
QObject::connect(_timer, &QTimer::timeout, this, &MavlinkMessagesTimer::timerTimeout);
}


HeartbeatTimer::~HeartbeatTimer()
MavlinkMessagesTimer::~MavlinkMessagesTimer()
{
if (_timer) {
QObject::disconnect(_timer, &QTimer::timeout, this, &HeartbeatTimer::timerTimeout);
QObject::disconnect(_timer, &QTimer::timeout, this, &MavlinkMessagesTimer::timerTimeout);
_timer->stop();
delete _timer;
_timer = nullptr;
Expand All @@ -43,7 +43,7 @@ HeartbeatTimer::~HeartbeatTimer()
emit activeChanged(false, _vehicleID);
}

void HeartbeatTimer::restartTimer()
void MavlinkMessagesTimer::restartTimer()
{
if (!_active) {
_active = true;
Expand All @@ -53,7 +53,7 @@ void HeartbeatTimer::restartTimer()
_timer->start();
}

void HeartbeatTimer::timerTimeout()
void MavlinkMessagesTimer::timerTimeout()
{
if (!_high_latency) {
if (_active) {
Expand Down
24 changes: 12 additions & 12 deletions src/comm/HeartbeatTimer.h → src/comm/MavlinkMessagesTimer.h
Original file line number Diff line number Diff line change
Expand Up @@ -7,34 +7,34 @@
*
****************************************************************************/

#ifndef _HEARTBEATTIMER_H_
#define _HEARTBEATTIMER_H_
#ifndef _MAVLINKMESSAGESTIMER_H_
#define _MAVLINKMESSAGESTIMER_H_

#include <QTimer>
#include <QObject>

/**
* @brief The HeartbeatTimer class
* @brief The MavlinkMessagesTimer class
*
* Track the heartbeat for a single vehicle on one link.
* As long as regular heartbeats are received the status is active. On the timer timeout
* Track the mavlink messages for a single vehicle on one link.
* As long as regular messages are received the status is active. On the timer timeout
* status is set to inactive. On any status change the activeChanged signal is emitted.
* If high_latency is true then active is always true.
*/
class HeartbeatTimer : public QObject
class MavlinkMessagesTimer : public QObject
{
Q_OBJECT

public:
/**
* @brief HeartbeatTimer
* @brief MavlinkMessagesTimer
*
* Constructor
*
* @param vehicle_id: The vehicle ID for which the heartbeat will be tracked.
* @param high_latency: Indicates if the link is a high latency one.
*/
HeartbeatTimer(int vehicle_id, bool high_latency);
MavlinkMessagesTimer(int vehicle_id, bool high_latency);

/**
* @brief init
Expand All @@ -43,7 +43,7 @@ class HeartbeatTimer : public QObject
*/
void init();

~HeartbeatTimer();
~MavlinkMessagesTimer();

/**
* @brief getActive
Expand All @@ -68,7 +68,7 @@ class HeartbeatTimer : public QObject
/**
* @brief heartbeatTimeout
*
* Emitted if no heartbeat is received over the specified time.
* Emitted if no mavlink message is received over the specified time.
*
* @param vehicle_id: The vehicle ID for which the heartbeat timed out.
*/
Expand Down Expand Up @@ -100,7 +100,7 @@ private slots:
int _vehicleID = -1; // Vehicle ID for which the heartbeat is tracked.
bool _high_latency = false; // Indicates if the link is a high latency link or not.

static const int _heartbeatReceivedTimeoutMSecs = 3500; // Signal connection lost after 3.5 seconds of no messages
static const int _messageReceivedTimeoutMSecs = 3500; // Signal connection lost after 3.5 seconds of no messages
};

#endif // _HEARTBEATTIMER_H_
#endif // _MAVLINKMESSAGESTIMER_H_

0 comments on commit 3a83977

Please sign in to comment.