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

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

The newest version!
package aQute.bnd.classfile;

import java.io.DataInput;
import java.io.DataOutput;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.util.Arrays;

import aQute.lib.io.IO;

public class CodeAttribute implements Attribute {
	public static final String		NAME	= "Code";
	public final int				max_stack;
	public final int				max_locals;
	public final ByteBuffer			code;
	public final ExceptionHandler[]	exception_table;
	public final Attribute[]		attributes;

	public CodeAttribute(int max_stack, int max_locals, ByteBuffer code, ExceptionHandler[] exception_table,
		Attribute[] attributes) {
		this.max_stack = max_stack;
		this.max_locals = max_locals;
		this.code = code;
		this.exception_table = exception_table;
		this.attributes = attributes;
	}

	@Override
	public String name() {
		return NAME;
	}

	@Override
	public String toString() {
		return NAME + " (" + code.limit() + " bytes) " + Arrays.toString(attributes);
	}

	public static CodeAttribute read(DataInput in, ConstantPool constant_pool) throws IOException {
		int max_stack = in.readUnsignedShort();
		int max_locals = in.readUnsignedShort();
		int code_length = in.readInt();
		ByteBuffer code = ClassFile.slice(in, code_length);
		int exception_table_length = in.readUnsignedShort();
		ExceptionHandler[] exception_table = new ExceptionHandler[exception_table_length];
		for (int i = 0; i < exception_table_length; i++) {
			exception_table[i] = ExceptionHandler.read(in, constant_pool);
		}
		Attribute[] attributes = Attribute.readAttributes(in, constant_pool);
		return new CodeAttribute(max_stack, max_locals, code, exception_table, attributes);
	}

	@Override
	public void write(DataOutput out, ConstantPool constant_pool) throws IOException {
		int attribute_name_index = constant_pool.utf8Info(name());
		int attribute_length = attribute_length();
		out.writeShort(attribute_name_index);
		out.writeInt(attribute_length);
		out.writeShort(max_stack);
		out.writeShort(max_locals);
		ByteBuffer duplicate = code.duplicate();
		duplicate.rewind();
		int code_length = duplicate.limit();
		out.writeInt(code_length);
		IO.copy(duplicate, out);
		out.writeShort(exception_table.length);
		for (ExceptionHandler exception : exception_table) {
			exception.write(out, constant_pool);
		}
		Attribute.writeAttributes(out, constant_pool, attributes);
	}

	@Override
	public int attribute_length() {
		int attribute_length = 3 * Short.BYTES + 1 * Integer.BYTES + code.limit();
		for (ExceptionHandler exception : exception_table) {
			attribute_length += exception.value_length();
		}
		attribute_length += Attribute.attributes_length(attributes);
		return attribute_length;
	}

	public static class ExceptionHandler {
		public final int	start_pc;
		public final int	end_pc;
		public final int	handler_pc;
		public final String	catch_type;

		public ExceptionHandler(int start_pc, int end_pc, int handler_pc, String catch_type) {
			this.start_pc = start_pc;
			this.end_pc = end_pc;
			this.handler_pc = handler_pc;
			this.catch_type = catch_type;
		}

		@Override
		public String toString() {
			return String.format("from=%d to=%d target=%d type=%s", start_pc, end_pc, handler_pc, catch_type);
		}

		static ExceptionHandler read(DataInput in, ConstantPool constant_pool) throws IOException {
			int start_pc = in.readUnsignedShort();
			int end_pc = in.readUnsignedShort();
			int handler_pc = in.readUnsignedShort();
			int catch_type = in.readUnsignedShort();
			String catch_type_name = (catch_type != 0) ? constant_pool.className(catch_type) : null;
			return new ExceptionHandler(start_pc, end_pc, handler_pc, catch_type_name);
		}

		void write(DataOutput out, ConstantPool constant_pool) throws IOException {
			out.writeShort(start_pc);
			out.writeShort(end_pc);
			out.writeShort(handler_pc);
			int catch_type_index = (catch_type != null) ? constant_pool.classInfo(catch_type) : 0;
			out.writeShort(catch_type_index);
		}

		int value_length() {
			return 4 * Short.BYTES;
		}
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy