Fix volume setting while not playing on PulseAudio.

We now only set the volume when starting playout if the user has called
SetSpeakerVolume while we weren't playing. This now also ensures it will
actually get set to what the user requested rather than being overridden
by a default value.

Add tests to voe_auto_test.

BUG=6140661
TEST=voe_auto_test

Review URL: https://webrtc-codereview.appspot.com/566006

git-svn-id: http://webrtc.googlecode.com/svn/trunk@2188 4adac7df-926f-26a2-2b94-8c16560cd09d
This commit is contained in:
andrew@webrtc.org
2012-05-07 21:42:49 +00:00
parent 20a05cd778
commit 589673f1cb
6 changed files with 82 additions and 26 deletions

View File

@@ -93,6 +93,7 @@ AudioDeviceLinuxPulse::AudioDeviceLinuxPulse(const WebRtc_Word32 id) :
_startPlay(false), _startPlay(false),
_stopPlay(false), _stopPlay(false),
_AGC(false), _AGC(false),
update_speaker_volume_at_startup_(false),
_playBufDelayFixed(20), _playBufDelayFixed(20),
_sndCardPlayDelay(0), _sndCardPlayDelay(0),
_sndCardRecDelay(0), _sndCardRecDelay(0),
@@ -127,7 +128,9 @@ AudioDeviceLinuxPulse::AudioDeviceLinuxPulse(const WebRtc_Word32 id) :
_recStream(NULL), _recStream(NULL),
_playStream(NULL), _playStream(NULL),
_recStreamFlags(0), _recStreamFlags(0),
_playStreamFlags(0) _playStreamFlags(0),
_playBufferAttr(),
_recBufferAttr()
{ {
WEBRTC_TRACE(kTraceMemory, kTraceAudioDevice, id, WEBRTC_TRACE(kTraceMemory, kTraceAudioDevice, id,
"%s created", __FUNCTION__); "%s created", __FUNCTION__);
@@ -539,7 +542,10 @@ WebRtc_Word32 AudioDeviceLinuxPulse::SpeakerVolumeIsAvailable(bool& available)
WebRtc_Word32 AudioDeviceLinuxPulse::SetSpeakerVolume(WebRtc_UWord32 volume) WebRtc_Word32 AudioDeviceLinuxPulse::SetSpeakerVolume(WebRtc_UWord32 volume)
{ {
if (!_playing) {
// Only update the volume if it's been set while we weren't playing.
update_speaker_volume_at_startup_ = true;
}
return (_mixerManager.SetSpeakerVolume(volume)); return (_mixerManager.SetSpeakerVolume(volume));
} }
@@ -2818,15 +2824,23 @@ bool AudioDeviceLinuxPulse::PlayThreadProcess()
// Get the currently saved speaker volume // Get the currently saved speaker volume
WebRtc_UWord32 volume = 0; WebRtc_UWord32 volume = 0;
_mixerManager.SpeakerVolume(volume); if (update_speaker_volume_at_startup_)
_mixerManager.SpeakerVolume(volume);
PaLock(); PaLock();
// Set the same volume for all channels // NULL gives PA the choice of startup volume.
pa_cvolume cVolumes; pa_cvolume* ptr_cvolume = NULL;
const pa_sample_spec *spec = if (update_speaker_volume_at_startup_) {
LATE(pa_stream_get_sample_spec)(_playStream); pa_cvolume cVolumes;
LATE(pa_cvolume_set)(&cVolumes, spec->channels, volume); ptr_cvolume = &cVolumes;
// Set the same volume for all channels
const pa_sample_spec *spec =
LATE(pa_stream_get_sample_spec)(_playStream);
LATE(pa_cvolume_set)(&cVolumes, spec->channels, volume);
update_speaker_volume_at_startup_ = false;
}
// Connect the stream to a sink // Connect the stream to a sink
if (LATE(pa_stream_connect_playback)( if (LATE(pa_stream_connect_playback)(
@@ -2834,7 +2848,7 @@ bool AudioDeviceLinuxPulse::PlayThreadProcess()
_playDeviceName, _playDeviceName,
&_playBufferAttr, &_playBufferAttr,
(pa_stream_flags_t) _playStreamFlags, (pa_stream_flags_t) _playStreamFlags,
&cVolumes, NULL) != PA_OK) ptr_cvolume, NULL) != PA_OK)
{ {
WEBRTC_TRACE(kTraceError, kTraceAudioDevice, _id, WEBRTC_TRACE(kTraceError, kTraceAudioDevice, _id,
" failed to connect play stream, err=%d", " failed to connect play stream, err=%d",

View File

@@ -330,6 +330,7 @@ private:
bool _startPlay; bool _startPlay;
bool _stopPlay; bool _stopPlay;
bool _AGC; bool _AGC;
bool update_speaker_volume_at_startup_;
private: private:
WebRtc_UWord16 _playBufDelayFixed; // fixed playback delay WebRtc_UWord16 _playBufDelayFixed; // fixed playback delay

View File

@@ -39,7 +39,7 @@ AudioMixerManagerLinuxPulse::AudioMixerManagerLinuxPulse(const WebRtc_Word32 id)
_paMute(0), _paMute(0),
_paVolSteps(0), _paVolSteps(0),
_paSpeakerMute(false), _paSpeakerMute(false),
_paSpeakerVolume(0), _paSpeakerVolume(PA_VOLUME_NORM),
_paChannels(0), _paChannels(0),
_paObjectsSet(false), _paObjectsSet(false),
_callbackValues(false) _callbackValues(false)
@@ -176,9 +176,6 @@ WebRtc_Word32 AudioMixerManagerLinuxPulse::OpenSpeaker(
// output device to control // output device to control
_paOutputDeviceIndex = deviceIndex; _paOutputDeviceIndex = deviceIndex;
// Init the speaker volume to the normal volume
_paSpeakerVolume = PA_VOLUME_NORM;
WEBRTC_TRACE(kTraceInfo, kTraceAudioDevice, _id, WEBRTC_TRACE(kTraceInfo, kTraceAudioDevice, _id,
" the output mixer device is now open"); " the output mixer device is now open");

View File

@@ -22,14 +22,13 @@ AfterStreamingFixture::AfterStreamingFixture()
EXPECT_FALSE(fake_microphone_input_file_.empty()); EXPECT_FALSE(fake_microphone_input_file_.empty());
SetUpLocalPlayback(); SetUpLocalPlayback();
StartPlaying(); ResumePlaying();
RestartFakeMicrophone();
} }
AfterStreamingFixture::~AfterStreamingFixture() { AfterStreamingFixture::~AfterStreamingFixture() {
voe_file_->StopPlayingFileAsMicrophone(channel_); voe_file_->StopPlayingFileAsMicrophone(channel_);
voe_base_->StopSend(channel_); PausePlaying();
voe_base_->StopPlayout(channel_);
voe_base_->StopReceive(channel_);
voe_base_->DeleteChannel(channel_); voe_base_->DeleteChannel(channel_);
} }
@@ -47,6 +46,18 @@ void AfterStreamingFixture::RestartFakeMicrophone() {
channel_, fake_microphone_input_file_.c_str(), true, true)); channel_, fake_microphone_input_file_.c_str(), true, true));
} }
void AfterStreamingFixture::PausePlaying() {
EXPECT_EQ(0, voe_base_->StopSend(channel_));
EXPECT_EQ(0, voe_base_->StopPlayout(channel_));
EXPECT_EQ(0, voe_base_->StopReceive(channel_));
}
void AfterStreamingFixture::ResumePlaying() {
EXPECT_EQ(0, voe_base_->StartReceive(channel_));
EXPECT_EQ(0, voe_base_->StartPlayout(channel_));
EXPECT_EQ(0, voe_base_->StartSend(channel_));
}
void AfterStreamingFixture::SetUpLocalPlayback() { void AfterStreamingFixture::SetUpLocalPlayback() {
EXPECT_EQ(0, voe_base_->SetSendDestination(channel_, 8000, kLoopbackIp)); EXPECT_EQ(0, voe_base_->SetSendDestination(channel_, 8000, kLoopbackIp));
EXPECT_EQ(0, voe_base_->SetLocalReceiver(0, 8000)); EXPECT_EQ(0, voe_base_->SetLocalReceiver(0, 8000));
@@ -61,10 +72,3 @@ void AfterStreamingFixture::SetUpLocalPlayback() {
voe_codec_->SetSendCodec(channel_, codec); voe_codec_->SetSendCodec(channel_, codec);
} }
void AfterStreamingFixture::StartPlaying() {
EXPECT_EQ(0, voe_base_->StartReceive(channel_));
EXPECT_EQ(0, voe_base_->StartPlayout(channel_));
EXPECT_EQ(0, voe_base_->StartSend(channel_));
RestartFakeMicrophone();
}

View File

@@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2011 The WebRTC project authors. All Rights Reserved. * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
* *
* Use of this source code is governed by a BSD-style license * Use of this source code is governed by a BSD-style license
* that can be found in the LICENSE file in the root of the source * that can be found in the LICENSE file in the root of the source
@@ -34,9 +34,14 @@ class AfterStreamingFixture : public AfterInitializationFixture {
// Restarts the fake microphone if it's been shut off earlier. // Restarts the fake microphone if it's been shut off earlier.
void RestartFakeMicrophone(); void RestartFakeMicrophone();
// Stops all sending and playout.
void PausePlaying();
// Resumes all sending and playout.
void ResumePlaying();
private: private:
void SetUpLocalPlayback(); void SetUpLocalPlayback();
void StartPlaying();
}; };

View File

@@ -19,6 +19,34 @@ TEST_F(VolumeTest, DefaultSpeakerVolumeIsAtMost255) {
EXPECT_LE(volume, 255u); EXPECT_LE(volume, 255u);
} }
TEST_F(VolumeTest, SetVolumeBeforePlayoutWorks) {
// This is a rather specialized test, intended to exercise some PulseAudio
// code. However, these conditions should be satisfied on any platform.
unsigned int original_volume = 0;
EXPECT_EQ(0, voe_volume_control_->GetSpeakerVolume(original_volume));
Sleep(1000);
EXPECT_EQ(0, voe_volume_control_->SetSpeakerVolume(200));
unsigned int volume;
EXPECT_EQ(0, voe_volume_control_->GetSpeakerVolume(volume));
EXPECT_EQ(200u, volume);
PausePlaying();
ResumePlaying();
EXPECT_EQ(0, voe_volume_control_->GetSpeakerVolume(volume));
// Ensure the volume has not changed after resuming playout.
EXPECT_EQ(200u, volume);
PausePlaying();
EXPECT_EQ(0, voe_volume_control_->SetSpeakerVolume(100));
ResumePlaying();
// Ensure the volume set while paused is retained.
EXPECT_EQ(0, voe_volume_control_->GetSpeakerVolume(volume));
EXPECT_EQ(100u, volume);
EXPECT_EQ(0, voe_volume_control_->SetSpeakerVolume(original_volume));
}
TEST_F(VolumeTest, ManualSetVolumeWorks) { TEST_F(VolumeTest, ManualSetVolumeWorks) {
unsigned int original_volume = 0; unsigned int original_volume = 0;
EXPECT_EQ(0, voe_volume_control_->GetSpeakerVolume(original_volume)); EXPECT_EQ(0, voe_volume_control_->GetSpeakerVolume(original_volume));
@@ -26,15 +54,22 @@ TEST_F(VolumeTest, ManualSetVolumeWorks) {
TEST_LOG("Setting speaker volume to 0 out of 255.\n"); TEST_LOG("Setting speaker volume to 0 out of 255.\n");
EXPECT_EQ(0, voe_volume_control_->SetSpeakerVolume(0)); EXPECT_EQ(0, voe_volume_control_->SetSpeakerVolume(0));
unsigned int volume;
EXPECT_EQ(0, voe_volume_control_->GetSpeakerVolume(volume));
EXPECT_EQ(0u, volume);
Sleep(1000); Sleep(1000);
TEST_LOG("Setting speaker volume to 100 out of 255.\n"); TEST_LOG("Setting speaker volume to 100 out of 255.\n");
EXPECT_EQ(0, voe_volume_control_->SetSpeakerVolume(100)); EXPECT_EQ(0, voe_volume_control_->SetSpeakerVolume(100));
EXPECT_EQ(0, voe_volume_control_->GetSpeakerVolume(volume));
EXPECT_EQ(100u, volume);
Sleep(1000); Sleep(1000);
// Set the volume to 255 very briefly so we don't blast the poor user // Set the volume to 255 very briefly so we don't blast the poor user
// listening to this. This is just to test the call succeeds. // listening to this. This is just to test the call succeeds.
EXPECT_EQ(0, voe_volume_control_->SetSpeakerVolume(255)); EXPECT_EQ(0, voe_volume_control_->SetSpeakerVolume(255));
EXPECT_EQ(0, voe_volume_control_->GetSpeakerVolume(volume));
EXPECT_EQ(255u, volume);
TEST_LOG("Setting speaker volume to the original %d out of 255.\n", TEST_LOG("Setting speaker volume to the original %d out of 255.\n",
original_volume); original_volume);