From fc29843477cce292d4d5cf754b779c641daf66a5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Matou=C5=A1=20Ku=C4=8Dera?= Date: Tue, 20 Aug 2024 12:11:15 +0200 Subject: [PATCH] refactor: strip down optimizer --- README.md | 7 +- .../kotlin/poke.base-conventions.gradle.kts | 2 +- .../main/java/run/slicer/poke/cli/Main.java | 4 + .../main/java/run/slicer/poke/Analyzer.java | 6 + .../java/run/slicer/poke/AnalyzerImpl.java | 17 +- .../run/slicer/poke/proguard/Optimizer.java | 1198 +++++++++++++++++ .../SideEffectVisitorMarkerFactory.java | 120 ++ gradle/libs.versions.toml | 2 +- js/poke.d.ts | 1 + js/src/main/java/run/slicer/poke/js/Main.java | 1 + .../main/java/run/slicer/poke/js/Options.java | 3 + 11 files changed, 1353 insertions(+), 8 deletions(-) create mode 100644 core/src/main/java/run/slicer/poke/proguard/Optimizer.java create mode 100644 core/src/main/java/run/slicer/poke/proguard/SideEffectVisitorMarkerFactory.java diff --git a/README.md b/README.md index d090d3b..4e6e156 100644 --- a/README.md +++ b/README.md @@ -12,19 +12,20 @@ For use as a library, see [how the CLI uses it](./cli/src/main/java/run/slicer/p For use of the actual CLI, grab a build from GitHub Packages and run it with the `--help` option, you should see something like this: ``` -Usage: poke [-hV] [--[no-]optimize] [--[no-]verify] [-p=] - +Usage: poke [-hV] [--[no-]inline] [--[no-]optimize] [--[no-]verify] + [-p=] A Java library for performing bytecode normalization and generic deobfuscation. The class/JAR file to be analyzed. The analyzed class/JAR file destination. -h, --help Show this help message and exit. + --[no-]inline Performs method inlining. --[no-]optimize Performs optimizations. -p, --passes= The amount of optimization passes. -V, --version Print version information and exit. --[no-]verify Performs preemptive verification and correction. ``` -In most use cases, you'll want to use both `--optimize` and `--verify` with a decent amount of passes (5-10). +In most use cases, you'll want to use `--optimize`, `--verify` and `--inline` with a decent amount of passes (5-10). ## Licensing diff --git a/build-logic/src/main/kotlin/poke.base-conventions.gradle.kts b/build-logic/src/main/kotlin/poke.base-conventions.gradle.kts index 4198ce2..40e5211 100644 --- a/build-logic/src/main/kotlin/poke.base-conventions.gradle.kts +++ b/build-logic/src/main/kotlin/poke.base-conventions.gradle.kts @@ -3,7 +3,7 @@ plugins { } group = "run.slicer" -version = "1.0.2" +version = "1.0.3" description = "A Java library for performing bytecode normalization and generic deobfuscation." repositories { diff --git a/cli/src/main/java/run/slicer/poke/cli/Main.java b/cli/src/main/java/run/slicer/poke/cli/Main.java index cfe726f..00908a2 100644 --- a/cli/src/main/java/run/slicer/poke/cli/Main.java +++ b/cli/src/main/java/run/slicer/poke/cli/Main.java @@ -39,12 +39,16 @@ public final class Main implements Callable { @CommandLine.Option(names = "--verify", description = "Performs preemptive verification and correction.", negatable = true) private boolean verify; + @CommandLine.Option(names = "--inline", description = "Performs method inlining.", negatable = true) + private boolean inline; + @Override public Integer call() throws Exception { final Analyzer analyzer = Analyzer.builder() .passes(this.passes) .optimize(this.optimize) .verify(this.verify) + .inline(this.inline) .build(); boolean isClass = false; diff --git a/core/src/main/java/run/slicer/poke/Analyzer.java b/core/src/main/java/run/slicer/poke/Analyzer.java index 4c464f6..71a57dd 100644 --- a/core/src/main/java/run/slicer/poke/Analyzer.java +++ b/core/src/main/java/run/slicer/poke/Analyzer.java @@ -32,6 +32,12 @@ default Builder optimize() { return this.optimize(true); } + Builder inline(boolean inline); + + default Builder inline() { + return this.inline(true); + } + Analyzer build(); } } diff --git a/core/src/main/java/run/slicer/poke/AnalyzerImpl.java b/core/src/main/java/run/slicer/poke/AnalyzerImpl.java index 4e12494..c681bbb 100644 --- a/core/src/main/java/run/slicer/poke/AnalyzerImpl.java +++ b/core/src/main/java/run/slicer/poke/AnalyzerImpl.java @@ -9,11 +9,11 @@ import proguard.classfile.pass.PrimitiveArrayConstantIntroducer; import proguard.classfile.util.PrimitiveArrayConstantReplacer; import proguard.optimize.LineNumberTrimmer; -import proguard.optimize.Optimizer; import proguard.optimize.peephole.LineNumberLinearizer; import proguard.preverify.PreverificationClearer; import proguard.preverify.Preverifier; import proguard.preverify.SubroutineInliner; +import run.slicer.poke.proguard.Optimizer; import java.io.*; import java.util.ArrayList; @@ -90,6 +90,7 @@ static final class Builder implements Analyzer.Builder { private int passes = 1; private boolean verify = false; private boolean optimize = false; + private boolean inline = false; Builder() { } @@ -116,23 +117,33 @@ public Builder optimize(boolean optimize) { return this; } + @Override + public Builder inline(boolean inline) { + this.inline = inline; + return this; + } + @Override public Analyzer build() { final var config = new Configuration(); config.optimize = this.optimize; config.preverify = this.verify; - config.keep = List.of(); config.optimizationPasses = this.passes; final List optimizations = new ArrayList<>(List.of( + "field/*", + "method/generalization/*", + "method/specialization/*", "method/propagation/*", - "method/inlining/*", "code/merging", "code/removal/*", "code/allocation/*" )); + if (this.inline) { + optimizations.add("method/inlining/*"); + } if (PEEPHOLE) { optimizations.add("code/simplification/variable"); optimizations.add("code/simplification/arithmetic"); diff --git a/core/src/main/java/run/slicer/poke/proguard/Optimizer.java b/core/src/main/java/run/slicer/poke/proguard/Optimizer.java new file mode 100644 index 0000000..09ad105 --- /dev/null +++ b/core/src/main/java/run/slicer/poke/proguard/Optimizer.java @@ -0,0 +1,1198 @@ +/* + * ProGuard -- shrinking, optimization, obfuscation, and preverification + * of Java bytecode. + * + * Copyright (c) 2002-2022 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 run.slicer.poke.proguard; + +import proguard.AppView; +import proguard.ClassSpecificationVisitorFactory; +import proguard.Configuration; +import proguard.classfile.AccessConstants; +import proguard.classfile.ClassPool; +import proguard.classfile.VersionConstants; +import proguard.classfile.attribute.Attribute; +import proguard.classfile.attribute.visitor.*; +import proguard.classfile.constant.Constant; +import proguard.classfile.constant.visitor.*; +import proguard.classfile.editor.*; +import proguard.classfile.instruction.visitor.*; +import proguard.classfile.kotlin.visitor.AllFunctionVisitor; +import proguard.classfile.kotlin.visitor.AllPropertyVisitor; +import proguard.classfile.kotlin.visitor.MultiKotlinMetadataVisitor; +import proguard.classfile.kotlin.visitor.ReferencedKotlinMetadataVisitor; +import proguard.classfile.util.BranchTargetFinder; +import proguard.classfile.util.MethodLinker; +import proguard.classfile.visitor.*; +import proguard.evaluation.*; +import proguard.evaluation.value.*; +import proguard.optimize.*; +import proguard.optimize.evaluation.*; +import proguard.optimize.info.*; +import proguard.optimize.kotlin.KotlinContextReceiverUsageMarker; +import proguard.optimize.peephole.*; +import proguard.pass.Pass; +import proguard.util.*; + +import java.io.IOException; +import java.util.ArrayList; +import java.util.List; + +/** + * This pass optimizes class pools according to a given configuration. + *

+ * This pass is stateful. It tracks when no more optimizations are + * possible, and then all further runs of this pass will have no effect. + * + * @author Eric Lafortune + */ +public class Optimizer implements Pass { + private static final Logger logger = Logger.getLogger(Optimizer.class); + + private static final String FIELD_GENERALIZATION_CLASS = "field/generalization/class"; + private static final String FIELD_SPECIALIZATION_TYPE = "field/specialization/type"; + private static final String FIELD_PROPAGATION_VALUE = "field/propagation/value"; + private static final String METHOD_GENERALIZATION_CLASS = "method/generalization/class"; + private static final String METHOD_SPECIALIZATION_PARAMETER_TYPE = "method/specialization/parametertype"; + private static final String METHOD_SPECIALIZATION_RETURN_TYPE = "method/specialization/returntype"; + private static final String METHOD_PROPAGATION_PARAMETER = "method/propagation/parameter"; + private static final String METHOD_PROPAGATION_RETURNVALUE = "method/propagation/returnvalue"; + private static final String METHOD_INLINING_SHORT = "method/inlining/short"; + private static final String METHOD_INLINING_UNIQUE = "method/inlining/unique"; + private static final String METHOD_INLINING_TAILRECURSION = "method/inlining/tailrecursion"; + private static final String CODE_MERGING = "code/merging"; + private static final String CODE_SIMPLIFICATION_VARIABLE = "code/simplification/variable"; + private static final String CODE_SIMPLIFICATION_ARITHMETIC = "code/simplification/arithmetic"; + private static final String CODE_SIMPLIFICATION_CAST = "code/simplification/cast"; + private static final String CODE_SIMPLIFICATION_FIELD = "code/simplification/field"; + private static final String CODE_SIMPLIFICATION_BRANCH = "code/simplification/branch"; + private static final String CODE_SIMPLIFICATION_OBJECT = "code/simplification/object"; + private static final String CODE_SIMPLIFICATION_STRING = "code/simplification/string"; + private static final String CODE_SIMPLIFICATION_MATH = "code/simplification/math"; + private static final String CODE_SIMPLIFICATION_ADVANCED = "code/simplification/advanced"; + private static final String CODE_REMOVAL_ADVANCED = "code/removal/advanced"; + private static final String CODE_REMOVAL_SIMPLE = "code/removal/simple"; + private static final String CODE_REMOVAL_VARIABLE = "code/removal/variable"; + private static final String CODE_REMOVAL_EXCEPTION = "code/removal/exception"; + private static final String CODE_ALLOCATION_VARIABLE = "code/allocation/variable"; + + + private boolean fieldGeneralizationClass; + private boolean fieldSpecializationType; + private boolean fieldPropagationValue; + private boolean methodGeneralizationClass; + private boolean methodSpecializationParametertype; + private boolean methodSpecializationReturntype; + private boolean methodPropagationParameter; + private boolean methodPropagationReturnvalue; + private boolean methodInliningShort; + private boolean methodInliningUnique; + private boolean methodInliningTailrecursion; + private boolean codeMerging; + private boolean codeSimplificationVariable; + private boolean codeSimplificationArithmetic; + private boolean codeSimplificationCast; + private boolean codeSimplificationField; + private boolean codeSimplificationBranch; + private boolean codeSimplificationObject; + private boolean codeSimplificationString; + private boolean codeSimplificationMath; + private boolean codeSimplificationPeephole; + private boolean codeSimplificationAdvanced; + private boolean codeRemovalAdvanced; + private boolean codeRemovalSimple; + private boolean codeRemovalVariable; + private boolean codeRemovalException; + private boolean codeAllocationVariable; + + + // The optimizer uses this field to communicate to its following + // invocations that no further optimizations are possible. + private boolean moreOptimizationsPossible = true; + private int passIndex = 0; + + private final Configuration configuration; + + public Optimizer(Configuration configuration) { + this.configuration = configuration; + } + + + @Override + public void execute(AppView appView) throws IOException { + if (!moreOptimizationsPossible) { + return; + } + + // Create a matcher for filtering optimizations. + StringMatcher filter = configuration.optimizations != null ? + new ListParser(new NameParser()).parse(configuration.optimizations) : + new ConstantMatcher(true); + + fieldGeneralizationClass = filter.matches(FIELD_GENERALIZATION_CLASS); + fieldSpecializationType = filter.matches(FIELD_SPECIALIZATION_TYPE); + fieldPropagationValue = filter.matches(FIELD_PROPAGATION_VALUE); + methodGeneralizationClass = filter.matches(METHOD_GENERALIZATION_CLASS); + methodSpecializationParametertype = filter.matches(METHOD_SPECIALIZATION_PARAMETER_TYPE); + methodSpecializationReturntype = filter.matches(METHOD_SPECIALIZATION_RETURN_TYPE); + methodPropagationParameter = filter.matches(METHOD_PROPAGATION_PARAMETER); + methodPropagationReturnvalue = filter.matches(METHOD_PROPAGATION_RETURNVALUE); + methodInliningShort = filter.matches(METHOD_INLINING_SHORT); + methodInliningUnique = filter.matches(METHOD_INLINING_UNIQUE); + methodInliningTailrecursion = filter.matches(METHOD_INLINING_TAILRECURSION); + codeMerging = filter.matches(CODE_MERGING); + codeSimplificationVariable = filter.matches(CODE_SIMPLIFICATION_VARIABLE); + codeSimplificationArithmetic = filter.matches(CODE_SIMPLIFICATION_ARITHMETIC); + codeSimplificationCast = filter.matches(CODE_SIMPLIFICATION_CAST); + codeSimplificationField = filter.matches(CODE_SIMPLIFICATION_FIELD); + codeSimplificationBranch = filter.matches(CODE_SIMPLIFICATION_BRANCH); + codeSimplificationObject = filter.matches(CODE_SIMPLIFICATION_OBJECT); + codeSimplificationString = filter.matches(CODE_SIMPLIFICATION_STRING); + codeSimplificationMath = filter.matches(CODE_SIMPLIFICATION_MATH); + codeSimplificationAdvanced = filter.matches(CODE_SIMPLIFICATION_ADVANCED); + codeRemovalAdvanced = filter.matches(CODE_REMOVAL_ADVANCED); + codeRemovalSimple = filter.matches(CODE_REMOVAL_SIMPLE); + codeRemovalVariable = filter.matches(CODE_REMOVAL_VARIABLE); + codeRemovalException = filter.matches(CODE_REMOVAL_EXCEPTION); + codeAllocationVariable = filter.matches(CODE_ALLOCATION_VARIABLE); + + // Some optimizations are required by other optimizations. + codeSimplificationAdvanced = + codeSimplificationAdvanced || + fieldPropagationValue || + methodPropagationParameter || + methodPropagationReturnvalue; + + codeRemovalSimple = + codeRemovalSimple || + codeSimplificationBranch; + + codeRemovalException = + codeRemovalException || + codeRemovalAdvanced || + codeRemovalSimple; + + codeSimplificationPeephole = + codeSimplificationVariable || + codeSimplificationArithmetic || + codeSimplificationCast || + codeSimplificationField || + codeSimplificationBranch || + codeSimplificationObject || + codeSimplificationString || + codeSimplificationMath || + fieldGeneralizationClass || + methodGeneralizationClass; + + logger.info("Optimizing (pass {}/{})...", passIndex + 1, configuration.optimizationPasses); + + optimize(configuration, + appView.programClassPool, + appView.libraryClassPool); + + passIndex++; + } + + + /** + * Performs optimization of the given program class pool. + */ + private void optimize(Configuration configuration, + ClassPool programClassPool, + ClassPool libraryClassPool) + throws IOException { + // Create counters to count the numbers of optimizations. + final InstructionCounter fieldGeneralizationClassCounter = new InstructionCounter(); + final MemberCounter fieldSpecializationTypeCounter = new MemberCounter(); + final MemberCounter fieldPropagationValueCounter = new MemberCounter(); + final InstructionCounter methodGeneralizationClassCounter = new InstructionCounter(); + final MemberCounter methodSpecializationParametertypeCounter = new MemberCounter(); + final MemberCounter methodSpecializationReturntypeCounter = new MemberCounter(); + final MemberCounter methodPropagationParameterCounter = new MemberCounter(); + final MemberCounter methodPropagationReturnvalueCounter = new MemberCounter(); + final InstructionCounter methodInliningShortCounter = new InstructionCounter(); + final InstructionCounter methodInliningUniqueCounter = new InstructionCounter(); + final InstructionCounter methodInliningTailrecursionCounter = new InstructionCounter(); + final InstructionCounter codeMergingCounter = new InstructionCounter(); + final InstructionCounter codeSimplificationVariableCounter = new InstructionCounter(); + final InstructionCounter codeSimplificationArithmeticCounter = new InstructionCounter(); + final InstructionCounter codeSimplificationCastCounter = new InstructionCounter(); + final InstructionCounter codeSimplificationFieldCounter = new InstructionCounter(); + final InstructionCounter codeSimplificationBranchCounter = new InstructionCounter(); + final InstructionCounter codeSimplificationObjectCounter = new InstructionCounter(); + final InstructionCounter codeSimplificationStringCounter = new InstructionCounter(); + final InstructionCounter codeSimplificationMathCounter = new InstructionCounter(); + final InstructionCounter codeSimplificationAndroidMathCounter = new InstructionCounter(); + final InstructionCounter codeSimplificationAdvancedCounter = new InstructionCounter(); + final InstructionCounter deletedCounter = new InstructionCounter(); + final InstructionCounter addedCounter = new InstructionCounter(); + final MemberCounter codeRemovalVariableCounter = new MemberCounter(); + final ExceptionCounter codeRemovalExceptionCounter = new ExceptionCounter(); + final MemberCounter codeAllocationVariableCounter = new MemberCounter(); + + // Clean up any old processing info. + programClassPool.classesAccept(new ClassCleaner()); + libraryClassPool.classesAccept(new ClassCleaner()); + + // Link all methods that should get the same optimization info. + programClassPool.classesAccept(new BottomClassFilter( + new MethodLinker())); + libraryClassPool.classesAccept(new BottomClassFilter( + new MethodLinker())); + + // Create a visitor for marking the seeds. + final KeepMarker keepMarker = new KeepMarker(); + + // All library classes and library class members remain unchanged. + libraryClassPool.classesAccept(keepMarker); + libraryClassPool.classesAccept(new AllMemberVisitor(keepMarker)); + + // Mark classes that have the DONT_OPTIMIZE flag set. + programClassPool.classesAccept( + new MultiClassVisitor( + new ClassProcessingFlagFilter(ProcessingFlags.DONT_OPTIMIZE, 0, + keepMarker), + + new AllMemberVisitor( + new MultiMemberVisitor( + new MemberProcessingFlagFilter(ProcessingFlags.DONT_OPTIMIZE, 0, + keepMarker), + + new AllAttributeVisitor( + new AttributeProcessingFlagFilter(ProcessingFlags.DONT_OPTIMIZE, 0, + keepMarker)) + )) + )); + + // We also keep all members that have the DONT_OPTIMIZE flag set on their code attribute. + programClassPool.classesAccept( + new AllMemberVisitor( + new AllAttributeVisitor( + new AttributeNameFilter(Attribute.CODE, + new AttributeProcessingFlagFilter(ProcessingFlags.DONT_OPTIMIZE, 0, + new CodeAttributeToMethodVisitor(keepMarker)))))); + + // We also keep all classes that are involved in .class constructs. + // We're not looking at enum classes though, so they can be simplified. + programClassPool.classesAccept( + new ClassAccessFilter(0, AccessConstants.ENUM, + new AllMethodVisitor( + new AllAttributeVisitor( + new AllInstructionVisitor( + new DotClassClassVisitor(keepMarker)))))); + + // We also keep all classes that are accessed dynamically. + programClassPool.classesAccept( + new AllConstantVisitor( + new ConstantTagFilter(Constant.STRING, + new ReferencedClassVisitor(keepMarker)))); + + // We also keep all class members that are accessed dynamically. + programClassPool.classesAccept( + new AllConstantVisitor( + new ConstantTagFilter(Constant.STRING, + new ReferencedMemberVisitor(keepMarker)))); + + // We also keep all bootstrap method signatures. + programClassPool.classesAccept( + new ClassVersionFilter(VersionConstants.CLASS_VERSION_1_7, + new AllAttributeVisitor( + new AttributeNameFilter(Attribute.BOOTSTRAP_METHODS, + new AllBootstrapMethodInfoVisitor( + new BootstrapMethodHandleTraveler( + new MethodrefTraveler( + new ReferencedMemberVisitor(keepMarker)))))))); + + // We also keep classes and methods referenced from bootstrap + // method arguments. + programClassPool.classesAccept( + new ClassVersionFilter(VersionConstants.CLASS_VERSION_1_7, + new AllAttributeVisitor( + new AttributeNameFilter(Attribute.BOOTSTRAP_METHODS, + new AllBootstrapMethodInfoVisitor( + new AllBootstrapMethodArgumentVisitor( + new MultiConstantVisitor( + // Class constants refer to additional functional + // interfaces (with LambdaMetafactory.altMetafactory). + new ConstantTagFilter(Constant.CLASS, + new ReferencedClassVisitor( + new FunctionalInterfaceFilter( + new ClassHierarchyTraveler(true, false, true, false, + new MultiClassVisitor( + keepMarker, + new AllMethodVisitor( + new MemberAccessFilter(AccessConstants.ABSTRACT, 0, + keepMarker)) + ))))), + + // Method handle constants refer to synthetic lambda + // methods (with LambdaMetafactory.metafactory and + // altMetafactory). + new MethodrefTraveler( + new ReferencedMemberVisitor(keepMarker))))))))); + + // We also keep the classes and abstract methods of functional + // interfaces that are returned by dynamic method invocations. + // These functional interfaces have to remain suitable for the + // dynamic method invocations with LambdaMetafactory. + programClassPool.classesAccept( + new ClassVersionFilter(VersionConstants.CLASS_VERSION_1_7, + new AllConstantVisitor( + new DynamicReturnedClassVisitor( + new FunctionalInterfaceFilter( + new ClassHierarchyTraveler(true, false, true, false, + new MultiClassVisitor( + keepMarker, + new AllMethodVisitor( + new MemberAccessFilter(AccessConstants.ABSTRACT, 0, + keepMarker)) + ))))))); + + // Attach some optimization info to all classes and class members, so + // it can be filled out later. + programClassPool.classesAccept(new ProgramClassOptimizationInfoSetter()); + + programClassPool.classesAccept(new AllMemberVisitor( + new ProgramMemberOptimizationInfoSetter( + false, configuration.optimizeConservatively))); + + if (configuration.assumeNoSideEffects != null) { + // Create a visitor for marking classes and methods that don't have + // any side effects. + NoSideEffectClassMarker noSideEffectClassMarker = new NoSideEffectClassMarker(); + NoSideEffectMethodMarker noSideEffectMethodMarker = new NoSideEffectMethodMarker(); + ClassPoolVisitor classPoolVisitor = + new ClassSpecificationVisitorFactory() + .createClassPoolVisitor(configuration.assumeNoSideEffects, + noSideEffectClassMarker, + noSideEffectMethodMarker); + + // Mark the seeds. + programClassPool.accept(classPoolVisitor); + libraryClassPool.accept(classPoolVisitor); + } + + if (configuration.assumeNoExternalSideEffects != null) { + // Create a visitor for marking classes and methods that don't have + // any external side effects. + NoSideEffectClassMarker noSideEffectClassMarker = new NoSideEffectClassMarker(); + NoExternalSideEffectMethodMarker noSideEffectMethodMarker = new NoExternalSideEffectMethodMarker(); + ClassPoolVisitor classPoolVisitor = + new ClassSpecificationVisitorFactory() + .createClassPoolVisitor(configuration.assumeNoExternalSideEffects, + noSideEffectClassMarker, + noSideEffectMethodMarker); + + // Mark the seeds. + programClassPool.accept(classPoolVisitor); + libraryClassPool.accept(classPoolVisitor); + } + + if (configuration.assumeNoEscapingParameters != null) { + // Create a visitor for marking methods that don't let any + // reference parameters escape. + NoEscapingParametersMethodMarker noEscapingParametersMethodMarker = new NoEscapingParametersMethodMarker(); + ClassPoolVisitor classPoolVisitor = + new ClassSpecificationVisitorFactory() + .createClassPoolVisitor(configuration.assumeNoEscapingParameters, + null, + noEscapingParametersMethodMarker); + + // Mark the seeds. + programClassPool.accept(classPoolVisitor); + libraryClassPool.accept(classPoolVisitor); + } + + if (configuration.assumeNoExternalReturnValues != null) { + // Create a visitor for marking methods that don't let any + // reference parameters escape. + NoExternalReturnValuesMethodMarker noExternalReturnValuesMethodMarker = new NoExternalReturnValuesMethodMarker(); + ClassPoolVisitor classPoolVisitor = + new ClassSpecificationVisitorFactory() + .createClassPoolVisitor(configuration.assumeNoExternalReturnValues, + null, + noExternalReturnValuesMethodMarker); + + // Mark the seeds. + programClassPool.accept(classPoolVisitor); + libraryClassPool.accept(classPoolVisitor); + } + + // Give initial marks to read/written fields. side-effect methods, and + // escaping parameters. + final MutableBoolean mutableBoolean = new MutableBoolean(); + + // Mark all fields as read and written. + programClassPool.classesAccept( + new AllFieldVisitor( + new ReadWriteFieldMarker(mutableBoolean))); + + // Mark methods based on their headers. + programClassPool.classesAccept( + new AllMethodVisitor( + new OptimizationInfoMemberFilter( + new MultiMemberVisitor( + new SideEffectMethodMarker(configuration.optimizeConservatively), + new ParameterEscapeMarker() + )))); + + programClassPool.accept(new InfluenceFixpointVisitor( + new SideEffectVisitorMarkerFactory(configuration.optimizeConservatively))); + + // Mark all used parameters, including the 'this' parameters. + ParallelAllClassVisitor.ClassVisitorFactory markingUsedParametersClassVisitor = + new ParallelAllClassVisitor.ClassVisitorFactory() { + public ClassVisitor createClassVisitor() { + return + new AllMethodVisitor( + new OptimizationInfoMemberFilter( + new ParameterUsageMarker(true, true))); + } + }; + + programClassPool.accept( + new TimedClassPoolVisitor("Marking used parameters", + new ParallelAllClassVisitor( + markingUsedParametersClassVisitor))); + + // Mark all parameters of referenced methods in methods whose code must + // be kept. This prevents shrinking of method descriptors which may not + // be propagated correctly otherwise. + programClassPool.accept( + new TimedClassPoolVisitor("Marking used parameters in kept code attributes", + new AllClassVisitor( + new AllMethodVisitor( + new OptimizationInfoMemberFilter( + null, + + // visit all methods that are kept + new AllAttributeVisitor( + new OptimizationCodeAttributeFilter( + null, + + // visit all code attributes that are kept + new AllInstructionVisitor( + new InstructionConstantVisitor( + new ConstantTagFilter(new int[]{Constant.METHODREF, + Constant.INTERFACE_METHODREF}, + new ReferencedMemberVisitor( + new OptimizationInfoMemberFilter( + // Mark all parameters including "this" of referenced methods + new ParameterUsageMarker(true, true, false)))))))) + ))))); + + // Mark parameter usage based on Kotlin Context receivers + KotlinContextReceiverUsageMarker kotlinContextReceiverUsageMarker = new KotlinContextReceiverUsageMarker(); + programClassPool.accept( + new AllClassVisitor( + new ReferencedKotlinMetadataVisitor( + new MultiKotlinMetadataVisitor( + kotlinContextReceiverUsageMarker, + new AllFunctionVisitor(kotlinContextReceiverUsageMarker), + new AllPropertyVisitor(kotlinContextReceiverUsageMarker) + ))) + ); + + // Perform partial evaluation for filling out fields, method parameters, + // and method return values, so they can be propagated. + if (fieldSpecializationType || + methodSpecializationParametertype || + methodSpecializationReturntype || + fieldPropagationValue || + methodPropagationParameter || + methodPropagationReturnvalue) { + // We'll create values to be stored with fields, method parameters, + // and return values. + ValueFactory valueFactory = new BasicRangeValueFactory(); + ValueFactory detailedValueFactory = new DetailedArrayValueFactory(); + + InvocationUnit storingInvocationUnit = + new StoringInvocationUnit(valueFactory, + fieldSpecializationType || fieldPropagationValue, + methodSpecializationParametertype || methodPropagationParameter, + methodSpecializationReturntype || methodPropagationReturnvalue); + + // Evaluate synthetic classes in more detail, notably to propagate + // the arrays of the classes generated for enum switch statements. + programClassPool.classesAccept( + new ClassAccessFilter(AccessConstants.SYNTHETIC, 0, + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Filling out fields, method parameters, and return values in synthetic classes", + new PartialEvaluator(detailedValueFactory, storingInvocationUnit, false)))))); + + // Evaluate non-synthetic classes. We may need to evaluate all + // casts, to account for downcasts when specializing descriptors. + + ParallelAllClassVisitor.ClassVisitorFactory fillingOutValuesClassVisitor = + new ParallelAllClassVisitor.ClassVisitorFactory() { + public ClassVisitor createClassVisitor() { + ValueFactory valueFactory = new ParticularValueFactory(); + + InvocationUnit storingInvocationUnit = + new StoringInvocationUnit(valueFactory, + fieldSpecializationType || fieldPropagationValue, + methodSpecializationParametertype || methodPropagationParameter, + methodSpecializationReturntype || methodPropagationReturnvalue); + + return + new ClassAccessFilter(0, AccessConstants.SYNTHETIC, + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Filling out fields, method parameters, and return values", + new PartialEvaluator(valueFactory, storingInvocationUnit, + fieldSpecializationType || + methodSpecializationParametertype || + methodSpecializationReturntype))))); + } + }; + + programClassPool.accept( + new TimedClassPoolVisitor("Filling out values in non-synthetic classes", + new ParallelAllClassVisitor( + fillingOutValuesClassVisitor))); + + if (fieldSpecializationType || + methodSpecializationParametertype || + methodSpecializationReturntype) { + // Specialize class member descriptors, based on partial evaluation. + programClassPool.classesAccept( + new AllMemberVisitor( + new OptimizationInfoMemberFilter( + new MemberDescriptorSpecializer(fieldSpecializationType, + methodSpecializationParametertype, + methodSpecializationReturntype, + fieldSpecializationTypeCounter, + methodSpecializationParametertypeCounter, + methodSpecializationReturntypeCounter)))); + + if (fieldSpecializationTypeCounter.getCount() > 0 || + methodSpecializationParametertypeCounter.getCount() > 0 || + methodSpecializationReturntypeCounter.getCount() > 0) { + // Fix all references to specialized members. + programClassPool.classesAccept(new MemberReferenceFixer(configuration.android)); + } + } + + if (configuration.assumeValues != null) { + // Create a visitor for setting assumed values. + ClassPoolVisitor classPoolVisitor = + new AssumeClassSpecificationVisitorFactory(valueFactory) + .createClassPoolVisitor(configuration.assumeValues, + null, + new MultiMemberVisitor()); + + // Set the assumed values. + programClassPool.accept(classPoolVisitor); + libraryClassPool.accept(classPoolVisitor); + } + + if (fieldPropagationValue) { + // Count the constant fields. + programClassPool.classesAccept( + new AllFieldVisitor( + new ConstantMemberFilter(fieldPropagationValueCounter))); + } + + if (methodPropagationParameter) { + // Count the constant method parameters. + programClassPool.classesAccept( + new AllMethodVisitor( + new ConstantParameterFilter(methodPropagationParameterCounter))); + } + + if (methodPropagationReturnvalue) { + // Count the constant method return values. + programClassPool.classesAccept( + new AllMethodVisitor( + new ConstantMemberFilter(methodPropagationReturnvalueCounter))); + } + + if (codeSimplificationAdvanced) { + // Fill out constants into the arrays of synthetic classes, + // notably the arrays of the classes generated for enum switch + // statements. + InvocationUnit loadingInvocationUnit = + new LoadingInvocationUnit(valueFactory, + fieldPropagationValue, + methodPropagationParameter, + methodPropagationReturnvalue); + + programClassPool.classesAccept( + new ClassAccessFilter(AccessConstants.SYNTHETIC, 0, + new AllMethodVisitor( + new AllAttributeVisitor( + new PartialEvaluator(valueFactory, loadingInvocationUnit, false))))); + } + } + + if (codeSimplificationAdvanced) { + ParallelAllClassVisitor.ClassVisitorFactory simplifyingCodeVisitor = + new ParallelAllClassVisitor.ClassVisitorFactory() { + public ClassVisitor createClassVisitor() { + // Perform partial evaluation again, now loading any previously stored + // values for fields, method parameters, and method return values. + ValueFactory valueFactory = new IdentifiedValueFactory(); + + SimplifiedInvocationUnit loadingInvocationUnit = + new LoadingInvocationUnit(valueFactory, + fieldPropagationValue, + methodPropagationParameter, + methodPropagationReturnvalue); + + return + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Simplifying code", + new OptimizationCodeAttributeFilter( + new EvaluationSimplifier( + new PartialEvaluator(valueFactory, loadingInvocationUnit, false), + codeSimplificationAdvancedCounter, + configuration.optimizeConservatively))))); + } + }; + + // Simplify based on partial evaluation, propagating constant + // field values, method parameter values, and return values. + programClassPool.accept( + new TimedClassPoolVisitor("Simplifying code", + new ParallelAllClassVisitor( + simplifyingCodeVisitor))); + } + + if (codeRemovalAdvanced) { + ParallelAllClassVisitor.ClassVisitorFactory shrinkingCodeVisitor = + new ParallelAllClassVisitor.ClassVisitorFactory() { + public ClassVisitor createClassVisitor() { + // Perform partial evaluation again, now loading any previously stored + // values for fields, method parameters, and method return values. + ValueFactory valueFactory = new IdentifiedValueFactory(); + + SimplifiedInvocationUnit loadingInvocationUnit = + new LoadingInvocationUnit(valueFactory, + fieldPropagationValue, + methodPropagationParameter, + methodPropagationReturnvalue); + + // Trace the construction of reference values. + ReferenceTracingValueFactory referenceTracingValueFactory = + new ReferenceTracingValueFactory(valueFactory); + + return + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Shrinking code", + new OptimizationCodeAttributeFilter( + new EvaluationShrinker( + new InstructionUsageMarker( + new PartialEvaluator(referenceTracingValueFactory, + new ParameterTracingInvocationUnit(loadingInvocationUnit), + !codeSimplificationAdvanced, + referenceTracingValueFactory), + true, configuration.optimizeConservatively), true, deletedCounter, addedCounter))))); + } + }; + + // Remove code based on partial evaluation, also removing unused + // parameters from method invocations, and making methods static + // if possible. + programClassPool.accept( + new TimedClassPoolVisitor("Shrinking code", + new ParallelAllClassVisitor( + shrinkingCodeVisitor))); + } + + if (codeRemovalAdvanced) { + // Just update the local variable frame sizes. + programClassPool.classesAccept( + new AllMethodVisitor( + new AllAttributeVisitor( + new OptimizationCodeAttributeFilter( + new StackSizeUpdater())))); + } + + // Mark all classes with package visible members. + // Mark all exception catches of methods. + // Count all method invocations. + // Mark super invocations and other access of methods. + StackSizeComputer stackSizeComputer = new StackSizeComputer(); + + programClassPool.accept( + new TimedClassPoolVisitor("Marking method and referenced class properties", + new MultiClassVisitor( + // Mark classes. + new OptimizationInfoClassFilter( + new MultiClassVisitor( + new PackageVisibleMemberContainingClassMarker(), + new WrapperClassMarker(), + + new AllConstantVisitor( + new PackageVisibleMemberInvokingClassMarker()), + + new AllMemberVisitor( + new ContainsConstructorsMarker()) + )), + + // Mark methods. + new AllMethodVisitor( + new OptimizationInfoMemberFilter( + new AllAttributeVisitor( + new DebugAttributeVisitor("Marking method properties", + new MultiAttributeVisitor( + stackSizeComputer, + new CatchExceptionMarker(), + + new AllInstructionVisitor( + new MultiInstructionVisitor( + new SuperInvocationMarker(), + new DynamicInvocationMarker(), + new BackwardBranchMarker(), + new AccessMethodMarker(), + new SynchronizedBlockMethodMarker(), + new FinalFieldAssignmentMarker(), + new NonEmptyStackReturnMarker(stackSizeComputer) + )) + ))))), + + // Mark referenced classes and methods. + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Marking referenced class properties", + new MultiAttributeVisitor( + new AllExceptionInfoVisitor( + new ExceptionHandlerConstantVisitor( + new ReferencedClassVisitor( + new OptimizationInfoClassFilter( + new CaughtClassMarker())))), + + new AllInstructionVisitor( + new MultiInstructionVisitor( + new InstantiationClassMarker(), + new InstanceofClassMarker(), + new DotClassMarker(), + new MethodInvocationMarker() + )) + )))) + ))); + + if (methodInliningUnique) { + // Inline methods that are only invoked once. + programClassPool.accept( + new TimedClassPoolVisitor("Inlining single methods", + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Inlining single methods", + new OptimizationCodeAttributeFilter( + new SingleInvocationMethodInliner(configuration.microEdition, + configuration.android, + configuration.allowAccessModification, + methodInliningUniqueCounter))))))); + } + + if (methodInliningShort) { + // Inline short methods. + programClassPool.accept( + new TimedClassPoolVisitor("Inlining short methods", + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Inlining short methods", + new OptimizationCodeAttributeFilter( + new ShortMethodInliner(configuration.microEdition, + configuration.android, + configuration.allowAccessModification, + methodInliningShortCounter))))))); + } + + if (methodInliningTailrecursion) { + // Simplify tail recursion calls. + programClassPool.accept( + new TimedClassPoolVisitor("Simplifying tail recursion", + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Simplifying tail recursion", + new OptimizationCodeAttributeFilter( + new TailRecursionSimplifier(methodInliningTailrecursionCounter))))))); + } + + if ((methodInliningUniqueCounter.getCount() > 0 || + methodInliningShortCounter.getCount() > 0 || + methodInliningTailrecursionCounter.getCount() > 0) && + configuration.allowAccessModification) { + // Fix the access flags of referenced classes and class members, + // for MethodInliner. + programClassPool.classesAccept(new AccessFixer()); + + // Fix invocations of interface methods, or methods that have become + // non-abstract or private, and of methods that have moved to a + // different package. + programClassPool.classesAccept( + new AllMemberVisitor( + new AllAttributeVisitor( + new MethodInvocationFixer()))); + } + + if (codeMerging) { + // Share common blocks of code at branches. + programClassPool.accept( + new TimedClassPoolVisitor("Sharing common code", + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Sharing common code", + new OptimizationCodeAttributeFilter( + new GotoCommonCodeReplacer(codeMergingCounter))))))); + } + + if (codeSimplificationPeephole) { + ParallelAllClassVisitor.ClassVisitorFactory peepHoleOptimizer = + new ParallelAllClassVisitor.ClassVisitorFactory() { + public ClassVisitor createClassVisitor() { + // Create a branch target marker and a code attribute editor that can + // be reused for all code attributes. + BranchTargetFinder branchTargetFinder = new BranchTargetFinder(); + CodeAttributeEditor codeAttributeEditor = new CodeAttributeEditor(); + + InstructionSequenceConstants sequences = + new InstructionSequenceConstants(programClassPool, + libraryClassPool); + + List peepholeOptimizations = createPeepholeOptimizations(configuration, + sequences, + branchTargetFinder, + codeAttributeEditor, + codeSimplificationVariableCounter, + codeSimplificationArithmeticCounter, + codeSimplificationCastCounter, + codeSimplificationFieldCounter, + codeSimplificationBranchCounter, + codeSimplificationObjectCounter, + codeSimplificationStringCounter, + codeSimplificationMathCounter, + codeSimplificationAndroidMathCounter, + fieldGeneralizationClassCounter, + methodGeneralizationClassCounter); + + // Convert the list into an array. + InstructionVisitor[] peepholeOptimizationsArray = + new InstructionVisitor[peepholeOptimizations.size()]; + peepholeOptimizations.toArray(peepholeOptimizationsArray); + + return + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Peephole optimizations", + new OptimizationCodeAttributeFilter( + new PeepholeEditor(branchTargetFinder, codeAttributeEditor, + new MultiInstructionVisitor( + peepholeOptimizationsArray)))))); + } + }; + + // Perform the peephole optimisations. + programClassPool.accept( + new TimedClassPoolVisitor("Peephole optimizations", + new ParallelAllClassVisitor( + peepHoleOptimizer))); + } + + if (codeRemovalException) { + // Remove unnecessary exception handlers. + programClassPool.accept( + new TimedClassPoolVisitor("Unreachable exception removal", + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Unreachable exception removal", + new OptimizationCodeAttributeFilter( + new UnreachableExceptionRemover(codeRemovalExceptionCounter))))))); + } + + if (codeRemovalSimple) { + // Remove unreachable code. + programClassPool.accept( + new TimedClassPoolVisitor("Unreachable code removal", + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Unreachable code removal", + new OptimizationCodeAttributeFilter( + new UnreachableCodeRemover(deletedCounter))))))); + } + + if (codeRemovalVariable) { + // Remove all unused local variables. + programClassPool.accept( + new TimedClassPoolVisitor("Variable shrinking", + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Variable shrinking", + new OptimizationCodeAttributeFilter( + new VariableShrinker(codeRemovalVariableCounter))))))); + } + + if (codeAllocationVariable) { + ParallelAllClassVisitor.ClassVisitorFactory optimizingVariablesVisitor = + new ParallelAllClassVisitor.ClassVisitorFactory() { + public ClassVisitor createClassVisitor() { + return + new AllMethodVisitor( + new AllAttributeVisitor( + new DebugAttributeVisitor("Variable optimizations", + new OptimizationCodeAttributeFilter( + new VariableOptimizer(false, codeAllocationVariableCounter))))); + } + }; + + // Optimize the variables. + programClassPool.accept( + new TimedClassPoolVisitor("Variable optimizations", + new ParallelAllClassVisitor( + optimizingVariablesVisitor))); + } + + // Remove unused constants. + programClassPool.accept( + new TimedClassPoolVisitor("Shrinking constant pool", + new ConstantPoolShrinker())); + + int fieldGeneralizationClassCount = fieldGeneralizationClassCounter.getCount(); + int fieldSpecializationTypeCount = fieldSpecializationTypeCounter.getCount(); + int fieldPropagationValueCount = fieldPropagationValueCounter.getCount(); + int methodGeneralizationClassCount = methodGeneralizationClassCounter.getCount(); + int methodSpecializationParametertypeCount = methodSpecializationParametertypeCounter.getCount(); + int methodSpecializationReturntypeCount = methodSpecializationReturntypeCounter.getCount(); + int methodPropagationParameterCount = methodPropagationParameterCounter.getCount(); + int methodPropagationReturnvalueCount = methodPropagationReturnvalueCounter.getCount(); + int methodInliningShortCount = methodInliningShortCounter.getCount(); + int methodInliningUniqueCount = methodInliningUniqueCounter.getCount(); + int methodInliningTailrecursionCount = methodInliningTailrecursionCounter.getCount(); + int codeMergingCount = codeMergingCounter.getCount(); + int codeSimplificationVariableCount = codeSimplificationVariableCounter.getCount(); + int codeSimplificationArithmeticCount = codeSimplificationArithmeticCounter.getCount(); + int codeSimplificationCastCount = codeSimplificationCastCounter.getCount(); + int codeSimplificationFieldCount = codeSimplificationFieldCounter.getCount(); + int codeSimplificationBranchCount = codeSimplificationBranchCounter.getCount(); + int codeSimplificationObjectCount = codeSimplificationObjectCounter.getCount(); + int codeSimplificationStringCount = codeSimplificationStringCounter.getCount(); + int codeSimplificationMathCount = codeSimplificationMathCounter.getCount(); + int codeSimplificationAndroidMathCount = codeSimplificationAndroidMathCounter.getCount(); + int codeSimplificationAdvancedCount = codeSimplificationAdvancedCounter.getCount(); + int codeRemovalCount = deletedCounter.getCount() - addedCounter.getCount(); + int codeRemovalVariableCount = codeRemovalVariableCounter.getCount(); + int codeRemovalExceptionCount = codeRemovalExceptionCounter.getCount(); + int codeAllocationVariableCount = codeAllocationVariableCounter.getCount(); + + // Forget about constant fields, parameters, and return values, if they + // didn't lead to any useful optimizations. We want to avoid fruitless + // additional optimization passes. + if (codeSimplificationAdvancedCount == 0) { + fieldPropagationValueCount = 0; + methodPropagationParameterCount = 0; + methodPropagationReturnvalueCount = 0; + } + + logger.info(" Number of generalized field accesses: {}{}", fieldGeneralizationClassCount, disabled(fieldGeneralizationClass)); + logger.info(" Number of specialized field types: {}{}", fieldSpecializationTypeCount, disabled(fieldSpecializationType)); + logger.info(" Number of inlined constant fields: {}{}", fieldPropagationValueCount, disabled(fieldPropagationValue)); + logger.info(" Number of generalized method invocations: {}{}", methodGeneralizationClassCount, disabled(methodGeneralizationClass)); + logger.info(" Number of specialized method parameter types: {}{}", methodSpecializationParametertypeCount, disabled(methodSpecializationParametertype)); + logger.info(" Number of specialized method return types: {}{}", methodSpecializationReturntypeCount, disabled(methodSpecializationReturntype)); + logger.info(" Number of inlined constant parameters: {}{}", methodPropagationParameterCount, disabled(methodPropagationParameter)); + logger.info(" Number of inlined constant return values: {}{}", methodPropagationReturnvalueCount, disabled(methodPropagationReturnvalue)); + logger.info(" Number of inlined short method calls: {}{}", methodInliningShortCount, disabled(methodInliningShort)); + logger.info(" Number of inlined unique method calls: {}{}", methodInliningUniqueCount, disabled(methodInliningUnique)); + logger.info(" Number of inlined tail recursion calls: {}{}", methodInliningTailrecursionCount, disabled(methodInliningTailrecursion)); + logger.info(" Number of merged code blocks: {}{}", codeMergingCount, disabled(codeMerging)); + logger.info(" Number of variable peephole optimizations: {}{}", codeSimplificationVariableCount, disabled(codeSimplificationVariable)); + logger.info(" Number of arithmetic peephole optimizations: {}{}", codeSimplificationArithmeticCount, disabled(codeSimplificationArithmetic)); + logger.info(" Number of cast peephole optimizations: {}{}", codeSimplificationCastCount, disabled(codeSimplificationCast)); + logger.info(" Number of field peephole optimizations: {}{}", codeSimplificationFieldCount, disabled(codeSimplificationField)); + logger.info(" Number of branch peephole optimizations: {}{}", codeSimplificationBranchCount, disabled(codeSimplificationBranch)); + logger.info(" Number of object peephole optimizations: {}{}", codeSimplificationObjectCount, disabled(codeSimplificationObject)); + logger.info(" Number of string peephole optimizations: {}{}", codeSimplificationStringCount, disabled(codeSimplificationString)); + logger.info(" Number of math peephole optimizations: {}{}", codeSimplificationMathCount, disabled(codeSimplificationMath)); + if (configuration.android) { + logger.info(" Number of Android math peephole optimizations: {}{}", codeSimplificationAndroidMathCount, disabled(codeSimplificationMath)); + } + logger.info(" Number of simplified instructions: {}{}", codeSimplificationAdvancedCount, disabled(codeSimplificationAdvanced)); + logger.info(" Number of removed instructions: {}{}", codeRemovalCount, disabled(codeRemovalAdvanced)); + logger.info(" Number of removed local variables: {}{}", codeRemovalVariableCount, disabled(codeRemovalVariable)); + logger.info(" Number of removed exception blocks: {}{}", codeRemovalExceptionCount, disabled(codeRemovalException)); + logger.info(" Number of optimized local variable frames: {}{}", codeAllocationVariableCount, disabled(codeAllocationVariable)); + + moreOptimizationsPossible = + fieldGeneralizationClassCount > 0 || + fieldSpecializationTypeCount > 0 || + fieldPropagationValueCount > 0 || + methodGeneralizationClassCount > 0 || + methodSpecializationParametertypeCount > 0 || + methodSpecializationReturntypeCount > 0 || + methodPropagationParameterCount > 0 || + methodPropagationReturnvalueCount > 0 || + methodInliningShortCount > 0 || + methodInliningUniqueCount > 0 || + methodInliningTailrecursionCount > 0 || + codeMergingCount > 0 || + codeSimplificationVariableCount > 0 || + codeSimplificationArithmeticCount > 0 || + codeSimplificationCastCount > 0 || + codeSimplificationFieldCount > 0 || + codeSimplificationBranchCount > 0 || + codeSimplificationObjectCount > 0 || + codeSimplificationStringCount > 0 || + codeSimplificationMathCount > 0 || + codeSimplificationAndroidMathCount > 0 || + codeSimplificationAdvancedCount > 0 || + codeRemovalCount > 0 || + codeRemovalVariableCount > 0 || + codeRemovalExceptionCount > 0 || + codeAllocationVariableCount > 0; + } + + + private List createPeepholeOptimizations(Configuration configuration, + InstructionSequenceConstants sequences, + BranchTargetFinder branchTargetFinder, + CodeAttributeEditor codeAttributeEditor, + InstructionCounter codeSimplificationVariableCounter, + InstructionCounter codeSimplificationArithmeticCounter, + InstructionCounter codeSimplificationCastCounter, + InstructionCounter codeSimplificationFieldCounter, + InstructionCounter codeSimplificationBranchCounter, + InstructionCounter codeSimplificationObjectCounter, + InstructionCounter codeSimplificationStringCounter, + InstructionCounter codeSimplificationMathCounter, + InstructionCounter codeSimplificationAndroidMathCounter, + InstructionCounter fieldGeneralizationClassCounter, + InstructionCounter methodGeneralizationClassCounter) { + List peepholeOptimizations = new ArrayList<>(); + + if (codeSimplificationVariable) { + // Peephole optimizations involving local variables. + peepholeOptimizations.add(new InstructionSequencesReplacer(sequences.CONSTANTS, + sequences.VARIABLE_SEQUENCES, + branchTargetFinder, + codeAttributeEditor, + codeSimplificationVariableCounter)); + } + + if (codeSimplificationArithmetic) { + // Peephole optimizations involving arithmetic operations. + peepholeOptimizations.add(new InstructionSequencesReplacer(sequences.CONSTANTS, + sequences.ARITHMETIC_SEQUENCES, + branchTargetFinder, + codeAttributeEditor, + codeSimplificationArithmeticCounter)); + } + + if (codeSimplificationCast) { + // Peephole optimizations involving cast operations. + peepholeOptimizations.add(new InstructionSequencesReplacer(sequences.CONSTANTS, + sequences.CAST_SEQUENCES, + branchTargetFinder, + codeAttributeEditor, + codeSimplificationCastCounter)); + } + + if (codeSimplificationField) { + // Peephole optimizations involving fields. + peepholeOptimizations.add(new InstructionSequencesReplacer(sequences.CONSTANTS, + sequences.FIELD_SEQUENCES, + branchTargetFinder, + codeAttributeEditor, + codeSimplificationFieldCounter)); + } + + if (codeSimplificationBranch) { + // Peephole optimizations involving branches. + peepholeOptimizations.add(new InstructionSequencesReplacer(sequences.CONSTANTS, + sequences.BRANCH_SEQUENCES, + branchTargetFinder, + codeAttributeEditor, + codeSimplificationBranchCounter)); + } + + if (codeSimplificationObject) { + // Peephole optimizations involving objects. + peepholeOptimizations.add(new InstructionSequencesReplacer(sequences.CONSTANTS, + sequences.OBJECT_SEQUENCES, + branchTargetFinder, + codeAttributeEditor, + codeSimplificationObjectCounter)); + + // Include optimizations of instance references on classes without + // constructors. + peepholeOptimizations.add( + new NoConstructorReferenceReplacer(codeAttributeEditor, codeSimplificationObjectCounter)); + } + + if (codeSimplificationString) { + // Peephole optimizations involving branches. + peepholeOptimizations.add(new InstructionSequencesReplacer(sequences.CONSTANTS, + sequences.STRING_SEQUENCES, + branchTargetFinder, codeAttributeEditor, + codeSimplificationStringCounter)); + } + + if (codeSimplificationMath) { + // Peephole optimizations involving math. + peepholeOptimizations.add(new InstructionSequencesReplacer(sequences.CONSTANTS, + sequences.MATH_SEQUENCES, + branchTargetFinder, + codeAttributeEditor, + codeSimplificationMathCounter)); + + if (configuration.android) { + peepholeOptimizations.add(new InstructionSequencesReplacer(sequences.CONSTANTS, + sequences.MATH_ANDROID_SEQUENCES, + branchTargetFinder, + codeAttributeEditor, + codeSimplificationAndroidMathCounter)); + } + } + + if (codeSimplificationBranch) { + // Include optimization of branches to branches and returns. + peepholeOptimizations.add( + new GotoGotoReplacer(codeAttributeEditor, codeSimplificationBranchCounter)); + peepholeOptimizations.add( + new GotoReturnReplacer(codeAttributeEditor, codeSimplificationBranchCounter)); + } + + if (fieldGeneralizationClass || + methodGeneralizationClass) { + // Generalize the target classes of method invocations, to + // reduce the number of descriptors. + peepholeOptimizations.add( + new MemberReferenceGeneralizer(fieldGeneralizationClass, + methodGeneralizationClass, + codeAttributeEditor, + fieldGeneralizationClassCounter, + methodGeneralizationClassCounter)); + } + + + return peepholeOptimizations; + } + + + /** + * Returns a String indicating whether the given flag is enabled or + * disabled. + */ + private String disabled(boolean flag) { + return flag ? "" : " (disabled)"; + } +} diff --git a/core/src/main/java/run/slicer/poke/proguard/SideEffectVisitorMarkerFactory.java b/core/src/main/java/run/slicer/poke/proguard/SideEffectVisitorMarkerFactory.java new file mode 100644 index 0000000..bdd762a --- /dev/null +++ b/core/src/main/java/run/slicer/poke/proguard/SideEffectVisitorMarkerFactory.java @@ -0,0 +1,120 @@ +/* + * ProGuard -- shrinking, optimization, obfuscation, and preverification + * of Java bytecode. + * + * Copyright (c) 2002-2021 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 run.slicer.poke.proguard; + +import proguard.classfile.attribute.visitor.AllAttributeVisitor; +import proguard.classfile.attribute.visitor.DebugAttributeVisitor; +import proguard.classfile.attribute.visitor.MultiAttributeVisitor; +import proguard.classfile.instruction.visitor.AllInstructionVisitor; +import proguard.classfile.instruction.visitor.MultiInstructionVisitor; +import proguard.classfile.visitor.MemberVisitor; +import proguard.evaluation.BasicInvocationUnit; +import proguard.evaluation.PartialEvaluator; +import proguard.evaluation.ReferenceTracingValueFactory; +import proguard.evaluation.value.TypedReferenceValueFactory; +import proguard.optimize.InfluenceFixpointVisitor; +import proguard.optimize.OptimizationInfoMemberFilter; +import proguard.optimize.evaluation.InstructionUsageMarker; +import proguard.optimize.evaluation.ParameterTracingInvocationUnit; +import proguard.optimize.info.ParameterEscapeMarker; +import proguard.optimize.info.SideEffectMethodMarker; + +/** + * The Member Visitors created by this factory checks whether a particular member contains side effects. + */ +class SideEffectVisitorMarkerFactory + implements InfluenceFixpointVisitor.MemberVisitorFactory { + private final boolean optimizeConservatively; + + /** + * @param optimizeConservatively specifies whether conservative + * optimization should be applied + */ + public SideEffectVisitorMarkerFactory(boolean optimizeConservatively) { + this.optimizeConservatively = optimizeConservatively; + } + + // Implementations for MemberVisitorFactory + + public MemberVisitor createMemberVisitor(MemberVisitor influencedMethodCollector) { + ReferenceTracingValueFactory referenceTracingValueFactory1 = + new ReferenceTracingValueFactory(new TypedReferenceValueFactory()); + PartialEvaluator partialEvaluator = + new PartialEvaluator(referenceTracingValueFactory1, + new ParameterTracingInvocationUnit(new BasicInvocationUnit(referenceTracingValueFactory1)), + false, + referenceTracingValueFactory1); + InstructionUsageMarker instructionUsageMarker = + new InstructionUsageMarker(partialEvaluator, false, false); + + + // Create the various markers. + // They will be used as code attribute visitors and + // instruction visitors this time. + // We're currently marking read and written fields once, + // outside of these iterations, for better performance, + // at the cost of some effectiveness (test2209). + //ReadWriteFieldMarker readWriteFieldMarker = + // new ReadWriteFieldMarker(repeatTrigger); + SideEffectMethodMarker sideEffectMethodMarker = new SideEffectMethodMarker(influencedMethodCollector, + optimizeConservatively); + ParameterEscapeMarker parameterEscapeMarker = + new ParameterEscapeMarker(partialEvaluator, false, influencedMethodCollector); + + return + new OptimizationInfoMemberFilter( + // Methods with editable optimization info. + new AllAttributeVisitor( + new DebugAttributeVisitor("Marking fields, methods, and parameters", + new MultiAttributeVisitor( + partialEvaluator, + parameterEscapeMarker, + instructionUsageMarker, + new AllInstructionVisitor( + instructionUsageMarker.necessaryInstructionFilter( + new MultiInstructionVisitor( + // All read / write field instruction are already marked + // for all code (see above), there is no need to mark them again. + // If unused code is removed that accesses fields, the + // respective field will be removed in the next iteration. + // This is a trade-off between performance and correctness. + // TODO: improve the marking for read / write fields after + // performance improvements have been implemented. + //readWriteFieldMarker, + sideEffectMethodMarker, + parameterEscapeMarker + )))))) + + // TODO: disabled for now, see comment above. + // Methods without editable optimization info, for + // which we can't mark side-effects or escaping + // parameters, so we can save some effort. + //new AllAttributeVisitor( + //new DebugAttributeVisitor("Marking fields", + //new MultiAttributeVisitor( + // partialEvaluator, + // instructionUsageMarker, + // new AllInstructionVisitor( + // instructionUsageMarker.necessaryInstructionFilter( + // readWriteFieldMarker))))) + ); + } +} diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index e924b51..3a93da4 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -6,7 +6,7 @@ teavm = "0.10.0" jspecify = { group = "org.jspecify", name = "jspecify", version = "0.3.0" } picocli = { group = "info.picocli", name = "picocli", version.ref = "picocli" } picocli-codegen = { group = "info.picocli", name = "picocli-codegen", version.ref = "picocli" } -proguard = { group = "com.github.run-slicer", name = "proguard", version = "82ace0a065" } +proguard = { group = "com.github.run-slicer", name = "proguard", version = "1d630e03ef" } teavm-core = { group = "org.teavm", name = "teavm-core", version.ref = "teavm" } [plugins] diff --git a/js/poke.d.ts b/js/poke.d.ts index a150bee..24b161a 100644 --- a/js/poke.d.ts +++ b/js/poke.d.ts @@ -3,6 +3,7 @@ declare module "@run-slicer/poke" { passes?: number; optimize?: boolean; verify?: boolean; + inline?: boolean; } export function analyze(data: Uint8Array, config?: Config): Promise; diff --git a/js/src/main/java/run/slicer/poke/js/Main.java b/js/src/main/java/run/slicer/poke/js/Main.java index b51f0be..f657c16 100644 --- a/js/src/main/java/run/slicer/poke/js/Main.java +++ b/js/src/main/java/run/slicer/poke/js/Main.java @@ -31,6 +31,7 @@ private static JSPromise analyze0(byte[] data, Options options) { .passes(options.passes()) .optimize(options.optimize()) .verify(options.verify()) + .inline(options.inline()) .build(); resolve.accept(wrapByteArray(analyzer.analyze(data))); diff --git a/js/src/main/java/run/slicer/poke/js/Options.java b/js/src/main/java/run/slicer/poke/js/Options.java index a142655..0063074 100644 --- a/js/src/main/java/run/slicer/poke/js/Options.java +++ b/js/src/main/java/run/slicer/poke/js/Options.java @@ -12,4 +12,7 @@ public interface Options extends JSObject { @JSBody(script = "return Boolean(this.verify);") boolean verify(); + + @JSBody(script = "return Boolean(this.inline);") + boolean inline(); }