0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
translation_unit_visitor.cc
Go to the documentation of this file.
1/**
2 * @file src/class_diagram/visitor/translation_unit_visitor.cc
3 *
4 * Copyright (c) 2021-2025 Bartek Kryza <bkryza@gmail.com>
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 */
18
20#include "common/clang_utils.h"
21
22#include <clang/AST/ExprConcepts.h>
23#include <clang/Basic/FileManager.h>
24#include <clang/Lex/Preprocessor.h>
25#include <spdlog/spdlog.h>
26
28
33 , template_builder_{diagram, config, *this}
34{
35}
36
38 const clang::NamedDecl *decl) const
39{
40 auto cls = std::make_unique<class_>(config().using_namespace());
41 cls->is_struct(common::is_struct(decl));
42 return cls;
43}
44
46{
47 assert(ns != nullptr);
48
49 if (config().package_type() == config::package_type_t::kDirectory)
50 return true;
51
52 if (ns->isAnonymousNamespace() || ns->isInline())
53 return true;
54
55 LOG_DBG("= Visiting namespace declaration {} at {}",
56 ns->getQualifiedNameAsString(),
57 ns->getLocation().printToString(source_manager()));
58
59 auto package_path = namespace_{common::get_qualified_name(*ns)};
60
61 auto package_parent = package_path;
62 bool is_root =
63 (package_path.size() == 1) && !config().using_namespace().is_empty();
64 package_path.is_root(is_root);
65
66 std::string name;
67 if (!package_path.is_empty())
68 name = package_path.name();
69
70 if (!package_parent.is_empty())
71 package_parent.pop_back();
72
73 const auto usn = config().using_namespace();
74
75 auto p = std::make_unique<common::model::package>(usn);
76 package_path = package_path.relative_to(usn);
77
78 p->set_name(name);
79 p->set_namespace(package_parent);
80 p->set_id(common::to_id(*ns));
81 p->is_root(is_root);
82 id_mapper().add(ns->getID(), p->id());
83
84 if (config().filter_mode() == config::filter_mode_t::advanced ||
85 (diagram().should_include(*p) && !diagram().get(p->id()))) {
86 process_comment(*ns, *p);
87 set_source_location(*ns, *p);
88
89 p->set_style(p->style_spec());
90
91 for (const auto *attr : ns->attrs()) {
92 if (attr->getKind() == clang::attr::Kind::Deprecated) {
93 p->set_deprecated(true);
94 break;
95 }
96 }
97
98 if (!p->skip()) {
99 diagram().add(package_path, std::move(p));
100 }
101 }
102
103 return true;
104}
105
106bool translation_unit_visitor::VisitTypedefDecl(clang::TypedefDecl *decl)
107{
108 if (const auto *enm = common::get_typedef_enum_decl(decl)) {
109 // Associate a typedef with an anonymous declaration so that it can
110 // be later used to assign proper name to enum
111 if (!should_include(enm))
112 return true;
113
114 LOG_DBG("= Visiting typedef enum declaration {} at {}",
115 enm->getQualifiedNameAsString(),
116 enm->getLocation().printToString(source_manager()));
117
118 auto e_ptr = create_declaration(enm, decl);
119
120 if (!e_ptr)
121 return true;
122
123 if (enm->isComplete()) {
124 process_declaration(*enm, *e_ptr);
125 }
126
127 if (e_ptr && diagram().should_include(*e_ptr))
128 add_enum(std::move(e_ptr));
129 }
130
131 return true; // Continue traversing
132}
133
135{
136 assert(enm != nullptr);
137
138 // Anonymous enum values are either class fields with type enum or
139 // `typedef enum` declarations
140 if (enm->getNameAsString().empty()) {
141 return true;
142 }
143
144 if (!should_include(enm))
145 return true;
146
147 LOG_DBG("= Visiting enum declaration {} at {}",
148 enm->getQualifiedNameAsString(),
149 enm->getLocation().printToString(source_manager()));
150
151 auto e_ptr = create_declaration(enm, nullptr);
152 if (!e_ptr)
153 return true;
154
155 return add_or_update(enm, std::move(e_ptr));
156}
157
158std::unique_ptr<clanguml::class_diagram::model::enum_>
160 const clang::EnumDecl *enm, const clang::TypedefDecl *typedef_decl)
161{
162 auto e_ptr = std::make_unique<enum_>(config().using_namespace());
163 auto &e = *e_ptr;
164
165 auto ns{common::get_tag_namespace(*enm)};
166
167 // Id of parent class or struct in which this enum is potentially nested
168 std::optional<eid_t> parent_id_opt;
169 [[maybe_unused]] namespace_ parent_ns;
170 find_record_parent_id(enm, parent_id_opt, parent_ns);
171
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();
177 else
178 enm_name = common::get_tag_name(*enm);
179
180 if (parent_id_opt && diagram().find<class_>(*parent_id_opt)) {
181 auto parent_class = diagram().find<class_>(*parent_id_opt);
182
183 e.set_namespace(ns);
184 e.set_name(parent_class.value().name(), enm_name);
185 e.set_id(common::to_id(e.full_name(false)));
186 e.add_relationship({relationship_t::kContainment, *parent_id_opt});
187 e.nested(true);
188 }
189 else if (parent_id_opt && diagram().find<objc_interface>(*parent_id_opt)) {
190 auto parent_class = diagram().find<objc_interface>(*parent_id_opt);
191
192 e.set_namespace(ns);
193 e.set_name(parent_class.value().name(), enm_name);
194 e.set_id(common::to_id(e.full_name(false)));
195 e.add_relationship({relationship_t::kContainment, *parent_id_opt});
196 e.nested(true);
197 }
198 else {
199 e.set_name(enm_name);
200 e.set_namespace(ns);
201 e.set_id(common::to_id(e.full_name(false)));
202 }
203
204 id_mapper().add(enm->getID(), e.id());
205
206 process_comment(*enm, e);
207 set_source_location(*enm, e);
208 set_owning_module(*enm, e);
209
210 if (e.skip())
211 return {};
212
213 e.set_style(e.style_spec());
214
215 return e_ptr;
216}
217
219 const clang::EnumDecl &enm, clanguml::class_diagram::model::enum_ &e)
220{
221 for (const auto &ev : enm.enumerators()) {
222 e.constants().push_back(ev->getNameAsString());
223 }
224
225 e.complete(true);
226}
227
229 clang::ClassTemplateSpecializationDecl *cls)
230{
231 if (!should_include(cls))
232 return true;
233
234 LOG_DBG("= Visiting template specialization declaration {} at {} "
235 "(described class id {})",
236 cls->getQualifiedNameAsString(),
237 cls->getLocation().printToString(source_manager()),
238 cls->getSpecializedTemplate()
239 ? cls->getSpecializedTemplate()->getTemplatedDecl()->getID()
240 : 0);
241
242 // TODO: Add support for classes defined in function/method bodies
243 if (cls->isLocalClass() != nullptr)
244 return true;
245
246 auto template_specialization_ptr = process_template_specialization(cls);
247
248 if (!template_specialization_ptr)
249 return true;
250
251 return add_or_update(cls, std::move(template_specialization_ptr));
252}
253
255 clang::TypeAliasTemplateDecl *cls)
256{
257 if (!should_include(cls))
258 return true;
259
260 LOG_DBG("= Visiting template type alias declaration {} at {}",
261 cls->getQualifiedNameAsString(),
262 cls->getLocation().printToString(source_manager()));
263
264 const auto *template_type_specialization_ptr =
265 cls->getTemplatedDecl()
266 ->getUnderlyingType()
267 ->getAs<clang::TemplateSpecializationType>();
268
269 if (template_type_specialization_ptr == nullptr)
270 return true;
271
272 auto template_specialization_ptr =
273 std::make_unique<class_>(config().using_namespace());
274 tbuilder().build_from_template_specialization_type(*cls->getTemplatedDecl(),
275 *template_specialization_ptr, cls, *template_type_specialization_ptr);
276
277 template_specialization_ptr->is_template(true);
278
279 if (diagram().should_include(*template_specialization_ptr)) {
280 const auto name = template_specialization_ptr->full_name(true);
281 const auto id = template_specialization_ptr->id();
282
283 LOG_DBG("Adding class {} with id {}", name, id);
284
285 set_source_location(*cls, *template_specialization_ptr);
286 set_owning_module(*cls, *template_specialization_ptr);
287
288 add_class(std::move(template_specialization_ptr));
289 }
290
291 return true;
292}
293
295 clang::ClassTemplateDecl *cls)
296{
297 if (!should_include(cls))
298 return true;
299
300 LOG_DBG("= Visiting class template declaration {} at {}",
301 cls->getQualifiedNameAsString(),
302 cls->getLocation().printToString(source_manager()));
303
304 auto c_ptr = create_declaration(cls->getTemplatedDecl());
305
306 if (!c_ptr)
307 return true;
308
309 add_processed_template_class(cls->getQualifiedNameAsString());
310
311 tbuilder().build_from_template_declaration(*c_ptr, *cls, *c_ptr);
312
313 // Override the id with the template id, for now we don't care about the
314 // underlying templated class id
315 const auto cls_full_name = c_ptr->full_name(false);
316 const auto id = common::to_id(cls_full_name);
317
318 c_ptr->set_id(id);
319 c_ptr->is_template(true);
320
321 id_mapper().add(cls->getID(), id);
322
323 constexpr auto kMaxConstraintCount = 24U;
324 llvm::SmallVector<const clang::Expr *, kMaxConstraintCount> constraints{};
325 if (cls->hasAssociatedConstraints()) {
326 cls->getAssociatedConstraints(constraints);
327 }
328
329 for (const auto *expr : constraints) {
331 }
332
333 return add_or_update(cls->getTemplatedDecl(), std::move(c_ptr));
334}
335
337{
338 if (rec == nullptr)
339 return true;
340
341 if (clang::dyn_cast_or_null<clang::CXXRecordDecl>(rec) != nullptr)
342 // This is handled by VisitCXXRecordDecl()
343 return true;
344
345 // It seems we are in a C (not C++) translation unit
346 if (!should_include(rec))
347 return true;
348
349 LOG_DBG("= Visiting record declaration {} at {}",
350 rec->getQualifiedNameAsString(),
351 rec->getLocation().printToString(source_manager()));
352
353 auto record_ptr = create_declaration(rec);
354
355 if (!record_ptr)
356 return true;
357
358 return add_or_update(rec, std::move(record_ptr));
359}
360
362 clang::ObjCCategoryDecl *decl)
363{
364 if (!should_include(decl))
365 return true;
366
367 LOG_DBG("= Visiting ObjC category declaration {} at {}",
368 decl->getQualifiedNameAsString(),
369 decl->getLocation().printToString(source_manager()));
370
371 auto category_ptr = create_objc_category_declaration(decl);
372
373 if (!category_ptr)
374 return true;
375
376 const auto category_id = category_ptr->id();
377
378 id_mapper().add(decl->getID(), category_id);
379
380 auto &category_model =
381 diagram().find<objc_interface>(category_id).has_value()
382 ? *diagram().find<objc_interface>(category_id).get()
383 : *category_ptr;
384
385 process_objc_category_declaration(*decl, category_model);
386
387 if (diagram().should_include(category_model)) {
388 LOG_DBG("Adding ObjC category {} with id {}",
389 category_model.full_name(false), category_model.id());
390
391 add_objc_interface(std::move(category_ptr));
392 }
393 else {
394 LOG_DBG("Skipping ObjC category {} with id {}",
395 category_model.full_name(true), category_model.id());
396 }
397
398 return true;
399}
400
402 clang::ObjCProtocolDecl *decl)
403{
404 if (!should_include(decl))
405 return true;
406
407 LOG_DBG("= Visiting ObjC protocol declaration {} at {}",
408 decl->getQualifiedNameAsString(),
409 decl->getLocation().printToString(source_manager()));
410
411 auto protocol_ptr = create_objc_protocol_declaration(decl);
412
413 if (!protocol_ptr)
414 return true;
415
416 const auto protocol_id = protocol_ptr->id();
417
418 id_mapper().add(decl->getID(), protocol_id);
419
420 auto &protocol_model =
421 diagram().find<objc_interface>(protocol_id).has_value()
422 ? *diagram().find<objc_interface>(protocol_id).get()
423 : *protocol_ptr;
424
425 process_objc_protocol_declaration(*decl, protocol_model);
426
427 if (diagram().should_include(protocol_model)) {
428 LOG_DBG("Adding ObjC protocol {} with id {}",
429 protocol_model.full_name(false), protocol_model.id());
430
431 add_objc_interface(std::move(protocol_ptr));
432 }
433 else {
434 LOG_DBG("Skipping ObjC protocol {} with id {}",
435 protocol_model.full_name(true), protocol_model.id());
436 }
437
438 return true;
439}
440
442 clang::ObjCInterfaceDecl *decl)
443{
444 if (!should_include(decl))
445 return true;
446
447 LOG_DBG("= Visiting ObjC interface declaration {} at {}",
448 decl->getQualifiedNameAsString(),
449 decl->getLocation().printToString(source_manager()));
450
451 auto interface_ptr = create_objc_interface_declaration(decl);
452
453 if (!interface_ptr)
454 return true;
455
456 const auto protocol_id = interface_ptr->id();
457
458 id_mapper().add(decl->getID(), protocol_id);
459
460 auto &interface_model =
461 diagram().find<objc_interface>(protocol_id).has_value()
462 ? *diagram().find<objc_interface>(protocol_id).get()
463 : *interface_ptr;
464
465 if (!interface_model.complete())
466 process_objc_interface_declaration(*decl, interface_model);
467
468 if (diagram().should_include(interface_model)) {
469 LOG_DBG("Adding ObjC interface {} with id {}",
470 interface_model.full_name(false), interface_model.id());
471
472 add_objc_interface(std::move(interface_ptr));
473 }
474 else {
475 LOG_DBG("Skipping ObjC interface {} with id {}",
476 interface_model.full_name(true), interface_model.id());
477 }
478
479 return true;
480}
481
483{
484 if (!should_include(cpt))
485 return true;
486
487 LOG_DBG("= Visiting concept (isType: {}) declaration {} at {}",
488 cpt->isTypeConcept(), cpt->getQualifiedNameAsString(),
489 cpt->getLocation().printToString(source_manager()));
490
491 auto concept_model = create_concept_declaration(cpt);
492
493 if (!concept_model)
494 return true;
495
496 const auto concept_id = concept_model->id();
497
498 id_mapper().add(cpt->getID(), concept_id);
499
500 tbuilder().build_from_template_declaration(*concept_model, *cpt);
501
502 constexpr auto kMaxConstraintCount = 24U;
503 llvm::SmallVector<const clang::Expr *, kMaxConstraintCount> constraints{};
504 if (cpt->hasAssociatedConstraints()) {
505 cpt->getAssociatedConstraints(constraints);
506 }
507
508 for (const auto *expr : constraints) {
510 }
511
512 if (cpt->getConstraintExpr() != nullptr) {
514 cpt, cpt->getConstraintExpr(), *concept_model);
515
517 *concept_model, cpt->getConstraintExpr());
518 }
519
520 if (diagram().should_include(*concept_model)) {
521 LOG_DBG("Adding concept {} with id {}", concept_model->full_name(false),
522 concept_model->id());
523
524 add_concept(std::move(concept_model));
525 }
526 else {
527 LOG_DBG("Skipping concept {} with id {}",
528 concept_model->full_name(true), concept_model->id());
529 }
530
531 return true;
532}
533
535 const clang::ConceptDecl *cpt, const clang::Expr *expr,
536 model::concept_ &concept_model) const
537{
538 if (const auto *constraint = llvm::dyn_cast<clang::RequiresExpr>(expr);
539 constraint) {
540
541 auto constraint_source = common::to_string(constraint);
542
543 LOG_DBG("== Processing constraint: '{}'", constraint_source);
544
545 for ([[maybe_unused]] const auto *requirement :
546 constraint->getRequirements()) {
547 // TODO
548 }
549
550 // process 'requires (...)' declaration
551 for (const auto *decl : constraint->getBody()->decls()) {
552 if (const auto *parm_var_decl =
553 llvm::dyn_cast<clang::ParmVarDecl>(decl);
554 parm_var_decl) {
555 parm_var_decl->getQualifiedNameAsString();
556
557 auto param_name = parm_var_decl->getNameAsString();
558 auto param_type = common::to_string(
559 parm_var_decl->getType(), cpt->getASTContext());
560
561 LOG_DBG("=== Processing parameter variable declaration: {}, {}",
562 param_type, param_name);
563
564 concept_model.add_parameter(
565 {std::move(param_type), std::move(param_name)});
566 }
567 else {
568 LOG_DBG("=== Processing some other concept declaration: {}",
569 decl->getID());
570 }
571 }
572
573 // process concept body requirements '{ }' if any
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);
578
579 if (simple_req != nullptr) {
581 simple_req->getExpr(), [&concept_model](const auto *e) {
582 auto simple_expr = common::to_string(e);
583
584 LOG_DBG("=== Processing expression requirement: {}",
585 simple_expr);
586
587 concept_model.add_statement(std::move(simple_expr));
588 });
589 }
590 }
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) {
595 auto type_name = common::to_string(
596 t->getType()->getType(), cpt->getASTContext());
597
598 LOG_DBG(
599 "=== Processing type requirement: {}", type_name);
600
601 concept_model.add_statement(std::move(type_name));
602 });
603 }
604 else if (req->getKind() ==
605 clang::concepts::Requirement::RK_Nested) {
606 const auto *nested_req =
607 llvm::dyn_cast<clang::concepts::NestedRequirement>(req);
608
609 if (nested_req != nullptr) {
611 nested_req->getConstraintExpr(), [](const auto *e) {
612 LOG_DBG("=== Processing nested requirement: {}",
613 common::to_string(e));
614 });
615 }
616 }
617 else if (req->getKind() ==
618 clang::concepts::Requirement::RK_Compound) {
619 const auto *compound_req =
620 llvm::dyn_cast<clang::concepts::ExprRequirement>(req);
621
622 if (compound_req != nullptr) {
623 const auto *compound_expr_ptr = compound_req->getExpr();
624
625 if (compound_expr_ptr != nullptr) {
626 auto compound_expr =
627 common::to_string(compound_expr_ptr);
628
629 auto req_return_type =
630 compound_req->getReturnTypeRequirement();
631
632 if (!req_return_type.isEmpty()) {
633 compound_expr =
634 fmt::format("{{{}}} -> {}", compound_expr,
636 req_return_type.getTypeConstraint()));
637 }
638 else if (compound_req->hasNoexceptRequirement()) {
639 compound_expr =
640 fmt::format("{{{}}} noexcept", compound_expr);
641 }
642
643 LOG_DBG("=== Processing compound requirement: {}",
644 compound_expr);
645
646 concept_model.add_statement(std::move(compound_expr));
647 }
648 }
649 }
650 }
651 }
652 else if (const auto *binop = llvm::dyn_cast<clang::BinaryOperator>(expr);
653 binop) {
654 process_constraint_requirements(cpt, binop->getLHS(), concept_model);
655 process_constraint_requirements(cpt, binop->getRHS(), concept_model);
656 }
657 else if (const auto *unop = llvm::dyn_cast<clang::UnaryOperator>(expr);
658 unop) {
659 process_constraint_requirements(cpt, unop->getSubExpr(), concept_model);
660 }
661}
662
664 clanguml::common::model::element &c, const clang::Expr *expr)
665{
666 if (expr == nullptr)
667 return;
668 found_relationships_t relationships;
669
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)
675 continue;
676 find_relationships({}, ta.getArgument().getAsType(),
677 relationships, relationship_t::kConstraint);
678 }
679 });
680
681 common::if_dyn_cast<clang::ConceptSpecializationExpr>(
682 expr, [&](const auto *cs) {
684 });
685
686 common::if_dyn_cast<clang::RequiresExpr>(expr, [&](const auto *re) {
687 // TODO
688 });
689
690 common::if_dyn_cast<clang::BinaryOperator>(expr, [&](const auto *op) {
693 });
694
695 common::if_dyn_cast<clang::UnaryOperator>(expr, [&](const auto *op) {
697 });
698
699 for (const auto &[type_element_id, relationship_type, source_decl] :
700 relationships) {
701 if (type_element_id != c.id() &&
702 (relationship_type != relationship_t::kNone)) {
703
704 relationship r{relationship_type, type_element_id};
705
706 if (source_decl != nullptr) {
707 set_source_location(*source_decl, r);
708 }
709
710 c.add_relationship(std::move(r));
711 }
712 }
713}
714
717 const clang::ConceptSpecializationExpr *concept_specialization)
718{
719
720 if (const auto *cpt = concept_specialization->getNamedConcept();
721 should_include(cpt)) {
722
723 const auto cpt_name = cpt->getNameAsString();
724 const eid_t ast_id{cpt->getID()};
725 const auto maybe_id = id_mapper().get_global_id(ast_id);
726 if (!maybe_id)
727 return;
728
729 const auto target_id = maybe_id.value();
730
731 std::vector<std::string> constrained_template_params;
732
733 size_t argument_index{};
734
735 for (const auto ta : concept_specialization->getTemplateArguments()) {
736 if (ta.getKind() == clang::TemplateArgument::Type) {
737 auto type_name =
738 common::to_string(ta.getAsType(), cpt->getASTContext());
739 extract_constrained_template_param_name(concept_specialization,
740 cpt, constrained_template_params, argument_index,
741 type_name);
742 }
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();
748 auto type_name =
749 common::to_string(pack_head, cpt->getASTContext());
751 concept_specialization, cpt,
752 constrained_template_params, argument_index, type_name);
753 }
754 }
755 else {
756 LOG_DBG("Unsupported concept type parameter in concept: {}",
757 cpt_name);
758 }
759 argument_index++;
760 }
761
762 if (!constrained_template_params.empty())
764 {relationship_t::kConstraint, target_id, access_t::kNone,
765 fmt::format(
766 "{}", fmt::join(constrained_template_params, ","))});
767 }
768}
769
770bool translation_unit_visitor::VisitCXXRecordDecl(clang::CXXRecordDecl *cls)
771{
772 if (!should_include(cls))
773 return true;
774
775 LOG_DBG("= Visiting class declaration {} at {}",
776 cls->getQualifiedNameAsString(),
777 cls->getLocation().printToString(source_manager()));
778
779 LOG_DBG(
780 "== getQualifiedNameAsString() = {}", cls->getQualifiedNameAsString());
781 if (cls->getOwningModule() != nullptr)
782 LOG_DBG(
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());
789
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());
795 }
796
797 if (has_processed_template_class(cls->getQualifiedNameAsString()))
798 // If we have already processed the template of this class
799 // skip it
800 return true;
801
802 if (cls->isTemplated() && (cls->getDescribedTemplate() != nullptr)) {
803 // If the described templated of this class is already in the model
804 // skip it:
805 const eid_t ast_id{cls->getDescribedTemplate()->getID()};
806 if (id_mapper().get_global_id(ast_id))
807 return true;
808 }
809
810 // TODO: Add support for classes defined in function/method bodies
811 if (cls->isLocalClass() != nullptr)
812 return true;
813
814 auto c_ptr = create_declaration(cls);
815
816 if (!c_ptr)
817 return true;
818
819 return add_or_update(cls, std::move(c_ptr));
820}
821
822std::unique_ptr<clanguml::class_diagram::model::concept_>
824{
825 assert(cpt != nullptr);
826
827 if (!should_include(cpt))
828 return {};
829
830 auto concept_ptr{
831 std::make_unique<model::concept_>(config().using_namespace())};
832 auto &concept_model = *concept_ptr;
833
834 auto ns = common::get_template_namespace(*cpt);
835
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)));
839
840 process_comment(*cpt, concept_model);
841 set_source_location(*cpt, concept_model);
842 set_owning_module(*cpt, concept_model);
843
844 if (concept_model.skip())
845 return {};
846
847 concept_model.set_style(concept_model.style_spec());
848
849 return concept_ptr;
850}
851
853 clang::RecordDecl *rec)
854{
855 assert(rec != nullptr);
856
857 if (!should_include(rec))
858 return {};
859
860 auto record_ptr{std::make_unique<class_>(config().using_namespace())};
861 auto &record = *record_ptr;
862
863 process_record_parent(rec, record, namespace_{});
864
865 if (!record.is_nested()) {
866 auto record_name = rec->getQualifiedNameAsString();
867
868#if LLVM_VERSION_MAJOR < 16
869 if (record_name == "(anonymous)") {
870 util::if_not_null(rec->getTypedefNameForAnonDecl(),
871 [&record_name](const clang::TypedefNameDecl *name) {
872 record_name = name->getNameAsString();
873 });
874 }
875#endif
876
877 record.set_name(record_name);
878 record.set_id(common::to_id(record.full_name(false)));
879 }
880
881 process_comment(*rec, record);
882 set_source_location(*rec, record);
883 set_owning_module(*rec, record);
884
885 const auto record_full_name = record_ptr->full_name(false);
886
887 record.is_struct(rec->isStruct());
888 record.is_union(rec->isUnion());
889
890 if (record.skip())
891 return {};
892
893 record.set_style(record.style_spec());
894
895 return record_ptr;
896}
897
899 clang::CXXRecordDecl *cls)
900{
901 assert(cls != nullptr);
902
903 if (!should_include(cls))
904 return {};
905
906 auto c_ptr{std::make_unique<class_>(config().using_namespace())};
907 auto &c = *c_ptr;
908
909 auto ns{common::get_tag_namespace(*cls)};
910
911 process_record_parent(cls, c, ns);
912
913 if (!c.is_nested()) {
914 c.set_name(common::get_tag_name(*cls));
915 c.set_namespace(ns);
916 c.set_id(common::to_id(c.full_name(false)));
917 }
918
919 c.is_struct(cls->isStruct());
920
921 process_comment(*cls, c);
922 set_source_location(*cls, c);
923 set_owning_module(*cls, c);
924
925 if (c.skip())
926 return {};
927
928 c.set_style(c.style_spec());
929
930 return c_ptr;
931}
932
933std::unique_ptr<clanguml::class_diagram::model::objc_interface>
935 clang::ObjCCategoryDecl *decl)
936{
937 assert(decl != nullptr);
938
939 if (!should_include(decl))
940 return {};
941
942 auto c_ptr{std::make_unique<class_diagram::model::objc_interface>(
943 config().using_namespace())};
944 auto &c = *c_ptr;
945
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())));
950 c.is_category(true);
951
952 process_comment(*decl, c);
953 set_source_location(*decl, c);
954
955 if (c.skip())
956 return {};
957
958 c.set_style(c.style_spec());
959
960 return c_ptr;
961}
962
963std::unique_ptr<clanguml::class_diagram::model::objc_interface>
965 clang::ObjCProtocolDecl *decl)
966{
967 assert(decl != nullptr);
968
969 if (!should_include(decl))
970 return {};
971
972 auto c_ptr{std::make_unique<class_diagram::model::objc_interface>(
973 config().using_namespace())};
974 auto &c = *c_ptr;
975
976 c.set_name(decl->getNameAsString());
977 c.set_id(common::to_id(*decl));
978 c.is_protocol(true);
979
980 process_comment(*decl, c);
981 set_source_location(*decl, c);
982
983 if (c.skip())
984 return {};
985
986 c.set_style(c.style_spec());
987
988 return c_ptr;
989}
990
991std::unique_ptr<clanguml::class_diagram::model::objc_interface>
993 clang::ObjCInterfaceDecl *decl)
994{
995 assert(decl != nullptr);
996
997 if (!should_include(decl))
998 return {};
999
1000 auto c_ptr{std::make_unique<class_diagram::model::objc_interface>(
1001 config().using_namespace())};
1002 auto &c = *c_ptr;
1003
1004 c.set_name(decl->getNameAsString());
1005 c.set_id(common::to_id(*decl));
1006
1007 process_comment(*decl, c);
1008 set_source_location(*decl, c);
1009
1010 if (c.skip())
1011 return {};
1012
1013 c.set_style(c.style_spec());
1014
1015 return c_ptr;
1016}
1017
1019 clang::RecordDecl *cls, class_ &c, const namespace_ &ns)
1020{
1021 std::optional<eid_t> id_opt;
1022 namespace_ parent_ns = ns;
1023
1024 find_record_parent_id(cls, id_opt, parent_ns);
1025
1026 if (id_opt && diagram().find<class_>(*id_opt)) {
1027 process_record_parent_by_type<class_>(*id_opt, c, parent_ns, cls);
1028 }
1029
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);
1033 }
1034}
1035
1037 const clang::CXXRecordDecl &cls, clanguml::class_diagram::model::class_ &c)
1038{
1039 // Process class child entities
1040 process_children(&cls, c);
1041
1042 // Process class bases
1043 process_class_bases(&cls, c);
1044
1045 c.complete(true);
1046}
1047
1049 const clang::ObjCCategoryDecl &cls, objc_interface &c)
1050{
1051 assert(c.is_category());
1052
1053 // Iterate over class methods (both regular and static)
1054 for (const auto *method : cls.methods()) {
1055 if (method != nullptr) {
1056 process_objc_method(*method, c);
1057 }
1058 }
1059
1060 // Add relationship to the ObjC Interface being extended by this
1061 // category
1062 if (cls.getClassInterface() != nullptr) {
1063 eid_t objc_interface_id = common::to_id(*cls.getClassInterface());
1065 relationship_t::kInstantiation, objc_interface_id, access_t::kNone};
1066
1067 LOG_DBG("Found protocol {} [{}] for ObjC interface {}",
1068 cls.getClassInterface()->getNameAsString(),
1069 objc_interface_id.value(), c.name());
1070
1071 c.add_relationship(std::move(r));
1072 }
1073
1074 c.complete(true);
1075}
1076
1078 const clang::ObjCProtocolDecl &cls, objc_interface &c)
1079{
1080 assert(c.is_protocol());
1081
1082 // Iterate over class methods (both regular and static)
1083 for (const auto *method : cls.methods()) {
1084 if (method != nullptr) {
1085 process_objc_method(*method, c);
1086 }
1087 }
1088
1089 c.complete(true);
1090}
1091
1093 const clang::ObjCInterfaceDecl &cls, objc_interface &c)
1094{
1095 assert(!c.is_protocol() && !c.is_category());
1096
1097 // Iterate over class methods (both regular and static)
1098 for (const auto *method : cls.methods()) {
1099 if (method != nullptr) {
1100 process_objc_method(*method, c);
1101 }
1102 }
1103
1104 for (const auto *ivar : cls.ivars()) {
1105 if (ivar != nullptr) {
1106 process_objc_ivar(*ivar, c);
1107 }
1108 }
1109
1111
1112 c.complete(true);
1113}
1115 const clang::ObjCInterfaceDecl &cls, objc_interface &c)
1116{
1117 if (const auto *base = cls.getSuperClass(); base != nullptr) {
1118 eid_t parent_id = common::to_id(*base);
1119 common::model::relationship cp{parent_id, access_t::kNone, false};
1120
1121 LOG_DBG("Found base class {} [{}] for ObjC interface {}",
1122 base->getNameAsString(), parent_id.value(), c.name());
1123
1124 c.add_relationship(std::move(cp));
1125 }
1126
1127 for (const auto *protocol : cls.protocols()) {
1128 eid_t parent_id = common::to_id(*protocol);
1130 relationship_t::kInstantiation, parent_id, access_t::kNone};
1131
1132 LOG_DBG("Found protocol {} [{}] for ObjC interface {}",
1133 protocol->getNameAsString(), parent_id.value(), c.name());
1134
1135 c.add_relationship(std::move(cp));
1136 }
1137}
1138
1140 const clang::ObjCIvarDecl &ivar, objc_interface &c)
1141{
1142 LOG_DBG("== Visiting ObjC ivar {}", ivar.getNameAsString());
1143
1144 // Default hint for relationship is aggregation
1145 auto relationship_hint = relationship_t::kAggregation;
1146
1147 auto field_type = ivar.getType();
1148 auto field_type_str =
1149 common::to_string(field_type, ivar.getASTContext(), false);
1150
1151 auto type_name = common::to_string(field_type, ivar.getASTContext());
1152
1153 const auto field_name = ivar.getNameAsString();
1154
1155 objc_member field{
1156 common::access_specifier_to_access_t(ivar.getAccessControl()),
1157 field_name, field_type_str};
1158
1159 field.set_qualified_name(ivar.getQualifiedNameAsString());
1160
1161 process_comment(ivar, field);
1162 set_source_location(ivar, field);
1163
1164 if (field.skip())
1165 return;
1166
1167 if (field_type->isObjCObjectPointerType()) {
1168 relationship_hint = relationship_t::kAggregation;
1169 field_type = field_type->getPointeeType();
1170 }
1171 else if (field_type->isPointerType()) {
1172 relationship_hint = relationship_t::kAssociation;
1173 field_type = field_type->getPointeeType();
1174 }
1175 else if (field_type->isLValueReferenceType()) {
1176 relationship_hint = relationship_t::kAssociation;
1177 field_type = field_type.getNonReferenceType();
1178 }
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)
1184 field.set_destination_multiplicity(common::get_array_size(
1185 *field_type->getAsArrayTypeUnsafe()));
1186 else {
1187 auto maybe_array_size =
1188 common::get_array_size(*field_type->getAsArrayTypeUnsafe());
1189 if (maybe_array_size.has_value()) {
1190 field.set_destination_multiplicity(
1191 current_multiplicity.value() *
1192 maybe_array_size.value());
1193 }
1194 }
1195
1196 field_type = field_type->getAsArrayTypeUnsafe()->getElementType();
1197 }
1198 }
1199 else if (field_type->isRValueReferenceType()) {
1200 field_type = field_type.getNonReferenceType();
1201 }
1202
1203 found_relationships_t relationships;
1204
1205 if (!field.skip_relationship()) {
1206 // Find relationship for the type if the type has not been added
1207 // as aggregation
1208 if (field_type->getAsObjCInterfaceType() != nullptr &&
1209 field_type->getAsObjCInterfaceType()->getInterface() != nullptr) {
1210 const auto *objc_iface =
1211 field_type->getAsObjCInterfaceType()->getInterface();
1212
1213 relationships.emplace_back(common::to_id(*objc_iface),
1214 relationship_t::kAggregation, &ivar);
1215 }
1216 else if ((field_type->getAsRecordDecl() != nullptr) &&
1217 field_type->getAsRecordDecl()->getNameAsString().empty()) {
1218 // Relationships to fields whose type is an anonymous and nested
1219 // struct have to be handled separately here
1220 anonymous_struct_relationships_[field_type->getAsRecordDecl()
1221 ->getID()] =
1222 std::make_tuple(field.name(), relationship_hint, field.access(),
1223 field.destination_multiplicity());
1224 }
1225 else
1227 &ivar, field_type, relationships, relationship_hint);
1228
1229 add_relationships(c, field, relationships);
1230 }
1231
1232 // If this is an anonymous struct - replace the anonymous_XYZ part with
1233 // field name
1234 if ((field_type->getAsRecordDecl() != nullptr) &&
1235 field_type->getAsRecordDecl()->getNameAsString().empty()) {
1236 if (util::contains(field.type(), "(anonymous_")) {
1237 std::regex anonymous_re("anonymous_(\\d*)");
1238 field.set_type(
1239 std::regex_replace(field.type(), anonymous_re, field_name));
1240 }
1241 }
1242
1243 c.add_member(std::move(field));
1244}
1245
1247 const clang::CXXRecordDecl *cls, class_ &c)
1248{
1249 for (const auto &base : cls->bases()) {
1250 eid_t parent_id;
1251 if (const auto *tsp =
1252 base.getType()->getAs<clang::TemplateSpecializationType>();
1253 tsp != nullptr) {
1254 auto template_specialization_ptr =
1255 std::make_unique<class_>(config().using_namespace());
1257 *cls, *template_specialization_ptr, cls, *tsp, {});
1258
1259 parent_id = template_specialization_ptr->id();
1260
1261 if (diagram().should_include(*template_specialization_ptr)) {
1262 add_class(std::move(template_specialization_ptr));
1263 }
1264 }
1265 else if (const auto *record_type =
1266 base.getType()->getAs<clang::RecordType>();
1267 record_type != nullptr) {
1268 parent_id = common::to_id(*record_type->getDecl());
1269 }
1270 else
1271 // This could be a template parameter - we don't want it here
1272 continue;
1273
1274 common::model::relationship cp{parent_id,
1275 common::access_specifier_to_access_t(base.getAccessSpecifier()),
1276 base.isVirtual()};
1277
1278 LOG_DBG("Found base class {} [{}] for class {}",
1279 common::to_string(base.getType(), cls->getASTContext()),
1280 parent_id.value(), c.name());
1281
1282 c.add_relationship(std::move(cp));
1283 }
1284}
1285
1287 const clang::RecordDecl &cls, class_ &c)
1288{
1289 // Iterate over C struct fields
1290 for (const auto *field : cls.fields()) {
1291 if (field != nullptr)
1292 process_field(*field, c);
1293 }
1294
1295 c.complete(true);
1296}
1297
1299 const clang::CXXRecordDecl *cls, clanguml::class_diagram::model::class_ &c)
1300{
1301 assert(cls != nullptr);
1302
1303 // Iterate over class methods (both regular and static)
1304 for (const auto *method : cls->methods()) {
1305 if (method != nullptr) {
1306 process_method(*method, c);
1307 }
1308 }
1309
1310 // Iterate over class template methods
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>(
1317 decl_iterator);
1318 if (method_template == nullptr)
1319 continue;
1320
1321 process_template_method(*method_template, c);
1322 }
1323 }
1324
1325 // Iterate over regular class fields
1326 for (const auto *field : cls->fields()) {
1327 if (field != nullptr)
1328 process_field(*field, c);
1329 }
1330
1331 // First we have to collect any `typedef enum` declarations, which should
1332 // not be rendered as members (typedefs are visited here after enums)
1333 std::set<const clang::EnumDecl *> typedeffed_enums;
1334 for (const auto *decl : cls->decls()) {
1335 if (decl->getKind() == clang::Decl::Typedef) {
1336 const auto *typedeffed_enum = common::get_typedef_enum_decl(
1337 clang::dyn_cast<clang::TypedefDecl>(decl));
1338 if (typedeffed_enum != nullptr)
1339 typedeffed_enums.emplace(typedeffed_enum);
1340 }
1341 }
1342
1343 // Static fields have to be processed by iterating over variable
1344 // declarations
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()) {
1351 process_static_field(*variable_declaration, c);
1352 }
1353 }
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)
1360 continue;
1361
1362 if (enum_decl->getNameAsString().empty()) {
1363 for (const auto *enum_const : enum_decl->enumerators()) {
1365 enum_decl->getAccess()),
1366 enum_const->getNameAsString(), "enum"};
1367 c.add_member(std::move(m));
1368 }
1369 }
1370 }
1371 }
1372
1373 if (cls->isCompleteDefinition())
1374 for (const auto *friend_declaration : cls->friends()) {
1375 if (friend_declaration != nullptr)
1376 process_friend(*friend_declaration, c);
1377 }
1378}
1379
1381 const clang::FriendDecl &f, class_ &c)
1382{
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>() !=
1386 nullptr) {
1387 // TODO: handle template friend
1388 }
1389 else if (friend_type->getAs<clang::RecordType>() != nullptr) {
1390 if (should_include(friend_type->getAsRecordDecl())) {
1391 relationship r{relationship_t::kFriendship,
1392 common::to_id(*friend_type->getAsRecordDecl()),
1394 "<<friend>>"};
1395
1396 c.add_relationship(std::move(r));
1397 }
1398 }
1399 }
1400}
1401
1403 const clang::CXXMethodDecl &mf, class_ &c)
1404{
1405 // TODO: For now skip implicitly default methods
1406 // in the future, add config option to choose
1407 if (mf.isDefaulted() && !mf.isExplicitlyDefaulted())
1408 return;
1409
1410 auto method_return_type =
1411 common::to_string(mf.getReturnType(), mf.getASTContext());
1412
1413 common::ensure_lambda_type_is_relative(config(), method_return_type);
1414
1415 auto method_name = mf.getNameAsString();
1416 if (mf.isTemplated()) {
1417 // Sometimes in template specializations method names contain the
1418 // template parameters for some reason - drop them
1419 // Is there a better way to do this?
1420 method_name = method_name.substr(0, method_name.find('<'));
1421 }
1422
1424 util::trim(method_name),
1425 config().simplify_template_type(method_return_type)};
1426
1427 method.set_qualified_name(mf.getQualifiedNameAsString());
1428
1429 process_method_properties(mf, c, method_name, method);
1430
1431 process_comment(mf, method);
1432
1433 // Register the source location of the field declaration
1434 set_source_location(mf, method);
1435
1436 if (method.skip())
1437 return;
1438
1439 for (const auto *param : mf.parameters()) {
1440 if (param != nullptr)
1441 process_function_parameter(*param, method, c);
1442 }
1443
1444 // find relationship for return type
1445 found_relationships_t relationships;
1446
1447 // Move dereferencing to build() method of template_builder
1448 if (const auto *templ = mf.getReturnType()
1449 .getNonReferenceType()
1450 .getUnqualifiedType()
1451 ->getAs<clang::TemplateSpecializationType>();
1452 templ != nullptr) {
1453 const auto *unaliased_type = templ;
1454 if (unaliased_type->isTypeAlias())
1455 unaliased_type = unaliased_type->getAliasedType()
1456 ->getAs<clang::TemplateSpecializationType>();
1457
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);
1465
1466 template_specialization_ptr->is_template();
1467
1468 if (diagram().should_include(*template_specialization_ptr)) {
1469 relationships.emplace_back(template_specialization_ptr->id(),
1470 relationship_t::kDependency, &mf);
1471
1472 add_class(std::move(template_specialization_ptr));
1473 }
1474 }
1475 }
1476
1478 &mf, mf.getReturnType(), relationships, relationship_t::kDependency);
1479
1480 for (const auto &[type_element_id, relationship_type, source_decl] :
1481 relationships) {
1482 if (type_element_id != c.id() &&
1483 (relationship_type != relationship_t::kNone)) {
1484 relationship r{relationship_t::kDependency, type_element_id};
1485
1486 if (source_decl != nullptr) {
1487 set_source_location(*source_decl, r);
1488 }
1489
1490 LOG_DBG("Adding method return type relationship from {}::{} to "
1491 "{}: {}",
1492 c, mf.getNameAsString(), r.type(), r.label());
1493
1494 c.add_relationship(std::move(r));
1495 }
1496 }
1497
1498 // Also consider the container itself if it is a template
1499 // instantiation it's arguments could count as reference to relevant
1500 // types
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();
1506
1507 if (const auto *atsp = underlying_type->getAs<clang::AutoType>();
1508 atsp != nullptr) {
1510 }
1511
1512 method.update(config().using_namespace());
1513
1514 if (diagram().should_include(method)) {
1515 LOG_DBG("Adding method: {}", method.name());
1516
1517 c.add_method(std::move(method));
1518 }
1519}
1520
1522 const clang::ObjCMethodDecl &mf, objc_interface &c)
1523{
1524 auto method_return_type =
1525 common::to_string(mf.getReturnType(), mf.getASTContext());
1526
1528 util::trim(mf.getNameAsString()), method_return_type};
1529
1530 method.set_qualified_name(mf.getQualifiedNameAsString());
1531
1532 process_comment(mf, method);
1533
1534 // Register the source location of the field declaration
1535 set_source_location(mf, method);
1536
1537 if (method.skip())
1538 return;
1539
1540 method.is_static(mf.isClassMethod());
1541 method.is_optional(mf.isOptional());
1542
1543 for (const auto *param : mf.parameters()) {
1544 if (param != nullptr)
1545 process_objc_method_parameter(*param, method, c);
1546 }
1547
1548 // find relationship for return type
1549 found_relationships_t relationships;
1550
1552 &mf, mf.getReturnType(), relationships, relationship_t::kDependency);
1553
1554 for (const auto &[type_element_id, relationship_type, source_decl] :
1555 relationships) {
1556 if (type_element_id != c.id() &&
1557 (relationship_type != relationship_t::kNone)) {
1558 relationship r{relationship_t::kDependency, type_element_id};
1559
1560 if (source_decl != nullptr) {
1561 set_source_location(*source_decl, r);
1562 }
1563
1564 LOG_DBG("Adding method return type relationship from {}::{} to "
1565 "{}: {}",
1566 c, mf.getNameAsString(), r.type(), r.label());
1567
1568 c.add_relationship(std::move(r));
1569 }
1570 }
1571
1572 // TODO
1573 if (diagram().should_include(method)) {
1574 LOG_DBG("Adding ObjC method: {}", method.name());
1575
1576 c.add_method(std::move(method));
1577 }
1578}
1579
1581 const clang::CXXMethodDecl &mf, const class_ &c,
1582 const std::string &method_name, class_method &method) const
1583{
1584 const bool is_constructor = c.name() == method_name;
1585 const bool is_destructor = fmt::format("~{}", c.name()) == method_name;
1586
1587#if LLVM_VERSION_MAJOR > 17
1588 method.is_pure_virtual(mf.isPureVirtual());
1589#else
1590 method.is_pure_virtual(mf.isPure());
1591#endif
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()));
1605 method.is_coroutine(common::is_coroutine(mf));
1606}
1607
1610 class_ &c, const clang::AutoType *atsp)
1611{
1612 auto desugared_atsp = atsp->getDeducedType();
1613
1614 if (atsp->isSugared()) {
1615 const auto *deduced_type =
1616 atsp->desugar()->getAs<clang::DeducedTemplateSpecializationType>();
1617
1618 if (deduced_type != nullptr)
1619 desugared_atsp = deduced_type->getDeducedType();
1620 }
1621
1622 if (desugared_atsp.isNull())
1623 return;
1624
1625 const auto *deduced_record_type = desugared_atsp->isRecordType()
1626 ? desugared_atsp->getAs<clang::RecordType>()
1627 : nullptr;
1628
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) {
1634
1635 const auto diagram_class_count_before_visit =
1636 diagram().classes().size();
1637
1638 VisitClassTemplateSpecializationDecl(deduced_auto_decl);
1639
1640 const bool visitor_added_new_template_specialization =
1641 (diagram().classes().size() -
1642 diagram_class_count_before_visit) > 0;
1643
1644 if (visitor_added_new_template_specialization) {
1645 const auto &template_specialization_model =
1646 diagram().classes().back();
1647
1648 if (should_include(deduced_auto_decl)) {
1649 relationship r{relationship_t::kDependency,
1650 template_specialization_model.get().id()};
1651
1652 c.add_relationship(std::move(r));
1653 }
1654 }
1655 }
1656 }
1657}
1658
1660 const clang::FunctionTemplateDecl &mf, class_ &c)
1661{
1662 // TODO: For now skip implicitly default methods
1663 // in the future, add config option to choose
1664 if (mf.getTemplatedDecl()->isDefaulted() &&
1665 !mf.getTemplatedDecl()->isExplicitlyDefaulted())
1666 return;
1667
1668 auto method_name = mf.getNameAsString();
1669 if (mf.isTemplated()) {
1670 // Sometimes in template specializations method names contain the
1671 // template parameters for some reason - drop them
1672 // Is there a better way to do this?
1673 method_name = util::trim(method_name.substr(0, method_name.find('<')));
1674 }
1675
1677 method_name, mf.getTemplatedDecl()->getReturnType().getAsString()};
1678
1680 clang::dyn_cast<clang::CXXMethodDecl>(mf.getTemplatedDecl()),
1681 [&](const auto *decl) {
1682 process_method_properties(*decl, c, method_name, method);
1683 });
1684
1686
1687 process_comment(mf, method);
1688
1689 if (method.skip())
1690 return;
1691
1692 for (const auto *param : mf.getTemplatedDecl()->parameters()) {
1693 if (param != nullptr)
1694 process_function_parameter(*param, method, c);
1695 }
1696
1697 method.update(config().using_namespace());
1698
1699 if (diagram().should_include(method)) {
1700 LOG_DBG("Adding method: {}", method.name());
1701
1702 c.add_method(std::move(method));
1703 }
1704}
1705
1707 const clang::QualType &type, found_relationships_t &relationships,
1709{
1710 bool result{false};
1711
1712 if (type->isPointerType()) {
1713 relationship_hint = relationship_t::kAssociation;
1715 decl, type->getPointeeType(), relationships, relationship_hint);
1716 }
1717 else if (type->isRValueReferenceType()) {
1718 relationship_hint = relationship_t::kAggregation;
1720 decl, type.getNonReferenceType(), relationships, relationship_hint);
1721 }
1722 else if (type->isLValueReferenceType()) {
1723 relationship_hint = relationship_t::kAssociation;
1725 decl, type.getNonReferenceType(), relationships, relationship_hint);
1726 }
1727 else if (type->isArrayType()) {
1728 find_relationships(decl, type->getAsArrayTypeUnsafe()->getElementType(),
1729 relationships, relationship_t::kAggregation);
1730 }
1731 else if (type->isEnumeralType()) {
1732 if (const auto *enum_type = type->getAs<clang::EnumType>();
1733 enum_type != nullptr) {
1734 // Use AST's local ID here for relationship target, as we can't
1735 // calculate here properly the ID for nested enums. It will be
1736 // resolved properly in finalize().
1737 relationships.emplace_back(
1738 enum_type->getDecl()->getID(), relationship_hint, decl);
1739 }
1740 }
1741 // TODO: Objc support
1742 else if (type->isRecordType()) {
1743 const auto *type_instantiation_type =
1744 type->getAs<clang::TemplateSpecializationType>();
1745
1746 if (type_instantiation_type != nullptr) {
1747 const auto *type_instantiation_template_decl =
1748 type_instantiation_type->getTemplateName().getAsTemplateDecl();
1749
1750 // If this template should be included in the diagram
1751 // add it - and then process recursively its arguments
1752 if (should_include(type_instantiation_template_decl)) {
1753 relationships.emplace_back(
1754 type_instantiation_type->getTemplateName()
1755 .getAsTemplateDecl()
1756 ->getID(),
1757 relationship_hint, decl);
1758 }
1759
1760 auto idx{0};
1761 for (const auto &template_argument :
1762 type_instantiation_type->template_arguments()) {
1763
1764 auto [overridden_relationship_hint, overridden] =
1765 override_relationship_hint(type_instantiation_template_decl
1766 ->getQualifiedNameAsString(),
1767 idx, relationship_hint);
1768
1769 const auto template_argument_kind = template_argument.getKind();
1770 if (template_argument_kind ==
1771 clang::TemplateArgument::ArgKind::Integral) {
1772 // pass
1773 }
1774 else if (template_argument_kind ==
1775 clang::TemplateArgument::ArgKind::Null) {
1776 // pass
1777 }
1778 else if (template_argument_kind ==
1779 clang::TemplateArgument::ArgKind::Expression) {
1780 // pass
1781 }
1782 else if (template_argument.getKind() ==
1783 clang::TemplateArgument::ArgKind::NullPtr) {
1784 // pass
1785 }
1786 else if (template_argument_kind ==
1787 clang::TemplateArgument::ArgKind::Template) {
1788 // pass
1789 }
1790 else if (template_argument_kind ==
1791 clang::TemplateArgument::ArgKind::TemplateExpansion) {
1792 // pass
1793 }
1794 else if (const auto *function_type =
1795 template_argument.getAsType()
1796 ->getAs<clang::FunctionProtoType>();
1797 function_type != nullptr) {
1798 for (const auto &param_type :
1799 function_type->param_types()) {
1800 result = find_relationships(decl, param_type,
1801 relationships, relationship_t::kDependency);
1802 }
1803 }
1804 else if (template_argument_kind ==
1805 clang::TemplateArgument::ArgKind::Type) {
1806 result =
1807 find_relationships(decl, template_argument.getAsType(),
1808 relationships, overridden_relationship_hint);
1809 }
1810 idx++;
1811 }
1812 }
1813 else if (type->getAsCXXRecordDecl() != nullptr) {
1814 relationships.emplace_back(
1815 type->getAsCXXRecordDecl()->getID(), relationship_hint, decl);
1816 result = true;
1817 }
1818 else {
1819 relationships.emplace_back(
1820 type->getAsRecordDecl()->getID(), relationship_hint, decl);
1821 result = true;
1822 }
1823 }
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()
1834 ->getID(),
1835 relationship_hint, decl);
1836 }
1837 auto idx{0};
1838 for (const auto &template_argument :
1839 template_specialization_type->template_arguments()) {
1840
1841 auto [overridden_relationship_hint, overridden] =
1842 override_relationship_hint(type_instantiation_template_decl
1843 ->getQualifiedNameAsString(),
1844 idx, relationship_hint);
1845
1846 const auto template_argument_kind = template_argument.getKind();
1847 if (template_argument_kind ==
1848 clang::TemplateArgument::ArgKind::Integral) {
1849 // pass
1850 }
1851 else if (template_argument_kind ==
1852 clang::TemplateArgument::ArgKind::Null) {
1853 // pass
1854 }
1855 else if (template_argument_kind ==
1856 clang::TemplateArgument::ArgKind::Expression) {
1857 // pass
1858 }
1859 else if (template_argument.getKind() ==
1860 clang::TemplateArgument::ArgKind::NullPtr) {
1861 // pass
1862 }
1863 else if (template_argument_kind ==
1864 clang::TemplateArgument::ArgKind::Template) {
1865 // pass
1866 }
1867 else if (template_argument_kind ==
1868 clang::TemplateArgument::ArgKind::TemplateExpansion) {
1869 // pass
1870 }
1871 else if (const auto *function_type =
1872 template_argument.getAsType()
1873 ->getAs<clang::FunctionProtoType>();
1874 function_type != nullptr) {
1875 for (const auto &param_type : function_type->param_types()) {
1876 result = find_relationships(decl, param_type, relationships,
1877 relationship_t::kDependency);
1878 }
1879 }
1880 else if (template_argument_kind ==
1881 clang::TemplateArgument::ArgKind::Type) {
1882 result = find_relationships(decl, template_argument.getAsType(),
1883 relationships, overridden_relationship_hint);
1884 }
1885 idx++;
1886 }
1887 }
1888
1889 return result;
1890}
1891
1893 const clang::ParmVarDecl &param, objc_method &method, objc_interface &c)
1894{
1895 method_parameter parameter;
1896 parameter.set_name(param.getNameAsString());
1897
1898 process_comment(param, parameter);
1899
1900 if (parameter.skip())
1901 return;
1902
1903 auto parameter_type =
1904 common::to_string(param.getType(), param.getASTContext());
1905 parameter.set_type(parameter_type);
1906
1907 if (!parameter.skip_relationship()) {
1908 // find relationship for the type
1909 found_relationships_t relationships;
1910
1911 LOG_DBG("Looking for relationships in type: {}",
1912 common::to_string(param.getType(), param.getASTContext()));
1913
1914 find_relationships(&param, param.getType(), relationships,
1915 relationship_t::kDependency);
1916
1917 for (const auto &[type_element_id, relationship_type, source_decl] :
1918 relationships) {
1919 if (type_element_id != c.id() &&
1920 (relationship_type != relationship_t::kNone)) {
1921 relationship r{relationship_t::kDependency, type_element_id};
1922
1923 if (source_decl != nullptr) {
1924 set_source_location(*source_decl, r);
1925 }
1926
1927 LOG_DBG("Adding ObjC method parameter relationship from {} to "
1928 "{}: {}",
1929 c, r.type(), r.label());
1930
1931 c.add_relationship(std::move(r));
1932 }
1933 }
1934 }
1935
1936 method.add_parameter(std::move(parameter));
1937}
1938
1940 const clang::ParmVarDecl &p, class_method &method, class_ &c,
1941 const std::set<std::string> & /*template_parameter_names*/)
1942{
1943 method_parameter parameter;
1944 parameter.set_name(p.getNameAsString());
1945
1946 process_comment(p, parameter);
1947
1948 if (parameter.skip())
1949 return;
1950
1951 auto parameter_type = common::to_string(p.getType(), p.getASTContext());
1952
1953 // Is there no better way to determine that 'type' is a lambda?
1955
1956 parameter.set_type(parameter_type);
1957
1958 if (p.hasDefaultArg()) {
1959 const auto *default_arg = p.getDefaultArg();
1960 if (default_arg != nullptr) {
1961 auto default_arg_str = common::get_source_text(
1962 default_arg->getSourceRange(), source_manager());
1963 parameter.set_default_value(default_arg_str);
1964 }
1965 }
1966
1967 if (!parameter.skip_relationship()) {
1968 // find relationship for the type
1969 found_relationships_t relationships;
1970
1971 LOG_DBG("Looking for relationships in type: {}",
1972 common::to_string(p.getType(), p.getASTContext()));
1973
1974 if (const auto *templ =
1975 p.getType()
1976 .getNonReferenceType()
1977 .getUnqualifiedType()
1978 ->getAs<clang::TemplateSpecializationType>();
1979 templ != nullptr) {
1980 auto template_specialization_ptr =
1981 std::make_unique<class_>(config().using_namespace());
1983 *template_specialization_ptr,
1984 templ->getTemplateName().getAsTemplateDecl(), *templ, &c);
1985
1986 template_specialization_ptr->is_template(true);
1987
1988 if (diagram().should_include(*template_specialization_ptr)) {
1989 relationships.emplace_back(template_specialization_ptr->id(),
1990 relationship_t::kDependency, &p);
1991
1992 add_class(std::move(template_specialization_ptr));
1993 }
1994 }
1995
1997 &p, p.getType(), relationships, relationship_t::kDependency);
1998
1999 for (const auto &[type_element_id, relationship_type, source_decl] :
2000 relationships) {
2001 if (type_element_id != c.id() &&
2002 (relationship_type != relationship_t::kNone)) {
2003 relationship r{relationship_t::kDependency, type_element_id};
2004
2005 if (source_decl != nullptr) {
2006 set_source_location(*source_decl, r);
2007 }
2008
2009 LOG_DBG("Adding function parameter relationship from {} to "
2010 "{}: {}",
2011 c, r.type(), r.label());
2012
2013 c.add_relationship(std::move(r));
2014 }
2015 }
2016 }
2017
2018 method.add_parameter(std::move(parameter));
2019}
2020
2023 const found_relationships_t &relationships, bool break_on_first_aggregation)
2024{
2025 auto [decorator_rtype, decorator_rmult] = field.get_relationship();
2026
2027 for (const auto &[target, relationship_type, source_decl] : relationships) {
2028 if (relationship_type != relationship_t::kNone) {
2029 relationship r{relationship_type, target};
2030 r.set_label(field.name());
2031 r.set_access(field.access());
2032 if (source_decl != nullptr) {
2033 set_source_location(*source_decl, r);
2034 }
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]);
2043 }
2044 }
2045 if (!mulitplicity_provided_in_comment &&
2046 field.destination_multiplicity().has_value()) {
2047 r.set_multiplicity_destination(
2048 std::to_string(*field.destination_multiplicity()));
2049 }
2050
2051 r.set_style(field.style_spec());
2052
2053 LOG_DBG("Adding relationship from {} to {} with label {}", c,
2054 r.destination(), r.type(), r.label());
2055
2056 c.add_relationship(std::move(r));
2057
2058 if (break_on_first_aggregation &&
2059 relationship_type == relationship_t::kAggregation)
2060 break;
2061 }
2062 }
2063}
2064
2065std::pair<relationship_t, bool>
2067 const std::string &type_name, int index, relationship_t hint)
2068{
2069 bool overridden{false};
2070
2071 for (const auto &[pattern, rel_hint] : config().relationship_hints()) {
2072 if (type_name.find(pattern) == 0) {
2073 if (index == -1)
2074 hint = rel_hint.default_hint;
2075 else
2076 hint = rel_hint.get(index, hint);
2077
2078 overridden = true;
2079 break;
2080 }
2081 }
2082
2083 return {hint, overridden};
2084}
2085
2087 const clang::VarDecl &field_declaration, class_ &c)
2088{
2089 const auto field_type = field_declaration.getType();
2090 auto type_name =
2091 common::to_string(field_type, field_declaration.getASTContext());
2092 if (type_name.empty())
2093 type_name = "<<anonymous>>";
2094
2095 class_member field{
2096 common::access_specifier_to_access_t(field_declaration.getAccess()),
2097 field_declaration.getNameAsString(),
2098 config().simplify_template_type(type_name)};
2099
2100 field.set_qualified_name(field_declaration.getQualifiedNameAsString());
2101
2102 field.is_static(true);
2103
2104 process_comment(field_declaration, field);
2105 set_source_location(field_declaration, field);
2106
2107 if (field.skip() || !diagram().should_include(field))
2108 return;
2109
2110 if (!field.skip_relationship()) {
2111 found_relationships_t relationships;
2112
2113 // find relationship for the type
2114 find_relationships(&field_declaration, field_declaration.getType(),
2115 relationships, relationship_t::kAssociation);
2116
2117 add_relationships(c, field, relationships);
2118 }
2119
2120 c.add_member(std::move(field));
2121}
2122
2123std::unique_ptr<class_>
2125 clang::ClassTemplateSpecializationDecl *cls)
2126{
2127 LOG_DBG("Processing template specialization {} at {}",
2128 cls->getQualifiedNameAsString(),
2129 cls->getLocation().printToString(source_manager()));
2130
2131 auto c_ptr = std::make_unique<class_>(config().using_namespace());
2133
2134 auto &template_instantiation = *c_ptr;
2135 template_instantiation.is_template(true);
2136
2137 // TODO: refactor to method get_qualified_name()
2138 auto qualified_name = cls->getQualifiedNameAsString();
2139 util::replace_all(qualified_name, "(anonymous namespace)", "");
2140 util::replace_all(qualified_name, "::::", "::");
2141
2142 namespace_ ns{qualified_name};
2143 ns.pop_back();
2144 template_instantiation.set_name(cls->getNameAsString());
2145 template_instantiation.set_namespace(ns);
2146
2147 template_instantiation.is_struct(cls->isStruct());
2148
2149 process_record_parent(cls, template_instantiation, ns);
2150
2151 if (!template_instantiation.is_nested()) {
2152 template_instantiation.set_name(common::get_tag_name(*cls));
2153 template_instantiation.set_id(
2154 common::to_id(template_instantiation.full_name(false)));
2155 }
2156
2157 process_comment(*cls, template_instantiation);
2158 set_source_location(*cls, template_instantiation);
2159 set_owning_module(*cls, template_instantiation);
2160
2161 if (template_instantiation.skip())
2162 return {};
2163
2164 id_mapper().add(cls->getID(), template_instantiation.id());
2165
2166 return c_ptr;
2167}
2168
2170 const clang::FieldDecl &field_declaration, class_ &c)
2171{
2172 LOG_DBG(
2173 "== Visiting record member {}", field_declaration.getNameAsString());
2174
2175 // Default hint for relationship is aggregation
2176 auto relationship_hint = relationship_t::kAggregation;
2177 // If the first type of the template instantiation of this field type
2178 // has been added as aggregation relationship with class 'c', don't
2179 // add it's nested template types as aggregation
2180 [[maybe_unused]] bool template_instantiation_added_as_aggregation{false};
2181 // The actual field type
2182 auto field_type = field_declaration.getType();
2183
2184 // String representation of the field type
2185 auto type_name =
2186 common::to_string(field_type, field_declaration.getASTContext());
2187 // The field name
2188 const auto field_name = field_declaration.getNameAsString();
2189
2190 auto field_type_str =
2191 common::to_string(field_type, field_declaration.getASTContext(), false);
2192
2194
2195 class_member field{
2196 common::access_specifier_to_access_t(field_declaration.getAccess()),
2197 field_name, config().simplify_template_type(field_type_str)};
2198
2199 field.set_qualified_name(field_declaration.getQualifiedNameAsString());
2200
2201 // Parse the field comment
2202 process_comment(field_declaration, field);
2203 // Register the source location of the field declaration
2204 set_source_location(field_declaration, field);
2205
2206 // If the comment contains a skip directive, just return
2207 if (field.skip())
2208 return;
2209
2210 if (field_type->isPointerType()) {
2211 relationship_hint = relationship_t::kAssociation;
2212 field_type = field_type->getPointeeType();
2213 }
2214 else if (field_type->isLValueReferenceType()) {
2215 relationship_hint = relationship_t::kAssociation;
2216 field_type = field_type.getNonReferenceType();
2217 }
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)
2223 field.set_destination_multiplicity(common::get_array_size(
2224 *field_type->getAsArrayTypeUnsafe()));
2225 else {
2226 auto maybe_array_size =
2227 common::get_array_size(*field_type->getAsArrayTypeUnsafe());
2228 if (maybe_array_size.has_value()) {
2229 field.set_destination_multiplicity(
2230 current_multiplicity.value() *
2231 maybe_array_size.value());
2232 }
2233 }
2234
2235 field_type = field_type->getAsArrayTypeUnsafe()->getElementType();
2236 }
2237 }
2238 else if (field_type->isRValueReferenceType()) {
2239 field_type = field_type.getNonReferenceType();
2240 }
2241
2242 auto [overridden_relationship_hint, overridden] =
2243 override_relationship_hint(type_name, -1, relationship_hint);
2244 if (overridden)
2245 relationship_hint = overridden_relationship_hint;
2246
2247 found_relationships_t relationships;
2248
2249 const auto *template_field_type =
2250 field_type->getAs<clang::TemplateSpecializationType>();
2251 // TODO: Refactor to an unalias_type() method
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>();
2257
2258 bool field_type_is_template_template_parameter{false};
2259 if (template_field_type != nullptr) {
2260 // Skip types which are template template parameters of the parent
2261 // template
2262 for (const auto &class_template_param : c.template_params()) {
2263 if (class_template_param.name() ==
2264 template_field_type->getTemplateName()
2265 .getAsTemplateDecl()
2266 ->getNameAsString() +
2267 "<>") {
2268 field_type_is_template_template_parameter = true;
2269 }
2270 }
2271 }
2272
2273 // Process the type which is template instantiation of some sort
2274 if (template_field_type != nullptr &&
2275 !field_type_is_template_template_parameter) {
2276 // Build the template instantiation for the field type
2277 auto template_specialization_ptr =
2278 std::make_unique<class_>(config().using_namespace());
2280 *template_specialization_ptr,
2281 field_type->getAs<clang::TemplateSpecializationType>()
2282 ->getTemplateName()
2283 .getAsTemplateDecl(),
2284 *template_field_type, {&c});
2285 template_specialization_ptr->is_template(true);
2286
2287 if (!field.skip_relationship() && template_specialization_ptr) {
2288 const auto &template_specialization = *template_specialization_ptr;
2289
2290 // Check if this template instantiation should be added to the
2291 // current diagram. Even if the top level template type for
2292 // this instantiation should not be part of the diagram, e.g.
2293 // it's a std::vector<>, it's nested types might be added
2294 bool add_template_instantiation_to_diagram{false};
2295 if (diagram().should_include(template_specialization)) {
2296
2297 found_relationships_t::value_type r{
2298 template_specialization.id(), relationship_hint,
2299 &field_declaration};
2300
2301 add_template_instantiation_to_diagram = true;
2302
2303 // If the template instantiation for the build type has been
2304 // added as aggregation, skip its nested templates
2305 template_instantiation_added_as_aggregation =
2306 relationship_hint == relationship_t::kAggregation;
2307 relationships.emplace_back(std::move(r));
2308 }
2309
2310 // Try to find relationships to types nested in the template
2311 // instantiation
2312 found_relationships_t nested_relationships;
2313 auto idx{0};
2314 if (!template_instantiation_added_as_aggregation) {
2315 for (const auto &template_argument :
2316 template_specialization.template_params()) {
2317
2318 auto template_argument_str = template_argument.to_string(
2319 config().using_namespace(), false);
2320
2321 LOG_DBG("Looking for nested relationships from {}::{} in "
2322 "template argument {}",
2323 c, field_name, template_argument_str);
2324
2325 auto [overridden_relationship_hint_param,
2326 overridden_param] =
2328 template_specialization.full_name(false), idx,
2329 relationship_hint);
2330
2331 template_instantiation_added_as_aggregation =
2332 template_argument.find_nested_relationships(
2333 &field_declaration, nested_relationships,
2334 overridden_relationship_hint_param,
2335 !overridden_param,
2336 [&d = diagram()](const std::string &full_name) {
2337 if (full_name.empty())
2338 return false;
2339 auto [ns, name] = common::split_ns(full_name);
2340 return d.should_include(ns, name);
2341 });
2342 }
2343
2344 // Add any relationships to the class 'c' to the diagram,
2345 // unless the top level type has been added as aggregation
2346 add_relationships(c, field, nested_relationships,
2347 /* break on first aggregation */ false);
2348
2349 idx++;
2350 }
2351
2352 // Add the template instantiation object to the diagram if it
2353 // matches the include pattern
2354 if (add_template_instantiation_to_diagram)
2355 add_class(std::move(template_specialization_ptr));
2356 }
2357 }
2358
2359 if (!field.skip_relationship()) {
2360 // Find relationship for the type if the type has not been added
2361 // as aggregation
2362 if (!template_instantiation_added_as_aggregation) {
2363 if ((field_type->getAsRecordDecl() != nullptr) &&
2364 field_type->getAsRecordDecl()->getNameAsString().empty()) {
2365 // Relationships to fields whose type is an anonymous nested
2366 // struct have to be handled separately here
2367 anonymous_struct_relationships_[field_type->getAsRecordDecl()
2368 ->getID()] =
2369 std::make_tuple(field.name(), relationship_hint,
2370 field.access(), field.destination_multiplicity());
2371 }
2372 else
2373 find_relationships(&field_declaration, field_type,
2374 relationships, relationship_hint);
2375 }
2376
2377 add_relationships(c, field, relationships);
2378 }
2379
2380 // If this is an anonymous struct - replace the anonymous_XYZ part with
2381 // field name
2382 if ((field_type->getAsRecordDecl() != nullptr) &&
2383 field_type->getAsRecordDecl()->getNameAsString().empty()) {
2384 if (util::contains(field.type(), "(anonymous_")) {
2385 std::regex anonymous_re("anonymous_(\\d*)");
2386 field.set_type(
2387 std::regex_replace(field.type(), anonymous_re, field_name));
2388 }
2389 }
2390
2391 if (diagram().should_include(field)) {
2392 c.add_member(std::move(field));
2393 }
2394}
2395
2397 std::optional<eid_t> &parent_id_opt, namespace_ &parent_ns) const
2398{
2399 const auto *parent = decl->getParent();
2400
2401 if (parent != nullptr) {
2402 if (const auto *parent_record_decl =
2403 clang::dyn_cast<clang::RecordDecl>(parent);
2404 parent_record_decl != nullptr) {
2405 parent_ns = common::get_tag_namespace(*parent_record_decl);
2406
2407 eid_t local_id{parent_record_decl->getID()};
2408
2409 // First check if the parent has been added to the diagram as
2410 // regular class
2411 parent_id_opt = id_mapper().get_global_id(local_id);
2412
2413 // If not, check if the parent template declaration is in the
2414 // model
2415 if (!parent_id_opt) {
2416 if (parent_record_decl->getDescribedTemplate() != nullptr) {
2417 local_id =
2418 parent_record_decl->getDescribedTemplate()->getID();
2419 parent_id_opt = id_mapper().get_global_id(local_id);
2420 }
2421 }
2422 }
2423 }
2424
2425 if (parent_id_opt)
2426 return;
2427
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) {
2433
2434 eid_t ast_id{parent_interface_decl->getID()};
2435
2436 // First check if the parent has been added to the diagram as
2437 // regular class
2438 parent_id_opt = id_mapper().get_global_id(ast_id);
2439 }
2440 }
2441}
2442
2444{
2445 for (auto &[id, c] : forward_declarations_.get<class_>()) {
2446 if (!diagram().find<class_>(id).has_value() &&
2447 diagram().should_include(c->get_namespace())) {
2448 add_class(std::move(c));
2449 }
2450 }
2451 forward_declarations_.get<class_>().clear();
2452
2453 for (auto &[id, e] : forward_declarations_.get<enum_>()) {
2454 if (!diagram().find<enum_>(id).has_value() &&
2455 diagram().should_include(e->get_namespace())) {
2456 add_enum(std::move(e));
2457 }
2458 }
2459 forward_declarations_.get<enum_>().clear();
2460}
2461
2463{
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 =
2469 id_mapper().get_global_id(rel.destination());
2470 if (maybe_id) {
2471 LOG_TRACE("= Resolved instantiation destination "
2472 "from local "
2473 "id {} to global id {}",
2474 rel.destination(), *maybe_id);
2475 rel.set_destination(*maybe_id);
2476 }
2477 }
2478 }
2479 el.get().remove_duplicate_relationships();
2480 }
2481 });
2482}
2483
2485{
2488 if (config().skip_redundant_dependencies()) {
2489 diagram().remove_redundant_dependencies();
2490 }
2491}
2492
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
2498{
2499 const auto full_declaration_text = common::get_source_text_raw(
2500 concept_specialization->getSourceRange(), source_manager());
2501
2502 if (!full_declaration_text.empty()) {
2503 // Handle typename constraint in requires clause
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);
2508
2509 auto template_params = common::parse_unexposed_template_params(
2510 concept_declaration_text, [](const auto &t) { return t; });
2511
2512 if (template_params.size() > argument_index)
2513 type_name = template_params[argument_index].to_string(
2514 config().using_namespace(), false);
2515 }
2516 constrained_template_params.push_back(type_name);
2517 }
2518}
2519
2521 std::string qualified_name)
2522{
2523 processed_template_qualified_names_.emplace(std::move(qualified_name));
2524}
2525
2527 const std::string &qualified_name) const
2528{
2530}
2531
2533 std::unique_ptr<common::model::template_element> element)
2534{
2535 add_class(util::unique_pointer_cast<class_>(std::move(element)));
2536}
2537
2538void translation_unit_visitor::add_class(std::unique_ptr<class_> &&c)
2539{
2540 if ((config().generate_packages() &&
2541 config().package_type() == config::package_type_t::kDirectory)) {
2542 assert(!c->file().empty());
2543
2544 const auto file = config().make_path_relative(c->file());
2545
2548 p.pop_back();
2549
2550 diagram().add(p, std::move(c));
2551 }
2552 else if ((config().generate_packages() &&
2553 config().package_type() == config::package_type_t::kModule)) {
2554
2555 const auto module_path = config().make_module_relative(c->module());
2556
2558
2559 diagram().add(p, std::move(c));
2560 }
2561 else {
2562 diagram().add(c->path(), std::move(c));
2563 }
2564}
2565
2567 std::unique_ptr<objc_interface> &&c)
2568{
2569 if ((config().generate_packages() &&
2570 config().package_type() == config::package_type_t::kDirectory)) {
2571 assert(!c->file().empty());
2572
2573 const auto file = config().make_path_relative(c->file());
2574
2577 p.pop_back();
2578
2579 diagram().add(p, std::move(c));
2580 }
2581 else {
2582 diagram().add(c->path(), std::move(c));
2583 }
2584}
2585
2586void translation_unit_visitor::add_enum(std::unique_ptr<enum_> &&e)
2587{
2588 if ((config().generate_packages() &&
2589 config().package_type() == config::package_type_t::kDirectory)) {
2590 assert(!e->file().empty());
2591
2592 const auto file = config().make_path_relative(e->file());
2593
2596 p.pop_back();
2597
2598 diagram().add(p, std::move(e));
2599 }
2600 else if ((config().generate_packages() &&
2601 config().package_type() == config::package_type_t::kModule)) {
2602
2603 const auto module_path = config().make_module_relative(e->module());
2604
2606
2607 diagram().add(p, std::move(e));
2608 }
2609 else {
2610 diagram().add(e->path(), std::move(e));
2611 }
2612}
2613
2614void translation_unit_visitor::add_concept(std::unique_ptr<concept_> &&c)
2615{
2616 if ((config().generate_packages() &&
2617 config().package_type() == config::package_type_t::kDirectory)) {
2618 assert(!c->file().empty());
2619
2620 const auto file = config().make_path_relative(c->file());
2621
2624 p.pop_back();
2625
2626 diagram().add(p, std::move(c));
2627 }
2628 else if ((config().generate_packages() &&
2629 config().package_type() == config::package_type_t::kModule)) {
2630
2631 const auto module_path = config().make_module_relative(c->module());
2632
2634
2635 diagram().add(p, std::move(c));
2636 }
2637 else {
2638 diagram().add(c->path(), std::move(c));
2639 }
2640}
2641
2643 common::model::template_element &template_instantiation_base,
2644 const std::string &full_name, eid_t templated_decl_id)
2645{
2646 auto &template_instantiation = dynamic_cast<class_diagram::model::class_ &>(
2647 template_instantiation_base);
2648
2649 // First try to find the best match for this template in partially
2650 // specialized templates
2651 std::string destination{};
2652 std::string best_match_full_name{};
2653 auto full_template_name = template_instantiation.full_name(false);
2654 int best_match{};
2655 eid_t best_match_id{};
2656
2657 for (const auto templ : diagram().classes()) {
2658 if (templ.get() == template_instantiation)
2659 continue;
2660
2661 auto c_full_name = templ.get().full_name(false);
2662 auto match =
2663 template_instantiation.calculate_template_specialization_match(
2664 templ.get());
2665
2666 if (match > best_match) {
2667 best_match = match;
2668 best_match_full_name = c_full_name;
2669 best_match_id = templ.get().id();
2670 }
2671 }
2672
2673 auto templated_decl_global_id =
2674 id_mapper().get_global_id(templated_decl_id).value_or(eid_t{});
2675
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);
2681 }
2682 // If we can't find optimal match for parent template specialization,
2683 // just use whatever clang suggests
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);
2689 }
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);
2694 }
2695 else if (diagram().should_include(common::model::namespace_{full_name})) {
2696 LOG_DBG("Skipping instantiation relationship from {} to {}",
2697 template_instantiation, templated_decl_global_id);
2698 }
2699 else {
2700 LOG_DBG("== Cannot determine global id for specialization template {} "
2701 "- delaying until the translation unit is complete ",
2702 templated_decl_global_id);
2703
2704 template_instantiation.add_relationship(
2706 }
2707}
2708
2709} // namespace clanguml::class_diagram::visitor