0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
template_builder.h
Go to the documentation of this file.
1/**
2 * @file src/class_diagram/visitor/template_builder.h
3 *
4 * Copyright (c) 2021-2025 Bartek Kryza <bkryza@gmail.com>
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 */
18#pragma once
19
25#include "config/config.h"
26
28
31using common::model::template_parameter;
32
33namespace detail {
34
36 const clang::ClassTemplateSpecializationDecl *decl, const std::string &tp);
37
39 const clang::TypeAliasTemplateDecl *decl, const std::string &tp);
40
41} // namespace detail
42
44 const clang::Decl *decl, const std::string &type_parameter);
45
46/**
47 * @brief Class responsible for building all kinds of templates from Clang AST.
48 */
49template <typename VisitorT> class template_builder {
50public:
51 /**
52 * @brief Constructor.
53 *
54 * @param visitor Reference to class diagram translation_unit_visitor
55 */
57 const clanguml::config::diagram &config_, VisitorT &visitor);
58
59 /**
60 * @brief Get reference to the current diagram model
61 *
62 * @return Reference to the current diagram model
63 */
65
66 /**
67 * @brief Get reference to the current diagram configuration
68 *
69 * @return Reference to the current diagram configuration
70 */
71 const config::diagram &config() const;
72
73 /**
74 * @brief Get diagram relative namespace
75 *
76 * @return Diagram relative namespace
77 */
78 const namespace_ &using_namespace() const;
79
80 /**
81 * @brief Simplify system templates
82 *
83 * This method tries to simplify all fully qualified template names
84 * in the `full_name` using substitutions from the configuration file
85 * ().
86 *
87 * Typical example is replace every `std::basic_string<char>` with
88 * `std::string`.
89 *
90 * @param ct Template parameter
91 * @param full_name Full template name
92 * @return
93 */
95 template_parameter &ct, const std::string &full_name) const;
96
99 const clang::TemplateDecl &template_declaration,
100 common::optional_ref<common::model::element> templated_element = {});
101
102 /**
103 * @brief Basic template class build method
104 *
105 * @param cls Clang template declaration
106 * @param template_type_decl Template specialization type
107 * @param parent Optional class in which this template is contained
108 * @return Created template class model
109 */
111 const clang::NamedDecl &location_declaration,
112 clanguml::common::model::template_element &template_instantiation,
113 const clang::NamedDecl *cls,
114 const clang::TemplateSpecializationType &template_type_decl,
115 std::optional<clanguml::common::model::template_element *> parent = {});
116
117 void build(const clang::NamedDecl &location_declaration,
118 clanguml::common::model::template_element &template_instantiation,
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 = {});
123
124 /**
125 * @brief Build template class from class template specialization decl
126 *
127 * @param template_specialization Class template specialization declaration
128 * @param parent Optional class in which this template is contained
129 * @return Created template class model
130 */
132 clanguml::common::model::template_element &template_instantiation,
133 const clang::ClassTemplateSpecializationDecl &template_specialization,
134 std::optional<clanguml::common::model::template_element *> parent = {});
135
136 /**
137 * @brief Add base classes to the template class, if any.
138 *
139 * This method adds base classes to a template declaration or
140 * specialization, including inferring whether variadic template
141 * parameter bases.
142 *
143 * @param tinst Class template model
144 * @param template_base_params List of base class template parameters
145 * @param arg_index Index of the template argument used for base class
146 * @param variadic_params Whether the parameter is variadic
147 * @param ct Template parameter model
148 * @return True, if any base classes were added
149 */
151 std::deque<std::tuple<std::string, int, bool>> &template_base_params,
152 int arg_index, bool variadic_params,
154
155 /**
156 * @brief Process template class parameters and arguments
157 *
158 * @param parent Optional class in which this template is contained
159 * @param cls Template class specialization declaration
160 * @param template_base_params List of base class template parameters
161 * @param template_args List of template arguments
162 * @param template_instantiation Template class model to add template args
163 * @param template_decl Base template declaration
164 */
165 void process_template_arguments(const clang::NamedDecl &location_decl,
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,
170 clanguml::common::model::template_element &template_instantiation,
171 const clang::TemplateDecl *template_decl);
172
173 /**
174 * @brief Process template arguments based on their type
175 *
176 * @param parent Optional class in which this template is contained
177 * @param cls Template class specialization declaration
178 * @param template_instantiation Template class model to add template args
179 * @param template_decl Base template declaration
180 * @param arg Template argument
181 * @param argument_index Argument index
182 * @param argument Output list of arguments (can be more than one for
183 * variadic parameters)
184 */
185 void argument_process_dispatch(const clang::NamedDecl &location_decl,
186 std::optional<clanguml::common::model::template_element *> &parent,
187 const clang::NamedDecl *cls,
188 clanguml::common::model::template_element &template_instantiation,
189 const clang::TemplateDecl *template_decl,
190 const clang::TemplateArgument &arg, size_t argument_index,
191 std::vector<template_parameter> &argument);
192
193 /**
194 * @brief Process `clang::TemplateArgument::Expression`
195 *
196 * @note The template argument is a pack expansion of a template name that
197 * was provided for a template template parameter.
198 *
199 * @param arg Template argument
200 * @return Return template argument model
201 */
203 const clang::TemplateArgument &arg);
204
205 /**
206 * @brief Process `clang::TemplateArgument::Integral`
207 *
208 * @note The template argument is an integral value stored in an
209 * llvm::APSInt that was provided for an integral non-type template
210 * parameter.
211 *
212 * @param arg Template argument
213 * @return Return template argument model
214 */
216 const clang::TemplateArgument &arg,
217 const clang::ASTContext &ast_context);
218
219#if LLVM_VERSION_MAJOR > 17
220 /**
221 * @brief Process `clang::TemplateArgument::StructuralValue`
222 *
223 * @note The template argument is a non-type template argument that can't be
224 * represented by the special-case Declaration, NullPtr, or Integral
225 * forms.
226 *
227 * @param arg Template argument
228 * @return Return template argument model
229 */
230 template_parameter process_structural_argument(
231 const clang::TemplateArgument &arg,
232 const clang::ASTContext &ast_context);
233#endif
234
235 /**
236 * @brief Process `clang::TemplateArgument::NullPtr`
237 *
238 * @note The template argument is a null pointer or null pointer to member
239 * that was provided for a non-type template parameter.
240 *
241 * @param arg Template argument
242 * @return Return template argument model
243 */
245 const clang::TemplateArgument &arg);
246
247 /**
248 * @brief Process `clang::TemplateArgument::Null`
249 *
250 * @note Represents an empty template argument, e.g., one that has not
251 * been deduced.
252 *
253 * @param arg Template argument
254 * @return Return template argument model
255 */
257 const clang::TemplateArgument &arg);
258
259 /**
260 * @brief Process `clang::TemplateArgument::Pack`
261 *
262 * @note The template argument is actually a parameter pack. Arguments are
263 * stored in the Args struct.
264 *
265 * @param arg Template argument
266 * @return Return template argument model
267 */
268 std::vector<template_parameter> process_pack_argument(
269 const clang::NamedDecl &location_decl,
270 std::optional<clanguml::common::model::template_element *> &parent,
271 const clang::NamedDecl *cls,
272 clanguml::common::model::template_element &template_instantiation,
273 const clang::TemplateDecl *base_template_decl,
274 const clang::TemplateArgument &arg, size_t argument_index,
275 std::vector<template_parameter> &argument);
276
277 /**
278 * @brief Process `clang::TemplateArgument::Type`
279 *
280 * @note The template argument is a type.
281 *
282 * @param arg Template argument
283 * @return Return template argument model
284 */
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,
290 clanguml::common::model::template_element &template_instantiation,
291 size_t argument_index);
292
293 /**
294 * @brief Process `clang::TemplateArgument::Template`
295 *
296 * @note The template argument is a template name that was provided for a
297 * template template parameter.
298 *
299 * @param arg Template argument
300 * @return Return template argument model
301 */
303 const clang::TemplateArgument &arg);
304
305 /**
306 * @brief Process `clang::TemplateArgument::TemplateExpansion`
307 *
308 * @note The template argument is a pack expansion of a template name that
309 * was provided for a template template parameter.
310 *
311 * @param arg Template argument
312 * @return Return template argument model
313 */
315 const clang::TemplateArgument &arg);
316
317 /**
318 * @brief Try to process template type argument as function template
319 *
320 * @param parent Optional class in which this template is contained
321 * @param cls Template class specialization declaration
322 * @param template_decl Base template declaration
323 * @param type Template type
324 * @param template_instantiation Template class model
325 * @param argument_index Argument index
326 * @return Function template argument if succeeds, or std::nullopt
327 */
328 std::optional<template_parameter> try_as_function_prototype(
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,
333 clanguml::common::model::template_element &template_instantiation,
334 size_t argument_index);
335
336 /**
337 * @brief Try to process template type argument as array
338 *
339 * @param parent Optional class in which this template is contained
340 * @param cls Template class specialization declaration
341 * @param template_decl Base template declaration
342 * @param type Template type
343 * @param template_instantiation Template class model
344 * @param argument_index Argument index
345 * @return Array template argument if succeeds, or std::nullopt
346 */
347 std::optional<template_parameter> try_as_array(
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,
352 clanguml::common::model::template_element &template_instantiation,
353 size_t argument_index);
354
355 /**
356 * @brief Try to process template type argument as specialization type
357 *
358 * @param parent Optional class in which this template is contained
359 * @param cls Template class specialization declaration
360 * @param template_decl Base template declaration
361 * @param type Template type
362 * @param template_instantiation Template class model
363 * @param argument_index Argument index
364 * @return Template specialization template argument if succeeds,
365 * or std::nullopt
366 */
367 std::optional<template_parameter> try_as_template_specialization_type(
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,
372 clanguml::common::model::template_element &template_instantiation,
373 size_t argument_index);
374
375 /**
376 * @brief Try to process template type argument as template parameter
377 *
378 * @param cls Template class specialization declaration
379 * @param template_decl Base template declaration
380 * @param type Template type
381 * @return Template parameter if succeeds, or std::nullopt
382 */
383 std::optional<template_parameter> try_as_template_parm_type(
384 const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl,
385 clang::QualType &type);
386
387 /**
388 * @brief Try to process template type argument as lambda
389 *
390 * @param cls Template class specialization declaration
391 * @param template_decl Base template declaration
392 * @param type Template type
393 * @return Lambda template argument if succeeds, or std::nullopt
394 */
395 std::optional<template_parameter> try_as_lambda(const clang::NamedDecl *cls,
396 const clang::TemplateDecl *template_decl, clang::QualType &type);
397
398 /**
399 * @brief Try to process template type argument as record type
400 *
401 * @param parent Optional class in which this template is contained
402 * @param cls Template class specialization declaration
403 * @param template_decl Base template declaration
404 * @param type Template type
405 * @param template_instantiation Template class model
406 * @param argument_index Argument index
407 * @return Record type template argument if succeeds, or std::nullopt
408 */
409 std::optional<template_parameter> try_as_record_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,
414 clanguml::common::model::template_element &template_instantiation,
415 size_t argument_index);
416
417 /**
418 * @brief Try to process template type argument as enum type
419 *
420 * @param parent Optional class in which this template is contained
421 * @param cls Template class specialization declaration
422 * @param template_decl Base template declaration
423 * @param type Template type
424 * @param template_instantiation Template class model
425 * @return Enum type template argument if succeeds, or std::nullopt
426 */
427 std::optional<template_parameter> try_as_enum_type(
428 std::optional<clanguml::common::model::template_element *> &parent,
429 const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl,
430 clang::QualType &type,
431 clanguml::common::model::template_element &template_instantiation);
432
433 /**
434 * @brief Try to process template type argument as builtin type
435 *
436 * @param parent Optional class in which this template is contained
437 * @param type Template type
438 * @param template_decl Base template declaration
439 * @return Builtin type template argument if succeeds, or std::nullopt
440 */
441 std::optional<template_parameter> try_as_builtin_type(
442 std::optional<clanguml::common::model::template_element *> &parent,
443 clang::QualType &type, const clang::TemplateDecl *template_decl);
444
445 /**
446 * @brief Try to process template type argument as member pointer type
447 *
448 * @param parent Optional class in which this template is contained
449 * @param cls Template class specialization declaration
450 * @param template_decl Base template declaration
451 * @param type Template type
452 * @param template_instantiation Template class model
453 * @param argument_index Argument index
454 * @return Member pointer type template argument if succeeds,
455 * or std::nullopt
456 */
457 std::optional<template_parameter> try_as_member_pointer(
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,
462 clanguml::common::model::template_element &template_instantiation,
463 size_t argument_index);
464
465 /**
466 * @brief Try to process template type argument as `decltype()` type
467 *
468 * @param parent Optional class in which this template is contained
469 * @param cls Template class specialization declaration
470 * @param template_decl Base template declaration
471 * @param type Template type
472 * @param template_instantiation Template class model
473 * @param argument_index Argument index
474 * @return `decltype()` type template argument if succeeds, or std::nullopt
475 */
476 std::optional<template_parameter> try_as_decl_type(
477 std::optional<clanguml::common::model::template_element *> &parent,
478 const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl,
479 clang::QualType &type,
480 clanguml::common::model::template_element &template_instantiation,
481 size_t argument_index);
482
483 /**
484 * @brief Try to process template type argument as typedef/using type
485 *
486 * @param parent Optional class in which this template is contained
487 * @param cls Template class specialization declaration
488 * @param template_decl Base template declaration
489 * @param type Template type
490 * @param template_instantiation Template class model
491 * @param argument_index Argument index
492 * @return Typedef type template argument if succeeds, or std::nullopt
493 */
494 std::optional<template_parameter> try_as_typedef_type(
495 std::optional<clanguml::common::model::template_element *> &parent,
496 const clang::NamedDecl *cls, const clang::TemplateDecl *template_decl,
497 clang::QualType &type,
498 clanguml::common::model::template_element &template_instantiation,
499 size_t argument_index);
500
501 /**
502 * @brief Remove types context (e.g. const or reference qualifiers)
503 *
504 * This method removes all const and reference/pointer qualifiers from
505 * `type`, adds them to the template parameter model `tp` and returns
506 * a type without context.
507 *
508 * @param type Type to remove context from
509 * @param tp Template model to add context to
510 * @return Type without context
511 */
512 clang::QualType consume_context(
513 clang::QualType type, template_parameter &tp) const;
514
515 /**
516 * @brief Try to find additional relationships in unexposed parameters
517 *
518 * Sometimes, Clang will report a template parameter as unexposed, which
519 * means all we get a is a string representation of the type, sometimes
520 * with template parameter names replaced with `type-parameter-X-Y`
521 * string.
522 *
523 * This method tries to find any type names, which might be relevant for
524 * the diagram relationships.
525 *
526 * @param ct Template argument model
527 * @param relationships List of discovered relationships
528 * @return True, if any relationships were found
529 */
531 const template_parameter &ct, found_relationships_t &relationships);
532
534 common::model::template_element &template_instantiation, eid_t id,
535 const std::string &qualified_name) const;
536
537 /**
538 * @brief Get reference to Clang AST to clang-uml id mapper
539 *
540 * @return Reference to Clang AST to clang-uml id mapper
541 */
543
544 /**
545 * @brief Get reference to the current source manager
546 *
547 * @return Reference to the current source manager
548 */
549 clang::SourceManager &source_manager() const;
550
551private:
552 // Reference to the output diagram model
554
555 // Reference to diagram config
557
559
560 clang::SourceManager &source_manager_;
561
562 VisitorT &visitor_;
563};
564
565template <typename VisitorT>
568 const clanguml::config::diagram &config_, VisitorT &visitor)
569 : diagram_{diagram_}
570 , config_{config_}
571 , id_mapper_{visitor.id_mapper()}
572 , source_manager_{visitor.source_manager()}
573 , visitor_{visitor}
574{
575}
576
577template <typename VisitorT>
579{
580 return diagram_;
581}
582
583template <typename VisitorT>
585{
586 return config_;
587}
588
589template <typename VisitorT>
591{
592 return config_.using_namespace();
593}
594
595template <typename VisitorT>
597{
598 return id_mapper_;
599}
600
601template <typename VisitorT>
603{
604 return source_manager_;
605}
606
607template <typename VisitorT>
609 template_parameter &ct, const std::string &full_name) const
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}
625
626template <typename VisitorT>
629 const clang::TemplateDecl &template_declaration,
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}
742
743template <typename VisitorT>
745 const clang::NamedDecl &location_declaration,
746 clanguml::common::model::template_element &template_instantiation,
747 const clang::NamedDecl *cls,
748 const clang::TemplateSpecializationType &template_type_decl,
749 std::optional<clanguml::common::model::template_element *> parent)
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}
775
776template <typename VisitorT>
777void template_builder<VisitorT>::build(const clang::NamedDecl &location_decl,
778 clanguml::common::model::template_element &template_instantiation,
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)
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,
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}
902
903template <typename VisitorT>
905 clanguml::common::model::template_element &template_instantiation,
906 const clang::ClassTemplateSpecializationDecl &template_specialization,
907 std::optional<clanguml::common::model::template_element *> parent)
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,
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}
948
949template <typename VisitorT>
951 common::model::template_element &template_instantiation, eid_t id,
952 const std::string &qualified_name) const
953{
955 template_instantiation, qualified_name, id);
956}
957
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,
965 clanguml::common::model::template_element &template_instantiation,
966 const clang::TemplateDecl *template_decl)
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(
979 template_parameter::make_empty());
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,
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) {
1033 simplify_system_template(
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}
1051
1052template <typename VisitorT>
1054 const clang::NamedDecl &location_decl,
1055 std::optional<clanguml::common::model::template_element *> &parent,
1056 const clang::NamedDecl *cls,
1057 clanguml::common::model::template_element &template_instantiation,
1058 const clang::TemplateDecl *template_decl,
1059 const clang::TemplateArgument &arg, size_t argument_index,
1060 std::vector<template_parameter> &argument)
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}
1106
1107template <typename VisitorT>
1109 const clang::TemplateArgument &arg)
1110{
1111 auto arg_name = common::to_string(arg);
1112
1113 LOG_DBG("Processing template argument: {}", arg_name);
1114
1115 return template_parameter::make_template_type(arg_name);
1116}
1117
1118template <typename VisitorT>
1120 const clang::TemplateArgument &arg)
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}
1136
1137template <typename VisitorT>
1139 clang::QualType type, template_parameter &tp) const
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}
1147
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,
1154 clanguml::common::model::template_element &template_instantiation,
1155 size_t argument_index)
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
1213 argument = try_as_enum_type(
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}
1225
1226template <typename VisitorT>
1229 const template_parameter &ct, found_relationships_t &relationships)
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()) {
1256 found = find_relationships_in_unexposed_template_params(
1257 nested_template_params, relationships) ||
1258 found;
1259 }
1260
1261 return found;
1262}
1263
1264template <typename VisitorT>
1266 const clang::TemplateArgument &arg, const clang::ASTContext &ast_context)
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}
1282
1283#if LLVM_VERSION_MAJOR > 17
1284template <typename VisitorT>
1286 const clang::TemplateArgument &arg, const clang::ASTContext &ast_context)
1287{
1288 assert(arg.getKind() == clang::TemplateArgument::StructuralValue);
1289
1290 std::string result;
1291 llvm::raw_string_ostream ostream(result);
1292 clang::PrintingPolicy policy(ast_context.getLangOpts());
1293
1294#if LLVM_VERSION_MAJOR > 18
1295 arg.print(policy, ostream, false);
1296#else
1297 arg.dump(ostream);
1298#endif
1299
1300 return template_parameter::make_argument(result);
1301}
1302#endif
1303
1304template <typename VisitorT>
1306 const clang::TemplateArgument &arg)
1307{
1308 assert(arg.getKind() == clang::TemplateArgument::Null);
1309
1310 return template_parameter::make_argument("");
1311}
1312
1313template <typename VisitorT>
1315 const clang::TemplateArgument &arg)
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}
1323
1324template <typename VisitorT>
1326 const clang::TemplateArgument &arg)
1327{
1328 assert(arg.getKind() == clang::TemplateArgument::Expression);
1329 return template_parameter::make_argument(common::get_source_text(
1330 arg.getAsExpr()->getSourceRange(), source_manager()));
1331}
1332
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,
1339 clanguml::common::model::template_element &template_instantiation,
1340 const clang::TemplateDecl *base_template_decl,
1341 const clang::TemplateArgument &arg, size_t argument_index,
1342 std::vector<template_parameter> & /*argument*/)
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}
1358
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,
1366 clanguml::common::model::template_element &template_instantiation,
1367 size_t argument_index)
1368{
1369 const auto *mp_type =
1370 common::dereference(type)->getAs<clang::MemberPointerType>();
1371 if (mp_type == nullptr)
1372 return {};
1373
1374 auto argument = template_parameter::make_template_type("");
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}
1438
1439template <typename VisitorT>
1440std::optional<template_parameter> template_builder<VisitorT>::try_as_array(
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,
1445 clanguml::common::model::template_element &template_instantiation,
1446 size_t argument_index)
1447{
1448 const auto *array_type = common::dereference(type)->getAsArrayTypeUnsafe();
1449 if (array_type == nullptr)
1450 return {};
1451
1452 auto argument = template_parameter::make_template_type("");
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(
1469 template_parameter::make_template_type(common::to_string(
1470 ((clang::DependentSizedArrayType *)array_type) // NOLINT
1471 ->getSizeExpr())));
1472 }
1473 else if (array_type->isConstantArrayType()) {
1474 argument.add_template_param(template_parameter::make_argument(
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}
1484
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,
1492 clanguml::common::model::template_element &template_instantiation,
1493 size_t argument_index)
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
1509 auto argument = template_parameter::make_template_type("");
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}
1538
1539template <typename VisitorT>
1540std::optional<template_parameter> template_builder<VisitorT>::try_as_decl_type(
1541 std::optional<clanguml::common::model::template_element *> & /*parent*/,
1542 const clang::NamedDecl * /*cls*/,
1543 const clang::TemplateDecl * /*template_decl*/, clang::QualType &type,
1544 clanguml::common::model::template_element & /*template_instantiation*/,
1545 size_t /*argument_index*/)
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}
1558
1559template <typename VisitorT>
1560std::optional<template_parameter>
1562 std::optional<clanguml::common::model::template_element *> &parent,
1563 const clang::NamedDecl * /*cls*/,
1564 const clang::TemplateDecl * /*template_decl*/, clang::QualType &type,
1565 clanguml::common::model::template_element & /*template_instantiation*/,
1566 size_t /*argument_index*/)
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) {
1583 return template_parameter::make_argument(
1584 fmt::format("{}::{}", parent.value()->full_name(false),
1585 typedef_type->getDecl()->getNameAsString()));
1586 }
1587 // - namespace
1588 return template_parameter::make_argument(
1589 typedef_type->getDecl()->getQualifiedNameAsString());
1590 }
1591
1592 return {};
1593}
1594
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,
1602 clanguml::common::model::template_element &template_instantiation,
1603 size_t argument_index)
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
1613 auto argument = template_parameter::make_argument("");
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());
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)
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')
1660 simplify_system_template(
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})) {
1672 if (config_.generate_template_argument_dependencies()) {
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}
1697
1698template <typename VisitorT>
1699std::optional<template_parameter>
1701 const clang::NamedDecl *cls, const clang::TemplateDecl * /*template_decl*/,
1702 clang::QualType &type)
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
1726 auto argument = template_parameter::make_template_type("");
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}
1742
1743template <typename VisitorT>
1744std::optional<template_parameter> template_builder<VisitorT>::try_as_lambda(
1745 const clang::NamedDecl *cls, const clang::TemplateDecl * /*template_decl*/,
1746 clang::QualType &type)
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
1755 auto argument = template_parameter::make_argument("");
1756 type = consume_context(type, argument);
1757
1758 common::ensure_lambda_type_is_relative(config(), type_name);
1759 argument.set_type(type_name);
1760
1761 return argument;
1762}
1763
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 * /*cls*/, const clang::TemplateDecl *template_decl,
1770 clang::QualType &type,
1771 clanguml::common::model::template_element &template_instantiation,
1772 size_t /*argument_index*/)
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
1781 auto argument = template_parameter::make_argument({});
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
1800 build_from_class_template_specialization(
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
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()});
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) {
1826 if (config_.generate_template_argument_dependencies() &&
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}
1837
1838template <typename VisitorT>
1839std::optional<template_parameter> template_builder<VisitorT>::try_as_enum_type(
1840 std::optional<clanguml::common::model::template_element *> & /*parent*/,
1841 const clang::NamedDecl * /*cls*/, const clang::TemplateDecl *template_decl,
1842 clang::QualType &type,
1843 clanguml::common::model::template_element &template_instantiation)
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
1851 auto argument = template_parameter::make_argument({});
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 &&
1860 config_.generate_template_argument_dependencies()) {
1861 template_instantiation.add_relationship(
1862 {relationship_t::kDependency, type_id});
1863 }
1864
1865 return argument;
1866}
1867
1868template <typename VisitorT>
1869std::optional<template_parameter>
1871 std::optional<clanguml::common::model::template_element *> & /*parent*/,
1872 clang::QualType &type, const clang::TemplateDecl *template_decl)
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());
1881 auto argument = template_parameter::make_argument(type_name);
1882
1883 type = consume_context(type, argument);
1884 argument.set_type(type_name);
1885
1886 return argument;
1887}
1888
1889template <typename VisitorT>
1892 std::deque<std::tuple<std::string, int, bool>> &template_base_params,
1893 int arg_index, bool variadic_params, const template_parameter &ct)
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(
1920 maybe_id.value(), common::model::access_t::kPublic, false});
1921 }
1922
1923 return variadic_params;
1924}
1925
1926} // namespace clanguml::common::visitor