| //===- RegAllocEvictionAdvisor.h - Interference resolution ------*- C++ -*-===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #ifndef LLVM_CODEGEN_REGALLOCEVICTIONADVISOR_H |
| #define LLVM_CODEGEN_REGALLOCEVICTIONADVISOR_H |
| |
| #include "llvm/ADT/ArrayRef.h" |
| #include "llvm/ADT/SmallSet.h" |
| #include "llvm/ADT/StringRef.h" |
| #include "llvm/CodeGen/Register.h" |
| #include "llvm/Config/llvm-config.h" |
| #include "llvm/MC/MCRegister.h" |
| #include "llvm/Pass.h" |
| |
| namespace llvm { |
| class AllocationOrder; |
| class LiveInterval; |
| class LiveIntervals; |
| class LiveRegMatrix; |
| class MachineFunction; |
| class MachineRegisterInfo; |
| class RegisterClassInfo; |
| class TargetRegisterInfo; |
| class VirtRegMap; |
| |
| using SmallVirtRegSet = SmallSet<Register, 16>; |
| |
| // Live ranges pass through a number of stages as we try to allocate them. |
| // Some of the stages may also create new live ranges: |
| // |
| // - Region splitting. |
| // - Per-block splitting. |
| // - Local splitting. |
| // - Spilling. |
| // |
| // Ranges produced by one of the stages skip the previous stages when they are |
| // dequeued. This improves performance because we can skip interference checks |
| // that are unlikely to give any results. It also guarantees that the live |
| // range splitting algorithm terminates, something that is otherwise hard to |
| // ensure. |
| enum LiveRangeStage { |
| /// Newly created live range that has never been queued. |
| RS_New, |
| |
| /// Only attempt assignment and eviction. Then requeue as RS_Split. |
| RS_Assign, |
| |
| /// Attempt live range splitting if assignment is impossible. |
| RS_Split, |
| |
| /// Attempt more aggressive live range splitting that is guaranteed to make |
| /// progress. This is used for split products that may not be making |
| /// progress. |
| RS_Split2, |
| |
| /// Live range will be spilled. No more splitting will be attempted. |
| RS_Spill, |
| |
| /// Live range is in memory. Because of other evictions, it might get moved |
| /// in a register in the end. |
| RS_Memory, |
| |
| /// There is nothing more we can do to this live range. Abort compilation |
| /// if it can't be assigned. |
| RS_Done |
| }; |
| |
| /// Cost of evicting interference - used by default advisor, and the eviction |
| /// chain heuristic in RegAllocGreedy. |
| // FIXME: this can be probably made an implementation detail of the default |
| // advisor, if the eviction chain logic can be refactored. |
| struct EvictionCost { |
| unsigned BrokenHints = 0; ///< Total number of broken hints. |
| float MaxWeight = 0; ///< Maximum spill weight evicted. |
| |
| EvictionCost() = default; |
| |
| bool isMax() const { return BrokenHints == ~0u; } |
| |
| void setMax() { BrokenHints = ~0u; } |
| |
| void setBrokenHints(unsigned NHints) { BrokenHints = NHints; } |
| |
| bool operator<(const EvictionCost &O) const { |
| return std::tie(BrokenHints, MaxWeight) < |
| std::tie(O.BrokenHints, O.MaxWeight); |
| } |
| }; |
| |
| /// Interface to the eviction advisor, which is responsible for making a |
| /// decision as to which live ranges should be evicted (if any). |
| class RAGreedy; |
| class RegAllocEvictionAdvisor { |
| public: |
| RegAllocEvictionAdvisor(const RegAllocEvictionAdvisor &) = delete; |
| RegAllocEvictionAdvisor(RegAllocEvictionAdvisor &&) = delete; |
| virtual ~RegAllocEvictionAdvisor() = default; |
| |
| /// Find a physical register that can be freed by evicting the FixedRegisters, |
| /// or return NoRegister. The eviction decision is assumed to be correct (i.e. |
| /// no fixed live ranges are evicted) and profitable. |
| virtual MCRegister tryFindEvictionCandidate( |
| const LiveInterval &VirtReg, const AllocationOrder &Order, |
| uint8_t CostPerUseLimit, const SmallVirtRegSet &FixedRegisters) const = 0; |
| |
| /// Find out if we can evict the live ranges occupying the given PhysReg, |
| /// which is a hint (preferred register) for VirtReg. |
| virtual bool |
| canEvictHintInterference(const LiveInterval &VirtReg, MCRegister PhysReg, |
| const SmallVirtRegSet &FixedRegisters) const = 0; |
| |
| /// Returns true if the given \p PhysReg is a callee saved register and has |
| /// not been used for allocation yet. |
| bool isUnusedCalleeSavedReg(MCRegister PhysReg) const; |
| |
| protected: |
| RegAllocEvictionAdvisor(const MachineFunction &MF, const RAGreedy &RA); |
| |
| Register canReassign(const LiveInterval &VirtReg, Register PrevReg) const; |
| |
| // Get the upper limit of elements in the given Order we need to analize. |
| // TODO: is this heuristic, we could consider learning it. |
| std::optional<unsigned> getOrderLimit(const LiveInterval &VirtReg, |
| const AllocationOrder &Order, |
| unsigned CostPerUseLimit) const; |
| |
| // Determine if it's worth trying to allocate this reg, given the |
| // CostPerUseLimit |
| // TODO: this is a heuristic component we could consider learning, too. |
| bool canAllocatePhysReg(unsigned CostPerUseLimit, MCRegister PhysReg) const; |
| |
| const MachineFunction &MF; |
| const RAGreedy &RA; |
| LiveRegMatrix *const Matrix; |
| LiveIntervals *const LIS; |
| VirtRegMap *const VRM; |
| MachineRegisterInfo *const MRI; |
| const TargetRegisterInfo *const TRI; |
| const RegisterClassInfo &RegClassInfo; |
| const ArrayRef<uint8_t> RegCosts; |
| |
| /// Run or not the local reassignment heuristic. This information is |
| /// obtained from the TargetSubtargetInfo. |
| const bool EnableLocalReassign; |
| }; |
| |
| /// ImmutableAnalysis abstraction for fetching the Eviction Advisor. We model it |
| /// as an analysis to decouple the user from the implementation insofar as |
| /// dependencies on other analyses goes. The motivation for it being an |
| /// immutable pass is twofold: |
| /// - in the ML implementation case, the evaluator is stateless but (especially |
| /// in the development mode) expensive to set up. With an immutable pass, we set |
| /// it up once. |
| /// - in the 'development' mode ML case, we want to capture the training log |
| /// during allocation (this is a log of features encountered and decisions |
| /// made), and then measure a score, potentially a few steps after allocation |
| /// completes. So we need the properties of an immutable pass to keep the logger |
| /// state around until we can make that measurement. |
| /// |
| /// Because we need to offer additional services in 'development' mode, the |
| /// implementations of this analysis need to implement RTTI support. |
| class RegAllocEvictionAdvisorAnalysis : public ImmutablePass { |
| public: |
| enum class AdvisorMode : int { Default, Release, Development }; |
| |
| RegAllocEvictionAdvisorAnalysis(AdvisorMode Mode) |
| : ImmutablePass(ID), Mode(Mode){}; |
| static char ID; |
| |
| /// Get an advisor for the given context (i.e. machine function, etc) |
| virtual std::unique_ptr<RegAllocEvictionAdvisor> |
| getAdvisor(const MachineFunction &MF, const RAGreedy &RA) = 0; |
| AdvisorMode getAdvisorMode() const { return Mode; } |
| virtual void logRewardIfNeeded(const MachineFunction &MF, |
| llvm::function_ref<float()> GetReward){}; |
| |
| protected: |
| // This analysis preserves everything, and subclasses may have additional |
| // requirements. |
| void getAnalysisUsage(AnalysisUsage &AU) const override { |
| AU.setPreservesAll(); |
| } |
| |
| private: |
| StringRef getPassName() const override; |
| const AdvisorMode Mode; |
| }; |
| |
| /// Specialization for the API used by the analysis infrastructure to create |
| /// an instance of the eviction advisor. |
| template <> Pass *callDefaultCtor<RegAllocEvictionAdvisorAnalysis>(); |
| |
| RegAllocEvictionAdvisorAnalysis *createReleaseModeAdvisor(); |
| |
| RegAllocEvictionAdvisorAnalysis *createDevelopmentModeAdvisor(); |
| |
| // TODO: move to RegAllocEvictionAdvisor.cpp when we move implementation |
| // out of RegAllocGreedy.cpp |
| class DefaultEvictionAdvisor : public RegAllocEvictionAdvisor { |
| public: |
| DefaultEvictionAdvisor(const MachineFunction &MF, const RAGreedy &RA) |
| : RegAllocEvictionAdvisor(MF, RA) {} |
| |
| private: |
| MCRegister tryFindEvictionCandidate(const LiveInterval &, |
| const AllocationOrder &, uint8_t, |
| const SmallVirtRegSet &) const override; |
| bool canEvictHintInterference(const LiveInterval &, MCRegister, |
| const SmallVirtRegSet &) const override; |
| bool canEvictInterferenceBasedOnCost(const LiveInterval &, MCRegister, bool, |
| EvictionCost &, |
| const SmallVirtRegSet &) const; |
| bool shouldEvict(const LiveInterval &A, bool, const LiveInterval &B, |
| bool) const; |
| }; |
| } // namespace llvm |
| |
| #endif // LLVM_CODEGEN_REGALLOCEVICTIONADVISOR_H |