Added support for C++20 coroutines in class diagrams (#221)

This commit is contained in:
Bartek Kryza
2023-12-15 20:00:56 +01:00
parent 7be848b8a1
commit f2fe1ca2cf
16 changed files with 218 additions and 1 deletions

View File

@@ -41,6 +41,7 @@ Main features supported so far include:
* Interactive links to online code or docs for classes, methods and class fields in SVG diagrams - [_example_](https://raw.githubusercontent.com/bkryza/clang-uml/master/docs/test_cases/t00002_class.svg)
* Support for plain C99/C11 code (struct, units and their relationships) - [_example_](docs/test_cases/t00057.md)
* C++20 concept constraints - [_example_](docs/test_cases/t00059.md)
* C++20 coroutines - [_example_](docs/test_cases/t00069.md)
* **Sequence diagram generation**
* Generation of sequence diagram from specific method or function - [_example_](docs/test_cases/t20001.md)
* Generation of loop and conditional statements - [_example_](docs/test_cases/t20021.md)

View File

@@ -63,6 +63,7 @@ void to_json(nlohmann::json &j, const class_method &c)
j["is_noexcept"] = c.is_noexcept();
j["is_constexpr"] = c.is_constexpr();
j["is_consteval"] = c.is_consteval();
j["is_coroutine"] = c.is_coroutine();
j["is_constructor"] = c.is_constructor();
j["is_move_assignment"] = c.is_move_assignment();
j["is_copy_assignment"] = c.is_copy_assignment();

View File

@@ -251,6 +251,9 @@ void generator::generate_method(
if (m.is_consteval()) {
method_mods.emplace_back("consteval");
}
if (m.is_coroutine()) {
method_mods.emplace_back("coroutine");
}
if (!method_mods.empty()) {
ostr << fmt::format("[{}] ", fmt::join(method_mods, ","));

View File

@@ -338,6 +338,9 @@ void generator::generate_method(
else if (m.is_deleted())
ostr << " = deleted";
if (m.is_coroutine())
ostr << " [coroutine]";
ostr << " : " << type;
if (config().generate_links) {

View File

@@ -70,6 +70,13 @@ void class_method::is_consteval(bool is_consteval)
is_consteval_ = is_consteval;
}
bool class_method::is_coroutine() const { return is_coroutine_; }
void class_method::is_coroutine(bool is_coroutine)
{
is_coroutine_ = is_coroutine;
}
bool class_method::is_noexcept() const { return is_noexcept_; }
void class_method::is_noexcept(bool is_noexcept) { is_noexcept_ = is_noexcept; }

View File

@@ -152,6 +152,20 @@ public:
*/
void is_consteval(bool is_consteval);
/**
* @brief Whether the method is a C++20 coroutine.
*
* @return True, if the method is a coroutine
*/
bool is_coroutine() const;
/**
* @brief Set whether the method is a C++20 coroutine.
*
* @param is_coroutine True, if the method is a coroutine
*/
void is_coroutine(bool is_coroutine);
/**
* @brief Whether the method is noexcept.
*
@@ -262,6 +276,7 @@ private:
bool is_noexcept_{false};
bool is_constexpr_{false};
bool is_consteval_{false};
bool is_coroutine_{false};
bool is_constructor_{false};
bool is_destructor_{false};
bool is_move_assignment_{false};

View File

@@ -1369,6 +1369,7 @@ void translation_unit_visitor::process_method_properties(
method.is_move_assignment(mf.isMoveAssignmentOperator());
method.is_copy_assignment(mf.isCopyAssignmentOperator());
method.is_noexcept(isNoexceptExceptionSpec(mf.getExceptionSpecType()));
method.is_coroutine(common::is_coroutine(mf));
}
void translation_unit_visitor::

View File

@@ -876,4 +876,10 @@ clang::RawComment *get_expression_raw_comment(const clang::SourceManager &sm,
return {};
}
bool is_coroutine(const clang::FunctionDecl &decl)
{
const auto *body = decl.getBody();
return clang::isa_and_nonnull<clang::CoroutineBodyStmt>(body);
}
} // namespace clanguml::common

View File

@@ -295,4 +295,12 @@ consume_type_context(clang::QualType type);
clang::RawComment *get_expression_raw_comment(const clang::SourceManager &sm,
const clang::ASTContext &context, const clang::Stmt *stmt);
/**
* Check if function or method declaration is a C++20 coroutine.
*
* @param decl Function declaration
* @return True, if the function is a C++20 coroutine.
*/
bool is_coroutine(const clang::FunctionDecl &decl);
} // namespace clanguml::common

View File

@@ -5,7 +5,7 @@ file(GLOB_RECURSE TEST_CONFIG_YMLS test_config_data/*.yml
test_compilation_database_data/*.yml
test_compilation_database_data/*.json)
set(TEST_CASES_REQUIRING_CXX20 t00056 t00058 t00059 t00065)
set(TEST_CASES_REQUIRING_CXX20 t00056 t00058 t00059 t00065 t00069)
set(CLANG_UML_TEST_LIBRARIES
clang-umllib

9
tests/t00069/.clang-uml Normal file
View File

@@ -0,0 +1,9 @@
diagrams:
t00069_class:
type: class
glob:
- t00069.cc
include:
namespaces:
- clanguml::t00069
using_namespace: clanguml::t00069

63
tests/t00069/t00069.cc Normal file
View File

@@ -0,0 +1,63 @@
#include <coroutine>
#include <optional>
namespace clanguml {
namespace t00069 {
template <typename T> struct generator {
struct promise_type;
using handle_type = std::coroutine_handle<promise_type>;
generator(handle_type h)
: h_(h)
{
}
~generator() { h_.destroy(); }
struct promise_type {
T value_;
std::exception_ptr exception_;
generator get_return_object()
{
return generator(handle_type::from_promise(*this));
}
std::suspend_always initial_suspend() { return {}; }
std::suspend_always final_suspend() noexcept { return {}; }
void unhandled_exception() { exception_ = std::current_exception(); }
template <std::convertible_to<T> From>
std::suspend_always yield_value(From &&from)
{
value_ = std::forward<From>(from);
return {};
}
void return_void() { }
};
handle_type h_;
private:
bool full_ = false;
};
class A {
public:
generator<unsigned long> iota() { co_yield counter_++; }
generator<unsigned long> seed()
{
counter_ = 42;
co_return;
}
private:
unsigned long counter_;
};
} // namespace t00069
} // namespace clanguml

87
tests/t00069/test_case.h Normal file
View File

@@ -0,0 +1,87 @@
/**
* tests/t00069/test_case.h
*
* Copyright (c) 2021-2023 Bartek Kryza <bkryza@gmail.com>
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
TEST_CASE("t00069", "[test-case][class]")
{
auto [config, db] = load_config("t00069");
auto diagram = config.diagrams["t00069_class"];
REQUIRE(diagram->name == "t00069_class");
auto model = generate_class_diagram(*db, diagram);
REQUIRE(model->name() == "t00069_class");
{
auto src = generate_class_puml(diagram, *model);
AliasMatcher _A(src);
REQUIRE_THAT(src, StartsWith("@startuml"));
REQUIRE_THAT(src, EndsWith("@enduml\n"));
// Check if all classes exist
REQUIRE_THAT(src, IsClass(_A("A")));
// Check if class templates exist
REQUIRE_THAT(src, IsClassTemplate("generator", "T"));
// Check if all inner classes exist
REQUIRE_THAT(src,
IsInnerClass(_A("generator<T>"), _A("generator::promise_type")));
// Check if all methods exist
REQUIRE_THAT(src,
(IsMethod<Public, Coroutine>("iota", "generator<unsigned long>")));
REQUIRE_THAT(src,
(IsMethod<Public, Coroutine>("seed", "generator<unsigned long>")));
// Check if all relationships exist
REQUIRE_THAT(
src, IsDependency(_A("A"), _A("generator<unsigned long>")));
REQUIRE_THAT(src,
IsInstantiation(
_A("generator<T>"), _A("generator<unsigned long>")));
save_puml(config.output_directory(), diagram->name + ".puml", src);
}
{
auto j = generate_class_json(diagram, *model);
using namespace json;
save_json(config.output_directory(), diagram->name + ".json", j);
}
{
auto src = generate_class_mermaid(diagram, *model);
mermaid::AliasMatcher _A(src);
using mermaid::IsClass;
using mermaid::IsMethod;
REQUIRE_THAT(src, IsClass(_A("A")));
REQUIRE_THAT(src,
(IsMethod<Public, Coroutine>("iota", "generator<unsigned long>")));
REQUIRE_THAT(src,
(IsMethod<Public, Coroutine>("seed", "generator<unsigned long>")));
save_mermaid(config.output_directory(), diagram->name + ".mmd", src);
}
}

View File

@@ -404,6 +404,9 @@ using namespace clanguml::test::matchers;
#include "t00066/test_case.h"
#include "t00067/test_case.h"
#include "t00068/test_case.h"
#if defined(ENABLE_CXX_STD_20_TEST_CASES)
#include "t00069/test_case.h"
#endif
///
/// Sequence diagram tests

View File

@@ -108,6 +108,8 @@ struct Constexpr { };
struct Consteval { };
struct Coroutine { };
struct Noexcept { };
struct Default { };
@@ -962,6 +964,9 @@ ContainsMatcher IsMethod(std::string const &name,
if constexpr (has_type<Deleted, Ts...>())
pattern += " = deleted";
if constexpr (has_type<Coroutine, Ts...>())
pattern += " [coroutine]";
pattern += " : " + type;
return ContainsMatcher(CasedString(pattern, caseSensitivity));
@@ -995,6 +1000,8 @@ ContainsMatcher IsMethod(std::string const &name, std::string type = "void",
method_mods.push_back("constexpr");
if constexpr (has_type<Consteval, Ts...>())
method_mods.push_back("consteval");
if constexpr (has_type<Coroutine, Ts...>())
method_mods.push_back("coroutine");
pattern += " : ";

View File

@@ -201,6 +201,9 @@ test_cases:
- name: t00068
title: Context filter radius parameter test case
description:
- name: t00069
title: Coroutine methods in class diagrams
description:
Sequence diagrams:
- name: t20001
title: Basic sequence diagram test case