12 "human-readable",
"Whether to output human-readable JSON",
true);
22 ABORT_MSG(kind <<
" is an impossible SVFTyKind in create()");
24 ABORT_MSG(
"Creation of RAW SVFType isn't allowed");
26 ABORT_IFNOT(isSingleValTy,
"Pointer type must be single-valued");
29 ABORT_IFNOT(isSingleValTy,
"Integer type must be single-valued");
32 ABORT_IFNOT(!isSingleValTy,
"Function type must be multi-valued");
35 ABORT_IFNOT(!isSingleValTy,
"Struct type must be multi-valued");
38 ABORT_IFNOT(!isSingleValTy,
"Array type must be multi-valued");
45template <
typename SmallNumberType>
51template <
typename BigNumberType,
typename CStrToVal>
55 "Expect (number) string JSON for " <<
JSON_KEY(
obj));
104 auto kind =
type->getKind();
109 assert(
false &&
"Impossible SVFType kind");
112 case SVFType::Kind: \
113 return contentToJson(static_cast<const Kind##pe*>(type))
133 assert(
false &&
"Impossible SVFValue kind");
135#define CASE(ValueKind, type) \
136 case SVFValue::ValueKind: \
137 return contentToJson(static_cast<const type*>(value))
160 switch (
var->getNodeKind())
163 assert(
false &&
"Unknown SVFVar kind");
165#define CASE(VarKind, VarType) \
166 case SVFVar::VarKind: \
167 return contentToJson(static_cast<const VarType*>(var))
202 switch (
stmt->getEdgeKind())
205 assert(
false &&
"Unknown SVFStmt kind");
207#define CASE(EdgeKind, EdgeType) \
208 case SVFStmt::EdgeKind: \
209 return contentToJson(static_cast<const EdgeType*>(stmt))
235 assert(
false &&
"Unknown ICFGNode kind");
237#define CASE(NodeKind, NodeType) \
238 case ICFGNode::NodeKind: \
239 return contentToJson(static_cast<const NodeType*>(node))
253 switch (
edge->getEdgeKind())
256 assert(
false &&
"Unknown ICFGEdge kind");
506#define F(f) JSON_WRITE_FIELD(root, value, f);
623#define F(field) JSON_WRITE_FIELD(root, loop, field)
924 const auto& svfTypes =
symTab->getSVFTypes();
931 const auto& stInfos =
symTab->getStInfos();
944 : svfIR(svfir), svfModuleWriter(svfir->svfModule), icfgWriter(svfir->icfg),
945 chgWriter(SVFUtil::dyn_cast<
CHGraph>(svfir->chgraph)),
953 os <<
writer.generateJsonString().get() <<
'\n';
967 <<
"' to write SVFIR's JSON\n";
976 return it->second.c_str();
978 return numToStrMap.emplace_hint(
it,
n, std::to_string(
n))->second.c_str();
996#define F(field) JSON_WRITE_FIELD(root, svfIR, field)
1002 F(icfgNode2SVFStmtsMap);
1003 F(icfgNode2PTASVFStmtsMap);
1011 F(callSiteArgsListMap);
1014 F(indCallSiteToFunPtrMap);
1015 F(funPtrToCallSitesMap);
1016 F(candidatePointers);
1039#define F(field) JSON_WRITE_FIELD(root, graph, field)
1040 F(KindToSVFStmtSetMap);
1041 F(KindToPTASVFStmtSetMap);
1043 F(nodeNumAfterPAGBuild);
1069#define F(field) JSON_WRITE_FIELD(root, icfg, field)
1073 F(FunToFunEntryNodeMap);
1074 F(FunToFunExitNodeMap);
1076 F(icfgNodeToSVFLoopVec);
1083 assert(node &&
"ICFGNode is null!");
1095 auto chg = SVFUtil::dyn_cast<CHGraph>(graph);
1096 assert(chg &&
"Unsupported CHGraph type!");
1103#define F(field) JSON_WRITE_FIELD(root, graph, field)
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);
1139#define F(field) JSON_WRITE_FIELD(root, stInfo, field)
1141 F(numOfFlattenElements);
1142 F(numOfFlattenFields);
1147 F(flattenElementTypes);
1212#define F(field) JSON_WRITE_FIELD(root, symTable, field)
1252#define F(field) JSON_WRITE_FIELD(root, module, field)
1257 F(moduleIdentifier);
1283 auto icfg =
new ICFG();
1284 auto chgraph =
new CHGraph(svfModule);
1286 symInfo->mod = svfModule;
1292#define F(field) JSON_READ_FIELD_FWD(svfirField, svfIR, field)
1299 F(icfgNode2SVFStmtsMap);
1300 F(icfgNode2PTASVFStmtsMap);
1308 F(callSiteArgsListMap);
1311 F(indCallSiteToFunPtrMap);
1312 F(funPtrToCallSitesMap);
1313 F(candidatePointers);
1358 return std::strtoul(
s,
nullptr, 10);
1367 return std::strtoll(
s,
nullptr, 10);
1376 return std::strtoull(
s,
nullptr, 10);
1391 ABORT_MSG(kind <<
" is an impossible ICFGNodeKind in create()");
1392#define CASE(kind, constructor) \
1393 case ICFGNode::kind: \
1394 return new constructor(id);
1413 ABORT_MSG(kind <<
" is an impossible ICFGEdgeKind in create()");
1425 ABORT_IFNOT(kind == 0,
"Impossible CHNode kind " << kind);
1426 return new CHNode(
"",
id);
1431 ABORT_IFNOT(kind == 0,
"Unsupported CHEdge kind " << kind);
1432 return new CHEdge(
nullptr,
nullptr, {});
1440 ABORT_MSG(kind <<
" is an impossible SVFVarKind in create()");
1441#define CASE(kind, constructor) \
1442 case SVFVar::kind: \
1443 return new constructor(id);
1462 ABORT_MSG(kind <<
" is an impossible SVFStmtKind in create()");
1463#define CASE(kind, constructor) \
1464 case SVFStmt::kind: \
1465 return new constructor;
1493 static_assert(
sizeof(
idAllocator->strategy) ==
sizeof(strategy),
1494 "idAllocator->strategy should be represented by int");
1507#define F(field) JSON_READ_FIELD_FWD(obj, symTabInfo, field)
1531#define F(field) JSON_READ_FIELD_FWD(obj, graph, field)
1533 F(KindToSVFStmtSetMap);
1534 F(KindToPTASVFStmtSetMap);
1536 F(nodeNumAfterPAGBuild);
1550#define F(field) JSON_READ_FIELD_FWD(obj, icfg, field)
1552 F(FunToFunEntryNodeMap);
1553 F(FunToFunExitNodeMap);
1555 F(icfgNodeToSVFLoopVec);
1563#define F(field) JSON_READ_FIELD_FWD(obj, graph, field)
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);
1582 assert(symInfo &&
"SymbolTableInfo should be non-NULL");
1586#define F(field) JSON_READ_FIELD_FWD(obj, module, field)
1588 F(moduleIdentifier);
1611 assert(!value &&
"SVFValue already read?");
1632 assert(!node &&
"ICFGNode already read?");
1645 assert(!node &&
"CHNode already read?");
1711 switch (
var->getNodeKind())
1714 assert(
false &&
"Unknown SVFVar kind");
1716#define CASE(VarKind, VarType) \
1717 case SVFVar::VarKind: \
1718 return fill(fieldJson, static_cast<VarType*>(var))
1793 auto kind =
stmt->getEdgeKind();
1798 ABORT_MSG(
"Unknown SVFStmt kind " << kind);
1800#define CASE(EdgeKind, EdgeType) \
1801 case SVFStmt::EdgeKind: \
1802 return fill(fieldJson, static_cast<EdgeType*>(stmt))
1935#define F(field) JSON_READ_FIELD_FWD(fieldJson, stInfo, field)
1937 F(numOfFlattenElements);
1938 F(numOfFlattenFields);
1943 F(flattenElementTypes);
1954#define CASE(NodeKind, NodeType) \
1955 case ICFGNode::NodeKind: \
1956 return fill(fieldJson, static_cast<NodeType*>(node))
2021 auto kind =
edge->getEdgeKind();
2025 ABORT_MSG(
"Unknown ICFGEdge kind " << kind);
2061#define F(field) JSON_READ_FIELD_FWD(fieldJson, loop, field)
2083 assert(
edge->getEdgeKind() == 0 &&
"Unknown CHEdge kind");
2092 ABORT_MSG(
"Unknown CHEdge type " << edgeType);
2102 ABORT_MSG(
"Impossible SVFValue kind " << kind);
2104#define CASE(ValueKind, Type) \
2105 case SVFValue::ValueKind: \
2106 return fill(fieldJson, static_cast<Type*>(value))
2138#define F(f) JSON_READ_FIELD_FWD(fieldJson, value, f)
2236 auto kind =
type->getKind();
2241 assert(
false &&
"Impossible SVFType kind");
2244 case SVFType::Kind: \
2245 return fill(fieldJson, SVFUtil::dyn_cast<Kind##pe>(type))
2250 CASE(SVFFunctionTy);
2306 std::string
info =
"open(\"" +
path +
"\")";
2312 std::string
info =
"fstate(\"" +
path +
"\")";
2320 std::string
info =
"mmap(content of \"" +
path +
"\")";
#define ABORT_IFNOT(condition, msg)
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)
const char *const const double number
Class representing a function argument variable in the SVFIR.
Common base for class hierarchy graph. Only implements what PointerAnalysis needs.
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.
SVFLoop * getSVFLoopPtr(size_t id) const
WriterPtrPool< SVFLoop > svfLoopPool
size_t getSvfLoopID(const SVFLoop *loop)
ICFGWriter(const ICFG *icfg)
const ICFGNodeToSVFLoopVec & getIcfgNodeToSVFLoopVec() const
ValueToEdgeMap valueToEdgeMap
Map SVFValues (e.g., ICFGNodes) to all corresponding PAGEdges.
SymbolTableInfo * symInfo
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
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)
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)
autoCStr generateJsonString()
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)
static SVFIR * getPAG(bool buildFromFile=false)
Singleton design here to make sure we only have one instance during any analysis.
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
const FunctionSetType & getFunctionSet() const
static SVFModule * getSVFModule()
const ConstantType & getConstantSet() const
GNodeK getKind() const
Get the type of this SVFValue.
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()
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)
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
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)
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)