LLVM 23.0.0git
SPIRVPrepareGlobals.cpp
Go to the documentation of this file.
1//===-- SPIRVPrepareGlobals.cpp - Prepare IR SPIRV globals ------*- 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// The pass:
10// - transforms IR globals that cannot be trivially mapped to SPIRV into
11// something that is trival to lower;
12// - for AMDGCN flavoured SPIRV, it assigns unique IDs to the specialisation
13// constants associated with feature predicates, which were inserted by the
14// FE when expanding calls to __builtin_amdgcn_processor_is or
15// __builtin_amdgcn_is_invocable
16//
17//===----------------------------------------------------------------------===//
18
19#include "SPIRVPrepareGlobals.h"
20#include "SPIRV.h"
21#include "SPIRVUtils.h"
22
23#include "llvm/ADT/STLExtras.h"
25#include "llvm/ADT/StringMap.h"
26#include "llvm/IR/IntrinsicsSPIRV.h"
27#include "llvm/IR/Module.h"
28#include "llvm/Support/Debug.h"
29
30#include <climits>
31#include <string>
32
33#define DEBUG_TYPE "spirv-prepare-globals"
34
35using namespace llvm;
36
37namespace {
38
39struct SPIRVPrepareGlobalsImpl {
40 bool runOnModule(Module &M);
41};
42
43struct SPIRVPrepareGlobalsLegacy : public ModulePass {
44 static char ID;
45 SPIRVPrepareGlobalsLegacy() : ModulePass(ID) {}
46
47 StringRef getPassName() const override {
48 return "SPIRV prepare global variables";
49 }
50
51 bool runOnModule(Module &M) override {
52 return SPIRVPrepareGlobalsImpl().runOnModule(M);
53 }
54};
55
56// The backend does not support GlobalAlias. Replace aliases with their aliasees
57// when possible and remove them from the module.
58bool tryReplaceAliasWithAliasee(GlobalAlias &GA) {
59 // According to the lang ref, aliases cannot be replaced if either the alias
60 // or the aliasee are interposable. We only replace in the case that both
61 // are not interposable.
62 if (GA.isInterposable()) {
63 LLVM_DEBUG(dbgs() << "Skipping interposable alias: " << GA.getName()
64 << "\n");
65 return false;
66 }
67
68 auto *AO = dyn_cast<GlobalObject>(GA.getAliasee());
69 if (!AO) {
70 LLVM_DEBUG(dbgs() << "Skipping alias whose aliasee is not a GlobalObject: "
71 << GA.getName() << "\n");
72 return false;
73 }
74
75 if (AO->isInterposable()) {
76 LLVM_DEBUG(dbgs() << "Skipping interposable aliasee: " << AO->getName()
77 << "\n");
78 return false;
79 }
80
81 LLVM_DEBUG(dbgs() << "Replacing alias " << GA.getName()
82 << " with aliasee: " << AO->getName() << "\n");
83
84 GA.replaceAllUsesWith(AO);
85 if (GA.isDiscardableIfUnused()) {
86 GA.eraseFromParent();
87 }
88
89 return true;
90}
91
92bool tryAssignPredicateSpecConstIDs(Module &M, Function *F) {
94 for (auto &&U : F->users()) {
95 auto *CI = dyn_cast<CallInst>(U);
96 if (!CI)
97 continue;
98
99 auto *SpecID = dyn_cast<ConstantInt>(CI->getArgOperand(0));
100 if (!SpecID)
101 continue;
102
103 unsigned ID = SpecID->getZExtValue();
104 if (ID != UINT32_MAX)
105 continue;
106
107 // Replace placeholder Specialisation Constant IDs with unique IDs
108 // associated with the predicate being evaluated, which is encoded via
109 // spv_assign_name.
110 auto *MD =
111 cast<MDNode>(cast<MetadataAsValue>(CI->getOperand(2))->getMetadata());
112 auto *P = cast<MDString>(MD->getOperand(0));
113
114 ID = IDs.try_emplace(P->getString(), IDs.size()).first->second;
115 CI->setArgOperand(0, ConstantInt::get(CI->getArgOperand(0)->getType(), ID));
116 }
117
118 if (IDs.empty())
119 return false;
120
121 // Store the predicate -> ID mapping as a fixed format string
122 // (predicate ID\0...), for later use during SPIR-V consumption.
123 std::string Tmp;
124 for (auto &&[Predicate, SpecID] : IDs)
125 Tmp.append(Predicate).append(" ").append(utostr(SpecID)).push_back('\0');
126
127 Constant *PredSpecIDStr =
128 ConstantDataArray::getString(M.getContext(), Tmp, false);
129
130 new GlobalVariable(M, PredSpecIDStr->getType(), true,
132 PredSpecIDStr, "llvm.amdgcn.feature.predicate.ids");
133
134 return true;
135}
136
137bool SPIRVPrepareGlobalsImpl::runOnModule(Module &M) {
138 bool Changed = false;
139
140 for (GlobalAlias &GA : make_early_inc_range(M.aliases())) {
141 Changed |= tryReplaceAliasWithAliasee(GA);
142 }
143
144 if (M.getTargetTriple().getVendor() != Triple::AMD)
145 return Changed;
146
147 // TODO: Currently, for AMDGCN flavoured SPIR-V, the symbol can only be
148 // inserted via feature predicate use, but in the future this will need
149 // revisiting if we start making more liberal use of the intrinsic.
151 &M, Intrinsic::spv_named_boolean_spec_constant))
152 Changed |= tryAssignPredicateSpecConstIDs(M, F);
153
154 return Changed;
155}
156char SPIRVPrepareGlobalsLegacy::ID = 0;
157
158} // namespace
159
160INITIALIZE_PASS(SPIRVPrepareGlobalsLegacy, "spirv-prepare-globals",
161 "SPIRV prepare global variables", false, false)
162
165 return SPIRVPrepareGlobalsImpl().runOnModule(M) ? PreservedAnalyses::none()
167}
168
169namespace llvm {
171 return new SPIRVPrepareGlobalsLegacy();
172}
173} // namespace llvm
This file defines the StringMap class.
Module.h This file contains the declarations for the Module class.
#define F(x, y, z)
Definition MD5.cpp:54
#define P(N)
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
Definition PassSupport.h:56
This file contains some templates that are useful if you are working with the STL at all.
This file contains some functions that are useful when dealing with strings.
#define LLVM_DEBUG(...)
Definition Debug.h:114
static LLVM_ABI Constant * getString(LLVMContext &Context, StringRef Initializer, bool AddNull=true, bool ByteString=false)
This method constructs a CDS and initializes it with a text string.
This is an important base class in LLVM.
Definition Constant.h:43
LLVM_ABI void eraseFromParent()
eraseFromParent - This method unlinks 'this' from the containing module and deletes it.
Definition Globals.cpp:651
const Constant * getAliasee() const
Definition GlobalAlias.h:87
LLVM_ABI bool isInterposable() const
Return true if this global's definition can be substituted with an arbitrary definition at link time ...
Definition Globals.cpp:116
static bool isDiscardableIfUnused(LinkageTypes Linkage)
Whether the definition of this global may be discarded if it is not used in its compilation unit.
@ ExternalLinkage
Externally visible function.
Definition GlobalValue.h:53
ModulePass class - This class is used to implement unstructured interprocedural optimizations and ana...
Definition Pass.h:255
A Module instance is used to store all the information related to an LLVM module.
Definition Module.h:67
A set of analyses that are preserved following a run of a transformation pass.
Definition Analysis.h:112
static PreservedAnalyses none()
Convenience factory function for the empty preserved set.
Definition Analysis.h:115
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
Definition Analysis.h:118
unsigned size() const
Definition StringMap.h:109
bool empty() const
Definition StringMap.h:108
StringMap - This is an unconventional map that is specialized for handling keys that are "strings",...
Definition StringMap.h:133
std::pair< iterator, bool > try_emplace(StringRef Key, ArgsTy &&...Args)
Emplace a new element for the specified key into the map if the key isn't already in the map.
Definition StringMap.h:381
Represent a constant reference to a string, i.e.
Definition StringRef.h:55
Type * getType() const
All values are typed, get the type of this value.
Definition Value.h:255
LLVM_ABI void replaceAllUsesWith(Value *V)
Change all uses of this to point to a new Value.
Definition Value.cpp:549
LLVM_ABI StringRef getName() const
Return a constant reference to the value's name.
Definition Value.cpp:318
Changed
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
Definition CallingConv.h:24
LLVM_ABI Function * getDeclarationIfExists(const Module *M, ID id)
Look up the Function declaration of the intrinsic id in the Module M and return it if it exists.
This is an optimization pass for GlobalISel generic memory operations.
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< early_inc_iterator_impl< detail::IterOfRange< RangeT > > > make_early_inc_range(RangeT &&Range)
Make a range that does early increment to allow mutation of the underlying range without disrupting i...
Definition STLExtras.h:633
std::string utostr(uint64_t X, bool isNeg=false)
ModulePass * createSPIRVPrepareGlobalsPass()
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
Definition Debug.cpp:207
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:559
AnalysisManager< Module > ModuleAnalysisManager
Convenience typedef for the Module analysis manager.
Definition MIRParser.h:39