Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(67)

Side by Side Diff: content/renderer/pepper/pepper_platform_audio_input_impl.cc

Issue 12383016: Merge AssociateStreamWithProducer message into CreateStream message for both audio output and input. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: rebase Created 7 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "content/renderer/pepper/pepper_platform_audio_input_impl.h" 5 #include "content/renderer/pepper/pepper_platform_audio_input_impl.h"
6 6
7 #include "base/bind.h" 7 #include "base/bind.h"
8 #include "base/logging.h" 8 #include "base/logging.h"
9 #include "base/message_loop_proxy.h" 9 #include "base/message_loop_proxy.h"
10 #include "build/build_config.h" 10 #include "build/build_config.h"
11 #include "content/common/child_process.h" 11 #include "content/common/child_process.h"
12 #include "content/common/media/audio_messages.h"
13 #include "content/renderer/media/audio_input_message_filter.h" 12 #include "content/renderer/media/audio_input_message_filter.h"
14 #include "content/renderer/pepper/pepper_plugin_delegate_impl.h" 13 #include "content/renderer/pepper/pepper_plugin_delegate_impl.h"
15 #include "content/renderer/render_thread_impl.h"
16 #include "media/audio/audio_manager_base.h" 14 #include "media/audio/audio_manager_base.h"
17 15
18 namespace content { 16 namespace content {
19 17
20 // static 18 // static
21 PepperPlatformAudioInputImpl* PepperPlatformAudioInputImpl::Create( 19 PepperPlatformAudioInputImpl* PepperPlatformAudioInputImpl::Create(
22 const base::WeakPtr<PepperPluginDelegateImpl>& plugin_delegate, 20 const base::WeakPtr<PepperPluginDelegateImpl>& plugin_delegate,
23 const std::string& device_id, 21 const std::string& device_id,
24 int sample_rate, 22 int sample_rate,
25 int frames_per_buffer, 23 int frames_per_buffer,
(...skipping 22 matching lines...) Expand all
48 DCHECK(main_message_loop_proxy_->BelongsToCurrentThread()); 46 DCHECK(main_message_loop_proxy_->BelongsToCurrentThread());
49 47
50 ChildProcess::current()->io_message_loop()->PostTask( 48 ChildProcess::current()->io_message_loop()->PostTask(
51 FROM_HERE, 49 FROM_HERE,
52 base::Bind(&PepperPlatformAudioInputImpl::StopCaptureOnIOThread, this)); 50 base::Bind(&PepperPlatformAudioInputImpl::StopCaptureOnIOThread, this));
53 } 51 }
54 52
55 void PepperPlatformAudioInputImpl::ShutDown() { 53 void PepperPlatformAudioInputImpl::ShutDown() {
56 DCHECK(main_message_loop_proxy_->BelongsToCurrentThread()); 54 DCHECK(main_message_loop_proxy_->BelongsToCurrentThread());
57 55
56 // Make sure we don't call shutdown more than once.
57 if (!client_)
58 return;
59
58 // Called on the main thread to stop all audio callbacks. We must only change 60 // Called on the main thread to stop all audio callbacks. We must only change
59 // the client on the main thread, and the delegates from the I/O thread. 61 // the client on the main thread, and the delegates from the I/O thread.
60 client_ = NULL; 62 client_ = NULL;
61 ChildProcess::current()->io_message_loop()->PostTask( 63 ChildProcess::current()->io_message_loop()->PostTask(
62 FROM_HERE, 64 FROM_HERE,
63 base::Bind(&PepperPlatformAudioInputImpl::ShutDownOnIOThread, this)); 65 base::Bind(&PepperPlatformAudioInputImpl::ShutDownOnIOThread, this));
64 } 66 }
65 67
66 void PepperPlatformAudioInputImpl::OnStreamCreated( 68 void PepperPlatformAudioInputImpl::OnStreamCreated(
67 base::SharedMemoryHandle handle, 69 base::SharedMemoryHandle handle,
68 base::SyncSocket::Handle socket_handle, 70 base::SyncSocket::Handle socket_handle,
69 int length) { 71 int length) {
70 #if defined(OS_WIN) 72 #if defined(OS_WIN)
71 DCHECK(handle); 73 DCHECK(handle);
72 DCHECK(socket_handle); 74 DCHECK(socket_handle);
73 #else 75 #else
74 DCHECK_NE(-1, handle.fd); 76 DCHECK_NE(-1, handle.fd);
75 DCHECK_NE(-1, socket_handle); 77 DCHECK_NE(-1, socket_handle);
76 #endif 78 #endif
77 DCHECK(length); 79 DCHECK(length);
78 80
79 if (base::MessageLoopProxy::current() != main_message_loop_proxy_) { 81 if (base::MessageLoopProxy::current() != main_message_loop_proxy_) {
80 // No need to check |shutdown_called_| here. If shutdown has occurred, 82 // If shutdown has occurred, |client_| will be NULL and the handles will be
81 // |client_| will be NULL and the handles will be cleaned up on the main 83 // cleaned up on the main thread.
82 // thread.
83 main_message_loop_proxy_->PostTask( 84 main_message_loop_proxy_->PostTask(
84 FROM_HERE, 85 FROM_HERE,
85 base::Bind(&PepperPlatformAudioInputImpl::OnStreamCreated, this, 86 base::Bind(&PepperPlatformAudioInputImpl::OnStreamCreated, this,
86 handle, socket_handle, length)); 87 handle, socket_handle, length));
87 } else { 88 } else {
88 // Must dereference the client only on the main thread. Shutdown may have 89 // Must dereference the client only on the main thread. Shutdown may have
89 // occurred while the request was in-flight, so we need to NULL check. 90 // occurred while the request was in-flight, so we need to NULL check.
90 if (client_) { 91 if (client_) {
91 client_->StreamCreated(handle, length, socket_handle); 92 client_->StreamCreated(handle, length, socket_handle);
92 } else { 93 } else {
93 // Clean up the handles. 94 // Clean up the handles.
94 base::SyncSocket temp_socket(socket_handle); 95 base::SyncSocket temp_socket(socket_handle);
95 base::SharedMemory temp_shared_memory(handle, false); 96 base::SharedMemory temp_shared_memory(handle, false);
96 } 97 }
97 } 98 }
98 } 99 }
99 100
100 void PepperPlatformAudioInputImpl::OnVolume(double volume) {} 101 void PepperPlatformAudioInputImpl::OnVolume(double volume) {}
101 102
102 void PepperPlatformAudioInputImpl::OnStateChanged( 103 void PepperPlatformAudioInputImpl::OnStateChanged(
103 media::AudioInputIPCDelegate::State state) { 104 media::AudioInputIPCDelegate::State state) {
104 } 105 }
105 106
106 void PepperPlatformAudioInputImpl::OnDeviceReady(const std::string& device_id) { 107 void PepperPlatformAudioInputImpl::OnDeviceReady(const std::string& device_id) {
107 DCHECK(ChildProcess::current()->io_message_loop_proxy()-> 108 DCHECK(ChildProcess::current()->io_message_loop_proxy()->
108 BelongsToCurrentThread()); 109 BelongsToCurrentThread());
109 110
110 if (shutdown_called_) 111 if (!ipc_)
111 return; 112 return;
112 113
113 if (device_id.empty()) { 114 if (device_id.empty()) {
114 main_message_loop_proxy_->PostTask( 115 main_message_loop_proxy_->PostTask(
115 FROM_HERE, 116 FROM_HERE,
116 base::Bind(&PepperPlatformAudioInputImpl::NotifyStreamCreationFailed, 117 base::Bind(&PepperPlatformAudioInputImpl::NotifyStreamCreationFailed,
117 this)); 118 this));
118 } else { 119 } else {
119 // We will be notified by OnStreamCreated(). 120 // We will be notified by OnStreamCreated().
120 ipc_->CreateStream(stream_id_, params_, device_id, false); 121 ipc_->CreateStream(this, params_, device_id, false);
121 } 122 }
122 } 123 }
123 124
124 void PepperPlatformAudioInputImpl::OnIPCClosed() { 125 void PepperPlatformAudioInputImpl::OnIPCClosed() {
125 ipc_ = NULL; 126 ipc_.reset();
126 } 127 }
127 128
128 PepperPlatformAudioInputImpl::~PepperPlatformAudioInputImpl() { 129 PepperPlatformAudioInputImpl::~PepperPlatformAudioInputImpl() {
129 // Make sure we have been shut down. Warning: this may happen on the I/O 130 // Make sure we have been shut down. Warning: this may happen on the I/O
130 // thread! 131 // thread!
131 // Although these members should be accessed on a specific thread (either the 132 // Although these members should be accessed on a specific thread (either the
132 // main thread or the I/O thread), it should be fine to examine their value 133 // main thread or the I/O thread), it should be fine to examine their value
133 // here. 134 // here.
134 DCHECK_EQ(0, stream_id_); 135 DCHECK(!ipc_);
135 DCHECK(!client_); 136 DCHECK(!client_);
136 DCHECK(label_.empty()); 137 DCHECK(label_.empty());
137 DCHECK(shutdown_called_);
138 } 138 }
139 139
140 PepperPlatformAudioInputImpl::PepperPlatformAudioInputImpl() 140 PepperPlatformAudioInputImpl::PepperPlatformAudioInputImpl()
141 : client_(NULL), 141 : client_(NULL),
142 stream_id_(0), 142 main_message_loop_proxy_(base::MessageLoopProxy::current()) {
143 render_view_id_(MSG_ROUTING_NONE),
144 main_message_loop_proxy_(base::MessageLoopProxy::current()),
145 shutdown_called_(false) {
146 ipc_ = RenderThreadImpl::current()->audio_input_message_filter();
147 } 143 }
148 144
149 bool PepperPlatformAudioInputImpl::Initialize( 145 bool PepperPlatformAudioInputImpl::Initialize(
150 const base::WeakPtr<PepperPluginDelegateImpl>& plugin_delegate, 146 const base::WeakPtr<PepperPluginDelegateImpl>& plugin_delegate,
151 const std::string& device_id, 147 const std::string& device_id,
152 int sample_rate, 148 int sample_rate,
153 int frames_per_buffer, 149 int frames_per_buffer,
154 webkit::ppapi::PluginDelegate::PlatformAudioInputClient* client) { 150 webkit::ppapi::PluginDelegate::PlatformAudioInputClient* client) {
155 DCHECK(main_message_loop_proxy_->BelongsToCurrentThread()); 151 DCHECK(main_message_loop_proxy_->BelongsToCurrentThread());
156 152
157 if (!plugin_delegate || !client) 153 if (!plugin_delegate || !client)
158 return false; 154 return false;
159 155
156 ipc_ = AudioInputMessageFilter::Get()->CreateAudioInputIPC(
157 plugin_delegate->GetRoutingID());
158 CHECK(ipc_);
159
160 plugin_delegate_ = plugin_delegate; 160 plugin_delegate_ = plugin_delegate;
161 render_view_id_ = plugin_delegate_->GetRoutingID();
162 client_ = client; 161 client_ = client;
163 162
164 params_.Reset(media::AudioParameters::AUDIO_PCM_LINEAR, 163 params_.Reset(media::AudioParameters::AUDIO_PCM_LINEAR,
165 media::CHANNEL_LAYOUT_MONO, 0, 164 media::CHANNEL_LAYOUT_MONO, 0,
166 sample_rate, 16, frames_per_buffer); 165 sample_rate, 16, frames_per_buffer);
167 166
168 if (device_id.empty()) { 167 if (device_id.empty()) {
169 // Use the default device. 168 // Use the default device.
170 ChildProcess::current()->io_message_loop()->PostTask( 169 ChildProcess::current()->io_message_loop()->PostTask(
171 FROM_HERE, 170 FROM_HERE,
172 base::Bind(&PepperPlatformAudioInputImpl::InitializeOnIOThread, 171 base::Bind(&PepperPlatformAudioInputImpl::InitializeOnIOThread,
173 this, 0)); 172 this, 0));
174 } else { 173 } else {
175 // We need to open the device and obtain the label and session ID before 174 // We need to open the device and obtain the label and session ID before
176 // initializing. 175 // initializing.
177 plugin_delegate_->OpenDevice( 176 plugin_delegate_->OpenDevice(
178 PP_DEVICETYPE_DEV_AUDIOCAPTURE, device_id, 177 PP_DEVICETYPE_DEV_AUDIOCAPTURE, device_id,
179 base::Bind(&PepperPlatformAudioInputImpl::OnDeviceOpened, this)); 178 base::Bind(&PepperPlatformAudioInputImpl::OnDeviceOpened, this));
180 } 179 }
181 return true; 180 return true;
182 } 181 }
183 182
184 void PepperPlatformAudioInputImpl::InitializeOnIOThread(int session_id) { 183 void PepperPlatformAudioInputImpl::InitializeOnIOThread(int session_id) {
185 DCHECK(ChildProcess::current()->io_message_loop_proxy()-> 184 DCHECK(ChildProcess::current()->io_message_loop_proxy()->
186 BelongsToCurrentThread()); 185 BelongsToCurrentThread());
187 186
188 if (shutdown_called_) 187 if (!ipc_)
189 return; 188 return;
190 189
191 // Make sure we don't call init more than once.
192 DCHECK_EQ(0, stream_id_);
193 stream_id_ = ipc_->AddDelegate(this);
194 DCHECK_NE(0, stream_id_);
195
196 if (!session_id) { 190 if (!session_id) {
197 // We will be notified by OnStreamCreated(). 191 // We will be notified by OnStreamCreated().
198 ipc_->CreateStream(stream_id_, params_, 192 ipc_->CreateStream(
199 media::AudioManagerBase::kDefaultDeviceId, false); 193 this, params_, media::AudioManagerBase::kDefaultDeviceId, false);
200 } else { 194 } else {
201 // We will be notified by OnDeviceReady(). 195 // We will be notified by OnDeviceReady().
202 ipc_->StartDevice(stream_id_, session_id); 196 ipc_->StartDevice(this, session_id);
203 } 197 }
204 } 198 }
205 199
206 void PepperPlatformAudioInputImpl::StartCaptureOnIOThread() { 200 void PepperPlatformAudioInputImpl::StartCaptureOnIOThread() {
207 DCHECK(ChildProcess::current()->io_message_loop_proxy()-> 201 DCHECK(ChildProcess::current()->io_message_loop_proxy()->
208 BelongsToCurrentThread()); 202 BelongsToCurrentThread());
209 203
210 if (stream_id_) { 204 if (ipc_)
211 ipc_->AssociateStreamWithConsumer(stream_id_, render_view_id_); 205 ipc_->RecordStream();
212 ipc_->RecordStream(stream_id_);
213 }
214 } 206 }
215 207
216 void PepperPlatformAudioInputImpl::StopCaptureOnIOThread() { 208 void PepperPlatformAudioInputImpl::StopCaptureOnIOThread() {
217 DCHECK(ChildProcess::current()->io_message_loop_proxy()-> 209 DCHECK(ChildProcess::current()->io_message_loop_proxy()->
218 BelongsToCurrentThread()); 210 BelongsToCurrentThread());
219 211
220 // TODO(yzshen): We cannot re-start capturing if the stream is closed. 212 // TODO(yzshen): We cannot re-start capturing if the stream is closed.
221 if (stream_id_) 213 if (ipc_) {
222 ipc_->CloseStream(stream_id_); 214 ipc_->CloseStream();
215 ipc_.reset();
216 }
223 } 217 }
224 218
225 void PepperPlatformAudioInputImpl::ShutDownOnIOThread() { 219 void PepperPlatformAudioInputImpl::ShutDownOnIOThread() {
226 DCHECK(ChildProcess::current()->io_message_loop_proxy()-> 220 DCHECK(ChildProcess::current()->io_message_loop_proxy()->
227 BelongsToCurrentThread()); 221 BelongsToCurrentThread());
228 222
229 // Make sure we don't call shutdown more than once. 223 StopCaptureOnIOThread();
230 if (shutdown_called_)
231 return;
232 shutdown_called_ = true;
233
234 if (stream_id_) {
235 ipc_->CloseStream(stream_id_);
236 ipc_->RemoveDelegate(stream_id_);
237 stream_id_ = 0;
238 }
239 224
240 main_message_loop_proxy_->PostTask( 225 main_message_loop_proxy_->PostTask(
241 FROM_HERE, 226 FROM_HERE,
242 base::Bind(&PepperPlatformAudioInputImpl::CloseDevice, this)); 227 base::Bind(&PepperPlatformAudioInputImpl::CloseDevice, this));
243 228
244 Release(); // Release for the delegate, balances out the reference taken in 229 Release(); // Release for the delegate, balances out the reference taken in
245 // PepperPluginDelegateImpl::CreateAudioInput. 230 // PepperPluginDelegateImpl::CreateAudioInput.
246 } 231 }
247 232
248 void PepperPlatformAudioInputImpl::OnDeviceOpened(int request_id, 233 void PepperPlatformAudioInputImpl::OnDeviceOpened(int request_id,
(...skipping 31 matching lines...) Expand 10 before | Expand all | Expand 10 after
280 } 265 }
281 266
282 void PepperPlatformAudioInputImpl::NotifyStreamCreationFailed() { 267 void PepperPlatformAudioInputImpl::NotifyStreamCreationFailed() {
283 DCHECK(main_message_loop_proxy_->BelongsToCurrentThread()); 268 DCHECK(main_message_loop_proxy_->BelongsToCurrentThread());
284 269
285 if (client_) 270 if (client_)
286 client_->StreamCreationFailed(); 271 client_->StreamCreationFailed();
287 } 272 }
288 273
289 } // namespace content 274 } // namespace content
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698