mirror of
https://github.com/hyperion-project/hyperion.ng.git
synced 2023-10-10 13:36:59 +02:00
164 lines
3.7 KiB
C++
164 lines
3.7 KiB
C++
|
#include "LedDeviceUdpDdp.h"
|
||
|
|
||
|
#include <QtEndian>
|
||
|
|
||
|
#include <utils/NetUtils.h>
|
||
|
|
||
|
// DDP header format
|
||
|
// header is 10 bytes (14 if TIME flag used)
|
||
|
struct ddp_hdr_struct {
|
||
|
uint8_t flags1;
|
||
|
uint8_t flags2;
|
||
|
uint8_t type;
|
||
|
uint8_t id;
|
||
|
uint32_t offset;
|
||
|
uint16_t len;
|
||
|
};
|
||
|
|
||
|
// Constants
|
||
|
namespace {
|
||
|
|
||
|
const char CONFIG_HOST[] = "host";
|
||
|
const char CONFIG_PORT[] = "port";
|
||
|
|
||
|
const ushort DDP_DEFAULT_PORT = 4048;
|
||
|
|
||
|
namespace DDP {
|
||
|
|
||
|
// DDP protocol header definitions
|
||
|
struct Header {
|
||
|
uint8_t flags1;
|
||
|
uint8_t flags2;
|
||
|
uint8_t type;
|
||
|
uint8_t id;
|
||
|
uint8_t offset[4];
|
||
|
uint8_t len[2];
|
||
|
};
|
||
|
|
||
|
static constexpr int HEADER_LEN = (sizeof(struct Header)); // header is 10 bytes (14 if TIME flag used)
|
||
|
static constexpr int MAX_LEDS = 480;
|
||
|
static constexpr int CHANNELS_PER_PACKET = MAX_LEDS*3;
|
||
|
|
||
|
namespace flags1 {
|
||
|
static constexpr auto VER_MASK = 0xc0;
|
||
|
static constexpr auto VER1 = 0x40;
|
||
|
static constexpr auto PUSH = 0x01;
|
||
|
static constexpr auto QUERY = 0x02;
|
||
|
static constexpr auto REPLY = 0x04;
|
||
|
static constexpr auto STORAGE = 0x08;
|
||
|
static constexpr auto TIME = 0x10;
|
||
|
} // namespace flags1
|
||
|
|
||
|
namespace id {
|
||
|
static constexpr auto DISPLAY = 1;
|
||
|
static constexpr auto CONTROL = 246;
|
||
|
static constexpr auto CONFIG = 250;
|
||
|
static constexpr auto STATUS = 251;
|
||
|
static constexpr auto DMXTRANSIT = 254;
|
||
|
static constexpr auto ALLDEVICES = 255;
|
||
|
} // namespace id
|
||
|
|
||
|
} // namespace DDP
|
||
|
|
||
|
} //End of constants
|
||
|
|
||
|
LedDeviceUdpDdp::LedDeviceUdpDdp(const QJsonObject &deviceConfig)
|
||
|
: ProviderUdp(deviceConfig)
|
||
|
,_packageSequenceNumber(0)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
LedDevice* LedDeviceUdpDdp::construct(const QJsonObject &deviceConfig)
|
||
|
{
|
||
|
return new LedDeviceUdpDdp(deviceConfig);
|
||
|
}
|
||
|
|
||
|
bool LedDeviceUdpDdp::init(const QJsonObject &deviceConfig)
|
||
|
{
|
||
|
bool isInitOK {false};
|
||
|
|
||
|
if ( ProviderUdp::init(deviceConfig) )
|
||
|
{
|
||
|
_hostName = _devConfig[ CONFIG_HOST ].toString();
|
||
|
_port = deviceConfig[CONFIG_PORT].toInt(DDP_DEFAULT_PORT);
|
||
|
|
||
|
Debug(_log, "Hostname/IP : %s", QSTRING_CSTR(_hostName) );
|
||
|
Debug(_log, "Port : %d", _port );
|
||
|
|
||
|
_ddpData.resize(DDP::HEADER_LEN + DDP::CHANNELS_PER_PACKET);
|
||
|
_ddpData[0] = DDP::flags1::VER1; // flags1
|
||
|
_ddpData[1] = 0; // flags2
|
||
|
_ddpData[2] = 1; // type
|
||
|
_ddpData[3] = DDP::id::DISPLAY; // id
|
||
|
|
||
|
isInitOK = true;
|
||
|
}
|
||
|
return isInitOK;
|
||
|
}
|
||
|
|
||
|
int LedDeviceUdpDdp::open()
|
||
|
{
|
||
|
int retval = -1;
|
||
|
_isDeviceReady = false;
|
||
|
|
||
|
if (NetUtils::resolveHostToAddress(_log, _hostName, _address))
|
||
|
{
|
||
|
if (ProviderUdp::open() == 0)
|
||
|
{
|
||
|
// Everything is OK, device is ready
|
||
|
_isDeviceReady = true;
|
||
|
retval = 0;
|
||
|
}
|
||
|
}
|
||
|
return retval;
|
||
|
}
|
||
|
|
||
|
int LedDeviceUdpDdp::write(const std::vector<ColorRgb> &ledValues)
|
||
|
{
|
||
|
int rc {0};
|
||
|
|
||
|
int channelCount = static_cast<int>(_ledCount) * 3; // 1 channel for every R,G,B value
|
||
|
int packetCount = ((channelCount-1) / DDP::CHANNELS_PER_PACKET) + 1;
|
||
|
int channel = 0;
|
||
|
|
||
|
_ddpData[0] = DDP::flags1::VER1;
|
||
|
|
||
|
for (int currentPacket = 0; currentPacket < packetCount; currentPacket++)
|
||
|
{
|
||
|
if (_packageSequenceNumber > 15)
|
||
|
{
|
||
|
_packageSequenceNumber = 0;
|
||
|
}
|
||
|
|
||
|
int packetSize = DDP::CHANNELS_PER_PACKET;
|
||
|
|
||
|
if (currentPacket == (packetCount - 1))
|
||
|
{
|
||
|
// last packet, set the push flag
|
||
|
/*0*/_ddpData[0] = DDP::flags1::VER1 | DDP::flags1::PUSH;
|
||
|
|
||
|
if (channelCount % DDP::CHANNELS_PER_PACKET != 0)
|
||
|
{
|
||
|
packetSize = channelCount % DDP::CHANNELS_PER_PACKET;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*1*/_ddpData[1] = static_cast<char>(_packageSequenceNumber++ & 0x0F);
|
||
|
/*4*/qToBigEndian<quint32>(static_cast<quint32>(channel), _ddpData.data() + 4);
|
||
|
/*8*/qToBigEndian<quint16>(static_cast<quint16>(packetSize), _ddpData.data() + 8);
|
||
|
|
||
|
_ddpData.replace(DDP::HEADER_LEN, channel, reinterpret_cast<const char*>(ledValues.data())+channel, packetSize);
|
||
|
_ddpData.resize(DDP::HEADER_LEN + packetSize);
|
||
|
|
||
|
rc = writeBytes(_ddpData);
|
||
|
|
||
|
if (rc != 0)
|
||
|
{
|
||
|
break;
|
||
|
}
|
||
|
channel += packetSize;
|
||
|
}
|
||
|
return rc;
|
||
|
}
|
||
|
|