Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.oracle.truffle.api.debug.BreakpointFactory Maven / Gradle / Ivy
Go to download
Truffle is a multi-language framework for executing dynamic languages
that achieves high performance when combined with Graal.
/*
* Copyright (c) 2016, 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package com.oracle.truffle.api.debug;
import static com.oracle.truffle.api.debug.Breakpoint.State.DISABLED;
import static com.oracle.truffle.api.debug.Breakpoint.State.DISABLED_UNRESOLVED;
import static com.oracle.truffle.api.debug.Breakpoint.State.DISPOSED;
import static com.oracle.truffle.api.debug.Breakpoint.State.ENABLED;
import static com.oracle.truffle.api.debug.Breakpoint.State.ENABLED_UNRESOLVED;
import java.io.IOException;
import java.io.PrintStream;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import com.oracle.truffle.api.Assumption;
import com.oracle.truffle.api.CallTarget;
import com.oracle.truffle.api.CompilerDirectives.CompilationFinal;
import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary;
import com.oracle.truffle.api.Truffle;
import com.oracle.truffle.api.TruffleLanguage;
import com.oracle.truffle.api.debug.Breakpoint.State;
import com.oracle.truffle.api.debug.Debugger.BreakpointCallback;
import com.oracle.truffle.api.debug.Debugger.WarningLog;
import com.oracle.truffle.api.frame.VirtualFrame;
import com.oracle.truffle.api.instrumentation.EventBinding;
import com.oracle.truffle.api.instrumentation.EventContext;
import com.oracle.truffle.api.instrumentation.ExecutionEventListener;
import com.oracle.truffle.api.instrumentation.ExecutionEventNode;
import com.oracle.truffle.api.instrumentation.ExecutionEventNodeFactory;
import com.oracle.truffle.api.instrumentation.Instrumenter;
import com.oracle.truffle.api.instrumentation.SourceSectionFilter;
import com.oracle.truffle.api.instrumentation.StandardTags;
import com.oracle.truffle.api.instrumentation.SourceSectionFilter.IndexRange;
import com.oracle.truffle.api.nodes.DirectCallNode;
import com.oracle.truffle.api.nodes.InvalidAssumptionException;
import com.oracle.truffle.api.nodes.Node;
import com.oracle.truffle.api.source.LineLocation;
import com.oracle.truffle.api.source.Source;
import com.oracle.truffle.api.utilities.CyclicAssumption;
import java.lang.ref.Reference;
import java.lang.ref.WeakReference;
import java.net.URI;
import java.util.HashSet;
import java.util.Objects;
import java.util.Set;
/**
* Creator and manager of program breakpoints.
*/
final class BreakpointFactory {
private static final boolean TRACE = Boolean.getBoolean("truffle.debug.trace");
private static final PrintStream OUT = System.out;
private static final String TRACE_PREFIX = "Brkpt";
@TruffleBoundary
private static void trace(String format, Object... args) {
if (TRACE) {
OUT.println(String.format("%s: %s", TRACE_PREFIX, String.format(format, args)));
}
}
/**
* Map: Location key ==> attached breakpoints, where key is either a {@link LineLocation} or
* {@linkplain String tag}; there may be no more than one breakpoint per key.
*/
private final Map breakpoints = new HashMap<>();
/**
* Breakpoints that are internal to the debugger infrastructure. These are not exposed to
* clients.
*/
private final Map breakpointsInternal = new HashMap<>();
private final Map> uriLocations = new HashMap<>();
private final Map> sources = new HashMap<>();
private static final Comparator> BREAKPOINT_COMPARATOR = new Comparator>() {
@Override
public int compare(Entry entry1, Entry entry2) {
final Object key1 = entry1.getKey();
final Object key2 = entry2.getKey();
if (key1 instanceof LineLocation && key2 instanceof LineLocation) {
final LineLocation line1 = (LineLocation) key1;
final LineLocation line2 = (LineLocation) key2;
final int nameOrder = line1.getSource().getName().compareTo(line2.getSource().getName());
if (nameOrder != 0) {
return nameOrder;
}
return Integer.compare(line1.getLineNumber(), line2.getLineNumber());
}
return key1.toString().compareTo(key2.toString());
}
};
/**
* Globally suspends all line breakpoint activity when {@code false}, ignoring whether
* individual breakpoints are enabled.
*/
@CompilationFinal boolean breakpointsActive = true;
private final CyclicAssumption breakpointsActiveUnchanged = new CyclicAssumption("All breakpoints globally active");
private final Instrumenter instrumenter;
private final BreakpointCallback breakpointCallback;
private final WarningLog warningLog;
BreakpointFactory(Instrumenter instrumenter, BreakpointCallback breakpointCallback, final WarningLog warningLog) {
this.instrumenter = instrumenter;
this.breakpointCallback = breakpointCallback;
this.warningLog = warningLog;
createDefaultBreakpoints();
}
private void createDefaultBreakpoints() {
Class> tag = DebuggerTags.AlwaysHalt.class;
SourceSectionFilter query = SourceSectionFilter.newBuilder().tagIs(tag).build();
BreakpointImpl breakpoint = createBreakpoint(tag, query, 0, false);
breakpointsInternal.put(tag, breakpoint);
}
/**
* Globally enables breakpoint activity; all breakpoints are ignored when set to {@code false}.
* When set to {@code true}, the enabled/disabled status of each breakpoint determines whether
* it will trigger when flow of execution reaches it.
*
* @param breakpointsActive
*/
void setActive(boolean breakpointsActive) {
if (this.breakpointsActive != breakpointsActive) {
breakpointsActiveUnchanged.invalidate();
this.breakpointsActive = breakpointsActive;
}
}
/**
* Creates a new line breakpoint if one doesn't already exist. If one does exist, then resets
* the ignore count .
*
* If a breakpoint condition is applied to the breakpoint, then the condition will be
* assumed to be in the same language as the code location where attached.
*
* @param lineLocation where to set the breakpoint
* @param ignoreCount number of initial hits before the breakpoint starts causing breaks.
* @param oneShot whether the breakpoint should dispose itself after one hit
* @return a possibly new breakpoint
* @throws IOException if a breakpoint already exists at the location and the ignore count is
* the same
*/
Breakpoint create(int ignoreCount, LineLocation lineLocation, boolean oneShot) throws IOException {
BreakpointImpl breakpoint = breakpoints.get(lineLocation);
if (breakpoint == null) {
final SourceSectionFilter query = SourceSectionFilter.newBuilder().sourceIs(lineLocation.getSource()).lineStartsIn(IndexRange.byLength(lineLocation.getLineNumber(), 1)).tagIs(
StandardTags.StatementTag.class).build();
breakpoint = createBreakpoint(lineLocation, query, ignoreCount, oneShot);
if (TRACE) {
trace("NEW " + breakpoint.getShortDescription());
}
breakpoints.put(lineLocation, breakpoint);
} else {
if (ignoreCount == breakpoint.getIgnoreCount()) {
throw new IOException("Breakpoint already set at location " + lineLocation);
}
breakpoint.setIgnoreCount(ignoreCount);
if (TRACE) {
trace("CHANGED ignoreCount %s", breakpoint.getShortDescription());
}
}
return breakpoint;
}
Breakpoint create(int ignoreCount, URI sourceUri, int line, int column, boolean oneShot) throws IOException {
URILocation uriLocation = new URILocation(sourceUri, line, column);
BreakpointImpl breakpoint = breakpoints.get(uriLocation);
if (breakpoint == null) {
Set locations = uriLocations.get(sourceUri);
if (locations == null) {
locations = new HashSet<>();
uriLocations.put(sourceUri, locations);
}
locations.add(uriLocation);
breakpoint = createBreakpoint(uriLocation, null, ignoreCount, oneShot);
Reference sourceRef = sources.get(sourceUri);
if (sourceRef != null) {
Source source = sourceRef.get();
if (source != null) {
breakpoint.resolve(source);
} else {
sources.remove(sourceUri);
}
}
if (TRACE) {
trace("NEW " + breakpoint.getShortDescription());
}
breakpoints.put(uriLocation, breakpoint);
} else {
if (ignoreCount == breakpoint.getIgnoreCount()) {
throw new IOException("Breakpoint already set at location " + uriLocation);
}
breakpoint.setIgnoreCount(ignoreCount);
if (TRACE) {
trace("CHANGED ignoreCount %s", breakpoint.getShortDescription());
}
}
return breakpoint;
}
/**
* Creates a new line breakpoint if one doesn't already exist. If one does exist, then resets
* the ignore count .
*
* If a breakpoint condition is applied to the breakpoint, then the condition will be
* assumed to be in the same language as the code location where attached.
*
* @param lineLocation where to set the breakpoint
* @param ignoreCount number of initial hits before the breakpoint starts causing breaks.
* @param oneShot whether the breakpoint should dispose itself after one hit
* @return a possibly new breakpoint
* @throws IOException if a breakpoint already exists at the location and the ignore count is
* the same
*/
Breakpoint create(int ignoreCount, Class> tag, boolean oneShot) throws IOException {
BreakpointImpl breakpoint = breakpoints.get(tag);
if (breakpoint == null) {
final SourceSectionFilter query = SourceSectionFilter.newBuilder().tagIs(tag).build();
breakpoint = createBreakpoint(tag, query, ignoreCount, oneShot);
if (TRACE) {
trace("NEW " + breakpoint.getShortDescription());
}
breakpoints.put(tag, breakpoint);
} else {
if (ignoreCount == breakpoint.getIgnoreCount()) {
throw new IOException("Breakpoint already set at location " + tag);
}
breakpoint.setIgnoreCount(ignoreCount);
if (TRACE) {
trace("CHANGED ignoreCount %s", breakpoint.getShortDescription());
}
}
return breakpoint;
}
/**
* Returns the {@link Breakpoint} for a given, if any, either a {@link LineLocation} or
* {@linkplain String tag}. There should only ever be one breakpoint per location.
*
* @param lineLocation The {@link LineLocation} to get the breakpoint for.
* @return The breakpoint for the given line.
*/
Breakpoint get(Object key) {
return breakpoints.get(key);
}
/**
* Gets all current line breakpoints,regardless of status; sorted and modification safe.
*/
List getAll() {
ArrayList> entries = new ArrayList<>(breakpoints.entrySet());
Collections.sort(entries, BREAKPOINT_COMPARATOR);
final ArrayList breakpointList = new ArrayList<>(entries.size());
for (Entry entry : entries) {
breakpointList.add(entry.getValue());
}
return breakpointList;
}
/**
* Removes the associated instrumentation for all one-shot breakpoints only.
*/
void disposeOneShots() {
final Collection oneShots = new ArrayList<>(breakpoints.values());
for (BreakpointImpl breakpoint : oneShots) {
if (breakpoint.isOneShot()) {
breakpoint.dispose();
}
}
}
/**
* Removes all knowledge of a breakpoint, presumed disposed.
*/
private void forget(BreakpointImpl breakpoint) {
assert breakpoint.getState() == State.DISPOSED;
Object key = breakpoint.getKey();
breakpoints.remove(key);
if (key instanceof URILocation) {
URILocation ul = (URILocation) key;
Set locations = uriLocations.get(ul.uri);
locations.remove(ul);
if (locations.isEmpty()) {
uriLocations.remove(ul.uri);
}
}
}
BreakpointImpl createBreakpoint(Object key, SourceSectionFilter query, int ignoreCount, boolean isOneShot) {
BreakpointImpl breakpoint = new BreakpointImpl(key, query, ignoreCount, isOneShot);
// Register listener after breakpoint has been constructed and JMM
// allows for safe publication. Otherwise, we can't be sure that the
// assumption fields are visible by other threads, which would lead to
// a race with object initialization.
if (query != null) {
breakpoint.binding = instrumenter.attachListener(query, new BreakpointListener(breakpoint));
}
return breakpoint;
}
void notifySourceLoaded(Source source) {
if (source == null) {
return;
}
URI uri = source.getURI();
assert uri != null;
Reference sourceRef = sources.get(uri);
if (sourceRef != null && source == sourceRef.get()) {
// We know about this source already
return;
}
Set locations = uriLocations.get(uri);
if (locations != null) {
for (URILocation l : locations) {
breakpoints.get(l).resolve(source);
}
}
sources.put(uri, new WeakReference<>(source));
}
private final class BreakpointImpl extends Breakpoint implements ExecutionEventNodeFactory {
private static final String SHOULD_NOT_HAPPEN = "BreakpointImpl: should not happen";
private final Object locationKey;
private SourceSectionFilter locationQuery;
private final boolean isOneShot;
private int ignoreCount;
private int hitCount = 0;
private State state = ENABLED_UNRESOLVED;
@SuppressWarnings("rawtypes") private EventBinding binding;
// Cached assumption that the global status of line breakpoint activity has not changed.
@CompilationFinal private Assumption breakpointsActiveAssumption;
// Whether this breakpoint is enable/disabled
@CompilationFinal private boolean isEnabled;
@CompilationFinal private Assumption enabledUnchangedAssumption;
private String conditionExpr;
private Source conditionSource;
@SuppressWarnings("rawtypes") private Class extends TruffleLanguage> condLangClass;
private BreakpointImpl(Object key, SourceSectionFilter query, int ignoreCount, boolean isOneShot) {
super();
this.ignoreCount = ignoreCount;
this.isOneShot = isOneShot;
this.locationKey = key;
this.locationQuery = query;
this.isEnabled = true;
this.breakpointsActiveAssumption = BreakpointFactory.this.breakpointsActiveUnchanged.getAssumption();
this.enabledUnchangedAssumption = Truffle.getRuntime().createAssumption("Breakpoint enabled state unchanged");
}
@Override
public String getLocationDescription() {
if (locationKey instanceof LineLocation) {
return ((LineLocation) locationKey).getShortDescription();
}
return "Tag: " + locationKey.toString();
}
@Override
public boolean isEnabled() {
return isEnabled;
}
@Override
public void setEnabled(boolean enabled) {
assert getState() != DISPOSED : "disposed breakpoints are unusable";
if (enabled != isEnabled) {
switch (getState()) {
case ENABLED:
assert !enabled : SHOULD_NOT_HAPPEN;
doSetEnabled(false);
changeState(DISABLED);
break;
case ENABLED_UNRESOLVED:
assert !enabled : SHOULD_NOT_HAPPEN;
doSetEnabled(false);
changeState(DISABLED_UNRESOLVED);
break;
case DISABLED:
assert enabled : SHOULD_NOT_HAPPEN;
doSetEnabled(true);
changeState(ENABLED);
break;
case DISABLED_UNRESOLVED:
assert enabled : SHOULD_NOT_HAPPEN;
doSetEnabled(true);
changeState(ENABLED_UNRESOLVED);
break;
case DISPOSED:
assert false : "breakpoint disposed";
break;
default:
assert false : SHOULD_NOT_HAPPEN;
break;
}
}
}
@Override
public Source getCondition() {
return conditionSource;
}
@Override
public void setCondition(String expr) throws IOException {
assert getState() != DISPOSED : "disposed breakpoints are unusable";
if (binding != null) {
binding.dispose();
if (expr == null) {
conditionSource = null;
binding = instrumenter.attachListener(locationQuery, new BreakpointListener(this));
} else {
conditionSource = Source.newBuilder(expr).name("breakpoint condition from text: " + expr).mimeType("text/plain").build();
binding = instrumenter.attachFactory(locationQuery, this);
}
}
conditionExpr = expr;
}
@Override
public boolean isOneShot() {
return isOneShot;
}
@Override
public int getIgnoreCount() {
return ignoreCount;
}
@Override
public void setIgnoreCount(int ignoreCount) {
assert getState() != DISPOSED : "disposed breakpoints are unusable";
this.ignoreCount = ignoreCount;
}
@Override
public int getHitCount() {
return hitCount;
}
@Override
public State getState() {
return state;
}
@TruffleBoundary
@Override
public void dispose() {
if (getState() != DISPOSED) {
binding.dispose();
changeState(DISPOSED);
isEnabled = false;
BreakpointFactory.this.forget(this);
}
}
/* EventNodeFactory for breakpoint condition */
@Override
public ExecutionEventNode create(EventContext context) {
assert conditionSource != null;
final Node instrumentedNode = context.getInstrumentedNode();
if (condLangClass == null) {
condLangClass = Debugger.AccessorDebug.nodesAccess().findLanguage(instrumentedNode.getRootNode());
if (condLangClass == null) {
warningLog.addWarning("Unable to find language for condition: \"" + conditionSource.getCode() + "\" at " + getLocationDescription());
return null;
}
}
try {
final CallTarget callTarget = Debugger.ACCESSOR.parse(condLangClass, conditionSource, instrumentedNode, new String[0]);
final DirectCallNode callNode = Truffle.getRuntime().createDirectCallNode(callTarget);
return new BreakpointConditionEventNode(context, callNode);
} catch (IOException e) {
warningLog.addWarning("Unable to parse breakpoint condition: \"" + conditionSource.getCode() + "\" at " + getLocationDescription());
return null;
}
}
private String getShortDescription() {
if (locationKey instanceof LineLocation) {
return "Breakpoint@" + ((LineLocation) locationKey).getShortDescription();
}
return "Breakpoint@" + locationKey.toString();
}
private void doSetEnabled(boolean enabled) {
if (this.isEnabled != enabled) {
enabledUnchangedAssumption.invalidate();
this.isEnabled = enabled;
}
}
private Object getKey() {
return locationKey;
}
private void changeState(State after) {
if (TRACE) {
trace("STATE %s-->%s %s", getState().getName(), after.getName(), getShortDescription());
}
this.state = after;
}
private void doBreak(EventContext context, VirtualFrame vFrame) {
if (++hitCount > ignoreCount) {
breakpointCallback.haltedAt(context, vFrame.materialize(), this);
}
}
/**
* Receives notification from an attached listener that execution is about to enter a node
* where the breakpoint is set. Designed so that when in the fast path, there is either an
* unconditional "halt" call to the debugger or nothing.
*/
private void nodeEnter(EventContext context, VirtualFrame vFrame) {
try {
// Deopt if the global active/inactive flag has changed
this.breakpointsActiveAssumption.check();
} catch (InvalidAssumptionException ex) {
this.breakpointsActiveAssumption = BreakpointFactory.this.breakpointsActiveUnchanged.getAssumption();
}
try {
// Deopt if the enabled/disabled state of this breakpoint has changed
this.enabledUnchangedAssumption.check();
} catch (InvalidAssumptionException ex) {
this.enabledUnchangedAssumption = Truffle.getRuntime().createAssumption("Breakpoint enabled state unchanged");
}
if (BreakpointFactory.this.breakpointsActive && this.isEnabled) {
if (isOneShot()) {
dispose();
}
BreakpointImpl.this.doBreak(context, vFrame);
}
}
private void conditionFailure(EventContext context, VirtualFrame vFrame, Exception ex) {
addExceptionWarning(ex);
if (TRACE) {
trace("breakpoint failure = %s %s", ex, getShortDescription());
}
// Take the breakpoint if evaluation fails.
nodeEnter(context, vFrame);
}
@TruffleBoundary
private void addExceptionWarning(Exception ex) {
warningLog.addWarning(String.format("Exception in %s: %s", getShortDescription(), ex.getMessage()));
}
private void resolve(Source source) {
int line = ((URILocation) locationKey).line;
LineLocation lineLocation = source.createLineLocation(line);
final SourceSectionFilter query = SourceSectionFilter.newBuilder().sourceIs(lineLocation.getSource()).lineStartsIn(IndexRange.byLength(lineLocation.getLineNumber(), 1)).tagIs(
StandardTags.StatementTag.class).build();
locationQuery = query;
if (conditionExpr != null) {
// @formatter:off
conditionSource = Source.newBuilder(conditionExpr).
name("breakpoint condition from text: " + conditionExpr).
mimeType(source.getMimeType()).
build();
// @formatter:on
binding = instrumenter.attachFactory(locationQuery, this);
} else {
binding = instrumenter.attachListener(query, new BreakpointListener(this));
}
}
/** Attached to implement a conditional breakpoint. */
private class BreakpointConditionEventNode extends ExecutionEventNode {
@Child DirectCallNode callNode;
final EventContext context;
BreakpointConditionEventNode(EventContext context, DirectCallNode callNode) {
this.context = context;
this.callNode = callNode;
}
@Override
protected void onEnter(VirtualFrame frame) {
try {
Object result = callNode.call(frame, new Object[0]);
if (result instanceof Boolean) {
if (TRACE) {
trace("breakpoint cond=%b %s %s", result, conditionSource.getCode(), getShortDescription());
}
if ((Boolean) result) {
nodeEnter(context, frame); // as if unconditional
}
} else {
conditionFailure(context, frame, new RuntimeException("breakpoint condition failure: non-boolean result " + conditionSource.getCode()));
}
} catch (Exception ex) {
conditionFailure(context, frame, new RuntimeException("breakpoint condition failure: " + conditionSource.getCode() + ex.getMessage()));
}
}
}
@Override
public String toString() {
final StringBuilder sb = new StringBuilder(getClass().getSimpleName());
sb.append(" state=");
sb.append(getState() == null ? "" : getState().getName());
if (isOneShot()) {
sb.append(", " + "One-Shot");
}
if (getCondition() != null) {
sb.append(", condition=\"" + getCondition() + "\"");
}
return sb.toString();
}
}
/** Attached to implement an unconditional breakpoint. */
private static final class BreakpointListener implements ExecutionEventListener {
private final BreakpointImpl breakpoint;
BreakpointListener(BreakpointImpl breakpoint) {
this.breakpoint = breakpoint;
}
@Override
public void onEnter(EventContext context, VirtualFrame frame) {
if (TRACE) {
trace("BEGIN HIT " + breakpoint.getShortDescription());
}
breakpoint.nodeEnter(context, frame);
if (TRACE) {
trace("END HIT " + breakpoint.getShortDescription());
}
}
@Override
public void onReturnValue(EventContext context, VirtualFrame frame, Object result) {
}
@Override
public void onReturnExceptional(EventContext context, VirtualFrame frame, Throwable exception) {
}
@Override
public String toString() {
return breakpoint.getShortDescription();
}
}
private static final class URILocation {
private final URI uri;
private final int line;
private final int column;
URILocation(URI uri, int line, int column) {
this.uri = uri;
this.line = line;
if (column < 0) {
this.column = -1;
} else {
this.column = column;
}
}
@Override
public int hashCode() {
return Objects.hash(this.uri, this.line, this.column);
}
@Override
public boolean equals(Object obj) {
if (this == obj) {
return true;
}
if (obj == null) {
return false;
}
if (getClass() != obj.getClass()) {
return false;
}
final URILocation other = (URILocation) obj;
if (this.line != other.line) {
return false;
}
if (this.column != other.column) {
return false;
}
if (!Objects.equals(this.uri, other.uri)) {
return false;
}
return true;
}
@Override
public String toString() {
return "URILocation{" + "uri=" + uri + ", line=" + line + ", column=" + column + '}';
}
}
}