summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorsergeyu@chromium.org <sergeyu@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-09-17 22:51:16 +0000
committersergeyu@chromium.org <sergeyu@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2010-09-17 22:51:16 +0000
commit7307035b2c12088ba241dc49098204d4beff52d4 (patch)
tree9511385010f72b15bf355f0e3c0a841257a87077
parent8e4040197b15b7af9d1510e47a7fea1d72ec6a06 (diff)
downloadchromium_src-7307035b2c12088ba241dc49098204d4beff52d4.zip
chromium_src-7307035b2c12088ba241dc49098204d4beff52d4.tar.gz
chromium_src-7307035b2c12088ba241dc49098204d4beff52d4.tar.bz2
Make AudioOutputController.Close() truly asynchronous.
Added closed_task parameter in this method. This parameter is used to notify the caller when the stream is actually closed. Callbacks may be called until closed_task is executed. BUG=55755 TEST=Unittests, audio still works, doesn't crash Review URL: http://codereview.chromium.org/3415007 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@59869 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r--chrome/browser/renderer_host/audio_renderer_host.cc65
-rw-r--r--chrome/browser/renderer_host/audio_renderer_host.h13
-rw-r--r--chrome/browser/renderer_host/audio_renderer_host_unittest.cc30
-rw-r--r--media/audio/audio_output_controller.cc102
-rw-r--r--media/audio/audio_output_controller.h15
-rw-r--r--media/audio/audio_output_controller_unittest.cc45
6 files changed, 154 insertions, 116 deletions
diff --git a/chrome/browser/renderer_host/audio_renderer_host.cc b/chrome/browser/renderer_host/audio_renderer_host.cc
index 3219333..b4ba6ba 100644
--- a/chrome/browser/renderer_host/audio_renderer_host.cc
+++ b/chrome/browser/renderer_host/audio_renderer_host.cc
@@ -335,7 +335,14 @@ void AudioRendererHost::OnCreateStream(
}
scoped_ptr<AudioEntry> entry(new AudioEntry());
- scoped_refptr<media::AudioOutputController> controller = NULL;
+ // Create the shared memory and share with the renderer process.
+ if (!entry->shared_memory.Create(L"", false, false, hardware_packet_size) ||
+ !entry->shared_memory.Map(entry->shared_memory.max_size())) {
+ // If creation of shared memory failed then send an error message.
+ SendErrorMessage(msg.routing_id(), stream_id);
+ return;
+ }
+
if (low_latency) {
// If this is the low latency mode, we need to construct a SyncReader first.
scoped_ptr<AudioSyncReader> reader(
@@ -350,44 +357,32 @@ void AudioRendererHost::OnCreateStream(
// If we have successfully created the SyncReader then assign it to the
// entry and construct an AudioOutputController.
entry->reader.reset(reader.release());
- controller =
+ entry->controller =
media::AudioOutputController::CreateLowLatency(
this, params.params,
hardware_packet_size,
entry->reader.get());
} else {
// The choice of buffer capacity is based on experiment.
- controller =
+ entry->controller =
media::AudioOutputController::Create(this, params.params,
hardware_packet_size,
3 * hardware_packet_size);
}
- if (!controller) {
+ if (!entry->controller) {
SendErrorMessage(msg.routing_id(), stream_id);
return;
}
// If we have created the controller successfully create a entry and add it
// to the map.
- entry->controller = controller;
entry->render_view_id = msg.routing_id();
entry->stream_id = stream_id;
- // Create the shared memory and share with the renderer process.
- if (!entry->shared_memory.Create(L"", false, false, hardware_packet_size) ||
- !entry->shared_memory.Map(entry->shared_memory.max_size())) {
- // If creation of shared memory failed then close the controller and
- // sends an error message.
- controller->Close();
- SendErrorMessage(msg.routing_id(), stream_id);
- return;
- }
-
- // If everything is successful then add it to the map.
- audio_entries_.insert(std::make_pair(
- AudioEntryId(msg.routing_id(), stream_id),
- entry.release()));
+ audio_entries_.insert(std::make_pair(
+ AudioEntryId(msg.routing_id(), stream_id),
+ entry.release()));
}
void AudioRendererHost::OnPlayStream(const IPC::Message& msg, int stream_id) {
@@ -431,10 +426,8 @@ void AudioRendererHost::OnCloseStream(const IPC::Message& msg, int stream_id) {
AudioEntry* entry = LookupById(msg.routing_id(), stream_id);
- // Note that closing an audio stream is a blocking operation. This call may
- // block the IO thread for up to 100ms.
if (entry)
- DeleteEntry(entry);
+ CloseAndDeleteStream(entry);
}
void AudioRendererHost::OnSetVolume(const IPC::Message& msg, int stream_id,
@@ -519,10 +512,25 @@ void AudioRendererHost::SendErrorMessage(int32 render_view_id,
void AudioRendererHost::DeleteEntries() {
DCHECK(ChromeThread::CurrentlyOn(ChromeThread::IO));
- while (!audio_entries_.empty()) {
- DeleteEntry(audio_entries_.begin()->second);
+ for (AudioEntryMap::iterator i = audio_entries_.begin();
+ i != audio_entries_.end(); ++i) {
+ CloseAndDeleteStream(i->second);
}
- DCHECK(audio_entries_.empty());
+}
+
+void AudioRendererHost::CloseAndDeleteStream(AudioEntry* entry) {
+ if (!entry->pending_close) {
+ entry->controller->Close(
+ NewRunnableMethod(this, &AudioRendererHost::OnStreamClosed, entry));
+ entry->pending_close = true;
+ }
+}
+
+void AudioRendererHost::OnStreamClosed(AudioEntry* entry) {
+ // Delete the entry after we've closed the stream.
+ ChromeThread::PostTask(
+ ChromeThread::IO, FROM_HERE,
+ NewRunnableMethod(this, &AudioRendererHost::DeleteEntry, entry));
}
void AudioRendererHost::DeleteEntry(AudioEntry* entry) {
@@ -531,10 +539,7 @@ void AudioRendererHost::DeleteEntry(AudioEntry* entry) {
// Delete the entry when this method goes out of scope.
scoped_ptr<AudioEntry> entry_deleter(entry);
- // Close the audio stream then remove the entry.
- entry->controller->Close();
-
- // Entry the entry from the map.
+ // Erase the entry from the map.
audio_entries_.erase(
AudioEntryId(entry->render_view_id, entry->stream_id));
}
@@ -545,7 +550,7 @@ void AudioRendererHost::DeleteEntryOnError(AudioEntry* entry) {
// Sends the error message first before we close the stream because
// |entry| is destroyed in DeleteEntry().
SendErrorMessage(entry->render_view_id, entry->stream_id);
- DeleteEntry(entry);
+ CloseAndDeleteStream(entry);
}
AudioRendererHost::AudioEntry* AudioRendererHost::LookupById(
diff --git a/chrome/browser/renderer_host/audio_renderer_host.h b/chrome/browser/renderer_host/audio_renderer_host.h
index 074ec66..c67e0e7 100644
--- a/chrome/browser/renderer_host/audio_renderer_host.h
+++ b/chrome/browser/renderer_host/audio_renderer_host.h
@@ -86,7 +86,8 @@ class AudioRendererHost : public base::RefCountedThreadSafe<
AudioEntry()
: render_view_id(0),
stream_id(0),
- pending_buffer_request(false) {
+ pending_buffer_request(false),
+ pending_close(false) {
}
// The AudioOutputController that manages the audio stream.
@@ -106,6 +107,9 @@ class AudioRendererHost : public base::RefCountedThreadSafe<
scoped_ptr<media::AudioOutputController::SyncReader> reader;
bool pending_buffer_request;
+
+ // Set to true after we called Close() for the controller.
+ bool pending_close;
};
typedef std::map<AudioEntryId, AudioEntry*> AudioEntryMap;
@@ -219,6 +223,13 @@ class AudioRendererHost : public base::RefCountedThreadSafe<
// Delete all audio entry and all audio streams
void DeleteEntries();
+ // Closes the stream. The stream is then deleted in DeleteEntry() after it
+ // is closed.
+ void CloseAndDeleteStream(AudioEntry* entry);
+
+ // Called on the audio thread after the audio stream is closed.
+ void OnStreamClosed(AudioEntry* entry);
+
// Delete an audio entry and close the related audio stream.
void DeleteEntry(AudioEntry* entry);
diff --git a/chrome/browser/renderer_host/audio_renderer_host_unittest.cc b/chrome/browser/renderer_host/audio_renderer_host_unittest.cc
index ad2fa7e..08734ee 100644
--- a/chrome/browser/renderer_host/audio_renderer_host_unittest.cc
+++ b/chrome/browser/renderer_host/audio_renderer_host_unittest.cc
@@ -12,6 +12,7 @@
#include "chrome/common/render_messages.h"
#include "chrome/common/render_messages_params.h"
#include "ipc/ipc_message_utils.h"
+#include "media/audio/audio_manager.h"
#include "media/audio/fake_audio_output_stream.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
@@ -192,7 +193,7 @@ class AudioRendererHostTest : public testing::Test {
host_ = NULL;
// We need to continue running message_loop_ to complete all destructions.
- message_loop_->RunAllPending();
+ SyncWithAudioThread();
io_thread_.reset();
}
@@ -309,17 +310,38 @@ class AudioRendererHostTest : public testing::Test {
CHECK(controller) << "AudioOutputController not found";
// Expect an error signal sent through IPC.
- EXPECT_CALL(*host_, OnStreamError(kRouteId, kStreamId))
- .WillOnce(QuitMessageLoop(message_loop_.get()));
+ EXPECT_CALL(*host_, OnStreamError(kRouteId, kStreamId));
// Simulate an error sent from the audio device.
host_->OnError(controller, 0);
- message_loop_->Run();
+ SyncWithAudioThread();
// Expect the audio stream record is removed.
EXPECT_EQ(0u, host_->audio_entries_.size());
}
+ // Called on the audio thread.
+ static void PostQuitMessageLoop(MessageLoop* message_loop) {
+ message_loop->PostTask(FROM_HERE, new MessageLoop::QuitTask());
+ }
+
+ // Called on the main thread.
+ static void PostQuitOnAudioThread(MessageLoop* message_loop) {
+ AudioManager::GetAudioManager()->GetMessageLoop()->PostTask(
+ FROM_HERE, NewRunnableFunction(&PostQuitMessageLoop, message_loop));
+ }
+
+ // SyncWithAudioThread() waits until all pending tasks on the audio thread
+ // are executed while also processing pending task in message_loop_ on the
+ // current thread. It is used to synchronize with the audio thread when we are
+ // closing an audio stream.
+ void SyncWithAudioThread() {
+ message_loop_->PostTask(
+ FROM_HERE, NewRunnableFunction(&PostQuitOnAudioThread,
+ message_loop_.get()));
+ message_loop_->Run();
+ }
+
MessageLoop* message_loop() { return message_loop_.get(); }
MockAudioRendererHost* host() { return host_; }
void EnableRealDevice() { mock_stream_ = false; }
diff --git a/media/audio/audio_output_controller.cc b/media/audio/audio_output_controller.cc
index 6ff02aa..439583c 100644
--- a/media/audio/audio_output_controller.cc
+++ b/media/audio/audio_output_controller.cc
@@ -115,18 +115,12 @@ void AudioOutputController::Flush() {
NewRunnableMethod(this, &AudioOutputController::DoFlush));
}
-void AudioOutputController::Close() {
- {
- AutoLock auto_lock(lock_);
- // Don't do anything if the stream is already closed.
- if (state_ == kClosed)
- return;
- state_ = kClosed;
- }
-
+void AudioOutputController::Close(Task* closed_task) {
+ DCHECK(closed_task);
+ DCHECK(message_loop_);
message_loop_->PostTask(
FROM_HERE,
- NewRunnableMethod(this, &AudioOutputController::DoClose));
+ NewRunnableMethod(this, &AudioOutputController::DoClose, closed_task));
}
void AudioOutputController::SetVolume(double volume) {
@@ -147,8 +141,6 @@ void AudioOutputController::DoCreate(AudioParameters params,
uint32 hardware_buffer_size) {
DCHECK_EQ(message_loop_, MessageLoop::current());
- AutoLock auto_lock(lock_);
-
// Close() can be called before DoCreate() is executed.
if (state_ == kClosed)
return;
@@ -180,6 +172,7 @@ void AudioOutputController::DoCreate(AudioParameters params,
// If in normal latency mode then start buffering.
if (!LowLatencyMode()) {
+ AutoLock auto_lock(lock_);
SubmitOnMoreData_Locked();
}
}
@@ -187,35 +180,25 @@ void AudioOutputController::DoCreate(AudioParameters params,
void AudioOutputController::DoPlay() {
DCHECK_EQ(message_loop_, MessageLoop::current());
- {
- AutoLock auto_lock(lock_);
- // We can start from created or paused state.
- if (state_ != kCreated && state_ != kPaused)
- return;
- state_ = kPlaying;
- }
+ // We can start from created or paused state.
+ if (state_ != kCreated && state_ != kPaused)
+ return;
+ state_ = kPlaying;
// We start the AudioOutputStream lazily.
stream_->Start(this);
- {
- AutoLock auto_lock(lock_);
- // Tell the event handler that we are now playing.
- if (state_ != kClosed)
- handler_->OnPlaying(this);
- }
+ // Tell the event handler that we are now playing.
+ handler_->OnPlaying(this);
}
void AudioOutputController::DoPause() {
DCHECK_EQ(message_loop_, MessageLoop::current());
- {
- AutoLock auto_lock(lock_);
- // We can pause from started state.
- if (state_ != kPlaying)
- return;
- state_ = kPaused;
- }
+ // We can pause from started state.
+ if (state_ != kPlaying)
+ return;
+ state_ = kPaused;
// Then we stop the audio device. This is not the perfect solution because
// it discards all the internal buffer in the audio device.
@@ -227,11 +210,7 @@ void AudioOutputController::DoPause() {
sync_reader_->UpdatePendingBytes(kPauseMark);
}
- {
- AutoLock auto_lock(lock_);
- if (state_ != kClosed)
- handler_->OnPaused(this);
- }
+ handler_->OnPaused(this);
}
void AudioOutputController::DoFlush() {
@@ -241,23 +220,30 @@ void AudioOutputController::DoFlush() {
// If we are in the regular latency mode then flush the push source.
if (!sync_reader_) {
- AutoLock auto_lock(lock_);
if (state_ != kPaused)
return;
push_source_.ClearAll();
}
}
-void AudioOutputController::DoClose() {
+void AudioOutputController::DoClose(Task* closed_task) {
DCHECK_EQ(message_loop_, MessageLoop::current());
- DCHECK_EQ(kClosed, state_);
- // |stream_| can be null if creating the device failed in DoCreate().
- if (stream_) {
- stream_->Stop();
- stream_->Close();
- // After stream is closed it is destroyed, so don't keep a reference to it.
- stream_ = NULL;
+
+ if (state_ != kClosed) {
+ // |stream_| can be null if creating the device failed in DoCreate().
+ if (stream_) {
+ stream_->Stop();
+ stream_->Close();
+ // After stream is closed it is destroyed, so don't keep a reference to
+ // it.
+ stream_ = NULL;
+ }
+
+ state_ = kClosed;
}
+
+ closed_task->Run();
+ delete closed_task;
}
void AudioOutputController::DoSetVolume(double volume) {
@@ -267,22 +253,16 @@ void AudioOutputController::DoSetVolume(double volume) {
// right away but when the stream is created we'll set the volume.
volume_ = volume;
- {
- AutoLock auto_lock(lock_);
- if (state_ != kPlaying && state_ != kPaused && state_ != kCreated)
- return;
- }
+ if (state_ != kPlaying && state_ != kPaused && state_ != kCreated)
+ return;
stream_->SetVolume(volume_);
}
void AudioOutputController::DoReportError(int code) {
DCHECK_EQ(message_loop_, MessageLoop::current());
- {
- AutoLock auto_lock(lock_);
- if (state_ != kClosed)
- handler_->OnError(this, code);
- }
+ if (state_ != kClosed)
+ handler_->OnError(this, code);
}
uint32 AudioOutputController::OnMoreData(AudioOutputStream* stream,
@@ -302,8 +282,8 @@ uint32 AudioOutputController::OnMoreData(AudioOutputStream* stream,
return 0;
}
- // Push source doesn't need to know the stream and number of pending bytes.
- // So just pass in NULL and 0.
+ // Push source doesn't need to know the stream and number of pending
+ // bytes. So just pass in NULL and 0.
uint32 size = push_source_.OnMoreData(NULL, dest, max_size, 0);
hardware_pending_bytes_ = pending_bytes + size;
SubmitOnMoreData_Locked();
@@ -317,6 +297,8 @@ uint32 AudioOutputController::OnMoreData(AudioOutputStream* stream,
}
void AudioOutputController::OnClose(AudioOutputStream* stream) {
+ DCHECK_EQ(message_loop_, MessageLoop::current());
+
// Push source doesn't need to know the stream so just pass in NULL.
if (LowLatencyMode()) {
sync_reader_->Close();
@@ -343,6 +325,10 @@ void AudioOutputController::SubmitOnMoreData_Locked() {
uint32 pending_bytes = hardware_pending_bytes_ +
push_source_.UnProcessedBytes();
+ // If we need more data then call the event handler to ask for more data.
+ // It is okay that we don't lock in this block because the parameters are
+ // correct and in the worst case we are just asking more data than needed.
+ AutoUnlock auto_unlock(lock_);
handler_->OnMoreData(this, timestamp, pending_bytes);
}
diff --git a/media/audio/audio_output_controller.h b/media/audio/audio_output_controller.h
index e9ed321..8208e28 100644
--- a/media/audio/audio_output_controller.h
+++ b/media/audio/audio_output_controller.h
@@ -14,6 +14,7 @@
#include "media/audio/simple_sources.h"
class MessageLoop;
+class Task;
// An AudioOutputController controls an AudioOutputStream and provides data
// to this output stream. It has an important function that it executes
@@ -134,13 +135,14 @@ class AudioOutputController
// has effect when the stream is paused.
void Flush();
- // Closes the audio output stream. It changes state to kClosed and returns
- // right away. The physical resources are freed on the audio thread if
- // neccessary.
+ // Closes the audio output stream. The state is changed and the resources
+ // are freed on the audio thread. closed_task is executed after that.
+ // Callbacks (EventHandler and SyncReader) must exist until closed_task is
+ // called.
//
// It is safe to call this method more than once. Calls after the first one
// will have no effect.
- void Close();
+ void Close(Task* closed_task);
// Sets the volume of the audio output stream.
void SetVolume(double volume);
@@ -168,7 +170,7 @@ class AudioOutputController
void DoPlay();
void DoPause();
void DoFlush();
- void DoClose();
+ void DoClose(Task* closed_task);
void DoSetVolume(double volume);
void DoReportError(int code);
@@ -190,8 +192,7 @@ class AudioOutputController
uint32 hardware_pending_bytes_;
base::Time last_callback_time_;
- // The |lock_| must be acquired whenever we access |state_| or call
- // |handler_|.
+ // The |lock_| must be acquired whenever we access |push_source_|.
Lock lock_;
// PushSource role is to buffer and it's only used in regular latency mode.
diff --git a/media/audio/audio_output_controller_unittest.cc b/media/audio/audio_output_controller_unittest.cc
index db1716c..b479974d 100644
--- a/media/audio/audio_output_controller_unittest.cc
+++ b/media/audio/audio_output_controller_unittest.cc
@@ -5,6 +5,7 @@
#include "base/environment.h"
#include "base/basictypes.h"
#include "base/logging.h"
+#include "base/task.h"
#include "base/waitable_event.h"
#include "media/audio/audio_output_controller.h"
#include "testing/gmock/include/gmock/gmock.h"
@@ -80,6 +81,18 @@ ACTION_P3(SignalEvent, event, count, limit) {
}
}
+// Helper functions used to close audio controller.
+static void SignalClosedEvent(base::WaitableEvent* event) {
+ event->Signal();
+}
+
+// Closes AudioOutputController synchronously.
+static void CloseAudioController(AudioOutputController* controller) {
+ base::WaitableEvent closed_event(true, false);
+ controller->Close(NewRunnableFunction(&SignalClosedEvent, &closed_event));
+ closed_event.Wait();
+}
+
TEST(AudioOutputControllerTest, CreateAndClose) {
if (!HasAudioOutputDevices() || IsRunningHeadless())
return;
@@ -92,11 +105,8 @@ TEST(AudioOutputControllerTest, CreateAndClose) {
kHardwareBufferSize, kBufferCapacity);
ASSERT_TRUE(controller.get());
- // Close the controller immediately. At this point, chances are that
- // DoCreate() hasn't been called yet. In any case, it should be safe to call
- // Close() and it should not try to call |event_handler| later (the test
- // would crash otherwise).
- controller->Close();
+ // Close the controller immediately.
+ CloseAudioController(controller);
}
TEST(AudioOutputControllerTest, PlayAndClose) {
@@ -134,9 +144,8 @@ TEST(AudioOutputControllerTest, PlayAndClose) {
controller->Play();
event.Wait();
- // Now stop the controller. The object is freed later after DoClose() is
- // executed.
- controller->Close();
+ // Now stop the controller.
+ CloseAudioController(controller);
}
TEST(AudioOutputControllerTest, PlayPauseClose) {
@@ -186,9 +195,8 @@ TEST(AudioOutputControllerTest, PlayPauseClose) {
controller->Pause();
event.Wait();
- // Now stop the controller. The object is freed later after DoClose() is
- // executed.
- controller->Close();
+ // Now stop the controller.
+ CloseAudioController(controller);
}
TEST(AudioOutputControllerTest, PlayPausePlay) {
@@ -250,9 +258,8 @@ TEST(AudioOutputControllerTest, PlayPausePlay) {
controller->Play();
event.Wait();
- // Now stop the controller. The object is freed later after DoClose() is
- // executed.
- controller->Close();
+ // Now stop the controller.
+ CloseAudioController(controller);
}
TEST(AudioOutputControllerTest, HardwareBufferTooLarge) {
@@ -302,8 +309,14 @@ TEST(AudioOutputControllerTest, CloseTwice) {
// Wait for OnMoreData() to be called.
event.Wait();
- controller->Close();
- controller->Close();
+ base::WaitableEvent closed_event_1(true, false);
+ controller->Close(NewRunnableFunction(&SignalClosedEvent, &closed_event_1));
+
+ base::WaitableEvent closed_event_2(true, false);
+ controller->Close(NewRunnableFunction(&SignalClosedEvent, &closed_event_2));
+
+ closed_event_1.Wait();
+ closed_event_2.Wait();
}
} // namespace media