Index: include/lldb/Core/Broadcaster.h =================================================================== --- include/lldb/Core/Broadcaster.h +++ include/lldb/Core/Broadcaster.h @@ -24,6 +24,8 @@ #include "lldb/lldb-private.h" #include "lldb/Core/ConstString.h" +#include "llvm/ADT/SmallVector.h" + namespace lldb_private { //---------------------------------------------------------------------- @@ -615,12 +617,11 @@ //------------------------------------------------------------------ // //------------------------------------------------------------------ - typedef std::list< std::pair > collection; + typedef llvm::SmallVector, 4> collection; typedef std::map event_names_map; - void - ListenerIterator (std::function const &callback); - + llvm::SmallVector, 4> + GetListeners(); Broadcaster &m_broadcaster; ///< The broadcsater that this implements event_names_map m_event_names; ///< Optionally define event names for readability and logging for each event bit Index: source/Core/Broadcaster.cpp =================================================================== --- source/Core/Broadcaster.cpp +++ source/Core/Broadcaster.cpp @@ -56,31 +56,25 @@ } } -void -Broadcaster::BroadcasterImpl::ListenerIterator (std::function const &callback) -{ - // Private iterator that should be used by everyone except BroadcasterImpl::RemoveListener(). - // We have weak pointers to our listeners which means that at any point the listener can - // expire which means we will need to take it out of our list. To take care of this, we - // iterate and check that the weak pointer can be made into a valid shared pointer before - // we call the callback. If the weak pointer has expired, we remove it from our list. - collection::iterator pos = m_listeners.begin(); - while (pos != m_listeners.end()) +llvm::SmallVector, 4> +Broadcaster::BroadcasterImpl::GetListeners() +{ + llvm::SmallVector, 4> listeners; + listeners.reserve(m_listeners.size()); + + for (auto it = m_listeners.begin(); it != m_listeners.end();) { - lldb::ListenerSP curr_listener_sp(pos->first.lock()); - if (curr_listener_sp) + lldb::ListenerSP curr_listener_sp(it->first.lock()); + if (curr_listener_sp && it->second) { - if (callback(curr_listener_sp, pos->second)) - ++pos; // Keep iterating - else - return; // Done iterating + listeners.emplace_back(std::move(curr_listener_sp), it->second); + ++it; } else - { - // The listener has been expired. Remove this entry. - pos = m_listeners.erase(pos); - } + it = m_listeners.erase(it); } + + return listeners; } void @@ -91,11 +85,8 @@ // Make sure the listener forgets about this broadcaster. We do // this in the broadcaster in case the broadcaster object initiates // the removal. - - ListenerIterator([this](const lldb::ListenerSP &curr_listener_sp, uint32_t &curr_event_mask) -> bool { - curr_listener_sp->BroadcasterWillDestruct (&m_broadcaster); - return true; // Keep iterating - }); + for(auto &pair : GetListeners()) + pair.first->BroadcasterWillDestruct (&m_broadcaster); m_listeners.clear(); } @@ -154,16 +145,16 @@ bool handled = false; - ListenerIterator([this, &listener_sp, &handled, event_mask](const lldb::ListenerSP &curr_listener_sp, uint32_t &curr_event_mask) -> bool { - if (curr_listener_sp == listener_sp) + for(auto &pair: GetListeners()) + { + if (pair.first == listener_sp) { handled = true; - curr_event_mask |= event_mask; + pair.second |= event_mask; m_broadcaster.AddInitialEventsToListener (listener_sp, event_mask); - return false; // Stop iterating + break; } - return true; // Keep iterating - }); + } if (!handled) { @@ -187,55 +178,27 @@ if (!m_hijacking_listeners.empty() && event_type & m_hijacking_masks.back()) return true; - if (m_listeners.empty()) - return false; - - bool result = false; - ListenerIterator([this, event_type, &result](const lldb::ListenerSP &curr_listener_sp, uint32_t &curr_event_mask) -> bool { - - if (curr_event_mask & event_type) - { - result = true; - return false; // Stop iterating - } - else - { - return true; // Keep iterating - } - }); - return result; + for(auto &pair: GetListeners()) + { + if (pair.second & event_type) + return true; + } + return false; } bool Broadcaster::BroadcasterImpl::RemoveListener (lldb_private::Listener *listener, uint32_t event_mask) { - if (listener) + if (!listener) + return false; + + std::lock_guard guard(m_listeners_mutex); + for (auto &pair : GetListeners()) { - std::lock_guard guard(m_listeners_mutex); - collection::iterator pos = m_listeners.begin(); - // See if we already have this listener, and if so, update its mask - while (pos != m_listeners.end()) + if (pair.first.get() == listener) { - lldb::ListenerSP curr_listener_sp(pos->first.lock()); - if (curr_listener_sp) - { - if (curr_listener_sp.get() == listener) - { - // Relinquish all event bits in "event_mask" - pos->second &= ~event_mask; - // If all bits have been relinquished then remove this listener - if (pos->second == 0) - m_listeners.erase (pos); - return true; - } - ++pos; - } - else - { - // The listener has been destroyed since we couldn't turn the std::weak_ptr - // into a valid shared pointer, so we can remove it. - pos = m_listeners.erase (pos); - } + pair.second &= ~event_mask; + return true; } } return false; @@ -302,16 +265,15 @@ } else { + for (auto &pair : GetListeners()) + { + if (!(pair.second & event_type)) + continue; + if (unique && pair.first->PeekAtNextEventForBroadcasterWithType (&m_broadcaster, event_type)) + continue; - ListenerIterator([this, unique, event_type, &event_sp](const lldb::ListenerSP &curr_listener_sp, uint32_t &curr_event_mask) -> bool { - - if (event_type & curr_event_mask) - { - if (!unique || curr_listener_sp->PeekAtNextEventForBroadcasterWithType (&m_broadcaster, event_type) == nullptr) - curr_listener_sp->AddEvent (event_sp); - } - return true; // Keep iterating - }); + pair.first->AddEvent (event_sp); + } } } Index: unittests/Core/BroadcasterTest.cpp =================================================================== --- /dev/null +++ unittests/Core/BroadcasterTest.cpp @@ -0,0 +1,72 @@ +//===-- BroadcasterTest.cpp -------------------------------------*- C++ -*-===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#include "gtest/gtest.h" + +#include "lldb/Core/Broadcaster.h" +#include "lldb/Core/Listener.h" +#include "lldb/Host/Predicate.h" + +#include + +using namespace lldb; +using namespace lldb_private; + +TEST(BroadcasterTest, BroadcastEvent) +{ + EventSP event_sp; + Broadcaster broadcaster(nullptr, "test-broadcaster"); + + // Create a listener, sign it up, make sure it recieves an event. + ListenerSP listener1_sp = Listener::MakeListener("test-listener1"); + const uint32_t event_mask1 = 1; + EXPECT_EQ(event_mask1, listener1_sp->StartListeningForEvents(&broadcaster, event_mask1)); + broadcaster.BroadcastEvent(event_mask1, nullptr); + EXPECT_TRUE(listener1_sp->GetNextEvent(event_sp)); + EXPECT_EQ(event_mask1, event_sp->GetType()); + + { + // Add one more listener, make sure it works as well. + ListenerSP listener2_sp = Listener::MakeListener("test-listener2"); + const uint32_t event_mask2 = 1; + EXPECT_EQ(event_mask2, listener2_sp->StartListeningForEvents(&broadcaster, event_mask1 | event_mask2)); + broadcaster.BroadcastEvent(event_mask2, nullptr); + EXPECT_TRUE(listener2_sp->GetNextEvent(event_sp)); + EXPECT_EQ(event_mask2, event_sp->GetType()); + + // Both listeners should get this event. + broadcaster.BroadcastEvent(event_mask1, nullptr); + EXPECT_TRUE(listener1_sp->GetNextEvent(event_sp)); + EXPECT_EQ(event_mask1, event_sp->GetType()); + EXPECT_TRUE(listener2_sp->GetNextEvent(event_sp)); + EXPECT_EQ(event_mask2, event_sp->GetType()); + } + + // Now again only one listener should be active. + broadcaster.BroadcastEvent(event_mask1, nullptr); + EXPECT_TRUE(listener1_sp->GetNextEvent(event_sp)); + EXPECT_EQ(event_mask1, event_sp->GetType()); +} + +TEST(BroadcasterTest, EventTypeHasListeners) +{ + EventSP event_sp; + Broadcaster broadcaster(nullptr, "test-broadcaster"); + + const uint32_t event_mask = 1; + EXPECT_FALSE(broadcaster.EventTypeHasListeners(event_mask)); + + { + ListenerSP listener_sp = Listener::MakeListener("test-listener"); + EXPECT_EQ(event_mask, listener_sp->StartListeningForEvents(&broadcaster, event_mask)); + EXPECT_TRUE(broadcaster.EventTypeHasListeners(event_mask)); + } + + EXPECT_FALSE(broadcaster.EventTypeHasListeners(event_mask)); +} Index: unittests/Core/CMakeLists.txt =================================================================== --- unittests/Core/CMakeLists.txt +++ unittests/Core/CMakeLists.txt @@ -1,4 +1,5 @@ add_lldb_unittest(LLDBCoreTests + BroadcasterTest.cpp DataExtractorTest.cpp ScalarTest.cpp )