Static Value-Flow Analysis
Loading...
Searching...
No Matches
Public Member Functions | Protected Member Functions | Private Attributes | Friends | List of all members
SVF::SymbolTableBuilder Class Reference

#include <SymbolTableBuilder.h>

Public Member Functions

 SymbolTableBuilder (SVFIR *ir)
 Constructor.
 
void buildMemModel ()
 Start building memory model.
 
u32_t getNumOfElements (const Type *ety)
 Return size of this object based on LLVM value.
 

Protected Member Functions

void collectSVFTypeInfo (const Value *val)
 collect the syms
 
void collectSym (const Value *val)
 
void collectVal (const Value *val)
 
void collectObj (const Value *val)
 
void collectRet (const Function *val)
 
void collectVararg (const Function *val)
 
void handleGlobalCE (const GlobalVariable *G)
 Handle constant expression.
 
void handleGlobalInitializerCE (const Constant *C)
 
void handleCE (const Value *val)
 
LLVMModuleSetllvmModuleSet ()
 
ObjTypeInferencegetTypeInference ()
 
const TypeinferObjType (const Value *startValue)
 Forward collect all possible infer sites starting from a value.
 
const TypeinferTypeOfHeapObjOrStaticObj (const Instruction *inst)
 Get the reference type of heap/static object from an allocation site.
 
ObjTypeInfocreateObjTypeInfo (const Value *val)
 Create an objectInfo based on LLVM value.
 
void initTypeInfo (ObjTypeInfo *typeinfo, const Value *value, const Type *ty)
 Initialize TypeInfo based on LLVM Value.
 
void analyzeObjType (ObjTypeInfo *typeinfo, const Value *val)
 Analyse types of all flattened fields of this object.
 
u32_t analyzeHeapObjType (ObjTypeInfo *typeinfo, const Value *val)
 Analyse types of heap and static objects.
 
void analyzeStaticObjType (ObjTypeInfo *typeinfo, const Value *val)
 Analyse types of heap and static objects.
 
u32_t analyzeHeapAllocByteSize (const Value *val)
 Analyze byte size of heap alloc function (e.g. malloc/calloc/...)
 
u32_t getNumOfFlattenElements (const Type *T)
 Number of flattened elements of an array or struct.
 
StInfogetOrAddSVFTypeInfo (const Type *T)
 Get a reference to StructInfo.
 
ObjTypeInfocreateBlkObjTypeInfo (NodeID symId)
 
ObjTypeInfocreateConstantObjTypeInfo (NodeID symId)
 

Private Attributes

SVFIRsvfir
 

Friends

class SVFIRBuilder
 

Detailed Description

Definition at line 44 of file SymbolTableBuilder.h.

Constructor & Destructor Documentation

◆ SymbolTableBuilder()

SVF::SymbolTableBuilder::SymbolTableBuilder ( SVFIR ir)
inline

Constructor.

Definition at line 52 of file SymbolTableBuilder.h.

52 : svfir(ir)
53 {
54 }
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74

Member Function Documentation

◆ analyzeHeapAllocByteSize()

u32_t SymbolTableBuilder::analyzeHeapAllocByteSize ( const Value val)
protected

Analyze byte size of heap alloc function (e.g. malloc/calloc/...)

Analyze byte size of heap alloc function (e.g. malloc/calloc/...) 1) attribute((annotate("ALLOC_HEAP_RET"), annotate("AllocSize:Arg0"))) void* safe_malloc(unsigned long size). Byte Size is the size(Arg0) 2)__attribute__((annotate("ALLOC_HEAP_RET"), annotate("AllocSize:Arg0*Arg1"))) char* safecalloc(int a, int b) Byte Size is a(Arg0) * b(Arg1) 3)__attribute__((annotate("ALLOC_HEAP_RET"), annotate("UNKNOWN"))) void* __sysv_signal(int a, void *b) Byte Size is Unknown If all required arg values are constant, byte Size is also constant, otherwise return ByteSize 0

Definition at line 730 of file SymbolTableBuilder.cpp.

731{
732 if(const llvm::CallInst* callInst = llvm::dyn_cast<llvm::CallInst>(val))
733 {
734 if (const llvm::Function* calledFunction =
735 callInst->getCalledFunction())
736 {
737 std::vector<const Value*> args;
738 // Heap alloc functions have annoation like "AllocSize:Arg1"
739 for (std::string annotation : llvmModuleSet()->getExtFuncAnnotations(calledFunction))
740 {
741 if (annotation.find("AllocSize:") != std::string::npos)
742 {
743 std::string allocSize = annotation.substr(10);
744 std::stringstream ss(allocSize);
745 std::string token;
746 // Analyaze annotation string and attract Arg list
747 while (std::getline(ss, token, '*'))
748 {
749 if (token.rfind("Arg", 0) == 0)
750 {
752 std::istringstream(token.substr(3)) >> argIndex;
754 {
755 args.push_back(
756 callInst->getArgOperand(argIndex));
757 }
758 }
759 }
760 }
761 }
762 u64_t product = 1;
763 if (args.size() > 0)
764 {
765 // for annotations like "AllocSize:Arg0*Arg1"
766 for (const llvm::Value* arg : args)
767 {
768 if (const llvm::ConstantInt* constIntArg =
769 llvm::dyn_cast<llvm::ConstantInt>(arg))
770 {
771 // Multiply the constant Value if all Args are const
773 }
774 else
775 {
776 // if Arg list has non-const value, return 0 to indicate it is non const byte size
777 return 0;
778 }
779 }
780 // If all the Args are const, return product
781 return product;
782 }
783 else
784 {
785 // for annotations like "AllocSize:UNKNOWN"
786 return 0;
787 }
788 }
789 }
790 // if it is not CallInst or CallInst has no CalledFunction, return 0 to indicate it is non const byte size
791 return 0;
792}
unsigned u32_t
Definition CommandLine.h:18
LLVMModuleSet * llvmModuleSet()
std::pair< s64_t, u64_t > getIntegerValue(const ConstantInt *intValue)
Definition LLVMUtil.h:82
unsigned long long u64_t
Definition GeneralType.h:49

◆ analyzeHeapObjType()

u32_t SymbolTableBuilder::analyzeHeapObjType ( ObjTypeInfo typeinfo,
const Value val 
)
protected

Analyse types of heap and static objects.

Analyse types of heap and static objects

For an C++ class, it can have variant elements depending on the vtable size, Hence we only handle non-cpp-class object, the type of the cpp class is treated as default PointerType

Definition at line 797 of file SymbolTableBuilder.cpp.

798{
800 analyzeObjType(typeinfo, val);
801 const Type* objTy = llvmModuleSet()->getLLVMType(typeinfo->getType());
802 if(SVFUtil::isa<ArrayType>(objTy))
803 return getNumOfElements(objTy);
804 else if(const StructType* st = SVFUtil::dyn_cast<StructType>(objTy))
805 {
809 typeinfo->resetTypeForHeapStaticObj(llvmModuleSet()->getSVFType(
810 llvmModuleSet()->getTypeInference()->ptrType()));
811 else
812 return getNumOfElements(objTy);
813 }
814 return typeinfo->getMaxFieldOffsetLimit();
815}
const Type * getLLVMType(const SVFType *T) const
Get LLVM Type.
void resetTypeForHeapStaticObj(const SVFType *t)
Definition ObjTypeInfo.h:79
u32_t getMaxFieldOffsetLimit()
Get max field offset limit.
const SVFType * getType() const
Get LLVM type.
Definition ObjTypeInfo.h:98
void setFlag(MEMTYPE mask)
Flag for this object type.
ObjTypeInference * getTypeInference()
u32_t getNumOfElements(const Type *ety)
Return size of this object based on LLVM value.
void analyzeObjType(ObjTypeInfo *typeinfo, const Value *val)
Analyse types of all flattened fields of this object.
bool classTyHasVTable(const StructType *ty)
Definition CppUtil.cpp:569
llvm::Type Type
Definition BasicTypes.h:83
llvm::StructType StructType
LLVM types.
Definition BasicTypes.h:94

◆ analyzeObjType()

void SymbolTableBuilder::analyzeObjType ( ObjTypeInfo typeinfo,
const Value val 
)
protected

Analyse types of all flattened fields of this object.

Analyse types of all flattened fields of this object

Definition at line 689 of file SymbolTableBuilder.cpp.

690{
691 const Type *elemTy = llvmModuleSet()->getLLVMType(typeinfo->getType());
692 // Find the inter nested array element
693 while (const ArrayType* AT = SVFUtil::dyn_cast<ArrayType>(elemTy))
694 {
695 elemTy = AT->getElementType();
696 if (SVFUtil::isa<GlobalVariable>(val) &&
697 SVFUtil::cast<GlobalVariable>(val)->hasInitializer() &&
698 SVFUtil::isa<ConstantArray>(
699 SVFUtil::cast<GlobalVariable>(val)->getInitializer()))
701 else
703 }
704 if (SVFUtil::isa<StructType>(elemTy))
705 {
706 if (SVFUtil::isa<GlobalVariable>(val) &&
707 SVFUtil::cast<GlobalVariable>(val)->hasInitializer() &&
708 SVFUtil::isa<ConstantStruct>(
709 SVFUtil::cast<GlobalVariable>(val)->getInitializer()))
711 else
713 }
714}
llvm::ArrayType ArrayType
Definition BasicTypes.h:95

◆ analyzeStaticObjType()

void SymbolTableBuilder::analyzeStaticObjType ( ObjTypeInfo typeinfo,
const Value val 
)
protected

Analyse types of heap and static objects.

Analyse types of heap and static objects

Definition at line 820 of file SymbolTableBuilder.cpp.

821{
823 {
825 analyzeObjType(typeinfo,castUse);
826 }
827 else
828 {
830 }
831}
const Value * getFirstUseViaCastInst(const Value *val)
Definition LLVMUtil.cpp:277
llvm::Value Value
LLVM Basic classes.
Definition BasicTypes.h:82

◆ buildMemModel()

void SymbolTableBuilder::buildMemModel ( )

Start building memory model.

This method identify which is value sym and which is object sym

if (SVFUtil::isa<InlineAsm>(Callee))

Definition at line 78 of file SymbolTableBuilder.cpp.

79{
81
82 // Pointer #0 always represents the null pointer.
83 assert(svfir->totalSymNum++ == IRGraph::NullPtr && "Something changed!");
84
85 // Pointer #1 always represents the pointer points-to black hole.
86 assert(svfir->totalSymNum++ == IRGraph::BlkPtr && "Something changed!");
87
88 // Object #2 is black hole the object that may point to any object
89 assert(svfir->totalSymNum++ == IRGraph::BlackHole && "Something changed!");
91
92 // Object #3 always represents the unique constant of a program (merging all constants if Options::ModelConsts is disabled)
93 assert(svfir->totalSymNum++ == IRGraph::ConstantObj && "Something changed!");
95
96 for (Module &M : llvmModuleSet()->getLLVMModules())
97 {
98 // Add symbols for all the globals .
99 for (const GlobalVariable& gv : M.globals())
100 {
101 collectSym(&gv);
102 }
103
104 // Add symbols for all the global aliases
105 for (const GlobalAlias& ga : M.aliases())
106 {
107 collectSym(&ga);
108 collectSym(ga.getAliasee());
109 }
110
111 // Add symbols for all of the functions and the instructions in them.
112 for (const Function& fun : M.functions())
113 {
114 collectSym(&fun);
115 collectRet(&fun);
116 if (fun.getFunctionType()->isVarArg())
117 collectVararg(&fun);
118
119 // Add symbols for all formal parameters.
120 for (const Argument& arg : fun.args())
121 {
122 collectSym(&arg);
123 }
124
125 // collect and create symbols inside the function body
126 for (const Instruction& inst : instructions(fun))
127 {
128 collectSym(&inst);
129
130 // initialization for some special instructions
131 //{@
132 if (const StoreInst* st = SVFUtil::dyn_cast<StoreInst>(&inst))
133 {
134 collectSym(st->getPointerOperand());
135 collectSym(st->getValueOperand());
136 }
137 else if (const LoadInst* ld =
138 SVFUtil::dyn_cast<LoadInst>(&inst))
139 {
140 collectSym(ld->getPointerOperand());
141 }
142 else if (const AllocaInst* alloc =
143 SVFUtil::dyn_cast<AllocaInst>(&inst))
144 {
145 collectSym(alloc->getArraySize());
146 }
147 else if (const PHINode* phi = SVFUtil::dyn_cast<PHINode>(&inst))
148 {
149 for (u32_t i = 0; i < phi->getNumIncomingValues(); ++i)
150 {
151 collectSym(phi->getIncomingValue(i));
152 }
153 }
154 else if (const GetElementPtrInst* gep =
155 SVFUtil::dyn_cast<GetElementPtrInst>(&inst))
156 {
157 collectSym(gep->getPointerOperand());
158 for (u32_t i = 0; i < gep->getNumOperands(); ++i)
159 {
160 collectSym(gep->getOperand(i));
161 }
162 }
163 else if (const SelectInst* sel =
164 SVFUtil::dyn_cast<SelectInst>(&inst))
165 {
166 collectSym(sel->getTrueValue());
167 collectSym(sel->getFalseValue());
168 collectSym(sel->getCondition());
169 }
170 else if (const BinaryOperator* binary =
171 SVFUtil::dyn_cast<BinaryOperator>(&inst))
172 {
173 for (u32_t i = 0; i < binary->getNumOperands(); i++)
174 collectSym(binary->getOperand(i));
175 }
176 else if (const UnaryOperator* unary =
177 SVFUtil::dyn_cast<UnaryOperator>(&inst))
178 {
179 for (u32_t i = 0; i < unary->getNumOperands(); i++)
180 collectSym(unary->getOperand(i));
181 }
182 else if (const CmpInst* cmp = SVFUtil::dyn_cast<CmpInst>(&inst))
183 {
184 for (u32_t i = 0; i < cmp->getNumOperands(); i++)
185 collectSym(cmp->getOperand(i));
186 }
187 else if (const CastInst* cast =
188 SVFUtil::dyn_cast<CastInst>(&inst))
189 {
190 collectSym(cast->getOperand(0));
191 }
192 else if (const ReturnInst* ret =
193 SVFUtil::dyn_cast<ReturnInst>(&inst))
194 {
195 if (ret->getReturnValue())
196 collectSym(ret->getReturnValue());
197 }
198 else if (const BranchInst* br =
199 SVFUtil::dyn_cast<BranchInst>(&inst))
200 {
201 Value* opnd = br->isConditional() ? br->getCondition() : br->getOperand(0);
203 }
204 else if (const SwitchInst* sw =
205 SVFUtil::dyn_cast<SwitchInst>(&inst))
206 {
207 collectSym(sw->getCondition());
208 }
209 else if (isNonInstricCallSite(&inst))
210 {
211
212 const CallBase* cs = LLVMUtil::getLLVMCallSite(&inst);
213 for (u32_t i = 0; i < cs->arg_size(); i++)
214 {
215 collectSym(cs->getArgOperand(i));
216 }
217 // Calls to inline asm need to be added as well because the
218 // callee isn't referenced anywhere else.
219 const Value* Callee = cs->getCalledOperand();
221
222 // TODO handle inlineAsm
225 {
227 }
228 }
230 }
231 }
232 }
233
236 {
238 }
239}
NodeID totalSymNum
total number of symbols
Definition IRGraph.h:101
static NodeIDAllocator * get(void)
Return (singleton) allocator.
NodeID endSymbolAllocation(void)
Notify the allocator that all symbols have had IDs allocated.
void validateTypeCheck(const CallBase *cs)
validate type inference
static const Option< bool > EnableTypeCheck
Definition Options.h:131
static const Option< bool > SymTabPrint
Definition Options.h:190
void collectRet(const Function *val)
ObjTypeInfo * createBlkObjTypeInfo(NodeID symId)
ObjTypeInfo * createConstantObjTypeInfo(NodeID symId)
void collectSym(const Value *val)
void collectVararg(const Function *val)
const CallBase * getLLVMCallSite(const Value *value)
Return LLVM callsite given a value.
Definition LLVMUtil.h:91
bool isNonInstricCallSite(const Instruction *inst)
Whether an instruction is a callsite in the application code, excluding llvm intrinsic calls.
Definition LLVMUtil.cpp:720
void increaseStackSize()
Increase the stack size limit.
Definition SVFUtil.cpp:229
std::enable_if_t<!is_simple_type< Y >::value, typename cast_retty< X, const Y >::ret_type > cast(const Y &Val)
Definition Casting.h:360
llvm::GlobalVariable GlobalVariable
Definition BasicTypes.h:130
llvm::GlobalAlias GlobalAlias
Definition BasicTypes.h:128
llvm::CallBase CallBase
Definition BasicTypes.h:146
llvm::UnaryOperator UnaryOperator
Definition BasicTypes.h:180
llvm::AllocaInst AllocaInst
Definition BasicTypes.h:150
llvm::SwitchInst SwitchInst
Definition BasicTypes.h:155
llvm::Argument Argument
Definition BasicTypes.h:145
llvm::LoadInst LoadInst
Definition BasicTypes.h:149
llvm::CmpInst CmpInst
Definition BasicTypes.h:159
llvm::Function Function
Definition BasicTypes.h:85
llvm::Instruction Instruction
Definition BasicTypes.h:87
llvm::CastInst CastInst
Definition BasicTypes.h:158
llvm::Module Module
Definition BasicTypes.h:84
llvm::BinaryOperator BinaryOperator
Definition BasicTypes.h:179
llvm::StoreInst StoreInst
Definition BasicTypes.h:148
llvm::SelectInst SelectInst
Definition BasicTypes.h:174
llvm::GetElementPtrInst GetElementPtrInst
Definition BasicTypes.h:162
llvm::ReturnInst ReturnInst
Definition BasicTypes.h:157
llvm::PHINode PHINode
Definition BasicTypes.h:165
llvm::BranchInst BranchInst
Definition BasicTypes.h:154

◆ collectObj()

void SymbolTableBuilder::collectObj ( const Value val)
protected

Get memory object sym, if not available create a new one

Definition at line 317 of file SymbolTableBuilder.cpp.

318{
320 LLVMModuleSet::ValueToIDMapTy::iterator iter = llvmModuleSet()->objSymMap.find(val);
321 if (iter == llvmModuleSet()->objSymMap.end())
322 {
323 // if the object pointed by the pointer is a constant data (e.g., i32 0) or a global constant object (e.g. string)
324 // then we treat them as one ConstantObj
326 {
327 llvmModuleSet()->objSymMap.insert(std::make_pair(val, svfir->constantSymID()));
328 }
329 // otherwise, we will create an object for each abstract memory location
330 else
331 {
332 // create obj sym and sym type
334 llvmModuleSet()->objSymMap.insert(std::make_pair(val, id));
336 outs() << "create a new obj sym " << id << "\n");
337
338 // create a memory object
340 assert(svfir->objTypeInfoMap.find(id) == svfir->objTypeInfoMap.end());
341 svfir->objTypeInfoMap[id] = ti;
342 }
343 }
344}
#define DBOUT(TYPE, X)
LLVM debug macros, define type of your DBUG model of each pass.
Definition SVFType.h:498
#define DMemModel
Definition SVFType.h:507
NodeID constantSymID() const
Definition IRGraph.h:188
IDToTypeInfoMapTy objTypeInfoMap
map a memory sym id to its obj
Definition IRGraph.h:88
ValueToIDMapTy objSymMap
map a obj reference to its sym id
Definition LLVMModule.h:113
NodeID allocateObjectId(void)
Allocate an object ID as determined by the strategy.
static Option< bool > ModelConsts
Definition Options.h:187
ObjTypeInfo * createObjTypeInfo(const Value *val)
Create an objectInfo based on LLVM value.
bool isConstantObjSym(const Value *val)
Check whether this value points-to a constant object.
Definition CppUtil.cpp:672
const Value * getGlobalRep(const Value *val)
find the unique defined global across multiple modules
Definition LLVMUtil.cpp:439
std::ostream & outs()
Overwrite llvm::outs()
Definition SVFUtil.h:52
u32_t NodeID
Definition GeneralType.h:56

◆ collectRet()

void SymbolTableBuilder::collectRet ( const Function val)
protected

Create unique return sym, if not available create a new one

Definition at line 349 of file SymbolTableBuilder.cpp.

350{
351
352 LLVMModuleSet::FunToIDMapTy::iterator iter =
354 if (iter == llvmModuleSet()->returnSymMap.end())
355 {
357 llvmModuleSet()->returnSymMap.insert(std::make_pair(val, id));
358 DBOUT(DMemModel, outs() << "create a return sym " << id << "\n");
359 }
360}
FunToIDMapTy returnSymMap
return map
Definition LLVMModule.h:114
NodeID allocateValueId(void)
Allocate a value ID as determined by the strategy.

◆ collectSVFTypeInfo()

void SymbolTableBuilder::collectSVFTypeInfo ( const Value val)
protected

collect the syms

Definition at line 241 of file SymbolTableBuilder.cpp.

242{
243 Type *valType = val->getType();
245 if(isGepConstantExpr(val) || SVFUtil::isa<GetElementPtrInst>(val))
246 {
248 gi = bridge_gep_begin(SVFUtil::cast<User>(val)),
249 ge = bridge_gep_end(SVFUtil::cast<User>(val));
250 gi != ge; ++gi)
251 {
252 const Type* gepTy = *gi;
254 }
255 }
256}
StInfo * getOrAddSVFTypeInfo(const Type *T)
Get a reference to StructInfo.
const ConstantExpr * isGepConstantExpr(const Value *val)
Return corresponding constant expression, otherwise return nullptr.
Definition LLVMUtil.h:215
bridge_gep_iterator bridge_gep_end(const User *GEP)
bridge_gep_iterator bridge_gep_begin(const User *GEP)

◆ collectSym()

void SymbolTableBuilder::collectSym ( const Value val)
protected

Collect symbols, including value and object syms

Definition at line 261 of file SymbolTableBuilder.cpp.

262{
263
264 //TODO: filter the non-pointer type // if (!SVFUtil::isa<PointerType>(val->getType())) return;
265
267 outs()
268 << "collect sym from ##"
269 << llvmModuleSet()->getSVFValue(val)->toString()
270 << " \n");
271 //TODO handle constant expression value here??
272 handleCE(val);
273
274 // create a value sym
276
279
280 // create an object If it is a heap, stack, global, function.
281 if (isObject(val))
282 {
284 }
285}
void collectSVFTypeInfo(const Value *val)
collect the syms
void collectObj(const Value *val)
void collectVal(const Value *val)
void handleCE(const Value *val)
bool isObject(const Value *ref)
Return true if this value refers to a object.
Definition LLVMUtil.cpp:59

◆ collectVal()

void SymbolTableBuilder::collectVal ( const Value val)
protected

Get value sym, if not available create a new one

handle global constant expression here

Definition at line 290 of file SymbolTableBuilder.cpp.

291{
292 // collect and record special sym here
294 {
295 return;
296 }
297 LLVMModuleSet::ValueToIDMapTy::iterator iter = llvmModuleSet()->valSymMap.find(val);
298 if (iter == llvmModuleSet()->valSymMap.end())
299 {
300 // create val sym and sym type
302 llvmModuleSet()->valSymMap.insert(std::make_pair(val, id));
304 outs() << "create a new value sym " << id << "\n");
306 if (const GlobalVariable* globalVar = SVFUtil::dyn_cast<GlobalVariable>(val))
308 }
309
312}
ValueToIDMapTy valSymMap
map a value to its sym id
Definition LLVMModule.h:112
void handleGlobalCE(const GlobalVariable *G)
Handle constant expression.
bool isNullPtrSym(const Value *val)
Check whether this value is a black hole.
Definition LLVMUtil.h:125
bool isBlackholeSym(const Value *val)
Check whether this value is a black hole.
Definition LLVMUtil.h:119

◆ collectVararg()

void SymbolTableBuilder::collectVararg ( const Function val)
protected

Create vararg sym, if not available create a new one

Definition at line 365 of file SymbolTableBuilder.cpp.

366{
367 LLVMModuleSet::FunToIDMapTy::iterator iter =
369 if (iter == llvmModuleSet()->varargSymMap.end())
370 {
372 llvmModuleSet()->varargSymMap.insert(std::make_pair(val, id));
373 DBOUT(DMemModel, outs() << "create a vararg sym " << id << "\n");
374 }
375}
FunToIDMapTy varargSymMap
vararg map
Definition LLVMModule.h:115

◆ createBlkObjTypeInfo()

ObjTypeInfo * SymbolTableBuilder::createBlkObjTypeInfo ( NodeID  symId)
protected

Definition at line 46 of file SymbolTableBuilder.cpp.

47{
51 {
53 IntegerType::get(llvmset->getContext(), 32)));
55 }
57 return ti;
58}
ObjTypeInfo * createObjTypeInfo(const SVFType *type)
Create an objectInfo based on LLVM type (value is null, and type could be null, representing a dummy ...
Definition IRGraph.cpp:231
static bool isBlkObj(NodeID id)
Definition IRGraph.h:165

◆ createConstantObjTypeInfo()

ObjTypeInfo * SymbolTableBuilder::createConstantObjTypeInfo ( NodeID  symId)
protected

Definition at line 60 of file SymbolTableBuilder.cpp.

61{
65 {
67 llvmset->getSVFType(IntegerType::get(llvmset->getContext(), 32)));
69 }
71 return ti;
72}
static bool isConstantSym(NodeID id)
Definition IRGraph.h:169

◆ createObjTypeInfo()

ObjTypeInfo * SymbolTableBuilder::createObjTypeInfo ( const Value val)
protected

Create an objectInfo based on LLVM value.

Definition at line 620 of file SymbolTableBuilder.cpp.

621{
622 const Type* objTy = nullptr;
623
624 const Instruction* I = SVFUtil::dyn_cast<Instruction>(val);
625
626 // We consider two types of objects:
627 // (1) A heap/static object from a callsite
628 if (I && isNonInstricCallSite(I))
629 {
631 }
632 // (2) Other objects (e.g., alloca, global, etc.)
633 else
634 {
635 if (SVFUtil::isa<PointerType>(val->getType()))
636 {
637 if (const AllocaInst *allocaInst = SVFUtil::dyn_cast<AllocaInst>(val))
638 {
639 // get the type of the allocated memory
640 // e.g., for `%retval = alloca i64, align 4`, we return i64
641 objTy = allocaInst->getAllocatedType();
642 }
643 else if (const GlobalValue *global = SVFUtil::dyn_cast<GlobalValue>(val))
644 {
645 // get the pointee type of the global pointer (begins with @ symbol in llvm)
646 objTy = global->getValueType();
647 }
648 else
649 {
651 assert(false && "not an allocation or global?");
652 }
653 }
654 }
655
656 if (objTy)
657 {
659 ObjTypeInfo* typeInfo = new ObjTypeInfo(
660 llvmModuleSet()->getSVFType(objTy),
662 initTypeInfo(typeInfo,val, objTy);
663 return typeInfo;
664 }
665 else
666 {
667 writeWrnMsg("try to create an object with a non-pointer type.");
668 writeWrnMsg(val->getName().str());
669 writeWrnMsg("(" + getSourceLoc(val) + ")");
671 {
672 ObjTypeInfo* typeInfo = new ObjTypeInfo(
673 llvmModuleSet()->getSVFType(val->getType()),
674 0);
675 initTypeInfo(typeInfo,val, val->getType());
676 return typeInfo;
677 }
678 else
679 {
680 assert(false && "Memory object must be either (1) held by a pointer-typed ref value or (2) a constant value (e.g., 10).");
681 abort();
682 }
683 }
684}
static const Option< u32_t > MaxFieldLimit
Maximum number of field derivations for an object.
Definition Options.h:38
const Type * inferTypeOfHeapObjOrStaticObj(const Instruction *inst)
Get the reference type of heap/static object from an allocation site.
void initTypeInfo(ObjTypeInfo *typeinfo, const Value *value, const Type *ty)
Initialize TypeInfo based on LLVM Value.
std::string dumpValueAndDbgInfo(const Value *val)
Definition LLVMUtil.cpp:622
const std::string getSourceLoc(const Value *val)
Definition LLVMUtil.cpp:452
std::ostream & errs()
Overwrite llvm::errs()
Definition SVFUtil.h:58
void writeWrnMsg(const std::string &msg)
Writes a message run through wrnMsg.
Definition SVFUtil.cpp:68
llvm::GlobalValue GlobalValue
Definition BasicTypes.h:88

◆ getNumOfElements()

u32_t SymbolTableBuilder::getNumOfElements ( const Type ety)

Return size of this object based on LLVM value.

Return size of this Object

Definition at line 928 of file SymbolTableBuilder.cpp.

929{
930 assert(ety && "type is null?");
931 u32_t numOfFields = 1;
932 if (SVFUtil::isa<StructType, ArrayType>(ety))
933 {
935 }
936 return numOfFields;
937}
u32_t getNumOfFlattenElements(const Type *T)
Number of flattened elements of an array or struct.

◆ getNumOfFlattenElements()

u32_t SymbolTableBuilder::getNumOfFlattenElements ( const Type T)
protected

Number of flattened elements of an array or struct.

Get a reference to the components of struct_info. Number of flattened elements of an array or struct

Definition at line 940 of file SymbolTableBuilder.cpp.

941{
944 else
946}
static Option< bool > ModelArrays
Definition Options.h:188
u32_t getNumOfFlattenElements() const
Return number of elements after flattening (including array elements)
Definition SVFType.h:139
u32_t getNumOfFlattenFields() const
Return the number of fields after flattening (ignoring array elements)
Definition SVFType.h:145

◆ getOrAddSVFTypeInfo()

StInfo * SymbolTableBuilder::getOrAddSVFTypeInfo ( const Type T)
protected

Get a reference to StructInfo.

Definition at line 948 of file SymbolTableBuilder.cpp.

949{
951}
SVFType * getSVFType(const Type *T)
Get or create SVFType and typeinfo.
StInfo * getTypeInfo()
Definition SVFType.h:230

◆ getTypeInference()

ObjTypeInference * SymbolTableBuilder::getTypeInference ( )
protected

Definition at line 568 of file SymbolTableBuilder.cpp.

569{
571}
ObjTypeInference * getTypeInference()

◆ handleCE()

void SymbolTableBuilder::handleCE ( const Value val)
protected

Handle constant expression

Definition at line 380 of file SymbolTableBuilder.cpp.

381{
382 if (const Constant* ref = SVFUtil::dyn_cast<Constant>(val))
383 {
384 if (const ConstantExpr* ce = isGepConstantExpr(ref))
385 {
386 DBOUT(DMemModelCE, outs() << "handle constant expression "
387 << llvmModuleSet()
389 ->toString()
390 << "\n");
391 collectVal(ce);
392
393 // handle the recursive constant express case
394 // like (gep (bitcast (gep X 1)) 1); the inner gep is ce->getOperand(0)
395 for (u32_t i = 0; i < ce->getNumOperands(); ++i)
396 {
397 collectVal(ce->getOperand(i));
398 handleCE(ce->getOperand(i));
399 }
400 }
401 else if (const ConstantExpr* ce = isCastConstantExpr(ref))
402 {
403 DBOUT(DMemModelCE, outs() << "handle constant expression "
404 << llvmModuleSet()
406 ->toString()
407 << "\n");
408 collectVal(ce);
409 collectVal(ce->getOperand(0));
410 // handle the recursive constant express case
411 // like (gep (bitcast (gep X 1)) 1); the inner gep is ce->getOperand(0)
412 handleCE(ce->getOperand(0));
413 }
414 else if (const ConstantExpr* ce = isSelectConstantExpr(ref))
415 {
416 DBOUT(DMemModelCE, outs() << "handle constant expression "
417 << llvmModuleSet()
419 ->toString()
420 << "\n");
421 collectVal(ce);
422 collectVal(ce->getOperand(0));
423 collectVal(ce->getOperand(1));
424 collectVal(ce->getOperand(2));
425 // handle the recursive constant express case
426 // like (gep (bitcast (gep X 1)) 1); the inner gep is ce->getOperand(0)
427 handleCE(ce->getOperand(0));
428 handleCE(ce->getOperand(1));
429 handleCE(ce->getOperand(2));
430 }
431 // if we meet a int2ptr, then it points-to black hole
433 {
435 const Constant* opnd = int2Ptrce->getOperand(0);
436 handleCE(opnd);
437 }
439 {
441 const Constant* opnd = ptr2Intce->getOperand(0);
442 handleCE(opnd);
443 }
445 {
447 }
448 else if (isBinaryConstantExpr(ref))
449 {
451 }
452 else if (isUnaryConstantExpr(ref))
453 {
454 // we don't handle unary constant expression like fneg(x) now
456 }
457 else if (SVFUtil::isa<ConstantAggregate>(ref))
458 {
459 // we don't handle constant aggregate like constant vectors
461 }
462 else
463 {
464 assert(!SVFUtil::isa<ConstantExpr>(val) &&
465 "we don't handle all other constant expression for now!");
467 }
468 }
469}
#define DMemModelCE
Definition SVFType.h:508
const ConstantExpr * isBinaryConstantExpr(const Value *val)
Definition LLVMUtil.h:290
const ConstantExpr * isInt2PtrConstantExpr(const Value *val)
Definition LLVMUtil.h:225
const ConstantExpr * isSelectConstantExpr(const Value *val)
Definition LLVMUtil.h:255
const ConstantExpr * isTruncConstantExpr(const Value *val)
Definition LLVMUtil.h:265
const ConstantExpr * isPtr2IntConstantExpr(const Value *val)
Definition LLVMUtil.h:235
const ConstantExpr * isUnaryConstantExpr(const Value *val)
Definition LLVMUtil.h:301
const ConstantExpr * isCastConstantExpr(const Value *val)
Definition LLVMUtil.h:245
const ConstantExpr * isCmpConstantExpr(const Value *val)
Definition LLVMUtil.h:279
llvm::Constant Constant
Definition BasicTypes.h:124
llvm::ConstantExpr ConstantExpr
Definition BasicTypes.h:120

◆ handleGlobalCE()

void SymbolTableBuilder::handleGlobalCE ( const GlobalVariable G)
protected

Handle constant expression.

Handle global constant expression

Definition at line 474 of file SymbolTableBuilder.cpp.

475{
476 assert(G);
477
478 //The type this global points to
479 const Type* T = G->getValueType();
480 bool is_array = 0;
481 //An array is considered a single variable of its type.
482 while (const ArrayType* AT = SVFUtil::dyn_cast<ArrayType>(T))
483 {
484 T = AT->getElementType();
485 is_array = true;
486 }
487
488 if (SVFUtil::isa<StructType>(T))
489 {
490 //A struct may be used in constant GEP expr.
491 for (const User* user : G->users())
492 {
493 handleCE(user);
494 }
495 }
496 else if (is_array)
497 {
498 for (const User* user : G->users())
499 {
500 handleCE(user);
501 }
502 }
503
504 if (G->hasInitializer())
505 {
506 handleGlobalInitializerCE(G->getInitializer());
507 }
508}
void handleGlobalInitializerCE(const Constant *C)
llvm::User User
Definition BasicTypes.h:142

◆ handleGlobalInitializerCE()

void SymbolTableBuilder::handleGlobalInitializerCE ( const Constant C)
protected

Handle global variable initialization

Definition at line 513 of file SymbolTableBuilder.cpp.

514{
515
516 if (C->getType()->isSingleValueType())
517 {
518 if (const ConstantExpr* E = SVFUtil::dyn_cast<ConstantExpr>(C))
519 {
520 handleCE(E);
521 }
522 else
523 {
524 collectVal(C);
525 }
526 }
527 else if (SVFUtil::isa<ConstantArray>(C))
528 {
529 for (u32_t i = 0, e = C->getNumOperands(); i != e; i++)
530 {
531 handleGlobalInitializerCE(SVFUtil::cast<Constant>(C->getOperand(i)));
532 }
533 }
534 else if (SVFUtil::isa<ConstantStruct>(C))
535 {
536 for (u32_t i = 0, e = C->getNumOperands(); i != e; i++)
537 {
538 handleGlobalInitializerCE(SVFUtil::cast<Constant>(C->getOperand(i)));
539 }
540 }
541 else if(const ConstantData* data = SVFUtil::dyn_cast<ConstantData>(C))
542 {
544 {
545 if (const ConstantDataSequential* seq =
546 SVFUtil::dyn_cast<ConstantDataSequential>(data))
547 {
548 for(u32_t i = 0; i < seq->getNumElements(); i++)
549 {
550 const Constant* ct = seq->getElementAsConstant(i);
552 }
553 }
554 else
555 {
556 assert(
557 (SVFUtil::isa<ConstantAggregateZero, UndefValue>(data)) &&
558 "Single value type data should have been handled!");
559 }
560 }
561 }
562 else
563 {
564 //TODO:assert(SVFUtil::isa<ConstantVector>(C),"what else do we have");
565 }
566}
llvm::ConstantData ConstantData
Definition BasicTypes.h:116
llvm::ConstantDataSequential ConstantDataSequential
Definition BasicTypes.h:119

◆ inferObjType()

const Type * SymbolTableBuilder::inferObjType ( const Value startValue)
protected

Forward collect all possible infer sites starting from a value.

Definition at line 574 of file SymbolTableBuilder.cpp.

575{
577}
const Type * inferObjType(const Value *var)
get or infer the type of the object pointed by the value

◆ inferTypeOfHeapObjOrStaticObj()

const Type * SymbolTableBuilder::inferTypeOfHeapObjOrStaticObj ( const Instruction inst)
protected

Get the reference type of heap/static object from an allocation site.

Return the type of the object from a heap allocation

Definition at line 582 of file SymbolTableBuilder.cpp.

583{
584 const Value* startValue = inst;
585 const PointerType *originalPType = SVFUtil::dyn_cast<PointerType>(inst->getType());
586 const Type* inferedType = nullptr;
587 assert(originalPType && "empty type?");
589 {
590 if(const Value* v = getFirstUseViaCastInst(inst))
591 {
592 if (const PointerType *newTy = SVFUtil::dyn_cast<PointerType>(v->getType()))
593 {
595 }
596 }
598 }
600 {
601 const CallBase* cs = LLVMUtil::getLLVMCallSite(inst);
602 u32_t arg_pos = LLVMUtil::getHeapAllocHoldingArgPosition(cs->getCalledFunction());
603 const Value* arg = cs->getArgOperand(arg_pos);
604 originalPType = SVFUtil::dyn_cast<PointerType>(arg->getType());
606 }
607 else
608 {
609 assert( false && "not a heap allocation instruction?");
610 }
611
613
614 return inferedType;
615}
void typeSizeDiffTest(const PointerType *oPTy, const Type *iTy, const Value *val)
const Type * inferObjType(const Value *startValue)
Forward collect all possible infer sites starting from a value.
bool isHeapAllocExtCallViaRet(const Instruction *inst)
Definition LLVMUtil.cpp:633
bool isHeapAllocExtCallViaArg(const Instruction *inst)
Definition LLVMUtil.cpp:648
u32_t getHeapAllocHoldingArgPosition(const Function *fun)
Definition LLVMUtil.cpp:400
llvm::PointerType PointerType
Definition BasicTypes.h:96

◆ initTypeInfo()

void SymbolTableBuilder::initTypeInfo ( ObjTypeInfo typeinfo,
const Value val,
const Type objTy 
)
protected

Initialize TypeInfo based on LLVM Value.

Initialize the type info of an object

if val is AllocaInst, byteSize is Type's LLVM ByteSize * ArraySize e.g. alloc i32, 10. byteSize is 4 (i32's size) * 10 (ArraySize) = 40

This is for alloca <ty> <NumElements>. For example, alloca i64 3 allocates 3 i64 on the stack (objSize=3) In most cases, NumElements is not specified in the instruction, which means there is only one element (objSize=1).

if ArraySize is not constant, byteSize is not static determined.

if val is GlobalVar, byteSize is Type's LLVM ByteSize All GlobalVariable must have constant size

if val is heap alloc

Definition at line 836 of file SymbolTableBuilder.cpp.

838{
839
840 u32_t elemNum = 1;
841 // init byteSize = 0, If byteSize is changed in the following process,
842 // it means that ObjTypeInfo has a Constant Byte Size
843 u32_t byteSize = 0;
844 // Global variable
845 // if val is Function Obj, byteSize is not set
846 if (SVFUtil::isa<Function>(val))
847 {
849 analyzeObjType(typeinfo,val);
850 elemNum = getNumOfElements(objTy);
851 }
854 else if(const AllocaInst* allocaInst = SVFUtil::dyn_cast<AllocaInst>(val))
855 {
857 analyzeObjType(typeinfo,val);
860 if(const ConstantInt* sz = SVFUtil::dyn_cast<ConstantInt>(allocaInst->getArraySize()))
861 {
863 byteSize = LLVMUtil::getIntegerValue(sz).second * typeinfo->getType()->getByteSize();
864 }
866 else
867 {
868 elemNum = getNumOfElements(objTy);
869 byteSize = 0;
870 }
871 }
874 else if(SVFUtil::isa<GlobalVariable>(val))
875 {
879 analyzeObjType(typeinfo,val);
880 elemNum = getNumOfElements(objTy);
881 byteSize = typeinfo->getType()->getByteSize();
882 }
884 else if (SVFUtil::isa<Instruction>(val) &&
886 SVFUtil::cast<Instruction>(val)))
887 {
888 elemNum = analyzeHeapObjType(typeinfo,val);
889 // analyze heap alloc like (malloc/calloc/...), the alloc functions have
890 // annotation like "AllocSize:Arg1". Please refer to extapi.c.
891 // e.g. calloc(4, 10), annotation is "AllocSize:Arg0*Arg1",
892 // it means byteSize = 4 (Arg0) * 10 (Arg1) = 40
893 byteSize = analyzeHeapAllocByteSize(val);
894 }
896 {
897 analyzeStaticObjType(typeinfo,val);
898 // user input data, label its field as infinite here
899 elemNum = typeinfo->getMaxFieldOffsetLimit();
900 byteSize = typeinfo->getType()->getByteSize();
901 }
903 {
905 elemNum = getNumOfFlattenElements(val->getType());
906 byteSize = typeinfo->getType()->getByteSize();
907 }
908 else
909 {
910 assert("what other object do we have??");
911 abort();
912 }
913
914 // Reset maxOffsetLimit if it is over the total fieldNum of this object
915 if(typeinfo->getMaxFieldOffsetLimit() > elemNum)
916 typeinfo->setNumOfElements(elemNum);
917
918 // set ByteSize. If ByteSize > 0, this typeinfo has constant type.
919 // If ByteSize == 0, this typeinfo has 1) zero byte 2) non-const byte size
920 // If ByteSize>MaxFieldLimit, set MaxFieldLimit to the byteSize;
921 byteSize = Options::MaxFieldLimit() > byteSize? byteSize: Options::MaxFieldLimit();
922 typeinfo->setByteSizeOfObj(byteSize);
923}
void setByteSizeOfObj(u32_t size)
Set the byte size of this object.
void setNumOfElements(u32_t num)
Set the number of elements of this object.
u32_t getByteSize() const
Definition SVFType.h:244
u32_t analyzeHeapObjType(ObjTypeInfo *typeinfo, const Value *val)
Analyse types of heap and static objects.
u32_t analyzeHeapAllocByteSize(const Value *val)
Analyze byte size of heap alloc function (e.g. malloc/calloc/...)
void analyzeStaticObjType(ObjTypeInfo *typeinfo, const Value *val)
Analyse types of heap and static objects.
bool isHeapAllocExtCall(const Instruction *inst)
Definition LLVMUtil.h:396
bool isConstDataOrAggData(const Value *val)
Return true if the value refers to constant data, e.g., i32 0.
Definition LLVMUtil.h:368
bool ArgInProgEntryFunction(const Value *val)
Return true if this is an argument of a program entry function (e.g. main)
Definition LLVMUtil.h:164
llvm::ConstantInt ConstantInt
Definition BasicTypes.h:125

◆ llvmModuleSet()

LLVMModuleSet * SVF::SymbolTableBuilder::llvmModuleSet ( )
inlineprotected

Definition at line 87 of file SymbolTableBuilder.h.

88 {
90 }
static LLVMModuleSet * getLLVMModuleSet()
Definition LLVMModule.h:129

Friends And Related Symbol Documentation

◆ SVFIRBuilder

Definition at line 46 of file SymbolTableBuilder.h.

Member Data Documentation

◆ svfir

SVFIR* SVF::SymbolTableBuilder::svfir
private

Definition at line 48 of file SymbolTableBuilder.h.


The documentation for this class was generated from the following files: