22#include <clang/AST/ExprConcepts.h>
23#include <clang/Basic/FileManager.h>
24#include <clang/Lex/Preprocessor.h>
25#include <spdlog/spdlog.h>
33 , template_builder_{
diagram, config, *this}
38 const clang::NamedDecl *decl)
const
40 auto cls = std::make_unique<class_>(
config().using_namespace());
47 assert(ns !=
nullptr);
52 if (ns->isAnonymousNamespace() || ns->isInline())
55 LOG_DBG(
"= Visiting namespace declaration {} at {}",
56 ns->getQualifiedNameAsString(),
61 auto package_parent = package_path;
63 (package_path.size() == 1) && !
config().using_namespace().is_empty();
64 package_path.is_root(is_root);
67 if (!package_path.is_empty())
68 name = package_path.name();
70 if (!package_parent.is_empty())
71 package_parent.pop_back();
73 const auto usn =
config().using_namespace();
75 auto p = std::make_unique<common::model::package>(usn);
76 package_path = package_path.relative_to(usn);
79 p->set_namespace(package_parent);
89 p->set_style(p->style_spec());
91 for (
const auto *attr : ns->attrs()) {
92 if (attr->getKind() == clang::attr::Kind::Deprecated) {
93 p->set_deprecated(
true);
99 diagram().add(package_path, std::move(p));
114 LOG_DBG(
"= Visiting typedef enum declaration {} at {}",
115 enm->getQualifiedNameAsString(),
123 if (enm->isComplete()) {
136 assert(enm !=
nullptr);
140 if (enm->getNameAsString().empty()) {
147 LOG_DBG(
"= Visiting enum declaration {} at {}",
148 enm->getQualifiedNameAsString(),
158std::unique_ptr<clanguml::class_diagram::model::enum_>
160 const clang::EnumDecl *enm,
const clang::TypedefDecl *typedef_decl)
162 auto e_ptr = std::make_unique<enum_>(
config().using_namespace());
168 std::optional<eid_t> parent_id_opt;
169 [[maybe_unused]] namespace_ parent_ns;
172 std::string enm_name;
173 if (enm->getNameAsString().empty() && typedef_decl !=
nullptr)
174 enm_name = typedef_decl->getNameAsString();
175 else if (parent_id_opt)
176 enm_name = enm->getNameAsString();
180 if (parent_id_opt &&
diagram().find<class_>(*parent_id_opt)) {
184 e.set_name(parent_class.value().name(), enm_name);
186 e.add_relationship({relationship_t::kContainment, *parent_id_opt});
189 else if (parent_id_opt &&
diagram().find<objc_interface>(*parent_id_opt)) {
193 e.set_name(parent_class.value().name(), enm_name);
195 e.add_relationship({relationship_t::kContainment, *parent_id_opt});
199 e.set_name(enm_name);
213 e.set_style(e.style_spec());
221 for (
const auto &ev : enm.enumerators()) {
222 e.
constants().push_back(ev->getNameAsString());
229 clang::ClassTemplateSpecializationDecl *cls)
234 LOG_DBG(
"= Visiting template specialization declaration {} at {} "
235 "(described class id {})",
236 cls->getQualifiedNameAsString(),
238 cls->getSpecializedTemplate()
239 ? cls->getSpecializedTemplate()->getTemplatedDecl()->getID()
243 if (cls->isLocalClass() !=
nullptr)
248 if (!template_specialization_ptr)
251 return add_or_update(cls, std::move(template_specialization_ptr));
255 clang::TypeAliasTemplateDecl *cls)
260 LOG_DBG(
"= Visiting template type alias declaration {} at {}",
261 cls->getQualifiedNameAsString(),
264 const auto *template_type_specialization_ptr =
265 cls->getTemplatedDecl()
266 ->getUnderlyingType()
267 ->getAs<clang::TemplateSpecializationType>();
269 if (template_type_specialization_ptr ==
nullptr)
272 auto template_specialization_ptr =
273 std::make_unique<class_>(
config().using_namespace());
275 *template_specialization_ptr, cls, *template_type_specialization_ptr);
277 template_specialization_ptr->is_template(
true);
280 const auto name = template_specialization_ptr->full_name(
true);
281 const auto id = template_specialization_ptr->id();
283 LOG_DBG(
"Adding class {} with id {}", name,
id);
288 add_class(std::move(template_specialization_ptr));
295 clang::ClassTemplateDecl *cls)
300 LOG_DBG(
"= Visiting class template declaration {} at {}",
301 cls->getQualifiedNameAsString(),
315 const auto cls_full_name = c_ptr->full_name(
false);
319 c_ptr->is_template(
true);
323 constexpr auto kMaxConstraintCount = 24U;
324 llvm::SmallVector<const clang::Expr *, kMaxConstraintCount> constraints{};
325 if (cls->hasAssociatedConstraints()) {
326 cls->getAssociatedConstraints(constraints);
329 for (
const auto *expr : constraints) {
333 return add_or_update(cls->getTemplatedDecl(), std::move(c_ptr));
341 if (clang::dyn_cast_or_null<clang::CXXRecordDecl>(rec) !=
nullptr)
349 LOG_DBG(
"= Visiting record declaration {} at {}",
350 rec->getQualifiedNameAsString(),
362 clang::ObjCCategoryDecl *decl)
367 LOG_DBG(
"= Visiting ObjC category declaration {} at {}",
368 decl->getQualifiedNameAsString(),
376 const auto category_id = category_ptr->id();
380 auto &category_model =
388 LOG_DBG(
"Adding ObjC category {} with id {}",
389 category_model.full_name(
false), category_model.id());
394 LOG_DBG(
"Skipping ObjC category {} with id {}",
395 category_model.full_name(
true), category_model.id());
402 clang::ObjCProtocolDecl *decl)
407 LOG_DBG(
"= Visiting ObjC protocol declaration {} at {}",
408 decl->getQualifiedNameAsString(),
416 const auto protocol_id = protocol_ptr->id();
420 auto &protocol_model =
428 LOG_DBG(
"Adding ObjC protocol {} with id {}",
429 protocol_model.full_name(
false), protocol_model.id());
434 LOG_DBG(
"Skipping ObjC protocol {} with id {}",
435 protocol_model.full_name(
true), protocol_model.id());
442 clang::ObjCInterfaceDecl *decl)
447 LOG_DBG(
"= Visiting ObjC interface declaration {} at {}",
448 decl->getQualifiedNameAsString(),
456 const auto protocol_id = interface_ptr->id();
460 auto &interface_model =
465 if (!interface_model.complete())
469 LOG_DBG(
"Adding ObjC interface {} with id {}",
470 interface_model.full_name(
false), interface_model.id());
475 LOG_DBG(
"Skipping ObjC interface {} with id {}",
476 interface_model.full_name(
true), interface_model.id());
487 LOG_DBG(
"= Visiting concept (isType: {}) declaration {} at {}",
488 cpt->isTypeConcept(), cpt->getQualifiedNameAsString(),
496 const auto concept_id = concept_model->id();
502 constexpr auto kMaxConstraintCount = 24U;
503 llvm::SmallVector<const clang::Expr *, kMaxConstraintCount> constraints{};
504 if (cpt->hasAssociatedConstraints()) {
505 cpt->getAssociatedConstraints(constraints);
508 for (
const auto *expr : constraints) {
512 if (cpt->getConstraintExpr() !=
nullptr) {
514 cpt, cpt->getConstraintExpr(), *concept_model);
517 *concept_model, cpt->getConstraintExpr());
521 LOG_DBG(
"Adding concept {} with id {}", concept_model->full_name(
false),
522 concept_model->id());
527 LOG_DBG(
"Skipping concept {} with id {}",
528 concept_model->full_name(
true), concept_model->id());
535 const clang::ConceptDecl *cpt,
const clang::Expr *expr,
538 if (
const auto *constraint = llvm::dyn_cast<clang::RequiresExpr>(expr);
543 LOG_DBG(
"== Processing constraint: '{}'", constraint_source);
545 for ([[maybe_unused]]
const auto *requirement :
546 constraint->getRequirements()) {
551 for (
const auto *decl : constraint->getBody()->decls()) {
552 if (
const auto *parm_var_decl =
553 llvm::dyn_cast<clang::ParmVarDecl>(decl);
555 parm_var_decl->getQualifiedNameAsString();
557 auto param_name = parm_var_decl->getNameAsString();
559 parm_var_decl->getType(), cpt->getASTContext());
561 LOG_DBG(
"=== Processing parameter variable declaration: {}, {}",
562 param_type, param_name);
565 {std::move(param_type), std::move(param_name)});
568 LOG_DBG(
"=== Processing some other concept declaration: {}",
574 for (
const auto *req : constraint->getRequirements()) {
575 if (req->getKind() == clang::concepts::Requirement::RK_Simple) {
576 const auto *simple_req =
577 llvm::dyn_cast<clang::concepts::ExprRequirement>(req);
579 if (simple_req !=
nullptr) {
581 simple_req->getExpr(), [&concept_model](
const auto *e) {
582 auto simple_expr = common::to_string(e);
584 LOG_DBG(
"=== Processing expression requirement: {}",
587 concept_model.add_statement(std::move(simple_expr));
591 else if (req->getKind() == clang::concepts::Requirement::RK_Type) {
593 llvm::dyn_cast<clang::concepts::TypeRequirement>(req),
594 [&concept_model, cpt](
const auto *t) {
596 t->getType()->getType(), cpt->getASTContext());
599 "=== Processing type requirement: {}", type_name);
604 else if (req->getKind() ==
605 clang::concepts::Requirement::RK_Nested) {
606 const auto *nested_req =
607 llvm::dyn_cast<clang::concepts::NestedRequirement>(req);
609 if (nested_req !=
nullptr) {
611 nested_req->getConstraintExpr(), [](
const auto *e) {
612 LOG_DBG(
"=== Processing nested requirement: {}",
613 common::to_string(e));
617 else if (req->getKind() ==
618 clang::concepts::Requirement::RK_Compound) {
619 const auto *compound_req =
620 llvm::dyn_cast<clang::concepts::ExprRequirement>(req);
622 if (compound_req !=
nullptr) {
623 const auto *compound_expr_ptr = compound_req->getExpr();
625 if (compound_expr_ptr !=
nullptr) {
629 auto req_return_type =
630 compound_req->getReturnTypeRequirement();
632 if (!req_return_type.isEmpty()) {
634 fmt::format(
"{{{}}} -> {}", compound_expr,
636 req_return_type.getTypeConstraint()));
638 else if (compound_req->hasNoexceptRequirement()) {
640 fmt::format(
"{{{}}} noexcept", compound_expr);
643 LOG_DBG(
"=== Processing compound requirement: {}",
652 else if (
const auto *binop = llvm::dyn_cast<clang::BinaryOperator>(expr);
657 else if (
const auto *unop = llvm::dyn_cast<clang::UnaryOperator>(expr);
668 found_relationships_t relationships;
670 common::if_dyn_cast<clang::UnresolvedLookupExpr>(
671 expr, [&](
const clang::UnresolvedLookupExpr *ul) {
672 for (
const auto ta : ul->template_arguments()) {
673 if (ta.getArgument().getKind() !=
674 clang::TemplateArgument::ArgKind::Type)
677 relationships, relationship_t::kConstraint);
681 common::if_dyn_cast<clang::ConceptSpecializationExpr>(
682 expr, [&](
const auto *cs) {
686 common::if_dyn_cast<clang::RequiresExpr>(expr, [&](
const auto *re) {
690 common::if_dyn_cast<clang::BinaryOperator>(expr, [&](
const auto *op) {
695 common::if_dyn_cast<clang::UnaryOperator>(expr, [&](
const auto *op) {
699 for (
const auto &[type_element_id, relationship_type, source_decl] :
701 if (type_element_id != c.
id() &&
702 (relationship_type != relationship_t::kNone)) {
706 if (source_decl !=
nullptr) {
717 const clang::ConceptSpecializationExpr *concept_specialization)
720 if (
const auto *cpt = concept_specialization->getNamedConcept();
723 const auto cpt_name = cpt->getNameAsString();
724 const eid_t ast_id{cpt->getID()};
729 const auto target_id = maybe_id.value();
731 std::vector<std::string> constrained_template_params;
733 size_t argument_index{};
735 for (
const auto ta : concept_specialization->getTemplateArguments()) {
736 if (ta.getKind() == clang::TemplateArgument::Type) {
740 cpt, constrained_template_params, argument_index,
743 else if (ta.getKind() == clang::TemplateArgument::Pack) {
744 if (!ta.getPackAsArray().empty() &&
745 ta.getPackAsArray().front().isPackExpansion()) {
746 const auto &pack_head =
747 ta.getPackAsArray().front().getAsType();
751 concept_specialization, cpt,
752 constrained_template_params, argument_index, type_name);
756 LOG_DBG(
"Unsupported concept type parameter in concept: {}",
762 if (!constrained_template_params.empty())
764 {relationship_t::kConstraint, target_id, access_t::kNone,
766 "{}", fmt::join(constrained_template_params,
","))});
775 LOG_DBG(
"= Visiting class declaration {} at {}",
776 cls->getQualifiedNameAsString(),
780 "== getQualifiedNameAsString() = {}", cls->getQualifiedNameAsString());
781 if (cls->getOwningModule() !=
nullptr)
783 "== getOwningModule()->Name = {}", cls->getOwningModule()->Name);
784 LOG_DBG(
"== getID() = {}", cls->getID());
785 LOG_DBG(
"== isTemplateDecl() = {}", cls->isTemplateDecl());
786 LOG_DBG(
"== isTemplated() = {}", cls->isTemplated());
787 LOG_DBG(
"== getParent()->isRecord()() = {}", cls->getParent()->isRecord());
788 LOG_DBG(
"== isCompleteDefinition() = {}", cls->isCompleteDefinition());
790 if (
const auto *parent_record =
791 clang::dyn_cast<clang::RecordDecl>(cls->getParent());
792 parent_record !=
nullptr) {
793 LOG_DBG(
"== getParent()->getQualifiedNameAsString() = {}",
794 parent_record->getQualifiedNameAsString());
802 if (cls->isTemplated() && (cls->getDescribedTemplate() !=
nullptr)) {
805 const eid_t ast_id{cls->getDescribedTemplate()->getID()};
811 if (cls->isLocalClass() !=
nullptr)
822std::unique_ptr<clanguml::class_diagram::model::concept_>
825 assert(cpt !=
nullptr);
831 std::make_unique<model::concept_>(
config().using_namespace())};
832 auto &concept_model = *concept_ptr;
836 concept_model.set_name(cpt->getNameAsString());
837 concept_model.set_namespace(ns);
838 concept_model.set_id(
common::to_id(concept_model.full_name(
false)));
844 if (concept_model.skip())
847 concept_model.set_style(concept_model.style_spec());
853 clang::RecordDecl *rec)
855 assert(rec !=
nullptr);
860 auto record_ptr{std::make_unique<class_>(
config().using_namespace())};
861 auto &record = *record_ptr;
865 if (!record.is_nested()) {
866 auto record_name = rec->getQualifiedNameAsString();
868#if LLVM_VERSION_MAJOR < 16
869 if (record_name ==
"(anonymous)") {
871 [&record_name](
const clang::TypedefNameDecl *name) {
872 record_name = name->getNameAsString();
877 record.set_name(record_name);
885 const auto record_full_name = record_ptr->full_name(
false);
887 record.is_struct(rec->isStruct());
888 record.is_union(rec->isUnion());
893 record.set_style(record.style_spec());
899 clang::CXXRecordDecl *cls)
901 assert(cls !=
nullptr);
906 auto c_ptr{std::make_unique<class_>(
config().using_namespace())};
913 if (!c.is_nested()) {
919 c.is_struct(cls->isStruct());
928 c.set_style(c.style_spec());
933std::unique_ptr<clanguml::class_diagram::model::objc_interface>
935 clang::ObjCCategoryDecl *decl)
937 assert(decl !=
nullptr);
942 auto c_ptr{std::make_unique<class_diagram::model::objc_interface>(
943 config().using_namespace())};
946 decl->getClassInterface()->getNameAsString();
947 c.set_name(fmt::format(
"{}({})",
948 decl->getClassInterface()->getNameAsString(), decl->getNameAsString()));
949 c.set_id(
common::to_id(fmt::format(
"__objc__category__{}", c.name())));
958 c.set_style(c.style_spec());
963std::unique_ptr<clanguml::class_diagram::model::objc_interface>
965 clang::ObjCProtocolDecl *decl)
967 assert(decl !=
nullptr);
972 auto c_ptr{std::make_unique<class_diagram::model::objc_interface>(
973 config().using_namespace())};
976 c.set_name(decl->getNameAsString());
986 c.set_style(c.style_spec());
991std::unique_ptr<clanguml::class_diagram::model::objc_interface>
993 clang::ObjCInterfaceDecl *decl)
995 assert(decl !=
nullptr);
1000 auto c_ptr{std::make_unique<class_diagram::model::objc_interface>(
1001 config().using_namespace())};
1004 c.set_name(decl->getNameAsString());
1013 c.set_style(c.style_spec());
1019 clang::RecordDecl *cls,
class_ &c,
const namespace_ &ns)
1021 std::optional<eid_t> id_opt;
1022 namespace_ parent_ns = ns;
1026 if (id_opt &&
diagram().find<class_>(*id_opt)) {
1027 process_record_parent_by_type<class_>(*id_opt, c, parent_ns, cls);
1030 if (id_opt &&
diagram().find<objc_interface>(*id_opt)) {
1031 process_record_parent_by_type<objc_interface>(
1032 *id_opt, c, parent_ns, cls);
1054 for (
const auto *method : cls.methods()) {
1055 if (method !=
nullptr) {
1062 if (cls.getClassInterface() !=
nullptr) {
1065 relationship_t::kInstantiation, objc_interface_id, access_t::kNone};
1067 LOG_DBG(
"Found protocol {} [{}] for ObjC interface {}",
1068 cls.getClassInterface()->getNameAsString(),
1083 for (
const auto *method : cls.methods()) {
1084 if (method !=
nullptr) {
1098 for (
const auto *method : cls.methods()) {
1099 if (method !=
nullptr) {
1104 for (
const auto *ivar : cls.ivars()) {
1105 if (ivar !=
nullptr) {
1117 if (
const auto *base = cls.getSuperClass(); base !=
nullptr) {
1121 LOG_DBG(
"Found base class {} [{}] for ObjC interface {}",
1122 base->getNameAsString(), parent_id.
value(), c.
name());
1127 for (
const auto *protocol : cls.protocols()) {
1130 relationship_t::kInstantiation, parent_id, access_t::kNone};
1132 LOG_DBG(
"Found protocol {} [{}] for ObjC interface {}",
1133 protocol->getNameAsString(), parent_id.
value(), c.
name());
1142 LOG_DBG(
"== Visiting ObjC ivar {}", ivar.getNameAsString());
1145 auto relationship_hint = relationship_t::kAggregation;
1147 auto field_type = ivar.getType();
1148 auto field_type_str =
1153 const auto field_name = ivar.getNameAsString();
1157 field_name, field_type_str};
1159 field.set_qualified_name(ivar.getQualifiedNameAsString());
1167 if (field_type->isObjCObjectPointerType()) {
1168 relationship_hint = relationship_t::kAggregation;
1169 field_type = field_type->getPointeeType();
1171 else if (field_type->isPointerType()) {
1172 relationship_hint = relationship_t::kAssociation;
1173 field_type = field_type->getPointeeType();
1175 else if (field_type->isLValueReferenceType()) {
1176 relationship_hint = relationship_t::kAssociation;
1177 field_type = field_type.getNonReferenceType();
1179 else if (field_type->isArrayType()) {
1180 relationship_hint = relationship_t::kAggregation;
1181 while (field_type->isArrayType()) {
1182 auto current_multiplicity = field.destination_multiplicity();
1183 if (!current_multiplicity)
1185 *field_type->getAsArrayTypeUnsafe()));
1187 auto maybe_array_size =
1189 if (maybe_array_size.has_value()) {
1190 field.set_destination_multiplicity(
1191 current_multiplicity.value() *
1192 maybe_array_size.value());
1196 field_type = field_type->getAsArrayTypeUnsafe()->getElementType();
1199 else if (field_type->isRValueReferenceType()) {
1200 field_type = field_type.getNonReferenceType();
1203 found_relationships_t relationships;
1205 if (!field.skip_relationship()) {
1208 if (field_type->getAsObjCInterfaceType() !=
nullptr &&
1209 field_type->getAsObjCInterfaceType()->getInterface() !=
nullptr) {
1210 const auto *objc_iface =
1211 field_type->getAsObjCInterfaceType()->getInterface();
1214 relationship_t::kAggregation, &ivar);
1216 else if ((field_type->getAsRecordDecl() !=
nullptr) &&
1217 field_type->getAsRecordDecl()->getNameAsString().empty()) {
1222 std::make_tuple(field.name(), relationship_hint, field.access(),
1223 field.destination_multiplicity());
1227 &ivar, field_type, relationships, relationship_hint);
1234 if ((field_type->getAsRecordDecl() !=
nullptr) &&
1235 field_type->getAsRecordDecl()->getNameAsString().empty()) {
1237 std::regex anonymous_re(
"anonymous_(\\d*)");
1239 std::regex_replace(field.type(), anonymous_re, field_name));
1247 const clang::CXXRecordDecl *cls,
class_ &c)
1249 for (
const auto &base : cls->bases()) {
1251 if (
const auto *tsp =
1252 base.getType()->getAs<clang::TemplateSpecializationType>();
1254 auto template_specialization_ptr =
1255 std::make_unique<class_>(
config().using_namespace());
1257 *cls, *template_specialization_ptr, cls, *tsp, {});
1259 parent_id = template_specialization_ptr->id();
1262 add_class(std::move(template_specialization_ptr));
1265 else if (
const auto *record_type =
1266 base.getType()->getAs<clang::RecordType>();
1267 record_type !=
nullptr) {
1278 LOG_DBG(
"Found base class {} [{}] for class {}",
1287 const clang::RecordDecl &cls,
class_ &c)
1290 for (
const auto *field : cls.fields()) {
1291 if (field !=
nullptr)
1301 assert(cls !=
nullptr);
1304 for (
const auto *method : cls->methods()) {
1305 if (method !=
nullptr) {
1311 if (
const auto *cls_decl_context =
1312 clang::dyn_cast_or_null<clang::DeclContext>(cls);
1313 cls_decl_context !=
nullptr) {
1314 for (
auto const *decl_iterator : cls_decl_context->decls()) {
1315 auto const *method_template =
1316 llvm::dyn_cast_or_null<clang::FunctionTemplateDecl>(
1318 if (method_template ==
nullptr)
1326 for (
const auto *field : cls->fields()) {
1327 if (field !=
nullptr)
1333 std::set<const clang::EnumDecl *> typedeffed_enums;
1334 for (
const auto *decl : cls->decls()) {
1335 if (decl->getKind() == clang::Decl::Typedef) {
1337 clang::dyn_cast<clang::TypedefDecl>(decl));
1338 if (typedeffed_enum !=
nullptr)
1339 typedeffed_enums.emplace(typedeffed_enum);
1345 for (
const auto *decl : cls->decls()) {
1346 if (decl->getKind() == clang::Decl::Var) {
1347 const clang::VarDecl *variable_declaration{
1348 clang::dyn_cast_or_null<clang::VarDecl>(decl)};
1349 if ((variable_declaration !=
nullptr) &&
1350 variable_declaration->isStaticDataMember()) {
1354 else if (decl->getKind() == clang::Decl::Enum &&
1355 typedeffed_enums.count(
1356 clang::dyn_cast_or_null<clang::EnumDecl>(decl)) == 0) {
1357 const auto *enum_decl =
1358 clang::dyn_cast_or_null<clang::EnumDecl>(decl);
1359 if (enum_decl ==
nullptr)
1362 if (enum_decl->getNameAsString().empty()) {
1363 for (
const auto *enum_const : enum_decl->enumerators()) {
1365 enum_decl->getAccess()),
1366 enum_const->getNameAsString(),
"enum"};
1373 if (cls->isCompleteDefinition())
1374 for (
const auto *friend_declaration : cls->friends()) {
1375 if (friend_declaration !=
nullptr)
1381 const clang::FriendDecl &f,
class_ &c)
1383 if (
const auto *friend_type_info = f.getFriendType()) {
1384 const auto friend_type = friend_type_info->getType();
1385 if (friend_type->getAs<clang::TemplateSpecializationType>() !=
1389 else if (friend_type->getAs<clang::RecordType>() !=
nullptr) {
1403 const clang::CXXMethodDecl &mf,
class_ &c)
1407 if (mf.isDefaulted() && !mf.isExplicitlyDefaulted())
1410 auto method_return_type =
1415 auto method_name = mf.getNameAsString();
1416 if (mf.isTemplated()) {
1420 method_name = method_name.substr(0, method_name.find(
'<'));
1425 config().simplify_template_type(method_return_type)};
1427 method.set_qualified_name(mf.getQualifiedNameAsString());
1439 for (
const auto *param : mf.parameters()) {
1440 if (param !=
nullptr)
1445 found_relationships_t relationships;
1448 if (
const auto *templ = mf.getReturnType()
1449 .getNonReferenceType()
1450 .getUnqualifiedType()
1451 ->getAs<clang::TemplateSpecializationType>();
1453 const auto *unaliased_type = templ;
1454 if (unaliased_type->isTypeAlias())
1455 unaliased_type = unaliased_type->getAliasedType()
1456 ->getAs<clang::TemplateSpecializationType>();
1458 if (unaliased_type !=
nullptr) {
1459 auto template_specialization_ptr =
1460 std::make_unique<class_>(
config().using_namespace());
1462 *template_specialization_ptr,
1463 unaliased_type->getTemplateName().getAsTemplateDecl(),
1464 *unaliased_type, &c);
1466 template_specialization_ptr->is_template();
1469 relationships.emplace_back(template_specialization_ptr->id(),
1470 relationship_t::kDependency, &mf);
1472 add_class(std::move(template_specialization_ptr));
1478 &mf, mf.getReturnType(), relationships, relationship_t::kDependency);
1480 for (
const auto &[type_element_id, relationship_type, source_decl] :
1482 if (type_element_id != c.
id() &&
1483 (relationship_type != relationship_t::kNone)) {
1484 relationship r{relationship_t::kDependency, type_element_id};
1486 if (source_decl !=
nullptr) {
1490 LOG_DBG(
"Adding method return type relationship from {}::{} to "
1492 c, mf.getNameAsString(), r.type(), r.label());
1501 auto underlying_type = mf.getReturnType();
1502 if (underlying_type->isReferenceType())
1503 underlying_type = underlying_type.getNonReferenceType();
1504 if (underlying_type->isPointerType())
1505 underlying_type = underlying_type->getPointeeType();
1507 if (
const auto *atsp = underlying_type->getAs<clang::AutoType>();
1512 method.update(
config().using_namespace());
1515 LOG_DBG(
"Adding method: {}", method.name());
1524 auto method_return_type =
1528 util::trim(mf.getNameAsString()), method_return_type};
1530 method.set_qualified_name(mf.getQualifiedNameAsString());
1540 method.is_static(mf.isClassMethod());
1541 method.is_optional(mf.isOptional());
1543 for (
const auto *param : mf.parameters()) {
1544 if (param !=
nullptr)
1549 found_relationships_t relationships;
1552 &mf, mf.getReturnType(), relationships, relationship_t::kDependency);
1554 for (
const auto &[type_element_id, relationship_type, source_decl] :
1556 if (type_element_id != c.
id() &&
1557 (relationship_type != relationship_t::kNone)) {
1558 relationship r{relationship_t::kDependency, type_element_id};
1560 if (source_decl !=
nullptr) {
1564 LOG_DBG(
"Adding method return type relationship from {}::{} to "
1566 c, mf.getNameAsString(), r.type(), r.label());
1574 LOG_DBG(
"Adding ObjC method: {}", method.name());
1581 const clang::CXXMethodDecl &mf,
const class_ &c,
1582 const std::string &method_name,
class_method &method)
const
1584 const bool is_constructor = c.
name() == method_name;
1585 const bool is_destructor = fmt::format(
"~{}", c.
name()) == method_name;
1587#if LLVM_VERSION_MAJOR > 17
1588 method.is_pure_virtual(mf.isPureVirtual());
1590 method.is_pure_virtual(mf.isPure());
1592 method.is_virtual(mf.isVirtual());
1593 method.is_const(mf.isConst());
1594 method.is_defaulted(mf.isDefaulted());
1595 method.is_deleted(mf.isDeleted());
1596 method.is_static(mf.isStatic());
1597 method.is_operator(mf.isOverloadedOperator());
1598 method.is_constexpr(mf.isConstexprSpecified() && !is_constructor);
1599 method.is_consteval(mf.isConsteval());
1600 method.is_constructor(is_constructor);
1601 method.is_destructor(is_destructor);
1602 method.is_move_assignment(mf.isMoveAssignmentOperator());
1603 method.is_copy_assignment(mf.isCopyAssignmentOperator());
1604 method.is_noexcept(isNoexceptExceptionSpec(mf.getExceptionSpecType()));
1610 class_ &c,
const clang::AutoType *atsp)
1612 auto desugared_atsp = atsp->getDeducedType();
1614 if (atsp->isSugared()) {
1615 const auto *deduced_type =
1616 atsp->desugar()->getAs<clang::DeducedTemplateSpecializationType>();
1618 if (deduced_type !=
nullptr)
1619 desugared_atsp = deduced_type->getDeducedType();
1622 if (desugared_atsp.isNull())
1625 const auto *deduced_record_type = desugared_atsp->isRecordType()
1626 ? desugared_atsp->getAs<clang::RecordType>()
1629 if (deduced_record_type !=
nullptr) {
1630 if (
auto *deduced_auto_decl =
1631 llvm::dyn_cast_or_null<clang::ClassTemplateSpecializationDecl>(
1632 deduced_record_type->getDecl());
1633 deduced_auto_decl !=
nullptr) {
1635 const auto diagram_class_count_before_visit =
1640 const bool visitor_added_new_template_specialization =
1642 diagram_class_count_before_visit) > 0;
1644 if (visitor_added_new_template_specialization) {
1645 const auto &template_specialization_model =
1650 template_specialization_model.get().id()};
1660 const clang::FunctionTemplateDecl &mf,
class_ &c)
1664 if (mf.getTemplatedDecl()->isDefaulted() &&
1665 !mf.getTemplatedDecl()->isExplicitlyDefaulted())
1668 auto method_name = mf.getNameAsString();
1669 if (mf.isTemplated()) {
1673 method_name =
util::trim(method_name.substr(0, method_name.find(
'<')));
1677 method_name, mf.getTemplatedDecl()->getReturnType().getAsString()};
1680 clang::dyn_cast<clang::CXXMethodDecl>(mf.getTemplatedDecl()),
1681 [&](
const auto *decl) {
1682 process_method_properties(*decl, c, method_name, method);
1692 for (
const auto *param : mf.getTemplatedDecl()->parameters()) {
1693 if (param !=
nullptr)
1697 method.update(
config().using_namespace());
1700 LOG_DBG(
"Adding method: {}", method.name());
1707 const clang::QualType &type, found_relationships_t &relationships,
1712 if (type->isPointerType()) {
1713 relationship_hint = relationship_t::kAssociation;
1715 decl, type->getPointeeType(), relationships, relationship_hint);
1717 else if (type->isRValueReferenceType()) {
1718 relationship_hint = relationship_t::kAggregation;
1720 decl, type.getNonReferenceType(), relationships, relationship_hint);
1722 else if (type->isLValueReferenceType()) {
1723 relationship_hint = relationship_t::kAssociation;
1725 decl, type.getNonReferenceType(), relationships, relationship_hint);
1727 else if (type->isArrayType()) {
1729 relationships, relationship_t::kAggregation);
1731 else if (type->isEnumeralType()) {
1732 if (
const auto *enum_type = type->getAs<clang::EnumType>();
1733 enum_type !=
nullptr) {
1737 relationships.emplace_back(
1738 enum_type->getDecl()->getID(), relationship_hint, decl);
1742 else if (type->isRecordType()) {
1743 const auto *type_instantiation_type =
1744 type->getAs<clang::TemplateSpecializationType>();
1746 if (type_instantiation_type !=
nullptr) {
1747 const auto *type_instantiation_template_decl =
1748 type_instantiation_type->getTemplateName().getAsTemplateDecl();
1753 relationships.emplace_back(
1754 type_instantiation_type->getTemplateName()
1755 .getAsTemplateDecl()
1757 relationship_hint, decl);
1761 for (
const auto &template_argument :
1762 type_instantiation_type->template_arguments()) {
1764 auto [overridden_relationship_hint, overridden] =
1766 ->getQualifiedNameAsString(),
1767 idx, relationship_hint);
1769 const auto template_argument_kind = template_argument.getKind();
1770 if (template_argument_kind ==
1771 clang::TemplateArgument::ArgKind::Integral) {
1774 else if (template_argument_kind ==
1775 clang::TemplateArgument::ArgKind::Null) {
1778 else if (template_argument_kind ==
1779 clang::TemplateArgument::ArgKind::Expression) {
1782 else if (template_argument.getKind() ==
1783 clang::TemplateArgument::ArgKind::NullPtr) {
1786 else if (template_argument_kind ==
1787 clang::TemplateArgument::ArgKind::Template) {
1790 else if (template_argument_kind ==
1791 clang::TemplateArgument::ArgKind::TemplateExpansion) {
1794 else if (
const auto *function_type =
1795 template_argument.getAsType()
1796 ->getAs<clang::FunctionProtoType>();
1797 function_type !=
nullptr) {
1798 for (
const auto ¶m_type :
1799 function_type->param_types()) {
1801 relationships, relationship_t::kDependency);
1804 else if (template_argument_kind ==
1805 clang::TemplateArgument::ArgKind::Type) {
1808 relationships, overridden_relationship_hint);
1813 else if (type->getAsCXXRecordDecl() !=
nullptr) {
1814 relationships.emplace_back(
1815 type->getAsCXXRecordDecl()->getID(), relationship_hint, decl);
1819 relationships.emplace_back(
1820 type->getAsRecordDecl()->getID(), relationship_hint, decl);
1824 else if (
const auto *template_specialization_type =
1825 type->getAs<clang::TemplateSpecializationType>();
1826 template_specialization_type !=
nullptr) {
1827 const auto *type_instantiation_template_decl =
1828 template_specialization_type->getTemplateName().getAsTemplateDecl();
1829 if (
should_include(template_specialization_type->getTemplateName()
1830 .getAsTemplateDecl())) {
1831 relationships.emplace_back(
1832 template_specialization_type->getTemplateName()
1833 .getAsTemplateDecl()
1835 relationship_hint, decl);
1838 for (
const auto &template_argument :
1839 template_specialization_type->template_arguments()) {
1841 auto [overridden_relationship_hint, overridden] =
1843 ->getQualifiedNameAsString(),
1844 idx, relationship_hint);
1846 const auto template_argument_kind = template_argument.getKind();
1847 if (template_argument_kind ==
1848 clang::TemplateArgument::ArgKind::Integral) {
1851 else if (template_argument_kind ==
1852 clang::TemplateArgument::ArgKind::Null) {
1855 else if (template_argument_kind ==
1856 clang::TemplateArgument::ArgKind::Expression) {
1859 else if (template_argument.getKind() ==
1860 clang::TemplateArgument::ArgKind::NullPtr) {
1863 else if (template_argument_kind ==
1864 clang::TemplateArgument::ArgKind::Template) {
1867 else if (template_argument_kind ==
1868 clang::TemplateArgument::ArgKind::TemplateExpansion) {
1871 else if (
const auto *function_type =
1872 template_argument.getAsType()
1873 ->getAs<clang::FunctionProtoType>();
1874 function_type !=
nullptr) {
1875 for (
const auto ¶m_type : function_type->param_types()) {
1877 relationship_t::kDependency);
1880 else if (template_argument_kind ==
1881 clang::TemplateArgument::ArgKind::Type) {
1883 relationships, overridden_relationship_hint);
1896 parameter.
set_name(param.getNameAsString());
1900 if (parameter.
skip())
1903 auto parameter_type =
1905 parameter.
set_type(parameter_type);
1909 found_relationships_t relationships;
1911 LOG_DBG(
"Looking for relationships in type: {}",
1915 relationship_t::kDependency);
1917 for (
const auto &[type_element_id, relationship_type, source_decl] :
1919 if (type_element_id != c.
id() &&
1920 (relationship_type != relationship_t::kNone)) {
1921 relationship r{relationship_t::kDependency, type_element_id};
1923 if (source_decl !=
nullptr) {
1927 LOG_DBG(
"Adding ObjC method parameter relationship from {} to "
1929 c, r.type(), r.label());
1936 method.add_parameter(std::move(parameter));
1941 const std::set<std::string> & )
1944 parameter.
set_name(p.getNameAsString());
1948 if (parameter.
skip())
1956 parameter.
set_type(parameter_type);
1958 if (p.hasDefaultArg()) {
1959 const auto *default_arg = p.getDefaultArg();
1960 if (default_arg !=
nullptr) {
1969 found_relationships_t relationships;
1971 LOG_DBG(
"Looking for relationships in type: {}",
1974 if (
const auto *templ =
1976 .getNonReferenceType()
1977 .getUnqualifiedType()
1978 ->getAs<clang::TemplateSpecializationType>();
1980 auto template_specialization_ptr =
1981 std::make_unique<class_>(
config().using_namespace());
1983 *template_specialization_ptr,
1984 templ->getTemplateName().getAsTemplateDecl(), *templ, &c);
1986 template_specialization_ptr->is_template(
true);
1989 relationships.emplace_back(template_specialization_ptr->id(),
1990 relationship_t::kDependency, &p);
1992 add_class(std::move(template_specialization_ptr));
1997 &p, p.getType(), relationships, relationship_t::kDependency);
1999 for (
const auto &[type_element_id, relationship_type, source_decl] :
2001 if (type_element_id != c.
id() &&
2002 (relationship_type != relationship_t::kNone)) {
2003 relationship r{relationship_t::kDependency, type_element_id};
2005 if (source_decl !=
nullptr) {
2009 LOG_DBG(
"Adding function parameter relationship from {} to "
2011 c, r.type(), r.label());
2018 method.add_parameter(std::move(parameter));
2023 const found_relationships_t &relationships,
bool break_on_first_aggregation)
2027 for (
const auto &[target, relationship_type, source_decl] : relationships) {
2028 if (relationship_type != relationship_t::kNone) {
2031 r.set_access(field.
access());
2032 if (source_decl !=
nullptr) {
2035 bool mulitplicity_provided_in_comment{
false};
2036 if (decorator_rtype != relationship_t::kNone) {
2037 r.set_type(decorator_rtype);
2038 auto mult =
util::split(decorator_rmult,
":",
false);
2039 if (mult.size() == 2) {
2040 mulitplicity_provided_in_comment =
true;
2041 r.set_multiplicity_source(mult[0]);
2042 r.set_multiplicity_destination(mult[1]);
2045 if (!mulitplicity_provided_in_comment &&
2047 r.set_multiplicity_destination(
2053 LOG_DBG(
"Adding relationship from {} to {} with label {}", c,
2054 r.destination(), r.type(), r.label());
2058 if (break_on_first_aggregation &&
2059 relationship_type == relationship_t::kAggregation)
2065std::pair<relationship_t, bool>
2067 const std::string &type_name,
int index, relationship_t hint)
2069 bool overridden{
false};
2071 for (
const auto &[pattern, rel_hint] :
config().relationship_hints()) {
2072 if (type_name.find(pattern) == 0) {
2074 hint = rel_hint.default_hint;
2076 hint = rel_hint.get(index, hint);
2083 return {hint, overridden};
2087 const clang::VarDecl &field_declaration,
class_ &c)
2089 const auto field_type = field_declaration.getType();
2092 if (type_name.empty())
2093 type_name =
"<<anonymous>>";
2097 field_declaration.getNameAsString(),
2098 config().simplify_template_type(type_name)};
2100 field.set_qualified_name(field_declaration.getQualifiedNameAsString());
2102 field.is_static(
true);
2110 if (!field.skip_relationship()) {
2111 found_relationships_t relationships;
2115 relationships, relationship_t::kAssociation);
2123std::unique_ptr<class_>
2125 clang::ClassTemplateSpecializationDecl *cls)
2127 LOG_DBG(
"Processing template specialization {} at {}",
2128 cls->getQualifiedNameAsString(),
2131 auto c_ptr = std::make_unique<class_>(
config().using_namespace());
2134 auto &template_instantiation = *c_ptr;
2135 template_instantiation.is_template(
true);
2138 auto qualified_name = cls->getQualifiedNameAsString();
2142 namespace_ ns{qualified_name};
2144 template_instantiation.set_name(cls->getNameAsString());
2145 template_instantiation.set_namespace(ns);
2147 template_instantiation.is_struct(cls->isStruct());
2151 if (!template_instantiation.is_nested()) {
2153 template_instantiation.set_id(
2161 if (template_instantiation.skip())
2164 id_mapper().
add(cls->getID(), template_instantiation.id());
2170 const clang::FieldDecl &field_declaration,
class_ &c)
2173 "== Visiting record member {}", field_declaration.getNameAsString());
2176 auto relationship_hint = relationship_t::kAggregation;
2180 [[maybe_unused]]
bool template_instantiation_added_as_aggregation{
false};
2182 auto field_type = field_declaration.getType();
2188 const auto field_name = field_declaration.getNameAsString();
2190 auto field_type_str =
2197 field_name,
config().simplify_template_type(field_type_str)};
2199 field.set_qualified_name(field_declaration.getQualifiedNameAsString());
2210 if (field_type->isPointerType()) {
2211 relationship_hint = relationship_t::kAssociation;
2212 field_type = field_type->getPointeeType();
2214 else if (field_type->isLValueReferenceType()) {
2215 relationship_hint = relationship_t::kAssociation;
2216 field_type = field_type.getNonReferenceType();
2218 else if (field_type->isArrayType()) {
2219 relationship_hint = relationship_t::kAggregation;
2220 while (field_type->isArrayType()) {
2221 auto current_multiplicity = field.destination_multiplicity();
2222 if (!current_multiplicity)
2224 *field_type->getAsArrayTypeUnsafe()));
2226 auto maybe_array_size =
2228 if (maybe_array_size.has_value()) {
2229 field.set_destination_multiplicity(
2230 current_multiplicity.value() *
2231 maybe_array_size.value());
2235 field_type = field_type->getAsArrayTypeUnsafe()->getElementType();
2238 else if (field_type->isRValueReferenceType()) {
2239 field_type = field_type.getNonReferenceType();
2242 auto [overridden_relationship_hint, overridden] =
2245 relationship_hint = overridden_relationship_hint;
2247 found_relationships_t relationships;
2249 const auto *template_field_type =
2250 field_type->getAs<clang::TemplateSpecializationType>();
2252 if (template_field_type !=
nullptr)
2253 if (template_field_type->isTypeAlias())
2254 template_field_type =
2255 template_field_type->getAliasedType()
2256 ->getAs<clang::TemplateSpecializationType>();
2258 bool field_type_is_template_template_parameter{
false};
2259 if (template_field_type !=
nullptr) {
2263 if (class_template_param.name() ==
2264 template_field_type->getTemplateName()
2265 .getAsTemplateDecl()
2266 ->getNameAsString() +
2268 field_type_is_template_template_parameter =
true;
2274 if (template_field_type !=
nullptr &&
2275 !field_type_is_template_template_parameter) {
2277 auto template_specialization_ptr =
2278 std::make_unique<class_>(
config().using_namespace());
2280 *template_specialization_ptr,
2281 field_type->getAs<clang::TemplateSpecializationType>()
2283 .getAsTemplateDecl(),
2284 *template_field_type, {&c});
2285 template_specialization_ptr->is_template(
true);
2287 if (!field.skip_relationship() && template_specialization_ptr) {
2288 const auto &template_specialization = *template_specialization_ptr;
2294 bool add_template_instantiation_to_diagram{
false};
2297 found_relationships_t::value_type r{
2298 template_specialization.id(), relationship_hint,
2299 &field_declaration};
2301 add_template_instantiation_to_diagram =
true;
2305 template_instantiation_added_as_aggregation =
2306 relationship_hint == relationship_t::kAggregation;
2307 relationships.emplace_back(std::move(r));
2312 found_relationships_t nested_relationships;
2314 if (!template_instantiation_added_as_aggregation) {
2315 for (
const auto &template_argument :
2316 template_specialization.template_params()) {
2318 auto template_argument_str = template_argument.to_string(
2319 config().using_namespace(),
false);
2321 LOG_DBG(
"Looking for nested relationships from {}::{} in "
2322 "template argument {}",
2323 c, field_name, template_argument_str);
2325 auto [overridden_relationship_hint_param,
2328 template_specialization.full_name(
false), idx,
2331 template_instantiation_added_as_aggregation =
2332 template_argument.find_nested_relationships(
2333 &field_declaration, nested_relationships,
2334 overridden_relationship_hint_param,
2336 [&d =
diagram()](
const std::string &full_name) {
2337 if (full_name.empty())
2340 return d.should_include(ns, name);
2354 if (add_template_instantiation_to_diagram)
2355 add_class(std::move(template_specialization_ptr));
2359 if (!field.skip_relationship()) {
2362 if (!template_instantiation_added_as_aggregation) {
2363 if ((field_type->getAsRecordDecl() !=
nullptr) &&
2364 field_type->getAsRecordDecl()->getNameAsString().empty()) {
2369 std::make_tuple(field.name(), relationship_hint,
2370 field.access(), field.destination_multiplicity());
2374 relationships, relationship_hint);
2382 if ((field_type->getAsRecordDecl() !=
nullptr) &&
2383 field_type->getAsRecordDecl()->getNameAsString().empty()) {
2385 std::regex anonymous_re(
"anonymous_(\\d*)");
2387 std::regex_replace(field.type(), anonymous_re, field_name));
2397 std::optional<eid_t> &parent_id_opt, namespace_ &parent_ns)
const
2399 const auto *parent = decl->getParent();
2401 if (parent !=
nullptr) {
2402 if (
const auto *parent_record_decl =
2403 clang::dyn_cast<clang::RecordDecl>(parent);
2404 parent_record_decl !=
nullptr) {
2407 eid_t local_id{parent_record_decl->getID()};
2415 if (!parent_id_opt) {
2416 if (parent_record_decl->getDescribedTemplate() !=
nullptr) {
2418 parent_record_decl->getDescribedTemplate()->getID();
2428 const auto *lexical_parent = decl->getLexicalParent();
2429 if (lexical_parent !=
nullptr) {
2430 if (
const auto *parent_interface_decl =
2431 clang::dyn_cast<clang::ObjCInterfaceDecl>(lexical_parent);
2432 parent_interface_decl !=
nullptr) {
2434 eid_t ast_id{parent_interface_decl->getID()};
2447 diagram().should_include(c->get_namespace())) {
2455 diagram().should_include(e->get_namespace())) {
2464 diagram().for_all_elements([&](
auto &element_view) {
2465 for (
const auto &el : element_view) {
2466 for (
auto &rel : el.get().relationships()) {
2467 if (!rel.destination().is_global()) {
2468 const auto maybe_id =
2471 LOG_TRACE(
"= Resolved instantiation destination "
2473 "id {} to global id {}",
2474 rel.destination(), *maybe_id);
2475 rel.set_destination(*maybe_id);
2479 el.get().remove_duplicate_relationships();
2488 if (
config().skip_redundant_dependencies()) {
2489 diagram().remove_redundant_dependencies();
2494 const clang::ConceptSpecializationExpr *concept_specialization,
2495 const clang::ConceptDecl *cpt,
2496 std::vector<std::string> &constrained_template_params,
2497 size_t argument_index, std::string &type_name)
const
2502 if (!full_declaration_text.empty()) {
2504 if (type_name.find(
"type-parameter-") == 0) {
2505 const auto concept_declaration_text = full_declaration_text.substr(
2506 full_declaration_text.find(cpt->getNameAsString()) +
2507 cpt->getNameAsString().size() + 1);
2510 concept_declaration_text, [](
const auto &t) {
return t; });
2512 if (template_params.size() > argument_index)
2513 type_name = template_params[argument_index].to_string(
2514 config().using_namespace(),
false);
2516 constrained_template_params.push_back(type_name);
2521 std::string qualified_name)
2527 const std::string &qualified_name)
const
2533 std::unique_ptr<common::model::template_element> element)
2535 add_class(util::unique_pointer_cast<class_>(std::move(element)));
2540 if ((
config().generate_packages() &&
2542 assert(!c->file().empty());
2544 const auto file =
config().make_path_relative(c->file());
2550 diagram().add(p, std::move(c));
2552 else if ((
config().generate_packages() &&
2555 const auto module_path =
config().make_module_relative(c->module());
2559 diagram().add(p, std::move(c));
2562 diagram().add(c->path(), std::move(c));
2567 std::unique_ptr<objc_interface> &&c)
2569 if ((
config().generate_packages() &&
2571 assert(!c->file().empty());
2573 const auto file =
config().make_path_relative(c->file());
2579 diagram().add(p, std::move(c));
2582 diagram().add(c->path(), std::move(c));
2588 if ((
config().generate_packages() &&
2590 assert(!e->file().empty());
2592 const auto file =
config().make_path_relative(e->file());
2598 diagram().add(p, std::move(e));
2600 else if ((
config().generate_packages() &&
2603 const auto module_path =
config().make_module_relative(e->module());
2607 diagram().add(p, std::move(e));
2610 diagram().add(e->path(), std::move(e));
2616 if ((
config().generate_packages() &&
2618 assert(!c->file().empty());
2620 const auto file =
config().make_path_relative(c->file());
2626 diagram().add(p, std::move(c));
2628 else if ((
config().generate_packages() &&
2631 const auto module_path =
config().make_module_relative(c->module());
2635 diagram().add(p, std::move(c));
2638 diagram().add(c->path(), std::move(c));
2644 const std::string &full_name,
eid_t templated_decl_id)
2647 template_instantiation_base);
2651 std::string destination{};
2652 std::string best_match_full_name{};
2653 auto full_template_name = template_instantiation.
full_name(
false);
2655 eid_t best_match_id{};
2657 for (
const auto templ :
diagram().classes()) {
2658 if (templ.get() == template_instantiation)
2661 auto c_full_name = templ.get().full_name(
false);
2663 template_instantiation.calculate_template_specialization_match(
2666 if (match > best_match) {
2668 best_match_full_name = c_full_name;
2669 best_match_id = templ.get().id();
2673 auto templated_decl_global_id =
2676 if (best_match_id.value() > 0) {
2677 destination = best_match_full_name;
2678 template_instantiation.add_relationship(
2680 template_instantiation.template_specialization_found(
true);
2684 else if (
diagram().has_element(templated_decl_global_id)) {
2685 template_instantiation.add_relationship(
2687 templated_decl_global_id});
2688 template_instantiation.template_specialization_found(
true);
2690 else if (
id_mapper().get_global_id(templated_decl_id).has_value()) {
2691 template_instantiation.add_relationship(
2693 template_instantiation.template_specialization_found(
true);
2696 LOG_DBG(
"Skipping instantiation relationship from {} to {}",
2697 template_instantiation, templated_decl_global_id);
2700 LOG_DBG(
"== Cannot determine global id for specialization template {} "
2701 "- delaying until the translation unit is complete ",
2702 templated_decl_global_id);
2704 template_instantiation.add_relationship(