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

serp.bytecode.CmpInstruction Maven / Gradle / Ivy

package serp.bytecode;

import serp.bytecode.visitor.*;

/**
 * An instruction comparing two stack values. Examples include
 * lcmp, fcmpl, etc.
 *
 * @author Abe White
 */
public class CmpInstruction extends TypedInstruction {
    private static Class[][] _mappings = new Class[][] {
        { int.class, long.class },
        { byte.class, long.class },
        { char.class, long.class },
        { short.class, long.class },
        { boolean.class, long.class },
        { void.class, long.class },
        { Object.class, long.class },
    };

    CmpInstruction(Code owner) {
        super(owner);
    }

    CmpInstruction(Code owner, int opcode) {
        super(owner, opcode);
    }

    public int getLogicalStackChange() {
        switch (getOpcode()) {
        case Constants.NOP:
            return 0;
        default:
            return -1;
        }
    }

    public int getStackChange() {
        switch (getOpcode()) {
        case Constants.LCMP:
        case Constants.DCMPL:
        case Constants.DCMPG:
            return -3;
        case Constants.NOP:
            return 0;
        default:
            return -1;
        }
    }

    public String getTypeName() {
        switch (getOpcode()) {
        case Constants.LCMP:
            return long.class.getName();
        case Constants.FCMPL:
        case Constants.FCMPG:
            return float.class.getName();
        case Constants.DCMPL:
        case Constants.DCMPG:
            return double.class.getName();
        default:
            return null;
        }
    }

    public TypedInstruction setType(String type) {
        type = mapType(type, _mappings, true);
        if (type == null)
            return (TypedInstruction) setOpcode(Constants.NOP);

        int opcode = getOpcode();
        switch (type.charAt(0)) {
        case 'l':
            return (TypedInstruction) setOpcode(Constants.LCMP);
        case 'f':
            if ((opcode == Constants.FCMPL) || (opcode == Constants.DCMPL))
                return (TypedInstruction) setOpcode(Constants.FCMPL);
            return (TypedInstruction) setOpcode(Constants.FCMPG);
        case 'd':
            if ((opcode == Constants.FCMPL) || (opcode == Constants.DCMPL))
                return (TypedInstruction) setOpcode(Constants.DCMPL);
            return (TypedInstruction) setOpcode(Constants.DCMPG);
        default:
            throw new IllegalStateException();
        }
    }

    /**
     * Return the number that will be placed on the stack if this instruction
     * is of type float or double and one of the operands is NaN. For
     * FCMPG or DCMPG, this value will be 1; for FCMPL or DCMPL this value
     * will be -1. For LCMP or if the type is unset, this value will be 0.
     */
    public int getNaNValue() {
        switch (getOpcode()) {
        case Constants.FCMPL:
        case Constants.DCMPL:
            return -1;
        case Constants.FCMPG:
        case Constants.DCMPG:
            return 1;
        default:
            return 0;
        }
    }

    /**
     * Set the number that will be placed on the stack if this instruction
     * is of type float or double and one of the operands is NaN. For
     * FCMPG or DCMPG, this value should be 1; for FCMPL or DCMPL this value
     * should be -1. For LCMP, this value should be 0.
     *
     * @return this instruction, for method chaining
     */
    public CmpInstruction setNaNValue(int nan) {
        switch (getOpcode()) {
        case Constants.FCMPL:
        case Constants.FCMPG:
            if (nan == 1)
                setOpcode(Constants.FCMPG);
            else if (nan == -1)
                setOpcode(Constants.FCMPL);
            else
                throw new IllegalArgumentException("Invalid nan for type");
        case Constants.DCMPL:
        case Constants.DCMPG:
            if (nan == 1)
                setOpcode(Constants.DCMPG);
            else if (nan == -1)
                setOpcode(Constants.DCMPL);
            else
                throw new IllegalArgumentException("Invalid nan for type");
        default:
            if (nan != 0)
                throw new IllegalArgumentException("Invalid nan for type");
        }
        return this;
    }

    public void acceptVisit(BCVisitor visit) {
        visit.enterCmpInstruction(this);
        visit.exitCmpInstruction(this);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy