blob: 2fae686d28cb9bd8ec90beb78d0e64bc48156b16 [file] [log] [blame]
henrika86d907c2015-09-07 16:09:50 +02001/*
2 * Copyright (c) 2013 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_device/fine_audio_buffer.h"
henrika86d907c2015-09-07 16:09:50 +020012
13#include <memory.h>
14#include <stdio.h>
15#include <algorithm>
16
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020017#include "modules/audio_device/audio_device_buffer.h"
18#include "rtc_base/checks.h"
19#include "rtc_base/logging.h"
henrika86d907c2015-09-07 16:09:50 +020020
21namespace webrtc {
22
23FineAudioBuffer::FineAudioBuffer(AudioDeviceBuffer* device_buffer,
henrikabb6f7522017-05-30 02:01:30 -070024 int sample_rate,
25 size_t capacity)
henrika86d907c2015-09-07 16:09:50 +020026 : device_buffer_(device_buffer),
henrika86d907c2015-09-07 16:09:50 +020027 sample_rate_(sample_rate),
28 samples_per_10_ms_(static_cast<size_t>(sample_rate_ * 10 / 1000)),
henrikabb6f7522017-05-30 02:01:30 -070029 playout_buffer_(0, capacity),
30 record_buffer_(0, capacity) {
henrika883d00f2018-03-16 10:09:49 +010031 RTC_LOG(INFO) << "samples_per_10_ms_: " << samples_per_10_ms_;
henrika86d907c2015-09-07 16:09:50 +020032}
33
34FineAudioBuffer::~FineAudioBuffer() {}
35
henrika86d907c2015-09-07 16:09:50 +020036void FineAudioBuffer::ResetPlayout() {
henrikab3ebc1a2017-02-27 05:14:17 -080037 playout_buffer_.Clear();
henrika86d907c2015-09-07 16:09:50 +020038}
39
40void FineAudioBuffer::ResetRecord() {
henrikaf166e1b2017-02-23 02:44:55 -080041 record_buffer_.Clear();
henrika86d907c2015-09-07 16:09:50 +020042}
43
henrika8d7393b2018-04-19 13:40:15 +020044void FineAudioBuffer::GetPlayoutData(rtc::ArrayView<int16_t> audio_buffer,
henrika883d00f2018-03-16 10:09:49 +010045 int playout_delay_ms) {
henrikab3ebc1a2017-02-27 05:14:17 -080046 // Ask WebRTC for new data in chunks of 10ms until we have enough to
47 // fulfill the request. It is possible that the buffer already contains
48 // enough samples from the last round.
henrika8d7393b2018-04-19 13:40:15 +020049 while (playout_buffer_.size() < audio_buffer.size()) {
henrikab3ebc1a2017-02-27 05:14:17 -080050 // Get 10ms decoded audio from WebRTC.
henrika86d907c2015-09-07 16:09:50 +020051 device_buffer_->RequestPlayoutData(samples_per_10_ms_);
henrikab3ebc1a2017-02-27 05:14:17 -080052 // Append |bytes_per_10_ms_| elements to the end of the buffer.
henrika8d7393b2018-04-19 13:40:15 +020053 const size_t samples_written = playout_buffer_.AppendData(
54 samples_per_10_ms_, [&](rtc::ArrayView<int16_t> buf) {
henrikab3ebc1a2017-02-27 05:14:17 -080055 const size_t samples_per_channel =
56 device_buffer_->GetPlayoutData(buf.data());
57 // TODO(henrika): this class is only used on mobile devices and is
58 // currently limited to mono. Modifications are needed for stereo.
henrika8d7393b2018-04-19 13:40:15 +020059 return samples_per_channel;
henrikab3ebc1a2017-02-27 05:14:17 -080060 });
henrika8d7393b2018-04-19 13:40:15 +020061 RTC_DCHECK_EQ(samples_per_10_ms_, samples_written);
henrika86d907c2015-09-07 16:09:50 +020062 }
henrika8d7393b2018-04-19 13:40:15 +020063
64 const size_t num_bytes = audio_buffer.size() * sizeof(int16_t);
henrikab3ebc1a2017-02-27 05:14:17 -080065 // Provide the requested number of bytes to the consumer.
henrikabb6f7522017-05-30 02:01:30 -070066 memcpy(audio_buffer.data(), playout_buffer_.data(), num_bytes);
henrikab3ebc1a2017-02-27 05:14:17 -080067 // Move remaining samples to start of buffer to prepare for next round.
henrika8d7393b2018-04-19 13:40:15 +020068 memmove(playout_buffer_.data(), playout_buffer_.data() + audio_buffer.size(),
69 (playout_buffer_.size() - audio_buffer.size()) * sizeof(int16_t));
70 playout_buffer_.SetSize(playout_buffer_.size() - audio_buffer.size());
henrika883d00f2018-03-16 10:09:49 +010071 // Cache playout latency for usage in DeliverRecordedData();
72 playout_delay_ms_ = playout_delay_ms;
henrika86d907c2015-09-07 16:09:50 +020073}
74
henrikabb6f7522017-05-30 02:01:30 -070075void FineAudioBuffer::DeliverRecordedData(
henrika8d7393b2018-04-19 13:40:15 +020076 rtc::ArrayView<const int16_t> audio_buffer,
henrikabb6f7522017-05-30 02:01:30 -070077 int record_delay_ms) {
henrikaf166e1b2017-02-23 02:44:55 -080078 // Always append new data and grow the buffer if needed.
henrikabb6f7522017-05-30 02:01:30 -070079 record_buffer_.AppendData(audio_buffer.data(), audio_buffer.size());
henrikaf166e1b2017-02-23 02:44:55 -080080 // Consume samples from buffer in chunks of 10ms until there is not
henrika8d7393b2018-04-19 13:40:15 +020081 // enough data left. The number of remaining samples in the cache is given by
henrikaf166e1b2017-02-23 02:44:55 -080082 // the new size of the buffer.
henrika8d7393b2018-04-19 13:40:15 +020083 while (record_buffer_.size() >= samples_per_10_ms_) {
henrikaf166e1b2017-02-23 02:44:55 -080084 device_buffer_->SetRecordedBuffer(record_buffer_.data(),
85 samples_per_10_ms_);
henrika883d00f2018-03-16 10:09:49 +010086 device_buffer_->SetVQEData(playout_delay_ms_, record_delay_ms);
henrika86d907c2015-09-07 16:09:50 +020087 device_buffer_->DeliverRecordedData();
henrika8d7393b2018-04-19 13:40:15 +020088 memmove(record_buffer_.data(), record_buffer_.data() + samples_per_10_ms_,
89 (record_buffer_.size() - samples_per_10_ms_) * sizeof(int16_t));
90 record_buffer_.SetSize(record_buffer_.size() - samples_per_10_ms_);
henrika86d907c2015-09-07 16:09:50 +020091 }
92}
93
94} // namespace webrtc