37#include "llvm/IR/IntrinsicsSPIRV.h"
47class SPIRVPrepareFunctionsImpl {
48 const SPIRVTargetMachine &TM;
49 bool substituteIntrinsicCalls(Function *
F);
50 bool substituteAbortKHRCalls(Function *
F);
52 Function *removeAggregateTypesFromSignature(Function *
F);
53 bool removeAggregateTypesFromCalls(Function *
F);
56 SPIRVPrepareFunctionsImpl(
const SPIRVTargetMachine &TM) : TM(TM) {}
57 bool runOnModule(
Module &M);
60class SPIRVPrepareFunctionsLegacy :
public ModulePass {
61 const SPIRVTargetMachine &TM;
65 SPIRVPrepareFunctionsLegacy(
const SPIRVTargetMachine &TM)
66 : ModulePass(ID), TM(TM) {}
68 bool runOnModule(
Module &M)
override {
69 return SPIRVPrepareFunctionsImpl(TM).runOnModule(M);
72 StringRef getPassName()
const override {
return "SPIRV prepare functions"; }
77 cl::desc(
"Emit unknown intrinsics as calls to external functions. A "
78 "comma-separated input list of intrinsic prefixes must be "
79 "provided, and only intrinsics carrying a listed prefix get "
80 "emitted as described."),
84char SPIRVPrepareFunctionsLegacy::ID = 0;
87 "SPIRV prepare functions",
false,
false)
90 Function *IntrinsicFunc =
II->getCalledFunction();
91 assert(IntrinsicFunc &&
"Missing function");
92 std::string FuncName = IntrinsicFunc->
getName().
str();
94 FuncName =
"spirv." + FuncName;
103 if (
F &&
F->getFunctionType() == FT)
128 "cannot lower the intrinsic '" +
129 Intrinsic->getCalledFunction()->getName() +
130 "' that takes a metadata argument",
139 std::string FuncName = lowerLLVMIntrinsicName(
Intrinsic);
141 FuncName +=
".volatile";
150 M->getOrInsertFunction(FuncName,
Intrinsic->getFunctionType());
151 auto IntrinsicID =
Intrinsic->getIntrinsicID();
155 assert(
F &&
"Callee must be a function");
157 switch (IntrinsicID) {
158 case Intrinsic::memset: {
167 IsVolatile->setName(
"isvolatile");
170 auto *MemSet = IRB.
CreateMemSet(Dest, Val, Len, MSI->getDestAlign(),
174 MemSet->eraseFromParent();
177 case Intrinsic::bswap: {
181 Intrinsic::bswap,
Intrinsic->getType(),
F->getArg(0));
195 AnnoVal =
Ref->getOperand(0);
197 OptAnnoVal =
Ref->getOperand(0);
208 C &&
C->getNumOperands()) {
209 Value *MaybeStruct =
C->getOperand(0);
211 for (
unsigned I = 0,
E = Struct->getNumOperands();
I !=
E; ++
I) {
213 Anno += (
I == 0 ?
": " :
", ") +
214 std::to_string(CInt->getType()->getIntegerBitWidth() == 1
215 ? CInt->getZExtValue()
216 : CInt->getSExtValue());
220 for (
unsigned I = 0,
E = Struct->getType()->getStructNumElements();
222 Anno +=
I == 0 ?
": 0" :
", 0";
229 const std::string &Anno,
236 static const std::regex R(
237 "\\{(\\d+)(?:[:,](\\d+|\"[^\"]*\")(?:,(\\d+|\"[^\"]*\"))*)?\\}");
240 for (std::sregex_iterator
241 It = std::sregex_iterator(Anno.begin(), Anno.end(), R),
242 ItEnd = std::sregex_iterator();
244 if (It->position() != Pos)
246 Pos = It->position() + It->length();
247 std::smatch Match = *It;
249 for (std::size_t i = 1; i < Match.size(); ++i) {
250 std::ssub_match SMatch = Match[i];
251 std::string Item = SMatch.str();
252 if (Item.length() == 0)
254 if (Item[0] ==
'"') {
255 Item = Item.substr(1, Item.length() - 2);
257 static const std::regex RStr(
"^(\\d+)(?:,(\\d+))*$");
258 if (std::smatch MatchStr; std::regex_match(Item, MatchStr, RStr)) {
259 for (std::size_t SubIdx = 1; SubIdx < MatchStr.size(); ++SubIdx)
260 if (std::string SubStr = MatchStr[SubIdx].str(); SubStr.length())
262 ConstantInt::get(
Int32Ty, std::stoi(SubStr))));
273 if (MDsItem.
size() == 0)
277 return Pos ==
static_cast<int>(Anno.length()) ? std::move(MDs)
286 Value *PtrArg =
nullptr;
288 PtrArg = BI->getOperand(0);
290 PtrArg =
II->getOperand(0);
293 4 <
II->arg_size() ?
II->getArgOperand(4) :
nullptr);
302 if (MDs.
size() == 0) {
312 Intrinsic::spv_assign_decoration, {PtrArg->
getType()},
314 II->replaceAllUsesWith(
II->getOperand(0));
324 Type *FSHRetTy = FSHFuncTy->getReturnType();
325 const std::string FuncName = lowerLLVMIntrinsicName(FSHIntrinsic);
329 if (!FSHFunc->
empty()) {
341 unsigned BitWidth = IntTy->getIntegerBitWidth();
343 Value *BitWidthForInsts =
347 Value *RotateModVal =
349 Value *FirstShift =
nullptr, *SecShift =
nullptr;
362 Value *SubRotateVal = IRB.
CreateSub(BitWidthForInsts, RotateModVal);
381 if (!ConstrainedCmpIntrinsic)
403 if (
II->getIntrinsicID() == Intrinsic::assume) {
405 II->getModule(), Intrinsic::SPVIntrinsics::spv_assume);
406 II->setCalledFunction(
F);
407 }
else if (
II->getIntrinsicID() == Intrinsic::expect) {
409 II->getModule(), Intrinsic::SPVIntrinsics::spv_expect,
410 {II->getOperand(0)->getType()});
411 II->setCalledFunction(
F);
418 auto *LifetimeArg0 =
II->getArgOperand(0);
422 II->eraseFromParent();
428 std::optional<TypeSize>
Size =
429 Alloca->getAllocationSize(Alloca->getDataLayout());
431 Builder.CreateIntrinsic(NewID, Alloca->getType(), {SizeVal, LifetimeArg0});
432 II->eraseFromParent();
440 Value *
A = FPI->getArgOperand(0);
444 Builder.SetInsertPoint(
II);
445 std::optional<RoundingMode> Rounding = FPI->getRoundingMode();
446 Value *Product = Builder.CreateFMul(
A,
Mul,
II->getName() +
".mul");
447 Value *Result = Builder.CreateConstrainedFPBinOp(
448 Intrinsic::experimental_constrained_fadd, Product,
Add, {},
449 II->getName() +
".add",
nullptr, Rounding);
450 II->replaceAllUsesWith(Result);
456bool SPIRVPrepareFunctionsImpl::substituteIntrinsicCalls(
Function *
F) {
458 const SPIRVSubtarget &STI = TM.
getSubtarget<SPIRVSubtarget>(*F);
461 for (BasicBlock &BB : *
F) {
471 II->getCalledOperand()->getName().starts_with(
"llvm.spv"))
473 switch (
II->getIntrinsicID()) {
474 case Intrinsic::memset:
475 case Intrinsic::bswap:
478 case Intrinsic::fshl:
479 case Intrinsic::fshr:
483 case Intrinsic::assume:
484 case Intrinsic::expect:
489 case Intrinsic::lifetime_start:
492 II, Intrinsic::SPVIntrinsics::spv_lifetime_start);
494 II->eraseFromParent();
498 case Intrinsic::lifetime_end:
501 II, Intrinsic::SPVIntrinsics::spv_lifetime_end);
503 II->eraseFromParent();
507 case Intrinsic::ptr_annotation:
511 case Intrinsic::experimental_constrained_fmuladd:
515 case Intrinsic::experimental_constrained_fcmp:
516 case Intrinsic::experimental_constrained_fcmps:
523 if (
II->isAssumeLikeIntrinsic()) {
524 if (!
II->getType()->isVoidTy())
526 II->eraseFromParent();
531 any_of(SPVAllowUnknownIntrinsics, [
II](
auto &&Prefix) {
534 return II->getCalledFunction()->getName().starts_with(Prefix);
541 for (
auto *
I : EraseFromParent)
542 I->eraseFromParent();
556 transform(ChangedTys, std::back_inserter(MDArgs), [=, &Ctx](
auto &&CTy) {
561 if (!AsmConstraints.empty())
570SPIRVPrepareFunctionsImpl::removeAggregateTypesFromSignature(Function *
F) {
571 bool IsRetAggr =
F->getReturnType()->isAggregateType();
575 if (
F->isIntrinsic())
580 bool HasAggrArg =
llvm::any_of(
F->args(), [](Argument &Arg) {
581 return Arg.getType()->isAggregateType();
583 bool DoClone = IsRetAggr || HasAggrArg;
587 Type *RetType = IsRetAggr ?
B.getInt32Ty() :
F->getReturnType();
589 ChangedTypes.
push_back(std::pair<int, Type *>(-1,
F->getReturnType()));
591 for (
const auto &Arg :
F->args()) {
592 if (Arg.getType()->isAggregateType()) {
595 std::pair<int, Type *>(Arg.getArgNo(), Arg.getType()));
599 FunctionType *NewFTy =
600 FunctionType::get(RetType, ArgTypes,
F->getFunctionType()->isVarArg());
603 F->getName(),
F->getParent());
607 for (
auto &Arg :
F->args()) {
608 StringRef ArgName = Arg.getName();
609 NewFArgIt->setName(ArgName);
610 VMap[&Arg] = &(*NewFArgIt++);
620 std::move(ChangedTypes), NewF->
getName());
627 C->handleOperandChange(
F, NewF);
629 U->replaceUsesOfWith(
F, NewF);
633 if (RetType !=
F->getReturnType())
634 TM.
getSubtarget<SPIRVSubtarget>(*F).getSPIRVGlobalRegistry()->addMutated(
635 NewF,
F->getReturnType());
646 if (!Name.contains(
"__spirv_AbortKHR"))
649 if (Demangled.empty())
669 Msg =
B.CreateLoad(AggTy, Msg);
671 B.CreateIntrinsic(Intrinsic::spv_abort, {Msg->
getType()}, {Msg});
678bool SPIRVPrepareFunctionsImpl::substituteAbortKHRCalls(Function *
F) {
683 for (User *U :
F->users()) {
685 if (!CI || CI->getCalledFunction() !=
F)
687 if (CI->arg_size() != 1)
692 for (CallInst *CI : Calls)
695 return !Calls.empty();
705bool SPIRVPrepareFunctionsImpl::terminateBlocksAfterTrap(
Module &M,
707 assert((IID == Intrinsic::trap || IID == Intrinsic::ubsantrap) &&
708 "Expected trap intrinsic ID");
718 if (!
ST.canUseExtension(SPIRV::Extension::SPV_KHR_abort))
722 for (User *U :
F->users()) {
724 if (!CI || CI->getCalledFunction() !=
F)
730 for (CallInst *CI : Calls) {
744 std::string SafeConstraints(
"=r,");
745 for (
unsigned I = 0u;
I != Tmp.
size() - 1; ++
I) {
746 if (Tmp[
I].
starts_with(
'=') && (Tmp[
I][1] ==
'&' || isalnum(Tmp[
I][1])))
748 SafeConstraints.append(Tmp[
I]).append({
','});
750 SafeConstraints.append(Tmp.
back());
752 return SafeConstraints;
760bool SPIRVPrepareFunctionsImpl::removeAggregateTypesFromCalls(Function *
F) {
761 if (
F->isDeclaration() ||
F->isIntrinsic())
767 if (!CB->getCalledOperand() || CB->getCalledFunction())
769 if (CB->getType()->isAggregateType() ||
771 [](
auto &&Arg) { return Arg->getType()->isAggregateType(); }))
781 unsigned MutatedCallIdx = 0;
782 for (
auto &&[CB, NewFnTy] : Calls) {
786 Type *RetTy = CB->getType();
792 for (
auto &&Arg : CB->args()) {
793 if (Arg->getType()->isAggregateType()) {
795 ChangedTypes.
emplace_back(Arg.getOperandNo(), Arg->getType());
800 NewFnTy = FunctionType::get(RetTy, NewArgTypes,
801 CB->getFunctionType()->isVarArg());
805 (
"spv.mutated_callsite." +
F->getName() +
"." + Twine(MutatedCallIdx++))
808 "spv.mutated_callsite",
811 std::string Constraints;
813 Constraints = ASM->getConstraintString();
816 NewFnTy, ASM->getAsmString(),
818 ASM->isAlignStack(), ASM->getDialect(), ASM->canThrow()));
822 F->getParent()->getOrInsertNamedMetadata(
"spv.mutated_callsites"),
823 std::move(ChangedTypes),
Key, Constraints);
826 for (
auto &&[CB, NewFTy] : Calls) {
827 if (NewFTy->getReturnType() != CB->getType())
828 TM.
getSubtarget<SPIRVSubtarget>(*F).getSPIRVGlobalRegistry()->addMutated(
830 CB->mutateFunctionType(NewFTy);
836bool SPIRVPrepareFunctionsImpl::runOnModule(
Module &M) {
840 const_cast<SPIRVTargetMachine &
>(TM)
841 .getMutableSubtargetImpl()
842 ->resolveEnvFromModule(M);
845 if (
M.getFunctionDefs().empty()) {
857 Changed |= terminateBlocksAfterTrap(M, Intrinsic::trap);
858 Changed |= terminateBlocksAfterTrap(M, Intrinsic::ubsantrap);
860 for (GlobalVariable &GV :
M.globals()) {
863 if (GV.hasAvailableExternallyLinkage() && !GV.isDeclaration()) {
870 for (Function &
F : M) {
873 if (
F.hasAvailableExternallyLinkage() && !
F.isDeclaration()) {
878 Changed |= substituteAbortKHRCalls(&
F);
879 Changed |= substituteIntrinsicCalls(&
F);
881 Changed |= removeAggregateTypesFromCalls(&
F);
884 std::vector<Function *> FuncsWorklist;
886 FuncsWorklist.push_back(&
F);
888 for (
auto *
F : FuncsWorklist) {
889 Function *NewF = removeAggregateTypesFromSignature(
F);
892 F->eraseFromParent();
901 return SPIRVPrepareFunctionsImpl(TM).runOnModule(M)
908 return new SPIRVPrepareFunctionsLegacy(TM);
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
Expand Atomic instructions
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
Machine Check Debug Module
uint64_t IntrinsicInst * II
#define INITIALIZE_PASS(passName, arg, name, cfg, analysis)
static void lowerFunnelShifts(IntrinsicInst *FSHIntrinsic)
static std::string getAnnotation(Value *AnnoVal, Value *OptAnnoVal)
static void lowerConstrainedFPCmpIntrinsic(ConstrainedFPCmpIntrinsic *ConstrainedCmpIntrinsic, SmallVector< Instruction * > &EraseFromParent)
static void lowerConstrainedFmuladd(IntrinsicInst *II, SmallVector< Instruction * > &EraseFromParent)
static void lowerPtrAnnotation(IntrinsicInst *II)
static bool lowerIntrinsicToFunction(IntrinsicInst *Intrinsic, const TargetTransformInfo &TTI)
static bool isAbortKHRBuiltin(const Function &F)
static SmallVector< Metadata * > parseAnnotation(Value *I, const std::string &Anno, LLVMContext &Ctx, Type *Int32Ty)
static std::string fixMultiOutputConstraintString(StringRef Constraints)
static void rewriteAbortKHRCall(CallInst *CI)
static void addFunctionTypeMutation(NamedMDNode *NMD, SmallVector< std::pair< int, Type * > > ChangedTys, StringRef Name, StringRef AsmConstraints="")
static bool toSpvLifetimeIntrinsic(IntrinsicInst *II, Intrinsic::ID NewID)
static void lowerExpectAssume(IntrinsicInst *II)
static Function * getOrCreateFunction(Module *M, Type *RetTy, ArrayRef< Type * > ArgTypes, StringRef Name)
#define SPIRV_WAS_AVAILABLE_EXTERNALLY_ATTR
This class represents an incoming formal argument to a Function.
Represent a constant reference to an array (0 or more elements consecutively in memory),...
LLVM Basic Block Representation.
static BasicBlock * Create(LLVMContext &Context, const Twine &Name="", Function *Parent=nullptr, BasicBlock *InsertBefore=nullptr)
Creates a new BasicBlock.
Function * getCalledFunction() const
Returns the function called, or null if this is an indirect function invocation or the function signa...
bool isByValArgument(unsigned ArgNo) const
Determine whether this argument is passed by value.
Type * getParamByValType(unsigned ArgNo) const
Extract the byval type for a call or parameter.
Value * getArgOperand(unsigned i) const
FunctionType * getFunctionType() const
void setCalledFunction(Function *Fn)
Sets the function called, including updating the function type.
This class represents a function call, abstracting a target machine's calling convention.
Predicate
This enumeration lists the possible predicates for CmpInst subclasses.
This is the shared class of boolean and integer constants.
static LLVM_ABI Constant * getNullValue(Type *Ty)
Constructor to create a '0' constant of arbitrary type.
Constrained floating point compare intrinsics.
LLVM_ABI FCmpInst::Predicate getPredicate() const
Diagnostic information for unsupported feature in backend.
Class to represent fixed width SIMD vectors.
unsigned getNumElements() const
A handy container for a FunctionType+Callee-pointer pair, which can be passed around as a single enti...
static LLVM_ABI FunctionType * get(Type *Result, ArrayRef< Type * > Params, bool isVarArg)
This static method is the primary way of constructing a FunctionType.
static Function * Create(FunctionType *Ty, LinkageTypes Linkage, unsigned AddrSpace, const Twine &N="", Module *M=nullptr)
FunctionType * getFunctionType() const
Returns the FunctionType for me.
bool isIntrinsic() const
isIntrinsic - Returns true if the function's name starts with "llvm.".
Type * getReturnType() const
Returns the type of the ret val.
void setCallingConv(CallingConv::ID CC)
Argument * getArg(unsigned i) const
Module * getParent()
Get the module that this global value is contained inside of...
void setDSOLocal(bool Local)
@ ExternalLinkage
Externally visible function.
LLVM_ABI CallInst * CreateIntrinsicWithoutFolding(Intrinsic::ID ID, ArrayRef< Type * > OverloadTypes, ArrayRef< Value * > Args, FMFSource FMFSource={}, const Twine &Name="", ArrayRef< OperandBundleDef > OpBundles={})
Create a call to intrinsic ID with Args, mangled using OverloadTypes.
LLVM_ABI Value * CreateVectorSplat(unsigned NumElts, Value *V, const Twine &Name="")
Return a vector value that contains.
Value * CreateLShr(Value *LHS, Value *RHS, const Twine &Name="", bool isExact=false)
ReturnInst * CreateRet(Value *V)
Create a 'ret <val>' instruction.
Value * CreateSub(Value *LHS, Value *RHS, const Twine &Name="", bool HasNUW=false, bool HasNSW=false)
Value * CreateShl(Value *LHS, Value *RHS, const Twine &Name="", bool HasNUW=false, bool HasNSW=false)
CallInst * CreateMemSet(Value *Ptr, Value *Val, uint64_t Size, MaybeAlign Align, bool isVolatile=false, const AAMDNodes &AAInfo=AAMDNodes())
Create and insert a memset to the specified pointer and the specified value.
ReturnInst * CreateRetVoid()
Create a 'ret void' instruction.
LLVM_ABI Value * CreateIntrinsic(Intrinsic::ID ID, ArrayRef< Type * > OverloadTypes, ArrayRef< Value * > Args, FMFSource FMFSource={}, const Twine &Name="", ArrayRef< OperandBundleDef > OpBundles={}, function_ref< void(CallInst *)> SetFn=[](CallInst *) {})
Variant to create a possibly constant-folded intrinsic.
void SetInsertPoint(BasicBlock *TheBB)
This specifies that created instructions should be appended to the end of the specified block.
Value * CreateOr(Value *LHS, Value *RHS, const Twine &Name="", bool IsDisjoint=false)
ConstantInt * getInt(const APInt &AI)
Get a constant integer value.
Value * CreateURem(Value *LHS, Value *RHS, const Twine &Name="")
This provides a uniform API for creating instructions and inserting them into a basic block: either a...
static LLVM_ABI InlineAsm * get(FunctionType *Ty, StringRef AsmString, StringRef Constraints, bool hasSideEffects, bool isAlignStack=false, AsmDialect asmDialect=AD_ATT, bool canThrow=false)
InlineAsm::get - Return the specified uniqued inline asm string.
LLVM_ABI const Module * getModule() const
Return the module owning the function this instruction belongs to or nullptr it the function does not...
A wrapper class for inspecting calls to intrinsic functions.
Intrinsic::ID getIntrinsicID() const
Return the intrinsic ID of this intrinsic.
LLVM_ABI void LowerIntrinsicCall(CallInst *CI)
Replace a call to the specified intrinsic function.
This is an important class for using LLVM in a threaded context.
static MDTuple * get(LLVMContext &Context, ArrayRef< Metadata * > MDs)
static LLVM_ABI MDString * get(LLVMContext &Context, StringRef Str)
This class wraps the llvm.memset and llvm.memset.inline intrinsics.
ModulePass class - This class is used to implement unstructured interprocedural optimizations and ana...
A Module instance is used to store all the information related to an LLVM module.
LLVMContext & getContext() const
Get the global data context.
NamedMDNode * getOrInsertNamedMetadata(StringRef Name)
Return the named MDNode in the module with the specified name.
Module * getParent()
Get the module that holds this named metadata collection.
LLVM_ABI void addOperand(MDNode *M)
static LLVM_ABI PoisonValue * get(Type *T)
Static factory methods - Return an 'poison' object of the specified type.
A set of analyses that are preserved following a run of a transformation pass.
static PreservedAnalyses none()
Convenience factory function for the empty preserved set.
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
PreservedAnalyses run(Module &M, ModuleAnalysisManager &AM)
bool canUseExtension(SPIRV::Extension::Extension E) const
TargetTransformInfo getTargetTransformInfo(const Function &F) const override
Get a TargetTransformInfo implementation for the target.
reference emplace_back(ArgTypes &&... Args)
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
Represent a constant reference to a string, i.e.
std::string str() const
Get the contents as an std::string.
const Triple & getTargetTriple() const
const STC & getSubtarget(const Function &F) const
This method returns a pointer to the specified type of TargetSubtargetInfo.
VendorType getVendor() const
Get the parsed vendor type of this triple.
The instances of the Type class are immutable: once they are created, they are never changed.
static LLVM_ABI IntegerType * getInt32Ty(LLVMContext &C)
bool isAggregateType() const
Return true if the type is an aggregate type.
LLVM Value Representation.
Type * getType() const
All values are typed, get the type of this value.
LLVM_ABI void setName(const Twine &Name)
Change the name of the value.
LLVM_ABI void replaceAllUsesWith(Value *V)
Change all uses of this to point to a new Value.
LLVM_ABI StringRef getName() const
Return a constant reference to the value's name.
LLVM_ABI void takeName(Value *V)
Transfer the name from V to this value.
Type * getElementType() const
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
@ SPIR_FUNC
Used for SPIR non-kernel device functions.
@ C
The default llvm calling convention, compatible with C.
@ BasicBlock
Various leaf nodes.
This namespace contains an enum with a value for every intrinsic/builtin function known by LLVM.
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.
LLVM_ABI Function * getOrInsertDeclaration(Module *M, ID id, ArrayRef< Type * > OverloadTys={})
Look up the Function declaration of the intrinsic id in the Module M.
LLVM_ABI bool isTargetIntrinsic(ID IID)
isTargetIntrinsic - Returns true if IID is an intrinsic specific to a certain target.
std::string lookupBuiltinNameHelper(StringRef DemangledCall, FPDecorationId *DecorationId)
Parses the name part of the demangled builtin call.
friend class Instruction
Iterator for Instructions in a `BasicBlock.
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.
FunctionAddr VTableAddr uintptr_t uintptr_t Int32Ty
LLVM_ABI bool getConstantStringInfo(const Value *V, StringRef &Str, bool TrimAtNul=true)
This function computes the length of a null-terminated C string pointed to by V.
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...
bool sortBlocks(Function &F)
LLVM_ABI void SplitString(StringRef Source, SmallVectorImpl< StringRef > &OutFragments, StringRef Delimiters=" \t\n\v\f\r")
SplitString - Split up the specified string according to the specified delimiters,...
Function * getOrCreateBackendServiceFunction(Module &M)
std::string getOclOrSpirvBuiltinDemangledName(StringRef Name)
auto dyn_cast_or_null(const Y &Val)
OutputIt transform(R &&Range, OutputIt d_first, UnaryFunction F)
Wrapper function around std::transform to apply a function to a range and store the result elsewhere.
bool any_of(R &&range, UnaryPredicate P)
Provide wrappers to std::any_of which take ranges instead of having to pass begin/end explicitly.
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
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...
LLVM_ATTRIBUTE_VISIBILITY_DEFAULT AnalysisKey InnerAnalysisManagerProxy< AnalysisManagerT, IRUnitT, ExtraArgTs... >::Key
LLVM_ABI unsigned changeToUnreachable(Instruction *I, bool PreserveLCSSA=false, DomTreeUpdater *DTU=nullptr, MemorySSAUpdater *MSSAU=nullptr)
Insert an unreachable instruction before the specified instruction, making it and the rest of the cod...
@ Ref
The access may reference the value stored in memory.
IRBuilder(LLVMContext &, FolderTy, InserterTy, MDNode *, ArrayRef< OperandBundleDef >) -> IRBuilder< FolderTy, InserterTy >
void replace(R &&Range, const T &OldValue, const T &NewValue)
Provide wrappers to std::replace which take ranges instead of having to pass begin/end explicitly.
FunctionAddr VTableAddr Next
constexpr unsigned BitWidth
ValueMap< const Value *, WeakTrackingVH > ValueToValueMapTy
LLVM_ABI void expandMemSetAsLoop(MemSetInst *MemSet, const TargetTransformInfo *TTI=nullptr)
Expand MemSet as a loop.
LLVM_ABI void CloneFunctionInto(Function *NewFunc, const Function *OldFunc, ValueToValueMapTy &VMap, CloneFunctionChangeType Changes, SmallVectorImpl< ReturnInst * > &Returns, const char *NameSuffix="", ClonedCodeInfo *CodeInfo=nullptr, ValueMapTypeRemapper *TypeMapper=nullptr, ValueMaterializer *Materializer=nullptr)
Clone OldFunc into NewFunc, transforming the old arguments into references to VMap values.
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
bool to_integer(StringRef S, N &Num, unsigned Base=0)
Convert the string S to an integer of the specified type using the radix Base. If Base is 0,...
ModulePass * createSPIRVPrepareFunctionsPass(const SPIRVTargetMachine &TM)
constexpr detail::IsaCheckPredicate< Types... > IsaPred
Function object wrapper for the llvm::isa type check.
AnalysisManager< Module > ModuleAnalysisManager
Convenience typedef for the Module analysis manager.
Implement std::hash so that hash_code can be used in STL containers.