diff options
author | Mathias Agopian <mathias@google.com> | 2011-12-05 21:14:03 -0800 |
---|---|---|
committer | Android (Google) Code Review <android-gerrit@google.com> | 2011-12-05 21:14:03 -0800 |
commit | 82fbc9be7cc7a7d98ac0cd791e7d4fe9f7deaa92 (patch) | |
tree | e3f03bf484900ec20f9da7867179c939d0012003 /services | |
parent | f697e5d6b54e84f67b41ad8bbd933af05244c5a3 (diff) | |
parent | 2871ce953c6e02db5a7fa1b5dda7cd91e0f6a305 (diff) | |
download | frameworks_base-82fbc9be7cc7a7d98ac0cd791e7d4fe9f7deaa92.zip frameworks_base-82fbc9be7cc7a7d98ac0cd791e7d4fe9f7deaa92.tar.gz frameworks_base-82fbc9be7cc7a7d98ac0cd791e7d4fe9f7deaa92.tar.bz2 |
Merge "fix a deadlock when removing a DisplayEventConnection"
Diffstat (limited to 'services')
-rw-r--r-- | services/surfaceflinger/EventThread.cpp | 64 | ||||
-rw-r--r-- | services/surfaceflinger/EventThread.h | 3 |
2 files changed, 46 insertions, 21 deletions
diff --git a/services/surfaceflinger/EventThread.cpp b/services/surfaceflinger/EventThread.cpp index edb06ba..42477a9 100644 --- a/services/surfaceflinger/EventThread.cpp +++ b/services/surfaceflinger/EventThread.cpp @@ -60,36 +60,51 @@ status_t EventThread::unregisterDisplayEventConnection( return NO_ERROR; } +status_t EventThread::removeDisplayEventConnection( + const wp<DisplayEventConnection>& connection) { + Mutex::Autolock _l(mLock); + mDisplayEventConnections.remove(connection); + return NO_ERROR; +} + bool EventThread::threadLoop() { nsecs_t timestamp; - Mutex::Autolock _l(mLock); - do { - // wait for listeners - while (!mDisplayEventConnections.size()) { - mCondition.wait(mLock); - } + DisplayEventReceiver::Event vsync; + SortedVector<wp<DisplayEventConnection> > displayEventConnections; + + { // scope for the lock + Mutex::Autolock _l(mLock); + do { + // wait for listeners + while (!mDisplayEventConnections.size()) { + mCondition.wait(mLock); + } - // wait for vsync - mLock.unlock(); - timestamp = mHw.waitForVSync(); - mLock.lock(); + // wait for vsync + mLock.unlock(); + timestamp = mHw.waitForVSync(); + mLock.lock(); - // make sure we still have some listeners - } while (!mDisplayEventConnections.size()); + // make sure we still have some listeners + } while (!mDisplayEventConnections.size()); - // dispatch vsync events to listeners... - mDeliveredEvents++; - const size_t count = mDisplayEventConnections.size(); + // dispatch vsync events to listeners... + mDeliveredEvents++; - DisplayEventReceiver::Event vsync; - vsync.header.type = DisplayEventReceiver::DISPLAY_EVENT_VSYNC; - vsync.header.timestamp = timestamp; - vsync.vsync.count = mDeliveredEvents; + vsync.header.type = DisplayEventReceiver::DISPLAY_EVENT_VSYNC; + vsync.header.timestamp = timestamp; + vsync.vsync.count = mDeliveredEvents; + // make a copy of our connection list, so we can + // dispatch events without holding mLock + displayEventConnections = mDisplayEventConnections; + } + + const size_t count = displayEventConnections.size(); for (size_t i=0 ; i<count ; i++) { - sp<DisplayEventConnection> conn(mDisplayEventConnections.itemAt(i).promote()); + sp<DisplayEventConnection> conn(displayEventConnections.itemAt(i).promote()); // make sure the connection didn't die if (conn != NULL) { status_t err = conn->postEvent(vsync); @@ -103,11 +118,18 @@ bool EventThread::threadLoop() { // handle any other error on the pipe as fatal. the only // reasonable thing to do is to clean-up this connection. // The most common error we'll get here is -EPIPE. - mDisplayEventConnections.remove(conn); + removeDisplayEventConnection(displayEventConnections.itemAt(i)); } + } else { + // somehow the connection is dead, but we still have it in our list + // just clean the list. + removeDisplayEventConnection(displayEventConnections.itemAt(i)); } } + // clear all our references without holding mLock + displayEventConnections.clear(); + return true; } diff --git a/services/surfaceflinger/EventThread.h b/services/surfaceflinger/EventThread.h index 0482ab7..4872c2b 100644 --- a/services/surfaceflinger/EventThread.h +++ b/services/surfaceflinger/EventThread.h @@ -58,6 +58,9 @@ private: virtual status_t readyToRun(); virtual void onFirstRef(); + status_t removeDisplayEventConnection( + const wp<DisplayEventConnection>& connection); + // constants sp<SurfaceFlinger> mFlinger; const DisplayHardware& mHw; |