31using common::model::template_parameter;
36 const clang::ClassTemplateSpecializationDecl *decl,
const std::string &tp);
39 const clang::TypeAliasTemplateDecl *decl,
const std::string &tp);
44 const clang::Decl *decl,
const std::string &type_parameter);
99 const clang::TemplateDecl &template_declaration,
111 const clang::NamedDecl &location_declaration,
113 const clang::NamedDecl *cls,
114 const clang::TemplateSpecializationType &template_type_decl,
115 std::optional<clanguml::common::model::template_element *> parent = {});
117 void build(
const clang::NamedDecl &location_declaration,
119 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
120 clang::ArrayRef<clang::TemplateArgument> template_arguments,
121 std::string full_template_specialization_name,
122 std::optional<clanguml::common::model::template_element *> parent = {});
133 const clang::ClassTemplateSpecializationDecl &template_specialization,
134 std::optional<clanguml::common::model::template_element *> parent = {});
151 std::deque<std::tuple<std::string, int, bool>> &template_base_params,
152 int arg_index,
bool variadic_params,
166 std::optional<clanguml::common::model::template_element *> &parent,
167 const clang::NamedDecl *cls,
168 std::deque<std::tuple<std::string, int, bool>> &template_base_params,
169 const clang::ArrayRef<clang::TemplateArgument> &template_args,
171 const clang::TemplateDecl *template_decl);
186 std::optional<clanguml::common::model::template_element *> &parent,
187 const clang::NamedDecl *cls,
189 const clang::TemplateDecl *template_decl,
190 const clang::TemplateArgument &arg,
size_t argument_index,
191 std::vector<template_parameter> &argument);
203 const clang::TemplateArgument &arg);
216 const clang::TemplateArgument &arg,
217 const clang::ASTContext &ast_context);
219#if LLVM_VERSION_MAJOR > 17
231 const clang::TemplateArgument &arg,
232 const clang::ASTContext &ast_context);
245 const clang::TemplateArgument &arg);
257 const clang::TemplateArgument &arg);
269 const clang::NamedDecl &location_decl,
270 std::optional<clanguml::common::model::template_element *> &parent,
271 const clang::NamedDecl *cls,
273 const clang::TemplateDecl *base_template_decl,
274 const clang::TemplateArgument &arg,
size_t argument_index,
275 std::vector<template_parameter> &argument);
286 const clang::NamedDecl &location_decl,
287 std::optional<clanguml::common::model::template_element *> &parent,
288 const clang::NamedDecl *cls,
289 const clang::TemplateDecl *base_template_decl, clang::QualType type,
291 size_t argument_index);
303 const clang::TemplateArgument &arg);
315 const clang::TemplateArgument &arg);
329 const clang::NamedDecl &location_decl,
330 std::optional<clanguml::common::model::template_element *> &parent,
331 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
332 clang::QualType &type,
334 size_t argument_index);
348 const clang::NamedDecl &location_decl,
349 std::optional<clanguml::common::model::template_element *> &parent,
350 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
351 clang::QualType &type,
353 size_t argument_index);
368 const clang::NamedDecl &location_decl,
369 std::optional<clanguml::common::model::template_element *> &parent,
370 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
371 clang::QualType &type,
373 size_t argument_index);
384 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
385 clang::QualType &type);
395 std::optional<template_parameter>
try_as_lambda(
const clang::NamedDecl *cls,
396 const clang::TemplateDecl *template_decl, clang::QualType &type);
410 const clang::NamedDecl &location_decl,
411 std::optional<clanguml::common::model::template_element *> &parent,
412 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
413 clang::QualType &type,
415 size_t argument_index);
428 std::optional<clanguml::common::model::template_element *> &parent,
429 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
430 clang::QualType &type,
442 std::optional<clanguml::common::model::template_element *> &parent,
443 clang::QualType &type,
const clang::TemplateDecl *template_decl);
458 const clang::NamedDecl &location_decl,
459 std::optional<clanguml::common::model::template_element *> &parent,
460 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
461 clang::QualType &type,
463 size_t argument_index);
477 std::optional<clanguml::common::model::template_element *> &parent,
478 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
479 clang::QualType &type,
481 size_t argument_index);
495 std::optional<clanguml::common::model::template_element *> &parent,
496 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
497 clang::QualType &type,
499 size_t argument_index);
535 const std::string &qualified_name)
const;
565template <
typename VisitorT>
571 , id_mapper_{visitor.id_mapper()}
572 , source_manager_{visitor.source_manager()}
577template <
typename VisitorT>
583template <
typename VisitorT>
589template <
typename VisitorT>
592 return config_.using_namespace();
595template <
typename VisitorT>
601template <
typename VisitorT>
604 return source_manager_;
607template <
typename VisitorT>
614 auto simplified = config().simplify_template_type(full_name);
616 if (simplified != full_name) {
626template <
typename VisitorT>
629 const clang::TemplateDecl &template_declaration,
632 LOG_DBG(
"Processing {} template parameters...",
635 if (template_declaration.getTemplateParameters() ==
nullptr)
638 for (
const auto *parameter :
639 *template_declaration.getTemplateParameters()) {
640 if (clang::dyn_cast_or_null<clang::TemplateTypeParmDecl>(parameter) !=
642 const auto *template_type_parameter =
643 clang::dyn_cast_or_null<clang::TemplateTypeParmDecl>(parameter);
645 std::optional<std::string> default_arg;
646 if (template_type_parameter->hasDefaultArgument()) {
648#if LLVM_VERSION_MAJOR > 18
650 template_type_parameter->getDefaultArgument(),
651 template_declaration.getASTContext());
653 template_type_parameter->getDefaultArgument().getAsString();
657 auto parameter_name = template_type_parameter->getNameAsString();
658 if (parameter_name.empty())
659 parameter_name =
"typename";
662 default_arg, template_type_parameter->isParameterPack());
664 if constexpr (std::is_same_v<
typename VisitorT::diagram_t,
666 if (template_type_parameter->getTypeConstraint() !=
nullptr) {
668 template_type_parameter->getTypeConstraint()
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,
680 eid_t{named_concept->getID()})
682 model::access_t::kNone,
689 (void)templated_element;
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>(
700 std::optional<std::string> default_arg;
702 if (template_nontype_parameter->hasDefaultArgument()) {
704#if LLVM_VERSION_MAJOR > 18
706 template_nontype_parameter->getDefaultArgument(),
707 template_declaration.getASTContext());
710 template_nontype_parameter->getDefaultArgument());
714 template_nontype_parameter->getType().getAsString(),
715 template_nontype_parameter->getNameAsString(), default_arg,
716 template_nontype_parameter->isParameterPack());
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>(
725 std::optional<std::string> default_arg;
726 if (template_template_parameter->hasDefaultArgument()) {
728 template_template_parameter->getDefaultArgument()
732 template_template_parameter->getNameAsString(), default_arg,
733 template_template_parameter->isParameterPack());
743template <
typename VisitorT>
745 const clang::NamedDecl &location_declaration,
747 const clang::NamedDecl *cls,
748 const clang::TemplateSpecializationType &template_type_decl,
749 std::optional<clanguml::common::model::template_element *> parent)
751 const auto *template_type_ptr = &template_type_decl;
753 if (template_type_decl.isTypeAlias()) {
754 if (
const auto *tsp =
755 template_type_decl.getAliasedType()
756 ->template getAs<clang::TemplateSpecializationType>();
758 template_type_ptr = tsp;
761 const auto &template_type = *template_type_ptr;
766 template_type.desugar(),
767 template_type.getTemplateName().getAsTemplateDecl()->getASTContext());
769 auto *template_decl{template_type.getTemplateName().getAsTemplateDecl()};
771 build(location_declaration, template_instantiation, cls, template_decl,
772 template_type.template_arguments(), full_template_specialization_name,
776template <
typename VisitorT>
779 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
780 const clang::ArrayRef<clang::TemplateArgument> template_arguments,
781 std::string full_template_specialization_name,
782 std::optional<clanguml::common::model::template_element *> parent)
788 std::deque<std::tuple< std::string, int,
790 template_base_params{};
792 auto template_decl_qualified_name =
793 template_decl->getQualifiedNameAsString();
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()) {
805 ->getOuterLexicalRecordContext())};
808 std::string name = template_decl->getQualifiedNameAsString();
809 if (!ns_str.empty()) {
810 name = name.substr(ns_str.size() + 2);
814 template_instantiation.
set_name(name);
819 namespace_ ns{template_decl_qualified_name};
821 template_instantiation.
set_name(template_decl->getNameAsString());
830 std::pair< std::string,
bool>>
831 template_parameter_names{};
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);
841 template_parameter_names.emplace_back(
842 parameter->getNameAsString(),
false);
848 const auto *templated_class_decl =
849 clang::dyn_cast_or_null<clang::CXXRecordDecl>(
850 template_decl->getTemplatedDecl());
852 if ((templated_class_decl !=
nullptr) &&
853 templated_class_decl->hasDefinition())
854 for (
const auto &base : templated_class_decl->bases()) {
856 base.getType(), templated_class_decl->getASTContext(),
false);
858 LOG_DBG(
"Found template instantiation base: {}, {}",
859 base_class_name, base_index);
863 auto it = std::find_if(template_parameter_names.begin(),
864 template_parameter_names.end(),
866 const auto &p) { return p.first == base_class_name; });
868 if (it != template_parameter_names.end()) {
869 const auto ¶meter_name = it->first;
870 const bool is_variadic = it->second;
872 LOG_DBG(
"Found base class which is a template parameter "
874 parameter_name, is_variadic,
875 std::distance(template_parameter_names.begin(), it));
877 template_base_params.emplace_back(parameter_name,
878 std::distance(template_parameter_names.begin(), it),
888 process_template_arguments(location_decl, parent, cls, template_base_params,
889 template_arguments, template_instantiation, template_decl);
891 if constexpr (std::is_same_v<
typename VisitorT::diagram_t,
893 find_instantiation_relationships(template_instantiation,
894 eid_t{template_decl->getID()}, full_template_specialization_name);
897 template_instantiation.
set_id(
900 visitor_.set_source_location(location_decl, template_instantiation);
903template <
typename VisitorT>
906 const clang::ClassTemplateSpecializationDecl &template_specialization,
907 std::optional<clanguml::common::model::template_element *> parent)
909 LOG_DBG(
"Building class template specialization model: {}",
910 template_specialization.getQualifiedNameAsString());
916 std::deque<std::tuple< std::string, int,
918 template_base_params{};
920 const clang::ClassTemplateDecl *template_decl =
921 template_specialization.getSpecializedTemplate();
923 auto qualified_name = template_decl->getQualifiedNameAsString();
925 namespace_ ns{qualified_name};
927 template_instantiation.
set_name(template_decl->getNameAsString());
930 process_template_arguments(template_specialization, parent,
931 &template_specialization, template_base_params,
932 template_specialization.getTemplateArgs().asArray(),
933 template_instantiation, template_decl);
936 template_instantiation.
set_id(
939 if constexpr (std::is_same_v<
typename VisitorT::diagram_t,
941 find_instantiation_relationships(template_instantiation,
942 eid_t{template_specialization.getID()}, qualified_name);
945 visitor_.set_source_location(
946 template_specialization, template_instantiation);
949template <
typename VisitorT>
952 const std::string &qualified_name)
const
955 template_instantiation, qualified_name,
id);
958template <
typename VisitorT>
960 const clang::NamedDecl &location_decl,
961 std::optional<clanguml::common::model::template_element *> &parent,
962 const clang::NamedDecl *cls,
963 std::deque<std::tuple<std::string, int, bool>> &template_base_params,
964 const clang::ArrayRef<clang::TemplateArgument> &template_args,
966 const clang::TemplateDecl *template_decl)
970 for (
const auto &arg : template_args) {
973 std::vector<template_parameter> arguments;
976 if (arg.getKind() == clang::TemplateArgument::ArgKind::Pack &&
977 arg.getPackAsArray().empty()) {
979 template_parameter::make_empty());
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()
998 if (maybe_type_parm_decl !=
nullptr &&
999 maybe_type_parm_decl->hasDefaultArgument()) {
1007 argument_process_dispatch(location_decl, parent, cls,
1008 template_instantiation, template_decl, arg, arg_index, arguments);
1010 if (arguments.empty()) {
1019 [[maybe_unused]]
auto variadic_params{
false};
1021 if constexpr (std::is_same_v<
typename VisitorT::diagram_t,
1025 if (!template_base_params.empty()) {
1026 variadic_params = add_base_classes(template_instantiation,
1027 template_base_params, arg_index, variadic_params,
1032 for (
auto &argument : arguments) {
1033 simplify_system_template(
1034 argument, argument.to_string(using_namespace(),
false,
true));
1036 LOG_DBG(
"Adding template argument {} to template "
1037 "specialization/instantiation {}",
1038 argument.to_string(using_namespace(),
false),
1039 template_instantiation.
name());
1041 template_instantiation.
add_template(std::move(argument));
1048 template_instantiation.
set_id(
1052template <
typename VisitorT>
1054 const clang::NamedDecl &location_decl,
1055 std::optional<clanguml::common::model::template_element *> &parent,
1056 const clang::NamedDecl *cls,
1058 const clang::TemplateDecl *template_decl,
1059 const clang::TemplateArgument &arg,
size_t argument_index,
1060 std::vector<template_parameter> &argument)
1062 LOG_DBG(
"Processing argument {} in template class: {}", argument_index,
1063 cls->getQualifiedNameAsString());
1065 switch (arg.getKind()) {
1066 case clang::TemplateArgument::Null:
1067 argument.push_back(process_null_argument(arg));
1069 case clang::TemplateArgument::Template:
1070 argument.push_back(process_template_argument(arg));
1072 case clang::TemplateArgument::Type: {
1074 process_type_argument(location_decl, parent, cls, template_decl,
1075 arg.getAsType(), template_instantiation, argument_index));
1078 case clang::TemplateArgument::Declaration:
1080 case clang::TemplateArgument::NullPtr:
1081 argument.push_back(process_nullptr_argument(arg));
1083 case clang::TemplateArgument::Integral:
1085 process_integral_argument(arg, template_decl->getASTContext()));
1087 case clang::TemplateArgument::TemplateExpansion:
1088 argument.push_back(process_template_expansion(arg));
1090 case clang::TemplateArgument::Expression:
1091 argument.push_back(process_expression_argument(arg));
1093 case clang::TemplateArgument::Pack:
1094 for (
auto &a : process_pack_argument(location_decl, parent, cls,
1095 template_instantiation, template_decl, arg, argument_index,
1097 argument.push_back(a);
1100#if LLVM_VERSION_MAJOR > 17
1101 case clang::TemplateArgument::StructuralValue:
1107template <
typename VisitorT>
1109 const clang::TemplateArgument &arg)
1113 LOG_DBG(
"Processing template argument: {}", arg_name);
1115 return template_parameter::make_template_type(arg_name);
1118template <
typename VisitorT>
1120 const clang::TemplateArgument &arg)
1124 LOG_DBG(
"Processing template expansion argument: {}", arg_name);
1127 arg.getAsTemplate().getAsTemplateDecl(), [&arg_name](
const auto *decl) {
1128 arg_name = decl->getQualifiedNameAsString();
1131 auto param = template_parameter::make_template_type(arg_name);
1132 param.is_variadic(
true);
1137template <
typename VisitorT>
1145 return unqualified_type;
1148template <
typename VisitorT>
1150 const clang::NamedDecl &location_decl,
1151 std::optional<clanguml::common::model::template_element *> &parent,
1152 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
1153 clang::QualType type,
1155 size_t argument_index)
1157 std::optional<template_parameter> argument;
1159 if (type->getAs<clang::ElaboratedType>() !=
nullptr) {
1160 type = type->getAs<clang::ElaboratedType>()->getNamedType();
1165 LOG_DBG(
"Processing template {} type argument {}: {}, {}, {}",
1166 template_decl->getQualifiedNameAsString(), argument_index, type_name,
1167 type->getTypeClassName(),
1170 argument = try_as_function_prototype(location_decl, parent, cls,
1171 template_decl, type, template_instantiation, argument_index);
1175 argument = try_as_member_pointer(location_decl, parent, cls, template_decl,
1176 type, template_instantiation, argument_index);
1180 argument = try_as_array(location_decl, parent, cls, template_decl, type,
1181 template_instantiation, argument_index);
1185 argument = try_as_template_parm_type(cls, template_decl, type);
1189 argument = try_as_template_specialization_type(location_decl, parent, cls,
1190 template_decl, type, template_instantiation, argument_index);
1194 argument = try_as_decl_type(parent, cls, template_decl, type,
1195 template_instantiation, argument_index);
1199 argument = try_as_typedef_type(parent, cls, template_decl, type,
1200 template_instantiation, argument_index);
1204 argument = try_as_lambda(cls, template_decl, type);
1208 argument = try_as_record_type(location_decl, parent, cls, template_decl,
1209 type, template_instantiation, argument_index);
1213 argument = try_as_enum_type(
1214 parent, cls, template_decl, type, template_instantiation);
1218 argument = try_as_builtin_type(parent, type, template_decl);
1223 return template_parameter::make_argument(type_name);
1226template <
typename VisitorT>
1231 const auto &type = ct.
type();
1237 LOG_DBG(
"Finding relationships in user defined type: {}",
1238 ct.
to_string(config().using_namespace(),
false));
1240 auto type_with_namespace =
1241 std::make_optional<common::model::namespace_>(type.value());
1243 if (!type_with_namespace.has_value()) {
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);
1256 found = find_relationships_in_unexposed_template_params(
1257 nested_template_params, relationships) ||
1264template <
typename VisitorT>
1266 const clang::TemplateArgument &arg,
const clang::ASTContext &ast_context)
1268 assert(arg.getKind() == clang::TemplateArgument::Integral);
1271 llvm::raw_string_ostream ostream(result);
1272 clang::PrintingPolicy policy(ast_context.getLangOpts());
1274#if LLVM_VERSION_MAJOR > 18
1275 arg.print(policy, ostream,
false);
1280 return template_parameter::make_argument(result);
1283#if LLVM_VERSION_MAJOR > 17
1284template <
typename VisitorT>
1286 const clang::TemplateArgument &arg,
const clang::ASTContext &ast_context)
1288 assert(arg.getKind() == clang::TemplateArgument::StructuralValue);
1291 llvm::raw_string_ostream ostream(result);
1292 clang::PrintingPolicy policy(ast_context.getLangOpts());
1294#if LLVM_VERSION_MAJOR > 18
1295 arg.print(policy, ostream,
false);
1300 return template_parameter::make_argument(result);
1304template <
typename VisitorT>
1306 const clang::TemplateArgument &arg)
1308 assert(arg.getKind() == clang::TemplateArgument::Null);
1310 return template_parameter::make_argument(
"");
1313template <
typename VisitorT>
1315 const clang::TemplateArgument &arg)
1317 assert(arg.getKind() == clang::TemplateArgument::NullPtr);
1321 return template_parameter::make_argument(
"nullptr");
1324template <
typename VisitorT>
1326 const clang::TemplateArgument &arg)
1328 assert(arg.getKind() == clang::TemplateArgument::Expression);
1330 arg.getAsExpr()->getSourceRange(), source_manager()));
1333template <
typename VisitorT>
1334std::vector<template_parameter>
1336 const clang::NamedDecl &location_decl,
1337 std::optional<clanguml::common::model::template_element *> &parent,
1338 const clang::NamedDecl *cls,
1340 const clang::TemplateDecl *base_template_decl,
1341 const clang::TemplateArgument &arg,
size_t argument_index,
1342 std::vector<template_parameter> & )
1344 assert(arg.getKind() == clang::TemplateArgument::Pack);
1346 std::vector<template_parameter> res;
1348 auto pack_argument_index = argument_index;
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);
1359template <
typename VisitorT>
1360std::optional<template_parameter>
1362 const clang::NamedDecl &location_decl,
1363 std::optional<clanguml::common::model::template_element *> &parent,
1364 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
1365 clang::QualType &type,
1367 size_t argument_index)
1369 const auto *mp_type =
1371 if (mp_type ==
nullptr)
1374 auto argument = template_parameter::make_template_type(
"");
1375 type = consume_context(type, argument);
1378 if (mp_type->isMemberDataPointer()) {
1379 argument.is_member_pointer(
false);
1380 argument.is_data_pointer(
true);
1382 auto pointee_arg = process_type_argument(location_decl, parent, cls,
1383 template_decl, mp_type->getPointeeType(), template_instantiation,
1386 argument.add_template_param(std::move(pointee_arg));
1388 const auto *member_class_type = mp_type->getClass();
1390 if (member_class_type ==
nullptr)
1393 auto class_type_arg = process_type_argument(location_decl, parent, cls,
1394 template_decl, mp_type->getClass()->getCanonicalTypeUnqualified(),
1395 template_instantiation, argument_index);
1397 argument.add_template_param(std::move(class_type_arg));
1401 argument.is_member_pointer(
true);
1402 argument.is_data_pointer(
false);
1404 const auto *function_type =
1405 mp_type->getPointeeType()->getAs<clang::FunctionProtoType>();
1407 assert(function_type !=
nullptr);
1409 auto return_type_arg = process_type_argument(location_decl, parent, cls,
1410 template_decl, function_type->getReturnType(),
1411 template_instantiation, argument_index);
1414 argument.add_template_param(std::move(return_type_arg));
1416 const auto *member_class_type = mp_type->getClass();
1418 if (member_class_type ==
nullptr)
1421 auto class_type_arg = process_type_argument(location_decl, parent, cls,
1422 template_decl, mp_type->getClass()->getCanonicalTypeUnqualified(),
1423 template_instantiation, argument_index);
1426 argument.add_template_param(std::move(class_type_arg));
1429 for (
const auto ¶m_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));
1439template <
typename VisitorT>
1441 const clang::NamedDecl &location_decl,
1442 std::optional<clanguml::common::model::template_element *> &parent,
1443 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
1444 clang::QualType &type,
1446 size_t argument_index)
1449 if (array_type ==
nullptr)
1452 auto argument = template_parameter::make_template_type(
"");
1454 type = consume_context(type, argument);
1456 argument.is_array(
true);
1459 auto element_type = process_type_argument(location_decl, parent, cls,
1460 template_decl, array_type->getElementType(), template_instantiation,
1463 argument.add_template_param(element_type);
1465 if (array_type->isDependentSizedArrayType() &&
1466 array_type->getDependence() ==
1467 clang::TypeDependence::DependentInstantiation) {
1468 argument.add_template_param(
1470 ((clang::DependentSizedArrayType *)array_type)
1473 else if (array_type->isConstantArrayType()) {
1474 argument.add_template_param(template_parameter::make_argument(
1475 std::to_string(((clang::ConstantArrayType *)array_type)
1477 .getLimitedValue())));
1485template <
typename VisitorT>
1486std::optional<template_parameter>
1488 const clang::NamedDecl &location_decl,
1489 std::optional<clanguml::common::model::template_element *> &parent,
1490 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
1491 clang::QualType &type,
1493 size_t argument_index)
1495 const auto *function_type = type->getAs<clang::FunctionProtoType>();
1497 if (function_type ==
nullptr && type->isFunctionPointerType()) {
1499 type->getPointeeType()->getAs<clang::FunctionProtoType>();
1500 if (function_type ==
nullptr)
1504 if (function_type ==
nullptr)
1507 LOG_DBG(
"Template argument is a function prototype");
1509 auto argument = template_parameter::make_template_type(
"");
1511 type = consume_context(type, argument);
1513 argument.is_function_template(
true);
1516 auto return_arg = process_type_argument(location_decl, parent, cls,
1517 template_decl, function_type->getReturnType(), template_instantiation,
1520 argument.add_template_param(return_arg);
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));
1529 for (
const auto ¶m_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));
1539template <
typename VisitorT>
1541 std::optional<clanguml::common::model::template_element *> & ,
1542 const clang::NamedDecl * ,
1543 const clang::TemplateDecl * , clang::QualType &type,
1547 const auto *decl_type =
1549 if (decl_type ==
nullptr) {
1553 LOG_DBG(
"Template argument is a decltype()");
1559template <
typename VisitorT>
1560std::optional<template_parameter>
1562 std::optional<clanguml::common::model::template_element *> &parent,
1563 const clang::NamedDecl * ,
1564 const clang::TemplateDecl * , clang::QualType &type,
1568 const auto *typedef_type =
1570 if (typedef_type ==
nullptr) {
1574 LOG_DBG(
"Template argument is a typedef/using");
1578 if (typedef_type->getAs<clang::DecltypeType>() !=
nullptr) {
1582 if (typedef_type->getDecl()->isCXXClassMember() && parent) {
1583 return template_parameter::make_argument(
1584 fmt::format(
"{}::{}", parent.value()->full_name(
false),
1585 typedef_type->getDecl()->getNameAsString()));
1588 return template_parameter::make_argument(
1589 typedef_type->getDecl()->getQualifiedNameAsString());
1595template <
typename VisitorT>
1596std::optional<template_parameter>
1598 const clang::NamedDecl &location_decl,
1599 std::optional<clanguml::common::model::template_element *> &parent,
1600 const clang::NamedDecl *cls,
const clang::TemplateDecl *template_decl,
1601 clang::QualType &type,
1603 size_t argument_index)
1605 const auto *nested_template_type =
1607 if (nested_template_type ==
nullptr) {
1611 LOG_DBG(
"Template argument is a template specialization type");
1613 auto argument = template_parameter::make_argument(
"");
1614 type = consume_context(type, argument);
1616 auto nested_type_name = nested_template_type->getTemplateName()
1617 .getAsTemplateDecl()
1618 ->getQualifiedNameAsString();
1620 if (clang::dyn_cast<clang::TemplateTemplateParmDecl>(
1621 nested_template_type->getTemplateName().getAsTemplateDecl()) !=
1623 if (
const auto *template_specialization_decl =
1624 clang::dyn_cast<clang::ClassTemplateSpecializationDecl>(cls);
1625 template_specialization_decl !=
nullptr) {
1627 template_specialization_decl->getDescribedTemplateParams()
1628 ->getParam(argument_index)
1629 ->getNameAsString();
1633 nested_type_name =
"template";
1636 argument.is_template_template_parameter(
true);
1639 argument.set_type(nested_type_name);
1641 auto nested_template_instantiation =
1642 visitor_.create_element(nested_template_type->getTemplateName()
1643 .getAsTemplateDecl()
1644 ->getTemplatedDecl());
1645 build_from_template_specialization_type(location_decl,
1646 *nested_template_instantiation, cls, *nested_template_type,
1647 diagram().should_include(
1648 namespace_{template_decl->getQualifiedNameAsString()})
1649 ? std::make_optional(&template_instantiation)
1652 argument.set_id(nested_template_instantiation->id());
1654 for (
const auto &t : nested_template_instantiation->template_params())
1655 argument.add_template_param(t);
1660 simplify_system_template(
1661 argument, argument.to_string(using_namespace(),
false));
1664 common::to_id(argument.to_string(using_namespace(),
false)));
1666 const auto nested_template_instantiation_full_name =
1667 nested_template_instantiation->full_name(
false);
1669 if (nested_template_instantiation &&
1670 diagram().should_include(
1671 namespace_{nested_template_instantiation_full_name})) {
1672 if (config_.generate_template_argument_dependencies()) {
1673 if (diagram().should_include(
1674 namespace_{template_decl->getQualifiedNameAsString()})) {
1676 {relationship_t::kDependency,
1677 nested_template_instantiation->id()});
1680 if (parent.has_value())
1681 parent.value()->add_relationship(
1682 {relationship_t::kDependency,
1683 nested_template_instantiation->id()});
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));
1698template <
typename VisitorT>
1699std::optional<template_parameter>
1701 const clang::NamedDecl *cls,
const clang::TemplateDecl * ,
1702 clang::QualType &type)
1704 auto is_variadic{
false};
1706 const auto *type_parameter =
1711 if (type_parameter ==
nullptr) {
1712 if (
const auto *pet =
1717 pet->getPattern()->getAs<clang::TemplateTypeParmType>();
1721 if (type_parameter ==
nullptr)
1724 LOG_DBG(
"Template argument is a template parameter type");
1726 auto argument = template_parameter::make_template_type(
"");
1727 type = consume_context(type, argument);
1730 if (type_parameter_name.empty())
1731 type_parameter_name =
"typename";
1734 cls, type_parameter_name));
1736 argument.is_variadic(is_variadic);
1743template <
typename VisitorT>
1745 const clang::NamedDecl *cls,
const clang::TemplateDecl * ,
1746 clang::QualType &type)
1750 if (type_name.find(
"(lambda at ") != 0)
1753 LOG_DBG(
"Template argument is a lambda");
1755 auto argument = template_parameter::make_argument(
"");
1756 type = consume_context(type, argument);
1759 argument.set_type(type_name);
1764template <
typename VisitorT>
1765std::optional<template_parameter>
1767 const clang::NamedDecl &location_decl,
1768 std::optional<clanguml::common::model::template_element *> &parent,
1769 const clang::NamedDecl * ,
const clang::TemplateDecl *template_decl,
1770 clang::QualType &type,
1774 const auto *record_type =
1776 if (record_type ==
nullptr)
1779 LOG_DBG(
"Template argument is a c++ record");
1781 auto argument = template_parameter::make_argument({});
1782 type = consume_context(type, argument);
1784 const auto type_name = config().simplify_template_type(
1787 argument.set_type(type_name);
1790 argument.set_id(type_id);
1792 const auto *class_template_specialization =
1793 clang::dyn_cast<clang::ClassTemplateSpecializationDecl>(
1794 record_type->getAsRecordDecl());
1796 if (class_template_specialization !=
nullptr) {
1798 visitor_.create_element(class_template_specialization);
1800 build_from_class_template_specialization(
1801 *tag_argument, *class_template_specialization);
1803 tag_argument->is_template(
true);
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()) {
1813 if (config_.generate_template_argument_dependencies() &&
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()});
1819 visitor_.set_source_location(location_decl, *tag_argument);
1820 visitor_.add_diagram_element(std::move(tag_argument));
1824 else if (
const auto *record_type_decl = record_type->getAsRecordDecl();
1825 record_type_decl !=
nullptr) {
1826 if (config_.generate_template_argument_dependencies() &&
1827 diagram().should_include(namespace_{type_name})) {
1831 {relationship_t::kDependency, type_id});
1838template <
typename VisitorT>
1840 std::optional<clanguml::common::model::template_element *> & ,
1841 const clang::NamedDecl * ,
const clang::TemplateDecl *template_decl,
1842 clang::QualType &type,
1845 const auto *enum_type = type->getAs<clang::EnumType>();
1846 if (enum_type ==
nullptr)
1849 LOG_DBG(
"Template argument is a an enum");
1851 auto argument = template_parameter::make_argument({});
1852 type = consume_context(type, argument);
1855 argument.set_type(type_name);
1857 argument.set_id(type_id);
1859 if (enum_type->getAsTagDecl() !=
nullptr &&
1860 config_.generate_template_argument_dependencies()) {
1862 {relationship_t::kDependency, type_id});
1868template <
typename VisitorT>
1869std::optional<template_parameter>
1871 std::optional<clanguml::common::model::template_element *> & ,
1872 clang::QualType &type,
const clang::TemplateDecl *template_decl)
1874 const auto *builtin_type = type->getAs<clang::BuiltinType>();
1875 if (builtin_type ==
nullptr)
1878 LOG_DBG(
"Template argument is a builtin type");
1881 auto argument = template_parameter::make_argument(type_name);
1883 type = consume_context(type, argument);
1884 argument.set_type(type_name);
1889template <
typename VisitorT>
1892 std::deque<std::tuple<std::string, int, bool>> &template_base_params,
1895 bool add_template_argument_as_base_class =
false;
1897 if (variadic_params) {
1898 add_template_argument_as_base_class =
true;
1901 auto [arg_name, index, is_variadic] = template_base_params.front();
1903 variadic_params = is_variadic;
1904 if ((arg_index == index) || (is_variadic && arg_index >= index)) {
1905 add_template_argument_as_base_class =
true;
1908 template_base_params.pop_front();
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 '{}'",
1923 return variadic_params;