X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=mixer.h;h=252f961f7668cbf4ceaeab2364bf2196c799131d;hb=3b77fb6a74f7a58033168202a772f3630fb9a30f;hp=02c06edfa46f2d8ee12278dee03320efe979e673;hpb=9f1e8fb59e1b68b68b4bb1a05e1f4ee37ea47471;p=nageru diff --git a/mixer.h b/mixer.h index 02c06ed..252f961 100644 --- a/mixer.h +++ b/mixer.h @@ -10,6 +10,8 @@ #include #include +#include +#include #include #include #include @@ -26,9 +28,11 @@ #include "pbo_frame_allocator.h" #include "ref_counted_frame.h" #include "ref_counted_gl_sync.h" -#include "resampler.h" +#include "resampling_queue.h" #include "theme.h" #include "timebase.h" +#include "stereocompressor.h" +#include "filter.h" class H264Encoder; class QSurface; @@ -95,7 +99,9 @@ public: output_channel[output].set_frame_ready_callback(callback); } - typedef std::function audio_level_callback_t; + typedef std::function audio_level_callback_t; void set_audio_level_callback(audio_level_callback_t callback) { audio_level_callback = callback; @@ -126,12 +132,50 @@ public: theme->set_wb(channel, r, g, b); } + void set_locut_cutoff(float cutoff_hz) + { + locut_cutoff_hz = cutoff_hz; + } + + float get_limiter_threshold_dbfs() + { + return limiter_threshold_dbfs; + } + + float get_compressor_threshold_dbfs() + { + return compressor_threshold_dbfs; + } + + void set_limiter_threshold_dbfs(float threshold_dbfs) + { + limiter_threshold_dbfs = threshold_dbfs; + } + + void set_compressor_threshold_dbfs(float threshold_dbfs) + { + compressor_threshold_dbfs = threshold_dbfs; + } + + void set_limiter_enabled(bool enabled) + { + limiter_enabled = enabled; + } + + void set_compressor_enabled(bool enabled) + { + compressor_enabled = enabled; + } + + void reset_meters(); + private: 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; } @@ -160,16 +204,15 @@ private: QSurface *surface; QOpenGLContext *context; - bool new_data_ready = false; // Whether new_frame and new_frame_audio contains anything. + 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::vector new_frame_audio; 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. + std::unique_ptr resampling_queue; // Under audio_mutex. int last_timecode = -1; // Unwrapped. }; CaptureCard cards[MAX_CARDS]; // protected by @@ -201,8 +244,24 @@ private: audio_level_callback_t audio_level_callback = nullptr; Ebu_r128_proc r128; - // TODO: Implement oversampled peak detection. - float peak = 0.0f; + Resampler peak_resampler; + 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; + + static constexpr float ref_level_dbfs = -14.0f; + + StereoCompressor limiter; + std::atomic limiter_threshold_dbfs{ref_level_dbfs + 4.0f}; // 4 dB. + std::atomic limiter_enabled{true}; + StereoCompressor compressor; + std::atomic compressor_threshold_dbfs{ref_level_dbfs - 12.0f}; // -12 dB. + std::atomic compressor_enabled{true}; }; extern Mixer *global_mixer;