LLVM 23.0.0git
VecUtils.h
Go to the documentation of this file.
1//===- VecUtils.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// Collector for SandboxVectorizer related convenience functions that don't
10// belong in other classes.
11
12#ifndef LLVM_TRANSFORMS_VECTORIZE_SANDBOXVECTORIZER_VECUTILS_H
13#define LLVM_TRANSFORMS_VECTORIZE_SANDBOXVECTORIZER_VECUTILS_H
14
16#include "llvm/IR/DataLayout.h"
17#include "llvm/SandboxIR/Type.h"
20#include <iterator>
21
22namespace llvm {
23/// Traits for DenseMap.
24template <> struct DenseMapInfo<SmallVector<sandboxir::Value *>> {
25 static unsigned getHashValue(const SmallVector<sandboxir::Value *> &Vec) {
26 return hash_combine_range(Vec);
27 }
30 return Vec1 == Vec2;
31 }
32};
33
34namespace sandboxir {
35
36class VecUtils {
37public:
38 /// \Returns the number of elements in \p Ty. That is the number of lanes if a
39 /// fixed vector or 1 if scalar. ScalableVectors have unknown size and
40 /// therefore are unsupported.
41 static int getNumElements(Type *Ty) {
43 return Ty->isVectorTy() ? cast<FixedVectorType>(Ty)->getNumElements() : 1;
44 }
45 /// Returns \p Ty if scalar or its element type if vector.
46 static Type *getElementType(Type *Ty) {
47 return Ty->isVectorTy() ? cast<FixedVectorType>(Ty)->getElementType() : Ty;
48 }
49
50 /// \Returns true if \p I1 and \p I2 are load/stores accessing consecutive
51 /// memory addresses.
52 template <typename LoadOrStoreT>
53 static bool areConsecutive(LoadOrStoreT *I1, LoadOrStoreT *I2,
54 ScalarEvolution &SE, const DataLayout &DL) {
55 static_assert(std::is_same<LoadOrStoreT, LoadInst>::value ||
56 std::is_same<LoadOrStoreT, StoreInst>::value,
57 "Expected Load or Store!");
58 auto Diff = Utils::getPointerDiffInBytes(I1, I2, SE);
59 if (!Diff)
60 return false;
61 int ElmBytes = Utils::getNumBits(I1) / 8;
62 return *Diff == ElmBytes;
63 }
64
65 template <typename LoadOrStoreT, typename ValT>
67 const DataLayout &DL) {
68 static_assert(std::is_same<LoadOrStoreT, LoadInst>::value ||
69 std::is_same<LoadOrStoreT, StoreInst>::value,
70 "Expected Load or Store!");
71 assert(isa<LoadOrStoreT>(Bndl[0]) && "Expected Load or Store!");
72 auto *LastLS = cast<LoadOrStoreT>(Bndl[0]);
73 for (Value *V : drop_begin(Bndl)) {
75 "Unimplemented: we only support StoreInst!");
76 auto *LS = cast<LoadOrStoreT>(V);
77 if (!VecUtils::areConsecutive(LastLS, LS, SE, DL))
78 return false;
79 LastLS = LS;
80 }
81 return true;
82 }
83
84 /// \Returns the number of vector lanes of \p Ty or 1 if not a vector.
85 /// NOTE: It asserts that \p Ty is a fixed vector type.
86 static unsigned getNumLanes(Type *Ty) {
87 assert(!isa<ScalableVectorType>(Ty) && "Expect scalar or fixed vector");
88 if (auto *FixedVecTy = dyn_cast<FixedVectorType>(Ty))
89 return FixedVecTy->getNumElements();
90 return 1u;
91 }
92
93 /// \Returns the expected vector lanes of \p V or 1 if not a vector.
94 /// NOTE: It asserts that \p V is a fixed vector.
95 static unsigned getNumLanes(Value *V) {
97 }
98
99 /// \Returns the total number of lanes across all values in \p Bndl.
100 static unsigned getNumLanes(ArrayRef<Value *> Bndl) {
101 unsigned Lanes = 0;
102 for (Value *V : Bndl)
103 Lanes += getNumLanes(V);
104 return Lanes;
105 }
106
107 /// \Returns <NumElts x ElemTy>.
108 /// It works for both scalar and vector \p ElemTy.
109 static Type *getWideType(Type *ElemTy, unsigned NumElts) {
110 if (ElemTy->isVectorTy()) {
111 auto *VecTy = cast<FixedVectorType>(ElemTy);
112 ElemTy = VecTy->getElementType();
113 NumElts = VecTy->getNumElements() * NumElts;
114 }
115 return FixedVectorType::get(ElemTy, NumElts);
116 }
117 /// \Returns the combined vector type for \p Bndl, even when the element types
118 /// differ. For example: i8,i8,i16 will return <4 x i8>. \Returns null if
119 /// types are of mixed float/integer types.
121 const DataLayout &DL) {
122 assert(!Bndl.empty() && "Expected non-empty Bndl!");
123 unsigned TotalBits = 0;
124 unsigned MinElmBits = std::numeric_limits<unsigned>::max();
125 Type *MinElmTy = nullptr;
126 for (auto [Idx, V] : enumerate(Bndl)) {
128
129 unsigned ElmBits = Utils::getNumBits(ElmTy, DL);
130 TotalBits += ElmBits * VecUtils::getNumLanes(V);
131 if (ElmBits < MinElmBits) {
132 MinElmBits = ElmBits;
133 MinElmTy = ElmTy;
134 }
135 }
136 unsigned NumElms = TotalBits / MinElmBits;
137 return FixedVectorType::get(MinElmTy, NumElms);
138 }
139 /// \Returns the instruction in \p Instrs that is lowest in the BB. Expects
140 /// that all instructions are in the same BB.
142 Instruction *LowestI = Instrs.front();
143 for (auto *I : drop_begin(Instrs)) {
144 if (LowestI->comesBefore(I))
145 LowestI = I;
146 }
147 return LowestI;
148 }
149 /// \Returns the instruction in \p Instrs that is highest in the BB. Expects
150 /// that all instructions are in the same BB.
152 Instruction *HighestI = Instrs.front();
153 for (auto *I : drop_begin(Instrs)) {
154 if (I->comesBefore(HighestI))
155 HighestI = I;
156 }
157 return HighestI;
158 }
159 /// \Returns the lowest instruction in \p Vals, or nullptr if no instructions
160 /// are found. Skips instructions not in \p BB.
162 // Find the first Instruction in Vals that is also in `BB`.
163 auto It = find_if(Vals, [BB](Value *V) {
164 return isa<Instruction>(V) && cast<Instruction>(V)->getParent() == BB;
165 });
166 // If we couldn't find an instruction return nullptr.
167 if (It == Vals.end())
168 return nullptr;
169 Instruction *FirstI = cast<Instruction>(*It);
170 // Now look for the lowest instruction in Vals starting from one position
171 // after FirstI.
172 Instruction *LowestI = FirstI;
173 for (auto *V : make_range(std::next(It), Vals.end())) {
174 auto *I = dyn_cast<Instruction>(V);
175 // Skip non-instructions.
176 if (I == nullptr)
177 continue;
178 // Skips instructions not in \p BB.
179 if (I->getParent() != BB)
180 continue;
181 // If `LowestI` comes before `I` then `I` is the new lowest.
182 if (LowestI->comesBefore(I))
183 LowestI = I;
184 }
185 return LowestI;
186 }
187
188 /// If \p I is not a PHI it returns it. Else it walks down the instruction
189 /// chain looking for the last PHI and returns it. \Returns nullptr if \p I is
190 /// nullptr.
192 Instruction *LastI = I;
193 while (I != nullptr && isa<PHINode>(I)) {
194 LastI = I;
195 I = I->getNextNode();
196 }
197 return LastI;
198 }
199
200 /// If all values in \p Bndl are of the same scalar type then return it,
201 /// otherwise return nullptr.
203 Value *V0 = Bndl[0];
204 Type *Ty0 = Utils::getExpectedType(V0);
205 Type *ScalarTy = VecUtils::getElementType(Ty0);
206 for (auto *V : drop_begin(Bndl)) {
208 Type *NScalarTy = VecUtils::getElementType(NTy);
209 if (NScalarTy != ScalarTy)
210 return nullptr;
211 }
212 return ScalarTy;
213 }
214
215 /// Similar to tryGetCommonScalarType() but will assert that there is a common
216 /// type. So this is faster in release builds as it won't iterate through the
217 /// values.
219 Value *V0 = Bndl[0];
220 Type *Ty0 = Utils::getExpectedType(V0);
221 Type *ScalarTy = VecUtils::getElementType(Ty0);
222 assert(tryGetCommonScalarType(Bndl) && "Expected common scalar type!");
223 return ScalarTy;
224 }
225 /// \Returns the first integer power of 2 that is <= Num.
226 LLVM_ABI static unsigned getFloorPowerOf2(unsigned Num);
227
228 /// Helper struct for `matchPack()`. Describes the instructions and operands
229 /// of a pack pattern.
230 struct PackPattern {
231 /// The insertelement instructions that form the pack pattern in bottom-up
232 /// order, i.e., the first instruction in `Instrs` is the bottom-most
233 /// InsertElement instruction of the pack pattern.
234 /// For example in this simple pack pattern:
235 /// %Pack0 = insertelement <2 x i8> poison, i8 %v0, i64 0
236 /// %Pack1 = insertelement <2 x i8> %Pack0, i8 %v1, i64 1
237 /// this is [ %Pack1, %Pack0 ].
239 /// The "external" operands of the pack pattern, i.e., the values that get
240 /// packed into a vector, skipping the ones in `Instrs`. The operands are in
241 /// bottom-up order, starting from the operands of the bottom-most insert.
242 /// So in our example this would be [ %v1, %v0 ].
244 };
245
246 /// If \p I is the last instruction of a pack pattern (i.e., an InsertElement
247 /// into a vector), then this function returns the instructions in the pack
248 /// and the operands in the pack, else returns nullopt.
249 /// Here is an example of a matched pattern:
250 /// %PackA0 = insertelement <2 x i8> poison, i8 %v0, i64 0
251 /// %PackA1 = insertelement <2 x i8> %PackA0, i8 %v1, i64 1
252 /// TODO: this currently detects only simple canonicalized patterns.
253 static std::optional<PackPattern> matchPack(Instruction *I) {
254 // TODO: Support vector pack patterns.
255 // TODO: Support out-of-order inserts.
256
257 // Early return if `I` is not an Insert.
259 return std::nullopt;
260 auto *BB0 = I->getParent();
261 // The pack contains as many instrs as the lanes of the bottom-most Insert
262 unsigned ExpectedNumInserts = VecUtils::getNumLanes(I);
263 assert(ExpectedNumInserts >= 2 && "Expected at least 2 inserts!");
265 Pack.Operands.resize(ExpectedNumInserts);
266 // Collect the inserts by walking up the use-def chain.
267 Instruction *InsertI = I;
268 for (auto ExpectedLane : reverse(seq<unsigned>(ExpectedNumInserts))) {
269 if (InsertI == nullptr)
270 return std::nullopt;
271 if (InsertI->getParent() != BB0)
272 return std::nullopt;
273 // Check the lane.
274 auto *LaneC = dyn_cast<ConstantInt>(InsertI->getOperand(2));
275 if (LaneC == nullptr || LaneC->getSExtValue() != ExpectedLane)
276 return std::nullopt;
277 Pack.Instrs.push_back(InsertI);
278 Pack.Operands[ExpectedLane] = InsertI->getOperand(1);
279
280 Value *Op = InsertI->getOperand(0);
281 if (ExpectedLane == 0) {
282 // Check the topmost insert. The operand should be a Poison.
283 if (!isa<PoisonValue>(Op))
284 return std::nullopt;
285 } else {
287 }
288 }
289 return Pack;
290 }
291
292 /// Emits the necessary instruction sequence to extract element of type \p
293 /// ExtrTy at \p Lane from \p FromVec. Emits instructions before \p WhereIt.
294 /// Returns the extracted value.
295 /// Note: This handles both vectors and scalars. In the vector case it
296 /// extracts an N-wide element (with N dictated by \p ExtrTy).
297 static Value *unpack(Value *FromVec, Type *ExtrTy, unsigned Lane,
298 BasicBlock::iterator WhereIt) {
299 assert(isa<FixedVectorType>(FromVec->getType()) && "Expected vector!");
300 auto &Ctx = FromVec->getContext();
301 if (!ExtrTy->isVectorTy()) {
302 // For scalar elements we emit a single ExtractElementInst.
303 assert(Lane <
304 cast<FixedVectorType>(FromVec->getType())->getNumElements() &&
305 "Out of bounds!");
306 assert(ExtrTy ==
307 cast<FixedVectorType>(FromVec->getType())->getElementType() &&
308 "Expected same element type!");
309 Constant *ExtractLaneC =
311 // Note: This may be folded into a Constant if FromVec is a Constant.
312 return ExtractElementInst::create(FromVec, ExtractLaneC, WhereIt, Ctx,
313 "Unpack");
314 }
315 // For vector elements we emit a shuffle.
316 // For example, extracting lanes 2 and 3 of a <4 x i32> vector %vec:
317 // shufflevector <4 x i32> %vec, <4 x i32> poison, <2 x i32> <i32 2, i32 3>
318 auto *VecTy = cast<FixedVectorType>(FromVec->getType());
319 auto *ExtrVecTy = cast<FixedVectorType>(ExtrTy);
320 assert(ExtrVecTy->getElementType() == VecTy->getElementType() &&
321 "Expected same element type!");
323 for (unsigned Idx = 0, E = ExtrVecTy->getNumElements(); Idx != E; ++Idx) {
324 int MaskLane = Lane + Idx;
325 assert((unsigned)MaskLane <
326 cast<FixedVectorType>(FromVec->getType())->getNumElements() &&
327 "Out of bounds!");
328 Mask.push_back(MaskLane);
329 }
330 return ShuffleVectorInst::create(FromVec, PoisonValue::get(VecTy), Mask,
331 WhereIt, Ctx, "Unpack");
332 }
333
334 /// Iterate over all lanes and Value pairs.
335 // For example, given a range: {i32 %v0, <2 x i32> %v1, i32 %v2} we get:
336 // Lane Elm
337 // 0 %v0
338 // 1 %v1
339 // 3 %v2
340 template <typename RangeIteratorT> class LaneValueEnumerator {
341 /// Points to current element.
342 RangeIteratorT It;
343 RangeIteratorT ItE;
344 /// Accumulator of lanes.
345 unsigned Lane;
346
347 public:
348 // Note that We can start counting from a non-zero BeginLane, though the
349 // user must make sure it corresponds to the correct lane matching Begin.
350 LaneValueEnumerator(RangeIteratorT Begin, RangeIteratorT End,
351 unsigned BeginLane)
352 : It(Begin), ItE(End), Lane(BeginLane) {}
353 using iterator_catecotry = std::input_iterator_tag;
354 // NOTE: dereference returns by value instead of by reference.
355 using value_type = std::pair<unsigned, Value *>;
356 using difference_type = std::ptrdiff_t;
357 using pointer = std::pair<unsigned, Value *> *;
358 using reference = std::pair<unsigned, Value *> &;
360 assert(It != ItE && "Already at end!");
361 auto *Ty = Utils::getExpectedType(*It);
362 if (auto *VecTy = dyn_cast<FixedVectorType>(Ty)) {
363 Lane += VecTy->getNumElements();
364 } else {
365 assert(!isa<VectorType>(Ty) && "Expected scalar type!");
366 Lane += 1;
367 }
368 ++It;
369 return *this;
370 }
371 value_type operator*() const { return {Lane, *It}; }
373 return It == Other.It;
374 }
376 return !(*this == Other);
377 }
378 };
379
380 /// Helper for creating LaneValueEnumerator ranges. Can be used in for loops
381 /// like: `for (auto [Lane, V] : enumerateLanes(Range))`
382 template <typename ValueContainerT>
383 static auto enumerateLanes(const ValueContainerT &Range) {
384 auto Begin = LaneValueEnumerator<decltype(Range.begin())>(Range.begin(),
385 Range.end(), 0);
386 auto End = LaneValueEnumerator<decltype(Range.begin())>(Range.end(),
387 Range.end(), 0);
388 return make_range(Begin, End);
389 }
390
391#ifndef NDEBUG
392 /// Helper dump function for debugging.
393 LLVM_DUMP_METHOD static void dump(ArrayRef<Value *> Bndl);
395#endif // NDEBUG
396};
397
398} // namespace sandboxir
399
400} // namespace llvm
401
402#endif // LLVM_TRANSFORMS_VECTORIZE_SANDBOXVECTORIZER_VECUTILS_H
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
#define LLVM_ABI
Definition Compiler.h:215
#define LLVM_DUMP_METHOD
Mark debug helper function definitions like dump() that should not be stripped from debug builds.
Definition Compiler.h:663
#define I(x, y, z)
Definition MD5.cpp:57
ConstantRange Range(APInt(BitWidth, Low), APInt(BitWidth, High))
Represent a constant reference to an array (0 or more elements consecutively in memory),...
Definition ArrayRef.h:40
const T & front() const
Get the first element.
Definition ArrayRef.h:144
iterator end() const
Definition ArrayRef.h:130
bool empty() const
Check if the array is empty.
Definition ArrayRef.h:136
InstListType::iterator iterator
Instruction iterators...
Definition BasicBlock.h:170
A parsed version of the target data layout string in and methods for querying it.
Definition DataLayout.h:64
The main scalar evolution driver.
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
LLVM Value Representation.
Definition Value.h:75
static LLVM_ABI ConstantInt * getSigned(IntegerType *Ty, int64_t V)
Return a ConstantInt with the specified value for the specified type.
Definition Constant.cpp:56
static LLVM_ABI Value * create(Value *Vec, Value *Idx, InsertPosition Pos, Context &Ctx, const Twine &Name="")
static LLVM_ABI FixedVectorType * get(Type *ElementType, unsigned NumElts)
A sandboxir::User with operands, opcode and linked with previous/next instructions in an instruction ...
Definition Instruction.h:43
bool comesBefore(const Instruction *Other) const
Given an instruction Other in the same basic block as this instruction, return true if this instructi...
LLVM_ABI BasicBlock * getParent() const
\Returns the BasicBlock containing this Instruction, or null if it is detached.
static LLVM_ABI PoisonValue * get(Type *T)
Static factory methods - Return an 'poison' object of the specified type.
Definition Constant.cpp:263
static LLVM_ABI Value * create(Value *V1, Value *V2, Value *Mask, InsertPosition Pos, Context &Ctx, const Twine &Name="")
Just like llvm::Type these are immutable, unique, never get freed and can only be created via static ...
Definition Type.h:50
static LLVM_ABI IntegerType * getInt32Ty(Context &Ctx)
Definition Type.cpp:21
bool isVectorTy() const
True if this is an instance of VectorType.
Definition Type.h:214
Value * getOperand(unsigned OpIdx) const
Definition User.h:123
static std::optional< int > getPointerDiffInBytes(LoadOrStoreT *I0, LoadOrStoreT *I1, ScalarEvolution &SE)
\Returns the gap between the memory locations accessed by I0 and I1 in bytes.
Definition Utils.h:92
static unsigned getNumBits(Type *Ty, const DataLayout &DL)
\Returns the number of bits of Ty.
Definition Utils.h:66
static Type * getExpectedType(const Value *V)
\Returns the expected type of Value V.
Definition Utils.h:32
A SandboxIR Value has users. This is the base class.
Definition Value.h:72
LLVM_ABI Type * getType() const
Definition Value.cpp:46
Context & getContext() const
Definition Value.h:285
Iterate over all lanes and Value pairs.
Definition VecUtils.h:340
bool operator==(const LaneValueEnumerator &Other) const
Definition VecUtils.h:372
bool operator!=(const LaneValueEnumerator &Other) const
Definition VecUtils.h:375
std::pair< unsigned, Value * > value_type
Definition VecUtils.h:355
std::pair< unsigned, Value * > & reference
Definition VecUtils.h:358
LaneValueEnumerator(RangeIteratorT Begin, RangeIteratorT End, unsigned BeginLane)
Definition VecUtils.h:350
std::pair< unsigned, Value * > * pointer
Definition VecUtils.h:357
static Type * tryGetCommonScalarType(ArrayRef< Value * > Bndl)
If all values in Bndl are of the same scalar type then return it, otherwise return nullptr.
Definition VecUtils.h:202
static Instruction * getLowest(ArrayRef< Instruction * > Instrs)
\Returns the instruction in Instrs that is lowest in the BB.
Definition VecUtils.h:141
static Type * getCommonScalarType(ArrayRef< Value * > Bndl)
Similar to tryGetCommonScalarType() but will assert that there is a common type.
Definition VecUtils.h:218
static int getNumElements(Type *Ty)
\Returns the number of elements in Ty.
Definition VecUtils.h:41
static std::optional< PackPattern > matchPack(Instruction *I)
If I is the last instruction of a pack pattern (i.e., an InsertElement into a vector),...
Definition VecUtils.h:253
static Type * getCombinedVectorTypeFor(ArrayRef< Instruction * > Bndl, const DataLayout &DL)
\Returns the combined vector type for Bndl, even when the element types differ.
Definition VecUtils.h:120
static Instruction * getLastPHIOrSelf(Instruction *I)
If I is not a PHI it returns it.
Definition VecUtils.h:191
static unsigned getNumLanes(Type *Ty)
\Returns the number of vector lanes of Ty or 1 if not a vector.
Definition VecUtils.h:86
static Instruction * getLowest(ArrayRef< Value * > Vals, BasicBlock *BB)
\Returns the lowest instruction in Vals, or nullptr if no instructions are found.
Definition VecUtils.h:161
static Value * unpack(Value *FromVec, Type *ExtrTy, unsigned Lane, BasicBlock::iterator WhereIt)
Emits the necessary instruction sequence to extract element of type ExtrTy at Lane from FromVec.
Definition VecUtils.h:297
static LLVM_DUMP_METHOD void dump(ArrayRef< Value * > Bndl)
Helper dump function for debugging.
Definition VecUtils.cpp:28
static Type * getWideType(Type *ElemTy, unsigned NumElts)
\Returns <NumElts x ElemTy>.
Definition VecUtils.h:109
static Instruction * getHighest(ArrayRef< Instruction * > Instrs)
\Returns the instruction in Instrs that is highest in the BB.
Definition VecUtils.h:151
static auto enumerateLanes(const ValueContainerT &Range)
Helper for creating LaneValueEnumerator ranges.
Definition VecUtils.h:383
static bool areConsecutive(LoadOrStoreT *I1, LoadOrStoreT *I2, ScalarEvolution &SE, const DataLayout &DL)
\Returns true if I1 and I2 are load/stores accessing consecutive memory addresses.
Definition VecUtils.h:53
static bool areConsecutive(ArrayRef< ValT * > Bndl, ScalarEvolution &SE, const DataLayout &DL)
Definition VecUtils.h:66
static Type * getElementType(Type *Ty)
Returns Ty if scalar or its element type if vector.
Definition VecUtils.h:46
static unsigned getNumLanes(Value *V)
\Returns the expected vector lanes of V or 1 if not a vector.
Definition VecUtils.h:95
static unsigned getNumLanes(ArrayRef< Value * > Bndl)
\Returns the total number of lanes across all values in Bndl.
Definition VecUtils.h:100
static LLVM_ABI unsigned getFloorPowerOf2(unsigned Num)
\Returns the first integer power of 2 that is <= Num.
Definition VecUtils.cpp:13
BasicBlock(llvm::BasicBlock *BB, Context &SBCtx)
Definition BasicBlock.h:75
This is an optimization pass for GlobalISel generic memory operations.
auto drop_begin(T &&RangeOrContainer, size_t N=1)
Return a range covering RangeOrContainer with the first N elements excluded.
Definition STLExtras.h:315
auto enumerate(FirstRange &&First, RestRanges &&...Rest)
Given two or more input ranges, returns a new range whose values are tuples (A, B,...
Definition STLExtras.h:2554
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:643
iterator_range< T > make_range(T x, T y)
Convenience function for iterating over sub-ranges.
auto reverse(ContainerTy &&C)
Definition STLExtras.h:407
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
Definition Casting.h:547
@ Other
Any other memory.
Definition ModRef.h:68
DWARFExpression::Operation Op
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:559
auto find_if(R &&Range, UnaryPredicate P)
Provide wrappers to std::find_if which take ranges instead of having to pass begin/end explicitly.
Definition STLExtras.h:1772
auto seq(T Begin, T End)
Iterate over an integral type from Begin up to - but not including - End.
Definition Sequence.h:305
hash_code hash_combine_range(InputIteratorT first, InputIteratorT last)
Compute a hash_code for a sequence of values.
Definition Hashing.h:285
static bool isEqual(const SmallVector< sandboxir::Value * > &Vec1, const SmallVector< sandboxir::Value * > &Vec2)
Definition VecUtils.h:28
static unsigned getHashValue(const SmallVector< sandboxir::Value * > &Vec)
Definition VecUtils.h:25
An information struct used to provide DenseMap with the various necessary components for a given valu...
Helper struct for matchPack().
Definition VecUtils.h:230
SmallVector< Value * > Operands
The "external" operands of the pack pattern, i.e., the values that get packed into a vector,...
Definition VecUtils.h:243
SmallVector< Instruction * > Instrs
The insertelement instructions that form the pack pattern in bottom-up order, i.e....
Definition VecUtils.h:238