// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "media/base/audio_pull_fifo.h"
#include <algorithm>
#include "base/logging.h"
#include "media/base/audio_bus.h"
namespace media {
AudioPullFifo::AudioPullFifo(int channels, int frames, const ReadCB& read_cb)
: read_cb_(read_cb),
fifo_(AudioBus::Create(channels, frames)),
fifo_index_(frames) {}
AudioPullFifo::~AudioPullFifo() {}
void AudioPullFifo::Consume(AudioBus* destination, int frames_to_consume) {
DCHECK_LE(frames_to_consume, destination->frames());
int remaining_frames_to_provide = frames_to_consume;
// Try to fulfill the request using what's available in the FIFO.
int frames_read = ReadFromFifo(destination, remaining_frames_to_provide, 0);
int write_pos = frames_read;
remaining_frames_to_provide -= frames_read;
// Get the remaining audio frames from the producer using the callback.
while (remaining_frames_to_provide > 0) {
DCHECK_EQ(fifo_index_, fifo_->frames());
fifo_index_ = 0;
// Fill up the FIFO by acquiring audio data from the producer.
read_cb_.Run(write_pos, fifo_.get());
// Try to fulfill the request using what's available in the FIFO.
frames_read =
ReadFromFifo(destination, remaining_frames_to_provide, write_pos);
write_pos += frames_read;
remaining_frames_to_provide -= frames_read;
}
}
void AudioPullFifo::Clear() { fifo_index_ = fifo_->frames(); }
int AudioPullFifo::ReadFromFifo(AudioBus* destination,
int frames_to_provide,
int write_pos) {
int frames = std::min(frames_to_provide, fifo_->frames() - fifo_index_);
if (frames <= 0)
return 0;
for (int ch = 0; ch < fifo_->channels(); ++ch) {
const float* src = fifo_->channel(ch) + fifo_index_;
float* dest = destination->channel(ch) + write_pos;
memcpy(dest, src, frames * sizeof(*src));
}
fifo_index_ += frames;
return frames;
}
} // namespace media