// Copyright (c) 2009 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. #include "base/logging.h" #include "base/basictypes.h" #include "base/scoped_ptr.h" #include "base/time.h" #include "media/audio/fake_audio_output_stream.h" #include "media/audio/simple_sources.h" #include "testing/gtest/include/gtest/gtest.h" namespace { void GenerateRandomData(char* buffer, size_t len) { static bool called = false; if (!called) { called = true; int seed = static_cast(base::Time::Now().ToInternalValue()); srand(seed); LOG(INFO) << "Random seed: " << seed; } for (size_t i = 0; i < len; i++) { buffer[i] = static_cast(rand()); } } } // namespace // To test write size smaller than read size. TEST(SimpleSourcesTest, PushSourceSmallerWrite) { const size_t kDataSize = 40960; scoped_array data(new char[kDataSize]); GenerateRandomData(data.get(), kDataSize); // Choose two prime numbers for read and write sizes. const size_t kWriteSize = 283; const size_t kReadSize = 293; scoped_array read_data(new char[kReadSize]); // Create a PushSource that assumes the hardware audio buffer size is always // bigger than the write size. PushSource push_source(kReadSize); EXPECT_EQ(0u, push_source.UnProcessedBytes()); // Write everything into this push source. for (size_t i = 0; i < kDataSize; i += kWriteSize) { size_t size = std::min(kDataSize - i, kWriteSize); EXPECT_TRUE(push_source.Write(data.get() + i, size)); } EXPECT_EQ(kDataSize, push_source.UnProcessedBytes()); // Read everything from the push source. for (size_t i = 0; i < kDataSize; i += kReadSize) { size_t size = std::min(kDataSize - i , kReadSize); EXPECT_EQ(size, push_source.OnMoreData(NULL, read_data.get(), size)); EXPECT_EQ(0, memcmp(data.get() + i, read_data.get(), size)); } EXPECT_EQ(0u, push_source.UnProcessedBytes()); push_source.OnClose(NULL); } // Validate that the SineWaveAudioSource writes the expected values for // the FORMAT_16BIT_MONO. The values are carefully selected so rounding issues // do not affect the result. We also test that AudioManager::GetLastMockBuffer // works. TEST(SimpleSources, SineWaveAudio16MonoTest) { const size_t samples = 1024; const size_t bytes_per_sample = 2; const int freq = 200; SineWaveAudioSource source(SineWaveAudioSource::FORMAT_16BIT_LINEAR_PCM, 1, freq, AudioManager::kTelephoneSampleRate); AudioManager* audio_man = AudioManager::GetAudioManager(); ASSERT_TRUE(NULL != audio_man); AudioOutputStream* oas = audio_man->MakeAudioStream(AudioManager::AUDIO_MOCK, 1, AudioManager::kTelephoneSampleRate, bytes_per_sample * 2); ASSERT_TRUE(NULL != oas); EXPECT_TRUE(oas->Open(samples * bytes_per_sample)); oas->Start(&source); oas->Stop(); oas->Close(); ASSERT_TRUE(FakeAudioOutputStream::GetLastFakeStream()); const int16* last_buffer = reinterpret_cast( FakeAudioOutputStream::GetLastFakeStream()->buffer()); ASSERT_TRUE(NULL != last_buffer); size_t half_period = AudioManager::kTelephoneSampleRate / (freq * 2); // Spot test positive incursion of sine wave. EXPECT_EQ(0, last_buffer[0]); EXPECT_EQ(5126, last_buffer[1]); EXPECT_TRUE(last_buffer[1] < last_buffer[2]); EXPECT_TRUE(last_buffer[2] < last_buffer[3]); // Spot test negative incursion of sine wave. EXPECT_EQ(0, last_buffer[half_period]); EXPECT_EQ(-5126, last_buffer[half_period + 1]); EXPECT_TRUE(last_buffer[half_period + 1] > last_buffer[half_period + 2]); EXPECT_TRUE(last_buffer[half_period + 2] > last_buffer[half_period + 3]); }