Newer
Older
/*=====================================================================
QGroundControl Open Source Ground Control Station
(c) 2009, 2010 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>
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 <http://www.gnu.org/licenses/>.
======================================================================*/
/**
* @file
* @brief Brief Description
*
* @author Lorenz Meier <mavteam@student.ethz.ch>
*
*/
#include <QList>
#include <QApplication>
#include <QDebug>
#include "LinkManager.h"
#include "MainWindow.h"
LinkManager* LinkManager::instance()
{
static LinkManager* _instance = 0;
if(_instance == 0) {
_instance = new LinkManager();
/* Set the application as parent to ensure that this object
* will be destroyed when the main application exits */
_instance->setParent(qApp);
}
return _instance;
* This class implements the singleton design pattern and has therefore only a private constructor.
**/
LinkManager::LinkManager() :
_connectionsSuspended(false)
_links = QList<LinkInterface*>();
_protocolLinks = QMap<ProtocolInterface*, LinkInterface*>();
_dataMutex.lock();
foreach (LinkInterface* link, _links) {
Q_ASSERT(link);
link->deleteLater();
_links.clear();
_dataMutex.unlock();
Q_ASSERT(link);
_dataMutex.lock();
if (!_links.contains(link))
connect(link, SIGNAL(destroyed(QObject*)), this, SLOT(removeObj(QObject*)));
_links.append(link);
_dataMutex.unlock();
emit newLink(link);
}
void LinkManager::addProtocol(LinkInterface* link, ProtocolInterface* protocol)
{
Q_ASSERT(link);
Q_ASSERT(protocol);
// Connect link to protocol
// the protocol will receive new bytes from the link
_dataMutex.lock();
QList<LinkInterface*> linkList = _protocolLinks.values(protocol);
// If protocol has not been added before (list length == 0)
// OR if link has not been added to protocol, add
// Protocol is new, add
connect(link, SIGNAL(bytesReceived(LinkInterface*, QByteArray)), protocol, SLOT(receiveBytes(LinkInterface*, QByteArray)));
// Add status
connect(link, SIGNAL(connected(bool)), protocol, SLOT(linkStatusChanged(bool)));
// Store the connection information in the protocol links map
_protocolLinks.insertMulti(protocol, link);
_dataMutex.unlock();
// Make sure the protocol clears its metadata for this link.
protocol->resetMetadataForLink(link);
//qDebug() << __FILE__ << __LINE__ << "ADDED LINK TO PROTOCOL" << link->getName() << protocol->getName() << "NEW SIZE OF LINK LIST:" << _protocolLinks.size();
}
QList<LinkInterface*> LinkManager::getLinksForProtocol(ProtocolInterface* protocol)
{
_dataMutex.lock();
QList<LinkInterface*> links = _protocolLinks.values(protocol);
_dataMutex.unlock();
ProtocolInterface* LinkManager::getProtocolForLink(LinkInterface* link)
{
_dataMutex.lock();
ProtocolInterface* interface = _protocolLinks.key(link);
_dataMutex.unlock();
}
if (_connectionsSuspendedMsg()) {
return false;
}
_dataMutex.lock();
foreach (LinkInterface* link, _links) {
Q_ASSERT(link);
if (!link->_connect()) {
allConnected = false;
}
_dataMutex.lock();
foreach (LinkInterface* link, _links)
Q_ASSERT(link);
if (!link->disconnect()) {
allDisconnected = false;
}
}
bool LinkManager::connectLink(LinkInterface* link)
{
Q_ASSERT(link);
if (_connectionsSuspendedMsg()) {
return false;
}
return link->_connect();
}
bool LinkManager::disconnectLink(LinkInterface* link)
{
Q_ASSERT(link);
return link->_disconnect();
void LinkManager::removeObj(QObject* link)
// Be careful of the fact that by the time this signal makes it through the queue
// the link object has already been destructed.
removeLink((LinkInterface*)link);
bool LinkManager::removeLink(LinkInterface* link)
{
_dataMutex.lock();
for (int i=0; i < _links.size(); i++)
_links.removeAt(i); //remove from link list
QList<ProtocolInterface* > protocols = _protocolLinks.keys(link);
_protocolLinks.remove(proto, link);
// Emit removal of link
emit linkRemoved(link);
return true;
}
return false;
/**
* The access time is linear in the number of links.
*
* @param id link identifier to search for
* @return A pointer to the link or NULL if not found
*/
LinkInterface* LinkManager::getLinkForId(int id)
{
LinkInterface* linkret = NULL;
foreach (LinkInterface* link, _links)
if (link->getId() == id)
{
linkret = link;
}
}
/**
*
*/
const QList<LinkInterface*> LinkManager::getLinks()
{
_dataMutex.lock();
QList<LinkInterface*> ret(_links);
_dataMutex.unlock();
const QList<SerialLink*> LinkManager::getSerialLinks()
{
foreach (LinkInterface* link, _links)
Q_ASSERT(link);
SerialLink* serialLink = qobject_cast<SerialLink*>(link);
if (serialLink)
s.append(serialLink);
/// @brief If all new connections should be suspended a message is displayed to the user and true
/// is returned;
bool LinkManager::_connectionsSuspendedMsg(void)
{
if (_connectionsSuspended) {
QMessageBox::information(MainWindow::instance(),
tr("Connect not allowed"),
tr("Connect not allowed: %1").arg(_connectionsSuspendedReason));
return true;
} else {
return false;
}
}
void LinkManager::setConnectionsSuspended(QString reason)
{
_connectionsSuspended = true;
_connectionsSuspendedReason = reason;
Q_ASSERT(!reason.isEmpty());
}