// Copyright 2018 Citra Emulator Project // Licensed under GPLv2 or any later version // Refer to the license.txt file included. #pragma once #include #include #include #include #include #include #include #include #include "common/common_types.h" #include "common/thread.h" #include "common/threadsafe_queue.h" #include "core/dumping/backend.h" extern "C" { #include #include #include #include #include } namespace VideoDumper { using VariableAudioFrame = std::vector; void InitFFmpegLibraries(); /** * Wrapper around FFmpeg AVCodecContext + AVStream. * Rescales/Resamples, encodes and writes a frame. */ class FFmpegStream { public: bool Init(AVFormatContext* format_context); void Free(); void Flush(); protected: ~FFmpegStream(); void WritePacket(AVPacket& packet); void SendFrame(AVFrame* frame); struct AVCodecContextDeleter { void operator()(AVCodecContext* codec_context) const { avcodec_free_context(&codec_context); } }; struct AVFrameDeleter { void operator()(AVFrame* frame) const { av_frame_free(&frame); } }; AVFormatContext* format_context{}; std::unique_ptr codec_context{}; AVStream* stream{}; }; /** * A FFmpegStream used for video data. * Rescales, encodes and writes a frame. */ class FFmpegVideoStream : public FFmpegStream { public: ~FFmpegVideoStream(); bool Init(AVFormatContext* format_context, AVOutputFormat* output_format, const Layout::FramebufferLayout& layout); void Free(); void ProcessFrame(VideoFrame& frame); private: struct SwsContextDeleter { void operator()(SwsContext* sws_context) const { sws_freeContext(sws_context); } }; u64 frame_count{}; std::unique_ptr current_frame{}; std::unique_ptr scaled_frame{}; std::unique_ptr sws_context{}; Layout::FramebufferLayout layout; /// The pixel format the frames are stored in static constexpr AVPixelFormat pixel_format = AVPixelFormat::AV_PIX_FMT_BGRA; }; /** * A FFmpegStream used for audio data. * Resamples (converts), encodes and writes a frame. * This also temporarily stores resampled audio data before there are enough to form a frame. */ class FFmpegAudioStream : public FFmpegStream { public: ~FFmpegAudioStream(); bool Init(AVFormatContext* format_context); void Free(); void ProcessFrame(const VariableAudioFrame& channel0, const VariableAudioFrame& channel1); void Flush(); private: struct SwrContextDeleter { void operator()(SwrContext* swr_context) const { swr_free(&swr_context); } }; u64 frame_size{}; u64 frame_count{}; std::unique_ptr audio_frame{}; std::unique_ptr swr_context{}; u8** resampled_data{}; u64 offset{}; // Number of output samples that are currently in resampled_data. }; /** * Wrapper around FFmpeg AVFormatContext. * Manages the video and audio streams, and accepts video and audio data. */ class FFmpegMuxer { public: ~FFmpegMuxer(); bool Init(const std::string& path, const Layout::FramebufferLayout& layout); void Free(); void ProcessVideoFrame(VideoFrame& frame); void ProcessAudioFrame(const VariableAudioFrame& channel0, const VariableAudioFrame& channel1); void FlushVideo(); void FlushAudio(); void WriteTrailer(); private: struct AVFormatContextDeleter { void operator()(AVFormatContext* format_context) const { avio_closep(&format_context->pb); avformat_free_context(format_context); } }; FFmpegAudioStream audio_stream{}; FFmpegVideoStream video_stream{}; std::unique_ptr format_context{}; }; /** * FFmpeg video dumping backend. * This class implements a double buffer. */ class FFmpegBackend : public Backend { public: FFmpegBackend(); ~FFmpegBackend() override; bool StartDumping(const std::string& path, const Layout::FramebufferLayout& layout) override; void AddVideoFrame(VideoFrame frame) override; void AddAudioFrame(AudioCore::StereoFrame16 frame) override; void AddAudioSample(const std::array& sample) override; void StopDumping() override; bool IsDumping() const override; Layout::FramebufferLayout GetLayout() const override; private: void EndDumping(); std::atomic_bool is_dumping = false; ///< Whether the backend is currently dumping FFmpegMuxer ffmpeg{}; Layout::FramebufferLayout video_layout; std::array video_frame_buffers; u32 current_buffer = 0, next_buffer = 1; Common::Event event1, event2; std::thread video_processing_thread; std::array, 2> audio_frame_queues; std::thread audio_processing_thread; Common::Event processing_ended; }; /// Struct describing encoder/muxer options struct OptionInfo { std::string name; std::string description; AVOptionType type; std::string default_value; struct NamedConstant { std::string name; std::string description; s64 value; }; std::vector named_constants; // If this is a scalar type double min; double max; }; /// Struct describing an encoder struct EncoderInfo { std::string name; std::string long_name; AVCodecID codec; std::vector options; }; /// Struct describing a format struct FormatInfo { std::string name; std::string long_name; std::vector extensions; std::set supported_video_codecs; std::set supported_audio_codecs; std::vector options; }; std::vector ListEncoders(AVMediaType type); std::vector ListFormats(); } // namespace VideoDumper