Summary: One standard way to schedule work is by starting a thread. We treat this by - Treating invocations of `start` on a receiver with the `Runnable _` attribute as scheduling that runnable for parallel execution in the background (as opposed to on the UI thread). - If `start` is used on an object of a subclass of `Thread` everything already works thanks to the `get_exp_attributes` function which will implicitly ascribe to an expression the attribute `Runnable _` if the expression points to an object with a known `run` method. This will even take care of some degree of dynamic dispatch, yay! - If `start` is used on a `Thread` object which has been created with a constructor call provided with a `Runnable` argument, we have to appropriately model that constructor call, which is what is done in `do_call`. Reviewed By: jvillard Differential Revision: D18726676 fbshipit-source-id: 0bd83c28emaster
parent
aef34d8384
commit
7a538c5004
@ -0,0 +1,98 @@
|
||||
/*
|
||||
* 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.os.Binder;
|
||||
import android.os.RemoteException;
|
||||
|
||||
class ThreadScheduling {
|
||||
static Binder binder;
|
||||
|
||||
private static void doTransact() {
|
||||
try {
|
||||
binder.transact(0, null, null, 0);
|
||||
} catch (RemoteException e) {
|
||||
}
|
||||
}
|
||||
|
||||
Object monitorA;
|
||||
|
||||
public void scheduleBlockingCallOnContendedLockBad() {
|
||||
Thread t =
|
||||
new Thread(
|
||||
new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
synchronized (monitorA) {
|
||||
doTransact();
|
||||
}
|
||||
}
|
||||
});
|
||||
t.start();
|
||||
|
||||
Executors.runOnUiThread(
|
||||
new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
synchronized (monitorA) {
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
Object monitorB, monitorC;
|
||||
|
||||
public void scheduleDeadlockBad() {
|
||||
Thread t =
|
||||
new Thread(
|
||||
new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
synchronized (monitorB) {
|
||||
synchronized (monitorC) {
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
t.start();
|
||||
|
||||
Executors.runOnUiThread(
|
||||
new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
synchronized (monitorC) {
|
||||
synchronized (monitorB) {
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
Object monitorD;
|
||||
|
||||
class BadThread extends Thread {
|
||||
@Override
|
||||
public void run() {
|
||||
synchronized (monitorD) {
|
||||
doTransact();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public void scheduleBlockingCallOnContendedLockViaInheritanceBad() {
|
||||
Thread t = new BadThread();
|
||||
t.start();
|
||||
|
||||
Executors.runOnUiThread(
|
||||
new Runnable() {
|
||||
@Override
|
||||
public void run() {
|
||||
synchronized (monitorD) {
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
Loading…
Reference in new issue