31 , template_builder_{diagram, config, *this}
35std::unique_ptr<sequence_diagram::model::class_>
37 const clang::NamedDecl * )
const
39 return std::make_unique<sequence_diagram::model::class_>(
40 config().using_namespace());
59 clang::ObjCProtocolDecl *declaration)
64 LOG_TRACE(
"Visiting ObjC interface declaration at {}",
69 if (!objc_protocol_model_ptr)
74 const auto class_id = objc_protocol_model_ptr->id();
85 : *objc_protocol_model_ptr;
88 LOG_DBG(
"Adding ObjC protocol participant {} with id {}",
89 class_model.full_name(
false), class_model.id());
91 assert(class_model.id() == class_id);
96 diagram().add_participant(std::move(objc_protocol_model_ptr));
99 LOG_DBG(
"Skipping ObjC protocol {} with id {}",
100 class_model.full_name(
true), class_id);
107 clang::ObjCInterfaceDecl *declaration)
112 LOG_TRACE(
"Visiting ObjC interface declaration at {}",
119 if (!objc_interface_model_ptr)
124 const auto class_id = objc_interface_model_ptr->id();
135 : *objc_interface_model_ptr;
138 LOG_DBG(
"Adding ObjC interface participant {} with id {}",
139 class_model.full_name(
false), class_model.id());
141 assert(class_model.id() == class_id);
146 diagram().add_participant(std::move(objc_interface_model_ptr));
149 LOG_DBG(
"Skipping ObjC interface {} with id {}",
150 class_model.full_name(
true), class_id);
157 clang::CXXRecordDecl *declaration)
159 auto context_backup =
context();
161 RecursiveASTVisitor<translation_unit_visitor>::TraverseCXXRecordDecl(
170 clang::CXXRecordDecl *declaration)
176 if (declaration->isTemplated() &&
177 declaration->getDescribedTemplate() !=
nullptr) {
182 LOG_TRACE(
"Visiting class declaration at {}",
189 if (!class_model_ptr)
194 const auto class_id = class_model_ptr->id();
207 if (!declaration->isCompleteDefinition()) {
215 LOG_DBG(
"Adding class participant {} with id {}",
216 class_model.full_name(
false), class_model.id());
218 assert(class_model.id() == class_id);
223 diagram().add_participant(std::move(class_model_ptr));
226 LOG_DBG(
"Skipping class {} with id {}", class_model.full_name(
true),
234 clang::ClassTemplateDecl *declaration)
239 LOG_TRACE(
"Visiting class template declaration {} at {} [{}]",
240 declaration->getQualifiedNameAsString(),
242 (
void *)declaration);
246 if (!class_model_ptr)
251 const auto class_full_name = class_model_ptr->full_name(
false);
256 class_model_ptr->set_id(
id);
260 if (!declaration->getTemplatedDecl()->isCompleteDefinition()) {
267 LOG_DBG(
"Adding class template participant {} with id {}",
268 class_full_name,
id);
273 diagram().add_participant(std::move(class_model_ptr));
280 clang::ClassTemplateSpecializationDecl *declaration)
285 LOG_TRACE(
"Visiting template specialization declaration {} at {}",
286 declaration->getQualifiedNameAsString(),
289 if (declaration->isLocalClass() !=
nullptr)
292 auto template_specialization_ptr =
295 if (!template_specialization_ptr)
298 const auto class_full_name = template_specialization_ptr->full_name(
false);
301 template_specialization_ptr->set_id(
id);
305 if (!declaration->isCompleteDefinition()) {
307 id, std::move(template_specialization_ptr));
314 "Adding class template specialization participant {} with id {}",
315 class_full_name,
id);
320 diagram().add_participant(std::move(template_specialization_ptr));
327 clang::ObjCMethodDecl *declaration)
332 auto context_backup =
context();
334 RecursiveASTVisitor<translation_unit_visitor>::TraverseObjCMethodDecl(
343 clang::ObjCMethodDecl *declaration)
348 LOG_TRACE(
"Visiting ObjC method {} in class",
349 declaration->getQualifiedNameAsString());
355 if (!method_model_ptr)
362 const auto method_full_name = method_model_ptr->full_name(
false);
368 LOG_TRACE(
"Set id {} --> {} for method name {} [{}]", declaration->getID(),
369 method_model_ptr->id(), method_full_name,
370 declaration->isThisDeclarationADefinition());
376 diagram().add_participant(std::move(method_model_ptr));
382 clang::CXXMethodDecl *declaration)
387 auto context_backup =
context();
389 RecursiveASTVisitor<translation_unit_visitor>::TraverseCXXMethodDecl(
398 clang::CXXMethodDecl *declaration)
403 if (!declaration->isThisDeclarationADefinition()) {
404 if (
auto *declaration_definition = declaration->getDefinition();
405 declaration_definition !=
nullptr) {
406 if (
auto *method_definition = clang::dyn_cast<clang::CXXMethodDecl>(
407 declaration_definition);
408 method_definition !=
nullptr) {
409 LOG_DBG(
"Calling VisitCXXMethodDecl recursively for forward "
417 LOG_TRACE(
"Visiting method {} in class {} [{}]",
418 declaration->getQualifiedNameAsString(),
419 declaration->getParent()->getQualifiedNameAsString(),
420 (
void *)declaration->getParent());
426 if (!method_model_ptr)
433 const auto method_full_name = method_model_ptr->full_name(
false);
439 if (declaration->isThisDeclarationADefinition()) {
441 declaration->getFirstDecl()->getID(), method_model_ptr->id());
446 LOG_TRACE(
"Set id {} --> {} for method name {} [{}]", declaration->getID(),
447 method_model_ptr->id(), method_full_name,
448 declaration->isThisDeclarationADefinition());
454 diagram().add_participant(std::move(method_model_ptr));
460 clang::FunctionDecl *declaration)
465 auto context_backup =
context();
467 RecursiveASTVisitor<translation_unit_visitor>::TraverseFunctionDecl(
476 clang::FunctionDecl *declaration)
478 if (declaration->isCXXClassMember())
484 if (!declaration->isThisDeclarationADefinition()) {
485 if (
auto *declaration_definition = declaration->getDefinition();
486 declaration_definition !=
nullptr)
488 static_cast<clang::FunctionDecl *
>(declaration_definition));
491 LOG_TRACE(
"Visiting function declaration {} at {}",
492 declaration->getQualifiedNameAsString(),
495 if (declaration->isTemplated()) {
496 if (declaration->getDescribedTemplate() !=
nullptr) {
500 eid_t{declaration->getDescribedTemplate()->getID()}))
505 std::unique_ptr<model::function> function_model_ptr{};
507 if (declaration->isFunctionTemplateSpecialization()) {
515 if (!function_model_ptr)
518 function_model_ptr->set_id(
521 function_model_ptr->is_void(declaration->getReturnType()->isVoidType());
523 function_model_ptr->is_operator(declaration->isOverloadedOperator());
525 function_model_ptr->is_cuda_kernel(
528 function_model_ptr->is_cuda_device(
537 if (declaration->isThisDeclarationADefinition()) {
539 declaration->getFirstDecl()->getID(), function_model_ptr->id());
542 set_unique_id(declaration->getID(), function_model_ptr->id());
548 diagram().add_participant(std::move(function_model_ptr));
554 clang::FunctionTemplateDecl *declaration)
559 auto context_backup =
context();
561 RecursiveASTVisitor<translation_unit_visitor>::TraverseFunctionTemplateDecl(
570 clang::FunctionTemplateDecl *declaration)
575 const auto function_name = declaration->getQualifiedNameAsString();
577 LOG_TRACE(
"Visiting function template declaration {} at {}", function_name,
587 function_template_model->is_void(
588 declaration->getAsFunction()->getReturnType()->isVoidType());
590 function_template_model->set_id(
593 function_template_model->is_operator(
594 declaration->getAsFunction()->isOverloadedOperator());
600 set_unique_id(declaration->getID(), function_template_model->id());
602 diagram().add_participant(std::move(function_template_model));
612 const auto lambda_full_name =
613 expr->getLambdaClass()->getCanonicalDecl()->getNameAsString();
615 LOG_TRACE(
"Visiting lambda expression {} at {} [caller_id = {}]",
616 lambda_full_name, expr->getBeginLoc().printToString(
source_manager()),
619 LOG_TRACE(
"Lambda call operator ID {} - lambda class ID {}, class call "
621 expr->getCallOperator()->getID(), expr->getLambdaClass()->getID(),
622 expr->getLambdaClass()->getLambdaCallOperator()->getID());
625 auto *cls = expr->getLambdaClass();
628 if (!lambda_class_model_ptr)
631 lambda_class_model_ptr->is_lambda(
true);
633 const auto cls_id = lambda_class_model_ptr->id();
637 auto lambda_method_model_ptr =
640 lambda_method_model_ptr->set_class_id(cls_id);
643 auto lambda_class_full_name = lambda_class_model_ptr->full_name(
false);
644 lambda_method_model_ptr->set_class_full_name(lambda_class_full_name);
646 diagram().add_participant(std::move(lambda_class_model_ptr));
648 lambda_method_model_ptr->set_id(
650 "::" + lambda_method_model_ptr->full_name_no_ns()));
652 get_participant<model::class_>(cls_id).value().set_lambda_operator_id(
653 lambda_method_model_ptr->id());
658 if (std::holds_alternative<clang::CallExpr *>(
659 context().current_callexpr()) &&
660 (!
context().lambda_caller_id().has_value()) &&
662 std::get<clang::CallExpr *>(
context().current_callexpr()))) {
668 m.set_from(
context().caller_id());
669 m.set_to(lambda_method_model_ptr->id());
673 diagram().add_active_participant(m.from());
674 diagram().add_active_participant(m.to());
676 LOG_DBG(
"Found call in lambda expression {} from {} [{}] to {} [{}]",
677 m.message_name(), m.from(), m.from(), m.to(), m.to());
686 expr->getCallOperator()->getID(), lambda_method_model_ptr->id());
688 diagram().add_participant(std::move(lambda_method_model_ptr));
690 [[maybe_unused]]
const auto is_generic_lambda = expr->isGenericLambda();
697 auto context_backup =
context();
699 RecursiveASTVisitor<translation_unit_visitor>::TraverseLambdaExpr(expr);
710 clang::ObjCMessageExpr *expr)
712 if (!
config().include_system_headers() &&
713 source_manager().isInSystemHeader(expr->getSourceRange().getBegin()))
716 LOG_TRACE(
"Entering ObjC message expression at {}",
721 RecursiveASTVisitor<translation_unit_visitor>::TraverseObjCMessageExpr(
724 LOG_TRACE(
"Leaving ObjC message expression at {}",
736 if (!
config().include_system_headers() &&
737 source_manager().isInSystemHeader(expr->getSourceRange().getBegin()))
740 LOG_TRACE(
"Entering co_yield expression at {}",
745 RecursiveASTVisitor<translation_unit_visitor>::TraverseCoyieldExpr(expr);
747 LOG_TRACE(
"Leaving co_yield expression at {}",
759 if (!
config().include_system_headers() &&
760 source_manager().isInSystemHeader(expr->getSourceRange().getBegin()))
763 LOG_TRACE(
"Entering co_await expression at {}",
768 RecursiveASTVisitor<translation_unit_visitor>::TraverseCoawaitExpr(expr);
770 LOG_TRACE(
"Leaving co_await expression at {}",
782 if (!
config().include_system_headers() &&
783 source_manager().isInSystemHeader(stmt->getSourceRange().getBegin()))
786 LOG_TRACE(
"Entering co_return statement at {}",
791 RecursiveASTVisitor<translation_unit_visitor>::TraverseCoreturnStmt(stmt);
793 LOG_TRACE(
"Leaving co_return statement at {}",
805 if (!
config().include_system_headers() &&
806 source_manager().isInSystemHeader(expr->getSourceRange().getBegin()))
809 LOG_TRACE(
"Entering call expression at {}",
814 RecursiveASTVisitor<translation_unit_visitor>::TraverseCallExpr(expr);
816 LOG_TRACE(
"Leaving call expression at {}",
827 clang::CUDAKernelCallExpr *expr)
829 if (!
config().include_system_headers() &&
830 source_manager().isInSystemHeader(expr->getSourceRange().getBegin()))
833 LOG_TRACE(
"Entering CUDA kernel call expression at {}",
838 RecursiveASTVisitor<translation_unit_visitor>::TraverseCallExpr(expr);
840 LOG_TRACE(
"Leaving CUDA kernel call expression at {}",
851 clang::CXXMemberCallExpr *expr)
853 if (!
config().include_system_headers() &&
854 source_manager().isInSystemHeader(expr->getSourceRange().getBegin()))
857 LOG_TRACE(
"Entering member call expression at {}",
862 RecursiveASTVisitor<translation_unit_visitor>::TraverseCXXMemberCallExpr(
865 LOG_TRACE(
"Leaving member call expression at {}",
876 clang::CXXOperatorCallExpr *expr)
880 RecursiveASTVisitor<translation_unit_visitor>::TraverseCXXOperatorCallExpr(
891 clang::CXXTemporaryObjectExpr *expr)
899 clang::dyn_cast<clang::CXXConstructExpr>(expr));
909 clang::CXXConstructExpr *expr)
911 LOG_TRACE(
"Entering cxx construct call expression at {}",
916 RecursiveASTVisitor<translation_unit_visitor>::TraverseCXXConstructExpr(
921 LOG_TRACE(
"Leaving cxx construct call expression at {}",
933 LOG_TRACE(
"Entering return statement at {}",
938 RecursiveASTVisitor<translation_unit_visitor>::TraverseReturnStmt(stmt);
940 LOG_TRACE(
"Leaving return statement at {}",
961 const auto *current_elseifstmt =
967 if (current_elseifstmt !=
nullptr) {
968 if (current_elseifstmt->getElse() == stmt) {
971 if (current_caller_id.value() != 0) {
974 diagram().add_message(std::move(m));
978 else if (current_ifstmt !=
nullptr) {
979 if (current_ifstmt->getElse() == stmt) {
982 if (current_caller_id.value() != 0) {
985 diagram().add_message(std::move(m));
990 RecursiveASTVisitor<translation_unit_visitor>::TraverseCompoundStmt(stmt);
1002 bool elseif_block{
false};
1006 const auto *current_elseifstmt =
1009 std::string condition_text;
1010 if (
config().generate_condition_statements())
1015 auto child_stmt_compare = [stmt](
auto *child_stmt) {
1016 return child_stmt == stmt;
1019 if (current_ifstmt !=
nullptr)
1020 elseif_block = elseif_block ||
1021 std::any_of(current_ifstmt->children().begin(),
1022 current_ifstmt->children().end(), child_stmt_compare);
1023 if (current_elseifstmt !=
nullptr)
1024 elseif_block = elseif_block ||
1025 std::any_of(current_elseifstmt->children().begin(),
1026 current_elseifstmt->children().end(), child_stmt_compare);
1028 if ((current_caller_id.value() != 0) && !stmt->isConstexpr()) {
1032 message m{message_t::kElseIf, current_caller_id};
1034 m.condition_text(condition_text);
1036 *
context().get_ast_context(), current_caller_id, stmt));
1037 diagram().add_block_message(std::move(m));
1044 message m{message_t::kIf, current_caller_id};
1046 m.condition_text(condition_text);
1048 *
context().get_ast_context(), current_caller_id, stmt));
1049 diagram().add_block_message(std::move(m));
1053 RecursiveASTVisitor<translation_unit_visitor>::TraverseIfStmt(stmt);
1055 if ((current_caller_id.value() != 0) && !stmt->isConstexpr()) {
1056 if (!elseif_block) {
1058 {message_t::kIfEnd, current_caller_id}, message_t::kIf);
1074 std::string condition_text;
1075 if (
config().generate_condition_statements())
1078 if (current_caller_id.value() != 0) {
1079 LOG_TRACE(
"Entering while statement at {}",
1083 message m{message_t::kWhile, current_caller_id};
1085 m.condition_text(condition_text);
1087 *
context().get_ast_context(), current_caller_id, stmt));
1088 diagram().add_block_message(std::move(m));
1091 RecursiveASTVisitor<translation_unit_visitor>::TraverseWhileStmt(stmt);
1093 if (current_caller_id.value() != 0) {
1095 {message_t::kWhileEnd, current_caller_id}, message_t::kWhile);
1110 std::string condition_text;
1111 if (
config().generate_condition_statements())
1114 if (current_caller_id.value() != 0) {
1116 message m{message_t::kDo, current_caller_id};
1118 m.condition_text(condition_text);
1120 *
context().get_ast_context(), current_caller_id, stmt));
1121 diagram().add_block_message(std::move(m));
1124 RecursiveASTVisitor<translation_unit_visitor>::TraverseDoStmt(stmt);
1126 if (current_caller_id.value() != 0) {
1128 {message_t::kDoEnd, current_caller_id}, message_t::kDo);
1143 std::string condition_text;
1144 if (
config().generate_condition_statements())
1147 if (current_caller_id.value() != 0) {
1149 message m{message_t::kFor, current_caller_id};
1151 m.condition_text(condition_text);
1154 *
context().get_ast_context(), current_caller_id, stmt));
1156 diagram().add_block_message(std::move(m));
1159 RecursiveASTVisitor<translation_unit_visitor>::TraverseForStmt(stmt);
1161 if (current_caller_id.value() != 0) {
1163 {message_t::kForEnd, current_caller_id}, message_t::kFor);
1178 if (current_caller_id.value() != 0) {
1180 message m{message_t::kTry, current_caller_id};
1183 *
context().get_ast_context(), current_caller_id, stmt));
1184 diagram().add_block_message(std::move(m));
1187 RecursiveASTVisitor<translation_unit_visitor>::TraverseCXXTryStmt(stmt);
1189 if (current_caller_id.value() != 0) {
1191 {message_t::kTryEnd, current_caller_id}, message_t::kTry);
1206 if ((current_caller_id.value() != 0) &&
1207 (
context().current_trystmt() !=
nullptr)) {
1208 std::string caught_type;
1209 if (stmt->getCaughtType().isNull())
1210 caught_type =
"...";
1213 stmt->getCaughtType(), *
context().get_ast_context());
1217 diagram().add_message(std::move(m));
1220 RecursiveASTVisitor<translation_unit_visitor>::TraverseCXXCatchStmt(stmt);
1226 clang::CXXForRangeStmt *stmt)
1234 std::string condition_text;
1235 if (
config().generate_condition_statements())
1238 if (current_caller_id.value() != 0) {
1240 message m{message_t::kFor, current_caller_id};
1242 m.condition_text(condition_text);
1244 *
context().get_ast_context(), current_caller_id, stmt));
1245 diagram().add_block_message(std::move(m));
1248 RecursiveASTVisitor<translation_unit_visitor>::TraverseCXXForRangeStmt(
1251 if (current_caller_id.value() != 0) {
1253 {message_t::kForEnd, current_caller_id}, message_t::kFor);
1266 if (current_caller_id.value() != 0) {
1271 *
context().get_ast_context(), current_caller_id, stmt));
1272 diagram().add_block_message(std::move(m));
1275 RecursiveASTVisitor<translation_unit_visitor>::TraverseSwitchStmt(stmt);
1277 if (current_caller_id.value() != 0) {
1280 {message_t::kSwitchEnd, current_caller_id}, message_t::kSwitch);
1292 if ((current_caller_id.value() != 0) &&
1293 (
context().current_switchstmt() !=
nullptr)) {
1296 diagram().add_case_stmt_message(std::move(m));
1299 RecursiveASTVisitor<translation_unit_visitor>::TraverseCaseStmt(stmt);
1310 if ((current_caller_id.value() != 0) &&
1311 (
context().current_switchstmt() !=
nullptr)) {
1314 diagram().add_case_stmt_message(std::move(m));
1317 RecursiveASTVisitor<translation_unit_visitor>::TraverseDefaultStmt(stmt);
1323 clang::ConditionalOperator *stmt)
1329 std::string condition_text;
1330 if (
config().generate_condition_statements())
1333 if (current_caller_id.value() != 0) {
1337 m.condition_text(condition_text);
1339 *
context().get_ast_context(), current_caller_id, stmt));
1340 diagram().add_block_message(std::move(m));
1343 RecursiveASTVisitor<translation_unit_visitor>::TraverseStmt(
1346 RecursiveASTVisitor<translation_unit_visitor>::TraverseStmt(
1347 stmt->getTrueExpr());
1349 if (current_caller_id.value() != 0) {
1350 model::message m{message_t::kConditionalElse, current_caller_id};
1352 diagram().add_message(std::move(m));
1355 RecursiveASTVisitor<translation_unit_visitor>::TraverseStmt(
1356 stmt->getFalseExpr());
1358 if (current_caller_id.value() != 0) {
1361 {message_t::kConditionalEnd, current_caller_id},
1362 message_t::kConditional);
1369 clang::ObjCMessageExpr *expr)
1377 if (!
context().valid() ||
context().get_ast_context() ==
nullptr)
1380 LOG_TRACE(
"Visiting ObjC message expression at {} [caller_id = {}]",
1391 raw_expr_comment,
context().get_ast_context()->getDiagnostics(), m);
1399 LOG_DBG(
"Skipping call expression due to filter at: {}",
1406 if (
context().is_expr_in_current_control_statement_condition(expr)) {
1410 if (generated_message_from_comment) {
1412 "Message for this call expression is taken from comment directive");
1419 if (m.from().value() > 0 && m.to().value() > 0) {
1420 if (raw_expr_comment !=
nullptr)
1421 m.set_comment(raw_expr_comment->getBeginLoc().getHashValue(),
1426 diagram().add_active_participant(m.from());
1427 diagram().add_active_participant(m.to());
1429 LOG_DBG(
"Found ObjC message {} from {} [{}] to {} [{}] ",
1430 m.message_name(), m.from(), m.from(), m.to(), m.to());
1439 clang::ObjCPropertyRefExpr *expr)
1447 if (!
context().valid() ||
context().get_ast_context() ==
nullptr)
1450 LOG_TRACE(
"Visiting ObjC property ref expression at {} [caller_id = {}]",
1465 if (!
context().valid() ||
context().get_ast_context() ==
nullptr)
1468 LOG_TRACE(
"Visiting call expression at {} [caller_id = {}]",
1481 raw_expr_comment,
context().get_ast_context()->getDiagnostics(), m);
1489 LOG_DBG(
"Skipping call expression due to filter at: {}",
1496 if (
context().is_expr_in_current_control_statement_condition(expr)) {
1500 auto result =
process_callee(expr, m, generated_message_from_comment);
1506 if (m.from().value() > 0 && m.to().value() > 0) {
1507 if (raw_expr_comment !=
nullptr)
1508 m.set_comment(raw_expr_comment->getBeginLoc().getHashValue(),
1513 diagram().add_active_participant(m.from());
1514 diagram().add_active_participant(m.to());
1516 LOG_DBG(
"Found call {} from {} [{}] to {} [{}] ", m.message_name(),
1517 m.from(), m.from(), m.to(), m.to());
1530 if (expr ==
nullptr)
1533 if (generated_message_from_comment) {
1535 "Message for this call expression is taken from comment directive");
1540 else if (
const auto *cuda_call_expr =
1541 llvm::dyn_cast_or_null<clang::CUDAKernelCallExpr>(expr);
1542 cuda_call_expr !=
nullptr) {
1548 else if (
const auto *operator_call_expr =
1549 llvm::dyn_cast_or_null<clang::CXXOperatorCallExpr>(expr);
1550 operator_call_expr !=
nullptr) {
1557 else if (
const auto *method_call_expr =
1558 llvm::dyn_cast_or_null<clang::CXXMemberCallExpr>(expr);
1559 method_call_expr !=
nullptr) {
1567 auto *callee_decl = expr->getCalleeDecl();
1569 if (callee_decl ==
nullptr) {
1570 LOG_DBG(
"Cannot get callee declaration - trying direct function "
1573 callee_decl = expr->getDirectCallee();
1575 if (callee_decl !=
nullptr)
1576 LOG_DBG(
"Found function/method callee in: {}",
1580 if (callee_decl ==
nullptr) {
1584 if (llvm::dyn_cast_or_null<clang::CXXDependentScopeMemberExpr>(
1585 expr->getCallee()) !=
nullptr) {
1592 else if (llvm::dyn_cast_or_null<clang::UnresolvedLookupExpr>(
1593 expr->getCallee()) !=
nullptr) {
1597 LOG_DBG(
"Processing lambda expression callee");
1600 else if (llvm::dyn_cast_or_null<clang::DependentScopeDeclRefExpr>(
1601 expr->getCallee()) !=
nullptr) {
1602 LOG_DBG(
"Processing dependent scope declaration expression "
1603 "callee - not able to infer the template parameter "
1604 "type at this point: {}",
1608 LOG_DBG(
"Found unsupported callee decl type for: {} at {}",
1617 LOG_DBG(
"Skipping call expression at: {}",
1636 if (!
context().valid() ||
context().get_ast_context() ==
nullptr)
1639 LOG_TRACE(
"Visiting co_await expression at {} [caller_id = {}]",
1646 if (expr->getOperand() !=
nullptr) {
1654 raw_expr_comment,
context().get_ast_context()->getDiagnostics(), m);
1661 if (!generated_message_from_comment &&
1663 clang::dyn_cast_or_null<clang::CallExpr>(expr->getResumeExpr()))) {
1664 LOG_DBG(
"Skipping call expression due to filter at: {}",
1671 if (
context().is_expr_in_current_control_statement_condition(expr)) {
1676 clang::dyn_cast_or_null<clang::CallExpr>(expr->getResumeExpr()), m,
1677 generated_message_from_comment);
1684 if (m.from().value() > 0) {
1685 if (raw_expr_comment !=
nullptr)
1686 m.set_comment(raw_expr_comment->getBeginLoc().getHashValue(),
1691 diagram().add_active_participant(m.from());
1693 LOG_DBG(
"Found return call {} from {} [{}] to {} [{}] ",
1694 m.message_name(), m.from(), m.from(), m.to(), m.to());
1710 if (!
context().valid() ||
context().get_ast_context() ==
nullptr)
1713 LOG_TRACE(
"Visiting return statement at {}",
1723 raw_expr_comment,
context().get_ast_context()->getDiagnostics(), m);
1728 if (stmt->getRetValue() !=
nullptr) {
1733 if (
context().lambda_caller_id().has_value() &&
1734 !
context().is_local_class()) {
1735 const auto &lambda_model = get_participant<model::method>(
1736 *
context().lambda_caller_id());
1738 if (lambda_model.has_value()) {
1739 if (lambda_model.has_value())
1740 m.set_return_type(lambda_model.value().return_type());
1743 else if (
context().current_function_decl_ !=
nullptr) {
1745 context().current_function_decl_->getReturnType().getAsString());
1747 else if (
context().current_function_template_decl_ !=
nullptr) {
1749 .current_function_template_decl_->getAsFunction()
1753 else if (
context().current_method_decl_ !=
nullptr) {
1755 context().current_method_decl_->getReturnType().getAsString());
1757 else if (
context().current_objc_method_decl_ !=
nullptr) {
1759 context().current_objc_method_decl_->getReturnType().getAsString());
1764 if (m.from().value() > 0) {
1765 if (raw_expr_comment !=
nullptr)
1766 m.set_comment(raw_expr_comment->getBeginLoc().getHashValue(),
1771 diagram().add_active_participant(m.from());
1773 LOG_DBG(
"Found return call {} from {} [{}] to {} [{}] ",
1774 m.message_name(), m.from(), m.from(), m.to(), m.to());
1790 if (!
context().valid() ||
context().get_ast_context() ==
nullptr)
1793 LOG_TRACE(
"Visiting co_yield expression at {}",
1803 raw_expr_comment,
context().get_ast_context()->getDiagnostics(), m);
1808 if (expr->getOperand() !=
nullptr) {
1813 if (
context().current_function_decl_ !=
nullptr) {
1815 context().current_function_decl_->getReturnType().getAsString());
1817 else if (
context().current_function_template_decl_ !=
nullptr) {
1819 .current_function_template_decl_->getAsFunction()
1823 else if (
context().current_method_decl_ !=
nullptr) {
1825 context().current_method_decl_->getReturnType().getAsString());
1827 else if (
context().current_objc_method_decl_ !=
nullptr) {
1829 context().current_objc_method_decl_->getReturnType().getAsString());
1834 if (m.from().value() > 0) {
1835 if (raw_expr_comment !=
nullptr)
1836 m.set_comment(raw_expr_comment->getBeginLoc().getHashValue(),
1841 diagram().add_active_participant(m.from());
1843 LOG_DBG(
"Found co_yield call {} from {} [{}] to {} [{}] ",
1844 m.message_name(), m.from(), m.from(), m.to(), m.to());
1860 if (!
context().valid() ||
context().get_ast_context() ==
nullptr)
1863 LOG_TRACE(
"Visiting co_return statement at {}",
1873 raw_expr_comment,
context().get_ast_context()->getDiagnostics(), m);
1878 if (stmt->getOperand() !=
nullptr) {
1883 if (
context().current_function_decl_ !=
nullptr) {
1885 context().current_function_decl_->getReturnType().getAsString());
1887 else if (
context().current_function_template_decl_ !=
nullptr) {
1889 .current_function_template_decl_->getAsFunction()
1893 else if (
context().current_method_decl_ !=
nullptr) {
1895 context().current_method_decl_->getReturnType().getAsString());
1897 else if (
context().current_objc_method_decl_ !=
nullptr) {
1899 context().current_objc_method_decl_->getReturnType().getAsString());
1904 if (m.from().value() > 0) {
1905 if (raw_expr_comment !=
nullptr)
1906 m.set_comment(raw_expr_comment->getBeginLoc().getHashValue(),
1911 diagram().add_active_participant(m.from());
1913 LOG_DBG(
"Found co_return call {} from {} [{}] to {} [{}] ",
1914 m.message_name(), m.from(), m.from(), m.to(), m.to());
1926 diagram().sequences().insert({m.
from(), std::move(a)});
1936 auto generated_message_from_comment{
false};
1937 for (
const auto &decorator : m.
decorators()) {
1938 auto call_decorator =
1939 std::dynamic_pointer_cast<decorators::call>(decorator);
1940 if (call_decorator &&
1941 call_decorator->applies_to_diagram(
config().name)) {
1943 generated_message_from_comment =
true;
1947 return generated_message_from_comment;
1952 if (decl->isStaticLocal())
1955 RecursiveASTVisitor::TraverseVarDecl(decl);
1957 if (decl->isStaticLocal())
1964 clang::CXXConstructExpr *expr)
1972 if (expr ==
nullptr)
1975 if (
const auto *ctor = expr->getConstructor();
1979 LOG_TRACE(
"Visiting cxx construct expression at {} [caller_id = {}]",
1989 if (
context().is_expr_in_current_control_statement_condition(expr)) {
1996 if (m.from().value() > 0 && m.to().value() > 0) {
1999 diagram().add_active_participant(m.from());
2000 diagram().add_active_participant(m.to());
2002 LOG_DBG(
"Found constructor call {} from {} [{}] to {} [{}] ",
2003 m.message_name(), m.from(), m.from(), m.to(), m.to());
2014 const auto *callee_decl = expr->getCalleeDecl();
2016 if (callee_decl ==
nullptr)
2019 const auto *callee_function = callee_decl->getAsFunction();
2021 if (callee_function ==
nullptr)
2028 if (
config().combine_free_functions_into_file_participants() &&
2032 auto callee_name = callee_function->getQualifiedNameAsString() +
"()";
2041 model::message &m,
const clang::CXXOperatorCallExpr *operator_call_expr)
2043 if (operator_call_expr->getCalleeDecl() ==
nullptr)
2046 LOG_DBG(
"Operator '{}' call expression to {} at {}",
2047 getOperatorSpelling(operator_call_expr->getOperator()),
2048 operator_call_expr->getCalleeDecl()->getID(),
2049 operator_call_expr->getBeginLoc().printToString(
source_manager()));
2052 if (
const auto *lambda_method = clang::dyn_cast<clang::CXXMethodDecl>(
2053 operator_call_expr->getCalleeDecl());
2054 lambda_method !=
nullptr && lambda_method->getParent()->isLambda()) {
2056 LOG_DBG(
"Operator callee is a lambda: {}",
2059 const auto source_location{
2064 m.
set_to(
eid_t{lambda_method->getParent()->getID()});
2067 const auto operator_ast_id =
2068 operator_call_expr->getCalleeDecl()->getID();
2073 "operator{}", getOperatorSpelling(operator_call_expr->getOperator())));
2079 model::message &m,
const clang::CXXConstructExpr *construct_expr)
2081 const auto *constructor = construct_expr->getConstructor();
2082 if (constructor ==
nullptr)
2085 const auto *constructor_parent = constructor->getParent();
2086 if (constructor_parent ==
nullptr)
2089 LOG_DBG(
"Constructor '{}' call expression to {} at {}",
2090 construct_expr->getConstructor()->getNameAsString(),
2091 constructor->getID(),
2096 fmt::format(
"{}::{}", constructor_parent->getQualifiedNameAsString(),
2097 constructor_parent->getNameAsString()));
2099 diagram().add_active_participant(
eid_t{constructor->getID()});
2107 const auto *method_decl = message_expr->getMethodDecl();
2109 if (method_decl ==
nullptr)
2112 std::string method_name = method_decl->getQualifiedNameAsString();
2114 if (message_expr->getReceiverInterface() ==
nullptr)
2117 const auto *callee_decl = message_expr->getReceiverInterface();
2119 if (callee_decl ==
nullptr)
2125 if (callee_decl->getImplementation() !=
nullptr &&
2126 callee_decl->getImplementation()->getMethod(method_decl->getSelector(),
2127 method_decl->isInstanceMethod(),
true) !=
nullptr) {
2128 const auto *impl_method_decl =
2129 callee_decl->getImplementation()->getMethod(
2130 method_decl->getSelector(), method_decl->isInstanceMethod(),
2140 message_expr->getCallReturnType(*
context().get_ast_context())
2143 LOG_TRACE(
"Set callee ObjC method id {} for method name {}", m.
to(),
2144 method_decl->getQualifiedNameAsString());
2146 diagram().add_active_participant(
eid_t{method_decl->getID()});
2152 model::message &m,
const clang::CXXMemberCallExpr *method_call_expr)
2155 const auto *method_decl = method_call_expr->getMethodDecl();
2157 if (method_decl ==
nullptr)
2160 std::string method_name = method_decl->getQualifiedNameAsString();
2162 const auto *callee_decl =
2163 method_decl !=
nullptr ? method_decl->getParent() :
nullptr;
2165 if (callee_decl ==
nullptr)
2174 method_call_expr->getCallReturnType(*
context().get_ast_context())
2177 LOG_TRACE(
"Set callee method id {} for method name {}", m.
to(),
2178 method_decl->getQualifiedNameAsString());
2180 diagram().add_active_participant(
eid_t{method_decl->getID()});
2188 const auto *dependent_member_callee =
2189 clang::dyn_cast_or_null<clang::CXXDependentScopeMemberExpr>(
2192 if (dependent_member_callee ==
nullptr)
2196 if (
const auto *tst = dependent_member_callee->getBaseType()
2197 ->getAs<clang::TemplateSpecializationType>();
2199 const auto *template_declaration =
2200 tst->getTemplateName().getAsTemplateDecl();
2202 std::string callee_method_full_name;
2210 "::" + dependent_member_callee->getMember().getAsString();
2212 for (
const auto &[
id, p] :
diagram().participants()) {
2213 const auto p_full_name = p->full_name(
false);
2215 if (p_full_name.find(callee_method_full_name +
"(") == 0) {
2225 const auto *argument_template =
2226 template_declaration->getTemplateParameters()
2235 dependent_member_callee->getMember().getAsString();
2237 for (
const auto &[
id, p] :
diagram().participants()) {
2238 const auto p_full_name = p->full_name(
false);
2239 if (p_full_name.find(callee_method_full_name +
"(") ==
2253 dependent_member_callee->getMember().getAsString());
2255 if (
const auto maybe_id =
2257 maybe_id.has_value())
2258 diagram().add_active_participant(maybe_id.value());
2262 LOG_DBG(
"Skipping call due to unresolvable "
2263 "CXXDependentScopeMemberExpr at {}",
2273 const auto *callee_decl = expr->getCalleeDecl();
2275 if (callee_decl ==
nullptr)
2278 const auto *callee_function = callee_decl->getAsFunction();
2280 if (callee_function ==
nullptr)
2287 if (
config().combine_free_functions_into_file_participants() &&
2291 auto callee_name = callee_function->getQualifiedNameAsString() +
"()";
2302 const auto *lambda_expr =
2303 clang::dyn_cast_or_null<clang::LambdaExpr>(expr->getCallee());
2305 if (lambda_expr ==
nullptr)
2308 const auto lambda_class_id =
eid_t{lambda_expr->getLambdaClass()->getID()};
2318 const auto *unresolved_expr =
2319 clang::dyn_cast_or_null<clang::UnresolvedLookupExpr>(expr->getCallee());
2321 if (unresolved_expr !=
nullptr) {
2322 for (
const auto *decl : unresolved_expr->decls()) {
2323 if (clang::dyn_cast_or_null<clang::FunctionTemplateDecl>(decl) !=
2326 clang::dyn_cast_or_null<clang::FunctionTemplateDecl>(decl);
2331 if (clang::dyn_cast_or_null<clang::FunctionDecl>(decl) !=
nullptr) {
2333 clang::dyn_cast_or_null<clang::FunctionDecl>(decl);
2338 LOG_DBG(
"Unknown unresolved lookup expression");
2346 const clang::CXXDependentScopeMemberExpr *dependent_member_expr)
const
2348 if (dependent_member_expr ==
nullptr)
2351 if (dependent_member_expr->getBaseType().isNull())
2354 const auto *tst = dependent_member_expr->getBaseType()
2355 ->getAs<clang::TemplateSpecializationType>();
2360 return !(tst->isPointerType());
2364 const clang::TemplateDecl *primary_template)
const
2366 return primary_template->getQualifiedNameAsString().find(
2367 "std::unique_ptr") == 0 ||
2368 primary_template->getQualifiedNameAsString().find(
"std::shared_ptr") ==
2370 primary_template->getQualifiedNameAsString().find(
"std::weak_ptr") == 0;
2373std::unique_ptr<clanguml::sequence_diagram::model::class_>
2375 clang::ObjCProtocolDecl *cls)
2377 assert(cls !=
nullptr);
2379 auto c_ptr{std::make_unique<clanguml::sequence_diagram::model::class_>(
2380 config().using_namespace())};
2383 auto qualified_name = cls->getQualifiedNameAsString();
2385 c.is_objc_interface();
2387 c.set_name(cls->getQualifiedNameAsString());
2396 c.set_style(c.style_spec());
2401std::unique_ptr<clanguml::sequence_diagram::model::class_>
2403 clang::ObjCInterfaceDecl *cls)
2405 assert(cls !=
nullptr);
2407 auto c_ptr{std::make_unique<clanguml::sequence_diagram::model::class_>(
2408 config().using_namespace())};
2411 auto qualified_name = cls->getQualifiedNameAsString();
2413 c.is_objc_interface(
true);
2415 c.set_name(cls->getQualifiedNameAsString());
2424 c.set_style(c.style_spec());
2429std::unique_ptr<clanguml::sequence_diagram::model::class_>
2432 assert(cls !=
nullptr);
2434 auto c_ptr{std::make_unique<clanguml::sequence_diagram::model::class_>(
2435 config().using_namespace())};
2438 auto qualified_name = cls->getQualifiedNameAsString();
2440 if (!cls->isLambda())
2449 const auto *parent = cls->getParent();
2451 if ((parent !=
nullptr) && parent->isRecord()) {
2456 std::optional<eid_t> id_opt;
2457 const auto *parent_record_decl =
2458 clang::dyn_cast<clang::RecordDecl>(parent);
2460 assert(parent_record_decl !=
nullptr);
2462 const eid_t ast_id{parent_record_decl->getID()};
2470 (parent_record_decl->getDescribedTemplate() !=
nullptr)) {
2471 parent_record_decl->getDescribedTemplate()->getID();
2472 if (parent_record_decl->getDescribedTemplate() !=
nullptr)
2484 if (!parent_class) {
2489 if (cls->getNameAsString().empty()) {
2492 const auto &[label, hint, access] =
2495 c.set_name(parent_class.value().name() +
2496 "::" + fmt::format(
"({})", label));
2498 parent_class.value().add_relationship(
2502 c.set_name(parent_class.value().name() +
"::" +
2504 "(anonymous_{})", std::to_string(cls->getID())));
2508 parent_class.value().name() +
"::" + cls->getNameAsString());
2515 else if (cls->isLambda()) {
2517 if (cls->getParent() !=
nullptr) {
2520 c.set_name(type_name);
2521 c.set_namespace(ns);
2525 LOG_WARN(
"Cannot find parent declaration for lambda {}",
2526 cls->getQualifiedNameAsString());
2531 else if (cls->isLocalClass() !=
nullptr) {
2532 const auto *func_declaration = cls->isLocalClass();
2534 eid_t local_parent_id{int64_t{}};
2537 LOG_DBG(
"The local class is defined in a lambda operator()");
2538 const auto *method_declaration =
2539 clang::dyn_cast<clang::CXXMethodDecl>(func_declaration);
2541 if (method_declaration !=
nullptr &&
2542 method_declaration->getParent() !=
nullptr) {
2543 local_parent_id = method_declaration->getParent()->getID();
2547 local_parent_id = func_declaration->getID();
2554 const auto &func_model =
2557 if (!func_model.has_value())
2560 LOG_DBG(
"Visiting local class declaration: {} in {}",
2561 cls->getQualifiedNameAsString(),
2562 func_model.value().full_name(
false));
2564 auto local_cls_ns = func_model.value().get_namespace();
2568 c.set_namespace(local_cls_ns);
2574 c.set_namespace(ns);
2578 c.is_struct(cls->isStruct());
2586 c.set_style(c.style_spec());
2593 LOG_TRACE(
"Setting local element mapping {} --> {}", local_id, global_id);
2601 eid_t local_id)
const
2609std::unique_ptr<model::function_template>
2611 const clang::FunctionTemplateDecl &declaration)
2613 auto function_template_model_ptr =
2614 std::make_unique<sequence_diagram::model::function_template>(
2615 config().using_namespace());
2620 *function_template_model_ptr, declaration);
2622 function_template_model_ptr->return_type(
2624 declaration.getASTContext()));
2626 for (
const auto *param : declaration.getTemplatedDecl()->parameters()) {
2627 function_template_model_ptr->add_parameter(
2629 param->getType(), declaration.getASTContext(),
false)));
2632 return function_template_model_ptr;
2635std::unique_ptr<model::function_template>
2637 const clang::FunctionDecl &decl)
2639 auto template_instantiation_ptr =
2640 std::make_unique<model::function_template>(
config().using_namespace());
2641 auto &template_instantiation = *template_instantiation_ptr;
2646 decl.getPrimaryTemplate(),
2647 decl.getTemplateSpecializationArgs()->asArray(),
2651 for (
const auto *param : decl.parameters()) {
2652 template_instantiation_ptr->add_parameter(
2656 return template_instantiation_ptr;
2660 const clang::FunctionDecl &declaration)
2662 auto function_model_ptr =
2663 std::make_unique<sequence_diagram::model::function>(
2664 config().using_namespace());
2667 function_model_ptr->set_name(ns.name());
2669 function_model_ptr->set_namespace(ns);
2672 declaration.getReturnType(), declaration.getASTContext()));
2674 for (
const auto *param : declaration.parameters()) {
2675 function_model_ptr->add_parameter(
2677 param->getType(), declaration.getASTContext(),
false)));
2680 if (declaration.isVariadic()) {
2681 function_model_ptr->add_parameter(
"...");
2684 return function_model_ptr;
2687std::unique_ptr<model::class_>
2689 clang::ClassTemplateSpecializationDecl *cls)
2691 auto c_ptr{std::make_unique<model::class_>(
config().using_namespace())};
2695 auto &template_instantiation = *c_ptr;
2698 auto qualified_name = cls->getQualifiedNameAsString();
2704 template_instantiation.set_name(cls->getNameAsString());
2705 template_instantiation.set_namespace(ns);
2707 template_instantiation.is_struct(cls->isStruct());
2713 if (template_instantiation.skip())
2716 template_instantiation.set_id(
2725 const std::string &full_name)
const
2727 return config().simplify_template_type(full_name);
2731 const clang::SourceLocation &source_location)
const
2733 const auto file_line =
2735 const auto file_column =
2737 const std::string file_name =
2739 .make_path_relative(
2742 return fmt::format(
"{}:{}:{}", file_name, file_line, file_column);
2746 const clang::CXXRecordDecl *cls)
const
2749 const auto location = cls->getLocation();
2753 if (maybe_lambda_caller_id.has_value()) {
2755 std::string parent_lambda_class_name{
"()"};
2756 if (
diagram().get_participant<model::method>(
2757 maybe_lambda_caller_id.value())) {
2758 auto parent_lambda_class_id =
2761 maybe_lambda_caller_id.value())
2766 parent_lambda_class_id)) {
2767 parent_lambda_class_name =
2775 result = fmt::format(
2776 "{}##(lambda {})", parent_lambda_class_name, source_location);
2778 else if (
context().caller_id().value() != 0 &&
2780 auto parent_full_name =
2784 fmt::format(
"{}##(lambda {})", parent_full_name, source_location);
2787 result = fmt::format(
"(lambda {})", source_location);
2837 assert(expr !=
nullptr);
2850 auto caller_id = msg.from();
2855 if (
diagram().has_activity(caller_id))
2856 diagram().get_activity(caller_id).add_message(std::move(msg));
2858 LOG_DBG(
"Skipping message due to missing activity: {}", caller_id);
2867 clang::CXXConstructExpr *expr)
2869 assert(expr !=
nullptr);
2879 auto caller_id = msg.from();
2880 diagram().get_activity(caller_id).add_message(std::move(msg));
2887 assert(stmt !=
nullptr);
2896 auto caller_id = msg.from();
2897 diagram().get_activity(caller_id).add_message(std::move(msg));
2904 assert(stmt !=
nullptr);
2914 auto caller_id = msg.from();
2915 diagram().get_activity(caller_id).add_message(std::move(msg));
2922 assert(expr !=
nullptr);
2932 auto caller_id = msg.from();
2933 diagram().get_activity(caller_id).add_message(std::move(msg));
2940 assert(expr !=
nullptr);
2950 auto caller_id = msg.from();
2951 diagram().get_activity(caller_id).add_message(std::move(msg));
2957 clang::ObjCMessageExpr *expr)
2959 assert(expr !=
nullptr);
2968 auto caller_id = msg.from();
2969 diagram().get_activity(caller_id).add_message(std::move(msg));
2986 if (
config().inline_lambda_messages())
2987 diagram().inline_lambda_operator_calls();
2992 for (
auto &[
id, activity] :
diagram().sequences()) {
2993 for (
auto &m : activity.messages()) {
2996 if (participant && participant.value().is_lambda() &&
2997 participant.value().lambda_operator_id().value() != 0) {
2998 LOG_DBG(
"Changing lambda expression target id from {} to {}",
2999 m.to(), participant.value().lambda_operator_id());
3001 m.set_to(participant.value().lambda_operator_id());
3002 m.set_message_name(
"operator()");
3012 std::set<eid_t> active_participants_unique;
3015 for (
auto id :
diagram().active_participants()) {
3017 !
id.is_global() && unique_id.has_value()) {
3018 active_participants_unique.emplace(unique_id.value());
3020 else if (
id.is_global()) {
3021 active_participants_unique.emplace(
id);
3025 diagram().active_participants() = std::move(active_participants_unique);
3028 for (
auto &[
id, activity] :
diagram().sequences()) {
3029 for (
auto &m : activity.messages()) {
3031 !m.to().is_global() && unique_id.has_value()) {
3032 m.set_to(unique_id.value());
3033 assert(m.to().is_global());
3042 std::map<eid_t, std::set<eid_t>> acs;
3047 std::set<eid_t> unique_caller_ids;
3048 for (
const auto &caller_id : caller_ids) {
3050 if (unique_caller_id)
3051 unique_caller_ids.emplace(*unique_caller_id);
3053 acs.emplace(*unique_id, std::move(unique_caller_ids));
3057 for (
auto &[
id, activity] :
diagram().sequences()) {
3058 assert(
id.is_global());
3060 if (acs.count(
id) > 0) {
3061 activity.set_callers(acs.at(
id));
3066std::unique_ptr<clanguml::sequence_diagram::model::method>
3068 clang::CXXMethodDecl *declaration)
3070 auto method_model_ptr = std::make_unique<sequence_diagram::model::method>(
3071 config().using_namespace());
3074 auto method_name = ns.
name();
3075 method_model_ptr->set_method_name(method_name);
3077 method_model_ptr->set_name(ns.name());
3079 method_model_ptr->set_namespace(ns);
3081 method_model_ptr->is_defaulted(declaration->isDefaulted());
3082 method_model_ptr->is_assignment(declaration->isCopyAssignmentOperator() ||
3083 declaration->isMoveAssignmentOperator());
3084 method_model_ptr->is_const(declaration->isConst());
3085 method_model_ptr->is_static(declaration->isStatic());
3086 method_model_ptr->is_operator(declaration->isOverloadedOperator());
3087 method_model_ptr->is_constructor(
3088 clang::dyn_cast<clang::CXXConstructorDecl>(declaration) !=
nullptr);
3090 method_model_ptr->is_void(declaration->getReturnType()->isVoidType());
3093 declaration->getReturnType(), declaration->getASTContext()));
3095 for (
const auto *param : declaration->parameters()) {
3096 auto parameter_type =
3100 method_model_ptr->add_parameter(
config().using_namespace().relative(
3104 return method_model_ptr;
3107std::unique_ptr<clanguml::sequence_diagram::model::objc_method>
3109 clang::ObjCMethodDecl *declaration)
3111 auto method_model_ptr =
3112 std::make_unique<sequence_diagram::model::objc_method>(
3113 config().using_namespace());
3116 auto method_name = ns.
name();
3117 method_model_ptr->set_method_name(method_name);
3119 method_model_ptr->set_name(ns.name());
3120 method_model_ptr->set_namespace({});
3122 clang::Decl *parent_decl = declaration->getClassInterface();
3124 if (parent_decl ==
nullptr) {
3125 LOG_DBG(
"Cannot find ObjC interface for method, probably it is a "
3132 LOG_DBG(
"Getting ObjC method's interface with local id {}",
3133 parent_decl->getID());
3135 const auto maybe_method_class = get_participant<model::class_>(parent_decl);
3137 if (!maybe_method_class) {
3138 LOG_DBG(
"Cannot find parent class_ for method {} in class {}",
3139 declaration->getQualifiedNameAsString(),
3140 declaration->getClassInterface()->getQualifiedNameAsString());
3144 const auto &method_class = maybe_method_class.value();
3146 method_model_ptr->is_void(declaration->getReturnType()->isVoidType());
3148 method_model_ptr->set_class_id(method_class.id());
3149 method_model_ptr->set_class_full_name(method_class.full_name(
false));
3150 method_model_ptr->set_name(
get_participant(method_model_ptr->class_id())
3152 .full_name_no_ns() +
3153 "::" + declaration->getNameAsString());
3156 declaration->getReturnType(), declaration->getASTContext()));
3158 for (
const auto *param : declaration->parameters()) {
3159 auto parameter_type =
3163 method_model_ptr->add_parameter(parameter_type);
3166 return method_model_ptr;
3169std::unique_ptr<clanguml::sequence_diagram::model::method>
3172 auto method_model_ptr = std::make_unique<sequence_diagram::model::method>(
3173 config().using_namespace());
3176 auto method_name = ns.
name();
3177 method_model_ptr->set_method_name(method_name);
3179 method_model_ptr->set_name(ns.name());
3181 method_model_ptr->set_namespace(ns);
3183 method_model_ptr->is_defaulted(declaration->isDefaulted());
3184 method_model_ptr->is_assignment(declaration->isCopyAssignmentOperator() ||
3185 declaration->isMoveAssignmentOperator());
3186 method_model_ptr->is_const(declaration->isConst());
3187 method_model_ptr->is_static(declaration->isStatic());
3188 method_model_ptr->is_operator(declaration->isOverloadedOperator());
3189 method_model_ptr->is_constructor(
3190 clang::dyn_cast<clang::CXXConstructorDecl>(declaration) !=
nullptr);
3193 clang::Decl *parent_decl = declaration->getParent();
3195 if (
context().current_class_template_decl_ !=
nullptr)
3198 LOG_DBG(
"Getting method's class with local id {}", parent_decl->getID());
3200 const auto maybe_method_class = get_participant<model::class_>(parent_decl);
3202 if (!maybe_method_class) {
3203 LOG_DBG(
"Cannot find parent class_ for method {} in class {}",
3204 declaration->getQualifiedNameAsString(),
3205 declaration->getParent()->getQualifiedNameAsString());
3210 "Found method class: {}", maybe_method_class.value().full_name(
false));
3212 const auto &method_class = maybe_method_class.value();
3214 method_model_ptr->is_void(declaration->getReturnType()->isVoidType());
3216 method_model_ptr->set_class_id(method_class.id());
3217 method_model_ptr->set_class_full_name(method_class.full_name(
false));
3218 method_model_ptr->set_name(
get_participant(method_model_ptr->class_id())
3220 .full_name_no_ns() +
3221 "::" + declaration->getNameAsString());
3224 declaration->getReturnType(), declaration->getASTContext()));
3226 for (
const auto *param : declaration->parameters()) {
3227 auto parameter_type =
3231 method_model_ptr->add_parameter(
config().using_namespace().relative(
3235 return method_model_ptr;
3241 dynamic_cast<const clang::NamedDecl *
>(decl));
3245 const clang::ObjCContainerDecl *decl)
const
3248 dynamic_cast<const clang::NamedDecl *
>(decl));
3252 const clang::LambdaExpr *expr)
const
3254 if (
context().caller_id() == 0)
3260 const auto expr_file = expr->getBeginLoc().printToString(
source_manager());
3270 const clang::ObjCMessageExpr *expr)
const
3272 if (
context().caller_id() == 0)
3278 const auto expr_file = expr->getBeginLoc().printToString(
source_manager());
3289 if (
context().caller_id() == 0)
3293 if (expr->isCallToStdMove())
3296 if (expr->isImplicitCXXThis())
3299 if (clang::dyn_cast_or_null<clang::ImplicitCastExpr>(expr) !=
nullptr)
3305 const auto expr_file = expr->getBeginLoc().printToString(
source_manager());
3311 const auto *callee_decl = expr->getCalleeDecl();
3313 if (callee_decl !=
nullptr) {
3314 const auto *callee_function = callee_decl->getAsFunction();
3316 if ((callee_function ==
nullptr) || !
should_include(callee_function)) {
3317 LOG_DBG(
"Skipping call expression at {}",
3329 const clang::CXXMethodDecl *decl)
const
3338 LOG_DBG(
"Including method {}", decl->getQualifiedNameAsString());
3344 const clang::ObjCMethodDecl *decl)
const
3350 LOG_DBG(
"Including ObjC method {}", decl->getQualifiedNameAsString());
3356 const clang::FunctionDecl *decl)
const
3362 const clang::FunctionTemplateDecl *decl)
const
3368 const clang::ClassTemplateDecl *decl)
const
3373std::optional<std::pair<unsigned int, std::string>>
3375 const clang::ASTContext &context,
const eid_t caller_id,
3376 const clang::Stmt *stmt)
3378 const auto *raw_comment =
3381 if (raw_comment ==
nullptr)
3392 raw_comment->getFormattedText(sm, sm.getDiagnostics()));
3394 if (stripped_comment.empty())
3397 return {{raw_comment->getBeginLoc().getHashValue(), stripped_comment}};