All Downloads are FREE. Search and download functionalities are using the official Maven repository.

aQute.bnd.classfile.ClassFile Maven / Gradle / Ivy

The newest version!
package aQute.bnd.classfile;

import java.io.ByteArrayOutputStream;
import java.io.DataInput;
import java.io.DataOutput;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.lang.reflect.Modifier;
import java.nio.ByteBuffer;
import java.util.Arrays;

import aQute.lib.io.ByteBufferDataInput;
import aQute.lib.io.ByteBufferDataOutput;
import aQute.lib.io.IO;

public class ClassFile extends ElementInfo {
	public static final int		MAJOR_VERSION	= 61;
	public static final int		ACC_ANNOTATION	= 0x2000;
	public static final int		ACC_ENUM		= 0x4000;
	public static final int		ACC_MODULE		= 0x8000;

	public final int			minor_version;
	public final int			major_version;
	public final ConstantPool	constant_pool;
	public final String			this_class;
	public final String			super_class;
	public final String[]		interfaces;
	public final FieldInfo[]	fields;
	public final MethodInfo[]	methods;

	public ClassFile(int minor_version, int major_version, ConstantPool constant_pool, int access_flags,
		String this_class, String super_class, String[] interfaces, FieldInfo[] fields, MethodInfo[] methods,
		Attribute[] attributes) {
		super(access_flags, attributes);
		this.minor_version = minor_version;
		this.major_version = major_version;
		this.constant_pool = constant_pool;
		this.this_class = this_class;
		this.super_class = super_class;
		this.interfaces = interfaces;
		this.fields = fields;
		this.methods = methods;
	}

	@Override
	public String toString() {
		StringBuilder sb = new StringBuilder();
		sb.append(Modifier.toString(access & Modifier.classModifiers()));
		if (sb.length() > 0) {
			sb.append(' ');
		}
		if ((access & Modifier.INTERFACE) != 0) {
			if ((access & ACC_ANNOTATION) != 0) {
				sb.append('@');
			}
			sb.append("interface ");
		} else if ((access & ACC_ENUM) != 0) {
			sb.append("enum ");
		} else if ((access & ACC_MODULE) == 0) {
			sb.append("class ");
		}
		return sb.append(this_class)
			.append(' ')
			.append(Arrays.toString(attributes))
			.toString();
	}

	public static ClassFile parseClassFile(DataInput in) throws IOException {
		int magic = in.readInt();
		if (magic != 0xCAFEBABE) {
			throw new IOException("Not a valid class file (no CAFEBABE header)");
		}

		int minor_version = in.readUnsignedShort();
		int major_version = in.readUnsignedShort();

		ConstantPool constant_pool = ConstantPool.read(in);

		int access_flags = in.readUnsignedShort();

		int this_class_index = in.readUnsignedShort();
		String this_class = constant_pool.className(this_class_index);

		int super_class_index = in.readUnsignedShort();
		String super_class = (super_class_index != 0) ? constant_pool.className(super_class_index) : null;

		int interfaces_count = in.readUnsignedShort();
		String[] interfaces = new String[interfaces_count];
		for (int i = 0; i < interfaces_count; i++) {
			int interface_index = in.readUnsignedShort();
			interfaces[i] = constant_pool.className(interface_index);
		}

		int fields_count = in.readUnsignedShort();
		FieldInfo[] fields = new FieldInfo[fields_count];
		for (int i = 0; i < fields_count; i++) {
			fields[i] = FieldInfo.read(in, constant_pool);
		}

		int methods_count = in.readUnsignedShort();
		MethodInfo[] methods = new MethodInfo[methods_count];
		for (int i = 0; i < methods_count; i++) {
			methods[i] = MethodInfo.read(in, constant_pool);
		}

		Attribute[] attributes = Attribute.readAttributes(in, constant_pool);

		ClassFile class_file = new ClassFile(minor_version, major_version, constant_pool, access_flags, this_class,
			super_class, interfaces, fields, methods, attributes);
		return class_file;
	}

	public void write(DataOutput out) throws IOException {
		ByteBufferDataOutput bbout = new ByteBufferDataOutput();

		bbout.writeShort(access);

		int this_class_index = constant_pool.classInfo(this_class);
		bbout.writeShort(this_class_index);
		int super_class_index = (super_class != null) ? constant_pool.classInfo(super_class) : 0;
		bbout.writeShort(super_class_index);

		bbout.writeShort(interfaces.length);
		for (String interf : interfaces) {
			int interf_class_index = constant_pool.classInfo(interf);
			bbout.writeShort(interf_class_index);
		}

		bbout.writeShort(fields.length);
		for (FieldInfo field : fields) {
			field.write(bbout, constant_pool);
		}

		bbout.writeShort(methods.length);
		for (MethodInfo method : methods) {
			method.write(bbout, constant_pool);
		}

		Attribute.writeAttributes(bbout, constant_pool, attributes);

		out.writeInt(0xCAFEBABE);
		out.writeShort(minor_version);
		out.writeShort(major_version);

		constant_pool.write(out);
		IO.copy(bbout.toByteBuffer(), out);
	}

	static ByteBuffer slice(DataInput in, int length) throws IOException {
		if (in instanceof ByteBufferDataInput bbin) {
			return bbin.slice(length);
		}
		byte[] array = new byte[length];
		in.readFully(array, 0, length);
		return ByteBuffer.wrap(array, 0, length);
	}

	public void write(OutputStream out) throws IOException {
		DataOutput dout = new DataOutputStream(out);
		write(dout);
	}

	public byte[] write() throws IOException {
		ByteArrayOutputStream bout = new ByteArrayOutputStream();
		DataOutput dout = new DataOutputStream(bout);
		write(dout);
		return bout.toByteArray();
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy