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

proguard.backport.StringConcatenationConverter Maven / Gradle / Ivy

Go to download

ProGuard is a free shrinker, optimizer, obfuscator, and preverifier for Java bytecode

The newest version!
/*
 * ProGuard -- shrinking, optimization, obfuscation, and preverification
 *             of Java bytecode.
 *
 * Copyright (c) 2002-2020 Guardsquare NV
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License as published by the Free
 * Software Foundation; either version 2 of the License, or (at your option)
 * any later version.
 *
 * This program 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 General Public License for
 * more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write to the Free Software Foundation, Inc.,
 * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
 */
package proguard.backport;

import proguard.classfile.*;
import proguard.classfile.attribute.*;
import proguard.classfile.attribute.visitor.*;
import proguard.classfile.constant.*;
import proguard.classfile.constant.visitor.ConstantVisitor;
import proguard.classfile.editor.*;
import proguard.classfile.instruction.*;
import proguard.classfile.instruction.visitor.InstructionVisitor;
import proguard.classfile.util.*;
import proguard.classfile.visitor.*;

import java.util.*;

/**
 * This InstructionVisitor converts all indy String Concatenations in the visited
 * classes to StringBuilder-append chains.
 *
 * @author Tim Van Den Broecke
 */
public class StringConcatenationConverter
implements InstructionVisitor,

           // Implementation interfaces.
           AttributeVisitor,
           BootstrapMethodInfoVisitor,
           ConstantVisitor
{
    private static final int MAXIMUM_BOOLEAN_AS_STRING_LENGTH =  5;  // "false"
    private static final int MAXIMUM_CHAR_AS_STRING_LENGTH    =  1;  // "any char"
    private static final int MAXIMUM_BYTE_AS_STRING_LENGTH    =  3;  // "255"
    private static final int MAXIMUM_SHORT_AS_STRING_LENGTH   =  6;  // "-32768"
    private static final int MAXIMUM_INT_AS_STRING_LENGTH     = 11; // "-2147483648"
    private static final int MAXIMUM_LONG_AS_STRING_LENGTH    = 20; // "-9223372036854775808"
    private static final int MAXIMUM_FLOAT_AS_STRING_LENGTH   = 13; // "-3.4028235E38"
    private static final int MAXIMUM_DOUBLE_AS_STRING_LENGTH  = 23; // "-1.7976931348623157E308"
    private static final int MAXIMUM_AT_HASHCODE_LENGTH       = MAXIMUM_CHAR_AS_STRING_LENGTH +
                                                                MAXIMUM_INT_AS_STRING_LENGTH;

    private static final int DEFAULT_STRINGBUILDER_INIT_SIZE  = 16;

    // Constants as per specification
    private static final char C_VARIABLE_ARGUMENT = '\u0001';
    private static final char C_CONSTANT_ARGUMENT = '\u0002';

    private final InstructionVisitor  extraInstructionVisitor;
    private final CodeAttributeEditor codeAttributeEditor;

    private InstructionSequenceBuilder appendChainComposer;
    private int                        estimatedStringLength;
    private int                        referencedBootstrapMethodIndex;
    private String                     concatenationRecipe;
    private int[]                      concatenationConstants;

    public StringConcatenationConverter(InstructionVisitor  extraInstructionVisitor,
                                        CodeAttributeEditor codeAttributeEditor)
    {
        this.extraInstructionVisitor = extraInstructionVisitor;
        this.codeAttributeEditor     = codeAttributeEditor;
    }


    // Implementations for InstructionVisitor.

    @Override
    public void visitAnyInstruction(Clazz clazz, Method method, CodeAttribute codeAttribute, int offset, Instruction instruction) {}

    @Override
    public void visitConstantInstruction(Clazz clazz, Method method, CodeAttribute codeAttribute, int offset, ConstantInstruction constantInstruction)
    {
        if (constantInstruction.opcode == Instruction.OP_INVOKEDYNAMIC)
        {
            ProgramClass programClass = (ProgramClass) clazz;

            InvokeDynamicConstant invokeDynamicConstant =
                (InvokeDynamicConstant) programClass.getConstant(constantInstruction.constantIndex);

            // Remember the referenced bootstrap method index and extract the recipe from it.
            referencedBootstrapMethodIndex = invokeDynamicConstant.getBootstrapMethodAttributeIndex();
            concatenationRecipe            = null;
            concatenationConstants         = null;

            programClass.attributesAccept(this);

            if (concatenationRecipe != null)
            {
                //if (isMakeConcatWithConstants(invokeDynamicConstant.getName(programClass)))
                String descriptor = invokeDynamicConstant.getType(programClass);

                InstructionSequenceBuilder mainReplacementComposer = new InstructionSequenceBuilder(programClass);
                appendChainComposer                                = new InstructionSequenceBuilder(programClass);
                estimatedStringLength                              = 0;

                // Collect the argument types.
                InternalTypeEnumeration typeEnumeration = new InternalTypeEnumeration(descriptor);
                List            types           = new ArrayList();
                while (typeEnumeration.hasMoreTypes())
                {
                    types.add(typeEnumeration.nextType());
                }

                // Store the correct number of stack values in reverse
                // order in local variables
                int                  variableIndex = codeAttribute.u2maxLocals;
                ListIterator typeIterator  = types.listIterator(types.size());
                while (typeIterator.hasPrevious())
                {
                    String type = typeIterator.previous();

                    mainReplacementComposer.store(variableIndex, type);
                    variableIndex += ClassUtil.internalTypeSize(type);
                }

                // Loop over the recipe.
                // Push the local variables one by one, insert
                // constants where necessary and create and append
                // instruction chain.
                typeIterator = types.listIterator();
                for (int argIndex = 0, constantCounter = 0; argIndex < concatenationRecipe.length(); argIndex++)
                {
                    switch (concatenationRecipe.charAt(argIndex))
                    {
                        case C_VARIABLE_ARGUMENT:
                            String type = typeIterator.next();
                            estimatedStringLength += typicalStringLengthFromType(type);

                            int variableSize = ClassUtil.internalTypeSize(type);
                            variableIndex -= variableSize;

                            appendChainComposer.load(variableIndex, type)
                                               .invokevirtual(ClassConstants.NAME_JAVA_LANG_STRING_BUILDER,
                                                              ClassConstants.METHOD_NAME_APPEND,
                                                              appendDescriptorFromInternalType(type));
                            break;

                        case C_CONSTANT_ARGUMENT:
                            int constantIndex = concatenationConstants[constantCounter++];

                            appendChainComposer.ldc_(constantIndex);
                            // Visit the constant to decide how it needs to be appended.
                            programClass.constantPoolEntryAccept(constantIndex, this);
                            break;

                        default:
                            // Find where the String stops and append it
                            int nextArgIndex = nextArgIndex(concatenationRecipe, argIndex);
                            estimatedStringLength += nextArgIndex - argIndex;
                            appendChainComposer.ldc(concatenationRecipe.substring(argIndex, nextArgIndex))
                                               .invokevirtual(ClassConstants.NAME_JAVA_LANG_STRING_BUILDER,
                                                              ClassConstants.METHOD_NAME_APPEND,
                                                              ClassConstants.METHOD_TYPE_STRING_STRING_BUILDER);

                            // Jump forward to the end of the String
                            argIndex = nextArgIndex - 1;
                            break;
                    }
                }

                // Create a StringBuilder with the estimated initial size
                mainReplacementComposer.new_(         ClassConstants.NAME_JAVA_LANG_STRING_BUILDER)
                                       .dup()
                                       .pushInt(      estimatedStringLength)
                                       .invokespecial(ClassConstants.NAME_JAVA_LANG_STRING_BUILDER,
                                                      ClassConstants.METHOD_NAME_INIT,
                                                      ClassConstants.METHOD_TYPE_INT_VOID);

                // Attach the 'append' instruction chain
                mainReplacementComposer.appendInstructions(appendChainComposer.instructions());

                // Finish with StringBuilder.toString()
                mainReplacementComposer.invokevirtual(ClassConstants.NAME_JAVA_LANG_STRING_BUILDER,
                                                      ClassConstants.METHOD_NAME_TOSTRING,
                                                      ClassConstants.METHOD_TYPE_TOSTRING);

                // Commit the code changes
                codeAttributeEditor.replaceInstruction(offset,
                                                       mainReplacementComposer.instructions());

                // Optionally let this instruction be visited some more
                if (extraInstructionVisitor != null)
                {
                    extraInstructionVisitor.visitConstantInstruction(clazz,
                                                                     method,
                                                                     codeAttribute,
                                                                     offset,
                                                                     constantInstruction);
                }
            }
        }
    }


    // Implementations for AttributeVisitor.

    @Override
    public void visitAnyAttribute(Clazz clazz, Attribute attribute) {}

    @Override
    public void visitBootstrapMethodsAttribute(Clazz                     clazz,
                                               BootstrapMethodsAttribute bootstrapMethodsAttribute)
    {
        bootstrapMethodsAttribute.bootstrapMethodEntryAccept(clazz, referencedBootstrapMethodIndex, this);
    }


    // Implementations for BootstrapMethodInfoVisitor.

    @Override
    public void visitBootstrapMethodInfo(Clazz               clazz,
                                         BootstrapMethodInfo bootstrapMethodInfo)
    {
        ProgramClass programClass = (ProgramClass) clazz;

        MethodHandleConstant bootstrapMethodHandle =
            (MethodHandleConstant) programClass.getConstant(bootstrapMethodInfo.u2methodHandleIndex);

        if (isStringConcatFactory(bootstrapMethodHandle.getClassName(clazz)))
        {
            concatenationRecipe    =
                ((StringConstant) programClass.getConstant(bootstrapMethodInfo.u2methodArguments[0])).getString(programClass);
            concatenationConstants = bootstrapMethodInfo.u2methodArgumentCount > 1 ?
                Arrays.copyOfRange(bootstrapMethodInfo.u2methodArguments, 1, bootstrapMethodInfo.u2methodArgumentCount) :
                new int[0];
        }
    }


    // Implementations for ConstantVisitor.

    @Override
    public void visitAnyConstant(Clazz clazz, Constant constant)
    {
        // append as Object by default. Override below if necessary.
        estimatedStringLength += 16;

        appendChainComposer.invokevirtual(ClassConstants.NAME_JAVA_LANG_STRING_BUILDER,
                                          ClassConstants.METHOD_NAME_APPEND,
                                          ClassConstants.METHOD_TYPE_OBJECT_STRING_BUILDER);
    }

    @Override
    public void visitStringConstant(Clazz clazz, StringConstant stringConstant)
    {
        estimatedStringLength += stringConstant.getString(clazz).length();

        appendChainComposer.invokevirtual(ClassConstants.NAME_JAVA_LANG_STRING_BUILDER,
                                          ClassConstants.METHOD_NAME_APPEND,
                                          ClassConstants.METHOD_TYPE_STRING_STRING_BUILDER);
    }


    // Small utility methods.

    private static boolean isStringConcatFactory(String className)
    {
        return ClassConstants.NAME_JAVA_LANG_INVOKE_STRING_CONCAT_FACTORY.equals(className);
    }

    private static boolean isMakeConcat(String methodName)
    {
        return ClassConstants.METHOD_NAME_MAKE_CONCAT.equals(methodName);
    }

    private static boolean isMakeConcatWithConstants(String methodName)
    {
        return ClassConstants.METHOD_NAME_MAKE_CONCAT_WITH_CONSTANTS.equals(methodName);
    }

    private static int typicalStringLengthFromType(String internalTypeName)
    {
        return internalTypeName.equals(String.valueOf(TypeConstants.BOOLEAN)) ? MAXIMUM_BOOLEAN_AS_STRING_LENGTH :
               internalTypeName.equals(String.valueOf(TypeConstants.CHAR))    ? MAXIMUM_CHAR_AS_STRING_LENGTH    :
               internalTypeName.equals(String.valueOf(TypeConstants.BYTE))    ? MAXIMUM_BYTE_AS_STRING_LENGTH    :
               internalTypeName.equals(String.valueOf(TypeConstants.SHORT))   ? MAXIMUM_SHORT_AS_STRING_LENGTH   :
               internalTypeName.equals(String.valueOf(TypeConstants.INT))     ? MAXIMUM_INT_AS_STRING_LENGTH     :
               internalTypeName.equals(String.valueOf(TypeConstants.LONG))    ? MAXIMUM_LONG_AS_STRING_LENGTH    :
               internalTypeName.equals(String.valueOf(TypeConstants.FLOAT))   ? MAXIMUM_FLOAT_AS_STRING_LENGTH   :
               internalTypeName.equals(String.valueOf(TypeConstants.DOUBLE))  ? MAXIMUM_DOUBLE_AS_STRING_LENGTH  :
                                                                                DEFAULT_STRINGBUILDER_INIT_SIZE  ;
    }

    private static String appendDescriptorFromInternalType(String internalTypeName)
    {
        return internalTypeName.equals(String.valueOf(TypeConstants.BOOLEAN)) ? ClassConstants.METHOD_TYPE_BOOLEAN_STRING_BUILDER :
               internalTypeName.equals(String.valueOf(TypeConstants.CHAR))    ? ClassConstants.METHOD_TYPE_CHAR_STRING_BUILDER    :
               internalTypeName.equals(String.valueOf(TypeConstants.BYTE))  ||
               internalTypeName.equals(String.valueOf(TypeConstants.SHORT)) ||
               internalTypeName.equals(String.valueOf(TypeConstants.INT))     ? ClassConstants.METHOD_TYPE_INT_STRING_BUILDER     :
               internalTypeName.equals(String.valueOf(TypeConstants.LONG))    ? ClassConstants.METHOD_TYPE_LONG_STRING_BUILDER    :
               internalTypeName.equals(String.valueOf(TypeConstants.FLOAT))   ? ClassConstants.METHOD_TYPE_FLOAT_STRING_BUILDER   :
               internalTypeName.equals(String.valueOf(TypeConstants.DOUBLE))  ? ClassConstants.METHOD_TYPE_DOUBLE_STRING_BUILDER  :
               internalTypeName.equals(ClassConstants.NAME_JAVA_LANG_STRING)  ? ClassConstants.METHOD_TYPE_STRING_STRING_BUILDER  :
                                                                                ClassConstants.METHOD_TYPE_OBJECT_STRING_BUILDER;
    }

    private static int nextArgIndex(String recipe, int fromIndex)
    {
        for(int i = fromIndex; i < recipe.length(); i++)
        {
            char c = recipe.charAt(i);
            if (c == C_VARIABLE_ARGUMENT ||
                c == C_CONSTANT_ARGUMENT)
            {
                return i;
            }
        }
        return recipe.length();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy