/* Copyright (C) 2012 Carl Hetherington This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #ifndef DVDOMATIC_ENCODER_H #define DVDOMATIC_ENCODER_H /** @file src/encoder.h * @brief Parent class for classes which can encode video and audio frames. */ #include #include #include #include #include #include #include extern "C" { #include } #ifdef HAVE_SWRESAMPLE extern "C" { #include } #endif #include #include "util.h" #include "video_sink.h" #include "audio_sink.h" class EncodeOptions; class Image; class Subtitle; class AudioBuffers; class Film; class ServerDescription; class DCPVideoFrame; /** @class Encoder * @brief Encoder to J2K and WAV for DCP. * * Video is supplied to process_video as YUV frames, and audio * is supplied as uncompressed PCM in blocks of various sizes. */ class Encoder : public VideoSink, public AudioSink { public: Encoder (boost::shared_ptr f, boost::shared_ptr o); virtual ~Encoder (); /** Called to indicate that a processing run is about to begin */ virtual void process_begin (); /** Call with a frame of video. * @param i Video frame image. * @param same true if i is the same as the last time we were called. * @param s A subtitle that should be on this frame, or 0. */ void process_video (boost::shared_ptr i, bool same, boost::shared_ptr s); /** Call with some audio data */ void process_audio (boost::shared_ptr); /** Called when a processing run has finished */ virtual void process_end (); float current_frames_per_second () const; bool skipping () const; SourceFrame video_frame () const; protected: void frame_done (); void frame_skipped (); /** Film that we are encoding */ boost::shared_ptr _film; /** Options */ boost::shared_ptr _opt; /** Mutex for _time_history, _just_skipped and _last_frame */ mutable boost::mutex _history_mutex; /** List of the times of completion of the last _history_size frames; first is the most recently completed. */ std::list _time_history; /** Number of frames that we should keep history for */ static int const _history_size; /** true if the last frame we processed was skipped (because it was already done) */ bool _just_skipped; /** Number of video frames received so far */ SourceFrame _video_frame; /** Number of audio frames received so far */ int64_t _audio_frame; private: void close_sound_files (); void write_audio (boost::shared_ptr audio); void encoder_thread (ServerDescription *); void terminate_worker_threads (); void link (std::string, std::string) const; #if HAVE_SWRESAMPLE SwrContext* _swr_context; #endif std::vector _sound_files; int64_t _audio_frames_written; boost::optional _last_real_frame; bool _process_end; std::list > _queue; std::list _worker_threads; mutable boost::mutex _worker_mutex; boost::condition _worker_condition; }; #endif