/*\r
-* Copyright (c) 2011 Sveriges Television AB <info@casparcg.com>\r
+* Copyright 2013 Sveriges Television AB http://casparcg.com/\r
*\r
* This file is part of CasparCG (www.casparcg.com).\r
*\r
#include "consumer/output.h"\r
#include "mixer/mixer.h"\r
#include "mixer/gpu/ogl_device.h"\r
+#include "mixer/audio/audio_util.h"\r
#include "producer/stage.h"\r
\r
#include <common/diagnostics/graph.h>\r
\r
struct video_channel::implementation : boost::noncopyable\r
{\r
- const int index_;\r
- video_format_desc format_desc_;\r
- const safe_ptr<ogl_device> ogl_;\r
- safe_ptr<diagnostics::graph> graph_;\r
-\r
- safe_ptr<caspar::core::output> output_;\r
- safe_ptr<caspar::core::mixer> mixer_;\r
- safe_ptr<caspar::core::stage> stage_;\r
+ video_channel& self_;\r
+ const int index_;\r
+ video_format_desc format_desc_;\r
+ const safe_ptr<ogl_device> ogl_;\r
+ const safe_ptr<diagnostics::graph> graph_;\r
+\r
+ const safe_ptr<caspar::core::output> output_;\r
+ const safe_ptr<caspar::core::mixer> mixer_;\r
+ const safe_ptr<caspar::core::stage> stage_;\r
+\r
+ safe_ptr<monitor::subject> monitor_subject_;\r
\r
public:\r
- implementation(int index, const video_format_desc& format_desc, const safe_ptr<ogl_device>& ogl) \r
- : index_(index)\r
+ implementation(video_channel& self, int index, const video_format_desc& format_desc, const safe_ptr<ogl_device>& ogl, const channel_layout& audio_channel_layout) \r
+ : self_(self)\r
+ , index_(index)\r
, format_desc_(format_desc)\r
, ogl_(ogl)\r
- , output_(new caspar::core::output(graph_, format_desc, index))\r
- , mixer_(new caspar::core::mixer(graph_, output_, format_desc, ogl))\r
+ , output_(new caspar::core::output(graph_, format_desc, audio_channel_layout, index))\r
+ , mixer_(new caspar::core::mixer(graph_, output_, format_desc, ogl, audio_channel_layout))\r
, stage_(new caspar::core::stage(graph_, mixer_, format_desc)) \r
+ , monitor_subject_(make_safe<monitor::subject>("/channel/" + boost::lexical_cast<std::string>(index)))\r
{\r
graph_->set_text(print());\r
diagnostics::register_graph(graph_);\r
for(int n = 0; n < std::max(1, env::properties().get(L"configuration.pipeline-tokens", 2)); ++n)\r
stage_->spawn_token();\r
\r
+ stage_->monitor_output().attach_parent(monitor_subject_);\r
+ mixer_->monitor_output().attach_parent(monitor_subject_);\r
+ output_->monitor_output().attach_parent(monitor_subject_);\r
+\r
CASPAR_LOG(info) << print() << " Successfully Initialized.";\r
}\r
- \r
+\r
void set_video_format_desc(const video_format_desc& format_desc)\r
{\r
if(format_desc.format == core::video_format::invalid)\r
{\r
output_->set_video_format_desc(format_desc);\r
mixer_->set_video_format_desc(format_desc);\r
+ stage_->set_video_format_desc(format_desc);\r
ogl_->gc();\r
}\r
catch(...)\r
{\r
output_->set_video_format_desc(format_desc_);\r
mixer_->set_video_format_desc(format_desc_);\r
+ stage_->set_video_format_desc(format_desc_);\r
throw;\r
}\r
format_desc_ = format_desc;\r
auto mixer_info = mixer_->info();\r
auto output_info = output_->info();\r
\r
- stage_info.timed_wait(boost::posix_time::seconds(2));\r
- mixer_info.timed_wait(boost::posix_time::seconds(2));\r
- output_info.timed_wait(boost::posix_time::seconds(2));\r
- \r
info.add(L"video-mode", format_desc_.name);\r
- info.add_child(L"stage", stage_info.get());\r
- info.add_child(L"mixer", mixer_info.get());\r
- info.add_child(L"output", output_info.get());\r
+\r
+ if (stage_info.timed_wait(boost::posix_time::seconds(2)))\r
+ info.add_child(L"stage", stage_info.get());\r
+\r
+ if (mixer_info.timed_wait(boost::posix_time::seconds(2)))\r
+ info.add_child(L"mixer", mixer_info.get());\r
+\r
+ if (output_info.timed_wait(boost::posix_time::seconds(2)))\r
+ info.add_child(L"output", output_info.get());\r
\r
return info; \r
}\r
+\r
+ boost::property_tree::wptree delay_info() const\r
+ {\r
+ boost::property_tree::wptree info;\r
+\r
+ auto stage_info = stage_->delay_info();\r
+ auto mixer_info = mixer_->delay_info();\r
+ auto output_info = output_->delay_info();\r
+\r
+ if (stage_info.timed_wait(boost::posix_time::seconds(2)))\r
+ info.add_child(L"layers", stage_info.get());\r
+\r
+ if (mixer_info.timed_wait(boost::posix_time::seconds(2)))\r
+ info.add_child(L"mix-time", mixer_info.get());\r
+\r
+ if (output_info.timed_wait(boost::posix_time::seconds(2)))\r
+ info.add_child(L"consumers", output_info.get());\r
+\r
+ return info;\r
+ }\r
};\r
\r
-video_channel::video_channel(int index, const video_format_desc& format_desc, const safe_ptr<ogl_device>& ogl) : impl_(new implementation(index, format_desc, ogl)){}\r
+video_channel::video_channel(int index, const video_format_desc& format_desc, const safe_ptr<ogl_device>& ogl, const channel_layout& audio_channel_layout) \r
+ : impl_(new implementation(*this, index, format_desc, ogl, audio_channel_layout)){}\r
safe_ptr<stage> video_channel::stage() { return impl_->stage_;} \r
safe_ptr<mixer> video_channel::mixer() { return impl_->mixer_;} \r
safe_ptr<output> video_channel::output() { return impl_->output_;} \r
video_format_desc video_channel::get_video_format_desc() const{return impl_->format_desc_;}\r
void video_channel::set_video_format_desc(const video_format_desc& format_desc){impl_->set_video_format_desc(format_desc);}\r
boost::property_tree::wptree video_channel::info() const{return impl_->info();}\r
-\r
+int video_channel::index() const {return impl_->index_;}\r
+monitor::subject& video_channel::monitor_output(){return *impl_->monitor_subject_;}\r
+boost::property_tree::wptree video_channel::delay_info() const { return impl_->delay_info(); }\r
}}
\ No newline at end of file