Skip to content

Commit

Permalink
fix timer user-after-free
Browse files Browse the repository at this point in the history
  • Loading branch information
poor-circle committed Jan 23, 2025
1 parent a1ebf51 commit 3983b43
Show file tree
Hide file tree
Showing 3 changed files with 44 additions and 19 deletions.
15 changes: 11 additions & 4 deletions include/ylt/coro_io/coro_io.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -111,12 +111,19 @@ class callback_awaitor_base {
resume();
}
template <typename... Args>
void set_value(Args &&...args) const {
if constexpr (!std::is_void_v<Arg>) {
obj->arg_ = {std::forward<Args>(args)...};
void set_value(std::error_code ec, Args &&...args) const {
if constexpr (!std::is_same_v<Arg, std::error_code>) {
obj->arg_ = {std::move(ec), std::forward<Args>(args)...};
}
else {
obj->arg_ = std::move(ec);
}
}
template <typename Args>
void set_value(Args &&args) const {
obj->arg_ = std::move(args);
}
void set_value(std::error_code &&ec) const {
void set_value(std::error_code ec) const {
if constexpr (std::is_same_v<Arg, std::error_code>) {
obj->arg_ = std::move(ec);
}
Expand Down
40 changes: 29 additions & 11 deletions include/ylt/coro_io/io_context_pool.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -105,17 +105,35 @@ class ExecutorWrapper : public async_simple::Executor {
}
void schedule(Func func, Duration dur, uint64_t hint,
async_simple::Slot *slot = nullptr) override {
auto timer = std::make_shared<asio::steady_timer>(executor_, dur);
timer->async_wait([fn = std::move(func), timer](const auto &ec) {
fn();
});
if (!async_simple::signalHelper{async_simple::SignalType::Terminate}
.tryEmplace(slot, [timer](auto signalType, auto *signal) mutable {
asio::dispatch(timer->get_executor(), [timer]() {
timer->cancel();
});
})) {
timer->cancel();
auto timer =
std::make_shared<std::pair<asio::steady_timer, std::atomic<bool>>>(
asio::steady_timer{executor_, dur}, false);
if (!slot) {
timer->first.async_wait([fn = std::move(func), timer](const auto &ec) {
fn();
});
}
else {
if (!async_simple::signalHelper{async_simple::SignalType::Terminate}
.tryEmplace(
slot, [timer](auto signalType, auto *signal) mutable {
if (bool expected = false;
!timer->second.compare_exchange_strong(
expected, true, std::memory_order_release)) {
timer->first.cancel();
}
})) {
asio::dispatch(timer->first.get_executor(), func);
}
else {
timer->first.async_wait([fn = std::move(func), timer](const auto &ec) {
fn();
});
if (bool expected = false; !timer->second.compare_exchange_strong(
expected, true, std::memory_order_release)) {
timer->first.cancel();
}
}
}
}
};
Expand Down
8 changes: 4 additions & 4 deletions src/coro_io/tests/test_client_pool.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -91,8 +91,8 @@ TEST_CASE("test client pool") {
REQUIRE(is_started.hasResult() == false);
auto pool = coro_io::client_pool<coro_rpc::coro_rpc_client>::create(
"127.0.0.1:8801", {.max_connection = 100,
.idle_timeout = 300ms,
.short_connect_idle_timeout = 50ms});
.idle_timeout = 700ms,
.short_connect_idle_timeout = 200ms});
SpinLock lock;
ConditionVariable<SpinLock> cv;
auto res = co_await event(20, *pool, cv, lock);
Expand All @@ -102,10 +102,10 @@ TEST_CASE("test client pool") {
CHECK(res);
auto sz = pool->free_client_count();
CHECK(sz == 200);
co_await coro_io::sleep_for(150ms);
co_await coro_io::sleep_for(500ms);
sz = pool->free_client_count();
CHECK((sz >= 100 && sz <= 105));
co_await coro_io::sleep_for(550ms);
co_await coro_io::sleep_for(1000ms);
CHECK(pool->free_client_count() == 0);
server.stop();
}());
Expand Down

0 comments on commit 3983b43

Please sign in to comment.