mediapipe/mediapipe2/framework/tool/switch_demux_calculator.cc
2021-06-10 23:01:19 +00:00

165 lines
6.4 KiB
C++

// Copyright 2019 The MediaPipe Authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#include <algorithm>
#include <memory>
#include <set>
#include <string>
#include "absl/strings/str_cat.h"
#include "mediapipe/framework/calculator_framework.h"
#include "mediapipe/framework/collection_item_id.h"
#include "mediapipe/framework/port/integral_types.h"
#include "mediapipe/framework/port/logging.h"
#include "mediapipe/framework/port/ret_check.h"
#include "mediapipe/framework/port/status.h"
#include "mediapipe/framework/port/status_macros.h"
#include "mediapipe/framework/tool/container_util.h"
namespace mediapipe {
// A calculator to redirect a set of input streams to one of several output
// channels, each consisting of corresponding output streams. Each channel
// is distinguished by a tag-prefix such as "C1__". For example:
//
// node {
// calculator: "SwitchDemuxCalculator"
// input_stream: "ENABLE:enable"
// input_stream: "FUNC_INPUT:foo"
// input_stream: "FUNC_INPUT:bar"
// output_stream: "C0__FUNC_INPUT:foo_0"
// output_stream: "C0__FUNC_INPUT:bar_0"
// output_stream: "C1__FUNC_INPUT:foo_1"
// output_stream: "C1__FUNC_INPUT:bar_1"
// }
//
// Input stream "ENABLE" specifies routing of packets to either channel 0
// or channel 1, given "ENABLE:false" or "ENABLE:true" respectively.
// Input-side-packet "ENABLE" and input-stream "SELECT" can also be used
// similarly to specify the active channel.
//
// SwitchDemuxCalculator is used by SwitchContainer to enable one of several
// contained subgraph or calculator nodes.
//
class SwitchDemuxCalculator : public CalculatorBase {
static constexpr char kSelectTag[] = "SELECT";
static constexpr char kEnableTag[] = "ENABLE";
public:
static absl::Status GetContract(CalculatorContract* cc);
absl::Status Open(CalculatorContext* cc) override;
absl::Status Process(CalculatorContext* cc) override;
private:
int channel_index_;
std::set<std::string> channel_tags_;
};
REGISTER_CALCULATOR(SwitchDemuxCalculator);
absl::Status SwitchDemuxCalculator::GetContract(CalculatorContract* cc) {
// Allow any one of kSelectTag, kEnableTag.
cc->Inputs().Tag(kSelectTag).Set<int>().Optional();
cc->Inputs().Tag(kEnableTag).Set<bool>().Optional();
// Allow any one of kSelectTag, kEnableTag.
cc->InputSidePackets().Tag(kSelectTag).Set<int>().Optional();
cc->InputSidePackets().Tag(kEnableTag).Set<bool>().Optional();
// Set the types for all output channels to corresponding input types.
std::set<std::string> channel_tags = ChannelTags(cc->Outputs().TagMap());
int channel_count = ChannelCount(cc->Outputs().TagMap());
for (const std::string& tag : channel_tags) {
for (int index = 0; index < cc->Inputs().NumEntries(tag); ++index) {
auto input_id = cc->Inputs().GetId(tag, index);
if (input_id.IsValid()) {
cc->Inputs().Get(tag, index).SetAny();
for (int channel = 0; channel < channel_count; ++channel) {
auto output_id =
cc->Outputs().GetId(tool::ChannelTag(tag, channel), index);
if (output_id.IsValid()) {
cc->Outputs().Get(output_id).SetSameAs(&cc->Inputs().Get(input_id));
}
}
}
}
}
channel_tags = ChannelTags(cc->OutputSidePackets().TagMap());
channel_count = ChannelCount(cc->OutputSidePackets().TagMap());
for (const std::string& tag : channel_tags) {
int num_entries = cc->InputSidePackets().NumEntries(tag);
for (int index = 0; index < num_entries; ++index) {
auto input_id = cc->InputSidePackets().GetId(tag, index);
if (input_id.IsValid()) {
cc->InputSidePackets().Get(tag, index).SetAny();
for (int channel = 0; channel < channel_count; ++channel) {
auto output_id = cc->OutputSidePackets().GetId(
tool::ChannelTag(tag, channel), index);
if (output_id.IsValid()) {
cc->OutputSidePackets().Get(output_id).SetSameAs(
&cc->InputSidePackets().Get(input_id));
}
}
}
}
}
cc->SetInputStreamHandler("ImmediateInputStreamHandler");
cc->SetProcessTimestampBounds(true);
return absl::OkStatus();
}
absl::Status SwitchDemuxCalculator::Open(CalculatorContext* cc) {
channel_index_ = tool::GetChannelIndex(*cc, channel_index_);
channel_tags_ = ChannelTags(cc->Outputs().TagMap());
// Relay side packets to all channels.
// Note: This is necessary because Calculator::Open only proceeds when every
// anticipated side-packet arrives.
int channel_count = tool::ChannelCount(cc->OutputSidePackets().TagMap());
for (const std::string& tag : ChannelTags(cc->OutputSidePackets().TagMap())) {
int num_entries = cc->InputSidePackets().NumEntries(tag);
for (int index = 0; index < num_entries; ++index) {
Packet input = cc->InputSidePackets().Get(tag, index);
for (int channel = 0; channel < channel_count; ++channel) {
std::string output_tag = tool::ChannelTag(tag, channel);
auto output_id = cc->OutputSidePackets().GetId(output_tag, index);
if (output_id.IsValid()) {
cc->OutputSidePackets().Get(output_tag, index).Set(input);
}
}
}
}
return absl::OkStatus();
}
absl::Status SwitchDemuxCalculator::Process(CalculatorContext* cc) {
// Update the input channel index if specified.
channel_index_ = tool::GetChannelIndex(*cc, channel_index_);
// Relay packets and timestamps only to channel_index_.
for (const std::string& tag : channel_tags_) {
for (int index = 0; index < cc->Inputs().NumEntries(tag); ++index) {
auto& input = cc->Inputs().Get(tag, index);
std::string output_tag = tool::ChannelTag(tag, channel_index_);
auto output_id = cc->Outputs().GetId(output_tag, index);
if (output_id.IsValid()) {
auto& output = cc->Outputs().Get(output_tag, index);
tool::Relay(input, &output);
}
}
}
return absl::OkStatus();
}
} // namespace mediapipe