diff options
-rw-r--r-- | media/audio/fake_audio_input_stream.cc | 80 | ||||
-rw-r--r-- | media/audio/fake_audio_input_stream.h | 51 | ||||
-rw-r--r-- | media/audio/fake_audio_input_stream_unittest.cc | 51 | ||||
-rw-r--r-- | media/audio/linux/audio_manager_linux.cc | 6 | ||||
-rw-r--r-- | media/audio/mac/audio_manager_mac.cc | 6 | ||||
-rw-r--r-- | media/audio/win/audio_manager_win.cc | 5 | ||||
-rw-r--r-- | media/media.gyp | 3 |
7 files changed, 201 insertions, 1 deletions
diff --git a/media/audio/fake_audio_input_stream.cc b/media/audio/fake_audio_input_stream.cc new file mode 100644 index 0000000..9748d8e --- /dev/null +++ b/media/audio/fake_audio_input_stream.cc @@ -0,0 +1,80 @@ +// Copyright (c) 2010 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 "media/audio/fake_audio_input_stream.h" + +#include "base/at_exit.h" + +using base::Time; +using base::TimeDelta; + +AudioInputStream* FakeAudioInputStream::MakeFakeStream(int channels, + int bits_per_sample, + int sampling_rate, + int samples_per_packet) { + return new FakeAudioInputStream(channels, bits_per_sample, sampling_rate, + samples_per_packet); +} + +FakeAudioInputStream::FakeAudioInputStream(int channels, int bits_per_sample, + int sampling_rate, + int samples_per_packet) + : callback_(NULL), + buffer_size_((channels * bits_per_sample * samples_per_packet) / 8), + thread_("FakeAudioRecordingThread"), + callback_interval_ms_((samples_per_packet * 1000) / sampling_rate) { + // This object is ref counted (so that it can be used with Thread, PostTask) + // but the caller expects a plain pointer. So we take a reference here and + // will Release() ourselves in Close(). + AddRef(); +} + +bool FakeAudioInputStream::Open() { + buffer_.reset(new uint8[buffer_size_]); + memset(buffer_.get(), 0, buffer_size_); + return true; +} + +void FakeAudioInputStream::Start(AudioInputCallback* callback) { + DCHECK(!thread_.IsRunning()); + callback_ = callback; + last_callback_time_ = Time::Now(); + thread_.Start(); + thread_.message_loop()->PostDelayedTask( + FROM_HERE, + NewRunnableMethod(this, &FakeAudioInputStream::DoCallback), + callback_interval_ms_); +} + +void FakeAudioInputStream::DoCallback() { + DCHECK(callback_); + callback_->OnData(this, buffer_.get(), buffer_size_); + + Time now = Time::Now(); + int64 next_callback_ms = (last_callback_time_ + + TimeDelta::FromMilliseconds(callback_interval_ms_ * 2) - + now).InMilliseconds(); + // If we are falling behind, try to catch up as much as we can in the next + // callback. + if (next_callback_ms < 0) + next_callback_ms = 0; + + last_callback_time_ = now; + thread_.message_loop()->PostDelayedTask( + FROM_HERE, + NewRunnableMethod(this, &FakeAudioInputStream::DoCallback), + next_callback_ms); +} + +void FakeAudioInputStream::Stop() { + thread_.Stop(); +} + +void FakeAudioInputStream::Close() { + if (callback_) { + callback_->OnClose(this); + callback_ = NULL; + } + Release(); // Destoys this object. +} diff --git a/media/audio/fake_audio_input_stream.h b/media/audio/fake_audio_input_stream.h new file mode 100644 index 0000000..1cca917 --- /dev/null +++ b/media/audio/fake_audio_input_stream.h @@ -0,0 +1,51 @@ +// Copyright (c) 2010 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. +// +// A fake implementation of AudioInputStream, useful for testing purpose. + +#ifndef MEDIA_AUDIO_FAKE_AUDIO_INPUT_STREAM_H_ +#define MEDIA_AUDIO_FAKE_AUDIO_INOUT_STREAM_H_ + +#include <vector> + +#include "base/ref_counted.h" +#include "base/scoped_ptr.h" +#include "base/thread.h" +#include "base/time.h" +#include "media/audio/audio_io.h" + +class FakeAudioInputStream : + public AudioInputStream, + public base::RefCountedThreadSafe<FakeAudioInputStream> { + public: + static AudioInputStream* MakeFakeStream(int channels, int bits_per_sample, + int sampling_rate, int samples_per_packet); + + virtual bool Open(); + virtual void Start(AudioInputCallback* callback); + virtual void Stop(); + virtual void Close(); + + private: + // Give RefCountedThreadSafe access our destructor. + friend class base::RefCountedThreadSafe<FakeAudioInputStream>; + + FakeAudioInputStream(int channels, int bits_per_sample, int sampling_rate, + int samples_per_packet); + virtual ~FakeAudioInputStream() {} + + void DoCallback(); + + AudioInputCallback* callback_; + scoped_array<uint8> buffer_; + int buffer_size_; + base::Thread thread_; + base::Time last_callback_time_; + int callback_interval_ms_; + + DISALLOW_COPY_AND_ASSIGN(FakeAudioInputStream); +}; + +#endif // MEDIA_AUDIO_FAKE_AUDIO_INPUT_STREAM_H_ + diff --git a/media/audio/fake_audio_input_stream_unittest.cc b/media/audio/fake_audio_input_stream_unittest.cc new file mode 100644 index 0000000..16e86a4 --- /dev/null +++ b/media/audio/fake_audio_input_stream_unittest.cc @@ -0,0 +1,51 @@ +// Copyright (c) 2010 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/basictypes.h" +#include "base/platform_thread.h" +#include "media/audio/audio_io.h" +#include "testing/gmock/include/gmock/gmock.h" +#include "testing/gtest/include/gtest/gtest.h" + +using ::testing::_; +using ::testing::AtLeast; +using ::testing::Exactly; +using ::testing::NotNull; + +namespace { + +class MockAudioInputCallback : public AudioInputStream::AudioInputCallback { + public: + MockAudioInputCallback() {} + + MOCK_METHOD1(OnClose, void(AudioInputStream* stream)); + MOCK_METHOD2(OnError, void(AudioInputStream* stream, int error_code)); + MOCK_METHOD3(OnData, void(AudioInputStream* stream, const uint8* src, + uint32 size)); + private: + DISALLOW_COPY_AND_ASSIGN(MockAudioInputCallback); +}; + +} + +// ============================================================================ +// Validate that the AudioManager::AUDIO_MOCK callbacks work. +TEST(FakeAudioInputTest, BasicCallbacks) { + MockAudioInputCallback callback; + EXPECT_CALL(callback, OnData(NotNull(), _, _)).Times(AtLeast(5)); + EXPECT_CALL(callback, OnError(NotNull(), _)).Times(Exactly(0)); + + AudioManager* audio_man = AudioManager::GetAudioManager(); + ASSERT_TRUE(NULL != audio_man); + // Ask for one recorded packet every 50ms. + AudioInputStream* stream = audio_man->MakeAudioInputStream( + AudioManager::AUDIO_MOCK, 2, 8000, 8, 400); + ASSERT_TRUE(NULL != stream); + EXPECT_TRUE(stream->Open()); + stream->Start(&callback); + PlatformThread::Sleep(340); // Give sufficient time to receive 5 / 6 packets. + stream->Stop(); + stream->Close(); +} + diff --git a/media/audio/linux/audio_manager_linux.cc b/media/audio/linux/audio_manager_linux.cc index 6822c8b..17a1f74 100644 --- a/media/audio/linux/audio_manager_linux.cc +++ b/media/audio/linux/audio_manager_linux.cc @@ -7,6 +7,7 @@ #include "base/at_exit.h" #include "base/command_line.h" #include "base/logging.h" +#include "media/audio/fake_audio_input_stream.h" #include "media/audio/fake_audio_output_stream.h" #include "media/audio/linux/alsa_output.h" #include "media/audio/linux/alsa_wrapper.h" @@ -35,6 +36,11 @@ AudioInputStream* AudioManagerLinux::MakeAudioInputStream( int sample_rate, char bits_per_sample, uint32 samples_per_packet) { + if (format == AUDIO_MOCK) { + return FakeAudioInputStream::MakeFakeStream(channels, bits_per_sample, + sample_rate, + samples_per_packet); + } // TODO(satish): implement. return NULL; } diff --git a/media/audio/mac/audio_manager_mac.cc b/media/audio/mac/audio_manager_mac.cc index 84f76de..312ecbb 100644 --- a/media/audio/mac/audio_manager_mac.cc +++ b/media/audio/mac/audio_manager_mac.cc @@ -5,6 +5,7 @@ #include <CoreAudio/AudioHardware.h> #include "base/at_exit.h" +#include "media/audio/fake_audio_input_stream.h" #include "media/audio/fake_audio_output_stream.h" #include "media/audio/mac/audio_manager_mac.h" #include "media/audio/mac/audio_output_mac.h" @@ -38,6 +39,11 @@ AudioInputStream* AudioManagerMac::MakeAudioInputStream( int sample_rate, char bits_per_sample, uint32 samples_per_packet) { + if (format == AUDIO_MOCK) { + return FakeAudioInputStream::MakeFakeStream(channels, bits_per_sample, + sample_rate, + samples_per_packet); + } // TODO(satish): implement. return NULL; } diff --git a/media/audio/win/audio_manager_win.cc b/media/audio/win/audio_manager_win.cc index 0c42015..061d2d8 100644 --- a/media/audio/win/audio_manager_win.cc +++ b/media/audio/win/audio_manager_win.cc @@ -9,6 +9,7 @@ #include "base/at_exit.h" #include "base/basictypes.h" +#include "media/audio/fake_audio_input_stream.h" #include "media/audio/fake_audio_output_stream.h" #include "media/audio/win/audio_manager_win.h" #include "media/audio/win/wavein_input_win.h" @@ -94,7 +95,9 @@ AudioInputStream* AudioManagerWin::MakeAudioInputStream( return NULL; if (format == AUDIO_MOCK) { - // TODO(satish): Add mock audio input stream. + return FakeAudioInputStream::MakeFakeStream(channels, bits_per_sample, + sample_rate, + samples_per_packet); } else if (format == AUDIO_PCM_LINEAR) { return new PCMWaveInAudioInputStream(this, channels, sample_rate, kNumInputBuffers, bits_per_sample, diff --git a/media/media.gyp b/media/media.gyp index 86966d6..554dcd1 100644 --- a/media/media.gyp +++ b/media/media.gyp @@ -25,6 +25,8 @@ 'audio/audio_output_controller.h', 'audio/audio_util.cc', 'audio/audio_util.h', + 'audio/fake_audio_input_stream.cc', + 'audio/fake_audio_input_stream.h', 'audio/fake_audio_output_stream.cc', 'audio/fake_audio_output_stream.h', 'audio/linux/audio_manager_linux.cc', @@ -190,6 +192,7 @@ 'sources': [ 'audio/audio_output_controller_unittest.cc', 'audio/audio_util_unittest.cc', + 'audio/fake_audio_input_stream_unittest.cc', 'audio/linux/alsa_output_unittest.cc', 'audio/mac/audio_output_mac_unittest.cc', 'audio/simple_sources_unittest.cc', |