1 //===- RDFRegisters.h -------------------------------------------*- C++ -*-===//
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
7 //===----------------------------------------------------------------------===//
9 #ifndef LLVM_LIB_TARGET_HEXAGON_RDFREGISTERS_H
10 #define LLVM_LIB_TARGET_HEXAGON_RDFREGISTERS_H
12 #include "llvm/ADT/BitVector.h"
13 #include "llvm/ADT/STLExtras.h"
14 #include "llvm/CodeGen/TargetRegisterInfo.h"
15 #include "llvm/MC/LaneBitmask.h"
24 class MachineFunction
;
29 using RegisterId
= uint32_t;
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>
37 IndexedSet() { Map
.reserve(N
); }
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());
45 uint32_t insert(T Val
) {
47 auto F
= llvm::find(Map
, Val
);
49 return F
- Map
.begin() + 1;
51 return Map
.size(); // Return actual_index + 1.
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;
60 uint32_t size() const { return Map
.size(); }
62 using const_iterator
= typename
std::vector
<T
>::const_iterator
;
64 const_iterator
begin() const { return Map
.begin(); }
65 const_iterator
end() const { return Map
.end(); }
73 LaneBitmask Mask
= LaneBitmask::getNone();
75 RegisterRef() = default;
76 explicit RegisterRef(RegisterId R
, LaneBitmask M
= LaneBitmask::getAll())
77 : Reg(R
), Mask(R
!= 0 ? M
: LaneBitmask::getNone()) {}
79 operator bool() const {
80 return Reg
!= 0 && Mask
.any();
83 bool operator== (const RegisterRef
&RR
) const {
84 return Reg
== RR
.Reg
&& Mask
== RR
.Mask
;
87 bool operator!= (const RegisterRef
&RR
) const {
88 return !operator==(RR
);
91 bool operator< (const RegisterRef
&RR
) const {
92 return Reg
< RR
.Reg
|| (Reg
== RR
.Reg
&& Mask
< RR
.Mask
);
97 struct PhysicalRegisterInfo
{
98 PhysicalRegisterInfo(const TargetRegisterInfo
&tri
,
99 const MachineFunction
&mf
);
101 static bool isRegMaskId(RegisterId R
) {
102 return TargetRegisterInfo::isStackSlot(R
);
105 RegisterId
getRegMaskId(const uint32_t *RM
) const {
106 return TargetRegisterInfo::index2StackSlot(RegMasks
.find(RM
));
109 const uint32_t *getRegMaskBits(RegisterId R
) const {
110 return RegMasks
.get(TargetRegisterInfo::stackSlot2Index(R
));
113 RegisterRef
normalize(RegisterRef RR
) const;
115 bool alias(RegisterRef RA
, RegisterRef RB
) const {
116 if (!isRegMaskId(RA
.Reg
))
117 return !isRegMaskId(RB
.Reg
) ? aliasRR(RA
, RB
) : aliasRM(RA
, RB
);
118 return !isRegMaskId(RB
.Reg
) ? aliasRM(RB
, RA
) : aliasMM(RA
, RB
);
121 std::set
<RegisterId
> getAliasSet(RegisterId Reg
) const;
123 RegisterRef
getRefForUnit(uint32_t U
) const {
124 return RegisterRef(UnitInfos
[U
].Reg
, UnitInfos
[U
].Mask
);
127 const BitVector
&getMaskUnits(RegisterId MaskId
) const {
128 return MaskInfos
[TargetRegisterInfo::stackSlot2Index(MaskId
)].Units
;
131 RegisterRef
mapTo(RegisterRef RR
, unsigned R
) const;
132 const TargetRegisterInfo
&getTRI() const { return TRI
; }
136 const TargetRegisterClass
*RegClass
= nullptr;
146 const TargetRegisterInfo
&TRI
;
147 IndexedSet
<const uint32_t*> RegMasks
;
148 std::vector
<RegInfo
> RegInfos
;
149 std::vector
<UnitInfo
> UnitInfos
;
150 std::vector
<MaskInfo
> MaskInfos
;
152 bool aliasRR(RegisterRef RA
, RegisterRef RB
) const;
153 bool aliasRM(RegisterRef RR
, RegisterRef RM
) const;
154 bool aliasMM(RegisterRef RM
, RegisterRef RN
) const;
157 struct RegisterAggr
{
158 RegisterAggr(const PhysicalRegisterInfo
&pri
)
159 : Units(pri
.getTRI().getNumRegUnits()), PRI(pri
) {}
160 RegisterAggr(const RegisterAggr
&RG
) = default;
162 bool empty() const { return Units
.none(); }
163 bool hasAliasOf(RegisterRef RR
) const;
164 bool hasCoverOf(RegisterRef RR
) const;
166 static bool isCoverOf(RegisterRef RA
, RegisterRef RB
,
167 const PhysicalRegisterInfo
&PRI
) {
168 return RegisterAggr(PRI
).insert(RA
).hasCoverOf(RB
);
171 RegisterAggr
&insert(RegisterRef RR
);
172 RegisterAggr
&insert(const RegisterAggr
&RG
);
173 RegisterAggr
&intersect(RegisterRef RR
);
174 RegisterAggr
&intersect(const RegisterAggr
&RG
);
175 RegisterAggr
&clear(RegisterRef RR
);
176 RegisterAggr
&clear(const RegisterAggr
&RG
);
178 RegisterRef
intersectWith(RegisterRef RR
) const;
179 RegisterRef
clearIn(RegisterRef RR
) const;
180 RegisterRef
makeRegRef() const;
182 void print(raw_ostream
&OS
) const;
185 using MapType
= std::map
<RegisterId
, LaneBitmask
>;
189 MapType::iterator Pos
;
191 const RegisterAggr
*Owner
;
194 rr_iterator(const RegisterAggr
&RG
, bool End
);
196 RegisterRef
operator*() const {
197 return RegisterRef(Pos
->first
, Pos
->second
);
200 rr_iterator
&operator++() {
206 bool operator==(const rr_iterator
&I
) const {
207 assert(Owner
== I
.Owner
);
209 return Index
== I
.Index
;
212 bool operator!=(const rr_iterator
&I
) const {
213 return !(*this == I
);
217 rr_iterator
rr_begin() const {
218 return rr_iterator(*this, false);
220 rr_iterator
rr_end() const {
221 return rr_iterator(*this, true);
226 const PhysicalRegisterInfo
&PRI
;
229 // Optionally print the lane mask, if it is not ~0.
230 struct PrintLaneMaskOpt
{
231 PrintLaneMaskOpt(LaneBitmask M
) : Mask(M
) {}
234 raw_ostream
&operator<< (raw_ostream
&OS
, const PrintLaneMaskOpt
&P
);
236 } // end namespace rdf
238 } // end namespace llvm
240 #endif // LLVM_LIB_TARGET_HEXAGON_RDFREGISTERS_H