X-Git-Url: https://git.carlh.net/gitweb/?a=blobdiff_plain;f=src%2Flib%2Fbutler.h;h=56374deeb31a9c904b7f2dd175e64b82463ceffc;hb=1fe6bd7f8ba059322b8357b2210f0fd590567ce2;hp=943aa9e0fa7dadeefe3cf9b33d45e9d2e2313b76;hpb=9e58ca88bdb5a108cb7b2adedb2aa54e620e7cd2;p=dcpomatic.git diff --git a/src/lib/butler.h b/src/lib/butler.h index 943aa9e0f..56374deeb 100644 --- a/src/lib/butler.h +++ b/src/lib/butler.h @@ -20,6 +20,7 @@ #include "video_ring_buffers.h" #include "audio_ring_buffers.h" +#include "text_ring_buffers.h" #include "audio_mapping.h" #include "exception_store.h" #include @@ -27,49 +28,78 @@ #include #include #include +#include -class Film; class Player; class PlayerVideo; +class Log; class Butler : public ExceptionStore, public boost::noncopyable { public: - Butler (boost::weak_ptr film, boost::shared_ptr player, AudioMapping map, int audio_channels); + Butler (boost::shared_ptr player, boost::shared_ptr log, AudioMapping map, int audio_channels); ~Butler (); void seek (DCPTime position, bool accurate); std::pair, DCPTime> get_video (); - void get_audio (float* out, Frame frames); + boost::optional get_audio (float* out, Frame frames); + boost::optional > get_closed_caption (); + + void disable_audio (); + + std::pair memory_used () const; private: void thread (); void video (boost::shared_ptr video, DCPTime time); void audio (boost::shared_ptr audio, DCPTime time); - void player_changed (); + void text (PlayerText pt, TextType type, DCPTimePeriod period); bool should_run () const; + void prepare (boost::weak_ptr video) const; + void return_seek (bool frequent); + void seek_unlocked (DCPTime position, bool accurate); - boost::weak_ptr _film; boost::shared_ptr _player; + boost::shared_ptr _log; boost::thread* _thread; + /** mutex to protect _video, _audio and _closed_caption for when we are clearing them and they all need to be + cleared together without any data being inserted in the interim; + XXX: is this necessary now that all butler output data is timestamped? Perhaps the locked clear-out + is only required if we guarantee that get_video() and get_audio() calls are in sync. + */ + boost::mutex _buffers_mutex; VideoRingBuffers _video; AudioRingBuffers _audio; + TextRingBuffers _closed_caption; + boost::thread_group _prepare_pool; + boost::asio::io_service _prepare_service; + boost::shared_ptr _prepare_work; + + /** mutex to protect _pending_seek_position, _pending_seek_acurate, _finished, _died, _stop_thread */ boost::mutex _mutex; boost::condition _summon; boost::condition _arrived; boost::optional _pending_seek_position; bool _pending_seek_accurate; - bool _finished; + bool _died; + bool _stop_thread; AudioMapping _audio_mapping; int _audio_channels; - bool _stop_thread; + bool _disable_audio; + + /** If we are waiting to be refilled following a seek, this is the time we were + seeking to. + */ + boost::optional _awaiting; boost::signals2::scoped_connection _player_video_connection; boost::signals2::scoped_connection _player_audio_connection; + boost::signals2::scoped_connection _player_text_connection; boost::signals2::scoped_connection _player_changed_connection; + boost::signals2::scoped_connection _player_not_changed_connection; };