0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
translation_unit_visitor.h
Go to the documentation of this file.
1/**
2 * @file src/class_diagram/visitor/translation_unit_visitor.h
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#pragma once
19
24#include "common/model/enums.h"
28#include "config/config.h"
29
30#include <clang/AST/RecursiveASTVisitor.h>
31#include <clang/Basic/SourceManager.h>
32
33#include <deque>
34#include <functional>
35#include <map>
36#include <memory>
37#include <string>
38
40
63
67
68template <typename T> class typed_storage_t {
69protected:
70 std::map<eid_t, std::unique_ptr<T>> values;
71
72 std::map<eid_t, std::unique_ptr<T>> &get() { return values; }
73
74 const std::map<eid_t, std::unique_ptr<T>> &get() const { return values; }
75};
76
77template <typename... Ts>
79public:
80 template <typename T> auto &get() { return typed_storage_t<T>::get(); }
81};
82
83/**
84 * @brief Class diagram translation unit visitor
85 *
86 * This class implements the `clang::RecursiveASTVisitor` interface
87 * for selected visitors relevant to generating class diagrams.
88 */
90 : public clang::RecursiveASTVisitor<translation_unit_visitor>,
92public:
95
97
98 /**
99 * @brief Constructor.
100 *
101 * @param sm Current source manager reference
102 * @param diagram Diagram model
103 * @param config Diagram configuration
104 */
105 explicit translation_unit_visitor(clang::SourceManager &sm,
108
109 /**
110 * \defgroup Implementation of ResursiveASTVisitor methods
111 * @{
112 */
113 bool shouldVisitTemplateInstantiations() const { return false; }
114
115 bool shouldVisitImplicitCode() const { return false; }
116
117 virtual bool VisitNamespaceDecl(clang::NamespaceDecl *ns);
118
119 virtual bool VisitRecordDecl(clang::RecordDecl *D);
120
121 virtual bool VisitCXXRecordDecl(clang::CXXRecordDecl *d);
122
123 virtual bool VisitTypedefDecl(clang::TypedefDecl *decl);
124
125 virtual bool VisitEnumDecl(clang::EnumDecl *e);
126
127 virtual bool VisitClassTemplateDecl(
128 clang::ClassTemplateDecl *class_template_declaration);
129
131 clang::ClassTemplateSpecializationDecl *cls);
132
133 virtual bool VisitTypeAliasTemplateDecl(clang::TypeAliasTemplateDecl *cls);
134
135 virtual bool TraverseConceptDecl(clang::ConceptDecl *cpt);
136
137 virtual bool VisitObjCCategoryDecl(clang::ObjCCategoryDecl *decl);
138
139 virtual bool VisitObjCProtocolDecl(clang::ObjCProtocolDecl *decl);
140
141 virtual bool VisitObjCInterfaceDecl(clang::ObjCInterfaceDecl *decl);
142 /** @} */
143
144 /**
145 * @brief Finalize diagram model
146 *
147 * This method is called after the entire AST has been visited by this
148 * visitor. It is used to perform necessary post processing on the
149 * diagram (e.g. resolve translation unit local element ID's into global
150 * ID's based on elements full names).
151 */
152 void finalize();
153
154 /**
155 * @brief Add class (or template class) to the diagram.
156 *
157 * @param c Class model
158 */
159 void add_class(std::unique_ptr<class_> &&c);
160
161 /**
162 * @brief Add enum to the diagram.
163 *
164 * @param e Enum model
165 */
166 void add_enum(std::unique_ptr<enum_> &&e);
167
168 /**
169 * @brief Add concept to the diagram.
170 *
171 * @param c Concept model
172 */
173 void add_concept(std::unique_ptr<concept_> &&c);
174
175 void add_objc_interface(std::unique_ptr<objc_interface> &&c);
176
178 std::unique_ptr<common::model::template_element> element) override;
179
180 std::unique_ptr<class_> create_element(const clang::NamedDecl *decl) const;
181
183 common::model::template_element &template_instantiation_base,
184 const std::string &full_name, eid_t templated_decl_id);
185
186private:
187 /**
188 * @brief Create class element model from class declaration
189 *
190 * @param cls Class declaration
191 * @return Class diagram element model
192 */
193 std::unique_ptr<clanguml::class_diagram::model::class_> create_declaration(
194 clang::CXXRecordDecl *cls);
195
196 /**
197 * @brief Add the element model or update update if the model already
198 * exists
199 *
200 * @tparam T Type of clang declaration
201 * @tparam ElementT Type of model element
202 * @param cls Pointer to clang declaration
203 * @param c_ptr Pointer to the element model
204 * @return
205 */
206 template <typename T, typename ElementT>
207 bool add_or_update(const T *cls, std::unique_ptr<ElementT> &&c_ptr);
208
209 /**
210 * @brief Create enum element model from enum (e.g. struct) declaration
211 *
212 * @param rec Enum declaration
213 * @return Enum diagram element model
214 */
215 std::unique_ptr<clanguml::class_diagram::model::enum_> create_declaration(
216 const clang::EnumDecl *enm, const clang::TypedefDecl *typedef_decl);
217
218 /**
219 * @brief Create class element model from record (e.g. struct)
220 * declaration
221 *
222 * @param rec Record declaration
223 * @return Class diagram element model
224 */
225 std::unique_ptr<clanguml::class_diagram::model::class_> create_declaration(
226 clang::RecordDecl *rec);
227
228 /**
229 * @brief Create class element model from Objective-C protocol
230 *
231 * @param decl Objective-C protocol declaration
232 * @return Class diagram element model
233 */
234 std::unique_ptr<clanguml::class_diagram::model::objc_interface>
235 create_objc_protocol_declaration(clang::ObjCProtocolDecl *decl);
236
237 /**
238 * @brief Create class element model from Objective-C interface
239 *
240 * @param decl Objective-C protocol declaration
241 * @return Class diagram element model
242 */
243 std::unique_ptr<clanguml::class_diagram::model::objc_interface>
244 create_objc_interface_declaration(clang::ObjCInterfaceDecl *decl);
245
246 /**
247 * @brief Create class element model from Objective-C category
248 *
249 * @param decl Objective-C protocol declaration
250 * @return Class diagram element model
251 */
252 std::unique_ptr<clanguml::class_diagram::model::objc_interface>
253 create_objc_category_declaration(clang::ObjCCategoryDecl *decl);
254
255 /**
256 * @brief Create concept element model from concept declaration
257 * @param cpt Concept declaration
258 * @return Concept diagram element model
259 */
260 std::unique_ptr<clanguml::class_diagram::model::concept_>
261 create_concept_declaration(clang::ConceptDecl *cpt);
262
263 /**
264 * @brief Process class declaration
265 *
266 * @param cls Class declaration
267 * @param c Class diagram element returned from `create_declaration`
268 */
269 void process_declaration(const clang::CXXRecordDecl &cls,
271
272 /**
273 * @brief Process enum declaration
274 *
275 * @param enm Enum declaration
276 * @param e Enum diagram element returned from `create_declaration`
277 */
279 const clang::EnumDecl &enm, clanguml::class_diagram::model::enum_ &e);
280
281 /**
282 * @brief Process Objective-C category declaration
283 *
284 * @param cls Objective-C category declaration
285 * @param c Class diagram element returned from
286 * `create_objc_category_declaration`
287 */
289 const clang::ObjCCategoryDecl &cls, objc_interface &c);
290
291 /**
292 * @brief Process Objective-C protocol declaration
293 *
294 * @param cls Objective-C protocol declaration
295 * @param c Class diagram element returned from
296 * `create_objc_protocol_declaration`
297 */
299 const clang::ObjCProtocolDecl &cls, objc_interface &c);
300
301 /**
302 * @brief Process Objective-C interface declaration
303 *
304 * @param cls Objective-C interface declaration
305 * @param c Class diagram element returned from
306 * `create_objc_interface_declaration`
307 */
309 const clang::ObjCInterfaceDecl &cls, objc_interface &c);
310
311 /**
312 * @brief Process class declaration bases (parents), if any
313 *
314 * @param cls Class declaration
315 * @param c Class diagram element model
316 */
317 void process_class_bases(const clang::CXXRecordDecl *cls,
319
320 /**
321 * @brief Process class children elements (members and methods)
322 *
323 * @param cls Class declaration
324 * @param c Class diagram element model
325 */
326 void process_children(const clang::CXXRecordDecl *cls,
328
329 /**
330 * @brief Process record members
331 * @param cls Class declaration
332 * @param c Class diagram element model
333 */
334 void process_declaration(const clang::RecordDecl &cls, class_ &c);
335
336 /**
337 * @brief Process class template specialization/instantiation
338 *
339 * @param cls Class template specialization declaration
340 * @return Class diagram element model
341 */
342 std::unique_ptr<clanguml::class_diagram::model::class_>
344 clang::ClassTemplateSpecializationDecl *cls);
345
346 /**
347 * @brief Process template specialization children (members and methods)
348 * @param cls Class template specialization declaration
349 * @param c Class diagram element model
350 */
352 const clang::ClassTemplateSpecializationDecl *cls, class_ &c);
353
354 /**
355 * @brief Process class method
356 *
357 * @param mf Method declaration
358 * @param c Class diagram element model
359 */
360 void process_method(const clang::CXXMethodDecl &mf,
362
363 /**
364 * @brief Process Objective-C method
365 *
366 * @param mf Method declaration
367 * @param c Class diagram element model
368 */
370 const clang::ObjCMethodDecl &mf, objc_interface &c);
371
372 /**
373 * @brief Process class method properties
374 * @param mf Method declaration
375 * @param c Class diagram element model
376 * @param method_name Method name
377 * @param method Method model
378 */
379 void process_method_properties(const clang::CXXMethodDecl &mf,
380 const class_ &c, const std::string &method_name,
381 class_method &method) const;
382
383 /**
384 * @brief Process class template method
385 *
386 * @param mf Method declaration
387 * @param c Class diagram element model
388 */
389 void process_template_method(const clang::FunctionTemplateDecl &mf,
391
392 /**
393 * @brief Process class static data member
394 *
395 * @param field_declaration Static data member declaration
396 * @param c Class diagram element model
397 */
398 void process_static_field(const clang::VarDecl &field_declaration,
400
401 /**
402 * @brief Process class data member
403 *
404 * @param field_declaration Data member declaration
405 * @param c Class diagram element model
406 */
407 void process_field(const clang::FieldDecl &field_declaration,
409
410 /**
411 * @brief Process Objective-C data member
412 *
413 * @param field_declaration Data member declaration
414 * @param c Class diagram element model
415 */
416 void process_objc_ivar(const clang::ObjCIvarDecl &ivar, objc_interface &c);
417
418 /**
419 * @brief Process Objective-C class base
420 *
421 * @param cls Objective-C interface declaration
422 * @param c Class diagram element model
423 */
425 const clang::ObjCInterfaceDecl &cls, objc_interface &c);
426
427 /**
428 * @brief Process function/method parameter
429 *
430 * @param param Parameter declaration
431 * @param method Class method model
432 * @param c Class diagram element model
433 * @param template_parameter_names Ignored
434 */
435 void process_function_parameter(const clang::ParmVarDecl &param,
436 class_method &method, class_ &c,
437 const std::set<std::string> &template_parameter_names = {});
438
439 /**
440 * @brief Process Objective-C class method parameter
441 *
442 * @param param Parameter declaration
443 * @param method Class method model
444 * @param c Class diagram element model
445 * @param template_parameter_names Ignored
446 */
447 void process_objc_method_parameter(const clang::ParmVarDecl &param,
448 objc_method &method, objc_interface &c);
449
450 /**
451 * @brief Process class friend
452 *
453 * @param f Friend declaration
454 * @param c Class diagram element model
455 */
456 void process_friend(const clang::FriendDecl &f, class_ &c);
457
458 /**
459 * @brief Find relationships in a specific type
460 *
461 * @param decl Source declaration from which this relationship
462 * originates
463 * @param type Type to search for relationships
464 * @param relationship_hint Default relationship type to infer from this
465 * type
466 * @return True, if any relationships were found
467 */
468 bool find_relationships(const clang::Decl *decl,
469 const clang::QualType &type, found_relationships_t & /*relationships*/,
471
472 /**
473 * @brief Add relationships from relationship list to a diagram element
474 * model
475 *
476 * This method takes a list of relationships whose originating element
477 * is class `c` and adds them to it, ignoring any duplicates and
478 * skipping relationships that should be excluded from the diagram.
479 *
480 * @param c Diagram element model
481 * @param field Class member model
482 * @param relationships List of found relationships
483 * @param break_on_first_aggregation Stop adding relatinoships, after
484 * first aggregation is found
485 */
487 const found_relationships_t &relationships,
488 bool break_on_first_aggregation = false);
489
490 /**
491 * @brief Try to override relationship hint using configuration file
492 *
493 * @param type_name
494 * @param hint
495 * @return Maybe overridden relationship type hint
496 */
497 std::pair<relationship_t, bool> override_relationship_hint(
498 const std::string &type_name, int index, relationship_t hint);
499
500 /**
501 * @brief Process record parent element (e.g. for nested classes)
502 *
503 * This method handles nested classes or structs.
504 *
505 * @param cls Record declaration
506 * @param c Class diagram element model
507 * @param ns Package in the diagram to which the class `c` should belong
508 */
510 clang::RecordDecl *cls, class_ &c, const namespace_ &ns);
511
512 /**
513 * @brief Find relationships in function parameter
514 *
515 * @param c Class diagram element model
516 * @param atsp `auto` type
517 */
519 model::class_ &c, const clang::AutoType *atsp);
520
521 /**
522 * @brief Find relationships in concept constraint expression
523 *
524 * @param c Diagram element model (concept)
525 * @param expr Concept constraint expression
526 */
528 clanguml::common::model::element &c, const clang::Expr *expr);
529
530 /**
531 * @brief Register incomplete forward declaration to be updated later
532 */
534
535 /**
536 * @brief Replace any AST local ids in diagram elements with global ones
537 *
538 * Not all elements global ids can be set in relationships during
539 * traversal of the AST. In such cases, a local id (obtained from
540 * `getID()`) and at after the traversal is complete, the id is replaced
541 * with the global diagram id.
542 */
544
545 /**
546 * @brief Process concept constraint requirements
547 *
548 * @param cpt Concept declaration
549 * @param expr Requires expression
550 * @param concept_model Concept diagram element model
551 */
552 void process_constraint_requirements(const clang::ConceptDecl *cpt,
553 const clang::Expr *expr, model::concept_ &concept_model) const;
554
555 /**
556 * @brief Find concept specializations relationships
557 *
558 * @param c Concept element model
559 * @param concept_specialization Concept specialization expression
560 */
562 const clang::ConceptSpecializationExpr *concept_specialization);
563
564 /**
565 * @brief Extract template contraint parameter name from raw source code
566 *
567 * @param concept_specialization Concept specialization expression
568 * @param cpt Concept declaration
569 * @param constrained_template_params Found constraint template param
570 * names
571 * @param argument_index Argument index
572 * @param type_name Type parameter name - used if extraction fails
573 */
575 const clang::ConceptSpecializationExpr *concept_specialization,
576 const clang::ConceptDecl *cpt,
577 std::vector<std::string> &constrained_template_params,
578 size_t argument_index, std::string &type_name) const;
579
580 /**
581 * @brief Register already processed template class name
582 *
583 * @param qualified_name Fully qualified template class name
584 */
585 void add_processed_template_class(std::string qualified_name);
586
587 /**
588 * @brief Check if template class has already been processed
589 *
590 * @param qualified_name Fully qualified template class name
591 * @return True, if template class has already been processed
592 */
593 bool has_processed_template_class(const std::string &qualified_name) const;
594
595 /**
596 * @brief Get template builder reference
597 *
598 * @return Reference to 'template_builder' instance
599 */
601
602 template <typename T>
603 void process_record_parent_by_type(eid_t parent_id, class_ &c,
604 namespace_ parent_ns, const clang::RecordDecl *decl);
605
606 void find_record_parent_id(const clang::TagDecl *decl,
607 std::optional<eid_t> &parent_id_opt, namespace_ &parent_ns) const;
608
610
612
613 std::map<int64_t /* local anonymous struct id */,
614 std::tuple<std::string /* field name */, common::model::relationship_t,
616 std::optional<size_t> /* destination_multiplicity */>>
618
619 std::map<const clang::EnumDecl *, const clang::TypedefDecl *>
621
622 /**
623 * When visiting CXX records we need to know if they have already been
624 * process in VisitClassTemplateDecl or
625 * VisitClassTemplateSpecializationDecl. If yes, then we need to skip it
626 *
627 * @todo There must be a better way to do this...
628 */
630};
631
632template <typename T>
634 class_ &c, namespace_ parent_ns, const clang::RecordDecl *decl)
635{
636 // Here we have 2 options, either:
637 // - the parent is a regular C++ class/struct
638 // - the parent is a class template declaration/specialization
639 auto parent_class = diagram().find<T>(parent_id);
640
641 c.set_namespace(parent_ns);
642 const auto cls_name = decl->getNameAsString();
643 if (cls_name.empty()) {
644 // Nested structs can be anonymous
645 if (anonymous_struct_relationships_.count(decl->getID()) > 0) {
646 const auto &[label, hint, access, destination_multiplicity] =
647 anonymous_struct_relationships_[decl->getID()];
648
649 c.set_name(parent_class.value().name() + "##" +
650 fmt::format("({})", label));
651
652 std::string destination_multiplicity_str{};
653 if (destination_multiplicity.has_value()) {
654 destination_multiplicity_str =
655 std::to_string(*destination_multiplicity);
656 }
657
658 parent_class.value().add_relationship(
659 {hint, common::to_id(c.full_name(false)), access, label, "",
660 destination_multiplicity_str});
661 }
662 else
663 c.set_name(parent_class.value().name() + "##" +
664 fmt::format("(anonymous_{})", std::to_string(decl->getID())));
665 }
666 else {
667 c.set_name(
668 parent_class.value().name() + "##" + decl->getNameAsString());
669 }
670
671 c.set_id(common::to_id(c.full_name(false)));
672
673 if (!(decl->getNameAsString().empty())) {
674 // Don't add anonymous structs as contained in the class
675 // as they are already added as aggregations
676 c.add_relationship({relationship_t::kContainment, parent_id});
677 }
678
679 c.nested(true);
680}
681
682template <typename T, typename ElementT>
684 const T *cls, std::unique_ptr<ElementT> &&c_ptr)
685{
686 static_assert(
687 std::is_same_v<ElementT, class_> || std::is_same_v<ElementT, enum_>);
688
689 const auto cls_id = c_ptr->id();
690
691 id_mapper().add(cls->getID(), cls_id);
692
693 auto maybe_existing_model = diagram().find<ElementT>(cls_id);
694
695 ElementT &class_model =
696 maybe_existing_model.has_value() ? *maybe_existing_model.get() : *c_ptr;
697
698 auto id = class_model.id();
699
700 if (cls->isCompleteDefinition() && !class_model.complete()) {
701 process_declaration(*cls, class_model);
702
703 // Update the source location for the element, otherwise
704 // it can point to the location of first encountered forward
705 // declaration
706 set_source_location(*cls, class_model);
707 }
708
709 if (cls->isCompleteDefinition()) {
710 if (maybe_existing_model &&
711 config().package_type() == config::package_type_t::kDirectory) {
712 // Move the class model to current filesystem path
713 // Eventually, this should be refactored so that it's
714 // not needed
715 const auto file = config().make_path_relative(class_model.file());
718 p.pop_back();
719 diagram().move<ElementT>(id, p);
720 }
721 }
722 else {
723 forward_declarations_.get<ElementT>().emplace(id, std::move(c_ptr));
724 return true;
725 }
726
727 forward_declarations_.get<ElementT>().erase(id);
728
729 if constexpr (std::is_same_v<T, clang::ClassTemplateSpecializationDecl>) {
730 if (!class_model.template_specialization_found()) {
731 // Only do this if we haven't found a better specialization
732 // during construction of the template specialization
733 const eid_t ast_id{cls->getSpecializedTemplate()->getID()};
734 const auto maybe_id = id_mapper().get_global_id(ast_id);
735 if (maybe_id.has_value())
736 class_model.add_relationship(
737 {relationship_t::kInstantiation, maybe_id.value()});
738 }
739 }
740
741 if (maybe_existing_model)
742 return true;
743
744 if (diagram().should_include(class_model)) {
745 LOG_DBG("Adding {} {} with id {}", class_model.type_name(), class_model,
746 class_model.id());
747 if constexpr (std::is_same_v<ElementT, class_>)
748 add_class(std::move(c_ptr));
749 else
750 add_enum(std::move(c_ptr));
751 }
752 else {
753 LOG_DBG("Skipping {} {} with id {}", class_model.type_name(),
754 class_model, class_model.id());
755 }
756
757 return true;
758}
759} // namespace clanguml::class_diagram::visitor