2016-08-11 07:13:55 +02:00
|
|
|
// Hyperion includes
|
|
|
|
#include <hyperion/GrabberWrapper.h>
|
2017-08-12 07:55:32 +02:00
|
|
|
#include <hyperion/Grabber.h>
|
2017-02-17 08:33:34 +01:00
|
|
|
#include <HyperionConfig.h>
|
2016-08-11 07:13:55 +02:00
|
|
|
|
2019-01-01 19:47:07 +01:00
|
|
|
// utils includes
|
|
|
|
#include <utils/GlobalSignals.h>
|
|
|
|
|
2018-12-27 23:11:32 +01:00
|
|
|
// qt
|
|
|
|
#include <QTimer>
|
|
|
|
|
2020-04-17 16:59:20 +02:00
|
|
|
GrabberWrapper* GrabberWrapper::instance = nullptr;
|
|
|
|
|
2020-08-08 23:12:43 +02:00
|
|
|
GrabberWrapper::GrabberWrapper(const QString& grabberName, Grabber * ggrabber, unsigned width, unsigned height, unsigned updateRate_Hz)
|
2016-08-11 07:13:55 +02:00
|
|
|
: _grabberName(grabberName)
|
2018-12-27 23:11:32 +01:00
|
|
|
, _timer(new QTimer(this))
|
2017-08-12 07:55:32 +02:00
|
|
|
, _updateInterval_ms(1000/updateRate_Hz)
|
2021-01-26 20:01:23 +01:00
|
|
|
, _log(Logger::getInstance(grabberName.toUpper()))
|
2017-08-12 07:55:32 +02:00
|
|
|
, _ggrabber(ggrabber)
|
|
|
|
, _image(0,0)
|
2016-08-11 07:13:55 +02:00
|
|
|
{
|
2020-04-17 16:59:20 +02:00
|
|
|
GrabberWrapper::instance = this;
|
|
|
|
|
2017-08-12 07:55:32 +02:00
|
|
|
// Configure the timer to generate events every n milliseconds
|
2018-12-27 23:11:32 +01:00
|
|
|
_timer->setInterval(_updateInterval_ms);
|
2016-08-11 07:13:55 +02:00
|
|
|
|
2017-08-12 07:55:32 +02:00
|
|
|
_image.resize(width, height);
|
2017-11-22 00:52:55 +01:00
|
|
|
|
2018-12-27 23:11:32 +01:00
|
|
|
connect(_timer, &QTimer::timeout, this, &GrabberWrapper::action);
|
2019-01-01 19:47:07 +01:00
|
|
|
|
|
|
|
// connect the image forwarding
|
2019-07-14 22:43:22 +02:00
|
|
|
(_grabberName.startsWith("V4L"))
|
2019-01-01 19:47:07 +01:00
|
|
|
? connect(this, &GrabberWrapper::systemImage, GlobalSignals::getInstance(), &GlobalSignals::setV4lImage)
|
|
|
|
: connect(this, &GrabberWrapper::systemImage, GlobalSignals::getInstance(), &GlobalSignals::setSystemImage);
|
2020-03-26 17:49:36 +01:00
|
|
|
|
|
|
|
// listen for source requests
|
|
|
|
connect(GlobalSignals::getInstance(), &GlobalSignals::requestSource, this, &GrabberWrapper::handleSourceRequest);
|
2016-08-11 07:13:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
GrabberWrapper::~GrabberWrapper()
|
|
|
|
{
|
2017-02-17 08:33:34 +01:00
|
|
|
Debug(_log,"Close grabber: %s", QSTRING_CSTR(_grabberName));
|
2016-08-11 07:13:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool GrabberWrapper::start()
|
|
|
|
{
|
2020-12-31 11:12:29 +01:00
|
|
|
if (!_timer->isActive())
|
|
|
|
{
|
|
|
|
// Start the timer with the pre configured interval
|
|
|
|
Debug(_log,"Grabber start()");
|
|
|
|
_timer->start();
|
|
|
|
}
|
2021-01-01 14:40:04 +01:00
|
|
|
|
|
|
|
return _timer->isActive();
|
2016-08-11 07:13:55 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void GrabberWrapper::stop()
|
|
|
|
{
|
2020-07-12 18:27:24 +02:00
|
|
|
if (_timer->isActive())
|
|
|
|
{
|
|
|
|
// Stop the timer, effectivly stopping the process
|
|
|
|
Debug(_log,"Grabber stop()");
|
|
|
|
_timer->stop();
|
|
|
|
}
|
2016-10-10 18:29:54 +02:00
|
|
|
}
|
2017-02-17 08:33:34 +01:00
|
|
|
|
2020-07-23 16:50:37 +02:00
|
|
|
bool GrabberWrapper::isActive() const
|
|
|
|
{
|
|
|
|
return _timer->isActive();
|
|
|
|
}
|
|
|
|
|
2020-12-18 17:38:21 +01:00
|
|
|
QStringList GrabberWrapper::getActive(int inst) const
|
2020-11-14 16:22:21 +01:00
|
|
|
{
|
2020-12-18 17:38:21 +01:00
|
|
|
QStringList result = QStringList();
|
|
|
|
|
|
|
|
if(GRABBER_V4L_CLIENTS.contains(inst))
|
|
|
|
result << GRABBER_V4L_CLIENTS.value(inst);
|
|
|
|
|
|
|
|
if(GRABBER_SYS_CLIENTS.contains(inst))
|
|
|
|
result << GRABBER_SYS_CLIENTS.value(inst);
|
|
|
|
|
|
|
|
return result;
|
2020-11-14 16:22:21 +01:00
|
|
|
}
|
|
|
|
|
2017-02-17 08:33:34 +01:00
|
|
|
QStringList GrabberWrapper::availableGrabbers()
|
|
|
|
{
|
|
|
|
QStringList grabbers;
|
|
|
|
|
|
|
|
#ifdef ENABLE_DISPMANX
|
|
|
|
grabbers << "dispmanx";
|
|
|
|
#endif
|
|
|
|
|
2020-12-18 17:38:21 +01:00
|
|
|
#if defined(ENABLE_V4L2) || defined(ENABLE_MF)
|
2017-02-17 08:33:34 +01:00
|
|
|
grabbers << "v4l2";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef ENABLE_FB
|
|
|
|
grabbers << "framebuffer";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef ENABLE_AMLOGIC
|
|
|
|
grabbers << "amlogic";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef ENABLE_OSX
|
|
|
|
grabbers << "osx";
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef ENABLE_X11
|
|
|
|
grabbers << "x11";
|
|
|
|
#endif
|
|
|
|
|
2020-08-03 12:31:39 +02:00
|
|
|
#ifdef ENABLE_XCB
|
|
|
|
grabbers << "xcb";
|
|
|
|
#endif
|
|
|
|
|
2019-01-06 19:49:56 +01:00
|
|
|
#ifdef ENABLE_QT
|
|
|
|
grabbers << "qt";
|
|
|
|
#endif
|
|
|
|
|
2020-11-14 16:22:21 +01:00
|
|
|
#ifdef ENABLE_DX
|
|
|
|
grabbers << "dx";
|
|
|
|
#endif
|
|
|
|
|
2017-02-17 08:33:34 +01:00
|
|
|
return grabbers;
|
|
|
|
}
|
2017-08-12 07:55:32 +02:00
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void GrabberWrapper::setVideoMode(VideoMode mode)
|
2017-08-12 07:55:32 +02:00
|
|
|
{
|
|
|
|
if (_ggrabber != nullptr)
|
|
|
|
{
|
|
|
|
Info(_log,"setvideomode");
|
|
|
|
_ggrabber->setVideoMode(mode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GrabberWrapper::setCropping(unsigned cropLeft, unsigned cropRight, unsigned cropTop, unsigned cropBottom)
|
|
|
|
{
|
|
|
|
_ggrabber->setCropping(cropLeft, cropRight, cropTop, cropBottom);
|
|
|
|
}
|
|
|
|
|
2020-07-23 16:50:37 +02:00
|
|
|
void GrabberWrapper::updateTimer(int interval)
|
|
|
|
{
|
|
|
|
if(_updateInterval_ms != interval)
|
|
|
|
{
|
|
|
|
_updateInterval_ms = interval;
|
|
|
|
|
|
|
|
const bool& timerWasActive = _timer->isActive();
|
|
|
|
_timer->stop();
|
|
|
|
_timer->setInterval(_updateInterval_ms);
|
|
|
|
|
|
|
|
if(timerWasActive)
|
|
|
|
_timer->start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void GrabberWrapper::handleSettingsUpdate(settings::type type, const QJsonDocument& config)
|
2017-08-12 07:55:32 +02:00
|
|
|
{
|
2020-07-23 16:50:37 +02:00
|
|
|
if(type == settings::SYSTEMCAPTURE && !_grabberName.startsWith("V4L"))
|
2018-12-27 23:11:32 +01:00
|
|
|
{
|
|
|
|
// extract settings
|
2019-05-03 17:54:43 +02:00
|
|
|
const QJsonObject& obj = config.object();
|
2018-12-27 23:11:32 +01:00
|
|
|
|
2020-07-23 16:50:37 +02:00
|
|
|
// width/height
|
|
|
|
_ggrabber->setWidthHeight(obj["width"].toInt(96), obj["height"].toInt(96));
|
|
|
|
|
|
|
|
// display index for MAC
|
|
|
|
_ggrabber->setDisplayIndex(obj["display"].toInt(0));
|
|
|
|
|
|
|
|
// device path for Framebuffer
|
|
|
|
_ggrabber->setDevicePath(obj["device"].toString("/dev/fb0"));
|
|
|
|
|
|
|
|
// pixel decimation for x11
|
|
|
|
_ggrabber->setPixelDecimation(obj["pixelDecimation"].toInt(8));
|
|
|
|
|
|
|
|
// crop for system capture
|
|
|
|
_ggrabber->setCropping(
|
|
|
|
obj["cropLeft"].toInt(0),
|
|
|
|
obj["cropRight"].toInt(0),
|
|
|
|
obj["cropTop"].toInt(0),
|
|
|
|
obj["cropBottom"].toInt(0));
|
|
|
|
|
|
|
|
// eval new update time
|
|
|
|
updateTimer(1000/obj["frequency_Hz"].toInt(10));
|
2018-12-27 23:11:32 +01:00
|
|
|
}
|
2017-08-12 07:55:32 +02:00
|
|
|
}
|
2020-03-26 17:49:36 +01:00
|
|
|
|
2020-08-08 13:09:15 +02:00
|
|
|
void GrabberWrapper::handleSourceRequest(hyperion::Components component, int hyperionInd, bool listen)
|
2020-03-26 17:49:36 +01:00
|
|
|
{
|
|
|
|
if(component == hyperion::Components::COMP_GRABBER && !_grabberName.startsWith("V4L"))
|
|
|
|
{
|
2020-12-31 11:12:29 +01:00
|
|
|
if(listen)
|
2020-12-18 17:38:21 +01:00
|
|
|
GRABBER_SYS_CLIENTS.insert(hyperionInd, _grabberName);
|
2020-12-31 11:12:29 +01:00
|
|
|
else
|
2020-12-18 17:38:21 +01:00
|
|
|
GRABBER_SYS_CLIENTS.remove(hyperionInd);
|
2020-03-26 17:49:36 +01:00
|
|
|
|
|
|
|
if(GRABBER_SYS_CLIENTS.empty())
|
|
|
|
stop();
|
|
|
|
else
|
|
|
|
start();
|
|
|
|
}
|
|
|
|
else if(component == hyperion::Components::COMP_V4L && _grabberName.startsWith("V4L"))
|
|
|
|
{
|
2020-12-31 11:12:29 +01:00
|
|
|
if(listen)
|
2020-12-18 17:38:21 +01:00
|
|
|
GRABBER_V4L_CLIENTS.insert(hyperionInd, _grabberName);
|
2020-12-31 11:12:29 +01:00
|
|
|
else
|
2020-12-18 17:38:21 +01:00
|
|
|
GRABBER_V4L_CLIENTS.remove(hyperionInd);
|
2020-03-26 17:49:36 +01:00
|
|
|
|
|
|
|
if(GRABBER_V4L_CLIENTS.empty())
|
|
|
|
stop();
|
|
|
|
else
|
|
|
|
start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GrabberWrapper::tryStart()
|
|
|
|
{
|
2020-07-12 18:27:24 +02:00
|
|
|
// verify start condition
|
2020-03-26 17:49:36 +01:00
|
|
|
if((_grabberName.startsWith("V4L") && !GRABBER_V4L_CLIENTS.empty()) || (!_grabberName.startsWith("V4L") && !GRABBER_SYS_CLIENTS.empty()))
|
|
|
|
{
|
|
|
|
start();
|
|
|
|
}
|
|
|
|
}
|
2020-04-17 16:59:20 +02:00
|
|
|
|
2021-01-31 13:49:31 +01:00
|
|
|
QStringList GrabberWrapper::getDevices() const
|
2020-04-17 16:59:20 +02:00
|
|
|
{
|
|
|
|
if(_grabberName.startsWith("V4L"))
|
2021-01-31 13:49:31 +01:00
|
|
|
return _ggrabber->getDevices();
|
2020-04-17 16:59:20 +02:00
|
|
|
|
|
|
|
return QStringList();
|
|
|
|
}
|
|
|
|
|
2021-01-31 13:49:31 +01:00
|
|
|
QString GrabberWrapper::getDeviceName(const QString& devicePath) const
|
2020-04-17 16:59:20 +02:00
|
|
|
{
|
|
|
|
if(_grabberName.startsWith("V4L"))
|
2021-01-31 13:49:31 +01:00
|
|
|
return _ggrabber->getDeviceName(devicePath);
|
2020-04-17 16:59:20 +02:00
|
|
|
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
2021-01-31 13:49:31 +01:00
|
|
|
QMultiMap<QString, int> GrabberWrapper::getDeviceInputs(const QString& devicePath) const
|
2020-06-17 20:55:57 +02:00
|
|
|
{
|
|
|
|
if(_grabberName.startsWith("V4L"))
|
2021-01-31 13:49:31 +01:00
|
|
|
return _ggrabber->getDeviceInputs(devicePath);
|
2020-06-17 20:55:57 +02:00
|
|
|
|
2021-02-05 21:52:02 +01:00
|
|
|
return QMultiMap<QString, int>();
|
2020-06-17 20:55:57 +02:00
|
|
|
}
|
|
|
|
|
2021-01-31 13:49:31 +01:00
|
|
|
QStringList GrabberWrapper::getAvailableEncodingFormats(const QString& devicePath, const int& deviceInput) const
|
2020-12-18 17:38:21 +01:00
|
|
|
{
|
|
|
|
if(_grabberName.startsWith("V4L"))
|
2021-01-31 13:49:31 +01:00
|
|
|
return _ggrabber->getAvailableEncodingFormats(devicePath, deviceInput);
|
2020-12-18 17:38:21 +01:00
|
|
|
|
|
|
|
return QStringList();
|
|
|
|
}
|
|
|
|
|
2021-02-05 21:52:02 +01:00
|
|
|
QMultiMap<int, int> GrabberWrapper::getAvailableDeviceResolutions(const QString& devicePath, const int& deviceInput, const PixelFormat& encFormat) const
|
2020-04-17 16:59:20 +02:00
|
|
|
{
|
|
|
|
if(_grabberName.startsWith("V4L"))
|
2021-01-31 13:49:31 +01:00
|
|
|
return _ggrabber->getAvailableDeviceResolutions(devicePath, deviceInput, encFormat);
|
2020-04-17 16:59:20 +02:00
|
|
|
|
2021-02-05 21:52:02 +01:00
|
|
|
return QMultiMap<int, int>();
|
2020-04-17 16:59:20 +02:00
|
|
|
}
|
|
|
|
|
2021-01-31 13:49:31 +01:00
|
|
|
QStringList GrabberWrapper::getAvailableDeviceFramerates(const QString& devicePath, const int& deviceInput, const PixelFormat& encFormat, const unsigned width, const unsigned height) const
|
2020-04-17 16:59:20 +02:00
|
|
|
{
|
|
|
|
if(_grabberName.startsWith("V4L"))
|
2021-01-31 13:49:31 +01:00
|
|
|
return _ggrabber->getAvailableDeviceFramerates(devicePath, deviceInput, encFormat, width, height);
|
2020-04-17 16:59:20 +02:00
|
|
|
|
|
|
|
return QStringList();
|
|
|
|
}
|