mirror of
https://github.com/hyperion-project/hyperion.ng.git
synced 2023-10-10 13:36:59 +02:00
206 lines
6.1 KiB
C++
206 lines
6.1 KiB
C++
|
#include <cassert>
|
||
|
#include <csignal>
|
||
|
#include <unistd.h>
|
||
|
#include <sys/prctl.h>
|
||
|
|
||
|
#include <QCoreApplication>
|
||
|
#include <QResource>
|
||
|
#include <QLocale>
|
||
|
#include <QFile>
|
||
|
|
||
|
#include "HyperionConfig.h"
|
||
|
|
||
|
#include <getoptPlusPlus/getoptpp.h>
|
||
|
#include <utils/jsonschema/JsonFactory.h>
|
||
|
#include <utils/Logger.h>
|
||
|
|
||
|
#include <hyperion/Hyperion.h>
|
||
|
#include <webconfig/WebConfig.h>
|
||
|
|
||
|
#include "hyperiond.h"
|
||
|
|
||
|
using namespace vlofgren;
|
||
|
|
||
|
void signal_handler(const int signum)
|
||
|
{
|
||
|
QCoreApplication::quit();
|
||
|
|
||
|
// reset signal handler to default (in case this handler is not capable of stopping)
|
||
|
signal(signum, SIG_DFL);
|
||
|
}
|
||
|
|
||
|
|
||
|
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)
|
||
|
{
|
||
|
if ( fork() == 0 )
|
||
|
{
|
||
|
sleep(3);
|
||
|
execl(hyperionFile.c_str(), hyperionFile.c_str(), "--parent", QString::number(parentPid).toStdString().c_str(), configFile.c_str(), NULL);
|
||
|
exit(0);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
int main(int argc, char** argv)
|
||
|
{
|
||
|
Logger* log = Logger::getInstance("MAIN", Logger::INFO);
|
||
|
|
||
|
// Initialising QCoreApplication
|
||
|
QCoreApplication app(argc, argv);
|
||
|
|
||
|
signal(SIGINT, signal_handler);
|
||
|
signal(SIGTERM, signal_handler);
|
||
|
signal(SIGCHLD, signal_handler);
|
||
|
|
||
|
// force the locale
|
||
|
setlocale(LC_ALL, "C");
|
||
|
QLocale::setDefault(QLocale::c());
|
||
|
|
||
|
OptionsParser optionParser("Hyperion Daemon");
|
||
|
ParameterSet & parameters = optionParser.getParameters();
|
||
|
|
||
|
SwitchParameter<> & argVersion = parameters.add<SwitchParameter<>> (0x0, "version", "Show version information");
|
||
|
IntParameter & argParentPid = parameters.add<IntParameter> (0x0, "parent", "pid of parent hyperiond");
|
||
|
SwitchParameter<> & argHelp = parameters.add<SwitchParameter<>> ('h', "help", "Show this help message and exit");
|
||
|
|
||
|
argParentPid.setDefault(0);
|
||
|
optionParser.parse(argc, const_cast<const char **>(argv));
|
||
|
const std::vector<std::string> configFiles = optionParser.getFiles();
|
||
|
|
||
|
// check if we need to display the usage. exit if we do.
|
||
|
if (argHelp.isSet())
|
||
|
{
|
||
|
optionParser.usage();
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
if (argVersion.isSet())
|
||
|
{
|
||
|
std::cout
|
||
|
<< "Hyperion Ambilight Deamon (" << getpid() << ")" << std::endl
|
||
|
<< "\tVersion : " << HYPERION_VERSION_ID << std::endl
|
||
|
<< "\tBuild Time: " << __DATE__ << " " << __TIME__ << std::endl;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
if (configFiles.size() == 0)
|
||
|
{
|
||
|
Error(log, "Missing required configuration file. Usage: hyperiond <options ...> [config.file ...]");
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
|
||
|
if (argParentPid.getValue() > 0 )
|
||
|
{
|
||
|
Info(log, "hyperiond client, parent is pid %d",argParentPid.getValue());
|
||
|
prctl(PR_SET_PDEATHSIG, SIGHUP);
|
||
|
}
|
||
|
|
||
|
int argvId = -1;
|
||
|
for(size_t idx=0; idx < configFiles.size(); idx++) {
|
||
|
if ( QFile::exists(configFiles[idx].c_str()))
|
||
|
{
|
||
|
if (argvId < 0) argvId=idx;
|
||
|
else startNewHyperion(getpid(), argv[0], configFiles[idx].c_str());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( argvId < 0)
|
||
|
{
|
||
|
Error(log, "No valid config found");
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
const std::string configFile = configFiles[argvId];
|
||
|
Info(log, "Selected configuration file: %s", configFile.c_str() );
|
||
|
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" << std::endl;
|
||
|
#endif
|
||
|
|
||
|
// run the application
|
||
|
int rc = app.exec();
|
||
|
Info(log, "INFO: Application closed with code %d", rc);
|
||
|
|
||
|
// Delete all components
|
||
|
delete amlGrabber;
|
||
|
delete dispmanx;
|
||
|
delete fbGrabber;
|
||
|
delete osxGrabber;
|
||
|
delete v4l2Grabber;
|
||
|
delete xbmcVideoChecker;
|
||
|
delete jsonServer;
|
||
|
delete protoServer;
|
||
|
delete boblightServer;
|
||
|
|
||
|
return rc;
|
||
|
}
|