Chromium Code Reviews| Index: media/audio/audio_output_dispatcher_impl.cc |
| =================================================================== |
| --- media/audio/audio_output_dispatcher_impl.cc (revision 0) |
| +++ media/audio/audio_output_dispatcher_impl.cc (revision 0) |
| @@ -0,0 +1,195 @@ |
| +// 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/audio/audio_output_dispatcher_impl.h" |
| + |
| +#include <algorithm> |
| + |
| +#include "base/bind.h" |
| +#include "base/compiler_specific.h" |
| +#include "base/message_loop.h" |
| +#include "base/time.h" |
| +#include "media/audio/audio_io.h" |
| +#include "media/audio/audio_output_proxy.h" |
| +#include "media/audio/audio_util.h" |
| + |
| +AudioOutputDispatcherImpl::AudioOutputDispatcherImpl( |
| + AudioManager* audio_manager, |
| + const AudioParameters& params, |
| + const base::TimeDelta& close_delay) |
| + : AudioOutputDispatcher(audio_manager, params), |
| + pause_delay_(base::TimeDelta::FromMilliseconds( |
| + 2 * params.frames_per_buffer() * |
| + base::Time::kMillisecondsPerSecond / params.sample_rate())), |
| + paused_proxies_(0), |
| + ALLOW_THIS_IN_INITIALIZER_LIST(weak_this_(this)), |
| + close_timer_(FROM_HERE, |
| + close_delay, |
| + weak_this_.GetWeakPtr(), |
| + &AudioOutputDispatcherImpl::ClosePendingStreams) { |
| +} |
| + |
| +AudioOutputDispatcherImpl::~AudioOutputDispatcherImpl() { |
| + DCHECK(proxy_to_physical_map_.empty()); |
| +} |
| + |
| +bool AudioOutputDispatcherImpl::OpenStream() { |
| + DCHECK_EQ(MessageLoop::current(), message_loop_); |
| + |
| + paused_proxies_++; |
| + |
| + // Ensure that there is at least one open stream. |
| + if (idle_streams_.empty() && !CreateAndOpenStream()) |
| + return false; |
| + |
| + close_timer_.Reset(); |
| + return true; |
| +} |
| + |
| +bool AudioOutputDispatcherImpl::StartStream( |
| + AudioOutputStream::AudioSourceCallback* callback, |
| + AudioOutputProxy* stream_proxy) { |
| + DCHECK_EQ(MessageLoop::current(), message_loop_); |
| + |
| + if (idle_streams_.empty() && !CreateAndOpenStream()) |
| + return false; |
| + |
| + AudioOutputStream* physical_stream = idle_streams_.back(); |
| + DCHECK_NE(static_cast<AudioOutputStream*>(NULL), physical_stream); |
|
vrk (LEFT CHROMIUM)
2012/04/06 23:37:05
DCHECK(physical_stream)?
enal1
2012/04/16 22:01:35
Done.
|
| + idle_streams_.pop_back(); |
| + |
| + DCHECK_GT(paused_proxies_, 0u); |
| + --paused_proxies_; |
| + |
| + close_timer_.Reset(); |
| + |
| + // Schedule task to allocate streams for other proxies if we need to. |
| + message_loop_->PostTask(FROM_HERE, base::Bind( |
| + &AudioOutputDispatcherImpl::OpenTask, weak_this_.GetWeakPtr())); |
| + |
| + double volume = 0; |
| + stream_proxy->GetVolume(&volume); |
| + physical_stream->SetVolume(volume); |
| + physical_stream->Start(callback); |
| + proxy_to_physical_map_[stream_proxy] = physical_stream; |
| + return true; |
| +} |
| + |
| +void AudioOutputDispatcherImpl::StopStream(AudioOutputProxy* stream_proxy) { |
| + DCHECK_EQ(MessageLoop::current(), message_loop_); |
| + |
| + AudioStreamMap::iterator it = proxy_to_physical_map_.find(stream_proxy); |
| + DCHECK(it != proxy_to_physical_map_.end()); |
| + AudioOutputStream* physical_stream = it->second; |
| + proxy_to_physical_map_.erase(it); |
| + |
| + DCHECK_NE(static_cast<AudioOutputStream*>(NULL), physical_stream); |
|
vrk (LEFT CHROMIUM)
2012/04/06 23:37:05
Remove DCHECK: doing a null DCHECK before usage is
enal1
2012/04/16 22:01:35
Done.
|
| + physical_stream->Stop(); |
| + |
| + ++paused_proxies_; |
| + |
| + pausing_streams_.push_front(physical_stream); |
| + |
| + // Don't recycle stream until two buffers worth of time has elapsed. |
| + message_loop_->PostDelayedTask( |
| + FROM_HERE, |
| + base::Bind(&AudioOutputDispatcherImpl::StopStreamTask, |
| + weak_this_.GetWeakPtr()), |
| + pause_delay_); |
| +} |
| + |
| +void AudioOutputDispatcherImpl::StreamVolumeSet(AudioOutputProxy* stream_proxy, |
| + double volume) { |
| + DCHECK_EQ(MessageLoop::current(), message_loop_); |
| + AudioStreamMap::iterator it = proxy_to_physical_map_.find(stream_proxy); |
| + if (it != proxy_to_physical_map_.end()) { |
| + AudioOutputStream* physical_stream = it->second; |
| + physical_stream->SetVolume(volume); |
| + } |
| +} |
| + |
| +void AudioOutputDispatcherImpl::StopStreamTask() { |
| + DCHECK_EQ(MessageLoop::current(), message_loop_); |
| + |
| + if (pausing_streams_.empty()) |
| + return; |
| + |
| + AudioOutputStream* stream = pausing_streams_.back(); |
| + pausing_streams_.pop_back(); |
| + idle_streams_.push_back(stream); |
| + close_timer_.Reset(); |
| +} |
| + |
| +void AudioOutputDispatcherImpl::CloseStream(AudioOutputProxy* stream_proxy) { |
| + DCHECK_EQ(MessageLoop::current(), message_loop_); |
| + |
| + while (!pausing_streams_.empty()) { |
| + idle_streams_.push_back(pausing_streams_.back()); |
| + pausing_streams_.pop_back(); |
| + } |
| + |
| + DCHECK_GT(paused_proxies_, 0u); |
| + paused_proxies_--; |
| + |
| + while (idle_streams_.size() > paused_proxies_) { |
| + idle_streams_.back()->Close(); |
| + idle_streams_.pop_back(); |
| + } |
| +} |
| + |
| +void AudioOutputDispatcherImpl::Shutdown() { |
| + DCHECK_EQ(MessageLoop::current(), message_loop_); |
| + |
| + // Cancel any pending tasks to close paused streams or create new ones. |
| + weak_this_.InvalidateWeakPtrs(); |
| + |
| + // No AudioOutputProxy objects should hold a reference to us when we get |
| + // to this stage. |
| + DCHECK(HasOneRef()) << "Only the AudioManager should hold a reference"; |
| + |
| + AudioOutputStreamList::iterator it = idle_streams_.begin(); |
| + for (; it != idle_streams_.end(); ++it) |
| + (*it)->Close(); |
| + idle_streams_.clear(); |
| + |
| + it = pausing_streams_.begin(); |
| + for (; it != pausing_streams_.end(); ++it) |
| + (*it)->Close(); |
| + pausing_streams_.clear(); |
|
vrk (LEFT CHROMIUM)
2012/04/06 23:37:05
Do you need to clear the proxy_to_physical_map_ so
enal1
2012/04/16 22:01:35
StopStream() removes physical stream from the map.
|
| +} |
| + |
| +bool AudioOutputDispatcherImpl::CreateAndOpenStream() { |
| + DCHECK_EQ(MessageLoop::current(), message_loop_); |
| + AudioOutputStream* stream = audio_manager_->MakeAudioOutputStream(params_); |
| + if (!stream) |
| + return false; |
| + |
| + if (!stream->Open()) { |
| + stream->Close(); |
| + return false; |
| + } |
| + idle_streams_.push_back(stream); |
| + return true; |
| +} |
| + |
| +void AudioOutputDispatcherImpl::OpenTask() { |
| + DCHECK_EQ(MessageLoop::current(), message_loop_); |
| + // Make sure that we have at least one stream allocated if there |
| + // are paused streams. |
| + if (paused_proxies_ > 0 && idle_streams_.empty() && |
| + pausing_streams_.empty()) { |
| + CreateAndOpenStream(); |
| + } |
| + |
| + close_timer_.Reset(); |
| +} |
| + |
| +// This method is called by |close_timer_|. |
| +void AudioOutputDispatcherImpl::ClosePendingStreams() { |
| + DCHECK_EQ(MessageLoop::current(), message_loop_); |
| + while (!idle_streams_.empty()) { |
| + idle_streams_.back()->Close(); |
| + idle_streams_.pop_back(); |
| + } |
| +} |