]> git.sesse.net Git - casparcg/blob - modules/reroute/producer/reroute_producer.cpp
set svn:eol-style native on .h and .cpp files
[casparcg] / modules / reroute / producer / reroute_producer.cpp
1 /*
2 * Copyright (c) 2011 Sveriges Television AB <info@casparcg.com>
3 *
4 * This file is part of CasparCG (www.casparcg.com).
5 *
6 * CasparCG is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 *
11 * CasparCG is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with CasparCG. If not, see <http://www.gnu.org/licenses/>.
18 *
19 * Author: Robert Nagy, ronag89@gmail.com
20 */
21
22 #include "../stdafx.h"
23
24 #include "reroute_producer.h"
25
26 #include <core/producer/frame_producer.h>
27 #include <core/frame/draw_frame.h>
28 #include <core/frame/frame_factory.h>
29 #include <core/frame/pixel_format.h>
30 #include <core/frame/frame.h>
31 #include <core/video_channel.h>
32 #include <core/producer/stage.h>
33
34 #include <common/except.h>
35 #include <common/diagnostics/graph.h>
36 #include <common/log.h>
37 #include <common/reactive.h>
38
39 #include <asmlib.h>
40
41 #include <tbb/concurrent_queue.h>
42
43 #include <boost/property_tree/ptree.hpp>
44 #include <boost/foreach.hpp>
45 #include <boost/optional.hpp>
46 #include <boost/range/algorithm_ext/push_back.hpp>
47 #include <boost/range/numeric.hpp>
48 #include <boost/range/adaptor/map.hpp>
49
50 #include <queue>
51
52 namespace caspar { namespace reroute {
53                 
54 class reroute_producer : public reactive::observer<std::map<int, core::draw_frame>>
55                                            , public core::frame_producer_base
56 {
57         const spl::shared_ptr<diagnostics::graph>                                               graph_;
58         
59         tbb::concurrent_bounded_queue<std::map<int, core::draw_frame>>  input_buffer_;
60 public:
61         explicit reroute_producer() 
62         {
63                 graph_->set_color("late-frame", diagnostics::color(0.6f, 0.3f, 0.3f));
64                 graph_->set_color("dropped-frame", diagnostics::color(0.3f, 0.6f, 0.3f));
65                 graph_->set_text(print());
66                 diagnostics::register_graph(graph_);
67
68                 input_buffer_.set_capacity(1);
69         }
70                 
71         // observable
72
73         void on_next(const std::map<int, core::draw_frame>& frames)
74         {
75                 if(!input_buffer_.try_push(frames))
76                         graph_->set_tag("dropped-frame");               
77         }
78
79         // frame_producer
80                         
81         core::draw_frame receive_impl() override
82         {               
83                 std::map<int, core::draw_frame> frames;
84                 if(!input_buffer_.try_pop(frames))
85                 {
86                         graph_->set_tag("late-frame");
87                         return core::draw_frame::late();                
88                 }
89
90                 return boost::accumulate(frames | boost::adaptors::map_values, core::draw_frame::empty(), core::draw_frame::over);
91         }       
92                 
93         std::wstring print() const override
94         {
95                 return L"reroute[]";
96         }
97
98         std::wstring name() const override
99         {
100                 return L"reroute";
101         }
102
103         boost::property_tree::wptree info() const override
104         {
105                 boost::property_tree::wptree info;
106                 info.add(L"type", L"rerotue-producer");
107                 return info;
108         }
109                 
110         void subscribe(const monitor::observable::observer_ptr& o) override
111         {
112         }
113
114         void unsubscribe(const monitor::observable::observer_ptr& o) override
115         {
116         }
117 };
118
119 spl::shared_ptr<core::frame_producer> create_producer(core::video_channel& channel)
120 {
121         auto producer = spl::make_shared<reroute_producer>();
122         
123         std::weak_ptr<reactive::observer<std::map<int, core::draw_frame>>> o = producer;
124
125         channel.stage().subscribe(o);
126
127         return producer;
128 }
129
130 }}