Static Value-Flow Analysis
Loading...
Searching...
No Matches
SVFUtil.cpp
Go to the documentation of this file.
1//===- AnalysisUtil.cpp -- Helper functions for pointer analysis--------------//
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 * AnalysisUtil.cpp
25 *
26 * Created on: Apr 11, 2013
27 * Author: Yulei Sui
28 */
29
30#include "Util/Options.h"
31#include "Util/SVFUtil.h"
33#include "Graphs/CallGraph.h"
34#include "SVFIR/SVFVariables.h"
35
36#include <sys/resource.h>
37
38using namespace SVF;
39
41#define KNRM "\x1B[1;0m"
42#define KRED "\x1B[1;31m"
43#define KGRN "\x1B[1;32m"
44#define KYEL "\x1B[1;33m"
45#define KBLU "\x1B[1;34m"
46#define KPUR "\x1B[1;35m"
47#define KCYA "\x1B[1;36m"
48#define KWHT "\x1B[1;37m"
49
50
51
55std::string SVFUtil::sucMsg(const std::string& msg)
56{
57 return KGRN + msg + KNRM;
58}
59
63std::string SVFUtil::wrnMsg(const std::string& msg)
64{
65 return KYEL + msg + KNRM;
66}
67
68void SVFUtil::writeWrnMsg(const std::string& msg)
69{
71 return;
72 outs() << wrnMsg(msg) << "\n";
73}
74
78std::string SVFUtil::errMsg(const std::string& msg)
79{
80 return KRED + msg + KNRM;
81}
82
83std::string SVFUtil::bugMsg1(const std::string& msg)
84{
85 return KYEL + msg + KNRM;
86}
87
88std::string SVFUtil::bugMsg2(const std::string& msg)
89{
90 return KPUR + msg + KNRM;
91}
92
93std::string SVFUtil::bugMsg3(const std::string& msg)
94{
95 return KCYA + msg + KNRM;
96}
97
101std::string SVFUtil::pasMsg(const std::string& msg)
102{
103 return KBLU + msg + KNRM;
104}
105
110{
111 outs() << "node " << node << " points-to: {";
112 dumpSet(bs);
113 outs() << "}\n";
114}
115
116// For use from the debugger.
118{
119 outs() << "{";
120 dumpSet(bs);
121 outs() << "}\n";
122}
123
125{
126 outs() << "{";
127 for (PointsToList::const_iterator ii = ptl.begin(), ie = ptl.end();
128 ii != ie; ii++)
129 {
130 auto bs = *ii;
131 dumpSet(bs);
132 }
133 outs() << "}\n";
134}
135
139void SVFUtil::dumpAliasSet(unsigned node, NodeBS bs)
140{
141 outs() << "node " << node << " alias set: {";
142 dumpSet(bs);
143 outs() << "}\n";
144}
145
150{
151 for (NodeBS::iterator ii = bs.begin(), ie = bs.end();
152 ii != ie; ii++)
153 {
154 O << " " << *ii << " ";
155 }
156}
157
159{
160 for (NodeID n : pt)
161 {
162 o << " " << n << " ";
163 }
164}
165
169void SVFUtil::reportMemoryUsageKB(const std::string& infor, OutStream & O)
170{
173 O << infor << "\tVmRSS: " << vmrss << "\tVmSize: " << vmsize << "\n";
174}
175
180{
181 /* Get the current process' status file from the proc filesystem */
182 char buffer[8192];
183 FILE* procfile = fopen("/proc/self/status", "r");
184 if(procfile)
185 {
186 u32_t result = fread(buffer, sizeof(char), 8192, procfile);
187 if (result == 0)
188 {
189 fputs ("Reading error\n",stderr);
190 }
191 }
192 else
193 {
194 SVFUtil::writeWrnMsg(" /proc/self/status file not exit!");
195 return false;
196 }
198
199 /* Look through proc status contents line by line */
200 char delims[] = "\n";
201 char* line = strtok(buffer, delims);
202
203 bool found_vmrss = false;
204 bool found_vmsize = false;
205
206 while (line != nullptr && (found_vmrss == false || found_vmsize == false))
207 {
208 if (strstr(line, "VmRSS:") != nullptr)
209 {
210 sscanf(line, "%*s %u", vmrss_kb);
211 found_vmrss = true;
212 }
213
214 if (strstr(line, "VmSize:") != nullptr)
215 {
216 sscanf(line, "%*s %u", vmsize_kb);
217 found_vmsize = true;
218 }
219
220 line = strtok(nullptr, delims);
221 }
222
223 return (found_vmrss && found_vmsize);
224}
225
230{
231 const rlim_t kStackSize = 256L * 1024L * 1024L; // min stack size = 256 Mb
232 struct rlimit rl;
234 if (result == 0)
235 {
236 if (rl.rlim_cur < kStackSize)
237 {
238 rl.rlim_cur = kStackSize;
240 if (result != 0)
241 writeWrnMsg("setrlimit returned result !=0 \n");
242 }
243 }
244}
245
246
247
249{
250 switch (method)
251 {
253 return "single";
255 return "complete";
257 return "average";
259 return "median";
261 return "svf-best";
262 default:
263 assert(false && "SVFUtil::hclustMethodToString: unknown method");
264 abort();
265 }
266}
267
269{
270 SVFUtil::outs().flush();
271 // TODO: output does not indicate which time limit is reached.
272 // This can be better in the future.
273 SVFUtil::outs() << "WPA: time limit reached\n";
274 exit(101);
275}
276
278{
279 if (timeLimit == 0) return false;
280
281 // If an alarm is already set, don't set another. That means this analysis
282 // is part of another which has a time limit.
283 unsigned remainingSeconds = alarm(0);
284 if (remainingSeconds != 0)
285 {
286 // Continue the previous alarm and move on.
288 return false;
289 }
290
293 return true;
294}
295
302
309{
310 if (callee->isVarArg() || ThreadAPI::getThreadAPI()->isTDFork(call))
311 return call->arg_size() >= callee->arg_size();
312 else
313 return call->arg_size() == callee->arg_size();
314}
315
317{
318 return SVFUtil::isa<CallICFGNode>(inst);
319}
320
322{
323 if (const CallICFGNode* call = SVFUtil::dyn_cast<CallICFGNode>(inst))
324 {
325 const FunObjVar* func = call->getCalledFunction();
326 if (func && func->isIntrinsic())
327 {
328 return true;
329 }
330 }
331 return false;
332}
333
335{
336 return isExtCall(cs->getCalledFunction());
337}
338
343
344
346{
347 return getHeapAllocHoldingArgPosition(cs->getCalledFunction());
348}
349
350
352{
353 if(!isCallSite(node)) return false;
354 return isExtCall(cast<CallICFGNode>(node)->getCalledFunction());
355}
356
358{
359 if(!isCallSite(cs)) return false;
360 return isHeapAllocExtCallViaRet(cast<CallICFGNode>(cs)) || isHeapAllocExtCallViaArg(cast<CallICFGNode>(cs));
361}
362
368
370{
371 bool isPtrTy = cs->getType()->isPointerTy();
373}
374
375
377{
378 if (const auto& intraNode = dyn_cast<IntraICFGNode>(node))
379 return intraNode->isRetInst();
380 else
381 return false;
382}
383
385{
386 return fun && (fun->getName() == "exit" ||
387 fun->getName() == "__assert_rtn" ||
388 fun->getName() == "__assert_fail");
389}
390
395
398{
400 for (const auto& item: *svfirCallGraph)
401 {
402 const CallGraphNode*fun = item.second;
403 if (fun->getName()==funName)
404 return fun->getFunction();
405 }
406 return nullptr;
407}
408
411{
413 for (const auto& item: *svfirCallGraph)
414 {
415 const CallGraphNode*fun = item.second;
416 if (isProgEntryFunction(fun->getFunction()))
417 return (fun->getFunction());
418 }
419 return nullptr;
420}
421
423{
424 const ValVar* pVar = PAG::getPAG()->getBaseValVar(svfvar->getId());
425 if(const ArgValVar* arg = SVFUtil::dyn_cast<ArgValVar>(pVar))
426 return arg->isArgOfUncalledFunction();
427 else
428 return false;
429}
430
432{
433 assert(valVar->getInEdges().size() == 1);
434 return SVFUtil::dyn_cast<ObjVar>((*valVar->getInEdges().begin())->getSrcNode());
435}
436
438{
439 return fun && ExtAPI::getExtAPI()->is_ext(fun);
440}
441
443{
444 return funObjVar && funObjVar->getName() == "main";
445}
#define KNRM
Color for output format.
Definition SVFUtil.cpp:41
#define KPUR
Definition SVFUtil.cpp:46
#define KBLU
Definition SVFUtil.cpp:45
#define KRED
Definition SVFUtil.cpp:42
#define KYEL
Definition SVFUtil.cpp:44
#define KGRN
Definition SVFUtil.cpp:43
#define KCYA
Definition SVFUtil.cpp:47
cJSON * n
Definition cJSON.cpp:2558
char * buffer
Definition cJSON.h:163
cJSON * item
Definition cJSON.h:222
Class representing a function argument variable in the SVFIR.
const FunObjVar * getFunction() const
Get function of this call node.
Definition CallGraph.h:191
const std::string & getName() const
Definition CallGraph.cpp:45
const FunObjVar * getCalledFunction() const
Definition ICFGNode.h:512
u32_t arg_size() const
Definition ICFGNode.h:499
static ExtAPI * getExtAPI()
Definition ExtAPI.cpp:43
bool is_ext(const FunObjVar *funObjVar)
Definition ExtAPI.cpp:257
static const Option< bool > DisableWarn
Definition Options.h:205
CallGraph * getCallGraph()
Definition SVFIR.h:184
static SVFIR * getPAG(bool buildFromFile=false)
Singleton design here to make sure we only have one instance during any analysis.
Definition SVFIR.h:116
const ValVar * getBaseValVar(NodeID id) const
Definition SVFIR.h:433
bool isPointerTy() const
Definition SVFType.h:249
virtual const SVFType * getType() const
Definition SVFValue.h:169
virtual const std::string & getName() const
Definition SVFValue.h:184
bool isTDFork(const CallICFGNode *inst) const
Return true if this call create a new thread.
static ThreadAPI * getThreadAPI()
Return a static reference.
Definition ThreadAPI.h:98
char * strtok(char *str, const char *delim)
Definition extapi.c:463
char * strstr(const char *haystack, const char *needle)
Definition extapi.c:821
hclust_fast_methods
Definition fastcluster.h:66
@ HCLUST_METHOD_AVERAGE
Definition fastcluster.h:72
@ HCLUST_METHOD_COMPLETE
Definition fastcluster.h:70
@ HCLUST_METHOD_SVF_BEST
Definition fastcluster.h:76
@ HCLUST_METHOD_MEDIAN
Definition fastcluster.h:74
@ HCLUST_METHOD_SINGLE
Definition fastcluster.h:68
bool isReallocExtCall(const CallICFGNode *cs)
Definition SVFUtil.cpp:369
std::string sucMsg(const std::string &msg)
Returns successful message by converting a string into green string output.
Definition SVFUtil.cpp:55
void increaseStackSize()
Increase the stack size limit.
Definition SVFUtil.cpp:229
const ObjVar * getObjVarOfValVar(const ValVar *valVar)
Definition SVFUtil.cpp:431
std::string bugMsg1(const std::string &msg)
Definition SVFUtil.cpp:83
bool isProgEntryFunction(const FunObjVar *)
Program entry function e.g. main.
Definition SVFUtil.cpp:442
std::string hclustMethodToString(hclust_fast_methods method)
Returns a string representation of a hclust method.
Definition SVFUtil.cpp:248
void stopAnalysisLimitTimer(bool limitTimerSet)
Definition SVFUtil.cpp:298
bool isIntrinsicInst(const ICFGNode *inst)
Return true if it is an llvm intrinsic instruction.
Definition SVFUtil.cpp:321
u32_t getHeapAllocHoldingArgPosition(const FunObjVar *fun)
Get the position of argument that holds an allocated heap object.
Definition SVFUtil.h:293
const FunObjVar * getProgEntryFunction()
Get program entry function.
Definition SVFUtil.cpp:410
bool isHeapAllocExtFunViaRet(const FunObjVar *fun)
Return true if the call is a heap allocator/reallocator.
Definition SVFUtil.h:279
std::string pasMsg(const std::string &msg)
Print each pass/phase message by converting a string into blue string output.
Definition SVFUtil.cpp:101
OrderedSet< PointsTo, equalPointsTo > PointsToList
Definition SVFUtil.h:168
void dumpAliasSet(unsigned node, NodeBS To)
Dump alias set.
Definition SVFUtil.cpp:139
bool isHeapAllocExtFunViaArg(const FunObjVar *fun)
Definition SVFUtil.h:285
bool getMemoryUsageKB(u32_t *vmrss_kb, u32_t *vmsize_kb)
Get memory usage from system file. Return TRUE if succeed.
Definition SVFUtil.cpp:179
void reportMemoryUsageKB(const std::string &infor, OutStream &O=SVFUtil::outs())
Print memory usage in KB.
Definition SVFUtil.cpp:169
bool startAnalysisLimitTimer(unsigned timeLimit)
Definition SVFUtil.cpp:277
std::string errMsg(const std::string &msg)
Print error message by converting a string into red string output.
Definition SVFUtil.cpp:78
std::string bugMsg3(const std::string &msg)
Definition SVFUtil.cpp:93
bool isHeapAllocExtCallViaArg(const CallICFGNode *cs)
Definition SVFUtil.cpp:339
bool isHeapAllocExtCall(const ICFGNode *cs)
Definition SVFUtil.cpp:357
bool isArgOfUncalledFunction(const SVFVar *svfvar)
Definition SVFUtil.cpp:422
bool isExtCall(const FunObjVar *fun)
Definition SVFUtil.cpp:437
bool matchArgs(const CallICFGNode *cs, const FunObjVar *callee)
Definition SVFUtil.cpp:308
void writeWrnMsg(const std::string &msg)
Writes a message run through wrnMsg.
Definition SVFUtil.cpp:68
void dumpSparseSet(const NodeBS &To)
Definition SVFUtil.cpp:117
void dumpPointsToSet(unsigned node, NodeBS To)
Dump points-to set.
Definition SVFUtil.cpp:109
std::string bugMsg2(const std::string &msg)
Definition SVFUtil.cpp:88
bool isProgExitCall(const CallICFGNode *cs)
Definition SVFUtil.cpp:391
bool isCallSite(const ICFGNode *inst)
Definition SVFUtil.cpp:316
std::string wrnMsg(const std::string &msg)
Returns warning message by converting a string into yellow string output.
Definition SVFUtil.cpp:63
bool isReallocExtFun(const FunObjVar *fun)
Return true if the call is a heap reallocator.
Definition SVFUtil.h:301
bool isRetInstNode(const ICFGNode *node)
Definition SVFUtil.cpp:376
std::ostream & outs()
Overwrite llvm::outs()
Definition SVFUtil.h:52
void timeLimitReached(int signum)
Function to call when alarm for time limit hits.
Definition SVFUtil.cpp:268
bool isHeapAllocExtCallViaRet(const CallICFGNode *cs)
interfaces to be used externally
Definition SVFUtil.cpp:363
void dumpSet(NodeBS To, OutStream &O=SVFUtil::outs())
Dump sparse bitvector set.
Definition SVFUtil.cpp:149
void dumpPointsToList(const PointsToList &ptl)
Definition SVFUtil.cpp:124
bool isProgExitFunction(const FunObjVar *fun)
Return true if this is a program exit function call.
Definition SVFUtil.cpp:384
const FunObjVar * getProgFunction(const std::string &funName)
Get program entry function from function name.
Definition SVFUtil.cpp:397
for isBitcode
Definition BasicTypes.h:68
u32_t NodeID
Definition GeneralType.h:56
std::ostream OutStream
Definition GeneralType.h:46
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
unsigned u32_t
Definition GeneralType.h:47