Details | Last modification | View Log | RSS feed
Rev | Author | Line No. | Line |
---|---|---|---|
14 | pmbaty | 1 | //===- RDFRegisters.h -------------------------------------------*- C++ -*-===// |
2 | // |
||
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
||
4 | // See https://llvm.org/LICENSE.txt for license information. |
||
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
||
6 | // |
||
7 | //===----------------------------------------------------------------------===// |
||
8 | |||
9 | #ifndef LLVM_CODEGEN_RDFREGISTERS_H |
||
10 | #define LLVM_CODEGEN_RDFREGISTERS_H |
||
11 | |||
12 | #include "llvm/ADT/BitVector.h" |
||
13 | #include "llvm/ADT/STLExtras.h" |
||
14 | #include "llvm/CodeGen/TargetRegisterInfo.h" |
||
15 | #include "llvm/MC/LaneBitmask.h" |
||
16 | #include <cassert> |
||
17 | #include <cstdint> |
||
18 | #include <map> |
||
19 | #include <set> |
||
20 | #include <vector> |
||
21 | |||
22 | namespace llvm { |
||
23 | |||
24 | class MachineFunction; |
||
25 | class raw_ostream; |
||
26 | |||
27 | namespace rdf { |
||
28 | |||
29 | using RegisterId = uint32_t; |
||
30 | |||
31 | // Template class for a map translating uint32_t into arbitrary types. |
||
32 | // The map will act like an indexed set: upon insertion of a new object, |
||
33 | // it will automatically assign a new index to it. Index of 0 is treated |
||
34 | // as invalid and is never allocated. |
||
35 | template <typename T, unsigned N = 32> |
||
36 | struct IndexedSet { |
||
37 | IndexedSet() { Map.reserve(N); } |
||
38 | |||
39 | T get(uint32_t Idx) const { |
||
40 | // Index Idx corresponds to Map[Idx-1]. |
||
41 | assert(Idx != 0 && !Map.empty() && Idx-1 < Map.size()); |
||
42 | return Map[Idx-1]; |
||
43 | } |
||
44 | |||
45 | uint32_t insert(T Val) { |
||
46 | // Linear search. |
||
47 | auto F = llvm::find(Map, Val); |
||
48 | if (F != Map.end()) |
||
49 | return F - Map.begin() + 1; |
||
50 | Map.push_back(Val); |
||
51 | return Map.size(); // Return actual_index + 1. |
||
52 | } |
||
53 | |||
54 | uint32_t find(T Val) const { |
||
55 | auto F = llvm::find(Map, Val); |
||
56 | assert(F != Map.end()); |
||
57 | return F - Map.begin() + 1; |
||
58 | } |
||
59 | |||
60 | uint32_t size() const { return Map.size(); } |
||
61 | |||
62 | using const_iterator = typename std::vector<T>::const_iterator; |
||
63 | |||
64 | const_iterator begin() const { return Map.begin(); } |
||
65 | const_iterator end() const { return Map.end(); } |
||
66 | |||
67 | private: |
||
68 | std::vector<T> Map; |
||
69 | }; |
||
70 | |||
71 | struct RegisterRef { |
||
72 | RegisterId Reg = 0; |
||
73 | LaneBitmask Mask = LaneBitmask::getNone(); |
||
74 | |||
75 | RegisterRef() = default; |
||
76 | explicit RegisterRef(RegisterId R, LaneBitmask M = LaneBitmask::getAll()) |
||
77 | : Reg(R), Mask(R != 0 ? M : LaneBitmask::getNone()) {} |
||
78 | |||
79 | operator bool() const { |
||
80 | return Reg != 0 && Mask.any(); |
||
81 | } |
||
82 | |||
83 | bool operator== (const RegisterRef &RR) const { |
||
84 | return Reg == RR.Reg && Mask == RR.Mask; |
||
85 | } |
||
86 | |||
87 | bool operator!= (const RegisterRef &RR) const { |
||
88 | return !operator==(RR); |
||
89 | } |
||
90 | |||
91 | bool operator< (const RegisterRef &RR) const { |
||
92 | return Reg < RR.Reg || (Reg == RR.Reg && Mask < RR.Mask); |
||
93 | } |
||
94 | |||
95 | size_t hash() const { |
||
96 | return std::hash<RegisterId>{}(Reg) ^ |
||
97 | std::hash<LaneBitmask::Type>{}(Mask.getAsInteger()); |
||
98 | } |
||
99 | }; |
||
100 | |||
101 | |||
102 | struct PhysicalRegisterInfo { |
||
103 | PhysicalRegisterInfo(const TargetRegisterInfo &tri, |
||
104 | const MachineFunction &mf); |
||
105 | |||
106 | static bool isRegMaskId(RegisterId R) { |
||
107 | return Register::isStackSlot(R); |
||
108 | } |
||
109 | |||
110 | RegisterId getRegMaskId(const uint32_t *RM) const { |
||
111 | return Register::index2StackSlot(RegMasks.find(RM)); |
||
112 | } |
||
113 | |||
114 | const uint32_t *getRegMaskBits(RegisterId R) const { |
||
115 | return RegMasks.get(Register::stackSlot2Index(R)); |
||
116 | } |
||
117 | |||
118 | bool alias(RegisterRef RA, RegisterRef RB) const { |
||
119 | if (!isRegMaskId(RA.Reg)) |
||
120 | return !isRegMaskId(RB.Reg) ? aliasRR(RA, RB) : aliasRM(RA, RB); |
||
121 | return !isRegMaskId(RB.Reg) ? aliasRM(RB, RA) : aliasMM(RA, RB); |
||
122 | } |
||
123 | |||
124 | std::set<RegisterId> getAliasSet(RegisterId Reg) const; |
||
125 | |||
126 | RegisterRef getRefForUnit(uint32_t U) const { |
||
127 | return RegisterRef(UnitInfos[U].Reg, UnitInfos[U].Mask); |
||
128 | } |
||
129 | |||
130 | const BitVector &getMaskUnits(RegisterId MaskId) const { |
||
131 | return MaskInfos[Register::stackSlot2Index(MaskId)].Units; |
||
132 | } |
||
133 | |||
134 | const BitVector &getUnitAliases(uint32_t U) const { |
||
135 | return AliasInfos[U].Regs; |
||
136 | } |
||
137 | |||
138 | RegisterRef mapTo(RegisterRef RR, unsigned R) const; |
||
139 | const TargetRegisterInfo &getTRI() const { return TRI; } |
||
140 | |||
141 | private: |
||
142 | struct RegInfo { |
||
143 | const TargetRegisterClass *RegClass = nullptr; |
||
144 | }; |
||
145 | struct UnitInfo { |
||
146 | RegisterId Reg = 0; |
||
147 | LaneBitmask Mask; |
||
148 | }; |
||
149 | struct MaskInfo { |
||
150 | BitVector Units; |
||
151 | }; |
||
152 | struct AliasInfo { |
||
153 | BitVector Regs; |
||
154 | }; |
||
155 | |||
156 | const TargetRegisterInfo &TRI; |
||
157 | IndexedSet<const uint32_t*> RegMasks; |
||
158 | std::vector<RegInfo> RegInfos; |
||
159 | std::vector<UnitInfo> UnitInfos; |
||
160 | std::vector<MaskInfo> MaskInfos; |
||
161 | std::vector<AliasInfo> AliasInfos; |
||
162 | |||
163 | bool aliasRR(RegisterRef RA, RegisterRef RB) const; |
||
164 | bool aliasRM(RegisterRef RR, RegisterRef RM) const; |
||
165 | bool aliasMM(RegisterRef RM, RegisterRef RN) const; |
||
166 | }; |
||
167 | |||
168 | struct RegisterAggr { |
||
169 | RegisterAggr(const PhysicalRegisterInfo &pri) |
||
170 | : Units(pri.getTRI().getNumRegUnits()), PRI(pri) {} |
||
171 | RegisterAggr(const RegisterAggr &RG) = default; |
||
172 | |||
173 | unsigned count() const { return Units.count(); } |
||
174 | bool empty() const { return Units.none(); } |
||
175 | bool hasAliasOf(RegisterRef RR) const; |
||
176 | bool hasCoverOf(RegisterRef RR) const; |
||
177 | |||
178 | bool operator==(const RegisterAggr &A) const { |
||
179 | return DenseMapInfo<BitVector>::isEqual(Units, A.Units); |
||
180 | } |
||
181 | |||
182 | static bool isCoverOf(RegisterRef RA, RegisterRef RB, |
||
183 | const PhysicalRegisterInfo &PRI) { |
||
184 | return RegisterAggr(PRI).insert(RA).hasCoverOf(RB); |
||
185 | } |
||
186 | |||
187 | RegisterAggr &insert(RegisterRef RR); |
||
188 | RegisterAggr &insert(const RegisterAggr &RG); |
||
189 | RegisterAggr &intersect(RegisterRef RR); |
||
190 | RegisterAggr &intersect(const RegisterAggr &RG); |
||
191 | RegisterAggr &clear(RegisterRef RR); |
||
192 | RegisterAggr &clear(const RegisterAggr &RG); |
||
193 | |||
194 | RegisterRef intersectWith(RegisterRef RR) const; |
||
195 | RegisterRef clearIn(RegisterRef RR) const; |
||
196 | RegisterRef makeRegRef() const; |
||
197 | |||
198 | size_t hash() const { |
||
199 | return DenseMapInfo<BitVector>::getHashValue(Units); |
||
200 | } |
||
201 | |||
202 | void print(raw_ostream &OS) const; |
||
203 | |||
204 | struct rr_iterator { |
||
205 | using MapType = std::map<RegisterId, LaneBitmask>; |
||
206 | |||
207 | private: |
||
208 | MapType Masks; |
||
209 | MapType::iterator Pos; |
||
210 | unsigned Index; |
||
211 | const RegisterAggr *Owner; |
||
212 | |||
213 | public: |
||
214 | rr_iterator(const RegisterAggr &RG, bool End); |
||
215 | |||
216 | RegisterRef operator*() const { |
||
217 | return RegisterRef(Pos->first, Pos->second); |
||
218 | } |
||
219 | |||
220 | rr_iterator &operator++() { |
||
221 | ++Pos; |
||
222 | ++Index; |
||
223 | return *this; |
||
224 | } |
||
225 | |||
226 | bool operator==(const rr_iterator &I) const { |
||
227 | assert(Owner == I.Owner); |
||
228 | (void)Owner; |
||
229 | return Index == I.Index; |
||
230 | } |
||
231 | |||
232 | bool operator!=(const rr_iterator &I) const { |
||
233 | return !(*this == I); |
||
234 | } |
||
235 | }; |
||
236 | |||
237 | rr_iterator rr_begin() const { |
||
238 | return rr_iterator(*this, false); |
||
239 | } |
||
240 | rr_iterator rr_end() const { |
||
241 | return rr_iterator(*this, true); |
||
242 | } |
||
243 | |||
244 | private: |
||
245 | BitVector Units; |
||
246 | const PhysicalRegisterInfo &PRI; |
||
247 | }; |
||
248 | |||
249 | // Optionally print the lane mask, if it is not ~0. |
||
250 | struct PrintLaneMaskOpt { |
||
251 | PrintLaneMaskOpt(LaneBitmask M) : Mask(M) {} |
||
252 | LaneBitmask Mask; |
||
253 | }; |
||
254 | raw_ostream &operator<< (raw_ostream &OS, const PrintLaneMaskOpt &P); |
||
255 | |||
256 | raw_ostream &operator<< (raw_ostream &OS, const RegisterAggr &A); |
||
257 | } // end namespace rdf |
||
258 | |||
259 | } // end namespace llvm |
||
260 | |||
261 | namespace std { |
||
262 | template <> struct hash<llvm::rdf::RegisterRef> { |
||
263 | size_t operator()(llvm::rdf::RegisterRef A) const { |
||
264 | return A.hash(); |
||
265 | } |
||
266 | }; |
||
267 | template <> struct hash<llvm::rdf::RegisterAggr> { |
||
268 | size_t operator()(const llvm::rdf::RegisterAggr &A) const { |
||
269 | return A.hash(); |
||
270 | } |
||
271 | }; |
||
272 | template <> struct equal_to<llvm::rdf::RegisterAggr> { |
||
273 | bool operator()(const llvm::rdf::RegisterAggr &A, |
||
274 | const llvm::rdf::RegisterAggr &B) const { |
||
275 | return A == B; |
||
276 | } |
||
277 | }; |
||
278 | } |
||
279 | #endif // LLVM_CODEGEN_RDFREGISTERS_H |