#pragma once // c/cpp #include #include #include #include // qt #include #include #include // orpheus #include "constants.h" #include "icamera.h" #include "image.h" #include "utils/sem_queue.h" namespace veye { namespace imx287m { class i2c; } // namespace imx287m } // namespace veye class HttpServer; class VeyeIMX287m : public ICamera { constexpr static char videoDevice[] = "/dev/video0"; public: using buffer_t = std::array; public: VeyeIMX287m(); ~VeyeIMX287m() override; public: static std::vector> search(); public: bool startStream() override; bool dequeueImageBuffer(size_t &image); // bool getImage(Image &image); bool getImage(Image *image) override; std::shared_ptr getImage() override; bool init(); // parameters public: bool set_autoExposure(const bool enable) override; std::optional get_autoExposure() override; bool set_autoGain(const bool enable) override; std::optional get_autoGain() override; bool set_exposureTime(const std::chrono::microseconds us) override; std::optional get_exposureTime() override; bool set_gain(const float value) override; std::optional get_gain() override; public: /*! * \brief processedCounter - count of images processed in current second. * Used for performance measurement and bottlenecks analysing */ uint32_t processedCounter{0}; private: bool openCam(); bool initCam(); bool initI2C(); // bool initHttpServer(); void getFrameLoop(std::stop_token stopToken); void rotateFrameLoop(std::stop_token stopToken); void calcPixelsLoop(std::stop_token stopToken); private: /*! * \brief m_cam_fd - camera file descriptor */ int m_cam_fd{-1}; /*! * \brief m_previousFrameCounter - used to detect dropped frames */ std::optional m_previousFrameCounter{}; static constexpr uint8_t BUFFER_COUNT{4}; // std::array m_images; /*! * \brief m_imageMutexes - lock while processing image from m_images */ std::array m_imageMutexes; // TODO: split this // there should be no chance of changing image by ioctl during futher processing struct buffer { void *mem{nullptr}; // std::shared_ptr image{std::make_shared()}; }; std::vector m_rawBuffers; struct Semaphore { sem_queue, BUFFER_COUNT> rawSemQueue; // sem_queue rawSemQueue; sem_queue, BUFFER_COUNT> rotSemQueue; // sem_queue rotSemQueue; } m_sync; std::mutex m_camMtx; /*! * \brief m_buffersQueue - queue of buffers which require extracting pixels */ std::queue> m_buffersQueue; std::jthread m_streamThread; std::jthread m_getThreads[1]; // std::jthread m_getThreads[4]; // TODO: sync all loops somehow to guarantee frames order std::jthread m_rotateThreads[2]; std::jthread m_calcPixelsThreads[2]; std::mutex m_lastImageMtx; std::shared_ptr m_lastProcessedImage{}; std::shared_ptr m_i2c; std::shared_ptr m_httpServer; };