|  | /* | 
|  | *  Copyright (c) 2019 The WebRTC project authors. All Rights Reserved. | 
|  | * | 
|  | *  Use of this source code is governed by a BSD-style license | 
|  | *  that can be found in the LICENSE file in the root of the source | 
|  | *  tree. An additional intellectual property rights grant can be found | 
|  | *  in the file PATENTS.  All contributing project authors may | 
|  | *  be found in the AUTHORS file in the root of the source tree. | 
|  | */ | 
|  |  | 
|  | #include "api/audio_codecs/opus/audio_encoder_multi_channel_opus_config.h" | 
|  |  | 
|  | #include <cstddef> | 
|  | #include <vector> | 
|  |  | 
|  | #include "api/audio_codecs/audio_encoder.h" | 
|  |  | 
|  | namespace webrtc { | 
|  |  | 
|  | namespace { | 
|  | constexpr int kDefaultComplexity = 9; | 
|  | }  // namespace | 
|  |  | 
|  | AudioEncoderMultiChannelOpusConfig::AudioEncoderMultiChannelOpusConfig() | 
|  | : frame_size_ms(kDefaultFrameSizeMs), | 
|  | num_channels(1), | 
|  | application(ApplicationMode::kVoip), | 
|  | bitrate_bps(32000), | 
|  | fec_enabled(false), | 
|  | cbr_enabled(false), | 
|  | dtx_enabled(false), | 
|  | max_playback_rate_hz(48000), | 
|  | complexity(kDefaultComplexity), | 
|  | num_streams(-1), | 
|  | coupled_streams(-1) {} | 
|  | AudioEncoderMultiChannelOpusConfig::AudioEncoderMultiChannelOpusConfig( | 
|  | const AudioEncoderMultiChannelOpusConfig&) = default; | 
|  | AudioEncoderMultiChannelOpusConfig::~AudioEncoderMultiChannelOpusConfig() = | 
|  | default; | 
|  | AudioEncoderMultiChannelOpusConfig& | 
|  | AudioEncoderMultiChannelOpusConfig::operator=( | 
|  | const AudioEncoderMultiChannelOpusConfig&) = default; | 
|  |  | 
|  | bool AudioEncoderMultiChannelOpusConfig::IsOk() const { | 
|  | if (frame_size_ms <= 0 || frame_size_ms % 10 != 0) | 
|  | return false; | 
|  | if (num_channels > AudioEncoder::kMaxNumberOfChannels) { | 
|  | return false; | 
|  | } | 
|  | if (bitrate_bps < kMinBitrateBps || bitrate_bps > kMaxBitrateBps) | 
|  | return false; | 
|  | if (complexity < 0 || complexity > 10) | 
|  | return false; | 
|  |  | 
|  | // Check the lengths: | 
|  | if (num_streams < 0 || coupled_streams < 0) { | 
|  | return false; | 
|  | } | 
|  | if (num_streams < coupled_streams) { | 
|  | return false; | 
|  | } | 
|  | if (channel_mapping.size() != static_cast<size_t>(num_channels)) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // Every mono stream codes one channel, every coupled stream codes two. This | 
|  | // is the total coded channel count: | 
|  | const int max_coded_channel = num_streams + coupled_streams; | 
|  | for (const auto& x : channel_mapping) { | 
|  | // Coded channels >= max_coded_channel don't exist. Except for 255, which | 
|  | // tells Opus to ignore input channel x. | 
|  | if (x >= max_coded_channel && x != 255) { | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Inverse mapping. | 
|  | constexpr int kNotSet = -1; | 
|  | std::vector<int> coded_channels_to_input_channels(max_coded_channel, kNotSet); | 
|  | for (size_t i = 0; i < num_channels; ++i) { | 
|  | if (channel_mapping[i] == 255) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | // If it's not ignored, put it in the inverted mapping. But first check if | 
|  | // we've told Opus to use another input channel for this coded channel: | 
|  | const int coded_channel = channel_mapping[i]; | 
|  | if (coded_channels_to_input_channels[coded_channel] != kNotSet) { | 
|  | // Coded channel `coded_channel` comes from both input channels | 
|  | // `coded_channels_to_input_channels[coded_channel]` and `i`. | 
|  | return false; | 
|  | } | 
|  |  | 
|  | coded_channels_to_input_channels[coded_channel] = i; | 
|  | } | 
|  |  | 
|  | // Check that we specified what input the encoder should use to produce | 
|  | // every coded channel. | 
|  | for (int i = 0; i < max_coded_channel; ++i) { | 
|  | if (coded_channels_to_input_channels[i] == kNotSet) { | 
|  | // Coded channel `i` has unspecified input channel. | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (num_channels > 255 || max_coded_channel >= 255) { | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | }  // namespace webrtc |