Static Value-Flow Analysis
Loading...
Searching...
No Matches
LLVMModule.cpp
Go to the documentation of this file.
1//===----- SVFModule.cpp Base class of pointer analyses ---------------------//
2//
3// SVF: Static Value-Flow Analysis
4//
5// Copyright (C) <2013-2017> <Yulei Sui>
6//
7
8// This program is free software: you can redistribute it and/or modify
9// it under the terms of the GNU Affero General Public License as published by
10// the Free Software Foundation, either version 3 of the License, or
11// (at your option) any later version.
12
13// This program is distributed in the hope that it will be useful,
14// but WITHOUT ANY WARRANTY; without even the implied warranty of
15// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16// GNU Affero General Public License for more details.
17
18// You should have received a copy of the GNU Affero General Public License
19// along with this program. If not, see <http://www.gnu.org/licenses/>.
20//
21//===----------------------------------------------------------------------===//
22
23/*
24 * SVFModule.cpp
25 *
26 * Created on: Aug 4, 2017
27 * Author: Xiaokang Fan
28 */
29
30#include <queue>
31#include <algorithm>
32#include "Util/SVFUtil.h"
33#include "SVF-LLVM/BasicTypes.h"
34#include "SVF-LLVM/LLVMUtil.h"
35#include "SVF-LLVM/CppUtil.h"
38#include "MSSA/SVFGBuilder.h"
39#include "llvm/Support/FileSystem.h"
41#include "llvm/Transforms/Utils/Cloning.h"
43#include "Graphs/CallGraph.h"
45
46using namespace std;
47using namespace SVF;
48
49/*
50 svf.main() is used to model the real entry point of a C++ program, which
51 initializes all global C++ objects and then call main().
52 LLVM may generate two global arrays @llvm.global_ctors and @llvm.global_dtors
53 that contain constructor and destructor functions for global variables. They
54 are not called explicitly, so we have to add them in the svf.main function.
55 The order to call these constructor and destructor functions are also
56 specified in the global arrays.
57 Related part in LLVM language reference:
58 https://llvm.org/docs/LangRef.html#the-llvm-global-ctors-global-variable
59 For example, given a "int main(int argc, char * argv[])", the corresponding
60 svf.main will be generated as follows:
61 define void @svf.main(i32, i8**, i8**) {
62 entry:
63 call void @ctor1()
64 call void @ctor2()
65 %3 = call i32 @main(i32 %0, i8** %1)
66 call void @dtor1()
67 call void @dtor2()
68 ret void
69 }
70*/
71
72#define SVF_MAIN_FUNC_NAME "svf.main"
73#define SVF_GLOBAL_CTORS "llvm.global_ctors"
74#define SVF_GLOBAL_DTORS "llvm.global_dtors"
75
78
80 : svfir(PAG::getPAG()), typeInference(new ObjTypeInference())
81{
82}
83
85{
86
87 delete typeInference;
88 typeInference = nullptr;
89
90}
91
96
98{
99 auto it = FunToDominatorTree.find(fun);
100 if(it != FunToDominatorTree.end()) return it->second;
102 dt.recalculate(const_cast<Function&>(*fun));
103 return dt;
104}
105
107{
109
110 double startSVFModuleTime = SVFStat::getClk(true);
111 PAG::getPAG()->setModuleIdentifier(mod.getModuleIdentifier());
112 mset->modules.emplace_back(mod); // Populates `modules`; can get context via `this->getContext()`
113 mset->loadExtAPIModules(); // Uses context from module through `this->getContext()`
114 mset->build();
115 double endSVFModuleTime = SVFStat::getClk(true);
117
118 mset->buildSymbolTable();
119}
120
121void LLVMModuleSet::buildSVFModule(const std::vector<std::string> &moduleNameVec)
122{
123 double startSVFModuleTime = SVFStat::getClk(true);
124
126
127 mset->loadModules(moduleNameVec); // Populates `modules`; can get context via `this->getContext()`
128 mset->loadExtAPIModules(); // Uses context from first module through `this->getContext()`
129
130 if (!moduleNameVec.empty())
131 {
133 }
134
135 mset->build();
136
137 double endSVFModuleTime = SVFStat::getClk(true);
140
141 mset->buildSymbolTable();
142}
143
145{
146 double startSymInfoTime = SVFStat::getClk(true);
148 {
150 DBOUT(DGENERAL, SVFUtil::outs() << SVFUtil::pasMsg("Building Symbol table ...\n"));
152 builder.buildMemModel();
153 }
154 double endSymInfoTime = SVFStat::getClk(true);
157}
158
160{
161 if(preProcessed==false)
163
166
167 if (Options::SVFMain())
168 addSVFMain();
169
171
172}
173
175{
178 // Functions need to be retrieved in the order of insertion
179 // candidateDefs is the vector for all used defined functions
180 // candidateDecls is the vector for all used declared functions
181 std::vector<const Function*> candidateDefs, candidateDecls;
182
183 for (Module& mod : modules)
184 {
186 for (Function& func : mod.functions())
187 {
188 if (func.isDeclaration())
189 {
190 candidateDecls.push_back(&func);
191 }
192 else
193 {
194 candidateDefs.push_back(&func);
195 }
196 }
197 }
198
199 for (const Function* func: candidateDefs)
200 {
202 }
203 for (const Function* func: candidateDecls)
204 {
206 }
207
208 // set function exit block
209 for (const auto& func: funSet)
210 {
211 for (Function::const_iterator bit = func->begin(), ebit = func->end(); bit != ebit; ++bit)
212 {
213 const BasicBlock* bb = &*bit;
215 if (succ_size(bb) == 0)
216 {
218 {
220 SVFUtil::isa<ReturnInst>(bb->back())) &&
221 "last inst must be return inst");
222 setFunExitBB(func, bb);
223 }
224 }
225 }
226 // For no return functions, we set the last block as exit BB
227 // This ensures that each function that has definition must have an exit BB
228 if (func->size() != 0 && !getFunExitBB(func))
229 {
231 SVFUtil::isa<ReturnInst>(&func->back().back())) &&
232 "last inst must be return inst");
233 setFunExitBB(func, &func->back());
234 }
235 }
236
237 // Store annotations of functions in extapi.bc
238 for (const auto& pair : ExtFun2Annotations)
239 {
240 const Function* fun = getFunction(pair.first);
241 setExtFuncAnnotations(fun, pair.second);
242 }
243
244}
245
250{
252 std::unique_ptr<BreakConstantGEPs> p1 = std::make_unique<BreakConstantGEPs>();
253 for (Module &M : getLLVMModules())
254 {
255 p1->runOnModule(M);
256 }
257
259 std::unique_ptr<UnifyFunctionExitNodes> p2 =
260 std::make_unique<UnifyFunctionExitNodes>();
261 for (Module &M : getLLVMModules())
262 {
263 for (auto F = M.begin(), E = M.end(); F != E; ++F)
264 {
265 Function &fun = *F;
266 if (fun.isDeclaration())
267 continue;
268 p2->runOnFunction(fun);
269 }
270 }
271}
272
273void LLVMModuleSet::preProcessBCs(std::vector<std::string> &moduleNameVec)
274{
276 mset->loadModules(moduleNameVec);
277 mset->loadExtAPIModules();
278 mset->prePassSchedule();
279
280 std::string preProcessSuffix = ".pre.bc";
281 // Get the existing module names, remove old extension, add preProcessSuffix
282 for (u32_t i = 0; i < moduleNameVec.size(); i++)
283 {
284 u32_t lastIndex = moduleNameVec[i].find_last_of(".");
285 std::string rawName = moduleNameVec[i].substr(0, lastIndex);
287 }
288
289 mset->dumpModulesToFile(preProcessSuffix);
290 preProcessed = true;
292}
293
294void LLVMModuleSet::loadModules(const std::vector<std::string> &moduleNameVec)
295{
296
297 // We read SVFIR from LLVM IR
299 {
300 if(moduleNameVec.empty())
301 {
302 SVFUtil::outs() << "no LLVM bc file is found!\n";
303 exit(0);
304 }
305 //assert(!moduleNameVec.empty() && "no LLVM bc file is found!");
306 }
307 // We read SVFIR from a user-defined txt instead of parsing SVFIR from LLVM IR
308 else
309 {
311 }
312 //
313 // LLVMContext objects separate global LLVM settings (from which e.g. types are
314 // derived); multiple LLVMContext objects can coexist and each context can "own"
315 // multiple modules (modules can only have one context). Mixing contexts can lead
316 // to unintended inequalities, such as the following:
317 //
318 // ------------------------------------------------------------------
319 // LLVMContext ctxa,ctxb;
320 // IntegerType * t1 = IntegerType::get(ctxa,32);
321 // IntegerType * t2 = IntegerType::get(ctxa,32);
322 // assert(t1 == t2);
323 // IntegerType * t3 = IntegerType::get(ctxb,32);
324 // IntegerType * t4 = IntegerType::get(ctxb,32);
325 // assert(t3 == t4);
326 // assert(t1 != t3);
327 // ------------------------------------------------------------------
328 //
329 // When loading bytecode files, SVF will use the same LLVMContext object for all
330 // modules (i.e. the context owns all loaded modules). This applies to ExtAPI as
331 // well, which *must* be loaded using the same LLVMContext object. Hence, when
332 // loading modules from bitcode files, a new LLVMContext is created (using a
333 // `std::unique_ptr<LLVMContext>` type to ensure automatic garbage collection).
334 //
335 // This garbage collection should be avoided when building an SVF module from an LLVM
336 // module instance; see the comment(s) in `buildSVFModule` and `loadExtAPIModules()`
337
338 owned_ctx = std::make_unique<LLVMContext>();
339 for (const std::string& moduleName : moduleNameVec)
340 {
341 if (!LLVMUtil::isIRFile(moduleName))
342 {
343 SVFUtil::errs() << "not an IR file: " << moduleName << std::endl;
344 abort();
345 }
346
348 std::unique_ptr<Module> mod = parseIRFile(moduleName, Err, *owned_ctx);
349 if (mod == nullptr)
350 {
351 SVFUtil::errs() << "load module: " << moduleName << "failed!!\n\n";
352 Err.print("SVFModuleLoader", llvm::errs());
353 abort();
354 }
355 modules.emplace_back(*mod);
356 owned_modules.emplace_back(std::move(mod));
357 }
358}
359
361{
362 // This function loads the ExtAPI bitcode file as an LLVM module. Note that it is important that
363 // the same LLVMContext object is used to load this bitcode file as is used by the other modules
364 // being analysed.
365 // When the modules are loaded from bitcode files (i.e. passing filenames to files containing
366 // LLVM IR to `buildSVFModule({file1.bc, file2.bc, ...})) the context is created while loading
367 // the modules in `loadModules()`, which populates this->modules and this->owned_modules.
368 // If, however, an LLVM Module object is passed to `buildSVFModule` (e.g. from an LLVM pass),
369 // the context should be retrieved from the module itself (note that the garbage collection from
370 // `std::unique_ptr<LLVMContext> LLVMModuleSet::owned_ctx` should be avoided in this case). This
371 // function populates only this->modules.
372 // In both cases, fetching the context from the main LLVM module (through `getContext`) works
373 assert(!empty() && "LLVMModuleSet contains no modules; cannot load ExtAPI module without LLVMContext!");
374
375 // Load external API module (extapi.bc)
376 if (!ExtAPI::getExtAPI()->getExtBcPath().empty())
377 {
380 {
381 SVFUtil::errs() << "not an external IR file: " << extModuleName << std::endl;
382 abort();
383 }
385 std::unique_ptr<Module> mod = parseIRFile(extModuleName, Err, getContext());
386 if (mod == nullptr)
387 {
388 SVFUtil::errs() << "load external module: " << extModuleName << "failed!!\n\n";
389 Err.print("SVFModuleLoader", llvm::errs());
390 abort();
391 }
392 // The module of extapi.bc needs to be inserted before applications modules, like std::vector<std::reference_wrapper<Module>> modules{extapi_module, app_module}.
393 // Otherwise, when overwriting the app function with SVF extern function, the corresponding SVFFunction of the extern function will not be found.
394 modules.insert(modules.begin(), *mod);
395 owned_modules.insert(owned_modules.begin(),std::move(mod));
396 }
397}
398
400{
401 // This function is used to extract constructor and destructor functions
402 // sorted by their priority from @llvm.global_ctors or @llvm.global_dtors.
403 // For example, given following @llvm.global_ctors, the returning sorted
404 // function list should be [ctor3, ctor1, ctor2].
405 // ------------------------------------------------------------------
406 // ; Each struct in the array is {priority, function, associated data}
407 //
408 // @llvm.global_ctors = appending global [2 x { i32, void ()*, i8* }]
409 // [{ i32, void ()*, i8* } { i32 1234, void ()* @ctor1.cpp, i8* null },
410 // { i32, void ()*, i8* } { i32 2345, void ()* @ctor2.cpp, i8* null },
411 // { i32, void ()*, i8* } { i32 345, void ()* @ctor3.cpp, i8* null }]
412 // ------------------------------------------------------------------
413 // TODO: According to LLVM language reference, if the third field is
414 // non-null, and points to a global variable or function, the initializer
415 // function will only run if the associated data from the current module is
416 // not discarded. However the associated data is currently ignored.
417
418
419 // This class is used for the priority queue that sorts the functions by
420 // their priority. Each object of this class stands for an item in the
421 // function array.
423 {
424 public:
426 const Function* func;
430 bool operator>(const LLVMGlobalFunction &other) const
431 {
432 if (priority != other.priority)
433 {
434 return priority > other.priority;
435 }
436 else
437 {
438 return func > other.func;
439 }
440 }
441 };
442
443 std::priority_queue<LLVMGlobalFunction, std::vector<LLVMGlobalFunction>,
445 queue;
446 std::vector<const Function* > result;
447
448 // The @llvm.global_ctors/dtors global variable is an array of struct. Each
449 // struct has three fields: {i32 priority, void ()* @ctor/dtor, i8* @data}.
450 // First get the array here.
452 SVFUtil::dyn_cast<ConstantArray>(global->getInitializer()))
453 {
454 // Get each struct in the array.
455 for (unsigned int i = 0; i < globalFuncArray->getNumOperands(); ++i)
456 {
457 if (
459 SVFUtil::dyn_cast<ConstantStruct>(
460 globalFuncArray->getOperand(i)))
461 {
462
463 // Extract priority and function from the struct
464 const ConstantInt* priority = SVFUtil::dyn_cast<ConstantInt>(
465 globalFuncItem->getOperand(0));
466 const Function* func = SVFUtil::dyn_cast<Function>(
467 globalFuncItem->getOperand(1));
468
469 if (priority && func)
470 {
472 func));
473 }
474 }
475 }
476 }
477
478 // Generate a sorted vector of functions from the priority queue.
479 while (!queue.empty())
480 {
481 result.push_back(queue.top().func);
482 queue.pop();
483 }
484 return result;
485}
486
488{
489 std::vector<const Function*> ctor_funcs;
490 std::vector<const Function*> dtor_funcs;
491 Function* orgMain = 0;
492 Module* mainMod = nullptr;
493
494 for (Module &mod : modules)
495 {
496 // Collect ctor and dtor functions
497 for (const GlobalVariable& global : mod.globals())
498 {
499 if (global.getName().equals(SVF_GLOBAL_CTORS) && global.hasInitializer())
500 {
502 }
503 else if (global.getName().equals(SVF_GLOBAL_DTORS) && global.hasInitializer())
504 {
506 }
507 }
508
509 // Find main function
510 for (auto &func : mod)
511 {
512 auto funName = func.getName();
513
514 assert(!funName.equals(SVF_MAIN_FUNC_NAME) && SVF_MAIN_FUNC_NAME " already defined");
515
516 if (funName.equals("main"))
517 {
518 orgMain = &func;
519 mainMod = &mod;
520 }
521 }
522 }
523
524 // Only create svf.main when the original main function is found, and also
525 // there are global constructor or destructor functions.
526 if (orgMain && getModuleNum() > 0 &&
527 (ctor_funcs.size() > 0 || dtor_funcs.size() > 0))
528 {
529 assert(mainMod && "Module with main function not found.");
530 Module& M = *mainMod;
531 // char **
532 Type* ptr = PointerType::getUnqual(M.getContext());
533 Type* i32 = IntegerType::getInt32Ty(M.getContext());
534 // define void @svf.main(i32, i8**, i8**)
535#if (LLVM_VERSION_MAJOR >= 9)
536 FunctionCallee svfmainFn = M.getOrInsertFunction(
538 Type::getVoidTy(M.getContext()),
539 i32,ptr,ptr
540 );
541 Function* svfmain = SVFUtil::dyn_cast<Function>(svfmainFn.getCallee());
542#else
543 Function* svfmain = SVFUtil::dyn_cast<Function>(M.getOrInsertFunction(
545 Type::getVoidTy(M.getContext()),
547 ));
548#endif
549 svfmain->setCallingConv(llvm::CallingConv::C);
550 BasicBlock* block = BasicBlock::Create(M.getContext(), "entry", svfmain);
552 // emit "call void @ctor()". ctor_funcs is sorted so the functions are
553 // emitted in the order of priority
554 for (auto& ctor : ctor_funcs)
555 {
556 auto target = M.getOrInsertFunction(
557 ctor->getName(),
558 Type::getVoidTy(M.getContext())
559 );
560 Builder.CreateCall(target);
561 }
562 // main() should be called after all ctor functions and before dtor
563 // functions.
564 Function::arg_iterator arg_it = svfmain->arg_begin();
565 Value* args[] = {arg_it, arg_it + 1, arg_it + 2};
566 size_t cnt = orgMain->arg_size();
567 assert(cnt <= 3 && "Too many arguments for main()");
568 Builder.CreateCall(orgMain, llvm::ArrayRef<Value*>(args, args + cnt));
569 // emit "call void @dtor()". dtor_funcs is sorted so the functions are
570 // emitted in the order of priority
571 for (auto& dtor : dtor_funcs)
572 {
573 auto target = M.getOrInsertFunction(dtor->getName(), Type::getVoidTy(M.getContext()));
574 Builder.CreateCall(target);
575 }
576 // return;
577 Builder.CreateRetVoid();
578 }
579}
580
582{
583 GlobalVariable *glob = mod->getGlobalVariable("llvm.global.annotations");
584 if (glob == nullptr || !glob->hasInitializer())
585 return;
586
587 ConstantArray *ca = SVFUtil::dyn_cast<ConstantArray>(glob->getInitializer());
588 if (ca == nullptr)
589 return;
590
591 for (unsigned i = 0; i < ca->getNumOperands(); ++i)
592 {
593 ConstantStruct *structAn = SVFUtil::dyn_cast<ConstantStruct>(ca->getOperand(i));
594 if (structAn == nullptr || structAn->getNumOperands() == 0)
595 continue;
596
597 // Check if the annotation is for a function
598 Function* fun = nullptr;
599 GlobalVariable *annotateStr = nullptr;
601 if (ConstantExpr *expr = SVFUtil::dyn_cast<ConstantExpr>(structAn->getOperand(0)))
602 {
603 if (expr->getOpcode() == Instruction::BitCast && SVFUtil::isa<Function>(expr->getOperand(0)))
604 fun = SVFUtil::cast<Function>(expr->getOperand(0));
605
606 ConstantExpr *note = SVFUtil::cast<ConstantExpr>(structAn->getOperand(1));
607 if (note->getOpcode() != Instruction::GetElementPtr)
608 continue;
609
610 annotateStr = SVFUtil::dyn_cast<GlobalVariable>(note->getOperand(0));
611 }
613 else
614 {
615 fun = SVFUtil::dyn_cast<Function>(structAn->getOperand(0));
616 annotateStr = SVFUtil::dyn_cast<GlobalVariable>(structAn->getOperand(1));
617 }
618
619 if (!fun || annotateStr == nullptr || !annotateStr->hasInitializer())
620 continue;;
621
622 ConstantDataSequential *data = SVFUtil::dyn_cast<ConstantDataSequential>(annotateStr->getInitializer());
623 if (data && data->isString())
624 {
625 std::string annotation = data->getAsString().str();
626 if (!annotation.empty())
627 ExtFun2Annotations[fun->getName().str()].push_back(annotation);
628 }
629 }
630}
631
632/*
633 There are three types of functions(definitions) in extapi.c:
634 1. (Fun_Overwrite): Functions with "OVERWRITE" annotion:
635 These functions are used to replace the corresponding function definitions in the application.
636 2. (Fun_Annotation): Functions with annotation(s) but without "OVERWRITE" annotation:
637 These functions are used to tell SVF to do special processing, like malloc().
638 3. (Fun_Noraml): Functions without any annotation:
639 These functions are used to replace the corresponding function declarations in the application.
640
641
642 We will iterate over declarations (appFunDecl) and definitons (appFunDef) of functions in the application and extapi.c to do the following clone or replace operations:
643 1. appFuncDecl --> Fun_Normal: Clone the Fun_Overwrite and replace the appFuncDecl in application.
644 2. appFuncDecl --> Fun_Annotation: Move the annotions on Fun_Annotation to appFuncDecl in application.
645
646 3. appFunDef --> Fun_Overwrite: Clone the Fun_Overwrite and overwrite the appFunDef in application.
647 4. appFunDef --> Fun_Annotation: Replace the appFunDef with appFunDecl and move the annotions to appFunDecl in application
648*/
650{
654 Module* appModule = nullptr;
655 Module* extModule = nullptr;
656
657 for (Module& mod : modules)
658 {
659 // extapi.bc functions
660 if (mod.getName().str() == ExtAPI::getExtAPI()->getExtBcPath())
661 {
663 extModule = &mod;
664 for (const Function& fun : mod.functions())
665 {
666 // there is main declaration in ext bc, it should be mapped to
667 // main definition in app bc.
668 if (fun.getName().str() == "main")
669 {
670 appFunDecls.insert(&fun);
671 appFuncDeclNames.insert(fun.getName().str());
672 }
674 else if (fun.getName().str() == "svf__main")
675 {
676 ExtFuncsVec.push_back(&fun);
677 }
678 else
679 {
680 extFuncs.insert(&fun);
681 extFunDefNames.insert(fun.getName().str());
682 }
683 }
684 }
685 else
686 {
687 appModule = &mod;
689 for (const Function& fun : mod.functions())
690 {
691 if (fun.isDeclaration())
692 {
693 appFunDecls.insert(&fun);
694 appFuncDeclNames.insert(fun.getName().str());
695 }
696 else
697 {
698 appFunDefs.insert(&fun);
699 appFuncDefNames.insert(fun.getName().str());
700 }
701 }
702 }
703 }
704
705 // Find the intersectNames between appFuncDefNames and externalFunDefNames
706 std::set_intersection(
707 appFuncDefNames.begin(), appFuncDefNames.end(), extFunDefNames.begin(), extFunDefNames.end(),
708 std::inserter(intersectNames, intersectNames.end()));
709
711 {
712 assert(!(appFunToReplace == NULL && appModule == NULL) && "appFunToReplace and appModule cannot both be NULL");
713
714 if (appFunToReplace)
715 {
716 appModule = appFunToReplace->getParent();
717 }
718 // Create a new function with the same signature as extFunToClone
719 Function *clonedFunction = Function::Create(extFunToClone->getFunctionType(), Function::ExternalLinkage, extFunToClone->getName(), appModule);
720 // Map the arguments of the new function to the arguments of extFunToClone
721 llvm::ValueToValueMapTy valueMap;
722 Function::arg_iterator destArg = clonedFunction->arg_begin();
723 for (Function::const_arg_iterator srcArg = extFunToClone->arg_begin(); srcArg != extFunToClone->arg_end(); ++srcArg)
724 {
725 destArg->setName(srcArg->getName()); // Copy the name of the original argument
726 valueMap[&*srcArg] = &*destArg++; // Add a mapping from the old arg to the new arg
727 }
728 if (cloneBody)
729 {
730 // Clone the body of extFunToClone into clonedFunction
731 llvm::SmallVector<ReturnInst*, 8> ignoredReturns;
732 CloneFunctionInto(clonedFunction, extFunToClone, valueMap, llvm::CloneFunctionChangeType::LocalChangesOnly, ignoredReturns, "", nullptr);
733 }
734 if (appFunToReplace)
735 {
736 // Replace all uses of appFunToReplace with clonedFunction
737 appFunToReplace->replaceAllUsesWith(clonedFunction);
738 std::string oldFunctionName = appFunToReplace->getName().str();
739 // Delete the old function
740 appFunToReplace->eraseFromParent();
742 }
743 return clonedFunction;
744 };
745
747 for (const Function* appFunDecl : appFunDecls)
748 {
749 std::string appFunDeclName = LLVMUtil::restoreFuncName(appFunDecl->getName().str());
750 for (const Function* extFun : extFuncs)
751 {
752 if (extFun->getName().str().compare(appFunDeclName) == 0)
753 {
754 auto it = ExtFun2Annotations.find(extFun->getName().str());
755 // Without annotations, this function is normal function with useful function body
756 if (it == ExtFun2Annotations.end())
757 {
758 Function* clonedFunction = cloneAndReplaceFunction(const_cast<Function*>(extFun), const_cast<Function*>(appFunDecl), nullptr, true);
761 }
762 else
763 {
764 ExtFuncsVec.push_back(appFunDecl);
765 }
766 break;
767 }
768 }
769 }
770
773 for (string sameFuncDef: intersectNames)
774 {
775 Function* appFuncDef = appModule->getFunction(sameFuncDef);
776 Function* extFuncDef = extModule->getFunction(sameFuncDef);
777 if (appFuncDef == nullptr || extFuncDef == nullptr)
778 continue;
779
780 FunctionType *appFuncDefType = appFuncDef->getFunctionType();
781 FunctionType *extFuncDefType = extFuncDef->getFunctionType();
783 continue;
784
785 auto it = ExtFun2Annotations.find(sameFuncDef);
786 if (it != ExtFun2Annotations.end())
787 {
788 std::vector<std::string> annotations = it->second;
789 if (annotations.size() == 1 && annotations[0].find("OVERWRITE") != std::string::npos)
790 {
791 Function* clonedFunction = cloneAndReplaceFunction(const_cast<Function*>(extFuncDef), const_cast<Function*>(appFuncDef), nullptr, true);
794 }
795 else
796 {
797 if (annotations.size() >= 2)
798 {
799 for (const auto& annotation : annotations)
800 {
801 if(annotation.find("OVERWRITE") != std::string::npos)
802 {
803 assert(false && "overwrite and other annotations cannot co-exist");
804 }
805 }
806 }
807 }
808 }
809 }
810
812 {
813 for (inst_iterator I = inst_begin(caller), E = inst_end(caller); I != E; ++I)
814 {
815 Instruction *inst = &*I;
816
817 if (CallInst *callInst = SVFUtil::dyn_cast<CallInst>(inst))
818 {
819 Function *calledFunc = callInst->getCalledFunction();
820
821 if (calledFunc && calledFunc->getName() == callee->getName())
822 {
823 callInst->setCalledFunction(callee);
824 }
825 }
826 }
827 };
828
829 std::function<void(const Function*, Function*)> cloneAndLinkFunction;
831 {
832 if (clonedFuncs.find(extFunToClone) != clonedFuncs.end())
833 return;
834
835 Module* appModule = appClonedFun->getParent();
836 // Check if the function already exists in the parent module
837 if (appModule->getFunction(extFunToClone->getName()))
838 {
839 // The function already exists, no need to clone, but need to link it with the caller
840 Function* func = appModule->getFunction(extFunToClone->getName());
842 return;
843 }
844 // Decide whether to clone the function body based on ExtFun2Annotations
845 bool cloneBody = true;
846 auto it = ExtFun2Annotations.find(extFunToClone->getName().str());
847 if (it != ExtFun2Annotations.end())
848 {
849 std::vector<std::string> annotations = it->second;
850 if (!(annotations.size() == 1 && annotations[0].find("OVERWRITE") != std::string::npos))
851 {
852 cloneBody = false;
853 }
854 }
855
857
859 // Add the cloned function to ExtFuncsVec for further processing
860 ExtFuncsVec.push_back(clonedFunction);
861
863
864 std::vector<const Function*> calledFunctions = LLVMUtil::getCalledFunctions(extFunToClone);
865
866 for (const auto& calledFunction : calledFunctions)
867 {
869 }
870 };
871
872 // Recursive clone called functions
873 for (const auto& pair : extFuncs2ClonedFuncs)
874 {
875 Function* extFun = const_cast<Function*>(pair.first);
876 Function* clonedExtFun = const_cast<Function*>(pair.second);
877 std::vector<const Function*> extCalledFuns = LLVMUtil::getCalledFunctions(extFun);
878
879 for (const auto& extCalledFun : extCalledFuns)
880 {
882 }
883 }
884
885 // Remove unused annotations in ExtFun2Annotations according to the functions in ExtFuncsVec
887 for (const Function* extFun : ExtFuncsVec)
888 {
889 std::string name = LLVMUtil::restoreFuncName(extFun->getName().str());
890 auto it = ExtFun2Annotations.find(name);
891 if (it != ExtFun2Annotations.end())
892 {
893 std::string newKey = name;
894 if (name != extFun->getName().str())
895 {
896 newKey = extFun->getName().str();
897 }
898 newFun2AnnoMap.insert({newKey, it->second});
899 }
900 }
902
903 // Remove ExtAPI module from modules
904 auto it = std::find_if(modules.begin(), modules.end(),
905 [&extModule](const std::reference_wrapper<llvm::Module>& moduleRef)
906 {
907 return &moduleRef.get() == extModule;
908 });
909
910 if (it != modules.end())
911 {
912 size_t index = std::distance(modules.begin(), it);
913 modules.erase(it);
914 owned_modules.erase(owned_modules.begin() + index);
915 }
916}
917
919{
922 for (Module &mod : modules)
923 {
924 // Collect ctor and dtor functions
925 for (GlobalVariable& global : mod.globals())
926 {
927 if (global.hasPrivateLinkage())
928 continue;
929 string name = global.getName().str();
930 if (name.empty())
931 continue;
932 nameToGlobalsMap[std::move(name)].insert(&global);
933 }
934 }
935
936 for (const auto& pair : nameToGlobalsMap)
937 {
938 const Set<GlobalVariable*> &globals = pair.second;
939
940 const auto repIt =
941 std::find_if(globals.begin(), globals.end(),
942 [](GlobalVariable* g)
943 {
944 return g->hasInitializer();
945 });
946 GlobalVariable* rep =
947 repIt != globals.end()
948 ? *repIt
949 // When there is no initializer, just pick the first one.
950 : (assert(!globals.empty() && "Empty global set"),
951 *globals.begin());
952
953 for (const GlobalVariable* cur : globals)
954 {
955 GlobalDefToRepMap[cur] = rep;
956 }
957 }
958}
959
960// Dump modules to files
962{
963 for (Module& mod : modules)
964 {
965 std::string moduleName = mod.getName().str();
966 std::string OutputFilename;
967 std::size_t pos = moduleName.rfind('.');
968 if (pos != std::string::npos)
969 OutputFilename = moduleName.substr(0, pos) + suffix;
970 else
971 OutputFilename = moduleName + suffix;
972
973 std::error_code EC;
974 raw_fd_ostream OS(OutputFilename.c_str(), EC, llvm::sys::fs::OF_None);
975
976#if (LLVM_VERSION_MAJOR >= 7)
978#else
980#endif
981
982 OS.flush();
983 }
984}
985
987{
988 if (SVFUtil::isa<ConstantPointerNull>(llvm_value))
989 return svfir->nullPtrSymID();
990 else if (SVFUtil::isa<UndefValue>(llvm_value))
991 return svfir->blkPtrSymID();
992 else
993 {
994 ValueToIDMapTy::const_iterator iter = valSymMap.find(llvm_value);
995 assert(iter!=valSymMap.end() &&"value sym not found");
996 return iter->second;
997 }
998}
1000{
1001 if (SVFUtil::isa<ConstantPointerNull, UndefValue>(val))
1002 return true;
1003 else
1004 return (valSymMap.find(val) != valSymMap.end());
1005}
1006
1008{
1009 if (const GlobalVariable* glob = SVFUtil::dyn_cast<GlobalVariable>(llvm_value))
1011 ValueToIDMapTy::const_iterator iter = objSymMap.find(llvm_value);
1012 assert(iter!=objSymMap.end() && "obj sym not found");
1013 return iter->second;
1014}
1015
1016
1018{
1020 for (ValueToIDMapTy::iterator iter = valSymMap.begin(); iter != valSymMap.end();
1021 ++iter)
1022 {
1023 const NodeID i = iter->second;
1024 idmap[i] = iter->first;
1025 }
1026 for (ValueToIDMapTy::iterator iter = objSymMap.begin(); iter != objSymMap.end();
1027 ++iter)
1028 {
1029 const NodeID i = iter->second;
1030 idmap[i] = iter->first;
1031 }
1032 for (FunToIDMapTy::iterator iter = retSyms().begin(); iter != retSyms().end();
1033 ++iter)
1034 {
1035 const NodeID i = iter->second;
1036 idmap[i] = iter->first;
1037 }
1038 for (FunToIDMapTy::iterator iter = varargSyms().begin(); iter != varargSyms().end();
1039 ++iter)
1040 {
1041 const NodeID i = iter->second;
1042 idmap[i] = iter->first;
1043 }
1044 SVFUtil::outs() << "{SymbolTableInfo \n";
1045
1046
1047
1048
1049 for (auto iter : idmap)
1050 {
1051 std::string str;
1052 llvm::raw_string_ostream rawstr(str);
1053 auto llvmVal = iter.second;
1054 if (llvmVal)
1055 rawstr << " " << *llvmVal << " ";
1056 else
1057 rawstr << " No llvmVal found";
1059 SVFUtil::outs() << iter.first << " " << rawstr.str() << "\n";
1060 }
1061 SVFUtil::outs() << "}\n";
1062}
1063
1066{
1068 svfBaseNode->setSourceLoc(LLVMUtil::getSourceLoc(val));
1069 svfBaseNode->setName(val->getName().str());
1070}
1071
1073{
1074 Function* fun = nullptr;
1075
1076 for (u32_t i = 0; i < llvmModuleSet->getModuleNum(); ++i)
1077 {
1079 fun = mod->getFunction(name);
1080 if (fun)
1081 {
1082 return llvmModuleSet->getFunObjVar(fun);
1083 }
1084 }
1085 return nullptr;
1086}
1087
1088
1090{
1091 for(LLVMType2SVFTypeMap::const_iterator it = LLVMType2SVFType.begin(), eit = LLVMType2SVFType.end(); it!=eit; ++it)
1092 {
1093 if (it->second == T)
1094 return it->first;
1095 }
1096 assert(false && "can't find the corresponding LLVM Type");
1097 abort();
1098}
1099
1104{
1105 assert(T && "SVFType should not be null");
1106 LLVMType2SVFTypeMap::const_iterator it = LLVMType2SVFType.find(T);
1107 if (it != LLVMType2SVFType.end())
1108 return it->second;
1109
1112 svfType->setTypeInfo(stinfo);
1113 return svfType;
1114}
1115
1116
1119{
1120 ICFGNode* node;
1122 node = getCallICFGNode(inst);
1123 else if(LLVMUtil::isIntrinsicInst(inst))
1124 node = getIntraICFGNode(inst);
1125 else
1126 node = getIntraICFGNode(inst);
1127
1128 assert (node!=nullptr && "no ICFGNode for this instruction?");
1129 return node;
1130}
1131
1133{
1134 ICFGNode* node;
1136 node = getCallBlock(inst);
1137 else if(LLVMUtil::isIntrinsicInst(inst))
1138 node = getIntraBlock(inst);
1139 else
1140 node = getIntraBlock(inst);
1141
1142 return node != nullptr;
1143}
1144
1146{
1147 assert(LLVMUtil::isCallSite(inst) && "not a call instruction?");
1148 assert(LLVMUtil::isNonInstricCallSite(inst) && "associating an intrinsic debug instruction with an ICFGNode!");
1149 CallICFGNode* node = getCallBlock(inst);
1150 assert (node!=nullptr && "no CallICFGNode for this instruction?");
1151 return node;
1152}
1153
1155{
1156 assert(LLVMUtil::isCallSite(inst) && "not a call instruction?");
1157 assert(LLVMUtil::isNonInstricCallSite(inst) && "associating an intrinsic debug instruction with an ICFGNode!");
1158 RetICFGNode* node = getRetBlock(inst);
1159 assert (node!=nullptr && "no RetICFGNode for this instruction?");
1160 return node;
1161}
1162
1164{
1165 IntraICFGNode* node = getIntraBlock(inst);
1166 assert (node!=nullptr && "no IntraICFGNode for this instruction?");
1167 return node;
1168}
1169
1171{
1172 Type2TypeInfoMap::iterator tit = Type2TypeInfo.find(T);
1173 if (tit != Type2TypeInfo.end())
1174 {
1175 return tit->second;
1176 }
1177 // No such StInfo for T, create it now.
1178 StInfo* stInfo;
1179 if (const ArrayType* aty = SVFUtil::dyn_cast<ArrayType>(T))
1180 {
1182 }
1183 else if (const StructType* sty = SVFUtil::dyn_cast<StructType>(T))
1184 {
1185 u32_t nf;
1187 if (nf > svfir->maxStSize)
1188 {
1190 svfir->maxStSize = nf;
1191 }
1192 }
1193 else
1194 {
1196 }
1197 Type2TypeInfo.emplace(T, stInfo);
1199 return stInfo;
1200}
1201
1203{
1204 assert(LLVMType2SVFType.find(T) == LLVMType2SVFType.end() &&
1205 "SVFType has been added before");
1206
1207 // add SVFType's LLVM byte size iff T isSized(), otherwise byteSize is 1 (default value)
1208 u32_t byteSize = 1;
1209 if (T->isSized())
1210 {
1211 const llvm::DataLayout &DL = LLVMModuleSet::getLLVMModuleSet()->
1212 getMainLLVMModule()->getDataLayout();
1213 Type *mut_T = const_cast<Type *>(T);
1214 byteSize = DL.getTypeAllocSize(mut_T);
1215 }
1216
1218 if (SVFUtil::isa<PointerType>(T))
1219 {
1220 svftype = new SVFPointerType(byteSize);
1221 }
1222 else if (const IntegerType* intT = SVFUtil::dyn_cast<IntegerType>(T))
1223 {
1224 auto svfIntT = new SVFIntegerType(byteSize);
1225 unsigned signWidth = intT->getBitWidth();
1226 assert(signWidth < INT16_MAX && "Integer width too big");
1227 svfIntT->setSignAndWidth(intT->getSignBit() ? -signWidth : signWidth);
1228 svftype = svfIntT;
1229 }
1230 else if (const FunctionType* ft = SVFUtil::dyn_cast<FunctionType>(T))
1231 {
1232 std::vector<const SVFType*> paramTypes;
1233 for (const auto& t: ft->params())
1234 {
1235 paramTypes.push_back(getSVFType(t));
1236 }
1237 svftype = new SVFFunctionType(getSVFType(ft->getReturnType()), paramTypes, ft->isVarArg());
1238 }
1239 else if (const StructType* st = SVFUtil::dyn_cast<StructType>(T))
1240 {
1241 auto svfst = new SVFStructType(byteSize);
1242 if (st->hasName())
1243 svfst->setName(st->getName().str());
1244 svftype = svfst;
1245 }
1246 else if (const auto at = SVFUtil::dyn_cast<ArrayType>(T))
1247 {
1248 auto svfat = new SVFArrayType(byteSize);
1249 svfat->setNumOfElement(at->getNumElements());
1250 svfat->setTypeOfElement(getSVFType(at->getElementType()));
1251 svftype = svfat;
1252 }
1253 else
1254 {
1255 std::string buffer;
1256 auto ot = new SVFOtherType(T->isSingleValueType(), byteSize);
1257 llvm::raw_string_ostream(buffer) << *T;
1258 ot->setRepr(std::move(buffer));
1259 svftype = ot;
1260 }
1261
1264
1265 return svftype;
1266}
1267
1272{
1273 u64_t totalElemNum = ty->getNumElements();
1274 const Type* elemTy = ty->getElementType();
1275 while (const ArrayType* aty = SVFUtil::dyn_cast<ArrayType>(elemTy))
1276 {
1277 totalElemNum *= aty->getNumElements();
1278 elemTy = aty->getElementType();
1279 }
1280
1283
1285 if (totalElemNum == 0)
1286 {
1287 stInfo->addFldWithType(0, elemSvfType, 0);
1288 stInfo->setNumOfFieldsAndElems(1, 1);
1289 stInfo->getFlattenFieldTypes().push_back(elemSvfType);
1290 stInfo->getFlattenElementTypes().push_back(elemSvfType);
1291 return stInfo;
1292 }
1293
1297 u32_t nfF = elemStInfo->getNumOfFlattenFields();
1298 u32_t nfE = elemStInfo->getNumOfFlattenElements();
1299 for (u32_t j = 0; j < nfF; j++)
1300 {
1301 const SVFType* fieldTy = elemStInfo->getFlattenFieldTypes()[j];
1302 stInfo->getFlattenFieldTypes().push_back(fieldTy);
1303 }
1304
1307 u32_t outArrayElemNum = ty->getNumElements();
1308 for (u32_t i = 0; i < outArrayElemNum; ++i)
1309 {
1310 auto idx = (i * nfE * totalElemNum) / outArrayElemNum;
1311 stInfo->addFldWithType(0, elemSvfType, idx);
1312 }
1313
1314 for (u32_t i = 0; i < totalElemNum; ++i)
1315 {
1316 for (u32_t j = 0; j < nfE; ++j)
1317 {
1318 const SVFType* et = elemStInfo->getFlattenElementTypes()[j];
1319 stInfo->getFlattenElementTypes().push_back(et);
1320 }
1321 }
1322
1323 assert(stInfo->getFlattenElementTypes().size() == nfE * totalElemNum &&
1324 "typeForArray size incorrect!!!");
1325 stInfo->setNumOfFieldsAndElems(nfF, nfE * totalElemNum);
1326
1327 return stInfo;
1328}
1329
1336{
1338 StInfo* stInfo = new StInfo(1);
1339
1340 // Number of fields after flattening the struct
1341 numFields = 0;
1342 // The offset when considering array stride info
1343 u32_t strideOffset = 0;
1344 for (const Type* elemTy : structTy->elements())
1345 {
1347 // offset with int_32 (s32_t) is large enough and won't overflow
1348 stInfo->addFldWithType(numFields, elemSvfTy, strideOffset);
1349
1350 if (SVFUtil::isa<StructType, ArrayType>(elemTy))
1351 {
1353 u32_t nfF = subStInfo->getNumOfFlattenFields();
1354 u32_t nfE = subStInfo->getNumOfFlattenElements();
1355 // Copy ST's info, whose element 0 is the size of ST itself.
1356 for (u32_t j = 0; j < nfF; ++j)
1357 {
1358 const SVFType* elemTy = subStInfo->getFlattenFieldTypes()[j];
1359 stInfo->getFlattenFieldTypes().push_back(elemTy);
1360 }
1361 numFields += nfF;
1362 strideOffset += nfE;
1363 for (u32_t tpj = 0; tpj < nfE; ++tpj)
1364 {
1365 const SVFType* ty = subStInfo->getFlattenElementTypes()[tpj];
1366 stInfo->getFlattenElementTypes().push_back(ty);
1367 }
1368
1369 }
1370 else
1371 {
1372 // Simple type
1373 numFields += 1;
1374 strideOffset += 1;
1375 stInfo->getFlattenFieldTypes().push_back(elemSvfTy);
1376 stInfo->getFlattenElementTypes().push_back(elemSvfTy);
1377 }
1378 }
1379
1380 assert(stInfo->getFlattenElementTypes().size() == strideOffset &&
1381 "typeForStruct size incorrect!");
1382 stInfo->setNumOfFieldsAndElems(numFields,strideOffset);
1383
1384 return stInfo;
1385}
1386
1387
1392{
1394 StInfo* stInfo = new StInfo(1);
1396 stInfo->addFldWithType(0, svfType, 0);
1397
1398 stInfo->getFlattenFieldTypes().push_back(svfType);
1399 stInfo->getFlattenElementTypes().push_back(svfType);
1400 stInfo->setNumOfFieldsAndElems(1,1);
1401
1402 return stInfo;
1403}
1404
1405void LLVMModuleSet::setExtFuncAnnotations(const Function* fun, const std::vector<std::string>& funcAnnotations)
1406{
1407 assert(fun && "Null SVFFunction* pointer");
1409}
1410
1412{
1413 assert(fun && "Null SVFFunction* pointer");
1414 auto it = func2Annotations.find(fun);
1415 if (it != func2Annotations.end())
1416 {
1417 for (const std::string& annotation : it->second)
1418 if (annotation.find(funcAnnotation) != std::string::npos)
1419 return true;
1420 }
1421 return false;
1422}
1423
1424std::string LLVMModuleSet::getExtFuncAnnotation(const Function* fun, const std::string& funcAnnotation)
1425{
1426 assert(fun && "Null Function* pointer");
1427 auto it = func2Annotations.find(fun);
1428 if (it != func2Annotations.end())
1429 {
1430 for (const std::string& annotation : it->second)
1431 if (annotation.find(funcAnnotation) != std::string::npos)
1432 return annotation;
1433 }
1434 return "";
1435}
1436
1437const std::vector<std::string>& LLVMModuleSet::getExtFuncAnnotations(const Function* fun)
1438{
1439 assert(fun && "Null Function* pointer");
1440 auto it = func2Annotations.find(fun);
1441 if (it != func2Annotations.end())
1442 return it->second;
1443 return func2Annotations[fun];
1444}
1445
1447{
1448 return F &&
1449 (hasExtFuncAnnotation(F, "MEMCPY") || hasExtFuncAnnotation(F, "STRCPY")
1450 || hasExtFuncAnnotation(F, "STRCAT"));
1451}
1452
1454{
1455 return F && hasExtFuncAnnotation(F, "MEMSET");
1456}
1457
1459{
1460 return F && hasExtFuncAnnotation(F, "ALLOC_HEAP_RET");
1461}
1462
1463// Does (F) allocate a new object and assign it to one of its arguments?
1465{
1466 return F && hasExtFuncAnnotation(F, "ALLOC_HEAP_ARG");
1467}
1468
1470{
1471 return F && hasExtFuncAnnotation(F, "ALLOC_STACK_RET");
1472}
1473
1474// Get the position of argument which holds the new object
1476{
1477 std::string allocArg = getExtFuncAnnotation(F, "ALLOC_HEAP_ARG");
1478 assert(!allocArg.empty() && "Not an alloc call via argument or incorrect extern function annotation!");
1479
1480 std::string number;
1481 for (char c : allocArg)
1482 {
1483 if (isdigit(c))
1484 number.push_back(c);
1485 }
1486 assert(!number.empty() && "Incorrect naming convention for svf external functions(ALLOC_HEAP_ARG + number)?");
1487 return std::stoi(number);
1488}
1489
1490// Does (F) reallocate a new object?
1492{
1493 return F && hasExtFuncAnnotation(F, "REALLOC_HEAP_RET");
1494}
1495
1496
1497// Should (F) be considered "external" (either not defined in the program
1498// or a user-defined version of a known alloc or no-op)?
1500{
1501 assert(F && "Null SVFFunction* pointer");
1502 if (F->isDeclaration() || F->isIntrinsic())
1503 return true;
1504 else if (hasExtFuncAnnotation(F, "OVERWRITE") && getExtFuncAnnotations(F).size() == 1)
1505 return false;
1506 else
1507 return !getExtFuncAnnotations(F).empty();
1508}
#define SVF_MAIN_FUNC_NAME
#define SVF_GLOBAL_DTORS
#define SVF_GLOBAL_CTORS
#define DBOUT(TYPE, X)
LLVM debug macros, define type of your DBUG model of each pass.
Definition SVFType.h:498
#define TIMEINTERVAL
Definition SVFType.h:526
#define DGENERAL
Definition SVFType.h:504
const char *const name
Definition cJSON.h:264
char * buffer
Definition cJSON.h:163
const char *const const double number
Definition cJSON.h:268
int index
Definition cJSON.h:170
std::string getExtBcPath()
Definition ExtAPI.cpp:120
static ExtAPI * getExtAPI()
Definition ExtAPI.cpp:43
const SVFType * maxStruct
The struct type with the most fields.
Definition IRGraph.h:351
void addStInfo(StInfo *stInfo)
Definition IRGraph.h:363
NodeID blkPtrSymID() const
Definition IRGraph.h:178
NodeID nullPtrSymID() const
Definition IRGraph.h:183
void addTypeInfo(const SVFType *ty)
Definition IRGraph.h:356
u32_t maxStSize
The number of fields in max_struct.
Definition IRGraph.h:354
bool hasValueNode(const Value *V)
NodeID getValueNode(const Value *V)
IntraICFGNode * getIntraBlock(const Instruction *inst)
Definition LLVMModule.h:493
LLVMType2SVFTypeMap LLVMType2SVFType
Definition LLVMModule.h:99
bool is_alloc_stack_ret(const Function *F)
std::vector< const Function * > getLLVMGlobalFunctions(const GlobalVariable *global)
FunToIDMapTy & retSyms()
Definition LLVMModule.h:267
void buildSymbolTable() const
Module * getMainLLVMModule() const
Definition LLVMModule.h:359
const FunObjVar * getFunObjVar(const Function *fun) const
Definition LLVMModule.h:260
ValueToIDMapTy valSymMap
map a value to its sym id
Definition LLVMModule.h:112
StInfo * collectTypeInfo(const Type *ty)
Collect a type info.
static LLVMModuleSet * getLLVMModuleSet()
Definition LLVMModule.h:129
void loadModules(const std::vector< std::string > &moduleNameVec)
ObjTypeInference * typeInference
Definition LLVMModule.h:101
static void releaseLLVMModuleSet()
Definition LLVMModule.h:136
static void preProcessBCs(std::vector< std::string > &moduleNameVec)
static bool preProcessed
Definition LLVMModule.h:80
DominatorTree & getDomTree(const Function *fun)
void addToSVFVar2LLVMValueMap(const Value *val, SVFValue *svfBaseNode)
RetICFGNode * getRetBlock(const Instruction *cs)
Get/Add a return node.
Definition LLVMModule.h:485
std::unique_ptr< LLVMContext > owned_ctx
Definition LLVMModule.h:82
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)
Definition LLVMModule.h:438
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.
Definition LLVMModule.h:476
ValueToIDMapTy objSymMap
map a obj reference to its sym id
Definition LLVMModule.h:113
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.
Definition LLVMModule.h:299
CallICFGNode * getCallICFGNode(const Instruction *cs)
get a call node
Fun2AnnoMap ExtFun2Annotations
Record annotations of function in extapi.bc.
Definition LLVMModule.h:89
bool is_arg_alloc(const Function *F)
Map< const Function *, std::vector< std::string > > func2Annotations
Definition LLVMModule.h:92
const BasicBlock * getFunExitBB(const Function *fun) const
Definition LLVMModule.h:176
NodeID getObjectNode(const Value *V)
Module * getModule(u32_t idx) const
Definition LLVMModule.h:160
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)
Definition LLVMModule.h:433
const std::vector< std::reference_wrapper< Module > > & getLLVMModules() const
Definition LLVMModule.h:155
LLVMContext & getContext() const
Definition LLVMModule.h:374
bool is_ext(const Function *F)
Map< const Function *, DominatorTree > FunToDominatorTree
Definition LLVMModule.h:110
static void buildSVFModule(Module &mod)
SVFBaseNode2LLVMValueMap SVFBaseNode2LLVMValue
Definition LLVMModule.h:103
StInfo * collectArrayInfo(const ArrayType *T)
Collect the array info.
LLVMModuleSet()
Constructor.
void buildGlobalDefToRepMap()
SVFType * addSVFTypeInfo(const Type *t)
Create SVFTypes.
FunctionSet funSet
Definition LLVMModule.h:117
FunctionSetType ExtFuncsVec
Record some "sse_" function declarations used in other ext function definition, e....
Definition LLVMModule.h:87
bool hasExtFuncAnnotation(const Function *fun, const std::string &funcAnnotation)
static LLVMModuleSet * llvmModuleSet
Definition LLVMModule.h:79
IntraICFGNode * getIntraICFGNode(const Instruction *inst)
get a intra node
std::vector< std::reference_wrapper< Module > > modules
Definition LLVMModule.h:84
std::string getExtFuncAnnotation(const Function *fun, const std::string &funcAnnotation)
bool empty() const
Definition LLVMModule.h:380
Map< std::string, std::vector< std::string > > Fun2AnnoMap
Definition LLVMModule.h:60
u32_t getModuleNum() const
Definition LLVMModule.h:150
GlobalDefToRepMapTy GlobalDefToRepMap
Global definition to a rep definition map.
Definition LLVMModule.h:95
std::vector< std::unique_ptr< Module > > owned_modules
Definition LLVMModule.h:83
bool is_memset(const Function *F)
FunToIDMapTy & varargSyms()
Definition LLVMModule.h:272
Type2TypeInfoMap Type2TypeInfo
Definition LLVMModule.h:100
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
Definition Options.h:183
static const Option< std::string > Graphtxt
Definition Options.h:182
static bool pagReadFromTXT()
Definition SVFIR.h:211
void setModuleIdentifier(const std::string &moduleIdentifier)
Definition SVFIR.h:221
static SVFIR * getPAG(bool buildFromFile=false)
Singleton design here to make sure we only have one instance during any analysis.
Definition SVFIR.h:116
static void setPagFromTXT(const std::string &txt)
Definition SVFIR.h:216
static double timeOfBuildingLLVMModule
Definition SVFStat.h:93
static double getClk(bool mark=false)
Definition SVFStat.cpp:48
static double timeOfBuildingSymbolTable
Definition SVFStat.h:94
static SVFType * svfPtrTy
ptr type
Definition SVFType.h:192
static SVFType * svfI8Ty
8-bit int type
Definition SVFType.h:193
#define NULL
Definition extapi.c:2
int isdigit(int c)
Definition extapi.c:851
bool isIntrinsicInst(const Instruction *inst)
Return true if it is an intrinsic instruction.
Definition LLVMUtil.cpp:202
std::vector< const Function * > getCalledFunctions(const Function *F)
Get all called funcions in a parent function.
Definition LLVMUtil.cpp:363
bool isCallSite(const Instruction *inst)
Whether an instruction is a call or invoke instruction.
Definition LLVMUtil.h:44
bool functionDoesNotRet(const Function *fun)
Definition LLVMUtil.cpp:122
std::pair< s64_t, u64_t > getIntegerValue(const ConstantInt *intValue)
Definition LLVMUtil.h:82
const std::string getSourceLoc(const Value *val)
Definition LLVMUtil.cpp:452
const Value * getGlobalRep(const Value *val)
find the unique defined global across multiple modules
Definition LLVMUtil.cpp:439
bool basicBlockHasRetInst(const BasicBlock *bb)
Return true if the function has a return instruction.
Definition LLVMUtil.cpp:108
bool isIRFile(const std::string &filename)
Check whether a file is an LLVM IR file.
Definition LLVMUtil.cpp:314
std::string restoreFuncName(std::string funcName)
Definition LLVMUtil.cpp:406
bool isNonInstricCallSite(const Instruction *inst)
Whether an instruction is a callsite in the application code, excluding llvm intrinsic calls.
Definition LLVMUtil.cpp:720
std::string pasMsg(const std::string &msg)
Print each pass/phase message by converting a string into blue string output.
Definition SVFUtil.cpp:101
std::ostream & errs()
Overwrite llvm::errs()
Definition SVFUtil.h:58
std::ostream & outs()
Overwrite llvm::outs()
Definition SVFUtil.h:52
for isBitcode
Definition BasicTypes.h:68
llvm::GlobalVariable GlobalVariable
Definition BasicTypes.h:130
llvm::ArrayType ArrayType
Definition BasicTypes.h:95
llvm::raw_fd_ostream raw_fd_ostream
LLVM outputs.
Definition BasicTypes.h:264
llvm::Type Type
Definition BasicTypes.h:83
llvm::BasicBlock BasicBlock
Definition BasicTypes.h:86
llvm::inst_iterator inst_iterator
Definition BasicTypes.h:249
llvm::ConstantStruct ConstantStruct
Definition BasicTypes.h:106
llvm::StructType StructType
LLVM types.
Definition BasicTypes.h:94
unsigned long long u64_t
Definition GeneralType.h:49
llvm::IntegerType IntegerType
Definition BasicTypes.h:97
u32_t NodeID
Definition GeneralType.h:56
llvm::ConstantArray ConstantArray
Definition BasicTypes.h:123
llvm::Function Function
Definition BasicTypes.h:85
llvm::FunctionType FunctionType
Definition BasicTypes.h:98
llvm::Instruction Instruction
Definition BasicTypes.h:87
llvm::SMDiagnostic SMDiagnostic
Definition BasicTypes.h:90
llvm::ConstantDataSequential ConstantDataSequential
Definition BasicTypes.h:119
llvm::Value Value
LLVM Basic classes.
Definition BasicTypes.h:82
llvm::ConstantExpr ConstantExpr
Definition BasicTypes.h:120
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
signed s32_t
Definition GeneralType.h:48
llvm::Module Module
Definition BasicTypes.h:84
unsigned u32_t
Definition GeneralType.h:47
llvm::CallInst CallInst
Definition BasicTypes.h:147
llvm::ConstantInt ConstantInt
Definition BasicTypes.h:125
llvm::DominatorTree DominatorTree
LLVM Dominators.
Definition BasicTypes.h:133
IntervalValue operator>(const IntervalValue &lhs, const IntervalValue &rhs)