From 32092e51eded72eeb8a08b636945f9f028df0fee Mon Sep 17 00:00:00 2001 From: "mergify[bot]" <37929162+mergify[bot]@users.noreply.github.com> Date: Mon, 16 Dec 2024 12:01:44 +0100 Subject: [PATCH] Fix tsan potential deadlock between `StatefulWriter` and `FlowController` (#5432) (#5494) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Refs #22339: Add BB test Signed-off-by: Mario Dominguez * Refs #22339: Fix tsan deadlock report Signed-off-by: Mario Dominguez * Refs #22339: Take writer's mutex before rproxy->stop() and check_acked_status() Signed-off-by: Mario Dominguez * Refs #22339: Apply Miguels suggestion Signed-off-by: Mario Dominguez --------- Signed-off-by: Mario Dominguez (cherry picked from commit 8fcd7ca4833af25d0792524587792c750f41b717) Co-authored-by: Mario Domínguez López <116071334+Mario-DL@users.noreply.github.com> --- src/cpp/rtps/writer/StatefulWriter.cpp | 72 +++++++++---------- .../blackbox/common/DDSBlackboxTestsBasic.cpp | 37 ++++++++++ 2 files changed, 73 insertions(+), 36 deletions(-) diff --git a/src/cpp/rtps/writer/StatefulWriter.cpp b/src/cpp/rtps/writer/StatefulWriter.cpp index 4bfc21ff5b9..638cb201196 100644 --- a/src/cpp/rtps/writer/StatefulWriter.cpp +++ b/src/cpp/rtps/writer/StatefulWriter.cpp @@ -1213,55 +1213,58 @@ bool StatefulWriter::matched_reader_remove( { ReaderProxy* rproxy = nullptr; std::unique_lock lock(mp_mutex); - std::unique_lock guard_locator_selector_general(locator_selector_general_); - std::unique_lock guard_locator_selector_async(locator_selector_async_); - for (ReaderProxyIterator it = matched_local_readers_.begin(); - it != matched_local_readers_.end(); ++it) { - if ((*it)->guid() == reader_guid) - { - EPROSIMA_LOG_INFO(RTPS_WRITER, "Reader Proxy removed: " << reader_guid); - rproxy = std::move(*it); - it = matched_local_readers_.erase(it); - break; - } - } + std::lock_guard guard_locator_selector_general(locator_selector_general_); + std::lock_guard guard_locator_selector_async(locator_selector_async_); - if (rproxy == nullptr) - { - for (ReaderProxyIterator it = matched_datasharing_readers_.begin(); - it != matched_datasharing_readers_.end(); ++it) + for (ReaderProxyIterator it = matched_local_readers_.begin(); + it != matched_local_readers_.end(); ++it) { if ((*it)->guid() == reader_guid) { EPROSIMA_LOG_INFO(RTPS_WRITER, "Reader Proxy removed: " << reader_guid); rproxy = std::move(*it); - it = matched_datasharing_readers_.erase(it); + it = matched_local_readers_.erase(it); break; } } - } - if (rproxy == nullptr) - { - for (ReaderProxyIterator it = matched_remote_readers_.begin(); - it != matched_remote_readers_.end(); ++it) + if (rproxy == nullptr) { - if ((*it)->guid() == reader_guid) + for (ReaderProxyIterator it = matched_datasharing_readers_.begin(); + it != matched_datasharing_readers_.end(); ++it) { - EPROSIMA_LOG_INFO(RTPS_WRITER, "Reader Proxy removed: " << reader_guid); - rproxy = std::move(*it); - it = matched_remote_readers_.erase(it); - break; + if ((*it)->guid() == reader_guid) + { + EPROSIMA_LOG_INFO(RTPS_WRITER, "Reader Proxy removed: " << reader_guid); + rproxy = std::move(*it); + it = matched_datasharing_readers_.erase(it); + break; + } } } - } - locator_selector_general_.locator_selector.remove_entry(reader_guid); - locator_selector_async_.locator_selector.remove_entry(reader_guid); - update_reader_info(locator_selector_general_, false); - update_reader_info(locator_selector_async_, false); + if (rproxy == nullptr) + { + for (ReaderProxyIterator it = matched_remote_readers_.begin(); + it != matched_remote_readers_.end(); ++it) + { + if ((*it)->guid() == reader_guid) + { + EPROSIMA_LOG_INFO(RTPS_WRITER, "Reader Proxy removed: " << reader_guid); + rproxy = std::move(*it); + it = matched_remote_readers_.erase(it); + break; + } + } + } + + locator_selector_general_.locator_selector.remove_entry(reader_guid); + locator_selector_async_.locator_selector.remove_entry(reader_guid); + update_reader_info(locator_selector_general_, false); + update_reader_info(locator_selector_async_, false); + } if (get_matched_readers_size() == 0) { @@ -1277,11 +1280,8 @@ bool StatefulWriter::matched_reader_remove( if (nullptr != listener_) { - // call the listener without locks taken - guard_locator_selector_async.unlock(); - guard_locator_selector_general.unlock(); + // listener is called without locks taken lock.unlock(); - listener_->on_reader_discovery(this, ReaderDiscoveryStatus::REMOVED_READER, reader_guid, nullptr); } diff --git a/test/blackbox/common/DDSBlackboxTestsBasic.cpp b/test/blackbox/common/DDSBlackboxTestsBasic.cpp index 1670ec49b0c..540281d62d5 100644 --- a/test/blackbox/common/DDSBlackboxTestsBasic.cpp +++ b/test/blackbox/common/DDSBlackboxTestsBasic.cpp @@ -1014,6 +1014,43 @@ TEST(DDSBasic, successful_destruction_among_intraprocess_participants) } } } +TEST(DDSBasic, reliable_volatile_writer_secure_builtin_no_potential_deadlock) +{ + // Create + PubSubWriter writer("HelloWorldTopic_no_potential_deadlock"); + PubSubReader reader("HelloWorldTopic_no_potential_deadlock"); + + writer.asynchronously(eprosima::fastdds::dds::ASYNCHRONOUS_PUBLISH_MODE) + .durability_kind(eprosima::fastdds::dds::VOLATILE_DURABILITY_QOS) + .history_kind(eprosima::fastdds::dds::KEEP_LAST_HISTORY_QOS) + .history_depth(20) + .init(); + + ASSERT_TRUE(writer.isInitialized()); + + reader.reliability(eprosima::fastdds::dds::RELIABLE_RELIABILITY_QOS) + .history_kind(eprosima::fastdds::dds::KEEP_LAST_HISTORY_QOS) + .history_depth(20) + .durability_kind(eprosima::fastdds::dds::VOLATILE_DURABILITY_QOS) + .init(); + + ASSERT_TRUE(reader.isInitialized()); + + auto data = default_helloworld_data_generator(30); + + std::thread th([&]() + { + reader.startReception(data); + reader.block_for_at_least(5); + }); + + writer.wait_discovery(); + writer.send(data); + + th.join(); + reader.destroy(); + writer.destroy(); +} } // namespace dds } // namespace fastdds