From 7fee17cebccf832c4f56c57c960a992e4290be14 Mon Sep 17 00:00:00 2001 From: "A. Jiang" Date: Sun, 10 Mar 2024 15:19:00 +0800 Subject: [PATCH 1/2] Make iterator, promise, and awaitable types of `generator` ADL-proof --- stl/inc/generator | 132 ++++++++++++--------- tests/std/tests/P2502R2_generator/test.cpp | 39 ++++++ 2 files changed, 113 insertions(+), 58 deletions(-) diff --git a/stl/inc/generator b/stl/inc/generator index 74bbf78df2..20a3b4d404 100644 --- a/stl/inc/generator +++ b/stl/inc/generator @@ -188,8 +188,18 @@ using _Gen_reference_t = conditional_t, _Rty&&, _Rty>; template using _Gen_yield_t = conditional_t, _Ref, const _Ref&>; +template +struct _Gen_promise_base_provider { + class _Base; +}; + +template +struct _Gen_iter_provider { + class _Iterator; +}; + template -class _Gen_promise_base { +class _Gen_promise_base_provider<_Yielded>::_Base { public: _STL_INTERNAL_STATIC_ASSERT(is_reference_v<_Yielded>); @@ -220,20 +230,21 @@ public: template requires same_as<_Gen_yield_t<_Gen_reference_t<_Rty, _Vty>>, _Yielded> _NODISCARD auto yield_value(_RANGES elements_of&&, _Unused> _Elem) noexcept { - return _Nested_awaitable<_Rty, _Vty, _Alloc>{std::move(_Elem.range)}; + using _Nested_awaitable = _Nested_awaitable_provider<_Rty, _Vty, _Alloc>::_Awaitable; + return _Nested_awaitable{std::move(_Elem.range)}; } template <_RANGES input_range _Rng, class _Alloc> requires convertible_to<_RANGES range_reference_t<_Rng>, _Yielded> _NODISCARD auto yield_value(_RANGES elements_of<_Rng, _Alloc> _Elem) { - using _Vty = _RANGES range_value_t<_Rng>; - return _Nested_awaitable<_Yielded, _Vty, _Alloc>{ - [](allocator_arg_t, _Alloc, _RANGES iterator_t<_Rng> _It, - const _RANGES sentinel_t<_Rng> _Se) -> generator<_Yielded, _Vty, _Alloc> { - for (; _It != _Se; ++_It) { - co_yield static_cast<_Yielded>(*_It); - } - }(allocator_arg, _Elem.allocator, _RANGES begin(_Elem.range), _RANGES end(_Elem.range))}; + using _Vty = _RANGES range_value_t<_Rng>; + using _Nested_awaitable = _Nested_awaitable_provider<_Yielded, _Vty, _Alloc>::_Awaitable; + return _Nested_awaitable{[](allocator_arg_t, _Alloc, _RANGES iterator_t<_Rng> _It, + const _RANGES sentinel_t<_Rng> _Se) -> generator<_Yielded, _Vty, _Alloc> { + for (; _It != _Se; ++_It) { + co_yield static_cast<_Yielded>(*_It); + } + }(allocator_arg, _Elem.allocator, _RANGES begin(_Elem.range), _RANGES end(_Elem.range))}; } void await_transform() = delete; @@ -259,11 +270,11 @@ private: template constexpr void await_suspend(coroutine_handle<_Promise> _Handle) noexcept { #ifdef __cpp_lib_is_pointer_interconvertible // TRANSITION, LLVM-48860 - _STL_INTERNAL_STATIC_ASSERT(is_pointer_interconvertible_base_of_v<_Gen_promise_base, _Promise>); + _STL_INTERNAL_STATIC_ASSERT(is_pointer_interconvertible_base_of_v<_Base, _Promise>); #endif // ^^^ no workaround ^^^ - _Gen_promise_base& _Current = _Handle.promise(); - _Current._Ptr = _STD addressof(_Val); + _Base& _Current = _Handle.promise(); + _Current._Ptr = _STD addressof(_Val); } constexpr void await_resume() const noexcept {} @@ -271,8 +282,8 @@ private: struct _Nest_info { exception_ptr _Except; - coroutine_handle<_Gen_promise_base> _Parent; - coroutine_handle<_Gen_promise_base> _Root; + coroutine_handle<_Base> _Parent; + coroutine_handle<_Base> _Root; }; struct _Final_awaiter { @@ -283,17 +294,17 @@ private: template _NODISCARD coroutine_handle<> await_suspend(coroutine_handle<_Promise> _Handle) noexcept { #ifdef __cpp_lib_is_pointer_interconvertible // TRANSITION, LLVM-48860 - _STL_INTERNAL_STATIC_ASSERT(is_pointer_interconvertible_base_of_v<_Gen_promise_base, _Promise>); + _STL_INTERNAL_STATIC_ASSERT(is_pointer_interconvertible_base_of_v<_Base, _Promise>); #endif // ^^^ no workaround ^^^ - _Gen_promise_base& _Current = _Handle.promise(); + _Base& _Current = _Handle.promise(); if (!_Current._Info) { return _STD noop_coroutine(); } - coroutine_handle<_Gen_promise_base> _Cont = _Current._Info->_Parent; - _Current._Info->_Root.promise()._Top = _Cont; - _Current._Info = nullptr; + coroutine_handle<_Base> _Cont = _Current._Info->_Parent; + _Current._Info->_Root.promise()._Top = _Cont; + _Current._Info = nullptr; return _Cont; } @@ -301,63 +312,68 @@ private: }; template - struct _Nested_awaitable { - _STL_INTERNAL_STATIC_ASSERT(same_as<_Gen_yield_t<_Gen_reference_t<_Rty, _Vty>>, _Yielded>); + struct _Nested_awaitable_provider { + struct _Awaitable { + _STL_INTERNAL_STATIC_ASSERT(same_as<_Gen_yield_t<_Gen_reference_t<_Rty, _Vty>>, _Yielded>); - _Nest_info _Nested; - generator<_Rty, _Vty, _Alloc> _Gen; + _Nest_info _Nested; + generator<_Rty, _Vty, _Alloc> _Gen; - explicit _Nested_awaitable(generator<_Rty, _Vty, _Alloc>&& _Gen_) noexcept : _Gen(_STD move(_Gen_)) {} + explicit _Awaitable(generator<_Rty, _Vty, _Alloc>&& _Gen_) noexcept : _Gen(_STD move(_Gen_)) {} - _NODISCARD bool await_ready() noexcept { - return !_Gen._Coro; - } + _NODISCARD bool await_ready() noexcept { + return !_Gen._Coro; + } - template - _NODISCARD coroutine_handle<_Gen_promise_base> await_suspend(coroutine_handle<_Promise> _Current) noexcept { + template + _NODISCARD coroutine_handle<_Base> await_suspend(coroutine_handle<_Promise> _Current) noexcept { #ifdef __cpp_lib_is_pointer_interconvertible // TRANSITION, LLVM-48860 - _STL_INTERNAL_STATIC_ASSERT(is_pointer_interconvertible_base_of_v<_Gen_promise_base, _Promise>); + _STL_INTERNAL_STATIC_ASSERT(is_pointer_interconvertible_base_of_v<_Base, _Promise>); #endif // ^^^ no workaround ^^^ - auto _Target = coroutine_handle<_Gen_promise_base>::from_address(_Gen._Coro.address()); - _Nested._Parent = coroutine_handle<_Gen_promise_base>::from_address(_Current.address()); - _Gen_promise_base& _Parent_promise = _Nested._Parent.promise(); - if (_Parent_promise._Info) { - _Nested._Root = _Parent_promise._Info->_Root; - } else { - _Nested._Root = _Nested._Parent; + auto _Target = coroutine_handle<_Base>::from_address(_Gen._Coro.address()); + _Nested._Parent = coroutine_handle<_Base>::from_address(_Current.address()); + _Base& _Parent_promise = _Nested._Parent.promise(); + if (_Parent_promise._Info) { + _Nested._Root = _Parent_promise._Info->_Root; + } else { + _Nested._Root = _Nested._Parent; + } + _Nested._Root.promise()._Top = _Target; + _Target.promise()._Info = _STD addressof(_Nested); + return _Target; } - _Nested._Root.promise()._Top = _Target; - _Target.promise()._Info = _STD addressof(_Nested); - return _Target; - } - void await_resume() { - if (_Nested._Except) { - _STD rethrow_exception(_STD move(_Nested._Except)); + void await_resume() { + if (_Nested._Except) { + _STD rethrow_exception(_STD move(_Nested._Except)); + } } - } + }; }; template - friend class _Gen_iter; + friend struct _Gen_iter_provider; // _Top and _Info are mutually exclusive, and could potentially be merged. - coroutine_handle<_Gen_promise_base> _Top = coroutine_handle<_Gen_promise_base>::from_promise(*this); - add_pointer_t<_Yielded> _Ptr = nullptr; - _Nest_info* _Info = nullptr; + coroutine_handle<_Base> _Top = coroutine_handle<_Base>::from_promise(*this); + add_pointer_t<_Yielded> _Ptr = nullptr; + _Nest_info* _Info = nullptr; }; +template +using _Gen_promise_base = _Gen_promise_base_provider<_Yielded>::_Base; + struct _Gen_secret_tag {}; template -class _Gen_iter { +class _Gen_iter_provider<_Value, _Ref>::_Iterator { public: using value_type = _Value; using difference_type = ptrdiff_t; - _Gen_iter(_Gen_iter&& _That) noexcept : _Coro{_STD exchange(_That._Coro, {})} {} + _Iterator(_Iterator&& _That) noexcept : _Coro{_STD exchange(_That._Coro, {})} {} - _Gen_iter& operator=(_Gen_iter&& _That) noexcept { + _Iterator& operator=(_Iterator&& _That) noexcept { _Coro = _STD exchange(_That._Coro, {}); return *this; } @@ -367,7 +383,7 @@ public: return static_cast<_Ref>(*_Coro.promise()._Top.promise()._Ptr); } - _Gen_iter& operator++() { + _Iterator& operator++() { _STL_ASSERT(!_Coro.done(), "Can't increment generator end iterator"); _Coro.promise()._Top.resume(); return *this; @@ -377,7 +393,7 @@ public: ++*this; } - _NODISCARD_FRIEND bool operator==(const _Gen_iter& _It, default_sentinel_t) noexcept /* strengthened */ + _NODISCARD_FRIEND bool operator==(const _Iterator& _It, default_sentinel_t) noexcept /* strengthened */ { return _It._Coro.done(); } @@ -386,7 +402,7 @@ private: template friend class generator; - explicit _Gen_iter(_Gen_secret_tag, coroutine_handle<_Gen_promise_base<_Gen_yield_t<_Ref>>> _Coro_) noexcept + explicit _Iterator(_Gen_secret_tag, coroutine_handle<_Gen_promise_base<_Gen_yield_t<_Ref>>> _Coro_) noexcept : _Coro{_Coro_} {} coroutine_handle<_Gen_promise_base<_Gen_yield_t<_Ref>>> _Coro; @@ -440,11 +456,11 @@ public: return *this; } - _NODISCARD _Gen_iter<_Value, _Ref> begin() { + _NODISCARD _Gen_iter_provider<_Value, _Ref>::_Iterator begin() { // Pre: _Coro is suspended at its initial suspend point _STL_ASSERT(_Coro, "Can't call begin on moved-from generator"); _Coro.resume(); - return _Gen_iter<_Value, _Ref>{ + return typename _Gen_iter_provider<_Value, _Ref>::_Iterator{ _Gen_secret_tag{}, coroutine_handle<_Gen_promise_base<_Gen_yield_t<_Ref>>>::from_address(_Coro.address())}; } diff --git a/tests/std/tests/P2502R2_generator/test.cpp b/tests/std/tests/P2502R2_generator/test.cpp index 9f072f1cf1..75dcad61af 100644 --- a/tests/std/tests/P2502R2_generator/test.cpp +++ b/tests/std/tests/P2502R2_generator/test.cpp @@ -10,6 +10,8 @@ #include #include #include +#include +#include #include namespace ranges = std::ranges; @@ -281,6 +283,39 @@ void arbitrary_range_test() { assert(ranges::equal(yield_arbitrary_ranges(), std::array{40, 30, 20, 10, 0, 1, 2, 3, 500, 400, 300})); } + +#ifndef _M_CEE // TRANSITION, VSO-1659496 +template +struct holder { + T t; +}; + +struct incomplete; + +void adl_proof_test() { + using validator = holder*; + auto yield_range = []() -> std::generator { + co_yield ranges::elements_of( + ranges::views::repeat(nullptr, 42) | ranges::views::transform([](std::nullptr_t) { return validator{}; })); + }; + + using R = decltype(yield_range()); + static_assert(ranges::input_range); + + using It = ranges::iterator_t; + static_assert(std::is_same_v()), It*>); + + using Promise = R::promise_type; + static_assert(std::is_same_v()), Promise*>); + + std::size_t i = 0; + for (const auto elem : yield_range()) { + ++i; + assert(elem == nullptr); + } + assert(i == 42); +} +#endif // ^^^ no workaround ^^^ #endif // ^^^ no workaround ^^^ int main() { @@ -339,5 +374,9 @@ int main() { #if !(defined(__clang__) && defined(_M_IX86)) // TRANSITION, LLVM-56507 recursive_test(); arbitrary_range_test(); + +#ifndef _M_CEE // TRANSITION, VSO-1659496 + adl_proof_test(); +#endif // ^^^ no workaround ^^^ #endif // ^^^ no workaround ^^^ } From bc79ab2cf4a869b5c7c4c6b307a00c326d101443 Mon Sep 17 00:00:00 2001 From: "A. Jiang" Date: Sun, 10 Mar 2024 15:26:18 +0800 Subject: [PATCH 2/2] Make default template argument of `elements_of` conditionally present which is consistent with the handling of `polymorphic_allocator`. --- stl/inc/ranges | 19 +++++++++++-------- 1 file changed, 11 insertions(+), 8 deletions(-) diff --git a/stl/inc/ranges b/stl/inc/ranges index 26c0e708c0..c6851ca76c 100644 --- a/stl/inc/ranges +++ b/stl/inc/ranges @@ -86,20 +86,23 @@ namespace ranges { requires (_Extent != dynamic_extent) inline constexpr auto _Compile_time_max_size> = _Extent; -#ifdef __cpp_lib_byte - using _Elements_alloc_type = byte; -#else - using _Elements_alloc_type = char; -#endif - - _EXPORT_STD template > +#if defined(__cpp_lib_byte) + _EXPORT_STD template > +#else // ^^^ defined(__cpp_lib_byte) / !defined(__cpp_lib_byte) vvv + _EXPORT_STD template +#endif // ^^^ !defined(__cpp_lib_byte) ^^^ struct elements_of { /* [[no_unique_address]] */ _Rng range; /* [[no_unique_address]] */ _Alloc allocator{}; }; - template > +#if defined(__cpp_lib_byte) + template > elements_of(_Rng&&, _Alloc = {}) -> elements_of<_Rng&&, _Alloc>; +#else // ^^^ defined(__cpp_lib_byte) / !defined(__cpp_lib_byte) vvv + template + elements_of(_Rng&&, _Alloc) -> elements_of<_Rng&&, _Alloc>; +#endif // ^^^ !defined(__cpp_lib_byte) ^^^ #endif // _HAS_CXX23 // clang-format off