You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
353 lines
9.8 KiB
353 lines
9.8 KiB
11 years ago
|
/*
|
||
|
* Fernflower - The Analytical Java Decompiler
|
||
|
* http://www.reversed-java.com
|
||
|
*
|
||
|
* (C) 2008 - 2010, Stiver
|
||
|
*
|
||
|
* This software is NEITHER public domain NOR free software
|
||
|
* as per GNU License. See license.txt for more details.
|
||
|
*
|
||
|
* This software is distributed WITHOUT ANY WARRANTY; without
|
||
|
* even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
||
|
* A PARTICULAR PURPOSE.
|
||
|
*/
|
||
|
|
||
10 years ago
|
package org.jetbrains.java.decompiler.struct;
|
||
11 years ago
|
|
||
|
import java.io.DataOutputStream;
|
||
|
import java.io.File;
|
||
|
import java.io.FileInputStream;
|
||
|
import java.io.FileNotFoundException;
|
||
|
import java.io.FileOutputStream;
|
||
|
import java.io.IOException;
|
||
|
import java.io.InputStream;
|
||
|
|
||
10 years ago
|
import org.jetbrains.java.decompiler.code.CodeConstants;
|
||
|
import org.jetbrains.java.decompiler.struct.attr.StructGeneralAttribute;
|
||
|
import org.jetbrains.java.decompiler.struct.consts.ConstantPool;
|
||
|
import org.jetbrains.java.decompiler.struct.consts.PrimitiveConstant;
|
||
|
import org.jetbrains.java.decompiler.struct.lazy.LazyLoader;
|
||
|
import org.jetbrains.java.decompiler.util.DataInputFullStream;
|
||
|
import org.jetbrains.java.decompiler.util.InterpreterUtil;
|
||
|
import org.jetbrains.java.decompiler.util.VBStyleCollection;
|
||
11 years ago
|
|
||
|
/*
|
||
|
ClassFile {
|
||
|
u4 magic;
|
||
|
u2 minor_version;
|
||
|
u2 major_version;
|
||
|
u2 constant_pool_count;
|
||
|
cp_info constant_pool[constant_pool_count-1];
|
||
|
u2 access_flags;
|
||
|
u2 this_class;
|
||
|
u2 super_class;
|
||
|
u2 interfaces_count;
|
||
|
u2 interfaces[interfaces_count];
|
||
|
u2 fields_count;
|
||
|
field_info fields[fields_count];
|
||
|
u2 methods_count;
|
||
|
method_info methods[methods_count];
|
||
|
u2 attributes_count;
|
||
|
attribute_info attributes[attributes_count];
|
||
|
}
|
||
|
*/
|
||
|
|
||
|
public class StructClass {
|
||
|
|
||
|
// *****************************************************************************
|
||
|
// public fields
|
||
|
// *****************************************************************************
|
||
|
|
||
|
public int minor_version;
|
||
|
|
||
|
public int major_version;
|
||
|
|
||
|
public int access_flags;
|
||
|
|
||
|
public int this_class;
|
||
|
|
||
|
public int super_class;
|
||
|
|
||
|
public PrimitiveConstant thisClass;
|
||
|
|
||
|
public PrimitiveConstant superClass;
|
||
|
|
||
|
public String qualifiedName;
|
||
|
|
||
|
|
||
|
// *****************************************************************************
|
||
|
// private fields
|
||
|
// *****************************************************************************
|
||
|
|
||
|
private ConstantPool pool;
|
||
|
|
||
|
private int[] interfaces;
|
||
|
|
||
|
private String[] interfaceNames;
|
||
|
|
||
|
private VBStyleCollection<StructField, String> fields = new VBStyleCollection<StructField, String>();
|
||
|
|
||
|
private VBStyleCollection<StructMethod, String> methods = new VBStyleCollection<StructMethod, String>();
|
||
|
|
||
|
private VBStyleCollection<StructGeneralAttribute, String> attributes = new VBStyleCollection<StructGeneralAttribute, String>();
|
||
|
|
||
|
private boolean own = true;
|
||
|
|
||
|
private LazyLoader loader;
|
||
|
|
||
|
// *****************************************************************************
|
||
|
// constructors
|
||
|
// *****************************************************************************
|
||
|
|
||
|
public StructClass(String filename, boolean own, LazyLoader loader) throws FileNotFoundException, IOException {
|
||
|
this(new FileInputStream(filename), own, loader);
|
||
|
}
|
||
|
|
||
|
public StructClass(InputStream inStream, boolean own, LazyLoader loader) throws FileNotFoundException, IOException {
|
||
|
this(new DataInputFullStream(inStream), own, loader);
|
||
|
}
|
||
|
|
||
|
public StructClass(DataInputFullStream inStream, boolean own, LazyLoader loader) throws FileNotFoundException, IOException {
|
||
|
this.own = own;
|
||
|
this.loader = loader;
|
||
|
|
||
|
initStruct(inStream);
|
||
|
}
|
||
|
|
||
|
// *****************************************************************************
|
||
|
// public methods
|
||
|
// *****************************************************************************
|
||
|
|
||
10 years ago
|
public boolean hasField(String name, String descriptor) {
|
||
|
return getField(name, descriptor) != null;
|
||
|
}
|
||
|
|
||
11 years ago
|
public StructField getField(String name, String descriptor) {
|
||
|
return fields.getWithKey(InterpreterUtil.makeUniqueKey(name, descriptor));
|
||
|
}
|
||
|
|
||
11 years ago
|
public StructMethod getMethod(String key) {
|
||
|
return methods.getWithKey(key);
|
||
|
}
|
||
|
|
||
11 years ago
|
public StructMethod getMethod(String name, String descriptor) {
|
||
|
return methods.getWithKey(InterpreterUtil.makeUniqueKey(name, descriptor));
|
||
|
}
|
||
|
|
||
|
public void writeToFile(File file) throws IOException {
|
||
|
DataOutputStream out = new DataOutputStream(new FileOutputStream(file));
|
||
|
writeToOutputStream(out);
|
||
|
out.close();
|
||
|
}
|
||
|
|
||
|
public void writeToOutputStream(DataOutputStream out) throws IOException {
|
||
|
|
||
|
out.writeInt(0xCAFEBABE);
|
||
|
out.writeShort(minor_version);
|
||
|
out.writeShort(major_version);
|
||
|
|
||
|
getPool().writeToOutputStream(out);
|
||
|
|
||
|
out.writeShort(access_flags);
|
||
|
out.writeShort(this_class);
|
||
|
out.writeShort(super_class);
|
||
|
|
||
|
out.writeShort(interfaces.length);
|
||
|
for(int i=0;i<interfaces.length;i++) {
|
||
|
out.writeShort(interfaces[i]);
|
||
|
}
|
||
|
|
||
|
out.writeShort(fields.size());
|
||
|
for(int i=0;i<fields.size();i++) {
|
||
|
fields.get(i).writeToStream(out);
|
||
|
}
|
||
|
|
||
|
out.writeShort(methods.size());
|
||
|
for(int i=0;i<methods.size();i++) {
|
||
|
methods.get(i).writeToStream(out);
|
||
|
}
|
||
|
|
||
|
out.writeShort(attributes.size());
|
||
|
for(StructGeneralAttribute attr: attributes) {
|
||
|
attr.writeToStream(out);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
public String getInterface(int i) {
|
||
|
return interfaceNames[i];
|
||
|
}
|
||
|
|
||
|
public void releaseResources() {
|
||
|
if(loader != null) {
|
||
|
pool = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// *****************************************************************************
|
||
|
// private methods
|
||
|
// *****************************************************************************
|
||
|
|
||
|
private void initStruct(DataInputFullStream in) throws IOException {
|
||
|
|
||
|
in.skip(4);
|
||
|
|
||
|
this.minor_version = in.readUnsignedShort();
|
||
|
this.major_version = in.readUnsignedShort();
|
||
|
|
||
|
pool = new ConstantPool(in);
|
||
|
|
||
|
this.access_flags = in.readUnsignedShort();
|
||
|
|
||
|
this_class = in.readUnsignedShort();
|
||
|
thisClass = pool.getPrimitiveConstant(this_class);
|
||
|
qualifiedName = thisClass.getString();
|
||
|
|
||
|
super_class = in.readUnsignedShort();
|
||
|
superClass = pool.getPrimitiveConstant(super_class);
|
||
|
|
||
|
// interfaces
|
||
|
int length = in.readUnsignedShort();
|
||
|
int[] arrInterfaces = new int[length];
|
||
|
String[] arrInterfaceNames = new String[length];
|
||
|
|
||
|
for (int i = 0; i < length; i++) {
|
||
|
arrInterfaces[i] = in.readUnsignedShort();
|
||
|
arrInterfaceNames[i] = pool.getPrimitiveConstant(arrInterfaces[i]).getString();
|
||
|
}
|
||
|
this.interfaces = arrInterfaces;
|
||
|
this.interfaceNames = arrInterfaceNames;
|
||
|
|
||
|
// fields
|
||
|
VBStyleCollection<StructField, String> lstFields = new VBStyleCollection<StructField, String>();
|
||
|
length = in.readUnsignedShort();
|
||
|
for (int i = 0; i < length; i++) {
|
||
|
StructField field = new StructField();
|
||
|
field.access_flags = in.readUnsignedShort();
|
||
|
field.name_index = in.readUnsignedShort();
|
||
|
field.descriptor_index = in.readUnsignedShort();
|
||
|
|
||
|
field.initStrings(pool, this_class);
|
||
|
|
||
|
field.setAttributes(readAttributes(in));
|
||
|
|
||
|
lstFields.addWithKey(field, InterpreterUtil.makeUniqueKey(field.getName(), field.getDescriptor()));
|
||
|
}
|
||
|
this.fields = lstFields;
|
||
|
|
||
|
// methods
|
||
|
length = in.readUnsignedShort();
|
||
|
for (int i = 0; i < length; i++) {
|
||
|
StructMethod meth = new StructMethod(in, own, this);
|
||
|
|
||
|
//if(qualifiedName.endsWith("JUnitStatusLine") && !meth.getName().equals("onProcessStarted") && !meth.getName().startsWith("access")) {
|
||
|
//if(!meth.getName().equals("run")) {
|
||
|
// continue;
|
||
|
//}
|
||
|
|
||
|
methods.addWithKey(meth, InterpreterUtil.makeUniqueKey(meth.getName(), meth.getDescriptor()));
|
||
|
}
|
||
|
|
||
|
// attributes
|
||
|
this.attributes = readAttributes(in);
|
||
|
|
||
|
|
||
|
// release memory
|
||
|
if(loader != null) {
|
||
|
pool = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private VBStyleCollection<StructGeneralAttribute, String> readAttributes(DataInputFullStream in) throws IOException {
|
||
|
|
||
|
VBStyleCollection<StructGeneralAttribute, String> lstAttribute = new VBStyleCollection<StructGeneralAttribute, String>();
|
||
|
|
||
|
int length = in.readUnsignedShort();
|
||
|
for (int i = 0; i < length; i++) {
|
||
|
int attr_nameindex = in.readUnsignedShort();
|
||
|
String attrname = pool.getPrimitiveConstant(attr_nameindex).getString();
|
||
|
|
||
|
StructGeneralAttribute attr = StructGeneralAttribute.getMatchingAttributeInstance(attr_nameindex, attrname);
|
||
|
|
||
|
if(attr != null) {
|
||
|
byte[] arr = new byte[in.readInt()];
|
||
|
in.readFull(arr);
|
||
|
attr.setInfo(arr);
|
||
|
|
||
|
attr.initContent(pool);
|
||
|
lstAttribute.addWithKey(attr, attr.getName());
|
||
|
} else {
|
||
|
in.skip(in.readInt());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return lstAttribute;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// *****************************************************************************
|
||
|
// getter and setter methods
|
||
|
// *****************************************************************************
|
||
|
|
||
|
public ConstantPool getPool() {
|
||
|
|
||
|
if(pool == null && loader != null) {
|
||
|
pool = loader.loadPool(qualifiedName);
|
||
|
}
|
||
|
|
||
|
return pool;
|
||
|
}
|
||
|
|
||
|
public int[] getInterfaces() {
|
||
|
return interfaces;
|
||
|
}
|
||
|
|
||
|
public String[] getInterfaceNames() {
|
||
|
return interfaceNames;
|
||
|
}
|
||
|
|
||
|
public VBStyleCollection<StructMethod, String> getMethods() {
|
||
|
return methods;
|
||
|
}
|
||
|
|
||
|
public VBStyleCollection<StructField, String> getFields() {
|
||
|
return fields;
|
||
|
}
|
||
|
|
||
|
public VBStyleCollection<StructGeneralAttribute, String> getAttributes() {
|
||
|
return attributes;
|
||
|
}
|
||
|
|
||
|
public boolean isOwn() {
|
||
|
return own;
|
||
|
}
|
||
|
|
||
|
public LazyLoader getLoader() {
|
||
|
return loader;
|
||
|
}
|
||
|
|
||
11 years ago
|
public boolean isVersionGE_1_5() {
|
||
|
return (major_version > 48 || (major_version == 48 && minor_version > 0)); // FIXME: check second condition
|
||
|
}
|
||
|
|
||
|
public boolean isVersionGE_1_7() {
|
||
|
return (major_version >= 51);
|
||
|
}
|
||
|
|
||
|
public int getBytecodeVersion() {
|
||
|
switch(major_version) {
|
||
|
case 52:
|
||
|
return CodeConstants.BYTECODE_JAVA_8;
|
||
|
case 51:
|
||
|
return CodeConstants.BYTECODE_JAVA_7;
|
||
|
case 50:
|
||
|
return CodeConstants.BYTECODE_JAVA_6;
|
||
|
case 49:
|
||
|
return CodeConstants.BYTECODE_JAVA_5;
|
||
|
}
|
||
|
|
||
|
return CodeConstants.BYTECODE_JAVA_LE_4;
|
||
|
}
|
||
11 years ago
|
}
|