1 // Copyright 2013 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 "base/message_loop/message_loop.h" 6 #include "base/run_loop.h" 7 #include "media/audio/audio_parameters.h" 8 #include "media/base/fake_audio_render_callback.h" 9 #include "media/base/mock_audio_renderer_sink.h" 10 #include "media/blink/webaudiosourceprovider_impl.h" 11 #include "testing/gmock/include/gmock/gmock.h" 12 #include "testing/gtest/include/gtest/gtest.h" 13 #include "third_party/WebKit/public/platform/WebAudioSourceProviderClient.h" 14 15 namespace media { 16 17 namespace { 18 const float kTestVolume = 0.25; 19 } // namespace 20 21 class WebAudioSourceProviderImplTest 22 : public testing::Test, 23 public blink::WebAudioSourceProviderClient { 24 public: 25 WebAudioSourceProviderImplTest() 26 : params_(AudioParameters::AUDIO_PCM_LINEAR, 27 CHANNEL_LAYOUT_STEREO, 48000, 16, 64), 28 fake_callback_(0.1), 29 mock_sink_(new MockAudioRendererSink()), 30 wasp_impl_(new WebAudioSourceProviderImpl(mock_sink_)) { 31 } 32 33 virtual ~WebAudioSourceProviderImplTest() {} 34 35 void CallAllSinkMethodsAndVerify(bool verify) { 36 testing::InSequence s; 37 38 EXPECT_CALL(*mock_sink_.get(), Start()).Times(verify); 39 wasp_impl_->Start(); 40 41 EXPECT_CALL(*mock_sink_.get(), Play()).Times(verify); 42 wasp_impl_->Play(); 43 44 EXPECT_CALL(*mock_sink_.get(), Pause()).Times(verify); 45 wasp_impl_->Pause(); 46 47 EXPECT_CALL(*mock_sink_.get(), SetVolume(kTestVolume)).Times(verify); 48 wasp_impl_->SetVolume(kTestVolume); 49 50 EXPECT_CALL(*mock_sink_.get(), Stop()).Times(verify); 51 wasp_impl_->Stop(); 52 53 testing::Mock::VerifyAndClear(mock_sink_.get()); 54 } 55 56 void SetClient(blink::WebAudioSourceProviderClient* client) { 57 testing::InSequence s; 58 59 if (client) { 60 EXPECT_CALL(*mock_sink_.get(), Stop()); 61 EXPECT_CALL(*this, setFormat(params_.channels(), params_.sample_rate())); 62 } 63 wasp_impl_->setClient(client); 64 base::RunLoop().RunUntilIdle(); 65 66 testing::Mock::VerifyAndClear(mock_sink_.get()); 67 testing::Mock::VerifyAndClear(this); 68 } 69 70 bool CompareBusses(const AudioBus* bus1, const AudioBus* bus2) { 71 EXPECT_EQ(bus1->channels(), bus2->channels()); 72 EXPECT_EQ(bus1->frames(), bus2->frames()); 73 for (int ch = 0; ch < bus1->channels(); ++ch) { 74 if (memcmp(bus1->channel(ch), bus2->channel(ch), 75 sizeof(*bus1->channel(ch)) * bus1->frames()) != 0) { 76 return false; 77 } 78 } 79 return true; 80 } 81 82 // blink::WebAudioSourceProviderClient implementation. 83 MOCK_METHOD2(setFormat, void(size_t numberOfChannels, float sampleRate)); 84 85 protected: 86 AudioParameters params_; 87 FakeAudioRenderCallback fake_callback_; 88 scoped_refptr<MockAudioRendererSink> mock_sink_; 89 scoped_refptr<WebAudioSourceProviderImpl> wasp_impl_; 90 base::MessageLoop message_loop_; 91 92 DISALLOW_COPY_AND_ASSIGN(WebAudioSourceProviderImplTest); 93 }; 94 95 TEST_F(WebAudioSourceProviderImplTest, SetClientBeforeInitialize) { 96 // setClient() with a NULL client should do nothing if no client is set. 97 wasp_impl_->setClient(NULL); 98 99 EXPECT_CALL(*mock_sink_.get(), Stop()); 100 wasp_impl_->setClient(this); 101 base::RunLoop().RunUntilIdle(); 102 103 // When Initialize() is called after setClient(), the params should propagate 104 // to the client via setFormat() during the call. 105 EXPECT_CALL(*this, setFormat(params_.channels(), params_.sample_rate())); 106 wasp_impl_->Initialize(params_, &fake_callback_); 107 base::RunLoop().RunUntilIdle(); 108 109 // setClient() with the same client should do nothing. 110 wasp_impl_->setClient(this); 111 base::RunLoop().RunUntilIdle(); 112 } 113 114 // Verify AudioRendererSink functionality w/ and w/o a client. 115 TEST_F(WebAudioSourceProviderImplTest, SinkMethods) { 116 wasp_impl_->Initialize(params_, &fake_callback_); 117 ASSERT_EQ(mock_sink_->callback(), &fake_callback_); 118 119 // Without a client all WASP calls should fall through to the underlying sink. 120 CallAllSinkMethodsAndVerify(true); 121 122 // With a client no calls should reach the Stop()'d sink. Also, setClient() 123 // should propagate the params provided during Initialize() at call time. 124 SetClient(this); 125 CallAllSinkMethodsAndVerify(false); 126 127 // Removing the client should cause WASP to revert to the underlying sink. 128 EXPECT_CALL(*mock_sink_.get(), SetVolume(kTestVolume)); 129 SetClient(NULL); 130 CallAllSinkMethodsAndVerify(true); 131 } 132 133 // Verify underlying sink state is restored after client removal. 134 TEST_F(WebAudioSourceProviderImplTest, SinkStateRestored) { 135 wasp_impl_->Initialize(params_, &fake_callback_); 136 137 // Verify state set before the client is set propagates back afterward. 138 EXPECT_CALL(*mock_sink_.get(), Start()); 139 wasp_impl_->Start(); 140 SetClient(this); 141 142 EXPECT_CALL(*mock_sink_.get(), SetVolume(1.0)); 143 EXPECT_CALL(*mock_sink_.get(), Start()); 144 SetClient(NULL); 145 146 // Verify state set while the client was attached propagates back afterward. 147 SetClient(this); 148 wasp_impl_->Play(); 149 wasp_impl_->SetVolume(kTestVolume); 150 151 EXPECT_CALL(*mock_sink_.get(), SetVolume(kTestVolume)); 152 EXPECT_CALL(*mock_sink_.get(), Start()); 153 EXPECT_CALL(*mock_sink_.get(), Play()); 154 SetClient(NULL); 155 } 156 157 // Test the AudioRendererSink state machine and its effects on provideInput(). 158 TEST_F(WebAudioSourceProviderImplTest, ProvideInput) { 159 scoped_ptr<AudioBus> bus1 = AudioBus::Create(params_); 160 scoped_ptr<AudioBus> bus2 = AudioBus::Create(params_); 161 162 // Point the WebVector into memory owned by |bus1|. 163 blink::WebVector<float*> audio_data(static_cast<size_t>(bus1->channels())); 164 for (size_t i = 0; i < audio_data.size(); ++i) 165 audio_data[i] = bus1->channel(i); 166 167 // Verify provideInput() works before Initialize() and returns silence. 168 bus1->channel(0)[0] = 1; 169 bus2->Zero(); 170 wasp_impl_->provideInput(audio_data, params_.frames_per_buffer()); 171 ASSERT_TRUE(CompareBusses(bus1.get(), bus2.get())); 172 173 wasp_impl_->Initialize(params_, &fake_callback_); 174 SetClient(this); 175 176 // Verify provideInput() is muted prior to Start() and no calls to the render 177 // callback have occurred. 178 bus1->channel(0)[0] = 1; 179 bus2->Zero(); 180 wasp_impl_->provideInput(audio_data, params_.frames_per_buffer()); 181 ASSERT_TRUE(CompareBusses(bus1.get(), bus2.get())); 182 ASSERT_EQ(fake_callback_.last_audio_delay_milliseconds(), -1); 183 184 wasp_impl_->Start(); 185 186 // Ditto for Play(). 187 bus1->channel(0)[0] = 1; 188 wasp_impl_->provideInput(audio_data, params_.frames_per_buffer()); 189 ASSERT_TRUE(CompareBusses(bus1.get(), bus2.get())); 190 ASSERT_EQ(fake_callback_.last_audio_delay_milliseconds(), -1); 191 192 wasp_impl_->Play(); 193 194 // Now we should get real audio data. 195 wasp_impl_->provideInput(audio_data, params_.frames_per_buffer()); 196 ASSERT_FALSE(CompareBusses(bus1.get(), bus2.get())); 197 198 // Ensure volume adjustment is working. 199 fake_callback_.reset(); 200 fake_callback_.Render(bus2.get(), 0); 201 bus2->Scale(kTestVolume); 202 203 fake_callback_.reset(); 204 wasp_impl_->SetVolume(kTestVolume); 205 wasp_impl_->provideInput(audio_data, params_.frames_per_buffer()); 206 ASSERT_TRUE(CompareBusses(bus1.get(), bus2.get())); 207 208 // Pause should return to silence. 209 wasp_impl_->Pause(); 210 bus1->channel(0)[0] = 1; 211 bus2->Zero(); 212 wasp_impl_->provideInput(audio_data, params_.frames_per_buffer()); 213 ASSERT_TRUE(CompareBusses(bus1.get(), bus2.get())); 214 215 // Ensure if a renderer properly fill silence for partial Render() calls by 216 // configuring the fake callback to return half the data. After these calls 217 // bus1 is full of junk data, and bus2 is partially filled. 218 wasp_impl_->SetVolume(1); 219 fake_callback_.Render(bus1.get(), 0); 220 fake_callback_.reset(); 221 fake_callback_.Render(bus2.get(), 0); 222 bus2->ZeroFramesPartial(bus2->frames() / 2, 223 bus2->frames() - bus2->frames() / 2); 224 fake_callback_.reset(); 225 fake_callback_.set_half_fill(true); 226 wasp_impl_->Play(); 227 228 // Play should return real audio data again, but the last half should be zero. 229 wasp_impl_->provideInput(audio_data, params_.frames_per_buffer()); 230 ASSERT_TRUE(CompareBusses(bus1.get(), bus2.get())); 231 232 // Stop() should return silence. 233 wasp_impl_->Stop(); 234 bus1->channel(0)[0] = 1; 235 bus2->Zero(); 236 wasp_impl_->provideInput(audio_data, params_.frames_per_buffer()); 237 ASSERT_TRUE(CompareBusses(bus1.get(), bus2.get())); 238 } 239 240 } // namespace media 241