58#include <system_error>
64#define DEBUG_TYPE "instrumentor"
70 "instrumentor-write-config-file",
72 "Write the instrumentor configuration into the specified JSON file"),
77 ConfigFiles(
"instrumentor-read-config-files",
78 cl::desc(
"Read the instrumentor configuration from the "
79 "specified JSON files (comma separated)"),
85 "instrumentor-read-config-paths-file",
86 cl::desc(
"Read the instrumentor configuration file "
87 "paths from the specified file (newline separated)"),
92template <
typename IRBuilderTy>
void ensureDbgLoc(IRBuilderTy &IRB) {
93 if (IRB.getCurrentDebugLocation())
95 auto *BB = IRB.GetInsertBlock();
96 if (
auto *SP = BB->getParent()->getSubprogram())
97 IRB.SetCurrentDebugLocation(
DILocation::get(BB->getContext(), 0, 0, SP));
103template <
typename IRBTy>
105 bool AllowTruncate =
false) {
108 Type *VTy = V->getType();
113 TypeSize RequestedSize =
DL.getTypeSizeInBits(Ty);
114 TypeSize ValueSize =
DL.getTypeSizeInBits(VTy);
115 bool ShouldTruncate = RequestedSize < ValueSize;
116 if (ShouldTruncate && !AllowTruncate)
118 if (ShouldTruncate && AllowTruncate) {
122 IntV = IRB.CreateBitCast(V, IRB.getIntNTy(ValueSize));
123 return tryToCast(IRB,
124 IRB.CreateIntCast(IntV, IRB.getIntNTy(RequestedSize),
126 Ty,
DL, AllowTruncate);
129 return IRB.CreatePointerBitCastOrAddrSpaceCast(V, Ty);
131 return IRB.CreateIntCast(V, Ty,
false);
135 return tryToCast(IRB, IRB.CreateBitCast(V, IRB.getIntNTy(ValueSize)), Ty,
140 if (VTy->
isIntegerTy() && Ty->isFloatingPointTy()) {
141 if (ValueSize == RequestedSize)
142 return IRB.CreateBitCast(V, Ty);
145 IRB.CreateIntCast(V, IRB.getIntNTy(RequestedSize),
false),
146 Ty,
DL, AllowTruncate);
148 return IRB.CreateBitOrPointerCast(V, Ty);
152template <
typename Ty>
154 return ConstantInt::get(
IT, Val, IsSigned);
159class InstrumentorImpl final {
164 : IConf(IConf), M(M), IIRB(IIRB) {}
172 InstChoicesPRE.clear();
173 InstChoicesPOST.clear();
174 ParsedFunctionRegex =
Regex();
179 bool shouldInstrumentTarget();
182 bool shouldInstrumentFunction(
Function &Fn);
190 bool instrumentFunction(
Function &Fn);
191 bool instrumentModule();
202 Regex ParsedFunctionRegex;
220 Twine(
"failed to parse ") + Name +
" regex: " + ErrMsg,
DS_Error));
228bool InstrumentorImpl::shouldInstrumentTarget() {
230 const bool IsGPU =
T.
isAMDGPU() ||
T.isNVPTX();
232 bool RegexMatches =
true;
235 RegexMatches = RX.
match(
T.str());
238 return ((IsGPU && IConf.
GPUEnabled->getBool()) ||
243bool InstrumentorImpl::shouldInstrumentFunction(Function &Fn) {
246 bool RegexMatches =
true;
247 if (ParsedFunctionRegex.
isValid())
253bool InstrumentorImpl::shouldInstrumentGlobalVariable(GlobalVariable &GV) {
258bool InstrumentorImpl::instrumentInstruction(Instruction &
I,
259 InstrumentationCaches &ICaches) {
270 if (
auto *IO = InstChoicesPRE.lookup(
I.getOpcode())) {
271 IIRB.
IRB.SetInsertPoint(&
I);
272 ensureDbgLoc(IIRB.
IRB);
273 IO->instrument(IPtr,
Changed, IConf, IIRB, ICaches);
276 if (
auto *IO = InstChoicesPOST.lookup(
I.getOpcode())) {
277 IIRB.
IRB.SetInsertPoint(
I.getNextNode());
278 ensureDbgLoc(IIRB.
IRB);
279 IO->instrument(IPtr,
Changed, IConf, IIRB, ICaches);
286bool InstrumentorImpl::instrumentFunction(Function &Fn) {
288 if (!shouldInstrumentFunction(Fn))
291 InstrumentationCaches ICaches;
293 ReversePostOrderTraversal<Function *> RPOT(&Fn);
294 for (
auto &It : RPOT) {
296 Changed |= instrumentInstruction(
I, ICaches);
298 auto *TI = It->getTerminator();
299 if (!TI->getNumSuccessors())
304 for (
auto &[Name, IO] :
311 IIRB.
IRB.SetInsertPoint(
312 cast<Function>(FPtr)->getEntryBlock().getFirstNonPHIOrDbgOrAlloca());
313 ensureDbgLoc(IIRB.
IRB);
314 IO->instrument(FPtr,
Changed, IConf, IIRB, ICaches);
318 for (
auto &[Name, IO] :
325 for (Instruction *FinalTI : FinalTIs) {
326 IIRB.
IRB.SetInsertPoint(FinalTI);
327 ensureDbgLoc(IIRB.
IRB);
328 IO->instrument(FPtr,
Changed, IConf, IIRB, ICaches);
335bool InstrumentorImpl::instrumentModule() {
338 for (GlobalVariable &GV : M.
globals()) {
341 GV.
getName() ==
"llvm.global_dtors" ||
342 GV.
getName() ==
"llvm.global_ctors")
347 auto CreateYtor = [&](
bool Ctor) {
350 IConf.
getRTName(Ctor ?
"ctor" :
"dtor",
""), M);
353 IIRB.
IRB.SetInsertPoint(EntryBB, EntryBB->begin());
354 ensureDbgLoc(IIRB.
IRB);
355 IIRB.
IRB.CreateRetVoid();
364 InstrumentationCaches ICaches;
366 Function *CtorFn =
nullptr, *DtorFn =
nullptr;
371 Function *&YtorFn = IsPRE ? CtorFn : DtorFn;
372 for (
auto &ChoiceIt : IConf.
IChoices[Loc]) {
373 auto *IO = ChoiceIt.second;
377 YtorFn = CreateYtor(IsPRE);
380 IIRB.
IRB.SetInsertPointPastAllocas(YtorFn);
381 ensureDbgLoc(IIRB.
IRB);
382 Value *YtorPtr = YtorFn;
387 IO->instrument(YtorPtr,
Changed, IConf, IIRB, ICaches);
395 Function *&YtorFn = IsPRE ? CtorFn : DtorFn;
396 for (
auto &ChoiceIt : IConf.
IChoices[Loc]) {
397 auto *IO = ChoiceIt.second;
401 YtorFn = CreateYtor(IsPRE);
404 for (GlobalVariable *GV : Globals) {
405 if (!shouldInstrumentGlobalVariable(*GV))
410 IIRB.
IRB.SetInsertPointPastAllocas(YtorFn);
411 ensureDbgLoc(IIRB.
IRB);
417 IO->instrument(GVPtr,
Changed, IConf, IIRB, ICaches);
426bool InstrumentorImpl::instrument() {
428 if (!shouldInstrumentTarget())
431 StringRef FunctionRegexStr = IConf.
FunctionRegex->getString();
432 ParsedFunctionRegex =
createRegex(FunctionRegexStr,
"function", IIRB.
Ctx);
435 auto RegisterForAllOpcodes = [](
auto &InstChoices,
436 InstrumentationOpportunity *IO) {
437 ArrayRef<unsigned> Opcodes = IO->getAllOpcodes();
439 for (
unsigned Opcode : Opcodes)
440 InstChoices[Opcode] = IO;
443 for (
auto &[Name, IO] :
446 RegisterForAllOpcodes(InstChoicesPRE, IO);
447 for (
auto &[Name, IO] :
450 RegisterForAllOpcodes(InstChoicesPOST, IO);
453 for (Function &Fn : M)
454 Changed |= instrumentFunction(Fn);
460 InstrumentationConfig *IC,
461 InstrumentorIRBuilderTy *IIRB)
462 : FS(FS), UserIConf(IC), UserIIRB(IIRB) {
471 InstrumentorImpl Impl(IConf, IIRB, M);
479 bool MultipleConfigs = ConfigFiles.size() > 1;
482 std::string ConfigFile =
483 ReadConfig && !ConfigFiles.empty() ? ConfigFiles[Idx] :
"";
494 ? OutputConfigFile +
"." + std::to_string(Idx)
501 }
while (++Idx < ConfigFiles.size());
510 std::unique_ptr<InstrumentationConfig> IConfInt(
511 !UserIConf ?
new InstrumentationConfig() :
nullptr);
512 std::unique_ptr<InstrumentorIRBuilderTy> IIRBInt(
513 !UserIIRB ?
new InstrumentorIRBuilderTy(M) :
nullptr);
515 auto *IConf = IConfInt ? IConfInt.get() : UserIConf;
516 auto *IIRB = IIRBInt ? IIRBInt.get() : UserIIRB;
518 auto PA = run(M, *IConf, *IIRB, !UserIConf);
524std::unique_ptr<BaseConfigurationOption>
530 BCO->setBool(DefaultValue);
535std::unique_ptr<BaseConfigurationOption>
542 BCO->setString(DefaultValue);
566 Twine(
"registered two instrumentation opportunities for the same "
577 Function *Fn = IIRB.
IRB.GetInsertBlock()->getParent();
593 if (!BPIO || !BPIO->Enabled) {
595 "Base pointer info disabled but required, passing nullptr.",
602 std::optional<BasicBlock::iterator> IP =
603 BasePtrI->getInsertionPointAfterDef();
605 IIRB.
IRB.SetInsertPoint(*IP);
608 "Base pointer info could not be placed, passing nullptr.",
613 IIRB.
IRB.SetInsertPointPastAllocas(IIRB.
IRB.GetInsertBlock()->getParent());
618 ensureDbgLoc(IIRB.
IRB);
624 BPI = BPIO->instrument(Obj,
Changed, *
this, IIRB, ICaches);
645 if (V.getType()->isVoidTy())
647 return tryToCast(IIRB.
IRB, &V, &Ty,
648 IIRB.
IRB.GetInsertBlock()->getDataLayout());
654 if (V.getType()->isVoidTy())
657 auto *NewVCasted = &NewV;
660 IIRB.
IRB.SetInsertPoint(
I->getNextNode());
661 ensureDbgLoc(IIRB.
IRB);
662 NewVCasted = tryToCast(IIRB.
IRB, &NewV, V.
getType(), IIRB.
DL,
677 for (
auto &It :
IO.IRTArgs) {
680 NumReplaceableArgs += bool(It.Flags & IRTArg::REPLACABLE);
681 MightRequireIndirection |= It.Flags & IRTArg::POTENTIALLY_INDIRECT;
692 "Wrong indirection setting!");
695 for (
auto &It :
IO.IRTArgs) {
724 auto IP = IIRB.
IRB.GetInsertPoint();
727 for (
auto &It :
IO.IRTArgs) {
731 if (!Param || It.NoCache)
733 Param = It.GetterCB(*V, *It.Ty, IConf, IIRB);
736 if (Param->getType()->isVoidTy()) {
738 }
else if (Param->getType()->isAggregateType() ||
739 DL.getTypeSizeInBits(Param->getType()) >
740 DL.getTypeSizeInBits(It.Ty)) {
743 Twine(
"indirection needed for ") + It.Name +
Twine(
" in ") +
745 Twine(
", but not indicated. Instrumentation is skipped"),
749 ForceIndirection =
true;
751 Param = tryToCast(IIRB.
IRB, Param, It.Ty,
DL);
756 if (ForceIndirection) {
757 Function *Fn = IIRB.
IRB.GetInsertBlock()->getParent();
760 for (
auto &It :
IO.IRTArgs) {
768 auto *&CallParam = CallParams[
Offset++];
770 CallParams.
insert(&CallParam + 1, IIRB.
IRB.getInt32(
DL.getTypeStoreSize(
771 CallParam->getType())));
778 CallParam = CachedParam;
783 IIRB.
IRB.CreateStore(CallParam, AI);
784 CallParam = CachedParam = AI;
788 if (!ForceIndirection)
789 IIRB.
IRB.SetInsertPoint(IP);
790 ensureDbgLoc(IIRB.
IRB);
794 IConf.
getRTName(
IO.IP.isPRE() ?
"pre_" :
"post_",
IO.getName(),
795 ForceIndirection ?
"_ind" :
"");
796 auto FC = IIRB.
IRB.GetInsertBlock()->getModule()->getOrInsertFunction(
798 auto *CI = IIRB.
IRB.CreateCall(FC, CallParams);
801 for (
unsigned I = 0, E =
IO.IRTArgs.size();
I < E; ++
I) {
802 if (!
IO.IRTArgs[
I].Enabled)
807 Value *NewValue = FnTy->isVoidTy() || IsCustomReplaceable
812 if (ForceIndirection && !IsCustomReplaceable &&
817 NewValue = IIRB.
IRB.CreateLoad(V->getType(), Q);
819 V =
IO.IRTArgs[
I].SetterCB(*V, *NewValue, IConf, IIRB);
825 if constexpr (std::is_same<Ty, Use>::value)
826 return ValueOrUse.get();
828 return static_cast<Value *
>(&ValueOrUse);
831template <
typename Range>
834 auto *Fn = IIRB.
IRB.GetInsertBlock()->getParent();
835 auto *I32Ty = IIRB.
IRB.getInt32Ty();
841 if (!V->getType()->isSized())
844 ConstantValues.
push_back(getCI(I32Ty, VSize));
845 Types.push_back(I32Ty);
846 ConstantValues.
push_back(getCI(I32Ty, V->getType()->getTypeID()));
847 Types.push_back(I32Ty);
848 if (
uint32_t MisAlign = VSize % 8) {
852 Types.push_back(V->getType());
875 for (
auto [Param, Idx] : Values) {
876 auto *Ptr = IIRB.
IRB.CreateStructGEP(STy, AI, Idx);
877 IIRB.
IRB.CreateStore(Param, Ptr);
882template <
typename Range>
886 auto *Fn = IIRB.
IRB.GetInsertBlock()->getParent();
890 for (
const auto &[Idx, RE] :
enumerate(R)) {
892 if (!V->getType()->isSized())
895 auto VSize =
DL.getTypeAllocSize(V->getType());
896 auto Padding =
alignTo(VSize, 8) - VSize;
898 auto *Ptr = IIRB.
IRB.CreateConstInBoundsGEP1_32(IIRB.
Int8Ty, &Pack,
Offset);
899 auto *NewV = IIRB.
IRB.CreateLoad(V->getType(), Ptr);
909 using namespace std::placeholders;
923 "Number of function arguments (without varargs).",
IRTArg::NONE,
927 IIRB.
PtrTy,
"arguments",
"Description of the arguments.",
935 "Flag to indicate it is the main function.",
965 return getCI(&Ty, std::distance(FRange.begin(), FRange.end()));
981 auto CB = [&](
int Idx,
Value *ReplV) {
1000 return getCI(&Ty, Fn.
getName() ==
"main");
1006 return getCI(&Ty,
I.getOpcode());
1011 return getCI(&Ty, V.getType()->getTypeID());
1017 auto &
DL =
I.getDataLayout();
1018 return getCI(&Ty,
DL.getTypeStoreSize(V.getType()));
1042 IRTArg(IIRB.
PtrTy,
"address",
"The allocated memory address.",
1048 IIRB.
Int64Ty,
"size",
"The allocation size.",
1064 Value *SizeValue =
nullptr;
1070 SizeValue = IIRB.
IRB.CreatePtrToInt(
1071 IIRB.
IRB.CreateGEP(AI.getAllocatedType(), NullPtr,
1072 {IIRB.IRB.getInt32(1)}),
1075 if (AI.isArrayAllocation())
1076 SizeValue = IIRB.
IRB.CreateMul(
1077 SizeValue, IIRB.
IRB.CreateZExtOrBitCast(AI.getArraySize(), &Ty));
1085 auto *NewAI = IIRB.
IRB.CreateAlloca(IIRB.
IRB.getInt8Ty(),
1086 DL.getAllocaAddrSpace(), &NewV);
1087 NewAI->setAlignment(AI.getAlign());
1088 AI.replaceAllUsesWith(NewAI);
1107 IRTArg(IIRB.
PtrTy,
"pointer",
"The accessed pointer.",
1114 "The address space of the accessed pointer.",
1119 "The runtime provided base pointer info.",
1147 "The atomicity ordering of the store.",
1168 return SI.getPointerOperand();
1174 SI.setOperand(
SI.getPointerOperandIndex(), &NewV);
1181 return getCI(&Ty,
SI.getPointerAddressSpace());
1194 return SI.getValueOperand();
1200 auto &
DL =
SI.getDataLayout();
1201 return getCI(&Ty,
DL.getTypeStoreSize(
SI.getValueOperand()->getType()));
1207 return getCI(&Ty,
SI.getAlign().value());
1213 return getCI(&Ty,
SI.getValueOperand()->getType()->getTypeID());
1220 return getCI(&Ty,
uint64_t(
SI.getOrdering()));
1226 return getCI(&Ty,
uint64_t(
SI.getSyncScopeID()));
1232 return getCI(&Ty,
SI.isVolatile());
1242 IRTArg(IIRB.
PtrTy,
"pointer",
"The accessed pointer.",
1249 "The address space of the accessed pointer.",
1254 "The runtime provided base pointer info.",
1284 "The atomicity ordering of the load.",
1305 return LI.getPointerOperand();
1311 LI.setOperand(LI.getPointerOperandIndex(), &NewV);
1318 return getCI(&Ty, LI.getPointerAddressSpace());
1336 auto &
DL = LI.getDataLayout();
1337 return getCI(&Ty,
DL.getTypeStoreSize(LI.getType()));
1343 return getCI(&Ty, LI.getAlign().value());
1349 return getCI(&Ty, LI.getType()->getTypeID());
1356 return getCI(&Ty,
uint64_t(LI.getOrdering()));
1362 return getCI(&Ty,
uint64_t(LI.getSyncScopeID()));
1368 return getCI(&Ty, LI.isVolatile());
1377 "The base pointer in question.",
1381 IIRB.
Int32Ty,
"base_pointer_kind",
1382 "The base pointer kind (argument, global, instruction, unknown).",
1392 return getCI(&Ty, 0);
1394 return getCI(&Ty, 1);
1396 return getCI(&Ty, 2);
1397 return getCI(&Ty, 3);
1407 "The module/translation unit name.",
1438 IIRB.
PtrTy,
"address",
1439 "The address of the global (replaceable for definitions).",
1448 "The size of the declared type of the global.",
1459 IIRB.
Int64Ty,
"initial_value",
"The initial value of the global.",
1468 "Flag to indicate global definitions.",
1492 DL.getDefaultGlobalsAddressSpace());
1497 IIRB.
IRB.CreateStore(&NewV, ShadowGV);
1506 auto MakeInstForConst = [&](
Use &U) {
1512 I = CE->getAsInstruction();
1526 while (!Worklist.
empty()) {
1529 U->set(ReloadMap[
I->getFunction()]);
1532 if (
auto *CI = ConstToInstMap[*U]) {
1533 auto *CIClone = CI->clone();
1536 auto *BB =
PHI->getIncomingBlock(U->getOperandNo());
1537 CIClone->insertBefore(BB->getTerminator()->getIterator());
1539 CIClone->insertBefore(
I->getIterator());
1542 for (
auto &CICUse : CIClone->operands()) {
1550 while (!Worklist.
empty()) {
1552 if (!
Done.insert(U).second)
1554 MakeInstForConst(*U);
1565 if (
II->getIntrinsicID() == Intrinsic::eh_typeid_for)
1568 InsertConsts(
I, *U);
1571 for (
auto &It : ConstToInstMap)
1573 It.second->deleteValue();
1653 "The opcode of the cast instruction.",
1663 return CI.getOperand(0);
1669 return getCI(&Ty, CI.getSrcTy()->getTypeID());
1675 auto &
DL = CI.getDataLayout();
1676 return getCI(&Ty,
DL.getTypeStoreSize(CI.getSrcTy()));
1682 return getCI(&Ty, CI.getDestTy()->getTypeID());
1688 auto &
DL = CI.getDataLayout();
1689 return getCI(&Ty,
DL.getTypeStoreSize(CI.getDestTy()));
1696 return I.getOperand(0);
1702 if (
I.getNumOperands() > 1)
1703 return I.getOperand(1);
1713 const auto ValArgOpts =
1728 "The operation's left operand.", ValArgOpts,
1732 "The operation's right operand. This value is "
1733 "poison for unary operations.",
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
This file defines the StringMap class.
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
static cl::opt< ITMode > IT(cl::desc("IT block support"), cl::Hidden, cl::init(DefaultIT), cl::values(clEnumValN(DefaultIT, "arm-default-it", "Generate any type of IT block"), clEnumValN(RestrictedIT, "arm-restrict-it", "Disallow complex IT blocks")))
This file contains the declarations for the subclasses of Constant, which represent the different fla...
post inline ee instrument
Module.h This file contains the declarations for the Module class.
This header defines various interfaces for pass management in LLVM.
static void readValuePack(const Range &R, Value &Pack, InstrumentorIRBuilderTy &IIRB, function_ref< void(int, Value *)> SetterCB)
static Value * getTypeId(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static constexpr Value * getValue(Ty &ValueOrUse)
static Value * createValuePack(const Range &R, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getOpcode(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Regex createRegex(StringRef Str, StringRef Name, LLVMContext &Ctx)
ConstantRange Range(APInt(BitWidth, Low), APInt(BitWidth, High))
uint64_t IntrinsicInst * II
ModuleAnalysisManager MAM
This file builds on the ADT/GraphTraits.h file to build a generic graph post order iterator.
This file defines the SmallPtrSet class.
This file defines the SmallVector class.
Defines the virtual file system interface vfs::FileSystem.
static unsigned getSize(unsigned Kind)
static LLVM_ABI ArrayType * get(Type *ElementType, uint64_t NumElements)
This static method is the primary way to construct an ArrayType.
static LLVM_ABI Attribute get(LLVMContext &Context, AttrKind Kind, uint64_t Val=0)
Return a uniquified Attribute object.
static BasicBlock * Create(LLVMContext &Context, const Twine &Name="", Function *Parent=nullptr, BasicBlock *InsertBefore=nullptr)
Creates a new BasicBlock.
LLVM_ABI const_iterator getFirstNonPHIOrDbgOrAlloca() const
Returns an iterator to the first instruction in this block that is not a PHINode, a debug intrinsic,...
const Instruction * getTerminator() const LLVM_READONLY
Returns the terminator instruction; assumes that the block is well-formed.
This class represents a function call, abstracting a target machine's calling convention.
static LLVM_ABI Constant * getAddrSpaceCast(Constant *C, Type *Ty, bool OnlyIfReduced=false)
static LLVM_ABI ConstantPointerNull * get(PointerType *T)
Static factory methods - Return objects of the specified value.
static LLVM_ABI Constant * get(StructType *T, ArrayRef< Constant * > V)
This is an important base class in LLVM.
static LLVM_ABI Constant * getAllOnesValue(Type *Ty)
static LLVM_ABI Constant * getNullValue(Type *Ty)
Constructor to create a '0' constant of arbitrary type.
A parsed version of the target data layout string in and methods for querying it.
LLVM_ABI TypeSize getTypeAllocSize(Type *Ty) const
Returns the offset in bytes between successive objects of the specified type, including alignment pad...
Diagnostic information for IR instrumentation reporting.
Class to represent function types.
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)
const BasicBlock & getEntryBlock() const
const DataLayout & getDataLayout() const
Get the data layout of the module this function belongs to.
iterator_range< arg_iterator > args()
bool isIntrinsic() const
isIntrinsic - Returns true if the function's name starts with "llvm.".
LLVMContext & getContext() const
getContext - Return a reference to the LLVMContext associated with this function.
Argument * getArg(unsigned i) const
bool hasFnAttribute(Attribute::AttrKind Kind) const
Return true if the function has the attribute.
StringRef getSection() const
Get the custom section of this global if it has one.
LLVM_ABI bool isDeclaration() const
Return true if the primary definition of this global value is outside of the current translation unit...
LinkageTypes getLinkage() const
ThreadLocalMode getThreadLocalMode() const
unsigned getAddressSpace() const
Module * getParent()
Get the module that this global value is contained inside of...
PointerType * getType() const
Global values are always pointers.
LLVM_ABI const DataLayout & getDataLayout() const
Get the data layout of the module this global belongs to.
@ PrivateLinkage
Like Internal, but omit from symbol table.
@ InternalLinkage
Rename collisions when linking (static functions).
@ WeakODRLinkage
Same, but only replaced by something equivalent.
Type * getValueType() const
const Constant * getInitializer() const
getInitializer - Return the initializer for this global variable.
bool hasInitializer() const
Definitions have initializers, declarations don't.
uint64_t getAlignment() const
FIXME: Remove this function once transition to Align is over.
bool isConstant() const
If the value is a global constant, its value is immutable throughout the runtime execution of the pro...
LLVM_ABI const Function * getFunction() const
Return the function this instruction belongs to.
InstrumentorPass(IntrusiveRefCntPtr< vfs::FileSystem > FS=nullptr, InstrumentationConfig *IC=nullptr, InstrumentorIRBuilderTy *IIRB=nullptr)
Construct an instrumentor pass that will use the instrumentation configuration IC and the IR builder ...
A smart pointer to a reference-counted object that inherits from RefCountedBase or ThreadSafeRefCount...
This is an important class for using LLVM in a threaded context.
LLVM_ABI void diagnose(const DiagnosticInfo &DI)
Report a message to the currently installed diagnostic handler.
An instruction for reading from memory.
static MDTuple * get(LLVMContext &Context, ArrayRef< Metadata * > MDs)
A Module instance is used to store all the information related to an LLVM module.
const Triple & getTargetTriple() const
Get the target triple which is a string describing the target host.
size_t global_size() const
StringRef getName() const
Get a short "name" for the module.
iterator_range< global_iterator > globals()
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.
LLVM_ABI bool isValid(std::string &Error) const
isValid - returns the error encountered during regex compilation, if any.
LLVM_ABI bool match(StringRef String, SmallVectorImpl< StringRef > *Matches=nullptr, std::string *Error=nullptr) const
matches - Match the regex against a given String.
SmallPtrSet - This class implements a set which is optimized for holding SmallSize or less elements.
void reserve(size_type N)
iterator insert(iterator I, T &&Elt)
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.
bool starts_with(StringRef Prefix) const
Check if this string starts with the given Prefix.
Class to represent struct types.
static LLVM_ABI StructType * get(LLVMContext &Context, ArrayRef< Type * > Elements, bool isPacked=false)
This static method is the primary way to create a literal StructType.
const std::string & getTriple() const
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
The instances of the Type class are immutable: once they are created, they are never changed.
bool isPointerTy() const
True if this is an instance of PointerType.
bool isAggregateType() const
Return true if the type is an aggregate type.
bool isFloatingPointTy() const
Return true if this is one of the floating-point types.
bool isIntegerTy() const
True if this is an instance of IntegerType.
A Use represents the edge between a Value definition and its users.
LLVM Value Representation.
Type * getType() const
All values are typed, get the type of this value.
LLVM_ABI bool replaceUsesWithIf(Value *New, llvm::function_ref< bool(Use &U)> ShouldReplace)
Go through the uses list for this definition and make each use point to "V" if the callback ShouldRep...
iterator_range< use_iterator > uses()
LLVM_ABI StringRef getName() const
Return a constant reference to the value's name.
constexpr ScalarTy getFixedValue() const
constexpr bool isFixed() const
Returns true if the quantity is not scaled by vscale.
An efficient, type-erasing, non-owning reference to a callable.
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
@ C
The default llvm calling convention, compatible with C.
initializer< Ty > init(const Ty &Val)
void writeConfigToJSON(InstrumentationConfig &IConf, StringRef OutputFile, LLVMContext &Ctx)
Write the configuration in /p IConf to the file with path OutputFile.
bool readConfigPathsFile(StringRef InputFile, cl::list< std::string > &Configs, LLVMContext &Ctx, vfs::FileSystem &FS)
Read the configuration paths from the file with path InputFile into Configs.
bool readConfigFromJSON(InstrumentationConfig &IConf, StringRef InputFile, LLVMContext &Ctx, vfs::FileSystem &FS)
Read the configuration from the file with path InputFile into /p IConf.
void printRuntimeStub(const InstrumentationConfig &IConf, StringRef StubRuntimeName, LLVMContext &Ctx)
Print a runtime stub file with the implementation of the instrumentation runtime functions correspond...
LLVM_ABI IntrusiveRefCntPtr< FileSystem > getRealFileSystem()
Gets an vfs::FileSystem for the 'real' file system, as seen by the operating system.
This is an optimization pass for GlobalISel generic memory operations.
FunctionAddr VTableAddr Value
MaybeAlign getAlign(const CallInst &I, unsigned Index)
auto enumerate(FirstRange &&First, RestRanges &&...Rest)
Given two or more input ranges, returns a new range whose values are tuples (A, B,...
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
void append_range(Container &C, Range &&R)
Wrapper function to append range R to container C.
constexpr uint64_t alignTo(uint64_t Size, Align A)
Returns a multiple of A needed to store Size bytes.
iterator_range< filter_iterator< detail::IterOfRange< RangeT >, PredicateT > > make_filter_range(RangeT &&Range, PredicateT Pred)
Convenience function that takes a range of elements and a predicate, and return a new filter_iterator...
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_ABI raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
LLVM_ABI const Value * getUnderlyingObjectAggressive(const Value *V)
Like getUnderlyingObject(), but will try harder to find a single underlying object.
LLVM_ABI void appendToGlobalCtors(Module &M, Function *F, int Priority, Constant *Data=nullptr)
Append F to the list of global ctors of module M with the given Priority.
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
iterator_range< pointer_iterator< WrappedIteratorT > > make_pointer_range(RangeT &&Range)
LLVM_ABI void appendToGlobalDtors(Module &M, Function *F, int Priority, Constant *Data=nullptr)
Same as appendToGlobalCtors(), but for global dtors.
DEMANGLE_ABI std::string demangle(std::string_view MangledName)
Attempt to demangle a string using different demangling schemes.
LLVM_ABI bool verifyModule(const Module &M, raw_ostream *OS=nullptr, bool *BrokenDebugInfo=nullptr)
Check a module for errors.
AnalysisManager< Module > ModuleAnalysisManager
Convenience typedef for the Module analysis manager.
This struct is a compact representation of a valid (power of two) or undefined (0) alignment.
static void populate(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
void init(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB, ConfigTy *UserConfig=nullptr)
}
BaseConfigTy< ConfigKind > ConfigTy
static Value * getSize(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * setSize(Value &V, Value &NewV, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getAlignment(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static std::unique_ptr< BaseConfigurationOption > createStringOption(InstrumentationConfig &IC, StringRef Name, StringRef Description, StringRef DefaultValue)
Create a string option with Name name, Description description and DefaultValue as string default val...
static std::unique_ptr< BaseConfigurationOption > createBoolOption(InstrumentationConfig &IC, StringRef Name, StringRef Description, bool DefaultValue)
Create a boolean option with Name name, Description description and DefaultValue as boolean default v...
static void populate(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getPointerKind(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * setValueNoop(Value &V, Value &NewV, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
This is necessary to produce a return value that can be used by other IOs.
BaseConfigTy< ConfigKind > ConfigTy
void init(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB, ConfigTy *UserConfig=nullptr)
void init(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB, ConfigTy *UserConfig=nullptr)
CastIO {.
static Value * getResultTypeId(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getInputSize(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static void populate(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getResultSize(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getInput(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
BaseConfigTy< ConfigKind > ConfigTy
static Value * getInputTypeId(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
llvm::instrumentor::FunctionIO::ConfigTy Config
static void populate(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
Value * setArguments(Value &V, Value &NewV, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getFunctionAddress(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * isMainFunction(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
Value * getArguments(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
Value * getNumArguments(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getFunctionName(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
void init(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB, ConfigTy *UserConfig=nullptr)
FunctionIO {.
static Value * setAddress(Value &V, Value &NewV, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
void init(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB, ConfigTy *UserConfig=nullptr)
static Value * getAS(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getAlignment(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getInitialValue(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * isDefinition(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getDeclaredSize(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getSymbolName(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static void populate(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getAddress(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
BaseConfigTy< ConfigKind > ConfigTy
static Value * isConstant(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
bool isReplacable(IRTArg &IRTA) const
Return whether the IRTA argument can be replaced.
IRTCallDescription(InstrumentationOpportunity &IO, Type *RetTy=nullptr)
Construct an instrumentation function description linked to the IO instrumentation opportunity and Re...
bool MightRequireIndirection
Whether any argument may require indirection.
CallInst * createLLVMCall(Value *&V, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB, const DataLayout &DL, InstrumentationCaches &ICaches)
Create a call instruction that calls to the instrumentation function and passes the corresponding arg...
Type * RetTy
The return type of the instrumentation function.
InstrumentationOpportunity & IO
The instrumentation opportunity which it is linked to.
FunctionType * createLLVMSignature(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB, const DataLayout &DL, bool ForceIndirection)
Create the type of the instrumentation function.
unsigned NumReplaceableArgs
The number of arguments that can be replaced.
bool RequiresIndirection
Whether the function requires indirection in some argument.
bool isPotentiallyIndirect(IRTArg &IRTA) const
Return whether the function may have any indirect argument.
Helper that represent the caches for instrumentation call arguments.
DenseMap< std::tuple< unsigned, StringRef, StringRef >, Value * > DirectArgCache
A cache for direct and indirect arguments.
DenseMap< std::tuple< unsigned, StringRef, StringRef >, Value * > IndirectArgCache
The class that contains the configuration for the instrumentor.
virtual void populate(InstrumentorIRBuilderTy &IIRB)
Populate the instrumentation opportunities.
void addChoice(InstrumentationOpportunity &IO, LLVMContext &Ctx)
Register instrumentation opportunity IO.
Constant * getGlobalString(StringRef S, InstrumentorIRBuilderTy &IIRB)
DenseMap< Value *, Value * > UnderlyingObjsMap
Map to remember underlying objects for pointers.
std::unique_ptr< BaseConfigurationOption > HostEnabled
std::unique_ptr< BaseConfigurationOption > DemangleFunctionNames
void init(InstrumentorIRBuilderTy &IIRB)
Initialize the config to a clean base state without loosing cached values that can be reused across c...
DenseMap< std::pair< Value *, Function * >, Value * > BasePointerInfoMap
Map to remember base pointer info for values in a specific function.
EnumeratedArray< MapVector< StringRef, InstrumentationOpportunity * >, InstrumentationLocation::KindTy > IChoices
The map registered instrumentation opportunities.
std::unique_ptr< BaseConfigurationOption > GPUEnabled
DenseMap< Constant *, GlobalVariable * > ConstantGlobalsCache
Mapping from constants to globals with the constant as initializer.
Value * getBasePointerInfo(Value &V, InstrumentorIRBuilderTy &IIRB)
Return the base pointer info for V.
std::unique_ptr< BaseConfigurationOption > RuntimeStubsFile
StringRef getRTName() const
Get the runtime prefix for the instrumentation runtime functions.
void addBaseChoice(BaseConfigurationOption *BCO)
Add the base configuration option BCO into the list of base options.
std::unique_ptr< BaseConfigurationOption > FunctionRegex
std::unique_ptr< BaseConfigurationOption > TargetRegex
bool isPRE() const
Return whether the instrumentation location is before the event occurs.
Base class for instrumentation opportunities.
InstrumentationLocation::KindTy getLocationKind() const
Get the location kind of the instrumentation opportunity.
static Value * getIdPre(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
Get the opportunity identifier for the pre and post positions.
static Value * forceCast(Value &V, Type &Ty, InstrumentorIRBuilderTy &IIRB)
Helpers to cast values, pass them to the runtime, and replace them.
static int32_t getIdFromEpoch(uint32_t CurrentEpoch)
}
static Value * getIdPost(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getValue(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * replaceValue(Value &V, Value &NewV, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
virtual StringRef getName() const =0
Get the name of the instrumentation opportunity.
SmallVector< IRTArg > IRTArgs
The list of possible arguments for the instrumentation runtime function.
void addCommonArgs(InstrumentationConfig &IConf, LLVMContext &Ctx, bool PassId)
}
An IR builder augmented with extra information for the instrumentor pass.
IRBuilder< ConstantFolder, IRBuilderCallbackInserter > IRB
The underlying IR builder with insertion callback.
unsigned Epoch
The current epoch number.
AllocaInst * getAlloca(Function *Fn, Type *Ty, bool MatchType=false)
Get a temporary alloca to communicate (large) values with the runtime.
void returnAllocas()
Return the temporary allocas.
DenseMap< Instruction *, unsigned > NewInsts
A mapping from instrumentation instructions to the epoch they have been created.
void eraseLater(Instruction *I)
Save instruction I to be erased later.
static Value * getValueSize(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getSyncScopeId(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getAtomicityOrdering(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
virtual Type * getValueType(InstrumentorIRBuilderTy &IIRB) const
}
static Value * getValue(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getAlignment(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getPointer(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
Getters and setters for the arguments of the instrumentation function for the load opportunity.
static Value * isVolatile(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getBasePointerInfo(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * setPointer(Value &V, Value &NewV, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
BaseConfigTy< ConfigKind > ConfigTy
static Value * getPointerAS(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static void populate(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
}
static Value * getValueTypeId(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
void init(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB, ConfigTy *UserConfig=nullptr)
Initialize the load opportunity using the instrumentation config IConf and the user config UserConfig...
static Value * getModuleName(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static void populate(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getTargetTriple(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
void init(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB, ConfigTy *UserConfig=nullptr)
BaseConfigTy< ConfigKind > ConfigTy
static Value * getLeft(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
}
static Value * getRight(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
void init(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB, ConfigTy *UserConfig=nullptr)
static void populate(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
BaseConfigTy< ConfigKind > ConfigTy
static Value * getPointer(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
Getters and setters for the arguments of the instrumentation function for the store opportunity.
static void populate(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
}
static Value * getValueTypeId(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
virtual Type * getValueType(InstrumentorIRBuilderTy &IIRB) const
}
static Value * getSyncScopeId(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getPointerAS(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getAlignment(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getValue(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * setPointer(Value &V, Value &NewV, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * isVolatile(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getValueSize(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
BaseConfigTy< ConfigKind > ConfigTy
void init(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB, ConfigTy *UserConfig=nullptr)
Initialize the store opportunity using the instrumentation config IConf and the user config UserConfi...
static Value * getBasePointerInfo(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static Value * getAtomicityOrdering(Value &V, Type &Ty, InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
static void populate(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB)
void init(InstrumentationConfig &IConf, InstrumentorIRBuilderTy &IIRB, ConfigTy *UserConfig=nullptr)
UnreachableIO {.
BaseConfigTy< ConfigKind > ConfigTy