/* BSD 2-Clause License - see OPAL/LICENSE for details. */ package org.opalj package bi package reader import scala.reflect.ClassTag import java.io.DataInputStream import scala.collection.immutable.ArraySeq import org.opalj.control.fillArraySeq /** * Defines a template method to read in the code attribute. * * '''From the Specification''' * The Code attribute is a variable-length attribute in the attributes table * of a method_info structure. */ trait Code_attributeReader extends AttributeReader { // // TYPE DEFINITIONS AND FACTORY METHODS // type ExceptionTableEntry <: AnyRef: ClassTag type ExceptionHandlers = ArraySeq[ExceptionTableEntry] type Instructions type Code_attribute <: Attribute def Instructions( cp: Constant_Pool, ap_name_index: Constant_Pool_Index, ap_descriptor_index: Constant_Pool_Index, in: DataInputStream ): Instructions protected def Attributes( cp: Constant_Pool, ap: AttributeParent, ap_name_index: Constant_Pool_Index, ap_descriptor_index: Constant_Pool_Index, in: DataInputStream ): Attributes def Code_attribute( cp: Constant_Pool, ap_name_index: Constant_Pool_Index, ap_descriptor_index: Constant_Pool_Index, attribute_name_index: Constant_Pool_Index, max_stack: Int, max_locals: Int, instructions: Instructions, exception_handlers: ExceptionHandlers, attributes: Attributes ): Code_attribute def ExceptionTableEntry( constant_pool: Constant_Pool, start_pc: Int, end_pc: Int, handler_pc: Int, catch_type: Constant_Pool_Index // may be "0" in case of "finally" ): ExceptionTableEntry // // IMPLEMENTATION // /** * '''From the Specification''' *
* Code_attribute {
* u2 attribute_name_index; u4 attribute_length;
* u2 max_stack;
* u2 max_locals;
* u4 code_length;
* u1 code[code_length];
* u2 exception_table_length;
* { u2 start_pc;
* u2 end_pc;
* u2 handler_pc;
* u2 catch_type;
* } exception_table[exception_table_length];
* u2 attributes_count;
* attribute_info attributes[attributes_count];
* }
*
*/
private def parserFactory() = (
cp: Constant_Pool,
ap: AttributeParent,
ap_name_index: Constant_Pool_Index,
ap_descriptor_index: Constant_Pool_Index,
attribute_name_index: Constant_Pool_Index,
in: DataInputStream
) => {
/*val attribute_length = */
in.readInt()
Code_attribute(
cp,
ap_name_index,
ap_descriptor_index,
attribute_name_index,
in.readUnsignedShort(),
in.readUnsignedShort(),
Instructions(cp, ap_name_index, ap_descriptor_index, in),
fillArraySeq(in.readUnsignedShort()) { // "exception_table_length" times
ExceptionTableEntry(
cp,
in.readUnsignedShort,
in.readUnsignedShort,
in.readUnsignedShort,
in.readUnsignedShort
)
},
Attributes(cp, AttributesParent.Code, ap_name_index, ap_descriptor_index, in)
)
}
registerAttributeReader(CodeAttribute.Name -> parserFactory())
}