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

edu.umd.cs.findbugs.StackMapAnalyzer Maven / Gradle / Ivy

There is a newer version: 4.8.6
Show newest version
/*
 * FindBugs - Find Bugs in Java programs
 * Copyright (C) 2003-2008 University of Maryland
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */

package edu.umd.cs.findbugs;

import java.util.ArrayList;
import java.util.BitSet;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import javax.annotation.CheckForNull;

import org.apache.bcel.Const;
import org.apache.bcel.classfile.Attribute;
import org.apache.bcel.classfile.ClassFormatException;
import org.apache.bcel.classfile.Code;
import org.apache.bcel.classfile.ConstantClass;
import org.apache.bcel.classfile.Method;
import org.apache.bcel.classfile.StackMap;
import org.apache.bcel.classfile.StackMapEntry;
import org.apache.bcel.classfile.StackMapType;
import org.apache.bcel.generic.Type;

import edu.umd.cs.findbugs.OpcodeStack.Item;
import edu.umd.cs.findbugs.OpcodeStack.JumpInfo;
import edu.umd.cs.findbugs.classfile.CheckedAnalysisException;
import edu.umd.cs.findbugs.classfile.IAnalysisCache;
import edu.umd.cs.findbugs.classfile.MethodDescriptor;

/**
 * @author pugh
 */
public class StackMapAnalyzer {



    public static class StackMapAnalysisFactory extends edu.umd.cs.findbugs.classfile.engine.bcel.AnalysisFactory {
        public StackMapAnalysisFactory() {
            super("Jump info for opcode stack from stack map analysis", JumpInfoFromStackMap.class);
        }

        @Override
        public JumpInfoFromStackMap analyze(IAnalysisCache analysisCache, MethodDescriptor descriptor) {

            return getFromStackMap(analysisCache, descriptor);


        }
    }

    static class JumpInfoFromStackMap extends JumpInfo {


        JumpInfoFromStackMap(Map> jumpEntries, Map> jumpStackEntries, BitSet jumpEntryLocations) {
            super(jumpEntries, jumpStackEntries, jumpEntryLocations);
        }

    }

    static final boolean DEBUG = false;

    enum StackFrameType {
        SAME_FRAME, SAME_LOCALS_1_STACK_ITEM_FRAME, CHOP_FRAME, APPEND_FRAME, FULL_FRAME;

        static StackFrameType get(int frame_type) {
            if (frame_type >= Const.SAME_FRAME && frame_type <= Const.SAME_FRAME_MAX) {
                return SAME_FRAME;
            } else if (frame_type >= Const.SAME_LOCALS_1_STACK_ITEM_FRAME
                    && frame_type <= Const.SAME_LOCALS_1_STACK_ITEM_FRAME_MAX) {
                return SAME_LOCALS_1_STACK_ITEM_FRAME;
            } else if (frame_type == Const.SAME_LOCALS_1_STACK_ITEM_FRAME_EXTENDED) {
                return SAME_LOCALS_1_STACK_ITEM_FRAME;
            } else if (frame_type >= Const.CHOP_FRAME && frame_type <= Const.CHOP_FRAME_MAX) {
                return CHOP_FRAME;
            } else if (frame_type == Const.SAME_FRAME_EXTENDED) {
                return SAME_FRAME;
            } else if (frame_type >= Const.APPEND_FRAME && frame_type <= Const.APPEND_FRAME_MAX) {
                return APPEND_FRAME;
            } else if (frame_type == Const.FULL_FRAME) {
                return FULL_FRAME;
            } else {
                /* Can't happen */
                throw new ClassFormatException("Invalid frame type : " + frame_type);
            }
        }
    }

    static @CheckForNull StackMap getStackMapTable(Code code) {
        for (Attribute a : code.getAttributes()) {
            if (a instanceof StackMap) {
                return (StackMap) a;
            }
        }
        return null;
    }

    static List getInitialLocals(MethodDescriptor descriptor) {
        List locals = new ArrayList<>();
        Type[] argTypes = Type.getArgumentTypes(descriptor.getSignature());
        int reg = 0;
        if (!descriptor.isStatic()) {
            Item it = Item.typeOnly("L" + descriptor.getSlashedClassName() + ";");
            locals.add(it);
            reg += it.getSize();
        }
        for (Type argType : argTypes) {
            Item it = Item.typeOnly(argType.getSignature());
            locals.add(it);
            reg += it.getSize();
            if (it.usesTwoSlots()) {
                locals.add(null);
            }
        }
        return locals;
    }

    static private @CheckForNull JumpInfoFromStackMap getFromStackMap(IAnalysisCache analysisCache, MethodDescriptor descriptor) {
        Method method;
        try {
            method = analysisCache.getMethodAnalysis(Method.class, descriptor);
        } catch (CheckedAnalysisException e1) {
            analysisCache.getErrorLogger().logError("Unable to get method for " + descriptor, e1);
            return null;
        }

        Code code = method.getCode();
        if (code == null) {
            return null;
        }
        StackMap stackMapTable = getStackMapTable(code);
        if (stackMapTable == null) {
            return null;
        }
        Map> jumpEntries = new HashMap<>();

        Map> jumpStackEntries = new HashMap<>();

        List locals = getInitialLocals(descriptor);
        List stack = new ArrayList<>();
        BitSet jumpEntryLocations = new BitSet();
        if (DEBUG) {
            System.out.println(descriptor);
            System.out.println(locals);
        }
        int pc = 0;
        for (StackMapEntry e : stackMapTable.getStackMap()) {
            pc += e.getByteCodeOffset();
            int rawFrameType = e.getFrameType();
            StackFrameType stackFrameType = StackFrameType.get(rawFrameType);
            switch (stackFrameType) {
            case SAME_FRAME:
                stack.clear();
                break;
            case SAME_LOCALS_1_STACK_ITEM_FRAME:
                stack.clear();
                addStack(stack, e.getTypesOfStackItems());
                break;
            case CHOP_FRAME:
                stack.clear();
                int n = Const.CHOP_FRAME_MAX + 1 - rawFrameType;
                for (int i = 0; i < n; i++) {
                    Item it = locals.remove(locals.size() - 1);
                    if (it == null) {
                        it = locals.remove(locals.size() - 1);
                        assert it.usesTwoSlots();
                    }
                }
                break;

            case APPEND_FRAME:

                stack.clear();
                addLocals(locals, e.getTypesOfLocals());

                break;
            case FULL_FRAME:
                stack.clear();
                locals.clear();
                addLocals(locals, e.getTypesOfLocals());
                addStack(stack, e.getTypesOfStackItems());
                break;

            }
            if (DEBUG) {
                System.out.printf("%4d %2d %2d  %12s %s%n",

                        pc, e.getNumberOfLocals(), e.getNumberOfStackItems(), stackFrameType, e);
                System.out.printf("     %s :: %s%n", stack, locals);
            }
            if (pc > 0) {
                jumpEntries.put(pc, new ArrayList<>(locals));
                if (!stack.isEmpty()) {
                    jumpStackEntries.put(pc, new ArrayList<>(stack));
                }
                jumpEntryLocations.set(pc);
            }
            pc++;
        }
        if (DEBUG) {
            System.out.println("\n");
        }
        return new JumpInfoFromStackMap(jumpEntries, jumpStackEntries, jumpEntryLocations);

    }

    static private Item getItem(StackMapType t) {

        switch (t.getType()) {

        case Const.ITEM_Double:
            return Item.typeOnly("D");
        case Const.ITEM_Float:
            return Item.typeOnly("F");
        case Const.ITEM_Integer:
            return Item.typeOnly("I");
        case Const.ITEM_Long:
            return Item.typeOnly("J");
        case Const.ITEM_Bogus:
        case Const.ITEM_NewObject:
            return Item.typeOnly("Ljava/lang/Object;");
        case Const.ITEM_Null:
            Item it = new Item();
            it.setSpecialKind(Item.TYPE_ONLY);
            return it;
        case Const.ITEM_InitObject:
            return Item.typeOnly("Ljava/lang/Object;");
        case Const.ITEM_Object:
            int index = t.getIndex();
            ConstantClass c = (ConstantClass) t.getConstantPool().getConstant(index);
            String name = c.getBytes(t.getConstantPool());
            if (name.charAt(0) != '[') {
                name = "L" + name + ";";
            }
            return Item.typeOnly(name);
        default:
            throw new IllegalArgumentException("Bad item type: " + t.getType());

        }
    }

    static private void addLocals(List lst, StackMapType[] typesOfStackItems) {
        for (StackMapType t : typesOfStackItems) {
            Item item = getItem(t);
            lst.add(item);
            if (item.usesTwoSlots()) {
                lst.add(null);
            }
        }

    }

    static private void addStack(List lst, StackMapType[] typesOfStackItems) {
        for (StackMapType t : typesOfStackItems) {
            Item item = getItem(t);
            lst.add(item);
        }

    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy