0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
translation_unit_visitor.h
Go to the documentation of this file.
1/**
2 * @file src/sequence_diagram/visitor/translation_unit_visitor.h
3 *
4 * Copyright (c) 2021-2025 Bartek Kryza <bkryza@gmail.com>
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 */
18#pragma once
19
23#include "config/config.h"
25
26#include <clang/AST/Expr.h>
27#include <clang/AST/RecursiveASTVisitor.h>
28#include <clang/Basic/SourceManager.h>
29
30#include <deque>
31
33
35using common::model::template_parameter;
36
37std::string to_string(const clang::FunctionTemplateDecl *decl);
38
42
43/**
44 * @brief Sequence diagram translation unit visitor
45 *
46 * This class implements the `clang::RecursiveASTVisitor` interface
47 * for selected visitors relevant to generating sequence diagrams.
48 */
50 : public clang::RecursiveASTVisitor<translation_unit_visitor>,
52public:
55
58
59 /**
60 * @brief Constructor.
61 *
62 * @param sm Current source manager reference
63 * @param diagram Diagram model
64 * @param config Diagram configuration
65 */
66 translation_unit_visitor(clang::SourceManager &sm,
69
70 ~translation_unit_visitor() override = default;
71
72 /**
73 * \defgroup Implementation of ResursiveASTVisitor methods
74 * @{
75 */
77
78 bool VisitReturnStmt(clang::ReturnStmt *stmt);
79
80 bool VisitCallExpr(clang::CallExpr *expr);
81
82 bool VisitObjCMessageExpr(clang::ObjCMessageExpr *expr);
83
84 bool VisitObjCPropertyRefExpr(clang::ObjCPropertyRefExpr *expr);
85
86 bool TraverseVarDecl(clang::VarDecl *VD);
87
88 bool TraverseCoyieldExpr(clang::CoyieldExpr *expr);
89
90 bool TraverseCoawaitExpr(clang::CoawaitExpr *expr);
91
92 bool TraverseCoreturnStmt(clang::CoreturnStmt *stmt);
93
94 bool TraverseCallExpr(clang::CallExpr *expr);
95
96 bool TraverseObjCMessageExpr(clang::ObjCMessageExpr *expr);
97
98 bool TraverseCUDAKernelCallExpr(clang::CUDAKernelCallExpr *expr);
99
100 bool TraverseCXXMemberCallExpr(clang::CXXMemberCallExpr *expr);
101
102 bool TraverseCXXOperatorCallExpr(clang::CXXOperatorCallExpr *expr);
103
104 bool VisitCXXConstructExpr(clang::CXXConstructExpr *expr);
105
106 bool TraverseCXXConstructExpr(clang::CXXConstructExpr *expr);
107
108 bool TraverseCXXTemporaryObjectExpr(clang::CXXTemporaryObjectExpr *expr);
109
110 bool TraverseReturnStmt(clang::ReturnStmt *stmt);
111
112 bool VisitCoreturnStmt(clang::CoreturnStmt *stmt);
113
114 bool VisitCoyieldExpr(clang::CoyieldExpr *expr);
115
116 bool VisitCoawaitExpr(clang::CoawaitExpr *expr);
117
118 bool VisitLambdaExpr(clang::LambdaExpr *expr);
119
120 bool TraverseLambdaExpr(clang::LambdaExpr *expr);
121
122 bool TraverseCXXMethodDecl(clang::CXXMethodDecl *declaration);
123
124 bool TraverseObjCMethodDecl(clang::ObjCMethodDecl *declaration);
125
126 bool VisitObjCMethodDecl(clang::ObjCMethodDecl *declaration);
127
128 bool VisitCXXMethodDecl(clang::CXXMethodDecl *declaration);
129
130 bool TraverseCXXRecordDecl(clang::CXXRecordDecl *declaration);
131
132 bool VisitCXXRecordDecl(clang::CXXRecordDecl *declaration);
133
134 bool VisitClassTemplateDecl(clang::ClassTemplateDecl *declaration);
135
137 clang::ClassTemplateSpecializationDecl *declaration);
138
139 bool TraverseFunctionDecl(clang::FunctionDecl *declaration);
140
141 bool VisitFunctionDecl(clang::FunctionDecl *declaration);
142
143 bool TraverseFunctionTemplateDecl(clang::FunctionTemplateDecl *declaration);
144
146 clang::FunctionTemplateDecl *function_declaration);
147
149 clang::ObjCInterfaceDecl *interface_declaration);
150
151 bool VisitObjCProtocolDecl(clang::ObjCProtocolDecl *protocol_declaration);
152
153 bool TraverseCompoundStmt(clang::CompoundStmt *stmt);
154
155 bool TraverseIfStmt(clang::IfStmt *stmt);
156
157 bool TraverseWhileStmt(clang::WhileStmt *stmt);
158
159 bool TraverseDoStmt(clang::DoStmt *stmt);
160
161 bool TraverseForStmt(clang::ForStmt *stmt);
162
163 bool TraverseCXXForRangeStmt(clang::CXXForRangeStmt *stmt);
164
165 bool TraverseCXXTryStmt(clang::CXXTryStmt *stmt);
166
167 bool TraverseCXXCatchStmt(clang::CXXCatchStmt *stmt);
168
169 bool TraverseSwitchStmt(clang::SwitchStmt *stmt);
170
171 bool TraverseCaseStmt(clang::CaseStmt *stmt);
172
173 bool TraverseDefaultStmt(clang::DefaultStmt *stmt);
174
175 bool TraverseConditionalOperator(clang::ConditionalOperator *stmt);
176 /** @} */
177
178 /**
179 * @brief Get current call expression context reference
180 *
181 * @return Reference to the current call expression context
182 */
184
185 /**
186 * @brief Get current call expression context reference
187 *
188 * @return Reference to the current call expression context
189 */
190 const call_expression_context &context() const;
191
192 /**
193 * @brief Get participant by declaration
194 *
195 * @tparam T Participant type
196 * @param decl Clang entity declaration
197 * @return Optional reference to participant diagram element
198 */
199 template <typename T = model::participant>
201 {
202 assert(decl != nullptr);
203
204 auto unique_participant_id = get_unique_id(eid_t{decl->getID()});
205 if (!unique_participant_id.has_value())
206 return {};
207
208 return get_participant<T>(unique_participant_id.value());
209 }
210
211 /**
212 * @brief Get participant by declaration
213 *
214 * @tparam T Participant type
215 * @param decl Clang entity declaration
216 * @return Optional reference to participant diagram element
217 */
218 template <typename T = model::participant>
219 common::optional_ref<T> get_participant(const clang::Decl *decl) const
220 {
221 assert(decl != nullptr);
222
223 auto unique_participant_id = get_unique_id(eid_t{decl->getID()});
224 if (!unique_participant_id.has_value())
225 return {};
226
227 return get_participant<T>(unique_participant_id.value());
228 }
229
230 /**
231 * @brief Get participant by global element id
232 *
233 * @tparam T Participant type
234 * @param id Global element id
235 * @return Optional reference to participant diagram element
236 */
237 template <typename T = model::participant>
239 {
240 if (diagram().participants().find(id) == diagram().participants().end())
241 return {};
242
244 *(static_cast<T *>(diagram().participants().at(id).get())));
245 }
246
247 /**
248 * @brief Get participant by global element id
249 *
250 * @tparam T Participant type
251 * @param id Global element id
252 * @return Optional reference to participant diagram element
253 */
254 template <typename T = model::participant>
256 {
257 if (diagram().participants().find(id) == diagram().participants().end())
258 return {};
259
261 *(static_cast<T *>(diagram().participants().at(id).get())));
262 }
263
264 /**
265 * @brief Store the mapping from local clang entity id (obtained using
266 * getID()) method to clang-uml global id
267 *
268 * @todo Refactor to @ref ast_id_mapper
269 *
270 * @param local_id Local AST element id
271 * @param global_id Globa diagram element id
272 */
273 void set_unique_id(int64_t local_id, eid_t global_id);
274
275 /**
276 * @brief Retrieve the global `clang-uml` entity id based on the Clang
277 * local id
278 * @param local_id AST local element id
279 * @return Global diagram element id
280 */
281 std::optional<eid_t> get_unique_id(eid_t local_id) const;
282
283 /**
284 * @brief Finalize diagram model for this translation unit
285 */
286 void finalize();
287
288 std::unique_ptr<sequence_diagram::model::class_> create_element(
289 const clang::NamedDecl *decl) const;
290
291private:
292 /**
293 * @brief Check if the diagram should include a declaration.
294 *
295 * @param decl Clang declaration.
296 * @return True, if the entity should be included in the diagram.
297 */
298 bool should_include(const clang::TagDecl *decl) const;
299
300 /**
301 * @brief Check if the diagram should include an ObjC declaration.
302 *
303 * @param decl Clang declaration.
304 * @return True, if the entity should be included in the diagram.
305 */
306 bool should_include(const clang::ObjCContainerDecl *decl) const;
307
308 /**
309 * @brief Check if the diagram should include a lambda expression.
310 *
311 * @param expr Lambda expression.
312 * @return True, if the expression should be included in the diagram.
313 */
314 bool should_include(const clang::LambdaExpr *expr) const;
315
316 /**
317 * @brief Check if the diagram should include a call expression.
318 *
319 * @param expr Call expression.
320 * @return True, if the expression should be included in the diagram.
321 */
322 bool should_include(const clang::CallExpr *expr) const;
323
324 /**
325 * @brief Check if the diagram should include an ObjC message expression.
326 *
327 * @param expr ObjC message expression.
328 * @return True, if the expression should be included in the diagram.
329 */
330 bool should_include(const clang::ObjCMessageExpr *expr) const;
331
332 /**
333 * @brief Check if the diagram should include a declaration.
334 *
335 * @param decl Clang declaration.
336 * @return True, if the entity should be included in the diagram.
337 */
338 bool should_include(const clang::CXXMethodDecl *decl) const;
339
340 bool should_include(const clang::ObjCMethodDecl *decl) const;
341
342 /**
343 * @brief Check if the diagram should include a declaration.
344 *
345 * @param decl Clang declaration.
346 * @return True, if the entity should be included in the diagram.
347 */
348 bool should_include(const clang::FunctionDecl *decl) const;
349
350 /**
351 * @brief Check if the diagram should include a declaration.
352 *
353 * @param decl Clang declaration.
354 * @return True, if the entity should be included in the diagram.
355 */
356 bool should_include(const clang::FunctionTemplateDecl *decl) const;
357
358 /**
359 * @brief Check if the diagram should include a declaration.
360 *
361 * @param decl Clang declaration.
362 * @return True, if the entity should be included in the diagram.
363 */
364 bool should_include(const clang::ClassTemplateDecl *decl) const;
365
366 std::unique_ptr<clanguml::sequence_diagram::model::class_>
367 create_objc_interface_model(clang::ObjCInterfaceDecl *cls);
368
369 std::unique_ptr<clanguml::sequence_diagram::model::class_>
370 create_objc_protocol_model(clang::ObjCProtocolDecl *cls);
371
372 std::unique_ptr<clanguml::sequence_diagram::model::class_>
373 create_class_model(clang::CXXRecordDecl *cls);
374
375 std::unique_ptr<clanguml::sequence_diagram::model::method>
376 create_method_model(clang::CXXMethodDecl *cls);
377
378 std::unique_ptr<clanguml::sequence_diagram::model::objc_method>
379 create_objc_method_model(clang::ObjCMethodDecl *cls);
380
381 std::unique_ptr<clanguml::sequence_diagram::model::method>
382 create_lambda_method_model(clang::CXXMethodDecl *cls);
383
384 std::unique_ptr<model::function_template>
385 build_function_template_instantiation(const clang::FunctionDecl &pDecl);
386
387 std::unique_ptr<model::function> build_function_model(
388 const clang::FunctionDecl &declaration);
389
390 std::unique_ptr<model::function_template> build_function_template(
391 const clang::FunctionTemplateDecl &declaration);
392
393 std::unique_ptr<model::class_> process_class_template_specialization(
394 clang::ClassTemplateSpecializationDecl *cls);
395
396 std::string simplify_system_template(const std::string &full_name) const;
397
398 /**
399 * @brief Assuming `cls` is a lambda, create it's full name.
400 *
401 * @note Currently, lambda names are generated using their source code
402 * location including file path, line and column to ensure
403 * unique names.
404 *
405 * @param cls Lambda declaration
406 * @return Full lambda unique name
407 */
408 std::string make_lambda_name(const clang::CXXRecordDecl *cls) const;
409
410 /**
411 * @brief Render lambda source location to string
412 *
413 * Returns exact source code location of the lambda expression in the form
414 * <filepath>:<line>:<column>.
415 *
416 * The filepath is relative to the `relative_to` config option.
417 *
418 * @param source_location Clang SourceLocation instance associated with
419 * lambda expression
420 * @return String representation of the location
421 */
422 std::string lambda_source_location(
423 const clang::SourceLocation &source_location) const;
424
425 /**
426 * @brief Check if template is a smart pointer
427 *
428 * @param primary_template Template declaration
429 * @return True, if template declaration is a smart pointer
430 */
431 bool is_smart_pointer(const clang::TemplateDecl *primary_template) const;
432
433 /**
434 * @brief Check, the callee is a template specialization
435 *
436 * @param dependent_member_expr Dependent member expression
437 * @return True, if the callee is a template specialization
438 */
440 const clang::CXXDependentScopeMemberExpr *dependent_member_expr) const;
441
442 bool process_callee(clang::CallExpr *expr, model::message &m,
443 bool generated_message_from_comment);
444
445 /**
446 * @brief Handle CXX constructor call
447 *
448 * @param m Message model
449 * @param construct_expr CXX Construct expression
450 * @return True, if `m` contains a valid constructor call
451 */
453 model::message &m, const clang::CXXConstructExpr *construct_expr);
454
455 /**
456 * @brief Handle a operator call expression
457 *
458 * @param m Message model
459 * @param operator_call_expr Operator call expression
460 * @return True, if `m` contains now a valid call expression model
461 */
463 const clang::CXXOperatorCallExpr *operator_call_expr);
464
466 model::message &m, const clang::CUDAKernelCallExpr *cuda_call_expr);
467
468 /**
469 * @brief Handle a class method call expresion
470 *
471 * @param m Message model
472 * @param method_call_expr Operator call expression
473 * @return True, if `m` contains now a valid call expression model
474 */
476 model::message &m, const clang::CXXMemberCallExpr *method_call_expr);
477
479 model::message &m, const clang::ObjCMessageExpr *message_expr);
480
481 /**
482 * @brief Handle a class template method call expresion
483 *
484 * @param m Message model
485 * @param expr Class template method call expression
486 * @return True, if `m` contains now a valid call expression model
487 */
489 model::message &m, const clang::CallExpr *expr);
490
491 /**
492 * @brief Handle a function call expresion
493 *
494 * @param m Message model
495 * @param expr Function call expression
496 * @return True, if `m` contains now a valid call expression model
497 */
499 model::message &m, const clang::CallExpr *expr);
500
501 /**
502 * @brief Handle an unresolved lookup call expresion
503 *
504 * Unresolved lookup expression is a reference to a name which Clang was
505 * not able to look up during parsing but could not resolve to a
506 * specific declaration.
507 *
508 * @param m Message model
509 * @param expr Call expression
510 * @return True, if `m` contains now a valid call expression model
511 */
513 model::message &m, const clang::CallExpr *expr) const;
514
516 model::message &m, const clang::CallExpr *expr) const;
517
518 /**
519 * @brief Register a message model `m` with a call expression
520 *
521 * This is used to know whether a model for a specific call expression
522 * has already been created, but not yet added to the diagram.
523 *
524 * @param expr Call expresion
525 * @param m Message model
526 */
527 void push_message(clang::CallExpr *expr, model::message &&m);
528 void push_message(clang::CXXConstructExpr *expr, model::message &&m);
529 void push_message(clang::ObjCMessageExpr *expr, model::message &&m);
530 void push_message(clang::ReturnStmt *stmt, model::message &&m);
531 void push_message(clang::CoreturnStmt *stmt, model::message &&m);
532 void push_message(clang::CoyieldExpr *stmt, model::message &&m);
533 void push_message(clang::CoawaitExpr *stmt, model::message &&m);
534
535 /**
536 * @brief Move a message model to diagram.
537 *
538 * @param expr Call expression
539 */
540 void pop_message_to_diagram(clang::CallExpr *expr);
541 void pop_message_to_diagram(clang::CXXConstructExpr *expr);
542 void pop_message_to_diagram(clang::ObjCMessageExpr *expr);
543 void pop_message_to_diagram(clang::ReturnStmt *stmt);
544 void pop_message_to_diagram(clang::CoreturnStmt *stmt);
545 void pop_message_to_diagram(clang::CoyieldExpr *expr);
546 void pop_message_to_diagram(clang::CoawaitExpr *expr);
547
548 std::optional<std::pair<unsigned int, std::string>> get_expression_comment(
549 const clang::SourceManager &sm, const clang::ASTContext &context,
550 eid_t caller_id, const clang::Stmt *stmt);
551
552 /**
553 * @brief Initializes model message from comment call directive
554 *
555 * @param m Message instance
556 * @return True, if the comment associated with the call expression
557 * contained a call directive and it was parsed correctly.
558 */
560
561 /**
562 * @brief Get template builder reference
563 *
564 * @return Reference to 'template_builder' instance
565 */
567
569
571
573
575
576 /**
577 * This is used to generate messages in proper order in case of nested call
578 * expressions (e.g. a(b(c(), d())), as they need to be added to the diagram
579 * sequence after the visitor leaves the call expression AST node
580 */
581 std::map<clang::CallExpr *, std::deque<model::message>>
583 std::map<clang::ReturnStmt *, model::message> return_stmt_message_map_;
584 std::map<clang::CoreturnStmt *, model::message> co_return_stmt_message_map_;
585 std::map<clang::CoyieldExpr *, model::message> co_yield_stmt_message_map_;
586 std::map<clang::CoawaitExpr *, model::message> co_await_stmt_message_map_;
587
588 std::map<clang::CXXConstructExpr *, model::message>
590 std::map<clang::ObjCMessageExpr *, model::message> objc_message_map_;
591
592 std::map<eid_t, std::unique_ptr<clanguml::sequence_diagram::model::class_>>
594
595 std::map<int64_t /* local anonymous struct id */,
596 std::tuple<std::string /* field name */, common::model::relationship_t,
599
600 std::map<eid_t, std::set<eid_t>> activity_callers_;
601
603 mutable std::set<const clang::Expr *>
605
606 mutable std::set<std::pair<int64_t, const clang::RawComment *>>
608
611};
612} // namespace clanguml::sequence_diagram::visitor