Jim Stichnoth | d97c7df | 2014-06-04 11:57:08 -0700 | [diff] [blame] | 1 | //===- subzero/src/IceRegAlloc.h - Linear-scan reg. allocation --*- C++ -*-===// |
| 2 | // |
| 3 | // The Subzero Code Generator |
| 4 | // |
| 5 | // This file is distributed under the University of Illinois Open Source |
| 6 | // License. See LICENSE.TXT for details. |
| 7 | // |
| 8 | //===----------------------------------------------------------------------===// |
Andrew Scull | 9612d32 | 2015-07-06 14:53:25 -0700 | [diff] [blame] | 9 | /// |
| 10 | /// \file |
Andrew Scull | d24cfda | 2015-08-25 10:31:15 -0700 | [diff] [blame] | 11 | /// This file declares the LinearScan data structure used during linear-scan |
| 12 | /// register allocation, which holds the various work queues for the linear-scan |
| 13 | /// algorithm. |
Andrew Scull | 9612d32 | 2015-07-06 14:53:25 -0700 | [diff] [blame] | 14 | /// |
Jim Stichnoth | d97c7df | 2014-06-04 11:57:08 -0700 | [diff] [blame] | 15 | //===----------------------------------------------------------------------===// |
| 16 | |
| 17 | #ifndef SUBZERO_SRC_ICEREGALLOC_H |
| 18 | #define SUBZERO_SRC_ICEREGALLOC_H |
| 19 | |
| 20 | #include "IceDefs.h" |
Andrew Scull | d24cfda | 2015-08-25 10:31:15 -0700 | [diff] [blame] | 21 | #include "IceOperand.h" |
Jim Stichnoth | d97c7df | 2014-06-04 11:57:08 -0700 | [diff] [blame] | 22 | #include "IceTypes.h" |
| 23 | |
| 24 | namespace Ice { |
| 25 | |
Jim Stichnoth | d97c7df | 2014-06-04 11:57:08 -0700 | [diff] [blame] | 26 | class LinearScan { |
Jim Stichnoth | c6ead20 | 2015-02-24 09:30:30 -0800 | [diff] [blame] | 27 | LinearScan() = delete; |
Jim Stichnoth | 5ce0abb | 2014-10-15 10:16:54 -0700 | [diff] [blame] | 28 | LinearScan(const LinearScan &) = delete; |
| 29 | LinearScan &operator=(const LinearScan &) = delete; |
| 30 | |
Jim Stichnoth | d97c7df | 2014-06-04 11:57:08 -0700 | [diff] [blame] | 31 | public: |
Andrew Scull | d24cfda | 2015-08-25 10:31:15 -0700 | [diff] [blame] | 32 | explicit LinearScan(Cfg *Func); |
Jim Stichnoth | 70d0a05 | 2014-11-14 15:53:46 -0800 | [diff] [blame] | 33 | void init(RegAllocKind Kind); |
Jim Stichnoth | e6d2478 | 2014-12-19 05:42:24 -0800 | [diff] [blame] | 34 | void scan(const llvm::SmallBitVector &RegMask, bool Randomized); |
Jim Stichnoth | d97c7df | 2014-06-04 11:57:08 -0700 | [diff] [blame] | 35 | void dump(Cfg *Func) const; |
| 36 | |
Andrew Scull | d24cfda | 2015-08-25 10:31:15 -0700 | [diff] [blame] | 37 | // TODO(stichnot): Statically choose the size based on the target being |
| 38 | // compiled. |
| 39 | static constexpr size_t REGS_SIZE = 32; |
| 40 | |
Jim Stichnoth | d97c7df | 2014-06-04 11:57:08 -0700 | [diff] [blame] | 41 | private: |
Andrew Scull | 8072bae | 2015-09-14 16:01:26 -0700 | [diff] [blame] | 42 | using OrderedRanges = std::vector<Variable *>; |
| 43 | using UnorderedRanges = std::vector<Variable *>; |
Jim Stichnoth | 4ead35a | 2014-12-03 20:30:34 -0800 | [diff] [blame] | 44 | |
Andrew Scull | d24cfda | 2015-08-25 10:31:15 -0700 | [diff] [blame] | 45 | class IterationState { |
| 46 | IterationState(const IterationState &) = delete; |
| 47 | IterationState operator=(const IterationState &) = delete; |
| 48 | |
| 49 | public: |
| 50 | IterationState() = default; |
| 51 | Variable *Cur = nullptr; |
| 52 | Variable *Prefer = nullptr; |
| 53 | int32_t PreferReg = Variable::NoRegister; |
| 54 | bool AllowOverlap = false; |
| 55 | llvm::SmallBitVector RegMask; |
| 56 | llvm::SmallBitVector Free; |
| 57 | llvm::SmallBitVector PrecoloredUnhandledMask; // Note: only used for dumping |
| 58 | llvm::SmallVector<RegWeight, REGS_SIZE> Weights; |
| 59 | }; |
| 60 | |
Jim Stichnoth | 70d0a05 | 2014-11-14 15:53:46 -0800 | [diff] [blame] | 61 | void initForGlobal(); |
| 62 | void initForInfOnly(); |
Andrew Scull | d24cfda | 2015-08-25 10:31:15 -0700 | [diff] [blame] | 63 | /// Move an item from the From set to the To set. From[Index] is pushed onto |
| 64 | /// the end of To[], then the item is efficiently removed from From[] by |
| 65 | /// effectively swapping it with the last item in From[] and then popping it |
| 66 | /// from the back. As such, the caller is best off iterating over From[] in |
| 67 | /// reverse order to avoid the need for special handling of the iterator. |
Jim Stichnoth | 4ead35a | 2014-12-03 20:30:34 -0800 | [diff] [blame] | 68 | void moveItem(UnorderedRanges &From, SizeT Index, UnorderedRanges &To) { |
| 69 | To.push_back(From[Index]); |
| 70 | From[Index] = From.back(); |
| 71 | From.pop_back(); |
| 72 | } |
Jim Stichnoth | 70d0a05 | 2014-11-14 15:53:46 -0800 | [diff] [blame] | 73 | |
Andrew Scull | d24cfda | 2015-08-25 10:31:15 -0700 | [diff] [blame] | 74 | /// \name scan helper functions. |
| 75 | /// @{ |
| 76 | /// Free up a register for infinite-weight Cur by spilling and reloading some |
| 77 | /// register that isn't used during Cur's live range. |
| 78 | void addSpillFill(IterationState &Iter); |
| 79 | /// Check for active ranges that have expired or become inactive. |
| 80 | void handleActiveRangeExpiredOrInactive(const Variable *Cur); |
| 81 | /// Check for inactive ranges that have expired or reactivated. |
| 82 | void handleInactiveRangeExpiredOrReactivated(const Variable *Cur); |
| 83 | void findRegisterPreference(IterationState &Iter); |
| 84 | void filterFreeWithInactiveRanges(IterationState &Iter); |
| 85 | void filterFreeWithPrecoloredRanges(IterationState &Iter); |
| 86 | void allocatePrecoloredRegister(Variable *Cur); |
| 87 | void allocatePreferredRegister(IterationState &Iter); |
| 88 | void allocateFreeRegister(IterationState &Iter); |
| 89 | void handleNoFreeRegisters(IterationState &Iter); |
| 90 | void assignFinalRegisters(const llvm::SmallBitVector &RegMaskFull, |
| 91 | const llvm::SmallBitVector &PreDefinedRegisters, |
| 92 | bool Randomized); |
| 93 | /// @} |
| 94 | |
| 95 | void dumpLiveRangeTrace(const char *Label, const Variable *Item); |
| 96 | |
Jim Stichnoth | d97c7df | 2014-06-04 11:57:08 -0700 | [diff] [blame] | 97 | Cfg *const Func; |
Andrew Scull | d24cfda | 2015-08-25 10:31:15 -0700 | [diff] [blame] | 98 | GlobalContext *const Ctx; |
John Porto | bb0a5fe | 2015-09-04 11:23:41 -0700 | [diff] [blame] | 99 | TargetLowering *const Target; |
Andrew Scull | d24cfda | 2015-08-25 10:31:15 -0700 | [diff] [blame] | 100 | |
Jim Stichnoth | d97c7df | 2014-06-04 11:57:08 -0700 | [diff] [blame] | 101 | OrderedRanges Unhandled; |
Andrew Scull | d24cfda | 2015-08-25 10:31:15 -0700 | [diff] [blame] | 102 | /// UnhandledPrecolored is a subset of Unhandled, specially collected for |
| 103 | /// faster processing. |
Jim Stichnoth | 541ba66 | 2014-10-02 12:58:21 -0700 | [diff] [blame] | 104 | OrderedRanges UnhandledPrecolored; |
Jim Stichnoth | d97c7df | 2014-06-04 11:57:08 -0700 | [diff] [blame] | 105 | UnorderedRanges Active, Inactive, Handled; |
Jim Stichnoth | 87ff3a1 | 2014-11-14 10:27:29 -0800 | [diff] [blame] | 106 | std::vector<InstNumberT> Kills; |
Jim Stichnoth | a3f57b9 | 2015-07-30 12:46:04 -0700 | [diff] [blame] | 107 | RegAllocKind Kind = RAK_Unknown; |
Andrew Scull | d24cfda | 2015-08-25 10:31:15 -0700 | [diff] [blame] | 108 | /// RegUses[I] is the number of live ranges (variables) that register I is |
| 109 | /// currently assigned to. It can be greater than 1 as a result of |
| 110 | /// AllowOverlap inference. |
| 111 | llvm::SmallVector<int32_t, REGS_SIZE> RegUses; |
John Porto | bb0a5fe | 2015-09-04 11:23:41 -0700 | [diff] [blame] | 112 | // TODO(jpp): for some architectures a SmallBitVector might not be big enough. |
| 113 | // Evaluate what the performance impact on those architectures is. |
| 114 | llvm::SmallVector<const llvm::SmallBitVector *, REGS_SIZE> RegAliases; |
Jim Stichnoth | eafb56c | 2015-06-22 10:35:22 -0700 | [diff] [blame] | 115 | bool FindPreference = false; |
| 116 | bool FindOverlap = false; |
Andrew Scull | d24cfda | 2015-08-25 10:31:15 -0700 | [diff] [blame] | 117 | |
| 118 | const bool Verbose; |
Jim Stichnoth | d97c7df | 2014-06-04 11:57:08 -0700 | [diff] [blame] | 119 | }; |
| 120 | |
| 121 | } // end of namespace Ice |
| 122 | |
| 123 | #endif // SUBZERO_SRC_ICEREGALLOC_H |