diff options
author | Qin Zhao <zhaoqin@google.com> | 2016-06-03 02:33:04 +0000 |
---|---|---|
committer | Qin Zhao <zhaoqin@google.com> | 2016-06-03 02:33:04 +0000 |
commit | c14c249343893cdc1d93ef1856648910221d4682 (patch) | |
tree | 6c980f4dbb1fbb8115ee5e6a7e6a13f5a3069879 /llvm/lib/Transforms | |
parent | 50ce381a5cf5c158771ad5500d28113594849460 (diff) | |
download | bcm5719-llvm-c14c249343893cdc1d93ef1856648910221d4682.tar.gz bcm5719-llvm-c14c249343893cdc1d93ef1856648910221d4682.zip |
[esan|cfrag] Instrument GEP instr for struct field access.
Summary:
Instrument GEP instruction for counting the number of struct field
address calculation to approximate the number of struct field accesses.
Adds test struct_field_count_basic.ll to test the struct field
instrumentation.
Reviewers: bruening, aizatsky
Subscribers: junbuml, zhaoqin, llvm-commits, eugenis, vitalybuka, kcc, bruening
Differential Revision: http://reviews.llvm.org/D20892
llvm-svn: 271619
Diffstat (limited to 'llvm/lib/Transforms')
-rw-r--r-- | llvm/lib/Transforms/Instrumentation/EfficiencySanitizer.cpp | 61 |
1 files changed, 61 insertions, 0 deletions
diff --git a/llvm/lib/Transforms/Instrumentation/EfficiencySanitizer.cpp b/llvm/lib/Transforms/Instrumentation/EfficiencySanitizer.cpp index 36f361d7459..d29d400e72e 100644 --- a/llvm/lib/Transforms/Instrumentation/EfficiencySanitizer.cpp +++ b/llvm/lib/Transforms/Instrumentation/EfficiencySanitizer.cpp @@ -63,6 +63,8 @@ STATISTIC(NumFastpaths, "Number of instrumented fastpaths"); STATISTIC(NumAccessesWithIrregularSize, "Number of accesses with a size outside our targeted callout sizes"); STATISTIC(NumIgnoredStructs, "Number of ignored structs"); +STATISTIC(NumIgnoredGEPs, "Number of ignored GEP instructions"); +STATISTIC(NumInstrumentedGEPs, "Number of instrumented GEP instructions"); static const uint64_t EsanCtorAndDtorPriority = 0; static const char *const EsanModuleCtorName = "esan.module_ctor"; @@ -145,6 +147,7 @@ private: bool runOnFunction(Function &F, Module &M); bool instrumentLoadOrStore(Instruction *I, const DataLayout &DL); bool instrumentMemIntrinsic(MemIntrinsic *MI); + bool instrumentGetElementPtr(Instruction *I, Module &M); bool shouldIgnoreMemoryAccess(Instruction *I); int getMemoryAccessFuncIndex(Value *Addr, const DataLayout &DL); Value *appToShadow(Value *Shadow, IRBuilder<> &IRB); @@ -171,6 +174,9 @@ private: Function *MemmoveFn, *MemcpyFn, *MemsetFn; Function *EsanCtorFunction; Function *EsanDtorFunction; + // Remember the counter variable for each struct type to avoid + // recomputing the variable name later during instrumentation. + std::map<Type *, GlobalVariable *> StructTyMap; }; } // namespace @@ -321,6 +327,9 @@ GlobalVariable *EfficiencySanitizer::createCacheFragInfoGV( ConstantAggregateZero::get(CounterArrayTy), CounterNameStr); + // Remember the counter variable for each struct type. + StructTyMap.insert(std::pair<Type *, GlobalVariable *>(StructTy, Counters)); + // FieldTypeNames. // We pass the field type name array to the runtime for better reporting. auto *TypeNameArrayTy = ArrayType::get(Int8PtrTy, StructTy->getNumElements()); @@ -477,6 +486,7 @@ bool EfficiencySanitizer::runOnFunction(Function &F, Module &M) { return false; SmallVector<Instruction *, 8> LoadsAndStores; SmallVector<Instruction *, 8> MemIntrinCalls; + SmallVector<Instruction *, 8> GetElementPtrs; bool Res = false; const DataLayout &DL = M.getDataLayout(); @@ -488,6 +498,8 @@ bool EfficiencySanitizer::runOnFunction(Function &F, Module &M) { LoadsAndStores.push_back(&Inst); else if (isa<MemIntrinsic>(Inst)) MemIntrinCalls.push_back(&Inst); + else if (isa<GetElementPtrInst>(Inst)) + GetElementPtrs.push_back(&Inst); } } @@ -503,6 +515,12 @@ bool EfficiencySanitizer::runOnFunction(Function &F, Module &M) { } } + if (Options.ToolType == EfficiencySanitizerOptions::ESAN_CacheFrag) { + for (auto Inst : GetElementPtrs) { + Res |= instrumentGetElementPtr(Inst, M); + } + } + return Res; } @@ -591,6 +609,49 @@ bool EfficiencySanitizer::instrumentMemIntrinsic(MemIntrinsic *MI) { return Res; } +bool EfficiencySanitizer::instrumentGetElementPtr(Instruction *I, Module &M) { + GetElementPtrInst *GepInst = dyn_cast<GetElementPtrInst>(I); + if (GepInst == nullptr || !isa<StructType>(GepInst->getSourceElementType()) || + StructTyMap.count(GepInst->getSourceElementType()) == 0 || + !GepInst->hasAllConstantIndices() || + // Only handle simple struct field GEP. + GepInst->getNumIndices() != 2) { + ++NumIgnoredGEPs; + return false; + } + StructType *StructTy = dyn_cast<StructType>(GepInst->getSourceElementType()); + if (shouldIgnoreStructType(StructTy)) { + ++NumIgnoredGEPs; + return false; + } + ++NumInstrumentedGEPs; + // Use the last index as the index within the struct. + ConstantInt *Idx = dyn_cast<ConstantInt>(GepInst->getOperand(2)); + if (Idx == nullptr || Idx->getZExtValue() > StructTy->getNumElements()) + return false; + + GlobalVariable *CounterArray = StructTyMap[StructTy]; + if (CounterArray == nullptr) + return false; + IRBuilder<> IRB(I); + Constant *Indices[2]; + // Xref http://llvm.org/docs/LangRef.html#i-getelementptr and + // http://llvm.org/docs/GetElementPtr.html. + // The first index of the GEP instruction steps through the first operand, + // i.e., the array itself. + Indices[0] = ConstantInt::get(IRB.getInt32Ty(), 0); + // The second index is the index within the array. + Indices[1] = ConstantInt::get(IRB.getInt32Ty(), Idx->getZExtValue()); + Constant *Counter = + ConstantExpr::getGetElementPtr(ArrayType::get(IRB.getInt64Ty(), + StructTy->getNumElements()), + CounterArray, Indices); + Value *Load = IRB.CreateLoad(Counter); + IRB.CreateStore(IRB.CreateAdd(Load, ConstantInt::get(IRB.getInt64Ty(), 1)), + Counter); + return true; +} + int EfficiencySanitizer::getMemoryAccessFuncIndex(Value *Addr, const DataLayout &DL) { Type *OrigPtrTy = Addr->getType(); |