/* * Copyright (C) 2015 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef ART_COMPILER_OPTIMIZING_INDUCTION_VAR_RANGE_H_ #define ART_COMPILER_OPTIMIZING_INDUCTION_VAR_RANGE_H_ #include "base/macros.h" #include "induction_var_analysis.h" namespace art HIDDEN { /** * This class implements range analysis on expressions within loops. It takes the results * of induction variable analysis in the constructor and provides a public API to obtain * a conservative lower and upper bound value or last value on each instruction in the HIR. * The public API also provides a few general-purpose utility methods related to induction. * * The range analysis is done with a combination of symbolic and partial integral evaluation * of expressions. The analysis avoids complications with wrap-around arithmetic on the integral * parts but all clients should be aware that wrap-around may occur on any of the symbolic parts. * For example, given a known range for [0,100] for i, the evaluation yields range [-100,100] * for expression -2*i+100, which is exact, and range [x,x+100] for expression i+x, which may * wrap-around anywhere in the range depending on the actual value of x. */ class InductionVarRange { public: /* * A value that can be represented as "a * instruction + b" for 32-bit constants, where * Value() denotes an unknown lower and upper bound. Although range analysis could yield * more complex values, the format is sufficiently powerful to represent useful cases * and feeds directly into optimizations like bounds check elimination. */ struct Value { Value() : instruction(nullptr), a_constant(0), b_constant(0), is_known(false) {} Value(HInstruction* i, int32_t a, int32_t b) : instruction(a != 0 ? i : nullptr), a_constant(a), b_constant(b), is_known(true) {} explicit Value(int32_t b) : Value(nullptr, 0, b) {} // Representation as: a_constant x instruction + b_constant. HInstruction* instruction; int32_t a_constant; int32_t b_constant; // If true, represented by prior fields. Otherwise unknown value. bool is_known; }; explicit InductionVarRange(HInductionVarAnalysis* induction); /** * Given a context block, returns a possibly conservative lower * and upper bound on the instruction's value in the output parameters min_val and max_val, * respectively. The need_finite_test flag denotes if an additional finite-test is needed * to protect the range evaluation inside its loop. The parameter chase_hint defines an * instruction at which chasing may stop. Returns false on failure. */ bool GetInductionRange(const HBasicBlock* context, HInstruction* instruction, HInstruction* chase_hint, /*out*/ Value* min_val, /*out*/ Value* max_val, /*out*/ bool* needs_finite_test); /** * Returns true if range analysis is able to generate code for the lower and upper * bound expressions on the instruction in the given context. The need_finite_test * and need_taken test flags denote if an additional finite-test and/or taken-test * are needed to protect the range evaluation inside its loop. */ bool CanGenerateRange(const HBasicBlock* context, HInstruction* instruction, /*out*/ bool* needs_finite_test, /*out*/ bool* needs_taken_test); /** * Generates the actual code in the HIR for the lower and upper bound expressions on the * instruction in the given context. Code for the lower and upper bound expression are * generated in given block and graph and are returned in the output parameters lower and * upper, respectively. For a loop invariant, lower is not set. * * For example, given expression x+i with range [0, 5] for i, calling this method * will generate the following sequence: * * block: * lower: add x, 0 * upper: add x, 5 * * Precondition: CanGenerateRange() returns true. */ void GenerateRange(const HBasicBlock* context, HInstruction* instruction, HGraph* graph, HBasicBlock* block, /*out*/ HInstruction** lower, /*out*/ HInstruction** upper); /** * Generates explicit taken-test for the given `loop_control` instruction. Code is generated in * given block and graph. Returns generated taken-test. * * Precondition: CanGenerateRange() returns true and needs_taken_test is set. */ HInstruction* GenerateTakenTest(HInstruction* loop_control, HGraph* graph, HBasicBlock* block); /** * Returns true if induction analysis is able to generate code for last value of * the given instruction inside the closest enveloping loop. */ bool CanGenerateLastValue(HInstruction* instruction); /** * Generates last value of the given instruction in the closest enveloping loop. * Code is generated in given block and graph. Returns generated last value. * * Precondition: CanGenerateLastValue() returns true. */ HInstruction* GenerateLastValue(HInstruction* instruction, HGraph* graph, HBasicBlock* block); /** * Updates all matching fetches with the given replacement in all induction information * that is associated with the given instruction. */ void Replace(HInstruction* instruction, HInstruction* fetch, HInstruction* replacement); /** * Incrementally updates induction information for just the given loop. */ void ReVisit(const HLoopInformation* loop) { induction_analysis_->induction_.erase(loop); for (HInstructionIterator it(loop->GetHeader()->GetPhis()); !it.Done(); it.Advance()) { induction_analysis_->cycles_.erase(it.Current()->AsPhi()); } induction_analysis_->VisitLoop(loop); } /** * Lookup an interesting cycle associated with an entry phi. */ ArenaSet* LookupCycle(HPhi* phi) const { return induction_analysis_->LookupCycle(phi); } /** * Checks if the given phi instruction has been classified as anything by * induction variable analysis. Returns false for anything that cannot be * classified statically, such as reductions or other complex cycles. */ bool IsClassified(HPhi* phi) const { HLoopInformation* lp = phi->GetBlock()->GetLoopInformation(); // closest enveloping loop return (lp != nullptr) && (induction_analysis_->LookupInfo(lp, phi) != nullptr); } /** * Checks if header logic of a loop terminates. If trip count is known sets 'trip_count' to its * value. */ bool IsFinite(const HLoopInformation* loop, /*out*/ int64_t* trip_count) const; /** * Checks if a trip count is known for the loop and sets 'trip_count' to its value in this case. */ bool HasKnownTripCount(const HLoopInformation* loop, /*out*/ int64_t* trip_count) const; /** * Checks if the given instruction is a unit stride induction inside the closest enveloping * loop of the context that is defined by the first parameter (e.g. pass an array reference * as context and the index as instruction to make sure the stride is tested against the * loop that envelops the reference the closest). Returns invariant offset on success. */ bool IsUnitStride(const HBasicBlock* context, HInstruction* instruction, HGraph* graph, /*out*/ HInstruction** offset) const; /** * Generates the trip count expression for the given loop. Code is generated in given block * and graph. The expression is guarded by a taken test if needed. Returns the trip count * expression on success or null otherwise. */ HInstruction* GenerateTripCount(const HLoopInformation* loop, HGraph* graph, HBasicBlock* block); private: /* * Enum used in IsConstant() request. */ enum ConstantRequest { kExact, kAtMost, kAtLeast }; /** * Checks if header logic of a loop terminates. If trip count is known (constant) sets * 'is_constant' to true and 'trip_count' to the trip count value. */ bool CheckForFiniteAndConstantProps(const HLoopInformation* loop, /*out*/ bool* is_constant, /*out*/ int64_t* trip_count) const; /** * Returns true if exact or upper/lower bound on the given induction * information is known as a 64-bit constant, which is returned in value. */ bool IsConstant(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, ConstantRequest request, /*out*/ int64_t* value) const; /** Returns whether induction information can be obtained. */ bool HasInductionInfo(const HBasicBlock* context, HInstruction* instruction, /*out*/ const HLoopInformation** loop, /*out*/ HInductionVarAnalysis::InductionInfo** info, /*out*/ HInductionVarAnalysis::InductionInfo** trip) const; bool HasFetchInLoop(HInductionVarAnalysis::InductionInfo* info) const; bool NeedsTripCount(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, /*out*/ int64_t* stride_value) const; bool IsBodyTripCount(HInductionVarAnalysis::InductionInfo* trip) const; bool IsUnsafeTripCount(HInductionVarAnalysis::InductionInfo* trip) const; bool IsWellBehavedTripCount(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* trip) const; Value GetLinear(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HInductionVarAnalysis::InductionInfo* trip, bool is_min) const; Value GetPolynomial(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HInductionVarAnalysis::InductionInfo* trip, bool is_min) const; Value GetGeometric(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HInductionVarAnalysis::InductionInfo* trip, bool is_min) const; Value GetFetch(const HBasicBlock* context, const HLoopInformation* loop, HInstruction* instruction, HInductionVarAnalysis::InductionInfo* trip, bool is_min) const; Value GetVal(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HInductionVarAnalysis::InductionInfo* trip, bool is_min) const; Value GetMul(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info1, HInductionVarAnalysis::InductionInfo* info2, HInductionVarAnalysis::InductionInfo* trip, bool is_min) const; Value GetDiv(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info1, HInductionVarAnalysis::InductionInfo* info2, HInductionVarAnalysis::InductionInfo* trip, bool is_min) const; Value GetRem(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info1, HInductionVarAnalysis::InductionInfo* info2) const; Value GetXor(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info1, HInductionVarAnalysis::InductionInfo* info2) const; Value MulRangeAndConstant(const HBasicBlock* context, const HLoopInformation* loop, int64_t value, HInductionVarAnalysis::InductionInfo* info, HInductionVarAnalysis::InductionInfo* trip, bool is_min) const; Value DivRangeAndConstant(const HBasicBlock* context, const HLoopInformation* loop, int64_t value, HInductionVarAnalysis::InductionInfo* info, HInductionVarAnalysis::InductionInfo* trip, bool is_min) const; Value AddValue(Value v1, Value v2) const; Value SubValue(Value v1, Value v2) const; Value MulValue(Value v1, Value v2) const; Value DivValue(Value v1, Value v2) const; Value MergeVal(Value v1, Value v2, bool is_min) const; /** * Generates code for lower/upper/taken-test or last value in the HIR. Returns true on * success. With values nullptr, the method can be used to determine if code generation * would be successful without generating actual code yet. */ bool GenerateRangeOrLastValue(const HBasicBlock* context, HInstruction* instruction, bool is_last_val, HGraph* graph, HBasicBlock* block, /*out*/ HInstruction** lower, /*out*/ HInstruction** upper, /*out*/ HInstruction** taken_test, /*out*/ int64_t* stride_value, /*out*/ bool* needs_finite_test, /*out*/ bool* needs_taken_test) const; bool GenerateLastValueLinear(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HInductionVarAnalysis::InductionInfo* trip, HGraph* graph, HBasicBlock* block, bool is_min, /*out*/ HInstruction** result, /*inout*/ bool* needs_taken_test) const; bool GenerateLastValuePolynomial(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HInductionVarAnalysis::InductionInfo* trip, HGraph* graph, HBasicBlock* block, /*out*/HInstruction** result) const; bool GenerateLastValueGeometric(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HInductionVarAnalysis::InductionInfo* trip, HGraph* graph, HBasicBlock* block, /*out*/HInstruction** result) const; bool GenerateLastValueWrapAround(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HInductionVarAnalysis::InductionInfo* trip, HGraph* graph, HBasicBlock* block, /*out*/HInstruction** result) const; bool GenerateLastValuePeriodic(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HInductionVarAnalysis::InductionInfo* trip, HGraph* graph, HBasicBlock* block, /*out*/ HInstruction** result, /*inout*/ bool* needs_taken_test) const; bool GenerateCode(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HInductionVarAnalysis::InductionInfo* trip, HGraph* graph, HBasicBlock* block, bool is_min, /*out*/ HInstruction** result, // TODO(solanes): Remove default value when all cases have been assessed. bool allow_potential_overflow = true) const; bool TryGenerateAddWithoutOverflow(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HGraph* graph, /*in*/ HInstruction* opa, /*in*/ HInstruction* opb, /*out*/ HInstruction** result) const; bool TryGenerateSubWithoutOverflow(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HGraph* graph, /*in*/ HInstruction* opa, /*out*/ HInstruction** result) const; // Try to guard the taken test with an HSelect instruction. Returns true if it can generate the // code, or false otherwise. The caller is responsible of updating `needs_taken_test`. bool TryGenerateTakenTest(const HBasicBlock* context, const HLoopInformation* loop, HInductionVarAnalysis::InductionInfo* info, HGraph* graph, HBasicBlock* block, /*inout*/ HInstruction** result, /*inout*/ HInstruction* not_taken_result) const; void ReplaceInduction(HInductionVarAnalysis::InductionInfo* info, HInstruction* fetch, HInstruction* replacement); /** Results of prior induction variable analysis. */ HInductionVarAnalysis* induction_analysis_; /** Instruction at which chasing may stop. */ HInstruction* chase_hint_; friend class HInductionVarAnalysis; friend class InductionVarRangeTest; DISALLOW_COPY_AND_ASSIGN(InductionVarRange); }; } // namespace art #endif // ART_COMPILER_OPTIMIZING_INDUCTION_VAR_RANGE_H_