blob: 82b6fe34961994c872fa0889943e6c63c04162d9 [file] [log] [blame]
/*
* Copyright (c) 2002, 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 sun.jvm.hotspot.interpreter;
import java.util.*;
import java.lang.reflect.Constructor;
import sun.jvm.hotspot.oops.*;
import sun.jvm.hotspot.utilities.*;
public class BytecodeDisassembler {
private Method method;
private static Map bytecode2Class = new HashMap(); // Map<int, Class>
private static void addBytecodeClass(int bytecode, Class clazz) {
bytecode2Class.put(new Integer(bytecode), clazz);
}
private static Class getBytecodeClass(int bytecode) {
return (Class) bytecode2Class.get(new Integer(bytecode));
}
static {
addBytecodeClass(Bytecodes._anewarray, BytecodeANewArray.class);
addBytecodeClass(Bytecodes._bipush, BytecodeBipush.class);
addBytecodeClass(Bytecodes._checkcast, BytecodeCheckCast.class);
addBytecodeClass(Bytecodes._getfield, BytecodeGetField.class);
addBytecodeClass(Bytecodes._getstatic, BytecodeGetStatic.class);
addBytecodeClass(Bytecodes._goto, BytecodeGoto.class);
addBytecodeClass(Bytecodes._goto_w, BytecodeGotoW.class);
addBytecodeClass(Bytecodes._ifeq, BytecodeIf.class);
addBytecodeClass(Bytecodes._ifne, BytecodeIf.class);
addBytecodeClass(Bytecodes._iflt, BytecodeIf.class);
addBytecodeClass(Bytecodes._ifge, BytecodeIf.class);
addBytecodeClass(Bytecodes._ifgt, BytecodeIf.class);
addBytecodeClass(Bytecodes._ifle, BytecodeIf.class);
addBytecodeClass(Bytecodes._if_icmpeq, BytecodeIf.class);
addBytecodeClass(Bytecodes._if_icmpne, BytecodeIf.class);
addBytecodeClass(Bytecodes._if_icmplt, BytecodeIf.class);
addBytecodeClass(Bytecodes._if_icmpge, BytecodeIf.class);
addBytecodeClass(Bytecodes._if_icmpgt, BytecodeIf.class);
addBytecodeClass(Bytecodes._if_icmple, BytecodeIf.class);
addBytecodeClass(Bytecodes._if_acmpeq, BytecodeIf.class);
addBytecodeClass(Bytecodes._if_acmpne, BytecodeIf.class);
addBytecodeClass(Bytecodes._ifnull, BytecodeIf.class);
addBytecodeClass(Bytecodes._ifnonnull, BytecodeIf.class);
addBytecodeClass(Bytecodes._iinc, BytecodeIinc.class);
addBytecodeClass(Bytecodes._instanceof, BytecodeInstanceOf.class);
addBytecodeClass(Bytecodes._invokevirtual, BytecodeInvoke.class);
addBytecodeClass(Bytecodes._invokestatic, BytecodeInvoke.class);
addBytecodeClass(Bytecodes._invokespecial, BytecodeInvoke.class);
addBytecodeClass(Bytecodes._invokeinterface, BytecodeInvoke.class);
addBytecodeClass(Bytecodes._invokedynamic, BytecodeInvoke.class);
addBytecodeClass(Bytecodes._jsr, BytecodeJsr.class);
addBytecodeClass(Bytecodes._jsr_w, BytecodeJsrW.class);
addBytecodeClass(Bytecodes._iload, BytecodeLoad.class);
addBytecodeClass(Bytecodes._lload, BytecodeLoad.class);
addBytecodeClass(Bytecodes._fload, BytecodeLoad.class);
addBytecodeClass(Bytecodes._dload, BytecodeLoad.class);
addBytecodeClass(Bytecodes._aload, BytecodeLoad.class);
addBytecodeClass(Bytecodes._ldc, BytecodeLoadConstant.class);
addBytecodeClass(Bytecodes._ldc_w, BytecodeLoadConstant.class);
addBytecodeClass(Bytecodes._ldc2_w, BytecodeLoadConstant.class);
addBytecodeClass(Bytecodes._lookupswitch, BytecodeLookupswitch.class);
addBytecodeClass(Bytecodes._multianewarray, BytecodeMultiANewArray.class);
addBytecodeClass(Bytecodes._new, BytecodeNew.class);
addBytecodeClass(Bytecodes._newarray, BytecodeNewArray.class);
addBytecodeClass(Bytecodes._putfield, BytecodePutField.class);
addBytecodeClass(Bytecodes._putstatic, BytecodePutStatic.class);
addBytecodeClass(Bytecodes._ret, BytecodeRet.class);
addBytecodeClass(Bytecodes._sipush, BytecodeSipush.class);
addBytecodeClass(Bytecodes._istore, BytecodeStore.class);
addBytecodeClass(Bytecodes._lstore, BytecodeStore.class);
addBytecodeClass(Bytecodes._fstore, BytecodeStore.class);
addBytecodeClass(Bytecodes._dstore, BytecodeStore.class);
addBytecodeClass(Bytecodes._astore, BytecodeStore.class);
addBytecodeClass(Bytecodes._tableswitch, BytecodeTableswitch.class);
}
public BytecodeDisassembler(Method method) {
this.method = method;
}
public Method getMethod() {
return method;
}
public void decode(BytecodeVisitor visitor) {
visitor.prologue(method);
BytecodeStream stream = new BytecodeStream(method);
int javacode = Bytecodes._illegal;
while ( (javacode = stream.next()) != Bytecodes._illegal) {
// look for special Bytecode class
int bci = stream.bci();
int hotspotcode = method.getBytecodeOrBPAt(bci);
Class clazz = getBytecodeClass(javacode);
if (clazz == null) {
// check for fast_(i|a)_access_0
clazz = getBytecodeClass(hotspotcode);
if (clazz == null) {
// use generic bytecode class
clazz = Bytecode.class;
}
}
// All bytecode classes must have a constructor with signature
// (Lsun/jvm/hotspot/oops/Method;I)V
Constructor cstr = null;
try {
cstr = clazz.getDeclaredConstructor(new Class[] { Method.class, Integer.TYPE });
} catch(NoSuchMethodException nomethod) {
if (Assert.ASSERTS_ENABLED) {
Assert.that(false, "Bytecode class without proper constructor!");
}
}
Bytecode bytecodeObj = null;
try {
bytecodeObj = (Bytecode)cstr.newInstance(new Object[] { method, new Integer(bci) });
} catch (Exception exp) {
if (Assert.ASSERTS_ENABLED) {
Assert.that(false, "Bytecode instance of class "
+ clazz.getName() + " can not be created!");
}
}
if (stream.isWide()) {
visitor.visit(new Bytecode(method, bci - 1));
}
try {
visitor.visit(bytecodeObj);
} catch(ClassCastException castfail) {
castfail.printStackTrace();
System.err.println(method.getAddress() + " " + bci);
}
}
visitor.epilogue();
}
}