Skip to content

Commit

Permalink
类型萃取的代码添加更多的注释。这部分东西永远是 c++ 里最晦涩难懂的
Browse files Browse the repository at this point in the history
  • Loading branch information
microcai committed Oct 20, 2024
1 parent 1cd2fdf commit dc0af59
Show file tree
Hide file tree
Showing 2 changed files with 112 additions and 27 deletions.
123 changes: 102 additions & 21 deletions include/ucoro/awaitable.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -72,39 +72,112 @@ namespace ucoro
inline constexpr local_storage_t<void> local_storage;

//////////////////////////////////////////////////////////////////////////
namespace detail
namespace concepts
{
//////////////////////////////////////////////////////////////////////////
// 用于判定 T 是否是一个 U<anytype> 的类型
// 比如
// is_instance_of_v<std::vector<int>,std::vector>; // true
// is_instance_of_v<std::vector<int>,std::list>; // false
template<class T, template<class...> class U>
//
// 首先定义一个接受 is_instance_of_v<typename, template_name> 这样的一个默认模板萃取
template<typename T, template<typename...> typename U>
inline constexpr bool is_instance_of_v = std::false_type{};
template<template<class...> class U, class... Vs>
inline constexpr bool is_instance_of_v<U<Vs...>,U> = std::true_type{};

template<class LocalStorage>
struct local_storage_value_type;
// 接着为 is_instance_of_v<class_type<parameters>, class_type>
// 这种定义一个偏特化,于是把符合这个模式的特殊参数给匹配到这个偏特化来了
template<template<typename...> typename class_type, typename... parameters>
inline constexpr bool is_instance_of_v<class_type<parameters...>, class_type> = std::true_type{};

template<class ValueType>
struct local_storage_value_type<local_storage_t<ValueType>>
{
typedef ValueType value_type;
};
// 然后把模板偏特化的萃取重新定义为一个 concept
template<typename T, template<typename...> typename U>
concept is_instance_of = is_instance_of_v<typename std::decay<T>::type, U>;

// 再定义一个直接用来测试 local_storage_t<> 的辅助
template<typename T>
concept is_valid_await_suspend_return_value =
std::convertible_to<T, std::coroutine_handle<>> || std::is_void_v<T> || std::is_same_v<T, bool>;
concept LocalStorage = is_instance_of<T, local_storage_t>;

// 再定义一个直接用来测试 awaitable<> 的辅助
template<typename T>
concept awaitable_type = is_instance_of<T, awaitable>;

// 再定义一个直接用来测试 awaitable_promise<> 的辅助
template<typename T>
concept awaitable_promise_type = is_instance_of<T, awaitable_promise>;

// await_suspend 有三种返回值
template<typename T>
concept is_valid_await_suspend_return_value = std::convertible_to<T, std::coroutine_handle<>> ||
std::is_void_v<T> ||
std::is_same_v<T, bool>;

// 用于判定 T 是否是一个 awaiter 的类型, 即: 拥有 await_ready,await_suspend,await_resume 成员函数的结构或类.
template<typename T>
concept is_awaiter_v = requires (T a) {
{ a.await_ready() } -> std::convertible_to<bool>;
{ a.await_ready() } -> std::same_as<bool>;
{ a.await_suspend(std::coroutine_handle<>{}) } -> is_valid_await_suspend_return_value;
{ a.await_resume() };
};

} // namespace detail
template<typename T>
concept has_operator_co_await = requires (T a)
{
{ a.operator co_await() } -> is_awaiter_v;
};

// 用于判定 T 是可以用在 co_await 后面
template<typename T>
concept is_awaitable_v = is_awaiter_v<typename std::decay_t<T>> ||
awaitable_type<T> ||
has_operator_co_await<typename std::decay_t<T>>;


template<typename T>
concept has_user_defined_await_transformer = requires (T&& a)
{
await_transformer<T>::await_transform(std::move(a));
};


} // namespace concepts

namespace traits
{
//////////////////////////////////////////////////////////////////////////
// 用于从 A = U<T> 类型里提取 T 参数
// 比如
// template_parameter_of<local_storage_t<int>, local_storage_t>; // int
// template_parameter_of<decltype(local_storage), local_storage_t>; // void
//
// 首先定义一个接受 template_parameter_of<Testee, FromTemplate> 这样的一个默认模板萃取
template<typename Testee, template<typename> typename FromTemplate>
struct template_parameter_traits;

// 接着定义一个偏特化,匹配 template_parameter_traits<模板名<参数>, 模板名>
// 这样,这个偏特化的 template_parameter_traits 就有了一个
// 名为 template_parameter 的成员类型,其定义的类型就是 _template_parameter
// 于是就把 _template_parameter 这个类型给萃取出来了
template<template<typename> typename class_template, typename _template_parameter>
struct template_parameter_traits<class_template<_template_parameter>, class_template>
{
using template_parameter = _template_parameter ;
};

// 最后,定义一个简化用法的 using 让用户的地方代码变短点
template<typename TesteeType, template<typename> typename FromTemplate>
using template_parameter_of = typename template_parameter_traits<
std::decay_t<TesteeType>, FromTemplate>::template_parameter;

// 利用 通用工具 template_parameter_of 萃取 local_storage_t<T> 里的 T
template<concepts::LocalStorage LocalStorage>
using local_storage_value_type = template_parameter_of<LocalStorage, local_storage_t>;


// 利用 通用工具 template_parameter_of 萃取 awaitable<T> 里的 T
template<concepts::awaitable_type awaitable_type>
using awaitable_return_type = template_parameter_of<awaitable_type, awaitable>;

} // namespace traits

struct debug_coro_promise
{
Expand Down Expand Up @@ -259,19 +332,27 @@ namespace ucoro
template<typename A>
auto await_transform(A&& awaiter) const
{
if constexpr ( detail::is_instance_of_v<std::decay_t<A>, local_storage_t> )
if constexpr (concepts::is_instance_of<A, local_storage_t>)
{
return local_storage_awaiter<typename detail::local_storage_value_type<std::decay_t<A>>::value_type>{this};
// 调用 co_await local_storage_t<T>
return local_storage_awaiter<traits::local_storage_value_type<A>>{this};
}
else if constexpr ( detail::is_awaiter_v<std::decay_t<A>> )
else if constexpr (concepts::is_awaitable_v<A>)
{
static_assert(std::is_rvalue_reference_v<decltype(awaiter)>, "co_await must be used on rvalue");
// 调用 co_await awaitable<T>; 或者其他有三件套的类型
static_assert(std::is_rvalue_reference_v<A&&>, "co_await must be used on rvalue");
return std::forward<A>(awaiter);
}
else
else if constexpr ( concepts::has_user_defined_await_transformer<A> )
{
// 调用 co_await 其他写了 await_transformer 的自定义类型.
// 例如包含了 asio_glue.hpp 后,就可以 co_await asio::awaitable<T>;
return await_transformer<A>::await_transform(std::move(awaiter));
}
else
{
static_assert(0, "co_await must been used on an awaitable");
}
}

std::coroutine_handle<> continuation_;
Expand Down Expand Up @@ -344,7 +425,7 @@ namespace ucoro
template<typename PromiseType>
auto await_suspend(std::coroutine_handle<PromiseType> continuation)
{
if constexpr (detail::is_instance_of_v<PromiseType, awaitable_promise>)
if constexpr (concepts::is_instance_of<PromiseType, awaitable_promise>)
{
current_coro_handle_.promise().local_ = continuation.promise().local_;
}
Expand Down
16 changes: 10 additions & 6 deletions tests/test3/test.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -7,15 +7,19 @@ int main(int argc, char **argv)
using CallbackAwaiterType0 = ucoro::CallbackAwaiter<void, decltype([](auto h) {}) >;
using CallbackAwaiterType1 = ucoro::CallbackAwaiter<int, decltype([](auto h) {}) > ;

static_assert(ucoro::detail::is_instance_of_v<ucoro::local_storage_t<void>, ucoro::local_storage_t>, "not a local_storage_t");
static_assert(ucoro::concepts::is_instance_of_v<ucoro::local_storage_t<void>, ucoro::local_storage_t>, "not a local_storage_t");

static_assert(ucoro::detail::is_awaiter_v < CallbackAwaiterType0 >, "not a coroutine");
static_assert(ucoro::detail::is_awaiter_v < CallbackAwaiterType1 >, "not a coroutine");
using local_storage_template_parameter = ucoro::traits::template_parameter_of<decltype(ucoro::local_storage), ucoro::local_storage_t>;

static_assert(ucoro::detail::is_awaiter_v < ucoro::awaitable<void> >, "not a coroutine");
static_assert(ucoro::detail::is_awaiter_v < ucoro::awaitable<int> >, "not a coroutine");
static_assert(std::is_void_v<local_storage_template_parameter>, "local_storage is not local_storage_t<void>");

static_assert(!ucoro::detail::is_awaiter_v < int >, "not a coroutine");
static_assert(ucoro::concepts::is_awaiter_v < CallbackAwaiterType0 >, "not a coroutine");
static_assert(ucoro::concepts::is_awaiter_v < CallbackAwaiterType1 >, "not a coroutine");

static_assert(ucoro::concepts::is_awaiter_v < ucoro::awaitable<void> >, "not a coroutine");
static_assert(ucoro::concepts::is_awaiter_v < ucoro::awaitable<int> >, "not a coroutine");

static_assert(!ucoro::concepts::is_awaiter_v < int >, "not a coroutine");

return 0;
}

0 comments on commit dc0af59

Please sign in to comment.