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

src.main.java.com.mebigfatguy.fbcontrib.detect.NeedlessAutoboxing Maven / Gradle / Ivy

Go to download

An auxiliary findbugs.sourceforge.net plugin for java bug detectors that fall outside the narrow scope of detectors to be packaged with the product itself.

There is a newer version: 7.6.8
Show newest version
/*
 * fb-contrib - Auxiliary detectors for Java programs
 * Copyright (C) 2005-2018 Dave Brosius
 *
 * 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 com.mebigfatguy.fbcontrib.detect;

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

import org.apache.bcel.Constants;
import org.apache.bcel.classfile.Method;

import com.mebigfatguy.fbcontrib.utils.BugType;
import com.mebigfatguy.fbcontrib.utils.SignatureBuilder;
import com.mebigfatguy.fbcontrib.utils.ToString;
import com.mebigfatguy.fbcontrib.utils.Values;

import edu.umd.cs.findbugs.BugInstance;
import edu.umd.cs.findbugs.BugReporter;
import edu.umd.cs.findbugs.ba.ClassContext;
import edu.umd.cs.findbugs.bcel.OpcodeStackDetector;
import edu.umd.cs.findbugs.internalAnnotations.SlashedClassName;

/**
 * Looks for methods that pass a primitive wrapper class object, to the same classes Constructor.
 */
public class NeedlessAutoboxing extends OpcodeStackDetector {
    enum State {
        SEEN_NOTHING, SEEN_VALUE, SEEN_VALUEOFSTRING, SEEN_PARSE, SEEN_CTOR, SEEN_VALUEOFPRIMITIVE, SEEN_ICONST, SEEN_GETSTATIC
    }

    private static final Map boxClasses = new HashMap<>();

    private static final Map parseClasses = new HashMap<>();

    private BugReporter bugReporter;
    private State state;
    private @SlashedClassName String boxClass;
    private BitSet ternaryPCs;

    static {
        addBoxClass(boxClasses, Values.SLASHED_JAVA_LANG_BOOLEAN, "boolean", Values.SIG_PRIMITIVE_BOOLEAN);
        addBoxClass(boxClasses, Values.SLASHED_JAVA_LANG_CHARACTER, "char", Values.SIG_PRIMITIVE_CHAR);
        addBoxClass(boxClasses, Values.SLASHED_JAVA_LANG_BYTE, "byte", Values.SIG_PRIMITIVE_BYTE);
        addBoxClass(boxClasses, Values.SLASHED_JAVA_LANG_SHORT, "short", Values.SIG_PRIMITIVE_SHORT);
        addBoxClass(boxClasses, Values.SLASHED_JAVA_LANG_INTEGER, "int", Values.SIG_PRIMITIVE_INT);
        addBoxClass(boxClasses, Values.SLASHED_JAVA_LANG_LONG, "long", Values.SIG_PRIMITIVE_LONG);
        addBoxClass(boxClasses, Values.SLASHED_JAVA_LANG_FLOAT, "float", Values.SIG_PRIMITIVE_FLOAT);
        addBoxClass(boxClasses, Values.SLASHED_JAVA_LANG_DOUBLE, "double", Values.SIG_PRIMITIVE_DOUBLE);
    }

    private static void addBoxClass(Map map, String slashedClass, String primitiveName, String primitiveSig) {
        map.put(slashedClass,
                new BoxParms(new SignatureBuilder().withMethodName(primitiveName + "Value").withReturnType(primitiveSig).toString(),
                        new SignatureBuilder().withParamTypes(primitiveSig).toString(),
                        new SignatureBuilder().withParamTypes(primitiveSig).withReturnType(slashedClass).toString()));
    }

    static {
        addParseClass(parseClasses, Values.SLASHED_JAVA_LANG_BOOLEAN, "boolean", Values.SIG_PRIMITIVE_BOOLEAN);
        addParseClass(parseClasses, Values.SLASHED_JAVA_LANG_CHARACTER, "char", Values.SIG_PRIMITIVE_CHAR);
        addParseClass(parseClasses, Values.SLASHED_JAVA_LANG_BYTE, "byte", Values.SIG_PRIMITIVE_BYTE);
        addParseClass(parseClasses, Values.SLASHED_JAVA_LANG_SHORT, "short", Values.SIG_PRIMITIVE_SHORT);
        addParseClass(parseClasses, Values.SLASHED_JAVA_LANG_INTEGER, "int", Values.SIG_PRIMITIVE_INT);
        addParseClass(parseClasses, Values.SLASHED_JAVA_LANG_LONG, "long", Values.SIG_PRIMITIVE_LONG);
        addParseClass(parseClasses, Values.SLASHED_JAVA_LANG_FLOAT, "float", Values.SIG_PRIMITIVE_DOUBLE);
        addParseClass(parseClasses, Values.SLASHED_JAVA_LANG_DOUBLE, "double", Values.SIG_PRIMITIVE_DOUBLE);
    }

    private static void addParseClass(Map map, String slashedClass, String primitiveName, String primitiveSig) {
        map.put(slashedClass, new SignatureBuilder().withMethodName("parse" + Character.toUpperCase(primitiveName.charAt(0)) + primitiveName.substring(1))
                .withParamTypes(Values.SLASHED_JAVA_LANG_STRING).withReturnType(primitiveSig).toString());
    }

    /**
     * constructs a NAB detector given the reporter to report bugs on
     *
     * @param bugReporter
     *            the sync of bug reports
     */
    public NeedlessAutoboxing(BugReporter bugReporter) {
        this.bugReporter = bugReporter;
    }

    @Override
    public void visitClassContext(ClassContext classContext) {
        try {
            ternaryPCs = new BitSet();
            super.visitClassContext(classContext);
        } finally {
            ternaryPCs = null;
        }
    }

    @Override
    public void visitMethod(Method obj) {
        state = State.SEEN_NOTHING;
        ternaryPCs.clear();
    }

    @Override
    public void sawOpcode(int seen) {

        if (ternaryPCs.get(getPC())) {
            ternaryPCs.clear(getPC());
            state = State.SEEN_NOTHING;
            return;
        }

        switch (state) {
            case SEEN_NOTHING:
                sawOpcodeAfterNothing(seen);
            break;

            case SEEN_VALUE:
                sawOpcodeAfterValue(seen);
            break;

            case SEEN_CTOR:
            case SEEN_VALUEOFPRIMITIVE:
                if (seen == INVOKEVIRTUAL) {
                    BoxParms boxSigs = boxClasses.get(boxClass);
                    if (boxSigs.getPrimitiveValueSignature().equals(getNameConstantOperand() + getSigConstantOperand())) {
                        bugReporter.reportBug(new BugInstance(this, BugType.NAB_NEEDLESS_BOX_TO_UNBOX.name(), NORMAL_PRIORITY).addClass(this).addMethod(this)
                                .addSourceLine(this));
                    } else if (getSigConstantOperand().startsWith(SignatureBuilder.PARAM_NONE) && getNameConstantOperand().endsWith("Value")) {
                        bugReporter.reportBug(new BugInstance(this, BugType.NAB_NEEDLESS_BOX_TO_CAST.name(), NORMAL_PRIORITY).addClass(this).addMethod(this)
                                .addSourceLine(this));
                    }
                }
                state = State.SEEN_NOTHING;
            break;

            case SEEN_VALUEOFSTRING:
                if (seen == INVOKEVIRTUAL) {
                    BoxParms boxSigs = boxClasses.get(boxClass);
                    if (boxSigs.getPrimitiveValueSignature().equals(getNameConstantOperand() + getSigConstantOperand())) {
                        bugReporter.reportBug(new BugInstance(this, BugType.NAB_NEEDLESS_BOXING_PARSE.name(), NORMAL_PRIORITY).addClass(this).addMethod(this)
                                .addSourceLine(this));
                    }
                }
                state = State.SEEN_NOTHING;
            break;

            case SEEN_PARSE:
                if (seen == INVOKESTATIC) {
                    if (boxClass.equals(getClassConstantOperand()) && "valueOf".equals(getNameConstantOperand())) {
                        bugReporter.reportBug(new BugInstance(this, BugType.NAB_NEEDLESS_BOXING_VALUEOF.name(), NORMAL_PRIORITY).addClass(this).addMethod(this)
                                .addSourceLine(this));
                    }
                } else if ((seen == INVOKESPECIAL) && Values.CONSTRUCTOR.equals(getNameConstantOperand()) && boxClass.equals(getClassConstantOperand())) {
                    bugReporter.reportBug(new BugInstance(this, BugType.NAB_NEEDLESS_BOXING_STRING_CTOR.name(), NORMAL_PRIORITY).addClass(this).addMethod(this)
                            .addSourceLine(this));
                }
                state = State.SEEN_NOTHING;
            break;

            case SEEN_ICONST:
                if ((seen == INVOKESTATIC) && Values.SLASHED_JAVA_LANG_BOOLEAN.equals(getClassConstantOperand()) && "valueOf".equals(getNameConstantOperand())
                        && SignatureBuilder.SIG_PRIMITIVE_BOOLEAN_TO_BOOLEAN.equals(getSigConstantOperand())) {
                    bugReporter.reportBug(new BugInstance(this, BugType.NAB_NEEDLESS_BOOLEAN_CONSTANT_CONVERSION.name(), NORMAL_PRIORITY).addClass(this)
                            .addMethod(this).addSourceLine(this));
                }
                state = State.SEEN_NOTHING;
                sawOpcode(seen);
            break;

            case SEEN_GETSTATIC:
                if ((seen == INVOKEVIRTUAL) && Values.SLASHED_JAVA_LANG_BOOLEAN.equals(getClassConstantOperand())
                        && "booleanValue".equals(getNameConstantOperand()) && SignatureBuilder.SIG_VOID_TO_BOOLEAN.equals(getSigConstantOperand())) {
                    bugReporter.reportBug(new BugInstance(this, BugType.NAB_NEEDLESS_BOOLEAN_CONSTANT_CONVERSION.name(), NORMAL_PRIORITY).addClass(this)
                            .addMethod(this).addSourceLine(this));
                }
                state = State.SEEN_NOTHING;
                sawOpcode(seen);
            break;
        }
    }

    private void sawOpcodeAfterNothing(int seen) {
        BoxParms boxSigs;
        switch (seen) {
            case INVOKEVIRTUAL:
                boxClass = getClassConstantOperand();
                boxSigs = boxClasses.get(boxClass);
                if (boxSigs != null) {
                    String methodInfo = getNameConstantOperand() + getSigConstantOperand();
                    if (boxSigs.getPrimitiveValueSignature().equals(methodInfo)) {
                        state = State.SEEN_VALUE;
                    }
                }
            break;

            case INVOKESTATIC:
                boxClass = getClassConstantOperand();
                boxSigs = boxClasses.get(boxClass);
                if (boxSigs != null) {
                    if ("valueOf".equals(getNameConstantOperand())) {
                        String sig = getSigConstantOperand();
                        if (sig.startsWith(SignatureBuilder.PARAM_STRING)) {
                            if (!Values.SLASHED_JAVA_LANG_BOOLEAN.equals(boxClass) || (getClassContext().getJavaClass().getMajor() >= Constants.MAJOR_1_5)) {
                                state = State.SEEN_VALUEOFSTRING;
                            }
                        } else {
                            state = State.SEEN_VALUEOFPRIMITIVE;
                        }
                    } else {
                        String parseSig = parseClasses.get(boxClass);
                        if (parseSig != null) {
                            String methodInfo = getNameConstantOperand() + getSigConstantOperand();
                            if (parseSig.equals(methodInfo)) {
                                state = State.SEEN_PARSE;
                            }
                        }
                    }
                }
            break;

            case INVOKESPECIAL:
                boxClass = getClassConstantOperand();
                boxSigs = boxClasses.get(boxClass);
                if ((boxSigs != null) && Values.CONSTRUCTOR.equals(getNameConstantOperand()) && boxSigs.getCtorSignature().equals(getSigConstantOperand())) {
                    state = State.SEEN_CTOR;
                }
            break;

            case ICONST_0:
            case ICONST_1:
                if (state == State.SEEN_NOTHING) {
                    state = State.SEEN_ICONST;
                }
            break;

            case GETSTATIC:
                String clsName = getClassConstantOperand();
                if (Values.SLASHED_JAVA_LANG_BOOLEAN.equals(clsName)) {
                    String fldName = getNameConstantOperand();
                    if ("TRUE".equals(fldName) || "FALSE".equals(fldName)) {
                        state = State.SEEN_GETSTATIC;
                    }
                }
            break;

            case GOTO:
            case GOTO_W:
                if (stack.getStackDepth() > 0) {
                    ternaryPCs.set(getBranchTarget());
                }
            break;

            default:
            break;
        }
    }

    private void sawOpcodeAfterValue(int seen) {
        if (seen == INVOKESPECIAL) {
            if (boxClass.equals(getClassConstantOperand())) {
                String methodName = getNameConstantOperand();
                if (Values.CONSTRUCTOR.equals(methodName)) {
                    String boxSig = boxClasses.get(boxClass).getCtorSignature();
                    String methodSig = getSigConstantOperand();
                    if (boxSig.equals(methodSig)) {
                        bugReporter.reportBug(new BugInstance(this, BugType.NAB_NEEDLESS_AUTOBOXING_CTOR.name(), NORMAL_PRIORITY).addClass(this).addMethod(this)
                                .addSourceLine(this));
                    }
                }
            }
        } else if ((seen == INVOKESTATIC) && boxClass.equals(getClassConstantOperand())) {
            String methodName = getNameConstantOperand();
            if ("valueOf".equals(methodName)) {
                String boxSig = boxClasses.get(boxClass).getValueOfSignature();
                String methodSig = getSigConstantOperand();
                if (boxSig.equals(methodSig)) {
                    bugReporter.reportBug(new BugInstance(this, BugType.NAB_NEEDLESS_AUTOBOXING_VALUEOF.name(), NORMAL_PRIORITY).addClass(this).addMethod(this)
                            .addSourceLine(this));
                }
            }
        }
        state = State.SEEN_NOTHING;
    }

    static class BoxParms {

        private String primitiveValueSignature;
        private String ctorSignature;
        private String valueOfSignature;

        BoxParms(String primValueSig, String ctorSig, String valueOfSig) {
            primitiveValueSignature = primValueSig;
            ctorSignature = ctorSig;
            valueOfSignature = valueOfSig;
        }

        public String getPrimitiveValueSignature() {
            return primitiveValueSignature;
        }

        public String getCtorSignature() {
            return ctorSignature;
        }

        public String getValueOfSignature() {
            return valueOfSignature;
        }

        @Override
        public String toString() {
            return ToString.build(this);
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy