1 //===- VPlanValue.h - Represent Values in Vectorizer Plan -----------------===// 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 /// \file 10 /// This file contains the declarations of the entities induced by Vectorization 11 /// Plans, e.g. the instructions the VPlan intends to generate if executed. 12 /// VPlan models the following entities: 13 /// VPValue VPUser VPDef 14 /// | | 15 /// VPInstruction 16 /// These are documented in docs/VectorizationPlan.rst. 17 /// 18 //===----------------------------------------------------------------------===// 19 20 #ifndef LLVM_TRANSFORMS_VECTORIZE_VPLAN_VALUE_H 21 #define LLVM_TRANSFORMS_VECTORIZE_VPLAN_VALUE_H 22 23 #include "llvm/ADT/DenseMap.h" 24 #include "llvm/ADT/STLExtras.h" 25 #include "llvm/ADT/SmallVector.h" 26 #include "llvm/ADT/TinyPtrVector.h" 27 #include "llvm/ADT/iterator_range.h" 28 29 namespace llvm { 30 31 // Forward declarations. 32 class raw_ostream; 33 class Value; 34 class VPDef; 35 class VPSlotTracker; 36 class VPUser; 37 class VPRecipeBase; 38 class VPWidenMemoryInstructionRecipe; 39 40 // This is the base class of the VPlan Def/Use graph, used for modeling the data 41 // flow into, within and out of the VPlan. VPValues can stand for live-ins 42 // coming from the input IR, instructions which VPlan will generate if executed 43 // and live-outs which the VPlan will need to fix accordingly. 44 class VPValue { 45 friend class VPBuilder; 46 friend class VPDef; 47 friend class VPInstruction; 48 friend struct VPlanTransforms; 49 friend class VPBasicBlock; 50 friend class VPInterleavedAccessInfo; 51 friend class VPSlotTracker; 52 friend class VPRecipeBase; 53 friend class VPWidenMemoryInstructionRecipe; 54 55 const unsigned char SubclassID; ///< Subclass identifier (for isa/dyn_cast). 56 57 SmallVector<VPUser *, 1> Users; 58 59 protected: 60 // Hold the underlying Value, if any, attached to this VPValue. 61 Value *UnderlyingVal; 62 63 /// Pointer to the VPDef that defines this VPValue. If it is nullptr, the 64 /// VPValue is not defined by any recipe modeled in VPlan. 65 VPDef *Def; 66 67 VPValue(const unsigned char SC, Value *UV = nullptr, VPDef *Def = nullptr); 68 69 // DESIGN PRINCIPLE: Access to the underlying IR must be strictly limited to 70 // the front-end and back-end of VPlan so that the middle-end is as 71 // independent as possible of the underlying IR. We grant access to the 72 // underlying IR using friendship. In that way, we should be able to use VPlan 73 // for multiple underlying IRs (Polly?) by providing a new VPlan front-end, 74 // back-end and analysis information for the new IR. 75 76 // Set \p Val as the underlying Value of this VPValue. 77 void setUnderlyingValue(Value *Val) { 78 assert(!UnderlyingVal && "Underlying Value is already set."); 79 UnderlyingVal = Val; 80 } 81 82 public: 83 /// Return the underlying Value attached to this VPValue. 84 Value *getUnderlyingValue() { return UnderlyingVal; } 85 const Value *getUnderlyingValue() const { return UnderlyingVal; } 86 87 /// An enumeration for keeping track of the concrete subclass of VPValue that 88 /// are actually instantiated. Values of this enumeration are kept in the 89 /// SubclassID field of the VPValue objects. They are used for concrete 90 /// type identification. 91 enum { 92 VPValueSC, 93 VPVInstructionSC, 94 VPVMemoryInstructionSC, 95 VPVReductionSC, 96 VPVReplicateSC, 97 VPVWidenSC, 98 VPVWidenCallSC, 99 VPVWidenGEPSC, 100 VPVWidenSelectSC, 101 }; 102 103 VPValue(Value *UV = nullptr, VPDef *Def = nullptr) 104 : VPValue(VPValueSC, UV, Def) {} 105 VPValue(const VPValue &) = delete; 106 VPValue &operator=(const VPValue &) = delete; 107 108 virtual ~VPValue(); 109 110 /// \return an ID for the concrete type of this object. 111 /// This is used to implement the classof checks. This should not be used 112 /// for any other purpose, as the values may change as LLVM evolves. 113 unsigned getVPValueID() const { return SubclassID; } 114 115 void printAsOperand(raw_ostream &OS, VPSlotTracker &Tracker) const; 116 void print(raw_ostream &OS, VPSlotTracker &Tracker) const; 117 118 /// Dump the value to stderr (for debugging). 119 void dump() const; 120 121 unsigned getNumUsers() const { return Users.size(); } 122 void addUser(VPUser &User) { Users.push_back(&User); } 123 124 /// Remove a single \p User from the list of users. 125 void removeUser(VPUser &User) { 126 bool Found = false; 127 // The same user can be added multiple times, e.g. because the same VPValue 128 // is used twice by the same VPUser. Remove a single one. 129 erase_if(Users, [&User, &Found](VPUser *Other) { 130 if (Found) 131 return false; 132 if (Other == &User) { 133 Found = true; 134 return true; 135 } 136 return false; 137 }); 138 } 139 140 typedef SmallVectorImpl<VPUser *>::iterator user_iterator; 141 typedef SmallVectorImpl<VPUser *>::const_iterator const_user_iterator; 142 typedef iterator_range<user_iterator> user_range; 143 typedef iterator_range<const_user_iterator> const_user_range; 144 145 user_iterator user_begin() { return Users.begin(); } 146 const_user_iterator user_begin() const { return Users.begin(); } 147 user_iterator user_end() { return Users.end(); } 148 const_user_iterator user_end() const { return Users.end(); } 149 user_range users() { return user_range(user_begin(), user_end()); } 150 const_user_range users() const { 151 return const_user_range(user_begin(), user_end()); 152 } 153 154 /// Returns true if the value has more than one unique user. 155 bool hasMoreThanOneUniqueUser() { 156 if (getNumUsers() == 0) 157 return false; 158 159 // Check if all users match the first user. 160 auto Current = std::next(user_begin()); 161 while (Current != user_end() && *user_begin() == *Current) 162 Current++; 163 return Current != user_end(); 164 } 165 166 void replaceAllUsesWith(VPValue *New); 167 168 VPDef *getDef() { return Def; } 169 170 /// Returns the underlying IR value, if this VPValue is defined outside the 171 /// scope of VPlan. Returns nullptr if the VPValue is defined by a VPDef 172 /// inside a VPlan. 173 Value *getLiveInIRValue() { 174 assert(!getDef() && 175 "VPValue is not a live-in; it is defined by a VPDef inside a VPlan"); 176 return getUnderlyingValue(); 177 } 178 }; 179 180 typedef DenseMap<Value *, VPValue *> Value2VPValueTy; 181 typedef DenseMap<VPValue *, Value *> VPValue2ValueTy; 182 183 raw_ostream &operator<<(raw_ostream &OS, const VPValue &V); 184 185 /// This class augments VPValue with operands which provide the inverse def-use 186 /// edges from VPValue's users to their defs. 187 class VPUser { 188 SmallVector<VPValue *, 2> Operands; 189 190 protected: 191 /// Print the operands to \p O. 192 void printOperands(raw_ostream &O, VPSlotTracker &SlotTracker) const; 193 194 public: 195 VPUser() {} 196 VPUser(ArrayRef<VPValue *> Operands) { 197 for (VPValue *Operand : Operands) 198 addOperand(Operand); 199 } 200 201 VPUser(std::initializer_list<VPValue *> Operands) 202 : VPUser(ArrayRef<VPValue *>(Operands)) {} 203 template <typename IterT> VPUser(iterator_range<IterT> Operands) { 204 for (VPValue *Operand : Operands) 205 addOperand(Operand); 206 } 207 208 VPUser(const VPUser &) = delete; 209 VPUser &operator=(const VPUser &) = delete; 210 virtual ~VPUser() { 211 for (VPValue *Op : operands()) 212 Op->removeUser(*this); 213 } 214 215 void addOperand(VPValue *Operand) { 216 Operands.push_back(Operand); 217 Operand->addUser(*this); 218 } 219 220 unsigned getNumOperands() const { return Operands.size(); } 221 inline VPValue *getOperand(unsigned N) const { 222 assert(N < Operands.size() && "Operand index out of bounds"); 223 return Operands[N]; 224 } 225 226 void setOperand(unsigned I, VPValue *New) { 227 Operands[I]->removeUser(*this); 228 Operands[I] = New; 229 New->addUser(*this); 230 } 231 232 typedef SmallVectorImpl<VPValue *>::iterator operand_iterator; 233 typedef SmallVectorImpl<VPValue *>::const_iterator const_operand_iterator; 234 typedef iterator_range<operand_iterator> operand_range; 235 typedef iterator_range<const_operand_iterator> const_operand_range; 236 237 operand_iterator op_begin() { return Operands.begin(); } 238 const_operand_iterator op_begin() const { return Operands.begin(); } 239 operand_iterator op_end() { return Operands.end(); } 240 const_operand_iterator op_end() const { return Operands.end(); } 241 operand_range operands() { return operand_range(op_begin(), op_end()); } 242 const_operand_range operands() const { 243 return const_operand_range(op_begin(), op_end()); 244 } 245 246 /// Method to support type inquiry through isa, cast, and dyn_cast. 247 static inline bool classof(const VPDef *Recipe); 248 }; 249 250 /// This class augments a recipe with a set of VPValues defined by the recipe. 251 /// It allows recipes to define zero, one or multiple VPValues. A VPDef owns 252 /// the VPValues it defines and is responsible for deleting its defined values. 253 /// Single-value VPDefs that also inherit from VPValue must make sure to inherit 254 /// from VPDef before VPValue. 255 class VPDef { 256 friend class VPValue; 257 258 /// Subclass identifier (for isa/dyn_cast). 259 const unsigned char SubclassID; 260 261 /// The VPValues defined by this VPDef. 262 TinyPtrVector<VPValue *> DefinedValues; 263 264 /// Add \p V as a defined value by this VPDef. 265 void addDefinedValue(VPValue *V) { 266 assert(V->getDef() == this && 267 "can only add VPValue already linked with this VPDef"); 268 DefinedValues.push_back(V); 269 } 270 271 /// Remove \p V from the values defined by this VPDef. \p V must be a defined 272 /// value of this VPDef. 273 void removeDefinedValue(VPValue *V) { 274 assert(V->getDef() == this && 275 "can only remove VPValue linked with this VPDef"); 276 assert(is_contained(DefinedValues, V) && 277 "VPValue to remove must be in DefinedValues"); 278 erase_value(DefinedValues, V); 279 V->Def = nullptr; 280 } 281 282 public: 283 /// An enumeration for keeping track of the concrete subclass of VPRecipeBase 284 /// that is actually instantiated. Values of this enumeration are kept in the 285 /// SubclassID field of the VPRecipeBase objects. They are used for concrete 286 /// type identification. 287 using VPRecipeTy = enum { 288 VPBlendSC, 289 VPBranchOnMaskSC, 290 VPInstructionSC, 291 VPInterleaveSC, 292 VPPredInstPHISC, 293 VPReductionSC, 294 VPReplicateSC, 295 VPWidenCallSC, 296 VPWidenCanonicalIVSC, 297 VPWidenGEPSC, 298 VPWidenIntOrFpInductionSC, 299 VPWidenMemoryInstructionSC, 300 VPWidenPHISC, 301 VPWidenSC, 302 VPWidenSelectSC 303 }; 304 305 VPDef(const unsigned char SC) : SubclassID(SC) {} 306 307 virtual ~VPDef() { 308 for (VPValue *D : make_early_inc_range(DefinedValues)) { 309 assert(D->Def == this && 310 "all defined VPValues should point to the containing VPDef"); 311 assert(D->getNumUsers() == 0 && 312 "all defined VPValues should have no more users"); 313 D->Def = nullptr; 314 delete D; 315 } 316 } 317 318 /// Returns the VPValue with index \p I defined by the VPDef. 319 VPValue *getVPValue(unsigned I = 0) { 320 assert(DefinedValues[I] && "defined value must be non-null"); 321 return DefinedValues[I]; 322 } 323 const VPValue *getVPValue(unsigned I = 0) const { 324 assert(DefinedValues[I] && "defined value must be non-null"); 325 return DefinedValues[I]; 326 } 327 328 /// Returns an ArrayRef of the values defined by the VPDef. 329 ArrayRef<VPValue *> definedValues() { return DefinedValues; } 330 /// Returns an ArrayRef of the values defined by the VPDef. 331 ArrayRef<VPValue *> definedValues() const { return DefinedValues; } 332 333 /// Returns the number of values defined by the VPDef. 334 unsigned getNumDefinedValues() const { return DefinedValues.size(); } 335 336 /// \return an ID for the concrete type of this object. 337 /// This is used to implement the classof checks. This should not be used 338 /// for any other purpose, as the values may change as LLVM evolves. 339 unsigned getVPDefID() const { return SubclassID; } 340 341 /// Dump the VPDef to stderr (for debugging). 342 void dump() const; 343 344 /// Each concrete VPDef prints itself. 345 virtual void print(raw_ostream &O, const Twine &Indent, 346 VPSlotTracker &SlotTracker) const = 0; 347 }; 348 349 class VPlan; 350 class VPBasicBlock; 351 class VPRegionBlock; 352 353 /// This class can be used to assign consecutive numbers to all VPValues in a 354 /// VPlan and allows querying the numbering for printing, similar to the 355 /// ModuleSlotTracker for IR values. 356 class VPSlotTracker { 357 DenseMap<const VPValue *, unsigned> Slots; 358 unsigned NextSlot = 0; 359 360 void assignSlots(const VPBlockBase *VPBB); 361 void assignSlots(const VPRegionBlock *Region); 362 void assignSlots(const VPBasicBlock *VPBB); 363 void assignSlot(const VPValue *V); 364 365 void assignSlots(const VPlan &Plan); 366 367 public: 368 VPSlotTracker(const VPlan *Plan = nullptr) { 369 if (Plan) 370 assignSlots(*Plan); 371 } 372 373 unsigned getSlot(const VPValue *V) const { 374 auto I = Slots.find(V); 375 if (I == Slots.end()) 376 return -1; 377 return I->second; 378 } 379 }; 380 381 } // namespace llvm 382 383 #endif // LLVM_TRANSFORMS_VECTORIZE_VPLAN_VALUE_H 384