mirror of
https://github.com/hyperion-project/hyperion.ng.git
synced 2023-10-10 13:36:59 +02:00
e9936e131b
* Allow build, if no grabbers are enabled * Align available functions to right Qt version * Update to next development version * Align available functions to right Qt version * fix workflows (apt/nightly) * Disable QNetworkConfigurationManager deprecation warnings * Initial go on Smart Pointers * Add Deallocation * Correct QT_WARNING_DISABLE_DEPRECATED (available since 5.9) * Cluster Build Variables * Hyperion Light * Address build warnings * Hyperion Light - UI * Update Protobuf to latest master * Removed compiler warnings * Added restart ability to systray * Correct Protobuf * Ignore 'no-return' warning on protobuf build * hyperion-remote: Fix auto discovery of hyperion server * Fix Qt version override * Update changelog * Remove Grabber Components, if no Grabber exists * Standalone Grabber - Fix fps default * Remote Control - Have Source Selction accrosswhole screen * Enable Blackborder detection only, if relevant input sources available * Enable Blackborder detection only, if relevant input sources available * Remote UI - rearrange containers * Checkout * Fix compilation on windows * Re-added qmdnsengine template cmake * chrono added for linux * Removed existing AVAHI/Bonjour, allow to enable/disable mDNS * hyperiond macos typo fix * Fix macOS Bundle build * Fix macOS bundle info details * Correct CMake files * Removed existing AVAHI/Bonjour (2) * Share hyperion's services via mDNS * Add mDNS Browser and mDNS for LED-Devices * Support mDNS discovery for standalone grabbers * Remove ZLib Dependency & Cleanup * mDNS - hanle 2.local2 an ".local." domains equally * Hue - Link discovery to bridge class, workaround port 443 for mDNS discovery * Fix save button state when switching between devices * Removed sessions (of other hyperions) * mDNS Publisher - Simplify service naming * mDNS refactoring & Forwarder discovery * mDNS Updates to use device service name * Consistency of standalone grabbers with mDNS Service Registry * Merge branch 'hyperion-project:master' into mDNS * Start JSON and WebServers only after Instance 0 is available * Remove bespoke qDebug Output again * MDNS updates and refactor Forwarder * Minor updates * Upgrade to CMake 3.1 * typo * macOS fix * Correct merge * - Remove dynamic linker flag from standalone dispmanX Grabber - Added ability to use system qmdns libs * Cec handler library will load at runtime * typo fix * protobuf changes * mDNS changes for Windows/macOS * test window build qmdnsengine * absolute path to protobuf cmake dir * Rework Hue Wizard supporting mDNS * LED-Devices - Retry support + Refactoring (excl. Hue) * LED-Devices - Refactoring/Retry support Hue + additional alignments * Address LGTM findings * Fix CI-Build, revert test changes * Build Windows in Release mode to avoid python problem * Correct that WebServerObject is available earlier * Ensure that instance name in logs for one instance are presented * Update content LEDs * Rework mDNS Address lookup * Fix LED UI * Fix for non mDNS Services (ignore default port) * Disbale device when now input is available * Revert back some updates, ensure last color is updated when switched on * Handle reopening case and changed IP, port for API-calls * Add UPD-DDP Device * WLED support for DDP * Fix printout * LEDDevice - Allow more retries, udapte defaults * LED-Net Devices - Select Custom device, if configured Co-authored-by: Paulchen Panther <16664240+Paulchen-Panther@users.noreply.github.com> Co-authored-by: Paulchen Panther <Paulchen-Panter@protonmail.com>
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;
|
|
}
|
|
|