package task6.jvm.attr; import task6.jvm.clz.ClassFile; import task6.jvm.constant.ConstantPool; import task6.jvm.loader.ByteCodeIterator; public class CodeAttr extends AttributeInfo { private int maxStack ; private int maxLocals ; private int codeLen ; private String code; public String getCode() { return code; } //private ByteCodeCommand[] cmds ; //public ByteCodeCommand[] getCmds() { // return cmds; //} private LineNumberTable lineNumTable; private LocalVariableTable localVarTable; private StackMapTable stackMapTable; public CodeAttr(int attrNameIndex, int attrLen, int maxStack, int maxLocals, int codeLen,String code /*ByteCodeCommand[] cmds*/) { super(attrNameIndex, attrLen); this.maxStack = maxStack; this.maxLocals = maxLocals; this.codeLen = codeLen; this.code = code; //this.cmds = cmds; } public void setLineNumberTable(LineNumberTable t) { this.lineNumTable = t; } public void setLocalVariableTable(LocalVariableTable t) { this.localVarTable = t; } public static CodeAttr parse(ClassFile clzFile, ByteCodeIterator iter){ int attrNameIndex = iter.next2Bytes(); int attrLen = iter.next4Bytes(); int maxStack = iter.next2Bytes(); int maxLocals = iter.next2Bytes(); int codeLen = iter.next4Bytes(); String code = iter.nextUxToHexString(codeLen); System.out.println(code); //ByteCodeCommand[] cmds = ByteCodeCommand.parse(clzFile,code); CodeAttr codeAttr = new CodeAttr(attrNameIndex,attrLen, maxStack,maxLocals,codeLen,code); int exceptionTableLen = iter.next2Bytes(); //TODO 处理exception if(exceptionTableLen>0){ String exTable = iter.nextUxToHexString(exceptionTableLen); System.out.println("Encountered exception table , just ignore it :" + exTable); } int subAttrCount = iter.next2Bytes(); for(int x=1; x<=subAttrCount; x++){ int subAttrIndex = iter.next2Bytes(); String subAttrName = clzFile.getConstantPool().getUTF8String(subAttrIndex); //已经向前移动了U2, 现在退回去。 iter.back(2); //line item table if(AttributeInfo.LINE_NUM_TABLE.equalsIgnoreCase(subAttrName)){ LineNumberTable t = LineNumberTable.parse(iter); codeAttr.setLineNumberTable(t); } else if(AttributeInfo.LOCAL_VAR_TABLE.equalsIgnoreCase(subAttrName)){ LocalVariableTable t = LocalVariableTable.parse(iter); codeAttr.setLocalVariableTable(t); } else if (AttributeInfo.STACK_MAP_TABLE.equalsIgnoreCase(subAttrName)){ StackMapTable t = StackMapTable.parse(iter); codeAttr.setStackMapTable(t); } else{ throw new RuntimeException("Need code to process " + subAttrName); } } return codeAttr; } public String toString(ConstantPool pool){ StringBuilder buffer = new StringBuilder(); buffer.append("Code:").append(code).append("\n"); /*for(int i=0;i<cmds.length;i++){ buffer.append(cmds[i].toString(pool)).append("\n"); }*/ buffer.append("\n"); buffer.append(this.lineNumTable.toString()); buffer.append(this.localVarTable.toString(pool)); return buffer.toString(); } private void setStackMapTable(StackMapTable t) { this.stackMapTable = t; } }