31 case dwarf::DW_TAG_array_type:
35 case dwarf::DW_TAG_class_type:
36 case dwarf::DW_TAG_structure_type:
50 if (
fm->getTag() == dwarf::DW_TAG_member && !
fm->isStaticMember())
70 case dwarf::DW_TAG_union_type:
89 case dwarf::DW_TAG_enumeration_type:
93 assert(
false &&
"DCHGraph::buildCHG: unexpected CompositeType tag.");
101 case dwarf::DW_TAG_inheritance:
103 assert(SVFUtil::isa<DIType>(
derivedType->getScope()) &&
"inheriting from non-type?");
110 case dwarf::DW_TAG_member:
111 case dwarf::DW_TAG_friend:
114 case dwarf::DW_TAG_typedef:
117 case dwarf::DW_TAG_pointer_type:
118 case dwarf::DW_TAG_ptr_to_member_type:
119 case dwarf::DW_TAG_reference_type:
120 case dwarf::DW_TAG_rvalue_reference_type:
123 case dwarf::DW_TAG_const_type:
124 case dwarf::DW_TAG_atomic_type:
125 case dwarf::DW_TAG_volatile_type:
126 case dwarf::DW_TAG_restrict_type:
129 assert(
false &&
"DCHGraph::buildCHG: unexpected DerivedType tag.");
144 std::vector<const DIDerivedType *> typedefs;
170 for (Module::const_global_iterator
gvI =
M.global_begin();
gvI !=
M.global_end(); ++
gvI)
174 if (
gv ==
nullptr)
continue;
178 assert(
type &&
"DCHG::buildVTables: bad metadata for ctir.vt");
190 assert(
vtbl &&
"Element of vtbl struct not an array");
198 for (
unsigned cN = 0;
cN <
vtbl->getNumOperands(); ++
cN)
201 if (SVFUtil::isa<ConstantPointerNull>(
c))
208 assert(
ce &&
"non-ConstantExpr, non-ConstantPointerNull in vtable?");
209 if (
ce->getOpcode() == Instruction::BitCast)
212 const Function*
vfn = SVFUtil::dyn_cast<Function>(
ce->getOperand(0));
273 assert(
type &&
"DCHG::flatten: canon type of struct/class is not struct/class");
284 && (
type->getTag() == dwarf::DW_TAG_class_type
285 ||
type->getTag() == dwarf::DW_TAG_structure_type)
286 &&
"DCHG::flatten: expected a class/struct");
289 std::vector<const DIDerivedType *>
fields;
303 { return a->getOffsetInBits() < b->getOffsetInBits(); });
307 assert((
mt->getTag() == dwarf::DW_TAG_member ||
mt->getTag() == dwarf::DW_TAG_inheritance)
308 &&
"DCHG: expected member/inheritance");
311 if (
fieldType->getTag() == dwarf::DW_TAG_structure_type
312 ||
fieldType->getTag() == dwarf::DW_TAG_class_type)
322 else if (
fieldType->getTag() == dwarf::DW_TAG_array_type)
349 if (t ==
nullptr)
return false;
350 return t->getTag() == dwarf::DW_TAG_array_type
351 || t->getTag() == dwarf::DW_TAG_structure_type
352 || t->getTag() == dwarf::DW_TAG_class_type;
366 if (
type->getTag() == dwarf::DW_TAG_array_type)
385 for (
unsigned i = 0;
i <
fields.size(); ++
i)
429 assert(
md !=
nullptr &&
"Missing type metadata at virtual callsite");
430 DIType *diType = SVFUtil::dyn_cast<DIType>(
md);
431 assert(diType !=
nullptr &&
"Incorrect metadata type at virtual callsite");
461 if (node == dst && edgeType ==
et)
463 assert(SVFUtil::isa<DCHEdge>(
edge) &&
"Non-DCHEdge in DCHNode edge set.");
498 if (
basicType->getEncoding() == dwarf::DW_ATE_unsigned_char
499 ||
basicType->getEncoding() == dwarf::DW_ATE_signed_char)
520 assert(
false &&
"DCHGraph::buildCHG: unexpected DIType.");
535 if (
nodeI->second->getDIType() !=
nullptr
536 &&
nodeI->second->getOutEdges().size() == 0)
629 const std::string
suffix(
"[abi:cxx11]");
690 assert(
f &&
b &&
"DCHG::isFieldOf: given nullptr!");
694 if (
f ==
b)
return true;
696 if (
b->getTag() == dwarf::DW_TAG_array_type ||
b->getTag() == dwarf::DW_TAG_pointer_type)
703 else if (
const DIDerivedType *ptrType = SVFUtil::dyn_cast<DIDerivedType>(
b))
707 assert(
baseType &&
"DCHG::isFieldOf: baseType is neither DIComposite nor DIDerived!");
712 else if (
b->getTag() == dwarf::DW_TAG_class_type
713 ||
b->getTag() == dwarf::DW_TAG_structure_type)
771 || SVFUtil::isa<DIBasicType>(t)
772 || SVFUtil::isa<DISubroutineType>(t))
777 unsigned tag = t->getTag();
779 if (
tag == dwarf::DW_TAG_const_type
780 ||
tag == dwarf::DW_TAG_atomic_type
781 ||
tag == dwarf::DW_TAG_volatile_type
782 ||
tag == dwarf::DW_TAG_restrict_type
783 ||
tag == dwarf::DW_TAG_typedef)
787 assert(t &&
"DCHG: expected DerivedType");
788 t =
dt->getBaseType();
790 else if (
tag == dwarf::DW_TAG_array_type
791 ||
tag == dwarf::DW_TAG_class_type
792 ||
tag == dwarf::DW_TAG_structure_type
793 ||
tag == dwarf::DW_TAG_union_type
794 ||
tag == dwarf::DW_TAG_enumeration_type
795 ||
tag == dwarf::DW_TAG_member
796 ||
tag == dwarf::DW_TAG_pointer_type
797 ||
tag == dwarf::DW_TAG_ptr_to_member_type
798 ||
tag == dwarf::DW_TAG_reference_type
799 ||
tag == dwarf::DW_TAG_rvalue_reference_type)
804 else if (
tag == dwarf::DW_TAG_inheritance
805 ||
tag == dwarf::DW_TAG_friend)
807 assert(
false &&
"DCHG: unexpected tag when stripping qualifiers");
811 assert(
false &&
"DCHG: unhandled tag when stripping qualifiers");
821 if (t->getTag() == dwarf::DW_TAG_array_type)
841 if (
t1 ==
nullptr ||
t2 ==
nullptr)
849 if (SVFUtil::isa<DIBasicType>(
t1) && SVFUtil::isa<DIBasicType>(
t2))
854 unsigned enc1 =
b1->getEncoding();
855 unsigned enc2 =
b2->getEncoding();
856 bool okayEnc = ((
enc1 == dwarf::DW_ATE_signed ||
enc1 == dwarf::DW_ATE_unsigned ||
enc1 == dwarf::DW_ATE_boolean)
857 && (
enc2 == dwarf::DW_ATE_signed ||
enc2 == dwarf::DW_ATE_unsigned ||
enc2 == dwarf::DW_ATE_boolean))
859 (
enc1 == dwarf::DW_ATE_float &&
enc2 == dwarf::DW_ATE_float)
861 ((
enc1 == dwarf::DW_ATE_signed_char ||
enc1 == dwarf::DW_ATE_unsigned_char)
863 (
enc2 == dwarf::DW_ATE_signed_char ||
enc2 == dwarf::DW_ATE_unsigned_char));
868 return t1->getSizeInBits() ==
t2->getSizeInBits()
869 &&
t1->getAlignInBits() ==
t2->getAlignInBits();
875 if ((SVFUtil::isa<DIDerivedType>(
t1) ||
t1->getTag() == dwarf::DW_TAG_array_type)
876 && (SVFUtil::isa<DIDerivedType>(
t2) ||
t2->getTag() == dwarf::DW_TAG_array_type))
888 assert(
c1 &&
"teq: bad cast for array type");
900 assert(
c2 &&
"teq: bad cast for array type");
909 if (SVFUtil::isa<DICompositeType>(
t1) && SVFUtil::isa<DICompositeType>(
t2))
914 if (
ct1->getTag() !=
ct2->getTag())
return false;
917 if (
ct1->getTag() == dwarf::DW_TAG_enumeration_type)
923 if (
ct1->getTag() == dwarf::DW_TAG_class_type)
925 return ct1->getIdentifier() ==
ct2->getIdentifier();
931 return ct1->getName() ==
ct2->getName()
932 &&
ct1->getFile() ==
ct2->getFile()
933 &&
ct1->getLine() ==
ct2->getLine();
951 if (
f ==
b)
return true;
954 assert(node &&
"DCHG::isFirstField: node not found");
961 if (
edge->getSrcNode()->getDIType() ==
b)
return true;
971 std::stringstream
ss;
978 if (
const DIBasicType *
bt = SVFUtil::dyn_cast<DIBasicType>(t))
980 ss <<
bt->getName().str();
982 else if (
const DIDerivedType *
dt = SVFUtil::dyn_cast<DIDerivedType>(t))
984 if (
dt->getName() ==
"__vtbl_ptr_type")
986 ss <<
"(vtbl * =) __vtbl_ptr_type";
988 else if (
dt->getTag() == dwarf::DW_TAG_const_type)
992 else if (
dt->getTag() == dwarf::DW_TAG_volatile_type)
996 else if (
dt->getTag() == dwarf::DW_TAG_restrict_type)
1000 else if (
dt->getTag() == dwarf::DW_TAG_atomic_type)
1004 else if (
dt->getTag() == dwarf::DW_TAG_pointer_type)
1008 else if (
dt->getTag() == dwarf::DW_TAG_ptr_to_member_type)
1011 <<
" " <<
diTypeToStr(SVFUtil::dyn_cast<DIType>(
dt->getExtraData())) <<
"::*";
1013 else if (
dt->getTag() == dwarf::DW_TAG_reference_type)
1017 else if (
dt->getTag() == dwarf::DW_TAG_rvalue_reference_type)
1021 else if (
dt->getTag() == dwarf::DW_TAG_typedef)
1028 if (
ct->getTag() == dwarf::DW_TAG_class_type
1029 ||
ct->getTag() == dwarf::DW_TAG_structure_type
1030 ||
ct->getTag() == dwarf::DW_TAG_union_type)
1033 if (
ct->getTag() == dwarf::DW_TAG_class_type)
1037 else if (
ct->getTag() == dwarf::DW_TAG_structure_type)
1041 else if (
ct->getTag() == dwarf::DW_TAG_union_type)
1048 if (
ct->getName() !=
"")
1050 ss <<
ct->getName().str();
1058 for (
unsigned i = 0;
i <
fields.size(); ++
i)
1064 ss <<
sp->getName().str();
1068 assert(
mt->getTag() == dwarf::DW_TAG_member &&
"DCHG: expected member");
1081 else if (
ct->getTag() == dwarf::DW_TAG_array_type)
1085 for (
unsigned i = 0;
i <
sizes.size(); ++
i)
1088 assert(
sr !=
nullptr &&
"DCHG: non-subrange as array element?");
1098 else if (
ct->getTag() == dwarf::DW_TAG_enumeration_type)
1102 else if (
ct->getTag() == dwarf::DW_TAG_union_type)
1112 if (
types.size() == 1)
1118 for (
unsigned i = 1;
i <
types.size(); ++
i)
1121 if (
i + 1 !=
types.size())
1131 ss <<
st->getName().str();
1139 return std::string(
n,
' ');
1144 static const std::string
line =
"-------------------------------------\n";
1145 static const std::string
thickLine =
"=====================================\n";
1160 if (*
nodes.begin() != id)
1169 && (node->
getDIType()->getTag() == dwarf::DW_TAG_class_type
1171 node->
getDIType()->getTag() == dwarf::DW_TAG_structure_type))
1181 const std::vector<std::vector<const Function* >> &vfnVectors = node->
getVfnVectors();
1182 for (
unsigned i = 0;
i < vfnVectors.size(); ++
i)
1186 for (
unsigned j = 0;
j < vfnVectors[
i].size(); ++
j)
1190 <<
dname.className <<
"::" <<
dname.funcName <<
"\n";
1197 if (vfnVectors.size() == 0)
1211 arrow =
"--inheritance-->";
1215 arrow =
"--first-field-->";
1219 arrow =
"---instance---->";
1223 arrow =
"---standard---->";
1227 arrow =
"----unknown---->";
1249 if (
tdef !=
nullptr)
1257 if (typedefs.size() == 0)
static std::string indent(size_t n)
s32_t getFunIdxInVtable() const
const std::string & getFunNameOfVirtualCall() const
const DIType * getCSStaticType(CallBase *cs) const
Retrieves the metadata associated with a virtual callsite.
const DIType * getCanonicalType(const DIType *t)
const NodeBS & cha(const DIType *type, bool firstField)
Returns a set of all children of type (CHA). Also gradually builds chaMap.
static const DIType * stripQualifiers(const DIType *)
Returns the DIType beneath the qualifiers. Does not strip away "DW_TAG_members".
void handleDIDerivedType(const DIDerivedType *derivedType)
Construction helper to process DIDerivedTypes.
DCHEdge * addEdge(const DIType *t1, const DIType *t2, DCHEdge::GEdgeKind et)
Creates an edge between from t1 to t2.
bool isFirstField(const DIType *f, const DIType *b)
static std::string diTypeToStr(const DIType *)
Returns a human-readable version of the DIType.
void buildVTables()
Finds all defined virtual functions and attaches them to nodes.
void handleDICompositeType(const DICompositeType *compositeType)
Construction helper to process DICompositeTypes.
Map< const CallICFGNode *, VFunSet > csCHAMap
Maps callsites to a set of potential virtual functions based on CHA.
bool hasNode(const DIType *type)
Checks if a node exists for type.
Map< const DIType *, NodeBS > chaFFMap
Maps types to all children but also considering first field.
DCHNode * getNode(const DIType *type)
Returns the node for type (nullptr if it doesn't exist).
Map< const DIType *, const DIType * > canonicalTypeMap
Maps types to their canonical type (many-to-one).
void handleDISubroutineType(const DISubroutineType *subroutineType)
Construction helper to process DISubroutineTypes.
virtual void getVFnsFromVtbls(const CallICFGNode *cs, const VTableSet &vtbls, VFunSet &virtualFunctions) override
static const DIType * stripArray(const DIType *)
Returns the DIType beneath all qualifiers and arrays.
void handleTypedef(const DIType *typedefType)
Attaches the typedef(s) to the base node.
virtual void buildCHG(bool extend)
DCHEdge * hasEdge(const DIType *t1, const DIType *t2, DCHEdge::GEdgeKind et)
Returns the edge between t1 and t2 if it exists, returns nullptr otherwise.
NodeID numTypes
Number of types (nodes) in the graph.
void handleDIBasicType(const DIBasicType *basicType)
Construction helper to process DIBasicTypes.
virtual const VTableSet & getCSVtblsBasedonCHA(const CallICFGNode *cs) override
void gatherAggs(const DICompositeType *type)
Populates containingAggs for type and all its elements.
static bool isAgg(const DIType *t)
DCHNode * getOrCreateNode(const DIType *type)
Creates a node from type, or returns it if it exists.
Map< const DIType *, VTableSet > vtblCHAMap
Maps types to a set with their vtable and all their children's.
bool extended
Whether this CHG is an extended CHG (first-field). Set by buildCHG.
const std::vector< const DIType * > & getFieldTypes(const DIType *base)
Returns a vector of the types of all fields in base.
Map< const DIType *, Set< const DIType * > > containingAggs
Maps aggregate types to all the aggregate types it transitively contains.
virtual bool isBase(const DIType *a, const DIType *b, bool firstField)
void flatten(const DICompositeType *type)
Populates fieldTypes for type and all its elements.
Map< const GlobalObjVar *, const DIType * > vtblToTypeMap
Maps VTables to the DIType associated with them.
virtual const VFunSet & getCSVFsBasedonCHA(const CallICFGNode *cs) override
static bool teq(const DIType *t1, const DIType *t2)
Map< const DIType *, std::vector< const DIType * > > fieldTypes
Maps types to their flattened fields' types.
Map< const DIType *, NodeBS > chaMap
Maps types to all children (i.e. CHA).
Map< const DIType *, DCHNode * > diTypeToNodeMap
Maps DITypes to their nodes.
virtual bool isFieldOf(const DIType *f, const DIType *b)
Returns true if f is a field of b (fields from getFieldTypes).
Set< const DIType * > canonicalTypes
Set of all possible canonical types (i.e. values of canonicalTypeMap).
const std::vector< std::vector< const Function * > > & getVfnVectors(void) const
Returns the vector of virtual function vectors.
std::vector< const Function * > & getVfnVector(unsigned n)
Return the nth virtual function vector in the vtable.
const DIType * getDIType(void) const
const Set< const DIDerivedType * > & getTypedefs(void) const
void setVTable(const GlobalObjVar *vtbl)
void addGNode(NodeID id, NodeType *node)
Add a Node.
iterator begin()
Iterators.
IDToNodeMapTy::const_iterator const_iterator
IDToNodeMapTy::iterator iterator
Node Iterators.
NodeType * getGNode(NodeID id) const
Get a node.
const GEdgeSetTy & getOutEdges() const
const GEdgeSetTy & getInEdges() const
bool addIncomingEdge(EdgeType *inEdge)
Add incoming and outgoing edges.
bool addOutgoingEdge(EdgeType *outEdge)
static LLVMModuleSet * getLLVMModuleSet()
NodeID getObjectNode(const Value *V)
const std::vector< std::reference_wrapper< Module > > & getLLVMModules() const
static const Option< bool > PrintDCHG
static SVFIR * getPAG(bool buildFromFile=false)
Singleton design here to make sure we only have one instance during any analysis.
NodeID getId() const
Get ID.
std::pair< s64_t, u64_t > getIntegerValue(const ConstantInt *intValue)
const FunObjVar * getFunObjVar(const std::string &name)
std::ostream & outs()
Overwrite llvm::outs()
const std::string derefMDName
const std::string vtMDName
struct DemangledName demangle(const std::string &name)
const ConstantStruct * getVtblStruct(const GlobalValue *vtbl)
llvm::DISubroutineType DISubroutineType
llvm::GlobalVariable GlobalVariable
llvm::DINodeArray DINodeArray
llvm::ConstantStruct ConstantStruct
Set< const GlobalObjVar * > VTableSet
llvm::ConstantArray ConstantArray
llvm::DebugInfoFinder DebugInfoFinder
llvm::DIDerivedType DIDerivedType
llvm::DISubprogram DISubprogram
llvm::ConstantExpr ConstantExpr
llvm::IRBuilder IRBuilder
llvm::DISubrange DISubrange
llvm::DITypeRefArray DITypeRefArray
iter_range< typename GenericGraphTraits< GraphType >::nodes_iterator > nodes(const GraphType &G)
iter_range< typename GenericGraphTraits< GraphType >::ChildIteratorType > children(const typename GenericGraphTraits< GraphType >::NodeRef &G)
llvm::DIBasicType DIBasicType
llvm::DICompositeType DICompositeType
llvm::ConstantInt ConstantInt
Set< const FunObjVar * > VFunSet