Mercurial > hg > truffle
view graal/com.oracle.max.graal.debug/src/com/oracle/max/graal/debug/internal/DebugScope.java @ 4612:d8ff6f2bcabd
Use parameters in graph group names for the visualizer so that successive compilation of methods with same name but different signature don't get fused
Made CFGPrinterObserver a bit more robust to failed schedule
When an exception occurs while intercepting exception, this new exception should be seen
author | Gilles Duboscq <duboscq@ssw.jku.at> |
---|---|
date | Wed, 15 Feb 2012 20:05:06 +0100 |
parents | 14a00ee82980 |
children | 7a4d120da7a0 |
line wrap: on
line source
/* * Copyright (c) 2012, 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. * * 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.max.graal.debug.internal; import java.io.*; import java.util.*; import java.util.concurrent.*; import com.oracle.max.graal.debug.*; public final class DebugScope { private static ThreadLocal<DebugScope> instanceTL = new ThreadLocal<>(); private static ThreadLocal<DebugConfig> configTL = new ThreadLocal<>(); private static ThreadLocal<Throwable> lastExceptionThrownTL = new ThreadLocal<>(); private static DebugTimer scopeTime = Debug.timer("ScopeTime"); private static DebugMetric scopeCount = Debug.metric("ScopeCount"); private final DebugScope parent; private Object[] context; private List<DebugScope> children; private DebugValueMap valueMap; private String qualifiedName; private String name; private static final char SCOPE_SEP = '.'; private boolean logEnabled; private boolean meterEnabled; private boolean timeEnabled; private boolean dumpEnabled; public static DebugScope getInstance() { DebugScope result = instanceTL.get(); if (result == null) { DebugScope topLevelDebugScope = new DebugScope(Thread.currentThread().getName(), "", null); instanceTL.set(topLevelDebugScope); DebugValueMap.registerTopLevel(topLevelDebugScope.getValueMap()); return topLevelDebugScope; } else { return result; } } public static DebugConfig getConfig() { return configTL.get(); } private DebugScope(String name, String qualifiedName, DebugScope parent, Object... context) { this.name = name; this.parent = parent; this.context = context; this.qualifiedName = qualifiedName; assert context != null; } public boolean isDumpEnabled() { return dumpEnabled; } public boolean isLogEnabled() { return logEnabled; } public boolean isMeterEnabled() { return meterEnabled; } public boolean isTimeEnabled() { return timeEnabled; } public void log(String msg, Object... args) { if (isLogEnabled()) { cachedOut.println(String.format(msg, args)); } } public void dump(Object object, String formatString, Object[] args) { if (isDumpEnabled()) { DebugConfig config = getConfig(); if (config != null) { String message = String.format(formatString, args); for (DebugDumpHandler dumpHandler : config.dumpHandlers()) { dumpHandler.dump(object, message); } } } } public <T> T scope(String newName, Runnable runnable, Callable<T> callable, boolean sandbox, Object[] newContext) { DebugScope oldContext = getInstance(); DebugConfig oldConfig = getConfig(); DebugScope newChild = null; if (sandbox) { newChild = new DebugScope(newName, newName, null, newContext); setConfig(null); } else { newChild = oldContext.createChild(newName, newContext); } instanceTL.set(newChild); newChild.updateFlags(); scopeCount.increment(); try (TimerCloseable a = scopeTime.start()) { return executeScope(runnable, callable); } finally { if (!sandbox && newChild.hasValueMap()) { getValueMap().addChild(newChild.getValueMap()); } newChild.context = null; instanceTL.set(oldContext); setConfig(oldConfig); } } private <T> T executeScope(Runnable runnable, Callable<T> callable) { try { if (runnable != null) { runnable.run(); } if (callable != null) { return call(callable); } } catch (Throwable e) { if (e == lastExceptionThrownTL.get()) { throw e; } else { RuntimeException newException = interceptException(e); if (newException == null) { lastExceptionThrownTL.set(e); throw e; } else { lastExceptionThrownTL.set(newException); throw newException; } } } return null; } private void updateFlags() { DebugConfig config = getConfig(); if (config == null) { logEnabled = false; meterEnabled = false; timeEnabled = false; dumpEnabled = false; } else { logEnabled = config.isLogEnabled(); meterEnabled = config.isMeterEnabled(); timeEnabled = config.isTimeEnabled(); dumpEnabled = config.isDumpEnabled(); } } private RuntimeException interceptException(final Throwable e) { final DebugConfig config = getConfig(); if (config != null) { return scope("InterceptException", null, new Callable<RuntimeException>() { @Override public RuntimeException call() throws Exception { try { return config.interceptException(e); } catch (Throwable t) { return new RuntimeException("Exception while intercepting exception", t); } } }, false, new Object[] {e}); } return null; } private DebugValueMap getValueMap() { if (valueMap == null) { valueMap = new DebugValueMap(name); } return valueMap; } private boolean hasValueMap() { return valueMap != null; } long getCurrentValue(int index) { return getValueMap().getCurrentValue(index); } void setCurrentValue(int index, long l) { getValueMap().setCurrentValue(index, l); } private DebugScope createChild(String newName, Object[] newContext) { String newQualifiedName = newName; if (this.qualifiedName.length() > 0) { newQualifiedName = this.qualifiedName + SCOPE_SEP + newName; } DebugScope result = new DebugScope(newName, newQualifiedName, this, newContext); if (children == null) { children = new ArrayList<>(4); } children.add(result); return result; } public Iterable<Object> getCurrentContext() { return new Iterable<Object>() { @Override public Iterator<Object> iterator() { return new Iterator<Object>() { DebugScope currentScope = DebugScope.this; int objectIndex; @Override public boolean hasNext() { selectScope(); return currentScope != null; } private void selectScope() { while (currentScope != null && currentScope.context.length <= objectIndex) { currentScope = currentScope.parent; objectIndex = 0; } } @Override public Object next() { selectScope(); if (currentScope != null) { return currentScope.context[objectIndex++]; } throw new IllegalStateException("May only be called if there is a next element."); } @Override public void remove() { throw new UnsupportedOperationException("This iterator is read only."); } }; } }; } public static <T> T call(Callable<T> callable) { try { return callable.call(); } catch (Exception e) { if (e instanceof RuntimeException) { throw (RuntimeException) e; } else { throw new RuntimeException(e); } } } public void setConfig(DebugConfig newConfig) { configTL.set(newConfig); updateFlags(); } public String getQualifiedName() { return qualifiedName; } public static PrintStream cachedOut; public static void initialize() { cachedOut = System.out; } }