/*
 * Copyright (c) 2000, 2017, 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 sun.jvm.hotspot.runtime;

import java.io.*;
import java.util.*;
import sun.jvm.hotspot.oops.*;
import sun.jvm.hotspot.utilities.*;
import sun.jvm.hotspot.debugger.*;

public abstract class JavaVFrame extends VFrame {

  private static final String ADDRESS_FORMAT = VM.getVM().isLP64() ? "0x%016x"
                                                                   : "0x%08x";

  
JVM state
/** JVM state */
public abstract Method getMethod(); public abstract int getBCI(); public abstract StackValueCollection getLocals(); public abstract StackValueCollection getExpressions(); public abstract List<MonitorInfo> getMonitors();
Test operation
/** Test operation */
public boolean isJavaFrame() { return true; }
Package-internal constructor
/** Package-internal constructor */
JavaVFrame(Frame fr, RegisterMap regMap, JavaThread thread) { super(fr, regMap, thread); }
Get monitor (if any) that this JavaVFrame is trying to enter
/** Get monitor (if any) that this JavaVFrame is trying to enter */
// FIXME: not yet implemented // public Address getPendingMonitor(int frameCount); public void printLockedObjectClassName(PrintStream tty, OopHandle hobj, String lockState) { if (hobj.asLongValue() != 0L) { tty.format("\t- %s <" + ADDRESS_FORMAT + "> ", lockState, hobj.asLongValue()); Klass klass = Oop.getKlassForOopHandle(hobj); String klassName = klass.getName().asString(); tty.print("(a "); if (klassName.equals("java/lang/Class")) { Oop obj = VM.getVM().getObjectHeap().newOop(hobj); klassName = java_lang_Class.asExternalName(obj); tty.print("java.lang.Class for "); } tty.println(klassName.replace('/', '.') + ")"); } } private String identifyLockState(MonitorInfo monitor, String waitingState) { Mark mark = new Mark(monitor.owner()); if (mark.hasMonitor() && ( // we have marked ourself as pending on this monitor mark.monitor().equals(thread.getCurrentPendingMonitor()) || // we are not the owner of this monitor !mark.monitor().isEntered(thread) )) { return waitingState; } return "locked"; }
Printing used during stack dumps
/** Printing used during stack dumps */
public void printLockInfo(PrintStream tty, int frameCount) { // If this is the first frame and it is java.lang.Object.wait(...) // then print out the receiver. Locals are not always available, // e.g., compiled native frames have no scope so there are no locals. if (frameCount == 0) { if (getMethod().getName().asString().equals("wait") && getMethod().getMethodHolder().getName().asString().equals("java/lang/Object")) { String waitState = "waiting on"; // assume we are waiting // If earlier in the output we reported java.lang.Thread.State == // "WAITING (on object monitor)" and now we report "waiting on", then // we are still waiting for notification or timeout. Otherwise if // we earlier reported java.lang.Thread.State == "BLOCKED (on object // monitor)", then we are actually waiting to re-lock the monitor. // At this level we can't distinguish the two cases to report // "waited on" rather than "waiting on" for the second case. StackValueCollection locs = getLocals(); if (!locs.isEmpty()) { StackValue sv = locs.get(0); if (sv.getType() == BasicType.getTObject()) { OopHandle o = sv.getObject(); printLockedObjectClassName(tty, o, waitState); } } else { tty.println("\t- " + waitState + " <no object reference available>"); } } else if (thread.getCurrentParkBlocker() != null) { Oop obj = thread.getCurrentParkBlocker(); Klass k = obj.getKlass(); tty.format("\t- parking to wait for <" + ADDRESS_FORMAT + "> (a %s)", obj.getHandle().asLongValue(), k.getName().asString()); tty.println(); } } // Print out all monitors that we have locked, or are trying to lock, // including re-locking after being notified or timing out in a wait(). List<MonitorInfo> mons = getMonitors(); if (!mons.isEmpty()) { boolean foundFirstMonitor = false; for (int index = mons.size() - 1; index >= 0; index--) { MonitorInfo monitor = mons.get(index); if (monitor.eliminated() && isCompiledFrame()) { // Eliminated in compiled code if (monitor.ownerIsScalarReplaced()) { Klass k = Oop.getKlassForOopHandle(monitor.ownerKlass()); tty.println("\t- eliminated <owner is scalar replaced> (a " + k.getName().asString() + ")"); } else if (monitor.owner() != null) { printLockedObjectClassName(tty, monitor.owner(), "eliminated"); } continue; } if (monitor.owner() != null) { // the monitor is associated with an object, i.e., it is locked String lockState = "locked"; if (!foundFirstMonitor && frameCount == 0) { // If this is the first frame and we haven't found an owned // monitor before, then we need to see if we have completed // the lock or if we are blocked trying to acquire it. Only // an inflated monitor that is first on the monitor list in // the first frame can block us on a monitor enter. lockState = identifyLockState(monitor, "waiting to lock"); } else if (frameCount != 0) { // This is not the first frame so we either own this monitor // or we owned the monitor before and called wait(). Because // wait() could have been called on any monitor in a lower // numbered frame on the stack, we have to check all the // monitors on the list for this frame. lockState = identifyLockState(monitor, "waiting to re-lock in wait()"); } printLockedObjectClassName(tty, monitor.owner(), lockState); foundFirstMonitor = true; } } } }
Printing operations
/** Printing operations */
// // FIXME: implement visitor pattern for traversing vframe contents? // public void print() { printOn(System.out); } public void printOn(PrintStream tty) { super.printOn(tty); tty.print("\t"); getMethod().printValueOn(tty); tty.println(); tty.println("\tbci:\t" + getBCI()); printStackValuesOn(tty, "locals", getLocals()); printStackValuesOn(tty, "expressions", getExpressions()); } public void printActivation(int index) { printActivationOn(System.out, index); } public void printActivationOn(PrintStream tty, int index) { // frame number and method tty.print(index + " - "); printValueOn(tty); tty.println(); if (VM.getVM().wizardMode()) { printOn(tty); tty.println(); } }
Verification operations
/** Verification operations */
public void verify() { } public boolean equals(Object o) { if (o == null || !(o instanceof JavaVFrame)) { return false; } JavaVFrame other = (JavaVFrame) o; // Check static part if (!getMethod().equals(other.getMethod())) { return false; } if (getBCI() != other.getBCI()) { return false; } // dynamic part - we just compare the frame pointer if (! getFrame().equals(other.getFrame())) { return false; } return true; } public int hashCode() { return getMethod().hashCode() ^ getBCI() ^ getFrame().hashCode(); }
Structural compare
/** Structural compare */
public boolean structuralCompare(JavaVFrame other) { // Check static part if (!getMethod().equals(other.getMethod())) { return false; } if (getBCI() != other.getBCI()) { return false; } // Check locals StackValueCollection locs = getLocals(); StackValueCollection otherLocs = other.getLocals(); if (Assert.ASSERTS_ENABLED) { Assert.that(locs.size() == otherLocs.size(), "sanity check"); } for (int i = 0; i < locs.size(); i++) { // it might happen the compiler reports a conflict and // the interpreter reports a bogus int. if ( isCompiledFrame() && (locs.get(i)).getType() == BasicType.getTConflict()) continue; if (other.isCompiledFrame() && (otherLocs.get(i)).getType() == BasicType.getTConflict()) continue; if (!locs.get(i).equals(otherLocs.get(i))) { return false; } } // Check expressions StackValueCollection exprs = getExpressions(); StackValueCollection otherExprs = other.getExpressions(); if (Assert.ASSERTS_ENABLED) { Assert.that(exprs.size() == otherExprs.size(), "sanity check"); } for (int i = 0; i < exprs.size(); i++) { if (!exprs.get(i).equals(otherExprs.get(i))) { return false; } } return true; } //-------------------------------------------------------------------------------- // Internals only below this point // private void printStackValuesOn(PrintStream tty, String title, StackValueCollection values) { if (values.isEmpty()) { return; } tty.println("\t" + title + ":"); for (int index = 0; index < values.size(); index++) { tty.print("\t" + index + "\t"); values.get(index).printOn(tty); tty.println(); } } }