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

com.strobel.assembler.metadata.VariableDefinitionCollection Maven / Gradle / Ivy

There is a newer version: 2.5.0.Final
Show newest version
/*
 * VariableDefinitionCollection.java
 *
 * Copyright (c) 2013 Mike Strobel
 *
 * This source code is based on Mono.Cecil from Jb Evain, Copyright (c) Jb Evain;
 * and ILSpy/ICSharpCode from SharpDevelop, Copyright (c) AlphaSierraPapa.
 *
 * This source code is subject to terms and conditions of the Apache License, Version 2.0.
 * A copy of the license can be found in the License.html file at the root of this distribution.
 * By using this source code in any fashion, you are agreeing to be bound by the terms of the
 * Apache License, Version 2.0.
 *
 * You must not remove this notice, or any other, from this software.
 */

package com.strobel.assembler.metadata;

import com.strobel.assembler.Collection;
import com.strobel.assembler.ir.OpCode;
import com.strobel.core.VerifyArgument;
import com.strobel.util.ContractUtils;

import java.util.NoSuchElementException;

public final class VariableDefinitionCollection extends Collection {
    private final MethodDefinition _declaringMethod;

    public VariableDefinitionCollection(final MethodDefinition declaringMethod) {
        _declaringMethod = VerifyArgument.notNull(declaringMethod, "declaringMethod");
    }

    public int slotCount() {
        int count = 0;

        for (int i = 0; i < size(); i++) {
            final VariableDefinition v = get(i);
            count = Math.max(v.getSlot() + v.getSize(), count);
        }

        return count;
    }

    public VariableDefinition tryFind(final int slot) {
        return find(slot, -1);
    }

    public VariableDefinition tryFind(final int slot, final int instructionOffset) {
        VariableDefinition result = null;

        for (int i = 0; i < size(); i++) {
            final VariableDefinition variable = get(i);

            if (variable.getSlot() == slot &&
                (instructionOffset < 0 ||
                 variable.getScopeStart() >= 0 &&
                 variable.getScopeStart() <= instructionOffset &&
                 (variable.getScopeEnd() < 0 || variable.getScopeEnd() >= instructionOffset)) &&
                (result == null || variable.getScopeStart() > result.getScopeStart())) {

                result = variable;
            }
        }

        return result;
    }

    public VariableDefinition find(final int slot) {
        return find(slot, -1);
    }

    public VariableDefinition find(final int slot, final int instructionOffset) {
        final VariableDefinition variable = tryFind(slot, instructionOffset);

        if (variable != null) {
            return variable;
        }

        throw new NoSuchElementException(
            String.format(
                "Could not find variable at slot %d and offset %d.",
                slot,
                instructionOffset
            )
        );
    }

    public VariableReference tryFind(final int slot, final OpCode op, final int instructionOffset) {
        final int effectiveOffset;

        if (op.isStore()) {
            effectiveOffset = instructionOffset + op.getSize() + op.getOperandType().getBaseSize();
        }
        else {
            effectiveOffset = instructionOffset;
        }

        return tryFind(slot, effectiveOffset);
    }

    public VariableReference reference(final int slot, final OpCode op, final int instructionOffset) {
        final VariableReference variable = tryFind(slot, op, instructionOffset);

        if (variable != null) {
            return variable;
        }

        final TypeReference variableType;

        switch (op) {
            case ISTORE:
            case ISTORE_0:
            case ISTORE_1:
            case ISTORE_2:
            case ISTORE_3:
            case ISTORE_W:
            case ILOAD:
            case ILOAD_0:
            case ILOAD_1:
            case ILOAD_2:
            case ILOAD_3:
            case ILOAD_W:
                variableType = BuiltinTypes.Integer;
                break;

            case LSTORE:
            case LSTORE_0:
            case LSTORE_1:
            case LSTORE_2:
            case LSTORE_3:
            case LSTORE_W:
            case LLOAD:
            case LLOAD_0:
            case LLOAD_1:
            case LLOAD_2:
            case LLOAD_3:
            case LLOAD_W:
                variableType = BuiltinTypes.Long;
                break;

            case FSTORE:
            case FSTORE_0:
            case FSTORE_1:
            case FSTORE_2:
            case FSTORE_3:
            case FSTORE_W:
            case FLOAD:
            case FLOAD_0:
            case FLOAD_1:
            case FLOAD_2:
            case FLOAD_3:
            case FLOAD_W:
                variableType = BuiltinTypes.Float;
                break;

            case DSTORE:
            case DSTORE_0:
            case DSTORE_1:
            case DSTORE_2:
            case DSTORE_3:
            case DSTORE_W:
            case DLOAD:
            case DLOAD_0:
            case DLOAD_1:
            case DLOAD_2:
            case DLOAD_3:
            case DLOAD_W:
                variableType = BuiltinTypes.Double;
                break;

            case IINC:
            case IINC_W:
                variableType = BuiltinTypes.Integer;
                break;

            default:
                variableType = BuiltinTypes.Object;
                break;
        }

        return makeReference(slot, variableType);
    }

    public VariableReference makeReference(final int slot, final TypeReference variableType) {
        return new UnknownVariableReference(
            variableType,
            slot,
            _declaringMethod.getDeclaringType()
        );
    }

    // 

    private final static class UnknownVariableReference extends VariableReference {
        private final int _slot;
        private final TypeReference _declaringType;

        UnknownVariableReference(final TypeReference variableType, final int slot, final TypeReference declaringType) {
            super(variableType);

            _slot = slot;
            _declaringType = VerifyArgument.notNull(declaringType, "declaringType");
        }

        @Override
        public final TypeReference getDeclaringType() {
            return _declaringType;
        }

        @Override
        public final int getSlot() {
            return _slot;
        }

        @Override
        public final VariableDefinition resolve() {
            throw ContractUtils.unsupported();
        }
    }

    // 
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy