blob: 2556bcda3179acdd94e4dca75a2698a6ef0a2443 [file] [log] [blame]
niklase@google.com470e71d2011-07-07 08:21:25 +00001/*
andrew@webrtc.org63a50982012-05-02 23:56:37 +00002 * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
niklase@google.com470e71d2011-07-07 08:21:25 +00003 *
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
pbos@webrtc.org7fad4b82013-05-28 08:11:59 +000011#include "webrtc/modules/audio_processing/audio_buffer.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000012
andrew@webrtc.org17e40642014-03-04 20:58:13 +000013#include "webrtc/common_audio/include/audio_util.h"
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +000014#include "webrtc/common_audio/resampler/push_sinc_resampler.h"
pbos@webrtc.org7fad4b82013-05-28 08:11:59 +000015#include "webrtc/common_audio/signal_processing/include/signal_processing_library.h"
andrew@webrtc.org755b04a2011-11-15 16:57:56 +000016
niklase@google.com470e71d2011-07-07 08:21:25 +000017namespace webrtc {
18namespace {
19
20enum {
21 kSamplesPer8kHzChannel = 80,
22 kSamplesPer16kHzChannel = 160,
23 kSamplesPer32kHzChannel = 320
24};
25
andrew@webrtc.org103657b2014-04-24 18:28:56 +000026bool HasKeyboardChannel(AudioProcessing::ChannelLayout layout) {
27 switch (layout) {
28 case AudioProcessing::kMono:
29 case AudioProcessing::kStereo:
30 return false;
31 case AudioProcessing::kMonoAndKeyboard:
32 case AudioProcessing::kStereoAndKeyboard:
33 return true;
34 }
35 assert(false);
36 return false;
37}
38
39int KeyboardChannelIndex(AudioProcessing::ChannelLayout layout) {
40 switch (layout) {
41 case AudioProcessing::kMono:
42 case AudioProcessing::kStereo:
43 assert(false);
44 return -1;
45 case AudioProcessing::kMonoAndKeyboard:
46 return 1;
47 case AudioProcessing::kStereoAndKeyboard:
48 return 2;
49 }
50 assert(false);
51 return -1;
52}
53
andrew@webrtc.org8328e7c2014-10-31 04:58:14 +000054template <typename T>
55void StereoToMono(const T* left, const T* right, T* out,
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +000056 int samples_per_channel) {
andrew@webrtc.org8328e7c2014-10-31 04:58:14 +000057 for (int i = 0; i < samples_per_channel; ++i)
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +000058 out[i] = (left[i] + right[i]) / 2;
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +000059}
60
niklase@google.com470e71d2011-07-07 08:21:25 +000061} // namespace
62
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +000063// One int16_t and one float ChannelBuffer that are kept in sync. The sync is
64// broken when someone requests write access to either ChannelBuffer, and
65// reestablished when someone requests the outdated ChannelBuffer. It is
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +000066// therefore safe to use the return value of ibuf_const() and fbuf_const()
67// until the next call to ibuf() or fbuf(), and the return value of ibuf() and
68// fbuf() until the next call to any of the other functions.
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +000069class IFChannelBuffer {
70 public:
71 IFChannelBuffer(int samples_per_channel, int num_channels)
72 : ivalid_(true),
73 ibuf_(samples_per_channel, num_channels),
74 fvalid_(true),
75 fbuf_(samples_per_channel, num_channels) {}
76
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +000077 ChannelBuffer<int16_t>* ibuf() { return ibuf(false); }
78 ChannelBuffer<float>* fbuf() { return fbuf(false); }
79 const ChannelBuffer<int16_t>* ibuf_const() { return ibuf(true); }
80 const ChannelBuffer<float>* fbuf_const() { return fbuf(true); }
81
82 private:
83 ChannelBuffer<int16_t>* ibuf(bool readonly) {
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +000084 RefreshI();
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +000085 fvalid_ = readonly;
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +000086 return &ibuf_;
87 }
88
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +000089 ChannelBuffer<float>* fbuf(bool readonly) {
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +000090 RefreshF();
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +000091 ivalid_ = readonly;
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +000092 return &fbuf_;
93 }
94
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +000095 void RefreshF() {
96 if (!fvalid_) {
97 assert(ivalid_);
98 const int16_t* const int_data = ibuf_.data();
99 float* const float_data = fbuf_.data();
100 const int length = fbuf_.length();
101 for (int i = 0; i < length; ++i)
102 float_data[i] = int_data[i];
103 fvalid_ = true;
104 }
105 }
106
107 void RefreshI() {
108 if (!ivalid_) {
109 assert(fvalid_);
andrew@webrtc.org8328e7c2014-10-31 04:58:14 +0000110 FloatS16ToS16(fbuf_.data(), ibuf_.length(), ibuf_.data());
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +0000111 ivalid_ = true;
112 }
113 }
114
115 bool ivalid_;
116 ChannelBuffer<int16_t> ibuf_;
117 bool fvalid_;
118 ChannelBuffer<float> fbuf_;
119};
120
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000121AudioBuffer::AudioBuffer(int input_samples_per_channel,
122 int num_input_channels,
123 int process_samples_per_channel,
124 int num_process_channels,
125 int output_samples_per_channel)
126 : input_samples_per_channel_(input_samples_per_channel),
127 num_input_channels_(num_input_channels),
128 proc_samples_per_channel_(process_samples_per_channel),
129 num_proc_channels_(num_process_channels),
130 output_samples_per_channel_(output_samples_per_channel),
131 samples_per_split_channel_(proc_samples_per_channel_),
aluebs@webrtc.org2561d522014-07-17 08:27:39 +0000132 mixed_low_pass_valid_(false),
andrew@webrtc.orged083d42011-09-19 15:28:51 +0000133 reference_copied_(false),
134 activity_(AudioFrame::kVadUnknown),
andrew@webrtc.org103657b2014-04-24 18:28:56 +0000135 keyboard_data_(NULL),
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +0000136 channels_(new IFChannelBuffer(proc_samples_per_channel_,
137 num_proc_channels_)) {
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000138 assert(input_samples_per_channel_ > 0);
139 assert(proc_samples_per_channel_ > 0);
140 assert(output_samples_per_channel_ > 0);
141 assert(num_input_channels_ > 0 && num_input_channels_ <= 2);
142 assert(num_proc_channels_ <= num_input_channels);
niklase@google.com470e71d2011-07-07 08:21:25 +0000143
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000144 if (num_input_channels_ == 2 && num_proc_channels_ == 1) {
145 input_buffer_.reset(new ChannelBuffer<float>(input_samples_per_channel_,
146 num_proc_channels_));
147 }
148
149 if (input_samples_per_channel_ != proc_samples_per_channel_ ||
150 output_samples_per_channel_ != proc_samples_per_channel_) {
151 // Create an intermediate buffer for resampling.
152 process_buffer_.reset(new ChannelBuffer<float>(proc_samples_per_channel_,
153 num_proc_channels_));
154 }
155
156 if (input_samples_per_channel_ != proc_samples_per_channel_) {
157 input_resamplers_.reserve(num_proc_channels_);
158 for (int i = 0; i < num_proc_channels_; ++i) {
159 input_resamplers_.push_back(
160 new PushSincResampler(input_samples_per_channel_,
161 proc_samples_per_channel_));
162 }
163 }
164
165 if (output_samples_per_channel_ != proc_samples_per_channel_) {
166 output_resamplers_.reserve(num_proc_channels_);
167 for (int i = 0; i < num_proc_channels_; ++i) {
168 output_resamplers_.push_back(
169 new PushSincResampler(proc_samples_per_channel_,
170 output_samples_per_channel_));
171 }
172 }
173
174 if (proc_samples_per_channel_ == kSamplesPer32kHzChannel) {
niklase@google.com470e71d2011-07-07 08:21:25 +0000175 samples_per_split_channel_ = kSamplesPer16kHzChannel;
kwiberg@webrtc.org2b6bc8d2014-07-17 09:46:37 +0000176 split_channels_low_.reset(new IFChannelBuffer(samples_per_split_channel_,
177 num_proc_channels_));
178 split_channels_high_.reset(new IFChannelBuffer(samples_per_split_channel_,
179 num_proc_channels_));
aluebs@webrtc.orgbe05c742014-11-14 22:18:10 +0000180 splitting_filter_.reset(new SplittingFilter(num_proc_channels_));
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000181 }
182}
183
andrew@webrtc.org103657b2014-04-24 18:28:56 +0000184AudioBuffer::~AudioBuffer() {}
185
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000186void AudioBuffer::CopyFrom(const float* const* data,
187 int samples_per_channel,
188 AudioProcessing::ChannelLayout layout) {
189 assert(samples_per_channel == input_samples_per_channel_);
190 assert(ChannelsFromLayout(layout) == num_input_channels_);
191 InitForNewData();
192
andrew@webrtc.org103657b2014-04-24 18:28:56 +0000193 if (HasKeyboardChannel(layout)) {
194 keyboard_data_ = data[KeyboardChannelIndex(layout)];
195 }
196
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000197 // Downmix.
198 const float* const* data_ptr = data;
199 if (num_input_channels_ == 2 && num_proc_channels_ == 1) {
200 StereoToMono(data[0],
201 data[1],
202 input_buffer_->channel(0),
203 input_samples_per_channel_);
204 data_ptr = input_buffer_->channels();
205 }
206
207 // Resample.
208 if (input_samples_per_channel_ != proc_samples_per_channel_) {
209 for (int i = 0; i < num_proc_channels_; ++i) {
210 input_resamplers_[i]->Resample(data_ptr[i],
211 input_samples_per_channel_,
212 process_buffer_->channel(i),
213 proc_samples_per_channel_);
214 }
215 data_ptr = process_buffer_->channels();
216 }
217
andrew@webrtc.org8328e7c2014-10-31 04:58:14 +0000218 // Convert to the S16 range.
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000219 for (int i = 0; i < num_proc_channels_; ++i) {
andrew@webrtc.org8328e7c2014-10-31 04:58:14 +0000220 FloatToFloatS16(data_ptr[i], proc_samples_per_channel_,
221 channels_->fbuf()->channel(i));
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000222 }
223}
224
225void AudioBuffer::CopyTo(int samples_per_channel,
226 AudioProcessing::ChannelLayout layout,
227 float* const* data) {
228 assert(samples_per_channel == output_samples_per_channel_);
229 assert(ChannelsFromLayout(layout) == num_proc_channels_);
230
andrew@webrtc.org8328e7c2014-10-31 04:58:14 +0000231 // Convert to the float range.
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000232 float* const* data_ptr = data;
233 if (output_samples_per_channel_ != proc_samples_per_channel_) {
234 // Convert to an intermediate buffer for subsequent resampling.
235 data_ptr = process_buffer_->channels();
236 }
237 for (int i = 0; i < num_proc_channels_; ++i) {
andrew@webrtc.org8328e7c2014-10-31 04:58:14 +0000238 FloatS16ToFloat(channels_->fbuf()->channel(i), proc_samples_per_channel_,
239 data_ptr[i]);
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000240 }
241
242 // Resample.
243 if (output_samples_per_channel_ != proc_samples_per_channel_) {
244 for (int i = 0; i < num_proc_channels_; ++i) {
245 output_resamplers_[i]->Resample(data_ptr[i],
246 proc_samples_per_channel_,
247 data[i],
248 output_samples_per_channel_);
249 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000250 }
251}
252
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000253void AudioBuffer::InitForNewData() {
andrew@webrtc.org103657b2014-04-24 18:28:56 +0000254 keyboard_data_ = NULL;
aluebs@webrtc.org2561d522014-07-17 08:27:39 +0000255 mixed_low_pass_valid_ = false;
andrew@webrtc.org17e40642014-03-04 20:58:13 +0000256 reference_copied_ = false;
257 activity_ = AudioFrame::kVadUnknown;
andrew@webrtc.org17e40642014-03-04 20:58:13 +0000258}
259
andrew@webrtc.org65f93382014-04-30 16:44:13 +0000260const int16_t* AudioBuffer::data(int channel) const {
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +0000261 return channels_->ibuf_const()->channel(channel);
niklase@google.com470e71d2011-07-07 08:21:25 +0000262}
263
andrew@webrtc.org65f93382014-04-30 16:44:13 +0000264int16_t* AudioBuffer::data(int channel) {
aluebs@webrtc.org2561d522014-07-17 08:27:39 +0000265 mixed_low_pass_valid_ = false;
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +0000266 return channels_->ibuf()->channel(channel);
andrew@webrtc.org65f93382014-04-30 16:44:13 +0000267}
268
aluebs@webrtc.orgbe05c742014-11-14 22:18:10 +0000269const int16_t* const* AudioBuffer::channels() const {
270 return channels_->ibuf_const()->channels();
271}
272
273int16_t* const* AudioBuffer::channels() {
274 mixed_low_pass_valid_ = false;
275 return channels_->ibuf()->channels();
276}
277
kwiberg@webrtc.org38214d52014-07-03 09:47:33 +0000278const float* AudioBuffer::data_f(int channel) const {
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +0000279 return channels_->fbuf_const()->channel(channel);
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +0000280}
281
kwiberg@webrtc.org38214d52014-07-03 09:47:33 +0000282float* AudioBuffer::data_f(int channel) {
aluebs@webrtc.org2561d522014-07-17 08:27:39 +0000283 mixed_low_pass_valid_ = false;
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +0000284 return channels_->fbuf()->channel(channel);
kwiberg@webrtc.org38214d52014-07-03 09:47:33 +0000285}
286
claguna@google.combfacaab2014-09-25 20:52:08 +0000287const float* const* AudioBuffer::channels_f() const {
288 return channels_->fbuf_const()->channels();
289}
290
291float* const* AudioBuffer::channels_f() {
292 mixed_low_pass_valid_ = false;
293 return channels_->fbuf()->channels();
294}
295
andrew@webrtc.org65f93382014-04-30 16:44:13 +0000296const int16_t* AudioBuffer::low_pass_split_data(int channel) const {
kwiberg@webrtc.org2b6bc8d2014-07-17 09:46:37 +0000297 return split_channels_low_.get()
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +0000298 ? split_channels_low_->ibuf_const()->channel(channel)
kwiberg@webrtc.org2b6bc8d2014-07-17 09:46:37 +0000299 : data(channel);
niklase@google.com470e71d2011-07-07 08:21:25 +0000300}
301
andrew@webrtc.org65f93382014-04-30 16:44:13 +0000302int16_t* AudioBuffer::low_pass_split_data(int channel) {
aluebs@webrtc.org2561d522014-07-17 08:27:39 +0000303 mixed_low_pass_valid_ = false;
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +0000304 return split_channels_low_.get()
305 ? split_channels_low_->ibuf()->channel(channel)
306 : data(channel);
andrew@webrtc.org65f93382014-04-30 16:44:13 +0000307}
308
aluebs@webrtc.orgbe05c742014-11-14 22:18:10 +0000309const int16_t* const* AudioBuffer::low_pass_split_channels() const {
310 return split_channels_low_.get()
311 ? split_channels_low_->ibuf_const()->channels()
312 : channels();
313}
314
315int16_t* const* AudioBuffer::low_pass_split_channels() {
316 mixed_low_pass_valid_ = false;
317 return split_channels_low_.get() ? split_channels_low_->ibuf()->channels()
318 : channels();
319}
320
kwiberg@webrtc.org38214d52014-07-03 09:47:33 +0000321const float* AudioBuffer::low_pass_split_data_f(int channel) const {
kwiberg@webrtc.org2b6bc8d2014-07-17 09:46:37 +0000322 return split_channels_low_.get()
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +0000323 ? split_channels_low_->fbuf_const()->channel(channel)
kwiberg@webrtc.org2b6bc8d2014-07-17 09:46:37 +0000324 : data_f(channel);
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +0000325}
326
kwiberg@webrtc.org38214d52014-07-03 09:47:33 +0000327float* AudioBuffer::low_pass_split_data_f(int channel) {
aluebs@webrtc.org2561d522014-07-17 08:27:39 +0000328 mixed_low_pass_valid_ = false;
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +0000329 return split_channels_low_.get()
330 ? split_channels_low_->fbuf()->channel(channel)
331 : data_f(channel);
kwiberg@webrtc.org38214d52014-07-03 09:47:33 +0000332}
333
claguna@google.combfacaab2014-09-25 20:52:08 +0000334const float* const* AudioBuffer::low_pass_split_channels_f() const {
335 return split_channels_low_.get()
336 ? split_channels_low_->fbuf_const()->channels()
337 : channels_f();
338}
339
340float* const* AudioBuffer::low_pass_split_channels_f() {
341 mixed_low_pass_valid_ = false;
342 return split_channels_low_.get()
343 ? split_channels_low_->fbuf()->channels()
344 : channels_f();
345}
346
andrew@webrtc.org65f93382014-04-30 16:44:13 +0000347const int16_t* AudioBuffer::high_pass_split_data(int channel) const {
kwiberg@webrtc.org2b6bc8d2014-07-17 09:46:37 +0000348 return split_channels_high_.get()
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +0000349 ? split_channels_high_->ibuf_const()->channel(channel)
350 : NULL;
351}
352
353int16_t* AudioBuffer::high_pass_split_data(int channel) {
354 return split_channels_high_.get()
kwiberg@webrtc.org2b6bc8d2014-07-17 09:46:37 +0000355 ? split_channels_high_->ibuf()->channel(channel)
356 : NULL;
niklase@google.com470e71d2011-07-07 08:21:25 +0000357}
358
aluebs@webrtc.orgbe05c742014-11-14 22:18:10 +0000359const int16_t* const* AudioBuffer::high_pass_split_channels() const {
360 return split_channels_high_.get()
361 ? split_channels_high_->ibuf_const()->channels()
362 : NULL;
363}
364
365int16_t* const* AudioBuffer::high_pass_split_channels() {
366 return split_channels_high_.get() ? split_channels_high_->ibuf()->channels()
367 : NULL;
368}
369
kwiberg@webrtc.org38214d52014-07-03 09:47:33 +0000370const float* AudioBuffer::high_pass_split_data_f(int channel) const {
kwiberg@webrtc.org2b6bc8d2014-07-17 09:46:37 +0000371 return split_channels_high_.get()
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +0000372 ? split_channels_high_->fbuf_const()->channel(channel)
kwiberg@webrtc.org2b6bc8d2014-07-17 09:46:37 +0000373 : NULL;
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +0000374}
375
kwiberg@webrtc.org38214d52014-07-03 09:47:33 +0000376float* AudioBuffer::high_pass_split_data_f(int channel) {
kwiberg@webrtc.orge364ac92014-07-18 07:50:29 +0000377 return split_channels_high_.get()
378 ? split_channels_high_->fbuf()->channel(channel)
379 : NULL;
kwiberg@webrtc.org38214d52014-07-03 09:47:33 +0000380}
381
claguna@google.combfacaab2014-09-25 20:52:08 +0000382const float* const* AudioBuffer::high_pass_split_channels_f() const {
383 return split_channels_high_.get()
384 ? split_channels_high_->fbuf_const()->channels()
385 : NULL;
386}
387
388float* const* AudioBuffer::high_pass_split_channels_f() {
389 return split_channels_high_.get()
390 ? split_channels_high_->fbuf()->channels()
391 : NULL;
392}
393
aluebs@webrtc.org2561d522014-07-17 08:27:39 +0000394const int16_t* AudioBuffer::mixed_low_pass_data() {
395 // Currently only mixing stereo to mono is supported.
396 assert(num_proc_channels_ == 1 || num_proc_channels_ == 2);
andrew@webrtc.org755b04a2011-11-15 16:57:56 +0000397
aluebs@webrtc.org2561d522014-07-17 08:27:39 +0000398 if (num_proc_channels_ == 1) {
399 return low_pass_split_data(0);
400 }
401
402 if (!mixed_low_pass_valid_) {
403 if (!mixed_low_pass_channels_.get()) {
404 mixed_low_pass_channels_.reset(
405 new ChannelBuffer<int16_t>(samples_per_split_channel_, 1));
406 }
407 StereoToMono(low_pass_split_data(0),
408 low_pass_split_data(1),
409 mixed_low_pass_channels_->data(),
410 samples_per_split_channel_);
411 mixed_low_pass_valid_ = true;
412 }
413 return mixed_low_pass_channels_->data();
niklase@google.com470e71d2011-07-07 08:21:25 +0000414}
415
andrew@webrtc.org65f93382014-04-30 16:44:13 +0000416const int16_t* AudioBuffer::low_pass_reference(int channel) const {
niklase@google.com470e71d2011-07-07 08:21:25 +0000417 if (!reference_copied_) {
418 return NULL;
419 }
420
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000421 return low_pass_reference_channels_->channel(channel);
niklase@google.com470e71d2011-07-07 08:21:25 +0000422}
423
andrew@webrtc.org103657b2014-04-24 18:28:56 +0000424const float* AudioBuffer::keyboard_data() const {
425 return keyboard_data_;
426}
427
andrew@webrtc.orged083d42011-09-19 15:28:51 +0000428void AudioBuffer::set_activity(AudioFrame::VADActivity activity) {
429 activity_ = activity;
430}
431
andrew@webrtc.org755b04a2011-11-15 16:57:56 +0000432AudioFrame::VADActivity AudioBuffer::activity() const {
andrew@webrtc.orged083d42011-09-19 15:28:51 +0000433 return activity_;
434}
435
436int AudioBuffer::num_channels() const {
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000437 return num_proc_channels_;
niklase@google.com470e71d2011-07-07 08:21:25 +0000438}
439
andrew@webrtc.orged083d42011-09-19 15:28:51 +0000440int AudioBuffer::samples_per_channel() const {
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000441 return proc_samples_per_channel_;
niklase@google.com470e71d2011-07-07 08:21:25 +0000442}
443
andrew@webrtc.orged083d42011-09-19 15:28:51 +0000444int AudioBuffer::samples_per_split_channel() const {
niklase@google.com470e71d2011-07-07 08:21:25 +0000445 return samples_per_split_channel_;
446}
447
andrew@webrtc.org103657b2014-04-24 18:28:56 +0000448int AudioBuffer::samples_per_keyboard_channel() const {
449 // We don't resample the keyboard channel.
450 return input_samples_per_channel_;
451}
452
andrew@webrtc.orged083d42011-09-19 15:28:51 +0000453// TODO(andrew): Do deinterleaving and mixing in one step?
454void AudioBuffer::DeinterleaveFrom(AudioFrame* frame) {
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000455 assert(proc_samples_per_channel_ == input_samples_per_channel_);
andrew@webrtc.org30be8272014-09-24 20:06:23 +0000456 assert(frame->num_channels_ == num_input_channels_);
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000457 assert(frame->samples_per_channel_ == proc_samples_per_channel_);
458 InitForNewData();
andrew@webrtc.org63a50982012-05-02 23:56:37 +0000459 activity_ = frame->vad_activity_;
niklase@google.com470e71d2011-07-07 08:21:25 +0000460
andrew@webrtc.org30be8272014-09-24 20:06:23 +0000461 if (num_input_channels_ == 2 && num_proc_channels_ == 1) {
462 // Downmix directly; no explicit deinterleaving needed.
463 int16_t* downmixed = channels_->ibuf()->channel(0);
464 for (int i = 0; i < input_samples_per_channel_; ++i) {
andrew@webrtc.org8328e7c2014-10-31 04:58:14 +0000465 downmixed[i] = (frame->data_[i * 2] + frame->data_[i * 2 + 1]) / 2;
andrew@webrtc.org30be8272014-09-24 20:06:23 +0000466 }
467 } else {
468 assert(num_proc_channels_ == num_input_channels_);
469 int16_t* interleaved = frame->data_;
470 for (int i = 0; i < num_proc_channels_; ++i) {
471 int16_t* deinterleaved = channels_->ibuf()->channel(i);
472 int interleaved_idx = i;
473 for (int j = 0; j < proc_samples_per_channel_; ++j) {
474 deinterleaved[j] = interleaved[interleaved_idx];
475 interleaved_idx += num_proc_channels_;
476 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000477 }
478 }
479}
480
andrew@webrtc.org755b04a2011-11-15 16:57:56 +0000481void AudioBuffer::InterleaveTo(AudioFrame* frame, bool data_changed) const {
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000482 assert(proc_samples_per_channel_ == output_samples_per_channel_);
483 assert(num_proc_channels_ == num_input_channels_);
484 assert(frame->num_channels_ == num_proc_channels_);
485 assert(frame->samples_per_channel_ == proc_samples_per_channel_);
andrew@webrtc.org63a50982012-05-02 23:56:37 +0000486 frame->vad_activity_ = activity_;
niklase@google.com470e71d2011-07-07 08:21:25 +0000487
andrew@webrtc.org755b04a2011-11-15 16:57:56 +0000488 if (!data_changed) {
489 return;
490 }
491
andrew@webrtc.org63a50982012-05-02 23:56:37 +0000492 int16_t* interleaved = frame->data_;
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000493 for (int i = 0; i < num_proc_channels_; i++) {
mflodman@webrtc.orgd5da2502014-05-15 11:17:21 +0000494 int16_t* deinterleaved = channels_->ibuf()->channel(i);
andrew@webrtc.orged083d42011-09-19 15:28:51 +0000495 int interleaved_idx = i;
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000496 for (int j = 0; j < proc_samples_per_channel_; j++) {
niklase@google.com470e71d2011-07-07 08:21:25 +0000497 interleaved[interleaved_idx] = deinterleaved[j];
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000498 interleaved_idx += num_proc_channels_;
niklase@google.com470e71d2011-07-07 08:21:25 +0000499 }
500 }
501}
502
niklase@google.com470e71d2011-07-07 08:21:25 +0000503void AudioBuffer::CopyLowPassToReference() {
504 reference_copied_ = true;
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000505 if (!low_pass_reference_channels_.get()) {
506 low_pass_reference_channels_.reset(
507 new ChannelBuffer<int16_t>(samples_per_split_channel_,
508 num_proc_channels_));
509 }
510 for (int i = 0; i < num_proc_channels_; i++) {
511 low_pass_reference_channels_->CopyFrom(low_pass_split_data(i), i);
niklase@google.com470e71d2011-07-07 08:21:25 +0000512 }
513}
andrew@webrtc.orgddbb8a22014-04-22 21:00:04 +0000514
aluebs@webrtc.orgbe05c742014-11-14 22:18:10 +0000515void AudioBuffer::SplitIntoFrequencyBands() {
516 splitting_filter_->TwoBandsAnalysis(
517 channels(), samples_per_channel(), num_proc_channels_,
518 low_pass_split_channels(), high_pass_split_channels());
519}
520
521void AudioBuffer::MergeFrequencyBands() {
522 splitting_filter_->TwoBandsSynthesis(
523 low_pass_split_channels(), high_pass_split_channels(),
524 samples_per_split_channel(), num_proc_channels_, channels());
525}
526
niklase@google.com470e71d2011-07-07 08:21:25 +0000527} // namespace webrtc