Static Value-Flow Analysis
Loading...
Searching...
No Matches
MHP.cpp
Go to the documentation of this file.
1//===- MHP.cpp -- May-happen-in-parallel analysis-------------//
2//
3// SVF: Static Value-Flow Analysis
4//
5// Copyright (C) <2013-> <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 * MHP.cpp
25 *
26 * Created on: Jan 21, 2014
27 * Author: Yulei Sui, Peng Di
28 */
29
30#include "Util/Options.h"
31#include "MTA/MHP.h"
32#include "MTA/MTA.h"
33#include "MTA/LockAnalysis.h"
34#include "Util/SVFUtil.h"
35#include "Util/PTAStat.h"
36
37using namespace SVF;
38using namespace SVFUtil;
39
40
44MHP::MHP(TCT* t) : tcg(t->getThreadCallGraph()), tct(t), numOfTotalQueries(0), numOfMHPQueries(0),
45 interleavingTime(0), interleavingQueriesTime(0)
46{
49}
50
55{
56 delete fja;
57}
58
63{
64 DBOUT(DGENERAL, outs() << pasMsg("MHP interleaving analysis\n"));
65 DBOUT(DMTA, outs() << pasMsg("MHP interleaving analysis\n"));
70}
71
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}
140
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}
178
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}
200
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}
230
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}
287
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}
316
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}
363
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}
379
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}
408
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}
443
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}
476
483{
484 const CallICFGNode* call = SVFUtil::dyn_cast<CallICFGNode>(joinsite);
485 assert(call && isTDJoin(call) && "not a join site!");
487}
488
493{
494 CxtStmt cs(cxt, call);
495 return fja->getDirAndIndJoinedTid(cs);
496}
497
501bool MHP::hasJoinInSymmetricLoop(const CallStrCxt& cxt, const ICFGNode* call) const
502{
503 CxtStmt cs(cxt, call);
504 return fja->hasJoinInSymmetricLoop(cs);
505}
506
508const MHP::LoopBBs& MHP::getJoinInSymmetricLoop(const CallStrCxt& cxt, const ICFGNode* call) const
509{
510 CxtStmt cs(cxt, call);
511 return fja->getJoinInSymmetricLoop(cs);
512}
513
518{
519 return fja->isHBPair(tid1, tid2);
520}
521
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}
546
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}
592
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}
614
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}
626
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}
644
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}
661
669{
670 // typedef Set<const ICFGNode*> CallInstSet;
671 // typedef Map<const FunObjVar*, CallInstSet> FunToFJSites;
672 // FunToFJSites funToFJSites;
673
674 // for (ThreadCallGraph::CallSiteSet::const_iterator it = tct->getThreadCallGraph()->forksitesBegin(),
675 // eit = tct->getThreadCallGraph()->forksitesEnd();
676 // it != eit; ++it)
677 // {
678 // const ICFGNode* fork = *it;
679 // funToFJSites[fork->getFun()].insert(fork);
680 // }
681
682 // for (ThreadCallGraph::CallSiteSet::const_iterator it = tct->getThreadCallGraph()->joinsitesBegin(),
683 // eit = tct->getThreadCallGraph()->joinsitesEnd();
684 // it != eit; ++it)
685 // {
686 // const ICFGNode* join = *it;
687 // funToFJSites[join->getFun()].insert(join);
688 // }
689
690 // for(FunToFJSites::const_iterator it = funToFJSites.begin(), eit = funToFJSites.end(); it!=eit; ++it)
691 // {
692 // // ScalarEvolution* SE = MTA::getSE(it->first);
693 // for(CallInstSet::const_iterator sit = it->second.begin(), esit = it->second.end(); sit!=esit; ++sit)
694 // {
695 // const SVFInstruction* callInst = *sit;
696 // if(tct->getThreadCallGraph()->isForksite(getCBN(callInst)))
697 // {
698 // // const SVFValue* forkSiteTidPtr = getForkedThread(callInst);
699 // // const SCEV *forkSiteTidPtrSCEV = SE->getSCEV(const_cast<Value*>(forkSiteTidPtr));
700 // // const SCEV *baseForkTidPtrSCEV = SE->getSCEV(const_cast<Value*>(getBasePtr(forkSiteTidPtr)));
701 // // forkSiteTidPtrSCEV = getSCEVMinusExpr(forkSiteTidPtrSCEV, baseForkTidPtrSCEV, SE);
702 // // PTASCEV scev(forkSiteTidPtr,nullptr,nullptr);
703 // // fkjnToPTASCEVMap.insert(std::make_pair(callInst,scev));
704 // }
705 // else
706 // {
707 // // const SVFValue* joinSiteTidPtr = getJoinedThread(callInst);
708 // //const SCEV *joinSiteTidPtrSCEV = SE->getSCEV(const_cast<Value*>(joinSiteTidPtr));
709 // //const SCEV *baseJoinTidPtrSCEV = SE->getSCEV(const_cast<Value*>(getBasePtr(joinSiteTidPtr)));
710 // //joinSiteTidPtrSCEV = getSCEVMinusExpr(joinSiteTidPtrSCEV, baseJoinTidPtrSCEV, SE);
711
712 // // PTASCEV scev(joinSiteTidPtr,nullptr,nullptr);
713 // // fkjnToPTASCEVMap.insert(std::make_pair(callInst,scev));
714 // }
715 // }
716 // }
717}
718
723{
724 for (const std::pair<const NodeID, TCTNode*>& tpair : *tct)
725 {
726 const CxtThread& ct = tpair.second->getCxtThread();
727 const NodeID rootTid = tpair.first;
728 clearFlagMap();
729 if (const ICFGNode* forkInst = ct.getThread())
730 {
733
734 for(const ICFGEdge* outEdge : forkInst->getOutEdges())
735 {
736 if(outEdge->getDstNode()->getFun() == forkInst->getFun())
737 {
738 CxtStmt newCts(forkSiteCxt, outEdge->getDstNode());
740 }
741 }
742
743 while (!cxtStmtList.empty())
744 {
746 const ICFGNode* curInst = cts.getStmt();
747 DBOUT(DMTA, outs() << "-----\nForkJoinAnalysis root thread: " << tpair.first << " ");
748 DBOUT(DMTA, cts.dump());
749 DBOUT(DMTA, outs() << "-----\n");
751 if (isTDFork(curInst))
752 {
754 }
755 else if (isTDJoin(curInst))
756 {
758 }
760 {
761
763 }
764 else if (isRetInstNode(curInst))
765 {
766 handleRet(cts);
767 }
768 else
769 {
771 }
772
773 if (curInst == exitInst)
774 {
775 if (getMarkedFlag(cts) != TDAlive)
777 else
779 }
780 }
781 }
782 }
783}
784
787{
788 const ICFGNode* call = cts.getStmt();
789 const CallStrCxt& curCxt = cts.getContext();
790
791 assert(isTDFork(call));
792 const CallICFGNode* cbn = cast<CallICFGNode>(call);
793 if (getTCG()->hasThreadForkEdge(cbn))
794 {
795 for (ThreadCallGraph::ForkEdgeSet::const_iterator cgIt = getTCG()->getForkEdgeBegin(cbn),
796 ecgIt = getTCG()->getForkEdgeEnd(cbn);
797 cgIt != ecgIt; ++cgIt)
798 {
799 const FunObjVar* callee = (*cgIt)->getDstNode()->getFunction();
802 CxtThread ct(newCxt, call);
803 if (getMarkedFlag(cts) != TDAlive)
805 else
807 }
808 }
810}
811
814{
815 const ICFGNode* call = cts.getStmt();
816 const CallStrCxt& curCxt = cts.getContext();
817
818 assert(isTDJoin(call));
819 const CallICFGNode* cbn = cast<CallICFGNode>(call);
820 if (getTCG()->hasCallGraphEdge(cbn))
821 {
823 const ICFGNode* joinSite = cts.getStmt();
824
825 if (isAliasedForkJoin(SVFUtil::cast<CallICFGNode>(forkSite), SVFUtil::cast<CallICFGNode>(joinSite)))
826 {
827 if (hasJoinLoop(SVFUtil::cast<CallICFGNode>(forkSite)))
828 {
829 LoopBBs& joinLoop = getJoinLoop(SVFUtil::cast<CallICFGNode>(forkSite));
830 std::vector<const SVFBasicBlock *> exitbbs;
831 joinSite->getFun()->getExitBlocksOfLoop(joinSite->getBB(), exitbbs);
832 while (!exitbbs.empty())
833 {
834 const SVFBasicBlock* eb = exitbbs.back();
835 exitbbs.pop_back();
836 const ICFGNode* svfEntryInst = eb->front();
840 {
843 }
844 else
846 }
847 }
848 else
849 {
852 DBOUT(DMTA, outs() << "\n\t match join site " << call->toString() << "for thread " << rootTid << "\n");
853 }
854 }
857 else
858 {
859 if (hasJoinLoop(SVFUtil::cast<CallICFGNode>(forkSite)))
860 {
861 std::vector<const SVFBasicBlock*> exitbbs;
862 joinSite->getFun()->getExitBlocksOfLoop(joinSite->getBB(), exitbbs);
863 while (!exitbbs.empty())
864 {
865 const SVFBasicBlock* eb = exitbbs.back();
866 exitbbs.pop_back();
867 const ICFGNode* svfEntryInst = eb->front();
870 }
871 }
872 }
873 }
875}
876
879{
880
881 const ICFGNode* call = cts.getStmt();
882 const CallStrCxt& curCxt = cts.getContext();
883 const CallICFGNode* cbn = SVFUtil::cast<CallICFGNode>(call);
884 if (getTCG()->hasCallGraphEdge(cbn))
885 {
886 for (CallGraph::CallGraphEdgeSet::const_iterator cgIt = getTCG()->getCallEdgeBegin(cbn),
887 ecgIt = getTCG()->getCallEdgeEnd(cbn);
888 cgIt != ecgIt; ++cgIt)
889 {
890 const FunObjVar* svfcallee = (*cgIt)->getDstNode()->getFunction();
891 if (isExtCall(svfcallee))
892 continue;
895 const ICFGNode* svfEntryInst = svfcallee->getEntryBlock()->front();
898 }
899 }
900}
901
904{
905 const ICFGNode* curInst = cts.getStmt();
906 const CallStrCxt& curCxt = cts.getContext();
907
909 for (CallGraphEdge* edge : curFunNode->getInEdges())
910 {
911 if (SVFUtil::isa<ThreadForkEdge, ThreadJoinEdge>(edge))
912 continue;
913 for (CallGraphEdge::CallInstSet::const_iterator cit = edge->directCallsBegin(),
914 ecit = edge->directCallsEnd();
915 cit != ecit; ++cit)
916 {
918 const ICFGNode* curNode = (*cit);
919 if (matchCxt(newCxt, SVFUtil::cast<CallICFGNode>(curNode), curFunNode->getFunction()))
920 {
921 for(const ICFGEdge* outEdge : curNode->getOutEdges())
922 {
923 if(outEdge->getDstNode()->getFun() == curNode->getFun())
924 {
925 CxtStmt newCts(newCxt, outEdge->getDstNode());
927 }
928 }
929 }
930 }
931 for (CallGraphEdge::CallInstSet::const_iterator cit = edge->indirectCallsBegin(),
932 ecit = edge->indirectCallsEnd();
933 cit != ecit; ++cit)
934 {
936 const ICFGNode* curNode = (*cit);
937
938 if (matchCxt(newCxt, SVFUtil::cast<CallICFGNode>(curNode), curFunNode->getFunction()))
939 {
940 for(const ICFGEdge* outEdge : curNode->getOutEdges())
941 {
942 if(outEdge->getDstNode()->getFun() == curNode->getFun())
943 {
944 CxtStmt newCts(newCxt, outEdge->getDstNode());
946 }
947 }
948 }
949 }
950 }
951}
952
955{
956
957 const ICFGNode* curInst = cts.getStmt();
958 const CallStrCxt& curCxt = cts.getContext();
959
960 for(const ICFGEdge* outEdge : curInst->getOutEdges())
961 {
962 if(outEdge->getDstNode()->getFun() == curInst->getFun())
963 {
964 CxtStmt newCts(curCxt, outEdge->getDstNode());
966 }
967 }
968}
969
976{
977
978 CxtStmtToTIDMap::const_iterator it = dirAndIndJoinMap.find(cs);
979 if (it != dirAndIndJoinMap.end())
980 return it->second;
981
984
985 FIFOWorkList<NodeID> worklist;
986 for (unsigned id : directJoinTids)
987 {
988 worklist.push(id);
989 }
990
991 while (!worklist.empty())
992 {
993 NodeID tid = worklist.pop();
994 TCTNode* node = tct->getTCTNode(tid);
995 for (TCT::ThreadCreateEdgeSet::const_iterator it = tct->getChildrenBegin(node), eit = tct->getChildrenEnd(node); it != eit; ++it)
996 {
997 NodeID childTid = (*it)->getDstID();
998 if (isFullJoin(tid, childTid))
999 {
1000 allJoinTids.set(childTid);
1001 worklist.push(childTid);
1002 }
1003 }
1004 }
1005
1007
1008 return allJoinTids;
1009}
1010
1011// static bool accessSameArrayIndex(const GetElementPtrInst* ptr1, const GetElementPtrInst* ptr2)
1012// {
1013
1014// std::vector<u32_t> ptr1vec;
1015// for (gep_type_iterator gi = gep_type_begin(*ptr1), ge = gep_type_end(*ptr1);
1016// gi != ge; ++gi)
1017// {
1018// if(SVFConstantInt* ci = SVFUtil::dyn_cast<SVFConstantInt>(LLVMModuleSet::getLLVMModuleSet()->getSVFValue(gi.getOperand())))
1019// {
1020// s32_t idx = ci->getSExtValue();
1021// ptr1vec.push_back(idx);
1022// }
1023// else
1024// return false;
1025// }
1026
1027// std::vector<u32_t> ptr2vec;
1028// for (gep_type_iterator gi = gep_type_begin(*ptr2), ge = gep_type_end(*ptr2);
1029// gi != ge; ++gi)
1030// {
1031// if(SVFConstantInt* ci = SVFUtil::dyn_cast<SVFConstantInt>(LLVMModuleSet::getLLVMModuleSet()->getSVFValue(gi.getOperand())))
1032// {
1033// s32_t idx = ci->getSExtValue();
1034// ptr2vec.push_back(idx);
1035// }
1036// else
1037// return false;
1038// }
1039
1040// return ptr1vec==ptr2vec;
1041// }
1042
1051{
1052
1053 // const PTASCEV& forkse = fkjnToPTASCEVMap[forkSite];
1054 // const PTASCEV& joinse = fkjnToPTASCEVMap[joinSite];
1055
1056 // //if(sameLoopTripCount(forkSite,joinSite) == false)
1057 // // return false;
1058
1059 // if(forkse.inloop && joinse.inloop)
1060 // return forkse.start==joinse.start && forkse.step == joinse.step && forkse.tripcount <= joinse.tripcount;
1061 // else if(SVFUtil::isa<GetElementPtrInst>(forkse.ptr) && SVFUtil::isa<GetElementPtrInst>(joinse.ptr))
1062 // return accessSameArrayIndex(SVFUtil::cast<GetElementPtrInst>(forkse.ptr),SVFUtil::cast<GetElementPtrInst>(joinse.ptr));
1063 // else if(SVFUtil::isa<GetElementPtrInst, GetElementPtrInst>(joinse.ptr))
1064 // return false;
1065 // else
1066 // return true;
1067
1068 return false;
1069}
1070
1075{
1076
1077 // ScalarEvolution* forkSE = getSE(forkSite);
1078 // ScalarEvolution* joinSE = getSE(joinSite);
1079
1080 // if(tct->hasLoop(forkSite) == false || tct->hasLoop(joinSite) == false)
1081 // return false;
1082
1083 // // Get loops
1084 // const LoopBBs& forkSiteLoop = tct->getLoop(forkSite);
1085 // const LoopBBs& joinSiteLoop = tct->getLoop(joinSite);
1086
1087 // const SCEV* forkLoopCountScev = forkSE->getBackedgeTakenCount(forkSiteLoop);
1088 // const SCEV* joinLoopCountScev = joinSE->getBackedgeTakenCount(joinSiteLoop);
1089
1090 // if(forkLoopCountScev!=forkSE->getCouldNotCompute())
1091 // {
1092 // if(forkLoopCountScev==joinLoopCountScev)
1093 // {
1094 // return true;
1095 // }
1096 // }
1097 return false;
1098}
#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
cJSON * item
Definition cJSON.h:222
const FunObjVar * getFunction() const
Get function of this call node.
Definition CallGraph.h:191
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
Set< const FunObjVar * > FunctionSet
Definition CallGraph.h:244
CallGraphEdgeSet::const_iterator getCallEdgeEnd(const CallICFGNode *inst) const
Definition CallGraph.h:440
const CallGraphNode * getCallGraphNode(const std::string &name)
Get call graph node.
const std::string toString() const override
Definition ICFG.cpp:139
const ICFGNode * getThread() const
Return forksite.
Definition CxtStmt.h:211
bool push(const Data &data)
Definition WorkList.h:165
bool empty() const
Definition WorkList.h:146
void addToFullJoin(NodeID tid1, NodeID tid2)
full join and partial join
Definition MHP.h:517
bool matchCxt(CallStrCxt &cxt, const CallICFGNode *call, const FunObjVar *callee)
Match context.
Definition MHP.h:458
ValDomain getMarkedFlag(const CxtStmt &cs)
Mark thread flags for cxtStmt.
Definition MHP.h:389
void addToHPPair(NodeID tid1, NodeID tid2)
Definition MHP.h:504
SVFLoopAndDomInfo::LoopBBs LoopBBs
Definition MHP.h:285
void analyzeForkJoinPair()
Definition MHP.cpp:722
bool hasJoinLoop(const CallICFGNode *inst)
Definition MHP.h:354
const CallGraph::FunctionSet & getCallee(const ICFGNode *inst, CallGraph::FunctionSet &callees)
Definition MHP.h:485
void addSymmetricLoopJoin(const CxtStmt &cs, LoopBBs &lp)
Add inloop join.
Definition MHP.h:528
void handleRet(const CxtStmt &cts)
Handle return.
Definition MHP.cpp:903
NodeBS getDirAndIndJoinedTid(const CxtStmt &cs)
Get directly and indirectly joined threadIDs based on a context-sensitive join site.
Definition MHP.cpp:975
CxtStmt popFromCTSWorkList()
Definition MHP.h:445
void addToHBPair(NodeID tid1, NodeID tid2)
Definition MHP.h:509
ThreadCallGraph * getTCG() const
ThreadCallGraph.
Definition MHP.h:491
void addToPartial(NodeID tid1, NodeID tid2)
Definition MHP.h:521
void collectSCEVInfo()
functions
Definition MHP.cpp:668
void clearFlagMap()
Clear flags.
Definition MHP.h:432
bool isHBPair(NodeID tid1, NodeID tid2)
Whether thread t1 happens-before thread t2.
Definition MHP.h:326
void addDirectlyJoinTID(const CxtStmt &cs, NodeID tid)
maps a context-sensitive join site to a thread id
Definition MHP.h:496
LoopBBs & getJoinLoop(const CallICFGNode *inst)
Get loop for join site.
Definition MHP.h:350
void pushCxt(CallStrCxt &cxt, const CallICFGNode *call, const FunObjVar *callee)
Push calling context.
Definition MHP.h:453
const ICFGNode * getExitInstOfParentRoutineFun(NodeID tid) const
Get exit instruction of the start routine function of tid's parent thread.
Definition MHP.h:341
bool isTDFork(const ICFGNode *call)
Whether it is a fork site.
Definition MHP.h:464
bool isFullJoin(NodeID tid1, NodeID tid2)
Whether t1 fully joins t2.
Definition MHP.h:333
CxtStmtToTIDMap dirAndIndJoinMap
maps a context-sensitive join site to directly and indirectly joined thread ids
Definition MHP.h:536
void handleCall(const CxtStmt &cts, NodeID rootTid)
Handle call.
Definition MHP.cpp:878
bool hasJoinInSymmetricLoop(const CxtStmt &cs) const
Definition MHP.h:320
bool sameLoopTripCount(const ICFGNode *forkSite, const ICFGNode *joinSite)
Same loop trip count.
Definition MHP.cpp:1074
bool isTDJoin(const ICFGNode *call)
Whether it is a join site.
Definition MHP.h:470
void markCxtStmtFlag(const CxtStmt &tgr, ValDomain flag)
Initialize TDAlive and TDDead flags.
Definition MHP.h:401
CxtStmtWorkList cxtStmtList
context-sensitive statement worklist
Definition MHP.h:534
const LoopBBs & getJoinInSymmetricLoop(const CxtStmt &cs) const
Whether a context-sensitive join satisfies symmetric loop pattern.
Definition MHP.h:314
bool isAliasedForkJoin(const CallICFGNode *forkSite, const CallICFGNode *joinSite)
Whether it is a matched fork join pair.
Definition MHP.h:382
void handleIntra(const CxtStmt &cts)
Handle intra.
Definition MHP.cpp:954
void handleFork(const CxtStmt &cts, NodeID rootTid)
Handle fork.
Definition MHP.cpp:786
NodeBS & getDirectlyJoinedTid(const CxtStmt &cs)
Get directly joined threadIDs based on a context-sensitive join site.
Definition MHP.h:306
void handleJoin(const CxtStmt &cts, NodeID rootTid)
Handle join.
Definition MHP.cpp:813
bool isSameSCEV(const ICFGNode *forkSite, const ICFGNode *joinSite)
Return true if the fork and join have the same SCEV.
Definition MHP.cpp:1050
virtual const FunObjVar * getFunction() const
Get containing function, or null for globals/constants.
void getExitBlocksOfLoop(const SVFBasicBlock *bb, BBList &exitbbs) const
const SVFBasicBlock * getEntryBlock() const
iterator OutEdgeEnd()
const GEdgeSetTy & getInEdges() const
iterator OutEdgeBegin()
iterators
GEdgeSetTy::const_iterator const_iterator
iterator InEdgeBegin()
iterator InEdgeEnd()
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
virtual const std::string toString() const
Definition ICFG.cpp:60
void analyze()
Start analysis here.
Definition MHP.cpp:62
CxtThreadStmtWorkList cxtStmtList
CxtThreadStmt worklist.
Definition MHP.h:255
bool isRecurFullJoin(NodeID parentTid, NodeID curTid)
Thread curTid can be fully joined by parentTid recursively.
Definition MHP.cpp:447
bool isMultiForkedThread(NodeID curTid)
A thread is a multiForked thread if it is in a loop or recursion.
Definition MHP.h:200
void rmInterleavingThread(const CxtThreadStmt &tgr, const NodeBS &tids, const ICFGNode *joinsite)
Definition MHP.h:172
virtual bool mayHappenInParallelCache(const ICFGNode *i1, const ICFGNode *i2)
Definition MHP.cpp:593
TCT * tct
TCT.
Definition MHP.h:253
FuncPairToBool nonCandidateFuncMHPRelMap
Definition MHP.h:258
void printInterleaving()
Print interleaving results.
Definition MHP.cpp:648
void updateSiblingThreads(NodeID tid)
Definition MHP.cpp:419
u32_t numOfTotalQueries
Total number of queries.
Definition MHP.h:262
Set< CxtThreadStmt > CxtThreadStmtSet
Definition MHP.h:51
std::pair< const FunObjVar *, const FunObjVar * > FuncPair
Definition MHP.h:58
void handleNonCandidateFun(const CxtThreadStmt &cts)
Handle non-candidate function.
Definition MHP.cpp:182
SVFLoopAndDomInfo::LoopBBs LoopBBs
Definition MHP.h:54
void handleJoin(const CxtThreadStmt &cts, NodeID rootTid)
Handle join.
Definition MHP.cpp:234
void pushCxt(CallStrCxt &cxt, const CallICFGNode *call, const FunObjVar *callee)
Push calling context.
Definition MHP.h:205
bool matchCxt(CallStrCxt &cxt, const CallICFGNode *call, const FunObjVar *callee)
Match context.
Definition MHP.h:210
ThreadCallGraph * tcg
TCG.
Definition MHP.h:252
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
InstToThreadStmtSetMap instToTSMap
Map a statement to its thread interleavings.
Definition MHP.h:257
virtual ~MHP()
Destructor.
Definition MHP.cpp:54
bool isHBPair(NodeID tid1, NodeID tid2)
Whether thread t1 happens before t2 based on ForkJoin Analysis.
Definition MHP.cpp:517
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
virtual bool executedByTheSameThread(const ICFGNode *i1, const ICFGNode *i2)
Definition MHP.cpp:627
virtual bool mayHappenInParallel(const ICFGNode *i1, const ICFGNode *i2)
Interface to query whether two instructions may happen-in-parallel.
Definition MHP.cpp:615
const CxtThreadStmtSet & getThreadStmtSet(const ICFGNode *inst) const
Get/has ThreadStmt.
Definition MHP.h:109
void handleFork(const CxtThreadStmt &cts, NodeID rootTid)
Handle fork.
Definition MHP.cpp:204
const LoopBBs & getJoinInSymmetricLoop(const CallStrCxt &cxt, const ICFGNode *call) const
Whether a context-sensitive join satisfies symmetric loop pattern.
Definition MHP.cpp:508
ForkJoinAnalysis * fja
ForJoin Analysis.
Definition MHP.h:254
bool hasJoinInSymmetricLoop(const CallStrCxt &cxt, const ICFGNode *call) const
Whether a context-sensitive join satisfies symmetric loop pattern.
Definition MHP.cpp:501
const CallGraph::FunctionSet & getCallee(const CallICFGNode *inst, CallGraph::FunctionSet &callees)
Definition MHP.h:126
bool hasThreadStmtSet(const ICFGNode *inst) const
Definition MHP.h:115
double interleavingQueriesTime
Definition MHP.h:265
u32_t numOfMHPQueries
Number of queries are answered as may-happen-in-parallel.
Definition MHP.h:263
void updateNonCandidateFunInterleaving()
Definition MHP.cpp:144
MHP(TCT *t)
Constructor.
Definition MHP.cpp:44
bool isMustJoin(const NodeID curTid, const ICFGNode *joinsite)
Whether a join site must join a thread t.
Definition MHP.cpp:482
double interleavingTime
Definition MHP.h:264
bool isConnectedfromMain(const FunObjVar *fun)
Whether the function is connected from main function in thread call graph.
Definition MHP.cpp:522
CxtThreadStmt popFromCTSWorkList()
Definition MHP.h:221
void analyzeInterleaving()
Analyze thread interleaving.
Definition MHP.cpp:75
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
void updateAncestorThreads(NodeID tid)
Update Ancestor and sibling threads.
Definition MHP.cpp:383
virtual bool mayHappenInParallelInst(const ICFGNode *i1, const ICFGNode *i2)
Definition MHP.cpp:557
void handleIntra(const CxtThreadStmt &cts)
Handle intra.
Definition MHP.cpp:367
void handleCall(const CxtThreadStmt &cts, NodeID rootTid)
Handle call.
Definition MHP.cpp:291
ThreadStmtToThreadInterleav threadStmtToTheadInterLeav
Definition MHP.h:256
bool isTDJoin(const ICFGNode *call)
Whether it is a join site.
Definition MHP.h:234
static const Option< bool > PrintInterLev
Definition Options.h:159
const ICFGNode * front() const
const ICFGNode * back() const
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
static double getClk(bool mark=false)
Definition SVFStat.cpp:48
NodeID getId() const
Get ID.
Definition SVFValue.h:158
virtual const std::string & getName() const
Definition SVFValue.h:184
void set(unsigned Idx)
const CxtThread & getCxtThread() const
Get CxtThread.
Definition TCT.h:101
TCTNode * getTCTNode(NodeID id) const
Get TCT node.
Definition TCT.h:200
bool isJoinSiteInRecursion(const CallICFGNode *join) const
Whether a join site is in recursion.
Definition TCT.h:422
bool isCallSite(const ICFGNode *inst)
Whether it is a callsite.
Definition TCT.h:265
Set< const CallGraphNode * > PTACGNodeSet
Definition TCT.h:163
ThreadCreateEdgeSet::const_iterator getChildrenBegin(const TCTNode *node) const
Get children and parent nodes.
Definition TCT.h:211
NodeID getParentThread(NodeID tid) const
Get parent thread.
Definition TCT.h:314
ThreadCallGraph * getThreadCallGraph() const
Get TCG.
Definition TCT.h:190
const NodeBS getSiblingThread(NodeID tid) const
Get sibling threads.
Definition TCT.h:344
bool isCandidateFun(const CallGraph::FunctionSet &callees) const
Whether it is a candidate function for indirect call.
Definition TCT.h:285
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
bool isExtCall(const ICFGNode *inst)
Whether it is calling an external function.
Definition TCT.h:258
ThreadCreateEdgeSet::const_iterator getChildrenEnd(const TCTNode *node) const
Definition TCT.h:215
const FunObjVar * getStartRoutineOfCxtThread(const CxtThread &ct) const
get the start routine function of a thread
Definition TCT.h:371
ForkEdgeSet::const_iterator getForkEdgeEnd(const CallICFGNode *cs) const
ForkEdgeSet::const_iterator getForkEdgeBegin(const CallICFGNode *cs) const
std::string pasMsg(const std::string &msg)
Print each pass/phase message by converting a string into blue string output.
Definition SVFUtil.cpp:101
bool isExtCall(const FunObjVar *fun)
Definition SVFUtil.cpp:437
bool isRetInstNode(const ICFGNode *node)
Definition SVFUtil.cpp:376
std::ostream & outs()
Overwrite llvm::outs()
Definition SVFUtil.h:52
void dumpSet(NodeBS To, OutStream &O=SVFUtil::outs())
Dump sparse bitvector set.
Definition SVFUtil.cpp:149
for isBitcode
Definition BasicTypes.h:68
u32_t NodeID
Definition GeneralType.h:56
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
std::vector< u32_t > CallStrCxt