Static Value-Flow Analysis
Loading...
Searching...
No Matches
SVFFileSystem.cpp
Go to the documentation of this file.
2#include "Graphs/CHG.h"
3#include "SVFIR/SVFIR.h"
4#include "Util/CommandLine.h"
5#include <sys/fcntl.h>
6#include <sys/mman.h>
7#include <sys/stat.h>
8#include <sys/types.h>
9#include <unistd.h>
10
12 "human-readable", "Whether to output human-readable JSON", true);
13
14namespace SVF
15{
16
17SVFType* createSVFType(SVFType::GNodeK kind, bool isSingleValTy)
18{
19 switch (kind)
20 {
21 default:
22 ABORT_MSG(kind << " is an impossible SVFTyKind in create()");
23 case SVFType::SVFTy:
24 ABORT_MSG("Creation of RAW SVFType isn't allowed");
26 ABORT_IFNOT(isSingleValTy, "Pointer type must be single-valued");
27 return new SVFPointerType();
29 ABORT_IFNOT(isSingleValTy, "Integer type must be single-valued");
30 return new SVFIntegerType();
32 ABORT_IFNOT(!isSingleValTy, "Function type must be multi-valued");
33 return new SVFFunctionType(nullptr);
35 ABORT_IFNOT(!isSingleValTy, "Struct type must be multi-valued");
36 return new SVFStructType();
38 ABORT_IFNOT(!isSingleValTy, "Array type must be multi-valued");
39 return new SVFArrayType();
41 return new SVFOtherType(isSingleValTy);
42 }
43}
44
45template <typename SmallNumberType>
46static inline void readSmallNumber(const cJSON* obj, SmallNumberType& val)
47{
48 val = static_cast<SmallNumberType>(jsonGetNumber(obj));
49}
50
51template <typename BigNumberType, typename CStrToVal>
52static inline void readBigNumber(const cJSON* obj, BigNumberType& val, CStrToVal conv)
53{
55 "Expect (number) string JSON for " << JSON_KEY(obj));
56 val = conv(obj->valuestring);
57}
58
63
65{
66 // OK, double precision enough
68}
69
71{
72 // OK, double precision enough
74}
75
76cJSON* SVFIRWriter::toJson(const std::string& str)
77{
78 return jsonCreateString(str.c_str());
79}
80
85
87{
88 // unsigned long is subset of unsigned long long
89 return toJson(static_cast<unsigned long long>(number));
90}
91
93{
94 return toJson(static_cast<unsigned long long>(number));
95}
96
97cJSON* SVFIRWriter::toJson(unsigned long long number)
98{
100}
101
103{
104 auto kind = type->getKind();
105
106 switch (kind)
107 {
108 default:
109 assert(false && "Impossible SVFType kind");
110
111#define CASE(Kind) \
112 case SVFType::Kind: \
113 return contentToJson(static_cast<const Kind##pe*>(type))
114
115 CASE(SVFTy);
116 CASE(SVFPointerTy);
117 CASE(SVFIntegerTy);
118 CASE(SVFFunctionTy);
119 CASE(SVFStructTy);
120 CASE(SVFArrayTy);
121 CASE(SVFOtherTy);
122#undef CASE
123 }
124}
125
127{
128 auto kind = value->getKind();
129
130 switch (kind)
131 {
132 default:
133 assert(false && "Impossible SVFValue kind");
134
135#define CASE(ValueKind, type) \
136 case SVFValue::ValueKind: \
137 return contentToJson(static_cast<const type*>(value))
138
139 CASE(SVFVal, SVFValue);
140 CASE(SVFFunc, SVFFunction);
141 CASE(SVFBB, SVFBasicBlock);
142 CASE(SVFInst, SVFInstruction);
143 CASE(SVFCall, SVFCallInst);
144 CASE(SVFGlob, SVFGlobalValue);
145 CASE(SVFArg, SVFArgument);
146 CASE(SVFConst, SVFConstant);
147 CASE(SVFConstData, SVFConstantData);
148 CASE(SVFConstInt, SVFConstantInt);
149 CASE(SVFConstFP, SVFConstantFP);
150 CASE(SVFNullPtr, SVFConstantNullPtr);
151 CASE(SVFBlackHole, SVFBlackHoleValue);
152 CASE(SVFMetaAsValue, SVFMetadataAsValue);
153 CASE(SVFOther, SVFOtherValue);
154#undef CASE
155 }
156}
157
159{
160 switch (var->getNodeKind())
161 {
162 default:
163 assert(false && "Unknown SVFVar kind");
164
165#define CASE(VarKind, VarType) \
166 case SVFVar::VarKind: \
167 return contentToJson(static_cast<const VarType*>(var))
168
169 CASE(ValNode, ValVar);
170 CASE(ObjNode, ObjVar);
171 CASE(RetNode, RetPN);
172 CASE(VarargNode, VarArgPN);
173 CASE(GepValNode, GepValVar);
174 CASE(GepObjNode, GepObjVar);
175 CASE(BaseObjNode, BaseObjVar);
176 CASE(DummyValNode, DummyValVar);
177 CASE(DummyObjNode, DummyObjVar);
178 CASE(FunObjNode, FunObjVar);
179 CASE(ArgNode, ArgValVar);
180 CASE(FunValNode, FunValVar);
181 CASE(HeapObjNode, HeapObjVar);
182 CASE(StackObjNode, StackObjVar);
183 CASE(ConstantDataValNode, ConstantDataValVar);
184 CASE(ConstantAggValNode, ConstantAggValVar);
185 CASE(GlobalValNode, GlobalValVar);
186 CASE(BlackHoleNode, BlackHoleVar);
187 CASE(ConstantFPValNode, ConstantFPValVar);
188 CASE(ConstantIntValNode, ConstantIntValVar);
189 CASE(ConstantNullptrValNode, ConstantNullPtrValVar);
190 CASE(ConstantDataObjNode, ConstantDataObjVar);
191 CASE(ConstantAggObjNode, ConstantAggObjVar);
192 CASE(GlobalObjNode, GlobalObjVar);
193 CASE(ConstantFPObjNode, ConstantFPObjVar);
194 CASE(ConstantIntObjNode, ConstantIntObjVar);
195 CASE(ConstantNullptrObjNode, ConstantNullPtrObjVar);
196#undef CASE
197 }
198}
199
201{
202 switch (stmt->getEdgeKind())
203 {
204 default:
205 assert(false && "Unknown SVFStmt kind");
206
207#define CASE(EdgeKind, EdgeType) \
208 case SVFStmt::EdgeKind: \
209 return contentToJson(static_cast<const EdgeType*>(stmt))
210
211 CASE(Addr, AddrStmt);
212 CASE(Copy, CopyStmt);
213 CASE(Store, StoreStmt);
214 CASE(Load, LoadStmt);
215 CASE(Call, CallPE);
216 CASE(Ret, RetPE);
217 CASE(Gep, GepStmt);
218 CASE(Phi, PhiStmt);
219 CASE(Select, SelectStmt);
220 CASE(Cmp, CmpStmt);
221 CASE(BinaryOp, BinaryOPStmt);
222 CASE(UnaryOp, UnaryOPStmt);
223 CASE(Branch, BranchStmt);
224 CASE(ThreadFork, TDForkPE);
225 CASE(ThreadJoin, TDJoinPE);
226#undef CASE
227 }
228}
229
231{
232 switch (node->getNodeKind())
233 {
234 default:
235 assert(false && "Unknown ICFGNode kind");
236
237#define CASE(NodeKind, NodeType) \
238 case ICFGNode::NodeKind: \
239 return contentToJson(static_cast<const NodeType*>(node))
240
241 CASE(IntraBlock, IntraICFGNode);
242 CASE(FunEntryBlock, FunEntryICFGNode);
243 CASE(FunExitBlock, FunExitICFGNode);
244 CASE(FunCallBlock, CallICFGNode);
245 CASE(FunRetBlock, RetICFGNode);
246 CASE(GlobalBlock, GlobalICFGNode);
247#undef CASE
248 }
249}
250
252{
253 switch (edge->getEdgeKind())
254 {
255 default:
256 assert(false && "Unknown ICFGEdge kind");
258 return contentToJson(static_cast<const IntraCFGEdge*>(edge));
259 case ICFGEdge::CallCF:
260 return contentToJson(static_cast<const CallCFGEdge*>(edge));
261 case ICFGEdge::RetCF:
262 return contentToJson(static_cast<const RetCFGEdge*>(edge));
263 }
264}
265
267{
268 return contentToJson(node);
269}
270
275
277{
278 cJSON* root = genericNodeToJson(var);
279 JSON_WRITE_FIELD(root, var, value);
280 JSON_WRITE_FIELD(root, var, InEdgeKindToSetMap);
281 JSON_WRITE_FIELD(root, var, OutEdgeKindToSetMap);
282 JSON_WRITE_FIELD(root, var, isPtr);
283 return root;
284}
285
287{
288 return contentToJson(static_cast<const SVFVar*>(var));
289}
290
292{
293 cJSON* root = contentToJson(static_cast<const SVFVar*>(var));
294 return root;
295}
296
298{
299 cJSON* root = contentToJson(static_cast<const ValVar*>(var));
300 JSON_WRITE_FIELD(root, var, ap);
301 JSON_WRITE_FIELD(root, var, gepValType);
302 return root;
303}
304
306{
307 cJSON* root = contentToJson(static_cast<const ObjVar*>(var));
308 JSON_WRITE_FIELD(root, var, apOffset);
309 JSON_WRITE_FIELD(root, var, base);
310 return root;
311}
312
314{
315 return contentToJson(static_cast<const ObjVar*>(var));
316}
317
319{
320 return contentToJson(static_cast<const ValVar*>(var));
321}
322
324{
325 return contentToJson(static_cast<const ValVar*>(var));
326}
327
329{
330 return contentToJson(static_cast<const ValVar*>(var));
331}
332
334{
335 return contentToJson(static_cast<const ObjVar*>(var));
336}
337
339{
340 cJSON* root = genericNodeToJson(node);
341 JSON_WRITE_FIELD(root, node, fun);
342 JSON_WRITE_FIELD(root, node, bb);
343 // TODO: Ensure this?
344 assert(node->VFGNodes.empty() && "VFGNodes list not empty?");
345 JSON_WRITE_FIELD(root, node, pagEdges);
346 return root;
347}
348
350{
351 return contentToJson(static_cast<const ICFGNode*>(node));
352}
353
355{
356 cJSON* root = contentToJson(static_cast<const ICFGNode*>(node));
357 JSON_WRITE_FIELD(root, node, isRet);
358 return root;
359}
360
362{
363 return contentToJson(static_cast<const ICFGNode*>(node));
364}
365
367{
368 cJSON* root = contentToJson(static_cast<const ICFGNode*>(node));
369 JSON_WRITE_FIELD(root, node, FPNodes);
370 return root;
371}
372
374{
375 cJSON* root = contentToJson(static_cast<const ICFGNode*>(node));
376 JSON_WRITE_FIELD(root, node, formalRet);
377 return root;
378}
379
381{
382 cJSON* root = contentToJson(static_cast<const ICFGNode*>(node));
383 JSON_WRITE_FIELD(root, node, ret);
384 JSON_WRITE_FIELD(root, node, APNodes);
385 return root;
386}
387
389{
390 cJSON* root = contentToJson(static_cast<const ICFGNode*>(node));
391 JSON_WRITE_FIELD(root, node, actualRet);
392 JSON_WRITE_FIELD(root, node, callBlockNode);
393 return root;
394}
395
400
402{
403 cJSON* root = contentToJson(static_cast<const ICFGEdge*>(edge));
404 JSON_WRITE_FIELD(root, edge, conditionVar);
405 JSON_WRITE_FIELD(root, edge, branchCondVal);
406 return root;
407}
408
410{
411 cJSON* root = contentToJson(static_cast<const ICFGEdge*>(edge));
412 JSON_WRITE_FIELD(root, edge, callPEs);
413 return root;
414}
415
417{
418 cJSON* root = contentToJson(static_cast<const ICFGEdge*>(edge));
419 JSON_WRITE_FIELD(root, edge, retPE);
420 return root;
421}
422
424{
425 cJSON* root = genericNodeToJson(node);
426 JSON_WRITE_FIELD(root, node, vtable);
427 JSON_WRITE_FIELD(root, node, className);
428 JSON_WRITE_FIELD(root, node, flags);
429 JSON_WRITE_FIELD(root, node, virtualFunctionVectors);
430 return root;
431}
432
434{
436 JSON_WRITE_FIELD(root, edge, edgeType);
437 return root;
438}
439
441{
442 cJSON* root = jsonCreateObject();
443 JSON_WRITE_FIELD(root, type, kind);
444 JSON_WRITE_FIELD(root, type, isSingleValTy);
445 JSON_WRITE_FIELD(root, type, typeinfo);
446 return root;
447}
448
450{
451 cJSON* root = contentToJson(static_cast<const SVFType*>(type));
452 return root;
453}
454
456{
457 cJSON* root = contentToJson(static_cast<const SVFType*>(type));
458 JSON_WRITE_FIELD(root, type, signAndWidth);
459 return root;
460}
461
463{
464 cJSON* root = contentToJson(static_cast<const SVFType*>(type));
465 JSON_WRITE_FIELD(root, type, retTy);
466 return root;
467}
468
470{
471 cJSON* root = contentToJson(static_cast<const SVFType*>(type));
472 JSON_WRITE_FIELD(root, type, name);
473 return root;
474}
475
477{
478 cJSON* root = contentToJson(static_cast<const SVFType*>(type));
479 JSON_WRITE_FIELD(root, type, numOfElement);
480 JSON_WRITE_FIELD(root, type, typeOfElement);
481 return root;
482}
483
485{
486 cJSON* root = contentToJson(static_cast<const SVFType*>(type));
487 JSON_WRITE_FIELD(root, type, repr);
488 return root;
489}
490
492{
493 cJSON* root = jsonCreateObject();
494 JSON_WRITE_FIELD(root, value, kind);
495 JSON_WRITE_FIELD(root, value, type);
496 JSON_WRITE_FIELD(root, value, name);
497 JSON_WRITE_FIELD(root, value, ptrInUncalledFun);
498 JSON_WRITE_FIELD(root, value, constDataOrAggData);
499 JSON_WRITE_FIELD(root, value, sourceLoc);
500 return root;
501}
502
504{
505 cJSON* root = contentToJson(static_cast<const SVFValue*>(value));
506#define F(f) JSON_WRITE_FIELD(root, value, f);
507 F(isDecl);
508 F(intrinsic);
509 F(addrTaken);
510 F(isUncalled);
511 F(isNotRet);
512 F(varArg);
513 F(funcType);
514 F(loopAndDom);
515 F(realDefFun);
516 F(allBBs);
517 F(allArgs);
518#undef F
519 return root;
520}
521
523{
524 cJSON* root = contentToJson(static_cast<const SVFValue*>(value));
525 JSON_WRITE_FIELD(root, value, succBBs);
526 JSON_WRITE_FIELD(root, value, predBBs);
527 JSON_WRITE_FIELD(root, value, fun);
528 return root;
529}
530
532{
533 cJSON* root = contentToJson(static_cast<const SVFValue*>(value));
534 JSON_WRITE_FIELD(root, value, bb);
535 JSON_WRITE_FIELD(root, value, terminator);
536 JSON_WRITE_FIELD(root, value, ret);
537 return root;
538}
539
541{
542 cJSON* root = contentToJson(static_cast<const SVFInstruction*>(value));
543 JSON_WRITE_FIELD(root, value, args);
544 JSON_WRITE_FIELD(root, value, varArg);
545 JSON_WRITE_FIELD(root, value, calledVal);
546 return root;
547}
548
550{
551 return contentToJson(static_cast<const SVFValue*>(value));
552}
553
555{
556 cJSON* root = contentToJson(static_cast<const SVFConstant*>(value));
557 JSON_WRITE_FIELD(root, value, realDefGlobal);
558 return root;
559}
560
562{
563 cJSON* root = contentToJson(static_cast<const SVFValue*>(value));
564 JSON_WRITE_FIELD(root, value, fun);
565 JSON_WRITE_FIELD(root, value, argNo);
566 JSON_WRITE_FIELD(root, value, uncalled);
567 return root;
568}
569
571{
572 return contentToJson(static_cast<const SVFConstant*>(value));
573}
574
576{
577 cJSON* root = contentToJson(static_cast<const SVFConstantData*>(value));
578 JSON_WRITE_FIELD(root, value, zval);
579 JSON_WRITE_FIELD(root, value, sval);
580 return root;
581}
582
584{
585 cJSON* root = contentToJson(static_cast<const SVFConstantData*>(value));
586 JSON_WRITE_FIELD(root, value, dval);
587 return root;
588}
589
591{
592 return contentToJson(static_cast<const SVFConstantData*>(value));
593}
594
596{
597 return contentToJson(static_cast<const SVFConstantData*>(value));
598}
599
601{
602 return contentToJson(static_cast<const SVFValue*>(value));
603}
604
606{
607 return contentToJson(static_cast<const SVFOtherValue*>(value));
608}
609
611{
613 JSON_WRITE_FIELD(root, edge, value);
614 JSON_WRITE_FIELD(root, edge, basicBlock);
615 JSON_WRITE_FIELD(root, edge, icfgNode);
616 JSON_WRITE_FIELD(root, edge, edgeId);
617 return root;
618}
619
621{
622 cJSON* root = jsonCreateObject();
623#define F(field) JSON_WRITE_FIELD(root, loop, field)
624 F(entryICFGEdges);
625 F(backICFGEdges);
626 F(inICFGEdges);
627 F(outICFGEdges);
628 F(icfgNodes);
629 F(loopBound);
630#undef F
631 return root;
632}
633
635{
636 return contentToJson(static_cast<const SVFStmt*>(edge));
637}
638
640{
641 return contentToJson(static_cast<const AssignStmt*>(edge));
642}
643
645{
646 cJSON* root = contentToJson(static_cast<const AssignStmt*>(edge));
647 JSON_WRITE_FIELD(root, edge, copyKind);
648 return root;
649}
650
652{
653 return contentToJson(static_cast<const AssignStmt*>(edge));
654}
655
657{
658 return contentToJson(static_cast<const AssignStmt*>(edge));
659}
660
662{
663 cJSON* root = contentToJson(static_cast<const AssignStmt*>(edge));
664 JSON_WRITE_FIELD(root, edge, ap);
665 JSON_WRITE_FIELD(root, edge, variantField);
666 return root;
667}
668
670{
671 cJSON* root = contentToJson(static_cast<const AssignStmt*>(edge));
672 JSON_WRITE_FIELD(root, edge, call);
673 JSON_WRITE_FIELD(root, edge, entry);
674 return root;
675}
676
678{
679 cJSON* root = contentToJson(static_cast<const AssignStmt*>(edge));
680 JSON_WRITE_FIELD(root, edge, call);
681 JSON_WRITE_FIELD(root, edge, exit);
682 return root;
683}
684
686{
687 cJSON* root = contentToJson(static_cast<const SVFStmt*>(edge));
688 JSON_WRITE_FIELD(root, edge, opVars);
689 return root;
690}
691
693{
694 cJSON* root = contentToJson(static_cast<const MultiOpndStmt*>(edge));
695 JSON_WRITE_FIELD(root, edge, opICFGNodes);
696 return root;
697}
698
700{
701 cJSON* root = contentToJson(static_cast<const MultiOpndStmt*>(edge));
702 JSON_WRITE_FIELD(root, edge, condition);
703 return root;
704}
705
707{
708 cJSON* root = contentToJson(static_cast<const MultiOpndStmt*>(edge));
709 JSON_WRITE_FIELD(root, edge, predicate);
710 return root;
711}
712
714{
715 cJSON* root = contentToJson(static_cast<const MultiOpndStmt*>(edge));
716 JSON_WRITE_FIELD(root, edge, opcode);
717 return root;
718}
719
721{
722 cJSON* root = contentToJson(static_cast<const SVFStmt*>(edge));
723 JSON_WRITE_FIELD(root, edge, opcode);
724 return root;
725}
726
728{
729 cJSON* root = contentToJson(static_cast<const SVFStmt*>(edge));
730 JSON_WRITE_FIELD(root, edge, successors);
731 JSON_WRITE_FIELD(root, edge, cond);
732 JSON_WRITE_FIELD(root, edge, brInst);
733 return root;
734}
735
737{
738 return contentToJson(static_cast<const CallPE*>(edge));
739}
740
742{
743 return contentToJson(static_cast<const RetPE*>(edge));
744}
745
746bool jsonAddNumberToObject(cJSON* obj, const char* name, double number)
747{
749 return jsonAddItemToObject(obj, name, node);
750}
751
752bool jsonAddStringToObject(cJSON* obj, const char* name, const char* str)
753{
755 return jsonAddItemToObject(obj, name, node);
756}
757
758bool jsonAddStringToObject(cJSON* obj, const char* name, const std::string& s)
759{
760 return jsonAddStringToObject(obj, name, s.c_str());
761}
762
764{
765 return humanReadableOption()
767 : cJSON_IsNumber(item) &&
768 (item->valuedouble == 0 || item->valuedouble == 1);
769}
770
771bool jsonIsBool(const cJSON* item, bool& flag)
772{
774 {
775 if (!cJSON_IsBool(item))
776 return false;
778 return true;
779 }
780 else
781 {
782 if (!cJSON_IsNumber(item))
783 return false;
784 flag = item->valuedouble == 1;
785 return true;
786 }
787}
788
790{
791 return cJSON_IsNumber(item);
792}
793
795{
796 return cJSON_IsString(item);
797}
798
800{
801 // TODO: optimize
802 return cJSON_IsNull(item);
803}
804
806{
807 return cJSON_IsArray(item);
808}
809
810bool jsonIsMap(const cJSON* item)
811{
812 return cJSON_IsArray(item);
813}
814
819
820bool jsonKeyEquals(const cJSON* item, const char* key)
821{
822 return item && !(humanReadableOption() && std::strcmp(item->string, key));
823}
824
825std::pair<const cJSON*,const cJSON*> jsonUnpackPair(const cJSON* item)
826{
827 ABORT_IFNOT(jsonIsArray(item), "Expected array as pair");
829 ABORT_IFNOT(child1, "Missing first child of pair");
831 ABORT_IFNOT(child2, "Missing first child of pair");
832 ABORT_IFNOT(!child2->next, "Pair has more than two children");
833 return {child1, child2};
834}
835
837{
838 ABORT_IFNOT(jsonIsNumber(item), "Expected number for " << JSON_KEY(item));
839 return item->valuedouble;
840}
841
843{
844 // TODO: optimize
845 return cJSON_CreateNull();
846}
847
852
854{
855 return cJSON_CreateArray();
856}
857
859{
860 // Don't use cJSON_CreateObject() here, because it only allows string keys.
861 return cJSON_CreateArray();
862}
863
865{
867}
868
870{
871 constexpr size_t maxPreciseIntInDouble = (1ull << 53);
872 (void)maxPreciseIntInDouble; // silence unused warning
875}
876
882
884{
885 return cJSON_CreateNumber(num);
886}
887
896
902
907
909{
910 for (const auto& pair : icfg->getIcfgNodeToSVFLoopVec())
911 {
912 for (const SVFLoop* loop : pair.second)
913 {
914 svfLoopPool.saveID(loop);
915 }
916 }
917}
918
920{
921 // TODO: SVFType & StInfo are managed by SymbolTableInfo. Refactor it?
923
924 const auto& svfTypes = symTab->getSVFTypes();
925 svfTypePool.reserve(svfTypes.size());
926 for (const SVFType* type : svfTypes)
927 {
928 svfTypePool.saveID(type);
929 }
930
931 const auto& stInfos = symTab->getStInfos();
932 stInfoPool.reserve(stInfos.size());
933 for (const StInfo* stInfo : stInfos)
934 {
935 stInfoPool.saveID(stInfo);
936 }
937
938 svfValuePool.reserve(svfModule->getFunctionSet().size() +
939 svfModule->getConstantSet().size() +
940 svfModule->getOtherValueSet().size());
941}
942
944 : svfIR(svfir), svfModuleWriter(svfir->svfModule), icfgWriter(svfir->icfg),
945 chgWriter(SVFUtil::dyn_cast<CHGraph>(svfir->chgraph)),
946 irGraphWriter(svfir)
947{
948}
949
950void SVFIRWriter::writeJsonToOstream(const SVFIR* svfir, std::ostream& os)
951{
952 SVFIRWriter writer(svfir);
953 os << writer.generateJsonString().get() << '\n';
954}
955
956void SVFIRWriter::writeJsonToPath(const SVFIR* svfir, const std::string& path)
957{
958 std::ofstream jsonFile(path);
959 if (jsonFile.is_open())
960 {
962 jsonFile.close();
963 }
964 else
965 {
966 SVFUtil::errs() << "Failed to open file '" << path
967 << "' to write SVFIR's JSON\n";
968 }
969}
970
971const char* SVFIRWriter::numToStr(size_t n)
972{
973 auto it = numToStrMap.find(n);
974 if (it != numToStrMap.end() && !(numToStrMap.key_comp()(n, it->first)))
975 {
976 return it->second.c_str();
977 }
978 return numToStrMap.emplace_hint(it, n, std::to_string(n))->second.c_str();
979}
980
982{
983 autoJSON object = generateJson();
984 char* str = humanReadableOption() ? cJSON_Print(object.get())
985 : cJSON_PrintUnformatted(object.get());
986 return {str, cJSON_free};
987}
988
990{
991 const IRGraph* const irGraph = svfIR;
993 assert(nodeIDAllocator && "NodeIDAllocator is not initialized?");
994
995 cJSON* root = jsonCreateObject();
996#define F(field) JSON_WRITE_FIELD(root, svfIR, field)
997 F(svfModule);
998 F(symInfo);
999 F(icfg);
1000 F(chgraph);
1002 F(icfgNode2SVFStmtsMap);
1003 F(icfgNode2PTASVFStmtsMap);
1004 F(GepValObjMap);
1005 F(typeLocSetsMap);
1006 F(GepObjVarMap);
1007 F(memToFieldsMap);
1008 F(globSVFStmtSet);
1009 F(phiNodeMap);
1010 F(funArgsListMap);
1011 F(callSiteArgsListMap);
1012 F(callSiteRetMap);
1013 F(funRetMap);
1014 F(indCallSiteToFunPtrMap);
1015 F(funPtrToCallSitesMap);
1016 F(candidatePointers);
1017 F(callSiteSet);
1019#undef F
1020
1021 return {root, cJSON_Delete};
1022}
1023
1028
1030{
1032}
1033
1035{
1036 ENSURE_NOT_VISITED(graph);
1037
1039#define F(field) JSON_WRITE_FIELD(root, graph, field)
1040 F(KindToSVFStmtSetMap);
1041 F(KindToPTASVFStmtSetMap);
1042 F(fromFile);
1043 F(nodeNumAfterPAGBuild);
1044 F(totalPTAPAGEdge);
1045 F(valueToEdgeMap);
1046#undef F
1047 return root;
1048}
1049
1051{
1052 return var ? jsonCreateIndex(var->getId()) : jsonCreateNullId();
1053}
1054
1059
1061{
1062 cJSON* allSvfLoop = jsonCreateArray(); // all indices seen in constructor
1063 for (const SVFLoop* svfLoop : icfgWriter.svfLoopPool)
1064 {
1067 }
1068
1069#define F(field) JSON_WRITE_FIELD(root, icfg, field)
1071 jsonAddItemToObject(root, FIELD_NAME_ITEM(allSvfLoop)); // Meta field
1072 F(totalICFGNode);
1073 F(FunToFunEntryNodeMap);
1074 F(FunToFunExitNodeMap);
1075 F(globalBlockNode);
1076 F(icfgNodeToSVFLoopVec);
1077#undef F
1078 return root;
1079}
1080
1082{
1083 assert(node && "ICFGNode is null!");
1084 return jsonCreateIndex(node->getId());
1085}
1086
1088{
1089 assert(edge && "ICFGNode is null!");
1091}
1092
1094{
1095 auto chg = SVFUtil::dyn_cast<CHGraph>(graph);
1096 assert(chg && "Unsupported CHGraph type!");
1097 return toJson(chg);
1098}
1099
1101{
1103#define F(field) JSON_WRITE_FIELD(root, graph, field)
1104 // TODO: Ensure svfMod is the same as the SVFIR's?
1105 F(classNum);
1106 F(vfID);
1107 // F(buildingCHGTime); No need
1108 F(classNameToNodeMap);
1109 F(classNameToDescendantsMap);
1110 F(classNameToAncestorsMap);
1111 F(classNameToInstAndDescsMap);
1112 F(templateNameToInstancesMap);
1113 F(callNodeToClassesMap);
1114 F(virtualFunctionToIDMap);
1115 F(callNodeToCHAVtblsMap);
1116 F(callNodeToCHAVFnsMap);
1117#undef F
1118 return root;
1119}
1120
1122{
1123 return jsonCreateIndex(node->getId());
1124}
1125
1130
1135
1137{
1138 cJSON* root = jsonCreateObject();
1139#define F(field) JSON_WRITE_FIELD(root, stInfo, field)
1140 F(stride);
1141 F(numOfFlattenElements);
1142 F(numOfFlattenFields);
1143 F(fldIdxVec);
1144 F(elemIdxVec);
1145 F(fldIdx2TypeMap);
1146 F(finfo);
1147 F(flattenElementTypes);
1148#undef F
1149 return root;
1150}
1151
1153{
1155
1156 cJSON* root = jsonCreateObject();
1158 JSON_WRITE_FIELD(root, objTypeInfo, flags);
1159 JSON_WRITE_FIELD(root, objTypeInfo, maxOffsetLimit);
1160 JSON_WRITE_FIELD(root, objTypeInfo, elemNum);
1161 return root;
1162}
1163
1165{
1167
1168 cJSON* root = jsonCreateObject();
1169 JSON_WRITE_FIELD(root, ldInfo, reachableBBs);
1170 JSON_WRITE_FIELD(root, ldInfo, dtBBsMap);
1171 JSON_WRITE_FIELD(root, ldInfo, pdtBBsMap);
1172 JSON_WRITE_FIELD(root, ldInfo, dfBBsMap);
1173 JSON_WRITE_FIELD(root, ldInfo, bb2LoopMap);
1174 JSON_WRITE_FIELD(root, ldInfo, bb2PdomLevel);
1175 JSON_WRITE_FIELD(root, ldInfo, bb2PIdom);
1176 return root;
1177}
1178
1183
1185{
1186 cJSON* root = jsonCreateObject();
1187 JSON_WRITE_FIELD(root, &ap, fldIdx);
1188 JSON_WRITE_FIELD(root, &ap, idxOperandPairs);
1189 return root;
1190}
1191
1193{
1195
1196 cJSON* root = jsonCreateObject();
1197 JSON_WRITE_FIELD(root, nodeIDAllocator, strategy);
1198 JSON_WRITE_FIELD(root, nodeIDAllocator, numObjects);
1199 JSON_WRITE_FIELD(root, nodeIDAllocator, numValues);
1200 JSON_WRITE_FIELD(root, nodeIDAllocator, numSymbols);
1201 JSON_WRITE_FIELD(root, nodeIDAllocator, numNodes);
1202 return root;
1203}
1204
1206{
1208
1209 cJSON* root = jsonCreateObject();
1211
1212#define F(field) JSON_WRITE_FIELD(root, symTable, field)
1213 jsonAddItemToObject(root, FIELD_NAME_ITEM(allMemObj)); // Actual field
1214
1215 F(valSymMap);
1216 F(objSymMap);
1217 F(returnSymMap);
1218 F(varargSymMap);
1219 // Field objMap can be represented by allMemObj
1220 // Field mod can be represented by svfIR->svfModule. Delete it?
1221 assert(symTable->mod == svfIR->svfModule && "SVFModule mismatch!");
1222 F(modelConstants);
1223 F(totalSymNum);
1224 F(maxStruct);
1225 F(maxStSize);
1226 // Field svfTypes can be represented by svfModuleWriter.svfTypePool
1227 // Field stInfos can be represented by svfModuleWriter.stInfoPool
1228#undef F
1229
1230 return root;
1231}
1232
1234{
1235 cJSON* root = jsonCreateObject();
1239
1241 {
1244 }
1245
1247 {
1250 }
1251
1252#define F(field) JSON_WRITE_FIELD(root, module, field)
1253 jsonAddItemToObject(root, FIELD_NAME_ITEM(allSVFType)); // Meta field
1254 jsonAddItemToObject(root, FIELD_NAME_ITEM(allStInfo)); // Meta field
1255 jsonAddItemToObject(root, FIELD_NAME_ITEM(allSVFValue)); // Meta field
1256 F(pagReadFromTxt);
1257 F(moduleIdentifier);
1258
1259 F(FunctionSet);
1260 F(GlobalSet);
1261 F(AliasSet);
1262 F(ConstantSet);
1263 F(OtherValueSet);
1264#undef F
1265
1266 for (size_t i = 1; i <= svfModuleWriter.sizeSVFValuePool(); ++i)
1267 {
1270 }
1271
1272 return root;
1273}
1274
1276{
1277 const cJSON* svfirField = createObjs(root);
1278
1279 SVFIR* svfIR = SVFIR::getPAG(); // SVFIR constructor sets symInfo
1280 IRGraph* irGraph = svfIR;
1281
1282 auto svfModule = SVFModule::getSVFModule();
1283 auto icfg = new ICFG();
1284 auto chgraph = new CHGraph(svfModule);
1285 auto symInfo = SymbolTableInfo::SymbolInfo();
1286 symInfo->mod = svfModule;
1287
1288 svfIR->svfModule = svfModule;
1289 svfIR->icfg = icfg;
1290 svfIR->chgraph = chgraph;
1291
1292#define F(field) JSON_READ_FIELD_FWD(svfirField, svfIR, field)
1293 readJson(symInfo);
1295 readJson(icfg);
1296 readJson(chgraph);
1297 readJson(svfModule);
1298
1299 F(icfgNode2SVFStmtsMap);
1300 F(icfgNode2PTASVFStmtsMap);
1301 F(GepValObjMap);
1302 F(typeLocSetsMap);
1303 F(GepObjVarMap);
1304 F(memToFieldsMap);
1305 F(globSVFStmtSet);
1306 F(phiNodeMap);
1307 F(funArgsListMap);
1308 F(callSiteArgsListMap);
1309 F(callSiteRetMap);
1310 F(funRetMap);
1311 F(indCallSiteToFunPtrMap);
1312 F(funPtrToCallSitesMap);
1313 F(candidatePointers);
1314 F(callSiteSet);
1315#undef F
1316 assert(!NodeIDAllocator::allocator && "NodeIDAllocator should be NULL");
1319
1320 return svfIR;
1321}
1322
1324{
1325 return nullptr;
1326}
1327
1329{
1330 ABORT_IFNOT(jsonIsBool(obj, flag), "Expect bool for " << obj->string);
1331}
1332
1333void SVFIRReader::readJson(const cJSON* obj, unsigned& val)
1334{
1336}
1337
1339{
1341}
1342
1344{
1346}
1347
1349{
1351}
1352
1353void SVFIRReader::readJson(const cJSON* obj, unsigned long& val)
1354{
1356 [](const char* s)
1357 {
1358 return std::strtoul(s, nullptr, 10);
1359 });
1360}
1361
1362void SVFIRReader::readJson(const cJSON* obj, long long& val)
1363{
1365 [](const char* s)
1366 {
1367 return std::strtoll(s, nullptr, 10);
1368 });
1369}
1370
1371void SVFIRReader::readJson(const cJSON* obj, unsigned long long& val)
1372{
1374 [](const char* s)
1375 {
1376 return std::strtoull(s, nullptr, 10);
1377 });
1378}
1379
1380void SVFIRReader::readJson(const cJSON* obj, std::string& str)
1381{
1382 ABORT_IFNOT(jsonIsString(obj), "Expect string for " << obj->string);
1383 str = obj->valuestring;
1384}
1385
1387{
1388 switch (kind)
1389 {
1390 default:
1391 ABORT_MSG(kind << " is an impossible ICFGNodeKind in create()");
1392#define CASE(kind, constructor) \
1393 case ICFGNode::kind: \
1394 return new constructor(id);
1395 CASE(IntraBlock, IntraICFGNode);
1396 CASE(FunEntryBlock, FunEntryICFGNode);
1397 CASE(FunExitBlock, FunExitICFGNode);
1398 CASE(FunCallBlock, CallICFGNode);
1399 CASE(FunRetBlock, RetICFGNode);
1400 CASE(GlobalBlock, GlobalICFGNode);
1401#undef CASE
1402 }
1403}
1404
1406{
1407 constexpr ICFGNode* src = nullptr;
1408 constexpr ICFGNode* dst = nullptr;
1409
1410 switch (kind)
1411 {
1412 default:
1413 ABORT_MSG(kind << " is an impossible ICFGEdgeKind in create()");
1414 case ICFGEdge::IntraCF:
1415 return new IntraCFGEdge(src, dst);
1416 case ICFGEdge::CallCF:
1417 return new CallCFGEdge(src, dst);
1418 case ICFGEdge::RetCF:
1419 return new RetCFGEdge(src, dst);
1420 }
1421}
1422
1424{
1425 ABORT_IFNOT(kind == 0, "Impossible CHNode kind " << kind);
1426 return new CHNode("", id);
1427}
1428
1430{
1431 ABORT_IFNOT(kind == 0, "Unsupported CHEdge kind " << kind);
1432 return new CHEdge(nullptr, nullptr, {});
1433}
1434
1436{
1437 switch (kind)
1438 {
1439 default:
1440 ABORT_MSG(kind << " is an impossible SVFVarKind in create()");
1441#define CASE(kind, constructor) \
1442 case SVFVar::kind: \
1443 return new constructor(id);
1444 CASE(ValNode, ValVar);
1445 CASE(RetNode, RetPN);
1446 CASE(ObjNode, ObjVar);
1447 CASE(VarargNode, VarArgPN);
1448 CASE(GepValNode, GepValVar);
1449 CASE(GepObjNode, GepObjVar);
1450 CASE(BaseObjNode, BaseObjVar);
1451 CASE(DummyValNode, DummyValVar);
1452 CASE(DummyObjNode, DummyObjVar);
1453#undef CASE
1454 }
1455}
1456
1458{
1459 switch (kind)
1460 {
1461 default:
1462 ABORT_MSG(kind << " is an impossible SVFStmtKind in create()");
1463#define CASE(kind, constructor) \
1464 case SVFStmt::kind: \
1465 return new constructor;
1466 CASE(Addr, AddrStmt);
1467 CASE(Copy, CopyStmt);
1468 CASE(Store, StoreStmt);
1469 CASE(Load, LoadStmt);
1470 CASE(Call, CallPE);
1471 CASE(Ret, RetPE);
1472 CASE(Gep, GepStmt);
1473 CASE(Phi, PhiStmt);
1474 CASE(Select, SelectStmt);
1475 CASE(Cmp, CmpStmt);
1476 CASE(BinaryOp, BinaryOPStmt);
1477 CASE(UnaryOp, UnaryOPStmt);
1478 CASE(Branch, BranchStmt);
1479 CASE(ThreadFork, TDForkPE);
1480 CASE(ThreadJoin, TDJoinPE);
1481#undef CASE
1482 }
1483}
1484
1486{
1487 assert(idAllocator && "idAllocator should be nonempty");
1488
1489 ABORT_IFNOT(jsonIsObject(obj), "Expect object for " << JSON_KEY(obj));
1490 obj = obj->child;
1491
1492 JSON_DEF_READ_FWD(obj, int, strategy);
1493 static_assert(sizeof(idAllocator->strategy) == sizeof(strategy),
1494 "idAllocator->strategy should be represented by int");
1495 idAllocator->strategy = static_cast<NodeIDAllocator::Strategy>(strategy);
1496 JSON_READ_FIELD_FWD(obj, idAllocator, numObjects);
1497 JSON_READ_FIELD_FWD(obj, idAllocator, numValues);
1498 JSON_READ_FIELD_FWD(obj, idAllocator, numSymbols);
1500
1501 ABORT_IFNOT(!obj, "Extra field " << JSON_KEY(obj) << " in NodeIDAllocator");
1502}
1503
1505{
1507#define F(field) JSON_READ_FIELD_FWD(obj, symTabInfo, field)
1508 // `allMemObj` was consumed during create & fill phase.
1509 F(valSymMap);
1510 F(objSymMap);
1511 F(returnSymMap);
1512 F(varargSymMap);
1513 F(modelConstants);
1514 F(totalSymNum);
1515 F(maxStruct);
1516 F(maxStSize);
1517#undef F
1518 ABORT_IFNOT(!obj, "Extra field " << JSON_KEY(obj) << " in SymbolTableInfo");
1519}
1520
1522{
1523 assert(SymbolTableInfo::symInfo && "SymbolTableInfo should be nonempty");
1524 assert(graph->symInfo == SymbolTableInfo::SymbolInfo() && "symInfo differ");
1525
1526 auto& valToEdgeMap = graph->valueToEdgeMap;
1527 valToEdgeMap.clear();
1528
1531#define F(field) JSON_READ_FIELD_FWD(obj, graph, field)
1532 // base and symInfo have already been read
1533 F(KindToSVFStmtSetMap);
1534 F(KindToPTASVFStmtSetMap);
1535 F(fromFile);
1536 F(nodeNumAfterPAGBuild);
1537 F(totalPTAPAGEdge);
1538 F(valueToEdgeMap);
1539#undef F
1540
1541 auto nullit = valToEdgeMap.find(nullptr);
1542 ABORT_IFNOT(nullit != valToEdgeMap.end(), "valueToEdgeMap should has key NULL");
1543 ABORT_IFNOT(nullit->second.empty(), "valueToEdgeMap[NULL] should be empty");
1544}
1545
1547{
1549 const cJSON* obj = icfgReader.getFieldJson();
1550#define F(field) JSON_READ_FIELD_FWD(obj, icfg, field)
1551 F(totalICFGNode);
1552 F(FunToFunEntryNodeMap);
1553 F(FunToFunExitNodeMap);
1554 F(globalBlockNode);
1555 F(icfgNodeToSVFLoopVec);
1556#undef F
1557}
1558
1560{
1563#define F(field) JSON_READ_FIELD_FWD(obj, graph, field)
1564 F(classNum);
1565 F(vfID);
1566 F(classNameToNodeMap);
1567 F(classNameToDescendantsMap);
1568 F(classNameToAncestorsMap);
1569 F(classNameToInstAndDescsMap);
1570 F(templateNameToInstancesMap);
1571 F(callNodeToClassesMap);
1572 F(virtualFunctionToIDMap);
1573 F(callNodeToCHAVtblsMap);
1574 F(callNodeToCHAVFnsMap);
1575#undef F
1576}
1577
1579{
1581 auto symInfo = SymbolTableInfo::symInfo;
1582 assert(symInfo && "SymbolTableInfo should be non-NULL");
1583 svfModuleReader.svfTypePool.saveToSet(symInfo->svfTypes);
1584 svfModuleReader.stInfoPool.saveToSet(symInfo->stInfos);
1585
1586#define F(field) JSON_READ_FIELD_FWD(obj, module, field)
1587 F(pagReadFromTxt);
1588 F(moduleIdentifier);
1589 F(FunctionSet);
1590 F(GlobalSet);
1591 F(AliasSet);
1592 F(ConstantSet);
1593 F(OtherValueSet);
1594#undef F
1595}
1596
1598{
1599 assert(!type && "SVFType already read?");
1601}
1602
1604{
1605 assert(!stInfo && "StInfo already read?");
1607}
1608
1610{
1611 assert(!value && "SVFValue already read?");
1613}
1614
1616{
1617 assert(!var && "SVFVar already read?");
1618 if (jsonIsNullId(obj))
1619 var = nullptr;
1620 else
1622}
1623
1625{
1626 assert(!stmt && "SVFStmt already read?");
1628}
1629
1631{
1632 assert(!node && "ICFGNode already read?");
1633 NodeID id = jsonGetNumber(obj);
1634 node = icfgReader.getNodePtr(id);
1635}
1636
1638{
1639 assert(!edge && "ICFGEdge already read?");
1641}
1642
1644{
1645 assert(!node && "CHNode already read?");
1647}
1648
1650{
1651 assert(!edge && "CHEdge already read?");
1653}
1654
1656{
1657 ABORT_IFNOT(jsonIsObject(obj), "Expected obj for AccessPath");
1658 obj = obj->child;
1659 JSON_READ_FIELD_FWD(obj, &ap, fldIdx);
1660 JSON_READ_FIELD_FWD(obj, &ap, idxOperandPairs);
1661 ABORT_IFNOT(!obj, "Extra field " << JSON_KEY(obj) << " in AccessPath");
1662}
1663
1665{
1666 assert(!loop && "SVFLoop already read?");
1667 unsigned id = jsonGetNumber(obj);
1669}
1670
1671
1673{
1674 assert(!objTypeInfo && "ObjTypeInfo already read?");
1675 ABORT_IFNOT(jsonIsObject(obj), "Expected object for objTypeInfo");
1676 cJSON* field = obj->child;
1677
1679 JSON_DEF_READ_FWD(field, u32_t, flags);
1680 JSON_DEF_READ_FWD(field, u32_t, maxOffsetLimit);
1681 JSON_DEF_READ_FWD(field, u32_t, elemNum);
1682
1683 ABORT_IFNOT(!field, "Extra field in objTypeInfo: " << JSON_KEY(field));
1684 objTypeInfo = new ObjTypeInfo(type, maxOffsetLimit);
1685 objTypeInfo->flags = flags;
1686 objTypeInfo->elemNum = elemNum;
1687}
1688
1690{
1691 assert(!ldInfo && "SVFLoopAndDomInfo already read?");
1692 ABORT_IFNOT(jsonIsObject(obj), "Expected object for SVFLoopAndDomInfo");
1693 cJSON* field = obj->child;
1694
1695 ldInfo = new SVFLoopAndDomInfo();
1696
1697 JSON_READ_FIELD_FWD(field, ldInfo, reachableBBs);
1698 JSON_READ_FIELD_FWD(field, ldInfo, dtBBsMap);
1699 JSON_READ_FIELD_FWD(field, ldInfo, pdtBBsMap);
1700 JSON_READ_FIELD_FWD(field, ldInfo, dfBBsMap);
1701 JSON_READ_FIELD_FWD(field, ldInfo, bb2LoopMap);
1702 JSON_READ_FIELD_FWD(field, ldInfo, bb2PdomLevel);
1703 JSON_READ_FIELD_FWD(field, ldInfo, bb2PIdom);
1704
1706 "Extra field in SVFLoopAndDomInfo: " << JSON_KEY(field));
1707}
1708
1710{
1711 switch (var->getNodeKind())
1712 {
1713 default:
1714 assert(false && "Unknown SVFVar kind");
1715
1716#define CASE(VarKind, VarType) \
1717 case SVFVar::VarKind: \
1718 return fill(fieldJson, static_cast<VarType*>(var))
1719
1720 CASE(ValNode, ValVar);
1721 CASE(ObjNode, ObjVar);
1722 CASE(RetNode, RetPN);
1723 CASE(VarargNode, VarArgPN);
1724 CASE(GepValNode, GepValVar);
1725 CASE(GepObjNode, GepObjVar);
1726 CASE(BaseObjNode, BaseObjVar);
1727 CASE(DummyValNode, DummyValVar);
1728 CASE(DummyObjNode, DummyObjVar);
1729#undef CASE
1730 }
1731}
1732
1734{
1735 fill(fieldJson, static_cast<GenericPAGNodeTy*>(var));
1737 JSON_READ_FIELD_FWD(fieldJson, var, InEdgeKindToSetMap);
1738 JSON_READ_FIELD_FWD(fieldJson, var, OutEdgeKindToSetMap);
1740}
1741
1743{
1744 fill(fieldJson, static_cast<SVFVar*>(var));
1745}
1746
1748{
1749 fill(fieldJson, static_cast<SVFVar*>(var));
1750}
1751
1753{
1754 fill(fieldJson, static_cast<ValVar*>(var));
1756 JSON_READ_FIELD_FWD(fieldJson, var, gepValType);
1757}
1758
1760{
1761 fill(fieldJson, static_cast<ObjVar*>(var));
1762 JSON_READ_FIELD_FWD(fieldJson, var, apOffset);
1764}
1765
1767{
1768 fill(fieldJson, static_cast<ObjVar*>(var));
1769}
1770
1772{
1773 fill(fieldJson, static_cast<ValVar*>(var));
1774}
1775
1777{
1778 fill(fieldJson, static_cast<ValVar*>(var));
1779}
1780
1782{
1783 fill(fieldJson, static_cast<ValVar*>(var));
1784}
1785
1787{
1788 fill(fieldJson, static_cast<ObjVar*>(var));
1789}
1790
1792{
1793 auto kind = stmt->getEdgeKind();
1794
1795 switch (kind)
1796 {
1797 default:
1798 ABORT_MSG("Unknown SVFStmt kind " << kind);
1799
1800#define CASE(EdgeKind, EdgeType) \
1801 case SVFStmt::EdgeKind: \
1802 return fill(fieldJson, static_cast<EdgeType*>(stmt))
1803
1804 CASE(Addr, AddrStmt);
1805 CASE(Copy, CopyStmt);
1806 CASE(Store, StoreStmt);
1807 CASE(Load, LoadStmt);
1808 CASE(Call, CallPE);
1809 CASE(Ret, RetPE);
1810 CASE(Gep, GepStmt);
1811 CASE(Phi, PhiStmt);
1812 CASE(Select, SelectStmt);
1813 CASE(Cmp, CmpStmt);
1814 CASE(BinaryOp, BinaryOPStmt);
1815 CASE(UnaryOp, UnaryOPStmt);
1816 CASE(Branch, BranchStmt);
1817 CASE(ThreadFork, TDForkPE);
1818 CASE(ThreadJoin, TDJoinPE);
1819#undef CASE
1820 }
1821}
1822
1824{
1825 fill(fieldJson, static_cast<GenericPAGEdgeTy*>(stmt));
1827 JSON_READ_FIELD_FWD(fieldJson, stmt, basicBlock);
1830}
1831
1833{
1834 fill(fieldJson, static_cast<SVFStmt*>(stmt));
1835}
1836
1838{
1839 fill(fieldJson, static_cast<AssignStmt*>(stmt));
1840}
1841
1843{
1844 fill(fieldJson, static_cast<AssignStmt*>(stmt));
1846}
1847
1849{
1850 fill(fieldJson, static_cast<AssignStmt*>(stmt));
1851}
1852
1854{
1855 fill(fieldJson, static_cast<AssignStmt*>(stmt));
1856}
1857
1859{
1860 fill(fieldJson, static_cast<AssignStmt*>(stmt));
1862 JSON_READ_FIELD_FWD(fieldJson, stmt, variantField);
1863}
1864
1866{
1867 fill(fieldJson, static_cast<AssignStmt*>(stmt));
1870}
1871
1873{
1874 fill(fieldJson, static_cast<AssignStmt*>(stmt));
1877}
1878
1880{
1881 fill(fieldJson, static_cast<SVFStmt*>(stmt));
1883}
1884
1886{
1887 fill(fieldJson, static_cast<MultiOpndStmt*>(stmt));
1888 JSON_READ_FIELD_FWD(fieldJson, stmt, opICFGNodes);
1889}
1890
1892{
1893 fill(fieldJson, static_cast<MultiOpndStmt*>(stmt));
1894 JSON_READ_FIELD_FWD(fieldJson, stmt, condition);
1895}
1896
1898{
1899 fill(fieldJson, static_cast<MultiOpndStmt*>(stmt));
1900 JSON_READ_FIELD_FWD(fieldJson, stmt, predicate);
1901}
1902
1904{
1905 fill(fieldJson, static_cast<MultiOpndStmt*>(stmt));
1907}
1908
1910{
1911 fill(fieldJson, static_cast<SVFStmt*>(stmt));
1913}
1914
1916{
1917 fill(fieldJson, static_cast<SVFStmt*>(stmt));
1918 JSON_READ_FIELD_FWD(fieldJson, stmt, successors);
1921}
1922
1924{
1925 fill(fieldJson, static_cast<CallPE*>(stmt));
1926}
1927
1929{
1930 fill(fieldJson, static_cast<RetPE*>(stmt));
1931}
1932
1934{
1935#define F(field) JSON_READ_FIELD_FWD(fieldJson, stInfo, field)
1936 // stride has already been read upon construction
1937 F(numOfFlattenElements);
1938 F(numOfFlattenFields);
1939 F(fldIdxVec);
1940 F(elemIdxVec);
1941 F(fldIdx2TypeMap);
1942 F(finfo);
1943 F(flattenElementTypes);
1944#undef F
1945}
1946
1948{
1949 switch (node->getNodeKind())
1950 {
1951 default:
1952 ABORT_MSG("Unknown ICFGNode kind " << node->getNodeKind());
1953
1954#define CASE(NodeKind, NodeType) \
1955 case ICFGNode::NodeKind: \
1956 return fill(fieldJson, static_cast<NodeType*>(node))
1957
1958 CASE(IntraBlock, IntraICFGNode);
1959 CASE(FunEntryBlock, FunEntryICFGNode);
1960 CASE(FunExitBlock, FunExitICFGNode);
1961 CASE(FunCallBlock, CallICFGNode);
1962 CASE(FunRetBlock, RetICFGNode);
1963 CASE(GlobalBlock, GlobalICFGNode);
1964#undef CASE
1965 }
1966}
1967
1969{
1970 fill(fieldJson, static_cast<GenericICFGNodeTy*>(node));
1971 JSON_READ_FIELD_FWD(fieldJson, node, fun);
1972 JSON_READ_FIELD_FWD(fieldJson, node, bb);
1973 // Skip VFGNodes as it is empty
1974 JSON_READ_FIELD_FWD(fieldJson, node, pagEdges);
1975}
1976
1978{
1979 fill(fieldJson, static_cast<ICFGNode*>(node));
1980}
1981
1983{
1984 fill(fieldJson, static_cast<ICFGNode*>(node));
1985 JSON_READ_FIELD_FWD(fieldJson, node, isRet);
1986}
1987
1989{
1990 fill(fieldJson, static_cast<ICFGNode*>(node));
1991}
1992
1994{
1995 fill(fieldJson, static_cast<ICFGNode*>(node));
1996 JSON_READ_FIELD_FWD(fieldJson, node, FPNodes);
1997}
1998
2000{
2001 fill(fieldJson, static_cast<ICFGNode*>(node));
2002 JSON_READ_FIELD_FWD(fieldJson, node, formalRet);
2003}
2004
2006{
2007 fill(fieldJson, static_cast<ICFGNode*>(node));
2008 JSON_READ_FIELD_FWD(fieldJson, node, ret);
2009 JSON_READ_FIELD_FWD(fieldJson, node, APNodes);
2010}
2011
2013{
2014 fill(fieldJson, static_cast<ICFGNode*>(node));
2015 JSON_READ_FIELD_FWD(fieldJson, node, actualRet);
2016 JSON_READ_FIELD_FWD(fieldJson, node, callBlockNode);
2017}
2018
2020{
2021 auto kind = edge->getEdgeKind();
2022 switch (kind)
2023 {
2024 default:
2025 ABORT_MSG("Unknown ICFGEdge kind " << kind);
2026 case ICFGEdge::IntraCF:
2027 return fill(fieldJson, static_cast<IntraCFGEdge*>(edge));
2028 case ICFGEdge::CallCF:
2029 return fill(fieldJson, static_cast<CallCFGEdge*>(edge));
2030 case ICFGEdge::RetCF:
2031 return fill(fieldJson, static_cast<RetCFGEdge*>(edge));
2032 }
2033}
2034
2036{
2037 fill(fieldJson, static_cast<GenericICFGEdgeTy*>(edge));
2038}
2039
2041{
2042 fill(fieldJson, static_cast<ICFGEdge*>(edge));
2043 JSON_READ_FIELD_FWD(fieldJson, edge, conditionVar);
2044 JSON_READ_FIELD_FWD(fieldJson, edge, branchCondVal);
2045}
2046
2048{
2049 fill(fieldJson, static_cast<ICFGEdge*>(edge));
2051}
2052
2054{
2055 fill(fieldJson, static_cast<ICFGEdge*>(edge));
2057}
2058
2060{
2061#define F(field) JSON_READ_FIELD_FWD(fieldJson, loop, field)
2062 F(entryICFGEdges);
2063 F(backICFGEdges);
2064 F(inICFGEdges);
2065 F(outICFGEdges);
2066 F(icfgNodes);
2067 F(loopBound);
2068#undef F
2069}
2070
2072{
2073 assert(node->getNodeKind() == 0 && "Unknown CHNode kind");
2074 fill(fieldJson, static_cast<GenericCHNodeTy*>(node));
2075 JSON_READ_FIELD_FWD(fieldJson, node, vtable);
2076 JSON_READ_FIELD_FWD(fieldJson, node, className);
2077 JSON_READ_FIELD_FWD(fieldJson, node, flags);
2078 JSON_READ_FIELD_FWD(fieldJson, node, virtualFunctionVectors);
2079}
2080
2082{
2083 assert(edge->getEdgeKind() == 0 && "Unknown CHEdge kind");
2084 fill(fieldJson, static_cast<GenericCHEdgeTy*>(edge));
2085 // edgeType is a enum
2086 JSON_DEF_READ_FWD(fieldJson, unsigned, edgeType);
2087 if (edgeType == CHEdge::INHERITANCE)
2088 edge->edgeType = CHEdge::INHERITANCE;
2089 else if (edgeType == CHEdge::INSTANTCE)
2090 edge->edgeType = CHEdge::INSTANTCE;
2091 else
2092 ABORT_MSG("Unknown CHEdge type " << edgeType);
2093}
2094
2096{
2097 auto kind = value->getKind();
2098
2099 switch (kind)
2100 {
2101 default:
2102 ABORT_MSG("Impossible SVFValue kind " << kind);
2103
2104#define CASE(ValueKind, Type) \
2105 case SVFValue::ValueKind: \
2106 return fill(fieldJson, static_cast<Type*>(value))
2107
2108 CASE(SVFVal, SVFValue);
2109 CASE(SVFFunc, SVFFunction);
2110 CASE(SVFBB, SVFBasicBlock);
2111 CASE(SVFInst, SVFInstruction);
2112 CASE(SVFCall, SVFCallInst);
2113 CASE(SVFGlob, SVFGlobalValue);
2114 CASE(SVFArg, SVFArgument);
2115 CASE(SVFConst, SVFConstant);
2116 CASE(SVFConstData, SVFConstantData);
2117 CASE(SVFConstInt, SVFConstantInt);
2118 CASE(SVFConstFP, SVFConstantFP);
2119 CASE(SVFNullPtr, SVFConstantNullPtr);
2120 CASE(SVFBlackHole, SVFBlackHoleValue);
2121 CASE(SVFMetaAsValue, SVFMetadataAsValue);
2122 CASE(SVFOther, SVFOtherValue);
2123#undef CASE
2124 }
2125}
2126
2128{
2129 // kind, type, name have already been read.
2130 JSON_READ_FIELD_FWD(fieldJson, value, ptrInUncalledFun);
2131 JSON_READ_FIELD_FWD(fieldJson, value, constDataOrAggData);
2132 JSON_READ_FIELD_FWD(fieldJson, value, sourceLoc);
2133}
2134
2136{
2137 fill(fieldJson, static_cast<SVFValue*>(value));
2138#define F(f) JSON_READ_FIELD_FWD(fieldJson, value, f)
2139 F(isDecl);
2140 F(intrinsic);
2141 F(addrTaken);
2142 F(isUncalled);
2143 F(isNotRet);
2144 F(varArg);
2145 F(funcType);
2146 F(loopAndDom);
2147 F(realDefFun);
2148 F(allBBs);
2149 F(allArgs);
2150#undef F
2151}
2152
2154{
2155 fill(fieldJson, static_cast<SVFValue*>(value));
2156 JSON_READ_FIELD_FWD(fieldJson, value, succBBs);
2157 JSON_READ_FIELD_FWD(fieldJson, value, predBBs);
2158 JSON_READ_FIELD_FWD(fieldJson, value, fun);
2159}
2160
2162{
2163 fill(fieldJson, static_cast<SVFValue*>(value));
2164 JSON_READ_FIELD_FWD(fieldJson, value, bb);
2165 JSON_READ_FIELD_FWD(fieldJson, value, terminator);
2166 JSON_READ_FIELD_FWD(fieldJson, value, ret);
2167}
2168
2170{
2171 fill(fieldJson, static_cast<SVFInstruction*>(value));
2172 JSON_READ_FIELD_FWD(fieldJson, value, args);
2173 JSON_READ_FIELD_FWD(fieldJson, value, varArg);
2174 JSON_READ_FIELD_FWD(fieldJson, value, calledVal);
2175}
2176
2178{
2179 fill(fieldJson, static_cast<SVFValue*>(value));
2180}
2181
2183{
2184 fill(fieldJson, static_cast<SVFConstant*>(value));
2185 JSON_READ_FIELD_FWD(fieldJson, value, realDefGlobal);
2186}
2187
2189{
2190 fill(fieldJson, static_cast<SVFValue*>(value));
2191 JSON_READ_FIELD_FWD(fieldJson, value, fun);
2192 JSON_READ_FIELD_FWD(fieldJson, value, argNo);
2193 JSON_READ_FIELD_FWD(fieldJson, value, uncalled);
2194}
2195
2197{
2198 fill(fieldJson, static_cast<SVFConstant*>(value));
2199}
2200
2202{
2203 fill(fieldJson, static_cast<SVFConstantData*>(value));
2204 JSON_READ_FIELD_FWD(fieldJson, value, zval);
2205 JSON_READ_FIELD_FWD(fieldJson, value, sval);
2206}
2207
2209{
2210 fill(fieldJson, static_cast<SVFConstantData*>(value));
2211 JSON_READ_FIELD_FWD(fieldJson, value, dval);
2212}
2213
2215{
2216 fill(fieldJson, static_cast<SVFConstantData*>(value));
2217}
2218
2220{
2221 fill(fieldJson, static_cast<SVFConstantData*>(value));
2222}
2223
2225{
2226 fill(fieldJson, static_cast<SVFValue*>(value));
2227}
2228
2230{
2231 fill(fieldJson, static_cast<SVFOtherValue*>(value));
2232}
2233
2235{
2236 auto kind = type->getKind();
2237
2238 switch (kind)
2239 {
2240 default:
2241 assert(false && "Impossible SVFType kind");
2242
2243#define CASE(Kind) \
2244 case SVFType::Kind: \
2245 return fill(fieldJson, SVFUtil::dyn_cast<Kind##pe>(type))
2246
2247 CASE(SVFTy);
2248 CASE(SVFPointerTy);
2249 CASE(SVFIntegerTy);
2250 CASE(SVFFunctionTy);
2251 CASE(SVFStructTy);
2252 CASE(SVFArrayTy);
2253 CASE(SVFOtherTy);
2254#undef CASE
2255 }
2256}
2257
2259{
2260 // kind has already been read
2262}
2263
2265{
2266 fill(fieldJson, static_cast<SVFType*>(type));
2267}
2268
2270{
2271 fill(fieldJson, static_cast<SVFType*>(type));
2272 JSON_READ_FIELD_FWD(fieldJson, type, signAndWidth);
2273}
2274
2276{
2277 fill(fieldJson, static_cast<SVFType*>(type));
2279}
2280
2286
2288{
2289 fill(fieldJson, static_cast<SVFType*>(type));
2290 JSON_READ_FIELD_FWD(fieldJson, type, numOfElement);
2291 JSON_READ_FIELD_FWD(fieldJson, type, typeOfElement);
2292}
2293
2295{
2296 fill(fieldJson, static_cast<SVFType*>(type));
2298}
2299
2300SVFIR* SVFIRReader::read(const std::string& path)
2301{
2302 struct stat buf;
2303 int fd = open(path.c_str(), O_RDONLY);
2304 if (fd == -1)
2305 {
2306 std::string info = "open(\"" + path + "\")";
2307 perror(info.c_str());
2308 abort();
2309 }
2310 if (fstat(fd, &buf) == -1)
2311 {
2312 std::string info = "fstate(\"" + path + "\")";
2313 perror(info.c_str());
2314 abort();
2315 }
2316 auto addr =
2317 (char*)mmap(nullptr, buf.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
2318 if (addr == MAP_FAILED)
2319 {
2320 std::string info = "mmap(content of \"" + path + "\")";
2321 perror(info.c_str());
2322 abort();
2323 }
2324
2325 auto root = cJSON_ParseWithLength(addr, buf.st_size);
2326
2327 if (munmap(addr, buf.st_size) == -1)
2328 perror("munmap()");
2329
2330 if (close(fd) < 0)
2331 perror("close()");
2332
2334 SVFIR* ir = reader.read(root);
2335
2336 cJSON_Delete(root);
2337 return ir;
2338}
2339
2340} // namespace SVF
#define ABORT_MSG(msg)
#define ABORT_IFNOT(condition, msg)
#define CASE(Kind)
#define F(f)
static const Option< bool > humanReadableOption("human-readable", "Whether to output human-readable JSON", true)
#define JSON_READ_OBJ_FWD(json, obj)
#define FIELD_NAME_ITEM(field)
#define JSON_DEF_READ_FWD(json, type, obj,...)
#define JSON_READ_FIELD_FWD(json, objptr, field)
#define ENSURE_NOT_VISITED(graph)
#define JSON_WRITE_FIELD(root, objptr, field)
#define JSON_KEY(json)
newitem type
Definition cJSON.cpp:2739
cJSON * n
Definition cJSON.cpp:2558
cJSON_Delete(null)
const char *const name
Definition cJSON.h:264
const char *const const double number
Definition cJSON.h:268
int index
Definition cJSON.h:170
cJSON * item
Definition cJSON.h:222
Class representing a function argument variable in the SVFIR.
@ INHERITANCE
Definition CHG.h:87
@ INSTANTCE
Definition CHG.h:88
Common base for class hierarchy graph. Only implements what PointerAnalysis needs.
Definition CHG.h:52
EdgeTy * getEdgePtr(unsigned id) const
void saveToGenericGraph(GenericGraph< NodeTy, EdgeTy > *graph) const
NodeTy * getNodePtr(unsigned id) const
const cJSON * getFieldJson() const
size_t getEdgeID(const EdgeType *edge)
WriterPtrPool< EdgeType > edgePool
Class representing a heap object variable in the SVFIR.
VFGNodeList VFGNodes
Definition ICFGNode.h:150
SVFLoop * getSVFLoopPtr(size_t id) const
WriterPtrPool< SVFLoop > svfLoopPool
size_t getSvfLoopID(const SVFLoop *loop)
ICFGWriter(const ICFG *icfg)
const ICFGNodeToSVFLoopVec & getIcfgNodeToSVFLoopVec() const
Definition ICFG.h:137
ValueToEdgeMap valueToEdgeMap
Map SVFValues (e.g., ICFGNodes) to all corresponding PAGEdges.
Definition IRGraph.h:64
SymbolTableInfo * symInfo
Definition IRGraph.h:65
static NodeIDAllocator * get(void)
Return (singleton) allocator.
static NodeIDAllocator * allocator
Single allocator.
Strategy
Allocation strategy to use.
GNodeK getNodeKind() const
Get node kind.
NodeID getId() const
Get ID.
IRGraphReader irGraphReader
SVFModuleReader svfModuleReader
static CHNode * createCHNode(NodeID id, GNodeK kind)
static ICFGEdge * createICFGEdge(GEdgeKind kind)
void fill(const cJSON *&fieldJson, SVFVar *var)
GenericEdge< void >::GEdgeKind GEdgeKind
static void readJson(const cJSON *obj, bool &flag)
static ICFGNode * createICFGNode(NodeID id, GNodeK type)
static CHEdge * createCHEdge(GEdgeKind kind)
static SVFStmt * createPAGEdge(GEdgeKind kind)
CHGraphReader chGraphReader
ICFGReader icfgReader
static SVFIR * read(const std::string &path)
const cJSON * createObjs(const cJSON *root)
static SVFVar * createPAGNode(NodeID id, GNodeK kind)
void virtFill(const cJSON *&fieldJson, SVFVar *var)
SymbolTableInfoReader symTableReader
const char * numToStr(size_t n)
static void writeJsonToOstream(const SVFIR *svfir, std::ostream &os)
std::unique_ptr< char, decltype(&cJSON_free)> autoCStr
static void writeJsonToPath(const SVFIR *svfir, const std::string &path)
const SVFIR * svfIR
std::unique_ptr< cJSON, decltype(&cJSON_Delete)> autoJSON
cJSON * genericGraphToJson(const GenericGraph< NodeTy, EdgeTy > *graph, const std::vector< const EdgeTy * > &edgePool)
OrderedMap< size_t, std::string > numToStrMap
cJSON * genericNodeToJson(const GenericNode< NodeTy, EdgeTy > *node)
autoJSON generateJson()
Main logic to dump a SVFIR to a JSON object.
cJSON * toJson(const NodeIDAllocator *nodeIDAllocator)
IRGraphWriter irGraphWriter
SVFIRWriter(const SVFIR *svfir)
Constructor.
SVFModuleWriter svfModuleWriter
cJSON * contentToJson(const SVFVar *var)
CHGraphWriter chgWriter
autoCStr generateJsonString()
ICFGWriter icfgWriter
bool jsonAddJsonableToObject(cJSON *obj, const char *name, const T &item)
cJSON * virtToJson(const SVFType *type)
Parameter types of these functions are all pointers. When they are used as arguments of toJson(),...
cJSON * genericEdgeToJson(const GenericEdge< NodeTy > *edge)
CommonCHGraph * chgraph
Definition SVFIR.h:100
SVFModule * svfModule
Definition SVFIR.h:98
ICFG * icfg
SVF Module.
Definition SVFIR.h:99
static SVFIR * getPAG(bool buildFromFile=false)
Singleton design here to make sure we only have one instance during any analysis.
Definition SVFIR.h:116
SVFType * getSVFTypePtr(size_t id) const
const cJSON * getFieldJson() const
StInfo * getStInfoPtr(size_t id) const
SVFValue * getSVFValuePtr(size_t id) const
ReaderPtrPool< StInfo > stInfoPool
ReaderPtrPool< SVFType > svfTypePool
size_t getSVFValueID(const SVFValue *value)
size_t sizeSVFValuePool() const
SVFModuleWriter(const SVFModule *svfModule)
WriterPtrPool< SVFValue > svfValuePool
size_t getStInfoID(const StInfo *stInfo)
size_t getSVFTypeID(const SVFType *type)
WriterPtrPool< StInfo > stInfoPool
WriterPtrPool< SVFType > svfTypePool
const SVFValue * getSVFValuePtr(size_t id) const
const OtherValueType & getOtherValueSet() const
Definition SVFModule.h:215
const FunctionSetType & getFunctionSet() const
Definition SVFModule.h:199
static SVFModule * getSVFModule()
Definition SVFModule.cpp:60
const ConstantType & getConstantSet() const
Definition SVFModule.h:203
s64_t GNodeK
Definition SVFType.h:163
GNodeK getKind() const
Get the type of this SVFValue.
Definition SVFValue.h:238
Represents a stack-allocated object variable in the SVFIR (SVF Intermediate Representation) @inherits...
const cJSON * getFieldJson() const
static SymbolTableInfo * SymbolInfo()
Singleton design here to make sure we only have one instance during any analysis.
static SymbolTableInfo * symInfo
const std::vector< const T * > & getPool() const
std::ostream & errs()
Overwrite llvm::errs()
Definition SVFUtil.h:56
for isBitcode
Definition BasicTypes.h:68
bool jsonAddNumberToObject(cJSON *obj, const char *name, double number)
Helper function to write a number to a JSON object.
cJSON * jsonCreateNullId()
double jsonGetNumber(const cJSON *item)
static void readSmallNumber(const cJSON *obj, SmallNumberType &val)
cJSON * jsonCreateNumber(double num)
bool jsonKeyEquals(const cJSON *item, const char *key)
bool jsonIsNumber(const cJSON *item)
bool jsonIsNullId(const cJSON *item)
u32_t NodeID
Definition GeneralType.h:55
cJSON * jsonCreateMap()
bool jsonAddPairToMap(cJSON *obj, cJSON *key, cJSON *value)
bool jsonIsMap(const cJSON *item)
cJSON * jsonCreateBool(bool flag)
bool jsonIsArray(const cJSON *item)
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
bool jsonAddItemToArray(cJSON *array, cJSON *item)
bool jsonAddStringToObject(cJSON *obj, const char *name, const char *str)
std::pair< const cJSON *, const cJSON * > jsonUnpackPair(const cJSON *item)
bool jsonIsObject(const cJSON *item)
static void readBigNumber(const cJSON *obj, BigNumberType &val, CStrToVal conv)
cJSON * jsonCreateObject()
bool jsonAddItemToObject(cJSON *obj, const char *name, cJSON *item)
unsigned u32_t
Definition GeneralType.h:46
bool jsonIsBool(const cJSON *item)
cJSON * jsonCreateString(const char *str)
bool jsonIsString(const cJSON *item)
cJSON * jsonCreateIndex(size_t index)
cJSON * jsonCreateArray()
SVFType * createSVFType(SVFType::GNodeK kind, bool isSingleValTy)
Definition cJSON.h:104
struct cJSON * child
Definition cJSON.h:109
double valuedouble
Definition cJSON.h:119
char * string
Definition cJSON.h:122
struct cJSON * next
Definition cJSON.h:106