#pragma once // stl includes #include #include // Qt includes #include #include #include #include #include // util includes #include #include #include #include #include // Required to determine the cmake options #if defined(ENABLE_CEC) #include #endif // general JPEG decoder includes #ifdef HAVE_JPEG_DECODER #include #include #endif // System JPEG decoder #ifdef HAVE_JPEG #include #include #endif // TurboJPEG decoder #ifdef HAVE_TURBO_JPEG #include #endif /// /// Capture class for V4L2 devices /// class V4L2Grabber : public Grabber { Q_OBJECT public: struct DeviceProperties { QString name = QString(); struct InputProperties { QString inputName = QString(); QList standards = QList(); struct EncodingProperties { int width = 0; int height = 0; QList framerates = QList(); }; QMultiMap encodingFormats = QMultiMap(); }; QMap inputs = QMap(); }; V4L2Grabber(); ~V4L2Grabber() override; int grabFrame(Image &); void setDevice(const QString& device); bool setInput(int input) override; bool setWidthHeight(int width, int height) override; void setEncoding(QString enc); void setBrightnessContrastSaturationHue(int brightness, int contrast, int saturation, int hue); void setSignalThreshold(double redSignalThreshold, double greenSignalThreshold, double blueSignalThreshold, int noSignalCounterThreshold = 50); void setSignalDetectionOffset( double verticalMin, double horizontalMin, double verticalMax, double horizontalMax); void setSignalDetectionEnable(bool enable); void setCecDetectionEnable(bool enable); bool reload(bool force = false); QRectF getSignalDetectionOffset() const { return QRectF(_x_frac_min, _y_frac_min, _x_frac_max, _y_frac_max); } //used from hyperion-v4l2 /// /// @brief Discover available V4L2 USB devices (for configuration). /// @param[in] params Parameters used to overwrite discovery default behaviour /// @return A JSON structure holding a list of USB devices found /// QJsonArray discover(const QJsonObject& params); public slots: bool prepare() { return true; } bool start(); void stop(); #if defined(ENABLE_CEC) void handleCecEvent(CECEvent event); #endif signals: void newFrame(const Image & image); void readError(const char* err); private slots: int read_frame(); private: void enumVideoCaptureDevices(); bool init(); void uninit(); bool open_device(); void close_device(); void init_read(unsigned int buffer_size); void init_mmap(); void init_userp(unsigned int buffer_size); void init_device(VideoStandard videoStandard); void uninit_device(); void start_capturing(); void stop_capturing(); bool process_image(const void *p, int size); void process_image(const uint8_t *p, int size); int xioctl(int request, void *arg); int xioctl(int fileDescriptor, int request, void *arg); void throw_exception(const QString & error) { Error(_log, "Throws error: %s", QSTRING_CSTR(error)); } void throw_errno_exception(const QString & error) { Error(_log, "Throws error nr: %s", QSTRING_CSTR(QString(error + " error code " + QString::number(errno) + ", " + strerror(errno)))); } private: enum io_method { IO_METHOD_READ, IO_METHOD_MMAP, IO_METHOD_USERPTR }; struct buffer { void *start; size_t length; }; #ifdef HAVE_JPEG struct errorManager { jpeg_error_mgr pub; jmp_buf setjmp_buffer; }; static void errorHandler(j_common_ptr cInfo) { errorManager* mgr = reinterpret_cast(cInfo->err); longjmp(mgr->setjmp_buffer, 1); } static void outputHandler(j_common_ptr cInfo) { // Suppress fprintf warnings. } jpeg_decompress_struct* _decompress; errorManager* _error; #endif #ifdef HAVE_TURBO_JPEG tjhandle _decompress = nullptr; int _subsamp; #endif private: QString _currentDeviceName, _newDeviceName; QMap _deviceProperties; io_method _ioMethod; int _fileDescriptor; std::vector _buffers; PixelFormat _pixelFormat, _pixelFormatConfig; int _pixelDecimation; int _lineLength; int _frameByteSize; // signal detection int _noSignalCounterThreshold; ColorRgb _noSignalThresholdColor; bool _cecDetectionEnabled, _cecStandbyActivated, _signalDetectionEnabled, _noSignalDetected; int _noSignalCounter; double _x_frac_min; double _y_frac_min; double _x_frac_max; double _y_frac_max; QSocketNotifier *_streamNotifier; bool _initialized, _reload; protected: void enumFrameIntervals(QList &framerates, int fileDescriptor, int pixelformat, int width, int height); };