1 //===-- Listener.cpp --------------------------------------------*- C++ -*-===//
3 // The LLVM Compiler Infrastructure
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
8 //===----------------------------------------------------------------------===//
10 #include "lldb/Core/Listener.h"
14 // Other libraries and framework includes
16 #include "lldb/Core/Broadcaster.h"
17 #include "lldb/Core/Log.h"
18 #include "lldb/Core/StreamString.h"
19 #include "lldb/Core/Event.h"
20 #include "lldb/Host/TimeValue.h"
24 using namespace lldb_private;
26 Listener::Listener(const char *name) :
29 m_broadcasters_mutex (Mutex::eMutexTypeRecursive),
31 m_events_mutex (Mutex::eMutexTypeRecursive),
34 Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_OBJECT));
36 log->Printf ("%p Listener::Listener('%s')",
37 static_cast<void*>(this), m_name.c_str());
42 Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_OBJECT));
43 Mutex::Locker locker (m_broadcasters_mutex);
45 size_t num_managers = m_broadcaster_managers.size();
47 for (size_t i = 0; i < num_managers; i++)
48 m_broadcaster_managers[i]->RemoveListener(*this);
51 log->Printf ("%p Listener::~Listener('%s')",
52 static_cast<void*>(this), m_name.c_str());
59 Mutex::Locker locker(m_broadcasters_mutex);
60 broadcaster_collection::iterator pos, end = m_broadcasters.end();
61 for (pos = m_broadcasters.begin(); pos != end; ++pos)
62 pos->first->RemoveListener (this, pos->second.event_mask);
63 m_broadcasters.clear();
64 m_cond_wait.SetValue (false, eBroadcastNever);
65 m_broadcasters.clear();
66 Mutex::Locker event_locker(m_events_mutex);
71 Listener::StartListeningForEvents (Broadcaster* broadcaster, uint32_t event_mask)
76 // Tell the broadcaster to add this object as a listener
78 Mutex::Locker locker(m_broadcasters_mutex);
79 m_broadcasters.insert(std::make_pair(broadcaster, BroadcasterInfo(event_mask)));
82 uint32_t acquired_mask = broadcaster->AddListener (this, event_mask);
84 if (event_mask != acquired_mask)
88 Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_EVENTS));
90 log->Printf ("%p Listener::StartListeningForEvents (broadcaster = %p, mask = 0x%8.8x) acquired_mask = 0x%8.8x for %s",
91 static_cast<void*>(this),
92 static_cast<void*>(broadcaster), event_mask,
93 acquired_mask, m_name.c_str());
102 Listener::StartListeningForEvents (Broadcaster* broadcaster, uint32_t event_mask, HandleBroadcastCallback callback, void *callback_user_data)
106 // Scope for "locker"
107 // Tell the broadcaster to add this object as a listener
109 Mutex::Locker locker(m_broadcasters_mutex);
110 m_broadcasters.insert(std::make_pair(broadcaster, BroadcasterInfo(event_mask, callback, callback_user_data)));
113 uint32_t acquired_mask = broadcaster->AddListener (this, event_mask);
115 Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_EVENTS));
118 void **pointer = reinterpret_cast<void**>(&callback);
119 log->Printf ("%p Listener::StartListeningForEvents (broadcaster = %p, mask = 0x%8.8x, callback = %p, user_data = %p) acquired_mask = 0x%8.8x for %s",
120 static_cast<void*>(this),
121 static_cast<void*>(broadcaster), event_mask, *pointer,
122 static_cast<void*>(callback_user_data), acquired_mask,
126 return acquired_mask;
132 Listener::StopListeningForEvents (Broadcaster* broadcaster, uint32_t event_mask)
136 // Scope for "locker"
138 Mutex::Locker locker(m_broadcasters_mutex);
139 m_broadcasters.erase (broadcaster);
141 // Remove the broadcaster from our set of broadcasters
142 return broadcaster->RemoveListener (this, event_mask);
148 // Called when a Broadcaster is in its destructor. We need to remove all
149 // knowledge of this broadcaster and any events that it may have queued up
151 Listener::BroadcasterWillDestruct (Broadcaster *broadcaster)
153 // Scope for "broadcasters_locker"
155 Mutex::Locker broadcasters_locker(m_broadcasters_mutex);
156 m_broadcasters.erase (broadcaster);
159 // Scope for "event_locker"
161 Mutex::Locker event_locker(m_events_mutex);
162 // Remove all events for this broadcaster object.
163 event_collection::iterator pos = m_events.begin();
164 while (pos != m_events.end())
166 if ((*pos)->GetBroadcaster() == broadcaster)
167 pos = m_events.erase(pos);
172 if (m_events.empty())
173 m_cond_wait.SetValue (false, eBroadcastNever);
179 Listener::BroadcasterManagerWillDestruct (BroadcasterManager *manager)
181 // Just need to remove this broadcast manager from the list of managers:
182 broadcaster_manager_collection::iterator iter, end_iter = m_broadcaster_managers.end();
183 iter = find(m_broadcaster_managers.begin(), end_iter, manager);
184 if (iter != end_iter)
185 m_broadcaster_managers.erase (iter);
189 Listener::AddEvent (EventSP &event_sp)
191 Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_EVENTS));
193 log->Printf ("%p Listener('%s')::AddEvent (event_sp = {%p})",
194 static_cast<void*>(this), m_name.c_str(),
195 static_cast<void*>(event_sp.get()));
197 // Scope for "locker"
199 Mutex::Locker locker(m_events_mutex);
200 m_events.push_back (event_sp);
202 m_cond_wait.SetValue (true, eBroadcastAlways);
205 class EventBroadcasterMatches
208 EventBroadcasterMatches (Broadcaster *broadcaster) :
209 m_broadcaster (broadcaster) {
212 bool operator() (const EventSP &event_sp) const
214 if (event_sp->BroadcasterIs(m_broadcaster))
221 Broadcaster *m_broadcaster;
228 EventMatcher (Broadcaster *broadcaster, const ConstString *broadcaster_names, uint32_t num_broadcaster_names, uint32_t event_type_mask) :
229 m_broadcaster (broadcaster),
230 m_broadcaster_names (broadcaster_names),
231 m_num_broadcaster_names (num_broadcaster_names),
232 m_event_type_mask (event_type_mask)
236 bool operator() (const EventSP &event_sp) const
238 if (m_broadcaster && !event_sp->BroadcasterIs(m_broadcaster))
241 if (m_broadcaster_names)
243 bool found_source = false;
244 const ConstString &event_broadcaster_name = event_sp->GetBroadcaster()->GetBroadcasterName();
245 for (uint32_t i=0; i<m_num_broadcaster_names; ++i)
247 if (m_broadcaster_names[i] == event_broadcaster_name)
257 if (m_event_type_mask == 0 || m_event_type_mask & event_sp->GetType())
263 Broadcaster *m_broadcaster;
264 const ConstString *m_broadcaster_names;
265 const uint32_t m_num_broadcaster_names;
266 const uint32_t m_event_type_mask;
271 Listener::FindNextEventInternal
273 Broadcaster *broadcaster, // NULL for any broadcaster
274 const ConstString *broadcaster_names, // NULL for any event
275 uint32_t num_broadcaster_names,
276 uint32_t event_type_mask,
280 Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_EVENTS));
282 Mutex::Locker lock(m_events_mutex);
284 if (m_events.empty())
288 Listener::event_collection::iterator pos = m_events.end();
290 if (broadcaster == NULL && broadcaster_names == NULL && event_type_mask == 0)
292 pos = m_events.begin();
296 pos = std::find_if (m_events.begin(), m_events.end(), EventMatcher (broadcaster, broadcaster_names, num_broadcaster_names, event_type_mask));
299 if (pos != m_events.end())
304 log->Printf ("%p '%s' Listener::FindNextEventInternal(broadcaster=%p, broadcaster_names=%p[%u], event_type_mask=0x%8.8x, remove=%i) event %p",
305 static_cast<void*>(this), GetName(),
306 static_cast<void*>(broadcaster),
307 static_cast<const void*>(broadcaster_names),
308 num_broadcaster_names, event_type_mask, remove,
309 static_cast<void*>(event_sp.get()));
315 if (m_events.empty())
316 m_cond_wait.SetValue (false, eBroadcastNever);
319 // Unlock the event queue here. We've removed this event and are about to return
320 // it so it should be okay to get the next event off the queue here - and it might
321 // be useful to do that in the "DoOnRemoval".
324 // Don't call DoOnRemoval if you aren't removing the event...
326 event_sp->DoOnRemoval();
336 Listener::PeekAtNextEvent ()
339 if (FindNextEventInternal (NULL, NULL, 0, 0, event_sp, false))
340 return event_sp.get();
345 Listener::PeekAtNextEventForBroadcaster (Broadcaster *broadcaster)
348 if (FindNextEventInternal (broadcaster, NULL, 0, 0, event_sp, false))
349 return event_sp.get();
354 Listener::PeekAtNextEventForBroadcasterWithType (Broadcaster *broadcaster, uint32_t event_type_mask)
357 if (FindNextEventInternal (broadcaster, NULL, 0, event_type_mask, event_sp, false))
358 return event_sp.get();
364 Listener::GetNextEventInternal
366 Broadcaster *broadcaster, // NULL for any broadcaster
367 const ConstString *broadcaster_names, // NULL for any event
368 uint32_t num_broadcaster_names,
369 uint32_t event_type_mask,
373 return FindNextEventInternal (broadcaster, broadcaster_names, num_broadcaster_names, event_type_mask, event_sp, true);
377 Listener::GetNextEvent (EventSP &event_sp)
379 return GetNextEventInternal (NULL, NULL, 0, 0, event_sp);
384 Listener::GetNextEventForBroadcaster (Broadcaster *broadcaster, EventSP &event_sp)
386 return GetNextEventInternal (broadcaster, NULL, 0, 0, event_sp);
390 Listener::GetNextEventForBroadcasterWithType (Broadcaster *broadcaster, uint32_t event_type_mask, EventSP &event_sp)
392 return GetNextEventInternal (broadcaster, NULL, 0, event_type_mask, event_sp);
397 Listener::WaitForEventsInternal
399 const TimeValue *timeout,
400 Broadcaster *broadcaster, // NULL for any broadcaster
401 const ConstString *broadcaster_names, // NULL for any event
402 uint32_t num_broadcaster_names,
403 uint32_t event_type_mask,
407 Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_EVENTS));
408 bool timed_out = false;
411 log->Printf ("%p Listener::WaitForEventsInternal (timeout = { %p }) for %s",
412 static_cast<void*>(this), static_cast<const void*>(timeout),
417 // Note, we don't want to lock the m_events_mutex in the call to GetNextEventInternal, since the DoOnRemoval
418 // code might require that new events be serviced. For instance, the Breakpoint Command's
419 if (GetNextEventInternal (broadcaster, broadcaster_names, num_broadcaster_names, event_type_mask, event_sp))
423 // Reset condition value to false, so we can wait for new events to be
424 // added that might meet our current filter
425 // But first poll for any new event that might satisfy our condition, and if so consume it,
428 Mutex::Locker event_locker(m_events_mutex);
429 const bool remove = false;
430 if (FindNextEventInternal (broadcaster, broadcaster_names, num_broadcaster_names, event_type_mask, event_sp, remove))
433 m_cond_wait.SetValue (false, eBroadcastNever);
436 if (m_cond_wait.WaitForValueEqualTo (true, timeout, &timed_out))
441 log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_EVENTS);
443 log->Printf ("%p Listener::WaitForEventsInternal() timed out for %s",
444 static_cast<void*>(this), m_name.c_str());
449 log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_EVENTS);
451 log->Printf ("%p Listener::WaitForEventsInternal() unknown error for %s",
452 static_cast<void*>(this), m_name.c_str());
461 Listener::WaitForEventForBroadcasterWithType
463 const TimeValue *timeout,
464 Broadcaster *broadcaster,
465 uint32_t event_type_mask,
469 return WaitForEventsInternal (timeout, broadcaster, NULL, 0, event_type_mask, event_sp);
473 Listener::WaitForEventForBroadcaster
475 const TimeValue *timeout,
476 Broadcaster *broadcaster,
480 return WaitForEventsInternal (timeout, broadcaster, NULL, 0, 0, event_sp);
484 Listener::WaitForEvent (const TimeValue *timeout, EventSP &event_sp)
486 return WaitForEventsInternal (timeout, NULL, NULL, 0, 0, event_sp);
489 //Listener::broadcaster_collection::iterator
490 //Listener::FindBroadcasterWithMask (Broadcaster *broadcaster, uint32_t event_mask, bool exact)
492 // broadcaster_collection::iterator pos;
493 // broadcaster_collection::iterator end = m_broadcasters.end();
494 // for (pos = m_broadcasters.find (broadcaster);
495 // pos != end && pos->first == broadcaster;
500 // if ((event_mask & pos->second.event_mask) == event_mask)
505 // if (event_mask & pos->second.event_mask)
513 Listener::HandleBroadcastEvent (EventSP &event_sp)
515 size_t num_handled = 0;
516 Mutex::Locker locker(m_broadcasters_mutex);
517 Broadcaster *broadcaster = event_sp->GetBroadcaster();
518 broadcaster_collection::iterator pos;
519 broadcaster_collection::iterator end = m_broadcasters.end();
520 for (pos = m_broadcasters.find (broadcaster);
521 pos != end && pos->first == broadcaster;
524 BroadcasterInfo info = pos->second;
525 if (event_sp->GetType () & info.event_mask)
527 if (info.callback != NULL)
529 info.callback (event_sp, info.callback_user_data);
538 Listener::StartListeningForEventSpec (BroadcasterManager &manager,
539 const BroadcastEventSpec &event_spec)
541 // The BroadcasterManager mutex must be locked before m_broadcasters_mutex
542 // to avoid violating the lock hierarchy (manager before broadcasters).
543 Mutex::Locker manager_locker(manager.m_manager_mutex);
544 Mutex::Locker locker(m_broadcasters_mutex);
546 uint32_t bits_acquired = manager.RegisterListenerForEvents(*this, event_spec);
548 m_broadcaster_managers.push_back(&manager);
550 return bits_acquired;
554 Listener::StopListeningForEventSpec (BroadcasterManager &manager,
555 const BroadcastEventSpec &event_spec)
557 Mutex::Locker locker(m_broadcasters_mutex);
558 return manager.UnregisterListenerForEvents (*this, event_spec);