// 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/location.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; } OutputDevice* AudioRendererMixerInput::GetOutputDevice() { DVLOG(1) << __FUNCTION__; return mixer_->GetOutputDevice(); } double AudioRendererMixerInput::ProvideInput(AudioBus* audio_bus, base::TimeDelta buffer_delay) { int frames_filled = callback_->Render( audio_bus, static_cast(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