39#include "llvm/Support/FileSystem.h"
41#include "llvm/Transforms/Utils/Cloning.h"
72#define SVF_MAIN_FUNC_NAME "svf.main"
73#define SVF_GLOBAL_CTORS "llvm.global_ctors"
74#define SVF_GLOBAL_DTORS "llvm.global_dtors"
112 mset->modules.emplace_back(
mod);
113 mset->loadExtAPIModules();
118 mset->buildSymbolTable();
128 mset->loadExtAPIModules();
141 mset->buildSymbolTable();
188 if (
func.isDeclaration())
211 for (Function::const_iterator bit =
func->begin(),
ebit =
func->end(); bit !=
ebit; ++bit)
220 SVFUtil::isa<ReturnInst>(bb->back())) &&
221 "last inst must be return inst");
231 SVFUtil::isa<ReturnInst>(&
func->back().back())) &&
232 "last inst must be return inst");
252 std::unique_ptr<BreakConstantGEPs>
p1 = std::make_unique<BreakConstantGEPs>();
259 std::unique_ptr<UnifyFunctionExitNodes>
p2 =
260 std::make_unique<UnifyFunctionExitNodes>();
263 for (
auto F =
M.begin(),
E =
M.end(); F !=
E; ++F)
266 if (fun.isDeclaration())
268 p2->runOnFunction(fun);
277 mset->loadExtAPIModules();
278 mset->prePassSchedule();
338 owned_ctx = std::make_unique<LLVMContext>();
343 SVFUtil::errs() <<
"not an IR file: " << moduleName << std::endl;
351 SVFUtil::errs() <<
"load module: " << moduleName <<
"failed!!\n\n";
352 Err.print(
"SVFModuleLoader", llvm::errs());
373 assert(!
empty() &&
"LLVMModuleSet contains no modules; cannot load ExtAPI module without LLVMContext!");
389 Err.print(
"SVFModuleLoader", llvm::errs());
443 std::priority_queue<LLVMGlobalFunction, std::vector<LLVMGlobalFunction>,
446 std::vector<const Function* >
result;
452 SVFUtil::dyn_cast<ConstantArray>(
global->getInitializer()))
459 SVFUtil::dyn_cast<ConstantStruct>(
479 while (!
queue.empty())
532 Type*
ptr = PointerType::getUnqual(
M.getContext());
533 Type*
i32 = IntegerType::getInt32Ty(
M.getContext());
535#if (LLVM_VERSION_MAJOR >= 9)
538 Type::getVoidTy(
M.getContext()),
545 Type::getVoidTy(
M.getContext()),
549 svfmain->setCallingConv(llvm::CallingConv::C);
556 auto target =
M.getOrInsertFunction(
558 Type::getVoidTy(
M.getContext())
567 assert(
cnt <= 3 &&
"Too many arguments for main()");
573 auto target =
M.getOrInsertFunction(
dtor->getName(), Type::getVoidTy(
M.getContext()));
584 if (
glob ==
nullptr || !
glob->hasInitializer())
591 for (
unsigned i = 0;
i <
ca->getNumOperands(); ++
i)
603 if (
expr->getOpcode() == Instruction::BitCast && SVFUtil::isa<Function>(
expr->getOperand(0)))
604 fun = SVFUtil::cast<Function>(
expr->getOperand(0));
607 if (
note->getOpcode() != Instruction::GetElementPtr)
615 fun = SVFUtil::dyn_cast<Function>(
structAn->getOperand(0));
623 if (data && data->isString())
625 std::string
annotation = data->getAsString().str();
668 if (fun.getName().str() ==
"main")
674 else if (fun.getName().str() ==
"svf__main")
691 if (fun.isDeclaration())
706 std::set_intersection(
801 if(
annotation.find(
"OVERWRITE") != std::string::npos)
803 assert(
false &&
"overwrite and other annotations cannot co-exist");
817 if (
CallInst *callInst = SVFUtil::dyn_cast<CallInst>(inst))
819 Function *calledFunc = callInst->getCalledFunction();
821 if (calledFunc && calledFunc->getName() ==
callee->getName())
823 callInst->setCalledFunction(
callee);
907 return &moduleRef.get() == extModule;
927 if (
global.hasPrivateLinkage())
944 return g->hasInitializer();
965 std::string moduleName =
mod.getName().str();
967 std::size_t
pos = moduleName.rfind(
'.');
968 if (
pos != std::string::npos)
976#if (LLVM_VERSION_MAJOR >= 7)
988 if (SVFUtil::isa<ConstantPointerNull>(
llvm_value))
990 else if (SVFUtil::isa<UndefValue>(
llvm_value))
1001 if (SVFUtil::isa<ConstantPointerNull, UndefValue>(
val))
1013 return iter->second;
1057 rawstr <<
" No llvmVal found";
1093 if (
it->second ==
T)
1096 assert(
false &&
"can't find the corresponding LLVM Type");
1105 assert(
T &&
"SVFType should not be null");
1128 assert (node!=
nullptr &&
"no ICFGNode for this instruction?");
1142 return node !=
nullptr;
1150 assert (node!=
nullptr &&
"no CallICFGNode for this instruction?");
1159 assert (node!=
nullptr &&
"no RetICFGNode for this instruction?");
1166 assert (node!=
nullptr &&
"no IntraICFGNode for this instruction?");
1183 else if (
const StructType*
sty = SVFUtil::dyn_cast<StructType>(
T))
1205 "SVFType has been added before");
1214 byteSize =
DL.getTypeAllocSize(
mut_T);
1218 if (SVFUtil::isa<PointerType>(
T))
1233 for (
const auto& t:
ft->params())
1239 else if (
const StructType*
st = SVFUtil::dyn_cast<StructType>(
T))
1243 svfst->setName(
st->getName().str());
1246 else if (
const auto at = SVFUtil::dyn_cast<ArrayType>(
T))
1249 svfat->setNumOfElement(
at->getNumElements());
1257 llvm::raw_string_ostream(
buffer) << *
T;
1288 stInfo->setNumOfFieldsAndElems(1, 1);
1319 stInfo->getFlattenElementTypes().push_back(
et);
1324 "typeForArray size incorrect!!!");
1350 if (SVFUtil::isa<StructType, ArrayType>(
elemTy))
1366 stInfo->getFlattenElementTypes().push_back(
ty);
1381 "typeForStruct size incorrect!");
1400 stInfo->setNumOfFieldsAndElems(1,1);
1407 assert(fun &&
"Null SVFFunction* pointer");
1413 assert(fun &&
"Null SVFFunction* pointer");
1426 assert(fun &&
"Null Function* pointer");
1439 assert(fun &&
"Null Function* pointer");
1478 assert(!
allocArg.empty() &&
"Not an alloc call via argument or incorrect extern function annotation!");
1486 assert(!
number.empty() &&
"Incorrect naming convention for svf external functions(ALLOC_HEAP_ARG + number)?");
1487 return std::stoi(
number);
1501 assert(F &&
"Null SVFFunction* pointer");
1502 if (F->isDeclaration() || F->isIntrinsic())
#define SVF_MAIN_FUNC_NAME
#define DBOUT(TYPE, X)
LLVM debug macros, define type of your DBUG model of each pass.
const char *const const double number
std::string getExtBcPath()
static ExtAPI * getExtAPI()
const SVFType * maxStruct
The struct type with the most fields.
void addStInfo(StInfo *stInfo)
NodeID blkPtrSymID() const
NodeID nullPtrSymID() const
void addTypeInfo(const SVFType *ty)
u32_t maxStSize
The number of fields in max_struct.
bool hasValueNode(const Value *V)
NodeID getValueNode(const Value *V)
IntraICFGNode * getIntraBlock(const Instruction *inst)
LLVMType2SVFTypeMap LLVMType2SVFType
bool is_alloc_stack_ret(const Function *F)
std::vector< const Function * > getLLVMGlobalFunctions(const GlobalVariable *global)
void buildSymbolTable() const
Module * getMainLLVMModule() const
const FunObjVar * getFunObjVar(const Function *fun) const
ValueToIDMapTy valSymMap
map a value to its sym id
StInfo * collectTypeInfo(const Type *ty)
Collect a type info.
static LLVMModuleSet * getLLVMModuleSet()
void loadModules(const std::vector< std::string > &moduleNameVec)
ObjTypeInference * typeInference
static void releaseLLVMModuleSet()
static void preProcessBCs(std::vector< std::string > &moduleNameVec)
DominatorTree & getDomTree(const Function *fun)
void addToSVFVar2LLVMValueMap(const Value *val, SVFValue *svfBaseNode)
RetICFGNode * getRetBlock(const Instruction *cs)
Get/Add a return node.
std::unique_ptr< LLVMContext > owned_ctx
void createSVFDataStructure()
StInfo * collectSimpleTypeInfo(const Type *T)
Collect simple type (non-aggregate) info.
bool is_alloc(const Function *F)
const std::vector< std::string > & getExtFuncAnnotations(const Function *fun)
SVFType * getSVFType(const Type *T)
Get or create SVFType and typeinfo.
void setFunExitBB(const Function *fun, const BasicBlock *bb)
const Type * getLLVMType(const SVFType *T) const
Get LLVM Type.
void setExtFuncAnnotations(const Function *fun, const std::vector< std::string > &funcAnnotations)
bool hasICFGNode(const Instruction *inst)
CallICFGNode * getCallBlock(const Instruction *cs)
Get/Add a call node.
ValueToIDMapTy objSymMap
map a obj reference to its sym id
ICFGNode * getICFGNode(const Instruction *inst)
Get a basic block ICFGNode.
const Function * getFunction(const std::string &name)
Get the corresponding Function based on its name.
CallICFGNode * getCallICFGNode(const Instruction *cs)
get a call node
Fun2AnnoMap ExtFun2Annotations
Record annotations of function in extapi.bc.
bool is_arg_alloc(const Function *F)
Map< const Function *, std::vector< std::string > > func2Annotations
const BasicBlock * getFunExitBB(const Function *fun) const
NodeID getObjectNode(const Value *V)
Module * getModule(u32_t idx) const
RetICFGNode * getRetICFGNode(const Instruction *cs)
get a return node
void prePassSchedule()
Invoke llvm passes to modify module.
StInfo * collectStructInfo(const StructType *structTy, u32_t &numFields)
Collect the struct info and set the number of fields after flattening.
s32_t get_alloc_arg_pos(const Function *F)
void addFunctionSet(const Function *svfFunc)
const std::vector< std::reference_wrapper< Module > > & getLLVMModules() const
LLVMContext & getContext() const
bool is_ext(const Function *F)
Map< const Function *, DominatorTree > FunToDominatorTree
static void buildSVFModule(Module &mod)
SVFBaseNode2LLVMValueMap SVFBaseNode2LLVMValue
StInfo * collectArrayInfo(const ArrayType *T)
Collect the array info.
LLVMModuleSet()
Constructor.
void buildGlobalDefToRepMap()
SVFType * addSVFTypeInfo(const Type *t)
Create SVFTypes.
FunctionSetType ExtFuncsVec
Record some "sse_" function declarations used in other ext function definition, e....
bool hasExtFuncAnnotation(const Function *fun, const std::string &funcAnnotation)
static LLVMModuleSet * llvmModuleSet
IntraICFGNode * getIntraICFGNode(const Instruction *inst)
get a intra node
std::vector< std::reference_wrapper< Module > > modules
std::string getExtFuncAnnotation(const Function *fun, const std::string &funcAnnotation)
Map< std::string, std::vector< std::string > > Fun2AnnoMap
u32_t getModuleNum() const
GlobalDefToRepMapTy GlobalDefToRepMap
Global definition to a rep definition map.
std::vector< std::unique_ptr< Module > > owned_modules
bool is_memset(const Function *F)
FunToIDMapTy & varargSyms()
Type2TypeInfoMap Type2TypeInfo
void dumpModulesToFile(const std::string &suffix)
void collectExtFunAnnotations(const Module *mod)
bool is_memcpy(const Function *F)
ObjTypeInference * getTypeInference()
bool is_realloc(const Function *F)
static const Option< bool > SVFMain
static const Option< std::string > Graphtxt
static bool pagReadFromTXT()
void setModuleIdentifier(const std::string &moduleIdentifier)
static SVFIR * getPAG(bool buildFromFile=false)
Singleton design here to make sure we only have one instance during any analysis.
static void setPagFromTXT(const std::string &txt)
static double timeOfBuildingLLVMModule
static double getClk(bool mark=false)
static double timeOfBuildingSymbolTable
static SVFType * svfPtrTy
ptr type
static SVFType * svfI8Ty
8-bit int type
bool isIntrinsicInst(const Instruction *inst)
Return true if it is an intrinsic instruction.
std::vector< const Function * > getCalledFunctions(const Function *F)
Get all called funcions in a parent function.
bool isCallSite(const Instruction *inst)
Whether an instruction is a call or invoke instruction.
bool functionDoesNotRet(const Function *fun)
std::pair< s64_t, u64_t > getIntegerValue(const ConstantInt *intValue)
const std::string getSourceLoc(const Value *val)
const Value * getGlobalRep(const Value *val)
find the unique defined global across multiple modules
bool basicBlockHasRetInst(const BasicBlock *bb)
Return true if the function has a return instruction.
bool isIRFile(const std::string &filename)
Check whether a file is an LLVM IR file.
std::string restoreFuncName(std::string funcName)
bool isNonInstricCallSite(const Instruction *inst)
Whether an instruction is a callsite in the application code, excluding llvm intrinsic calls.
std::string pasMsg(const std::string &msg)
Print each pass/phase message by converting a string into blue string output.
std::ostream & errs()
Overwrite llvm::errs()
std::ostream & outs()
Overwrite llvm::outs()
llvm::GlobalVariable GlobalVariable
llvm::ArrayType ArrayType
llvm::raw_fd_ostream raw_fd_ostream
LLVM outputs.
llvm::BasicBlock BasicBlock
llvm::inst_iterator inst_iterator
llvm::ConstantStruct ConstantStruct
llvm::StructType StructType
LLVM types.
llvm::IntegerType IntegerType
llvm::ConstantArray ConstantArray
llvm::FunctionType FunctionType
llvm::Instruction Instruction
llvm::SMDiagnostic SMDiagnostic
llvm::ConstantDataSequential ConstantDataSequential
llvm::Value Value
LLVM Basic classes.
llvm::ConstantExpr ConstantExpr
llvm::IRBuilder IRBuilder
llvm::ConstantInt ConstantInt
llvm::DominatorTree DominatorTree
LLVM Dominators.
IntervalValue operator>(const IntervalValue &lhs, const IntervalValue &rhs)