<div dir="ltr">I committed in a fix as rL262925 (initialization order issue in a log line)</div><br><div class="gmail_quote"><div dir="ltr">On Tue, Mar 8, 2016 at 3:04 AM Jim Ingham via lldb-commits <<a href="mailto:lldb-commits@lists.llvm.org">lldb-commits@lists.llvm.org</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">I can’t get to a linux box right now, and I can’t tell from the bot logs what is going wrong. This change was pretty much all in generic code, the only places it touched platform specific code was to make sure the Process plugins used a Listener Shared pointer not a listener reference, and mechanical changes wherever folks had to make Listeners. So I’m kind of surprised this is causing problems.<br>
<br>
If somebody working on the Linux side could take a look at what’s going wrong, that would be great. Otherwise feel free to revert it and I’ll look into it when I can.<br>
<br>
Jim<br>
<br>
<br>
> On Mar 7, 2016, at 5:40 PM, Siva Chandra <<a href="mailto:sivachandra@google.com" target="_blank">sivachandra@google.com</a>> wrote:<br>
><br>
> I think this broke everything on Linux:<br>
> <a href="http://lab.llvm.org:8011/builders/lldb-x86_64-ubuntu-14.04-cmake/builds/12154" rel="noreferrer" target="_blank">http://lab.llvm.org:8011/builders/lldb-x86_64-ubuntu-14.04-cmake/builds/12154</a><br>
><br>
> On Mon, Mar 7, 2016 at 1:50 PM, Jim Ingham via lldb-commits<br>
> <<a href="mailto:lldb-commits@lists.llvm.org" target="_blank">lldb-commits@lists.llvm.org</a>> wrote:<br>
>> Author: jingham<br>
>> Date: Mon Mar 7 15:50:25 2016<br>
>> New Revision: 262863<br>
>><br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project?rev=262863&view=rev" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project?rev=262863&view=rev</a><br>
>> Log:<br>
>> Change over the broadcaster/listener process to hold shared or weak pointers<br>
>> to each other. This should remove some infrequent teardown crashes when the<br>
>> listener is not the debugger's listener.<br>
>><br>
>> Processes now need to take a ListenerSP, not a Listener&.<br>
>><br>
>> This required changing over the Process plugin class constructors to take a ListenerSP, instead<br>
>> of a Listener&. Other than that there should be no functional change.<br>
>><br>
>> <rdar://problem/24580184> CrashTracer: [USER] Xcode at …ework: lldb_private::Listener::BroadcasterWillDestruct + 39<br>
>><br>
>> Added:<br>
>> lldb/trunk/packages/Python/lldbsuite/test/api/listeners/<br>
>> lldb/trunk/packages/Python/lldbsuite/test/api/listeners/Makefile<br>
>> lldb/trunk/packages/Python/lldbsuite/test/api/listeners/TestListener.py<br>
>> lldb/trunk/packages/Python/lldbsuite/test/api/listeners/main.c<br>
>> Modified:<br>
>> lldb/trunk/include/lldb/API/SBListener.h<br>
>> lldb/trunk/include/lldb/Core/Broadcaster.h<br>
>> lldb/trunk/include/lldb/Core/Debugger.h<br>
>> lldb/trunk/include/lldb/Core/Event.h<br>
>> lldb/trunk/include/lldb/Core/Listener.h<br>
>> lldb/trunk/include/lldb/Target/Process.h<br>
>> lldb/trunk/include/lldb/Target/ProcessLaunchInfo.h<br>
>> lldb/trunk/include/lldb/Target/Target.h<br>
>> lldb/trunk/include/lldb/lldb-forward.h<br>
>> lldb/trunk/include/lldb/lldb-private-interfaces.h<br>
>> lldb/trunk/source/API/SBBroadcaster.cpp<br>
>> lldb/trunk/source/API/SBDebugger.cpp<br>
>> lldb/trunk/source/API/SBListener.cpp<br>
>> lldb/trunk/source/API/SBTarget.cpp<br>
>> lldb/trunk/source/Core/Broadcaster.cpp<br>
>> lldb/trunk/source/Core/Communication.cpp<br>
>> lldb/trunk/source/Core/Debugger.cpp<br>
>> lldb/trunk/source/Core/Event.cpp<br>
>> lldb/trunk/source/Core/IOHandler.cpp<br>
>> lldb/trunk/source/Core/Listener.cpp<br>
>> lldb/trunk/source/Interpreter/CommandInterpreter.cpp<br>
>> lldb/trunk/source/Plugins/Platform/Linux/PlatformLinux.cpp<br>
>> lldb/trunk/source/Plugins/Platform/POSIX/PlatformPOSIX.cpp<br>
>> lldb/trunk/source/Plugins/Platform/Windows/PlatformWindows.cpp<br>
>> lldb/trunk/source/Plugins/Platform/gdb-server/PlatformRemoteGDBServer.cpp<br>
>> lldb/trunk/source/Plugins/Process/FreeBSD/ProcessFreeBSD.cpp<br>
>> lldb/trunk/source/Plugins/Process/FreeBSD/ProcessFreeBSD.h<br>
>> lldb/trunk/source/Plugins/Process/MacOSX-Kernel/ProcessKDP.cpp<br>
>> lldb/trunk/source/Plugins/Process/MacOSX-Kernel/ProcessKDP.h<br>
>> lldb/trunk/source/Plugins/Process/Windows/Common/ProcessWindows.cpp<br>
>> lldb/trunk/source/Plugins/Process/Windows/Common/ProcessWindows.h<br>
>> lldb/trunk/source/Plugins/Process/Windows/Live/ProcessWindowsLive.cpp<br>
>> lldb/trunk/source/Plugins/Process/Windows/Live/ProcessWindowsLive.h<br>
>> lldb/trunk/source/Plugins/Process/Windows/MiniDump/ProcessWinMiniDump.cpp<br>
>> lldb/trunk/source/Plugins/Process/Windows/MiniDump/ProcessWinMiniDump.h<br>
>> lldb/trunk/source/Plugins/Process/elf-core/ProcessElfCore.cpp<br>
>> lldb/trunk/source/Plugins/Process/elf-core/ProcessElfCore.h<br>
>> lldb/trunk/source/Plugins/Process/gdb-remote/GDBRemoteCommunicationClient.cpp<br>
>> lldb/trunk/source/Plugins/Process/gdb-remote/ProcessGDBRemote.cpp<br>
>> lldb/trunk/source/Plugins/Process/gdb-remote/ProcessGDBRemote.h<br>
>> lldb/trunk/source/Plugins/Process/mach-core/ProcessMachCore.cpp<br>
>> lldb/trunk/source/Plugins/Process/mach-core/ProcessMachCore.h<br>
>> lldb/trunk/source/Target/Process.cpp<br>
>> lldb/trunk/source/Target/ProcessLaunchInfo.cpp<br>
>> lldb/trunk/source/Target/Target.cpp<br>
>> lldb/trunk/source/Target/TargetList.cpp<br>
>> lldb/trunk/source/Target/Thread.cpp<br>
>><br>
>> Modified: lldb/trunk/include/lldb/API/SBListener.h<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/API/SBListener.h?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/API/SBListener.h?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/include/lldb/API/SBListener.h (original)<br>
>> +++ lldb/trunk/include/lldb/API/SBListener.h Mon Mar 7 15:50:25 2016<br>
>> @@ -106,8 +106,6 @@ protected:<br>
>> friend class SBLaunchInfo;<br>
>> friend class SBTarget;<br>
>><br>
>> - SBListener (lldb_private::Listener &listener);<br>
>> -<br>
>> SBListener (const lldb::ListenerSP &listener_sp);<br>
>><br>
>> lldb::ListenerSP<br>
>> @@ -124,20 +122,11 @@ private:<br>
>> lldb_private::Listener *<br>
>> get() const;<br>
>><br>
>> - lldb_private::Listener &<br>
>> - ref() const;<br>
>> -<br>
>> - lldb_private::Listener &<br>
>> - operator *();<br>
>> -<br>
>> - const lldb_private::Listener &<br>
>> - operator *() const;<br>
>> -<br>
>> void<br>
>> - reset(lldb_private::Listener *listener, bool transfer_ownership);<br>
>> + reset(lldb::ListenerSP listener_sp);<br>
>><br>
>> lldb::ListenerSP m_opaque_sp;<br>
>> - lldb_private::Listener *m_opaque_ptr;<br>
>> + lldb_private::Listener *m_unused_ptr;<br>
>> };<br>
>><br>
>> } // namespace lldb<br>
>><br>
>> Modified: lldb/trunk/include/lldb/Core/Broadcaster.h<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Core/Broadcaster.h?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Core/Broadcaster.h?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/include/lldb/Core/Broadcaster.h (original)<br>
>> +++ lldb/trunk/include/lldb/Core/Broadcaster.h Mon Mar 7 15:50:25 2016<br>
>> @@ -19,9 +19,8 @@<br>
>> // Other libraries and framework includes<br>
>> // Project includes<br>
>> #include "lldb/lldb-private.h"<br>
>> -//#include "lldb/Core/Flags.h"<br>
>> #include "lldb/Core/ConstString.h"<br>
>> -#include "lldb/Core/Listener.h"<br>
>> +#include "lldb/Host/Mutex.h"<br>
>><br>
>> namespace lldb_private {<br>
>><br>
>> @@ -82,37 +81,47 @@ private:<br>
>> uint32_t m_event_bits;<br>
>> };<br>
>><br>
>> -class BroadcasterManager<br>
>> +class BroadcasterManager :<br>
>> + public std::enable_shared_from_this<BroadcasterManager><br>
>> {<br>
>> public:<br>
>> friend class Listener;<br>
>><br>
>> +protected:<br>
>> BroadcasterManager ();<br>
>> -<br>
>> +public:<br>
>> + // Listeners hold onto weak pointers to their broadcaster managers. So they must be<br>
>> + // made into shared pointers, which you do with MakeBroadcasterManager.<br>
>> +<br>
>> + static lldb::BroadcasterManagerSP<br>
>> + MakeBroadcasterManager();<br>
>> +<br>
>> ~BroadcasterManager() = default;<br>
>><br>
>> uint32_t<br>
>> - RegisterListenerForEvents (Listener &listener, BroadcastEventSpec event_spec);<br>
>> + RegisterListenerForEvents (lldb::ListenerSP listener_sp, BroadcastEventSpec event_spec);<br>
>><br>
>> bool<br>
>> - UnregisterListenerForEvents (Listener &listener, BroadcastEventSpec event_spec);<br>
>> + UnregisterListenerForEvents (lldb::ListenerSP listener_sp, BroadcastEventSpec event_spec);<br>
>><br>
>> - Listener *<br>
>> + lldb::ListenerSP<br>
>> GetListenerForEventSpec (BroadcastEventSpec event_spec) const;<br>
>><br>
>> void<br>
>> SignUpListenersForBroadcaster (Broadcaster &broadcaster);<br>
>><br>
>> void<br>
>> - RemoveListener (Listener &Listener);<br>
>> + RemoveListener (lldb::ListenerSP listener_sp);<br>
>> +<br>
>> + void<br>
>> + RemoveListener (Listener *listener);<br>
>><br>
>> -protected:<br>
>> void Clear();<br>
>><br>
>> private:<br>
>> - typedef std::pair<BroadcastEventSpec, Listener *> event_listener_key;<br>
>> - typedef std::map<BroadcastEventSpec, Listener *> collection;<br>
>> - typedef std::set<Listener *> listener_collection;<br>
>> + typedef std::pair<BroadcastEventSpec, lldb::ListenerSP> event_listener_key;<br>
>> + typedef std::map<BroadcastEventSpec, lldb::ListenerSP> collection;<br>
>> + typedef std::set<lldb::ListenerSP> listener_collection;<br>
>> collection m_event_map;<br>
>> listener_collection m_listeners;<br>
>><br>
>> @@ -162,9 +171,9 @@ private:<br>
>> {<br>
>> public:<br>
>> ListenerMatchesAndSharedBits (BroadcastEventSpec broadcaster_spec,<br>
>> - const Listener &listener) :<br>
>> + const lldb::ListenerSP listener_sp) :<br>
>> m_broadcaster_spec (broadcaster_spec),<br>
>> - m_listener (&listener)<br>
>> + m_listener_sp (listener_sp)<br>
>> {<br>
>> }<br>
>><br>
>> @@ -174,19 +183,19 @@ private:<br>
>> {<br>
>> return (input.first.GetBroadcasterClass() == m_broadcaster_spec.GetBroadcasterClass()<br>
>> && (input.first.GetEventBits() & m_broadcaster_spec.GetEventBits()) != 0<br>
>> - && input.second == m_listener);<br>
>> + && input.second == m_listener_sp);<br>
>> }<br>
>><br>
>> private:<br>
>> BroadcastEventSpec m_broadcaster_spec;<br>
>> - const Listener *m_listener;<br>
>> + const lldb::ListenerSP m_listener_sp;<br>
>> };<br>
>><br>
>> class ListenerMatches<br>
>> {<br>
>> public:<br>
>> - ListenerMatches (const Listener &in_listener) :<br>
>> - m_listener (&in_listener)<br>
>> + ListenerMatches (const lldb::ListenerSP in_listener_sp) :<br>
>> + m_listener_sp (in_listener_sp)<br>
>> {<br>
>> }<br>
>><br>
>> @@ -194,7 +203,40 @@ private:<br>
>><br>
>> bool operator () (const event_listener_key input) const<br>
>> {<br>
>> - return (input.second == m_listener);<br>
>> + if (input.second == m_listener_sp)<br>
>> + return true;<br>
>> + else<br>
>> + return false;<br>
>> + }<br>
>> +<br>
>> + private:<br>
>> + const lldb::ListenerSP m_listener_sp;<br>
>> + };<br>
>> +<br>
>> + class ListenerMatchesPointer<br>
>> + {<br>
>> + public:<br>
>> + ListenerMatchesPointer (const Listener *in_listener) :<br>
>> + m_listener (in_listener)<br>
>> + {<br>
>> + }<br>
>> +<br>
>> + ~ListenerMatchesPointer() = default;<br>
>> +<br>
>> + bool operator () (const event_listener_key input) const<br>
>> + {<br>
>> + if (input.second.get() == m_listener)<br>
>> + return true;<br>
>> + else<br>
>> + return false;<br>
>> + }<br>
>> +<br>
>> + bool operator () (const lldb::ListenerSP input) const<br>
>> + {<br>
>> + if (input.get() == m_listener)<br>
>> + return true;<br>
>> + else<br>
>> + return false;<br>
>> }<br>
>><br>
>> private:<br>
>> @@ -237,6 +279,8 @@ private:<br>
>> //----------------------------------------------------------------------<br>
>> class Broadcaster<br>
>> {<br>
>> +friend class Listener;<br>
>> +friend class Event;<br>
>> public:<br>
>> //------------------------------------------------------------------<br>
>> /// Construct with a broadcaster with a name.<br>
>> @@ -245,7 +289,7 @@ public:<br>
>> /// A NULL terminated C string that contains the name of the<br>
>> /// broadcaster object.<br>
>> //------------------------------------------------------------------<br>
>> - Broadcaster (BroadcasterManager *manager, const char *name);<br>
>> + Broadcaster (lldb::BroadcasterManagerSP manager_sp, const char *name);<br>
>><br>
>> //------------------------------------------------------------------<br>
>> /// Destructor.<br>
>> @@ -275,22 +319,37 @@ public:<br>
>> ///<br>
>> //------------------------------------------------------------------<br>
>> void<br>
>> - BroadcastEvent (lldb::EventSP &event_sp);<br>
>> + BroadcastEvent (lldb::EventSP &event_sp)<br>
>> + {<br>
>> + m_broadcaster_sp->BroadcastEvent(event_sp);<br>
>> + }<br>
>><br>
>> void<br>
>> - BroadcastEventIfUnique (lldb::EventSP &event_sp);<br>
>> + BroadcastEventIfUnique (lldb::EventSP &event_sp)<br>
>> + {<br>
>> + m_broadcaster_sp->BroadcastEventIfUnique(event_sp);<br>
>> + }<br>
>><br>
>> void<br>
>> - BroadcastEvent(uint32_t event_type, EventData *event_data = nullptr);<br>
>> + BroadcastEvent(uint32_t event_type, EventData *event_data = nullptr)<br>
>> + {<br>
>> + m_broadcaster_sp->BroadcastEvent(event_type, event_data);<br>
>> + }<br>
>><br>
>> void<br>
>> - BroadcastEventIfUnique(uint32_t event_type, EventData *event_data = nullptr);<br>
>> + BroadcastEventIfUnique(uint32_t event_type, EventData *event_data = nullptr)<br>
>> + {<br>
>> + m_broadcaster_sp->BroadcastEventIfUnique(event_type, event_data);<br>
>> + }<br>
>><br>
>> void<br>
>> - Clear();<br>
>> + Clear()<br>
>> + {<br>
>> + m_broadcaster_sp->Clear();<br>
>> + }<br>
>><br>
>> virtual void<br>
>> - AddInitialEventsToListener (Listener *listener, uint32_t requested_events);<br>
>> + AddInitialEventsToListener (lldb::ListenerSP listener_sp, uint32_t requested_events);<br>
>><br>
>> //------------------------------------------------------------------<br>
>> /// Listen for any events specified by \a event_mask.<br>
>> @@ -315,7 +374,10 @@ public:<br>
>> /// The actual event bits that were acquired by \a listener.<br>
>> //------------------------------------------------------------------<br>
>> uint32_t<br>
>> - AddListener (Listener* listener, uint32_t event_mask);<br>
>> + AddListener (lldb::ListenerSP listener_sp, uint32_t event_mask)<br>
>> + {<br>
>> + return m_broadcaster_sp->AddListener(listener_sp, event_mask);<br>
>> + }<br>
>><br>
>> //------------------------------------------------------------------<br>
>> /// Get the NULL terminated C string name of this Broadcaster<br>
>> @@ -325,7 +387,10 @@ public:<br>
>> /// The NULL terminated C string name of this Broadcaster.<br>
>> //------------------------------------------------------------------<br>
>> const ConstString &<br>
>> - GetBroadcasterName ();<br>
>> + GetBroadcasterName ()<br>
>> + {<br>
>> + return m_broadcaster_name;<br>
>> + }<br>
>><br>
>> //------------------------------------------------------------------<br>
>> /// Get the event name(s) for one or more event bits.<br>
>> @@ -337,7 +402,10 @@ public:<br>
>> /// The NULL terminated C string name of this Broadcaster.<br>
>> //------------------------------------------------------------------<br>
>> bool<br>
>> - GetEventNames (Stream &s, const uint32_t event_mask, bool prefix_with_broadcaster_name) const;<br>
>> + GetEventNames (Stream &s, const uint32_t event_mask, bool prefix_with_broadcaster_name) const<br>
>> + {<br>
>> + return m_broadcaster_sp->GetEventNames(s, event_mask, prefix_with_broadcaster_name);<br>
>> + }<br>
>><br>
>> //------------------------------------------------------------------<br>
>> /// Set the name for an event bit.<br>
>> @@ -352,20 +420,20 @@ public:<br>
>> void<br>
>> SetEventName (uint32_t event_mask, const char *name)<br>
>> {<br>
>> - m_event_names[event_mask] = name;<br>
>> + m_broadcaster_sp->SetEventName(event_mask, name);<br>
>> }<br>
>><br>
>> const char *<br>
>> GetEventName (uint32_t event_mask) const<br>
>> {<br>
>> - const auto pos = m_event_names.find (event_mask);<br>
>> - if (pos != m_event_names.end())<br>
>> - return pos->second.c_str();<br>
>> - return nullptr;<br>
>> + return m_broadcaster_sp->GetEventName(event_mask);<br>
>> }<br>
>><br>
>> bool<br>
>> - EventTypeHasListeners (uint32_t event_type);<br>
>> + EventTypeHasListeners (uint32_t event_type)<br>
>> + {<br>
>> + return m_broadcaster_sp->EventTypeHasListeners(event_type);<br>
>> + }<br>
>><br>
>> //------------------------------------------------------------------<br>
>> /// Removes a Listener from this broadcasters list and frees the<br>
>> @@ -386,7 +454,10 @@ public:<br>
>> /// @see uint32_t Broadcaster::AddListener (Listener*, uint32_t)<br>
>> //------------------------------------------------------------------<br>
>> bool<br>
>> - RemoveListener (Listener* listener, uint32_t event_mask = UINT32_MAX);<br>
>> + RemoveListener (lldb::ListenerSP listener_sp, uint32_t event_mask = UINT32_MAX)<br>
>> + {<br>
>> + return m_broadcaster_sp->RemoveListener(listener_sp, event_mask);<br>
>> + }<br>
>><br>
>> //------------------------------------------------------------------<br>
>> /// Provides a simple mechanism to temporarily redirect events from<br>
>> @@ -410,17 +481,26 @@ public:<br>
>> /// @see uint32_t Broadcaster::AddListener (Listener*, uint32_t)<br>
>> //------------------------------------------------------------------<br>
>> bool<br>
>> - HijackBroadcaster (Listener *listener, uint32_t event_mask = UINT32_MAX);<br>
>> + HijackBroadcaster (lldb::ListenerSP listener_sp, uint32_t event_mask = UINT32_MAX)<br>
>> + {<br>
>> + return m_broadcaster_sp->HijackBroadcaster(listener_sp, event_mask);<br>
>> + }<br>
>><br>
>> bool<br>
>> - IsHijackedForEvent (uint32_t event_mask);<br>
>> + IsHijackedForEvent (uint32_t event_mask)<br>
>> + {<br>
>> + return m_broadcaster_sp->IsHijackedForEvent(event_mask);<br>
>> + }<br>
>><br>
>> //------------------------------------------------------------------<br>
>> /// Restore the state of the Broadcaster from a previous hijack attempt.<br>
>> ///<br>
>> //------------------------------------------------------------------<br>
>> void<br>
>> - RestoreBroadcaster ();<br>
>> + RestoreBroadcaster ()<br>
>> + {<br>
>> + m_broadcaster_sp->RestoreBroadcaster();<br>
>> + }<br>
>><br>
>> // This needs to be filled in if you are going to register the broadcaster with the broadcaster<br>
>> // manager and do broadcaster class matching.<br>
>> @@ -428,32 +508,145 @@ public:<br>
>> // with the BroadcasterManager, so that it is clearer how to add one.<br>
>> virtual ConstString &GetBroadcasterClass() const;<br>
>><br>
>> - BroadcasterManager *GetManager();<br>
>> + lldb::BroadcasterManagerSP GetManager();<br>
>><br>
>> protected:<br>
>> - void<br>
>> - PrivateBroadcastEvent (lldb::EventSP &event_sp, bool unique);<br>
>> + // BroadcasterImpl contains the actual Broadcaster implementation. The Broadcaster makes a BroadcasterImpl<br>
>> + // which lives as long as it does. The Listeners & the Events hold a weak pointer to the BroadcasterImpl,<br>
>> + // so that they can survive if a Broadcaster they were listening to is destroyed w/o their being able to<br>
>> + // unregister from it (which can happen if the Broadcasters & Listeners are being destroyed on separate threads<br>
>> + // simultaneously.<br>
>> + // The Broadcaster itself can't be shared out as a weak pointer, because some things that are broadcasters<br>
>> + // (e.g. the Target and the Process) are shared in their own right.<br>
>> + //<br>
>> + // For the most part, the Broadcaster functions dispatch to the BroadcasterImpl, and are documented in the<br>
>> + // public Broadcaster API above.<br>
>> +<br>
>> +<br>
>> + class BroadcasterImpl<br>
>> + {<br>
>> + friend class Listener;<br>
>> + friend class Broadcaster;<br>
>> + public:<br>
>> + BroadcasterImpl (Broadcaster &broadcaster);<br>
>> +<br>
>> + ~BroadcasterImpl() = default;<br>
>> +<br>
>> + void<br>
>> + BroadcastEvent (lldb::EventSP &event_sp);<br>
>> +<br>
>> + void<br>
>> + BroadcastEventIfUnique (lldb::EventSP &event_sp);<br>
>> +<br>
>> + void<br>
>> + BroadcastEvent(uint32_t event_type, EventData *event_data = nullptr);<br>
>> +<br>
>> + void<br>
>> + BroadcastEventIfUnique(uint32_t event_type, EventData *event_data = nullptr);<br>
>> +<br>
>> + void<br>
>> + Clear();<br>
>> +<br>
>> + uint32_t<br>
>> + AddListener (lldb::ListenerSP listener_sp, uint32_t event_mask);<br>
>> +<br>
>> + const char *<br>
>> + GetBroadcasterName () const<br>
>> + {<br>
>> + return m_broadcaster.GetBroadcasterName().AsCString();<br>
>> + }<br>
>> +<br>
>> + Broadcaster *<br>
>> + GetBroadcaster();<br>
>> +<br>
>> + bool<br>
>> + GetEventNames (Stream &s, const uint32_t event_mask, bool prefix_with_broadcaster_name) const;<br>
>> +<br>
>> + void<br>
>> + SetEventName (uint32_t event_mask, const char *name)<br>
>> + {<br>
>> + m_event_names[event_mask] = name;<br>
>> + }<br>
>> +<br>
>> + const char *<br>
>> + GetEventName (uint32_t event_mask) const<br>
>> + {<br>
>> + const auto pos = m_event_names.find (event_mask);<br>
>> + if (pos != m_event_names.end())<br>
>> + return pos->second.c_str();<br>
>> + return nullptr;<br>
>> + }<br>
>> +<br>
>> + bool<br>
>> + EventTypeHasListeners (uint32_t event_type);<br>
>><br>
>> + bool<br>
>> + RemoveListener (lldb::ListenerSP listener_sp, uint32_t event_mask = UINT32_MAX);<br>
>> +<br>
>> + bool<br>
>> + HijackBroadcaster (lldb::ListenerSP listener_sp, uint32_t event_mask = UINT32_MAX);<br>
>> +<br>
>> + bool<br>
>> + IsHijackedForEvent (uint32_t event_mask);<br>
>> +<br>
>> + void<br>
>> + RestoreBroadcaster ();<br>
>> +<br>
>> + protected:<br>
>> + void<br>
>> + PrivateBroadcastEvent (lldb::EventSP &event_sp, bool unique);<br>
>> +<br>
>> + const char *<br>
>> + GetHijackingListenerName();<br>
>> +<br>
>> + //------------------------------------------------------------------<br>
>> + //<br>
>> + //------------------------------------------------------------------<br>
>> + typedef std::vector< std::pair<lldb::ListenerSP,uint32_t> > collection;<br>
>> + typedef std::map<uint32_t, std::string> event_names_map;<br>
>> +<br>
>> + Broadcaster &m_broadcaster; ///< The broadcsater that this implements<br>
>> + event_names_map m_event_names; ///< Optionally define event names for readability and logging for each event bit<br>
>> + collection m_listeners; ///< A list of Listener / event_mask pairs that are listening to this broadcaster.<br>
>> + Mutex m_listeners_mutex; ///< A mutex that protects \a m_listeners.<br>
>> + std::vector<lldb::ListenerSP> m_hijacking_listeners; // A simple mechanism to intercept events from a broadcaster<br>
>> + std::vector<uint32_t> m_hijacking_masks; // At some point we may want to have a stack or Listener<br>
>> + // collections, but for now this is just for private hijacking.<br>
>> +<br>
>> + private:<br>
>> + //------------------------------------------------------------------<br>
>> + // For Broadcaster only<br>
>> + //------------------------------------------------------------------<br>
>> + DISALLOW_COPY_AND_ASSIGN (BroadcasterImpl);<br>
>> + };<br>
>> +<br>
>> + typedef std::shared_ptr<BroadcasterImpl> BroadcasterImplSP;<br>
>> + typedef std::weak_ptr<BroadcasterImpl> BroadcasterImplWP;<br>
>> +<br>
>> + BroadcasterImplSP<br>
>> + GetBroadcasterImpl()<br>
>> + {<br>
>> + return m_broadcaster_sp;<br>
>> + }<br>
>> +<br>
>> + const char *<br>
>> + GetHijackingListenerName()<br>
>> + {<br>
>> + return m_broadcaster_sp->GetHijackingListenerName();<br>
>> + }<br>
>> //------------------------------------------------------------------<br>
>> // Classes that inherit from Broadcaster can see and modify these<br>
>> //------------------------------------------------------------------<br>
>> - typedef std::vector< std::pair<Listener*,uint32_t> > collection;<br>
>> - typedef std::map<uint32_t, std::string> event_names_map;<br>
>> - // Prefix the name of our member variables with "m_broadcaster_"<br>
>> - // since this is a class that gets subclassed.<br>
>> - const ConstString m_broadcaster_name; ///< The name of this broadcaster object.<br>
>> - event_names_map m_event_names; ///< Optionally define event names for readability and logging for each event bit<br>
>> - collection m_listeners; ///< A list of Listener / event_mask pairs that are listening to this broadcaster.<br>
>> - Mutex m_listeners_mutex; ///< A mutex that protects \a m_listeners.<br>
>> - std::vector<Listener *> m_hijacking_listeners; // A simple mechanism to intercept events from a broadcaster<br>
>> - std::vector<uint32_t> m_hijacking_masks; // At some point we may want to have a stack or Listener<br>
>> - // collections, but for now this is just for private hijacking.<br>
>> - BroadcasterManager *m_manager;<br>
>> +<br>
>><br>
>> private:<br>
>> //------------------------------------------------------------------<br>
>> // For Broadcaster only<br>
>> //------------------------------------------------------------------<br>
>> + BroadcasterImplSP m_broadcaster_sp;<br>
>> + lldb::BroadcasterManagerSP m_manager_sp;<br>
>> + const ConstString m_broadcaster_name; ///< The name of this broadcaster object.<br>
>> +<br>
>> DISALLOW_COPY_AND_ASSIGN (Broadcaster);<br>
>> };<br>
>><br>
>><br>
>> Modified: lldb/trunk/include/lldb/Core/Debugger.h<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Core/Debugger.h?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Core/Debugger.h?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/include/lldb/Core/Debugger.h (original)<br>
>> +++ lldb/trunk/include/lldb/Core/Debugger.h Mon Mar 7 15:50:25 2016<br>
>> @@ -53,8 +53,7 @@ namespace lldb_private {<br>
>> class Debugger :<br>
>> public std::enable_shared_from_this<Debugger>,<br>
>> public UserID,<br>
>> - public Properties,<br>
>> - public BroadcasterManager<br>
>> + public Properties<br>
>> {<br>
>> friend class SourceManager; // For GetSourceFileCache.<br>
>><br>
>> @@ -159,10 +158,10 @@ public:<br>
>> return *m_command_interpreter_ap;<br>
>> }<br>
>><br>
>> - Listener &<br>
>> + lldb::ListenerSP<br>
>> GetListener ()<br>
>> {<br>
>> - return m_listener;<br>
>> + return m_listener_sp;<br>
>> }<br>
>><br>
>> // This returns the Debugger's scratch source manager. It won't be able to look up files in debug<br>
>> @@ -392,6 +391,12 @@ public:<br>
>> Target *GetSelectedOrDummyTarget(bool prefer_dummy = false);<br>
>> Target *GetDummyTarget();<br>
>><br>
>> + lldb::BroadcasterManagerSP<br>
>> + GetBroadcasterManager()<br>
>> + {<br>
>> + return m_broadcaster_manager_sp;<br>
>> + }<br>
>> +<br>
>> protected:<br>
>> friend class CommandInterpreter;<br>
>> friend class REPL;<br>
>> @@ -446,15 +451,20 @@ protected:<br>
>><br>
>> void<br>
>> InstanceInitialize ();<br>
>> -<br>
>> +<br>
>> lldb::StreamFileSP m_input_file_sp;<br>
>> lldb::StreamFileSP m_output_file_sp;<br>
>> lldb::StreamFileSP m_error_file_sp;<br>
>> +<br>
>> + lldb::BroadcasterManagerSP m_broadcaster_manager_sp; // The debugger acts as a broadcaster manager of last resort.<br>
>> + // It needs to get constructed before the target_list or any other<br>
>> + // member that might want to broadcast through the debugger.<br>
>> +<br>
>> TerminalState m_terminal_state;<br>
>> TargetList m_target_list;<br>
>><br>
>> PlatformList m_platform_list;<br>
>> - Listener m_listener;<br>
>> + lldb::ListenerSP m_listener_sp;<br>
>> std::unique_ptr<SourceManager> m_source_manager_ap; // This is a scratch source manager that we return if we have no targets.<br>
>> SourceManager::SourceFileCache m_source_file_cache; // All the source managers for targets created in this debugger used this shared<br>
>> // source file cache.<br>
>><br>
>> Modified: lldb/trunk/include/lldb/Core/Event.h<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Core/Event.h?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Core/Event.h?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/include/lldb/Core/Event.h (original)<br>
>> +++ lldb/trunk/include/lldb/Core/Event.h Mon Mar 7 15:50:25 2016<br>
>> @@ -20,6 +20,7 @@<br>
>> #include "lldb/lldb-private.h"<br>
>> #include "lldb/Core/ConstString.h"<br>
>> #include "lldb/Host/Predicate.h"<br>
>> +#include "lldb/Core/Broadcaster.h"<br>
>><br>
>> namespace lldb_private {<br>
>><br>
>> @@ -118,9 +119,9 @@ private:<br>
>> //----------------------------------------------------------------------<br>
>> class Event<br>
>> {<br>
>> - friend class Broadcaster;<br>
>> friend class Listener;<br>
>> friend class EventData;<br>
>> + friend class Broadcaster::BroadcasterImpl;<br>
>><br>
>> public:<br>
>> Event(Broadcaster *broadcaster, uint32_t event_type, EventData *data = nullptr);<br>
>> @@ -165,13 +166,21 @@ public:<br>
>> Broadcaster *<br>
>> GetBroadcaster () const<br>
>> {<br>
>> - return m_broadcaster;<br>
>> + Broadcaster::BroadcasterImplSP broadcaster_impl_sp = m_broadcaster_wp.lock();<br>
>> + if (broadcaster_impl_sp)<br>
>> + return broadcaster_impl_sp->GetBroadcaster();<br>
>> + else<br>
>> + return nullptr;<br>
>> }<br>
>><br>
>> bool<br>
>> BroadcasterIs (Broadcaster *broadcaster)<br>
>> {<br>
>> - return broadcaster == m_broadcaster;<br>
>> + Broadcaster::BroadcasterImplSP broadcaster_impl_sp = m_broadcaster_wp.lock();<br>
>> + if (broadcaster_impl_sp)<br>
>> + return broadcaster_impl_sp->GetBroadcaster() == broadcaster;<br>
>> + else<br>
>> + return false;<br>
>> }<br>
>><br>
>> void<br>
>> @@ -194,10 +203,10 @@ private:<br>
>> void<br>
>> SetBroadcaster (Broadcaster *broadcaster)<br>
>> {<br>
>> - m_broadcaster = broadcaster;<br>
>> + m_broadcaster_wp = broadcaster->GetBroadcasterImpl();<br>
>> }<br>
>><br>
>> - Broadcaster * m_broadcaster; // The broadcaster that sent this event<br>
>> + Broadcaster::BroadcasterImplWP m_broadcaster_wp; // The broadcaster that sent this event<br>
>> uint32_t m_type; // The bit describing this event<br>
>> std::unique_ptr<EventData> m_data_ap; // User specific data for this event<br>
>><br>
>><br>
>> Modified: lldb/trunk/include/lldb/Core/Listener.h<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Core/Listener.h?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Core/Listener.h?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/include/lldb/Core/Listener.h (original)<br>
>> +++ lldb/trunk/include/lldb/Core/Listener.h Mon Mar 7 15:50:25 2016<br>
>> @@ -21,11 +21,13 @@<br>
>> // Project includes<br>
>> #include "lldb/lldb-private.h"<br>
>> #include "lldb/Host/Predicate.h"<br>
>> +#include "lldb/Core/Broadcaster.h"<br>
>> #include "lldb/Core/Event.h"<br>
>><br>
>> namespace lldb_private {<br>
>><br>
>> -class Listener<br>
>> +class Listener :<br>
>> + public std::enable_shared_from_this<Listener><br>
>> {<br>
>> public:<br>
>> typedef bool (*HandleBroadcastCallback) (lldb::EventSP &event_sp, void *baton);<br>
>> @@ -36,8 +38,16 @@ public:<br>
>> //------------------------------------------------------------------<br>
>> // Constructors and Destructors<br>
>> //------------------------------------------------------------------<br>
>> + //<br>
>> + // Listeners have to be constructed into shared pointers - at least if you want them to listen to<br>
>> + // Broadcasters,<br>
>> +protected:<br>
>> Listener (const char *name);<br>
>><br>
>> +public:<br>
>> + static lldb::ListenerSP<br>
>> + MakeListener(const char *name);<br>
>> +<br>
>> ~Listener ();<br>
>><br>
>> void<br>
>> @@ -53,11 +63,11 @@ public:<br>
>> }<br>
>><br>
>> uint32_t<br>
>> - StartListeningForEventSpec (BroadcasterManager &manager,<br>
>> + StartListeningForEventSpec (lldb::BroadcasterManagerSP manager_sp,<br>
>> const BroadcastEventSpec &event_spec);<br>
>><br>
>> bool<br>
>> - StopListeningForEventSpec (BroadcasterManager &manager,<br>
>> + StopListeningForEventSpec (lldb::BroadcasterManagerSP manager_sp,<br>
>> const BroadcastEventSpec &event_spec);<br>
>><br>
>> uint32_t<br>
>> @@ -133,9 +143,11 @@ private:<br>
>> void *callback_user_data;<br>
>> };<br>
>><br>
>> - typedef std::multimap<Broadcaster*, BroadcasterInfo> broadcaster_collection;<br>
>> + typedef std::multimap<Broadcaster::BroadcasterImplWP,<br>
>> + BroadcasterInfo,<br>
>> + std::owner_less<Broadcaster::BroadcasterImplWP>> broadcaster_collection;<br>
>> typedef std::list<lldb::EventSP> event_collection;<br>
>> - typedef std::vector<BroadcasterManager *> broadcaster_manager_collection;<br>
>> + typedef std::vector<lldb::BroadcasterManagerWP> broadcaster_manager_collection;<br>
>><br>
>> bool<br>
>> FindNextEventInternal(Broadcaster *broadcaster, // nullptr for any broadcaster<br>
>> @@ -172,7 +184,7 @@ private:<br>
>> BroadcasterWillDestruct (Broadcaster *);<br>
>><br>
>> void<br>
>> - BroadcasterManagerWillDestruct (BroadcasterManager *manager);<br>
>> + BroadcasterManagerWillDestruct (lldb::BroadcasterManagerSP manager_sp);<br>
>><br>
>><br>
>> // broadcaster_collection::iterator<br>
>><br>
>> Modified: lldb/trunk/include/lldb/Target/Process.h<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/Process.h?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/Process.h?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/include/lldb/Target/Process.h (original)<br>
>> +++ lldb/trunk/include/lldb/Target/Process.h Mon Mar 7 15:50:25 2016<br>
>> @@ -30,6 +30,7 @@<br>
>> #include "lldb/Core/Communication.h"<br>
>> #include "lldb/Core/Error.h"<br>
>> #include "lldb/Core/Event.h"<br>
>> +#include "lldb/Core/Listener.h"<br>
>> #include "lldb/Core/LoadedModuleInfoList.h"<br>
>> #include "lldb/Core/ThreadSafeValue.h"<br>
>> #include "lldb/Core/PluginInterface.h"<br>
>> @@ -400,7 +401,7 @@ public:<br>
>> m_listener_sp = listener_sp;<br>
>> }<br>
>><br>
>> - Listener &<br>
>> + lldb::ListenerSP<br>
>> GetListenerForProcess (Debugger &debugger);<br>
>><br>
>> protected:<br>
>> @@ -939,13 +940,13 @@ public:<br>
>> /// Construct with a shared pointer to a target, and the Process listener.<br>
>> /// Uses the Host UnixSignalsSP by default.<br>
>> //------------------------------------------------------------------<br>
>> - Process(lldb::TargetSP target_sp, Listener &listener);<br>
>> + Process(lldb::TargetSP target_sp, lldb::ListenerSP listener_sp);<br>
>><br>
>> //------------------------------------------------------------------<br>
>> /// Construct with a shared pointer to a target, the Process listener,<br>
>> /// and the appropriate UnixSignalsSP for the process.<br>
>> //------------------------------------------------------------------<br>
>> - Process(lldb::TargetSP target_sp, Listener &listener, const lldb::UnixSignalsSP &unix_signals_sp);<br>
>> + Process(lldb::TargetSP target_sp, lldb::ListenerSP listener_sp, const lldb::UnixSignalsSP &unix_signals_sp);<br>
>><br>
>> //------------------------------------------------------------------<br>
>> /// Destructor.<br>
>> @@ -985,7 +986,7 @@ public:<br>
>> static lldb::ProcessSP<br>
>> FindPlugin (lldb::TargetSP target_sp,<br>
>> const char *plugin_name,<br>
>> - Listener &listener,<br>
>> + lldb::ListenerSP listener_sp,<br>
>> const FileSpec *crash_file_path);<br>
>><br>
>> //------------------------------------------------------------------<br>
>> @@ -2851,7 +2852,7 @@ public:<br>
>> WaitForProcessToStop(const TimeValue *timeout,<br>
>> lldb::EventSP *event_sp_ptr = nullptr,<br>
>> bool wait_always = true,<br>
>> - Listener *hijack_listener = nullptr,<br>
>> + lldb::ListenerSP hijack_listener = lldb::ListenerSP(),<br>
>> Stream *stream = nullptr,<br>
>> bool use_run_lock = true);<br>
>><br>
>> @@ -2877,7 +2878,7 @@ public:<br>
>> lldb::StateType<br>
>> WaitForStateChangedEvents(const TimeValue *timeout,<br>
>> lldb::EventSP &event_sp,<br>
>> - Listener *hijack_listener); // Pass nullptr to use builtin listener<br>
>> + lldb::ListenerSP hijack_listener); // Pass an empty ListenerSP to use builtin listener<br>
>><br>
>> //--------------------------------------------------------------------------------------<br>
>> /// Centralize the code that handles and prints descriptions for process state changes.<br>
>> @@ -2908,10 +2909,10 @@ public:<br>
>> ProcessEventHijacker<br>
>> {<br>
>> public:<br>
>> - ProcessEventHijacker (Process &process, Listener *listener) :<br>
>> + ProcessEventHijacker (Process &process, lldb::ListenerSP listener_sp) :<br>
>> m_process (process)<br>
>> {<br>
>> - m_process.HijackProcessEvents (listener);<br>
>> + m_process.HijackProcessEvents (listener_sp);<br>
>> }<br>
>><br>
>> ~ProcessEventHijacker ()<br>
>> @@ -2940,7 +2941,7 @@ public:<br>
>> /// \b false otherwise.<br>
>> //------------------------------------------------------------------<br>
>> bool<br>
>> - HijackProcessEvents (Listener *listener);<br>
>> + HijackProcessEvents (lldb::ListenerSP listener_sp);<br>
>><br>
>> //------------------------------------------------------------------<br>
>> /// Restores the process event broadcasting to its normal state.<br>
>> @@ -3343,7 +3344,7 @@ protected:<br>
>> ThreadSafeValue<lldb::StateType> m_private_state; // The actual state of our process<br>
>> Broadcaster m_private_state_broadcaster; // This broadcaster feeds state changed events into the private state thread's listener.<br>
>> Broadcaster m_private_state_control_broadcaster; // This is the control broadcaster, used to pause, resume & stop the private state thread.<br>
>> - Listener m_private_state_listener; // This is the listener for the private state thread.<br>
>> + lldb::ListenerSP m_private_state_listener_sp; // This is the listener for the private state thread.<br>
>> Predicate<bool> m_private_state_control_wait; /// This Predicate is used to signal that a control operation is complete.<br>
>> HostThread m_private_state_thread; ///< Thread ID for the thread that watches internal state events<br>
>> ProcessModID m_mod_id; ///< Tracks the state of the process over stops and other alterations.<br>
>> @@ -3363,7 +3364,7 @@ protected:<br>
>> uint32_t m_queue_list_stop_id; ///< The natural stop id when queue list was last fetched<br>
>> std::vector<Notifications> m_notifications; ///< The list of notifications that this process can deliver.<br>
>> std::vector<lldb::addr_t> m_image_tokens;<br>
>> - Listener &m_listener;<br>
>> + lldb::ListenerSP m_listener_sp; ///< Shared pointer to the listener used for public events. Can not be empty.<br>
>> BreakpointSiteList m_breakpoint_site_list; ///< This is the list of breakpoint locations we intend to insert in the target.<br>
>> lldb::DynamicLoaderUP m_dyld_ap;<br>
>> lldb::JITLoaderListUP m_jit_loaders_ap;<br>
>><br>
>> Modified: lldb/trunk/include/lldb/Target/ProcessLaunchInfo.h<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/ProcessLaunchInfo.h?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/ProcessLaunchInfo.h?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/include/lldb/Target/ProcessLaunchInfo.h (original)<br>
>> +++ lldb/trunk/include/lldb/Target/ProcessLaunchInfo.h Mon Mar 7 15:50:25 2016<br>
>> @@ -196,7 +196,7 @@ namespace lldb_private<br>
>> m_listener_sp = listener_sp;<br>
>> }<br>
>><br>
>> - Listener &<br>
>> + lldb::ListenerSP<br>
>> GetListenerForProcess (Debugger &debugger);<br>
>><br>
>> lldb::ListenerSP<br>
>><br>
>> Modified: lldb/trunk/include/lldb/Target/Target.h<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/Target.h?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/Target.h?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/include/lldb/Target/Target.h (original)<br>
>> +++ lldb/trunk/include/lldb/Target/Target.h Mon Mar 7 15:50:25 2016<br>
>> @@ -719,7 +719,7 @@ public:<br>
>> Dump (Stream *s, lldb::DescriptionLevel description_level);<br>
>><br>
>> const lldb::ProcessSP &<br>
>> - CreateProcess (Listener &listener,<br>
>> + CreateProcess (lldb::ListenerSP listener,<br>
>> const char *plugin_name,<br>
>> const FileSpec *crash_file);<br>
>><br>
>><br>
>> Modified: lldb/trunk/include/lldb/lldb-forward.h<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/lldb-forward.h?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/lldb-forward.h?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/include/lldb/lldb-forward.h (original)<br>
>> +++ lldb/trunk/include/lldb/lldb-forward.h Mon Mar 7 15:50:25 2016<br>
>> @@ -310,6 +310,8 @@ namespace lldb {<br>
>> typedef std::weak_ptr<lldb_private::BreakpointLocation> BreakpointLocationWP;<br>
>> typedef std::shared_ptr<lldb_private::BreakpointResolver> BreakpointResolverSP;<br>
>> typedef std::shared_ptr<lldb_private::Broadcaster> BroadcasterSP;<br>
>> + typedef std::shared_ptr<lldb_private::BroadcasterManager> BroadcasterManagerSP;<br>
>> + typedef std::weak_ptr<lldb_private::BroadcasterManager> BroadcasterManagerWP;<br>
>> typedef std::unique_ptr<lldb_private::ClangASTContext> ClangASTContextUP;<br>
>> typedef std::shared_ptr<lldb_private::ClangASTImporter> ClangASTImporterSP;<br>
>> typedef std::unique_ptr<lldb_private::ClangModulesDeclVendor> ClangModulesDeclVendorUP;<br>
>><br>
>> Modified: lldb/trunk/include/lldb/lldb-private-interfaces.h<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/lldb-private-interfaces.h?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/lldb-private-interfaces.h?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/include/lldb/lldb-private-interfaces.h (original)<br>
>> +++ lldb/trunk/include/lldb/lldb-private-interfaces.h Mon Mar 7 15:50:25 2016<br>
>> @@ -35,7 +35,7 @@ namespace lldb_private<br>
>> typedef lldb::CommandObjectSP (*LanguageRuntimeGetCommandObject) (CommandInterpreter& interpreter);<br>
>> typedef SystemRuntime *(*SystemRuntimeCreateInstance) (Process *process);<br>
>> typedef lldb::PlatformSP (*PlatformCreateInstance) (bool force, const ArchSpec *arch);<br>
>> - typedef lldb::ProcessSP (*ProcessCreateInstance) (lldb::TargetSP target_sp, Listener &listener, const FileSpec *crash_file_path);<br>
>> + typedef lldb::ProcessSP (*ProcessCreateInstance) (lldb::TargetSP target_sp, lldb::ListenerSP listener_sp, const FileSpec *crash_file_path);<br>
>> typedef lldb::ScriptInterpreterSP (*ScriptInterpreterCreateInstance)(CommandInterpreter &interpreter);<br>
>> typedef SymbolFile* (*SymbolFileCreateInstance) (ObjectFile* obj_file);<br>
>> typedef SymbolVendor* (*SymbolVendorCreateInstance) (const lldb::ModuleSP &module_sp, lldb_private::Stream *feedback_strm); // Module can be NULL for default system symbol vendor<br>
>><br>
>> Added: lldb/trunk/packages/Python/lldbsuite/test/api/listeners/Makefile<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/api/listeners/Makefile?rev=262863&view=auto" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/api/listeners/Makefile?rev=262863&view=auto</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/packages/Python/lldbsuite/test/api/listeners/Makefile (added)<br>
>> +++ lldb/trunk/packages/Python/lldbsuite/test/api/listeners/Makefile Mon Mar 7 15:50:25 2016<br>
>> @@ -0,0 +1,6 @@<br>
>> +LEVEL = ../../make<br>
>> +<br>
>> +C_SOURCES := main.c<br>
>> +<br>
>> +include $(LEVEL)/Makefile.rules<br>
>> +<br>
>><br>
>> Added: lldb/trunk/packages/Python/lldbsuite/test/api/listeners/TestListener.py<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/api/listeners/TestListener.py?rev=262863&view=auto" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/api/listeners/TestListener.py?rev=262863&view=auto</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/packages/Python/lldbsuite/test/api/listeners/TestListener.py (added)<br>
>> +++ lldb/trunk/packages/Python/lldbsuite/test/api/listeners/TestListener.py Mon Mar 7 15:50:25 2016<br>
>> @@ -0,0 +1,55 @@<br>
>> +"""<br>
>> +Test that we can listen to modules loaded events.<br>
>> +"""<br>
>> +<br>
>> +from __future__ import print_function<br>
>> +<br>
>> +import copy<br>
>> +import os<br>
>> +import time<br>
>> +<br>
>> +import lldb<br>
>> +from lldbsuite.test.decorators import *<br>
>> +from lldbsuite.test.lldbtest import *<br>
>> +from lldbsuite.test import lldbutil<br>
>> +<br>
>> +import six<br>
>> +<br>
>> +class ListenToModuleLoadedEvents (TestBase):<br>
>> +<br>
>> + mydir = TestBase.compute_mydir(__file__)<br>
>> +<br>
>> + def setUp(self):<br>
>> + # Call super's setUp().<br>
>> + TestBase.setUp(self)<br>
>> + self.build()<br>
>> +<br>
>> + def test_receiving_breakpoint_added (self):<br>
>> + """Test that we get breakpoint added events, waiting on event classes on the debugger"""<br>
>> +<br>
>> + my_listener = lldb.SBListener("test_listener")<br>
>> +<br>
>> + my_listener.StartListeningForEventClass(self.dbg, lldb.SBTarget.GetBroadcasterClassName(), lldb.SBTarget.eBroadcastBitBreakpointChanged)<br>
>> +<br>
>> + exe = os.path.join (os.getcwd(), "a.out")<br>
>> +<br>
>> + target = self.dbg.CreateTarget(exe)<br>
>> +<br>
>> + bkpt = target.BreakpointCreateByName("main")<br>
>> +<br>
>> + event = lldb.SBEvent()<br>
>> + my_listener.WaitForEvent(1, event)<br>
>> +<br>
>> + self.assertTrue(event.IsValid(), "Got a valid event.")<br>
>> + self.assertTrue(lldb.SBBreakpoint.EventIsBreakpointEvent(event), "It is a breakpoint event.")<br>
>> + self.assertTrue(lldb.SBBreakpoint.GetBreakpointEventTypeFromEvent(event) == lldb.eBreakpointEventTypeAdded, "It is a breakpoint added event.")<br>
>> + self.assertTrue(bkpt == lldb.SBBreakpoint.GetBreakpointFromEvent(event), "It is our breakpoint.")<br>
>> +<br>
>> + # Now make sure if we stop listening for events we don't get them:<br>
>> +<br>
>> + my_listener.StopListeningForEventClass(self.dbg, lldb.SBTarget.GetBroadcasterClassName(), lldb.SBTarget.eBroadcastBitBreakpointChanged)<br>
>> + my_listener.StopListeningForEvents(target.GetBroadcaster(), lldb.SBTarget.eBroadcastBitBreakpointChanged)<br>
>> +<br>
>> + bkpt2 = target.BreakpointCreateByName("main")<br>
>> + my_listener.WaitForEvent(1, event)<br>
>> + self.assertTrue(not event.IsValid(), "We don't get events we aren't listening to.")<br>
>><br>
>> Added: lldb/trunk/packages/Python/lldbsuite/test/api/listeners/main.c<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/api/listeners/main.c?rev=262863&view=auto" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/packages/Python/lldbsuite/test/api/listeners/main.c?rev=262863&view=auto</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/packages/Python/lldbsuite/test/api/listeners/main.c (added)<br>
>> +++ lldb/trunk/packages/Python/lldbsuite/test/api/listeners/main.c Mon Mar 7 15:50:25 2016<br>
>> @@ -0,0 +1,7 @@<br>
>> +#include <stdio.h><br>
>> +<br>
>> +int<br>
>> +main()<br>
>> +{<br>
>> + printf ("Hello there.\n");<br>
>> +}<br>
>><br>
>> Modified: lldb/trunk/source/API/SBBroadcaster.cpp<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/API/SBBroadcaster.cpp?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/API/SBBroadcaster.cpp?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/source/API/SBBroadcaster.cpp (original)<br>
>> +++ lldb/trunk/source/API/SBBroadcaster.cpp Mon Mar 7 15:50:25 2016<br>
>> @@ -117,14 +117,14 @@ SBBroadcaster::AddInitialEventsToListene<br>
>> static_cast<void*>(m_opaque_ptr),<br>
>> static_cast<void*>(listener.get()), requested_events);<br>
>> if (m_opaque_ptr)<br>
>> - m_opaque_ptr->AddInitialEventsToListener (listener.get(), requested_events);<br>
>> + m_opaque_ptr->AddInitialEventsToListener (listener.m_opaque_sp, requested_events);<br>
>> }<br>
>><br>
>> uint32_t<br>
>> SBBroadcaster::AddListener (const SBListener &listener, uint32_t event_mask)<br>
>> {<br>
>> if (m_opaque_ptr)<br>
>> - return m_opaque_ptr->AddListener (listener.get(), event_mask);<br>
>> + return m_opaque_ptr->AddListener (listener.m_opaque_sp, event_mask);<br>
>> return 0;<br>
>> }<br>
>><br>
>> @@ -148,7 +148,7 @@ bool<br>
>> SBBroadcaster::RemoveListener (const SBListener &listener, uint32_t event_mask)<br>
>> {<br>
>> if (m_opaque_ptr)<br>
>> - return m_opaque_ptr->RemoveListener (listener.get(), event_mask);<br>
>> + return m_opaque_ptr->RemoveListener (listener.m_opaque_sp, event_mask);<br>
>> return false;<br>
>> }<br>
>><br>
>><br>
>> Modified: lldb/trunk/source/API/SBDebugger.cpp<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/API/SBDebugger.cpp?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/API/SBDebugger.cpp?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/source/API/SBDebugger.cpp (original)<br>
>> +++ lldb/trunk/source/API/SBDebugger.cpp Mon Mar 7 15:50:25 2016<br>
>> @@ -432,8 +432,8 @@ SBDebugger::HandleCommand (const char *c<br>
>> if (process_sp)<br>
>> {<br>
>> EventSP event_sp;<br>
>> - Listener &lldb_listener = m_opaque_sp->GetListener();<br>
>> - while (lldb_listener.GetNextEventForBroadcaster (process_sp.get(), event_sp))<br>
>> + ListenerSP lldb_listener_sp = m_opaque_sp->GetListener();<br>
>> + while (lldb_listener_sp->GetNextEventForBroadcaster (process_sp.get(), event_sp))<br>
>> {<br>
>> SBEvent event(event_sp);<br>
>> HandleProcessEvent (process, event, GetOutputFileHandle(), GetErrorFileHandle());<br>
>> @@ -450,7 +450,7 @@ SBDebugger::GetListener ()<br>
>><br>
>> SBListener sb_listener;<br>
>> if (m_opaque_sp)<br>
>> - sb_listener.reset(&m_opaque_sp->GetListener(), false);<br>
>> + sb_listener.reset(m_opaque_sp->GetListener());<br>
>><br>
>> if (log)<br>
>> log->Printf ("SBDebugger(%p)::GetListener () => SBListener(%p)",<br>
>><br>
>> Modified: lldb/trunk/source/API/SBListener.cpp<br>
>> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/API/SBListener.cpp?rev=262863&r1=262862&r2=262863&view=diff" rel="noreferrer" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/API/SBListener.cpp?rev=262863&r1=262862&r2=262863&view=diff</a><br>
>> ==============================================================================<br>
>> --- lldb/trunk/source/API/SBListener.cpp (original)<br>
>> +++ lldb/trunk/source/API/SBListener.cpp Mon Mar 7 15:50:25 2016<br>
>> @@ -26,27 +26,25 @@ using namespace lldb_private;<br>
>><br>
>> SBListener::SBListener () :<br>
>> m_opaque_sp (),<br>
>> - m_opaque_ptr (NULL)<br>
>> + m_unused_ptr (NULL)<br>
>> {<br>
>> }<br>
>><br>
>> SBListener::SBListener (const char *name) :<br>
>> - m_opaque_sp (new Listener (name)),<br>
>> - m_opaque_ptr (NULL)<br>
>> + m_opaque_sp (Listener::MakeListener(name)),<br>
>> + m_unused_ptr (nullptr)<br>
>> {<br>
>> - m_opaque_ptr = m_opaque_sp.get();<br>
>> -<br>
>> Log *log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API));<br>
>><br>
>> if (log)<br>
>> log->Printf ("SBListener::SBListener (name=\"%s\") => SBListener(%p)",<br>
>> - name, static_cast<void*>(m_opaque_ptr));<br>
>> + name, static_cast<void*>(m_opaque_sp.get()));<br>
>> }<br>
>><br>
>><br>
>> SBListener::SBListener (const SBListener &rhs) :<br>
>> m_opaque_sp (rhs.m_opaque_sp),<br>
>> - m_opaque_ptr (rhs.m_opaque_ptr)<br>
>> + m_unused_ptr (nullptr)<br>
>> {<br>
>> }<br>
>><br>
>> @@ -56,20 +54,14 @@ SBListener::operator = (const lldb::SBLi<br>
>> if (this != &rhs)<br>
>> {<br>
>> m_opaque_sp = rhs.m_opaque_sp;<br>
>> - m_opaque_ptr = rhs.m_opaque_ptr;<br>
>> + m_unused_ptr = nullptr;<br>
>> }<br>
>> return *this;<br>
>> }<br>
>><br>
>> -SBListener::SBListener (Listener &listener) :<br>
>> - m_opaque_sp (),<br>
>> - m_opaque_ptr (&listener)<br>
>> -{<br>
>> -}<br>
>> -<br>
>> SBListener::SBListener (const lldb::ListenerSP &listener_sp) :<br>
>> m_opaque_sp (listener_sp),<br>
>> - m_opaque_ptr (listener_sp.get())<br>
>> + m_unused_ptr (nullptr)<br>
>> {<br>
>> }<br>
>><br>
>> @@ -80,7 +72,7 @@ SBListener::~SBListener ()<br>
>> bool<br>
>> SBListener::IsValid() const<br>
>> {<br>
>> - return m_opaque_ptr != NULL;<br>
>> + return m_opaque_sp != nullptr;<br>
>> }<br>
>><br>
>> void<br>
>> @@ -88,14 +80,14 @@ SBListener::AddEvent (const SBEvent &eve<br>
>> {<br>
>> EventSP &event_sp = event.GetSP ();<br>
>> if (event_sp)<br>
>> - m_opaque_ptr->AddEvent (event_sp);<br>
>> + m_opaque_sp->AddEvent (event_sp);<br>
>> }<br>
>><br>
>> void<br>
>> SBListener::Clear ()<br>
>> {<br>
>> - if (m_opaque_ptr)<br>
>> - m_opaque_ptr->Clear ();<br>
>> + if (m_opaque_sp)<br>
>> + m_opaque_sp->Clear ();<br>
>> }<br>
>><br>
>> uint32_t<br>
>> @@ -103,13 +95,13 @@ SBListener::StartListeningForEventClass<br>
>> const char *broadcaster_class,<br>
>> uint32_t event_mask)<br>
>> {<br>
>> - if (m_opaque_ptr)<br>
>> + if (m_opaque_sp)<br>
>> {<br>
>> Debugger *lldb_debugger = debugger.get();<br>
>> if (!lldb_debugger)<br>
>> return 0;<br>
>> BroadcastEventSpec event_spec (ConstString (broadcaster_class), event_mask);<br>
>> - return m_opaque_ptr->StartListeningForEventSpec (*lldb_debugger, event_spec);<br>
>> + return m_opaque_sp->StartListeningForEventSpec (lldb_debugger->GetBroadcasterManager(), event_spec);<br>
>> }<br>
>> else<br>
>> return 0;<br>
>> @@ -120,13 +112,13 @@ SBListener::StopListeningForEventClass (<br>
>> const char *broadcaster_class,<br>
>> uint32_t event_mask)<br>
>> {<br>
>> - if (m_opaque_ptr)<br>
>> + if (m_opaque_sp)<br>
>> {<br>
>> Debugger *lldb_debugger = debugger.get();<br>
>> if (!lldb_debugger)<br>
>> return false;<br>
>> BroadcastEventSpec event_spec (ConstString (broadcaster_class), event_mask);<br>
>> - return m_opaque_ptr->StopListeningForEventSpec (*lldb_debugger, event_spec);<br>
>> + return m_opaque_sp->StopListeningForEventSpec (lldb_debugger->GetBroadcasterManager(), event_spec);<br>
>> }<br>
>> else<br>
>> return false;<br>
>> @@ -136,9 +128,9 @@ uint32_t<br>
>> SBListener::StartListeningForEvents (const SBBroadcaster& broadcaster, uint32_t event_mask)<br>
>> {<br>
>> uint32_t acquired_event_mask = 0;<br>
>> - if (m_opaque_ptr && broadcaster.IsValid())<br>
>> + if (m_opaque_sp && broadcaster.IsValid())<br>
>> {<br>
>> - acquired_event_mask = m_opaque_ptr->StartListeningForEvents (broadcaster.get(), event_mask);<br>
>> + acquired_event_mask = m_opaque_sp->StartListeningForEvents (broadcaster.get(), event_mask);<br>
>> }<br>
>><br>
>> Log *log = lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_API);<br>
>> @@ -153,7 +145,7 @@ SBListener::StartListeningForEvents (con<br>
>> const bool got_requested_names = lldb_broadcaster->GetEventNames (sstr_requested, event_mask, false);<br>
>> const bool got_acquired_names = lldb_broadcaster->GetEventNames (sstr_acquired, acquired_event_mask, false);<br>
>> log->Printf ("SBListener(%p)::StartListeneingForEvents (SBBroadcaster(%p): %s, event_mask=0x%8.8x%s%s%s) => 0x%8.8x%s%s%s",<br>
>> - static_cast<void*>(m_opaque_ptr),<br>
>> + static_cast<void*>(m_opaque_sp.get()),<br>
>> static_cast<void*>(lldb_broadcaster),<br>
>> lldb_broadcaster->GetBroadcasterName().GetCString(),<br>
>> event_mask,<br>
>> @@ -168,7 +160,7 @@ SBListener::StartListeningForEvents (con<br>
>> else<br>
>> {<br>
>> log->Printf ("SBListener(%p)::StartListeneingForEvents (SBBroadcaster(%p), event_mask=0x%8.8x) => 0x%8.8x",<br>
>> - static_cast<void*>(m_opaque_ptr),<br>
>> + static_cast<void*>(m_opaque_sp.get()),<br>
>> static_cast<void*>(lldb_broadcaster), event_mask,<br>
>> acquired_event_mask);<br>
>> }<br>
>> @@ -180,9 +172,9 @@ SBListener::StartListeningForEvents (con<br>
>> bool<br>
>> SBListener::StopListeningForEvents (const SBBroadcaster& broadcaster, uint32_t event_mask)<br>
>> {<br>
>> - if (m_opaque_ptr && broadcaster.IsValid())<br>
>> + if (m_opaque_sp && broadcaster.IsValid())<br>
>> {<br>
>> - return m_opaque_ptr->StopListeningForEvents (broadcaster.get(), event_mask);<br>
>> + return m_opaque_sp->StopListeningForEvents (broadcaster.get(), event_mask);<br>
>> }<br>
>> return false;<br>
>> }<br>
>> @@ -196,19 +188,19 @@ SBListener::WaitForEvent (uint32_t timeo<br>
>> if (timeout_secs == UINT32_MAX)<br>
>> {<br>
>> log->Printf ("SBListener(%p)::WaitForEvent (timeout_secs=INFINITE, SBEvent(%p))...",<br>
>> - static_cast<void*>(m_opaque_ptr),<br>
>> + static_cast<void*>(m_opaque_sp.get()),<br>
>> static_cast<void*>(event.get()));<br>
>> }<br>
>> else<br>
>> {<br>
>> log->Printf ("SBListener(%p)::WaitForEvent (timeout_secs=%d, SBEvent(%p))...",<br>
>> - static_cast<void*>(m_opaque_ptr), timeout_secs,<br>
>> + static_cast<void*>(m_opaque_sp.get()), timeout_secs,<br>
>> static_cast<void*>(event.get()));<br>
>> }<br>
>> }<br>
>> bool success = false;<br>
>><br>
>> - if (m_opaque_ptr)<br>
>> + if (m_opaque_sp)<br>
>> {<br>
>> TimeValue time_value;<br>
>> if (timeout_secs != UINT32_MAX)<br>
>> @@ -218,7 +210,7 @@ SBListener::WaitForEvent (uint32_t timeo<br>
>> time_value.OffsetWithSeconds (timeout_secs);<br>
>> }<br>
>> EventSP event_sp;<br>
>> - if (m_opaque_ptr->WaitForEvent (time_value.IsValid() ? &time_value : NULL, event_sp))<br>
>> + if (m_opaque_sp->WaitForEvent (time_value.IsValid() ? &time_value : NULL, event_sp))<br>
>> {<br>
>> event.reset (event_sp);<br>
>> success = true;<br>
>> @@ -230,13 +222,13 @@ SBListener::WaitForEvent (uint32_t timeo<br>
>> if (timeout_secs == UINT32_MAX)<br>
>> {<br>
>> log->Printf ("SBListener(%p)::WaitForEvent (timeout_secs=INFINITE, SBEvent(%p)) => %i",<br>
>> - static_cast<void*>(m_opaque_ptr),<br>
>> + static_cast<void*>(m_opaque_sp.get()),<br>
>> static_cast<void*>(event.get()), success);<br>
>> }<br>
>> else<br>
>> {<br>
>> log->Printf ("SBListener(%p)::WaitForEvent (timeout_secs=%d, SBEvent(%p)) => %i",<br>
>> - static_cast<void*>(m_opaque_ptr), timeout_secs,<br>
>> + static_cast<void*>(m_opaque_sp.get()), timeout_secs,<br>
>> static_cast<void*>(event.get()), success);<br>
>> }<br>
>> }<br>
>> @@ -253,7 +245,7 @@ SBListener::WaitForEventForBroadcaster<br>
>> SBEvent &event<br>
>> )<br>
>> {<br>
>> - if (m_opaque_ptr && broadcaster.IsValid())<br>
>> + if (m_opaque_sp && broadcaster.IsValid())<br>
>> {<br>
>> TimeValue time_value;<br>
>> if (num_seconds != UINT32_MAX)<br>
>> @@ -262,7 +254,7 @@ SBListener::WaitForEventForBroadcaster<br>
>> time_value.OffsetWithSeconds (num_seconds);<br>
>> }<br>
>> EventSP event_sp;<br>
>> - if (m_opaque_ptr->WaitForEventForBroadcaster (time_value.IsValid() ? &time_value : NULL,<br>
>> + if (m_opaque_sp->WaitForEventForBroadcaster (time_value.IsValid() ? &time_value : NULL,<br>
>> broadcaster.get(),<br>
>> event_sp))<br>
>> {<br>
>> @@ -284,7 +276,7 @@ SBListener::WaitForEventForBroadcasterWi<br>
>> SBEvent &event<br>
>> )<br>
>> {<br>
>> - if (m_opaque_ptr && broadcaster.IsValid())<br>
>> + if (m_opaque_sp && broadcaster.IsValid())<br>
>> {<br>
>> TimeValue time_value;<br>
>> if (num_seconds != UINT32_MAX)<br>
>> @@ -293,7 +285,7 @@ SBListener::WaitForEventForBroadcasterWi<br>
>> time_value.OffsetWithSeconds (num_seconds);<br>
>> }<br>
>> EventSP event_sp;<br>
>> - if (m_opaque_ptr->WaitForEventForBroadcasterWithType (time_value.IsValid() ? &time_value : NULL,<br>
>> + if (m_opaque_sp->WaitForEventForBroadcasterWithType (time_value.IsValid() ? &time_value : NULL,<br>
>></blockquote></div>