blob: ab9f0a7ecab20e0d2c6ddae12fc6bcc690009960 [file] [log] [blame]
// 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_renderer_mixer_input.h"
#include "base/bind.h"
#include "base/logging.h"
#include "media/base/audio_renderer_mixer.h"
namespace media {
AudioRendererMixerInput::AudioRendererMixerInput(
const GetMixerCB& get_mixer_cb, const RemoveMixerCB& remove_mixer_cb)
: playing_(false),
initialized_(false),
volume_(1.0f),
get_mixer_cb_(get_mixer_cb),
remove_mixer_cb_(remove_mixer_cb),
mixer_(NULL),
callback_(NULL),
error_cb_(base::Bind(
&AudioRendererMixerInput::OnRenderError, base::Unretained(this))) {
}
AudioRendererMixerInput::~AudioRendererMixerInput() {
DCHECK(!playing_);
DCHECK(!mixer_);
}
void AudioRendererMixerInput::Initialize(
const AudioParameters& params,
AudioRendererSink::RenderCallback* callback) {
DCHECK(callback);
DCHECK(!initialized_);
params_ = params;
callback_ = callback;
initialized_ = true;
}
void AudioRendererMixerInput::Start() {
DCHECK(initialized_);
DCHECK(!playing_);
DCHECK(!mixer_);
mixer_ = get_mixer_cb_.Run(params_);
// Note: OnRenderError() may be called immediately after this call returns.
mixer_->AddErrorCallback(error_cb_);
}
void AudioRendererMixerInput::Stop() {
// Stop() may be called at any time, if Pause() hasn't been called we need to
// remove our mixer input before shutdown.
if (playing_) {
mixer_->RemoveMixerInput(this);
playing_ = false;
}
if (mixer_) {
// TODO(dalecurtis): This is required so that |callback_| isn't called after
// Stop() by an error event since it may outlive this ref-counted object. We
// should instead have sane ownership semantics: http://crbug.com/151051
mixer_->RemoveErrorCallback(error_cb_);
remove_mixer_cb_.Run(params_);
mixer_ = NULL;
}
}
void AudioRendererMixerInput::Play() {
DCHECK(initialized_);
DCHECK(mixer_);
if (playing_)
return;
mixer_->AddMixerInput(this);
playing_ = true;
}
void AudioRendererMixerInput::Pause() {
DCHECK(initialized_);
DCHECK(mixer_);
if (!playing_)
return;
mixer_->RemoveMixerInput(this);
playing_ = false;
}
bool AudioRendererMixerInput::SetVolume(double volume) {
volume_ = volume;
return true;
}
double AudioRendererMixerInput::ProvideInput(AudioBus* audio_bus,
base::TimeDelta buffer_delay) {
int frames_filled = callback_->Render(
audio_bus, static_cast<int>(buffer_delay.InMillisecondsF() + 0.5));
// AudioConverter expects unfilled frames to be zeroed.
if (frames_filled < audio_bus->frames()) {
audio_bus->ZeroFramesPartial(
frames_filled, audio_bus->frames() - frames_filled);
}
return frames_filled > 0 ? volume_ : 0;
}
void AudioRendererMixerInput::OnRenderError() {
callback_->OnRenderError();
}
} // namespace media