3 changed files with 574 additions and 0 deletions
@ -0,0 +1,161 @@
@@ -0,0 +1,161 @@
|
||||
<PropertyList> |
||||
<generic> |
||||
<output> |
||||
<line_separator>newline</line_separator> |
||||
<var_separator>,</var_separator> |
||||
|
||||
<chunk> |
||||
<name>lat</name> |
||||
<type>float</type> |
||||
<format>%+1.8f</format> |
||||
<node>/position/latitude-deg</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>lon</name> |
||||
<type>float</type> |
||||
<format>%+1.8f</format> |
||||
<node>/position/longitude-deg</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>alt</name> |
||||
<type>float</type> |
||||
<format>%+1.4f</format> |
||||
<node>/position/altitude-ft</node> |
||||
</chunk> |
||||
|
||||
|
||||
<chunk> |
||||
<name>speed</name> |
||||
<type>float</type> |
||||
<format>%2.3f</format> |
||||
<node>/velocities/groundspeed-kt</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>airspeed</name> |
||||
<type>float</type> |
||||
<format>%2.3f</format> |
||||
<node>/velocities/airspeed-kt</node> |
||||
</chunk> |
||||
|
||||
|
||||
|
||||
<chunk> |
||||
<name>pitch</name> |
||||
<type>float</type> |
||||
<format>%+1.3f</format> |
||||
<node>/orientation/pitch-deg</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>roll</name> |
||||
<type>float</type> |
||||
<format>%+1.3f</format> |
||||
<node>/orientation/roll-deg</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>heading</name> |
||||
<type>float</type> |
||||
<format>%+1.3f</format> |
||||
<node>/orientation/heading-deg</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>v_n</name> |
||||
<type>float</type> |
||||
<format>%2.3f</format> |
||||
<node>/velocities/speed-north-fps</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>v_e</name> |
||||
<type>float</type> |
||||
<format>%2.3f</format> |
||||
<node>/velocities/speed-east-fps</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>v_d</name> |
||||
<type>float</type> |
||||
<format>%2.3f</format> |
||||
<node>/velocities/speed-down-fps</node> |
||||
</chunk> |
||||
|
||||
|
||||
<chunk> |
||||
<name>rate_phi</name> |
||||
<type>float</type> |
||||
<format>%2.3f</format> |
||||
<node>/orientation/roll-rate-degps</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>rate_theta</name> |
||||
<type>float</type> |
||||
<format>%2.3f</format> |
||||
<node>/orientation/pitch-rate-degps</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>rate_psi</name> |
||||
<type>float</type> |
||||
<format>%2.3f</format> |
||||
<node>/orientation/yaw-rate-degps</node> |
||||
</chunk> |
||||
|
||||
|
||||
</output> |
||||
|
||||
<input> |
||||
<line_separator>newline</line_separator> |
||||
<var_separator>,</var_separator> |
||||
|
||||
<chunk> |
||||
<name>pitch</name> |
||||
<type>float</type> |
||||
<format>%+1.3f</format> |
||||
<node>/autopilot/settings/target-pitch-deg</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>roll</name> |
||||
<type>float</type> |
||||
<format>%+1.3f</format> |
||||
<node>/autopilot/settings/target-roll-deg</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>throttle</name> |
||||
<type>float</type> |
||||
<format>%+1.3f</format> |
||||
<node>/controls/engines/engine/throttle</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>lat</name> |
||||
<type>float</type> |
||||
<format>%+1.8f</format> |
||||
<node>/sim/view/target/latitude-deg</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>lon</name> |
||||
<type>float</type> |
||||
<format>%+1.8f</format> |
||||
<node>/sim/view/target/longitude-deg</node> |
||||
</chunk> |
||||
|
||||
<chunk> |
||||
<name>alt</name> |
||||
<type>float</type> |
||||
<format>%+1.4f</format> |
||||
<node>/sim/view/target/alt</node> |
||||
</chunk> |
||||
|
||||
</input> |
||||
|
||||
</generic> |
||||
</PropertyList> |
@ -0,0 +1,301 @@
@@ -0,0 +1,301 @@
|
||||
/*=====================================================================
|
||||
|
||||
QGroundControl Open Source Ground Control Station |
||||
|
||||
(c) 2009 - 2011 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 Definition of UDP connection (server) for unmanned vehicles |
||||
* @author Lorenz Meier <mavteam@student.ethz.ch> |
||||
* |
||||
*/ |
||||
|
||||
#include <QTimer> |
||||
#include <QList> |
||||
#include <QDebug> |
||||
#include <QMutexLocker> |
||||
#include <iostream> |
||||
#include "QGCFlightGearLink.h" |
||||
#include "LinkManager.h" |
||||
#include "QGC.h" |
||||
#include <QHostInfo> |
||||
//#include <netinet/in.h>
|
||||
|
||||
QGCFlightGearLink::QGCFlightGearLink(QHostAddress host, quint16 port) |
||||
{ |
||||
this->host = host; |
||||
this->port = port; |
||||
this->connectState = false; |
||||
// Set unique ID and add link to the list of links
|
||||
this->name = tr("FlightGear Link (port:%1)").arg(5401); |
||||
LinkManager::instance()->add(this); |
||||
} |
||||
|
||||
QGCFlightGearLink::~QGCFlightGearLink() |
||||
{ |
||||
disconnect(); |
||||
} |
||||
|
||||
/**
|
||||
* @brief Runs the thread |
||||
* |
||||
**/ |
||||
void QGCFlightGearLink::run() |
||||
{ |
||||
// forever
|
||||
// {
|
||||
// QGC::SLEEP::msleep(5000);
|
||||
// }
|
||||
exec(); |
||||
} |
||||
|
||||
void QGCFlightGearLink::setPort(int port) |
||||
{ |
||||
this->port = port; |
||||
disconnect(); |
||||
connect(); |
||||
} |
||||
|
||||
/**
|
||||
* @param host Hostname in standard formatting, e.g. localhost:14551 or 192.168.1.1:14551 |
||||
*/ |
||||
void QGCFlightGearLink::setRemoteHost(const QString& host) |
||||
{ |
||||
//qDebug() << "UDP:" << "ADDING HOST:" << host;
|
||||
if (host.contains(":")) |
||||
{ |
||||
//qDebug() << "HOST: " << host.split(":").first();
|
||||
QHostInfo info = QHostInfo::fromName(host.split(":").first()); |
||||
if (info.error() == QHostInfo::NoError) |
||||
{ |
||||
// Add host
|
||||
QList<QHostAddress> hostAddresses = info.addresses(); |
||||
QHostAddress address; |
||||
for (int i = 0; i < hostAddresses.size(); i++) |
||||
{ |
||||
// Exclude loopback IPv4 and all IPv6 addresses
|
||||
if (!hostAddresses.at(i).toString().contains(":")) |
||||
{ |
||||
address = hostAddresses.at(i); |
||||
} |
||||
} |
||||
currentHost = address; |
||||
//qDebug() << "Address:" << address.toString();
|
||||
// Set port according to user input
|
||||
currentPort = host.split(":"); |
||||
} |
||||
} |
||||
else |
||||
{ |
||||
QHostInfo info = QHostInfo::fromName(host); |
||||
if (info.error() == QHostInfo::NoError) |
||||
{ |
||||
// Add host
|
||||
currentHost = info.addresses().first(); |
||||
} |
||||
} |
||||
} |
||||
|
||||
void QGCFlightGearLink::writeBytes(const char* data, qint64 size) |
||||
{ |
||||
// Broadcast to all connected systems
|
||||
for (int h = 0; h < hosts.size(); h++) |
||||
{ |
||||
quint16 currentPort = ports.at(h); |
||||
//#define QGCFlightGearLink_DEBUG
|
||||
#ifdef QGCFlightGearLink_DEBUG |
||||
QString bytes; |
||||
QString ascii; |
||||
for (int i=0; i<size; i++) |
||||
{ |
||||
unsigned char v = data[i]; |
||||
bytes.append(QString().sprintf("%02x ", v)); |
||||
if (data[i] > 31 && data[i] < 127) |
||||
{ |
||||
ascii.append(data[i]); |
||||
} |
||||
else |
||||
{ |
||||
ascii.append(219); |
||||
} |
||||
} |
||||
qDebug() << "Sent" << size << "bytes to" << currentHost.toString() << ":" << currentPort << "data:"; |
||||
qDebug() << bytes; |
||||
qDebug() << "ASCII:" << ascii; |
||||
#endif |
||||
socket->writeDatagram(data, size, currentHost, currentPort); |
||||
} |
||||
} |
||||
|
||||
/**
|
||||
* @brief Read a number of bytes from the interface. |
||||
* |
||||
* @param data Pointer to the data byte array to write the bytes to |
||||
* @param maxLength The maximum number of bytes to write |
||||
**/ |
||||
void QGCFlightGearLink::readBytes() |
||||
{ |
||||
const qint64 maxLength = 65536; |
||||
static char data[maxLength]; |
||||
QHostAddress sender; |
||||
quint16 senderPort; |
||||
|
||||
unsigned int s = socket->pendingDatagramSize(); |
||||
if (s > maxLength) std::cerr << __FILE__ << __LINE__ << " UDP datagram overflow, allowed to read less bytes than datagram size" << std::endl; |
||||
socket->readDatagram(data, maxLength, &sender, &senderPort); |
||||
|
||||
// FIXME TODO Check if this method is better than retrieving the data by individual processes
|
||||
QByteArray b(data, s); |
||||
emit bytesReceived(this, b); |
||||
|
||||
// // Echo data for debugging purposes
|
||||
// std::cerr << __FILE__ << __LINE__ << "Received datagram:" << std::endl;
|
||||
// int i;
|
||||
// for (i=0; i<s; i++)
|
||||
// {
|
||||
// unsigned int v=data[i];
|
||||
// fprintf(stderr,"%02x ", v);
|
||||
// }
|
||||
// std::cerr << std::endl;
|
||||
|
||||
|
||||
// Add host to broadcast list if not yet present
|
||||
if (!hosts.contains(sender)) |
||||
{ |
||||
hosts.append(sender); |
||||
ports.append(senderPort); |
||||
// ports->insert(sender, senderPort);
|
||||
} |
||||
else |
||||
{ |
||||
int index = hosts.indexOf(sender); |
||||
ports.replace(index, senderPort); |
||||
} |
||||
|
||||
} |
||||
|
||||
|
||||
/**
|
||||
* @brief Get the number of bytes to read. |
||||
* |
||||
* @return The number of bytes to read |
||||
**/ |
||||
qint64 QGCFlightGearLink::bytesAvailable() |
||||
{ |
||||
return socket->pendingDatagramSize(); |
||||
} |
||||
|
||||
/**
|
||||
* @brief Disconnect the connection. |
||||
* |
||||
* @return True if connection has been disconnected, false if connection couldn't be disconnected. |
||||
**/ |
||||
bool QGCFlightGearLink::disconnect() |
||||
{ |
||||
delete socket; |
||||
socket = NULL; |
||||
|
||||
connectState = false; |
||||
|
||||
emit disconnected(); |
||||
emit connected(false); |
||||
return !connectState; |
||||
} |
||||
|
||||
/**
|
||||
* @brief Connect the connection. |
||||
* |
||||
* @return True if connection has been established, false if connection couldn't be established. |
||||
**/ |
||||
bool QGCFlightGearLink::connect() |
||||
{ |
||||
socket = new QUdpSocket(this); |
||||
|
||||
//Check if we are using a multicast-address
|
||||
// bool multicast = false;
|
||||
// if (host.isInSubnet(QHostAddress("224.0.0.0"),4))
|
||||
// {
|
||||
// multicast = true;
|
||||
// connectState = socket->bind(port, QUdpSocket::ShareAddress);
|
||||
// }
|
||||
// else
|
||||
// {
|
||||
connectState = socket->bind(host, port); |
||||
// }
|
||||
|
||||
//Provides Multicast functionality to UdpSocket
|
||||
/* not working yet
|
||||
if (multicast) |
||||
{ |
||||
int sendingFd = socket->socketDescriptor(); |
||||
|
||||
if (sendingFd != -1) |
||||
{ |
||||
// set up destination address
|
||||
struct sockaddr_in sendAddr; |
||||
memset(&sendAddr,0,sizeof(sendAddr)); |
||||
sendAddr.sin_family=AF_INET; |
||||
sendAddr.sin_addr.s_addr=inet_addr(HELLO_GROUP); |
||||
sendAddr.sin_port=htons(port); |
||||
|
||||
// set TTL
|
||||
unsigned int ttl = 1; // restricted to the same subnet
|
||||
if (setsockopt(sendingFd, IPPROTO_IP, IP_MULTICAST_TTL, (unsigned int*)&ttl, sizeof(ttl) ) < 0) |
||||
{ |
||||
std::cout << "TTL failed\n"; |
||||
} |
||||
} |
||||
} |
||||
*/ |
||||
|
||||
//QObject::connect(socket, SIGNAL(readyRead()), this, SLOT(readPendingDatagrams()));
|
||||
QObject::connect(socket, SIGNAL(readyRead()), this, SLOT(readBytes())); |
||||
|
||||
emit connected(connectState); |
||||
if (connectState) { |
||||
emit connected(); |
||||
connectionStartTime = QGC::groundTimeUsecs()/1000; |
||||
} |
||||
|
||||
start(HighPriority); |
||||
return connectState; |
||||
} |
||||
|
||||
/**
|
||||
* @brief Check if connection is active. |
||||
* |
||||
* @return True if link is connected, false otherwise. |
||||
**/ |
||||
bool QGCFlightGearLink::isConnected() |
||||
{ |
||||
return connectState; |
||||
} |
||||
|
||||
QString QGCFlightGearLink::getName() |
||||
{ |
||||
return name; |
||||
} |
||||
|
||||
void QGCFlightGearLink::setName(QString name) |
||||
{ |
||||
this->name = name; |
||||
emit nameChanged(this->name); |
||||
} |
@ -0,0 +1,112 @@
@@ -0,0 +1,112 @@
|
||||
/*=====================================================================
|
||||
|
||||
QGroundControl Open Source Ground Control Station |
||||
|
||||
(c) 2009 - 2011 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 UDP connection (server) for unmanned vehicles |
||||
* @author Lorenz Meier <mavteam@student.ethz.ch> |
||||
* |
||||
*/ |
||||
|
||||
#ifndef QGCFLIGHTGEARLINK_H |
||||
#define QGCFLIGHTGEARLINK_H |
||||
|
||||
#include <QString> |
||||
#include <QList> |
||||
#include <QMap> |
||||
#include <QMutex> |
||||
#include <QUdpSocket> |
||||
#include <LinkInterface.h> |
||||
#include <configuration.h> |
||||
|
||||
class QGCFlightGearLink : public QThread |
||||
{ |
||||
Q_OBJECT |
||||
//Q_INTERFACES(QGCFlightGearLinkInterface:LinkInterface)
|
||||
|
||||
public: |
||||
QGCFlightGearLink(QHostAddress host = QHostAddress::Any, quint16 port = 5401); |
||||
//QGCFlightGearLink(QHostAddress host = "239.255.76.67", quint16 port = 7667);
|
||||
~QGCFlightGearLink(); |
||||
|
||||
bool isConnected(); |
||||
qint64 bytesAvailable(); |
||||
int getPort() const { |
||||
return port; |
||||
} |
||||
|
||||
/**
|
||||
* @brief The human readable port name |
||||
*/ |
||||
QString getName(); |
||||
|
||||
void run(); |
||||
|
||||
public slots: |
||||
void setAddress(QString address); |
||||
void setPort(int port); |
||||
/** @brief Add a new host to broadcast messages to */ |
||||
void addHost(const QString& host); |
||||
/** @brief Remove a host from broadcasting messages to */ |
||||
void removeHost(const QString& host); |
||||
// void readPendingDatagrams();
|
||||
|
||||
void readBytes(); |
||||
/**
|
||||
* @brief Write a number of bytes to the interface. |
||||
* |
||||
* @param data Pointer to the data byte array |
||||
* @param size The size of the bytes array |
||||
**/ |
||||
void writeBytes(const char* data, qint64 length); |
||||
bool connect(); |
||||
bool disconnect(); |
||||
|
||||
protected: |
||||
QString name; |
||||
QHostAddress host; |
||||
QHostAddress currentHost |
||||
quint16 currentPort; |
||||
quint16 port; |
||||
int id; |
||||
QUdpSocket* socket; |
||||
bool connectState; |
||||
|
||||
quint64 bitsSentTotal; |
||||
quint64 bitsSentCurrent; |
||||
quint64 bitsSentMax; |
||||
quint64 bitsReceivedTotal; |
||||
quint64 bitsReceivedCurrent; |
||||
quint64 bitsReceivedMax; |
||||
quint64 connectionStartTime; |
||||
QMutex statisticsMutex; |
||||
QMutex dataMutex; |
||||
|
||||
void setName(QString name); |
||||
|
||||
signals: |
||||
// Signals are defined by LinkInterface
|
||||
|
||||
}; |
||||
|
||||
#endif // QGCFLIGHTGEARLINK_H
|
Loading…
Reference in new issue