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 0bf511d commit f5fd4bf
Show file tree
Hide file tree
Showing 2 changed files with 56 additions and 24 deletions.
68 changes: 50 additions & 18 deletions include/ucoro/awaitable.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -75,25 +75,26 @@ 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>;

template<typename T>
concept is_valid_await_suspend_return_value =
Expand All @@ -102,12 +103,39 @@ namespace ucoro
// 用于判定 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>> || has_operator_co_await<typename std::decay_t<T>> || is_instance_of_v<typename std::decay_t<T>, awaitable>;

} // namespace concepts

namespace traits
{
template<typename LocalStorage>
struct local_storage_value_type_trait;

template<typename ValueType>
struct local_storage_value_type_trait<local_storage_t<ValueType>>
{
using value_type = ValueType ;
};

// 用来提取 local_storage_t<T> 里的那个 T.
template<typename LocalStorage>
using local_storage_value_type = typename local_storage_value_type_trait<std::decay_t<LocalStorage>>::value_type;

} // namespace traits

struct debug_coro_promise
{
Expand Down Expand Up @@ -262,19 +290,23 @@ 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};
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");
return std::forward<A>(awaiter);
}
else
else if constexpr ( requires { await_transformer<A>::await_transform; } )
{
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 @@ -347,7 +379,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
12 changes: 6 additions & 6 deletions tests/test3/test.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -7,15 +7,15 @@ 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");
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::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(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::detail::is_awaiter_v < int >, "not a coroutine");
static_assert(!ucoro::concepts::is_awaiter_v < int >, "not a coroutine");

return 0;
}

0 comments on commit f5fd4bf

Please sign in to comment.