aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license |
| 5 | * that can be found in the LICENSE file in the root of the source |
| 6 | * tree. An additional intellectual property rights grant can be found |
| 7 | * in the file PATENTS. All contributing project authors may |
| 8 | * be found in the AUTHORS file in the root of the source tree. |
| 9 | */ |
| 10 | |
aleloi | 5d167d6 | 2016-08-24 02:20:54 -0700 | [diff] [blame] | 11 | #include "webrtc/modules/audio_mixer/audio_mixer_impl.h" |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 12 | |
| 13 | #include <algorithm> |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 14 | #include <functional> |
aleloi | 623427c | 2016-12-08 02:37:58 -0800 | [diff] [blame^] | 15 | #include <iterator> |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 16 | #include <utility> |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 17 | |
aleloi | 6321b49 | 2016-12-05 01:46:09 -0800 | [diff] [blame] | 18 | #include "webrtc/audio/utility/audio_frame_operations.h" |
aleloi | 3654251 | 2016-10-07 05:28:32 -0700 | [diff] [blame] | 19 | #include "webrtc/base/logging.h" |
aleloi | 5bcc00e | 2016-08-15 03:01:31 -0700 | [diff] [blame] | 20 | #include "webrtc/modules/audio_mixer/audio_frame_manipulator.h" |
aleloi | 623427c | 2016-12-08 02:37:58 -0800 | [diff] [blame^] | 21 | #include "webrtc/modules/audio_mixer/default_output_rate_calculator.h" |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 22 | |
| 23 | namespace webrtc { |
| 24 | namespace { |
| 25 | |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 26 | struct SourceFrame { |
| 27 | SourceFrame(AudioMixerImpl::SourceStatus* source_status, |
aleloi | 3654251 | 2016-10-07 05:28:32 -0700 | [diff] [blame] | 28 | AudioFrame* audio_frame, |
| 29 | bool muted) |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 30 | : source_status(source_status), audio_frame(audio_frame), muted(muted) { |
| 31 | RTC_DCHECK(source_status); |
| 32 | RTC_DCHECK(audio_frame); |
| 33 | if (!muted) { |
| 34 | energy = AudioMixerCalculateEnergy(*audio_frame); |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 35 | } |
| 36 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 37 | |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 38 | SourceFrame(AudioMixerImpl::SourceStatus* source_status, |
aleloi | 3654251 | 2016-10-07 05:28:32 -0700 | [diff] [blame] | 39 | AudioFrame* audio_frame, |
| 40 | bool muted, |
aleloi | 652ac89 | 2016-09-07 07:42:14 -0700 | [diff] [blame] | 41 | uint32_t energy) |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 42 | : source_status(source_status), |
| 43 | audio_frame(audio_frame), |
| 44 | muted(muted), |
| 45 | energy(energy) { |
| 46 | RTC_DCHECK(source_status); |
| 47 | RTC_DCHECK(audio_frame); |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 48 | } |
| 49 | |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 50 | AudioMixerImpl::SourceStatus* source_status = nullptr; |
| 51 | AudioFrame* audio_frame = nullptr; |
| 52 | bool muted = true; |
| 53 | uint32_t energy = 0; |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 54 | }; |
terelius | ea4c141 | 2016-07-29 01:36:14 -0700 | [diff] [blame] | 55 | |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 56 | // ShouldMixBefore(a, b) is used to select mixer sources. |
| 57 | bool ShouldMixBefore(const SourceFrame& a, const SourceFrame& b) { |
| 58 | if (a.muted != b.muted) { |
| 59 | return b.muted; |
| 60 | } |
aleloi | 4496809 | 2016-08-08 10:18:58 -0700 | [diff] [blame] | 61 | |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 62 | const auto a_activity = a.audio_frame->vad_activity_; |
| 63 | const auto b_activity = b.audio_frame->vad_activity_; |
| 64 | |
| 65 | if (a_activity != b_activity) { |
| 66 | return a_activity == AudioFrame::kVadActive; |
| 67 | } |
| 68 | |
| 69 | return a.energy > b.energy; |
| 70 | } |
| 71 | |
| 72 | void RampAndUpdateGain( |
| 73 | const std::vector<SourceFrame>& mixed_sources_and_frames) { |
aleloi | 652ac89 | 2016-09-07 07:42:14 -0700 | [diff] [blame] | 74 | for (const auto& source_frame : mixed_sources_and_frames) { |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 75 | float target_gain = source_frame.source_status->is_mixed ? 1.0f : 0.0f; |
| 76 | Ramp(source_frame.source_status->gain, target_gain, |
| 77 | source_frame.audio_frame); |
| 78 | source_frame.source_status->gain = target_gain; |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 79 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 80 | } |
| 81 | |
aleloi | dc7669a | 2016-10-04 04:06:20 -0700 | [diff] [blame] | 82 | // Mix the AudioFrames stored in audioFrameList into mixed_audio. |
| 83 | int32_t MixFromList(AudioFrame* mixed_audio, |
| 84 | const AudioFrameList& audio_frame_list, |
aleloi | dc7669a | 2016-10-04 04:06:20 -0700 | [diff] [blame] | 85 | bool use_limiter) { |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 86 | if (audio_frame_list.empty()) { |
aleloi | dc7669a | 2016-10-04 04:06:20 -0700 | [diff] [blame] | 87 | return 0; |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 88 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 89 | |
aleloi | dc7669a | 2016-10-04 04:06:20 -0700 | [diff] [blame] | 90 | if (audio_frame_list.size() == 1) { |
| 91 | mixed_audio->timestamp_ = audio_frame_list.front()->timestamp_; |
| 92 | mixed_audio->elapsed_time_ms_ = audio_frame_list.front()->elapsed_time_ms_; |
| 93 | } else { |
| 94 | // TODO(wu): Issue 3390. |
| 95 | // Audio frame timestamp is only supported in one channel case. |
| 96 | mixed_audio->timestamp_ = 0; |
| 97 | mixed_audio->elapsed_time_ms_ = -1; |
| 98 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 99 | |
aleloi | dc7669a | 2016-10-04 04:06:20 -0700 | [diff] [blame] | 100 | for (const auto& frame : audio_frame_list) { |
| 101 | RTC_DCHECK_EQ(mixed_audio->sample_rate_hz_, frame->sample_rate_hz_); |
| 102 | RTC_DCHECK_EQ( |
| 103 | frame->samples_per_channel_, |
| 104 | static_cast<size_t>((mixed_audio->sample_rate_hz_ * |
| 105 | webrtc::AudioMixerImpl::kFrameDurationInMs) / |
| 106 | 1000)); |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 107 | |
aleloi | dc7669a | 2016-10-04 04:06:20 -0700 | [diff] [blame] | 108 | // Mix |f.frame| into |mixed_audio|, with saturation protection. |
| 109 | // These effect is applied to |f.frame| itself prior to mixing. |
| 110 | if (use_limiter) { |
aleloi | 6321b49 | 2016-12-05 01:46:09 -0800 | [diff] [blame] | 111 | // This is to avoid saturation in the mixing. It is only |
| 112 | // meaningful if the limiter will be used. |
| 113 | AudioFrameOperations::ApplyHalfGain(frame); |
aleloi | dc7669a | 2016-10-04 04:06:20 -0700 | [diff] [blame] | 114 | } |
| 115 | RTC_DCHECK_EQ(frame->num_channels_, mixed_audio->num_channels_); |
aleloi | 6321b49 | 2016-12-05 01:46:09 -0800 | [diff] [blame] | 116 | AudioFrameOperations::Add(*frame, mixed_audio); |
aleloi | dc7669a | 2016-10-04 04:06:20 -0700 | [diff] [blame] | 117 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 118 | return 0; |
| 119 | } |
| 120 | |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 121 | AudioMixerImpl::SourceStatusList::const_iterator FindSourceInList( |
aleloi | e891415 | 2016-10-11 06:18:31 -0700 | [diff] [blame] | 122 | AudioMixerImpl::Source const* audio_source, |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 123 | AudioMixerImpl::SourceStatusList const* audio_source_list) { |
aleloi | 6c27849 | 2016-10-20 14:24:39 -0700 | [diff] [blame] | 124 | return std::find_if( |
| 125 | audio_source_list->begin(), audio_source_list->end(), |
| 126 | [audio_source](const std::unique_ptr<AudioMixerImpl::SourceStatus>& p) { |
| 127 | return p->audio_source == audio_source; |
| 128 | }); |
aleloi | 3654251 | 2016-10-07 05:28:32 -0700 | [diff] [blame] | 129 | } |
| 130 | |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 131 | // TODO(aleloi): remove non-const version when WEBRTC only supports modern STL. |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 132 | AudioMixerImpl::SourceStatusList::iterator FindSourceInList( |
aleloi | e891415 | 2016-10-11 06:18:31 -0700 | [diff] [blame] | 133 | AudioMixerImpl::Source const* audio_source, |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 134 | AudioMixerImpl::SourceStatusList* audio_source_list) { |
aleloi | 6c27849 | 2016-10-20 14:24:39 -0700 | [diff] [blame] | 135 | return std::find_if( |
| 136 | audio_source_list->begin(), audio_source_list->end(), |
| 137 | [audio_source](const std::unique_ptr<AudioMixerImpl::SourceStatus>& p) { |
| 138 | return p->audio_source == audio_source; |
| 139 | }); |
aleloi | 3654251 | 2016-10-07 05:28:32 -0700 | [diff] [blame] | 140 | } |
| 141 | |
aleloi | 623427c | 2016-12-08 02:37:58 -0800 | [diff] [blame^] | 142 | std::unique_ptr<AudioProcessing> CreateLimiter() { |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 143 | Config config; |
| 144 | config.Set<ExperimentalAgc>(new ExperimentalAgc(false)); |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 145 | std::unique_ptr<AudioProcessing> limiter(AudioProcessing::Create(config)); |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 146 | if (!limiter.get()) { |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 147 | return nullptr; |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 148 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 149 | |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 150 | if (limiter->gain_control()->set_mode(GainControl::kFixedDigital) != |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 151 | limiter->kNoError) { |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 152 | return nullptr; |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 153 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 154 | |
| 155 | // We smoothly limit the mixed frame to -7 dbFS. -6 would correspond to the |
| 156 | // divide-by-2 but -7 is used instead to give a bit of headroom since the |
| 157 | // AGC is not a hard limiter. |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 158 | if (limiter->gain_control()->set_target_level_dbfs(7) != limiter->kNoError) { |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 159 | return nullptr; |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 160 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 161 | |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 162 | if (limiter->gain_control()->set_compression_gain_db(0) != |
| 163 | limiter->kNoError) { |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 164 | return nullptr; |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 165 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 166 | |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 167 | if (limiter->gain_control()->enable_limiter(true) != limiter->kNoError) { |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 168 | return nullptr; |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 169 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 170 | |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 171 | if (limiter->gain_control()->Enable(true) != limiter->kNoError) { |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 172 | return nullptr; |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 173 | } |
aleloi | 623427c | 2016-12-08 02:37:58 -0800 | [diff] [blame^] | 174 | return limiter; |
| 175 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 176 | |
aleloi | 623427c | 2016-12-08 02:37:58 -0800 | [diff] [blame^] | 177 | } // namespace |
| 178 | |
| 179 | AudioMixerImpl::AudioMixerImpl( |
| 180 | std::unique_ptr<AudioProcessing> limiter, |
| 181 | std::unique_ptr<OutputRateCalculator> output_rate_calculator) |
| 182 | : output_rate_calculator_(std::move(output_rate_calculator)), |
| 183 | output_frequency_(0), |
| 184 | sample_size_(0), |
| 185 | audio_source_list_(), |
| 186 | use_limiter_(true), |
| 187 | time_stamp_(0), |
| 188 | limiter_(std::move(limiter)) {} |
| 189 | |
| 190 | AudioMixerImpl::~AudioMixerImpl() {} |
| 191 | |
| 192 | rtc::scoped_refptr<AudioMixerImpl> AudioMixerImpl::Create() { |
| 193 | return CreateWithOutputRateCalculator( |
| 194 | std::unique_ptr<DefaultOutputRateCalculator>( |
| 195 | new DefaultOutputRateCalculator())); |
| 196 | } |
| 197 | |
| 198 | rtc::scoped_refptr<AudioMixerImpl> |
| 199 | AudioMixerImpl::CreateWithOutputRateCalculator( |
| 200 | std::unique_ptr<OutputRateCalculator> output_rate_calculator) { |
aleloi | 116ec6d | 2016-10-12 06:07:07 -0700 | [diff] [blame] | 201 | return rtc::scoped_refptr<AudioMixerImpl>( |
aleloi | 623427c | 2016-12-08 02:37:58 -0800 | [diff] [blame^] | 202 | new rtc::RefCountedObject<AudioMixerImpl>( |
| 203 | CreateLimiter(), std::move(output_rate_calculator))); |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 204 | } |
| 205 | |
aleloi | 9561183 | 2016-11-08 06:39:50 -0800 | [diff] [blame] | 206 | void AudioMixerImpl::Mix(size_t number_of_channels, |
aleloi | 5d167d6 | 2016-08-24 02:20:54 -0700 | [diff] [blame] | 207 | AudioFrame* audio_frame_for_mixing) { |
aleloi | 4496809 | 2016-08-08 10:18:58 -0700 | [diff] [blame] | 208 | RTC_DCHECK(number_of_channels == 1 || number_of_channels == 2); |
aleloi | 920d30b | 2016-10-20 14:23:24 -0700 | [diff] [blame] | 209 | RTC_DCHECK_RUNS_SERIALIZED(&race_checker_); |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 210 | |
aleloi | 623427c | 2016-12-08 02:37:58 -0800 | [diff] [blame^] | 211 | CalculateOutputFrequency(); |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 212 | |
aleloi | 652ac89 | 2016-09-07 07:42:14 -0700 | [diff] [blame] | 213 | AudioFrameList mix_list; |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 214 | { |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 215 | rtc::CritScope lock(&crit_); |
aleloi | 116ec6d | 2016-10-12 06:07:07 -0700 | [diff] [blame] | 216 | mix_list = GetAudioFromSources(); |
aleloi | 1655e45 | 2016-10-24 06:56:56 -0700 | [diff] [blame] | 217 | |
| 218 | for (const auto& frame : mix_list) { |
| 219 | RemixFrame(number_of_channels, frame); |
| 220 | } |
| 221 | |
| 222 | audio_frame_for_mixing->UpdateFrame( |
| 223 | -1, time_stamp_, NULL, 0, OutputFrequency(), AudioFrame::kNormalSpeech, |
| 224 | AudioFrame::kVadPassive, number_of_channels); |
| 225 | |
| 226 | time_stamp_ += static_cast<uint32_t>(sample_size_); |
| 227 | |
| 228 | use_limiter_ = mix_list.size() > 1; |
| 229 | |
| 230 | // We only use the limiter if we're actually mixing multiple streams. |
| 231 | MixFromList(audio_frame_for_mixing, mix_list, use_limiter_); |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 232 | } |
| 233 | |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 234 | if (audio_frame_for_mixing->samples_per_channel_ == 0) { |
| 235 | // Nothing was mixed, set the audio samples to silence. |
| 236 | audio_frame_for_mixing->samples_per_channel_ = sample_size_; |
aleloi | 6321b49 | 2016-12-05 01:46:09 -0800 | [diff] [blame] | 237 | AudioFrameOperations::Mute(audio_frame_for_mixing); |
aleloi | 311525e | 2016-09-07 06:13:12 -0700 | [diff] [blame] | 238 | } else { |
| 239 | // Only call the limiter if we have something to mix. |
| 240 | LimitMixedAudio(audio_frame_for_mixing); |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 241 | } |
aleloi | 616df1e | 2016-08-24 01:17:12 -0700 | [diff] [blame] | 242 | |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 243 | return; |
| 244 | } |
| 245 | |
aleloi | 623427c | 2016-12-08 02:37:58 -0800 | [diff] [blame^] | 246 | void AudioMixerImpl::CalculateOutputFrequency() { |
aleloi | 920d30b | 2016-10-20 14:23:24 -0700 | [diff] [blame] | 247 | RTC_DCHECK_RUNS_SERIALIZED(&race_checker_); |
aleloi | 623427c | 2016-12-08 02:37:58 -0800 | [diff] [blame^] | 248 | rtc::CritScope lock(&crit_); |
| 249 | |
| 250 | std::vector<int> preferred_rates; |
| 251 | std::transform(audio_source_list_.begin(), audio_source_list_.end(), |
| 252 | std::back_inserter(preferred_rates), |
| 253 | [&](std::unique_ptr<SourceStatus>& a) { |
| 254 | return a->audio_source->PreferredSampleRate(); |
| 255 | }); |
| 256 | |
| 257 | output_frequency_ = |
| 258 | output_rate_calculator_->CalculateOutputRate(preferred_rates); |
aleloi | dc7669a | 2016-10-04 04:06:20 -0700 | [diff] [blame] | 259 | sample_size_ = (output_frequency_ * kFrameDurationInMs) / 1000; |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 260 | } |
| 261 | |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 262 | int AudioMixerImpl::OutputFrequency() const { |
aleloi | 920d30b | 2016-10-20 14:23:24 -0700 | [diff] [blame] | 263 | RTC_DCHECK_RUNS_SERIALIZED(&race_checker_); |
aleloi | 6382a19 | 2016-08-08 10:25:04 -0700 | [diff] [blame] | 264 | return output_frequency_; |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 265 | } |
| 266 | |
aleloi | 116ec6d | 2016-10-12 06:07:07 -0700 | [diff] [blame] | 267 | bool AudioMixerImpl::AddSource(Source* audio_source) { |
| 268 | RTC_DCHECK(audio_source); |
| 269 | rtc::CritScope lock(&crit_); |
| 270 | RTC_DCHECK(FindSourceInList(audio_source, &audio_source_list_) == |
| 271 | audio_source_list_.end()) |
| 272 | << "Source already added to mixer"; |
aleloi | 6c27849 | 2016-10-20 14:24:39 -0700 | [diff] [blame] | 273 | audio_source_list_.emplace_back(new SourceStatus(audio_source, false, 0)); |
aleloi | 116ec6d | 2016-10-12 06:07:07 -0700 | [diff] [blame] | 274 | return true; |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 275 | } |
| 276 | |
aleloi | 76b3049 | 2016-11-18 02:03:04 -0800 | [diff] [blame] | 277 | void AudioMixerImpl::RemoveSource(Source* audio_source) { |
aleloi | 116ec6d | 2016-10-12 06:07:07 -0700 | [diff] [blame] | 278 | RTC_DCHECK(audio_source); |
| 279 | rtc::CritScope lock(&crit_); |
| 280 | const auto iter = FindSourceInList(audio_source, &audio_source_list_); |
| 281 | RTC_DCHECK(iter != audio_source_list_.end()) << "Source not present in mixer"; |
| 282 | audio_source_list_.erase(iter); |
aleloi | 116ec6d | 2016-10-12 06:07:07 -0700 | [diff] [blame] | 283 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 284 | |
aleloi | 116ec6d | 2016-10-12 06:07:07 -0700 | [diff] [blame] | 285 | AudioFrameList AudioMixerImpl::GetAudioFromSources() { |
aleloi | 920d30b | 2016-10-20 14:23:24 -0700 | [diff] [blame] | 286 | RTC_DCHECK_RUNS_SERIALIZED(&race_checker_); |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 287 | AudioFrameList result; |
aleloi | a4c2106 | 2016-09-08 01:25:46 -0700 | [diff] [blame] | 288 | std::vector<SourceFrame> audio_source_mixing_data_list; |
aleloi | 652ac89 | 2016-09-07 07:42:14 -0700 | [diff] [blame] | 289 | std::vector<SourceFrame> ramp_list; |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 290 | |
aleloi | 6c27849 | 2016-10-20 14:24:39 -0700 | [diff] [blame] | 291 | // Get audio from the audio sources and put it in the SourceFrame vector. |
aleloi | 3654251 | 2016-10-07 05:28:32 -0700 | [diff] [blame] | 292 | for (auto& source_and_status : audio_source_list_) { |
aleloi | 6c27849 | 2016-10-20 14:24:39 -0700 | [diff] [blame] | 293 | const auto audio_frame_info = |
| 294 | source_and_status->audio_source->GetAudioFrameWithInfo( |
| 295 | OutputFrequency(), &source_and_status->audio_frame); |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 296 | |
aleloi | e891415 | 2016-10-11 06:18:31 -0700 | [diff] [blame] | 297 | if (audio_frame_info == Source::AudioFrameInfo::kError) { |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 298 | LOG_F(LS_WARNING) << "failed to GetAudioFrameWithInfo() from source"; |
terelius | ea4c141 | 2016-07-29 01:36:14 -0700 | [diff] [blame] | 299 | continue; |
| 300 | } |
aleloi | a4c2106 | 2016-09-08 01:25:46 -0700 | [diff] [blame] | 301 | audio_source_mixing_data_list.emplace_back( |
aleloi | 6c27849 | 2016-10-20 14:24:39 -0700 | [diff] [blame] | 302 | source_and_status.get(), &source_and_status->audio_frame, |
aleloi | e891415 | 2016-10-11 06:18:31 -0700 | [diff] [blame] | 303 | audio_frame_info == Source::AudioFrameInfo::kMuted); |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 304 | } |
| 305 | |
| 306 | // Sort frames by sorting function. |
aleloi | a4c2106 | 2016-09-08 01:25:46 -0700 | [diff] [blame] | 307 | std::sort(audio_source_mixing_data_list.begin(), |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 308 | audio_source_mixing_data_list.end(), ShouldMixBefore); |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 309 | |
aleloi | a4c2106 | 2016-09-08 01:25:46 -0700 | [diff] [blame] | 310 | int max_audio_frame_counter = kMaximumAmountOfMixedAudioSources; |
| 311 | |
| 312 | // Go through list in order and put unmuted frames in result list. |
aleloi | 3654251 | 2016-10-07 05:28:32 -0700 | [diff] [blame] | 313 | for (const auto& p : audio_source_mixing_data_list) { |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 314 | // Filter muted. |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 315 | if (p.muted) { |
| 316 | p.source_status->is_mixed = false; |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 317 | continue; |
terelius | ea4c141 | 2016-07-29 01:36:14 -0700 | [diff] [blame] | 318 | } |
aleloi | 2942e24 | 2016-07-29 01:23:49 -0700 | [diff] [blame] | 319 | |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 320 | // Add frame to result vector for mixing. |
| 321 | bool is_mixed = false; |
aleloi | a4c2106 | 2016-09-08 01:25:46 -0700 | [diff] [blame] | 322 | if (max_audio_frame_counter > 0) { |
| 323 | --max_audio_frame_counter; |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 324 | result.push_back(p.audio_frame); |
| 325 | ramp_list.emplace_back(p.source_status, p.audio_frame, false, -1); |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 326 | is_mixed = true; |
terelius | ea4c141 | 2016-07-29 01:36:14 -0700 | [diff] [blame] | 327 | } |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 328 | p.source_status->is_mixed = is_mixed; |
terelius | ea4c141 | 2016-07-29 01:36:14 -0700 | [diff] [blame] | 329 | } |
aleloi | 4b8bfb8 | 2016-10-12 02:14:59 -0700 | [diff] [blame] | 330 | RampAndUpdateGain(ramp_list); |
aleloi | f388257 | 2016-07-29 02:12:41 -0700 | [diff] [blame] | 331 | return result; |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 332 | } |
| 333 | |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 334 | |
aleloi | a4c2106 | 2016-09-08 01:25:46 -0700 | [diff] [blame] | 335 | bool AudioMixerImpl::LimitMixedAudio(AudioFrame* mixed_audio) const { |
aleloi | 920d30b | 2016-10-20 14:23:24 -0700 | [diff] [blame] | 336 | RTC_DCHECK_RUNS_SERIALIZED(&race_checker_); |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 337 | if (!use_limiter_) { |
| 338 | return true; |
| 339 | } |
| 340 | |
| 341 | // Smoothly limit the mixed frame. |
aleloi | a4c2106 | 2016-09-08 01:25:46 -0700 | [diff] [blame] | 342 | const int error = limiter_->ProcessStream(mixed_audio); |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 343 | |
| 344 | // And now we can safely restore the level. This procedure results in |
| 345 | // some loss of resolution, deemed acceptable. |
| 346 | // |
| 347 | // It's possible to apply the gain in the AGC (with a target level of 0 dbFS |
| 348 | // and compression gain of 6 dB). However, in the transition frame when this |
aleloi | 09f4510 | 2016-07-28 03:52:15 -0700 | [diff] [blame] | 349 | // is enabled (moving from one to two audio sources) it has the potential to |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 350 | // create discontinuities in the mixed frame. |
| 351 | // |
| 352 | // Instead we double the frame (with addition since left-shifting a |
| 353 | // negative value is undefined). |
aleloi | 6321b49 | 2016-12-05 01:46:09 -0800 | [diff] [blame] | 354 | AudioFrameOperations::Add(*mixed_audio, mixed_audio); |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 355 | |
aleloi | 6382a19 | 2016-08-08 10:25:04 -0700 | [diff] [blame] | 356 | if (error != limiter_->kNoError) { |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 357 | LOG_F(LS_ERROR) << "Error from AudioProcessing: " << error; |
aleloi | 09f4510 | 2016-07-28 03:52:15 -0700 | [diff] [blame] | 358 | RTC_NOTREACHED(); |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 359 | return false; |
| 360 | } |
| 361 | return true; |
| 362 | } |
aleloi | 616df1e | 2016-08-24 01:17:12 -0700 | [diff] [blame] | 363 | |
aleloi | 3654251 | 2016-10-07 05:28:32 -0700 | [diff] [blame] | 364 | bool AudioMixerImpl::GetAudioSourceMixabilityStatusForTest( |
aleloi | e97974d | 2016-10-12 03:06:09 -0700 | [diff] [blame] | 365 | AudioMixerImpl::Source* audio_source) const { |
aleloi | 920d30b | 2016-10-20 14:23:24 -0700 | [diff] [blame] | 366 | RTC_DCHECK_RUNS_SERIALIZED(&race_checker_); |
aleloi | 3654251 | 2016-10-07 05:28:32 -0700 | [diff] [blame] | 367 | rtc::CritScope lock(&crit_); |
| 368 | |
aleloi | 6c27849 | 2016-10-20 14:24:39 -0700 | [diff] [blame] | 369 | const auto iter = FindSourceInList(audio_source, &audio_source_list_); |
| 370 | if (iter != audio_source_list_.end()) { |
| 371 | return (*iter)->is_mixed; |
aleloi | 3654251 | 2016-10-07 05:28:32 -0700 | [diff] [blame] | 372 | } |
| 373 | |
aleloi | 3654251 | 2016-10-07 05:28:32 -0700 | [diff] [blame] | 374 | LOG(LS_ERROR) << "Audio source unknown"; |
| 375 | return false; |
| 376 | } |
aleloi | 77ad394 | 2016-07-04 06:33:02 -0700 | [diff] [blame] | 377 | } // namespace webrtc |