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>
343 lines
8.3 KiB
C++
343 lines
8.3 KiB
C++
#include <cec/CECHandler.h>
|
|
#include <utils/Logger.h>
|
|
|
|
#include <algorithm>
|
|
|
|
#include <libcec/cecloader.h>
|
|
|
|
#include <QJsonArray>
|
|
#include <QJsonDocument>
|
|
#include <QJsonObject>
|
|
#include <QDebug>
|
|
#include <QFile>
|
|
|
|
/* Enable to turn on detailed CEC logs */
|
|
// #define VERBOSE_CEC
|
|
|
|
CECHandler::CECHandler()
|
|
{
|
|
qRegisterMetaType<CECEvent>("CECEvent");
|
|
|
|
_logger = Logger::getInstance("CEC");
|
|
|
|
_cecCallbacks = getCallbacks();
|
|
_cecConfig = getConfig();
|
|
_cecConfig.callbacks = &_cecCallbacks;
|
|
_cecConfig.callbackParam = this;
|
|
}
|
|
|
|
CECHandler::~CECHandler()
|
|
{
|
|
stop();
|
|
}
|
|
|
|
bool CECHandler::start()
|
|
{
|
|
if (_cecAdapter)
|
|
return true;
|
|
|
|
std::string library = std::string("" CEC_LIBRARY);
|
|
_cecAdapter = LibCecInitialise(&_cecConfig, QFile::exists(QString::fromStdString(library)) ? library.c_str() : nullptr);
|
|
if(!_cecAdapter)
|
|
{
|
|
Error(_logger, "Failed to loading libcec.so");
|
|
return false;
|
|
}
|
|
|
|
Info(_logger, "CEC handler started");
|
|
|
|
auto adapters = getAdapters();
|
|
if (adapters.isEmpty())
|
|
{
|
|
Error(_logger, "Failed to find CEC adapter");
|
|
UnloadLibCec(_cecAdapter);
|
|
_cecAdapter = nullptr;
|
|
|
|
return false;
|
|
}
|
|
|
|
Info(_logger, "Auto detecting CEC adapter");
|
|
bool opened = false;
|
|
for (const auto & adapter : adapters)
|
|
{
|
|
printAdapter(adapter);
|
|
|
|
if (!opened && openAdapter(adapter))
|
|
{
|
|
Info(_logger, "CEC Handler initialized with adapter : %s", adapter.strComName);
|
|
|
|
opened = true;
|
|
}
|
|
}
|
|
|
|
if (!opened)
|
|
{
|
|
UnloadLibCec(_cecAdapter);
|
|
_cecAdapter = nullptr;
|
|
}
|
|
|
|
return opened;
|
|
}
|
|
|
|
void CECHandler::stop()
|
|
{
|
|
if (_cecAdapter)
|
|
{
|
|
Info(_logger, "Stopping CEC handler");
|
|
|
|
_cecAdapter->Close();
|
|
UnloadLibCec(_cecAdapter);
|
|
_cecAdapter = nullptr;
|
|
}
|
|
}
|
|
|
|
CECConfig CECHandler::getConfig() const
|
|
{
|
|
CECConfig configuration;
|
|
|
|
const std::string name("HyperionCEC");
|
|
name.copy(configuration.strDeviceName, std::min(name.size(), sizeof(configuration.strDeviceName)));
|
|
|
|
configuration.deviceTypes.Add(CEC::CEC_DEVICE_TYPE_RECORDING_DEVICE);
|
|
configuration.clientVersion = CEC::LIBCEC_VERSION_CURRENT;
|
|
|
|
return configuration;
|
|
}
|
|
|
|
CECCallbacks CECHandler::getCallbacks() const
|
|
{
|
|
CECCallbacks callbacks;
|
|
|
|
callbacks.sourceActivated = onCecSourceActivated;
|
|
callbacks.commandReceived = onCecCommandReceived;
|
|
callbacks.alert = onCecAlert;
|
|
callbacks.logMessage = onCecLogMessage;
|
|
callbacks.keyPress = onCecKeyPress;
|
|
callbacks.configurationChanged = onCecConfigurationChanged;
|
|
callbacks.menuStateChanged = onCecMenuStateChanged;
|
|
|
|
return callbacks;
|
|
}
|
|
|
|
QVector<CECAdapterDescriptor> CECHandler::getAdapters() const
|
|
{
|
|
if (!_cecAdapter)
|
|
return {};
|
|
|
|
QVector<CECAdapterDescriptor> descriptors(16);
|
|
int8_t size = _cecAdapter->DetectAdapters(descriptors.data(), descriptors.size(), nullptr, true /*quickscan*/);
|
|
descriptors.resize(size);
|
|
|
|
return descriptors;
|
|
}
|
|
|
|
bool CECHandler::openAdapter(const CECAdapterDescriptor & descriptor)
|
|
{
|
|
if (!_cecAdapter)
|
|
return false;
|
|
|
|
if(!_cecAdapter->Open(descriptor.strComName))
|
|
{
|
|
Error(_logger, QString("Failed to open the CEC adaper on port %1")
|
|
.arg(descriptor.strComName)
|
|
.toLocal8Bit());
|
|
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void CECHandler::printAdapter(const CECAdapterDescriptor & descriptor) const
|
|
{
|
|
Info(_logger, QString("CEC Adapter:").toLocal8Bit());
|
|
Info(_logger, QString("\tName : %1").arg(descriptor.strComName).toLocal8Bit());
|
|
Info(_logger, QString("\tPath : %1").arg(descriptor.strComPath).toLocal8Bit());
|
|
}
|
|
|
|
QString CECHandler::scan() const
|
|
{
|
|
if (!_cecAdapter)
|
|
return {};
|
|
|
|
Info(_logger, "Starting CEC scan");
|
|
|
|
QJsonArray devices;
|
|
CECLogicalAddresses addresses = _cecAdapter->GetActiveDevices();
|
|
for (int address = CEC::CECDEVICE_TV; address <= CEC::CECDEVICE_BROADCAST; ++address)
|
|
{
|
|
if (addresses[address])
|
|
{
|
|
CECLogicalAddress logicalAddress = (CECLogicalAddress)address;
|
|
|
|
QJsonObject device;
|
|
CECVendorId vendor = (CECVendorId)_cecAdapter->GetDeviceVendorId(logicalAddress);
|
|
CECPowerStatus power = _cecAdapter->GetDevicePowerStatus(logicalAddress);
|
|
|
|
device["name" ] = _cecAdapter->GetDeviceOSDName(logicalAddress).c_str();
|
|
device["vendor" ] = _cecAdapter->ToString(vendor);
|
|
device["address" ] = _cecAdapter->ToString(logicalAddress);
|
|
device["power" ] = _cecAdapter->ToString(power);
|
|
|
|
devices << device;
|
|
|
|
Info(_logger, QString("\tCECDevice: %1 / %2 / %3 / %4")
|
|
.arg(device["name"].toString())
|
|
.arg(device["vendor"].toString())
|
|
.arg(device["address"].toString())
|
|
.arg(device["power"].toString())
|
|
.toLocal8Bit());
|
|
}
|
|
}
|
|
|
|
return QJsonDocument(devices).toJson(QJsonDocument::Compact);
|
|
}
|
|
|
|
void CECHandler::onCecLogMessage(void * context, const CECLogMessage * message)
|
|
{
|
|
#ifdef VERBOSE_CEC
|
|
CECHandler * handler = qobject_cast<CECHandler*>(static_cast<QObject*>(context));
|
|
if (!handler)
|
|
return;
|
|
|
|
switch (message->level)
|
|
{
|
|
case CEC::CEC_LOG_ERROR:
|
|
Error(handler->_logger, QString("%1")
|
|
.arg(message->message)
|
|
.toLocal8Bit());
|
|
break;
|
|
case CEC::CEC_LOG_WARNING:
|
|
Warning(handler->_logger, QString("%1")
|
|
.arg(message->message)
|
|
.toLocal8Bit());
|
|
break;
|
|
case CEC::CEC_LOG_TRAFFIC:
|
|
case CEC::CEC_LOG_NOTICE:
|
|
Info(handler->_logger, QString("%1")
|
|
.arg(message->message)
|
|
.toLocal8Bit());
|
|
break;
|
|
case CEC::CEC_LOG_DEBUG:
|
|
Debug(handler->_logger, QString("%1")
|
|
.arg(message->message)
|
|
.toLocal8Bit());
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void CECHandler::onCecKeyPress(void * context, const CECKeyPress * key)
|
|
{
|
|
#ifdef VERBOSE_CEC
|
|
CECHandler * handler = qobject_cast<CECHandler*>(static_cast<QObject*>(context));
|
|
if (!handler)
|
|
return;
|
|
|
|
CECAdapter * adapter = handler->_cecAdapter;
|
|
|
|
Debug(handler->_logger, QString("CECHandler::onCecKeyPress: %1")
|
|
.arg(adapter->ToString(key->keycode))
|
|
.toLocal8Bit());
|
|
#endif
|
|
}
|
|
|
|
void CECHandler::onCecAlert(void * context, const CECAlert alert, const CECParameter data)
|
|
{
|
|
#ifdef VERBOSE_CEC
|
|
CECHandler * handler = qobject_cast<CECHandler*>(static_cast<QObject*>(context));
|
|
if (!handler)
|
|
return;
|
|
|
|
Error(handler->_logger, QString("CECHandler::onCecAlert: %1")
|
|
.arg(alert)
|
|
.toLocal8Bit());
|
|
#endif
|
|
}
|
|
|
|
void CECHandler::onCecConfigurationChanged(void * context, const CECConfig * configuration)
|
|
{
|
|
#ifdef VERBOSE_CEC
|
|
CECHandler * handler = qobject_cast<CECHandler*>(static_cast<QObject*>(context));
|
|
if (!handler)
|
|
return;
|
|
|
|
Debug(handler->_logger, QString("CECHandler::onCecConfigurationChanged: %1")
|
|
.arg(configuration->strDeviceName)
|
|
.toLocal8Bit());
|
|
#endif
|
|
}
|
|
|
|
int CECHandler::onCecMenuStateChanged(void * context, const CECMenuState state)
|
|
{
|
|
#ifdef VERBOSE_CEC
|
|
CECHandler * handler = qobject_cast<CECHandler*>(static_cast<QObject*>(context));
|
|
if (!handler)
|
|
return 0;
|
|
|
|
CECAdapter * adapter = handler->_cecAdapter;
|
|
|
|
Debug(handler->_logger, QString("CECHandler::onCecMenuStateChanged: %1")
|
|
.arg(adapter->ToString(state))
|
|
.toLocal8Bit());
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
void CECHandler::onCecCommandReceived(void * context, const CECCommand * command)
|
|
{
|
|
CECHandler * handler = qobject_cast<CECHandler*>(static_cast<QObject*>(context));
|
|
if (!handler)
|
|
return;
|
|
|
|
CECAdapter * adapter = handler->_cecAdapter;
|
|
|
|
#ifdef VERBOSE_CEC
|
|
Debug(handler->_logger, QString("CECHandler::onCecCommandReceived: %1 (%2 > %3)")
|
|
.arg(adapter->ToString(command->opcode))
|
|
.arg(adapter->ToString(command->initiator))
|
|
.arg(adapter->ToString(command->destination))
|
|
.toLocal8Bit());
|
|
#endif
|
|
/* We do NOT check sender */
|
|
// if (address == CEC::CECDEVICE_TV)
|
|
{
|
|
if (command->opcode == CEC::CEC_OPCODE_SET_STREAM_PATH)
|
|
{
|
|
Info(handler->_logger, QString("CEC source activated: %1")
|
|
.arg(adapter->ToString(command->initiator))
|
|
.toLocal8Bit());
|
|
emit handler->cecEvent(CECEvent::On);
|
|
}
|
|
if (command->opcode == CEC::CEC_OPCODE_STANDBY)
|
|
{
|
|
Info(handler->_logger, QString("CEC source deactivated: %1")
|
|
.arg(adapter->ToString(command->initiator))
|
|
.toLocal8Bit());
|
|
emit handler->cecEvent(CECEvent::Off);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CECHandler::onCecSourceActivated(void * context, const CECLogicalAddress address, const uint8_t activated)
|
|
{
|
|
/* We use CECHandler::onCecCommandReceived for
|
|
* source activated/deactivated notifications. */
|
|
|
|
#ifdef VERBOSE_CEC
|
|
CECHandler * handler = qobject_cast<CECHandler*>(static_cast<QObject*>(context));
|
|
if (!handler)
|
|
return;
|
|
|
|
CECAdapter * adapter = handler->_cecAdapter;
|
|
|
|
Debug(handler->_logger, QString("CEC source %1 : %2")
|
|
.arg(activated ? "activated" : "deactivated")
|
|
.arg(adapter->ToString(address))
|
|
.toLocal8Bit());
|
|
#endif
|
|
}
|
|
|
|
|