0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
Public Member Functions | Private Attributes | List of all members
clanguml::common::visitor::template_builder< VisitorT > Class Template Reference

Class responsible for building all kinds of templates from Clang AST. More...

Detailed Description

template<typename VisitorT>
class clanguml::common::visitor::template_builder< VisitorT >

Class responsible for building all kinds of templates from Clang AST.

Definition at line 49 of file template_builder.h.

#include <template_builder.h>

Public Member Functions

 template_builder (clanguml::common::model::diagram &diagram_, const clanguml::config::diagram &config_, VisitorT &visitor)
 Constructor.
 
common::model::diagramdiagram ()
 Get reference to the current diagram model.
 
const config::diagramconfig () const
 Get reference to the current diagram configuration.
 
const namespace_ & using_namespace () const
 Get diagram relative namespace.
 
bool simplify_system_template (template_parameter &ct, const std::string &full_name) const
 Simplify system templates.
 
void build_from_template_declaration (clanguml::common::model::template_trait &template_model, const clang::TemplateDecl &template_declaration, common::optional_ref< common::model::element > templated_element={})
 
void build_from_template_specialization_type (const clang::NamedDecl &location_declaration, clanguml::common::model::template_element &template_instantiation, const clang::NamedDecl *cls, const clang::TemplateSpecializationType &template_type_decl, std::optional< clanguml::common::model::template_element * > parent={})
 Basic template class build method.
 
void build (const clang::NamedDecl &location_declaration, clanguml::common::model::template_element &template_instantiation, const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl, clang::ArrayRef< clang::TemplateArgument > template_arguments, std::string full_template_specialization_name, std::optional< clanguml::common::model::template_element * > parent={})
 
void build_from_class_template_specialization (clanguml::common::model::template_element &template_instantiation, const clang::ClassTemplateSpecializationDecl &template_specialization, std::optional< clanguml::common::model::template_element * > parent={})
 Build template class from class template specialization decl.
 
bool add_base_classes (clanguml::common::model::template_element &tinst, std::deque< std::tuple< std::string, int, bool > > &template_base_params, int arg_index, bool variadic_params, const clanguml::common::model::template_parameter &ct)
 Add base classes to the template class, if any.
 
void process_template_arguments (const clang::NamedDecl &location_decl, std::optional< clanguml::common::model::template_element * > &parent, const clang::NamedDecl *cls, std::deque< std::tuple< std::string, int, bool > > &template_base_params, const clang::ArrayRef< clang::TemplateArgument > &template_args, clanguml::common::model::template_element &template_instantiation, const clang::TemplateDecl *template_decl)
 Process template class parameters and arguments.
 
void argument_process_dispatch (const clang::NamedDecl &location_decl, std::optional< clanguml::common::model::template_element * > &parent, const clang::NamedDecl *cls, clanguml::common::model::template_element &template_instantiation, const clang::TemplateDecl *template_decl, const clang::TemplateArgument &arg, size_t argument_index, std::vector< template_parameter > &argument)
 Process template arguments based on their type.
 
template_parameter process_expression_argument (const clang::TemplateArgument &arg)
 Process clang::TemplateArgument::Expression
 
template_parameter process_integral_argument (const clang::TemplateArgument &arg, const clang::ASTContext &ast_context)
 Process clang::TemplateArgument::Integral
 
template_parameter process_nullptr_argument (const clang::TemplateArgument &arg)
 Process clang::TemplateArgument::NullPtr
 
template_parameter process_null_argument (const clang::TemplateArgument &arg)
 Process clang::TemplateArgument::Null
 
std::vector< template_parameterprocess_pack_argument (const clang::NamedDecl &location_decl, std::optional< clanguml::common::model::template_element * > &parent, const clang::NamedDecl *cls, clanguml::common::model::template_element &template_instantiation, const clang::TemplateDecl *base_template_decl, const clang::TemplateArgument &arg, size_t argument_index, std::vector< template_parameter > &argument)
 Process clang::TemplateArgument::Pack
 
template_parameter process_type_argument (const clang::NamedDecl &location_decl, std::optional< clanguml::common::model::template_element * > &parent, const clang::NamedDecl *cls, const clang::TemplateDecl *base_template_decl, clang::QualType type, clanguml::common::model::template_element &template_instantiation, size_t argument_index)
 Process clang::TemplateArgument::Type
 
common::model::template_parameter process_template_argument (const clang::TemplateArgument &arg)
 Process clang::TemplateArgument::Template
 
common::model::template_parameter process_template_expansion (const clang::TemplateArgument &arg)
 Process clang::TemplateArgument::TemplateExpansion
 
std::optional< template_parametertry_as_function_prototype (const clang::NamedDecl &location_decl, std::optional< clanguml::common::model::template_element * > &parent, const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl, clang::QualType &type, clanguml::common::model::template_element &template_instantiation, size_t argument_index)
 Try to process template type argument as function template.
 
std::optional< template_parametertry_as_array (const clang::NamedDecl &location_decl, std::optional< clanguml::common::model::template_element * > &parent, const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl, clang::QualType &type, clanguml::common::model::template_element &template_instantiation, size_t argument_index)
 Try to process template type argument as array.
 
std::optional< template_parametertry_as_template_specialization_type (const clang::NamedDecl &location_decl, std::optional< clanguml::common::model::template_element * > &parent, const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl, clang::QualType &type, clanguml::common::model::template_element &template_instantiation, size_t argument_index)
 Try to process template type argument as specialization type.
 
std::optional< template_parametertry_as_template_parm_type (const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl, clang::QualType &type)
 Try to process template type argument as template parameter.
 
std::optional< template_parametertry_as_lambda (const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl, clang::QualType &type)
 Try to process template type argument as lambda.
 
std::optional< template_parametertry_as_record_type (const clang::NamedDecl &location_decl, std::optional< clanguml::common::model::template_element * > &parent, const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl, clang::QualType &type, clanguml::common::model::template_element &template_instantiation, size_t argument_index)
 Try to process template type argument as record type.
 
std::optional< template_parametertry_as_enum_type (std::optional< clanguml::common::model::template_element * > &parent, const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl, clang::QualType &type, clanguml::common::model::template_element &template_instantiation)
 Try to process template type argument as enum type.
 
std::optional< template_parametertry_as_builtin_type (std::optional< clanguml::common::model::template_element * > &parent, clang::QualType &type, const clang::TemplateDecl *template_decl)
 Try to process template type argument as builtin type.
 
std::optional< template_parametertry_as_member_pointer (const clang::NamedDecl &location_decl, std::optional< clanguml::common::model::template_element * > &parent, const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl, clang::QualType &type, clanguml::common::model::template_element &template_instantiation, size_t argument_index)
 Try to process template type argument as member pointer type.
 
std::optional< template_parametertry_as_decl_type (std::optional< clanguml::common::model::template_element * > &parent, const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl, clang::QualType &type, clanguml::common::model::template_element &template_instantiation, size_t argument_index)
 Try to process template type argument as decltype() type.
 
std::optional< template_parametertry_as_typedef_type (std::optional< clanguml::common::model::template_element * > &parent, const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl, clang::QualType &type, clanguml::common::model::template_element &template_instantiation, size_t argument_index)
 Try to process template type argument as typedef/using type.
 
clang::QualType consume_context (clang::QualType type, template_parameter &tp) const
 Remove types context (e.g. const or reference qualifiers)
 
bool find_relationships_in_unexposed_template_params (const template_parameter &ct, found_relationships_t &relationships)
 Try to find additional relationships in unexposed parameters.
 
void find_instantiation_relationships (common::model::template_element &template_instantiation, eid_t id, const std::string &qualified_name) const
 
common::visitor::ast_id_mapperid_mapper ()
 Get reference to Clang AST to clang-uml id mapper.
 
clang::SourceManager & source_manager () const
 Get reference to the current source manager.
 

Private Attributes

clanguml::common::model::diagramdiagram_
 
const clanguml::config::diagramconfig_
 
common::visitor::ast_id_mapperid_mapper_
 
clang::SourceManager & source_manager_
 
VisitorT & visitor_
 

Constructor & Destructor Documentation

◆ template_builder()

template<typename VisitorT >
clanguml::common::visitor::template_builder< VisitorT >::template_builder ( clanguml::common::model::diagram diagram_,
const clanguml::config::diagram config_,
VisitorT &  visitor 
)

Constructor.

Parameters
visitorReference to class diagram translation_unit_visitor

Definition at line 566 of file template_builder.h.

571 , id_mapper_{visitor.id_mapper()}
572 , source_manager_{visitor.source_manager()}
573 , visitor_{visitor}
574{
575}

Member Function Documentation

◆ add_base_classes()

template<typename VisitorT >
bool clanguml::common::visitor::template_builder< VisitorT >::add_base_classes ( clanguml::common::model::template_element tinst,
std::deque< std::tuple< std::string, int, bool > > &  template_base_params,
int  arg_index,
bool  variadic_params,
const clanguml::common::model::template_parameter ct 
)

Add base classes to the template class, if any.

This method adds base classes to a template declaration or specialization, including inferring whether variadic template parameter bases.

Parameters
tinstClass template model
template_base_paramsList of base class template parameters
arg_indexIndex of the template argument used for base class
variadic_paramsWhether the parameter is variadic
ctTemplate parameter model
Returns
True, if any base classes were added

Definition at line 1890 of file template_builder.h.

1894{
1895 bool add_template_argument_as_base_class = false;
1896
1897 if (variadic_params) {
1898 add_template_argument_as_base_class = true;
1899 }
1900 else {
1901 auto [arg_name, index, is_variadic] = template_base_params.front();
1902
1903 variadic_params = is_variadic;
1904 if ((arg_index == index) || (is_variadic && arg_index >= index)) {
1905 add_template_argument_as_base_class = true;
1906 if (!is_variadic) {
1907 // Don't remove the remaining variadic parameter
1908 template_base_params.pop_front();
1909 }
1910 }
1911 }
1912
1913 const auto maybe_id = ct.id();
1914 if (add_template_argument_as_base_class && maybe_id) {
1915 LOG_DBG("Adding template argument as base class '{}'",
1916 ct.to_string({}, false));
1917
1918 dynamic_cast<class_diagram::model::class_ &>(tinst).add_relationship(
1919 common::model::relationship{
1920 maybe_id.value(), common::model::access_t::kPublic, false});
1921 }
1922
1923 return variadic_params;
1924}

◆ argument_process_dispatch()

template<typename VisitorT >
void clanguml::common::visitor::template_builder< VisitorT >::argument_process_dispatch ( const clang::NamedDecl &  location_decl,
std::optional< clanguml::common::model::template_element * > &  parent,
const clang::NamedDecl *  cls,
clanguml::common::model::template_element template_instantiation,
const clang::TemplateDecl *  template_decl,
const clang::TemplateArgument &  arg,
size_t  argument_index,
std::vector< template_parameter > &  argument 
)

Process template arguments based on their type.

Parameters
parentOptional class in which this template is contained
clsTemplate class specialization declaration
template_instantiationTemplate class model to add template args
template_declBase template declaration
argTemplate argument
argument_indexArgument index
argumentOutput list of arguments (can be more than one for variadic parameters)

Definition at line 1053 of file template_builder.h.

1061{
1062 LOG_DBG("Processing argument {} in template class: {}", argument_index,
1063 cls->getQualifiedNameAsString());
1064
1065 switch (arg.getKind()) {
1066 case clang::TemplateArgument::Null:
1067 argument.push_back(process_null_argument(arg));
1068 break;
1069 case clang::TemplateArgument::Template:
1070 argument.push_back(process_template_argument(arg));
1071 break;
1072 case clang::TemplateArgument::Type: {
1073 argument.push_back(
1074 process_type_argument(location_decl, parent, cls, template_decl,
1075 arg.getAsType(), template_instantiation, argument_index));
1076 break;
1077 }
1078 case clang::TemplateArgument::Declaration:
1079 break;
1080 case clang::TemplateArgument::NullPtr:
1081 argument.push_back(process_nullptr_argument(arg));
1082 break;
1083 case clang::TemplateArgument::Integral:
1084 argument.push_back(
1085 process_integral_argument(arg, template_decl->getASTContext()));
1086 break;
1087 case clang::TemplateArgument::TemplateExpansion:
1088 argument.push_back(process_template_expansion(arg));
1089 break;
1090 case clang::TemplateArgument::Expression:
1091 argument.push_back(process_expression_argument(arg));
1092 break;
1093 case clang::TemplateArgument::Pack:
1094 for (auto &a : process_pack_argument(location_decl, parent, cls,
1095 template_instantiation, template_decl, arg, argument_index,
1096 argument)) {
1097 argument.push_back(a);
1098 }
1099 break;
1100#if LLVM_VERSION_MAJOR > 17
1101 case clang::TemplateArgument::StructuralValue:
1102 break;
1103#endif
1104 }
1105}

◆ build()

template<typename VisitorT >
void clanguml::common::visitor::template_builder< VisitorT >::build ( const clang::NamedDecl &  location_declaration,
clanguml::common::model::template_element template_instantiation,
const clang::NamedDecl *  cls,
const clang::TemplateDecl *  template_decl,
clang::ArrayRef< clang::TemplateArgument >  template_arguments,
std::string  full_template_specialization_name,
std::optional< clanguml::common::model::template_element * >  parent = {} 
)

Definition at line 777 of file template_builder.h.

783{
784 //
785 // Here we'll hold the template base class params to replace with the
786 // instantiated values
787 //
788 std::deque<std::tuple</*parameter name*/ std::string, /* position */ int,
789 /*is variadic */ bool>>
790 template_base_params{};
791
792 auto template_decl_qualified_name =
793 template_decl->getQualifiedNameAsString();
794
795 if (const auto *class_template_decl =
796 clang::dyn_cast<clang::ClassTemplateDecl>(template_decl);
797 (class_template_decl != nullptr) &&
798 (class_template_decl->getTemplatedDecl() != nullptr) &&
799 (class_template_decl->getTemplatedDecl()->getParent() != nullptr) &&
800 class_template_decl->getTemplatedDecl()->getParent()->isRecord()) {
801
802 namespace_ ns{
803 common::get_tag_namespace(*class_template_decl->getTemplatedDecl()
804 ->getParent()
805 ->getOuterLexicalRecordContext())};
806
807 std::string ns_str = ns.to_string();
808 std::string name = template_decl->getQualifiedNameAsString();
809 if (!ns_str.empty()) {
810 name = name.substr(ns_str.size() + 2);
811 }
812
813 util::replace_all(name, "::", "##");
814 template_instantiation.set_name(name);
815
816 template_instantiation.set_namespace(ns);
817 }
818 else {
819 namespace_ ns{template_decl_qualified_name};
820 ns.pop_back();
821 template_instantiation.set_name(template_decl->getNameAsString());
822 template_instantiation.set_namespace(ns);
823 }
824
825 // TODO: Refactor handling of base parameters to a separate method
826
827 // We need this to match any possible base classes coming from template
828 // arguments
829 std::vector<
830 std::pair</* parameter name */ std::string, /* is variadic */ bool>>
831 template_parameter_names{};
832
833 for (const auto *parameter : *template_decl->getTemplateParameters()) {
834 if (parameter->isTemplateParameter() &&
835 (parameter->isTemplateParameterPack() ||
836 parameter->isParameterPack())) {
837 template_parameter_names.emplace_back(
838 parameter->getNameAsString(), true);
839 }
840 else
841 template_parameter_names.emplace_back(
842 parameter->getNameAsString(), false);
843 }
844
845 // Check if the primary template has any base classes
846 int base_index = 0;
847
848 const auto *templated_class_decl =
849 clang::dyn_cast_or_null<clang::CXXRecordDecl>(
850 template_decl->getTemplatedDecl());
851
852 if ((templated_class_decl != nullptr) &&
853 templated_class_decl->hasDefinition())
854 for (const auto &base : templated_class_decl->bases()) {
855 const auto base_class_name = common::to_string(
856 base.getType(), templated_class_decl->getASTContext(), false);
857
858 LOG_DBG("Found template instantiation base: {}, {}",
859 base_class_name, base_index);
860
861 // Check if any of the primary template arguments has a
862 // parameter equal to this type
863 auto it = std::find_if(template_parameter_names.begin(),
864 template_parameter_names.end(),
865 [&base_class_name](
866 const auto &p) { return p.first == base_class_name; });
867
868 if (it != template_parameter_names.end()) {
869 const auto &parameter_name = it->first;
870 const bool is_variadic = it->second;
871 // Found base class which is a template parameter
872 LOG_DBG("Found base class which is a template parameter "
873 "{}, {}, {}",
874 parameter_name, is_variadic,
875 std::distance(template_parameter_names.begin(), it));
876
877 template_base_params.emplace_back(parameter_name,
878 std::distance(template_parameter_names.begin(), it),
879 is_variadic);
880 }
881 else {
882 // This is a regular base class - it is handled by
883 // process_template
884 }
885 base_index++;
886 }
887
888 process_template_arguments(location_decl, parent, cls, template_base_params,
889 template_arguments, template_instantiation, template_decl);
890
891 if constexpr (std::is_same_v<typename VisitorT::diagram_t,
892 class_diagram::model::diagram>) {
893 find_instantiation_relationships(template_instantiation,
894 eid_t{template_decl->getID()}, full_template_specialization_name);
895 }
896
897 template_instantiation.set_id(
898 common::to_id(template_instantiation.full_name(false)));
899
900 visitor_.set_source_location(location_decl, template_instantiation);
901}

◆ build_from_class_template_specialization()

template<typename VisitorT >
void clanguml::common::visitor::template_builder< VisitorT >::build_from_class_template_specialization ( clanguml::common::model::template_element template_instantiation,
const clang::ClassTemplateSpecializationDecl &  template_specialization,
std::optional< clanguml::common::model::template_element * >  parent = {} 
)

Build template class from class template specialization decl.

Parameters
template_specializationClass template specialization declaration
parentOptional class in which this template is contained
Returns
Created template class model

Definition at line 904 of file template_builder.h.

908{
909 LOG_DBG("Building class template specialization model: {}",
910 template_specialization.getQualifiedNameAsString());
911
912 //
913 // Here we'll hold the template base params to replace with the
914 // instantiated values
915 //
916 std::deque<std::tuple</*parameter name*/ std::string, /* position */ int,
917 /*is variadic */ bool>>
918 template_base_params{};
919
920 const clang::ClassTemplateDecl *template_decl =
921 template_specialization.getSpecializedTemplate();
922
923 auto qualified_name = template_decl->getQualifiedNameAsString();
924
925 namespace_ ns{qualified_name};
926 ns.pop_back();
927 template_instantiation.set_name(template_decl->getNameAsString());
928 template_instantiation.set_namespace(ns);
929
930 process_template_arguments(template_specialization, parent,
931 &template_specialization, template_base_params,
932 template_specialization.getTemplateArgs().asArray(),
933 template_instantiation, template_decl);
934
935 // Update the id after the template parameters are processed
936 template_instantiation.set_id(
937 common::to_id(template_instantiation.full_name(false)));
938
939 if constexpr (std::is_same_v<typename VisitorT::diagram_t,
940 class_diagram::model::diagram>) {
941 find_instantiation_relationships(template_instantiation,
942 eid_t{template_specialization.getID()}, qualified_name);
943 }
944
945 visitor_.set_source_location(
946 template_specialization, template_instantiation);
947}

◆ build_from_template_declaration()

template<typename VisitorT >
void clanguml::common::visitor::template_builder< VisitorT >::build_from_template_declaration ( clanguml::common::model::template_trait template_model,
const clang::TemplateDecl &  template_declaration,
common::optional_ref< common::model::element templated_element = {} 
)

Definition at line 627 of file template_builder.h.

631{
632 LOG_DBG("Processing {} template parameters...",
633 common::get_qualified_name(template_declaration));
634
635 if (template_declaration.getTemplateParameters() == nullptr)
636 return;
637
638 for (const auto *parameter :
639 *template_declaration.getTemplateParameters()) {
640 if (clang::dyn_cast_or_null<clang::TemplateTypeParmDecl>(parameter) !=
641 nullptr) {
642 const auto *template_type_parameter =
643 clang::dyn_cast_or_null<clang::TemplateTypeParmDecl>(parameter);
644
645 std::optional<std::string> default_arg;
646 if (template_type_parameter->hasDefaultArgument()) {
647 default_arg =
648#if LLVM_VERSION_MAJOR > 18
650 template_type_parameter->getDefaultArgument(),
651 template_declaration.getASTContext());
652#else
653 template_type_parameter->getDefaultArgument().getAsString();
654#endif
655 }
656
657 auto parameter_name = template_type_parameter->getNameAsString();
658 if (parameter_name.empty())
659 parameter_name = "typename";
660
661 auto ct = template_parameter::make_template_type(parameter_name,
662 default_arg, template_type_parameter->isParameterPack());
663
664 if constexpr (std::is_same_v<typename VisitorT::diagram_t,
666 if (template_type_parameter->getTypeConstraint() != nullptr) {
668 template_type_parameter->getTypeConstraint()
669 ->getNamedConcept(),
670 [this, &ct, &templated_element](
671 const clang::ConceptDecl *named_concept) mutable {
672 ct.set_concept_constraint(
673 named_concept->getQualifiedNameAsString());
674 if (templated_element &&
675 visitor_.should_include(named_concept)) {
676 templated_element.value().add_relationship(
677 {relationship_t::kConstraint,
678 id_mapper()
679 .get_global_id(
680 eid_t{named_concept->getID()})
681 .value(),
682 model::access_t::kNone,
683 ct.name().value()});
684 }
685 });
686 }
687 }
688 else {
689 (void)templated_element; // NOLINT
690 }
691
692 template_model.add_template(std::move(ct));
693 }
694 else if (clang::dyn_cast_or_null<clang::NonTypeTemplateParmDecl>(
695 parameter) != nullptr) {
696 const auto *template_nontype_parameter =
697 clang::dyn_cast_or_null<clang::NonTypeTemplateParmDecl>(
698 parameter);
699
700 std::optional<std::string> default_arg;
701
702 if (template_nontype_parameter->hasDefaultArgument()) {
703 default_arg =
704#if LLVM_VERSION_MAJOR > 18
706 template_nontype_parameter->getDefaultArgument(),
707 template_declaration.getASTContext());
708#else
710 template_nontype_parameter->getDefaultArgument());
711#endif
712 }
714 template_nontype_parameter->getType().getAsString(),
715 template_nontype_parameter->getNameAsString(), default_arg,
716 template_nontype_parameter->isParameterPack());
717
718 template_model.add_template(std::move(ct));
719 }
720 else if (clang::dyn_cast_or_null<clang::TemplateTemplateParmDecl>(
721 parameter) != nullptr) {
722 const auto *template_template_parameter =
723 clang::dyn_cast_or_null<clang::TemplateTemplateParmDecl>(
724 parameter);
725 std::optional<std::string> default_arg;
726 if (template_template_parameter->hasDefaultArgument()) {
727 default_arg = common::to_string(
728 template_template_parameter->getDefaultArgument()
729 .getArgument());
730 }
732 template_template_parameter->getNameAsString(), default_arg,
733 template_template_parameter->isParameterPack());
734
735 template_model.add_template(std::move(ct));
736 }
737 else {
738 // pass
739 }
740 }
741}

◆ build_from_template_specialization_type()

template<typename VisitorT >
void clanguml::common::visitor::template_builder< VisitorT >::build_from_template_specialization_type ( const clang::NamedDecl &  location_declaration,
clanguml::common::model::template_element template_instantiation,
const clang::NamedDecl *  cls,
const clang::TemplateSpecializationType &  template_type_decl,
std::optional< clanguml::common::model::template_element * >  parent = {} 
)

Basic template class build method.

Parameters
clsClang template declaration
template_type_declTemplate specialization type
parentOptional class in which this template is contained
Returns
Created template class model

Definition at line 744 of file template_builder.h.

750{
751 const auto *template_type_ptr = &template_type_decl;
752
753 if (template_type_decl.isTypeAlias()) {
754 if (const auto *tsp =
755 template_type_decl.getAliasedType()
756 ->template getAs<clang::TemplateSpecializationType>();
757 tsp != nullptr)
758 template_type_ptr = tsp;
759 }
760
761 const auto &template_type = *template_type_ptr;
762
763 template_instantiation.is_template(true);
764
765 std::string full_template_specialization_name = common::to_string(
766 template_type.desugar(),
767 template_type.getTemplateName().getAsTemplateDecl()->getASTContext());
768
769 auto *template_decl{template_type.getTemplateName().getAsTemplateDecl()};
770
771 build(location_declaration, template_instantiation, cls, template_decl,
772 template_type.template_arguments(), full_template_specialization_name,
773 parent);
774}

◆ config()

template<typename VisitorT >
const config::diagram & clanguml::common::visitor::template_builder< VisitorT >::config

Get reference to the current diagram configuration.

Returns
Reference to the current diagram configuration

Definition at line 584 of file template_builder.h.

585{
586 return config_;
587}

◆ consume_context()

template<typename VisitorT >
clang::QualType clanguml::common::visitor::template_builder< VisitorT >::consume_context ( clang::QualType  type,
template_parameter tp 
) const

Remove types context (e.g. const or reference qualifiers)

This method removes all const and reference/pointer qualifiers from type, adds them to the template parameter model tp and returns a type without context.

Parameters
typeType to remove context from
tpTemplate model to add context to
Returns
Type without context

Definition at line 1138 of file template_builder.h.

1140{
1141 auto [unqualified_type, context] = common::consume_type_context(type);
1142
1143 tp.deduced_context(std::move(context));
1144
1145 return unqualified_type;
1146}

◆ diagram()

template<typename VisitorT >
common::model::diagram & clanguml::common::visitor::template_builder< VisitorT >::diagram

Get reference to the current diagram model.

Returns
Reference to the current diagram model

Definition at line 578 of file template_builder.h.

579{
580 return diagram_;
581}

◆ find_instantiation_relationships()

template<typename VisitorT >
void clanguml::common::visitor::template_builder< VisitorT >::find_instantiation_relationships ( common::model::template_element template_instantiation,
eid_t  id,
const std::string &  qualified_name 
) const

Definition at line 950 of file template_builder.h.

953{
954 visitor_.find_instantiation_relationships(
955 template_instantiation, qualified_name, id);
956}

◆ find_relationships_in_unexposed_template_params()

template<typename VisitorT >
bool clanguml::common::visitor::template_builder< VisitorT >::find_relationships_in_unexposed_template_params ( const template_parameter ct,
found_relationships_t relationships 
)

Try to find additional relationships in unexposed parameters.

Sometimes, Clang will report a template parameter as unexposed, which means all we get a is a string representation of the type, sometimes with template parameter names replaced with type-parameter-X-Y string.

This method tries to find any type names, which might be relevant for the diagram relationships.

Parameters
ctTemplate argument model
relationshipsList of discovered relationships
Returns
True, if any relationships were found

Definition at line 1227 of file template_builder.h.

1230{
1231 const auto &type = ct.type();
1232
1233 if (!type)
1234 return false;
1235
1236 bool found{false};
1237 LOG_DBG("Finding relationships in user defined type: {}",
1238 ct.to_string(config().using_namespace(), false));
1239
1240 auto type_with_namespace =
1241 std::make_optional<common::model::namespace_>(type.value());
1242
1243 if (!type_with_namespace.has_value()) {
1244 // Couldn't find declaration of this type
1245 type_with_namespace = common::model::namespace_{type.value()};
1246 }
1247
1248 auto element_opt = diagram().get(type_with_namespace.value().to_string());
1249 if (config_.generate_template_argument_dependencies() && element_opt) {
1250 relationships.emplace_back(
1251 element_opt.value().id(), relationship_t::kDependency);
1252 found = true;
1253 }
1254
1255 for (const auto &nested_template_params : ct.template_params()) {
1257 nested_template_params, relationships) ||
1258 found;
1259 }
1260
1261 return found;
1262}

◆ id_mapper()

template<typename VisitorT >
common::visitor::ast_id_mapper & clanguml::common::visitor::template_builder< VisitorT >::id_mapper

Get reference to Clang AST to clang-uml id mapper.

Returns
Reference to Clang AST to clang-uml id mapper

Definition at line 596 of file template_builder.h.

597{
598 return id_mapper_;
599}

◆ process_expression_argument()

template<typename VisitorT >
template_parameter clanguml::common::visitor::template_builder< VisitorT >::process_expression_argument ( const clang::TemplateArgument &  arg)

Process clang::TemplateArgument::Expression

Note
The template argument is a pack expansion of a template name that was provided for a template template parameter.
Parameters
argTemplate argument
Returns
Return template argument model

Definition at line 1325 of file template_builder.h.

1327{
1328 assert(arg.getKind() == clang::TemplateArgument::Expression);
1330 arg.getAsExpr()->getSourceRange(), source_manager()));
1331}

◆ process_integral_argument()

template<typename VisitorT >
template_parameter clanguml::common::visitor::template_builder< VisitorT >::process_integral_argument ( const clang::TemplateArgument &  arg,
const clang::ASTContext &  ast_context 
)

Process clang::TemplateArgument::Integral

Note
The template argument is an integral value stored in an llvm::APSInt that was provided for an integral non-type template parameter.
Parameters
argTemplate argument
Returns
Return template argument model

Definition at line 1265 of file template_builder.h.

1267{
1268 assert(arg.getKind() == clang::TemplateArgument::Integral);
1269
1270 std::string result;
1271 llvm::raw_string_ostream ostream(result);
1272 clang::PrintingPolicy policy(ast_context.getLangOpts());
1273
1274#if LLVM_VERSION_MAJOR > 18
1275 arg.print(policy, ostream, false);
1276#else
1277 arg.dump(ostream);
1278#endif
1279
1280 return template_parameter::make_argument(result);
1281}

◆ process_null_argument()

template<typename VisitorT >
template_parameter clanguml::common::visitor::template_builder< VisitorT >::process_null_argument ( const clang::TemplateArgument &  arg)

Process clang::TemplateArgument::Null

Note
Represents an empty template argument, e.g., one that has not been deduced.
Parameters
argTemplate argument
Returns
Return template argument model

Definition at line 1305 of file template_builder.h.

1307{
1308 assert(arg.getKind() == clang::TemplateArgument::Null);
1309
1311}

◆ process_nullptr_argument()

template<typename VisitorT >
template_parameter clanguml::common::visitor::template_builder< VisitorT >::process_nullptr_argument ( const clang::TemplateArgument &  arg)

Process clang::TemplateArgument::NullPtr

Note
The template argument is a null pointer or null pointer to member that was provided for a non-type template parameter.
Parameters
argTemplate argument
Returns
Return template argument model

Definition at line 1314 of file template_builder.h.

1316{
1317 assert(arg.getKind() == clang::TemplateArgument::NullPtr);
1318
1319 LOG_DBG("Processing nullptr argument: {}", common::to_string(arg));
1320
1321 return template_parameter::make_argument("nullptr");
1322}

◆ process_pack_argument()

template<typename VisitorT >
std::vector< template_parameter > clanguml::common::visitor::template_builder< VisitorT >::process_pack_argument ( const clang::NamedDecl &  location_decl,
std::optional< clanguml::common::model::template_element * > &  parent,
const clang::NamedDecl *  cls,
clanguml::common::model::template_element template_instantiation,
const clang::TemplateDecl *  base_template_decl,
const clang::TemplateArgument &  arg,
size_t  argument_index,
std::vector< template_parameter > &  argument 
)

Process clang::TemplateArgument::Pack

Note
The template argument is actually a parameter pack. Arguments are stored in the Args struct.
Parameters
argTemplate argument
Returns
Return template argument model

Definition at line 1335 of file template_builder.h.

1343{
1344 assert(arg.getKind() == clang::TemplateArgument::Pack);
1345
1346 std::vector<template_parameter> res;
1347
1348 auto pack_argument_index = argument_index;
1349
1350 for (const auto &a : arg.getPackAsArray()) {
1351 argument_process_dispatch(location_decl, parent, cls,
1352 template_instantiation, base_template_decl, a,
1353 pack_argument_index++, res);
1354 }
1355
1356 return res;
1357}

◆ process_template_argument()

template<typename VisitorT >
template_parameter clanguml::common::visitor::template_builder< VisitorT >::process_template_argument ( const clang::TemplateArgument &  arg)

Process clang::TemplateArgument::Template

Note
The template argument is a template name that was provided for a template template parameter.
Parameters
argTemplate argument
Returns
Return template argument model

Definition at line 1108 of file template_builder.h.

1110{
1111 auto arg_name = common::to_string(arg);
1112
1113 LOG_DBG("Processing template argument: {}", arg_name);
1114
1116}

◆ process_template_arguments()

template<typename VisitorT >
void clanguml::common::visitor::template_builder< VisitorT >::process_template_arguments ( const clang::NamedDecl &  location_decl,
std::optional< clanguml::common::model::template_element * > &  parent,
const clang::NamedDecl *  cls,
std::deque< std::tuple< std::string, int, bool > > &  template_base_params,
const clang::ArrayRef< clang::TemplateArgument > &  template_args,
clanguml::common::model::template_element template_instantiation,
const clang::TemplateDecl *  template_decl 
)

Process template class parameters and arguments.

Parameters
parentOptional class in which this template is contained
clsTemplate class specialization declaration
template_base_paramsList of base class template parameters
template_argsList of template arguments
template_instantiationTemplate class model to add template args
template_declBase template declaration

Definition at line 959 of file template_builder.h.

967{
968 auto arg_index{0};
969
970 for (const auto &arg : template_args) {
971 // Argument can be a parameter pack in which case it gives multiple
972 // arguments
973 std::vector<template_parameter> arguments;
974
975 // Handle empty pack expansion
976 if (arg.getKind() == clang::TemplateArgument::ArgKind::Pack &&
977 arg.getPackAsArray().empty()) {
978 template_instantiation.add_template(
980 }
981
982 // For now ignore the default template arguments of templates which
983 // do not match the inclusion filters, to make the system
984 // templates 'nicer' - i.e. skipping the allocators and comparators
985 // TODO: Change this to ignore only when the arguments are set to
986 // default values, and add them when they are specifically
987 // overridden
988 if (!diagram().should_include(
989 namespace_{template_decl->getQualifiedNameAsString()})) {
990 const auto *maybe_type_parm_decl =
991 clang::dyn_cast<clang::TemplateTypeParmDecl>(
992 template_decl->getTemplateParameters()->getParam(
993 std::min<unsigned>(arg_index,
994 static_cast<unsigned>(
995 template_decl->getTemplateParameters()
996 ->size()) -
997 1)));
998 if (maybe_type_parm_decl != nullptr &&
999 maybe_type_parm_decl->hasDefaultArgument()) {
1000 continue;
1001 }
1002 }
1003
1004 //
1005 // Handle the template parameter/argument based on its kind
1006 //
1007 argument_process_dispatch(location_decl, parent, cls,
1008 template_instantiation, template_decl, arg, arg_index, arguments);
1009
1010 if (arguments.empty()) {
1011 arg_index++;
1012 continue;
1013 }
1014
1015 // We can figure this only when we encounter variadic param in
1016 // the list of template params, from then this variable is true
1017 // and we can process following template parameters as belonging
1018 // to the variadic tuple
1019 [[maybe_unused]] auto variadic_params{false};
1020
1021 if constexpr (std::is_same_v<typename VisitorT::diagram_t,
1022 class_diagram::model::diagram>) {
1023 // In case any of the template arguments are base classes, add
1024 // them as parents of the current template instantiation class
1025 if (!template_base_params.empty()) {
1026 variadic_params = add_base_classes(template_instantiation,
1027 template_base_params, arg_index, variadic_params,
1028 arguments.front());
1029 }
1030 }
1031
1032 for (auto &argument : arguments) {
1034 argument, argument.to_string(using_namespace(), false, true));
1035
1036 LOG_DBG("Adding template argument {} to template "
1037 "specialization/instantiation {}",
1038 argument.to_string(using_namespace(), false),
1039 template_instantiation.name());
1040
1041 template_instantiation.add_template(std::move(argument));
1042 }
1043
1044 arg_index++;
1045 }
1046
1047 // Update id
1048 template_instantiation.set_id(
1049 common::to_id(template_instantiation.full_name(false)));
1050}

◆ process_template_expansion()

template<typename VisitorT >
template_parameter clanguml::common::visitor::template_builder< VisitorT >::process_template_expansion ( const clang::TemplateArgument &  arg)

Process clang::TemplateArgument::TemplateExpansion

Note
The template argument is a pack expansion of a template name that was provided for a template template parameter.
Parameters
argTemplate argument
Returns
Return template argument model

Definition at line 1119 of file template_builder.h.

1121{
1122 auto arg_name = common::to_string(arg);
1123
1124 LOG_DBG("Processing template expansion argument: {}", arg_name);
1125
1127 arg.getAsTemplate().getAsTemplateDecl(), [&arg_name](const auto *decl) {
1128 arg_name = decl->getQualifiedNameAsString();
1129 });
1130
1131 auto param = template_parameter::make_template_type(arg_name);
1132 param.is_variadic(true);
1133
1134 return param;
1135}

◆ process_type_argument()

template<typename VisitorT >
template_parameter clanguml::common::visitor::template_builder< VisitorT >::process_type_argument ( const clang::NamedDecl &  location_decl,
std::optional< clanguml::common::model::template_element * > &  parent,
const clang::NamedDecl *  cls,
const clang::TemplateDecl *  base_template_decl,
clang::QualType  type,
clanguml::common::model::template_element template_instantiation,
size_t  argument_index 
)

Process clang::TemplateArgument::Type

Note
The template argument is a type.
Parameters
argTemplate argument
Returns
Return template argument model

Definition at line 1149 of file template_builder.h.

1156{
1157 std::optional<template_parameter> argument;
1158
1159 if (type->getAs<clang::ElaboratedType>() != nullptr) {
1160 type = type->getAs<clang::ElaboratedType>()->getNamedType(); // NOLINT
1161 }
1162
1163 auto type_name = common::to_string(type, &cls->getASTContext());
1164
1165 LOG_DBG("Processing template {} type argument {}: {}, {}, {}",
1166 template_decl->getQualifiedNameAsString(), argument_index, type_name,
1167 type->getTypeClassName(),
1168 common::to_string(type, cls->getASTContext()));
1169
1170 argument = try_as_function_prototype(location_decl, parent, cls,
1171 template_decl, type, template_instantiation, argument_index);
1172 if (argument)
1173 return *argument;
1174
1175 argument = try_as_member_pointer(location_decl, parent, cls, template_decl,
1176 type, template_instantiation, argument_index);
1177 if (argument)
1178 return *argument;
1179
1180 argument = try_as_array(location_decl, parent, cls, template_decl, type,
1181 template_instantiation, argument_index);
1182 if (argument)
1183 return *argument;
1184
1185 argument = try_as_template_parm_type(cls, template_decl, type);
1186 if (argument)
1187 return *argument;
1188
1189 argument = try_as_template_specialization_type(location_decl, parent, cls,
1190 template_decl, type, template_instantiation, argument_index);
1191 if (argument)
1192 return *argument;
1193
1194 argument = try_as_decl_type(parent, cls, template_decl, type,
1195 template_instantiation, argument_index);
1196 if (argument)
1197 return *argument;
1198
1199 argument = try_as_typedef_type(parent, cls, template_decl, type,
1200 template_instantiation, argument_index);
1201 if (argument)
1202 return *argument;
1203
1204 argument = try_as_lambda(cls, template_decl, type);
1205 if (argument)
1206 return *argument;
1207
1208 argument = try_as_record_type(location_decl, parent, cls, template_decl,
1209 type, template_instantiation, argument_index);
1210 if (argument)
1211 return *argument;
1212
1214 parent, cls, template_decl, type, template_instantiation);
1215 if (argument)
1216 return *argument;
1217
1218 argument = try_as_builtin_type(parent, type, template_decl);
1219 if (argument)
1220 return *argument;
1221
1222 // fallback
1223 return template_parameter::make_argument(type_name);
1224}

◆ simplify_system_template()

template<typename VisitorT >
bool clanguml::common::visitor::template_builder< VisitorT >::simplify_system_template ( template_parameter ct,
const std::string &  full_name 
) const

Simplify system templates.

This method tries to simplify all fully qualified template names in the full_name using substitutions from the configuration file ().

Typical example is replace every std::basic_string<char> with std::string.

Parameters
ctTemplate parameter
full_nameFull template name
Returns

Definition at line 608 of file template_builder.h.

610{
612 return false;
613
614 auto simplified = config().simplify_template_type(full_name);
615
616 if (simplified != full_name) {
617 ct.set_type(simplified);
618 ct.set_id(common::to_id(simplified));
619 ct.clear_params();
620 return true;
621 }
622
623 return false;
624}

◆ source_manager()

template<typename VisitorT >
clang::SourceManager & clanguml::common::visitor::template_builder< VisitorT >::source_manager

Get reference to the current source manager.

Returns
Reference to the current source manager

Definition at line 602 of file template_builder.h.

603{
604 return source_manager_;
605}

◆ try_as_array()

template<typename VisitorT >
std::optional< template_parameter > clanguml::common::visitor::template_builder< VisitorT >::try_as_array ( const clang::NamedDecl &  location_decl,
std::optional< clanguml::common::model::template_element * > &  parent,
const clang::NamedDecl *  cls,
const clang::TemplateDecl *  template_decl,
clang::QualType &  type,
clanguml::common::model::template_element template_instantiation,
size_t  argument_index 
)

Try to process template type argument as array.

Parameters
parentOptional class in which this template is contained
clsTemplate class specialization declaration
template_declBase template declaration
typeTemplate type
template_instantiationTemplate class model
argument_indexArgument index
Returns
Array template argument if succeeds, or std::nullopt

Definition at line 1440 of file template_builder.h.

1447{
1448 const auto *array_type = common::dereference(type)->getAsArrayTypeUnsafe();
1449 if (array_type == nullptr)
1450 return {};
1451
1453
1454 type = consume_context(type, argument);
1455
1456 argument.is_array(true);
1457
1458 // Set function template return type
1459 auto element_type = process_type_argument(location_decl, parent, cls,
1460 template_decl, array_type->getElementType(), template_instantiation,
1461 argument_index);
1462
1463 argument.add_template_param(element_type);
1464
1465 if (array_type->isDependentSizedArrayType() &&
1466 array_type->getDependence() ==
1467 clang::TypeDependence::DependentInstantiation) {
1468 argument.add_template_param(
1470 ((clang::DependentSizedArrayType *)array_type) // NOLINT
1471 ->getSizeExpr())));
1472 }
1473 else if (array_type->isConstantArrayType()) {
1475 std::to_string(((clang::ConstantArrayType *)array_type) // NOLINT
1476 ->getSize()
1477 .getLimitedValue())));
1478 }
1479
1480 // TODO: Handle variable sized arrays
1481
1482 return argument;
1483}

◆ try_as_builtin_type()

template<typename VisitorT >
std::optional< template_parameter > clanguml::common::visitor::template_builder< VisitorT >::try_as_builtin_type ( std::optional< clanguml::common::model::template_element * > &  parent,
clang::QualType &  type,
const clang::TemplateDecl *  template_decl 
)

Try to process template type argument as builtin type.

Parameters
parentOptional class in which this template is contained
typeTemplate type
template_declBase template declaration
Returns
Builtin type template argument if succeeds, or std::nullopt

Definition at line 1870 of file template_builder.h.

1873{
1874 const auto *builtin_type = type->getAs<clang::BuiltinType>();
1875 if (builtin_type == nullptr)
1876 return {};
1877
1878 LOG_DBG("Template argument is a builtin type");
1879
1880 auto type_name = common::to_string(type, template_decl->getASTContext());
1882
1883 type = consume_context(type, argument);
1884 argument.set_type(type_name);
1885
1886 return argument;
1887}

◆ try_as_decl_type()

template<typename VisitorT >
std::optional< template_parameter > clanguml::common::visitor::template_builder< VisitorT >::try_as_decl_type ( std::optional< clanguml::common::model::template_element * > &  parent,
const clang::NamedDecl *  cls,
const clang::TemplateDecl *  template_decl,
clang::QualType &  type,
clanguml::common::model::template_element template_instantiation,
size_t  argument_index 
)

Try to process template type argument as decltype() type.

Parameters
parentOptional class in which this template is contained
clsTemplate class specialization declaration
template_declBase template declaration
typeTemplate type
template_instantiationTemplate class model
argument_indexArgument index
Returns
decltype() type template argument if succeeds, or std::nullopt

Definition at line 1540 of file template_builder.h.

1546{
1547 const auto *decl_type =
1548 common::dereference(type)->getAs<clang::DecltypeType>();
1549 if (decl_type == nullptr) {
1550 return {};
1551 }
1552
1553 LOG_DBG("Template argument is a decltype()");
1554
1555 // TODO
1556 return {};
1557}

◆ try_as_enum_type()

template<typename VisitorT >
std::optional< template_parameter > clanguml::common::visitor::template_builder< VisitorT >::try_as_enum_type ( std::optional< clanguml::common::model::template_element * > &  parent,
const clang::NamedDecl *  cls,
const clang::TemplateDecl *  template_decl,
clang::QualType &  type,
clanguml::common::model::template_element template_instantiation 
)

Try to process template type argument as enum type.

Parameters
parentOptional class in which this template is contained
clsTemplate class specialization declaration
template_declBase template declaration
typeTemplate type
template_instantiationTemplate class model
Returns
Enum type template argument if succeeds, or std::nullopt

Definition at line 1839 of file template_builder.h.

1844{
1845 const auto *enum_type = type->getAs<clang::EnumType>();
1846 if (enum_type == nullptr)
1847 return {};
1848
1849 LOG_DBG("Template argument is a an enum");
1850
1852 type = consume_context(type, argument);
1853
1854 auto type_name = common::to_string(type, template_decl->getASTContext());
1855 argument.set_type(type_name);
1856 const auto type_id = common::to_id(type_name);
1857 argument.set_id(type_id);
1858
1859 if (enum_type->getAsTagDecl() != nullptr &&
1861 template_instantiation.add_relationship(
1862 {relationship_t::kDependency, type_id});
1863 }
1864
1865 return argument;
1866}

◆ try_as_function_prototype()

template<typename VisitorT >
std::optional< template_parameter > clanguml::common::visitor::template_builder< VisitorT >::try_as_function_prototype ( const clang::NamedDecl &  location_decl,
std::optional< clanguml::common::model::template_element * > &  parent,
const clang::NamedDecl *  cls,
const clang::TemplateDecl *  template_decl,
clang::QualType &  type,
clanguml::common::model::template_element template_instantiation,
size_t  argument_index 
)

Try to process template type argument as function template.

Parameters
parentOptional class in which this template is contained
clsTemplate class specialization declaration
template_declBase template declaration
typeTemplate type
template_instantiationTemplate class model
argument_indexArgument index
Returns
Function template argument if succeeds, or std::nullopt

Definition at line 1487 of file template_builder.h.

1494{
1495 const auto *function_type = type->getAs<clang::FunctionProtoType>();
1496
1497 if (function_type == nullptr && type->isFunctionPointerType()) {
1498 function_type =
1499 type->getPointeeType()->getAs<clang::FunctionProtoType>();
1500 if (function_type == nullptr)
1501 return {};
1502 }
1503
1504 if (function_type == nullptr)
1505 return {};
1506
1507 LOG_DBG("Template argument is a function prototype");
1508
1510
1511 type = consume_context(type, argument);
1512
1513 argument.is_function_template(true);
1514
1515 // Set function template return type
1516 auto return_arg = process_type_argument(location_decl, parent, cls,
1517 template_decl, function_type->getReturnType(), template_instantiation,
1518 argument_index);
1519
1520 argument.add_template_param(return_arg);
1521
1522 // Set function template argument types
1523 if (function_type->isVariadic() && function_type->param_types().empty()) {
1524 auto fallback_arg = template_parameter::make_argument({});
1525 fallback_arg.is_ellipsis(true);
1526 argument.add_template_param(std::move(fallback_arg));
1527 }
1528 else {
1529 for (const auto &param_type : function_type->param_types()) {
1530 argument.add_template_param(
1531 process_type_argument(location_decl, parent, cls, template_decl,
1532 param_type, template_instantiation, argument_index));
1533 }
1534 }
1535
1536 return argument;
1537}

◆ try_as_lambda()

template<typename VisitorT >
std::optional< template_parameter > clanguml::common::visitor::template_builder< VisitorT >::try_as_lambda ( const clang::NamedDecl *  cls,
const clang::TemplateDecl *  template_decl,
clang::QualType &  type 
)

Try to process template type argument as lambda.

Parameters
clsTemplate class specialization declaration
template_declBase template declaration
typeTemplate type
Returns
Lambda template argument if succeeds, or std::nullopt

Definition at line 1744 of file template_builder.h.

1747{
1748 auto type_name = common::to_string(type, &cls->getASTContext());
1749
1750 if (type_name.find("(lambda at ") != 0)
1751 return {};
1752
1753 LOG_DBG("Template argument is a lambda");
1754
1756 type = consume_context(type, argument);
1757
1759 argument.set_type(type_name);
1760
1761 return argument;
1762}

◆ try_as_member_pointer()

template<typename VisitorT >
std::optional< template_parameter > clanguml::common::visitor::template_builder< VisitorT >::try_as_member_pointer ( const clang::NamedDecl &  location_decl,
std::optional< clanguml::common::model::template_element * > &  parent,
const clang::NamedDecl *  cls,
const clang::TemplateDecl *  template_decl,
clang::QualType &  type,
clanguml::common::model::template_element template_instantiation,
size_t  argument_index 
)

Try to process template type argument as member pointer type.

Parameters
parentOptional class in which this template is contained
clsTemplate class specialization declaration
template_declBase template declaration
typeTemplate type
template_instantiationTemplate class model
argument_indexArgument index
Returns
Member pointer type template argument if succeeds, or std::nullopt

Definition at line 1361 of file template_builder.h.

1368{
1369 const auto *mp_type =
1370 common::dereference(type)->getAs<clang::MemberPointerType>();
1371 if (mp_type == nullptr)
1372 return {};
1373
1375 type = consume_context(type, argument);
1376
1377 // Handle a pointer to a data member of a class
1378 if (mp_type->isMemberDataPointer()) {
1379 argument.is_member_pointer(false);
1380 argument.is_data_pointer(true);
1381
1382 auto pointee_arg = process_type_argument(location_decl, parent, cls,
1383 template_decl, mp_type->getPointeeType(), template_instantiation,
1384 argument_index);
1385
1386 argument.add_template_param(std::move(pointee_arg));
1387
1388 const auto *member_class_type = mp_type->getClass();
1389
1390 if (member_class_type == nullptr)
1391 return {};
1392
1393 auto class_type_arg = process_type_argument(location_decl, parent, cls,
1394 template_decl, mp_type->getClass()->getCanonicalTypeUnqualified(),
1395 template_instantiation, argument_index);
1396
1397 argument.add_template_param(std::move(class_type_arg));
1398 }
1399 // Handle pointer to class method member
1400 else {
1401 argument.is_member_pointer(true);
1402 argument.is_data_pointer(false);
1403
1404 const auto *function_type =
1405 mp_type->getPointeeType()->getAs<clang::FunctionProtoType>();
1406
1407 assert(function_type != nullptr);
1408
1409 auto return_type_arg = process_type_argument(location_decl, parent, cls,
1410 template_decl, function_type->getReturnType(),
1411 template_instantiation, argument_index);
1412
1413 // Add return type argument
1414 argument.add_template_param(std::move(return_type_arg));
1415
1416 const auto *member_class_type = mp_type->getClass();
1417
1418 if (member_class_type == nullptr)
1419 return {};
1420
1421 auto class_type_arg = process_type_argument(location_decl, parent, cls,
1422 template_decl, mp_type->getClass()->getCanonicalTypeUnqualified(),
1423 template_instantiation, argument_index);
1424
1425 // Add class type argument
1426 argument.add_template_param(std::move(class_type_arg));
1427
1428 // Add argument types
1429 for (const auto &param_type : function_type->param_types()) {
1430 argument.add_template_param(
1431 process_type_argument(location_decl, parent, cls, template_decl,
1432 param_type, template_instantiation, argument_index));
1433 }
1434 }
1435
1436 return argument;
1437}

◆ try_as_record_type()

template<typename VisitorT >
std::optional< template_parameter > clanguml::common::visitor::template_builder< VisitorT >::try_as_record_type ( const clang::NamedDecl &  location_decl,
std::optional< clanguml::common::model::template_element * > &  parent,
const clang::NamedDecl *  cls,
const clang::TemplateDecl *  template_decl,
clang::QualType &  type,
clanguml::common::model::template_element template_instantiation,
size_t  argument_index 
)

Try to process template type argument as record type.

Parameters
parentOptional class in which this template is contained
clsTemplate class specialization declaration
template_declBase template declaration
typeTemplate type
template_instantiationTemplate class model
argument_indexArgument index
Returns
Record type template argument if succeeds, or std::nullopt

Definition at line 1766 of file template_builder.h.

1773{
1774 const auto *record_type =
1775 common::dereference(type)->getAs<clang::RecordType>();
1776 if (record_type == nullptr)
1777 return {};
1778
1779 LOG_DBG("Template argument is a c++ record");
1780
1782 type = consume_context(type, argument);
1783
1784 const auto type_name = config().simplify_template_type(
1785 common::to_string(type, template_decl->getASTContext()));
1786
1787 argument.set_type(type_name);
1788 const auto type_id = common::to_id(type_name);
1789
1790 argument.set_id(type_id);
1791
1792 const auto *class_template_specialization =
1793 clang::dyn_cast<clang::ClassTemplateSpecializationDecl>(
1794 record_type->getAsRecordDecl());
1795
1796 if (class_template_specialization != nullptr) {
1797 auto tag_argument =
1798 visitor_.create_element(class_template_specialization);
1799
1801 *tag_argument, *class_template_specialization);
1802
1803 tag_argument->is_template(true);
1804
1805 if (tag_argument) {
1806 argument.set_type(tag_argument->name_and_ns());
1807 for (const auto &p : tag_argument->template_params())
1808 argument.add_template_param(p);
1809 for (auto &r : tag_argument->relationships()) {
1810 template_instantiation.add_relationship(std::move(r));
1811 }
1812
1814 diagram().should_include(tag_argument->get_namespace())) {
1815 if (parent.has_value())
1816 parent.value()->add_relationship(
1817 {relationship_t::kDependency, tag_argument->id()});
1818
1819 visitor_.set_source_location(location_decl, *tag_argument);
1820 visitor_.add_diagram_element(std::move(tag_argument));
1821 }
1822 }
1823 }
1824 else if (const auto *record_type_decl = record_type->getAsRecordDecl();
1825 record_type_decl != nullptr) {
1827 diagram().should_include(namespace_{type_name})) {
1828 // Add dependency relationship to the parent
1829 // template
1830 template_instantiation.add_relationship(
1831 {relationship_t::kDependency, type_id});
1832 }
1833 }
1834
1835 return argument;
1836}

◆ try_as_template_parm_type()

template<typename VisitorT >
std::optional< template_parameter > clanguml::common::visitor::template_builder< VisitorT >::try_as_template_parm_type ( const clang::NamedDecl *  cls,
const clang::TemplateDecl *  template_decl,
clang::QualType &  type 
)

Try to process template type argument as template parameter.

Parameters
clsTemplate class specialization declaration
template_declBase template declaration
typeTemplate type
Returns
Template parameter if succeeds, or std::nullopt

Definition at line 1700 of file template_builder.h.

1703{
1704 auto is_variadic{false};
1705
1706 const auto *type_parameter =
1707 common::dereference(type)->getAs<clang::TemplateTypeParmType>();
1708
1709 auto type_name = common::to_string(type, &cls->getASTContext());
1710
1711 if (type_parameter == nullptr) {
1712 if (const auto *pet =
1713 common::dereference(type)->getAs<clang::PackExpansionType>();
1714 pet != nullptr) {
1715 is_variadic = true;
1716 type_parameter =
1717 pet->getPattern()->getAs<clang::TemplateTypeParmType>();
1718 }
1719 }
1720
1721 if (type_parameter == nullptr)
1722 return {};
1723
1724 LOG_DBG("Template argument is a template parameter type");
1725
1727 type = consume_context(type, argument);
1728
1729 auto type_parameter_name = common::to_string(type, cls->getASTContext());
1730 if (type_parameter_name.empty())
1731 type_parameter_name = "typename";
1732
1734 cls, type_parameter_name));
1735
1736 argument.is_variadic(is_variadic);
1737
1738 common::ensure_lambda_type_is_relative(config(), type_parameter_name);
1739
1740 return argument;
1741}

◆ try_as_template_specialization_type()

template<typename VisitorT >
std::optional< template_parameter > clanguml::common::visitor::template_builder< VisitorT >::try_as_template_specialization_type ( const clang::NamedDecl &  location_decl,
std::optional< clanguml::common::model::template_element * > &  parent,
const clang::NamedDecl *  cls,
const clang::TemplateDecl *  template_decl,
clang::QualType &  type,
clanguml::common::model::template_element template_instantiation,
size_t  argument_index 
)

Try to process template type argument as specialization type.

Parameters
parentOptional class in which this template is contained
clsTemplate class specialization declaration
template_declBase template declaration
typeTemplate type
template_instantiationTemplate class model
argument_indexArgument index
Returns
Template specialization template argument if succeeds, or std::nullopt

Definition at line 1597 of file template_builder.h.

1604{
1605 const auto *nested_template_type =
1606 common::dereference(type)->getAs<clang::TemplateSpecializationType>();
1607 if (nested_template_type == nullptr) {
1608 return {};
1609 }
1610
1611 LOG_DBG("Template argument is a template specialization type");
1612
1614 type = consume_context(type, argument);
1615
1616 auto nested_type_name = nested_template_type->getTemplateName()
1617 .getAsTemplateDecl()
1618 ->getQualifiedNameAsString();
1619
1620 if (clang::dyn_cast<clang::TemplateTemplateParmDecl>(
1621 nested_template_type->getTemplateName().getAsTemplateDecl()) !=
1622 nullptr) {
1623 if (const auto *template_specialization_decl =
1624 clang::dyn_cast<clang::ClassTemplateSpecializationDecl>(cls);
1625 template_specialization_decl != nullptr) {
1626 nested_type_name =
1627 template_specialization_decl->getDescribedTemplateParams()
1628 ->getParam(argument_index)
1629 ->getNameAsString();
1630 }
1631 else {
1632 // fallback
1633 nested_type_name = "template";
1634 }
1635
1636 argument.is_template_template_parameter(true);
1637 }
1638
1639 argument.set_type(nested_type_name);
1640
1641 auto nested_template_instantiation =
1642 visitor_.create_element(nested_template_type->getTemplateName()
1643 .getAsTemplateDecl()
1644 ->getTemplatedDecl());
1646 *nested_template_instantiation, cls, *nested_template_type,
1647 diagram().should_include(
1648 namespace_{template_decl->getQualifiedNameAsString()})
1649 ? std::make_optional(&template_instantiation)
1650 : parent);
1651
1652 argument.set_id(nested_template_instantiation->id());
1653
1654 for (const auto &t : nested_template_instantiation->template_params())
1655 argument.add_template_param(t);
1656
1657 // Check if this template should be simplified (e.g. system
1658 // template aliases such as 'std:basic_string<char>' should
1659 // be simply 'std::string')
1661 argument, argument.to_string(using_namespace(), false));
1662
1663 argument.set_id(
1664 common::to_id(argument.to_string(using_namespace(), false)));
1665
1666 const auto nested_template_instantiation_full_name =
1667 nested_template_instantiation->full_name(false);
1668
1669 if (nested_template_instantiation &&
1670 diagram().should_include(
1671 namespace_{nested_template_instantiation_full_name})) {
1673 if (diagram().should_include(
1674 namespace_{template_decl->getQualifiedNameAsString()})) {
1675 template_instantiation.add_relationship(
1676 {relationship_t::kDependency,
1677 nested_template_instantiation->id()});
1678 }
1679 else {
1680 if (parent.has_value())
1681 parent.value()->add_relationship(
1682 {relationship_t::kDependency,
1683 nested_template_instantiation->id()});
1684 }
1685 }
1686 }
1687
1688 if (diagram().should_include(
1689 namespace_{nested_template_instantiation_full_name})) {
1690 visitor_.set_source_location(
1691 location_decl, *nested_template_instantiation);
1692 visitor_.add_diagram_element(std::move(nested_template_instantiation));
1693 }
1694
1695 return argument;
1696}

◆ try_as_typedef_type()

template<typename VisitorT >
std::optional< template_parameter > clanguml::common::visitor::template_builder< VisitorT >::try_as_typedef_type ( std::optional< clanguml::common::model::template_element * > &  parent,
const clang::NamedDecl *  cls,
const clang::TemplateDecl *  template_decl,
clang::QualType &  type,
clanguml::common::model::template_element template_instantiation,
size_t  argument_index 
)

Try to process template type argument as typedef/using type.

Parameters
parentOptional class in which this template is contained
clsTemplate class specialization declaration
template_declBase template declaration
typeTemplate type
template_instantiationTemplate class model
argument_indexArgument index
Returns
Typedef type template argument if succeeds, or std::nullopt

Definition at line 1561 of file template_builder.h.

1567{
1568 const auto *typedef_type =
1569 common::dereference(type)->getAs<clang::TypedefType>();
1570 if (typedef_type == nullptr) {
1571 return {};
1572 }
1573
1574 LOG_DBG("Template argument is a typedef/using");
1575
1576 // If this is a typedef/using alias to a decltype - we're not able
1577 // to figure out anything out of it probably
1578 if (typedef_type->getAs<clang::DecltypeType>() != nullptr) {
1579 // Here we need to figure out the parent context of this alias,
1580 // it can be a:
1581 // - class/struct
1582 if (typedef_type->getDecl()->isCXXClassMember() && parent) {
1584 fmt::format("{}::{}", parent.value()->full_name(false),
1585 typedef_type->getDecl()->getNameAsString()));
1586 }
1587 // - namespace
1589 typedef_type->getDecl()->getQualifiedNameAsString());
1590 }
1591
1592 return {};
1593}

◆ using_namespace()

template<typename VisitorT >
const namespace_ & clanguml::common::visitor::template_builder< VisitorT >::using_namespace

Get diagram relative namespace.

Returns
Diagram relative namespace

Definition at line 590 of file template_builder.h.

591{
592 return config_.using_namespace();
593}

Member Data Documentation

◆ config_

template<typename VisitorT >
const clanguml::config::diagram& clanguml::common::visitor::template_builder< VisitorT >::config_
private

Definition at line 556 of file template_builder.h.

◆ diagram_

template<typename VisitorT >
clanguml::common::model::diagram& clanguml::common::visitor::template_builder< VisitorT >::diagram_
private

Definition at line 553 of file template_builder.h.

◆ id_mapper_

template<typename VisitorT >
common::visitor::ast_id_mapper& clanguml::common::visitor::template_builder< VisitorT >::id_mapper_
private

Definition at line 558 of file template_builder.h.

◆ source_manager_

template<typename VisitorT >
clang::SourceManager& clanguml::common::visitor::template_builder< VisitorT >::source_manager_
private

Definition at line 560 of file template_builder.h.

◆ visitor_

template<typename VisitorT >
VisitorT& clanguml::common::visitor::template_builder< VisitorT >::visitor_
private

Definition at line 562 of file template_builder.h.


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