From 0e3d874160e6bb60237dbcc8f8c95e9252aab65c Mon Sep 17 00:00:00 2001 From: Jonathan Reams Date: Tue, 18 Oct 2022 12:47:03 -0400 Subject: [PATCH] Do not lock destroyed mutex after wait_for_upload timeout (#5944) --- test/object-store/util/test_file.cpp | 34 +++++++++++++++++----------- 1 file changed, 21 insertions(+), 13 deletions(-) diff --git a/test/object-store/util/test_file.cpp b/test/object-store/util/test_file.cpp index 5358bfd4eca..7a76de899c7 100644 --- a/test/object-store/util/test_file.cpp +++ b/test/object-store/util/test_file.cpp @@ -214,29 +214,37 @@ std::string SyncServer::url_for_realm(StringData realm_name) const return util::format("%1/%2", m_url, realm_name); } +struct WaitForSessionState { + std::condition_variable cv; + std::mutex mutex; + bool complete = false; + std::error_code ec; +}; + static std::error_code wait_for_session(Realm& realm, void (SyncSession::*fn)(util::UniqueFunction&&), std::chrono::seconds timeout) { - std::condition_variable cv; - std::mutex wait_mutex; - bool wait_flag(false); - std::error_code ec; + auto shared_state = std::make_shared(); auto& session = *realm.config().sync_config->user->session_for_on_disk_path(realm.config().path); - (session.*fn)([&](std::error_code error) { - std::unique_lock lock(wait_mutex); - wait_flag = true; - ec = error; - cv.notify_one(); + (session.*fn)([weak_state = std::weak_ptr(shared_state)](std::error_code error) { + auto shared_state = weak_state.lock(); + if (!shared_state) { + return; + } + std::lock_guard lock(shared_state->mutex); + shared_state->complete = true; + shared_state->ec = error; + shared_state->cv.notify_one(); }); - std::unique_lock lock(wait_mutex); - bool completed = cv.wait_for(lock, timeout, [&]() { - return wait_flag == true; + std::unique_lock lock(shared_state->mutex); + bool completed = shared_state->cv.wait_for(lock, timeout, [&]() { + return shared_state->complete == true; }); if (!completed) { throw std::runtime_error("wait_For_session() timed out"); } - return ec; + return shared_state->ec; } std::error_code wait_for_upload(Realm& realm, std::chrono::seconds timeout)