| // Copyright (c) 2017, the R8 project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| package com.android.tools.r8.debug; |
| |
| import com.android.tools.r8.CompilationMode; |
| import com.android.tools.r8.D8; |
| import com.android.tools.r8.D8Command; |
| import com.android.tools.r8.ToolHelper; |
| import com.android.tools.r8.ToolHelper.ArtCommandBuilder; |
| import com.android.tools.r8.ToolHelper.DexVm; |
| import com.google.common.collect.ImmutableList; |
| import java.io.File; |
| import java.nio.file.Path; |
| import java.nio.file.Paths; |
| import java.util.ArrayDeque; |
| import java.util.Arrays; |
| import java.util.Collections; |
| import java.util.List; |
| import java.util.Map; |
| import java.util.Queue; |
| import java.util.TreeMap; |
| import java.util.function.Consumer; |
| import java.util.stream.Collectors; |
| import org.apache.harmony.jpda.tests.framework.jdwp.CommandPacket; |
| import org.apache.harmony.jpda.tests.framework.jdwp.Event; |
| import org.apache.harmony.jpda.tests.framework.jdwp.EventBuilder; |
| import org.apache.harmony.jpda.tests.framework.jdwp.EventPacket; |
| import org.apache.harmony.jpda.tests.framework.jdwp.Frame.Variable; |
| import org.apache.harmony.jpda.tests.framework.jdwp.JDWPCommands; |
| import org.apache.harmony.jpda.tests.framework.jdwp.JDWPCommands.StackFrameCommandSet; |
| import org.apache.harmony.jpda.tests.framework.jdwp.JDWPConstants; |
| import org.apache.harmony.jpda.tests.framework.jdwp.JDWPConstants.EventKind; |
| import org.apache.harmony.jpda.tests.framework.jdwp.JDWPConstants.StepDepth; |
| import org.apache.harmony.jpda.tests.framework.jdwp.JDWPConstants.StepSize; |
| import org.apache.harmony.jpda.tests.framework.jdwp.JDWPConstants.SuspendPolicy; |
| import org.apache.harmony.jpda.tests.framework.jdwp.Location; |
| import org.apache.harmony.jpda.tests.framework.jdwp.ParsedEvent; |
| import org.apache.harmony.jpda.tests.framework.jdwp.ParsedEvent.EventThread; |
| import org.apache.harmony.jpda.tests.framework.jdwp.ReplyPacket; |
| import org.apache.harmony.jpda.tests.framework.jdwp.Value; |
| import org.apache.harmony.jpda.tests.framework.jdwp.VmMirror; |
| import org.apache.harmony.jpda.tests.jdwp.share.JDWPTestCase; |
| import org.apache.harmony.jpda.tests.share.JPDATestOptions; |
| import org.junit.Assert; |
| import org.junit.Assume; |
| import org.junit.BeforeClass; |
| import org.junit.ClassRule; |
| import org.junit.Ignore; |
| import org.junit.Rule; |
| import org.junit.rules.TemporaryFolder; |
| import org.junit.rules.TestName; |
| |
| /** |
| * |
| * Base class for debugging tests |
| */ |
| public abstract class DebugTestBase { |
| |
| public static final StepFilter NO_FILTER = new StepFilter.NoStepFilter(); |
| public static final StepFilter INTELLIJ_FILTER = new StepFilter.IntelliJStepFilter(); |
| private static final StepFilter DEFAULT_FILTER = NO_FILTER; |
| |
| // Set to true to run tests with java |
| private static final boolean RUN_DEBUGGEE_WITH_JAVA = false; |
| |
| // Set to true to enable verbose logs |
| private static final boolean DEBUG_TESTS = false; |
| |
| private static final List<DexVm> UNSUPPORTED_ART_VERSIONS = ImmutableList.of( |
| // Dalvik does not support command ReferenceType.Methods which is used to set breakpoint. |
| // TODO(shertz) use command ReferenceType.MethodsWithGeneric instead |
| DexVm.ART_4_4_4, |
| // Older runtimes fail on buildbot |
| // TODO(shertz) re-enable once issue is solved |
| DexVm.ART_5_1_1, |
| DexVm.ART_6_0_1); |
| |
| private static final Path DEBUGGEE_JAR = Paths |
| .get(ToolHelper.BUILD_DIR, "test", "debug_test_resources.jar"); |
| |
| @ClassRule |
| public static TemporaryFolder temp = new TemporaryFolder(); |
| private static Path jdwpDexD8 = null; |
| private static Path debuggeeDexD8 = null; |
| |
| @Rule |
| public TestName testName = new TestName(); |
| |
| @BeforeClass |
| public static void setUp() throws Exception { |
| // Convert jar to dex with d8 with debug info |
| int minSdk = ToolHelper.getMinApiLevelForDexVm(ToolHelper.getDexVm()); |
| { |
| Path jdwpJar = ToolHelper.getJdwpTestsJarPath(minSdk); |
| Path dexOutputDir = temp.newFolder("d8-jdwp-jar").toPath(); |
| jdwpDexD8 = dexOutputDir.resolve("classes.dex"); |
| D8.run( |
| D8Command.builder() |
| .addProgramFiles(jdwpJar) |
| .setOutputPath(dexOutputDir) |
| .setMinApiLevel(minSdk) |
| .setMode(CompilationMode.DEBUG) |
| .build()); |
| } |
| { |
| Path dexOutputDir = temp.newFolder("d8-debuggee-jar").toPath(); |
| debuggeeDexD8 = dexOutputDir.resolve("classes.dex"); |
| D8.run( |
| D8Command.builder() |
| .addProgramFiles(DEBUGGEE_JAR) |
| .setOutputPath(dexOutputDir) |
| .setMinApiLevel(minSdk) |
| .setMode(CompilationMode.DEBUG) |
| .build()); |
| } |
| } |
| |
| protected final void runDebugTest(String debuggeeClass, JUnit3Wrapper.Command... commands) |
| throws Throwable { |
| runDebugTest(debuggeeClass, Arrays.asList(commands)); |
| } |
| |
| protected final void runDebugTest(String debuggeeClass, List<JUnit3Wrapper.Command> commands) |
| throws Throwable { |
| // Skip test due to unsupported runtime. |
| Assume.assumeTrue("Skipping test " + testName.getMethodName() + " because ART is not supported", |
| ToolHelper.artSupported()); |
| Assume.assumeFalse( |
| "Skipping failing test " + testName.getMethodName() + " for runtime " + ToolHelper |
| .getDexVm(), UNSUPPORTED_ART_VERSIONS.contains(ToolHelper.getDexVm())); |
| |
| // Run with ART. |
| String[] paths = new String[]{jdwpDexD8.toString(), debuggeeDexD8.toString()}; |
| new JUnit3Wrapper(debuggeeClass, paths, commands).runBare(); |
| } |
| |
| protected final JUnit3Wrapper.Command run() { |
| return new JUnit3Wrapper.Command.RunCommand(); |
| } |
| |
| protected final JUnit3Wrapper.Command breakpoint(String className, String methodName) { |
| return new JUnit3Wrapper.Command.BreakpointCommand(className, methodName); |
| } |
| |
| protected final JUnit3Wrapper.Command stepOver() { |
| return stepOver(DEFAULT_FILTER); |
| } |
| |
| protected final JUnit3Wrapper.Command stepOver(StepFilter stepFilter) { |
| return step(StepDepth.OVER, stepFilter); |
| } |
| |
| protected final JUnit3Wrapper.Command stepOut() { |
| return stepOut(DEFAULT_FILTER); |
| } |
| |
| protected final JUnit3Wrapper.Command stepOut(StepFilter stepFilter) { |
| return step(StepDepth.OUT, stepFilter); |
| } |
| |
| protected final JUnit3Wrapper.Command stepInto() { |
| return stepInto(DEFAULT_FILTER); |
| } |
| |
| protected final JUnit3Wrapper.Command stepInto(StepFilter stepFilter) { |
| return step(StepDepth.INTO, stepFilter); |
| } |
| |
| private JUnit3Wrapper.Command step(byte stepDepth, |
| StepFilter stepFilter) { |
| return new JUnit3Wrapper.Command.StepCommand(stepDepth, stepFilter); |
| } |
| |
| protected final JUnit3Wrapper.Command checkLocal(String localName, Value expectedValue) { |
| return inspect(t -> t.checkLocal(localName, expectedValue)); |
| } |
| |
| protected final JUnit3Wrapper.Command checkNoLocal() { |
| return inspect(t -> Assert.assertTrue(t.getLocalNames().isEmpty())); |
| } |
| |
| protected final JUnit3Wrapper.Command checkLine(int line) { |
| return inspect(t -> t.checkLine(line)); |
| } |
| |
| protected final JUnit3Wrapper.Command checkMethod(String className, String methodName) { |
| return inspect(t -> t.checkMethod(className, methodName)); |
| } |
| |
| protected final JUnit3Wrapper.Command inspect(Consumer<JUnit3Wrapper.DebuggeeState> inspector) { |
| return t -> inspector.accept(t.debuggeeState); |
| } |
| |
| protected final JUnit3Wrapper.Command setLocal(String localName, Value newValue) { |
| return new JUnit3Wrapper.Command.SetLocalCommand(localName, newValue); |
| } |
| |
| @Ignore("Prevents Gradle from running the wrapper as a test.") |
| static class JUnit3Wrapper extends JDWPTestCase { |
| |
| private final String debuggeeClassName; |
| |
| private final String[] debuggeePath; |
| |
| // Initially, the runtime is suspended so we're ready to process commands. |
| private State state = State.ProcessCommand; |
| |
| /** |
| * Represents the context of the debuggee suspension. This is {@code null} when the debuggee is |
| * not suspended. |
| */ |
| private DebuggeeState debuggeeState = null; |
| |
| private final Queue<Command> commandsQueue; |
| |
| // Active event requests. |
| private final Map<Integer, EventHandler> events = new TreeMap<>(); |
| |
| JUnit3Wrapper(String debuggeeClassName, String[] debuggeePath, List<Command> commands) { |
| this.debuggeeClassName = debuggeeClassName; |
| this.debuggeePath = debuggeePath; |
| this.commandsQueue = new ArrayDeque<>(commands); |
| } |
| |
| @Override |
| protected void runTest() throws Throwable { |
| if (DEBUG_TESTS) { |
| logWriter.println("Starts loop with " + commandsQueue.size() + " command(s) to process"); |
| } |
| |
| boolean exited = false; |
| while (!exited) { |
| if (DEBUG_TESTS) { |
| logWriter.println("Loop on state " + state.name()); |
| } |
| switch (state) { |
| case ProcessCommand: { |
| Command command = commandsQueue.poll(); |
| assert command != null; |
| if (DEBUG_TESTS) { |
| logWriter.println("Process command " + command.toString()); |
| } |
| command.perform(this); |
| break; |
| } |
| case WaitForEvent: |
| processEvents(); |
| break; |
| case Exit: |
| exited = true; |
| break; |
| default: |
| throw new AssertionError(); |
| } |
| } |
| |
| assertTrue("All commands have NOT been processed", commandsQueue.isEmpty()); |
| |
| logWriter.println("Finish loop"); |
| } |
| |
| @Override |
| protected String getDebuggeeClassName() { |
| return debuggeeClassName; |
| } |
| |
| private enum State { |
| /** |
| * Process next command |
| */ |
| ProcessCommand, |
| /** |
| * Wait for the next event |
| */ |
| WaitForEvent, |
| /** |
| * The debuggee has exited |
| */ |
| Exit |
| } |
| |
| private void processEvents() { |
| EventPacket eventPacket = getMirror().receiveEvent(); |
| ParsedEvent[] parsedEvents = ParsedEvent.parseEventPacket(eventPacket); |
| if (DEBUG_TESTS) { |
| logWriter.println("Received " + parsedEvents.length + " event(s)"); |
| for (int i = 0; i < parsedEvents.length; ++i) { |
| String msg = String.format("#%d: %s (id=%d)", Integer.valueOf(i), |
| JDWPConstants.EventKind.getName(parsedEvents[i].getEventKind()), |
| Integer.valueOf(parsedEvents[i].getRequestID())); |
| logWriter.println(msg); |
| } |
| } |
| // We only expect one event at a time. |
| assertEquals(1, parsedEvents.length); |
| ParsedEvent parsedEvent = parsedEvents[0]; |
| byte eventKind = parsedEvent.getEventKind(); |
| int requestID = parsedEvent.getRequestID(); |
| |
| if (eventKind == JDWPConstants.EventKind.VM_DEATH) { |
| // Special event when debuggee is about to terminate. |
| assertEquals(0, requestID); |
| setState(State.Exit); |
| } else { |
| assert parsedEvent.getSuspendPolicy() == SuspendPolicy.ALL; |
| |
| // Capture the context of the event suspension. |
| updateEventContext((EventThread) parsedEvent); |
| |
| // Handle event. |
| EventHandler eh = events.get(requestID); |
| assert eh != null; |
| eh.handle(this); |
| } |
| } |
| |
| // |
| // Inspection |
| // |
| |
| /** |
| * Allows to inspect the state of a debuggee when it is suspended. |
| */ |
| public class DebuggeeState { |
| |
| private final long threadId; |
| private final long frameId; |
| private final Location location; |
| |
| public DebuggeeState(long threadId, long frameId, Location location) { |
| this.threadId = threadId; |
| this.frameId = frameId; |
| this.location = location; |
| } |
| |
| public long getThreadId() { |
| return threadId; |
| } |
| |
| public long getFrameId() { |
| return frameId; |
| } |
| |
| public Location getLocation() { |
| return this.location; |
| } |
| |
| public void checkLocal(String localName, Value expectedValue) { |
| Variable localVar = getVariableAt(getLocation(), localName); |
| |
| // Get value |
| CommandPacket commandPacket = new CommandPacket( |
| JDWPCommands.StackFrameCommandSet.CommandSetID, |
| JDWPCommands.StackFrameCommandSet.GetValuesCommand); |
| commandPacket.setNextValueAsThreadID(getThreadId()); |
| commandPacket.setNextValueAsFrameID(getFrameId()); |
| commandPacket.setNextValueAsInt(1); |
| commandPacket.setNextValueAsInt(localVar.getSlot()); |
| commandPacket.setNextValueAsByte(localVar.getTag()); |
| ReplyPacket replyPacket = getMirror().performCommand(commandPacket); |
| checkReplyPacket(replyPacket, "StackFrame.GetValues command"); |
| int valuesCount = replyPacket.getNextValueAsInt(); |
| assert valuesCount == 1; |
| Value localValue = replyPacket.getNextValueAsValue(); |
| assertAllDataRead(replyPacket); |
| |
| Assert.assertEquals(expectedValue, localValue); |
| } |
| |
| public void checkLine(int line) { |
| Location location = getLocation(); |
| int currentLine = getMirror() |
| .getLineNumber(location.classID, location.methodID, location.index); |
| Assert.assertEquals(line, currentLine); |
| } |
| |
| public List<String> getLocalNames() { |
| return getVariablesAt(location).stream().map(v -> v.getName()).collect(Collectors.toList()); |
| } |
| |
| public void checkMethod(String className, String methodName) { |
| String currentClassSig = getMirror().getClassSignature(location.classID); |
| assert currentClassSig.charAt(0) == 'L'; |
| String currentClassName = currentClassSig.substring(1, currentClassSig.length() - 1) |
| .replace('/', '.'); |
| Assert.assertEquals("Incorrect class name", className, currentClassName); |
| |
| String currentMethodName = getMirror().getMethodName(location.classID, location.methodID); |
| Assert.assertEquals("Incorrect method name", methodName, currentMethodName); |
| } |
| } |
| |
| private static boolean inScope(long index, Variable var) { |
| long varStart = var.getCodeIndex(); |
| long varEnd = varStart + var.getLength(); |
| return index >= varStart && index < varEnd; |
| } |
| |
| private Variable getVariableAt(Location location, String localName) { |
| return getVariablesAt(location).stream() |
| .filter(v -> localName.equals(v.getName())) |
| .findFirst() |
| .get(); |
| } |
| |
| private List<Variable> getVariablesAt(Location location) { |
| // Get variable table and keep only variables visible at this location. |
| return getVariables(location.classID, location.methodID).stream() |
| .filter(v -> inScope(location.index, v)) |
| .collect(Collectors.toList()); |
| } |
| |
| private List<Variable> getVariables(long classID, long methodID) { |
| List<Variable> list = getMirror().getVariableTable(classID, methodID); |
| return list != null ? list : Collections.emptyList(); |
| } |
| |
| private void setState(State state) { |
| this.state = state; |
| } |
| |
| public DebuggeeState getDebuggeeState() { |
| return debuggeeState; |
| } |
| |
| private void updateEventContext(EventThread event) { |
| long threadId = event.getThreadID(); |
| long frameId = -1; |
| Location location = null; |
| // ART returns an error if we ask for frames when there is none. Workaround by asking the frame |
| // count first. |
| int frameCount = getMirror().getFrameCount(threadId); |
| if (frameCount > 0) { |
| ReplyPacket replyPacket = getMirror().getThreadFrames(threadId, 0, 1); |
| { |
| int number = replyPacket.getNextValueAsInt(); |
| assertEquals(1, number); |
| } |
| frameId = replyPacket.getNextValueAsFrameID(); |
| location = replyPacket.getNextValueAsLocation(); |
| assertAllDataRead(replyPacket); |
| } |
| debuggeeState = new DebuggeeState(threadId, frameId, location); |
| } |
| |
| private VmMirror getMirror() { |
| return debuggeeWrapper.vmMirror; |
| } |
| |
| private void resume() { |
| debuggeeState = null; |
| getMirror().resume(); |
| setState(State.WaitForEvent); |
| } |
| |
| private boolean installBreakpoint(BreakpointInfo breakpointInfo) { |
| final long classId = getMirror().getClassID(getClassSignature(breakpointInfo.className)); |
| if (classId == -1) { |
| // The class is not ready yet. Request a CLASS_PREPARE to delay the installation of the |
| // breakpoint. |
| ReplyPacket replyPacket = getMirror().setClassPrepared(breakpointInfo.className); |
| int classPrepareRequestId = replyPacket.getNextValueAsInt(); |
| assertAllDataRead(replyPacket); |
| events.put(Integer.valueOf(classPrepareRequestId), |
| new ClassPrepareHandler(breakpointInfo, classPrepareRequestId)); |
| return false; |
| } else { |
| int breakpointId = getMirror() |
| .setBreakpointAtMethodBegin(classId, breakpointInfo.methodName); |
| // Nothing to do on breakpoint |
| events.put(Integer.valueOf(breakpointId), new DefaultEventHandler()); |
| return true; |
| } |
| } |
| |
| // |
| // Command processing |
| // |
| public interface Command { |
| |
| void perform(JUnit3Wrapper testBase); |
| |
| class RunCommand implements Command { |
| |
| @Override |
| public void perform(JUnit3Wrapper testBase) { |
| testBase.resume(); |
| } |
| |
| @Override |
| public String toString() { |
| return "run"; |
| } |
| } |
| |
| // TODO(shertz) add method signature support (when multiple methods have the same name) |
| class BreakpointCommand implements Command { |
| |
| private final String className; |
| private final String methodName; |
| |
| public BreakpointCommand(String className, String methodName) { |
| assert className != null; |
| assert methodName != null; |
| this.className = className; |
| this.methodName = methodName; |
| } |
| |
| @Override |
| public void perform(JUnit3Wrapper testBase) { |
| testBase.installBreakpoint(new BreakpointInfo(className, methodName)); |
| } |
| |
| @Override |
| public String toString() { |
| StringBuilder sb = new StringBuilder(); |
| sb.append("breakpoint"); |
| sb.append(" class="); |
| sb.append(className); |
| sb.append(" method="); |
| sb.append(methodName); |
| return sb.toString(); |
| } |
| } |
| |
| class StepCommand implements Command { |
| |
| private final byte stepDepth; |
| private final StepFilter stepFilter; |
| |
| public StepCommand(byte stepDepth, |
| StepFilter stepFilter) { |
| this.stepDepth = stepDepth; |
| this.stepFilter = stepFilter; |
| } |
| |
| @Override |
| public void perform(JUnit3Wrapper testBase) { |
| long threadId = testBase.getDebuggeeState().getThreadId(); |
| int stepRequestID; |
| { |
| EventBuilder eventBuilder = Event.builder(EventKind.SINGLE_STEP, SuspendPolicy.ALL); |
| eventBuilder.setStep(threadId, StepSize.LINE, stepDepth); |
| stepFilter.getExcludedClasses().stream().forEach(s -> eventBuilder.setClassExclude(s)); |
| ReplyPacket replyPacket = testBase.getMirror().setEvent(eventBuilder.build()); |
| stepRequestID = replyPacket.getNextValueAsInt(); |
| testBase.assertAllDataRead(replyPacket); |
| } |
| testBase.events.put(stepRequestID, new StepEventHandler(stepRequestID, stepFilter)); |
| |
| // Resume all threads. |
| testBase.resume(); |
| } |
| |
| @Override |
| public String toString() { |
| return "step " + JDWPConstants.StepDepth.getName(stepDepth); |
| } |
| } |
| |
| class SetLocalCommand implements Command { |
| |
| private final String localName; |
| private final Value newValue; |
| |
| public SetLocalCommand(String localName, Value newValue) { |
| this.localName = localName; |
| this.newValue = newValue; |
| } |
| |
| @Override |
| public void perform(JUnit3Wrapper testBase) { |
| Variable v = testBase.getVariableAt(testBase.debuggeeState.location, localName); |
| CommandPacket setValues = new CommandPacket(StackFrameCommandSet.CommandSetID, |
| StackFrameCommandSet.SetValuesCommand); |
| setValues.setNextValueAsThreadID(testBase.getDebuggeeState().getThreadId()); |
| setValues.setNextValueAsFrameID(testBase.getDebuggeeState().getFrameId()); |
| setValues.setNextValueAsInt(1); |
| setValues.setNextValueAsInt(v.getSlot()); |
| setValues.setNextValueAsValue(newValue); |
| ReplyPacket replyPacket = testBase.getMirror().performCommand(setValues); |
| testBase.checkReplyPacket(replyPacket, "StackFrame.SetValues"); |
| } |
| } |
| } |
| |
| // |
| // Event handling |
| // |
| private interface EventHandler { |
| |
| void handle(JUnit3Wrapper testBase); |
| } |
| |
| private static class DefaultEventHandler implements EventHandler { |
| |
| @Override |
| public void handle(JUnit3Wrapper testBase) { |
| testBase.setState(State.ProcessCommand); |
| } |
| } |
| |
| private static class StepEventHandler extends DefaultEventHandler { |
| |
| private final int stepRequestID; |
| private final StepFilter stepFilter; |
| |
| private StepEventHandler(int stepRequestID, |
| StepFilter stepFilter) { |
| this.stepRequestID = stepRequestID; |
| this.stepFilter = stepFilter; |
| } |
| |
| @Override |
| public void handle(JUnit3Wrapper testBase) { |
| if (stepFilter |
| .skipLocation(testBase.getMirror(), testBase.getDebuggeeState().getLocation())) { |
| // Keep the step active and resume so that we do another step. |
| testBase.resume(); |
| } else { |
| // When hit, the single step must be cleared. |
| testBase.getMirror().clearEvent(EventKind.SINGLE_STEP, stepRequestID); |
| testBase.events.remove(Integer.valueOf(stepRequestID)); |
| super.handle(testBase); |
| } |
| } |
| } |
| |
| private static class BreakpointInfo { |
| |
| private final String className; |
| private final String methodName; |
| |
| private BreakpointInfo(String className, String methodName) { |
| this.className = className; |
| this.methodName = methodName; |
| } |
| } |
| |
| /** |
| * CLASS_PREPARE signals us that we can install a breakpoint |
| */ |
| private static class ClassPrepareHandler implements EventHandler { |
| |
| private final BreakpointInfo breakpointInfo; |
| private final int classPrepareRequestId; |
| |
| private ClassPrepareHandler(BreakpointInfo breakpointInfo, int classPrepareRequestId) { |
| this.breakpointInfo = breakpointInfo; |
| this.classPrepareRequestId = classPrepareRequestId; |
| } |
| |
| @Override |
| public void handle(JUnit3Wrapper testBase) { |
| // Remove the CLASS_PREPARE |
| testBase.events.remove(Integer.valueOf(classPrepareRequestId)); |
| testBase.getMirror().clearEvent(JDWPConstants.EventKind.CLASS_PREPARE, |
| classPrepareRequestId); |
| |
| // Install breakpoint now. |
| boolean success = testBase.installBreakpoint(breakpointInfo); |
| Assert.assertTrue("Failed to insert breakpoint after class has been prepared", success); |
| |
| // Resume now |
| testBase.resume(); |
| } |
| } |
| |
| @Override |
| protected JPDATestOptions createTestOptions() { |
| if (RUN_DEBUGGEE_WITH_JAVA) { |
| return super.createTestOptions(); |
| } else { |
| // Override properties to run debuggee with ART/Dalvik. |
| class ArtTestOptions extends JPDATestOptions { |
| |
| ArtTestOptions(String[] debuggeePath) { |
| // Set debuggee command-line. |
| if (!RUN_DEBUGGEE_WITH_JAVA) { |
| ArtCommandBuilder artCommandBuilder = new ArtCommandBuilder(ToolHelper.getDexVm()); |
| if (ToolHelper.getDexVm().isNewerThan(DexVm.ART_5_1_1)) { |
| artCommandBuilder.appendArtOption("-Xcompiler-option"); |
| artCommandBuilder.appendArtOption("--debuggable"); |
| artCommandBuilder.appendArtOption("-Xcompiler-option"); |
| artCommandBuilder.appendArtOption("--compiler-filter=interpret-only"); |
| } |
| setProperty("jpda.settings.debuggeeJavaPath", artCommandBuilder.build()); |
| |
| // Set debuggee classpath |
| String debuggeeClassPath = String.join(File.pathSeparator, debuggeePath); |
| setProperty("jpda.settings.debuggeeClasspath", debuggeeClassPath); |
| } |
| |
| // Set verbosity |
| setProperty("jpda.settings.verbose", Boolean.toString(DEBUG_TESTS)); |
| } |
| } |
| return new ArtTestOptions(debuggeePath); |
| } |
| } |
| } |
| |
| // |
| // Step filtering |
| // |
| |
| interface StepFilter { |
| |
| /** |
| * Provides a list of class name to be skipped when single stepping. This can be a fully |
| * qualified name (like java.lang.String) or a subpackage (like java.util.*). |
| */ |
| List<String> getExcludedClasses(); |
| |
| /** |
| * Indicates whether the given location must be skipped. |
| */ |
| boolean skipLocation(VmMirror mirror, Location location); |
| |
| /** |
| * A {@link StepFilter} that does not filter anything. |
| */ |
| class NoStepFilter implements StepFilter { |
| |
| @Override |
| public List<String> getExcludedClasses() { |
| return Collections.emptyList(); |
| } |
| |
| @Override |
| public boolean skipLocation(VmMirror mirror, Location location) { |
| return false; |
| } |
| } |
| |
| /** |
| * A {@link StepFilter} that matches the default behavior of IntelliJ regarding single |
| * stepping. |
| */ |
| class IntelliJStepFilter implements StepFilter { |
| // This is the value specified by JDWP in documentation of ReferenceType.Methods command. |
| private static final int SYNTHETIC_FLAG = 0xF0000000; |
| |
| @Override |
| public List<String> getExcludedClasses() { |
| return Arrays.asList( |
| "com.sun.*", |
| "java.*", |
| "javax.*", |
| "org.omg.*", |
| "sun.*", |
| "jdk.internal.*", |
| "junit.*", |
| "com.intellij.rt.*", |
| "com.yourkit.runtime.*", |
| "com.springsource.loaded.*", |
| "org.springsource.loaded.*", |
| "javassist.*", |
| "org.apache.webbeans.*", |
| "com.ibm.ws.*", |
| "kotlin.*" |
| ); |
| } |
| |
| @Override |
| public boolean skipLocation(VmMirror mirror, Location location) { |
| // TODO(shertz) we also need to skip class loaders to act like IntelliJ. |
| // Skip synthetic methods. |
| return isSyntheticMethod(mirror, location); |
| } |
| |
| private static boolean isSyntheticMethod(VmMirror mirror, Location location) { |
| // We must gather the modifiers of the method. This is only possible using |
| // ReferenceType.Methods command which gather information about all methods in a class. |
| ReplyPacket reply = mirror.getMethods(location.classID); |
| int methodsCount = reply.getNextValueAsInt(); |
| for (int i = 0; i < methodsCount; ++i) { |
| long methodId = reply.getNextValueAsMethodID(); |
| reply.getNextValueAsString(); // skip method name |
| reply.getNextValueAsString(); // skip method signature |
| int modifiers = reply.getNextValueAsInt(); |
| if (methodId == location.methodID && |
| ((modifiers & SYNTHETIC_FLAG) != 0)) { |
| return true; |
| } |
| } |
| return false; |
| } |
| } |
| } |
| |
| } |