mirror of
https://github.com/hyperion-project/hyperion.ng.git
synced 2023-10-10 13:36:59 +02:00
508 lines
11 KiB
C++
508 lines
11 KiB
C++
#include <leddevice/LedDevice.h>
|
|
|
|
//QT include
|
|
#include <QResource>
|
|
#include <QStringList>
|
|
#include <QDir>
|
|
#include <QDateTime>
|
|
#include <QEventLoop>
|
|
#include <QTimer>
|
|
#include <QDateTime>
|
|
|
|
#include "hyperion/Hyperion.h"
|
|
#include <utils/JsonUtils.h>
|
|
|
|
//std includes
|
|
#include <sstream>
|
|
#include <iomanip>
|
|
|
|
// Constants
|
|
namespace {
|
|
|
|
// Configuration settings
|
|
const char CONFIG_CURRENT_LED_COUNT[] = "currentLedCount";
|
|
const char CONFIG_COLOR_ORDER[] = "colorOrder";
|
|
const char CONFIG_AUTOSTART[] = "autoStart";
|
|
const char CONFIG_LATCH_TIME[] = "latchTime";
|
|
const char CONFIG_REWRITE_TIME[] = "rewriteTime";
|
|
|
|
int DEFAULT_LED_COUNT = 1;
|
|
const char DEFAULT_COLOR_ORDER[] = "RGB";
|
|
const bool DEFAULT_IS_AUTOSTART = true;
|
|
|
|
} //End of constants
|
|
|
|
LedDevice::LedDevice(const QJsonObject& deviceConfig, QObject* parent)
|
|
: QObject(parent)
|
|
, _devConfig(deviceConfig)
|
|
, _log(Logger::getInstance("LEDDEVICE"))
|
|
, _ledBuffer(0)
|
|
, _refreshTimer(nullptr)
|
|
, _refreshTimerInterval_ms(0)
|
|
, _latchTime_ms(0)
|
|
, _ledCount(0)
|
|
, _isRestoreOrigState(false)
|
|
, _isEnabled(false)
|
|
, _isDeviceInitialised(false)
|
|
, _isDeviceReady(false)
|
|
, _isOn(false)
|
|
, _isDeviceInError(false)
|
|
, _isInSwitchOff (false)
|
|
, _lastWriteTime(QDateTime::currentDateTime())
|
|
, _isRefreshEnabled (false)
|
|
, _isAutoStart(true)
|
|
{
|
|
_activeDeviceType = deviceConfig["type"].toString("UNSPECIFIED").toLower();
|
|
}
|
|
|
|
LedDevice::~LedDevice()
|
|
{
|
|
delete _refreshTimer;
|
|
}
|
|
|
|
void LedDevice::start()
|
|
{
|
|
Info(_log, "Start LedDevice '%s'.", QSTRING_CSTR(_activeDeviceType));
|
|
|
|
// setup refreshTimer
|
|
if ( _refreshTimer == nullptr )
|
|
{
|
|
_refreshTimer = new QTimer(this);
|
|
_refreshTimer->setTimerType(Qt::PreciseTimer);
|
|
_refreshTimer->setInterval( _refreshTimerInterval_ms );
|
|
connect(_refreshTimer, &QTimer::timeout, this, &LedDevice::rewriteLEDs );
|
|
}
|
|
|
|
close();
|
|
|
|
_isDeviceInitialised = false;
|
|
// General initialisation and configuration of LedDevice
|
|
if ( init(_devConfig) )
|
|
{
|
|
// Everything is OK -> enable device
|
|
_isDeviceInitialised = true;
|
|
if (_isAutoStart)
|
|
{
|
|
this->enable();
|
|
}
|
|
}
|
|
}
|
|
|
|
void LedDevice::stop()
|
|
{
|
|
this->disable();
|
|
this->stopRefreshTimer();
|
|
Info(_log, " Stopped LedDevice '%s'", QSTRING_CSTR(_activeDeviceType) );
|
|
}
|
|
|
|
int LedDevice::open()
|
|
{
|
|
_isDeviceReady = true;
|
|
int retval = 0;
|
|
|
|
return retval;
|
|
}
|
|
|
|
int LedDevice::close()
|
|
{
|
|
_isDeviceReady = false;
|
|
int retval = 0;
|
|
|
|
return retval;
|
|
}
|
|
|
|
void LedDevice::setInError(const QString& errorMsg)
|
|
{
|
|
_isDeviceInError = true;
|
|
_isDeviceReady = false;
|
|
_isEnabled = false;
|
|
this->stopRefreshTimer();
|
|
|
|
Error(_log, "Device disabled, device '%s' signals error: '%s'", QSTRING_CSTR(_activeDeviceType), QSTRING_CSTR(errorMsg));
|
|
emit enableStateChanged(_isEnabled);
|
|
}
|
|
|
|
void LedDevice::enable()
|
|
{
|
|
if ( !_isEnabled )
|
|
{
|
|
_isDeviceInError = false;
|
|
|
|
if ( ! _isDeviceReady )
|
|
{
|
|
open();
|
|
}
|
|
|
|
if ( _isDeviceReady )
|
|
{
|
|
_isEnabled = true;
|
|
if ( switchOn() )
|
|
{
|
|
emit enableStateChanged(_isEnabled);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void LedDevice::disable()
|
|
{
|
|
if ( _isEnabled )
|
|
{
|
|
_isEnabled = false;
|
|
this->stopRefreshTimer();
|
|
|
|
switchOff();
|
|
close();
|
|
|
|
emit enableStateChanged(_isEnabled);
|
|
}
|
|
}
|
|
|
|
void LedDevice::setActiveDeviceType(const QString& deviceType)
|
|
{
|
|
_activeDeviceType = deviceType;
|
|
}
|
|
|
|
bool LedDevice::init(const QJsonObject &deviceConfig)
|
|
{
|
|
Debug(_log, "deviceConfig: [%s]", QString(QJsonDocument(_devConfig).toJson(QJsonDocument::Compact)).toUtf8().constData() );
|
|
|
|
_colorOrder = deviceConfig[CONFIG_COLOR_ORDER].toString(DEFAULT_COLOR_ORDER);
|
|
_isAutoStart = deviceConfig[CONFIG_AUTOSTART].toBool(DEFAULT_IS_AUTOSTART);
|
|
|
|
setLedCount( deviceConfig[CONFIG_CURRENT_LED_COUNT].toInt(DEFAULT_LED_COUNT) ); // property injected to reflect real led count
|
|
setLatchTime( deviceConfig[CONFIG_LATCH_TIME].toInt( _latchTime_ms ) );
|
|
setRewriteTime ( deviceConfig[CONFIG_REWRITE_TIME].toInt( _refreshTimerInterval_ms) );
|
|
|
|
return true;
|
|
}
|
|
|
|
void LedDevice::startRefreshTimer()
|
|
{
|
|
if ( _isDeviceReady && _isEnabled )
|
|
{
|
|
_refreshTimer->start();
|
|
}
|
|
}
|
|
|
|
void LedDevice::stopRefreshTimer()
|
|
{
|
|
if ( _refreshTimer != nullptr )
|
|
{
|
|
_refreshTimer->stop();
|
|
}
|
|
}
|
|
|
|
int LedDevice::updateLeds(const std::vector<ColorRgb>& ledValues)
|
|
{
|
|
int retval = 0;
|
|
if ( !_isEnabled || !_isOn || !_isDeviceReady || _isDeviceInError )
|
|
{
|
|
//std::cout << "LedDevice::updateLeds(), LedDevice NOT ready! ";
|
|
retval = -1;
|
|
}
|
|
else
|
|
{
|
|
qint64 elapsedTimeMs = _lastWriteTime.msecsTo( QDateTime::currentDateTime() );
|
|
if (_latchTime_ms == 0 || elapsedTimeMs >= _latchTime_ms)
|
|
{
|
|
//std::cout << "LedDevice::updateLeds(), Elapsed time since last write (" << elapsedTimeMs << ") ms > _latchTime_ms (" << _latchTime_ms << ") ms" << std::endl;
|
|
retval = write(ledValues);
|
|
_lastWriteTime = QDateTime::currentDateTime();
|
|
|
|
// if device requires refreshing, save Led-Values and restart the timer
|
|
if ( _isRefreshEnabled && _isEnabled )
|
|
{
|
|
this->startRefreshTimer();
|
|
_lastLedValues = ledValues;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//std::cout << "LedDevice::updateLeds(), Skip write. elapsedTime (" << elapsedTimeMs << ") ms < _latchTime_ms (" << _latchTime_ms << ") ms" << std::endl;
|
|
if ( _isRefreshEnabled )
|
|
{
|
|
//Stop timer to allow for next non-refresh update
|
|
this->stopRefreshTimer();
|
|
}
|
|
}
|
|
}
|
|
return retval;
|
|
}
|
|
|
|
int LedDevice::rewriteLEDs()
|
|
{
|
|
int retval = -1;
|
|
|
|
if ( _isDeviceReady && _isEnabled )
|
|
{
|
|
// qint64 elapsedTimeMs = _lastWriteTime.msecsTo(QDateTime::currentDateTime());
|
|
// std::cout << "LedDevice::rewriteLEDs(): Rewrite LEDs now, elapsedTime [" << elapsedTimeMs << "] ms" << std::endl;
|
|
// //:TESTING: Inject "white" output records to differentiate from normal writes
|
|
// _lastLedValues.clear();
|
|
// _lastLedValues.resize(static_cast<unsigned long>(_ledCount), ColorRgb::WHITE);
|
|
// printLedValues(_lastLedValues);
|
|
// //:TESTING:
|
|
|
|
retval = write(_lastLedValues);
|
|
_lastWriteTime = QDateTime::currentDateTime();
|
|
}
|
|
else
|
|
{
|
|
// If Device is not ready stop timer
|
|
this->stopRefreshTimer();
|
|
}
|
|
return retval;
|
|
}
|
|
|
|
int LedDevice::writeBlack(int numberOfWrites)
|
|
{
|
|
return writeColor(ColorRgb::BLACK, numberOfWrites);
|
|
}
|
|
|
|
int LedDevice::writeColor(const ColorRgb& color, int numberOfWrites)
|
|
{
|
|
int rc = -1;
|
|
|
|
for (int i = 0; i < numberOfWrites; i++)
|
|
{
|
|
if (_latchTime_ms > 0)
|
|
{
|
|
// Wait latch time before writing black
|
|
QEventLoop loop;
|
|
QTimer::singleShot(_latchTime_ms, &loop, &QEventLoop::quit);
|
|
loop.exec();
|
|
}
|
|
_lastLedValues = std::vector<ColorRgb>(static_cast<unsigned long>(_ledCount),color);
|
|
rc = write(_lastLedValues);
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
bool LedDevice::switchOn()
|
|
{
|
|
bool rc = false;
|
|
|
|
if ( _isOn )
|
|
{
|
|
rc = true;
|
|
}
|
|
else
|
|
{
|
|
if ( _isEnabled &&_isDeviceInitialised )
|
|
{
|
|
if ( storeState() )
|
|
{
|
|
if ( powerOn() )
|
|
{
|
|
_isOn = true;
|
|
_isInSwitchOff = false;
|
|
rc = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
bool LedDevice::switchOff()
|
|
{
|
|
bool rc = false;
|
|
|
|
if ( !_isOn )
|
|
{
|
|
rc = true;
|
|
}
|
|
else
|
|
{
|
|
if ( _isDeviceInitialised )
|
|
{
|
|
// Disable device to ensure no standard Led updates are written/processed
|
|
_isOn = false;
|
|
_isInSwitchOff = true;
|
|
|
|
rc = true;
|
|
|
|
if ( _isDeviceReady )
|
|
{
|
|
if ( _isRestoreOrigState )
|
|
{
|
|
//Restore devices state
|
|
restoreState();
|
|
}
|
|
else
|
|
{
|
|
powerOff();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
bool LedDevice::powerOff()
|
|
{
|
|
bool rc = false;
|
|
|
|
// Simulate power-off by writing a final "Black" to have a defined outcome
|
|
if ( writeBlack() >= 0 )
|
|
{
|
|
rc = true;
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
bool LedDevice::powerOn()
|
|
{
|
|
bool rc = true;
|
|
return rc;
|
|
}
|
|
|
|
bool LedDevice::storeState()
|
|
{
|
|
bool rc = true;
|
|
|
|
if ( _isRestoreOrigState )
|
|
{
|
|
// Save device's original state
|
|
// _originalStateValues = get device's state;
|
|
// store original power on/off state, if available
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
bool LedDevice::restoreState()
|
|
{
|
|
bool rc = true;
|
|
|
|
if ( _isRestoreOrigState )
|
|
{
|
|
// Restore device's original state
|
|
// update device using _originalStateValues
|
|
// update original power on/off state, if supported
|
|
}
|
|
return rc;
|
|
}
|
|
|
|
QJsonObject LedDevice::discover(const QJsonObject& /*params*/)
|
|
{
|
|
QJsonObject devicesDiscovered;
|
|
|
|
devicesDiscovered.insert("ledDeviceType", _activeDeviceType);
|
|
|
|
QJsonArray deviceList;
|
|
devicesDiscovered.insert("devices", deviceList);
|
|
|
|
Debug(_log, "devicesDiscovered: [%s]", QString(QJsonDocument(devicesDiscovered).toJson(QJsonDocument::Compact)).toUtf8().constData() );
|
|
return devicesDiscovered;
|
|
}
|
|
|
|
QString LedDevice::discoverFirst()
|
|
{
|
|
QString deviceDiscovered;
|
|
|
|
Debug(_log, "deviceDiscovered: [%s]", QSTRING_CSTR(deviceDiscovered) );
|
|
return deviceDiscovered;
|
|
}
|
|
|
|
|
|
QJsonObject LedDevice::getProperties(const QJsonObject& params)
|
|
{
|
|
Debug(_log, "params: [%s]", QString(QJsonDocument(params).toJson(QJsonDocument::Compact)).toUtf8().constData() );
|
|
|
|
QJsonObject properties;
|
|
|
|
QJsonObject deviceProperties;
|
|
properties.insert("properties", deviceProperties);
|
|
|
|
Debug(_log, "properties: [%s]", QString(QJsonDocument(properties).toJson(QJsonDocument::Compact)).toUtf8().constData() );
|
|
|
|
return properties;
|
|
}
|
|
|
|
void LedDevice::setLedCount(int ledCount)
|
|
{
|
|
assert(ledCount >= 0);
|
|
_ledCount = ledCount;
|
|
_ledRGBCount = _ledCount * sizeof(ColorRgb);
|
|
_ledRGBWCount = _ledCount * sizeof(ColorRgbw);
|
|
}
|
|
|
|
void LedDevice::setLatchTime( int latchTime_ms )
|
|
{
|
|
assert(latchTime_ms >= 0);
|
|
_latchTime_ms = latchTime_ms;
|
|
Debug(_log, "LatchTime updated to %dms", _latchTime_ms);
|
|
}
|
|
|
|
void LedDevice::setRewriteTime( int rewriteTime_ms )
|
|
{
|
|
assert(rewriteTime_ms >= 0);
|
|
|
|
//Check, if refresh timer was not initialised due to getProperties/identify sceanrios
|
|
if (_refreshTimer != nullptr)
|
|
{
|
|
_refreshTimerInterval_ms = rewriteTime_ms;
|
|
|
|
if (_refreshTimerInterval_ms > 0)
|
|
{
|
|
|
|
_isRefreshEnabled = true;
|
|
|
|
if (_refreshTimerInterval_ms <= _latchTime_ms)
|
|
{
|
|
int new_refresh_timer_interval = _latchTime_ms + 10;
|
|
Warning(_log, "latchTime(%d) is bigger/equal rewriteTime(%d), set rewriteTime to %dms", _latchTime_ms, _refreshTimerInterval_ms, new_refresh_timer_interval);
|
|
_refreshTimerInterval_ms = new_refresh_timer_interval;
|
|
_refreshTimer->setInterval(_refreshTimerInterval_ms);
|
|
}
|
|
|
|
Debug(_log, "Refresh interval = %dms", _refreshTimerInterval_ms);
|
|
_refreshTimer->setInterval(_refreshTimerInterval_ms);
|
|
|
|
_lastWriteTime = QDateTime::currentDateTime();
|
|
}
|
|
|
|
Debug(_log, "RewriteTime updated to %dms", _refreshTimerInterval_ms);
|
|
}
|
|
}
|
|
|
|
void LedDevice::printLedValues(const std::vector<ColorRgb>& ledValues)
|
|
{
|
|
std::cout << "LedValues [" << ledValues.size() <<"] [";
|
|
for (const ColorRgb& color : ledValues)
|
|
{
|
|
std::cout << color;
|
|
}
|
|
std::cout << "]" << std::endl;
|
|
}
|
|
|
|
QString LedDevice::uint8_t_to_hex_string(const uint8_t * data, const int size, int number) const
|
|
{
|
|
if ( number <= 0 || number > size)
|
|
{
|
|
number = size;
|
|
}
|
|
|
|
QByteArray bytes (reinterpret_cast<const char*>(data), number);
|
|
#if (QT_VERSION >= QT_VERSION_CHECK(5, 9, 0))
|
|
return bytes.toHex(':');
|
|
#else
|
|
return bytes.toHex();
|
|
#endif
|
|
}
|
|
|
|
QString LedDevice::toHex(const QByteArray& data, int number) const
|
|
{
|
|
if ( number <= 0 || number > data.size())
|
|
{
|
|
number = data.size();
|
|
}
|
|
|
|
#if (QT_VERSION >= QT_VERSION_CHECK(5, 9, 0))
|
|
return data.left(number).toHex(':');
|
|
#else
|
|
return data.left(number).toHex();
|
|
#endif
|
|
}
|