// Copyright (c) 2012 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 #include "base/logging.h" #include "base/memory/scoped_ptr.h" #include "media/audio/audio_io.h" #include "media/audio/audio_manager_base.h" #include "media/audio/audio_util.h" #include "testing/gtest/include/gtest/gtest.h" #if defined(OS_WIN) #include "base/win/scoped_com_initializer.h" #include "media/audio/win/core_audio_util_win.h" #endif namespace media { class AudioInputVolumeTest : public ::testing::Test { protected: AudioInputVolumeTest() : audio_manager_(AudioManager::Create()) #if defined(OS_WIN) , com_init_(base::win::ScopedCOMInitializer::kMTA) #endif { } bool CanRunAudioTests() { #if defined(OS_WIN) // TODO(henrika): add support for volume control on Windows XP as well. // For now, we might as well signal false already here to avoid running // these tests on Windows XP. if (!CoreAudioUtil::IsSupported()) return false; #endif if (!audio_manager_.get()) return false; return audio_manager_->HasAudioInputDevices(); } // Helper method which checks if the stream has volume support. bool HasDeviceVolumeControl(AudioInputStream* stream) { if (!stream) return false; return (stream->GetMaxVolume() != 0.0); } AudioInputStream* CreateAndOpenStream(const std::string& device_id) { AudioParameters::Format format = AudioParameters::AUDIO_PCM_LOW_LATENCY; ChannelLayout channel_layout = media::GetAudioInputHardwareChannelLayout(device_id); int bits_per_sample = 16; int sample_rate = static_cast(media::GetAudioInputHardwareSampleRate(device_id)); int samples_per_packet = 0; #if defined(OS_MACOSX) samples_per_packet = (sample_rate / 100); #elif defined(OS_LINUX) || defined(OS_OPENBSD) samples_per_packet = (sample_rate / 100); #elif defined(OS_WIN) if (sample_rate == 44100) samples_per_packet = 448; else samples_per_packet = (sample_rate / 100); #else #error Unsupported platform #endif AudioInputStream* ais = audio_manager_->MakeAudioInputStream( AudioParameters(format, channel_layout, sample_rate, bits_per_sample, samples_per_packet), device_id); EXPECT_TRUE(NULL != ais); #if defined(OS_LINUX) || defined(OS_OPENBSD) // Some linux devices do not support our settings, we may fail to open // those devices. if (!ais->Open()) { // Default device should always be able to be opened. EXPECT_TRUE(AudioManagerBase::kDefaultDeviceId != device_id); ais->Close(); ais = NULL; } #elif defined(OS_WIN) || defined(OS_MACOSX) EXPECT_TRUE(ais->Open()); #endif return ais; } scoped_ptr audio_manager_; #if defined(OS_WIN) base::win::ScopedCOMInitializer com_init_; #endif }; TEST_F(AudioInputVolumeTest, InputVolumeTest) { if (!CanRunAudioTests()) return; // Retrieve a list of all available input devices. AudioDeviceNames device_names; audio_manager_->GetAudioInputDeviceNames(&device_names); if (device_names.empty()) { LOG(WARNING) << "Could not find any available input device"; return; } // Scan all available input devices and repeat the same test for all of them. for (AudioDeviceNames::const_iterator it = device_names.begin(); it != device_names.end(); ++it) { AudioInputStream* ais = CreateAndOpenStream(it->unique_id); if (!ais) { DLOG(WARNING) << "Failed to open stream for device " << it->unique_id; continue; } if (!HasDeviceVolumeControl(ais)) { DLOG(WARNING) << "Device: " << it->unique_id << ", does not have volume control."; ais->Close(); continue; } double max_volume = ais->GetMaxVolume(); EXPECT_GT(max_volume, 0.0); // Store the current input-device volume level. double original_volume = ais->GetVolume(); EXPECT_GE(original_volume, 0.0); #if defined(OS_WIN) || defined(OS_MACOSX) // Note that |original_volume| can be higher than |max_volume| on Linux. EXPECT_LE(original_volume, max_volume); #endif // Set the volume to the maxiumum level.. ais->SetVolume(max_volume); double current_volume = ais->GetVolume(); EXPECT_EQ(max_volume, current_volume); // Set the volume to the mininum level (=0). double new_volume = 0.0; ais->SetVolume(new_volume); current_volume = ais->GetVolume(); EXPECT_EQ(new_volume, current_volume); // Set the volume to the mid level (50% of max). // Verify that the absolute error is small enough. new_volume = max_volume / 2; ais->SetVolume(new_volume); current_volume = ais->GetVolume(); EXPECT_LT(current_volume, max_volume); EXPECT_GT(current_volume, 0); EXPECT_NEAR(current_volume, new_volume, 0.25 * max_volume); // Restores the volume to the original value. ais->SetVolume(original_volume); current_volume = ais->GetVolume(); EXPECT_EQ(original_volume, current_volume); ais->Close(); } } } // namespace media