0.6.1
C++ to UML diagram generator based on Clang
Loading...
Searching...
No Matches
call_expression_context.h
Go to the documentation of this file.
1/**
2 * @file src/sequence_diagram/visitor/call_expression_context.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
20#include "common/clang_utils.h"
21#include "util/util.h"
22
23#include <clang/AST/Expr.h>
24#include <clang/AST/RecursiveASTVisitor.h>
25#include <clang/Basic/SourceManager.h>
26
27#include <stack>
28
30
32
33/**
34 * @brief This class is used to track current context of the call expressions.
35 *
36 * When traversing AST for call expressions, we need to keep the state of
37 * the current context, for instance whether we are in a `for` loop or
38 * an `if` block, as well as the current parent of the call expression
39 * e.g. a class method or function.
40 */
42 /**
43 * In Clang, call to a class constructor is represented by
44 * `clang::CXXConstructExpr`, which does inherit from `clang::CallExpr`.
45 * So to enable to track calls to constructors, we need to be able
46 * to add to the call stack either type.
47 */
48 using callexpr_stack_t = std::variant<std::monostate, clang::CallExpr *,
49 clang::CXXConstructExpr *, clang::ObjCMessageExpr *,
50 clang::ReturnStmt *, clang::CoreturnStmt *, clang::CoyieldExpr *,
51 clang::CoawaitExpr *>;
52
54
55 /**
56 * @brief Reset call expression context to the original state.
57 */
58 void reset();
59
60 /**
61 * @brief Verify that the context is in a valid state.
62 *
63 * Context can only be in a single state (for instance inside a function).
64 *
65 * @return True, if the context is in a valid state.
66 */
67 bool valid() const;
68
69 /**
70 * @brief
71 *
72 * @return Current AST context
73 */
74 clang::ASTContext *get_ast_context() const;
75
76 /**
77 * @brief Set the current context to a class.
78 *
79 * @param cls Class declaration.
80 */
81 void update(clang::CXXRecordDecl *cls);
82 void update(clang::ObjCInterfaceDecl *cls);
83 void update(clang::ObjCProtocolDecl *cls);
84
85 /**
86 * @brief Set the current context to a class template specialization.
87 *
88 * @param clst Class template specialization declaration.
89 */
90 void update(clang::ClassTemplateSpecializationDecl *clst);
91
92 /**
93 * @brief Set the current context to a class template.
94 *
95 * @param clst Class template declaration.
96 */
97 void update(clang::ClassTemplateDecl *clst);
98
99 /**
100 * @brief Set the current context to a class method.
101 *
102 * @param method Class method declaration.
103 */
104 void update(clang::CXXMethodDecl *method);
105 void update(clang::ObjCMethodDecl *method);
106
107 /**
108 * @brief Set the current context to a function.
109 *
110 * @param function Function declaration.
111 */
112 void update(clang::FunctionDecl *function);
113
114 /**
115 * @brief Set the current context to a function template.
116 *
117 * @param function_template Function template declaration.
118 */
119 void update(clang::FunctionTemplateDecl *function_template);
120
121 /**
122 * @brief Set current caller to id of the current participant.
123 *
124 * @param id Set current caller id.
125 */
126 void set_caller_id(eid_t id);
127
128 /**
129 * @brief Get current caller id
130 *
131 * @return Id of the current caller participant
132 */
133 eid_t caller_id() const;
134
135 /**
136 * @brief Get the id of the current lambda caller.
137 *
138 * Since lambdas can be nested within methods and functions, they have
139 * a separate caller id field.
140 *
141 * @return Current lambda caller id, or 0 if current caller is not lambda.
142 */
143 std::optional<eid_t> lambda_caller_id() const;
144
145 /**
146 * @brief Enter a lambda expression
147 *
148 * @param id Lambda id
149 */
151
152 /**
153 * @brief Leave current lambda expression
154 */
156
157 /**
158 * @brief Get current `if` statement block
159 *
160 * @return `if` statement block.
161 */
162 clang::IfStmt *current_ifstmt() const;
163
164 /**
165 * @brief Enter `if` statement block
166 *
167 * @param stmt `if` statement block
168 */
169 void enter_ifstmt(clang::IfStmt *stmt);
170
171 /**
172 * @brief Leave `if` statement block
173 */
174 void leave_ifstmt();
175
176 /**
177 * @brief Enter `else if` statement block
178 *
179 * @param stmt `if` statement block
180 */
181 void enter_elseifstmt(clang::IfStmt *stmt);
182
183 /**
184 * @brief Get current `else if` statement block
185 *
186 * @return `if` statement block.
187 */
188 clang::IfStmt *current_elseifstmt() const;
189
190 /**
191 * @brief Get current loop statement block
192 *
193 * @return Loop statement block.
194 */
195 clang::Stmt *current_loopstmt() const;
196
197 /**
198 * @brief Enter loop statement block
199 *
200 * @param stmt Loop statement block
201 */
202 void enter_loopstmt(clang::Stmt *stmt);
203
204 /**
205 * @brief Leave loop statement block
206 */
207 void leave_loopstmt();
208
209 /**
210 * @brief Get current `try` statement block
211 *
212 * @return `try` statement block.
213 */
214 clang::Stmt *current_trystmt() const;
215
216 /**
217 * @brief Enter `try` statement block
218 *
219 * @param stmt `try` statement block
220 */
221 void enter_trystmt(clang::Stmt *stmt);
222
223 /**
224 * @brief Leave `try` statement block
225 */
226 void leave_trystmt();
227
228 /**
229 * @brief Get current `switch` statement block
230 *
231 * @return `switch` statement block.
232 */
233 clang::SwitchStmt *current_switchstmt() const;
234
235 /**
236 * @brief Enter `switch` statement block
237 *
238 * @param stmt `switch` statement block
239 */
240 void enter_switchstmt(clang::SwitchStmt *stmt);
241
242 /**
243 * @brief Leave `switch` statement block
244 */
245 void leave_switchstmt();
246
247 /**
248 * @brief Get current `:?` statement block
249 *
250 * @return `:?` statement block.
251 */
252 clang::ConditionalOperator *current_conditionaloperator() const;
253
254 /**
255 * @brief Enter `:?` statement block
256 *
257 * @param stmt `:?` statement block
258 */
259 void enter_conditionaloperator(clang::ConditionalOperator *stmt);
260
261 /**
262 * @brief Leave `:?` statement block
263 */
265
266 /**
267 * @brief Get current call expression
268 *
269 * @return Call expression
270 */
272
273 /**
274 * @brief Enter a call expression
275 *
276 * @param expr Call expression
277 */
278 void enter_callexpr(clang::CallExpr *expr);
279
280 /**
281 * @brief Enter a constructor call expression
282 *
283 * @param expr Constructor call expression
284 */
285 void enter_callexpr(clang::CXXConstructExpr *expr);
286 void enter_callexpr(clang::ObjCMessageExpr *expr);
287 void enter_callexpr(clang::ReturnStmt *stmt);
288 void enter_callexpr(clang::CoreturnStmt *stmt);
289 void enter_callexpr(clang::CoyieldExpr *expr);
290 void enter_callexpr(clang::CoawaitExpr *expr);
291
292 /**
293 * @brief Leave call expression
294 */
295 void leave_callexpr();
296
297 /**
298 * @brief Check, if a statement is contained in a control statement
299 *
300 * This method is used to check if `stmt` is contained in control
301 * statement of a block, for instance:
302 *
303 * ```cpp
304 * if(a.method1()) {}
305 * ```
306 * it will return `true` for `stmt` representing `method1()` call
307 * expression.
308 *
309 * @param stmt Statement
310 * @return True, if `stmt` is contained in control expression of a
311 * statement block
312 */
314 const clang::Stmt *stmt) const;
315
316 /**
317 * @brief Print the current call expression stack for debugging.
318 */
319 void dump();
320
321 /**
322 * @brief Check if current context is inside a local class
323 *
324 * @return True, if current context is inside a local class
325 */
326 bool is_local_class() const;
327
328 clang::CXXRecordDecl *current_class_decl_{nullptr};
329 clang::ClassTemplateDecl *current_class_template_decl_{nullptr};
330 clang::ClassTemplateSpecializationDecl
332 clang::CXXMethodDecl *current_method_decl_{nullptr};
333 clang::ObjCMethodDecl *current_objc_method_decl_{nullptr};
334 clang::FunctionDecl *current_function_decl_{nullptr};
335 clang::FunctionTemplateDecl *current_function_template_decl_{nullptr};
336 clang::ObjCInterfaceDecl *objc_interface_decl_{nullptr};
337 clang::ObjCProtocolDecl *objc_protocol_decl_{nullptr};
338
339private:
341 std::stack<eid_t> current_lambda_caller_id_;
342
343 std::stack<callexpr_stack_t> call_expr_stack_;
344
345 std::stack<clang::IfStmt *> if_stmt_stack_;
346 std::map<clang::IfStmt *, std::stack<clang::IfStmt *>> elseif_stmt_stacks_;
347
348 std::stack<clang::Stmt *> loop_stmt_stack_;
349 std::stack<clang::Stmt *> try_stmt_stack_;
350 std::stack<clang::SwitchStmt *> switch_stmt_stack_;
351 std::stack<clang::ConditionalOperator *> conditional_operator_stack_;
352};
353
354} // namespace clanguml::sequence_diagram::visitor