Hyperiond refactoring + preparation for ip restriction settings (#12)

* make hyperion to singelton.
remove arguments for config and hyperion - both are gettable via Hyperion::getInstance

* refactor hyperiond

* remove qt4 comapt
make zeroconf mandatory
refactor hyperiond

* xbmcchecker is now a singleton

* cleanup in hyperiond
zeroconf switchable between static and shared linking

* fix xbmcchecker

* extensive refactoring of hyperiond
webserver: make client ip information availabel in request object - preparation for ip filters
proto/json server, use hyperion::getInstance instead of hyperion argument

* move creation of hyperion core into hyperionDeamon class
cleanup
This commit is contained in:
redPanther 2016-06-19 00:56:47 +02:00 committed by brindosch
parent 0e5607db65
commit 4533b34606
14 changed files with 261 additions and 226 deletions

View File

@ -27,7 +27,7 @@ public:
/// @param hyperion Hyperion instance /// @param hyperion Hyperion instance
/// @param port port number on which to start listening for connections /// @param port port number on which to start listening for connections
/// ///
JsonServer(Hyperion * hyperion, uint16_t port = 19444); JsonServer(uint16_t port = 19444);
~JsonServer(); ~JsonServer();
/// ///

View File

@ -41,7 +41,7 @@ public:
/// @param hyperion Hyperion instance /// @param hyperion Hyperion instance
/// @param port port number on which to start listening for connections /// @param port port number on which to start listening for connections
/// ///
ProtoServer(Hyperion * hyperion, uint16_t port = 19445); ProtoServer(uint16_t port = 19445);
~ProtoServer(); ~ProtoServer();
/// ///

View File

@ -5,9 +5,9 @@
#include <jsonserver/JsonServer.h> #include <jsonserver/JsonServer.h>
#include "JsonClientConnection.h" #include "JsonClientConnection.h"
JsonServer::JsonServer(Hyperion *hyperion, uint16_t port) : JsonServer::JsonServer(uint16_t port) :
QObject(), QObject(),
_hyperion(hyperion), _hyperion(Hyperion::getInstance()),
_server(), _server(),
_openConnections() _openConnections()
{ {

View File

@ -7,14 +7,14 @@
#include "protoserver/ProtoConnection.h" #include "protoserver/ProtoConnection.h"
#include "ProtoClientConnection.h" #include "ProtoClientConnection.h"
ProtoServer::ProtoServer(Hyperion *hyperion, uint16_t port) : ProtoServer::ProtoServer(uint16_t port) :
QObject(), QObject(),
_hyperion(hyperion), _hyperion(Hyperion::getInstance()),
_server(), _server(),
_openConnections() _openConnections()
{ {
MessageForwarder * forwarder = hyperion->getForwarder(); MessageForwarder * forwarder = _hyperion->getForwarder();
QStringList slaves = forwarder->getProtoSlaves(); QStringList slaves = forwarder->getProtoSlaves();
for (int i = 0; i < slaves.size(); ++i) { for (int i = 0; i < slaves.size(); ++i) {

View File

@ -4,6 +4,7 @@
#include <QByteArray> #include <QByteArray>
#include "CgiHandler.h" #include "CgiHandler.h"
#include "QtHttpHeader.h"
CgiHandler::CgiHandler (Hyperion * hyperion, QObject * parent) CgiHandler::CgiHandler (Hyperion * hyperion, QObject * parent)
: QObject(parent) : QObject(parent)
@ -16,10 +17,15 @@ CgiHandler::~CgiHandler()
{ {
} }
void CgiHandler::exec(const QStringList & args, QtHttpReply * reply) void CgiHandler::exec(const QStringList & args, QtHttpRequest * request, QtHttpReply * reply)
{ {
try try
{ {
// QByteArray header = reply->getHeader(QtHttpHeader::Host);
// QtHttpRequest::ClientInfo info = request->getClientInfo();
// qDebug() << info.clientAddress.toString();
// qDebug() << info.serverAddress.toString();
cmd_cfg_jsonserver(args,reply); cmd_cfg_jsonserver(args,reply);
cmd_cfg_hyperion(args,reply); cmd_cfg_hyperion(args,reply);
throw 1; throw 1;

View File

@ -9,6 +9,7 @@
#include <hyperion/Hyperion.h> #include <hyperion/Hyperion.h>
#include "QtHttpReply.h" #include "QtHttpReply.h"
#include "QtHttpRequest.h"
class CgiHandler : public QObject { class CgiHandler : public QObject {
Q_OBJECT Q_OBJECT
@ -17,7 +18,7 @@ public:
CgiHandler (Hyperion * hyperion, QObject * parent = NULL); CgiHandler (Hyperion * hyperion, QObject * parent = NULL);
virtual ~CgiHandler (void); virtual ~CgiHandler (void);
void exec(const QStringList & args, QtHttpReply * reply); void exec(const QStringList & args,QtHttpRequest * request, QtHttpReply * reply);
// cgi commands // cgi commands
void cmd_cfg_jsonserver(const QStringList & args, QtHttpReply * reply); void cmd_cfg_jsonserver(const QStringList & args, QtHttpReply * reply);

View File

@ -10,6 +10,7 @@
#include <QStringBuilder> #include <QStringBuilder>
#include <QStringList> #include <QStringList>
#include <QDateTime> #include <QDateTime>
#include <QHostAddress>
const QByteArray & QtHttpClientWrapper::CRLF = QByteArrayLiteral ("\r\n"); const QByteArray & QtHttpClientWrapper::CRLF = QByteArrayLiteral ("\r\n");
@ -54,6 +55,7 @@ void QtHttpClientWrapper::onClientDataReceived (void) {
// << "url :" << url // << "url :" << url
// << "version :" << version; // << "version :" << version;
m_currentRequest = new QtHttpRequest (m_serverHandle); m_currentRequest = new QtHttpRequest (m_serverHandle);
m_currentRequest->setClientInfo(m_sockClient->localAddress(), m_sockClient->peerAddress());
m_currentRequest->setUrl (QUrl (url)); m_currentRequest->setUrl (QUrl (url));
m_currentRequest->setCommand (command); m_currentRequest->setCommand (command);
m_parsingStatus = AwaitingHeaders; m_parsingStatus = AwaitingHeaders;

View File

@ -23,6 +23,10 @@ QString QtHttpRequest::getCommand (void) const {
return m_command; return m_command;
} }
QtHttpRequest::ClientInfo QtHttpRequest::getClientInfo (void) const {
return m_clientInfo;
}
int QtHttpRequest::getRawDataSize (void) const { int QtHttpRequest::getRawDataSize (void) const {
return m_data.size (); return m_data.size ();
} }
@ -48,6 +52,11 @@ void QtHttpRequest::setCommand (const QString & command) {
m_command = command; m_command = command;
} }
void QtHttpRequest::setClientInfo (const QHostAddress & server, const QHostAddress & client) {
m_clientInfo.serverAddress = server;
m_clientInfo.clientAddress = client;
}
void QtHttpRequest::addHeader (const QByteArray & header, const QByteArray & value) { void QtHttpRequest::addHeader (const QByteArray & header, const QByteArray & value) {
QByteArray key = header.trimmed (); QByteArray key = header.trimmed ();
if (!key.isEmpty ()) { if (!key.isEmpty ()) {

View File

@ -6,6 +6,7 @@
#include <QByteArray> #include <QByteArray>
#include <QHash> #include <QHash>
#include <QUrl> #include <QUrl>
#include <QHostAddress>
class QtHttpServer; class QtHttpServer;
@ -15,17 +16,24 @@ class QtHttpRequest : public QObject {
public: public:
explicit QtHttpRequest (QtHttpServer * parent); explicit QtHttpRequest (QtHttpServer * parent);
struct ClientInfo {
QHostAddress serverAddress;
QHostAddress clientAddress;
};
int getRawDataSize (void) const; int getRawDataSize (void) const;
QUrl getUrl (void) const; QUrl getUrl (void) const;
QString getCommand (void) const; QString getCommand (void) const;
QByteArray getRawData (void) const; QByteArray getRawData (void) const;
QList<QByteArray> getHeadersList (void) const; QList<QByteArray> getHeadersList (void) const;
ClientInfo getClientInfo (void) const;
QByteArray getHeader (const QByteArray & header) const; QByteArray getHeader (const QByteArray & header) const;
public slots: public slots:
void setUrl (const QUrl & url); void setUrl (const QUrl & url);
void setCommand (const QString & command); void setCommand (const QString & command);
void setClientInfo (const QHostAddress & server, const QHostAddress & client);
void addHeader (const QByteArray & header, const QByteArray & value); void addHeader (const QByteArray & header, const QByteArray & value);
void appendRawData (const QByteArray & data); void appendRawData (const QByteArray & data);
@ -35,6 +43,7 @@ private:
QByteArray m_data; QByteArray m_data;
QtHttpServer * m_serverHandle; QtHttpServer * m_serverHandle;
QHash<QByteArray, QByteArray> m_headersHash; QHash<QByteArray, QByteArray> m_headersHash;
ClientInfo m_clientInfo;
}; };
#endif // QTHTTPREQUEST_H #endif // QTHTTPREQUEST_H

View File

@ -66,7 +66,7 @@ void StaticFileServing::onRequestNeedsReply (QtHttpRequest * request, QtHttpRepl
uri_parts.removeAt(0); uri_parts.removeAt(0);
try try
{ {
_cgi.exec(uri_parts, reply); _cgi.exec(uri_parts, request, reply);
} }
catch(...) catch(...)
{ {

View File

@ -16,7 +16,7 @@ class StaticFileServing : public QObject {
Q_OBJECT Q_OBJECT
public: public:
explicit StaticFileServing (Hyperion *hyperion, QString baseUrl, quint16 port, QObject * parent = NULL); explicit StaticFileServing (Hyperion *hyperion, QString baseUrl, quint16 port, QObject * parent = nullptr);
virtual ~StaticFileServing (void); virtual ~StaticFileServing (void);
public slots: public slots:

View File

@ -9,7 +9,6 @@
#include "HyperionConfig.h" #include "HyperionConfig.h"
#include <utils/jsonschema/JsonFactory.h> #include <utils/jsonschema/JsonFactory.h>
#include <utils/Logger.h>
#include <hyperion/Hyperion.h> #include <hyperion/Hyperion.h>
#include <effectengine/EffectEngine.h> #include <effectengine/EffectEngine.h>
@ -22,15 +21,94 @@
#include "hyperiond.h" #include "hyperiond.h"
void startBootsequence() HyperionDaemon::HyperionDaemon(std::string configFile, QObject *parent)
: QObject(parent)
, _log(Logger::getInstance("MAIN"))
, _xbmcVideoChecker(nullptr)
, _jsonServer(nullptr)
, _protoServer(nullptr)
, _boblightServer(nullptr)
, _v4l2Grabber(nullptr)
, _dispmanx(nullptr)
, _amlGrabber(nullptr)
, _fbGrabber(nullptr)
, _osxGrabber(nullptr)
, _webConfig(nullptr)
{
loadConfig(configFile);
Hyperion::initInstance(_config, configFile);
Info(_log, "Hyperion started and initialised");
}
HyperionDaemon::~HyperionDaemon()
{
delete _amlGrabber;
delete _dispmanx;
delete _fbGrabber;
delete _osxGrabber;
delete _v4l2Grabber;
delete _xbmcVideoChecker;
delete _jsonServer;
delete _protoServer;
delete _boblightServer;
delete _webConfig;
}
void HyperionDaemon::run()
{
startBootsequence();
createXBMCVideoChecker();
// ---- network services -----
startNetworkServices();
_webConfig = new WebConfig(this);
// ---- grabber -----
createGrabberV4L2();
createGrabberDispmanx();
createGrabberAmlogic();
createGrabberFramebuffer();
createGrabberDispmanx();
#if !defined(ENABLE_DISPMANX) && !defined(ENABLE_OSX) && !defined(ENABLE_FB)
ErrorIf(_config.isMember("framegrabber"), log, "No grabber can be instantiated, because all grabbers have been left out from the build");
#endif
}
void HyperionDaemon::loadConfig(const std::string & configFile)
{
Info(_log, "Selected configuration file: %s", configFile.c_str() );
// make sure the resources are loaded (they may be left out after static linking)
Q_INIT_RESOURCE(resource);
// read the json schema from the resource
QResource schemaData(":/hyperion-schema");
assert(schemaData.isValid());
Json::Reader jsonReader;
Json::Value schemaJson;
if (!jsonReader.parse(reinterpret_cast<const char *>(schemaData.data()), reinterpret_cast<const char *>(schemaData.data()) + schemaData.size(), schemaJson, false))
{
throw std::runtime_error("ERROR: Json schema wrong: " + jsonReader.getFormattedErrorMessages()) ;
}
JsonSchemaChecker schemaChecker;
schemaChecker.setSchema(schemaJson);
_config = JsonFactory::readJson(configFile);
schemaChecker.validate(_config);
}
void HyperionDaemon::startBootsequence()
{ {
Hyperion *hyperion = Hyperion::getInstance(); Hyperion *hyperion = Hyperion::getInstance();
const Json::Value &config = hyperion->getJsonConfig();
// create boot sequence if the configuration is present // create boot sequence if the configuration is present
if (config.isMember("bootsequence")) if (_config.isMember("bootsequence"))
{ {
const Json::Value effectConfig = config["bootsequence"]; const Json::Value effectConfig = _config["bootsequence"];
// Get the parameters for the bootsequence // Get the parameters for the bootsequence
const std::string effectName = effectConfig["effect"].asString(); const std::string effectName = effectConfig["effect"].asString();
@ -75,15 +153,13 @@ void startBootsequence()
} }
// create XBMC video checker if the configuration is present // create XBMC video checker if the _configuration is present
XBMCVideoChecker* createXBMCVideoChecker() void HyperionDaemon::createXBMCVideoChecker()
{ {
XBMCVideoChecker* xbmcVideoChecker = nullptr; if (_config.isMember("xbmcVideoChecker"))
const Json::Value &config = Hyperion::getInstance()->getJsonConfig();
if (config.isMember("xbmcVideoChecker"))
{ {
const Json::Value & videoCheckerConfig = config["xbmcVideoChecker"]; const Json::Value & videoCheckerConfig = _config["xbmcVideoChecker"];
xbmcVideoChecker = XBMCVideoChecker::initInstance( _xbmcVideoChecker = XBMCVideoChecker::initInstance(
videoCheckerConfig["xbmcAddress"].asString(), videoCheckerConfig["xbmcAddress"].asString(),
videoCheckerConfig["xbmcTcpPort"].asUInt(), videoCheckerConfig["xbmcTcpPort"].asUInt(),
videoCheckerConfig["grabVideo"].asBool(), videoCheckerConfig["grabVideo"].asBool(),
@ -94,137 +170,133 @@ XBMCVideoChecker* createXBMCVideoChecker()
videoCheckerConfig.get("grabScreensaver", true).asBool(), videoCheckerConfig.get("grabScreensaver", true).asBool(),
videoCheckerConfig.get("enable3DDetection", true).asBool()); videoCheckerConfig.get("enable3DDetection", true).asBool());
xbmcVideoChecker->start(); _xbmcVideoChecker->start();
Info(Logger::getInstance("MAIN"), "Kodi checker created and started"); Info(_log, "Kodi checker created and started");
} }
return xbmcVideoChecker;
} }
void startNetworkServices(JsonServer* &jsonServer, ProtoServer* &protoServer, BoblightServer* &boblightServer) void HyperionDaemon::startNetworkServices()
{ {
Hyperion *hyperion = Hyperion::getInstance(); Hyperion *hyperion = Hyperion::getInstance();
XBMCVideoChecker* xbmcVideoChecker = XBMCVideoChecker::getInstance(); XBMCVideoChecker* xbmcVideoChecker = XBMCVideoChecker::getInstance();
const Json::Value &config = hyperion->getJsonConfig();
// Create Json server if configuration is present // Create Json server if configuration is present
unsigned int jsonPort = 19444; unsigned int jsonPort = 19444;
if (config.isMember("jsonServer")) if (_config.isMember("jsonServer"))
{ {
const Json::Value & jsonServerConfig = config["jsonServer"]; const Json::Value & jsonServerConfig = _config["jsonServer"];
//jsonEnable = jsonServerConfig.get("enable", true).asBool(); //jsonEnable = jsonServerConfig.get("enable", true).asBool();
jsonPort = jsonServerConfig.get("port", jsonPort).asUInt(); jsonPort = jsonServerConfig.get("port", jsonPort).asUInt();
} }
jsonServer = new JsonServer(hyperion, jsonPort ); _jsonServer = new JsonServer(jsonPort );
Info(Logger::getInstance("MAIN"), "Json server created and started on port %d", jsonServer->getPort()); Info(_log, "Json server created and started on port %d", _jsonServer->getPort());
// Create Proto server if configuration is present // Create Proto server if configuration is present
unsigned int protoPort = 19445; unsigned int protoPort = 19445;
if (config.isMember("protoServer")) if (_config.isMember("protoServer"))
{ {
const Json::Value & protoServerConfig = config["protoServer"]; const Json::Value & protoServerConfig = _config["protoServer"];
//protoEnable = protoServerConfig.get("enable", true).asBool(); //protoEnable = protoServerConfig.get("enable", true).asBool();
protoPort = protoServerConfig.get("port", protoPort).asUInt(); protoPort = protoServerConfig.get("port", protoPort).asUInt();
} }
protoServer = new ProtoServer(hyperion, protoPort ); _protoServer = new ProtoServer(protoPort );
if (xbmcVideoChecker != nullptr) if (xbmcVideoChecker != nullptr)
{ {
QObject::connect(xbmcVideoChecker, SIGNAL(grabbingMode(GrabbingMode)), protoServer, SIGNAL(grabbingMode(GrabbingMode))); QObject::connect(xbmcVideoChecker, SIGNAL(grabbingMode(GrabbingMode)), _protoServer, SIGNAL(grabbingMode(GrabbingMode)));
QObject::connect(xbmcVideoChecker, SIGNAL(videoMode(VideoMode)), protoServer, SIGNAL(videoMode(VideoMode))); QObject::connect(xbmcVideoChecker, SIGNAL(videoMode(VideoMode)), _protoServer, SIGNAL(videoMode(VideoMode)));
} }
Info(Logger::getInstance("MAIN"), "Proto server created and started on port %d", protoServer->getPort()); Info(_log, "Proto server created and started on port %d", _protoServer->getPort());
const Json::Value & deviceConfig = config["device"]; // Create Boblight server if configuration is present
if (_config.isMember("boblightServer"))
{
const Json::Value & boblightServerConfig = _config["boblightServer"];
_boblightServer = new BoblightServer(hyperion, boblightServerConfig.get("priority",900).asInt(), boblightServerConfig["port"].asUInt());
Info(_log, "Boblight server created and started on port %d", _boblightServer->getPort());
}
// zeroconf
const Json::Value & deviceConfig = _config["device"];
const std::string deviceName = deviceConfig.get("name", "").asString(); const std::string deviceName = deviceConfig.get("name", "").asString();
const std::string hostname = QHostInfo::localHostName().toStdString(); const std::string hostname = QHostInfo::localHostName().toStdString();
// zeroconf json
std::string mDNSDescr_json = hostname; std::string mDNSDescr_json = hostname;
std::string mDNSService_json = "_hyperiond_json._tcp"; std::string mDNSService_json = "_hyperiond_json._tcp";
if (config.isMember("jsonServer")) if (_config.isMember("jsonServer"))
{ {
const Json::Value & jsonServerConfig = config["jsonServer"]; const Json::Value & jsonServerConfig = _config["jsonServer"];
mDNSDescr_json = jsonServerConfig.get("mDNSDescr", mDNSDescr_json).asString(); mDNSDescr_json = jsonServerConfig.get("mDNSDescr", mDNSDescr_json).asString();
mDNSService_json = jsonServerConfig.get("mDNSService", mDNSService_json).asString(); mDNSService_json = jsonServerConfig.get("mDNSService", mDNSService_json).asString();
} }
BonjourServiceRegister *bonjourRegister_json = new BonjourServiceRegister(); BonjourServiceRegister *bonjourRegister_json = new BonjourServiceRegister();
bonjourRegister_json->registerService(BonjourRecord((deviceName + " @ " + mDNSDescr_json).c_str(), mDNSService_json.c_str(), bonjourRegister_json->registerService(BonjourRecord((deviceName + " @ " + mDNSDescr_json).c_str(), mDNSService_json.c_str(),
QString()), jsonServer->getPort() ); QString()), _jsonServer->getPort() );
Info(Logger::getInstance("MAIN"), "Json mDNS responder started"); Info(_log, "Json mDNS responder started");
// zeroconf proto
std::string mDNSDescr_proto = hostname; std::string mDNSDescr_proto = hostname;
std::string mDNSService_proto = "_hyperiond_proto._tcp"; std::string mDNSService_proto = "_hyperiond_proto._tcp";
if (config.isMember("protoServer")) if (_config.isMember("protoServer"))
{ {
const Json::Value & protoServerConfig = config["protoServer"]; const Json::Value & protoServerConfig = _config["protoServer"];
mDNSDescr_proto = protoServerConfig.get("mDNSDescr", mDNSDescr_proto).asString(); mDNSDescr_proto = protoServerConfig.get("mDNSDescr", mDNSDescr_proto).asString();
mDNSService_proto = protoServerConfig.get("mDNSService", mDNSService_proto).asString(); mDNSService_proto = protoServerConfig.get("mDNSService", mDNSService_proto).asString();
} }
BonjourServiceRegister *bonjourRegister_proto = new BonjourServiceRegister(); BonjourServiceRegister *bonjourRegister_proto = new BonjourServiceRegister();
bonjourRegister_proto->registerService(BonjourRecord((deviceName + " @ " + mDNSDescr_proto).c_str(), mDNSService_proto.c_str(), bonjourRegister_proto->registerService(BonjourRecord((deviceName + " @ " + mDNSDescr_proto).c_str(), mDNSService_proto.c_str(),
QString()), protoServer->getPort() ); QString()), _protoServer->getPort() );
Info(Logger::getInstance("MAIN"), "Proto mDNS responder started"); Info(_log, "Proto mDNS responder started");
// Create Boblight server if configuration is present
if (config.isMember("boblightServer"))
{
const Json::Value & boblightServerConfig = config["boblightServer"];
boblightServer = new BoblightServer(hyperion, boblightServerConfig.get("priority",900).asInt(), boblightServerConfig["port"].asUInt());
Info(Logger::getInstance("MAIN"), "Boblight server created and started on port %d", boblightServer->getPort());
}
} }
DispmanxWrapper* createGrabberDispmanx(ProtoServer* &protoServer) void HyperionDaemon::createGrabberDispmanx()
{ {
DispmanxWrapper* dispmanx = nullptr;
#ifdef ENABLE_DISPMANX #ifdef ENABLE_DISPMANX
XBMCVideoChecker* xbmcVideoChecker = XBMCVideoChecker::getInstance();
const Json::Value &config = Hyperion::getInstance()->getJsonConfig();
// Construct and start the frame-grabber if the configuration is present // Construct and start the frame-grabber if the configuration is present
if (config.isMember("framegrabber")) if (_config.isMember("framegrabber"))
{ {
const Json::Value & frameGrabberConfig = config["framegrabber"]; const Json::Value & frameGrabberConfig = _config["framegrabber"];
dispmanx = new DispmanxWrapper( _dispmanx = new DispmanxWrapper(
frameGrabberConfig["width"].asUInt(), frameGrabberConfig["width"].asUInt(),
frameGrabberConfig["height"].asUInt(), frameGrabberConfig["height"].asUInt(),
frameGrabberConfig["frequency_Hz"].asUInt(), frameGrabberConfig["frequency_Hz"].asUInt(),
frameGrabberConfig.get("priority",900).asInt()); frameGrabberConfig.get("priority",900).asInt());
dispmanx->setCropping( _dispmanx->setCropping(
frameGrabberConfig.get("cropLeft", 0).asInt(), frameGrabberConfig.get("cropLeft", 0).asInt(),
frameGrabberConfig.get("cropRight", 0).asInt(), frameGrabberConfig.get("cropRight", 0).asInt(),
frameGrabberConfig.get("cropTop", 0).asInt(), frameGrabberConfig.get("cropTop", 0).asInt(),
frameGrabberConfig.get("cropBottom", 0).asInt()); frameGrabberConfig.get("cropBottom", 0).asInt());
if (xbmcVideoChecker != nullptr) if (_xbmcVideoChecker != nullptr)
{ {
QObject::connect(xbmcVideoChecker, SIGNAL(grabbingMode(GrabbingMode)), dispmanx, SLOT(setGrabbingMode(GrabbingMode))); QObject::connect(_xbmcVideoChecker, SIGNAL(grabbingMode(GrabbingMode)), _dispmanx, SLOT(setGrabbingMode(GrabbingMode)));
QObject::connect(xbmcVideoChecker, SIGNAL(videoMode(VideoMode)), dispmanx, SLOT(setVideoMode(VideoMode))); QObject::connect(_xbmcVideoChecker, SIGNAL(videoMode(VideoMode)), _dispmanx, SLOT(setVideoMode(VideoMode)));
} }
QObject::connect(dispmanx, SIGNAL(emitImage(int, const Image<ColorRgb>&, const int)), protoServer, SLOT(sendImageToProtoSlaves(int, const Image<ColorRgb>&, const int)) ); QObject::connect(_dispmanx, SIGNAL(emitImage(int, const Image<ColorRgb>&, const int)), _protoServer, SLOT(sendImageToProtoSlaves(int, const Image<ColorRgb>&, const int)) );
dispmanx->start(); _dispmanx->start();
Info(Logger::getInstance("MAIN"), "Frame grabber created and started"); Info(_log, "Frame grabber created and started");
} }
#else
ErrorIf(_config.isMember("framegrabber"), _log, "The dispmanx framegrabber can not be instantiated, because it has been left out from the build");
#endif #endif
return dispmanx;
} }
V4L2Wrapper* createGrabberV4L2(ProtoServer* &protoServer ) void HyperionDaemon::createGrabberV4L2()
{ {
// construct and start the v4l2 grabber if the configuration is present // construct and start the v4l2 grabber if the configuration is present
V4L2Wrapper* v4l2Grabber = nullptr;
#ifdef ENABLE_V4L2 #ifdef ENABLE_V4L2
const Json::Value &config = Hyperion::getInstance()->getJsonConfig(); if (_config.isMember("grabber-v4l2"))
if (config.isMember("grabber-v4l2"))
{ {
const Json::Value & grabberConfig = config["grabber-v4l2"]; const Json::Value & grabberConfig = _config["grabber-v4l2"];
v4l2Grabber = new V4L2Wrapper( _v4l2Grabber = new V4L2Wrapper(
grabberConfig.get("device", "/dev/video0").asString(), grabberConfig.get("device", "/dev/video0").asString(),
grabberConfig.get("input", 0).asInt(), grabberConfig.get("input", 0).asInt(),
parseVideoStandard(grabberConfig.get("standard", "no-change").asString()), parseVideoStandard(grabberConfig.get("standard", "no-change").asString()),
@ -237,118 +309,110 @@ V4L2Wrapper* createGrabberV4L2(ProtoServer* &protoServer )
grabberConfig.get("greenSignalThreshold", 0.0).asDouble(), grabberConfig.get("greenSignalThreshold", 0.0).asDouble(),
grabberConfig.get("blueSignalThreshold", 0.0).asDouble(), grabberConfig.get("blueSignalThreshold", 0.0).asDouble(),
grabberConfig.get("priority", 900).asInt()); grabberConfig.get("priority", 900).asInt());
v4l2Grabber->set3D(parse3DMode(grabberConfig.get("mode", "2D").asString())); _v4l2Grabber->set3D(parse3DMode(grabberConfig.get("mode", "2D").asString()));
v4l2Grabber->setCropping( _v4l2Grabber->setCropping(
grabberConfig.get("cropLeft", 0).asInt(), grabberConfig.get("cropLeft", 0).asInt(),
grabberConfig.get("cropRight", 0).asInt(), grabberConfig.get("cropRight", 0).asInt(),
grabberConfig.get("cropTop", 0).asInt(), grabberConfig.get("cropTop", 0).asInt(),
grabberConfig.get("cropBottom", 0).asInt()); grabberConfig.get("cropBottom", 0).asInt());
QObject::connect(v4l2Grabber, SIGNAL(emitImage(int, const Image<ColorRgb>&, const int)), protoServer, SLOT(sendImageToProtoSlaves(int, const Image<ColorRgb>&, const int)) ); QObject::connect(_v4l2Grabber, SIGNAL(emitImage(int, const Image<ColorRgb>&, const int)), _protoServer, SLOT(sendImageToProtoSlaves(int, const Image<ColorRgb>&, const int)) );
v4l2Grabber->start(); _v4l2Grabber->start();
Info(Logger::getInstance("MAIN"), "V4L2 grabber created and started"); Info(_log, "V4L2 grabber created and started");
} }
#else
ErrorIf(_config.isMember("grabber-v4l2"), _log, "The v4l2 grabber can not be instantiated, because it has been left out from the build");
#endif #endif
return v4l2Grabber;
} }
AmlogicWrapper* createGrabberAmlogic(ProtoServer* &protoServer) void HyperionDaemon::createGrabberAmlogic()
{ {
AmlogicWrapper* amlGrabber = nullptr;
#ifdef ENABLE_AMLOGIC #ifdef ENABLE_AMLOGIC
XBMCVideoChecker* xbmcVideoChecker = XBMCVideoChecker::getInstance();
const Json::Value &config = Hyperion::getInstance()->getJsonConfig();
// Construct and start the framebuffer grabber if the configuration is present // Construct and start the framebuffer grabber if the configuration is present
if (config.isMember("amlgrabber")) if (_config.isMember("amlgrabber"))
{ {
const Json::Value & grabberConfig = config["amlgrabber"]; const Json::Value & grabberConfig = _config["amlgrabber"];
amlGrabber = new AmlogicWrapper( _amlGrabber = new AmlogicWrapper(
grabberConfig["width"].asUInt(), grabberConfig["width"].asUInt(),
grabberConfig["height"].asUInt(), grabberConfig["height"].asUInt(),
grabberConfig["frequency_Hz"].asUInt(), grabberConfig["frequency_Hz"].asUInt(),
grabberConfig.get("priority",900).asInt()); grabberConfig.get("priority",900).asInt());
if (xbmcVideoChecker != nullptr) if (_xbmcVideoChecker != nullptr)
{ {
QObject::connect(xbmcVideoChecker, SIGNAL(grabbingMode(GrabbingMode)), amlGrabber, SLOT(setGrabbingMode(GrabbingMode))); QObject::connect(_xbmcVideoChecker, SIGNAL(grabbingMode(GrabbingMode)), _amlGrabber, SLOT(setGrabbingMode(GrabbingMode)));
QObject::connect(xbmcVideoChecker, SIGNAL(videoMode(VideoMode)), amlGrabber, SLOT(setVideoMode(VideoMode))); QObject::connect(_xbmcVideoChecker, SIGNAL(videoMode(VideoMode)), _amlGrabber, SLOT(setVideoMode(VideoMode)));
} }
QObject::connect(amlGrabber, SIGNAL(emitImage(int, const Image<ColorRgb>&, const int)), protoServer, SLOT(sendImageToProtoSlaves(int, const Image<ColorRgb>&, const int)) ); QObject::connect(_amlGrabber, SIGNAL(emitImage(int, const Image<ColorRgb>&, const int)), _protoServer, SLOT(sendImageToProtoSlaves(int, const Image<ColorRgb>&, const int)) );
amlGrabber->start(); _amlGrabber->start();
Info(Logger::getInstance("MAIN"), "AMLOGIC grabber created and started"); Info(_log, "AMLOGIC grabber created and started");
} }
#else
ErrorIf(_config.isMember("amlgrabber"), _log, "The AMLOGIC grabber can not be instantiated, because it has been left out from the build");
#endif #endif
return amlGrabber;
} }
FramebufferWrapper* createGrabberFramebuffer(ProtoServer* &protoServer) void HyperionDaemon::createGrabberFramebuffer()
{ {
FramebufferWrapper* fbGrabber = nullptr;
#ifdef ENABLE_FB #ifdef ENABLE_FB
XBMCVideoChecker* xbmcVideoChecker = XBMCVideoChecker::getInstance();
const Json::Value &config = Hyperion::getInstance()->getJsonConfig();
// Construct and start the framebuffer grabber if the configuration is present // Construct and start the framebuffer grabber if the configuration is present
if (config.isMember("framebuffergrabber") || config.isMember("framegrabber")) if (_config.isMember("framebuffergrabber") || _config.isMember("framegrabber"))
{ {
const Json::Value & grabberConfig = config.isMember("framebuffergrabber")? config["framebuffergrabber"] : config["framegrabber"]; const Json::Value & grabberConfig = _config.isMember("framebuffergrabber")? _config["framebuffergrabber"] : _config["framegrabber"];
fbGrabber = new FramebufferWrapper( _fbGrabber = new FramebufferWrapper(
grabberConfig.get("device", "/dev/fb0").asString(), grabberConfig.get("device", "/dev/fb0").asString(),
grabberConfig["width"].asUInt(), grabberConfig["width"].asUInt(),
grabberConfig["height"].asUInt(), grabberConfig["height"].asUInt(),
grabberConfig["frequency_Hz"].asUInt(), grabberConfig["frequency_Hz"].asUInt(),
grabberConfig.get("priority",900).asInt()); grabberConfig.get("priority",900).asInt());
if (xbmcVideoChecker != nullptr) if (_xbmcVideoChecker != nullptr)
{ {
QObject::connect(xbmcVideoChecker, SIGNAL(grabbingMode(GrabbingMode)), fbGrabber, SLOT(setGrabbingMode(GrabbingMode))); QObject::connect(_xbmcVideoChecker, SIGNAL(grabbingMode(GrabbingMode)), _fbGrabber, SLOT(setGrabbingMode(GrabbingMode)));
QObject::connect(xbmcVideoChecker, SIGNAL(videoMode(VideoMode)), fbGrabber, SLOT(setVideoMode(VideoMode))); QObject::connect(_xbmcVideoChecker, SIGNAL(videoMode(VideoMode)), _fbGrabber, SLOT(setVideoMode(VideoMode)));
} }
QObject::connect(fbGrabber, SIGNAL(emitImage(int, const Image<ColorRgb>&, const int)), protoServer, SLOT(sendImageToProtoSlaves(int, const Image<ColorRgb>&, const int)) ); QObject::connect(_fbGrabber, SIGNAL(emitImage(int, const Image<ColorRgb>&, const int)), _protoServer, SLOT(sendImageToProtoSlaves(int, const Image<ColorRgb>&, const int)) );
fbGrabber->start(); _fbGrabber->start();
Info(Logger::getInstance("MAIN"), "Framebuffer grabber created and started"); Info(_log, "Framebuffer grabber created and started");
} }
#else
ErrorIf(_config.isMember("framebuffergrabber"), _log, "The framebuffer grabber can not be instantiated, because it has been left out from the build");
#endif #endif
return fbGrabber;
} }
OsxWrapper* createGrabberOsx(ProtoServer* &protoServer) void HyperionDaemon::createGrabberOsx()
{ {
OsxWrapper* osxGrabber = nullptr;
#ifdef ENABLE_OSX #ifdef ENABLE_OSX
XBMCVideoChecker* xbmcVideoChecker = XBMCVideoChecker::getInstance();
const Json::Value &config = Hyperion::getInstance()->getJsonConfig();
// Construct and start the osx grabber if the configuration is present // Construct and start the osx grabber if the configuration is present
if (config.isMember("osxgrabber") || config.isMember("framegrabber")) if (_config.isMember("framegrabber"))
{ {
const Json::Value & grabberConfig = config.isMember("osxgrabber")? config["osxgrabber"] : config["framegrabber"]; const Json::Value & grabberConfig = _config.isMember("osxgrabber")? _config["osxgrabber"] : _config["framegrabber"];
osxGrabber = new OsxWrapper( _osxGrabber = new OsxWrapper(
grabberConfig.get("display", 0).asUInt(), grabberConfig.get("display", 0).asUInt(),
grabberConfig["width"].asUInt(), grabberConfig["width"].asUInt(),
grabberConfig["height"].asUInt(), grabberConfig["height"].asUInt(),
grabberConfig["frequency_Hz"].asUInt(), grabberConfig["frequency_Hz"].asUInt(),
grabberConfig.get("priority",900).asInt()); grabberConfig.get("priority",900).asInt());
if (xbmcVideoChecker != nullptr) if (_xbmcVideoChecker != nullptr)
{ {
QObject::connect(xbmcVideoChecker, SIGNAL(grabbingMode(GrabbingMode)), osxGrabber, SLOT(setGrabbingMode(GrabbingMode))); QObject::connect(_xbmcVideoChecker, SIGNAL(grabbingMode(GrabbingMode)), _osxGrabber, SLOT(setGrabbingMode(GrabbingMode)));
QObject::connect(xbmcVideoChecker, SIGNAL(videoMode(VideoMode)), osxGrabber, SLOT(setVideoMode(VideoMode))); QObject::connect(_xbmcVideoChecker, SIGNAL(videoMode(VideoMode)), _osxGrabber, SLOT(setVideoMode(VideoMode)));
} }
QObject::connect(osxGrabber, SIGNAL(emitImage(int, const Image<ColorRgb>&, const int)), protoServer, SLOT(sendImageToProtoSlaves(int, const Image<ColorRgb>&, const int)) ); QObject::connect(_osxGrabber, SIGNAL(emitImage(int, const Image<ColorRgb>&, const int)), _protoServer, SLOT(sendImageToProtoSlaves(int, const Image<ColorRgb>&, const int)) );
osxGrabber->start(); _osxGrabber->start();
Info(Logger::getInstance("MAIN"), "OSX grabber created and started"); Info(_log, "OSX grabber created and started");
} }
#else
ErrorIf(_config.isMember("osxgrabber"), _log, "The osx grabber can not be instantiated, because it has been left out from the build");
#endif #endif
return osxGrabber;
} }

View File

@ -32,18 +32,45 @@
typedef QObject OsxWrapper; typedef QObject OsxWrapper;
#endif #endif
#include <utils/Logger.h>
#include <xbmcvideochecker/XBMCVideoChecker.h> #include <xbmcvideochecker/XBMCVideoChecker.h>
#include <jsonserver/JsonServer.h> #include <jsonserver/JsonServer.h>
#include <protoserver/ProtoServer.h> #include <protoserver/ProtoServer.h>
#include <boblightserver/BoblightServer.h> #include <boblightserver/BoblightServer.h>
#include <webconfig/WebConfig.h>
void startBootsequence(); class HyperionDaemon : public QObject
XBMCVideoChecker* createXBMCVideoChecker(); {
void startNetworkServices(JsonServer* &jsonServer, ProtoServer* &protoServer, BoblightServer* &boblightServer); public:
HyperionDaemon(std::string configFile, QObject *parent=nullptr);
~HyperionDaemon();
// grabber creators void loadConfig(const std::string & configFile);
DispmanxWrapper* createGrabberDispmanx(ProtoServer* &protoServer); void run();
V4L2Wrapper* createGrabberV4L2(ProtoServer* &protoServer );
AmlogicWrapper* createGrabberAmlogic(ProtoServer* &protoServer); void startBootsequence();
FramebufferWrapper* createGrabberFramebuffer(ProtoServer* &protoServer); void createXBMCVideoChecker();
OsxWrapper* createGrabberOsx(ProtoServer* &protoServer); void startNetworkServices();
// grabber creators
void createGrabberDispmanx();
void createGrabberV4L2();
void createGrabberAmlogic();
void createGrabberFramebuffer();
void createGrabberOsx();
private:
Logger* _log;
Json::Value _config;
XBMCVideoChecker* _xbmcVideoChecker;
JsonServer* _jsonServer;
ProtoServer* _protoServer;
BoblightServer* _boblightServer;
V4L2Wrapper* _v4l2Grabber;
DispmanxWrapper* _dispmanx;
AmlogicWrapper* _amlGrabber;
FramebufferWrapper* _fbGrabber;
OsxWrapper* _osxGrabber;
WebConfig* _webConfig;
};

View File

@ -4,18 +4,15 @@
#include <sys/prctl.h> #include <sys/prctl.h>
#include <QCoreApplication> #include <QCoreApplication>
#include <QResource>
#include <QLocale> #include <QLocale>
#include <QFile> #include <QFile>
#include "HyperionConfig.h" #include "HyperionConfig.h"
#include <getoptPlusPlus/getoptpp.h> #include <getoptPlusPlus/getoptpp.h>
#include <utils/jsonschema/JsonFactory.h>
#include <utils/Logger.h> #include <utils/Logger.h>
#include <hyperion/Hyperion.h>
#include <webconfig/WebConfig.h>
#include "hyperiond.h" #include "hyperiond.h"
@ -30,30 +27,6 @@ void signal_handler(const int signum)
} }
Json::Value loadConfig(const std::string & configFile)
{
// make sure the resources are loaded (they may be left out after static linking)
Q_INIT_RESOURCE(resource);
// read the json schema from the resource
QResource schemaData(":/hyperion-schema");
assert(schemaData.isValid());
Json::Reader jsonReader;
Json::Value schemaJson;
if (!jsonReader.parse(reinterpret_cast<const char *>(schemaData.data()), reinterpret_cast<const char *>(schemaData.data()) + schemaData.size(), schemaJson, false))
{
throw std::runtime_error("ERROR: Json schema wrong: " + jsonReader.getFormattedErrorMessages()) ;
}
JsonSchemaChecker schemaChecker;
schemaChecker.setSchema(schemaJson);
const Json::Value jsonConfig = JsonFactory::readJson(configFile);
schemaChecker.validate(jsonConfig);
return jsonConfig;
}
void startNewHyperion(int parentPid, std::string hyperionFile, std::string configFile) void startNewHyperion(int parentPid, std::string hyperionFile, std::string configFile)
{ {
if ( fork() == 0 ) if ( fork() == 0 )
@ -136,70 +109,14 @@ int main(int argc, char** argv)
return 1; return 1;
} }
const std::string configFile = configFiles[argvId]; HyperionDaemon* hyperiond = new HyperionDaemon(configFiles[argvId], &app);
Info(log, "Selected configuration file: %s", configFile.c_str() ); hyperiond->run();
const Json::Value config = loadConfig(configFile);
Hyperion::initInstance(config, configFile);
Info(log, "Hyperion started and initialised");
startBootsequence();
XBMCVideoChecker * xbmcVideoChecker = createXBMCVideoChecker();
// ---- network services -----
JsonServer * jsonServer = nullptr;
ProtoServer * protoServer = nullptr;
BoblightServer * boblightServer = nullptr;
startNetworkServices(jsonServer, protoServer, boblightServer);
WebConfig webConfig(&app);
// ---- grabber -----
// if a grabber is left out of build, then <grabber>Wrapper is set to QObject as dummy and has value nullptr
V4L2Wrapper * v4l2Grabber = createGrabberV4L2(protoServer);
#ifndef ENABLE_V4L2
ErrorIf(config.isMember("grabber-v4l2"), log, "The v4l2 grabber can not be instantiated, because it has been left out from the build");
#endif
DispmanxWrapper * dispmanx = createGrabberDispmanx(protoServer);
#ifndef ENABLE_DISPMANX
ErrorIf(config.isMember("framegrabber"), log, "The dispmanx framegrabber can not be instantiated, because it has been left out from the build");
#endif
AmlogicWrapper * amlGrabber = createGrabberAmlogic(protoServer);
#ifndef ENABLE_AMLOGIC
ErrorIf(config.isMember("amlgrabber"), log, "The AMLOGIC grabber can not be instantiated, because it has been left out from the build");
#endif
FramebufferWrapper * fbGrabber = createGrabberFramebuffer(protoServer);
#ifndef ENABLE_FB
ErrorIf(config.isMember("framebuffergrabber"), log, "The framebuffer grabber can not be instantiated, because it has been left out from the build");
#endif
OsxWrapper * osxGrabber = createGrabberDispmanx(protoServer);
#ifndef ENABLE_OSX
ErrorIf(config.isMember("osxgrabber"), log, "The osx grabber can not be instantiated, because it has been left out from the build");
#endif
#if !defined(ENABLE_DISPMANX) && !defined(ENABLE_OSX) && !defined(ENABLE_FB)
ErrorIf(config.isMember("framegrabber"), log, "No grabber can be instantiated, because all grabbers have been left out from the build");
#endif
// run the application // run the application
int rc = app.exec(); int rc = app.exec();
Info(log, "INFO: Application closed with code %d", rc); Info(log, "INFO: Application closed with code %d", rc);
// Delete all components delete hyperiond;
delete amlGrabber;
delete dispmanx;
delete fbGrabber;
delete osxGrabber;
delete v4l2Grabber;
delete xbmcVideoChecker;
delete jsonServer;
delete protoServer;
delete boblightServer;
return rc; return rc;
} }