X-Git-Url: https://main.carlh.net/gitweb/?p=dcpomatic.git;a=blobdiff_plain;f=src%2Flib%2Fbutler.h;h=c7e71658da25889107d033147e62b16897f92f4d;hp=fb133d10809413298364faba4e02a8d62c42c5b1;hb=ac34066d5e448d1984d11a180be74e31b6e13b5c;hpb=a57717d54c34096d80a467ad78e9fdcecca70b2c diff --git a/src/lib/butler.h b/src/lib/butler.h index fb133d108..c7e71658d 100644 --- a/src/lib/butler.h +++ b/src/lib/butler.h @@ -1,5 +1,5 @@ /* - Copyright (C) 2016-2017 Carl Hetherington + Copyright (C) 2016-2021 Carl Hetherington This file is part of DCP-o-matic. @@ -18,37 +18,67 @@ */ -#include "video_ring_buffers.h" -#include "audio_ring_buffers.h" -#include "text_ring_buffers.h" + #include "audio_mapping.h" +#include "audio_ring_buffers.h" +#include "change_signaller.h" #include "exception_store.h" -#include -#include +#include "text_ring_buffers.h" +#include "video_ring_buffers.h" +#include +#include #include #include -#include -#include + class Player; class PlayerVideo; -class Log; -class Butler : public ExceptionStore, public boost::noncopyable + +class Butler : public ExceptionStore { public: - Butler (boost::shared_ptr player, boost::shared_ptr log, AudioMapping map, int audio_channels); + Butler ( + std::weak_ptr film, + std::shared_ptr player, + AudioMapping map, + int audio_channels, + std::function pixel_format, + VideoRange video_range, + Image::Alignment alignment, + bool fast, + bool prepare_only_proxy + ); + ~Butler (); - void seek (DCPTime position, bool accurate); + Butler (Butler const&) = delete; + Butler& operator= (Butler const&) = delete; + + void seek (dcpomatic::DCPTime position, bool accurate); + + class Error { + public: + enum class Code { + NONE, + AGAIN, + DIED, + FINISHED + }; + + Code code = Code::NONE; + std::string message; + + std::string summary () const; + }; - enum Error { - NONE, - AGAIN + enum class Behaviour { + BLOCKING, + NON_BLOCKING }; - std::pair, DCPTime> get_video (Error* e = 0); - boost::optional get_audio (float* out, Frame frames); + std::pair, dcpomatic::DCPTime> get_video (Behaviour behaviour, Error* e = nullptr); + boost::optional get_audio (Behaviour behaviour, float* out, Frame frames); boost::optional get_closed_caption (); void disable_audio (); @@ -57,41 +87,36 @@ public: private: void thread (); - void video (boost::shared_ptr video, DCPTime time); - void audio (boost::shared_ptr audio, DCPTime time); - void text (PlayerText pt, TextType type, boost::optional track, DCPTimePeriod period); + void video (std::shared_ptr video, dcpomatic::DCPTime time); + void audio (std::shared_ptr audio, dcpomatic::DCPTime time, int frame_rate); + void text (PlayerText pt, TextType type, boost::optional track, dcpomatic::DCPTimePeriod period); bool should_run () const; - void prepare (boost::weak_ptr video) const; - void player_change (ChangeType type, bool frequent); - void seek_unlocked (DCPTime position, bool accurate); - - 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; + void prepare (std::weak_ptr video); + void player_change (ChangeType type, int property); + void seek_unlocked (dcpomatic::DCPTime position, bool accurate); + + std::weak_ptr _film; + std::shared_ptr _player; + boost::thread _thread; + VideoRingBuffers _video; AudioRingBuffers _audio; TextRingBuffers _closed_caption; boost::thread_group _prepare_pool; boost::asio::io_service _prepare_service; - boost::shared_ptr _prepare_work; + std::shared_ptr _prepare_work; - /** mutex to protect _pending_seek_position, _pending_seek_acurate, _finished, _died, _stop_thread */ + /** mutex to protect _pending_seek_position, _pending_seek_accurate, _finished, _died, _stop_thread */ boost::mutex _mutex; boost::condition _summon; boost::condition _arrived; - boost::optional _pending_seek_position; + boost::optional _pending_seek_position; bool _pending_seek_accurate; int _suspended; bool _finished; bool _died; + std::string _died_message; bool _stop_thread; AudioMapping _audio_mapping; @@ -99,10 +124,23 @@ private: bool _disable_audio; + std::function _pixel_format; + VideoRange _video_range; + Image::Alignment _alignment; + bool _fast; + + /** true to ask PlayerVideo::prepare to only prepare the ImageProxy and not also + * the final image. We want to do this when the viewer is intending to call + * PlayerVideo::raw_image() and do the things in PlayerVideo::make_imgae() itself: + * this is the case for the GLVideoView which can do scale, pixfmt conversion etc. + * in the shader. + */ + bool _prepare_only_proxy = false; + /** If we are waiting to be refilled following a seek, this is the time we were seeking to. */ - boost::optional _awaiting; + boost::optional _awaiting; boost::signals2::scoped_connection _player_video_connection; boost::signals2::scoped_connection _player_audio_connection;