X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=mixer.h;h=6725a7f46e242fe44bdba8a7c0e583b422b42102;hb=0d7183f398856c9331e58808d0374e63593334b6;hp=80d18ff00395c3651ff461b72b03e72da51b7f7a;hpb=9addf2b67e26eb7cb8d7d02a0b11a1f50e95b334;p=nageru diff --git a/mixer.h b/mixer.h index 80d18ff..6725a7f 100644 --- a/mixer.h +++ b/mixer.h @@ -5,18 +5,42 @@ #include #undef Success +#include +#include + #include #include +#include +#include +#include #include - -#include "bmusb.h" +#include +#include +#include +#include +#include + +#include "bmusb/bmusb.h" +#include "ebu_r128_proc.h" #include "h264encode.h" +#include "httpd.h" #include "pbo_frame_allocator.h" #include "ref_counted_frame.h" #include "ref_counted_gl_sync.h" +#include "resampler.h" #include "theme.h" +#include "timebase.h" +#include "stereocompressor.h" +#include "filter.h" -#define NUM_CARDS 2 +class H264Encoder; +class QSurface; +namespace movit { +class Effect; +class EffectChain; +class FlatInput; +class ResourcePool; +} // namespace movit namespace movit { class YCbCrInput; @@ -27,24 +51,19 @@ class QSurfaceFormat; class Mixer { public: // The surface format is used for offscreen destinations for OpenGL contexts we need. - Mixer(const QSurfaceFormat &format); + Mixer(const QSurfaceFormat &format, unsigned num_cards); ~Mixer(); void start(); void quit(); - enum Source { - SOURCE_INPUT1, - SOURCE_INPUT2, - SOURCE_SBS, - }; - void cut(Source source); + void transition_clicked(int transition_num); + void channel_clicked(int preview_num); enum Output { OUTPUT_LIVE = 0, OUTPUT_PREVIEW, - OUTPUT_INPUT0, - OUTPUT_INPUT1, - NUM_OUTPUTS + OUTPUT_INPUT0, // 1, 2, 3, up to 15 follow numerically. + NUM_OUTPUTS = 18 }; struct DisplayFrame { @@ -79,14 +98,59 @@ public: output_channel[output].set_frame_ready_callback(callback); } + typedef std::function audio_level_callback_t; + void set_audio_level_callback(audio_level_callback_t callback) + { + audio_level_callback = callback; + } + + std::vector get_transition_names() + { + return theme->get_transition_names(pts()); + } + + unsigned get_num_channels() const + { + return theme->get_num_channels(); + } + + std::string get_channel_name(unsigned channel) const + { + return theme->get_channel_name(channel); + } + + bool get_supports_set_wb(unsigned channel) const + { + return theme->get_supports_set_wb(channel); + } + + void set_wb(unsigned channel, double r, double g, double b) const + { + theme->set_wb(channel, r, g, b); + } + + void set_locut_cutoff(float cutoff_hz) + { + locut_cutoff_hz = cutoff_hz; + } + + void reset_meters(); + private: - void bm_frame(int card_index, uint16_t timecode, + void bm_frame(unsigned card_index, uint16_t timecode, FrameAllocator::Frame video_frame, size_t video_offset, uint16_t video_format, FrameAllocator::Frame audio_frame, size_t audio_offset, uint16_t audio_format); void place_rectangle(movit::Effect *resample_effect, movit::Effect *padding_effect, float x0, float y0, float x1, float y1); void thread_func(); + void process_audio_one_frame(); void subsample_chroma(GLuint src_tex, GLuint dst_dst); void release_display_frame(DisplayFrame *frame); + double pts() { return double(pts_int) / TIMEBASE; } + + HTTPD httpd; + unsigned num_cards; QSurface *mixer_surface, *h264_encoder_surface; std::unique_ptr resource_pool; @@ -98,30 +162,35 @@ private: // Effects part of . Owned by . movit::FlatInput *display_input; - Source current_source = SOURCE_INPUT1; - int frame = 0; + int64_t pts_int = 0; // In TIMEBASE units. std::mutex bmusb_mutex; struct CaptureCard { BMUSBCapture *usb; std::unique_ptr frame_allocator; - // Threading stuff - bool thread_initialized = false; + // Stuff for the OpenGL context (for texture uploading). QSurface *surface; QOpenGLContext *context; bool new_data_ready = false; // Whether new_frame contains anything. + bool should_quit = false; RefCountedFrame new_frame; GLsync new_data_ready_fence; // Whether new_frame is ready for rendering. std::condition_variable new_data_ready_changed; // Set whenever new_data_ready is changed. + unsigned dropped_frames = 0; // Before new_frame. + + std::mutex audio_mutex; + std::unique_ptr resampler; // Under audio_mutex. + int last_timecode = -1; // Unwrapped. }; - CaptureCard cards[NUM_CARDS]; // protected by + CaptureCard cards[MAX_CARDS]; // protected by - RefCountedFrame bmusb_current_rendering_frame[NUM_CARDS]; + RefCountedFrame bmusb_current_rendering_frame[MAX_CARDS]; class OutputChannel { public: + ~OutputChannel(); void output_frame(DisplayFrame frame); bool get_display_frame(DisplayFrame *frame); void set_frame_ready_callback(new_frame_ready_callback_t callback); @@ -140,6 +209,22 @@ private: std::thread mixer_thread; bool should_quit = false; + + audio_level_callback_t audio_level_callback = nullptr; + Ebu_r128_proc r128; + + // TODO: Implement oversampled peak detection. + std::atomic peak{0.0f}; + + StereoFilter locut; // Default cutoff 150 Hz, 24 dB/oct. + std::atomic locut_cutoff_hz; + + // First compressor; takes us up to about -12 dBFS. + StereoCompressor level_compressor; + float last_gain_staging_db = 0.0f; + + StereoCompressor limiter; + StereoCompressor compressor; }; extern Mixer *global_mixer;