diff --git a/silkworm/db/kv/api/service_router.cpp b/silkworm/db/kv/api/service_router.cpp index bf69b87a23..ed3f343e26 100644 --- a/silkworm/db/kv/api/service_router.cpp +++ b/silkworm/db/kv/api/service_router.cpp @@ -16,9 +16,7 @@ #include "service_router.hpp" -#include - -#include +#include namespace silkworm::db::kv::api { @@ -26,7 +24,7 @@ using namespace boost::asio; Task StateChangeRunner::run(std::shared_ptr self) { auto run = self->handle_calls(); - co_await concurrency::co_spawn(self->strand_, std::move(run), use_awaitable); + co_await concurrency::spawn_task(self->strand_, std::move(run)); } StateChangeRunner::StateChangeRunner(const boost::asio::any_io_executor& executor) diff --git a/silkworm/db/kv/state_changes_stream.cpp b/silkworm/db/kv/state_changes_stream.cpp index edcaad83d5..cd583fbcee 100644 --- a/silkworm/db/kv/state_changes_stream.cpp +++ b/silkworm/db/kv/state_changes_stream.cpp @@ -16,12 +16,11 @@ #include "state_changes_stream.hpp" -#include #include #include -#include #include +#include #include namespace silkworm::db::kv { @@ -32,7 +31,7 @@ StateChangesStream::StateChangesStream(rpc::ClientContext& context, api::Client& cache_(must_use_shared_service(scheduler_)) {} std::future StateChangesStream::open() { - return concurrency::co_spawn(scheduler_, run(), boost::asio::use_future); + return concurrency::spawn_future(scheduler_, run()); } void StateChangesStream::close() { diff --git a/silkworm/infra/concurrency/awaitable_wait_for_all.hpp b/silkworm/infra/concurrency/awaitable_wait_for_all.hpp index 183e942338..79a03894c7 100644 --- a/silkworm/infra/concurrency/awaitable_wait_for_all.hpp +++ b/silkworm/infra/concurrency/awaitable_wait_for_all.hpp @@ -33,8 +33,8 @@ #include #include -#include "co_spawn_sw.hpp" #include "parallel_group_utils.hpp" +#include "spawn.hpp" namespace silkworm::concurrency::awaitable_wait_for_all { @@ -71,8 +71,8 @@ awaitable operator&&( auto [order, ex0, ex1] = co_await make_parallel_group( - co_spawn_sw(ex, std::move(t), deferred), - co_spawn_sw(ex, std::move(u), deferred)) + co_spawn(ex, std::move(t), deferred), + co_spawn(ex, std::move(u), deferred)) .async_wait( wait_for_one_error(), use_awaitable_t{}); @@ -93,8 +93,8 @@ awaitable operator&&( auto [order, ex0, ex1, r1] = co_await make_parallel_group( - co_spawn_sw(ex, std::move(t), deferred), - co_spawn_sw(ex, detail::awaitable_wrap(std::move(u)), deferred)) + co_spawn(ex, std::move(t), deferred), + co_spawn(ex, detail::awaitable_wrap(std::move(u)), deferred)) .async_wait( wait_for_one_error(), use_awaitable_t{}); @@ -115,8 +115,8 @@ awaitable operator&&( auto [order, ex0, r0, ex1] = co_await make_parallel_group( - co_spawn_sw(ex, detail::awaitable_wrap(std::move(t)), deferred), - co_spawn_sw(ex, std::move(u), deferred)) + co_spawn(ex, detail::awaitable_wrap(std::move(t)), deferred), + co_spawn(ex, std::move(u), deferred)) .async_wait( wait_for_one_error(), use_awaitable_t{}); @@ -137,8 +137,8 @@ awaitable, Executor> operator&&( auto [order, ex0, r0, ex1, r1] = co_await make_parallel_group( - co_spawn_sw(ex, detail::awaitable_wrap(std::move(t)), deferred), - co_spawn_sw(ex, detail::awaitable_wrap(std::move(u)), deferred)) + co_spawn(ex, detail::awaitable_wrap(std::move(t)), deferred), + co_spawn(ex, detail::awaitable_wrap(std::move(u)), deferred)) .async_wait( wait_for_one_error(), use_awaitable_t{}); @@ -161,8 +161,8 @@ awaitable, Executor> operator&&( auto [order, ex0, r0, ex1, r1] = co_await make_parallel_group( - co_spawn_sw(ex, detail::awaitable_wrap(std::move(t)), deferred), - co_spawn_sw(ex, std::move(u), deferred)) + co_spawn(ex, detail::awaitable_wrap(std::move(t)), deferred), + co_spawn(ex, std::move(u), deferred)) .async_wait( wait_for_one_error(), use_awaitable_t{}); @@ -183,8 +183,8 @@ awaitable, Executor> operator&&( auto [order, ex0, r0, ex1, r1] = co_await make_parallel_group( - co_spawn_sw(ex, detail::awaitable_wrap(std::move(t)), deferred), - co_spawn_sw(ex, detail::awaitable_wrap(std::move(u)), deferred)) + co_spawn(ex, detail::awaitable_wrap(std::move(t)), deferred), + co_spawn(ex, detail::awaitable_wrap(std::move(u)), deferred)) .async_wait( wait_for_one_error(), use_awaitable_t{}); diff --git a/silkworm/infra/concurrency/awaitable_wait_for_one.hpp b/silkworm/infra/concurrency/awaitable_wait_for_one.hpp index b4df238ca5..855a877354 100644 --- a/silkworm/infra/concurrency/awaitable_wait_for_one.hpp +++ b/silkworm/infra/concurrency/awaitable_wait_for_one.hpp @@ -34,7 +34,7 @@ #include #include -#include "co_spawn_sw.hpp" +#include "spawn.hpp" namespace silkworm::concurrency::awaitable_wait_for_one { @@ -70,8 +70,8 @@ awaitable, Executor> operator||(awa auto ex = co_await this_coro::executor; auto [order, ex0, ex1] = - co_await make_parallel_group(co_spawn_sw(ex, std::move(t), deferred), - co_spawn_sw(ex, std::move(u), deferred)) + co_await make_parallel_group(co_spawn(ex, std::move(t), deferred), + co_spawn(ex, std::move(u), deferred)) .async_wait(wait_for_one(), use_awaitable_t{}); if (order[0] == 0) { @@ -93,8 +93,8 @@ awaitable, Executor> operator||(awaitable{}); if (order[0] == 0) { @@ -119,8 +119,8 @@ awaitable, Executor> operator||(awaitable{}); if (order[0] == 0) { @@ -145,8 +145,8 @@ awaitable, Executor> operator||(awaitable t, awa auto ex = co_await this_coro::executor; auto [order, ex0, r0, ex1, r1] = - co_await make_parallel_group(co_spawn_sw(ex, detail::awaitable_wrap(std::move(t)), deferred), - co_spawn_sw(ex, detail::awaitable_wrap(std::move(u)), deferred)) + co_await make_parallel_group(co_spawn(ex, detail::awaitable_wrap(std::move(t)), deferred), + co_spawn(ex, detail::awaitable_wrap(std::move(u)), deferred)) .async_wait(wait_for_one(), use_awaitable_t{}); if (order[0] == 0) { @@ -169,8 +169,8 @@ awaitable, Executor> operator||(awaitable{}); using widen = detail::widen_variant; @@ -194,8 +194,8 @@ awaitable, Executor> operator||(awaitable{}); using widen = detail::widen_variant; diff --git a/silkworm/infra/concurrency/co_spawn_sw.hpp b/silkworm/infra/concurrency/co_spawn_sw.hpp deleted file mode 100644 index 73c0298373..0000000000 --- a/silkworm/infra/concurrency/co_spawn_sw.hpp +++ /dev/null @@ -1,41 +0,0 @@ -/* - Copyright 2023 The Silkworm Authors - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. -*/ - -#pragma once - -#include - -#include - -#include -#include -#include - -#define co_spawn_sw co_spawn - -namespace silkworm::concurrency { - -using namespace boost::asio; - -template -inline BOOST_ASIO_INITFN_AUTO_RESULT_TYPE( - boost::asio::use_awaitable, - typename boost::asio::detail::awaitable_signature::type>::type = 0) - co_spawn_and_await(const Executor& ex, F&& f) { - return (co_spawn_sw)(ex, std::forward(f), boost::asio::use_awaitable); -} - -} // namespace silkworm::concurrency diff --git a/silkworm/infra/concurrency/parallel_group_test.cpp b/silkworm/infra/concurrency/parallel_group_test.cpp index 8183251863..624d3d195e 100644 --- a/silkworm/infra/concurrency/parallel_group_test.cpp +++ b/silkworm/infra/concurrency/parallel_group_test.cpp @@ -24,11 +24,10 @@ #include #include #include -#include #include #include -#include +#include using namespace boost::asio; using namespace boost::asio::experimental; @@ -52,12 +51,12 @@ awaitable throw_op() { } awaitable spawn_throw_op(strand& strand) { - co_await co_spawn_sw(strand, throw_op(), use_awaitable); + co_await spawn_task(strand, throw_op()); } awaitable spawn_noop_loop(strand& strand) { while (true) { - co_await co_spawn_sw(strand, noop(), use_awaitable); + co_await spawn_task(strand, noop()); } } @@ -74,6 +73,6 @@ awaitable co_spawn_cancellation_handler_bug() { TEST_CASE("parallel_group.co_spawn_cancellation_handler_bug") { io_context context; - co_spawn_sw(context, co_spawn_cancellation_handler_bug(), use_future); + spawn_future(context, co_spawn_cancellation_handler_bug()); context.run(); } diff --git a/silkworm/infra/concurrency/spawn.hpp b/silkworm/infra/concurrency/spawn.hpp new file mode 100644 index 0000000000..26bc8e3685 --- /dev/null +++ b/silkworm/infra/concurrency/spawn.hpp @@ -0,0 +1,69 @@ +/* + Copyright 2023 The Silkworm Authors + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. +*/ + +#pragma once + +#include + +#include + +#include +#include +#include +#include +#include + +#include + +namespace silkworm::concurrency { + +template +concept AsioExecutor = boost::asio::is_executor::value || boost::asio::execution::is_executor::value; + +template +concept AsioExecutionContext = std::is_convertible_v; + +template +auto spawn_task(const Executor& ex, F&& f) { + return boost::asio::co_spawn(ex, std::forward(f), boost::asio::use_awaitable); +} + +template +auto spawn_task(ExecutionContext& ctx, F&& f) { + return boost::asio::co_spawn(ctx, std::forward(f), boost::asio::use_awaitable); +} + +template +auto spawn_future(const Executor& ex, F&& f) { + return boost::asio::co_spawn(ex, std::forward(f), boost::asio::use_future); +} + +template +auto spawn_future(ExecutionContext& ctx, F&& f) { + return boost::asio::co_spawn(ctx, std::forward(f), boost::asio::use_future); +} + +template +auto spawn_future_and_wait(const Executor& ex, F&& f) { + return spawn_future(ex, std::forward(f)).get(); +} + +template +auto spawn_future_and_wait(ExecutionContext& ctx, F&& f) { + return spawn_future(ctx, std::forward(f)).get(); +} + +} // namespace silkworm::concurrency diff --git a/silkworm/infra/concurrency/sync_wait_test.cpp b/silkworm/infra/concurrency/spawn_test.cpp similarity index 71% rename from silkworm/infra/concurrency/sync_wait_test.cpp rename to silkworm/infra/concurrency/spawn_test.cpp index 63914e4679..d012c6fef8 100644 --- a/silkworm/infra/concurrency/sync_wait_test.cpp +++ b/silkworm/infra/concurrency/spawn_test.cpp @@ -14,7 +14,7 @@ limitations under the License. */ -#include "sync_wait.hpp" +#include "spawn.hpp" #include @@ -27,7 +27,7 @@ #include #include -namespace silkworm { +namespace silkworm::concurrency { namespace asio = boost::asio; @@ -55,31 +55,31 @@ class DummyEngine { } }; -TEST_CASE("sync wait") { - asio::io_context io; - asio::executor_work_guard work_guard{io.get_executor()}; - - SECTION("wait for function") { - std::thread io_execution([&io]() { io.run(); }); +struct SpawnTest { + SpawnTest() { + ioc_thread = std::thread{[this]() { ioc.run(); }}; + } + ~SpawnTest() { + ioc.stop(); + if (ioc_thread.joinable()) { + ioc_thread.join(); + } + } - sync_wait(io, dummy_task()); + asio::io_context ioc; + asio::executor_work_guard work_guard{ioc.get_executor()}; + std::thread ioc_thread; +}; - io.stop(); - io_execution.join(); +TEST_CASE_METHOD(SpawnTest, "spawn_and_wait") { + SECTION("wait for function") { + CHECK_NOTHROW(spawn_future_and_wait(ioc, dummy_task())); } SECTION("wait for method") { - std::thread io_execution([&io]() { io.run(); }); - - DummyEngine engine{io}; - - auto value = sync_wait(in(engine), DummyEngine::do_work()); - - CHECK(value == 42); - - io.stop(); - io_execution.join(); + DummyEngine engine{ioc}; + CHECK(spawn_future_and_wait(engine.get_executor(), DummyEngine::do_work()) == 42); } } -} // namespace silkworm \ No newline at end of file +} // namespace silkworm::concurrency \ No newline at end of file diff --git a/silkworm/infra/concurrency/sync_wait.hpp b/silkworm/infra/concurrency/sync_wait.hpp deleted file mode 100644 index 714b923959..0000000000 --- a/silkworm/infra/concurrency/sync_wait.hpp +++ /dev/null @@ -1,76 +0,0 @@ -/* - Copyright 2023 The Silkworm Authors - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. -*/ - -#pragma once - -#include - -#include -#include -#include - -namespace silkworm { - -/** - * Do a synchronous wait of a coroutine on the specified io_context - * - * sync_wait: - * - schedules a coroutine for execution in the specified io_context - * - blocks the calling thread until the coroutine completes - * - returns the result of the coroutine - * - * Rationale: doing an asynchronous wait of a coroutine is easy: - * auto result = co_await task(); - * Doing a synchronous wait of a coroutine is more verbose: - * auto result = co_spawn(io_context, task(), use_future).get(); - * also this exposes implementation details. - * Using sync_wait the call becomes: - * auto result = sync_wait(io_context, task()); - * Or, if the current object has an io_context: - * auto result = sync_wait(in(this), task()); - * - */ -template -T sync_wait(boost::asio::io_context& io_context, const Task& task) { - auto future_result = boost::asio::co_spawn(io_context, task, boost::asio::use_future); - return future_result.get(); -} - -template -T sync_wait(boost::asio::io_context& io_context, Task&& task) { - auto future_result = boost::asio::co_spawn(io_context, std::move(task), boost::asio::use_future); - return future_result.get(); -} - -/** - * Simplify the call to sync_wait - * - * When the desired io_context is not immediately available in the scope of the caller, - * but is owned by some object, this function can be used to retrieve it and simplify - * the call to sync_wait - * - * For example, provided that some class Engine has an io_context: - * sync_wait(in(engine), engine.do_work()); - * or provided that the current object has an io_context: - * sync_wait(in(this), engine.do_work()); - */ - -template -boost::asio::io_context& in(C& context) { - return context.get_executor(); -} - -} // namespace silkworm \ No newline at end of file diff --git a/silkworm/infra/grpc/client/client_context_pool.cpp b/silkworm/infra/grpc/client/client_context_pool.cpp index 1a7683a942..b394bd7a04 100644 --- a/silkworm/infra/grpc/client/client_context_pool.cpp +++ b/silkworm/infra/grpc/client/client_context_pool.cpp @@ -77,7 +77,8 @@ void ClientContext::execute_loop_single_threaded(IdleStrategy&& idle_strategy) { void ClientContext::execute_loop_multi_threaded() { SILK_DEBUG << "Multi-thread execution loop start [" << std::this_thread::get_id() << "]"; - std::thread grpc_context_thread{[grpc_context = grpc_context_]() { + std::thread grpc_context_thread{[context_id = context_id_, grpc_context = grpc_context_]() { + log::set_thread_name(("grpc_ctx_s" + std::to_string(context_id)).c_str()); grpc_context->run_completion_queue(); }}; std::exception_ptr run_exception; diff --git a/silkworm/infra/test_util/context_test_base.hpp b/silkworm/infra/test_util/context_test_base.hpp index 66d8c54483..4de8ace778 100644 --- a/silkworm/infra/test_util/context_test_base.hpp +++ b/silkworm/infra/test_util/context_test_base.hpp @@ -24,6 +24,7 @@ #include #include +#include #include #include @@ -35,7 +36,7 @@ class ContextTestBase { template auto spawn(AwaitableOrFunction&& awaitable) { - return boost::asio::co_spawn(io_context_, std::forward(awaitable), boost::asio::use_future); + return concurrency::spawn_future(io_context_, std::forward(awaitable)); } template diff --git a/silkworm/node/execution/api/active_direct_service.cpp b/silkworm/node/execution/api/active_direct_service.cpp index ba194351a7..57e18ae05b 100644 --- a/silkworm/node/execution/api/active_direct_service.cpp +++ b/silkworm/node/execution/api/active_direct_service.cpp @@ -16,7 +16,7 @@ #include "active_direct_service.hpp" -#include +#include namespace silkworm::execution::api { @@ -41,7 +41,7 @@ bool ActiveDirectService::stop() { // rpc InsertBlocks(InsertBlocksRequest) returns(InsertionResult); Task ActiveDirectService::insert_blocks(const Blocks& blocks) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, const auto& bb) { + return concurrency::spawn_task(executor_, [](auto* self, const auto& bb) { return self->DirectService::insert_blocks(bb); }(this, blocks)); } @@ -50,14 +50,14 @@ Task ActiveDirectService::insert_blocks(const Blocks& blocks) { // rpc ValidateChain(ValidationRequest) returns(ValidationReceipt); Task ActiveDirectService::validate_chain(BlockNumAndHash number_and_hash) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, auto num_and_hash) { + return concurrency::spawn_task(executor_, [](auto* self, auto num_and_hash) { return self->DirectService::validate_chain(num_and_hash); }(this, number_and_hash)); } // rpc UpdateForkChoice(ForkChoice) returns(ForkChoiceReceipt); Task ActiveDirectService::update_fork_choice(const ForkChoice& fork_choice) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, const auto& choice) { + return concurrency::spawn_task(executor_, [](auto* self, const auto& choice) { return self->DirectService::update_fork_choice(choice); }(this, fork_choice)); } @@ -66,14 +66,14 @@ Task ActiveDirectService::update_fork_choice(const ForkChoice& // rpc AssembleBlock(AssembleBlockRequest) returns(AssembleBlockResponse); Task ActiveDirectService::assemble_block(const api::BlockUnderConstruction& block) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, const auto& b) { + return concurrency::spawn_task(executor_, [](auto* self, const auto& b) { return self->DirectService::assemble_block(b); }(this, block)); } // rpc GetAssembledBlock(GetAssembledBlockRequest) returns(GetAssembledBlockResponse); Task ActiveDirectService::get_assembled_block(PayloadId payload_id) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, auto id) { + return concurrency::spawn_task(executor_, [](auto* self, auto id) { return self->DirectService::get_assembled_block(id); }(this, payload_id)); } @@ -82,35 +82,35 @@ Task ActiveDirectService::get_assembled_block(PayloadId pa // rpc CurrentHeader(google.protobuf.Empty) returns(GetHeaderResponse); Task> ActiveDirectService::current_header() { - return concurrency::co_spawn_and_await(executor_, [](auto* self) { + return concurrency::spawn_task(executor_, [](auto* self) { return self->DirectService::current_header(); }(this)); } // rpc GetTD(GetSegmentRequest) returns(GetTDResponse); Task> ActiveDirectService::get_td(BlockNumberOrHash number_or_hash) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, auto num_or_hash) { + return concurrency::spawn_task(executor_, [](auto* self, auto num_or_hash) { return self->DirectService::get_td(num_or_hash); }(this, number_or_hash)); } // rpc GetHeader(GetSegmentRequest) returns(GetHeaderResponse); Task> ActiveDirectService::get_header(BlockNumberOrHash number_or_hash) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, auto num_or_hash) { + return concurrency::spawn_task(executor_, [](auto* self, auto num_or_hash) { return self->DirectService::get_header(num_or_hash); }(this, number_or_hash)); } // rpc GetBody(GetSegmentRequest) returns(GetBodyResponse); Task> ActiveDirectService::get_body(BlockNumberOrHash number_or_hash) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, auto num_or_hash) { + return concurrency::spawn_task(executor_, [](auto* self, auto num_or_hash) { return self->DirectService::get_body(num_or_hash); }(this, number_or_hash)); } // rpc HasBlock(GetSegmentRequest) returns(HasBlockResponse); Task ActiveDirectService::has_block(BlockNumberOrHash number_or_hash) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, auto num_or_hash) { + return concurrency::spawn_task(executor_, [](auto* self, auto num_or_hash) { return self->DirectService::has_block(num_or_hash); }(this, number_or_hash)); } @@ -119,14 +119,14 @@ Task ActiveDirectService::has_block(BlockNumberOrHash number_or_hash) { // rpc GetBodiesByRange(GetBodiesByRangeRequest) returns(GetBodiesBatchResponse); Task ActiveDirectService::get_bodies_by_range(BlockNumRange number_range) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, auto num_range) { + return concurrency::spawn_task(executor_, [](auto* self, auto num_range) { return self->DirectService::get_bodies_by_range(num_range); }(this, number_range)); } // rpc GetBodiesByHashes(GetBodiesByHashesRequest) returns(GetBodiesBatchResponse); Task ActiveDirectService::get_bodies_by_hashes(const BlockHashes& hashes) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, const auto& hh) { + return concurrency::spawn_task(executor_, [](auto* self, const auto& hh) { return self->DirectService::get_bodies_by_hashes(hh); }(this, hashes)); } @@ -135,21 +135,21 @@ Task ActiveDirectService::get_bodies_by_hashes(const BlockHashes& h // rpc IsCanonicalHash(types.H256) returns(IsCanonicalResponse); Task ActiveDirectService::is_canonical_hash(Hash block_hash) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, auto h) { + return concurrency::spawn_task(executor_, [](auto* self, auto h) { return self->DirectService::is_canonical_hash(h); }(this, block_hash)); } // rpc GetHeaderHashNumber(types.H256) returns(GetHeaderHashNumberResponse); Task> ActiveDirectService::get_header_hash_number(Hash block_hash) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, auto h) { + return concurrency::spawn_task(executor_, [](auto* self, auto h) { return self->DirectService::get_header_hash_number(h); }(this, block_hash)); } // rpc GetForkChoice(google.protobuf.Empty) returns(ForkChoice); Task ActiveDirectService::get_fork_choice() { - return concurrency::co_spawn_and_await(executor_, [](auto* self) { + return concurrency::spawn_task(executor_, [](auto* self) { return self->DirectService::get_fork_choice(); }(this)); } @@ -158,14 +158,14 @@ Task ActiveDirectService::get_fork_choice() { // rpc Ready(google.protobuf.Empty) returns(ReadyResponse); Task ActiveDirectService::ready() { - return concurrency::co_spawn_and_await(executor_, [](auto* self) { + return concurrency::spawn_task(executor_, [](auto* self) { return self->DirectService::ready(); }(this)); } // rpc FrozenBlocks(google.protobuf.Empty) returns(FrozenBlocksResponse); Task ActiveDirectService::frozen_blocks() { - return concurrency::co_spawn_and_await(executor_, [](auto* self) { + return concurrency::spawn_task(executor_, [](auto* self) { return self->DirectService::frozen_blocks(); }(this)); } @@ -173,13 +173,13 @@ Task ActiveDirectService::frozen_blocks() { /** Additional non-RPC methods **/ Task ActiveDirectService::get_last_headers(uint64_t n) { - return concurrency::co_spawn_and_await(executor_, [](auto* self, auto how_many) { + return concurrency::spawn_task(executor_, [](auto* self, auto how_many) { return self->DirectService::get_last_headers(how_many); }(this, n)); } Task ActiveDirectService::block_progress() { - return concurrency::co_spawn_and_await(executor_, [](auto* self) { + return concurrency::spawn_task(executor_, [](auto* self) { return self->DirectService::block_progress(); }(this)); } diff --git a/silkworm/node/stagedsync/execution_engine.cpp b/silkworm/node/stagedsync/execution_engine.cpp index a36f770839..7d42aa4609 100644 --- a/silkworm/node/stagedsync/execution_engine.cpp +++ b/silkworm/node/stagedsync/execution_engine.cpp @@ -17,13 +17,10 @@ #include "execution_engine.hpp" #include -#include - -#include #include #include -#include +#include namespace silkworm::stagedsync { @@ -209,7 +206,7 @@ bool ExecutionEngine::notify_fork_choice_update(Hash head_block_hash, // notify the fork of the update - we need to block here to restore the invariant auto fork_choice_aw_future = (*f)->fork_choice(head_block_hash, finalized_block_hash, safe_block_hash); - std::future fork_choice_future = concurrency::co_spawn_sw(io_context_, fork_choice_aw_future.get(), use_future); + std::future fork_choice_future = concurrency::spawn_future(io_context_, fork_choice_aw_future.get()); bool updated = fork_choice_future.get(); // BLOCKING if (!updated) return false; diff --git a/silkworm/node/stagedsync/stages/stage_triggers.cpp b/silkworm/node/stagedsync/stages/stage_triggers.cpp index 2dbf1d9b3f..66753fe491 100644 --- a/silkworm/node/stagedsync/stages/stage_triggers.cpp +++ b/silkworm/node/stagedsync/stages/stage_triggers.cpp @@ -18,11 +18,9 @@ #include -#include -#include #include -#include +#include namespace silkworm::stagedsync { @@ -48,7 +46,7 @@ Task TriggersStage::schedule(std::function(db::RWTxn&)> task) { assert(tx); co_await t(*tx); }; - return concurrency::co_spawn_sw(io_context_, task_caller(), boost::asio::use_awaitable); + return concurrency::spawn_task(io_context_, task_caller()); } bool TriggersStage::stop() { diff --git a/silkworm/rpc/daemon.cpp b/silkworm/rpc/daemon.cpp index 18c205b750..994d28a760 100644 --- a/silkworm/rpc/daemon.cpp +++ b/silkworm/rpc/daemon.cpp @@ -36,6 +36,7 @@ #include #include #include +#include #include #include #include diff --git a/silkworm/sentry/discovery/node_db/serial_node_db.cpp b/silkworm/sentry/discovery/node_db/serial_node_db.cpp index 61805dff6c..8bd5cbcf7e 100644 --- a/silkworm/sentry/discovery/node_db/serial_node_db.cpp +++ b/silkworm/sentry/discovery/node_db/serial_node_db.cpp @@ -18,122 +18,122 @@ #include -#include +#include namespace silkworm::sentry::discovery::node_db { using namespace boost::asio; Task SerialNodeDb::upsert_node_address(NodeId id, NodeAddress address) { - return concurrency::co_spawn_sw(strand_, db_.upsert_node_address(std::move(id), std::move(address)), use_awaitable); + return concurrency::spawn_task(strand_, db_.upsert_node_address(std::move(id), std::move(address))); } Task> SerialNodeDb::find_node_address_v4(NodeId id) { - return concurrency::co_spawn_sw(strand_, db_.find_node_address_v4(std::move(id)), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_node_address_v4(std::move(id))); } Task> SerialNodeDb::find_node_address_v6(NodeId id) { - return concurrency::co_spawn_sw(strand_, db_.find_node_address_v6(std::move(id)), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_node_address_v6(std::move(id))); } Task SerialNodeDb::update_next_ping_time(NodeId id, Time value) { - return concurrency::co_spawn_sw(strand_, db_.update_next_ping_time(std::move(id), value), use_awaitable); + return concurrency::spawn_task(strand_, db_.update_next_ping_time(std::move(id), value)); } Task> SerialNodeDb::find_next_ping_time(NodeId id) { - return concurrency::co_spawn_sw(strand_, db_.find_next_ping_time(std::move(id)), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_next_ping_time(std::move(id))); } Task SerialNodeDb::update_last_pong_time(NodeId id, Time value) { - return concurrency::co_spawn_sw(strand_, db_.update_last_pong_time(std::move(id), value), use_awaitable); + return concurrency::spawn_task(strand_, db_.update_last_pong_time(std::move(id), value)); } Task> SerialNodeDb::find_last_pong_time(NodeId id) { - return concurrency::co_spawn_sw(strand_, db_.find_last_pong_time(std::move(id)), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_last_pong_time(std::move(id))); } Task SerialNodeDb::update_ping_fails(NodeId id, size_t value) { - return concurrency::co_spawn_sw(strand_, db_.update_ping_fails(std::move(id), value), use_awaitable); + return concurrency::spawn_task(strand_, db_.update_ping_fails(std::move(id), value)); } Task> SerialNodeDb::find_ping_fails(NodeId id) { - return concurrency::co_spawn_sw(strand_, db_.find_ping_fails(std::move(id)), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_ping_fails(std::move(id))); } Task SerialNodeDb::update_peer_disconnected_time(NodeId id, Time value) { - return concurrency::co_spawn_sw(strand_, db_.update_peer_disconnected_time(std::move(id), value), use_awaitable); + return concurrency::spawn_task(strand_, db_.update_peer_disconnected_time(std::move(id), value)); } Task> SerialNodeDb::find_peer_disconnected_time(NodeId id) { - return concurrency::co_spawn_sw(strand_, db_.find_peer_disconnected_time(std::move(id)), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_peer_disconnected_time(std::move(id))); } Task SerialNodeDb::update_peer_is_useless(NodeId id, bool value) { - return concurrency::co_spawn_sw(strand_, db_.update_peer_is_useless(std::move(id), value), use_awaitable); + return concurrency::spawn_task(strand_, db_.update_peer_is_useless(std::move(id), value)); } Task> SerialNodeDb::find_peer_is_useless(NodeId id) { - return concurrency::co_spawn_sw(strand_, db_.find_peer_is_useless(std::move(id)), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_peer_is_useless(std::move(id))); } Task SerialNodeDb::update_distance(NodeId id, size_t value) { - return concurrency::co_spawn_sw(strand_, db_.update_distance(std::move(id), value), use_awaitable); + return concurrency::spawn_task(strand_, db_.update_distance(std::move(id), value)); } Task> SerialNodeDb::find_distance(NodeId id) { - return concurrency::co_spawn_sw(strand_, db_.find_distance(std::move(id)), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_distance(std::move(id))); } Task SerialNodeDb::update_enr_seq_num(NodeId id, uint64_t value) { - return concurrency::co_spawn_sw(strand_, db_.update_enr_seq_num(std::move(id), value), use_awaitable); + return concurrency::spawn_task(strand_, db_.update_enr_seq_num(std::move(id), value)); } Task> SerialNodeDb::find_enr_seq_num(NodeId id) { - return concurrency::co_spawn_sw(strand_, db_.find_enr_seq_num(std::move(id)), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_enr_seq_num(std::move(id))); } Task SerialNodeDb::update_eth1_fork_id(NodeId id, std::optional value) { - return concurrency::co_spawn_sw(strand_, db_.update_eth1_fork_id(std::move(id), value), use_awaitable); + return concurrency::spawn_task(strand_, db_.update_eth1_fork_id(std::move(id), value)); } Task> SerialNodeDb::find_eth1_fork_id(NodeId id) { - return concurrency::co_spawn_sw(strand_, db_.find_eth1_fork_id(std::move(id)), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_eth1_fork_id(std::move(id))); } Task> SerialNodeDb::find_ping_candidates(Time time, size_t limit) { - return concurrency::co_spawn_sw(strand_, db_.find_ping_candidates(time, limit), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_ping_candidates(time, limit)); } Task> SerialNodeDb::find_useful_nodes(Time min_pong_time, size_t limit) { - return concurrency::co_spawn_sw(strand_, db_.find_useful_nodes(min_pong_time, limit), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_useful_nodes(min_pong_time, limit)); } Task> SerialNodeDb::find_lookup_candidates(FindLookupCandidatesQuery query) { - return concurrency::co_spawn_sw(strand_, db_.find_lookup_candidates(query), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_lookup_candidates(query)); } Task SerialNodeDb::mark_taken_lookup_candidates(const std::vector& ids, Time time) { - return concurrency::co_spawn_sw(strand_, db_.mark_taken_lookup_candidates(ids, time), use_awaitable); + return concurrency::spawn_task(strand_, db_.mark_taken_lookup_candidates(ids, time)); } Task> SerialNodeDb::take_lookup_candidates(FindLookupCandidatesQuery query, Time time) { - return concurrency::co_spawn_sw(strand_, db_.take_lookup_candidates(query, time), use_awaitable); + return concurrency::spawn_task(strand_, db_.take_lookup_candidates(query, time)); } Task> SerialNodeDb::find_peer_candidates(FindPeerCandidatesQuery query) { - return concurrency::co_spawn_sw(strand_, db_.find_peer_candidates(std::move(query)), use_awaitable); + return concurrency::spawn_task(strand_, db_.find_peer_candidates(std::move(query))); } Task SerialNodeDb::mark_taken_peer_candidates(const std::vector& ids, Time time) { - return concurrency::co_spawn_sw(strand_, db_.mark_taken_peer_candidates(ids, time), use_awaitable); + return concurrency::spawn_task(strand_, db_.mark_taken_peer_candidates(ids, time)); } Task> SerialNodeDb::take_peer_candidates(FindPeerCandidatesQuery query, Time time) { - return concurrency::co_spawn_sw(strand_, db_.take_peer_candidates(std::move(query), time), use_awaitable); + return concurrency::spawn_task(strand_, db_.take_peer_candidates(std::move(query), time)); } Task SerialNodeDb::delete_node(NodeId id) { - return concurrency::co_spawn_sw(strand_, db_.delete_node(std::move(id)), use_awaitable); + return concurrency::spawn_task(strand_, db_.delete_node(std::move(id))); } } // namespace silkworm::sentry::discovery::node_db diff --git a/silkworm/sentry/message_receiver.cpp b/silkworm/sentry/message_receiver.cpp index 480d405bc3..c69cb059cb 100644 --- a/silkworm/sentry/message_receiver.cpp +++ b/silkworm/sentry/message_receiver.cpp @@ -21,13 +21,12 @@ #include #include -#include #include #include #include #include -#include +#include namespace silkworm::sentry { @@ -42,7 +41,7 @@ Task MessageReceiver::run(std::shared_ptr self, PeerManag self->peer_tasks_.wait() && self->unsubscription_tasks_.wait() && self->handle_calls(); - co_await concurrency::co_spawn_sw(self->strand_, std::move(run), use_awaitable); + co_await concurrency::spawn_task(self->strand_, std::move(run)); } Task MessageReceiver::handle_calls() { diff --git a/silkworm/sentry/peer_manager.cpp b/silkworm/sentry/peer_manager.cpp index 31396234b8..2ef2b6d3fc 100644 --- a/silkworm/sentry/peer_manager.cpp +++ b/silkworm/sentry/peer_manager.cpp @@ -23,8 +23,8 @@ #include #include #include -#include #include +#include #include #include "peer_manager_observer.hpp" @@ -49,7 +49,7 @@ Task PeerManager::run( connect_peer_tasks_.wait() && drop_peer_tasks_.wait() && peer_tasks_.wait(); - co_await concurrency::co_spawn_sw(strand_, std::move(run), use_awaitable); + co_await concurrency::spawn_task(strand_, std::move(run)); } Task PeerManager::run_in_strand(concurrency::Channel>& peer_channel) { @@ -122,15 +122,15 @@ Task PeerManager::drop_peer( } Task PeerManager::count_peers() { - co_return (co_await concurrency::co_spawn_sw(strand_, count_peers_in_strand(), use_awaitable)); + co_return (co_await concurrency::spawn_task(strand_, count_peers_in_strand())); } Task PeerManager::enumerate_peers(EnumeratePeersCallback callback) { - co_await concurrency::co_spawn_sw(strand_, enumerate_peers_in_strand(callback), use_awaitable); + co_await concurrency::spawn_task(strand_, enumerate_peers_in_strand(callback)); } Task PeerManager::enumerate_random_peers(size_t max_count, EnumeratePeersCallback callback) { - co_await concurrency::co_spawn_sw(strand_, enumerate_random_peers_in_strand(max_count, callback), use_awaitable); + co_await concurrency::spawn_task(strand_, enumerate_random_peers_in_strand(max_count, callback)); } Task PeerManager::count_peers_in_strand() { @@ -251,7 +251,7 @@ Task PeerManager::connect_peer(EnodeUrl peer_url, bool is_static_peer, std [[maybe_unused]] auto _ = gsl::finally([this, peer_url] { this->connecting_peer_urls_.erase(peer_url); }); try { - auto peer1 = co_await concurrency::co_spawn_sw(executor_pool_.any_executor(), client->connect(peer_url, is_static_peer), use_awaitable); + auto peer1 = co_await concurrency::spawn_task(executor_pool_.any_executor(), client->connect(peer_url, is_static_peer)); auto peer = std::shared_ptr(std::move(peer1)); co_await client_peer_channel_.send(peer); } catch (const boost::system::system_error& ex) { diff --git a/silkworm/sentry/peer_manager_api.cpp b/silkworm/sentry/peer_manager_api.cpp index 2cf8f672bc..ce0f8018bb 100644 --- a/silkworm/sentry/peer_manager_api.cpp +++ b/silkworm/sentry/peer_manager_api.cpp @@ -22,13 +22,12 @@ #include #include -#include #include #include #include #include -#include +#include #include namespace silkworm::sentry { @@ -48,7 +47,7 @@ Task PeerManagerApi::run(std::shared_ptr self) { self->handle_peer_events_calls() && self->events_unsubscription_tasks_.wait() && self->forward_peer_events(); - co_await concurrency::co_spawn_sw(self->strand_, std::move(run), use_awaitable); + co_await concurrency::spawn_task(self->strand_, std::move(run)); } Task PeerManagerApi::handle_peer_count_calls() { diff --git a/silkworm/sentry/rlpx/peer.cpp b/silkworm/sentry/rlpx/peer.cpp index add694fb84..0fc939d197 100644 --- a/silkworm/sentry/rlpx/peer.cpp +++ b/silkworm/sentry/rlpx/peer.cpp @@ -27,8 +27,8 @@ #include #include #include -#include #include +#include #include #include "auth/auth_message_error.hpp" @@ -77,7 +77,7 @@ Task Peer::run(std::shared_ptr peer) { using namespace concurrency::awaitable_wait_for_one; auto run = peer->handle() || peer->send_message_tasks_.wait(); - co_await concurrency::co_spawn_sw(peer->strand_, std::move(run), use_awaitable); + co_await concurrency::spawn_task(peer->strand_, std::move(run)); } static bool is_fatal_network_error(const boost::system::system_error& ex) { @@ -244,7 +244,7 @@ Task Peer::handle() { } Task Peer::drop(const std::shared_ptr& peer, DisconnectReason reason) { - return concurrency::co_spawn_sw(peer->strand_, Peer::drop_in_strand(peer, reason), use_awaitable); + return concurrency::spawn_task(peer->strand_, Peer::drop_in_strand(peer, reason)); } Task Peer::drop_in_strand(std::shared_ptr peer, DisconnectReason reason) { diff --git a/silkworm/sync/sync_pos.cpp b/silkworm/sync/sync_pos.cpp index b4a722e7e7..89ba4e3909 100644 --- a/silkworm/sync/sync_pos.cpp +++ b/silkworm/sync/sync_pos.cpp @@ -220,7 +220,6 @@ Task PoSSync::new_payload(const rpc::NewPayloadRequest& requ } else if (std::holds_alternative(verification)) { // INVALID const auto invalid_chain = std::get(verification); - // auto latest_valid_height = sync_wait(in(exec_engine_), exec_engine_.get_block_num(invalid_chain.latest_valid_head)); auto unwind_point_td = chain_fork_view_.get_total_difficulty(invalid_chain.unwind_point.hash); Hash latest_valid_hash = unwind_point_td < terminal_total_difficulty ? kZeroHash diff --git a/silkworm/sync/sync_pow.cpp b/silkworm/sync/sync_pow.cpp index aa6edb2e0b..9932f741d5 100644 --- a/silkworm/sync/sync_pow.cpp +++ b/silkworm/sync/sync_pow.cpp @@ -21,12 +21,14 @@ #include #include #include -#include +#include #include #include namespace silkworm::chainsync { +using concurrency::spawn_future_and_wait; + PoWSync::PoWSync(BlockExchange& block_exchange, execution::api::Client& exec_engine) : ChainSync(block_exchange, exec_engine) {} @@ -38,21 +40,21 @@ PoWSync::NewHeight PoWSync::resume() { // find the point (head) where we left o BlockId head{}; // BlockExchange need a bunch of previous headers to attach the new ones - auto last_headers = sync_wait(io_context_, exec_engine_->get_last_headers(1000)); + auto last_headers = spawn_future_and_wait(io_context_, exec_engine_->get_last_headers(1000)); block_exchange_.initial_state(last_headers); // We calculate a provisional head based on the previous headers std::ranges::for_each(last_headers, [&, this](const auto& header) { auto hash = header.hash(); - auto td = sync_wait(io_context_, exec_engine_->get_td(hash)); + auto td = spawn_future_and_wait(io_context_, exec_engine_->get_td(hash)); chain_fork_view_.add(header, *td); // add to cache & compute a new canonical head }); // Now we can resume the sync from the canonical head - const auto last_fcu = sync_wait(io_context_, exec_engine_->get_fork_choice()); // previously was get_canonical_head() - const auto block_progress = sync_wait(io_context_, exec_engine_->block_progress()); + const auto last_fcu = spawn_future_and_wait(io_context_, exec_engine_->get_fork_choice()); // previously was get_canonical_head() + const auto block_progress = spawn_future_and_wait(io_context_, exec_engine_->block_progress()); - const auto last_fcu_number = sync_wait(io_context_, exec_engine_->get_header_hash_number(last_fcu.head_block_hash)); + const auto last_fcu_number = spawn_future_and_wait(io_context_, exec_engine_->get_header_hash_number(last_fcu.head_block_hash)); if (!last_fcu_number) return head; ensure_invariant(*last_fcu_number <= block_progress, "canonical head beyond block progress"); @@ -78,7 +80,7 @@ PoWSync::NewHeight PoWSync::forward_and_insert_blocks() { ResultQueue& downloading_queue = block_exchange_.result_queue(); - auto initial_block_progress = sync_wait(io_context_, exec_engine_->block_progress()); + auto initial_block_progress = spawn_future_and_wait(io_context_, exec_engine_->block_progress()); auto block_progress = initial_block_progress; block_exchange_.download_blocks(initial_block_progress, BlockExchange::Target_Tracking::kByAnnouncements); @@ -105,7 +107,7 @@ PoWSync::NewHeight PoWSync::forward_and_insert_blocks() { }); // Insert blocks into database - const auto insert_result{sync_wait(io_context_, exec_engine_->insert_blocks(to_plain_blocks(blocks)))}; + const auto insert_result{spawn_future_and_wait(io_context_, exec_engine_->insert_blocks(to_plain_blocks(blocks)))}; if (!insert_result) { log::Error("Sync") << "Cannot insert " << blocks.size() << " blocks, error=" << insert_result.status; continue; @@ -155,7 +157,7 @@ void PoWSync::execution_loop() { // Verify the new section of the chain log::Info("Sync") << "Verifying chain, head=(" << new_height.number << ", " << to_hex(new_height.hash) << ")"; - const auto verification = sync_wait(io_context_, exec_engine_->validate_chain(new_height)); // BLOCKING + const auto verification = spawn_future_and_wait(io_context_, exec_engine_->validate_chain(new_height)); // BLOCKING if (std::holds_alternative(verification)) { auto valid_chain = std::get(verification); @@ -167,14 +169,14 @@ void PoWSync::execution_loop() { // Notify the fork choice log::Info("Sync") << "Notifying fork choice updated, new head=" << new_height.number; - sync_wait(io_context_, exec_engine_->update_fork_choice({new_height.hash})); + spawn_future_and_wait(io_context_, exec_engine_->update_fork_choice({new_height.hash})); send_new_block_hash_announcements(); // according to eth/67 they must be done after a full block verification } else if (std::holds_alternative(verification)) { auto invalid_chain = std::get(verification); - const auto latest_valid_height = sync_wait(io_context_, exec_engine_->get_header_hash_number(invalid_chain.unwind_point.hash)); + const auto latest_valid_height = spawn_future_and_wait(io_context_, exec_engine_->get_header_hash_number(invalid_chain.unwind_point.hash)); ensure_invariant(latest_valid_height.has_value(), "wrong latest_valid_head"); log::Info("Sync") << "Invalid chain, unwinding down to=" << *latest_valid_height; @@ -188,14 +190,12 @@ void PoWSync::execution_loop() { // Notify the fork choice log::Info("Sync") << "Notifying fork choice updated, head=" << to_hex(invalid_chain.unwind_point.hash); - sync_wait(io_context_, exec_engine_->update_fork_choice({invalid_chain.unwind_point.hash})); - + spawn_future_and_wait(io_context_, exec_engine_->update_fork_choice({invalid_chain.unwind_point.hash})); } else if (std::holds_alternative(verification)) { // If it returned a validation error, raise an exception const auto validation_error = std::get(verification); throw std::logic_error("Consensus validation error: last point=" + validation_error.latest_valid_head.hash.to_hex() + ", error=" + validation_error.error); - } else { throw std::logic_error("Consensus, unknown error"); }