
org.apache.commons.compress.harmony.unpack200.bytecode.CodeAttribute Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.commons.compress.harmony.unpack200.bytecode;
import java.io.DataOutputStream;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import org.apache.commons.compress.harmony.unpack200.Segment;
public class CodeAttribute extends BCIRenumberedAttribute {
public List attributes = new ArrayList();
// instances
public List byteCodeOffsets = new ArrayList();
public List byteCodes = new ArrayList();
public int codeLength;
public List exceptionTable; // of ExceptionTableEntry
public int maxLocals;
public int maxStack;
private static CPUTF8 attributeName;
public CodeAttribute(final int maxStack, final int maxLocals, final byte codePacked[], final Segment segment,
final OperandManager operandManager, final List exceptionTable) {
super(attributeName);
this.maxLocals = maxLocals;
this.maxStack = maxStack;
this.codeLength = 0;
this.exceptionTable = exceptionTable;
byteCodeOffsets.add(Integer.valueOf(0));
int byteCodeIndex = 0;
for (int i = 0; i < codePacked.length; i++) {
final ByteCode byteCode = ByteCode.getByteCode(codePacked[i] & 0xff);
// Setting the offset must happen before extracting operands
// because label bytecodes need to know their offsets.
byteCode.setByteCodeIndex(byteCodeIndex);
byteCodeIndex++;
byteCode.extractOperands(operandManager, segment, codeLength);
byteCodes.add(byteCode);
codeLength += byteCode.getLength();
final int lastBytecodePosition = ((Integer) byteCodeOffsets.get(byteCodeOffsets.size() - 1)).intValue();
// This code assumes all multiple byte bytecodes are
// replaced by a single-byte bytecode followed by
// another bytecode.
if (byteCode.hasMultipleByteCodes()) {
byteCodeOffsets.add(Integer.valueOf(lastBytecodePosition + 1));
byteCodeIndex++;
}
// I've already added the first element (at 0) before
// entering this loop, so make sure I don't add one
// after the last element.
if (i < (codePacked.length - 1)) {
byteCodeOffsets.add(Integer.valueOf(lastBytecodePosition + byteCode.getLength()));
}
if (byteCode.getOpcode() == 0xC4) {
// Special processing for wide bytecode - it knows what its
// instruction is from the opcode manager, so ignore the
// next instruction
i++;
}
}
// Now that all the bytecodes know their positions and
// sizes, fix up the byte code targets
// At this point, byteCodes may be a different size than
// codePacked because of wide bytecodes.
for (int i = 0; i < byteCodes.size(); i++) {
final ByteCode byteCode = (ByteCode) byteCodes.get(i);
byteCode.applyByteCodeTargetFixup(this);
}
}
@Override
protected int getLength() {
int attributesSize = 0;
for (int it = 0; it < attributes.size(); it++) {
final Attribute attribute = (Attribute) attributes.get(it);
attributesSize += attribute.getLengthIncludingHeader();
}
return 2 + 2 + 4 + codeLength + 2 + exceptionTable.size() * (2 + 2 + 2 + 2) + 2 + attributesSize;
}
@Override
protected ClassFileEntry[] getNestedClassFileEntries() {
final ArrayList nestedEntries = new ArrayList(attributes.size() + byteCodes.size() + 10);
nestedEntries.add(getAttributeName());
nestedEntries.addAll(byteCodes);
nestedEntries.addAll(attributes);
// Don't forget to add the ExceptionTable catch_types
for (int iter = 0; iter < exceptionTable.size(); iter++) {
final ExceptionTableEntry entry = (ExceptionTableEntry) exceptionTable.get(iter);
final CPClass catchType = entry.getCatchType();
// If the catch type is null, this is a finally
// block. If it's not null, we need to add the
// CPClass to the list of nested class file entries.
if (catchType != null) {
nestedEntries.add(catchType);
}
}
final ClassFileEntry[] nestedEntryArray = new ClassFileEntry[nestedEntries.size()];
nestedEntries.toArray(nestedEntryArray);
return nestedEntryArray;
}
@Override
protected void resolve(final ClassConstantPool pool) {
super.resolve(pool);
for (int it = 0; it < attributes.size(); it++) {
final Attribute attribute = (Attribute) attributes.get(it);
attribute.resolve(pool);
}
for (int it = 0; it < byteCodes.size(); it++) {
final ByteCode byteCode = (ByteCode) byteCodes.get(it);
byteCode.resolve(pool);
}
for (int it = 0; it < exceptionTable.size(); it++) {
final ExceptionTableEntry entry = (ExceptionTableEntry) exceptionTable.get(it);
entry.resolve(pool);
}
}
@Override
public String toString() {
return "Code: " + getLength() + " bytes";
}
@Override
protected void writeBody(final DataOutputStream dos) throws IOException {
dos.writeShort(maxStack);
dos.writeShort(maxLocals);
dos.writeInt(codeLength);
for (int it = 0; it < byteCodes.size(); it++) {
final ByteCode byteCode = (ByteCode) byteCodes.get(it);
byteCode.write(dos);
}
dos.writeShort(exceptionTable.size());
for (int it = 0; it < exceptionTable.size(); it++) {
final ExceptionTableEntry entry = (ExceptionTableEntry) exceptionTable.get(it);
entry.write(dos);
}
dos.writeShort(attributes.size());
for (int it = 0; it < attributes.size(); it++) {
final Attribute attribute = (Attribute) attributes.get(it);
attribute.write(dos);
}
}
public void addAttribute(final Attribute attribute) {
attributes.add(attribute);
if (attribute instanceof LocalVariableTableAttribute) {
((LocalVariableTableAttribute) attribute).setCodeLength(codeLength);
}
if (attribute instanceof LocalVariableTypeTableAttribute) {
((LocalVariableTypeTableAttribute) attribute).setCodeLength(codeLength);
}
}
@Override
protected int[] getStartPCs() {
// Do nothing here as we've overriden renumber
return null;
}
@Override
public void renumber(final List byteCodeOffsets) {
for (int iter = 0; iter < exceptionTable.size(); iter++) {
final ExceptionTableEntry entry = (ExceptionTableEntry) exceptionTable.get(iter);
entry.renumber(byteCodeOffsets);
}
}
public static void setAttributeName(final CPUTF8 attributeName) {
CodeAttribute.attributeName = attributeName;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy