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

proguard.classfile.instruction.VariableInstruction Maven / Gradle / Ivy

Go to download

ProGuardCORE is a free library to read, analyze, modify, and write Java class files.

There is a newer version: 9.1.7
Show newest version
/*
 * ProGuardCORE -- library to process Java bytecode.
 *
 * Copyright (c) 2002-2020 Guardsquare NV
 *
 * Licensed 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 proguard.classfile.instruction;

import proguard.classfile.*;
import proguard.classfile.attribute.CodeAttribute;
import proguard.classfile.instruction.visitor.InstructionVisitor;

import java.util.Objects;

/**
 * This {@link Instruction} represents an instruction that refers to a variable on the
 * local variable stack.
 *
 * @author Eric Lafortune
 */
public class VariableInstruction extends Instruction
{
    public boolean wide;
    public int     variableIndex;
    public int     constant;


    /**
     * Creates an uninitialized VariableInstruction.
     */
    public VariableInstruction() {}


    public VariableInstruction(boolean wide)
    {
        this.wide = wide;
    }


    public VariableInstruction(byte opcode)
    {
        this(opcode, embeddedVariable(opcode), 0);
    }


    public VariableInstruction(byte opcode,
                               int  variableIndex)
    {
        this(opcode, variableIndex, 0);
    }


    public VariableInstruction(byte opcode,
                               int  variableIndex,
                               int  constant)
    {
        this.opcode        = opcode;
        this.variableIndex = variableIndex;
        this.constant      = constant;
        this.wide          = requiredVariableIndexSize() > 1 ||
                             requiredConstantSize()      > 1;
    }


    /**
     * Copies the given instruction into this instruction.
     * @param variableInstruction the instruction to be copied.
     * @return this instruction.
     */
    public VariableInstruction copy(VariableInstruction variableInstruction)
    {
        this.opcode        = variableInstruction.opcode;
        this.variableIndex = variableInstruction.variableIndex;
        this.constant      = variableInstruction.constant;
        this.wide          = variableInstruction.wide;

        return this;
    }


    /**
     * Return the embedded variable of the given opcode, or 0 if the opcode
     * doesn't have one.
     */
    private static int embeddedVariable(byte opcode)
    {
        switch (opcode)
        {
            case Instruction.OP_ILOAD_1:
            case Instruction.OP_LLOAD_1:
            case Instruction.OP_FLOAD_1:
            case Instruction.OP_DLOAD_1:
            case Instruction.OP_ALOAD_1:
            case Instruction.OP_ISTORE_1:
            case Instruction.OP_LSTORE_1:
            case Instruction.OP_FSTORE_1:
            case Instruction.OP_DSTORE_1:
            case Instruction.OP_ASTORE_1: return 1;

            case Instruction.OP_ILOAD_2:
            case Instruction.OP_LLOAD_2:
            case Instruction.OP_FLOAD_2:
            case Instruction.OP_DLOAD_2:
            case Instruction.OP_ALOAD_2:
            case Instruction.OP_ISTORE_2:
            case Instruction.OP_LSTORE_2:
            case Instruction.OP_FSTORE_2:
            case Instruction.OP_DSTORE_2:
            case Instruction.OP_ASTORE_2: return 2;

            case Instruction.OP_ILOAD_3:
            case Instruction.OP_LLOAD_3:
            case Instruction.OP_FLOAD_3:
            case Instruction.OP_DLOAD_3:
            case Instruction.OP_ALOAD_3:
            case Instruction.OP_ISTORE_3:
            case Instruction.OP_LSTORE_3:
            case Instruction.OP_FSTORE_3:
            case Instruction.OP_DSTORE_3:
            case Instruction.OP_ASTORE_3: return 3;

            default: return 0;
        }
    }


    /**
     * Returns whether this instruction stores the value of a variable.
     * The value is false for the ret instruction, but true for the iinc
     * instruction.
     */
    public boolean isStore()
    {
        // A store instruction can be recognized as follows. Note that this
        // excludes the ret instruction, which has a negative opcode.
        return opcode >= Instruction.OP_ISTORE ||
               opcode == Instruction.OP_IINC;
    }


    /**
     * Returns whether this instruction loads the value of a variable.
     * The value is true for the ret instruction and for the iinc
     * instruction.
     */
    public boolean isLoad()
    {
        // A load instruction can be recognized as follows. Note that this
        // includes the ret instruction, which has a negative opcode.
        return opcode < Instruction.OP_ISTORE;
    }


    // Implementations for Instruction.

    public byte canonicalOpcode()
    {
        // Remove the _0, _1, _2, _3 extension, if any.
        switch (opcode)
        {
            case Instruction.OP_ILOAD_0:
            case Instruction.OP_ILOAD_1:
            case Instruction.OP_ILOAD_2:
            case Instruction.OP_ILOAD_3: return Instruction.OP_ILOAD;
            case Instruction.OP_LLOAD_0:
            case Instruction.OP_LLOAD_1:
            case Instruction.OP_LLOAD_2:
            case Instruction.OP_LLOAD_3: return Instruction.OP_LLOAD;
            case Instruction.OP_FLOAD_0:
            case Instruction.OP_FLOAD_1:
            case Instruction.OP_FLOAD_2:
            case Instruction.OP_FLOAD_3: return Instruction.OP_FLOAD;
            case Instruction.OP_DLOAD_0:
            case Instruction.OP_DLOAD_1:
            case Instruction.OP_DLOAD_2:
            case Instruction.OP_DLOAD_3: return Instruction.OP_DLOAD;
            case Instruction.OP_ALOAD_0:
            case Instruction.OP_ALOAD_1:
            case Instruction.OP_ALOAD_2:
            case Instruction.OP_ALOAD_3: return Instruction.OP_ALOAD;

            case Instruction.OP_ISTORE_0:
            case Instruction.OP_ISTORE_1:
            case Instruction.OP_ISTORE_2:
            case Instruction.OP_ISTORE_3: return Instruction.OP_ISTORE;
            case Instruction.OP_LSTORE_0:
            case Instruction.OP_LSTORE_1:
            case Instruction.OP_LSTORE_2:
            case Instruction.OP_LSTORE_3: return Instruction.OP_LSTORE;
            case Instruction.OP_FSTORE_0:
            case Instruction.OP_FSTORE_1:
            case Instruction.OP_FSTORE_2:
            case Instruction.OP_FSTORE_3: return Instruction.OP_FSTORE;
            case Instruction.OP_DSTORE_0:
            case Instruction.OP_DSTORE_1:
            case Instruction.OP_DSTORE_2:
            case Instruction.OP_DSTORE_3: return Instruction.OP_DSTORE;
            case Instruction.OP_ASTORE_0:
            case Instruction.OP_ASTORE_1:
            case Instruction.OP_ASTORE_2:
            case Instruction.OP_ASTORE_3: return Instruction.OP_ASTORE;

            default: return opcode;
        }
    }

    public Instruction shrink()
    {
        opcode = canonicalOpcode();

        // Is this instruction pointing to a variable with index from 0 to 3?
        if (variableIndex <= 3)
        {
            switch (opcode)
            {
                case Instruction.OP_ILOAD: opcode = (byte)(Instruction.OP_ILOAD_0 + variableIndex); break;
                case Instruction.OP_LLOAD: opcode = (byte)(Instruction.OP_LLOAD_0 + variableIndex); break;
                case Instruction.OP_FLOAD: opcode = (byte)(Instruction.OP_FLOAD_0 + variableIndex); break;
                case Instruction.OP_DLOAD: opcode = (byte)(Instruction.OP_DLOAD_0 + variableIndex); break;
                case Instruction.OP_ALOAD: opcode = (byte)(Instruction.OP_ALOAD_0 + variableIndex); break;

                case Instruction.OP_ISTORE: opcode = (byte)(Instruction.OP_ISTORE_0 + variableIndex); break;
                case Instruction.OP_LSTORE: opcode = (byte)(Instruction.OP_LSTORE_0 + variableIndex); break;
                case Instruction.OP_FSTORE: opcode = (byte)(Instruction.OP_FSTORE_0 + variableIndex); break;
                case Instruction.OP_DSTORE: opcode = (byte)(Instruction.OP_DSTORE_0 + variableIndex); break;
                case Instruction.OP_ASTORE: opcode = (byte)(Instruction.OP_ASTORE_0 + variableIndex); break;
            }
        }

        // Only make the instruction wide if necessary.
        wide = requiredVariableIndexSize() > 1 ||
               requiredConstantSize()      > 1;

        return this;
    }


    protected boolean isWide()
    {
        return wide;
    }


    protected void readInfo(byte[] code, int offset)
    {
        int variableIndexSize = variableIndexSize();
        int constantSize      = constantSize();

        // Also initialize embedded variable indexes.
        if (variableIndexSize == 0)
        {
            // An embedded variable index can be decoded as follows.
            variableIndex = opcode < Instruction.OP_ISTORE_0 ?
                (opcode - Instruction.OP_ILOAD_0 ) & 3 :
                (opcode - Instruction.OP_ISTORE_0) & 3;
        }
        else
        {
            variableIndex = readValue(code, offset, variableIndexSize); offset += variableIndexSize;
        }

        constant = readSignedValue(code, offset, constantSize);
    }


    protected void writeInfo(byte[] code, int offset)
    {
        int variableIndexSize = variableIndexSize();
        int constantSize      = constantSize();

        if (requiredVariableIndexSize() > variableIndexSize)
        {
            throw new IllegalArgumentException("Instruction has invalid variable index size ("+this.toString(offset)+")");
        }

        if (requiredConstantSize() > constantSize)
        {
            throw new IllegalArgumentException("Instruction has invalid constant size ("+this.toString(offset)+")");
        }

        writeValue(code, offset, variableIndex, variableIndexSize); offset += variableIndexSize;
        writeSignedValue(code, offset, constant, constantSize);
    }


    public int length(int offset)
    {
        return (wide ? 2 : 1) + variableIndexSize() + constantSize();
    }


    public void accept(Clazz clazz, Method method, CodeAttribute codeAttribute, int offset, InstructionVisitor instructionVisitor)
    {
        instructionVisitor.visitVariableInstruction(clazz, method, codeAttribute, offset, this);
    }


    // Implementations for Object.

    @Override
    public String toString()
    {
        return getName() +
               (wide ? "_w" : "") +
               " v"+variableIndex +
               (constantSize() > 0 ? ", "+constant : "");
    }


    @Override
    public boolean equals(Object o)
    {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        VariableInstruction that = (VariableInstruction)o;
        return opcode == that.opcode &&
               wide == that.wide &&
               variableIndex == that.variableIndex &&
               constant == that.constant;
    }


    @Override
    public int hashCode()
    {
        return Objects.hash(opcode, wide, variableIndex, constant);
    }


    // Small utility methods.

    /**
     * Returns the variable index size for this instruction.
     */
    private int variableIndexSize()
    {
        return (opcode >= Instruction.OP_ILOAD_0 &&
                opcode <= Instruction.OP_ALOAD_3) ||
               (opcode >= Instruction.OP_ISTORE_0 &&
                opcode <= Instruction.OP_ASTORE_3) ? 0 :
               wide                                         ? 2 :
                                                              1;
    }


    /**
     * Computes the required variable index size for this instruction's variable
     * index.
     */
    private int requiredVariableIndexSize()
    {
        return (variableIndex &    0x3) == variableIndex ? 0 :
               (variableIndex &   0xff) == variableIndex ? 1 :
               (variableIndex & 0xffff) == variableIndex ? 2 :
                                                           4;
    }


    /**
     * Returns the constant size for this instruction.
     */
    private int constantSize()
    {
        return opcode != Instruction.OP_IINC ? 0 :
               wide                                   ? 2 :
                                                        1;
    }


    /**
     * Computes the required constant size for this instruction's constant.
     */
    private int requiredConstantSize()
    {
        return opcode != Instruction.OP_IINC ? 0 :
               (byte)constant  == constant            ? 1 :
               (short)constant == constant            ? 2 :
                                                        4;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy