/*
|
* Copyright (C) 2017 The Android Open Source Project
|
*
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
* you may not use this file except in compliance with the License.
|
* You may obtain a copy of the License at
|
*
|
* http://www.apache.org/licenses/LICENSE-2.0
|
*
|
* Unless required by applicable law or agreed to in writing, software
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
* See the License for the specific language governing permissions and
|
* limitations under the License.
|
*/
|
|
import art.Locals;
|
import art.StackTrace;
|
import art.Suspension;
|
import java.lang.reflect.Constructor;
|
import java.lang.reflect.Executable;
|
import java.lang.reflect.Method;
|
import java.nio.ByteBuffer;
|
import java.time.Instant;
|
import java.util.concurrent.Semaphore;
|
import java.util.Arrays;
|
import java.util.Collection;
|
import java.util.List;
|
import java.util.Set;
|
import java.util.function.Function;
|
import java.util.function.Predicate;
|
import java.util.function.Supplier;
|
import java.util.function.Consumer;
|
|
public class Main {
|
public static final int SET_VALUE = 1337;
|
public static final String TARGET_VAR = "TARGET";
|
|
public static void main(String[] args) throws Exception {
|
System.loadLibrary(args[0]);
|
Locals.EnableLocalVariableAccess();
|
runGet();
|
runSet();
|
}
|
|
public static void reportValue(Object val) {
|
System.out.println("\tValue is '" + val + "'");
|
}
|
|
public static class IntRunner implements Runnable {
|
private volatile boolean continueBusyLoop;
|
private volatile boolean inBusyLoop;
|
private final boolean expectOsr;
|
public IntRunner(boolean expectOsr) {
|
this.continueBusyLoop = true;
|
this.inBusyLoop = false;
|
this.expectOsr = expectOsr;
|
}
|
public void run() {
|
int TARGET = 42;
|
boolean normalJit = hasJit() && getJitThreshold() != 0; // Excluding JIT-at-first-use.
|
if (normalJit && expectOsr && !Main.isInterpreted()) {
|
System.out.println("Unexpectedly in jit code prior to restarting the JIT!");
|
}
|
startJit();
|
// We will suspend the thread during this loop.
|
while (continueBusyLoop) {
|
inBusyLoop = true;
|
}
|
// Wait up to 300 seconds for OSR to kick in if we expect it. If we don't give up after only
|
// 3 seconds.
|
Instant osrDeadline = Instant.now().plusSeconds(expectOsr ? 600 : 3);
|
do {
|
// Don't actually do anything here.
|
inBusyLoop = true;
|
} while (normalJit && !Main.isInOsrCode("run") && osrDeadline.compareTo(Instant.now()) > 0);
|
// We shouldn't be doing OSR since we are using JVMTI and the set prevents OSR.
|
// Set local will also push us to interpreter but the get local may remain in compiled code.
|
if (normalJit) {
|
boolean inOsr = Main.isInOsrCode("run");
|
if (expectOsr && !inOsr) {
|
throw new Error(
|
"Expected to be in OSR but was not. interpreter: " + Main.isInterpreted());
|
} else if (!expectOsr && inOsr) {
|
throw new Error(
|
"Expected not to be in OSR but was. interpreter: " + Main.isInterpreted());
|
}
|
}
|
reportValue(TARGET);
|
}
|
public void waitForBusyLoopStart() { while (!inBusyLoop) {} }
|
public void finish() {
|
continueBusyLoop = false;
|
}
|
}
|
|
public static void runGet() throws Exception {
|
Method target = IntRunner.class.getDeclaredMethod("run");
|
// Stop jit temporarily. It will be restarted by the test itself.
|
stopJit();
|
// Get Int.
|
IntRunner int_runner = new IntRunner(true);
|
Thread target_get = new Thread(int_runner, "GetLocalInt - Target");
|
target_get.start();
|
int_runner.waitForBusyLoopStart();
|
try {
|
Suspension.suspend(target_get);
|
} catch (Exception e) {
|
System.out.println("FAIL: got " + e);
|
e.printStackTrace();
|
int_runner.finish();
|
target_get.join();
|
return;
|
}
|
try {
|
StackTrace.StackFrameData frame = FindStackFrame(target_get, target);
|
int depth = frame.depth;
|
if (depth != 0) { throw new Error("Expected depth 0 but got " + depth); }
|
int slot = FindSlot(frame);
|
int value = Locals.GetLocalVariableInt(target_get, depth, slot);
|
System.out.println("From GetLocalInt(), value is " + value);
|
} finally {
|
Suspension.resume(target_get);
|
int_runner.finish();
|
target_get.join();
|
}
|
}
|
|
public static void runSet() throws Exception {
|
Method target = IntRunner.class.getDeclaredMethod("run");
|
// Stop jit temporarily. It will be restarted by the test itself.
|
stopJit();
|
// Set Int. Even if we start out in JIT code somehow we should be pushed out of it.
|
IntRunner int_runner = new IntRunner(false);
|
Thread target_set = new Thread(int_runner, "SetLocalInt - Target");
|
target_set.start();
|
int_runner.waitForBusyLoopStart();
|
try {
|
Suspension.suspend(target_set);
|
} catch (Exception e) {
|
System.out.println("FAIL: got " + e);
|
e.printStackTrace();
|
int_runner.finish();
|
target_set.join();
|
return;
|
}
|
try {
|
StackTrace.StackFrameData frame = FindStackFrame(target_set, target);
|
int depth = frame.depth;
|
if (depth != 0) { throw new Error("Expected depth 0 but got " + depth); }
|
int slot = FindSlot(frame);
|
System.out.println("Setting TARGET to " + SET_VALUE);
|
Locals.SetLocalVariableInt(target_set, depth, slot, SET_VALUE);
|
} finally {
|
Suspension.resume(target_set);
|
int_runner.finish();
|
target_set.join();
|
}
|
}
|
|
public static int FindSlot(StackTrace.StackFrameData frame) throws Exception {
|
long loc = frame.current_location;
|
for (Locals.VariableDescription var : Locals.GetLocalVariableTable(frame.method)) {
|
if (var.start_location <= loc &&
|
var.length + var.start_location > loc &&
|
var.name.equals(TARGET_VAR)) {
|
return var.slot;
|
}
|
}
|
throw new Error(
|
"Unable to find variable " + TARGET_VAR + " in " + frame.method + " at loc " + loc);
|
}
|
|
private static StackTrace.StackFrameData FindStackFrame(Thread thr, Method target) {
|
for (StackTrace.StackFrameData frame : StackTrace.GetStackTrace(thr)) {
|
if (frame.method.equals(target)) {
|
return frame;
|
}
|
}
|
throw new Error("Unable to find stack frame in method " + target + " on thread " + thr);
|
}
|
|
public static native boolean isInterpreted();
|
public static native boolean isInOsrCode(String methodName);
|
public static native boolean stopJit();
|
public static native boolean startJit();
|
public static native boolean hasJit();
|
public static native int getJitThreshold();
|
}
|