diff --git a/doc/api_reference.md b/doc/api_reference.md index c9ab3b5fd..50f9f711d 100644 --- a/doc/api_reference.md +++ b/doc/api_reference.md @@ -1194,6 +1194,20 @@ pass the state in as additional arguments. ## Other +### `async_resource` + +Provides _async RAII_ programming idiom. `async_resource_ptr` is like +a `std::unique_ptr` except that `async_resource_ptr<>`'s destructor destroys +its `T` asynchronously, in constrast to `unique_ptr<>`'s destructor's +synchronous invocation of `delete p_`. `async_resource` removes the need for +manual lifetime management of a `Resource` protected by an `async_scope` and +should be preferred over `async_scope`, if there is a need for one. + +1. explicitly supports parent / child relationship modeled as a tree +2. allows for subtree removal / swap +3. each `Resource` managed by `async_resource` has an associated, managed _inner_ + `async_scope` without an explicit option to `join()` + ### `async_scope` A place to safely spawn work such that it can be joined later. diff --git a/include/unifex/async_destroy.hpp b/include/unifex/async_destroy.hpp new file mode 100644 index 000000000..d4b024172 --- /dev/null +++ b/include/unifex/async_destroy.hpp @@ -0,0 +1,64 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ +#pragma once + +#include +#include + +#include + +namespace unifex { +namespace _async_destroy { + +inline constexpr struct _destroy_fn { +private: + template + static auto try_invoke_member(Resource& resource, int) noexcept( + noexcept(resource.destroy())) -> decltype(resource.destroy()) { + return resource.destroy(); + } + + template + [[deprecated( + "Can't async_destroy an object with no async_destroy customization; add " + "a no-op if that's what's intended.")]] // + static auto + try_invoke_member(Resource&, double) noexcept { + return just(); + } + +public: + template(typename Resource) // + (requires tag_invocable<_destroy_fn, Resource&>) // + constexpr auto + operator()(Resource& resource) const + noexcept(is_nothrow_tag_invocable_v<_destroy_fn, Resource&>) { + return tag_invoke(_destroy_fn{}, resource); + } + + template(typename Resource) // + (requires(!tag_invocable<_destroy_fn, Resource&>)) // + constexpr auto + operator()(Resource& resource) const + noexcept(noexcept(try_invoke_member(resource, 1))) { + return try_invoke_member(resource, 1); + } +} async_destroy{}; +} // namespace _async_destroy +using _async_destroy::async_destroy; +} // namespace unifex + +#include diff --git a/include/unifex/async_resource.hpp b/include/unifex/async_resource.hpp new file mode 100644 index 000000000..2375d0cef --- /dev/null +++ b/include/unifex/async_resource.hpp @@ -0,0 +1,473 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ +#pragma once + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include +#include +#include +#include +#include + +#include + +namespace unifex { +namespace _async_resource { +using Scope = v2::async_scope; +/* +* `ResourceFactory` takes an `async_scope_ref` that refers to the *inner* scope +* `ResourceFactory(async_scope_ref)` returns a `Resource` +* the `Resource` is stored next to the *inner* scope in a heap-allocated + `_container` +* `async_resource` takes an *outer* scope ref, returns a move-only + `async_resource_ptr` +* the returned `async_resource_ptr` is a *unique_ptr*-like object +* creation of the `Resource`, it's asynchronous destruction and `join()` of + the *inner* scope is spawned on the *outer* scope. Lifetime is controlled by + `async_resource_ptr`. +* `async_destroy` is a CPO algorithm that by default does nothing. Customizable + through `tag_invoke` to do async destruction if `Resource` provides a + `destroy() -> Sender` +* TBD customizations: + - for a range of T, async_destroy(range) is just async_destroy on each T, + followed by clearing the range + - for an optional, async_destroy(opt) is a no-op on empty + - async_destroy(*opt) on non-empty, followed by .reset() + - for a unique_ptr, async_destroy is async_destroy(*ptr) followed by + .reset(), unless it's nullptr, in which case no-op +*/ + +class _container_base { +protected: + enum class state_t : char { + ALLOCATED, + DESTRUCT_SPAWNED, + CONSTRUCTED, + DESTROYING, + }; + // type-erased _container func pointers + void (*destruct_this_resource_)(_container_base*) noexcept; + any_sender_of<> (*schedule_destruct_)(_container_base*, _container_base*); + any_sender_of<> (*destroy_this_resource_)(_container_base*); + void (*deleter_)(_container_base*) noexcept; + UNIFEX_NO_UNIQUE_ADDRESS manual_lifetime scope_; + // doubly linked list + _container_base* children_; + _container_base* next_; + _container_base* parent_; + // means destruction should start + async_manual_reset_event destructionEvt_; + std::mutex mutex_; + std::atomic state_; + + task close_child_scopes() { + auto* adopted = [&]() noexcept { + // terminate when mutex_ fails to lock + std::lock_guard lock{mutex_}; + return std::exchange(children_, nullptr); + }(); + { + for (auto* child = adopted; child != nullptr; child = child->next_) { + co_await child->close_scope(); + } + } + { + std::lock_guard lock{mutex_}; + auto** tail = &children_; + + while (*tail) { + tail = &(*tail)->next_; + } + + *tail = adopted; + } + } + + // asynchronous part of `Resource` & children destruction + task destroy_resource() { + co_await destroy_this_resource_(this); + for (auto* child = children_; child != nullptr; child = child->next_) { + co_await child->destroy_resource(); + } + } + + // synchronous part of `Resource` & children destruction: deallocation + void destruct_container() noexcept { + if (state() == state_t::DESTROYING) { + destruct_this_resource_(this); + scope_.destruct(); + _container_base* child = children_; + while (child != nullptr) { + auto next = child->next_; + child->destruct_container(); + child = next; + } + } + deleter_(this); + } + + void deregister_child(_container_base* child) noexcept { + std::lock_guard lock{mutex_}; + for (_container_base** head = &children_; *head; head = &((*head)->next_)) { + if (*head == child) { + *head = (*head)->next_; + return; + } + } + UNIFEX_ASSERT(false); // not found + } + + // pre-spawned async_destroy CPO prior to _container construction + auto destruct_impl() noexcept { + return let_value(destructionEvt_.async_wait(), [this]() noexcept { + bool value = true; + switch (state()) { + case state_t::ALLOCATED: + // scheduled destruct moved the state from ALLOCATED + UNIFEX_ASSERT(false); + break; + case state_t::CONSTRUCTED: value = false; break; + case state_t::DESTRUCT_SPAWNED: + if (!parent_) { + deleter_(this); + } + break; + case state_t::DESTROYING: + // parent tearing down + break; + } + return just_void_or_done(value) | + let_done( + [this]() noexcept(noexcept(let_done( + async_destroy(UNIFEX_DECLVAL(_container_base&)), just))) { + // TODO remove let_done once any_sender_of<>(s) are removed + return let_done(async_destroy(*this), just); + }); + }); + } + +public: + _container_base( + void (*destruct_this_resource)(_container_base*) noexcept, + any_sender_of<> (*schedule_destruct)(_container_base*, _container_base*), + any_sender_of<> (*destroy_this_resource)(_container_base*), + void (*deleter)(_container_base*) noexcept) noexcept + : destruct_this_resource_(destruct_this_resource) + , schedule_destruct_(schedule_destruct) + , destroy_this_resource_(destroy_this_resource) + , deleter_(deleter) + , children_(nullptr) + , next_(nullptr) + , parent_(nullptr) + , state_(state_t::ALLOCATED) {} + + state_t state() const noexcept { + // TODO optimize `memory_order_seq_cst` + return state_.load(std::memory_order_seq_cst); + } + + // recursively join() self and child scopes + task close_scope() noexcept { + if (auto expected = state_t::CONSTRUCTED; state_.compare_exchange_strong( + expected, state_t::DESTROYING, std::memory_order_seq_cst)) { + destructionEvt_.set(); + co_await when_all(scope_.get().join(), close_child_scopes()); + } + } + + // happens when _ptr is dropped (non-nullptr) + // destroys from the root of tree post-order (objects created later might have + // references to objects created earlier, opposite is not true): + // 1. close all scopes + // 2. async destroy all resources + // 3. delete all containers + task destroy() noexcept { + if (parent_) { + parent_->deregister_child(this); + } + co_await close_scope(); + co_await destroy_resource(); + destruct_container(); + } + + // accessor to async_destroy CPO, essentially: + // on(async_destroy) via type erased schedule_destruct_ + any_sender_of<> destruct(_container_base* parent) { + static_assert( + !sender_traits::sends_done, + "destruct() must be unstoppable"); + return schedule_destruct_(this, parent); + } + + void register_child(_container_base* child) noexcept { + // terminate if cannot lock mutex_ + std::lock_guard lock{mutex_}; + child->parent_ = this; + child->next_ = std::exchange(children_, child); + } + + task handle_construction_failure() noexcept { + switch (state()) { + case state_t::ALLOCATED: deleter_(this); break; + case state_t::DESTRUCT_SPAWNED: destructionEvt_.set(); break; + case state_t::CONSTRUCTED: + if (parent_ && parent_->state() == state_t::DESTROYING) { + co_await close_scope(); + // bubble up cancellation, prevent returning an empty _ptr + co_await just_done(); + } + break; + case state_t::DESTROYING: + // do nothing, ownership handed over to destroy() + break; + } + } +}; + +class async_scope_ref final { + Scope* scope_; + _container_base* container_; + +public: + /*implicit*/ async_scope_ref(Scope& scope) noexcept + : async_scope_ref(scope, nullptr) {} // unmanaged scope + + explicit async_scope_ref(Scope& scope, _container_base* container) noexcept + : scope_(&scope) + , container_(container) {} + + void register_child(_container_base* child) { + UNIFEX_ASSERT(child != nullptr); + spawn_detached(child->destruct(container_), *scope_); + } + + template + [[nodiscard]] auto nest(Sender&& sender) noexcept( + noexcept(scope_->nest(static_cast(sender)))) + -> decltype(scope_->nest(static_cast(sender))) { + return scope_->nest(static_cast(sender)); + } + + friend bool operator==(async_scope_ref lhs, async_scope_ref rhs) noexcept { + return lhs.scope_ == rhs.scope_ && lhs.container_ == rhs.container_; + } + + friend bool operator!=(async_scope_ref lhs, async_scope_ref rhs) noexcept { + return !(lhs == rhs); + } +}; + +template +struct _container final : _container_base { + explicit _container(Scheduler scheduler) noexcept( + std::is_nothrow_move_constructible_v) + : _container_base{destruct_this_resource, // + schedule_destruct, // + destroy_this_resource, // + deleter}, + scheduler_(std::move(scheduler)) {} + + template + auto schedule_construct(Sender&& factory) noexcept { + return on( + scheduler_, + sequence( + just_from([&]() noexcept { scope_.construct(); }), + std::move(factory), + just_from([&]() noexcept { return set_constructed(); }))); + } + + template + auto construct(ResourceFactory&& factory) noexcept { + if constexpr (noexcept( + factory(std::declval(), scheduler_))) { + return schedule_construct( + just_from([factory = static_cast(factory), + this]() mutable noexcept { + resource_.construct_with([&]() noexcept { + return factory(async_scope_ref{scope_.get(), this}, scheduler_); + }); + })); + } else { + return schedule_construct(let_error( + just_from([factory = static_cast(factory), + this]() mutable { + resource_.construct_with([&]() { + return factory(async_scope_ref{scope_.get(), this}, scheduler_); + }); + }), + [&](auto e) { + return sequence( + scope_.get().join(), + just_from([&]() noexcept { scope_.destruct(); }), + just_error(e)); + })); + } + } + + template + auto construct_as_sender(ResourceFactory&& factory) noexcept { + return schedule_construct(let_error( + defer( + [factory = static_cast(factory), this]() mutable noexcept( + noexcept( + factory(std::declval(), scheduler_))) { + return then( + factory(async_scope_ref{scope_.get(), this}, scheduler_), + [&](auto&&... args) noexcept(std::is_nothrow_constructible_v< + Resource, + decltype(args)...>) { + resource_.construct(static_cast(args)...); + }); + }), + [&](auto e) { + return sequence( + scope_.get().join(), + just_from([&]() noexcept { scope_.destruct(); }), + just_error(e)); + })); + } + + async_resource_ptr ptr() noexcept { + return {&resource_.get(), &destructionEvt_}; + } + +private: + _container* set_constructed() noexcept { + [[maybe_unused]] auto old = + state_.exchange(state_t::CONSTRUCTED, std::memory_order_seq_cst); + UNIFEX_ASSERT(old == state_t::DESTRUCT_SPAWNED); + return this; + } + + UNIFEX_NO_UNIQUE_ADDRESS Scheduler scheduler_; + UNIFEX_NO_UNIQUE_ADDRESS manual_lifetime resource_; + + // + // passed to _container_base as func pointers + // + static void destruct_this_resource(_container_base* base) noexcept { + UNIFEX_ASSERT(base != nullptr); + auto& self = static_cast<_container&>(*base); + self.resource_.destruct(); + } + + // TODO potential improvmement to avoid the any_sender_of<> static method: + // 1. add operation states to the _container + // 2. maybe a union + static any_sender_of<> + schedule_destruct(_container_base* base, _container_base* parent) { + UNIFEX_ASSERT(base != nullptr); + auto& self = static_cast<_container&>(*base); + return sequence( + just_from([&self, parent]() noexcept { + [[maybe_unused]] auto old = self.state_.exchange( + state_t::DESTRUCT_SPAWNED, std::memory_order_seq_cst); + UNIFEX_ASSERT(old == state_t::ALLOCATED); + if (parent) { + parent->register_child(&self); + } + }), + unstoppable(on(self.scheduler_, self.destruct_impl()))); + } + + static any_sender_of<> destroy_this_resource(_container_base* base) { + UNIFEX_ASSERT(base != nullptr); + auto& self = static_cast<_container&>(*base); + return with_query_value( + let_done(async_destroy(self.resource_.get()), just), + get_scheduler, + self.scheduler_); + } + + static void deleter(_container_base* base) noexcept { + UNIFEX_ASSERT(base != nullptr); + auto* self = static_cast<_container*>(base); + delete self; + } +}; // namespace unifex + +template < + typename Resource = void, + typename Scheduler, + typename ResourceFactory> +auto make_async_resource( + Scheduler scheduler, + async_scope_ref outerScope, + ResourceFactory&& factory) { + using factory_result_t = + decltype(factory(std::declval(), scheduler)); + auto result = []() noexcept { + if constexpr (sender) { + static_assert( + !std::is_void_v, + "Sender returning ResourceFactory must specify Resource"); + return single_overload{}; + } else { + static_assert(same_as, "must not specify Resource"); + return single_overload{}; + } + }; + using resource_t = typename decltype(result())::type; + // TODO support passing allocators + auto* container = new _container{scheduler}; + auto construct = [&]() { + if constexpr (sender) { + return container->construct_as_sender( + static_cast(factory)); + } else { + return container->construct(static_cast(factory)); + } + }(); + outerScope.register_child(container); + return finally( + nest(std::move(construct), outerScope), + container->handle_construction_failure()) | + then([](auto* container) noexcept { return container->ptr(); }); +} +} // namespace _async_resource +using _async_resource::async_scope_ref; +using _async_resource::make_async_resource; +} // namespace unifex + +#include diff --git a/include/unifex/async_resource_ptr.hpp b/include/unifex/async_resource_ptr.hpp new file mode 100644 index 000000000..24b77f533 --- /dev/null +++ b/include/unifex/async_resource_ptr.hpp @@ -0,0 +1,127 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ +#pragma once + +#include +#include + +#include +#include +#include +#include + +namespace unifex { +namespace _async_resource_ptr { + +template +struct [[nodiscard]] async_resource_ptr final { + template + friend struct async_resource_ptr; + + // @deprecated transitional adapter for v2/detached_spawn + explicit async_resource_ptr(std::unique_ptr resource) noexcept + : resource_(resource.release()) + , evt_(nullptr) { + UNIFEX_ASSERT(resource_ != nullptr); + UNIFEX_ASSERT(evt_ == nullptr); + } + + async_resource_ptr( + Resource* resource, unifex::async_manual_reset_event* evt) noexcept + : resource_(resource) + , evt_(evt) { + UNIFEX_ASSERT(resource_ != nullptr); + UNIFEX_ASSERT(evt_ != nullptr); + } + + template < + typename Resource2, + std::enable_if_t< + !std::is_same_v && + std::is_convertible_v>* = nullptr> + /* implicit */ async_resource_ptr(async_resource_ptr h) noexcept + : resource_(std::exchange(h.resource_, nullptr)) + , evt_(std::exchange(h.evt_, nullptr)) {} + + async_resource_ptr(async_resource_ptr&& h) noexcept + : resource_(std::exchange(h.resource_, nullptr)) + , evt_(std::exchange(h.evt_, nullptr)) {} + + async_resource_ptr(const async_resource_ptr& h) noexcept = delete; + + async_resource_ptr() noexcept : resource_(nullptr), evt_(nullptr) {} + + /* implicit */ async_resource_ptr(std::nullptr_t) noexcept + : resource_(nullptr) + , evt_(nullptr) {} + + ~async_resource_ptr() noexcept { reset(); } + + Resource* operator->() const noexcept { return resource_; } + + Resource& operator*() const noexcept { return *resource_; } + + async_resource_ptr& operator=(async_resource_ptr h) noexcept { + swap(h); + return *this; + } + + explicit operator bool() const noexcept { return resource_ != nullptr; } + + Resource* get() const noexcept { return resource_; } + + void reset() noexcept { + if (evt_) { + evt_->set(); + evt_ = nullptr; + } else { + delete resource_; + } + resource_ = nullptr; + } + + void swap(async_resource_ptr& other) noexcept { + std::swap(resource_, other.resource_); + std::swap(evt_, other.evt_); + } + + friend bool operator==( + const async_resource_ptr& a, const async_resource_ptr& b) noexcept { + return a.resource_ == b.resource_ && a.evt_ == b.evt_; + } + + friend bool operator!=( + const async_resource_ptr& a, const async_resource_ptr& b) noexcept { + return !(a == b); + } + +private: + Resource* resource_; + unifex::async_manual_reset_event* evt_; +}; +} // namespace _async_resource_ptr +using _async_resource_ptr::async_resource_ptr; +} // namespace unifex + +namespace std { +template +struct hash> { + std::size_t + operator()(const unifex::async_resource_ptr& p) noexcept { + return std::hash{}(p.get()); + } +}; +} // namespace std diff --git a/test/async_destroy_test.cpp b/test/async_destroy_test.cpp new file mode 100644 index 000000000..8680bcc8f --- /dev/null +++ b/test/async_destroy_test.cpp @@ -0,0 +1,96 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include +#include +#include + +#include +#include + +using namespace unifex; + +namespace { +struct NoexceptResource { + bool destroyed{false}; + auto destroy() noexcept { + return just_from([&]() { destroyed = true; }); + } +}; + +struct ThrowingResource { + bool destroyed{false}; + auto destroy() noexcept(false) { + return just_from([&]() { destroyed = true; }); + } +}; + +struct NoexceptTagResource { + bool destroyed{false}; + bool tagged{false}; + auto destroy() noexcept { + return just_from([&]() { destroyed = true; }); + } + +private: + // takes precedence over destroy() + friend auto + tag_invoke(tag_t, NoexceptTagResource& self) noexcept { + return just_from([&]() { self.tagged = true; }); + } +}; + +struct ThrowingTagResource { + bool destroyed{false}; + bool tagged{false}; + auto destroy() noexcept { + return just_from([&]() { destroyed = true; }); + } + +private: + // takes precedence over destroy() + friend auto + tag_invoke(tag_t, ThrowingTagResource& self) noexcept(false) { + return just_from([&]() noexcept { self.tagged = true; }); + } +}; + +TEST(AsyncDestroyTest, member) { + NoexceptResource r; + static_assert(noexcept(async_destroy(r))); + sync_wait(async_destroy(r)); + ASSERT_TRUE(r.destroyed); + + ThrowingResource t; + static_assert(!noexcept(async_destroy(t))); + sync_wait(async_destroy(t)); + ASSERT_TRUE(t.destroyed); +} + +TEST(AsyncDestroyTest, tag_invoke) { + NoexceptTagResource r; + static_assert(noexcept(async_destroy(r))); + sync_wait(async_destroy(r)); + ASSERT_FALSE(r.destroyed); + ASSERT_TRUE(r.tagged); + + ThrowingTagResource t; + static_assert(!noexcept(async_destroy(t))); + sync_wait(async_destroy(t)); + ASSERT_FALSE(t.destroyed); + ASSERT_TRUE(t.tagged); +} +} // namespace diff --git a/test/async_resource_cancel_task_async_destroy_test.cpp b/test/async_resource_cancel_task_async_destroy_test.cpp new file mode 100644 index 000000000..0c4a3acd8 --- /dev/null +++ b/test/async_resource_cancel_task_async_destroy_test.cpp @@ -0,0 +1,63 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +struct CancelTaskDestroyResource : public ResourceBase { + using ResourceBase::ResourceBase; + + task destroy() noexcept { + co_await just_from([&]() noexcept { destroyCalled_ = true; }); + } + + ~CancelTaskDestroyResource() noexcept { EXPECT_TRUE(destroyCalled_); } + +private: + bool destroyCalled_{false}; +}; + +task cancel_task_async_destroy(AsyncResourceTest* f) { + // drop immediately + (void)co_await make_async_resource( + f->ctx.get_scheduler(), f->outerScope, [f](auto, auto) noexcept { + return CancelTaskDestroyResource{&(f->objectCount)}; + }); + // TODO v2::async_scope does not support cleanup() + co_await f->outerScope.join(); +} + +} // namespace + +TEST_F(AsyncResourceTest, cancel_task_async_destroy) { + sync_wait(cancel_task_async_destroy(this)); +} +#endif diff --git a/test/async_resource_closed_scope_test.cpp b/test/async_resource_closed_scope_test.cpp new file mode 100644 index 000000000..f0b846c2a --- /dev/null +++ b/test/async_resource_closed_scope_test.cpp @@ -0,0 +1,55 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +task closed_scope(AsyncResourceTest* f) { + // no work can be spawned + co_await f->outerScope.join(); + // drop immediately + (void)co_await let_done( + make_async_resource( + f->ctx.get_scheduler(), + f->outerScope, + [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }), + []() noexcept { return just(async_resource_ptr{}); }); + co_await f->outerScope.join(); +} + +} // namespace + +TEST_F(AsyncResourceTest, closed_scope) { + sync_wait(closed_scope(this)); +} +#endif diff --git a/test/async_resource_drop_child_before_parent_test.cpp b/test/async_resource_drop_child_before_parent_test.cpp new file mode 100644 index 000000000..25dc54d6d --- /dev/null +++ b/test/async_resource_drop_child_before_parent_test.cpp @@ -0,0 +1,58 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +task drop_child_before_parent(AsyncResourceTest* f) { + { + auto parent = co_await make_async_resource< + SingleNestingResource>( // parent + f->ctx.get_scheduler(), + f->outerScope, + [f](auto scope, auto scheduler) noexcept { + return make_async_resource( // child + scheduler, + scope, + [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + }); + parent->drop_child(); + } + co_await f->outerScope.join(); // grandparent +} +} // namespace + +TEST_F(AsyncResourceTest, drop_child_before_parent) { + sync_wait(drop_child_before_parent(this)); +} +#endif diff --git a/test/async_resource_great_grandparent_test.cpp b/test/async_resource_great_grandparent_test.cpp new file mode 100644 index 000000000..47325bc2a --- /dev/null +++ b/test/async_resource_great_grandparent_test.cpp @@ -0,0 +1,62 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +task great_grandparent(AsyncResourceTest* f) { + // drop immediately + (void)co_await make_async_resource>>( // grandparent + f->ctx.get_scheduler(), + f->outerScope, + [f](auto scope, auto scheduler) noexcept { + return make_async_resource< + SingleNestingResource>( // parent + scheduler, + scope, + [f](auto scope, auto scheduler) noexcept { + return make_async_resource( // child + scheduler, + scope, + [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + }); + }); + co_await f->outerScope.join(); // great grandparent +} + +} // namespace + +TEST_F(AsyncResourceTest, great_grandparent) { + sync_wait(great_grandparent(this)); +} +#endif diff --git a/test/async_resource_in_place_test.cpp b/test/async_resource_in_place_test.cpp new file mode 100644 index 000000000..23d1f32a8 --- /dev/null +++ b/test/async_resource_in_place_test.cpp @@ -0,0 +1,72 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include + +# include +# include +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +class NoCopyNoMoveResource { + int i_; + double d_; + std::string s_; + +public: + NoCopyNoMoveResource(int i, double d, std::string s) noexcept + : i_(i) + , d_(d) + , s_(std::move(s)) {} + + NoCopyNoMoveResource(const NoCopyNoMoveResource&) = delete; + NoCopyNoMoveResource(NoCopyNoMoveResource&&) = delete; + + auto args() const noexcept { return std::make_tuple(i_, d_, s_); } + + // suppress deprecation warning - noop + auto destroy() noexcept { return just(); } +}; + +task in_place(AsyncResourceTest* f) { + { + auto r = co_await make_async_resource( + f->ctx.get_scheduler(), f->outerScope, [](auto, auto) noexcept { + return just(42, 42.42, "Fish"); + }); + EXPECT_EQ(r->args(), std::make_tuple(42, 42.42, "Fish")); + } + co_await f->outerScope.join(); +} + +} // namespace + +TEST_F(AsyncResourceTest, in_place) { + sync_wait(in_place(this)); +} +#endif diff --git a/test/async_resource_managed_any_sender_async_destroy_test.cpp b/test/async_resource_managed_any_sender_async_destroy_test.cpp new file mode 100644 index 000000000..6f4bf0d33 --- /dev/null +++ b/test/async_resource_managed_any_sender_async_destroy_test.cpp @@ -0,0 +1,63 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +struct ManagedAnySenderDestroyResource : public ResourceBase { + using ResourceBase::ResourceBase; + + any_sender_of<> destroy() noexcept { + return just_from([&]() noexcept { destroyCalled_ = true; }); + } + + ~ManagedAnySenderDestroyResource() noexcept { EXPECT_TRUE(destroyCalled_); } + +private: + bool destroyCalled_{false}; +}; + +task managed_any_sender_async_destroy(AsyncResourceTest* f) { + // drop immediately + (void)co_await make_async_resource( + f->ctx.get_scheduler(), f->outerScope, [f](auto, auto) noexcept { + return ManagedAnySenderDestroyResource{&(f->objectCount)}; + }); + co_await f->outerScope.join(); +} + +} // namespace + +TEST_F(AsyncResourceTest, managed_any_sender_async_destroy) { + sync_wait(managed_any_sender_async_destroy(this)); +} +#endif diff --git a/test/async_resource_managed_async_destroy_test.cpp b/test/async_resource_managed_async_destroy_test.cpp new file mode 100644 index 000000000..4a0b0c6b9 --- /dev/null +++ b/test/async_resource_managed_async_destroy_test.cpp @@ -0,0 +1,62 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +struct ManagedDestroyResource : public ResourceBase { + using ResourceBase::ResourceBase; + + auto destroy() noexcept { + return just_from([&]() noexcept { destroyCalled_ = true; }); + } + + ~ManagedDestroyResource() noexcept { EXPECT_TRUE(destroyCalled_); } + +private: + bool destroyCalled_{false}; +}; + +task managed_async_destroy(AsyncResourceTest* f) { + // drop immediately + (void)co_await make_async_resource( + f->ctx.get_scheduler(), f->outerScope, [f](auto, auto) noexcept { + return ManagedDestroyResource{&(f->objectCount)}; + }); + co_await f->outerScope.join(); +} + +} // namespace + +TEST_F(AsyncResourceTest, managed_async_destroy) { + sync_wait(managed_async_destroy(this)); +} +#endif diff --git a/test/async_resource_managed_task_async_destroy_test.cpp b/test/async_resource_managed_task_async_destroy_test.cpp new file mode 100644 index 000000000..495816f06 --- /dev/null +++ b/test/async_resource_managed_task_async_destroy_test.cpp @@ -0,0 +1,62 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +struct ManagedTaskDestroyResource : public ResourceBase { + using ResourceBase::ResourceBase; + + task destroy() noexcept { + co_await just_from([&]() noexcept { destroyCalled_ = true; }); + } + + ~ManagedTaskDestroyResource() noexcept { EXPECT_TRUE(destroyCalled_); } + +private: + bool destroyCalled_{false}; +}; + +task managed_task_async_destroy(AsyncResourceTest* f) { + // drop immediately + (void)co_await make_async_resource( + f->ctx.get_scheduler(), f->outerScope, [f](auto, auto) noexcept { + return ManagedTaskDestroyResource{&(f->objectCount)}; + }); + co_await f->outerScope.join(); +} + +} // namespace + +TEST_F(AsyncResourceTest, managed_task_async_destroy) { + sync_wait(managed_task_async_destroy(this)); +} +#endif diff --git a/test/async_resource_no_async_destroy_test.cpp b/test/async_resource_no_async_destroy_test.cpp new file mode 100644 index 000000000..e49a2e1ff --- /dev/null +++ b/test/async_resource_no_async_destroy_test.cpp @@ -0,0 +1,49 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +task no_async_destroy(AsyncResourceTest* f) { + // drop immediately + (void)co_await make_async_resource( + f->ctx.get_scheduler(), f->outerScope, [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + co_await f->outerScope.join(); +} + +} // namespace + +TEST_F(AsyncResourceTest, no_async_destroy) { + sync_wait(no_async_destroy(this)); +} +#endif diff --git a/test/async_resource_parent_drops_child_test.cpp b/test/async_resource_parent_drops_child_test.cpp new file mode 100644 index 000000000..17d7845ee --- /dev/null +++ b/test/async_resource_parent_drops_child_test.cpp @@ -0,0 +1,57 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +task parent_drops_child(AsyncResourceTest* f) { + // drop immediately + (void)co_await make_async_resource< + SingleNestingResource>( // parent + f->ctx.get_scheduler(), + f->outerScope, + [f](auto scope, auto scheduler) noexcept { + return make_async_resource( // child + scheduler, + scope, + [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + }); + co_await f->outerScope.join(); // grandparent +} + +} // namespace + +TEST_F(AsyncResourceTest, parent_drops_child) { + sync_wait(parent_drops_child(this)); +} +#endif diff --git a/test/async_resource_plain_sender_test.cpp b/test/async_resource_plain_sender_test.cpp new file mode 100644 index 000000000..17ac14194 --- /dev/null +++ b/test/async_resource_plain_sender_test.cpp @@ -0,0 +1,52 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +auto plain_sender(AsyncResourceTest* f) { + return sequence( + make_async_resource( + f->ctx.get_scheduler(), + f->outerScope, + [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }) | + then([](auto&&...) noexcept {}), + f->outerScope.join()); +} + +} // namespace + +TEST_F(AsyncResourceTest, plain_sender) { + sync_wait(plain_sender(this)); +} +#endif diff --git a/test/async_resource_ptr_test.cpp b/test/async_resource_ptr_test.cpp new file mode 100644 index 000000000..b0caf7c71 --- /dev/null +++ b/test/async_resource_ptr_test.cpp @@ -0,0 +1,99 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#include + +#include +#include + +#include + +using namespace unifex; + +namespace { +struct Dummy {}; + +TEST(AsyncResourcePtrTest, equality) { + Dummy r; + async_manual_reset_event evt; + // default + async_resource_ptr default1; + async_resource_ptr default2; + ASSERT_EQ(default1, default2); + // nullptr_t + async_resource_ptr null1{nullptr}; + async_resource_ptr null2{nullptr}; + ASSERT_EQ(null1, null2); + // dummy + async_resource_ptr dummy1{&r, &evt}; + async_resource_ptr dummy2{&r, &evt}; + ASSERT_EQ(dummy1, dummy2); + + ASSERT_EQ(null1, default1); +} + +TEST(AsyncResourcePtrTest, unequality) { + Dummy r1; + Dummy r2; + async_manual_reset_event evt1; + async_manual_reset_event evt2; + // dummy + async_resource_ptr dummy1{&r1, &evt1}; + async_resource_ptr dummy2{&r2, &evt1}; + ASSERT_NE(dummy1, dummy2); + + async_resource_ptr dummy3{&r1, &evt1}; + async_resource_ptr dummy4{&r1, &evt2}; + ASSERT_NE(dummy3, dummy4); +} + +TEST(AsyncResourcePtrTest, swap) { + Dummy r; + async_manual_reset_event evt; + // dummy + const async_resource_ptr dummy{&r, &evt}; + async_resource_ptr dummy1{&r, &evt}; + async_resource_ptr dummy2; + ASSERT_EQ(dummy, dummy1); + ASSERT_NE(dummy1, dummy2); + + // swap + std::swap(dummy1, dummy2); + ASSERT_EQ(dummy, dummy2); + ASSERT_NE(dummy, dummy1); +} + +TEST(AsyncResourcePtrTest, hash) { + using dummy_hash = std::hash>; + Dummy r1; + Dummy r2; + async_manual_reset_event evt1; + async_manual_reset_event evt2; + // hash equal + async_resource_ptr dummy1{&r1, &evt1}; + async_resource_ptr dummy2{&r1, &evt2}; + ASSERT_NE(dummy1, dummy2); + ASSERT_EQ(dummy_hash{}(dummy1), dummy_hash{}(dummy2)); + + // hash unequal + async_resource_ptr dummy3{&r1, &evt1}; + async_resource_ptr dummy4{&r2, &evt1}; + ASSERT_NE(dummy3, dummy4); + ASSERT_NE(dummy_hash{}(dummy3), dummy_hash{}(dummy4)); +} +} // namespace diff --git a/test/async_resource_reset_on_destroy_test.cpp b/test/async_resource_reset_on_destroy_test.cpp new file mode 100644 index 000000000..16b66aa3c --- /dev/null +++ b/test/async_resource_reset_on_destroy_test.cpp @@ -0,0 +1,81 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include +# include +# include +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +struct ManagedResetOnDestroyResource { + ManagedResetOnDestroyResource( + async_resource_ptr child) noexcept + : child_(std::move(child)) {} + + ManagedResetOnDestroyResource(ManagedResetOnDestroyResource&& other) noexcept + : child_(std::move(other.child_)) {} + + auto destroy() noexcept { + return just_from([this]() noexcept { child_.reset(); }); + } + + ~ManagedResetOnDestroyResource() noexcept { + // reset called + EXPECT_TRUE(child_.get() == nullptr); + } + +private: + async_resource_ptr child_; +}; + +task reset_on_destroy(AsyncResourceTest* f) { + // drop immediately + (void)co_await make_async_resource( // parent + f->ctx.get_scheduler(), + f->outerScope, + [f](auto scope, auto scheduler) noexcept { + return make_async_resource( // child + scheduler, + scope, + [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + }); + co_await f->outerScope.join(); // grandparent +} + +} // namespace + +TEST_F(AsyncResourceTest, reset_on_destroy) { + sync_wait(reset_on_destroy(this)); +} +#endif diff --git a/test/async_resource_reset_on_destruct_test.cpp b/test/async_resource_reset_on_destruct_test.cpp new file mode 100644 index 000000000..3de331584 --- /dev/null +++ b/test/async_resource_reset_on_destruct_test.cpp @@ -0,0 +1,74 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +struct ManagedResetOnDestructResource { + ManagedResetOnDestructResource( + async_resource_ptr child) noexcept + : child_(std::move(child)) {} + + ManagedResetOnDestructResource( + ManagedResetOnDestructResource&& other) noexcept + : child_(std::move(other.child_)) {} + + ~ManagedResetOnDestructResource() noexcept { std::move(child_).reset(); } + + // suppress deprecation warning - noop + auto destroy() noexcept { return just(); } + +private: + async_resource_ptr child_; +}; + +task reset_on_destruct(AsyncResourceTest* f) { + // drop immediately + (void)co_await make_async_resource( // parent + f->ctx.get_scheduler(), + f->outerScope, + [f](auto scope, auto scheduler) noexcept { + return make_async_resource( // child + scheduler, + scope, + [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + }); + co_await f->outerScope.join(); // grandparent +} + +} // namespace + +TEST_F(AsyncResourceTest, reset_on_destruct) { + sync_wait(reset_on_destruct(this)); +} +#endif diff --git a/test/async_resource_spawning_resource_test.cpp b/test/async_resource_spawning_resource_test.cpp new file mode 100644 index 000000000..4a3a7f404 --- /dev/null +++ b/test/async_resource_spawning_resource_test.cpp @@ -0,0 +1,93 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +class SpawningResource { + async_resource_ptr child1_; + async_resource_ptr child2_; + +public: + template + SpawningResource( + async_scope_ref scope, + Scheduler sched1, + Scheduler sched2, + AsyncResourceTest* f) noexcept { + // race for tear-down: child may never be constructed + spawn_detached( + on(sched1, + defer([sched1, scope, f]() noexcept { + return make_async_resource( + sched1, scope, [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + }) | then([this](auto&& child) noexcept { child1_.swap(child); })), + scope); + spawn_detached( + on(sched2, + defer([sched2, scope, f]() noexcept { + return make_async_resource( + sched2, scope, [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + }) | then([this](auto&& child) noexcept { child2_.swap(child); })), + scope); + } + + // suppress deprecation warning - noop + auto destroy() noexcept { return just(); } +}; + +task spawning_resource(AsyncResourceTest* f) { + single_thread_context ctx1; + single_thread_context ctx2; + // drop immediately + (void)co_await make_async_resource( + f->ctx.get_scheduler(), + f->outerScope, + [f, sched1 = ctx1.get_scheduler(), sched2 = ctx2.get_scheduler()]( + auto scope, auto) noexcept { + return SpawningResource{scope, sched1, sched2, f}; + }); + co_await f->outerScope.join(); +} + +} // namespace + +TEST_F(AsyncResourceTest, spawning_resource) { + sync_wait(spawning_resource(this)); +} +#endif diff --git a/test/async_resource_tag_invoke_async_destroy_test.cpp b/test/async_resource_tag_invoke_async_destroy_test.cpp new file mode 100644 index 000000000..24eccd0e0 --- /dev/null +++ b/test/async_resource_tag_invoke_async_destroy_test.cpp @@ -0,0 +1,64 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +struct ManagedTagInvokeResource : public ResourceBase { + using ResourceBase::ResourceBase; + + ~ManagedTagInvokeResource() noexcept { EXPECT_TRUE(destroyCalled_); } + +private: + friend auto + tag_invoke(tag_t, ManagedTagInvokeResource& self) noexcept { + return just_from([&]() noexcept { self.destroyCalled_ = true; }); + } + + bool destroyCalled_{false}; +}; + +task tag_invoke_async_destroy(AsyncResourceTest* f) { + // drop immediately + (void)co_await make_async_resource( + f->ctx.get_scheduler(), f->outerScope, [f](auto, auto) noexcept { + return ManagedTagInvokeResource{&(f->objectCount)}; + }); + co_await f->outerScope.join(); +} + +} // namespace + +TEST_F(AsyncResourceTest, tag_invoke_async_destroy) { + sync_wait(tag_invoke_async_destroy(this)); +} +#endif diff --git a/test/async_resource_tag_invoke_over_async_destroy_test.cpp b/test/async_resource_tag_invoke_over_async_destroy_test.cpp new file mode 100644 index 000000000..362e0d42b --- /dev/null +++ b/test/async_resource_tag_invoke_over_async_destroy_test.cpp @@ -0,0 +1,78 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +struct ManagedDestroyTagInvokeResource : public ResourceBase { + using ResourceBase::ResourceBase; + + ~ManagedDestroyTagInvokeResource() noexcept { + EXPECT_FALSE(destroyCalled_); + EXPECT_TRUE(tagInvokeCalled_); + } + + auto destroy() noexcept { + return just_from([&]() noexcept { destroyCalled_ = true; }); + } + +private: + // takes precedence over destroy() + friend auto tag_invoke( + tag_t, ManagedDestroyTagInvokeResource& self) noexcept { + return just_from([&]() noexcept { self.tagInvokeCalled_ = true; }); + } + + bool destroyCalled_{false}; + bool tagInvokeCalled_{false}; +}; + +task tag_invoke_over_async_destroy(AsyncResourceTest* f) { + // move-only _ptr + { + auto ptr = co_await make_async_resource( + f->ctx.get_scheduler(), f->outerScope, [f](auto, auto) noexcept { + return ManagedDestroyTagInvokeResource{&(f->objectCount)}; + }); + static_assert( + !std::is_copy_constructible_v, + "ptr is a move-only type"); + } // drop ptr + co_await f->outerScope.join(); +} + +} // namespace + +TEST_F(AsyncResourceTest, tag_invoke_over_async_destroy) { + sync_wait(tag_invoke_over_async_destroy(this)); +} +#endif diff --git a/test/async_resource_test.hpp b/test/async_resource_test.hpp new file mode 100644 index 000000000..68b16dc95 --- /dev/null +++ b/test/async_resource_test.hpp @@ -0,0 +1,134 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#include + +#include +#include + +#include +#include +#include +#include +#include +#include + +namespace unifex_test { + +struct AsyncResourceTest : testing::Test { + unifex::v2::async_scope outerScope; + unifex::single_thread_context ctx; + + ~AsyncResourceTest() override { EXPECT_EQ(objectCount, 0); } + + std::atomic_int objectCount{0}; +}; + +template +struct AsyncResourceTypedTest : public AsyncResourceTest {}; + +class ResourceBase { + std::atomic_int* objectCount_; + +public: + explicit ResourceBase(std::atomic_int* objectCount) noexcept + : objectCount_(objectCount) { + ++(*objectCount_); + } + + ~ResourceBase() { --(*objectCount_); } +}; + +struct UnmanagedResource : public ResourceBase { + using ResourceBase::ResourceBase; + + // suppress deprecation warning - noop + auto destroy() noexcept { return unifex::just(); } +}; + +struct TwinNestingResource { + TwinNestingResource() = default; + + TwinNestingResource( + unifex::async_resource_ptr child1, + unifex::async_resource_ptr child2) noexcept + : child1_(std::move(child1)) + , child2_(std::move(child2)) {} + + void swap1(unifex::async_resource_ptr child) noexcept { + child1_.swap(child); + } + + void swap2(unifex::async_resource_ptr child) noexcept { + child2_.swap(child); + } + + void drop_children() noexcept { + std::move(child1_).reset(); + std::move(child2_).reset(); + } + // + // suppress deprecation warning - noop + auto destroy() noexcept { return unifex::just(); } + +private: + unifex::async_resource_ptr child1_; + unifex::async_resource_ptr child2_; +}; + +template +struct SingleNestingResource { + SingleNestingResource(unifex::async_resource_ptr child) noexcept + : child_(std::move(child)) {} + + void drop_child() noexcept { std::move(child_).reset(); } + // + // suppress deprecation warning - noop + auto destroy() noexcept { return unifex::just(); } + +private: + unifex::async_resource_ptr child_; +}; + +struct ThrowingResource { + ThrowingResource() { throw 42; } + // suppress deprecation warning - noop + auto destroy() noexcept { return unifex::just(); } +}; + +template +struct ThrowingSpawningResource { + ThrowingSpawningResource( + unifex::async_scope_ref scope, Scheduler sched, AsyncResourceTest* f) { + unifex::spawn_detached( + unifex::on( + sched, + unifex::defer([sched, scope, f]() noexcept { + return unifex::make_async_resource( + sched, scope, [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + }) | unifex::then([](auto&&) noexcept { /* drop */ })), + scope); + + throw 42; + } + // suppress deprecation warning - noop + auto destroy() noexcept { return unifex::just(); } +}; +} // namespace unifex_test diff --git a/test/async_resource_throwing_plain_sender_test.cpp b/test/async_resource_throwing_plain_sender_test.cpp new file mode 100644 index 000000000..ff36e7dd5 --- /dev/null +++ b/test/async_resource_throwing_plain_sender_test.cpp @@ -0,0 +1,121 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# if !UNIFEX_NO_EXCEPTIONS +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { +struct Spawning { + auto operator()(AsyncResourceTest* f) const { + return make_async_resource( + f->ctx.get_scheduler(), + f->outerScope, + [f](auto scope, auto scheduler) { + // constructor throws after spawning + return ThrowingSpawningResource{scope, scheduler, f}; + }) | + then([](auto&&...) noexcept {}); + } +}; + +struct SpawningSenderFactory { + auto operator()(AsyncResourceTest* f) const { + return make_async_resource< + ThrowingSpawningResourcectx.get_scheduler())>>( + f->ctx.get_scheduler(), + f->outerScope, + [f](auto scope, auto scheduler) noexcept { + // constructor throws + return just(scope, scheduler, f); + }) | + then([](auto&&...) noexcept {}); + } +}; + +struct Throwing { + auto operator()(AsyncResourceTest* f) const { + return make_async_resource( + f->ctx.get_scheduler(), + f->outerScope, + [](auto, auto) { + // constructor throws + return ThrowingResource{}; + }) | + then([](auto&&...) noexcept {}); + } +}; + +struct SenderFactoryConstructor { + auto operator()(AsyncResourceTest* f) const { + return make_async_resource( + f->ctx.get_scheduler(), + f->outerScope, + [](auto, auto) noexcept { + // constructor throws + return just(); + }) | + then([](auto&&...) noexcept {}); + } +}; + +struct SenderFactory { + auto operator()(AsyncResourceTest* f) const { + return make_async_resource( + f->ctx.get_scheduler(), + f->outerScope, + [](auto, auto) -> any_sender_of<> { + // factory throws + throw 42; + }) | + then([](auto&&...) noexcept {}); + } +}; +} // namespace + +TYPED_TEST_SUITE_P(AsyncResourceTypedTest); +TYPED_TEST_P(AsyncResourceTypedTest, throwing_plain_sender) { + try { + sync_wait(TypeParam{}(this)); + FAIL() << "throwing ResourceFactory should not succeed"; + } catch (int i) { + ASSERT_EQ(i, 42); + } + sync_wait(this->outerScope.join()); +} +REGISTER_TYPED_TEST_SUITE_P(AsyncResourceTypedTest, throwing_plain_sender); +using TestTypes = ::testing::Types< + Spawning, + SpawningSenderFactory, + Throwing, + SenderFactoryConstructor, + SenderFactory>; +INSTANTIATE_TYPED_TEST_SUITE_P(Throwing, AsyncResourceTypedTest, TestTypes); + +# endif // !UNIFEX_NO_EXCEPTIONS +#endif // !UNIFEX_NO_COROUTINES diff --git a/test/async_resource_throwing_test.cpp b/test/async_resource_throwing_test.cpp new file mode 100644 index 000000000..9ea1618cb --- /dev/null +++ b/test/async_resource_throwing_test.cpp @@ -0,0 +1,111 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# if !UNIFEX_NO_EXCEPTIONS +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { +struct Spawning { + task operator()(AsyncResourceTest* f) const { + (void)co_await make_async_resource( + f->ctx.get_scheduler(), f->outerScope, [f](auto scope, auto scheduler) { + // constructor throws after spawning + return ThrowingSpawningResource{scope, scheduler, f}; + }); + } +}; + +struct SpawningSenderFactory { + task operator()(AsyncResourceTest* f) const { + (void)co_await make_async_resource< + ThrowingSpawningResourcectx.get_scheduler())>>( + f->ctx.get_scheduler(), + f->outerScope, + [f](auto scope, auto scheduler) noexcept { + // constructor throws + return just(scope, scheduler, f); + }); + } +}; + +struct Throwing { + task operator()(AsyncResourceTest* f) const { + (void)co_await make_async_resource( + f->ctx.get_scheduler(), f->outerScope, [](auto, auto) { + // constructor throws + return ThrowingResource{}; + }); + } +}; + +struct SenderFactoryConstructor { + task operator()(AsyncResourceTest* f) const { + (void)co_await make_async_resource( + f->ctx.get_scheduler(), f->outerScope, [](auto, auto) noexcept { + // constructor throws + return just(); + }); + } +}; + +struct SenderFactory { + task operator()(AsyncResourceTest* f) const { + (void)co_await make_async_resource( + f->ctx.get_scheduler(), + f->outerScope, + [](auto, auto) -> any_sender_of<> { + // factory throws + throw 42; + }); + } +}; +} // namespace + +TYPED_TEST_SUITE_P(AsyncResourceTypedTest); +TYPED_TEST_P(AsyncResourceTypedTest, throwing) { + try { + sync_wait(TypeParam{}(this)); + FAIL() << "throwing ResourceFactory should not succeed"; + } catch (int i) { + ASSERT_EQ(i, 42); + } + sync_wait(this->outerScope.join()); +} +REGISTER_TYPED_TEST_SUITE_P(AsyncResourceTypedTest, throwing); +using TestTypes = ::testing::Types< + Spawning, + SpawningSenderFactory, + Throwing, + SenderFactoryConstructor, + SenderFactory>; +INSTANTIATE_TYPED_TEST_SUITE_P(Throwing, AsyncResourceTypedTest, TestTypes); + +# endif // !UNIFEX_NO_EXCEPTIONS +#endif // !UNIFEX_NO_COROUTINES diff --git a/test/async_resource_two_children_later_test.cpp b/test/async_resource_two_children_later_test.cpp new file mode 100644 index 000000000..4d5e73f0d --- /dev/null +++ b/test/async_resource_two_children_later_test.cpp @@ -0,0 +1,74 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +template +struct Proxy { + Scope scope; + Scheduler scheduler; + TwinNestingResource resource; + + // suppress deprecation warning - noop + auto destroy() noexcept { return just(); } +}; + +task two_children_later(AsyncResourceTest* f) { + { + auto parent = co_await make_async_resource( // parent Proxy + f->ctx.get_scheduler(), + f->outerScope, + [](auto scope, auto scheduler) noexcept { + return Proxy{ + scope, scheduler, TwinNestingResource{}}; + }); + // children + auto child1 = co_await make_async_resource( + parent->scheduler, parent->scope, [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + parent->resource.swap1(std::move(child1)); + auto child2 = co_await make_async_resource( + parent->scheduler, parent->scope, [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + parent->resource.swap2(std::move(child2)); + } + co_await f->outerScope.join(); // grandparent +} + +} // namespace + +TEST_F(AsyncResourceTest, two_children_later) { + sync_wait(two_children_later(this)); +} +#endif diff --git a/test/async_resource_two_children_test.cpp b/test/async_resource_two_children_test.cpp new file mode 100644 index 000000000..560ac07b7 --- /dev/null +++ b/test/async_resource_two_children_test.cpp @@ -0,0 +1,60 @@ +/* + * Copyright (c) Facebook, Inc. and its affiliates. + * + * Licensed under the Apache License Version 2.0 with LLVM Exceptions + * (the "License"); you may not use this file except in compliance with + * the License. You may obtain a copy of the License at + * + * https://llvm.org/LICENSE.txt + * + * 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. + */ + +#include + +#if !UNIFEX_NO_COROUTINES +# include + +# include "async_resource_test.hpp" + +# include +# include + +# include +# include + +using namespace unifex; +using namespace unifex_test; + +namespace { + +task two_children(AsyncResourceTest* f) { + // drop immediately + (void)co_await make_async_resource( // parent + f->ctx.get_scheduler(), + f->outerScope, + [f](auto scope, auto scheduler) noexcept -> task { + // children + auto child1 = co_await make_async_resource( + scheduler, scope, [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + auto child2 = co_await make_async_resource( + scheduler, scope, [f](auto, auto) noexcept { + return UnmanagedResource{&(f->objectCount)}; + }); + co_return TwinNestingResource{std::move(child1), std::move(child2)}; + }); + co_await f->outerScope.join(); // grandparent +} + +} // namespace + +TEST_F(AsyncResourceTest, two_children) { + sync_wait(two_children(this)); +} +#endif