Summary: Android may spontaneously call these methods on the UI thread, so recognize the fact. Reviewed By: ezgicicek Differential Revision: D18530477 fbshipit-source-id: a8a798779master
							parent
							
								
									08df37ef76
								
							
						
					
					
						commit
						0c4d2d7a92
					
				| @ -0,0 +1,111 @@ | |||||||
|  | /* | ||||||
|  |  * Copyright (c) Facebook, Inc. and its affiliates. | ||||||
|  |  * | ||||||
|  |  * This source code is licensed under the MIT license found in the | ||||||
|  |  * LICENSE file in the root directory of this source tree. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | import android.app.Activity; | ||||||
|  | import android.os.Binder; | ||||||
|  | import android.os.Bundle; | ||||||
|  | import android.os.RemoteException; | ||||||
|  | import java.util.concurrent.Executor; | ||||||
|  | 
 | ||||||
|  | class MyActivity extends Activity { | ||||||
|  |   Binder b; | ||||||
|  | 
 | ||||||
|  |   private void bad() { | ||||||
|  |     try { | ||||||
|  |       b.transact(0, null, null, 0); | ||||||
|  |     } catch (RemoteException r) { | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   // overrides so no Bad suffixes
 | ||||||
|  | 
 | ||||||
|  |   @Override | ||||||
|  |   protected void onCreate(Bundle savedInstanceState) { | ||||||
|  |     bad(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   @Override | ||||||
|  |   public void onStart() { | ||||||
|  |     bad(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   @Override | ||||||
|  |   public void onRestart() { | ||||||
|  |     bad(); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   Object monitorA; | ||||||
|  |   @ForNonUiThread private final Executor mNonUiThreadExecutor = null; | ||||||
|  | 
 | ||||||
|  |   // method is a UI thread callback, and schedules a transaction in the background
 | ||||||
|  |   // but it synchronises on the lock protecting the transaction, thus stalling the main thread
 | ||||||
|  |   @Override | ||||||
|  |   public void onStop() { | ||||||
|  |     synchronized (monitorA) { | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     mNonUiThreadExecutor.execute( | ||||||
|  |         new Runnable() { | ||||||
|  |           @Override | ||||||
|  |           public void run() { | ||||||
|  |             synchronized (monitorA) { | ||||||
|  |               bad(); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   Object monitorB, monitorC; | ||||||
|  | 
 | ||||||
|  |   // method is a UI thread callback and deadlocks with work scheduled in
 | ||||||
|  |   // another callback (onPause) but which schedules work to a background thread
 | ||||||
|  |   @Override | ||||||
|  |   public void onDestroy() { | ||||||
|  |     synchronized (monitorC) { | ||||||
|  |       synchronized (monitorB) { | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   @Override | ||||||
|  |   public void onPause() { | ||||||
|  |     mNonUiThreadExecutor.execute( | ||||||
|  |         new Runnable() { | ||||||
|  |           @Override | ||||||
|  |           public void run() { | ||||||
|  |             synchronized (monitorB) { | ||||||
|  |               synchronized (monitorC) { | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         }); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   Object FP_monitorD, FP_monitorE; | ||||||
|  | 
 | ||||||
|  |   // False positive: by the time the work is scheduled, no lock is held, so no deadlock
 | ||||||
|  |   // Locks are named FP_* so that the report is clearly an FP (we can't change the name of the
 | ||||||
|  |   // override).
 | ||||||
|  |   @Override | ||||||
|  |   public void onResume() { | ||||||
|  |     synchronized (FP_monitorD) { | ||||||
|  |       synchronized (FP_monitorE) { | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     mNonUiThreadExecutor.execute( | ||||||
|  |         new Runnable() { | ||||||
|  |           @Override | ||||||
|  |           public void run() { | ||||||
|  |             synchronized (FP_monitorE) { | ||||||
|  |               synchronized (FP_monitorD) { | ||||||
|  |               } | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         }); | ||||||
|  |   } | ||||||
|  | } | ||||||
					Loading…
					
					
				
		Reference in new issue