0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
Public Types | Public Member Functions | Private Member Functions | Private Attributes | List of all members
clanguml::class_diagram::visitor::translation_unit_visitor Class Reference

Class diagram translation unit visitor. More...

Detailed Description

Class diagram translation unit visitor.

This class implements the clang::RecursiveASTVisitor interface for selected visitors relevant to generating class diagrams.

Definition at line 89 of file translation_unit_visitor.h.

#include <translation_unit_visitor.h>

Public Types

using template_builder_t = template_builder< translation_unit_visitor >
 
using config_t = ConfigT
 
using diagram_t = DiagramT
 
- Public Types inherited from clanguml::common::visitor::translation_unit_visitor< ConfigT, DiagramT >
using config_t = ConfigT
 
using diagram_t = DiagramT
 

Public Member Functions

 translation_unit_visitor (clang::SourceManager &sm, clanguml::class_diagram::model::diagram &diagram, const clanguml::config::class_diagram &config)
 Constructor.
 
bool shouldVisitTemplateInstantiations () const
 
bool shouldVisitImplicitCode () const
 
virtual bool VisitNamespaceDecl (clang::NamespaceDecl *ns)
 
virtual bool VisitRecordDecl (clang::RecordDecl *D)
 
virtual bool VisitCXXRecordDecl (clang::CXXRecordDecl *d)
 
virtual bool VisitTypedefDecl (clang::TypedefDecl *decl)
 
virtual bool VisitEnumDecl (clang::EnumDecl *e)
 
virtual bool VisitClassTemplateDecl (clang::ClassTemplateDecl *class_template_declaration)
 
virtual bool VisitClassTemplateSpecializationDecl (clang::ClassTemplateSpecializationDecl *cls)
 
virtual bool VisitTypeAliasTemplateDecl (clang::TypeAliasTemplateDecl *cls)
 
virtual bool TraverseConceptDecl (clang::ConceptDecl *cpt)
 
virtual bool VisitObjCCategoryDecl (clang::ObjCCategoryDecl *decl)
 
virtual bool VisitObjCProtocolDecl (clang::ObjCProtocolDecl *decl)
 
virtual bool VisitObjCInterfaceDecl (clang::ObjCInterfaceDecl *decl)
 
void finalize ()
 Finalize diagram model.
 
void add_class (std::unique_ptr< class_ > &&c)
 Add class (or template class) to the diagram.
 
void add_enum (std::unique_ptr< enum_ > &&e)
 Add enum to the diagram.
 
void add_concept (std::unique_ptr< concept_ > &&c)
 Add concept to the diagram.
 
void add_objc_interface (std::unique_ptr< objc_interface > &&c)
 
void add_diagram_element (std::unique_ptr< common::model::template_element > element) override
 
std::unique_ptr< class_create_element (const clang::NamedDecl *decl) const
 
void find_instantiation_relationships (common::model::template_element &template_instantiation_base, const std::string &full_name, eid_t templated_decl_id)
 
- Public Member Functions inherited from clanguml::common::visitor::translation_unit_visitor< ConfigT, DiagramT >
 translation_unit_visitor (clang::SourceManager &sm, DiagramT &diagram, const ConfigT &config)
 Constructor.
 
virtual ~translation_unit_visitor ()=default
 
void set_tu_path (const std::string &translation_unit_path)
 
const std::filesystem::path & tu_path () const
 Return relative path to current translation unit.
 
common::visitor::ast_id_mapperid_mapper () const
 Get reference to Clang AST to clang-uml id mapper.
 
clang::SourceManager & source_manager () const
 Get clang::SourceManager.
 
void set_source_location (const clang::Decl &decl, clanguml::common::model::source_location &element)
 Set source location in diagram element.
 
void set_source_location (const clang::Expr &expr, clanguml::common::model::source_location &element)
 Set source location in diagram element.
 
void set_source_location (const clang::Stmt &stmt, clanguml::common::model::source_location &element)
 
void set_qualified_name (const clang::NamedDecl &decl, clanguml::common::model::element &element)
 
void set_source_location (const clang::SourceLocation &location, clanguml::common::model::source_location &element)
 Set source location in diagram element.
 
void set_owning_module (const clang::Decl &decl, clanguml::common::model::element &element)
 
virtual void add_diagram_element (std::unique_ptr< common::model::template_element > element)
 
void process_comment (const clang::NamedDecl &decl, clanguml::common::model::decorated_element &e)
 Process comment directives in comment attached to a declaration.
 
std::string process_comment (const clang::RawComment *comment, clang::DiagnosticsEngine &de, clanguml::common::model::decorated_element &e)
 Process comment directives in raw comment.
 
bool skip_system_header_decl (const clang::NamedDecl *decl) const
 
bool should_include (const clang::NamedDecl *decl) const
 Check if the diagram should include a declaration.
 
DiagramT & diagram ()
 Get diagram model reference.
 
const DiagramT & diagram () const
 Get diagram model reference.
 
const ConfigT & config () const
 Get diagram config instance.
 

Private Member Functions

std::unique_ptr< clanguml::class_diagram::model::class_create_declaration (clang::CXXRecordDecl *cls)
 Create class element model from class declaration.
 
template<typename T , typename ElementT >
bool add_or_update (const T *cls, std::unique_ptr< ElementT > &&c_ptr)
 Add the element model or update update if the model already exists.
 
std::unique_ptr< clanguml::class_diagram::model::enum_create_declaration (const clang::EnumDecl *enm, const clang::TypedefDecl *typedef_decl)
 Create enum element model from enum (e.g. struct) declaration.
 
std::unique_ptr< clanguml::class_diagram::model::class_create_declaration (clang::RecordDecl *rec)
 Create class element model from record (e.g. struct) declaration.
 
std::unique_ptr< clanguml::class_diagram::model::objc_interfacecreate_objc_protocol_declaration (clang::ObjCProtocolDecl *decl)
 Create class element model from Objective-C protocol.
 
std::unique_ptr< clanguml::class_diagram::model::objc_interfacecreate_objc_interface_declaration (clang::ObjCInterfaceDecl *decl)
 Create class element model from Objective-C interface.
 
std::unique_ptr< clanguml::class_diagram::model::objc_interfacecreate_objc_category_declaration (clang::ObjCCategoryDecl *decl)
 Create class element model from Objective-C category.
 
std::unique_ptr< clanguml::class_diagram::model::concept_create_concept_declaration (clang::ConceptDecl *cpt)
 Create concept element model from concept declaration.
 
void process_declaration (const clang::CXXRecordDecl &cls, clanguml::class_diagram::model::class_ &c)
 Process class declaration.
 
void process_declaration (const clang::EnumDecl &enm, clanguml::class_diagram::model::enum_ &e)
 Process enum declaration.
 
void process_objc_category_declaration (const clang::ObjCCategoryDecl &cls, objc_interface &c)
 Process Objective-C category declaration.
 
void process_objc_protocol_declaration (const clang::ObjCProtocolDecl &cls, objc_interface &c)
 Process Objective-C protocol declaration.
 
void process_objc_interface_declaration (const clang::ObjCInterfaceDecl &cls, objc_interface &c)
 Process Objective-C interface declaration.
 
void process_class_bases (const clang::CXXRecordDecl *cls, clanguml::class_diagram::model::class_ &c)
 Process class declaration bases (parents), if any.
 
void process_children (const clang::CXXRecordDecl *cls, clanguml::class_diagram::model::class_ &c)
 Process class children elements (members and methods)
 
void process_declaration (const clang::RecordDecl &cls, class_ &c)
 Process record members.
 
std::unique_ptr< clanguml::class_diagram::model::class_process_template_specialization (clang::ClassTemplateSpecializationDecl *cls)
 Process class template specialization/instantiation.
 
void process_template_specialization_children (const clang::ClassTemplateSpecializationDecl *cls, class_ &c)
 Process template specialization children (members and methods)
 
void process_method (const clang::CXXMethodDecl &mf, clanguml::class_diagram::model::class_ &c)
 Process class method.
 
void process_objc_method (const clang::ObjCMethodDecl &mf, objc_interface &c)
 Process Objective-C method.
 
void process_method_properties (const clang::CXXMethodDecl &mf, const class_ &c, const std::string &method_name, class_method &method) const
 Process class method properties.
 
void process_template_method (const clang::FunctionTemplateDecl &mf, clanguml::class_diagram::model::class_ &c)
 Process class template method.
 
void process_static_field (const clang::VarDecl &field_declaration, clanguml::class_diagram::model::class_ &c)
 Process class static data member.
 
void process_field (const clang::FieldDecl &field_declaration, clanguml::class_diagram::model::class_ &c)
 Process class data member.
 
void process_objc_ivar (const clang::ObjCIvarDecl &ivar, objc_interface &c)
 Process Objective-C data member.
 
void process_objc_interface_base (const clang::ObjCInterfaceDecl &cls, objc_interface &c)
 Process Objective-C class base.
 
void process_function_parameter (const clang::ParmVarDecl &param, class_method &method, class_ &c, const std::set< std::string > &template_parameter_names={})
 Process function/method parameter.
 
void process_objc_method_parameter (const clang::ParmVarDecl &param, objc_method &method, objc_interface &c)
 Process Objective-C class method parameter.
 
void process_friend (const clang::FriendDecl &f, class_ &c)
 Process class friend.
 
bool find_relationships (const clang::Decl *decl, const clang::QualType &type, found_relationships_t &, clanguml::common::model::relationship_t relationship_hint)
 Find relationships in a specific type.
 
void add_relationships (diagram_element &c, const class_member_base &field, const found_relationships_t &relationships, bool break_on_first_aggregation=false)
 Add relationships from relationship list to a diagram element model.
 
std::pair< relationship_t, bool > override_relationship_hint (const std::string &type_name, int index, relationship_t hint)
 Try to override relationship hint using configuration file.
 
void process_record_parent (clang::RecordDecl *cls, class_ &c, const namespace_ &ns)
 Process record parent element (e.g. for nested classes)
 
void process_function_parameter_find_relationships_in_autotype (model::class_ &c, const clang::AutoType *atsp)
 Find relationships in function parameter.
 
void find_relationships_in_constraint_expression (clanguml::common::model::element &c, const clang::Expr *expr)
 Find relationships in concept constraint expression.
 
void add_incomplete_forward_declarations ()
 Register incomplete forward declaration to be updated later.
 
void resolve_local_to_global_ids ()
 Replace any AST local ids in diagram elements with global ones.
 
void process_constraint_requirements (const clang::ConceptDecl *cpt, const clang::Expr *expr, model::concept_ &concept_model) const
 Process concept constraint requirements.
 
void process_concept_specialization_relationships (common::model::element &c, const clang::ConceptSpecializationExpr *concept_specialization)
 Find concept specializations relationships.
 
void extract_constrained_template_param_name (const clang::ConceptSpecializationExpr *concept_specialization, const clang::ConceptDecl *cpt, std::vector< std::string > &constrained_template_params, size_t argument_index, std::string &type_name) const
 Extract template contraint parameter name from raw source code.
 
void add_processed_template_class (std::string qualified_name)
 Register already processed template class name.
 
bool has_processed_template_class (const std::string &qualified_name) const
 Check if template class has already been processed.
 
template_builder_ttbuilder ()
 Get template builder reference.
 
template<typename T >
void process_record_parent_by_type (eid_t parent_id, class_ &c, namespace_ parent_ns, const clang::RecordDecl *decl)
 
void find_record_parent_id (const clang::TagDecl *decl, std::optional< eid_t > &parent_id_opt, namespace_ &parent_ns) const
 

Private Attributes

template_builder_t template_builder_
 
forward_declarations_t< class_, enum_forward_declarations_
 
std::map< int64_t, std::tuple< std::string, common::model::relationship_t, common::model::access_t, std::optional< size_t > > > anonymous_struct_relationships_
 
std::map< const clang::EnumDecl *, const clang::TypedefDecl * > typedef_enum_decls_
 
std::set< std::string > processed_template_qualified_names_
 

Additional Inherited Members

- Protected Member Functions inherited from clanguml::common::visitor::translation_unit_visitor< ConfigT, DiagramT >
std::set< const clang::RawComment * > & processed_comments ()
 
std::string get_file_path (const std::string &file_location) const
 

Member Typedef Documentation

◆ config_t

using clanguml::common::visitor::translation_unit_visitor< ConfigT, DiagramT >::config_t = ConfigT

Definition at line 56 of file translation_unit_visitor.h.

◆ diagram_t

using clanguml::common::visitor::translation_unit_visitor< ConfigT, DiagramT >::diagram_t = DiagramT

Definition at line 57 of file translation_unit_visitor.h.

◆ template_builder_t

Definition at line 96 of file translation_unit_visitor.h.

Constructor & Destructor Documentation

◆ translation_unit_visitor()

clanguml::class_diagram::visitor::translation_unit_visitor::translation_unit_visitor ( clang::SourceManager &  sm,
clanguml::class_diagram::model::diagram diagram,
const clanguml::config::class_diagram config 
)
explicit

Constructor.

Parameters
smCurrent source manager reference
diagramDiagram model
configDiagram configuration

Definition at line 29 of file translation_unit_visitor.cc.

Member Function Documentation

◆ add_class()

void clanguml::class_diagram::visitor::translation_unit_visitor::add_class ( std::unique_ptr< class_ > &&  c)

Add class (or template class) to the diagram.

Parameters
cClass model

Definition at line 2538 of file translation_unit_visitor.cc.

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
2546 common::model::path p{
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
2557 common::model::path p{module_path, common::model::path_type::kModule};
2558
2559 diagram().add(p, std::move(c));
2560 }
2561 else {
2562 diagram().add(c->path(), std::move(c));
2563 }
2564}

◆ add_concept()

void clanguml::class_diagram::visitor::translation_unit_visitor::add_concept ( std::unique_ptr< concept_ > &&  c)

Add concept to the diagram.

Parameters
cConcept model

Definition at line 2614 of file translation_unit_visitor.cc.

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
2622 common::model::path p{
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
2633 common::model::path p{module_path, common::model::path_type::kModule};
2634
2635 diagram().add(p, std::move(c));
2636 }
2637 else {
2638 diagram().add(c->path(), std::move(c));
2639 }
2640}

◆ add_diagram_element()

void clanguml::class_diagram::visitor::translation_unit_visitor::add_diagram_element ( std::unique_ptr< common::model::template_element element)
overridevirtual

Reimplemented from clanguml::common::visitor::translation_unit_visitor< ConfigT, DiagramT >.

Definition at line 2532 of file translation_unit_visitor.cc.

2534{
2535 add_class(util::unique_pointer_cast<class_>(std::move(element)));
2536}

◆ add_enum()

void clanguml::class_diagram::visitor::translation_unit_visitor::add_enum ( std::unique_ptr< enum_ > &&  e)

Add enum to the diagram.

Parameters
eEnum model

Definition at line 2586 of file translation_unit_visitor.cc.

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
2594 common::model::path p{
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
2605 common::model::path p{module_path, common::model::path_type::kModule};
2606
2607 diagram().add(p, std::move(e));
2608 }
2609 else {
2610 diagram().add(e->path(), std::move(e));
2611 }
2612}

◆ add_incomplete_forward_declarations()

void clanguml::class_diagram::visitor::translation_unit_visitor::add_incomplete_forward_declarations ( )
private

Register incomplete forward declaration to be updated later.

Definition at line 2443 of file translation_unit_visitor.cc.

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}

◆ add_objc_interface()

void clanguml::class_diagram::visitor::translation_unit_visitor::add_objc_interface ( std::unique_ptr< objc_interface > &&  c)

Definition at line 2566 of file translation_unit_visitor.cc.

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
2575 common::model::path p{
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}

◆ add_or_update()

template<typename T , typename ElementT >
bool clanguml::class_diagram::visitor::translation_unit_visitor::add_or_update ( const T *  cls,
std::unique_ptr< ElementT > &&  c_ptr 
)
private

Add the element model or update update if the model already exists.

Template Parameters
TType of clang declaration
ElementTType of model element
Parameters
clsPointer to clang declaration
c_ptrPointer to the element model
Returns

Definition at line 683 of file translation_unit_visitor.h.

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());
716 common::model::path p{
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}

◆ add_processed_template_class()

void clanguml::class_diagram::visitor::translation_unit_visitor::add_processed_template_class ( std::string  qualified_name)
private

Register already processed template class name.

Parameters
qualified_nameFully qualified template class name

Definition at line 2520 of file translation_unit_visitor.cc.

2522{
2523 processed_template_qualified_names_.emplace(std::move(qualified_name));
2524}

◆ add_relationships()

void clanguml::class_diagram::visitor::translation_unit_visitor::add_relationships ( diagram_element c,
const class_member_base field,
const found_relationships_t &  relationships,
bool  break_on_first_aggregation = false 
)
private

Add relationships from relationship list to a diagram element model.

This method takes a list of relationships whose originating element is class c and adds them to it, ignoring any duplicates and skipping relationships that should be excluded from the diagram.

Parameters
cDiagram element model
fieldClass member model
relationshipsList of found relationships
break_on_first_aggregationStop adding relatinoships, after first aggregation is found

Definition at line 2021 of file translation_unit_visitor.cc.

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}

◆ create_concept_declaration()

std::unique_ptr< clanguml::class_diagram::model::concept_ > clanguml::class_diagram::visitor::translation_unit_visitor::create_concept_declaration ( clang::ConceptDecl *  cpt)
private

Create concept element model from concept declaration.

Parameters
cptConcept declaration
Returns
Concept diagram element model

Definition at line 823 of file translation_unit_visitor.cc.

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}

◆ create_declaration() [1/3]

std::unique_ptr< class_ > clanguml::class_diagram::visitor::translation_unit_visitor::create_declaration ( clang::CXXRecordDecl *  cls)
private

Create class element model from class declaration.

Parameters
clsClass declaration
Returns
Class diagram element model

Definition at line 898 of file translation_unit_visitor.cc.

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}

◆ create_declaration() [2/3]

std::unique_ptr< class_ > clanguml::class_diagram::visitor::translation_unit_visitor::create_declaration ( clang::RecordDecl *  rec)
private

Create class element model from record (e.g. struct) declaration.

Parameters
recRecord declaration
Returns
Class diagram element model

Definition at line 852 of file translation_unit_visitor.cc.

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}

◆ create_declaration() [3/3]

std::unique_ptr< clanguml::class_diagram::model::enum_ > clanguml::class_diagram::visitor::translation_unit_visitor::create_declaration ( const clang::EnumDecl *  enm,
const clang::TypedefDecl *  typedef_decl 
)
private

Create enum element model from enum (e.g. struct) declaration.

Parameters
recEnum declaration
Returns
Enum diagram element model

Definition at line 159 of file translation_unit_visitor.cc.

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}

◆ create_element()

std::unique_ptr< class_ > clanguml::class_diagram::visitor::translation_unit_visitor::create_element ( const clang::NamedDecl *  decl) const

Definition at line 37 of file translation_unit_visitor.cc.

39{
40 auto cls = std::make_unique<class_>(config().using_namespace());
41 cls->is_struct(common::is_struct(decl));
42 return cls;
43}

◆ create_objc_category_declaration()

std::unique_ptr< clanguml::class_diagram::model::objc_interface > clanguml::class_diagram::visitor::translation_unit_visitor::create_objc_category_declaration ( clang::ObjCCategoryDecl *  decl)
private

Create class element model from Objective-C category.

Parameters
declObjective-C protocol declaration
Returns
Class diagram element model

Definition at line 934 of file translation_unit_visitor.cc.

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}

◆ create_objc_interface_declaration()

std::unique_ptr< clanguml::class_diagram::model::objc_interface > clanguml::class_diagram::visitor::translation_unit_visitor::create_objc_interface_declaration ( clang::ObjCInterfaceDecl *  decl)
private

Create class element model from Objective-C interface.

Parameters
declObjective-C protocol declaration
Returns
Class diagram element model

Definition at line 992 of file translation_unit_visitor.cc.

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}

◆ create_objc_protocol_declaration()

std::unique_ptr< clanguml::class_diagram::model::objc_interface > clanguml::class_diagram::visitor::translation_unit_visitor::create_objc_protocol_declaration ( clang::ObjCProtocolDecl *  decl)
private

Create class element model from Objective-C protocol.

Parameters
declObjective-C protocol declaration
Returns
Class diagram element model

Definition at line 964 of file translation_unit_visitor.cc.

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}

◆ extract_constrained_template_param_name()

void clanguml::class_diagram::visitor::translation_unit_visitor::extract_constrained_template_param_name ( const clang::ConceptSpecializationExpr *  concept_specialization,
const clang::ConceptDecl *  cpt,
std::vector< std::string > &  constrained_template_params,
size_t  argument_index,
std::string &  type_name 
) const
private

Extract template contraint parameter name from raw source code.

Parameters
concept_specializationConcept specialization expression
cptConcept declaration
constrained_template_paramsFound constraint template param names
argument_indexArgument index
type_nameType parameter name - used if extraction fails

Definition at line 2493 of file translation_unit_visitor.cc.

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}

◆ finalize()

void clanguml::class_diagram::visitor::translation_unit_visitor::finalize ( )

Finalize diagram model.

This method is called after the entire AST has been visited by this visitor. It is used to perform necessary post processing on the diagram (e.g. resolve translation unit local element ID's into global ID's based on elements full names).

Definition at line 2484 of file translation_unit_visitor.cc.

2485{
2488 if (config().skip_redundant_dependencies()) {
2489 diagram().remove_redundant_dependencies();
2490 }
2491}

◆ find_instantiation_relationships()

void clanguml::class_diagram::visitor::translation_unit_visitor::find_instantiation_relationships ( common::model::template_element template_instantiation_base,
const std::string &  full_name,
eid_t  templated_decl_id 
)

Definition at line 2642 of file translation_unit_visitor.cc.

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}

◆ find_record_parent_id()

void clanguml::class_diagram::visitor::translation_unit_visitor::find_record_parent_id ( const clang::TagDecl *  decl,
std::optional< eid_t > &  parent_id_opt,
namespace_ &  parent_ns 
) const
private

Definition at line 2396 of file translation_unit_visitor.cc.

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}

◆ find_relationships()

bool clanguml::class_diagram::visitor::translation_unit_visitor::find_relationships ( const clang::Decl *  decl,
const clang::QualType &  type,
found_relationships_t &  relationships,
clanguml::common::model::relationship_t  relationship_hint 
)
private

Find relationships in a specific type.

Parameters
declSource declaration from which this relationship originates
typeType to search for relationships
relationship_hintDefault relationship type to infer from this type
Returns
True, if any relationships were found

Definition at line 1706 of file translation_unit_visitor.cc.

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}

◆ find_relationships_in_constraint_expression()

void clanguml::class_diagram::visitor::translation_unit_visitor::find_relationships_in_constraint_expression ( clanguml::common::model::element c,
const clang::Expr *  expr 
)
private

Find relationships in concept constraint expression.

Parameters
cDiagram element model (concept)
exprConcept constraint expression

Definition at line 663 of file translation_unit_visitor.cc.

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}

◆ has_processed_template_class()

bool clanguml::class_diagram::visitor::translation_unit_visitor::has_processed_template_class ( const std::string &  qualified_name) const
private

Check if template class has already been processed.

Parameters
qualified_nameFully qualified template class name
Returns
True, if template class has already been processed

Definition at line 2526 of file translation_unit_visitor.cc.

2528{
2530}

◆ override_relationship_hint()

std::pair< relationship_t, bool > clanguml::class_diagram::visitor::translation_unit_visitor::override_relationship_hint ( const std::string &  type_name,
int  index,
relationship_t  hint 
)
private

Try to override relationship hint using configuration file.

Parameters
type_name
hint
Returns
Maybe overridden relationship type hint

Definition at line 2066 of file translation_unit_visitor.cc.

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}

◆ process_children()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_children ( const clang::CXXRecordDecl *  cls,
clanguml::class_diagram::model::class_ c 
)
private

Process class children elements (members and methods)

Parameters
clsClass declaration
cClass diagram element model

Definition at line 1298 of file translation_unit_visitor.cc.

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}

◆ process_class_bases()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_class_bases ( const clang::CXXRecordDecl *  cls,
clanguml::class_diagram::model::class_ c 
)
private

Process class declaration bases (parents), if any.

Parameters
clsClass declaration
cClass diagram element model

Definition at line 1246 of file translation_unit_visitor.cc.

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}

◆ process_concept_specialization_relationships()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_concept_specialization_relationships ( common::model::element c,
const clang::ConceptSpecializationExpr *  concept_specialization 
)
private

Find concept specializations relationships.

Parameters
cConcept element model
concept_specializationConcept specialization expression

Definition at line 715 of file translation_unit_visitor.cc.

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())
763 c.add_relationship(
764 {relationship_t::kConstraint, target_id, access_t::kNone,
765 fmt::format(
766 "{}", fmt::join(constrained_template_params, ","))});
767 }
768}

◆ process_constraint_requirements()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_constraint_requirements ( const clang::ConceptDecl *  cpt,
const clang::Expr *  expr,
model::concept_ concept_model 
) const
private

Process concept constraint requirements.

Parameters
cptConcept declaration
exprRequires expression
concept_modelConcept diagram element model

Definition at line 534 of file translation_unit_visitor.cc.

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}

◆ process_declaration() [1/3]

void clanguml::class_diagram::visitor::translation_unit_visitor::process_declaration ( const clang::CXXRecordDecl &  cls,
clanguml::class_diagram::model::class_ c 
)
private

Process class declaration.

Parameters
clsClass declaration
cClass diagram element returned from create_declaration

Definition at line 1036 of file translation_unit_visitor.cc.

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}

◆ process_declaration() [2/3]

void clanguml::class_diagram::visitor::translation_unit_visitor::process_declaration ( const clang::EnumDecl &  enm,
clanguml::class_diagram::model::enum_ e 
)
private

Process enum declaration.

Parameters
enmEnum declaration
eEnum diagram element returned from create_declaration

Definition at line 218 of file translation_unit_visitor.cc.

220{
221 for (const auto &ev : enm.enumerators()) {
222 e.constants().push_back(ev->getNameAsString());
223 }
224
225 e.complete(true);
226}

◆ process_declaration() [3/3]

void clanguml::class_diagram::visitor::translation_unit_visitor::process_declaration ( const clang::RecordDecl &  cls,
class_ c 
)
private

Process record members.

Parameters
clsClass declaration
cClass diagram element model

Definition at line 1286 of file translation_unit_visitor.cc.

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}

◆ process_field()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_field ( const clang::FieldDecl &  field_declaration,
clanguml::class_diagram::model::class_ c 
)
private

Process class data member.

Parameters
field_declarationData member declaration
cClass diagram element model

Definition at line 2169 of file translation_unit_visitor.cc.

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}

◆ process_friend()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_friend ( const clang::FriendDecl &  f,
class_ c 
)
private

Process class friend.

Parameters
fFriend declaration
cClass diagram element model

Definition at line 1380 of file translation_unit_visitor.cc.

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}

◆ process_function_parameter()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_function_parameter ( const clang::ParmVarDecl &  param,
class_method method,
class_ c,
const std::set< std::string > &  template_parameter_names = {} 
)
private

Process function/method parameter.

Parameters
paramParameter declaration
methodClass method model
cClass diagram element model
template_parameter_namesIgnored

Definition at line 1939 of file translation_unit_visitor.cc.

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}

◆ process_function_parameter_find_relationships_in_autotype()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_function_parameter_find_relationships_in_autotype ( model::class_ c,
const clang::AutoType *  atsp 
)
private

Find relationships in function parameter.

Parameters
cClass diagram element model
atspauto type

Definition at line 1608 of file translation_unit_visitor.cc.

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}

◆ process_method()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_method ( const clang::CXXMethodDecl &  mf,
clanguml::class_diagram::model::class_ c 
)
private

Process class method.

Parameters
mfMethod declaration
cClass diagram element model

Definition at line 1402 of file translation_unit_visitor.cc.

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
1423 class_method method{common::access_specifier_to_access_t(mf.getAccess()),
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}

◆ process_method_properties()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_method_properties ( const clang::CXXMethodDecl &  mf,
const class_ c,
const std::string &  method_name,
class_method method 
) const
private

Process class method properties.

Parameters
mfMethod declaration
cClass diagram element model
method_nameMethod name
methodMethod model

Definition at line 1580 of file translation_unit_visitor.cc.

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}

◆ process_objc_category_declaration()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_objc_category_declaration ( const clang::ObjCCategoryDecl &  cls,
objc_interface c 
)
private

Process Objective-C category declaration.

Parameters
clsObjective-C category declaration
cClass diagram element returned from create_objc_category_declaration

Definition at line 1048 of file translation_unit_visitor.cc.

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());
1064 common::model::relationship r{
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}

◆ process_objc_interface_base()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_objc_interface_base ( const clang::ObjCInterfaceDecl &  cls,
objc_interface c 
)
private

Process Objective-C class base.

Parameters
clsObjective-C interface declaration
cClass diagram element model

Definition at line 1114 of file translation_unit_visitor.cc.

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);
1129 common::model::relationship cp{
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}

◆ process_objc_interface_declaration()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_objc_interface_declaration ( const clang::ObjCInterfaceDecl &  cls,
objc_interface c 
)
private

Process Objective-C interface declaration.

Parameters
clsObjective-C interface declaration
cClass diagram element returned from create_objc_interface_declaration

Definition at line 1092 of file translation_unit_visitor.cc.

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}

◆ process_objc_ivar()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_objc_ivar ( const clang::ObjCIvarDecl &  ivar,
objc_interface c 
)
private

Process Objective-C data member.

Parameters
field_declarationData member declaration
cClass diagram element model

Definition at line 1139 of file translation_unit_visitor.cc.

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}

◆ process_objc_method()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_objc_method ( const clang::ObjCMethodDecl &  mf,
objc_interface c 
)
private

Process Objective-C method.

Parameters
mfMethod declaration
cClass diagram element model

Definition at line 1521 of file translation_unit_visitor.cc.

1523{
1524 auto method_return_type =
1525 common::to_string(mf.getReturnType(), mf.getASTContext());
1526
1527 objc_method method{common::access_specifier_to_access_t(mf.getAccess()),
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}

◆ process_objc_method_parameter()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_objc_method_parameter ( const clang::ParmVarDecl &  param,
objc_method method,
objc_interface c 
)
private

Process Objective-C class method parameter.

Parameters
paramParameter declaration
methodClass method model
cClass diagram element model
template_parameter_namesIgnored

Definition at line 1892 of file translation_unit_visitor.cc.

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}

◆ process_objc_protocol_declaration()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_objc_protocol_declaration ( const clang::ObjCProtocolDecl &  cls,
objc_interface c 
)
private

Process Objective-C protocol declaration.

Parameters
clsObjective-C protocol declaration
cClass diagram element returned from create_objc_protocol_declaration

Definition at line 1077 of file translation_unit_visitor.cc.

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}

◆ process_record_parent()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_record_parent ( clang::RecordDecl *  cls,
class_ c,
const namespace_ &  ns 
)
private

Process record parent element (e.g. for nested classes)

This method handles nested classes or structs.

Parameters
clsRecord declaration
cClass diagram element model
nsPackage in the diagram to which the class c should belong

Definition at line 1018 of file translation_unit_visitor.cc.

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}

◆ process_record_parent_by_type()

template<typename T >
void clanguml::class_diagram::visitor::translation_unit_visitor::process_record_parent_by_type ( eid_t  parent_id,
class_ c,
namespace_  parent_ns,
const clang::RecordDecl *  decl 
)
private

Definition at line 633 of file translation_unit_visitor.h.

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}

◆ process_static_field()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_static_field ( const clang::VarDecl &  field_declaration,
clanguml::class_diagram::model::class_ c 
)
private

Process class static data member.

Parameters
field_declarationStatic data member declaration
cClass diagram element model

Definition at line 2086 of file translation_unit_visitor.cc.

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}

◆ process_template_method()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_template_method ( const clang::FunctionTemplateDecl &  mf,
clanguml::class_diagram::model::class_ c 
)
private

Process class template method.

Parameters
mfMethod declaration
cClass diagram element model

Definition at line 1659 of file translation_unit_visitor.cc.

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
1676 class_method method{common::access_specifier_to_access_t(mf.getAccess()),
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}

◆ process_template_specialization()

std::unique_ptr< class_ > clanguml::class_diagram::visitor::translation_unit_visitor::process_template_specialization ( clang::ClassTemplateSpecializationDecl *  cls)
private

Process class template specialization/instantiation.

Parameters
clsClass template specialization declaration
Returns
Class diagram element model

Definition at line 2124 of file translation_unit_visitor.cc.

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}

◆ process_template_specialization_children()

void clanguml::class_diagram::visitor::translation_unit_visitor::process_template_specialization_children ( const clang::ClassTemplateSpecializationDecl *  cls,
class_ c 
)
private

Process template specialization children (members and methods)

Parameters
clsClass template specialization declaration
cClass diagram element model

◆ resolve_local_to_global_ids()

void clanguml::class_diagram::visitor::translation_unit_visitor::resolve_local_to_global_ids ( )
private

Replace any AST local ids in diagram elements with global ones.

Not all elements global ids can be set in relationships during traversal of the AST. In such cases, a local id (obtained from getID()) and at after the traversal is complete, the id is replaced with the global diagram id.

Definition at line 2462 of file translation_unit_visitor.cc.

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}

◆ tbuilder()

template_builder_t & clanguml::class_diagram::visitor::translation_unit_visitor::tbuilder ( )
inlineprivate

Get template builder reference.

Returns
Reference to 'template_builder' instance

Definition at line 600 of file translation_unit_visitor.h.

600{ return template_builder_; }

Member Data Documentation

◆ anonymous_struct_relationships_

std::map<int64_t , std::tuple<std::string , common::model::relationship_t, common::model::access_t, std::optional<size_t> > > clanguml::class_diagram::visitor::translation_unit_visitor::anonymous_struct_relationships_
private

Definition at line 617 of file translation_unit_visitor.h.

◆ forward_declarations_

forward_declarations_t<class_, enum_> clanguml::class_diagram::visitor::translation_unit_visitor::forward_declarations_
private

Definition at line 611 of file translation_unit_visitor.h.

◆ processed_template_qualified_names_

std::set<std::string> clanguml::class_diagram::visitor::translation_unit_visitor::processed_template_qualified_names_
private

When visiting CXX records we need to know if they have already been process in VisitClassTemplateDecl or VisitClassTemplateSpecializationDecl. If yes, then we need to skip it

Todo:
There must be a better way to do this...

Definition at line 629 of file translation_unit_visitor.h.

◆ template_builder_

template_builder_t clanguml::class_diagram::visitor::translation_unit_visitor::template_builder_
private

Definition at line 609 of file translation_unit_visitor.h.

◆ typedef_enum_decls_

std::map<const clang::EnumDecl *, const clang::TypedefDecl *> clanguml::class_diagram::visitor::translation_unit_visitor::typedef_enum_decls_
private

Definition at line 620 of file translation_unit_visitor.h.


The documentation for this class was generated from the following files: