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

#include <MHP.h>

Public Types

typedef Set< const FunObjVar * > FunSet
 
typedef FIFOWorkList< CxtThreadStmtCxtThreadStmtWorkList
 
typedef Set< CxtThreadStmtCxtThreadStmtSet
 
typedef Map< CxtThreadStmt, NodeBSThreadStmtToThreadInterleav
 
typedef Map< const ICFGNode *, CxtThreadStmtSetInstToThreadStmtSetMap
 
typedef SVFLoopAndDomInfo::LoopBBs LoopBBs
 
typedef Set< CxtStmtLockSpan
 
typedef std::pair< const FunObjVar *, const FunObjVar * > FuncPair
 
typedef Map< FuncPair, boolFuncPairToBool
 

Public Member Functions

 MHP (TCT *t)
 Constructor.
 
virtual ~MHP ()
 Destructor.
 
void analyze ()
 Start analysis here.
 
void analyzeInterleaving ()
 Analyze thread interleaving.
 
ThreadCallGraphgetThreadCallGraph () const
 Get ThreadCallGraph.
 
TCTgetTCT () const
 Get Thread Creation Tree.
 
bool isConnectedfromMain (const FunObjVar *fun)
 Whether the function is connected from main function in thread call graph.
 
virtual bool mayHappenInParallel (const ICFGNode *i1, const ICFGNode *i2)
 Interface to query whether two instructions may happen-in-parallel.
 
virtual bool mayHappenInParallelCache (const ICFGNode *i1, const ICFGNode *i2)
 
virtual bool mayHappenInParallelInst (const ICFGNode *i1, const ICFGNode *i2)
 
virtual bool executedByTheSameThread (const ICFGNode *i1, const ICFGNode *i2)
 
const NodeBSgetInterleavingThreads (const CxtThreadStmt &cts)
 Get interleaving thread for statement inst.
 
bool hasInterleavingThreads (const CxtThreadStmt &cts) const
 
const CxtThreadStmtSetgetThreadStmtSet (const ICFGNode *inst) const
 Get/has ThreadStmt.
 
bool hasThreadStmtSet (const ICFGNode *inst) const
 
void printInterleaving ()
 Print interleaving results.
 

Public Attributes

u32_t numOfTotalQueries
 Total number of queries.
 
u32_t numOfMHPQueries
 Number of queries are answered as may-happen-in-parallel.
 
double interleavingTime
 
double interleavingQueriesTime
 

Private Member Functions

const CallGraph::FunctionSetgetCallee (const CallICFGNode *inst, CallGraph::FunctionSet &callees)
 
void updateNonCandidateFunInterleaving ()
 
void handleNonCandidateFun (const CxtThreadStmt &cts)
 Handle non-candidate function.
 
void handleFork (const CxtThreadStmt &cts, NodeID rootTid)
 Handle fork.
 
void handleJoin (const CxtThreadStmt &cts, NodeID rootTid)
 Handle join.
 
void handleCall (const CxtThreadStmt &cts, NodeID rootTid)
 Handle call.
 
void handleRet (const CxtThreadStmt &cts)
 Handle return.
 
void handleIntra (const CxtThreadStmt &cts)
 Handle intra.
 
void addInterleavingThread (const CxtThreadStmt &tgr, NodeID tid)
 Add/Remove interleaving thread for statement inst.
 
void addInterleavingThread (const CxtThreadStmt &tgr, const CxtThreadStmt &src)
 
void rmInterleavingThread (const CxtThreadStmt &tgr, const NodeBS &tids, const ICFGNode *joinsite)
 
void updateAncestorThreads (NodeID tid)
 Update Ancestor and sibling threads.
 
void updateSiblingThreads (NodeID tid)
 
bool isRecurFullJoin (NodeID parentTid, NodeID curTid)
 Thread curTid can be fully joined by parentTid recursively.
 
bool isMustJoin (const NodeID curTid, const ICFGNode *joinsite)
 Whether a join site must join a thread t.
 
bool isMultiForkedThread (NodeID curTid)
 A thread is a multiForked thread if it is in a loop or recursion.
 
void pushCxt (CallStrCxt &cxt, const CallICFGNode *call, const FunObjVar *callee)
 Push calling context.
 
bool matchCxt (CallStrCxt &cxt, const CallICFGNode *call, const FunObjVar *callee)
 Match context.
 
bool pushToCTSWorkList (const CxtThreadStmt &cs)
 WorkList helper functions.
 
CxtThreadStmt popFromCTSWorkList ()
 
bool isTDFork (const ICFGNode *call)
 Whether it is a fork site.
 
bool isTDJoin (const ICFGNode *call)
 Whether it is a join site.
 
NodeBS getDirAndIndJoinedTid (const CallStrCxt &cxt, const ICFGNode *call)
 Return thread id(s) which are directly or indirectly joined at this join site.
 
bool hasJoinInSymmetricLoop (const CallStrCxt &cxt, const ICFGNode *call) const
 Whether a context-sensitive join satisfies symmetric loop pattern.
 
const LoopBBsgetJoinInSymmetricLoop (const CallStrCxt &cxt, const ICFGNode *call) const
 Whether a context-sensitive join satisfies symmetric loop pattern.
 
bool isHBPair (NodeID tid1, NodeID tid2)
 Whether thread t1 happens before t2 based on ForkJoin Analysis.
 

Private Attributes

ThreadCallGraphtcg
 TCG.
 
TCTtct
 TCT.
 
ForkJoinAnalysisfja
 ForJoin Analysis.
 
CxtThreadStmtWorkList cxtStmtList
 CxtThreadStmt worklist.
 
ThreadStmtToThreadInterleav threadStmtToTheadInterLeav
 
InstToThreadStmtSetMap instToTSMap
 Map a statement to its thread interleavings.
 
FuncPairToBool nonCandidateFuncMHPRelMap
 

Detailed Description

This class serves as a base may-happen in parallel analysis for multithreaded program Given a statement under an abstract thread, it tells which abstract threads may be alive at the same time (May-happen-in-parallel).

Definition at line 45 of file MHP.h.

Member Typedef Documentation

◆ CxtThreadStmtSet

Definition at line 51 of file MHP.h.

◆ CxtThreadStmtWorkList

Definition at line 50 of file MHP.h.

◆ FuncPair

Definition at line 58 of file MHP.h.

◆ FuncPairToBool

Definition at line 59 of file MHP.h.

◆ FunSet

Definition at line 49 of file MHP.h.

◆ InstToThreadStmtSetMap

Definition at line 53 of file MHP.h.

◆ LockSpan

Definition at line 56 of file MHP.h.

◆ LoopBBs

Definition at line 54 of file MHP.h.

◆ ThreadStmtToThreadInterleav

Definition at line 52 of file MHP.h.

Constructor & Destructor Documentation

◆ MHP()

MHP::MHP ( TCT t)

Constructor.

Constructor

Definition at line 44 of file MHP.cpp.

46{
49}
void analyzeForkJoinPair()
Definition MHP.cpp:722
TCT * tct
TCT.
Definition MHP.h:253
u32_t numOfTotalQueries
Total number of queries.
Definition MHP.h:262
ThreadCallGraph * tcg
TCG.
Definition MHP.h:252
ForkJoinAnalysis * fja
ForJoin Analysis.
Definition MHP.h:254
double interleavingQueriesTime
Definition MHP.h:265
u32_t numOfMHPQueries
Number of queries are answered as may-happen-in-parallel.
Definition MHP.h:263
double interleavingTime
Definition MHP.h:264
ThreadCallGraph * getThreadCallGraph() const
Get TCG.
Definition TCT.h:190

◆ ~MHP()

MHP::~MHP ( )
virtual

Destructor.

Destructor

Definition at line 54 of file MHP.cpp.

55{
56 delete fja;
57}

Member Function Documentation

◆ addInterleavingThread() [1/2]

void SVF::MHP::addInterleavingThread ( const CxtThreadStmt tgr,
const CxtThreadStmt src 
)
inlineprivate

Definition at line 163 of file MHP.h.

164 {
166 if(changed)
167 {
168 instToTSMap[tgr.getStmt()].insert(tgr);
170 }
171 }
InstToThreadStmtSetMap instToTSMap
Map a statement to its thread interleavings.
Definition MHP.h:257
bool pushToCTSWorkList(const CxtThreadStmt &cs)
WorkList helper functions.
Definition MHP.h:217
ThreadStmtToThreadInterleav threadStmtToTheadInterLeav
Definition MHP.h:256
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74

◆ addInterleavingThread() [2/2]

void SVF::MHP::addInterleavingThread ( const CxtThreadStmt tgr,
NodeID  tid 
)
inlineprivate

Add/Remove interleaving thread for statement inst.

Definition at line 155 of file MHP.h.

156 {
157 if(threadStmtToTheadInterLeav[tgr].test_and_set(tid))
158 {
159 instToTSMap[tgr.getStmt()].insert(tgr);
161 }
162 }

◆ analyze()

void MHP::analyze ( )

Start analysis here.

Start analysis here

Definition at line 62 of file MHP.cpp.

63{
64 DBOUT(DGENERAL, outs() << pasMsg("MHP interleaving analysis\n"));
65 DBOUT(DMTA, outs() << pasMsg("MHP interleaving analysis\n"));
70}
#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 DMTA
Definition SVFType.h:519
#define DGENERAL
Definition SVFType.h:504
#define DOTIMESTAT(X)
Definition SVFType.h:500
void analyzeInterleaving()
Analyze thread interleaving.
Definition MHP.cpp:75
static double getClk(bool mark=false)
Definition SVFStat.cpp:48
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 & outs()
Overwrite llvm::outs()
Definition SVFUtil.h:52

◆ analyzeInterleaving()

void MHP::analyzeInterleaving ( )

Analyze thread interleaving.

Analyze thread interleaving

handle non-candidate function

handle candidate function

update non-candidate functions' interleaving

Definition at line 75 of file MHP.cpp.

76{
77 for (const std::pair<const NodeID, TCTNode*>& tpair : *tct)
78 {
79 const CxtThread& ct = tpair.second->getCxtThread();
80 NodeID rootTid = tpair.first;
82 const ICFGNode* svfInst = routine->getEntryBlock()->front();
83 CxtThreadStmt rootcts(rootTid, ct.getContext(), svfInst);
84
88
89 while (!cxtStmtList.empty())
90 {
92 const ICFGNode* curInst = cts.getStmt();
93 DBOUT(DMTA, outs() << "-----\nMHP analysis root thread: " << rootTid << " ");
94 DBOUT(DMTA, cts.dump());
95 DBOUT(DMTA, outs() << "current thread interleaving: < ");
97 DBOUT(DMTA, outs() << " >\n-----\n");
98
100 if (!tct->isCandidateFun(curInst->getFun()))
101 {
103 }
105 else
106 {
107 if (isTDFork(curInst))
108 {
110 }
111 else if (isTDJoin(curInst))
112 {
114 }
115 else if (tct->isCallSite(curInst) && !tct->isExtCall(curInst))
116 {
119 if (!tct->isCandidateFun(getCallee(SVFUtil::cast<CallICFGNode>(curInst), callees)))
121 }
122 else if (isRetInstNode(curInst))
123 {
124 handleRet(cts);
125 }
126 else
127 {
129 }
130 }
131 }
132 }
133
136
139}
Set< const FunObjVar * > FunctionSet
Definition CallGraph.h:244
bool empty() const
Definition WorkList.h:146
CxtThreadStmtWorkList cxtStmtList
CxtThreadStmt worklist.
Definition MHP.h:255
void printInterleaving()
Print interleaving results.
Definition MHP.cpp:648
void updateSiblingThreads(NodeID tid)
Definition MHP.cpp:419
void handleNonCandidateFun(const CxtThreadStmt &cts)
Handle non-candidate function.
Definition MHP.cpp:182
void handleJoin(const CxtThreadStmt &cts, NodeID rootTid)
Handle join.
Definition MHP.cpp:234
void addInterleavingThread(const CxtThreadStmt &tgr, NodeID tid)
Add/Remove interleaving thread for statement inst.
Definition MHP.h:155
const NodeBS & getInterleavingThreads(const CxtThreadStmt &cts)
Get interleaving thread for statement inst.
Definition MHP.h:97
bool isTDFork(const ICFGNode *call)
Whether it is a fork site.
Definition MHP.h:228
void handleRet(const CxtThreadStmt &cts)
Handle return.
Definition MHP.cpp:320
void handleFork(const CxtThreadStmt &cts, NodeID rootTid)
Handle fork.
Definition MHP.cpp:204
const CallGraph::FunctionSet & getCallee(const CallICFGNode *inst, CallGraph::FunctionSet &callees)
Definition MHP.h:126
void updateNonCandidateFunInterleaving()
Definition MHP.cpp:144
CxtThreadStmt popFromCTSWorkList()
Definition MHP.h:221
void updateAncestorThreads(NodeID tid)
Update Ancestor and sibling threads.
Definition MHP.cpp:383
void handleIntra(const CxtThreadStmt &cts)
Handle intra.
Definition MHP.cpp:367
void handleCall(const CxtThreadStmt &cts, NodeID rootTid)
Handle call.
Definition MHP.cpp:291
bool isTDJoin(const ICFGNode *call)
Whether it is a join site.
Definition MHP.h:234
static const Option< bool > PrintInterLev
Definition Options.h:159
bool isCallSite(const ICFGNode *inst)
Whether it is a callsite.
Definition TCT.h:265
bool isCandidateFun(const CallGraph::FunctionSet &callees) const
Whether it is a candidate function for indirect call.
Definition TCT.h:285
bool isExtCall(const ICFGNode *inst)
Whether it is calling an external function.
Definition TCT.h:258
const FunObjVar * getStartRoutineOfCxtThread(const CxtThread &ct) const
get the start routine function of a thread
Definition TCT.h:371
bool isRetInstNode(const ICFGNode *node)
Definition SVFUtil.cpp:376
void dumpSet(NodeBS To, OutStream &O=SVFUtil::outs())
Dump sparse bitvector set.
Definition SVFUtil.cpp:149
u32_t NodeID
Definition GeneralType.h:56

◆ executedByTheSameThread()

bool MHP::executedByTheSameThread ( const ICFGNode i1,
const ICFGNode i2 
)
virtual

Definition at line 627 of file MHP.cpp.

628{
630 return true;
631
634 for (const CxtThreadStmt&ts1 : tsSet1)
635 {
636 for (const CxtThreadStmt& ts2 : tsSet2)
637 {
638 if (ts1.getTid() != ts2.getTid() || isMultiForkedThread(ts1.getTid()))
639 return false;
640 }
641 }
642 return true;
643}
bool isMultiForkedThread(NodeID curTid)
A thread is a multiForked thread if it is in a loop or recursion.
Definition MHP.h:200
Set< CxtThreadStmt > CxtThreadStmtSet
Definition MHP.h:51
const CxtThreadStmtSet & getThreadStmtSet(const ICFGNode *inst) const
Get/has ThreadStmt.
Definition MHP.h:109
bool hasThreadStmtSet(const ICFGNode *inst) const
Definition MHP.h:115

◆ getCallee()

const CallGraph::FunctionSet & SVF::MHP::getCallee ( const CallICFGNode inst,
CallGraph::FunctionSet callees 
)
inlineprivate

Definition at line 126 of file MHP.h.

127 {
128 tcg->getCallees(inst, callees);
129 return callees;
130 }
void getCallees(const CallICFGNode *cs, FunctionSet &callees)
Get all callees for a callsite.
Definition CallGraph.h:414

◆ getDirAndIndJoinedTid()

NodeBS MHP::getDirAndIndJoinedTid ( const CallStrCxt cxt,
const ICFGNode call 
)
private

Return thread id(s) which are directly or indirectly joined at this join site.

Return thread id(s) which are directly or indirectly joined at this join site

Definition at line 492 of file MHP.cpp.

493{
494 CxtStmt cs(cxt, call);
495 return fja->getDirAndIndJoinedTid(cs);
496}
NodeBS getDirAndIndJoinedTid(const CxtStmt &cs)
Get directly and indirectly joined threadIDs based on a context-sensitive join site.
Definition MHP.cpp:975

◆ getInterleavingThreads()

const NodeBS & SVF::MHP::getInterleavingThreads ( const CxtThreadStmt cts)
inline

Get interleaving thread for statement inst.

Definition at line 97 of file MHP.h.

98 {
100 }

◆ getJoinInSymmetricLoop()

const MHP::LoopBBs & MHP::getJoinInSymmetricLoop ( const CallStrCxt cxt,
const ICFGNode call 
) const
private

Whether a context-sensitive join satisfies symmetric loop pattern.

Definition at line 508 of file MHP.cpp.

509{
510 CxtStmt cs(cxt, call);
511 return fja->getJoinInSymmetricLoop(cs);
512}
const LoopBBs & getJoinInSymmetricLoop(const CxtStmt &cs) const
Whether a context-sensitive join satisfies symmetric loop pattern.
Definition MHP.h:314

◆ getTCT()

TCT * SVF::MHP::getTCT ( ) const
inline

Get Thread Creation Tree.

Definition at line 80 of file MHP.h.

81 {
82 return tct;
83 }

◆ getThreadCallGraph()

ThreadCallGraph * SVF::MHP::getThreadCallGraph ( ) const
inline

Get ThreadCallGraph.

Definition at line 74 of file MHP.h.

75 {
76 return tcg;
77 }

◆ getThreadStmtSet()

const CxtThreadStmtSet & SVF::MHP::getThreadStmtSet ( const ICFGNode inst) const
inline

Get/has ThreadStmt.

Definition at line 109 of file MHP.h.

110 {
111 InstToThreadStmtSetMap::const_iterator it = instToTSMap.find(inst);
112 assert(it!=instToTSMap.end() && "no thread access the instruction?");
113 return it->second;
114 }

◆ handleCall()

void MHP::handleCall ( const CxtThreadStmt cts,
NodeID  rootTid 
)
private

Handle call.

Handle call instruction in the current thread scope (excluding any fork site)

Definition at line 291 of file MHP.cpp.

292{
293
294 const ICFGNode* call = cts.getStmt();
295 const CallStrCxt& curCxt = cts.getContext();
296 const CallICFGNode* cbn = cast<CallICFGNode>(call);
298 {
299 for (CallGraph::CallGraphEdgeSet::const_iterator cgIt = tcg->getCallEdgeBegin(cbn),
301 cgIt != ecgIt; ++cgIt)
302 {
303
304 const FunObjVar* svfcallee = (*cgIt)->getDstNode()->getFunction();
305 if (isExtCall(svfcallee))
306 continue;
308 const CallICFGNode* callicfgnode = SVFUtil::cast<CallICFGNode>(call);
310 const ICFGNode* svfEntryInst = svfcallee->getEntryBlock()->front();
313 }
314 }
315}
CallGraphEdgeSet::const_iterator getCallEdgeBegin(const CallICFGNode *inst) const
Definition CallGraph.h:433
bool hasCallGraphEdge(const CallICFGNode *inst) const
Get call graph edge via call instruction.
Definition CallGraph.h:429
CallGraphEdgeSet::const_iterator getCallEdgeEnd(const CallICFGNode *inst) const
Definition CallGraph.h:440
virtual const FunObjVar * getFunction() const
Get containing function, or null for globals/constants.
void pushCxt(CallStrCxt &cxt, const CallICFGNode *call, const FunObjVar *callee)
Push calling context.
Definition MHP.h:205
bool isExtCall(const FunObjVar *fun)
Definition SVFUtil.cpp:437
std::vector< u32_t > CallStrCxt

◆ handleFork()

void MHP::handleFork ( const CxtThreadStmt cts,
NodeID  rootTid 
)
private

Handle fork.

Handle fork

Definition at line 204 of file MHP.cpp.

205{
206
207 const ICFGNode* call = cts.getStmt();
208 const CallStrCxt& curCxt = cts.getContext();
209
210 assert(isTDFork(call));
211 const CallICFGNode* cbn = cast<CallICFGNode>(call);
213 {
214
215 for (ThreadCallGraph::ForkEdgeSet::const_iterator cgIt = tcg->getForkEdgeBegin(cbn),
217 cgIt != ecgIt; ++cgIt)
218 {
219 const FunObjVar* svfroutine = (*cgIt)->getDstNode()->getFunction();
222 const ICFGNode* stmt = svfroutine->getEntryBlock()->front();
223 CxtThread ct(newCxt, call);
224 CxtThreadStmt newcts(tct->getTCTNode(ct)->getId(), ct.getContext(), stmt);
226 }
227 }
229}
NodeID getId() const
Get ID.
Definition SVFValue.h:158
TCTNode * getTCTNode(NodeID id) const
Get TCT node.
Definition TCT.h:200
ForkEdgeSet::const_iterator getForkEdgeEnd(const CallICFGNode *cs) const
ForkEdgeSet::const_iterator getForkEdgeBegin(const CallICFGNode *cs) const

◆ handleIntra()

void MHP::handleIntra ( const CxtThreadStmt cts)
private

Handle intra.

Handling intraprocedural statements (successive statements on the CFG )

Definition at line 367 of file MHP.cpp.

368{
369
370 for(const ICFGEdge* outEdge : cts.getStmt()->getOutEdges())
371 {
372 if(outEdge->getDstNode()->getFun() == cts.getStmt()->getFun())
373 {
374 CxtThreadStmt newCts(cts.getTid(), cts.getContext(), outEdge->getDstNode());
376 }
377 }
378}

◆ handleJoin()

void MHP::handleJoin ( const CxtThreadStmt cts,
NodeID  rootTid 
)
private

Handle join.

Handle join

for the join site in a loop loop which does not join the current thread we process the loop exit

Definition at line 234 of file MHP.cpp.

235{
236
237 const CallStrCxt& curCxt = cts.getContext();
238
239 assert(isTDJoin(cts.getStmt()));
240
241 const CallICFGNode* call = SVFUtil::cast<CallICFGNode>(cts.getStmt());
242
244 if (!joinedTids.empty())
245 {
246 if (fja->hasJoinLoop(call))
247 {
248 std::vector<const SVFBasicBlock*> exitbbs;
249 call->getFun()->getExitBlocksOfLoop(call->getBB(), exitbbs);
250 while (!exitbbs.empty())
251 {
252 const SVFBasicBlock* eb = exitbbs.back();
253 exitbbs.pop_back();
254 const ICFGNode* svfEntryInst = eb->front();
259 }
260 }
261 else
262 {
264 DBOUT(DMTA, outs() << "\n\t match join site " << call->toString() << " for thread " << rootTid << "\n");
265 }
266 }
269 else
270 {
271 if (fja->hasJoinLoop(call))
272 {
273 std::vector<const SVFBasicBlock*> exitbbs;
274 call->getFun()->getExitBlocksOfLoop(call->getBB(), exitbbs);
275 while (!exitbbs.empty())
276 {
277 const SVFBasicBlock* eb = exitbbs.back();
278 exitbbs.pop_back();
279 const ICFGNode* svfEntryInst = eb->front();
280 CxtThreadStmt newCts(cts.getTid(), cts.getContext(), svfEntryInst);
282 }
283 }
284 }
286}
const std::string toString() const override
Definition ICFG.cpp:139
bool hasJoinLoop(const CallICFGNode *inst)
Definition MHP.h:354
void getExitBlocksOfLoop(const SVFBasicBlock *bb, BBList &exitbbs) const
virtual const FunObjVar * getFun() const
Return the function of this ICFGNode.
Definition ICFGNode.h:76
virtual const SVFBasicBlock * getBB() const
Return the basic block of this ICFGNode.
Definition ICFGNode.h:82
void rmInterleavingThread(const CxtThreadStmt &tgr, const NodeBS &tids, const ICFGNode *joinsite)
Definition MHP.h:172
bool hasJoinInSymmetricLoop(const CallStrCxt &cxt, const ICFGNode *call) const
Whether a context-sensitive join satisfies symmetric loop pattern.
Definition MHP.cpp:501
NodeBS getDirAndIndJoinedTid(const CallStrCxt &cxt, const ICFGNode *call)
Return thread id(s) which are directly or indirectly joined at this join site.
Definition MHP.cpp:492
const ICFGNode * back() const

◆ handleNonCandidateFun()

void MHP::handleNonCandidateFun ( const CxtThreadStmt cts)
private

Handle non-candidate function.

Handle call instruction in the current thread scope (excluding any fork site)

Definition at line 182 of file MHP.cpp.

183{
184 const ICFGNode* curInst = cts.getStmt();
185 const FunObjVar* curfun = curInst->getFun();
186 assert((curInst == curfun->getEntryBlock()->front()) && "curInst is not the entry of non candidate function.");
187 const CallStrCxt& curCxt = cts.getContext();
189 for (CallGraphNode::const_iterator nit = node->OutEdgeBegin(), neit = node->OutEdgeEnd(); nit != neit; nit++)
190 {
191 const FunObjVar* callee = (*nit)->getDstNode()->getFunction();
192 if (!isExtCall(callee))
193 {
194 const ICFGNode* calleeInst = callee->getEntryBlock()->front();
197 }
198 }
199}
const CallGraphNode * getCallGraphNode(const std::string &name)
Get call graph node.
iterator OutEdgeEnd()
iterator OutEdgeBegin()
iterators
GEdgeSetTy::const_iterator const_iterator

◆ handleRet()

void MHP::handleRet ( const CxtThreadStmt cts)
private

Handle return.

Handle return instruction in the current thread scope (excluding any join site)

Definition at line 320 of file MHP.cpp.

321{
322 CallGraphNode* curFunNode = tcg->getCallGraphNode(cts.getStmt()->getFun());
323 for (CallGraphEdge* edge : curFunNode->getInEdges())
324 {
325 if (SVFUtil::isa<ThreadForkEdge, ThreadJoinEdge>(edge))
326 continue;
327 for (CallGraphEdge::CallInstSet::const_iterator cit = (edge)->directCallsBegin(),
328 ecit = (edge)->directCallsEnd();
329 cit != ecit; ++cit)
330 {
331 CallStrCxt newCxt = cts.getContext();
332 if (matchCxt(newCxt, *cit, curFunNode->getFunction()))
333 {
334 for(const ICFGEdge* outEdge : cts.getStmt()->getOutEdges())
335 {
336 if(outEdge->getDstNode()->getFun() == cts.getStmt()->getFun())
337 {
338 CxtThreadStmt newCts(cts.getTid(), newCxt, outEdge->getDstNode());
340 }
341 }
342 }
343 }
344 for (CallGraphEdge::CallInstSet::const_iterator cit = (edge)->indirectCallsBegin(),
345 ecit = (edge)->indirectCallsEnd();
346 cit != ecit; ++cit)
347 {
348 CallStrCxt newCxt = cts.getContext();
349 if (matchCxt(newCxt, *cit, curFunNode->getFunction()))
350 {
351 for(const ICFGEdge* outEdge : cts.getStmt()->getOutEdges())
352 {
353 if(outEdge->getDstNode()->getFun() == cts.getStmt()->getFun())
354 {
355 CxtThreadStmt newCts(cts.getTid(), newCxt, outEdge->getDstNode());
357 }
358 }
359 }
360 }
361 }
362}
bool matchCxt(CallStrCxt &cxt, const CallICFGNode *call, const FunObjVar *callee)
Match context.
Definition MHP.h:210

◆ hasInterleavingThreads()

bool SVF::MHP::hasInterleavingThreads ( const CxtThreadStmt cts) const
inline

Definition at line 101 of file MHP.h.

102 {
104 }

◆ hasJoinInSymmetricLoop()

bool MHP::hasJoinInSymmetricLoop ( const CallStrCxt cxt,
const ICFGNode call 
) const
private

Whether a context-sensitive join satisfies symmetric loop pattern.

Whether a context-sensitive join satisfies symmetric loop pattern

Definition at line 501 of file MHP.cpp.

502{
503 CxtStmt cs(cxt, call);
504 return fja->hasJoinInSymmetricLoop(cs);
505}
bool hasJoinInSymmetricLoop(const CxtStmt &cs) const
Definition MHP.h:320

◆ hasThreadStmtSet()

bool SVF::MHP::hasThreadStmtSet ( const ICFGNode inst) const
inline

Definition at line 115 of file MHP.h.

116 {
117 return instToTSMap.find(inst)!=instToTSMap.end();
118 }

◆ isConnectedfromMain()

bool MHP::isConnectedfromMain ( const FunObjVar fun)

Whether the function is connected from main function in thread call graph.

Definition at line 522 of file MHP.cpp.

523{
526 TCT::PTACGNodeSet visited;
527 worklist.push(cgnode);
528 visited.insert(cgnode);
529 while (!worklist.empty())
530 {
531 const CallGraphNode* node = worklist.pop();
532 if ("main" == node->getFunction()->getName())
533 return true;
534 for (CallGraphNode::const_iterator nit = node->InEdgeBegin(), neit = node->InEdgeEnd(); nit != neit; nit++)
535 {
536 const CallGraphNode* srcNode = (*nit)->getSrcNode();
537 if (visited.find(srcNode) == visited.end())
538 {
539 visited.insert(srcNode);
540 worklist.push(srcNode);
541 }
542 }
543 }
544 return false;
545}
const FunObjVar * getFunction() const
Get function of this call node.
Definition CallGraph.h:191
bool push(const Data &data)
Definition WorkList.h:165
iterator InEdgeBegin()
iterator InEdgeEnd()
virtual const std::string & getName() const
Definition SVFValue.h:184
Set< const CallGraphNode * > PTACGNodeSet
Definition TCT.h:163

◆ isHBPair()

bool MHP::isHBPair ( NodeID  tid1,
NodeID  tid2 
)
private

Whether thread t1 happens before t2 based on ForkJoin Analysis.

Whether two thread t1 happens-fore t2

Definition at line 517 of file MHP.cpp.

518{
519 return fja->isHBPair(tid1, tid2);
520}
bool isHBPair(NodeID tid1, NodeID tid2)
Whether thread t1 happens-before thread t2.
Definition MHP.h:326

◆ isMultiForkedThread()

bool SVF::MHP::isMultiForkedThread ( NodeID  curTid)
inlineprivate

A thread is a multiForked thread if it is in a loop or recursion.

Definition at line 200 of file MHP.h.

201 {
203 }
bool isMultiforked() const
Definition TCT.h:120

◆ isMustJoin()

bool MHP::isMustJoin ( const NodeID  curTid,
const ICFGNode joinsite 
)
private

Whether a join site must join a thread t.

A join site must join t if (1) t is not a multiforked thread (2) the join site of t is not in recursion

Definition at line 482 of file MHP.cpp.

483{
484 const CallICFGNode* call = SVFUtil::dyn_cast<CallICFGNode>(joinsite);
485 assert(call && isTDJoin(call) && "not a join site!");
487}
bool isJoinSiteInRecursion(const CallICFGNode *join) const
Whether a join site is in recursion.
Definition TCT.h:422

◆ isRecurFullJoin()

bool MHP::isRecurFullJoin ( NodeID  parentTid,
NodeID  curTid 
)
private

Thread curTid can be fully joined by parentTid recursively.

Whether curTid can be fully joined by parentTid recursively

Definition at line 447 of file MHP.cpp.

448{
449 if (parentTid == curTid)
450 return true;
451
454 worklist.push(curNode);
455 while (!worklist.empty())
456 {
457 const TCTNode* node = worklist.pop();
458 for (TCTEdge* edge : node->getInEdges())
459 {
460 NodeID srcID = edge->getSrcID();
461 if (fja->isFullJoin(srcID, node->getId()))
462 {
463 if (srcID == parentTid)
464 return true;
465 else
466 worklist.push(edge->getSrcNode());
467 }
468 else
469 {
470 return false;
471 }
472 }
473 }
474 return false;
475}
bool isFullJoin(NodeID tid1, NodeID tid2)
Whether t1 fully joins t2.
Definition MHP.h:333

◆ isTDFork()

bool SVF::MHP::isTDFork ( const ICFGNode call)
inlineprivate

Whether it is a fork site.

Definition at line 228 of file MHP.h.

229 {
230 const CallICFGNode* fork = SVFUtil::dyn_cast<CallICFGNode>(call);
231 return fork && tcg->getThreadAPI()->isTDFork(fork);
232 }
bool isTDFork(const CallICFGNode *inst) const
Return true if this call create a new thread.
ThreadAPI * getThreadAPI() const
Thread API.

◆ isTDJoin()

bool SVF::MHP::isTDJoin ( const ICFGNode call)
inlineprivate

Whether it is a join site.

Definition at line 234 of file MHP.h.

235 {
236 const CallICFGNode* join = SVFUtil::dyn_cast<CallICFGNode>(call);
237 return join && tcg->getThreadAPI()->isTDJoin(join);
238 }
bool isTDJoin(const CallICFGNode *inst) const
Return true if this call wait for a worker thread.

◆ matchCxt()

bool SVF::MHP::matchCxt ( CallStrCxt cxt,
const CallICFGNode call,
const FunObjVar callee 
)
inlineprivate

Match context.

Definition at line 210 of file MHP.h.

211 {
212 return tct->matchCxt(cxt,call,callee);
213 }
bool matchCxt(CallStrCxt &cxt, const CallICFGNode *call, const FunObjVar *callee)
Match context.
Definition TCT.cpp:466

◆ mayHappenInParallel()

bool MHP::mayHappenInParallel ( const ICFGNode i1,
const ICFGNode i2 
)
virtual

Interface to query whether two instructions may happen-in-parallel.

Definition at line 615 of file MHP.cpp.

616{
618
619 DOTIMESTAT(double queryStart = PTAStat::getClk(true));
620 bool mhp = mayHappenInParallelCache(i1, i2);
621 DOTIMESTAT(double queryEnd = PTAStat::getClk(true));
623
624 return mhp;
625}
virtual bool mayHappenInParallelCache(const ICFGNode *i1, const ICFGNode *i2)
Definition MHP.cpp:593

◆ mayHappenInParallelCache()

bool MHP::mayHappenInParallelCache ( const ICFGNode i1,
const ICFGNode i2 
)
virtual

Definition at line 593 of file MHP.cpp.

594{
595 if (!tct->isCandidateFun(i1->getFun()) && !tct->isCandidateFun(i2->getFun()))
596 {
597 FuncPair funpair = std::make_pair(i1->getFun(), i2->getFun());
598 FuncPairToBool::const_iterator it = nonCandidateFuncMHPRelMap.find(funpair);
599 if (it == nonCandidateFuncMHPRelMap.end())
600 {
601 bool mhp = mayHappenInParallelInst(i1, i2);
603 return mhp;
604 }
605 else
606 {
607 if (it->second)
609 return it->second;
610 }
611 }
613}
FuncPairToBool nonCandidateFuncMHPRelMap
Definition MHP.h:258
std::pair< const FunObjVar *, const FunObjVar * > FuncPair
Definition MHP.h:58
virtual bool mayHappenInParallelInst(const ICFGNode *i1, const ICFGNode *i2)
Definition MHP.cpp:557

◆ mayHappenInParallelInst()

bool MHP::mayHappenInParallelInst ( const ICFGNode i1,
const ICFGNode i2 
)
virtual

Answer MHP queries For a pair of ThreadStmts (t1,s1) = <l1> (t2,s2) = <l2> They may happen in parallel if (1) t1 == t2 and t1 inloop/incycle (2) t1!=t2 and t1 \in l2 and t2 \in l1

TODO: Any instruction in dead function is assumed no MHP with others

Definition at line 557 of file MHP.cpp.

558{
559
562 return false;
563
566 for (const CxtThreadStmt& ts1 : tsSet1)
567 {
569 for (const CxtThreadStmt& ts2 : tsSet2)
570 {
572 if (ts1.getTid() != ts2.getTid())
573 {
574 if (l1.test(ts2.getTid()) && l2.test(ts1.getTid()))
575 {
577 return true;
578 }
579 }
580 else
581 {
582 if (isMultiForkedThread(ts1.getTid()))
583 {
585 return true;
586 }
587 }
588 }
589 }
590 return false;
591}

◆ popFromCTSWorkList()

CxtThreadStmt SVF::MHP::popFromCTSWorkList ( )
inlineprivate

Definition at line 221 of file MHP.h.

222 {
223 CxtThreadStmt ctp = cxtStmtList.pop();
224 return ctp;
225 }

◆ printInterleaving()

void MHP::printInterleaving ( )

Print interleaving results.

Print interleaving results

Definition at line 648 of file MHP.cpp.

649{
650 for (const auto& pair : threadStmtToTheadInterLeav)
651 {
652 outs() << "( t" << pair.first.getTid()
653 << pair.first.getStmt()->toString() << " ) ==> [";
654 for (unsigned i : pair.second)
655 {
656 outs() << " " << i << " ";
657 }
658 outs() << "]\n";
659 }
660}

◆ pushCxt()

void SVF::MHP::pushCxt ( CallStrCxt cxt,
const CallICFGNode call,
const FunObjVar callee 
)
inlineprivate

Push calling context.

Definition at line 205 of file MHP.h.

206 {
207 tct->pushCxt(cxt,call,callee);
208 }
void pushCxt(CallStrCxt &cxt, const CallICFGNode *call, const FunObjVar *callee)
Push calling context.
Definition TCT.cpp:445

◆ pushToCTSWorkList()

bool SVF::MHP::pushToCTSWorkList ( const CxtThreadStmt cs)
inlineprivate

WorkList helper functions.

Definition at line 217 of file MHP.h.

218 {
219 return cxtStmtList.push(cs);
220 }

◆ rmInterleavingThread()

void SVF::MHP::rmInterleavingThread ( const CxtThreadStmt tgr,
const NodeBS tids,
const ICFGNode joinsite 
)
inlineprivate

Definition at line 172 of file MHP.h.

173 {
175 for(NodeBS::iterator it = tids.begin(), eit = tids.end(); it!=eit; ++it)
176 {
177 if(isMustJoin(tgr.getTid(),joinsite))
178 joinedTids.set(*it);
179 }
180 if(threadStmtToTheadInterLeav[tgr].intersectWithComplement(joinedTids))
181 {
183 }
184 }
bool isMustJoin(const NodeID curTid, const ICFGNode *joinsite)
Whether a join site must join a thread t.
Definition MHP.cpp:482
SparseBitVectorIterator iterator
void set(unsigned Idx)
SparseBitVector NodeBS
Definition GeneralType.h:62

◆ updateAncestorThreads()

void MHP::updateAncestorThreads ( NodeID  curTid)
private

Update Ancestor and sibling threads.

Update interleavings of ancestor threads according to TCT

Definition at line 383 of file MHP.cpp.

384{
386 DBOUT(DMTA, outs() << "##Ancestor thread of " << curTid << " is : ");
388 DBOUT(DMTA, outs() << "\n");
389 tds.set(curTid);
390
391 for (const unsigned i : tds)
392 {
393 const CxtThread& ct = tct->getTCTNode(i)->getCxtThread();
394 if (const ICFGNode* forkInst = ct.getThread())
395 {
397 for(const ICFGEdge* outEdge : forkInst->getOutEdges())
398 {
399 if(outEdge->getDstNode()->getFun() == forkInst->getFun())
400 {
403 }
404 }
405 }
406 }
407}
const CxtThread & getCxtThread() const
Get CxtThread.
Definition TCT.h:101
NodeID getParentThread(NodeID tid) const
Get parent thread.
Definition TCT.h:314
const CallStrCxt & getCxtOfCxtThread(const CxtThread &ct) const
get the context of a thread at its spawning site (fork site)
Definition TCT.h:363
const NodeBS getAncestorThread(NodeID tid) const
Get all ancestor threads.
Definition TCT.h:323

◆ updateNonCandidateFunInterleaving()

void MHP::updateNonCandidateFunInterleaving ( )
private

Update non-candidate functions' interleaving. Copy interleaving threads of the entry inst to other insts.

Update non-candidate functions' interleaving

Definition at line 144 of file MHP.cpp.

145{
146 for (const auto& item : *PAG::getPAG()->getCallGraph())
147 {
148 const FunObjVar* fun = item.second->getFunction();
149 if (!tct->isCandidateFun(fun) && !isExtCall(fun))
150 {
151 const ICFGNode* entryNode = fun->getEntryBlock()->front();
152
154 continue;
155
157
158 for (const CxtThreadStmt& cts : tsSet)
159 {
160 const CallStrCxt& curCxt = cts.getContext();
161
162 for (auto it : *fun)
163 {
164 const SVFBasicBlock* svfbb = it.second;
165 for (const ICFGNode* curNode : svfbb->getICFGNodeList())
166 {
167 if (curNode == entryNode)
168 continue;
171 instToTSMap[curNode].insert(newCts);
172 }
173 }
174 }
175 }
176 }
177}
cJSON * item
Definition cJSON.h:222
const SVFBasicBlock * getEntryBlock() const
const ICFGNode * front() const

◆ updateSiblingThreads()

void MHP::updateSiblingThreads ( NodeID  curTid)
private

Update interleavings of sibling threads according to TCT

Exclude sibling thread that never happen in parallel based on ForkJoinAnalysis

The interleaving of a thread t is not unnecessary to be updated if (1) t HB Sibling and t fully joins curTid recursively or (2) Sibling HB t

Definition at line 419 of file MHP.cpp.

420{
422 tds.set(curTid);
423 for (const unsigned tid : tds)
424 {
426 for (const unsigned stid : siblingTds)
427 {
428 if ((isHBPair(tid, stid) && isRecurFullJoin(tid, curTid)) || isHBPair(stid, tid))
429 continue;
430
433 const ICFGNode* stmt = routine->getEntryBlock()->front();
434 CxtThreadStmt cts(stid, ct.getContext(), stmt);
436 }
437
438 DBOUT(DMTA, outs() << "##Sibling thread of " << curTid << " is : ");
440 DBOUT(DMTA, outs() << "\n");
441 }
442}
bool isRecurFullJoin(NodeID parentTid, NodeID curTid)
Thread curTid can be fully joined by parentTid recursively.
Definition MHP.cpp:447
bool isHBPair(NodeID tid1, NodeID tid2)
Whether thread t1 happens before t2 based on ForkJoin Analysis.
Definition MHP.cpp:517
const NodeBS getSiblingThread(NodeID tid) const
Get sibling threads.
Definition TCT.h:344

Member Data Documentation

◆ cxtStmtList

CxtThreadStmtWorkList SVF::MHP::cxtStmtList
private

CxtThreadStmt worklist.

Definition at line 255 of file MHP.h.

◆ fja

ForkJoinAnalysis* SVF::MHP::fja
private

ForJoin Analysis.

Definition at line 254 of file MHP.h.

◆ instToTSMap

InstToThreadStmtSetMap SVF::MHP::instToTSMap
private

Map a statement to its thread interleavings.

Map an instruction to its ThreadStmtSet

Definition at line 257 of file MHP.h.

◆ interleavingQueriesTime

double SVF::MHP::interleavingQueriesTime

Definition at line 265 of file MHP.h.

◆ interleavingTime

double SVF::MHP::interleavingTime

Definition at line 264 of file MHP.h.

◆ nonCandidateFuncMHPRelMap

FuncPairToBool SVF::MHP::nonCandidateFuncMHPRelMap
private

Definition at line 258 of file MHP.h.

◆ numOfMHPQueries

u32_t SVF::MHP::numOfMHPQueries

Number of queries are answered as may-happen-in-parallel.

Definition at line 263 of file MHP.h.

◆ numOfTotalQueries

u32_t SVF::MHP::numOfTotalQueries

Total number of queries.

Definition at line 262 of file MHP.h.

◆ tcg

ThreadCallGraph* SVF::MHP::tcg
private

TCG.

Definition at line 252 of file MHP.h.

◆ tct

TCT* SVF::MHP::tct
private

TCT.

Definition at line 253 of file MHP.h.

◆ threadStmtToTheadInterLeav

ThreadStmtToThreadInterleav SVF::MHP::threadStmtToTheadInterLeav
private

Definition at line 256 of file MHP.h.


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