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

serp.bytecode.WideInstruction Maven / Gradle / Ivy

There is a newer version: 10.0.0-M3
Show newest version
package serp.bytecode;

import java.io.*;

import serp.bytecode.visitor.*;

/**
 * The wide instruction, which is used to allow other
 * instructions to index values beyond what they can normally index baed
 * on the length of their arguments.
 *
 * @author Abe White
 */
public class WideInstruction extends LocalVariableInstruction {
    private static final Class[][] _mappings = new Class[][] {
        { byte.class, int.class },
        { boolean.class, int.class },
        { char.class, int.class },
        { short.class, int.class },
        { void.class, int.class },
    };
    private int _ins = Constants.NOP;
    private int _inc = -1;

    WideInstruction(Code owner) {
        super(owner, Constants.WIDE);
    }

    int getLength() {
        // opcode, ins, index
        int length = super.getLength() + 1 + 2;

        // increment
        if (_ins == Constants.IINC)
            length += 2;
        return length;
    }

    public int getStackChange() {
        switch (_ins) {
        case Constants.ILOAD:
        case Constants.FLOAD:
        case Constants.ALOAD:
            return 1;
        case Constants.LLOAD:
        case Constants.DLOAD:
            return 2;
        case Constants.ISTORE:
        case Constants.FSTORE:
        case Constants.ASTORE:
            return -1;
        case Constants.LSTORE:
        case Constants.DSTORE:
            return -2;
        default:
            return 0;
        }
    }

    public int getLogicalStackChange() {
        switch (_ins) {
        case Constants.ILOAD:
        case Constants.FLOAD:
        case Constants.ALOAD:
        case Constants.LLOAD:
        case Constants.DLOAD:
            return 1;
        case Constants.ISTORE:
        case Constants.FSTORE:
        case Constants.ASTORE:
        case Constants.LSTORE:
        case Constants.DSTORE:
            return -1;
        default:
            return 0;
        }
    }

    public String getTypeName() {
        switch (_ins) {
        case Constants.ILOAD:
        case Constants.ISTORE:
            return int.class.getName();
        case Constants.LLOAD:
        case Constants.LSTORE:
            return long.class.getName();
        case Constants.FLOAD:
        case Constants.FSTORE:
            return float.class.getName();
        case Constants.DLOAD:
        case Constants.DSTORE:
            return double.class.getName();
        case Constants.ALOAD:
        case Constants.ASTORE:
            return Object.class.getName();
        default:
            return null;
        }
    }

    public TypedInstruction setType(String type) {
        type = mapType(type, _mappings, true);
        switch (_ins) {
        case Constants.ILOAD:
        case Constants.LLOAD:
        case Constants.FLOAD:
        case Constants.DLOAD:
        case Constants.ALOAD:
            if (type == null)
                throw new IllegalStateException();
            switch (type.charAt(0)) {
            case 'i':
                return (TypedInstruction) setInstruction(Constants.ILOAD);
            case 'l':
                return (TypedInstruction) setInstruction(Constants.LLOAD);
            case 'f':
                return (TypedInstruction) setInstruction(Constants.FLOAD);
            case 'd':
                return (TypedInstruction) setInstruction(Constants.DLOAD);
            default:
                return (TypedInstruction) setInstruction(Constants.ALOAD);
            }
        case Constants.ISTORE:
        case Constants.LSTORE:
        case Constants.FSTORE:
        case Constants.DSTORE:
        case Constants.ASTORE:
            if (type == null)
                throw new IllegalStateException();
            switch (type.charAt(0)) {
            case 'i':
                return (TypedInstruction) setInstruction(Constants.ISTORE);
            case 'l':
                return (TypedInstruction) setInstruction(Constants.LSTORE);
            case 'f':
                return (TypedInstruction) setInstruction(Constants.FSTORE);
            case 'd':
                return (TypedInstruction) setInstruction(Constants.DSTORE);
            default:
                return (TypedInstruction) setInstruction(Constants.ASTORE);
            }
        default:
            if (type != null)
                throw new IllegalStateException("Augmented instruction not " 
                    + "typed");
            return this;
        }
    }

    /**
     * Return the opcode of the instruction to modify; this will return one
     * of the constants defined in {@link Constants}.
     */
    public int getInstruction() {
        return _ins;
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     */
    public WideInstruction setInstruction(Instruction ins) {
        if (ins == null)
            return setInstruction(Constants.NOP);
        setInstruction(ins.getOpcode());
        if (_ins == Constants.IINC)
            _inc = ((IIncInstruction) ins).getIncrement();
        return this;
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     */
    public WideInstruction setInstruction(int opcode) {
        int len = getLength();
        _ins = opcode;
        if (len != getLength())
            invalidateByteIndexes();
        return this;
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     *
     * @return this instruction, for method chaining
     */
    public WideInstruction iinc() {
        return setInstruction(Constants.IINC);
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     *
     * @return this instruction, for method chaining
     */
    public WideInstruction ret() {
        return setInstruction(Constants.RET);
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     *
     * @return this instruction, for method chaining
     */
    public WideInstruction iload() {
        return setInstruction(Constants.ILOAD);
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     *
     * @return this instruction, for method chaining
     */
    public WideInstruction fload() {
        return setInstruction(Constants.FLOAD);
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     *
     * @return this instruction, for method chaining
     */
    public WideInstruction aload() {
        return setInstruction(Constants.ALOAD);
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     *
     * @return this instruction, for method chaining
     */
    public WideInstruction lload() {
        return setInstruction(Constants.LLOAD);
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     *
     * @return this instruction, for method chaining
     */
    public WideInstruction dload() {
        return setInstruction(Constants.DLOAD);
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     *
     * @return this instruction, for method chaining
     */
    public WideInstruction istore() {
        return setInstruction(Constants.ISTORE);
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     *
     * @return this instruction, for method chaining
     */
    public WideInstruction fstore() {
        return setInstruction(Constants.FSTORE);
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     *
     * @return this instruction, for method chaining
     */
    public WideInstruction astore() {
        return setInstruction(Constants.ASTORE);
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     *
     * @return this instruction, for method chaining
     */
    public WideInstruction lstore() {
        return setInstruction(Constants.LSTORE);
    }

    /**
     * Set the type of instruction this wide instruction modifies.
     *
     * @return this instruction, for method chaining
     */
    public WideInstruction dstore() {
        return setInstruction(Constants.DSTORE);
    }

    /**
     * Return the increment for this instruction if it augments IINC, or -1
     * if unset.
     */
    public int getIncrement() {
        return _inc;
    }

    /**
     * Set the increment on this instruction if it augments IINC.
     *
     * @return this Instruction, for method chaining
     */
    public WideInstruction setIncrement(int val) {
        _inc = val;
        return this;
    }

    /**
     * WideInstructions are equal if the instruction they augment is the same
     * or unset.
     */
    public boolean equalsInstruction(Instruction other) {
        if (other == this)
            return true;
        if (!super.equalsInstruction(other))
            return false;
        if (!(other instanceof WideInstruction))
            return false;

        WideInstruction ins = (WideInstruction) other;
        int code = getInstruction();
        int otherCode = ins.getInstruction();
        if (code != otherCode)
            return false;
        if (code == Constants.IINC) {
            int inc = getIncrement();
            int otherInc = ins.getIncrement();
            return (inc == -1) || (otherInc == -1) || (inc == otherInc);
        }
        return true;
    }

    public void acceptVisit(BCVisitor visit) {
        visit.enterWideInstruction(this);
        visit.exitWideInstruction(this);
    }

    void read(Instruction orig) {
        super.read(orig);
        setInstruction(((WideInstruction) orig).getInstruction());
    }

    void read(DataInput in) throws IOException {
        super.read(in);
        _ins = in.readUnsignedByte();
        setLocal(in.readUnsignedShort());
        if (_ins == Constants.IINC)
            _inc = in.readUnsignedShort();
    }

    void write(DataOutput out) throws IOException {
        super.write(out);
        out.writeByte(_ins);
        out.writeShort(getLocal());
        if (_ins == Constants.IINC)
            out.writeShort(_inc);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy