Chromium Code Reviews| Index: media/audio/audio_output_dispatcher.cc |
| diff --git a/media/audio/audio_output_dispatcher.cc b/media/audio/audio_output_dispatcher.cc |
| index 3f9d848a2c9d1762155e7062aee36f479d6563d0..ee66b6b1efe338c469e2150f03457b2789c614b8 100644 |
| --- a/media/audio/audio_output_dispatcher.cc |
| +++ b/media/audio/audio_output_dispatcher.cc |
| @@ -1,4 +1,4 @@ |
| -// Copyright (c) 2010 The Chromium Authors. All rights reserved. |
| +// Copyright (c) 2011 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. |
| @@ -6,6 +6,7 @@ |
| #include "base/compiler_specific.h" |
| #include "base/message_loop.h" |
| +#include "base/time.h" |
| #include "media/audio/audio_io.h" |
| AudioOutputDispatcher::AudioOutputDispatcher( |
| @@ -14,6 +15,8 @@ AudioOutputDispatcher::AudioOutputDispatcher( |
| : audio_manager_(audio_manager), |
| message_loop_(audio_manager->GetMessageLoop()), |
| params_(params), |
| + pause_delay_milliseconds_(2 * params.samples_per_packet * |
| + base::Time::kMillisecondsPerSecond / params.sample_rate), |
| paused_proxies_(0), |
| ALLOW_THIS_IN_INITIALIZER_LIST(close_timer_( |
| base::TimeDelta::FromMilliseconds(close_delay_ms), |
| @@ -28,7 +31,7 @@ bool AudioOutputDispatcher::StreamOpened() { |
| paused_proxies_++; |
| // Ensure that there is at least one open stream. |
| - if (streams_.empty() && !CreateAndOpenStream()) { |
| + if (idle_streams_.empty() && !CreateAndOpenStream()) { |
| return false; |
| } |
| @@ -40,12 +43,12 @@ bool AudioOutputDispatcher::StreamOpened() { |
| AudioOutputStream* AudioOutputDispatcher::StreamStarted() { |
| DCHECK_EQ(MessageLoop::current(), message_loop_); |
| - if (streams_.empty() && !CreateAndOpenStream()) { |
| + if (idle_streams_.empty() && !CreateAndOpenStream()) { |
| return NULL; |
| } |
| - AudioOutputStream* stream = streams_.back(); |
| - streams_.pop_back(); |
| + AudioOutputStream* stream = idle_streams_.back(); |
| + idle_streams_.pop_back(); |
| DCHECK_GT(paused_proxies_, 0u); |
| paused_proxies_--; |
| @@ -61,20 +64,41 @@ AudioOutputStream* AudioOutputDispatcher::StreamStarted() { |
| void AudioOutputDispatcher::StreamStopped(AudioOutputStream* stream) { |
| DCHECK_EQ(MessageLoop::current(), message_loop_); |
| + |
| paused_proxies_++; |
| - streams_.push_back(stream); |
| + |
| + pausing_streams_.push_front(stream); |
| close_timer_.Reset(); |
| + |
| + // Don't recycle stream until two buffers worth of time has elapsed. |
| + message_loop_->PostDelayedTask( |
| + FROM_HERE, |
| + NewRunnableMethod(this, &AudioOutputDispatcher::StopStreamTask), |
| + pause_delay_milliseconds_); |
| +} |
| + |
| +void AudioOutputDispatcher::StopStreamTask() { |
| + if (pausing_streams_.empty()) |
| + return; |
| + AudioOutputStream* stream = pausing_streams_.back(); |
| + pausing_streams_.pop_back(); |
| + idle_streams_.push_back(stream); |
| } |
| void AudioOutputDispatcher::StreamClosed() { |
| DCHECK_EQ(MessageLoop::current(), message_loop_); |
|
davejcool
2011/04/14 00:43:26
When a stream is finally closed, it should be adde
|
| + while (!pausing_streams_.empty()) { |
| + idle_streams_.push_back(pausing_streams_.back()); |
|
Sergey Ulanov
2011/04/14 18:05:09
I don't like this. Basically when a single stream
davejcool
2011/04/15 01:12:51
Closing the pausing_streams_ later can not be done
|
| + pausing_streams_.pop_back(); |
| + } |
| + |
| DCHECK_GT(paused_proxies_, 0u); |
| paused_proxies_--; |
| - while (streams_.size() > paused_proxies_) { |
| - streams_.back()->Close(); |
| - streams_.pop_back(); |
| + while (idle_streams_.size() > paused_proxies_) { |
| + idle_streams_.back()->Close(); |
| + idle_streams_.pop_back(); |
| } |
| } |
| @@ -92,14 +116,15 @@ bool AudioOutputDispatcher::CreateAndOpenStream() { |
| stream->Close(); |
| return false; |
| } |
| - streams_.push_back(stream); |
| + idle_streams_.push_back(stream); |
| return true; |
| } |
| void AudioOutputDispatcher::OpenTask() { |
| // Make sure that we have at least one stream allocated if there |
| // are paused streams. |
| - if (paused_proxies_ > 0 && streams_.empty()) { |
| + if (paused_proxies_ > 0 && idle_streams_.empty() |
| + && pausing_streams_.empty()) { |
|
scherkus (not reviewing)
2011/04/14 17:11:08
nit: && goes on end of previous line
|
| CreateAndOpenStream(); |
| } |
| @@ -108,8 +133,8 @@ void AudioOutputDispatcher::OpenTask() { |
| // This method is called by |close_timer_|. |
| void AudioOutputDispatcher::ClosePendingStreams() { |
| - while (!streams_.empty()) { |
| - streams_.back()->Close(); |
| - streams_.pop_back(); |
| + while (!idle_streams_.empty()) { |
| + idle_streams_.back()->Close(); |
| + idle_streams_.pop_back(); |
| } |
| } |