2016-06-20 08:38:12 +02:00
|
|
|
// system includes
|
|
|
|
#include <stdexcept>
|
|
|
|
|
|
|
|
// project includes
|
|
|
|
#include <udplistener/UDPListener.h>
|
|
|
|
|
|
|
|
// hyperion util includes
|
|
|
|
#include "hyperion/ImageProcessorFactory.h"
|
|
|
|
#include "hyperion/ImageProcessor.h"
|
|
|
|
#include "utils/ColorRgb.h"
|
|
|
|
#include "HyperionConfig.h"
|
|
|
|
|
2016-06-26 20:08:03 +02:00
|
|
|
UDPListener::UDPListener(const int priority, const int timeout, const std::string& address, quint16 listenPort, bool shared) :
|
2016-06-20 08:38:12 +02:00
|
|
|
QObject(),
|
2016-06-20 23:41:07 +02:00
|
|
|
_hyperion(Hyperion::getInstance()),
|
2016-06-20 08:38:12 +02:00
|
|
|
_server(),
|
|
|
|
_openConnections(),
|
|
|
|
_priority(priority),
|
|
|
|
_timeout(timeout),
|
2016-06-25 15:15:23 +02:00
|
|
|
_ledColors(Hyperion::getInstance()->getLedCount(), ColorRgb::BLACK),
|
2016-06-27 09:27:11 +02:00
|
|
|
_log(Logger::getInstance("UDPLISTENER")),
|
|
|
|
_isActive(false),
|
2016-07-10 12:42:21 +02:00
|
|
|
_listenPort(listenPort),
|
2016-06-27 09:27:11 +02:00
|
|
|
_bondage(shared ? QAbstractSocket::ShareAddress : QAbstractSocket::DefaultForPlatform)
|
2016-06-20 08:38:12 +02:00
|
|
|
{
|
|
|
|
_server = new QUdpSocket(this);
|
2016-07-10 12:42:21 +02:00
|
|
|
_listenAddress = address.empty()
|
2016-06-27 09:27:11 +02:00
|
|
|
? QHostAddress::AnyIPv4
|
|
|
|
: QHostAddress( QString::fromStdString(address) );
|
2016-06-26 20:08:03 +02:00
|
|
|
|
2016-06-27 09:27:11 +02:00
|
|
|
// Set trigger for incoming connections
|
|
|
|
connect(_server, SIGNAL(readyRead()), this, SLOT(readPendingDatagrams()));
|
|
|
|
}
|
2016-06-25 15:15:23 +02:00
|
|
|
|
2016-06-27 09:27:11 +02:00
|
|
|
UDPListener::~UDPListener()
|
|
|
|
{
|
|
|
|
// clear the current channel
|
|
|
|
stop();
|
|
|
|
delete _server;
|
|
|
|
_hyperion->clear(_priority);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UDPListener::start()
|
|
|
|
{
|
|
|
|
if ( active() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
QHostAddress mcastGroup;
|
|
|
|
if (_listenAddress.isInSubnet(QHostAddress::parseSubnet("224.0.0.0/4"))) {
|
|
|
|
mcastGroup = _listenAddress;
|
2016-06-26 20:08:03 +02:00
|
|
|
}
|
2016-06-25 15:15:23 +02:00
|
|
|
|
2016-06-27 09:27:11 +02:00
|
|
|
if (!_server->bind(_listenAddress, _listenPort, _bondage))
|
2016-06-20 08:38:12 +02:00
|
|
|
{
|
2016-06-27 09:27:11 +02:00
|
|
|
Warning(_log, "Could not bind to %s:%d", _listenAddress.toString().toStdString().c_str(), _listenPort);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Info(_log, "Started, listening on %s:%d", _listenAddress.toString().toStdString().c_str(), _listenPort);
|
2016-06-26 20:08:03 +02:00
|
|
|
if (!mcastGroup.isNull()) {
|
2016-06-27 09:27:11 +02:00
|
|
|
bool joinGroupOK = _server->joinMulticastGroup(_listenAddress);
|
|
|
|
InfoIf ( joinGroupOK, _log, "Multicast enabled");
|
|
|
|
WarningIf( ! joinGroupOK, _log, "Multicast failed");
|
2016-06-25 15:15:23 +02:00
|
|
|
}
|
2016-06-27 09:27:11 +02:00
|
|
|
_isActive = true;
|
|
|
|
emit statusChanged(_isActive);
|
2016-06-20 08:38:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-27 09:27:11 +02:00
|
|
|
void UDPListener::stop()
|
2016-06-20 08:38:12 +02:00
|
|
|
{
|
2016-06-27 09:27:11 +02:00
|
|
|
if ( ! active() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
_server->close();
|
|
|
|
_isActive = false;
|
|
|
|
emit statusChanged(_isActive);
|
2016-06-20 08:38:12 +02:00
|
|
|
}
|
|
|
|
|
2016-06-27 09:27:11 +02:00
|
|
|
|
2016-06-20 08:38:12 +02:00
|
|
|
uint16_t UDPListener::getPort() const
|
|
|
|
{
|
|
|
|
return _server->localPort();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UDPListener::readPendingDatagrams()
|
|
|
|
{
|
|
|
|
while (_server->hasPendingDatagrams()) {
|
|
|
|
QByteArray datagram;
|
|
|
|
datagram.resize(_server->pendingDatagramSize());
|
|
|
|
QHostAddress sender;
|
|
|
|
quint16 senderPort;
|
|
|
|
|
|
|
|
_server->readDatagram(datagram.data(), datagram.size(),
|
|
|
|
&sender, &senderPort);
|
|
|
|
|
|
|
|
processTheDatagram(&datagram);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void UDPListener::processTheDatagram(const QByteArray * datagram)
|
|
|
|
{
|
|
|
|
int packlen = datagram->size()/3;
|
|
|
|
int ledlen = _ledColors.size();
|
|
|
|
int maxled = std::min(packlen , ledlen);
|
|
|
|
|
|
|
|
for (int ledIndex=0; ledIndex < maxled; ledIndex++) {
|
|
|
|
ColorRgb & rgb = _ledColors[ledIndex];
|
|
|
|
rgb.red = datagram->at(ledIndex*3+0);
|
|
|
|
rgb.green = datagram->at(ledIndex*3+1);
|
|
|
|
rgb.blue = datagram->at(ledIndex*3+2);
|
|
|
|
}
|
|
|
|
|
|
|
|
_hyperion->setColors(_priority, _ledColors, _timeout, -1);
|
|
|
|
|
|
|
|
}
|