diff options
| author | Robert Carr <racarr@google.com> | 2020-05-13 11:23:43 -0700 |
|---|---|---|
| committer | Rob Carr <racarr@google.com> | 2020-05-13 18:29:59 +0000 |
| commit | ebaaca1a46b792d4d79f87d02059686ad30b4fa2 (patch) | |
| tree | 48c383a3b955d4ed677996ce3cecedf4c5891da8 /core/java/android/view/SurfaceView.java | |
| parent | 43d32906a8a51b37bbb118c8febd4cea341331b4 (diff) | |
SurfaceView: positionLost locking fix
positionLost can be called from CanvasContext::destroyHardwareResources
which runs asynchronously to the UI thread. This means we could be
simultaneously executing releaseSurfaces on the UI thread. We need
to expand the scope of mSurfaceControl lock in positionLost. While
we are here we add a block comment explaining the previously
undocumented locking strategy.
Bug: 156264048
Test: Existing tests pass
Change-Id: I9cdb6a0f7aeffd878f1755f240e8896f0fb8bf01
Diffstat (limited to 'core/java/android/view/SurfaceView.java')
| -rw-r--r-- | core/java/android/view/SurfaceView.java | 35 |
1 files changed, 28 insertions, 7 deletions
diff --git a/core/java/android/view/SurfaceView.java b/core/java/android/view/SurfaceView.java index db6fe0f57d06..bd811fc1f052 100644 --- a/core/java/android/view/SurfaceView.java +++ b/core/java/android/view/SurfaceView.java @@ -134,6 +134,23 @@ public class SurfaceView extends View implements ViewRootImpl.SurfaceChangedCall // we need to preserve the old one until the new one has drawn. SurfaceControl mDeferredDestroySurfaceControl; SurfaceControl mBackgroundControl; + + /** + * We use this lock in SOME cases when reading or writing SurfaceControl, + * but use the following model so that the RenderThread can run locklessly + * in the position up-date case. + * + * 1. UI Thread can read from mSurfaceControl (use in Transactions) without + * holding the lock. + * 2. UI Thread will hold the lock when writing to mSurfaceControl (calling release + * or remove). + * 3. Render thread will also hold the lock when writing to mSurfaceControl (e.g. + * calling release from positionLost). + * 3. RenderNode.PositionUpdateListener::positionChanged will only be called + * when the UI thread is paused (blocked on the Render thread). + * 4. positionChanged thus will not be required to hold the lock as the + * UI thread is blocked, and the other writer is the RT itself. + */ final Object mSurfaceControlLock = new Object(); final Rect mTmpRect = new Rect(); @@ -1297,15 +1314,19 @@ public class SurfaceView extends View implements ViewRootImpl.SurfaceChangedCall (viewRoot != null ? viewRoot.getBLASTSyncTransaction() : mRtTransaction) : mRtTransaction; - if (frameNumber > 0 && viewRoot != null && !useBLAST) { - if (viewRoot.mSurface.isValid()) { - mRtTransaction.deferTransactionUntil(mSurfaceControl, - viewRoot.getRenderSurfaceControl(), frameNumber); + /** + * positionLost can be called while UI thread is un-paused so we + * need to hold the lock here. + */ + synchronized (mSurfaceControlLock) { + if (frameNumber > 0 && viewRoot != null && !useBLAST) { + if (viewRoot.mSurface.isValid()) { + mRtTransaction.deferTransactionUntil(mSurfaceControl, + viewRoot.getRenderSurfaceControl(), frameNumber); + } } - } - t.hide(mSurfaceControl); + t.hide(mSurfaceControl); - synchronized (mSurfaceControlLock) { if (mRtReleaseSurfaces) { mRtReleaseSurfaces = false; mRtTransaction.remove(mSurfaceControl); |
