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

Side by Side Diff: content/test/webrtc_audio_device_test.cc

Issue 194813002: Removing WebRTCAudioDeviceTest test suite (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: xians@ Created 6 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
(Empty)
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
3 // found in the LICENSE file.
4
5 #include "content/test/webrtc_audio_device_test.h"
6
7 #include "base/bind.h"
8 #include "base/bind_helpers.h"
9 #include "base/compiler_specific.h"
10 #include "base/file_util.h"
11 #include "base/run_loop.h"
12 #include "base/synchronization/waitable_event.h"
13 #include "base/test/test_timeouts.h"
14 #include "content/browser/media/capture/audio_mirroring_manager.h"
15 #include "content/browser/media/media_internals.h"
16 #include "content/browser/renderer_host/media/audio_input_renderer_host.h"
17 #include "content/browser/renderer_host/media/audio_renderer_host.h"
18 #include "content/browser/renderer_host/media/media_stream_manager.h"
19 #include "content/browser/renderer_host/media/mock_media_observer.h"
20 #include "content/common/media/media_param_traits.h"
21 #include "content/common/view_messages.h"
22 #include "content/public/browser/browser_thread.h"
23 #include "content/public/browser/resource_context.h"
24 #include "content/public/common/content_paths.h"
25 #include "content/public/test/test_browser_thread.h"
26 #include "content/renderer/media/audio_input_message_filter.h"
27 #include "content/renderer/media/audio_message_filter.h"
28 #include "content/renderer/media/webrtc_audio_device_impl.h"
29 #include "content/renderer/render_process.h"
30 #include "content/renderer/render_thread_impl.h"
31 #include "content/renderer/renderer_webkitplatformsupport_impl.h"
32 #include "media/audio/audio_parameters.h"
33 #include "media/base/audio_hardware_config.h"
34 #include "net/url_request/url_request_test_util.h"
35 #include "testing/gmock/include/gmock/gmock.h"
36 #include "testing/gtest/include/gtest/gtest.h"
37 #include "third_party/webrtc/voice_engine/include/voe_audio_processing.h"
38 #include "third_party/webrtc/voice_engine/include/voe_base.h"
39 #include "third_party/webrtc/voice_engine/include/voe_file.h"
40 #include "third_party/webrtc/voice_engine/include/voe_network.h"
41
42 #if defined(OS_WIN)
43 #include "base/win/scoped_com_initializer.h"
44 #endif
45
46 using media::AudioParameters;
47 using media::ChannelLayout;
48 using testing::_;
49 using testing::InvokeWithoutArgs;
50 using testing::Return;
51 using testing::StrEq;
52
53 namespace content {
54
55 // This class is a mock of the child process singleton which is needed
56 // to be able to create a RenderThread object.
57 class WebRTCMockRenderProcess : public RenderProcess {
58 public:
59 WebRTCMockRenderProcess() {}
60 virtual ~WebRTCMockRenderProcess() {}
61
62 // RenderProcess implementation.
63 virtual skia::PlatformCanvas* GetDrawingCanvas(
64 TransportDIB** memory, const gfx::Rect& rect) OVERRIDE {
65 return NULL;
66 }
67 virtual void ReleaseTransportDIB(TransportDIB* memory) OVERRIDE {}
68 virtual void AddBindings(int bindings) OVERRIDE {}
69 virtual int GetEnabledBindings() const OVERRIDE { return 0; }
70 virtual TransportDIB* CreateTransportDIB(size_t size) OVERRIDE {
71 return NULL;
72 }
73 virtual void FreeTransportDIB(TransportDIB*) OVERRIDE {}
74
75 private:
76 DISALLOW_COPY_AND_ASSIGN(WebRTCMockRenderProcess);
77 };
78
79 class TestAudioRendererHost : public AudioRendererHost {
80 public:
81 TestAudioRendererHost(
82 int render_process_id,
83 media::AudioManager* audio_manager,
84 AudioMirroringManager* mirroring_manager,
85 MediaInternals* media_internals,
86 MediaStreamManager* media_stream_manager,
87 IPC::Channel* channel)
88 : AudioRendererHost(render_process_id, audio_manager, mirroring_manager,
89 media_internals, media_stream_manager),
90 channel_(channel) {}
91 virtual bool Send(IPC::Message* message) OVERRIDE {
92 if (channel_)
93 return channel_->Send(message);
94 return false;
95 }
96 void ResetChannel() {
97 channel_ = NULL;
98 }
99
100 protected:
101 virtual ~TestAudioRendererHost() {}
102
103 private:
104 IPC::Channel* channel_;
105 };
106
107 class TestAudioInputRendererHost : public AudioInputRendererHost {
108 public:
109 TestAudioInputRendererHost(
110 media::AudioManager* audio_manager,
111 MediaStreamManager* media_stream_manager,
112 AudioMirroringManager* audio_mirroring_manager,
113 media::UserInputMonitor* user_input_monitor,
114 IPC::Channel* channel)
115 : AudioInputRendererHost(audio_manager, media_stream_manager,
116 audio_mirroring_manager, user_input_monitor),
117 channel_(channel) {}
118 virtual bool Send(IPC::Message* message) OVERRIDE {
119 if (channel_)
120 return channel_->Send(message);
121 return false;
122 }
123 void ResetChannel() {
124 channel_ = NULL;
125 }
126
127 protected:
128 virtual ~TestAudioInputRendererHost() {}
129
130 private:
131 IPC::Channel* channel_;
132 };
133
134 // Utility scoped class to replace the global content client's renderer for the
135 // duration of the test.
136 class ReplaceContentClientRenderer {
137 public:
138 explicit ReplaceContentClientRenderer(ContentRendererClient* new_renderer) {
139 saved_renderer_ = SetRendererClientForTesting(new_renderer);
140 }
141 ~ReplaceContentClientRenderer() {
142 // Restore the original renderer.
143 SetRendererClientForTesting(saved_renderer_);
144 }
145 private:
146 ContentRendererClient* saved_renderer_;
147 DISALLOW_COPY_AND_ASSIGN(ReplaceContentClientRenderer);
148 };
149
150 class MockRTCResourceContext : public ResourceContext {
151 public:
152 MockRTCResourceContext() : test_request_context_(NULL) {}
153 virtual ~MockRTCResourceContext() {}
154
155 void set_request_context(net::URLRequestContext* request_context) {
156 test_request_context_ = request_context;
157 }
158
159 // ResourceContext implementation:
160 virtual net::HostResolver* GetHostResolver() OVERRIDE {
161 return NULL;
162 }
163 virtual net::URLRequestContext* GetRequestContext() OVERRIDE {
164 return test_request_context_;
165 }
166
167 virtual bool AllowMicAccess(const GURL& origin) OVERRIDE {
168 return false;
169 }
170
171 virtual bool AllowCameraAccess(const GURL& origin) OVERRIDE {
172 return false;
173 }
174
175 private:
176 net::URLRequestContext* test_request_context_;
177
178 DISALLOW_COPY_AND_ASSIGN(MockRTCResourceContext);
179 };
180
181 ACTION_P(QuitMessageLoop, loop_or_proxy) {
182 loop_or_proxy->PostTask(FROM_HERE, base::MessageLoop::QuitClosure());
183 }
184
185 MAYBE_WebRTCAudioDeviceTest::MAYBE_WebRTCAudioDeviceTest()
186 : render_thread_(NULL), audio_hardware_config_(NULL),
187 has_input_devices_(false), has_output_devices_(false) {
188 }
189
190 MAYBE_WebRTCAudioDeviceTest::~MAYBE_WebRTCAudioDeviceTest() {}
191
192 void MAYBE_WebRTCAudioDeviceTest::SetUp() {
193 // This part sets up a RenderThread environment to ensure that
194 // RenderThread::current() (<=> TLS pointer) is valid.
195 // Main parts are inspired by the RenderViewFakeResourcesTest.
196 // Note that, the IPC part is not utilized in this test.
197 saved_content_renderer_.reset(
198 new ReplaceContentClientRenderer(&content_renderer_client_));
199 mock_process_.reset(new WebRTCMockRenderProcess());
200 ui_thread_.reset(
201 new TestBrowserThread(BrowserThread::UI, base::MessageLoop::current()));
202
203 // Construct the resource context on the UI thread.
204 resource_context_.reset(new MockRTCResourceContext);
205
206 static const char kThreadName[] = "RenderThread";
207 ChildProcess::current()->io_message_loop()->PostTask(FROM_HERE,
208 base::Bind(&MAYBE_WebRTCAudioDeviceTest::InitializeIOThread,
209 base::Unretained(this), kThreadName));
210 WaitForIOThreadCompletion();
211
212 sandbox_was_enabled_ =
213 RendererWebKitPlatformSupportImpl::SetSandboxEnabledForTesting(false);
214 // TODO(tommi): RenderThreadImpl no longer supports being instantiated in
215 // tests like this. Right now it initializes DiscardableMemory which can
216 // only be initialized once. Since all WebRTCAudioDeviceTest have been
217 // disabled, they are starting to bit rot :-(
218 // We should use a mocked render thread.
219 render_thread_ = new RenderThreadImpl(kThreadName);
220 }
221
222 void MAYBE_WebRTCAudioDeviceTest::TearDown() {
223 SetAudioHardwareConfig(NULL);
224
225 // Run any pending cleanup tasks that may have been posted to the main thread.
226 base::RunLoop().RunUntilIdle();
227
228 // Kick of the cleanup process by closing the channel. This queues up
229 // OnStreamClosed calls to be executed on the audio thread.
230 ChildProcess::current()->io_message_loop()->PostTask(FROM_HERE,
231 base::Bind(&MAYBE_WebRTCAudioDeviceTest::DestroyChannel,
232 base::Unretained(this)));
233 WaitForIOThreadCompletion();
234
235 // When audio [input] render hosts are notified that the channel has
236 // been closed, they post tasks to the audio thread to close the
237 // AudioOutputController and once that's completed, a task is posted back to
238 // the IO thread to actually delete the AudioEntry for the audio stream. Only
239 // then is the reference to the audio manager released, so we wait for the
240 // whole thing to be torn down before we finally uninitialize the io thread.
241 WaitForAudioManagerCompletion();
242
243 ChildProcess::current()->io_message_loop()->PostTask(FROM_HERE,
244 base::Bind(&MAYBE_WebRTCAudioDeviceTest::UninitializeIOThread,
245 base::Unretained((this))));
246 WaitForIOThreadCompletion();
247 mock_process_.reset();
248 media_stream_manager_.reset();
249 mirroring_manager_.reset();
250 RendererWebKitPlatformSupportImpl::SetSandboxEnabledForTesting(
251 sandbox_was_enabled_);
252 }
253
254 bool MAYBE_WebRTCAudioDeviceTest::Send(IPC::Message* message) {
255 return channel_->Send(message);
256 }
257
258 void MAYBE_WebRTCAudioDeviceTest::SetAudioHardwareConfig(
259 media::AudioHardwareConfig* hardware_config) {
260 audio_hardware_config_ = hardware_config;
261 }
262
263 scoped_refptr<WebRtcAudioRenderer>
264 MAYBE_WebRTCAudioDeviceTest::CreateDefaultWebRtcAudioRenderer(
265 int render_view_id,
266 const scoped_refptr<webrtc::MediaStreamInterface>& media_stream) {
267 media::AudioHardwareConfig* hardware_config =
268 RenderThreadImpl::current()->GetAudioHardwareConfig();
269 int sample_rate = hardware_config->GetOutputSampleRate();
270 int frames_per_buffer = hardware_config->GetOutputBufferSize();
271 return new WebRtcAudioRenderer(media_stream, render_view_id, MSG_ROUTING_NONE,
272 0, sample_rate, frames_per_buffer);
273 }
274
275 void MAYBE_WebRTCAudioDeviceTest::InitializeIOThread(const char* thread_name) {
276 #if defined(OS_WIN)
277 // We initialize COM (STA) on our IO thread as is done in Chrome.
278 // See BrowserProcessSubThread::Init.
279 initialize_com_.reset(new base::win::ScopedCOMInitializer());
280 #endif
281
282 // Set the current thread as the IO thread.
283 io_thread_.reset(
284 new TestBrowserThread(BrowserThread::IO, base::MessageLoop::current()));
285
286 // Populate our resource context.
287 test_request_context_.reset(new net::TestURLRequestContext());
288 MockRTCResourceContext* resource_context =
289 static_cast<MockRTCResourceContext*>(resource_context_.get());
290 resource_context->set_request_context(test_request_context_.get());
291
292 // Create our own AudioManager, AudioMirroringManager and MediaStreamManager.
293 audio_manager_.reset(media::AudioManager::CreateForTesting());
294 mirroring_manager_.reset(new AudioMirroringManager());
295 media_stream_manager_.reset(new MediaStreamManager(audio_manager_.get()));
296
297 has_input_devices_ = audio_manager_->HasAudioInputDevices();
298 has_output_devices_ = audio_manager_->HasAudioOutputDevices();
299
300 // Create an IPC channel that handles incoming messages on the IO thread.
301 CreateChannel(thread_name);
302 }
303
304 void MAYBE_WebRTCAudioDeviceTest::UninitializeIOThread() {
305 resource_context_.reset();
306
307 test_request_context_.reset();
308
309 #if defined(OS_WIN)
310 initialize_com_.reset();
311 #endif
312
313 audio_manager_.reset();
314 }
315
316 void MAYBE_WebRTCAudioDeviceTest::CreateChannel(const char* name) {
317 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
318
319 channel_.reset(new IPC::Channel(name, IPC::Channel::MODE_SERVER, this));
320 ASSERT_TRUE(channel_->Connect());
321
322 static const int kRenderProcessId = 1;
323 audio_render_host_ = new TestAudioRendererHost(kRenderProcessId,
324 audio_manager_.get(),
325 mirroring_manager_.get(),
326 MediaInternals::GetInstance(),
327 media_stream_manager_.get(),
328 channel_.get());
329 audio_render_host_->set_peer_pid_for_testing(base::GetCurrentProcId());
330
331 audio_input_renderer_host_ =
332 new TestAudioInputRendererHost(audio_manager_.get(),
333 media_stream_manager_.get(),
334 mirroring_manager_.get(),
335 NULL,
336 channel_.get());
337 audio_input_renderer_host_->set_peer_pid_for_testing(
338 base::GetCurrentProcId());
339 }
340
341 void MAYBE_WebRTCAudioDeviceTest::DestroyChannel() {
342 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::IO));
343 audio_render_host_->OnChannelClosing();
344 audio_render_host_->OnFilterRemoved();
345 audio_input_renderer_host_->OnChannelClosing();
346 audio_input_renderer_host_->OnFilterRemoved();
347 audio_render_host_->ResetChannel();
348 audio_input_renderer_host_->ResetChannel();
349 channel_.reset();
350 audio_render_host_ = NULL;
351 audio_input_renderer_host_ = NULL;
352 }
353
354 void MAYBE_WebRTCAudioDeviceTest::OnGetAudioHardwareConfig(
355 AudioParameters* input_params, AudioParameters* output_params) {
356 ASSERT_TRUE(audio_hardware_config_);
357 *input_params = audio_hardware_config_->GetInputConfig();
358 *output_params = audio_hardware_config_->GetOutputConfig();
359 }
360
361 // IPC::Listener implementation.
362 bool MAYBE_WebRTCAudioDeviceTest::OnMessageReceived(
363 const IPC::Message& message) {
364 if (render_thread_) {
365 IPC::ChannelProxy::MessageFilter* filter =
366 render_thread_->audio_input_message_filter();
367 if (filter->OnMessageReceived(message))
368 return true;
369
370 filter = render_thread_->audio_message_filter();
371 if (filter->OnMessageReceived(message))
372 return true;
373 }
374
375 if (audio_render_host_.get()) {
376 bool message_was_ok = false;
377 if (audio_render_host_->OnMessageReceived(message, &message_was_ok))
378 return true;
379 }
380
381 if (audio_input_renderer_host_.get()) {
382 bool message_was_ok = false;
383 if (audio_input_renderer_host_->OnMessageReceived(message, &message_was_ok))
384 return true;
385 }
386
387 bool handled ALLOW_UNUSED = true;
388 bool message_is_ok = true;
389 IPC_BEGIN_MESSAGE_MAP_EX(MAYBE_WebRTCAudioDeviceTest, message, message_is_ok)
390 IPC_MESSAGE_HANDLER(ViewHostMsg_GetAudioHardwareConfig,
391 OnGetAudioHardwareConfig)
392 IPC_MESSAGE_UNHANDLED(handled = false)
393 IPC_END_MESSAGE_MAP_EX()
394
395 EXPECT_TRUE(message_is_ok);
396
397 return true;
398 }
399
400 // Posts a final task to the IO message loop and waits for completion.
401 void MAYBE_WebRTCAudioDeviceTest::WaitForIOThreadCompletion() {
402 WaitForTaskRunnerCompletion(
403 ChildProcess::current()->io_message_loop()->message_loop_proxy());
404 }
405
406 void MAYBE_WebRTCAudioDeviceTest::WaitForAudioManagerCompletion() {
407 if (audio_manager_)
408 WaitForTaskRunnerCompletion(audio_manager_->GetTaskRunner());
409 }
410
411 void MAYBE_WebRTCAudioDeviceTest::WaitForTaskRunnerCompletion(
412 const scoped_refptr<base::SingleThreadTaskRunner>& task_runner) {
413 base::WaitableEvent* event = new base::WaitableEvent(false, false);
414 task_runner->PostTask(
415 FROM_HERE,
416 base::Bind(&base::WaitableEvent::Signal, base::Unretained(event)));
417 if (event->TimedWait(TestTimeouts::action_max_timeout())) {
418 delete event;
419 } else {
420 // Don't delete the event object in case the message ever gets processed.
421 // If we do, we will crash the test process.
422 ADD_FAILURE() << "Failed to wait for message loop";
423 }
424 }
425
426 std::string MAYBE_WebRTCAudioDeviceTest::GetTestDataPath(
427 const base::FilePath::StringType& file_name) {
428 base::FilePath path;
429 EXPECT_TRUE(PathService::Get(DIR_TEST_DATA, &path));
430 path = path.Append(file_name);
431 EXPECT_TRUE(base::PathExists(path));
432 #if defined(OS_WIN)
433 return base::WideToUTF8(path.value());
434 #else
435 return path.value();
436 #endif
437 }
438
439 WebRTCTransportImpl::WebRTCTransportImpl(webrtc::VoENetwork* network)
440 : network_(network) {
441 }
442
443 WebRTCTransportImpl::~WebRTCTransportImpl() {}
444
445 int WebRTCTransportImpl::SendPacket(int channel, const void* data, int len) {
446 return network_->ReceivedRTPPacket(channel, data, len);
447 }
448
449 int WebRTCTransportImpl::SendRTCPPacket(int channel, const void* data,
450 int len) {
451 return network_->ReceivedRTCPPacket(channel, data, len);
452 }
453
454 } // namespace content
OLDNEW
« no previous file with comments | « content/test/webrtc_audio_device_test.h ('k') | tools/valgrind/gtest_exclude/content_unittests.gtest.txt » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698