src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/alloc/trace/DefaultTraceRegisterAllocationPolicy.java
branchJDK-8200758-branch
changeset 57069 b8385a806d2b
parent 57068 eb6d315c4e39
parent 52934 8deeb7bba516
child 57070 42783e8e73de
--- a/src/jdk.internal.vm.compiler/share/classes/org.graalvm.compiler.lir/src/org/graalvm/compiler/lir/alloc/trace/DefaultTraceRegisterAllocationPolicy.java	Mon Dec 10 16:49:54 2018 -0500
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,420 +0,0 @@
-/*
- * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
- * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
- *
- * This code is free software; you can redistribute it and/or modify it
- * under the terms of the GNU General Public License version 2 only, as
- * published by the Free Software Foundation.
- *
- * This code 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
- * version 2 for more details (a copy is included in the LICENSE file that
- * accompanied this code).
- *
- * You should have received a copy of the GNU General Public License version
- * 2 along with this work; if not, write to the Free Software Foundation,
- * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
- *
- * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
- * or visit www.oracle.com if you need additional information or have any
- * questions.
- */
-
-
-package org.graalvm.compiler.lir.alloc.trace;
-
-import java.util.ArrayList;
-
-import org.graalvm.compiler.core.common.alloc.RegisterAllocationConfig;
-import org.graalvm.compiler.core.common.alloc.Trace;
-import org.graalvm.compiler.core.common.alloc.TraceBuilderResult;
-import org.graalvm.compiler.core.common.cfg.AbstractBlockBase;
-import org.graalvm.compiler.lir.alloc.trace.TraceAllocationPhase.TraceAllocationContext;
-import org.graalvm.compiler.lir.alloc.trace.TraceRegisterAllocationPolicy.AllocationStrategy;
-import org.graalvm.compiler.lir.alloc.trace.bu.BottomUpAllocator;
-import org.graalvm.compiler.lir.alloc.trace.lsra.TraceLinearScanPhase;
-import org.graalvm.compiler.lir.gen.LIRGenerationResult;
-import org.graalvm.compiler.lir.gen.LIRGeneratorTool.MoveFactory;
-import org.graalvm.compiler.options.EnumOptionKey;
-import org.graalvm.compiler.options.Option;
-import org.graalvm.compiler.options.OptionKey;
-import org.graalvm.compiler.options.OptionType;
-import org.graalvm.compiler.options.OptionValues;
-
-import jdk.vm.ci.code.TargetDescription;
-import jdk.vm.ci.common.JVMCIError;
-import jdk.vm.ci.meta.AllocatableValue;
-import jdk.vm.ci.meta.PlatformKind;
-
-/**
- * Manages the selection of allocation strategies.
- */
-public final class DefaultTraceRegisterAllocationPolicy {
-
-    public enum TraceRAPolicies {
-        Default,
-        LinearScanOnly,
-        BottomUpOnly,
-        AlmostTrivial,
-        NumVariables,
-        Ratio,
-        Loops,
-        MaxFreq,
-        FreqBudget,
-        LoopRatio,
-        LoopBudget,
-        LoopMaxFreq
-    }
-
-    public static class Options {
-        // @formatter:off
-        @Option(help = "Use special allocator for trivial blocks.", type = OptionType.Debug)
-        public static final OptionKey<Boolean> TraceRAtrivialBlockAllocator = new OptionKey<>(true);
-        @Option(help = "Use BottomUp if there is only one block with at most this number of instructions", type = OptionType.Debug)
-        public static final OptionKey<Integer> TraceRAalmostTrivialSize = new OptionKey<>(2);
-        @Option(help = "Use BottomUp for traces with low number of variables at block boundaries", type = OptionType.Debug)
-        public static final OptionKey<Integer> TraceRAnumVariables = new OptionKey<>(null);
-        @Option(help = "Use LSRA / BottomUp ratio", type = OptionType.Debug)
-        public static final OptionKey<Double> TraceRAbottomUpRatio = new OptionKey<>(0.0);
-        @Option(help = "Frequency Threshold", type = OptionType.Debug)
-        public static final OptionKey<Double> TraceRAfrequencyThreshold = new OptionKey<>(0.8);
-        @Option(help = "Sum Frequency Budget Threshold", type = OptionType.Debug)
-        public static final OptionKey<Double> TraceRAsumBudget = new OptionKey<>(0.5);
-        @Option(help = "TraceRA allocation policy to use.", type = OptionType.Debug)
-        public static final EnumOptionKey<TraceRAPolicies> TraceRAPolicy = new EnumOptionKey<>(TraceRAPolicies.Default);
-        // @formatter:on
-    }
-
-    public static final class TrivialTraceStrategy extends AllocationStrategy {
-
-        public TrivialTraceStrategy(TraceRegisterAllocationPolicy plan) {
-            plan.super();
-        }
-
-        @Override
-        public boolean shouldApplyTo(Trace trace) {
-            return TraceUtil.isTrivialTrace(getLIR(), trace);
-        }
-
-        @Override
-        protected TraceAllocationPhase<TraceAllocationContext> initAllocator(TargetDescription target, LIRGenerationResult lirGenRes, MoveFactory spillMoveFactory,
-                        RegisterAllocationConfig registerAllocationConfig, AllocatableValue[] cachedStackSlots, TraceBuilderResult resultTraces, boolean neverSpillConstant,
-                        GlobalLivenessInfo livenessInfo, ArrayList<AllocationStrategy> strategies) {
-            return new TrivialTraceAllocator();
-        }
-    }
-
-    public static class BottomUpStrategy extends AllocationStrategy {
-
-        public BottomUpStrategy(TraceRegisterAllocationPolicy plan) {
-            // explicitly specify the enclosing instance for the superclass constructor call
-            plan.super();
-        }
-
-        @Override
-        public boolean shouldApplyTo(Trace trace) {
-            return !containsExceptionEdge(trace);
-        }
-
-        private static boolean containsExceptionEdge(Trace trace) {
-            for (AbstractBlockBase<?> block : trace.getBlocks()) {
-                // check if one of the successors is an exception handler
-                for (AbstractBlockBase<?> succ : block.getSuccessors()) {
-                    if (succ.isExceptionEntry()) {
-                        return true;
-                    }
-                }
-            }
-            return false;
-        }
-
-        @Override
-        protected TraceAllocationPhase<TraceAllocationContext> initAllocator(TargetDescription target, LIRGenerationResult lirGenRes, MoveFactory spillMoveFactory,
-                        RegisterAllocationConfig registerAllocationConfig, AllocatableValue[] cachedStackSlots, TraceBuilderResult resultTraces, boolean neverSpillConstant,
-                        GlobalLivenessInfo livenessInfo, ArrayList<AllocationStrategy> strategies) {
-            return new BottomUpAllocator(target, lirGenRes, spillMoveFactory, registerAllocationConfig, cachedStackSlots, resultTraces, neverSpillConstant, livenessInfo);
-        }
-    }
-
-    public static final class BottomUpAlmostTrivialStrategy extends BottomUpStrategy {
-
-        private final int trivialTraceSize;
-
-        public BottomUpAlmostTrivialStrategy(TraceRegisterAllocationPolicy plan) {
-            // explicitly specify the enclosing instance for the superclass constructor call
-            super(plan);
-            trivialTraceSize = Options.TraceRAalmostTrivialSize.getValue(plan.getOptions());
-        }
-
-        @Override
-        public boolean shouldApplyTo(Trace trace) {
-            if (!super.shouldApplyTo(trace)) {
-                return false;
-            }
-            if (trace.size() != 1) {
-                return false;
-            }
-            return getLIR().getLIRforBlock(trace.getBlocks()[0]).size() <= trivialTraceSize;
-        }
-
-    }
-
-    public static final class BottomUpNumVariablesStrategy extends BottomUpStrategy {
-
-        private final int numVarLimit;
-
-        public BottomUpNumVariablesStrategy(TraceRegisterAllocationPolicy plan) {
-            // explicitly specify the enclosing instance for the superclass constructor call
-            super(plan);
-            Integer value = Options.TraceRAnumVariables.getValue(plan.getOptions());
-            if (value != null) {
-                numVarLimit = value;
-            } else {
-                /* Default to the number of allocatable word registers. */
-                PlatformKind wordKind = getTarget().arch.getWordKind();
-                int numWordRegisters = getRegisterAllocationConfig().getAllocatableRegisters(wordKind).allocatableRegisters.length;
-                numVarLimit = numWordRegisters;
-            }
-        }
-
-        @Override
-        public boolean shouldApplyTo(Trace trace) {
-            if (!super.shouldApplyTo(trace)) {
-                return false;
-            }
-            GlobalLivenessInfo livenessInfo = getGlobalLivenessInfo();
-            int maxNumVars = livenessInfo.getBlockIn(trace.getBlocks()[0]).length;
-            for (AbstractBlockBase<?> block : trace.getBlocks()) {
-                maxNumVars = Math.max(maxNumVars, livenessInfo.getBlockOut(block).length);
-            }
-            return maxNumVars <= numVarLimit;
-        }
-
-    }
-
-    public static final class BottomUpRatioStrategy extends BottomUpStrategy {
-
-        private final double ratio;
-
-        public BottomUpRatioStrategy(TraceRegisterAllocationPolicy plan) {
-            // explicitly specify the enclosing instance for the superclass constructor call
-            super(plan);
-            ratio = Options.TraceRAbottomUpRatio.getValue(plan.getOptions());
-        }
-
-        @Override
-        public boolean shouldApplyTo(Trace trace) {
-            if (!super.shouldApplyTo(trace)) {
-                return false;
-            }
-            double numTraces = getTraceBuilderResult().getTraces().size();
-            double traceId = trace.getId();
-            assert ratio >= 0 && ratio <= 1.0 : "Ratio out of range: " + ratio;
-            return (traceId / numTraces) >= ratio;
-        }
-
-    }
-
-    public abstract static class BottomUpLoopStrategyBase extends BottomUpStrategy {
-
-        public BottomUpLoopStrategyBase(TraceRegisterAllocationPolicy plan) {
-            // explicitly specify the enclosing instance for the superclass constructor call
-            super(plan);
-        }
-
-        @Override
-        public final boolean shouldApplyTo(Trace trace) {
-            if (!super.shouldApplyTo(trace)) {
-                return false;
-            }
-            if (getLIR().getControlFlowGraph().getLoops().isEmpty()) {
-                return shouldApplyToNoLoop(trace);
-            }
-            for (AbstractBlockBase<?> block : trace.getBlocks()) {
-                if (block.getLoopDepth() > 0) {
-                    return false;
-                }
-            }
-            return true;
-        }
-
-        protected abstract boolean shouldApplyToNoLoop(Trace trace);
-
-    }
-
-    public static final class BottomUpLoopStrategy extends BottomUpLoopStrategyBase {
-
-        public BottomUpLoopStrategy(TraceRegisterAllocationPolicy plan) {
-            // explicitly specify the enclosing instance for the superclass constructor call
-            super(plan);
-        }
-
-        @Override
-        protected boolean shouldApplyToNoLoop(Trace trace) {
-            // no loops at all -> use LSRA
-            return false;
-        }
-
-    }
-
-    public static final class BottomUpDelegatingLoopStrategy extends BottomUpLoopStrategyBase {
-
-        private final BottomUpStrategy delegate;
-
-        public BottomUpDelegatingLoopStrategy(TraceRegisterAllocationPolicy plan, BottomUpStrategy delegate) {
-            // explicitly specify the enclosing instance for the superclass constructor call
-            super(plan);
-            this.delegate = delegate;
-        }
-
-        @Override
-        protected boolean shouldApplyToNoLoop(Trace trace) {
-            return delegate.shouldApplyTo(trace);
-        }
-
-    }
-
-    public static final class BottomUpMaxFrequencyStrategy extends BottomUpStrategy {
-
-        private final double maxRelativeFrequency;
-        private final double frequencyThreshold;
-
-        public BottomUpMaxFrequencyStrategy(TraceRegisterAllocationPolicy plan) {
-            // explicitly specify the enclosing instance for the superclass constructor call
-            super(plan);
-            maxRelativeFrequency = maxRelativeFrequency(getLIR().getControlFlowGraph().getBlocks());
-            frequencyThreshold = Options.TraceRAfrequencyThreshold.getValue(plan.getOptions());
-        }
-
-        private static double maxRelativeFrequency(AbstractBlockBase<?>[] blocks) {
-            double max = 0;
-            for (AbstractBlockBase<?> block : blocks) {
-                double frequency = block.getRelativeFrequency();
-                if (frequency > max) {
-                    max = frequency;
-                }
-            }
-            return max;
-        }
-
-        @Override
-        public boolean shouldApplyTo(Trace trace) {
-            if (!super.shouldApplyTo(trace)) {
-                return false;
-            }
-            return maxRelativeFrequency(trace.getBlocks()) / maxRelativeFrequency <= frequencyThreshold;
-        }
-
-    }
-
-    public static final class BottomUpFrequencyBudgetStrategy extends BottomUpStrategy {
-
-        private final double[] cumulativeTraceFrequency;
-        private final double budget;
-
-        public BottomUpFrequencyBudgetStrategy(TraceRegisterAllocationPolicy plan) {
-            // explicitly specify the enclosing instance for the superclass constructor call
-            super(plan);
-            ArrayList<Trace> traces = getTraceBuilderResult().getTraces();
-            this.cumulativeTraceFrequency = new double[traces.size()];
-            double sumMethodFrequency = init(traces, this.cumulativeTraceFrequency);
-            this.budget = sumMethodFrequency * Options.TraceRAsumBudget.getValue(plan.getOptions());
-        }
-
-        private static double init(ArrayList<Trace> traces, double[] sumTraces) {
-            double sumMethod = 0;
-            for (Trace trace : traces) {
-                double traceSum = 0;
-                for (AbstractBlockBase<?> block : trace.getBlocks()) {
-                    traceSum += block.getRelativeFrequency();
-                }
-                sumMethod += traceSum;
-                // store cumulative sum for trace
-                sumTraces[trace.getId()] = sumMethod;
-            }
-            return sumMethod;
-        }
-
-        @Override
-        public boolean shouldApplyTo(Trace trace) {
-            if (!super.shouldApplyTo(trace)) {
-                return false;
-            }
-            double cumTraceFrequency = cumulativeTraceFrequency[trace.getId()];
-            return cumTraceFrequency > budget;
-        }
-
-    }
-
-    public static final class TraceLinearScanStrategy extends AllocationStrategy {
-
-        public TraceLinearScanStrategy(TraceRegisterAllocationPolicy plan) {
-            // explicitly specify the enclosing instance for the superclass constructor call
-            plan.super();
-        }
-
-        @Override
-        public boolean shouldApplyTo(Trace trace) {
-            return true;
-        }
-
-        @Override
-        protected TraceAllocationPhase<TraceAllocationContext> initAllocator(TargetDescription target, LIRGenerationResult lirGenRes, MoveFactory spillMoveFactory,
-                        RegisterAllocationConfig registerAllocationConfig, AllocatableValue[] cachedStackSlots, TraceBuilderResult resultTraces, boolean neverSpillConstant,
-                        GlobalLivenessInfo livenessInfo, ArrayList<AllocationStrategy> strategies) {
-            return new TraceLinearScanPhase(target, lirGenRes, spillMoveFactory, registerAllocationConfig, resultTraces, neverSpillConstant, cachedStackSlots, livenessInfo);
-        }
-
-    }
-
-    public static TraceRegisterAllocationPolicy allocationPolicy(TargetDescription target, LIRGenerationResult lirGenRes, MoveFactory spillMoveFactory,
-                    RegisterAllocationConfig registerAllocationConfig, AllocatableValue[] cachedStackSlots, TraceBuilderResult resultTraces, boolean neverSpillConstant,
-                    GlobalLivenessInfo livenessInfo, OptionValues options) {
-        TraceRegisterAllocationPolicy plan = new TraceRegisterAllocationPolicy(target, lirGenRes, spillMoveFactory, registerAllocationConfig, cachedStackSlots, resultTraces, neverSpillConstant,
-                        livenessInfo);
-        if (Options.TraceRAtrivialBlockAllocator.getValue(options)) {
-            plan.appendStrategy(new TrivialTraceStrategy(plan));
-        }
-        switch (Options.TraceRAPolicy.getValue(options)) {
-            case Default:
-            case LinearScanOnly:
-                break;
-            case BottomUpOnly:
-                plan.appendStrategy(new BottomUpStrategy(plan));
-                break;
-            case AlmostTrivial:
-                plan.appendStrategy(new BottomUpAlmostTrivialStrategy(plan));
-                break;
-            case NumVariables:
-                plan.appendStrategy(new BottomUpNumVariablesStrategy(plan));
-                break;
-            case Ratio:
-                plan.appendStrategy(new BottomUpRatioStrategy(plan));
-                break;
-            case Loops:
-                plan.appendStrategy(new BottomUpLoopStrategy(plan));
-                break;
-            case MaxFreq:
-                plan.appendStrategy(new BottomUpMaxFrequencyStrategy(plan));
-                break;
-            case FreqBudget:
-                plan.appendStrategy(new BottomUpFrequencyBudgetStrategy(plan));
-                break;
-            case LoopRatio:
-                plan.appendStrategy(new BottomUpDelegatingLoopStrategy(plan, new BottomUpRatioStrategy(plan)));
-                break;
-            case LoopMaxFreq:
-                plan.appendStrategy(new BottomUpDelegatingLoopStrategy(plan, new BottomUpMaxFrequencyStrategy(plan)));
-                break;
-            case LoopBudget:
-                plan.appendStrategy(new BottomUpDelegatingLoopStrategy(plan, new BottomUpFrequencyBudgetStrategy(plan)));
-                break;
-            default:
-                throw JVMCIError.shouldNotReachHere();
-        }
-        // Fallback
-        plan.appendStrategy(new TraceLinearScanStrategy(plan));
-        return plan;
-    }
-}