49 LOG_DBG(
"Skipping empty call from '{}' to '{}'", m.
from(), m.
to());
62 const auto &f =
dynamic_cast<const model::method &
>(to.value());
63 const std::string_view style = f.
is_static() ?
"__" :
"";
65 if (m.
type() == message_t::kCoAwait)
66 message = fmt::format(
"{}<< co_await >>\\n{}{}", style,
67 f.message_name(render_mode), style);
70 "{}{}{}", style, f.message_name(render_mode), style);
72 else if (to.value().
type_name() ==
"objc_method") {
74 const std::string_view style = f.
is_static() ?
"__" :
"";
76 fmt::format(
"{}{}{}", style, f.message_name(render_mode), style);
78 else if (
config().combine_free_functions_into_file_participants()) {
79 if (to.value().
type_name() ==
"function") {
83 if (f.is_cuda_kernel())
85 else if (f.is_cuda_device())
87 else if (f.is_coroutine())
90 else if (to.value().
type_name() ==
"function_template") {
94 if (f.is_cuda_kernel())
96 else if (f.is_cuda_device())
98 else if (f.is_coroutine())
112 ostr << from_alias <<
" "
117 if (
config().generate_links) {
145 std::string message_stereotype;
146 if (m.
type() == message_t::kCoReturn) {
147 message_stereotype =
"<< co_return >>";
149 else if (m.
type() == message_t::kCoYield) {
150 message_stereotype =
"<< co_yield >>";
153 std::string message_label;
157 if (to.has_value() && from.has_value() && !from.value().
is_void()) {
162 ostr << from_alias <<
" "
166 if (
config().generate_return_types())
168 else if (
config().generate_return_values())
171 else if (from.has_value() && !from.value().is_void() &&
172 (from.value().type_name() ==
"method" ||
173 from.value().type_name() ==
"objc_method" ||
174 config().combine_free_functions_into_file_participants())) {
177 ostr <<
"[<--" <<
" " << from_alias;
178 if (
config().generate_return_types())
180 else if (
config().generate_return_values())
184 if (!message_stereotype.empty()) {
185 if (message_label.empty())
186 message_label = fmt::format(
"//{}//", message_stereotype);
188 message_label = fmt::format(
189 "//{}//\\n//{}//", message_stereotype, message_label);
192 if (!message_label.empty())
193 message_label = fmt::format(
"//{}//", message_label);
196 if (!message_label.empty())
197 ostr <<
" : " << message_label;
203 eid_t activity_id, std::ostream &ostr, std::vector<eid_t> &visited)
const
205 const auto &a =
model().get_activity(activity_id);
209 if (
config().fold_repeated_activities() && !inserted &&
210 !a.messages().empty()) {
224 for (
const auto &m : a.messages()) {
225 if (m.in_static_declaration_context()) {
232 if (m.type() == message_t::kCall || m.type() == message_t::kCoAwait) {
236 if (!to.has_value()) {
237 LOG_DBG(
"Skipping activity {} due to missing target paricipant "
243 visited.push_back(m.from());
245 LOG_DBG(
"Generating message [{}] --> [{}]", m.from(), m.to());
251 ostr <<
"activate " << to_alias <<
'\n';
253 if (
model().sequences().find(m.to()) !=
model().sequences().end()) {
254 if (std::find(visited.begin(), visited.end(), m.to()) ==
258 LOG_DBG(
"Generating activity {} (called from {})", m.to(),
265 LOG_DBG(
"Skipping activity {} --> {} - missing sequence {}",
266 m.from(), m.to(), m.to());
268 ostr <<
"deactivate " << to_alias <<
'\n';
272 else if (m.type() == message_t::kReturn) {
275 auto return_message = m;
276 if (!visited.empty()) {
277 return_message.set_to(visited.back());
281 else if (m.type() == message_t::kCoReturn) {
284 auto return_message = m;
285 if (!visited.empty()) {
286 return_message.set_to(visited.back());
290 else if (m.type() == message_t::kCoYield) {
293 auto return_message = m;
294 if (!visited.empty()) {
295 return_message.set_to(visited.back());
299 else if (m.type() == message_t::kIf) {
303 if (
const auto &text = m.condition_text(); text.has_value())
304 ostr <<
" " << text.value();
307 else if (m.type() == message_t::kElseIf) {
310 if (
const auto &text = m.condition_text(); text.has_value())
311 ostr <<
" " << text.value();
314 else if (m.type() == message_t::kElse) {
318 else if (m.type() == message_t::kIfEnd) {
321 else if (m.type() == message_t::kWhile) {
325 if (
const auto &text = m.condition_text(); text.has_value())
326 ostr <<
" " << text.value();
329 else if (m.type() == message_t::kWhileEnd) {
332 else if (m.type() == message_t::kFor) {
336 if (
const auto &text = m.condition_text(); text.has_value())
337 ostr <<
" " << text.value();
340 else if (m.type() == message_t::kForEnd) {
343 else if (m.type() == message_t::kDo) {
347 if (
const auto &text = m.condition_text(); text.has_value())
348 ostr <<
" " << text.value();
351 else if (m.type() == message_t::kDoEnd) {
354 else if (m.type() == message_t::kTry) {
357 ostr <<
"group try\n";
359 else if (m.type() == message_t::kCatch) {
363 else if (m.type() == message_t::kTryEnd) {
367 else if (m.type() == message_t::kSwitch) {
370 ostr <<
"group switch\n";
372 else if (m.type() == message_t::kCase) {
376 else if (m.type() == message_t::kSwitchEnd) {
379 else if (m.type() == message_t::kConditional) {
383 if (
const auto &text = m.condition_text(); text.has_value())
384 ostr <<
" " << text.value();
387 else if (m.type() == message_t::kConditionalElse) {
391 else if (m.type() == message_t::kConditionalEnd) {
405 bool comment_generated_from_note_decorators{
false};
406 for (
const auto &decorator : m.
decorators()) {
407 auto note = std::dynamic_pointer_cast<decorators::note>(decorator);
408 if (note && note->applies_to_diagram(
config().name)) {
409 comment_generated_from_note_decorators =
true;
414 note->text,
config().message_comment_width())
417 ostr <<
"end note" <<
'\n';
421 if (comment_generated_from_note_decorators)
424 if (!
config().generate_message_comments())
428 if (
const auto &comment = m.
comment(); comment &&
434 config().message_comment_width())
437 ostr <<
"end note" <<
'\n';
442 std::ostream &ostr,
const std::string &name)
const
444 auto p =
model().get(name);
446 if (!p.has_value()) {
447 LOG_WARN(
"Cannot find participant {} from `participants_order` "
457 std::ostream &ostr,
eid_t id,
bool force)
const
459 eid_t participant_id{};
462 for (
const auto pid :
model().active_participants()) {
464 participant_id = pid;
472 if (participant_id == 0)
478 const auto &participant =
481 if (participant.type_name() ==
"method") {
482 const auto class_id =
491 const auto &class_participant =
496 auto participant_name =
config().simplify_template_type(
499 config().using_namespace().relative(participant_name);
503 ostr <<
"participant \"" << participant_name <<
"\" as "
504 << class_participant.alias();
506 if (
config().generate_links) {
508 ostr, class_participant);
515 else if (participant.type_name() ==
"objc_method") {
516 const auto class_id =
525 const auto &class_participant =
530 auto participant_name =
config().simplify_template_type(
533 config().using_namespace().relative(participant_name);
537 ostr <<
"participant \"" << participant_name <<
"\" as "
538 << class_participant.alias() <<
" <<ObjC Interface>>";
540 if (
config().generate_links) {
542 ostr, class_participant);
549 else if ((participant.type_name() ==
"function" ||
550 participant.type_name() ==
"function_template") &&
551 config().combine_free_functions_into_file_participants()) {
554 const auto &file_path =
560 assert(!file_path.empty());
568 std::filesystem::path{file_path},
config().root_directory())
571 ostr <<
"participant \"" << participant_name <<
"\" as "
572 << fmt::format(
"C_{:022}", file_id.value());
581 auto participant_name =
582 config().using_namespace().relative(
config().simplify_template_type(
586 ostr <<
"participant \"" << participant_name <<
"\" as "
587 << participant.alias();
589 if (
const auto *function_ptr =
592 if (function_ptr->is_cuda_kernel())
593 ostr <<
" << CUDA Kernel >>";
594 else if (function_ptr->is_cuda_device())
595 ostr <<
" << CUDA Device >>";
596 else if (function_ptr->is_coroutine())
597 ostr << fmt::format(
"<< Coroutine >>");
600 if (
config().generate_links) {
621 if ((participant.
type_name() ==
"function" ||
622 participant.
type_name() ==
"function_template") &&
623 config().combine_free_functions_into_file_participants()) {
626 return fmt::format(
"C_{:022}", file_id.value());
629 return participant.
alias();
636 if (
config().participants_order.has_value) {
637 for (
const auto &p :
config().participants_order()) {
638 LOG_DBG(
"Pregenerating participant {}", p);
655 std::vector<eid_t> visited_participants;
657 for (
const auto from_id : start_from) {
658 if (
model().participants().count(from_id) == 0)
663 if (!from.has_value()) {
664 LOG_WARN(
"Failed to find participant {} for 'from' "
680 if (from.value().type_name() ==
"method" ||
681 from.value().type_name() ==
"objc_method" ||
682 config().combine_free_functions_into_file_participants()) {
683 ostr <<
"[->" <<
" " << from_alias <<
" : "
688 ostr <<
"activate " << from_alias <<
'\n';
692 ostr <<
"deactivate " << from_alias <<
'\n';
698 std::vector<eid_t> start_from;
699 for (
const auto &sf :
config().from()) {
700 if (sf.location_type == location_t::function) {
702 for (
const auto &[k, v] :
model().sequences()) {
703 if (
model().participants().count(v.from()) == 0)
706 const auto &caller = *
model().participants().at(v.from());
707 std::string vfrom = caller.full_name(
false);
708 if (sf.location == vfrom) {
709 LOG_DBG(
"Found sequence diagram start point: {}", k);
710 start_from.push_back(k);
718 fmt::format(
"Failed to find participant matching '{}' for "
719 "'from' condition: ",
720 sf.location.to_string()));
729 std::vector<model::message_chain_t> result;
731 for (
const auto &to_location :
config().to()) {
732 auto to_activity_ids =
model().get_to_activity_ids(to_location);
734 if (to_activity_ids.empty()) {
735 LOG_WARN(
"Failed to find participant matching '{}' for "
737 to_location.location.to_string());
740 for (
const auto &to_activity_id : to_activity_ids) {
741 std::vector<model::message_chain_t> message_chains_unique =
742 model().get_all_from_to_message_chains(
eid_t{}, to_activity_id);
744 result.insert(result.end(), message_chains_unique.begin(),
745 message_chains_unique.end());
759 std::vector<model::message_chain_t> message_chains =
762 bool first_separator_skipped{
false};
763 for (
const auto &mc : message_chains) {
764 if (!first_separator_skipped)
765 first_separator_skipped =
true;
769 const auto from_activity_id = mc.front().from();
771 if (
model().participants().count(from_activity_id) == 0)
777 if (from.value().type_name() ==
"method" ||
778 from.value().type_name() ==
"objc_method" ||
779 config().combine_free_functions_into_file_participants()) {
787 for (
const auto &m : mc) {
795 for (
const auto &ft :
config().from_to()) {
798 assert(ft.size() == 2);
800 const auto &from_location = ft.front();
801 const auto &to_location = ft.back();
803 const auto from_activity_ids =
804 model().get_from_activity_ids(from_location);
806 const auto to_activity_ids =
model().get_to_activity_ids(to_location);
808 if (from_activity_ids.empty()) {
811 fmt::format(
"Failed to find participant matching '{}' for "
812 "'from' condition: ",
813 from_location.location.to_string()));
816 if (from_activity_ids.empty() || to_activity_ids.empty()) {
819 fmt::format(
"Failed to find participant matching '{}' for "
821 to_location.location.to_string()));
824 bool first_separator_skipped{
false};
826 for (
const auto from_activity_id : from_activity_ids) {
827 if (
model().participants().count(from_activity_id) == 0)
830 for (
const auto to_activity_id : to_activity_ids) {
831 if (
model().participants().count(to_activity_id) == 0)
834 auto message_chains_unique =
835 model().get_all_from_to_message_chains(
836 from_activity_id, to_activity_id);
838 for (
const auto &mc : message_chains_unique) {
839 if (!first_separator_skipped)
840 first_separator_skipped =
true;
847 if (from.value().type_name() ==
"method" ||
848 from.value().type_name() ==
"objc_method" ||
850 .combine_free_functions_into_file_participants()) {
859 for (
const auto &m : mc) {
871 if (
config().generate_method_arguments() ==