2 * Copyright 2013 NewTek
4 * This file is part of CasparCG (www.casparcg.com).
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.
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.
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/>.
19 * Author: Robert Nagy, ronag@live.com
22 #include "../StdAfx.h"
24 #include "newtek_ivga_consumer.h"
26 #include <core/consumer/frame_consumer.h>
27 #include <core/parameters/parameters.h>
28 #include <core/video_format.h>
29 #include <core/mixer/read_frame.h>
31 #include <common/utility/assert.h>
32 #include <common/concurrency/executor.h>
33 #include <common/diagnostics/graph.h>
35 #include <boost/algorithm/string.hpp>
36 #include <boost/timer.hpp>
38 #include <tbb/atomic.h>
40 #include "../util/air_send.h"
42 namespace caspar { namespace newtek {
44 struct newtek_ivga_consumer : public core::frame_consumer
46 std::shared_ptr<void> air_send_;
47 core::video_format_desc format_desc_;
48 core::channel_layout channel_layout_;
50 tbb::atomic<bool> connected_;
52 safe_ptr<diagnostics::graph> graph_;
53 boost::timer tick_timer_;
54 boost::timer frame_timer_;
58 newtek_ivga_consumer(core::channel_layout channel_layout)
60 , channel_layout_(channel_layout)
62 if (!airsend::is_available())
63 BOOST_THROW_EXCEPTION(caspar_exception() << msg_info(narrow(airsend::dll_name()) + " not available"));
67 graph_->set_text(print());
68 graph_->set_color("frame-time", diagnostics::color(0.5f, 1.0f, 0.2f));
69 graph_->set_color("tick-time", diagnostics::color(0.0f, 0.6f, 0.9f));
70 diagnostics::register_graph(graph_);
73 ~newtek_ivga_consumer()
79 virtual void initialize(const core::video_format_desc& format_desc, int channel_index) override
85 format_desc.time_scale,
87 format_desc.field_mode == core::field_mode::progressive,
88 static_cast<float>(format_desc.square_width) / static_cast<float>(format_desc.square_height),
90 channel_layout_.num_channels,
91 format_desc.audio_sample_rate),
94 CASPAR_VERIFY(air_send_);
96 format_desc_ = format_desc;
98 CASPAR_LOG(info) << print() << L" Successfully Initialized.";
101 virtual boost::unique_future<bool> send(const safe_ptr<core::read_frame>& frame) override
103 CASPAR_VERIFY(format_desc_.height * format_desc_.width * 4 == static_cast<unsigned>(frame->image_data().size()));
105 return executor_.begin_invoke([=]() -> bool
107 graph_->set_value("tick-time", tick_timer_.elapsed() * format_desc_.fps * 0.5);
108 tick_timer_.restart();
109 frame_timer_.restart();
113 std::vector<int16_t, tbb::cache_aligned_allocator<int16_t>> audio_buffer;
115 if (core::needs_rearranging(
116 frame->multichannel_view(),
118 channel_layout_.num_channels))
120 core::audio_buffer downmixed;
123 frame->multichannel_view().num_samples()
124 * channel_layout_.num_channels,
127 auto dest_view = core::make_multichannel_view<int32_t>(
128 downmixed.begin(), downmixed.end(), channel_layout_);
130 core::rearrange_or_rearrange_and_mix(
131 frame->multichannel_view(),
133 core::default_mix_config_repository());
135 audio_buffer = core::audio_32_to_16(downmixed);
139 audio_buffer = core::audio_32_to_16(frame->audio_data());
142 airsend::add_audio(air_send_.get(), audio_buffer.data(), audio_buffer.size() / channel_layout_.num_channels);
146 connected_ = airsend::add_frame_bgra(air_send_.get(), frame->image_data().begin());
148 graph_->set_text(print());
149 graph_->set_value("frame-time", frame_timer_.elapsed() * format_desc_.fps * 0.5);
155 virtual std::wstring print() const override
158 L"newtek-ivga[connected]" : L"newtek-ivga[not connected]";
161 virtual boost::property_tree::wptree info() const override
163 boost::property_tree::wptree info;
164 info.add(L"type", L"newtek-ivga-consumer");
165 info.add(L"connected", connected_ ? L"true" : L"false");
169 virtual size_t buffer_depth() const override
174 virtual int index() const override
179 virtual int64_t presentation_frame_age_millis() const override
184 virtual bool has_synchronization_clock() const override
190 safe_ptr<core::frame_consumer> create_ivga_consumer(const core::parameters& params)
192 if(params.size() < 1 || params[0] != L"NEWTEK_IVGA")
193 return core::frame_consumer::empty();
195 const auto channel_layout = core::default_channel_layout_repository()
197 params.get(L"CHANNEL_LAYOUT", L"STEREO"));
199 return make_safe<newtek_ivga_consumer>(channel_layout);
202 safe_ptr<core::frame_consumer> create_ivga_consumer(const boost::property_tree::wptree& ptree)
204 const auto channel_layout =
205 core::default_channel_layout_repository()
207 boost::to_upper_copy(ptree.get(L"channel-layout", L"STEREO")));
209 return make_safe<newtek_ivga_consumer>(channel_layout);