blob: 287e7c3c485944fee3fe44a163fe2db93de287eb [file] [log] [blame]
peah69221db2017-01-27 03:28:19 -08001/*
2 * Copyright (c) 2017 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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "modules/audio_processing/aec3/render_delay_buffer.h"
peah69221db2017-01-27 03:28:19 -080012
Yves Gerey988cc082018-10-23 12:03:01 +020013#include <stdlib.h>
peah69221db2017-01-27 03:28:19 -080014#include <algorithm>
Yves Gerey988cc082018-10-23 12:03:01 +020015#include <memory>
Per Åhgrenb6b00dc2018-02-20 22:18:27 +010016#include <numeric>
peah69221db2017-01-27 03:28:19 -080017
Yves Gerey988cc082018-10-23 12:03:01 +020018#include "absl/types/optional.h"
19#include "api/array_view.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020020#include "modules/audio_processing/aec3/aec3_common.h"
Per Åhgren8ba58612017-12-01 23:01:44 +010021#include "modules/audio_processing/aec3/aec3_fft.h"
Per Åhgren38e2d952017-11-17 14:54:28 +010022#include "modules/audio_processing/aec3/decimator.h"
Per Åhgren8ba58612017-12-01 23:01:44 +010023#include "modules/audio_processing/aec3/fft_buffer.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020024#include "modules/audio_processing/aec3/fft_data.h"
Per Åhgren8ba58612017-12-01 23:01:44 +010025#include "modules/audio_processing/aec3/matrix_buffer.h"
Yves Gerey988cc082018-10-23 12:03:01 +020026#include "modules/audio_processing/aec3/vector_buffer.h"
27#include "modules/audio_processing/logging/apm_data_dumper.h"
Steve Anton10542f22019-01-11 09:11:00 -080028#include "rtc_base/atomic_ops.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020029#include "rtc_base/checks.h"
Steve Anton10542f22019-01-11 09:11:00 -080030#include "rtc_base/constructor_magic.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020031#include "rtc_base/logging.h"
Per Åhgrene05c43c2018-05-09 12:26:51 +020032#include "system_wrappers/include/field_trial.h"
peah69221db2017-01-27 03:28:19 -080033
34namespace webrtc {
35namespace {
36
Per Åhgrene05c43c2018-05-09 12:26:51 +020037bool EnableZeroExternalDelayHeadroom() {
38 return !field_trial::IsEnabled(
39 "WebRTC-Aec3ZeroExternalDelayHeadroomKillSwitch");
40}
41
peahcf02cf12017-04-05 14:18:07 -070042class RenderDelayBufferImpl final : public RenderDelayBuffer {
43 public:
Per Åhgren8ba58612017-12-01 23:01:44 +010044 RenderDelayBufferImpl(const EchoCanceller3Config& config, size_t num_bands);
peahcf02cf12017-04-05 14:18:07 -070045 ~RenderDelayBufferImpl() override;
46
47 void Reset() override;
Per Åhgren8ba58612017-12-01 23:01:44 +010048 BufferingEvent Insert(const std::vector<std::vector<float>>& block) override;
Per Åhgrenc59a5762017-12-11 21:34:19 +010049 BufferingEvent PrepareCaptureProcessing() override;
50 bool SetDelay(size_t delay) override;
Per Åhgren5c532d32018-03-22 00:29:25 +010051 size_t Delay() const override { return MapInternalDelayToExternalDelay(); }
Per Åhgren8ba58612017-12-01 23:01:44 +010052 size_t MaxDelay() const override {
Per Åhgren09a718a2017-12-11 22:28:45 +010053 return blocks_.buffer.size() - 1 - buffer_headroom_;
Per Åhgren8ba58612017-12-01 23:01:44 +010054 }
Per Åhgrenc59a5762017-12-11 21:34:19 +010055 RenderBuffer* GetRenderBuffer() override { return &echo_remover_buffer_; }
peahcf02cf12017-04-05 14:18:07 -070056
57 const DownsampledRenderBuffer& GetDownsampledRenderBuffer() const override {
Per Åhgren8ba58612017-12-01 23:01:44 +010058 return low_rate_;
peahcf02cf12017-04-05 14:18:07 -070059 }
60
Per Åhgrena76ef9d2018-01-25 07:01:34 +010061 bool CausalDelay(size_t delay) const override;
Per Åhgrenc59a5762017-12-11 21:34:19 +010062
Per Åhgrend0fa8202018-04-18 09:35:13 +020063 void SetAudioBufferDelay(size_t delay_ms) override;
64
peahcf02cf12017-04-05 14:18:07 -070065 private:
Per Åhgren38e2d952017-11-17 14:54:28 +010066 static int instance_count_;
67 std::unique_ptr<ApmDataDumper> data_dumper_;
peahcf02cf12017-04-05 14:18:07 -070068 const Aec3Optimization optimization_;
Per Åhgrenc59a5762017-12-11 21:34:19 +010069 const EchoCanceller3Config config_;
Gustaf Ullberg78b1c4a2018-05-25 10:12:58 +020070 size_t down_sampling_factor_;
Per Åhgrene05c43c2018-05-09 12:26:51 +020071 const bool use_zero_external_delay_headroom_;
Per Åhgren8ba58612017-12-01 23:01:44 +010072 const int sub_block_size_;
73 MatrixBuffer blocks_;
74 VectorBuffer spectra_;
75 FftBuffer ffts_;
Danil Chapovalovdb9f7ab2018-06-19 10:50:11 +020076 absl::optional<size_t> delay_;
77 absl::optional<int> internal_delay_;
Per Åhgren8ba58612017-12-01 23:01:44 +010078 RenderBuffer echo_remover_buffer_;
79 DownsampledRenderBuffer low_rate_;
Per Åhgren38e2d952017-11-17 14:54:28 +010080 Decimator render_decimator_;
peahdebaa442017-05-03 05:39:09 -070081 const std::vector<std::vector<float>> zero_block_;
Per Åhgren8ba58612017-12-01 23:01:44 +010082 const Aec3Fft fft_;
Per Åhgren8ba58612017-12-01 23:01:44 +010083 std::vector<float> render_ds_;
Per Åhgren09a718a2017-12-11 22:28:45 +010084 const int buffer_headroom_;
Per Åhgrenb4c188d2017-12-20 00:25:18 +010085 bool last_call_was_render_ = false;
86 int num_api_calls_in_a_row_ = 0;
87 int max_observed_jitter_ = 1;
88 size_t capture_call_counter_ = 0;
89 size_t render_call_counter_ = 0;
Per Åhgrenb6b00dc2018-02-20 22:18:27 +010090 bool render_activity_ = false;
91 size_t render_activity_counter_ = 0;
Danil Chapovalovdb9f7ab2018-06-19 10:50:11 +020092 absl::optional<size_t> external_audio_buffer_delay_;
Per Åhgrend0fa8202018-04-18 09:35:13 +020093 bool external_delay_verified_after_reset_ = false;
Per Åhgren8ba58612017-12-01 23:01:44 +010094
Per Åhgrenc59a5762017-12-11 21:34:19 +010095 int LowRateBufferOffset() const { return DelayEstimatorOffset(config_) >> 1; }
Per Åhgren5c532d32018-03-22 00:29:25 +010096 int MapExternalDelayToInternalDelay(size_t external_delay_blocks) const;
97 int MapInternalDelayToExternalDelay() const;
Per Åhgrenc59a5762017-12-11 21:34:19 +010098 void ApplyDelay(int delay);
99 void InsertBlock(const std::vector<std::vector<float>>& block,
100 int previous_write);
Per Åhgrenb6b00dc2018-02-20 22:18:27 +0100101 bool DetectActiveRender(rtc::ArrayView<const float> x) const;
Per Åhgren8ba58612017-12-01 23:01:44 +0100102
peah69221db2017-01-27 03:28:19 -0800103 RTC_DISALLOW_IMPLICIT_CONSTRUCTORS(RenderDelayBufferImpl);
104};
105
Per Åhgrenc59a5762017-12-11 21:34:19 +0100106// Increases the write indices for the render buffers.
107void IncreaseWriteIndices(int sub_block_size,
108 MatrixBuffer* blocks,
109 VectorBuffer* spectra,
110 FftBuffer* ffts,
111 DownsampledRenderBuffer* low_rate) {
112 low_rate->UpdateWriteIndex(-sub_block_size);
113 blocks->IncWriteIndex();
114 spectra->DecWriteIndex();
115 ffts->DecWriteIndex();
116}
117
118// Increases the read indices for the render buffers.
Danil Chapovalovdb9f7ab2018-06-19 10:50:11 +0200119void IncreaseReadIndices(const absl::optional<int>& delay,
Per Åhgrenc59a5762017-12-11 21:34:19 +0100120 int sub_block_size,
121 MatrixBuffer* blocks,
122 VectorBuffer* spectra,
123 FftBuffer* ffts,
124 DownsampledRenderBuffer* low_rate) {
125 RTC_DCHECK_NE(low_rate->read, low_rate->write);
126 low_rate->UpdateReadIndex(-sub_block_size);
127
128 if (blocks->read != blocks->write) {
129 blocks->IncReadIndex();
130 spectra->DecReadIndex();
131 ffts->DecReadIndex();
132 } else {
133 // Only allow underrun for blocks_ when the delay is not set.
134 RTC_DCHECK(!delay);
135 }
136}
137
138// Checks for a render buffer overrun.
139bool RenderOverrun(const MatrixBuffer& b, const DownsampledRenderBuffer& l) {
140 return l.read == l.write || b.read == b.write;
141}
142
143// Checks for a render buffer underrun. If the delay is not specified, only the
144// low rate buffer underrun is counted as the delay offset for the other buffers
145// is unknown.
Danil Chapovalovdb9f7ab2018-06-19 10:50:11 +0200146bool RenderUnderrun(const absl::optional<int>& delay,
Per Åhgrenc59a5762017-12-11 21:34:19 +0100147 const MatrixBuffer& b,
148 const DownsampledRenderBuffer& l) {
149 return l.read == l.write || (delay && b.read == b.write);
150}
151
152// Computes the latency in the buffer (the number of unread elements).
153int BufferLatency(const DownsampledRenderBuffer& l) {
154 return (l.buffer.size() + l.read - l.write) % l.buffer.size();
155}
156
157// Computes the mismatch between the number of render and capture calls based on
158// the known offset (achieved during reset) of the low rate buffer.
159bool ApiCallSkew(const DownsampledRenderBuffer& low_rate_buffer,
160 int sub_block_size,
161 int low_rate_buffer_offset_sub_blocks) {
162 int latency = BufferLatency(low_rate_buffer);
163 int skew = abs(low_rate_buffer_offset_sub_blocks * sub_block_size - latency);
164 int skew_limit = low_rate_buffer_offset_sub_blocks * sub_block_size;
165 return skew >= skew_limit;
166}
167
Per Åhgren38e2d952017-11-17 14:54:28 +0100168int RenderDelayBufferImpl::instance_count_ = 0;
169
Per Åhgren8ba58612017-12-01 23:01:44 +0100170RenderDelayBufferImpl::RenderDelayBufferImpl(const EchoCanceller3Config& config,
171 size_t num_bands)
Per Åhgren38e2d952017-11-17 14:54:28 +0100172 : data_dumper_(
173 new ApmDataDumper(rtc::AtomicOps::Increment(&instance_count_))),
174 optimization_(DetectOptimization()),
Per Åhgrenc59a5762017-12-11 21:34:19 +0100175 config_(config),
Gustaf Ullberg53e22112018-10-11 15:27:26 +0200176 down_sampling_factor_(config.delay.down_sampling_factor),
Per Åhgrene05c43c2018-05-09 12:26:51 +0200177 use_zero_external_delay_headroom_(EnableZeroExternalDelayHeadroom()),
Gustaf Ullberg78b1c4a2018-05-25 10:12:58 +0200178 sub_block_size_(static_cast<int>(down_sampling_factor_ > 0
179 ? kBlockSize / down_sampling_factor_
180 : kBlockSize)),
181 blocks_(GetRenderDelayBufferSize(down_sampling_factor_,
Per Åhgren09a718a2017-12-11 22:28:45 +0100182 config.delay.num_filters,
Per Åhgren08ea5892018-01-15 08:07:41 +0100183 config.filter.main.length_blocks),
Per Åhgren8ba58612017-12-01 23:01:44 +0100184 num_bands,
185 kBlockSize),
186 spectra_(blocks_.buffer.size(), kFftLengthBy2Plus1),
187 ffts_(blocks_.buffer.size()),
Per Åhgren5c532d32018-03-22 00:29:25 +0100188 delay_(config_.delay.default_delay),
Per Åhgrenec22e3f2017-12-20 15:20:37 +0100189 echo_remover_buffer_(&blocks_, &spectra_, &ffts_),
Gustaf Ullberg78b1c4a2018-05-25 10:12:58 +0200190 low_rate_(GetDownSampledBufferSize(down_sampling_factor_,
Per Åhgren8ba58612017-12-01 23:01:44 +0100191 config.delay.num_filters)),
Gustaf Ullberg78b1c4a2018-05-25 10:12:58 +0200192 render_decimator_(down_sampling_factor_),
Per Åhgren8ba58612017-12-01 23:01:44 +0100193 zero_block_(num_bands, std::vector<float>(kBlockSize, 0.f)),
194 fft_(),
Per Åhgren09a718a2017-12-11 22:28:45 +0100195 render_ds_(sub_block_size_, 0.f),
Per Åhgren08ea5892018-01-15 08:07:41 +0100196 buffer_headroom_(config.filter.main.length_blocks) {
Per Åhgren8ba58612017-12-01 23:01:44 +0100197 RTC_DCHECK_EQ(blocks_.buffer.size(), ffts_.buffer.size());
198 RTC_DCHECK_EQ(spectra_.buffer.size(), ffts_.buffer.size());
Per Åhgrenc59a5762017-12-11 21:34:19 +0100199
200 // Necessary condition to avoid unrecoverable echp due to noncausal alignment.
201 RTC_DCHECK_EQ(DelayEstimatorOffset(config_), LowRateBufferOffset() * 2);
Per Åhgren8ba58612017-12-01 23:01:44 +0100202 Reset();
peahcf02cf12017-04-05 14:18:07 -0700203}
peah69221db2017-01-27 03:28:19 -0800204
205RenderDelayBufferImpl::~RenderDelayBufferImpl() = default;
206
Per Åhgrenc59a5762017-12-11 21:34:19 +0100207// Resets the buffer delays and clears the reported delays.
peahcf02cf12017-04-05 14:18:07 -0700208void RenderDelayBufferImpl::Reset() {
Per Åhgrenb4c188d2017-12-20 00:25:18 +0100209 last_call_was_render_ = false;
210 num_api_calls_in_a_row_ = 1;
211
Per Åhgrenc59a5762017-12-11 21:34:19 +0100212 // Pre-fill the low rate buffer (which is used for delay estimation) to add
213 // headroom for the allowed api call jitter.
214 low_rate_.read = low_rate_.OffsetIndex(
215 low_rate_.write, LowRateBufferOffset() * sub_block_size_);
216
Per Åhgrend0fa8202018-04-18 09:35:13 +0200217 // Check for any external audio buffer delay and whether it is feasible.
Gustaf Ullberga49eacb2018-05-04 16:43:56 +0200218 if (external_audio_buffer_delay_) {
Per Åhgrene05c43c2018-05-09 12:26:51 +0200219 const size_t headroom = use_zero_external_delay_headroom_ ? 0 : 2;
Per Åhgrend0fa8202018-04-18 09:35:13 +0200220 size_t external_delay_to_set = 0;
Per Åhgrene05c43c2018-05-09 12:26:51 +0200221 if (*external_audio_buffer_delay_ < headroom) {
Per Åhgrend0fa8202018-04-18 09:35:13 +0200222 external_delay_to_set = 0;
223 } else {
Per Åhgrene05c43c2018-05-09 12:26:51 +0200224 external_delay_to_set = *external_audio_buffer_delay_ - headroom;
Per Åhgrend0fa8202018-04-18 09:35:13 +0200225 }
Per Åhgrenc59a5762017-12-11 21:34:19 +0100226
Gustaf Ullberg623d2812018-05-07 13:36:07 +0200227 external_delay_to_set = std::min(external_delay_to_set, MaxDelay());
Per Åhgrend0fa8202018-04-18 09:35:13 +0200228
229 // When an external delay estimate is available, use that delay as the
Gustaf Ullberg623d2812018-05-07 13:36:07 +0200230 // initial render buffer delay.
231 internal_delay_ = external_delay_to_set;
232 ApplyDelay(*internal_delay_);
233 delay_ = MapInternalDelayToExternalDelay();
234
Per Åhgrend0fa8202018-04-18 09:35:13 +0200235 external_delay_verified_after_reset_ = false;
236 } else {
237 // If an external delay estimate is not available, use that delay as the
238 // initial delay. Set the render buffer delays to the default delay.
239 ApplyDelay(config_.delay.default_delay);
240
241 // Unset the delays which are set by SetDelay.
Danil Chapovalovdb9f7ab2018-06-19 10:50:11 +0200242 delay_ = absl::nullopt;
243 internal_delay_ = absl::nullopt;
Per Åhgrend0fa8202018-04-18 09:35:13 +0200244 }
peahcf02cf12017-04-05 14:18:07 -0700245}
peah69221db2017-01-27 03:28:19 -0800246
Per Åhgrenc59a5762017-12-11 21:34:19 +0100247// Inserts a new block into the render buffers.
Per Åhgren8ba58612017-12-01 23:01:44 +0100248RenderDelayBuffer::BufferingEvent RenderDelayBufferImpl::Insert(
peahcf02cf12017-04-05 14:18:07 -0700249 const std::vector<std::vector<float>>& block) {
Per Åhgrenb4c188d2017-12-20 00:25:18 +0100250 ++render_call_counter_;
251 if (delay_) {
252 if (!last_call_was_render_) {
253 last_call_was_render_ = true;
254 num_api_calls_in_a_row_ = 1;
255 } else {
256 if (++num_api_calls_in_a_row_ > max_observed_jitter_) {
257 max_observed_jitter_ = num_api_calls_in_a_row_;
Per Åhgrenf7ac09f2018-03-22 07:45:43 +0100258 RTC_LOG(LS_WARNING)
Per Åhgrenb4c188d2017-12-20 00:25:18 +0100259 << "New max number api jitter observed at render block "
260 << render_call_counter_ << ": " << num_api_calls_in_a_row_
261 << " blocks";
262 }
263 }
264 }
265
Per Åhgrenc59a5762017-12-11 21:34:19 +0100266 // Increase the write indices to where the new blocks should be written.
267 const int previous_write = blocks_.write;
268 IncreaseWriteIndices(sub_block_size_, &blocks_, &spectra_, &ffts_,
269 &low_rate_);
peahcf02cf12017-04-05 14:18:07 -0700270
Per Åhgrenc59a5762017-12-11 21:34:19 +0100271 // Allow overrun and do a reset when render overrun occurrs due to more render
272 // data being inserted than capture data is received.
273 BufferingEvent event = RenderOverrun(blocks_, low_rate_)
Per Åhgren2e27d1c2017-12-12 09:33:23 +0100274 ? BufferingEvent::kRenderOverrun
Per Åhgrenc59a5762017-12-11 21:34:19 +0100275 : BufferingEvent::kNone;
276
Per Åhgrenb6b00dc2018-02-20 22:18:27 +0100277 // Detect and update render activity.
278 if (!render_activity_) {
279 render_activity_counter_ += DetectActiveRender(block[0]) ? 1 : 0;
280 render_activity_ = render_activity_counter_ >= 20;
281 }
282
Per Åhgrenc59a5762017-12-11 21:34:19 +0100283 // Insert the new render block into the specified position.
284 InsertBlock(block, previous_write);
285
286 if (event != BufferingEvent::kNone) {
287 Reset();
peah69221db2017-01-27 03:28:19 -0800288 }
peah69221db2017-01-27 03:28:19 -0800289
Per Åhgren8ba58612017-12-01 23:01:44 +0100290 return event;
peah69221db2017-01-27 03:28:19 -0800291}
292
Per Åhgrenc59a5762017-12-11 21:34:19 +0100293// Prepares the render buffers for processing another capture block.
294RenderDelayBuffer::BufferingEvent
295RenderDelayBufferImpl::PrepareCaptureProcessing() {
Per Åhgren8ba58612017-12-01 23:01:44 +0100296 BufferingEvent event = BufferingEvent::kNone;
Per Åhgrenb4c188d2017-12-20 00:25:18 +0100297 ++capture_call_counter_;
298
299 if (delay_) {
300 if (last_call_was_render_) {
301 last_call_was_render_ = false;
302 num_api_calls_in_a_row_ = 1;
303 } else {
304 if (++num_api_calls_in_a_row_ > max_observed_jitter_) {
305 max_observed_jitter_ = num_api_calls_in_a_row_;
Per Åhgrenf7ac09f2018-03-22 07:45:43 +0100306 RTC_LOG(LS_WARNING)
Per Åhgrenb4c188d2017-12-20 00:25:18 +0100307 << "New max number api jitter observed at capture block "
308 << capture_call_counter_ << ": " << num_api_calls_in_a_row_
309 << " blocks";
310 }
311 }
312 }
peah69221db2017-01-27 03:28:19 -0800313
Per Åhgrenc59a5762017-12-11 21:34:19 +0100314 if (RenderUnderrun(internal_delay_, blocks_, low_rate_)) {
315 // Don't increase the read indices if there is a render underrun.
Per Åhgren8ba58612017-12-01 23:01:44 +0100316 event = BufferingEvent::kRenderUnderrun;
peahdebaa442017-05-03 05:39:09 -0700317 } else {
Per Åhgrenc59a5762017-12-11 21:34:19 +0100318 // Increase the read indices in the render buffers to point to the most
319 // recent block to use in the capture processing.
320 IncreaseReadIndices(internal_delay_, sub_block_size_, &blocks_, &spectra_,
321 &ffts_, &low_rate_);
Per Åhgren8ba58612017-12-01 23:01:44 +0100322
Per Åhgrenc59a5762017-12-11 21:34:19 +0100323 // Check for skew in the API calls which, if too large, causes the delay
324 // estimation to be noncausal. Doing this check after the render indice
325 // increase saves one unit of allowed skew. Note that the skew check only
326 // should need to be one-sided as one of the skew directions results in an
327 // underrun.
328 bool skew = ApiCallSkew(low_rate_, sub_block_size_, LowRateBufferOffset());
329 event = skew ? BufferingEvent::kApiCallSkew : BufferingEvent::kNone;
Per Åhgren8ba58612017-12-01 23:01:44 +0100330 }
331
Per Åhgrenc59a5762017-12-11 21:34:19 +0100332 if (event != BufferingEvent::kNone) {
333 Reset();
334 }
335
Per Åhgrenb6b00dc2018-02-20 22:18:27 +0100336 echo_remover_buffer_.SetRenderActivity(render_activity_);
337 if (render_activity_) {
338 render_activity_counter_ = 0;
339 render_activity_ = false;
340 }
341
Per Åhgren8ba58612017-12-01 23:01:44 +0100342 return event;
peah69221db2017-01-27 03:28:19 -0800343}
344
Per Åhgrenc59a5762017-12-11 21:34:19 +0100345// Sets the delay and returns a bool indicating whether the delay was changed.
346bool RenderDelayBufferImpl::SetDelay(size_t delay) {
Gustaf Ullberg623d2812018-05-07 13:36:07 +0200347 if (!external_delay_verified_after_reset_ && external_audio_buffer_delay_ &&
348 delay_) {
349 int difference = static_cast<int>(delay) - static_cast<int>(*delay_);
350 RTC_LOG(LS_WARNING) << "Mismatch between first estimated delay after reset "
351 "and external delay: "
352 << difference << " blocks";
Per Åhgrend0fa8202018-04-18 09:35:13 +0200353 external_delay_verified_after_reset_ = true;
354 }
Per Åhgrenc59a5762017-12-11 21:34:19 +0100355 if (delay_ && *delay_ == delay) {
356 return false;
peahcf02cf12017-04-05 14:18:07 -0700357 }
Per Åhgren8ba58612017-12-01 23:01:44 +0100358 delay_ = delay;
Per Åhgren8ba58612017-12-01 23:01:44 +0100359
Per Åhgrenc59a5762017-12-11 21:34:19 +0100360 // Compute the internal delay and limit the delay to the allowed range.
Per Åhgren5c532d32018-03-22 00:29:25 +0100361 int internal_delay = MapExternalDelayToInternalDelay(*delay_);
Per Åhgrenc59a5762017-12-11 21:34:19 +0100362 internal_delay_ =
363 std::min(MaxDelay(), static_cast<size_t>(std::max(internal_delay, 0)));
364
365 // Apply the delay to the buffers.
366 ApplyDelay(*internal_delay_);
367 return true;
peah69221db2017-01-27 03:28:19 -0800368}
369
Per Åhgrenc59a5762017-12-11 21:34:19 +0100370// Returns whether the specified delay is causal.
Per Åhgrena76ef9d2018-01-25 07:01:34 +0100371bool RenderDelayBufferImpl::CausalDelay(size_t delay) const {
372 // Compute the internal delay and limit the delay to the allowed range.
Per Åhgren5c532d32018-03-22 00:29:25 +0100373 int internal_delay = MapExternalDelayToInternalDelay(delay);
Per Åhgrena76ef9d2018-01-25 07:01:34 +0100374 internal_delay =
375 std::min(MaxDelay(), static_cast<size_t>(std::max(internal_delay, 0)));
376
377 return internal_delay >=
378 static_cast<int>(config_.delay.min_echo_path_delay_blocks);
Per Åhgrenc59a5762017-12-11 21:34:19 +0100379}
Per Åhgren8ba58612017-12-01 23:01:44 +0100380
Per Åhgrend0fa8202018-04-18 09:35:13 +0200381void RenderDelayBufferImpl::SetAudioBufferDelay(size_t delay_ms) {
Gustaf Ullberga49eacb2018-05-04 16:43:56 +0200382 if (!external_audio_buffer_delay_) {
Per Åhgrend0fa8202018-04-18 09:35:13 +0200383 RTC_LOG(LS_WARNING)
384 << "Receiving a first reported externally buffer delay of " << delay_ms
385 << " ms.";
386 }
Gustaf Ullberg623d2812018-05-07 13:36:07 +0200387
388 // Convert delay from milliseconds to blocks (rounded down).
Gustaf Ullberga49eacb2018-05-04 16:43:56 +0200389 external_audio_buffer_delay_ = delay_ms / 4;
Per Åhgrend0fa8202018-04-18 09:35:13 +0200390}
391
Per Åhgrenc59a5762017-12-11 21:34:19 +0100392// Maps the externally computed delay to the delay used internally.
Per Åhgren5c532d32018-03-22 00:29:25 +0100393int RenderDelayBufferImpl::MapExternalDelayToInternalDelay(
Per Åhgrenc59a5762017-12-11 21:34:19 +0100394 size_t external_delay_blocks) const {
395 const int latency = BufferLatency(low_rate_);
396 RTC_DCHECK_LT(0, sub_block_size_);
397 RTC_DCHECK_EQ(0, latency % sub_block_size_);
398 int latency_blocks = latency / sub_block_size_;
399 return latency_blocks + static_cast<int>(external_delay_blocks) -
400 DelayEstimatorOffset(config_);
401}
Per Åhgren8ba58612017-12-01 23:01:44 +0100402
Per Åhgren5c532d32018-03-22 00:29:25 +0100403// Maps the internally used delay to the delay used externally.
404int RenderDelayBufferImpl::MapInternalDelayToExternalDelay() const {
405 const int latency = BufferLatency(low_rate_);
406 int latency_blocks = latency / sub_block_size_;
407 int internal_delay = spectra_.read >= spectra_.write
408 ? spectra_.read - spectra_.write
409 : spectra_.size + spectra_.read - spectra_.write;
410
411 return internal_delay - latency_blocks + DelayEstimatorOffset(config_);
412}
413
Per Åhgrenc59a5762017-12-11 21:34:19 +0100414// Set the read indices according to the delay.
415void RenderDelayBufferImpl::ApplyDelay(int delay) {
Per Åhgrenf7ac09f2018-03-22 07:45:43 +0100416 RTC_LOG(LS_WARNING) << "Applying internal delay of " << delay << " blocks.";
Per Åhgrenc59a5762017-12-11 21:34:19 +0100417 blocks_.read = blocks_.OffsetIndex(blocks_.write, -delay);
418 spectra_.read = spectra_.OffsetIndex(spectra_.write, delay);
419 ffts_.read = ffts_.OffsetIndex(ffts_.write, delay);
420}
Per Åhgren8ba58612017-12-01 23:01:44 +0100421
Per Åhgrenc59a5762017-12-11 21:34:19 +0100422// Inserts a block into the render buffers.
423void RenderDelayBufferImpl::InsertBlock(
424 const std::vector<std::vector<float>>& block,
425 int previous_write) {
426 auto& b = blocks_;
427 auto& lr = low_rate_;
428 auto& ds = render_ds_;
429 auto& f = ffts_;
430 auto& s = spectra_;
431 RTC_DCHECK_EQ(block.size(), b.buffer[b.write].size());
432 for (size_t k = 0; k < block.size(); ++k) {
433 RTC_DCHECK_EQ(block[k].size(), b.buffer[b.write][k].size());
434 std::copy(block[k].begin(), block[k].end(), b.buffer[b.write][k].begin());
435 }
Per Åhgren8ba58612017-12-01 23:01:44 +0100436
Gustaf Ullberg43c707a2018-05-22 15:27:23 +0200437 data_dumper_->DumpWav("aec3_render_decimator_input", block[0].size(),
438 block[0].data(), 16000, 1);
Per Åhgrenc59a5762017-12-11 21:34:19 +0100439 render_decimator_.Decimate(block[0], ds);
Gustaf Ullberg43c707a2018-05-22 15:27:23 +0200440 data_dumper_->DumpWav("aec3_render_decimator_output", ds.size(), ds.data(),
Gustaf Ullberg78b1c4a2018-05-25 10:12:58 +0200441 16000 / down_sampling_factor_, 1);
Per Åhgrenc59a5762017-12-11 21:34:19 +0100442 std::copy(ds.rbegin(), ds.rend(), lr.buffer.begin() + lr.write);
443 fft_.PaddedFft(block[0], b.buffer[previous_write][0], &f.buffer[f.write]);
444 f.buffer[f.write].Spectrum(optimization_, s.buffer[s.write]);
445}
Per Åhgren8ba58612017-12-01 23:01:44 +0100446
Per Åhgrenb6b00dc2018-02-20 22:18:27 +0100447bool RenderDelayBufferImpl::DetectActiveRender(
448 rtc::ArrayView<const float> x) const {
449 const float x_energy = std::inner_product(x.begin(), x.end(), x.begin(), 0.f);
450 return x_energy > (config_.render_levels.active_render_limit *
451 config_.render_levels.active_render_limit) *
452 kFftLengthBy2;
453}
454
peah69221db2017-01-27 03:28:19 -0800455} // namespace
456
Per Åhgrenc59a5762017-12-11 21:34:19 +0100457int RenderDelayBuffer::RenderDelayBuffer::DelayEstimatorOffset(
458 const EchoCanceller3Config& config) {
459 return config.delay.api_call_jitter_blocks * 2;
460}
461
Per Åhgren8ba58612017-12-01 23:01:44 +0100462RenderDelayBuffer* RenderDelayBuffer::Create(const EchoCanceller3Config& config,
463 size_t num_bands) {
464 return new RenderDelayBufferImpl(config, num_bands);
peah69221db2017-01-27 03:28:19 -0800465}
466
467} // namespace webrtc