blob: b3986a32b489ab26712fadf276d31818a2a12539 [file] [log] [blame]
Jim Stichnothd97c7df2014-06-04 11:57:08 -07001//===- 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 Scull9612d322015-07-06 14:53:25 -07009///
10/// \file
Andrew Sculld24cfda2015-08-25 10:31:15 -070011/// 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 Scull9612d322015-07-06 14:53:25 -070014///
Jim Stichnothd97c7df2014-06-04 11:57:08 -070015//===----------------------------------------------------------------------===//
16
17#ifndef SUBZERO_SRC_ICEREGALLOC_H
18#define SUBZERO_SRC_ICEREGALLOC_H
19
20#include "IceDefs.h"
Andrew Sculld24cfda2015-08-25 10:31:15 -070021#include "IceOperand.h"
Jim Stichnothd97c7df2014-06-04 11:57:08 -070022#include "IceTypes.h"
23
24namespace Ice {
25
Jim Stichnothd97c7df2014-06-04 11:57:08 -070026class LinearScan {
Jim Stichnothc6ead202015-02-24 09:30:30 -080027 LinearScan() = delete;
Jim Stichnoth5ce0abb2014-10-15 10:16:54 -070028 LinearScan(const LinearScan &) = delete;
29 LinearScan &operator=(const LinearScan &) = delete;
30
Jim Stichnothd97c7df2014-06-04 11:57:08 -070031public:
Andrew Sculld24cfda2015-08-25 10:31:15 -070032 explicit LinearScan(Cfg *Func);
Jim Stichnoth70d0a052014-11-14 15:53:46 -080033 void init(RegAllocKind Kind);
Jim Stichnothe6d24782014-12-19 05:42:24 -080034 void scan(const llvm::SmallBitVector &RegMask, bool Randomized);
Jim Stichnothd97c7df2014-06-04 11:57:08 -070035 void dump(Cfg *Func) const;
36
Andrew Sculld24cfda2015-08-25 10:31:15 -070037 // TODO(stichnot): Statically choose the size based on the target being
38 // compiled.
39 static constexpr size_t REGS_SIZE = 32;
40
Jim Stichnothd97c7df2014-06-04 11:57:08 -070041private:
Andrew Scull8072bae2015-09-14 16:01:26 -070042 using OrderedRanges = std::vector<Variable *>;
43 using UnorderedRanges = std::vector<Variable *>;
Jim Stichnoth4ead35a2014-12-03 20:30:34 -080044
Andrew Sculld24cfda2015-08-25 10:31:15 -070045 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 Stichnoth70d0a052014-11-14 15:53:46 -080061 void initForGlobal();
62 void initForInfOnly();
Andrew Sculld24cfda2015-08-25 10:31:15 -070063 /// 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 Stichnoth4ead35a2014-12-03 20:30:34 -080068 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 Stichnoth70d0a052014-11-14 15:53:46 -080073
Andrew Sculld24cfda2015-08-25 10:31:15 -070074 /// \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 Stichnothd97c7df2014-06-04 11:57:08 -070097 Cfg *const Func;
Andrew Sculld24cfda2015-08-25 10:31:15 -070098 GlobalContext *const Ctx;
John Portobb0a5fe2015-09-04 11:23:41 -070099 TargetLowering *const Target;
Andrew Sculld24cfda2015-08-25 10:31:15 -0700100
Jim Stichnothd97c7df2014-06-04 11:57:08 -0700101 OrderedRanges Unhandled;
Andrew Sculld24cfda2015-08-25 10:31:15 -0700102 /// UnhandledPrecolored is a subset of Unhandled, specially collected for
103 /// faster processing.
Jim Stichnoth541ba662014-10-02 12:58:21 -0700104 OrderedRanges UnhandledPrecolored;
Jim Stichnothd97c7df2014-06-04 11:57:08 -0700105 UnorderedRanges Active, Inactive, Handled;
Jim Stichnoth87ff3a12014-11-14 10:27:29 -0800106 std::vector<InstNumberT> Kills;
Jim Stichnotha3f57b92015-07-30 12:46:04 -0700107 RegAllocKind Kind = RAK_Unknown;
Andrew Sculld24cfda2015-08-25 10:31:15 -0700108 /// 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 Portobb0a5fe2015-09-04 11:23:41 -0700112 // 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 Stichnotheafb56c2015-06-22 10:35:22 -0700115 bool FindPreference = false;
116 bool FindOverlap = false;
Andrew Sculld24cfda2015-08-25 10:31:15 -0700117
118 const bool Verbose;
Jim Stichnothd97c7df2014-06-04 11:57:08 -0700119};
120
121} // end of namespace Ice
122
123#endif // SUBZERO_SRC_ICEREGALLOC_H