]> git.sesse.net Git - casparcg/blobdiff - core/producer/layer.cpp
2.0.0.2: Moved ATL init to shell.
[casparcg] / core / producer / layer.cpp
index b145b9b88096ad1bb363bc4ea2d878f03241c59a..698aa4931a0eda999920374a9017f3b7a5fb97bc 100644 (file)
@@ -2,36 +2,63 @@
 \r
 #include "layer.h"\r
 \r
-#include "../processor/draw_frame.h"\r
+#include "../mixer/frame/draw_frame.h"\r
+#include "../mixer/image/image_mixer.h"\r
+#include "../mixer/audio/audio_mixer.h"\r
 #include "../producer/frame_producer.h"\r
 \r
-#include "../format/video_format.h"\r
+#include "../video_format.h"\r
+\r
+#include <common/utility/assert.h>\r
 \r
 namespace caspar { namespace core {\r
 \r
-struct layer::implementation\r
-{              \r
-       implementation(size_t index) : foreground_(frame_producer::empty()), background_(frame_producer::empty()), last_frame_(draw_frame::empty()), index_(index) {}\r
+struct layer::implementation : boost::noncopyable\r
+{                                      \r
+       safe_ptr<frame_producer>        foreground_;\r
+       safe_ptr<frame_producer>        background_;\r
+       safe_ptr<draw_frame>            last_frame_;\r
+       bool                                            is_paused_;\r
+\r
+public:\r
+       implementation() \r
+               : foreground_(frame_producer::empty())\r
+               , background_(frame_producer::empty())\r
+               , last_frame_(draw_frame::empty())\r
+               , is_paused_(false){}\r
        \r
-       void load(const safe_ptr<frame_producer>& frame_producer, load_option::type option)\r
+       void load(const safe_ptr<frame_producer>& frame_producer, bool play_on_load)\r
        {                       \r
                background_ = frame_producer;\r
-               if(option == load_option::preview)              \r
+               if(play_on_load)\r
+                       play();                 \r
+       }\r
+\r
+       void preview(const safe_ptr<frame_producer>& frame_producer)\r
+       {\r
+               stop();\r
+               load(frame_producer, false);            \r
+               try\r
                {\r
-                       foreground_ = frame_producer::empty();  \r
                        last_frame_ = frame_producer->receive();\r
                }\r
-               else if(option == load_option::auto_play)\r
-                       play();                 \r
+               catch(...)\r
+               {\r
+                       CASPAR_LOG_CURRENT_EXCEPTION();\r
+                       clear();\r
+               }\r
        }\r
        \r
        void play()\r
        {                       \r
-               background_->set_leading_producer(foreground_);\r
-               foreground_ = background_;\r
-               background_ = frame_producer::empty();\r
-               is_paused_ = false;\r
-               CASPAR_LOG(info) << L"layer[" << index_ << L"] Started: " << foreground_->print();\r
+               if(is_paused_)                  \r
+                       is_paused_ = false;\r
+               else\r
+               {\r
+                       background_->set_leading_producer(foreground_);\r
+                       foreground_ = background_;\r
+                       background_ = frame_producer::empty();\r
+               }\r
        }\r
 \r
        void pause()\r
@@ -41,73 +68,73 @@ struct layer::implementation
 \r
        void stop()\r
        {\r
-               foreground_ = frame_producer::empty();\r
+               pause();\r
                last_frame_ = draw_frame::empty();\r
+               foreground_ = frame_producer::empty();\r
        }\r
 \r
        void clear()\r
        {\r
-               foreground_ = frame_producer::empty();\r
+               stop();\r
                background_ = frame_producer::empty();\r
-               last_frame_ = draw_frame::empty();\r
        }\r
        \r
        safe_ptr<draw_frame> receive()\r
        {               \r
-               if(foreground_ == frame_producer::empty() || is_paused_)\r
+               if(is_paused_)\r
                        return last_frame_;\r
 \r
                try\r
                {\r
-                       last_frame_ = foreground_->receive();\r
-\r
+                       last_frame_ = foreground_->receive(); \r
                        if(last_frame_ == draw_frame::eof())\r
                        {\r
-                               CASPAR_LOG(info) << L"layer[" << index_ << L"] Ended:" << foreground_->print();\r
+                               CASPAR_ASSERT(foreground_ != frame_producer::empty());\r
+\r
                                auto following = foreground_->get_following_producer();\r
                                following->set_leading_producer(foreground_);\r
                                foreground_ = following;\r
-                               if(foreground_ != frame_producer::empty())\r
-                                       CASPAR_LOG(info) << L"layer[" << index_ << L"] Started:" << foreground_->print();\r
+\r
                                last_frame_ = receive();\r
                        }\r
                }\r
                catch(...)\r
                {\r
-                       try\r
-                       {\r
-                               CASPAR_LOG_CURRENT_EXCEPTION();\r
-                               CASPAR_LOG(warning) << L"layer[" << index_ << L"] Error. Removed " << foreground_->print() << L" from layer.";\r
-                               foreground_ = frame_producer::empty();\r
-                               last_frame_ = draw_frame::empty();\r
-                       }\r
-                       catch(...){}\r
+                       CASPAR_LOG_CURRENT_EXCEPTION();\r
+                       stop();\r
                }\r
 \r
                return last_frame_;\r
-       }       \r
-               \r
-       tbb::atomic<bool>                       is_paused_;\r
-       safe_ptr<draw_frame>            last_frame_;\r
-       safe_ptr<frame_producer>        foreground_;\r
-       safe_ptr<frame_producer>        background_;\r
-       size_t                                          index_;\r
+       }\r
 };\r
 \r
-layer::layer(size_t index) : impl_(new implementation(index)){}\r
-layer::layer(layer&& other) : impl_(std::move(other.impl_)){other.impl_ = nullptr;}\r
+layer::layer() \r
+{\r
+       impl_ = new implementation();\r
+}\r
+layer::layer(layer&& other) \r
+{\r
+       impl_ = other.impl_.compare_and_swap(nullptr, other.impl_);\r
+}\r
+layer::~layer()\r
+{\r
+       delete impl_.fetch_and_store(nullptr);\r
+}\r
 layer& layer::operator=(layer&& other)\r
 {\r
-       impl_ = std::move(other.impl_); \r
-       other.impl_ = nullptr;\r
+       impl_ = other.impl_.compare_and_swap(nullptr, other.impl_);\r
        return *this;\r
 }\r
-void layer::load(const safe_ptr<frame_producer>& frame_producer, load_option::type option){return impl_->load(frame_producer, option);}        \r
+void layer::swap(layer& other)\r
+{\r
+       impl_ = other.impl_.compare_and_swap(impl_, other.impl_);\r
+}\r
+void layer::load(const safe_ptr<frame_producer>& frame_producer, bool play_on_load){return impl_->load(frame_producer, play_on_load);} \r
+void layer::preview(const safe_ptr<frame_producer>& frame_producer){return impl_->preview(frame_producer);}    \r
 void layer::play(){impl_->play();}\r
 void layer::pause(){impl_->pause();}\r
 void layer::stop(){impl_->stop();}\r
 void layer::clear(){impl_->clear();}\r
-bool layer::empty() const { return impl_->foreground_ == frame_producer::empty() && impl_->background_ == frame_producer::empty();}\r
 safe_ptr<draw_frame> layer::receive() {return impl_->receive();}\r
 safe_ptr<frame_producer> layer::foreground() const { return impl_->foreground_;}\r
 safe_ptr<frame_producer> layer::background() const { return impl_->background_;}\r