#ifndef _MJPEG_ENCODER_H
#define _MJPEG_ENCODER_H 1
-#include "defs.h"
#include "shared/ffmpeg_raii.h"
+#include "shared/ff_maybe_const.h"
#include "shared/httpd.h"
+#include "shared/va_resource_pool.h"
#include "ref_counted_frame.h"
extern "C" {
#include <atomic>
#include <bmusb/bmusb.h>
#include <condition_variable>
-#include <list>
+#include <map>
+#include <memory>
#include <mutex>
#include <queue>
+#include <stddef.h>
#include <stdint.h>
#include <string>
#include <thread>
+#include <vector>
#include <movit/effect.h>
#include <va/va.h>
+#include <va/va_enc_jpeg.h>
struct jpeg_compress_struct;
struct VADisplayWithCleanup;
bool using_vaapi() const { return va_dpy != nullptr; }
bool should_encode_mjpeg_for_card(unsigned card_index);
+ VAResourcePool *get_va_pool() const { return va_pool.get(); }
private:
static constexpr int quality = 90;
- struct VAResources {
- unsigned width, height;
- uint32_t fourcc;
- VASurfaceID surface;
- VAContextID context;
- VABufferID data_buffer;
- VAImage image;
- };
-
- // RAII wrapper to release VAResources on return (even on error).
- class ReleaseVAResources {
- public:
- ReleaseVAResources() : committed(true) {}
-
- ReleaseVAResources(MJPEGEncoder *mjpeg, const VAResources &resources)
- : mjpeg(mjpeg), resources(resources) {}
-
- ReleaseVAResources(ReleaseVAResources &) = delete;
-
- ReleaseVAResources(ReleaseVAResources &&other)
- : mjpeg(other.mjpeg), resources(other.resources), committed(other.committed) {
- other.commit();
- }
-
- ReleaseVAResources &operator= (ReleaseVAResources &) = delete;
-
- ReleaseVAResources &operator= (ReleaseVAResources &&other) {
- if (!committed) {
- mjpeg->release_va_resources(resources);
- }
- mjpeg = other.mjpeg;
- resources = std::move(other.resources);
- committed = other.committed;
- other.commit();
- return *this;
- }
-
- ~ReleaseVAResources()
- {
- if (!committed) {
- mjpeg->release_va_resources(resources);
- }
- }
-
- void commit() { committed = true; }
-
- private:
- MJPEGEncoder *mjpeg = nullptr;
- VAResources resources;
- bool committed = false;
- };
-
struct QueuedFrame {
int64_t pts;
unsigned card_index;
movit::RGBTriplet white_balance;
// Only for frames in the process of being encoded by VA-API.
- VAResources resources;
+ VAResourcePool::VAResources resources;
ReleaseVAResources resource_releaser;
};
HTTPD::StreamID stream_id;
};
std::map<HTTPD::StreamID, WritePacket2Context> ffmpeg_contexts; // Statically set up, so we never need to worry about dangling pointers.
- static int write_packet2_thunk(void *opaque, uint8_t *buf, int buf_size, AVIODataMarkerType type, int64_t time);
- int write_packet2(HTTPD::StreamID stream_id, uint8_t *buf, int buf_size, AVIODataMarkerType type, int64_t time);
+ static int write_packet2_thunk(void *opaque, FF_MAYBE_CONST uint8_t *buf, int buf_size, AVIODataMarkerType type, int64_t time);
+ int write_packet2(HTTPD::StreamID stream_id, FF_MAYBE_CONST uint8_t *buf, int buf_size, AVIODataMarkerType type, int64_t time);
std::thread encoder_thread, va_receiver_thread;
bool running = false;
std::unique_ptr<VADisplayWithCleanup> va_dpy;
- VAConfigID config_id_422, config_id_420;
+ std::unique_ptr<VAResourcePool> va_pool;
struct VAKey {
unsigned width, height, y_h_samp_factor, y_v_samp_factor;
std::map<VAKey, VAData> va_data_for_parameters;
VAData get_va_data_for_parameters(unsigned width, unsigned height, unsigned y_h_samp_factor, unsigned y_v_samp_factor, const movit::RGBTriplet &white_balance);
- std::list<VAResources> va_resources_freelist;
- std::mutex va_resources_mutex;
- VAResources get_va_resources(unsigned width, unsigned height, uint32_t fourcc);
- void release_va_resources(VAResources resources);
-
uint8_t *tmp_y, *tmp_cbcr, *tmp_cb, *tmp_cr; // Private to the encoder thread. Used by the libjpeg backend only.
std::atomic<int64_t> metric_mjpeg_frames_zero_size_dropped{0};