From 296a0ec7e149b6b373b9271683eca286b1c376ed Mon Sep 17 00:00:00 2001 From: Eric Laurent Date: Tue, 15 Sep 2009 07:10:12 -0700 Subject: Fix issue 2118464: cannot play ring tones and notifications after disconnecting BT headset while in call. The problem comes from the fact that when the duplicated output is closed after BT headset disconnection, the OUTPUT_CLOSED notification is not sent to AudioSystem. Then the mapping between notification stream and duplicated output cached in AudioSystem is not cleared and next time a notification is played, the duplicated output is selected and the createTrack() request is refused by AudioFlinger as the selected output doesn't exist. The notification is ignored by AudioFlinger because when it is sent by the terminating playback thread, the thread has already been removed from the playback thread list. The fix consists in sending the notification in closeOutput() and not when exiting the playback thread. The same fix is applied to record threads. --- libs/audioflinger/AudioFlinger.cpp | 30 +++++++++++++----------------- libs/audioflinger/AudioFlinger.h | 2 +- 2 files changed, 14 insertions(+), 18 deletions(-) (limited to 'libs/audioflinger') diff --git a/libs/audioflinger/AudioFlinger.cpp b/libs/audioflinger/AudioFlinger.cpp index d9be007..4f7500f 100644 --- a/libs/audioflinger/AudioFlinger.cpp +++ b/libs/audioflinger/AudioFlinger.cpp @@ -677,8 +677,8 @@ void AudioFlinger::binderDied(const wp& who) { } } -void AudioFlinger::audioConfigChanged(int event, const sp& thread, void *param2) { - Mutex::Autolock _l(mLock); +// audioConfigChanged_l() must be called with AudioFlinger::mLock held +void AudioFlinger::audioConfigChanged_l(int event, const sp& thread, void *param2) { int ioHandle = 0; for (size_t i = 0; i < mPlaybackThreads.size(); i++) { @@ -700,7 +700,7 @@ void AudioFlinger::audioConfigChanged(int event, const sp& thread, v size_t size = mNotificationClients.size(); for (size_t i = 0; i < size; i++) { sp binder = mNotificationClients.itemAt(i); - LOGV("audioConfigChanged() Notifying change to client %p", binder.get()); + LOGV("audioConfigChanged_l() Notifying change to client %p", binder.get()); sp client = interface_cast (binder); client->ioConfigChanged(event, ioHandle, param2); } @@ -803,8 +803,8 @@ void AudioFlinger::ThreadBase::processConfigEvents() LOGV("processConfigEvents() remaining events %d", mConfigEvents.size()); ConfigEvent *configEvent = mConfigEvents[0]; mConfigEvents.removeAt(0); - // release mLock because audioConfigChanged() will call - // Audioflinger::audioConfigChanged() which locks AudioFlinger mLock thus creating + // release mLock because audioConfigChanged() will lock AudioFlinger mLock + // before calling Audioflinger::audioConfigChanged_l() thus creating // potential cross deadlock between AudioFlinger::mLock and mLock mLock.unlock(); audioConfigChanged(configEvent->mEvent, configEvent->mParam); @@ -1118,7 +1118,8 @@ void AudioFlinger::PlaybackThread::audioConfigChanged(int event, int param) { default: break; } - mAudioFlinger->audioConfigChanged(event, this, param2); + Mutex::Autolock _l(mAudioFlinger->mLock); + mAudioFlinger->audioConfigChanged_l(event, this, param2); } void AudioFlinger::PlaybackThread::readOutputParameters() @@ -1272,8 +1273,6 @@ bool AudioFlinger::MixerThread::threadLoop() if (!mStandby) { mOutput->standby(); } - sendConfigEvent(AudioSystem::OUTPUT_CLOSED); - processConfigEvents(); LOGV("MixerThread %p exiting", this); return false; @@ -1772,8 +1771,6 @@ bool AudioFlinger::DirectOutputThread::threadLoop() if (!mStandby) { mOutput->standby(); } - sendConfigEvent(AudioSystem::OUTPUT_CLOSED); - processConfigEvents(); LOGV("DirectOutputThread %p exiting", this); return false; @@ -1965,9 +1962,6 @@ bool AudioFlinger::DuplicatingThread::threadLoop() } } - sendConfigEvent(AudioSystem::OUTPUT_CLOSED); - processConfigEvents(); - return false; } @@ -3047,9 +3041,6 @@ bool AudioFlinger::RecordThread::threadLoop() } mActiveTrack.clear(); - sendConfigEvent(AudioSystem::INPUT_CLOSED); - processConfigEvents(); - LOGV("RecordThread %p exiting", this); return false; } @@ -3234,7 +3225,8 @@ void AudioFlinger::RecordThread::audioConfigChanged(int event, int param) { default: break; } - mAudioFlinger->audioConfigChanged(event, this, param2); + Mutex::Autolock _l(mAudioFlinger->mLock); + mAudioFlinger->audioConfigChanged_l(event, this, param2); } void AudioFlinger::RecordThread::readInputParameters() @@ -3379,6 +3371,8 @@ status_t AudioFlinger::closeOutput(int output) } } } + void *param2 = 0; + audioConfigChanged_l(AudioSystem::OUTPUT_CLOSED, thread, param2); mPlaybackThreads.removeItem(output); } thread->exit(); @@ -3498,6 +3492,8 @@ status_t AudioFlinger::closeInput(int input) } LOGV("closeInput() %d", input); + void *param2 = 0; + audioConfigChanged_l(AudioSystem::INPUT_CLOSED, thread, param2); mRecordThreads.removeItem(input); } thread->exit(); diff --git a/libs/audioflinger/AudioFlinger.h b/libs/audioflinger/AudioFlinger.h index 65c148e..7a6641f 100644 --- a/libs/audioflinger/AudioFlinger.h +++ b/libs/audioflinger/AudioFlinger.h @@ -616,7 +616,7 @@ private: MixerThread *checkMixerThread_l(int output) const; RecordThread *checkRecordThread_l(int input) const; float streamVolumeInternal(int stream) const { return mStreamTypes[stream].volume; } - void audioConfigChanged(int event, const sp& thread, void *param2); + void audioConfigChanged_l(int event, const sp& thread, void *param2); friend class AudioBuffer; -- cgit v1.1