<div dir="ltr">Thanks! Will be more careful next time.<div><br></div><div>Kuba<br><div class="gmail_extra"><br><div class="gmail_quote">On Fri, Oct 10, 2014 at 5:50 PM, Eric Christopher <span dir="ltr"><<a href="mailto:echristo@gmail.com" target="_blank">echristo@gmail.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">FWIW this didn't even vaguely build on anything other than xcode. Fixed thusly:<br>
<br>
dzur:~/sources/llvm/tools/lldb> svn ci lib/Makefile source/<br>
Sending lib/Makefile<br>
Sending source/CMakeLists.txt<br>
Sending source/Plugins/Makefile<br>
Transmitting file data ...<br>
Committed revision 219554.<br>
<br>
-eric<br>
<br>
On Fri, Oct 10, 2014 at 4:43 PM, Kuba Brecka <<a href="mailto:kuba.brecka@gmail.com">kuba.brecka@gmail.com</a>> wrote:<br>
> Author: kuba.brecka<br>
> Date: Fri Oct 10 18:43:03 2014<br>
> New Revision: 219546<br>
><br>
> URL: <a href="http://llvm.org/viewvc/llvm-project?rev=219546&view=rev" target="_blank">http://llvm.org/viewvc/llvm-project?rev=219546&view=rev</a><br>
> Log:<br>
> LLDB AddressSanitizer instrumentation runtime plugin, breakpint on error and report data extraction<br>
><br>
> Reviewed at <a href="http://reviews.llvm.org/D5592" target="_blank">http://reviews.llvm.org/D5592</a><br>
><br>
> This patch gives LLDB some ability to interact with AddressSanitizer runtime library, on top of what we already have (historical memory stack traces provided by ASan). Namely, that's the ability to stop on an error caught by ASan, and access the report information that are associated with it. The report information is also exposed into SB API.<br>
><br>
> More precisely this patch...<br>
><br>
> adds a new plugin type, InstrumentationRuntime, which should serve as a generic superclass for other instrumentation runtime libraries, these plugins get notified when modules are loaded, so they get a chance to "activate" when a specific dynamic library is loaded<br>
> an instance of this plugin type, AddressSanitizerRuntime, which activates itself when it sees the ASan dynamic library or founds ASan statically linked in the executable<br>
> adds a collection of these plugins into the Process class<br>
> AddressSanitizerRuntime sets an internal breakpoint on __asan::AsanDie(), and when this breakpoint gets hit, it retrieves the report information from ASan<br>
> this breakpoint is then exposed as a new StopReason, eStopReasonInstrumentation, with a new StopInfo subclass, InstrumentationRuntimeStopInfo<br>
> the StopInfo superclass is extended with a m_extended_info field (it's a StructuredData::ObjectSP), that can hold arbitrary JSON-like data, which is the way the new plugin provides the report data<br>
> the "thread info" command now accepts a "-s" flag that prints out the JSON data of a stop reason (same way the "-j" flag works now)<br>
> SBThread has a new API, GetStopReasonExtendedInfoAsJSON, which dumps the JSON string into a SBStream<br>
> adds a test case for all of this<br>
> I plan to also get rid of the original ASan plugin (memory history stack traces) and use an instance of AddressSanitizerRuntime for that purpose.<br>
><br>
> Kuba<br>
><br>
><br>
> Added:<br>
> lldb/trunk/include/lldb/Target/InstrumentationRuntime.h<br>
> lldb/trunk/include/lldb/Target/InstrumentationRuntimeStopInfo.h<br>
> lldb/trunk/source/Plugins/InstrumentationRuntime/<br>
> lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/<br>
> lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.cpp<br>
> lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.h<br>
> lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/CMakeLists.txt<br>
> lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/Makefile<br>
> lldb/trunk/source/Plugins/InstrumentationRuntime/CMakeLists.txt<br>
> lldb/trunk/source/Target/InstrumentationRuntime.cpp<br>
> lldb/trunk/source/Target/InstrumentationRuntimeStopInfo.cpp<br>
> lldb/trunk/test/functionalities/asan/TestMemoryHistory.py<br>
> - copied unchanged from r219545, lldb/trunk/test/functionalities/asan/TestAsan.py<br>
> lldb/trunk/test/functionalities/asan/TestReportData.py<br>
> Removed:<br>
> lldb/trunk/test/functionalities/asan/TestAsan.py<br>
> Modified:<br>
> lldb/trunk/include/lldb/API/SBThread.h<br>
> lldb/trunk/include/lldb/Core/PluginManager.h<br>
> lldb/trunk/include/lldb/Target/Process.h<br>
> lldb/trunk/include/lldb/Target/StopInfo.h<br>
> lldb/trunk/include/lldb/Target/Thread.h<br>
> lldb/trunk/include/lldb/lldb-enumerations.h<br>
> lldb/trunk/include/lldb/lldb-forward.h<br>
> lldb/trunk/include/lldb/lldb-private-interfaces.h<br>
> lldb/trunk/lldb.xcodeproj/project.pbxproj<br>
> lldb/trunk/scripts/Python/interface/SBThread.i<br>
> lldb/trunk/source/API/SBThread.cpp<br>
> lldb/trunk/source/Commands/CommandObjectThread.cpp<br>
> lldb/trunk/source/Core/Debugger.cpp<br>
> lldb/trunk/source/Core/PluginManager.cpp<br>
> lldb/trunk/source/Target/CMakeLists.txt<br>
> lldb/trunk/source/Target/Process.cpp<br>
> lldb/trunk/source/Target/StopInfo.cpp<br>
> lldb/trunk/source/Target/Thread.cpp<br>
> lldb/trunk/source/lldb.cpp<br>
> lldb/trunk/test/functionalities/asan/main.c<br>
><br>
> Modified: lldb/trunk/include/lldb/API/SBThread.h<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/API/SBThread.h?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/API/SBThread.h?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/include/lldb/API/SBThread.h (original)<br>
> +++ lldb/trunk/include/lldb/API/SBThread.h Fri Oct 10 18:43:03 2014<br>
> @@ -78,7 +78,10 @@ public:<br>
> //--------------------------------------------------------------------------<br>
> uint64_t<br>
> GetStopReasonDataAtIndex(uint32_t idx);<br>
> -<br>
> +<br>
> + bool<br>
> + GetStopReasonExtendedInfoAsJSON (lldb::SBStream &stream);<br>
> +<br>
> size_t<br>
> GetStopDescription (char *dst, size_t dst_len);<br>
><br>
><br>
> Modified: lldb/trunk/include/lldb/Core/PluginManager.h<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Core/PluginManager.h?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Core/PluginManager.h?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/include/lldb/Core/PluginManager.h (original)<br>
> +++ lldb/trunk/include/lldb/Core/PluginManager.h Fri Oct 10 18:43:03 2014<br>
> @@ -361,6 +361,28 @@ public:<br>
> GetMemoryHistoryCreateCallbackForPluginName (const ConstString &name);<br>
><br>
> //------------------------------------------------------------------<br>
> + // InstrumentationRuntime<br>
> + //------------------------------------------------------------------<br>
> + static bool<br>
> + RegisterPlugin (const ConstString &name,<br>
> + const char *description,<br>
> + InstrumentationRuntimeCreateInstance create_callback,<br>
> + InstrumentationRuntimeGetType get_type_callback);<br>
> +<br>
> + static bool<br>
> + UnregisterPlugin (InstrumentationRuntimeCreateInstance create_callback);<br>
> +<br>
> + static InstrumentationRuntimeGetType<br>
> + GetInstrumentationRuntimeGetTypeCallbackAtIndex (uint32_t idx);<br>
> +<br>
> + static InstrumentationRuntimeCreateInstance<br>
> + GetInstrumentationRuntimeCreateCallbackAtIndex (uint32_t idx);<br>
> +<br>
> + static InstrumentationRuntimeCreateInstance<br>
> + GetInstrumentationRuntimeCreateCallbackForPluginName (const ConstString &name);<br>
> +<br>
> +<br>
> + //------------------------------------------------------------------<br>
> // Some plug-ins might register a DebuggerInitializeCallback<br>
> // callback when registering the plug-in. After a new Debugger<br>
> // instance is created, this DebuggerInitialize function will get<br>
><br>
> Added: lldb/trunk/include/lldb/Target/InstrumentationRuntime.h<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/InstrumentationRuntime.h?rev=219546&view=auto" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/InstrumentationRuntime.h?rev=219546&view=auto</a><br>
> ==============================================================================<br>
> --- lldb/trunk/include/lldb/Target/InstrumentationRuntime.h (added)<br>
> +++ lldb/trunk/include/lldb/Target/InstrumentationRuntime.h Fri Oct 10 18:43:03 2014<br>
> @@ -0,0 +1,47 @@<br>
> +//===-- InstrumentationRuntime.h --------------------------------*- C++ -*-===//<br>
> +//<br>
> +// The LLVM Compiler Infrastructure<br>
> +//<br>
> +// This file is distributed under the University of Illinois Open Source<br>
> +// License. See LICENSE.TXT for details.<br>
> +//<br>
> +//===----------------------------------------------------------------------===//<br>
> +<br>
> +#ifndef liblldb_InstrumentationRuntime_h_<br>
> +#define liblldb_InstrumentationRuntime_h_<br>
> +<br>
> +// C Includes<br>
> +// C++ Includes<br>
> +#include <vector><br>
> +#include <map><br>
> +<br>
> +// Other libraries and framework includes<br>
> +// Project includes<br>
> +#include "lldb/lldb-private.h"<br>
> +#include "lldb/lldb-types.h"<br>
> +#include "lldb/Core/PluginInterface.h"<br>
> +<br>
> +namespace lldb_private {<br>
> +<br>
> +typedef std::map<lldb::InstrumentationRuntimeType, lldb::InstrumentationRuntimeSP> InstrumentationRuntimeCollection;<br>
> +<br>
> +class InstrumentationRuntime :<br>
> + public std::enable_shared_from_this<InstrumentationRuntime>,<br>
> + public PluginInterface<br>
> +{<br>
> +public:<br>
> +<br>
> + static void<br>
> + ModulesDidLoad(lldb_private::ModuleList &module_list, Process *process, InstrumentationRuntimeCollection &runtimes);<br>
> +<br>
> + virtual void<br>
> + ModulesDidLoad(lldb_private::ModuleList &module_list);<br>
> +<br>
> + virtual bool<br>
> + IsActive();<br>
> +<br>
> +};<br>
> +<br>
> +} // namespace lldb_private<br>
> +<br>
> +#endif // liblldb_InstrumentationRuntime_h_<br>
><br>
> Added: lldb/trunk/include/lldb/Target/InstrumentationRuntimeStopInfo.h<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/InstrumentationRuntimeStopInfo.h?rev=219546&view=auto" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/InstrumentationRuntimeStopInfo.h?rev=219546&view=auto</a><br>
> ==============================================================================<br>
> --- lldb/trunk/include/lldb/Target/InstrumentationRuntimeStopInfo.h (added)<br>
> +++ lldb/trunk/include/lldb/Target/InstrumentationRuntimeStopInfo.h Fri Oct 10 18:43:03 2014<br>
> @@ -0,0 +1,52 @@<br>
> +//===-- InstrumentationRuntimeStopInfo.h ------------------------*- C++ -*-===//<br>
> +//<br>
> +// The LLVM Compiler Infrastructure<br>
> +//<br>
> +// This file is distributed under the University of Illinois Open Source<br>
> +// License. See LICENSE.TXT for details.<br>
> +//<br>
> +//===----------------------------------------------------------------------===//<br>
> +<br>
> +#ifndef liblldb_InstrumentationRuntimeStopInfo_h_<br>
> +#define liblldb_InstrumentationRuntimeStopInfo_h_<br>
> +<br>
> +// C Includes<br>
> +// C++ Includes<br>
> +#include <string><br>
> +<br>
> +// Other libraries and framework includes<br>
> +// Project includes<br>
> +#include "lldb/Target/StopInfo.h"<br>
> +#include "lldb/Core/StructuredData.h"<br>
> +<br>
> +namespace lldb_private {<br>
> +<br>
> +class InstrumentationRuntimeStopInfo : public StopInfo<br>
> +{<br>
> +public:<br>
> +<br>
> + virtual ~InstrumentationRuntimeStopInfo()<br>
> + {<br>
> + }<br>
> +<br>
> + virtual lldb::StopReason<br>
> + GetStopReason () const<br>
> + {<br>
> + return lldb::eStopReasonInstrumentation;<br>
> + }<br>
> +<br>
> + virtual const char *<br>
> + GetDescription ();<br>
> +<br>
> + static lldb::StopInfoSP<br>
> + CreateStopReasonWithInstrumentationData (Thread &thread, std::string description, StructuredData::ObjectSP additional_data);<br>
> +<br>
> +private:<br>
> +<br>
> + InstrumentationRuntimeStopInfo(Thread &thread, std::string description, StructuredData::ObjectSP additional_data);<br>
> +<br>
> +};<br>
> +<br>
> +} // namespace lldb_private<br>
> +<br>
> +#endif // liblldb_InstrumentationRuntimeStopInfo_h_<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=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/Process.h?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/include/lldb/Target/Process.h (original)<br>
> +++ lldb/trunk/include/lldb/Target/Process.h Fri Oct 10 18:43:03 2014<br>
> @@ -52,6 +52,7 @@<br>
> #include "lldb/Target/ThreadList.h"<br>
> #include "lldb/Target/UnixSignals.h"<br>
> #include "lldb/Utility/PseudoTerminal.h"<br>
> +#include "lldb/Target/InstrumentationRuntime.h"<br>
><br>
> namespace lldb_private {<br>
><br>
> @@ -3082,6 +3083,7 @@ protected:<br>
> AllocatedMemoryCache m_allocated_memory_cache;<br>
> bool m_should_detach; /// Should we detach if the process object goes away with an explicit call to Kill or Detach?<br>
> LanguageRuntimeCollection m_language_runtimes;<br>
> + InstrumentationRuntimeCollection m_instrumentation_runtimes;<br>
> std::unique_ptr<NextEventAction> m_next_event_action_ap;<br>
> std::vector<PreResumeCallbackAndBaton> m_pre_resume_actions;<br>
> ProcessRunLock m_public_run_lock;<br>
><br>
> Modified: lldb/trunk/include/lldb/Target/StopInfo.h<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/StopInfo.h?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/StopInfo.h?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/include/lldb/Target/StopInfo.h (original)<br>
> +++ lldb/trunk/include/lldb/Target/StopInfo.h Fri Oct 10 18:43:03 2014<br>
> @@ -18,6 +18,7 @@<br>
> // Project includes<br>
> #include "lldb/lldb-public.h"<br>
> #include "lldb/Target/Process.h"<br>
> +#include "lldb/Core/StructuredData.h"<br>
><br>
> namespace lldb_private {<br>
><br>
> @@ -140,6 +141,12 @@ public:<br>
> return m_override_should_stop == eLazyBoolYes;<br>
> }<br>
><br>
> + StructuredData::ObjectSP<br>
> + GetExtendedInfo ()<br>
> + {<br>
> + return m_extended_info;<br>
> + }<br>
> +<br>
> static lldb::StopInfoSP<br>
> CreateStopReasonWithBreakpointSiteID (Thread &thread, lldb::break_id_t break_id);<br>
><br>
> @@ -211,6 +218,8 @@ protected:<br>
> LazyBool m_override_should_notify;<br>
> LazyBool m_override_should_stop;<br>
><br>
> + StructuredData::ObjectSP m_extended_info; // The extended info for this stop info<br>
> +<br>
> // This determines whether the target has run since this stop info.<br>
> // N.B. running to evaluate a user expression does not count.<br>
> bool HasTargetRunSinceMe ();<br>
><br>
> Modified: lldb/trunk/include/lldb/Target/Thread.h<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/Thread.h?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/Target/Thread.h?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/include/lldb/Target/Thread.h (original)<br>
> +++ lldb/trunk/include/lldb/Target/Thread.h Fri Oct 10 18:43:03 2014<br>
> @@ -535,7 +535,7 @@ public:<br>
> DumpUsingSettingsFormat (Stream &strm, uint32_t frame_idx);<br>
><br>
> bool<br>
> - GetDescription (Stream &s, lldb::DescriptionLevel level, bool json_output);<br>
> + GetDescription (Stream &s, lldb::DescriptionLevel level, bool print_json_thread, bool print_json_stopinfo);<br>
><br>
> //------------------------------------------------------------------<br>
> /// Default implementation for stepping into.<br>
><br>
> Modified: lldb/trunk/include/lldb/lldb-enumerations.h<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/lldb-enumerations.h?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/lldb-enumerations.h?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/include/lldb/lldb-enumerations.h (original)<br>
> +++ lldb/trunk/include/lldb/lldb-enumerations.h Fri Oct 10 18:43:03 2014<br>
> @@ -185,7 +185,8 @@ namespace lldb {<br>
> eStopReasonException,<br>
> eStopReasonExec, // Program was re-exec'ed<br>
> eStopReasonPlanComplete,<br>
> - eStopReasonThreadExiting<br>
> + eStopReasonThreadExiting,<br>
> + eStopReasonInstrumentation<br>
> } StopReason;<br>
><br>
> //----------------------------------------------------------------------<br>
> @@ -387,6 +388,11 @@ namespace lldb {<br>
> eLanguageTypeDylan = 0x0020, ///< Dylan.<br>
> eNumLanguageTypes<br>
> } LanguageType;<br>
> +<br>
> + typedef enum InstrumentationRuntimeType {<br>
> + eInstrumentationRuntimeTypeAddressSanitizer = 0x0000,<br>
> + eNumInstrumentationRuntimeTypes<br>
> + } InstrumentationRuntimeType;<br>
><br>
> typedef enum DynamicValueType<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=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/lldb-forward.h?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/include/lldb/lldb-forward.h (original)<br>
> +++ lldb/trunk/include/lldb/lldb-forward.h Fri Oct 10 18:43:03 2014<br>
> @@ -104,6 +104,7 @@ class FunctionInfo;<br>
> class InlineFunctionInfo;<br>
> class Instruction;<br>
> class InstructionList;<br>
> +class InstrumentationRuntime;<br>
> class IOHandler;<br>
> class IOObject;<br>
> class IRExecutionUnit;<br>
> @@ -312,6 +313,7 @@ namespace lldb {<br>
> typedef std::shared_ptr<lldb_private::FuncUnwinders> FuncUnwindersSP;<br>
> typedef std::shared_ptr<lldb_private::InlineFunctionInfo> InlineFunctionInfoSP;<br>
> typedef std::shared_ptr<lldb_private::Instruction> InstructionSP;<br>
> + typedef std::shared_ptr<lldb_private::InstrumentationRuntime> InstrumentationRuntimeSP;<br>
> typedef std::shared_ptr<lldb_private::IOHandler> IOHandlerSP;<br>
> typedef std::shared_ptr<lldb_private::IOObject> IOObjectSP;<br>
> typedef std::shared_ptr<lldb_private::JITLoader> JITLoaderSP;<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=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/include/lldb/lldb-private-interfaces.h?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/include/lldb/lldb-private-interfaces.h (original)<br>
> +++ lldb/trunk/include/lldb/lldb-private-interfaces.h Fri Oct 10 18:43:03 2014<br>
> @@ -40,6 +40,8 @@ namespace lldb_private<br>
> typedef lldb::ThreadPlanSP (*ThreadPlanStepFromHereCallback) (ThreadPlan *current_plan, Flags &flags, lldb::FrameComparison operation, void *baton);<br>
> typedef UnwindAssembly* (*UnwindAssemblyCreateInstance) (const ArchSpec &arch);<br>
> typedef lldb::MemoryHistorySP (*MemoryHistoryCreateInstance) (const lldb::ProcessSP &process_sp);<br>
> + typedef lldb::InstrumentationRuntimeType (*InstrumentationRuntimeGetType) ();<br>
> + typedef lldb::InstrumentationRuntimeSP (*InstrumentationRuntimeCreateInstance) (const lldb::ProcessSP &process_sp);<br>
> typedef int (*ComparisonFunction)(const void *, const void *);<br>
> typedef void (*DebuggerInitializeCallback)(Debugger &debugger);<br>
><br>
><br>
> Modified: lldb/trunk/lldb.xcodeproj/project.pbxproj<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/lldb.xcodeproj/project.pbxproj?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/lldb.xcodeproj/project.pbxproj?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/lldb.xcodeproj/project.pbxproj (original)<br>
> +++ lldb/trunk/lldb.xcodeproj/project.pbxproj Fri Oct 10 18:43:03 2014<br>
> @@ -637,6 +637,9 @@<br>
> 8CCB017E19BA28A80009FD44 /* ThreadCollection.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 8CCB017A19BA283D0009FD44 /* ThreadCollection.cpp */; };<br>
> 8CCB018219BA4E270009FD44 /* SBThreadCollection.h in Headers */ = {isa = PBXBuildFile; fileRef = 8CCB018119BA4E210009FD44 /* SBThreadCollection.h */; settings = {ATTRIBUTES = (Public, ); }; };<br>
> 8CCB018319BA51BF0009FD44 /* SBThreadCollection.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 8CCB017F19BA4DD00009FD44 /* SBThreadCollection.cpp */; };<br>
> + 8CF02AE919DCC01900B14BE0 /* InstrumentationRuntime.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 8CF02ADF19DCBF3B00B14BE0 /* InstrumentationRuntime.cpp */; };<br>
> + 8CF02AEA19DCC02100B14BE0 /* AddressSanitizerRuntime.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 8CF02AE519DCBF8400B14BE0 /* AddressSanitizerRuntime.cpp */; };<br>
> + 8CF02AEF19DD16B100B14BE0 /* InstrumentationRuntimeStopInfo.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 8CF02AED19DD15CF00B14BE0 /* InstrumentationRuntimeStopInfo.cpp */; };<br>
> 94094C6B163B6F840083A547 /* ValueObjectCast.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 94094C69163B6CD90083A547 /* ValueObjectCast.cpp */; };<br>
> 940B02F619DC96E700AD0F52 /* SBExecutionContext.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 940B02F519DC96E700AD0F52 /* SBExecutionContext.cpp */; };<br>
> 94145431175E63B500284436 /* lldb-versioning.h in Headers */ = {isa = PBXBuildFile; fileRef = 94145430175D7FDE00284436 /* lldb-versioning.h */; settings = {ATTRIBUTES = (Public, ); }; };<br>
> @@ -1965,6 +1968,12 @@<br>
> 8CCB017F19BA4DD00009FD44 /* SBThreadCollection.cpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.cpp; name = SBThreadCollection.cpp; path = source/API/SBThreadCollection.cpp; sourceTree = "<group>"; };<br>
> 8CCB018119BA4E210009FD44 /* SBThreadCollection.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = SBThreadCollection.h; path = include/lldb/API/SBThreadCollection.h; sourceTree = "<group>"; };<br>
> 8CCB018419BA54930009FD44 /* SBThreadCollection.i */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.c.preprocessed; path = SBThreadCollection.i; sourceTree = "<group>"; };<br>
> + 8CF02ADF19DCBF3B00B14BE0 /* InstrumentationRuntime.cpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.cpp; name = InstrumentationRuntime.cpp; path = source/Target/InstrumentationRuntime.cpp; sourceTree = "<group>"; };<br>
> + 8CF02AE019DCBF3B00B14BE0 /* InstrumentationRuntime.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = InstrumentationRuntime.h; path = include/lldb/Target/InstrumentationRuntime.h; sourceTree = "<group>"; };<br>
> + 8CF02AE519DCBF8400B14BE0 /* AddressSanitizerRuntime.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = AddressSanitizerRuntime.cpp; sourceTree = "<group>"; };<br>
> + 8CF02AE619DCBF8400B14BE0 /* AddressSanitizerRuntime.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = AddressSanitizerRuntime.h; sourceTree = "<group>"; };<br>
> + 8CF02AED19DD15CF00B14BE0 /* InstrumentationRuntimeStopInfo.cpp */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.cpp.cpp; name = InstrumentationRuntimeStopInfo.cpp; path = source/Target/InstrumentationRuntimeStopInfo.cpp; sourceTree = "<group>"; };<br>
> + 8CF02AEE19DD15CF00B14BE0 /* InstrumentationRuntimeStopInfo.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = InstrumentationRuntimeStopInfo.h; path = include/lldb/Target/InstrumentationRuntimeStopInfo.h; sourceTree = "<group>"; };<br>
> 94005E0313F438DF001EF42D /* python-wrapper.swig */ = {isa = PBXFileReference; lastKnownFileType = text; path = "python-wrapper.swig"; sourceTree = "<group>"; };<br>
> 94005E0513F45A1B001EF42D /* embedded_interpreter.py */ = {isa = PBXFileReference; lastKnownFileType = text.script.python; name = embedded_interpreter.py; path = source/Interpreter/embedded_interpreter.py; sourceTree = "<group>"; };<br>
> 94031A9F13CF5B3D00DCFF3C /* PriorityPointerPair.h */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.c.h; name = PriorityPointerPair.h; path = include/lldb/Utility/PriorityPointerPair.h; sourceTree = "<group>"; };<br>
> @@ -2423,6 +2432,7 @@<br>
> 260C897110F57C5600BB2B04 /* Plugins */ = {<br>
> isa = PBXGroup;<br>
> children = (<br>
> + 8CF02ADD19DCBEC200B14BE0 /* InstrumentationRuntime */,<br>
> 8C2D6A58197A1FB9006989C9 /* MemoryHistory */,<br>
> 26DB3E051379E7AD0080DC73 /* ABI */,<br>
> 260C897210F57C5600BB2B04 /* Disassembler */,<br>
> @@ -3746,6 +3756,10 @@<br>
> 26BC7DEF10F1B80200F91463 /* Target */ = {<br>
> isa = PBXGroup;<br>
> children = (<br>
> + 8CF02AE019DCBF3B00B14BE0 /* InstrumentationRuntime.h */,<br>
> + 8CF02ADF19DCBF3B00B14BE0 /* InstrumentationRuntime.cpp */,<br>
> + 8CF02AEE19DD15CF00B14BE0 /* InstrumentationRuntimeStopInfo.h */,<br>
> + 8CF02AED19DD15CF00B14BE0 /* InstrumentationRuntimeStopInfo.cpp */,<br>
> 3FDFD6C3199C396E009756A7 /* FileAction.h */,<br>
> 3FDFDDBC199C3A06009756A7 /* FileAction.cpp */,<br>
> 23EDE3311926843600F6A132 /* NativeRegisterContext.h */,<br>
> @@ -4264,6 +4278,23 @@<br>
> path = asan;<br>
> sourceTree = "<group>";<br>
> };<br>
> + 8CF02ADD19DCBEC200B14BE0 /* InstrumentationRuntime */ = {<br>
> + isa = PBXGroup;<br>
> + children = (<br>
> + 8CF02ADE19DCBEE600B14BE0 /* AddressSanitizer */,<br>
> + );<br>
> + path = InstrumentationRuntime;<br>
> + sourceTree = "<group>";<br>
> + };<br>
> + 8CF02ADE19DCBEE600B14BE0 /* AddressSanitizer */ = {<br>
> + isa = PBXGroup;<br>
> + children = (<br>
> + 8CF02AE519DCBF8400B14BE0 /* AddressSanitizerRuntime.cpp */,<br>
> + 8CF02AE619DCBF8400B14BE0 /* AddressSanitizerRuntime.h */,<br>
> + );<br>
> + path = AddressSanitizer;<br>
> + sourceTree = "<group>";<br>
> + };<br>
> 9457596415349416005A9070 /* POSIX */ = {<br>
> isa = PBXGroup;<br>
> children = (<br>
> @@ -4965,6 +4996,7 @@<br>
> 2689000713353DB600698AC0 /* BreakpointSite.cpp in Sources */,<br>
> 2689000913353DB600698AC0 /* BreakpointSiteList.cpp in Sources */,<br>
> 26474CC918D0CB5B0073DEBA /* RegisterContextMemory.cpp in Sources */,<br>
> + 8CF02AEA19DCC02100B14BE0 /* AddressSanitizerRuntime.cpp in Sources */,<br>
> 26474CB218D0CB180073DEBA /* RegisterContextLinux_i386.cpp in Sources */,<br>
> 26474CCB18D0CB5B0073DEBA /* RegisterContextPOSIX_mips64.cpp in Sources */,<br>
> 2689000B13353DB600698AC0 /* Stoppoint.cpp in Sources */,<br>
> @@ -5002,6 +5034,7 @@<br>
> 2689002813353DDE00698AC0 /* CommandObjectThread.cpp in Sources */,<br>
> 2689002913353DDE00698AC0 /* CommandObjectVersion.cpp in Sources */,<br>
> 2689002A13353E0400698AC0 /* Address.cpp in Sources */,<br>
> + 8CF02AE919DCC01900B14BE0 /* InstrumentationRuntime.cpp in Sources */,<br>
> 2689002B13353E0400698AC0 /* AddressRange.cpp in Sources */,<br>
> 236124A41986B4E2004EFC37 /* IOObject.cpp in Sources */,<br>
> 2689002C13353E0400698AC0 /* AddressResolver.cpp in Sources */,<br>
> @@ -5223,6 +5256,7 @@<br>
> 268900FA13353E6F00698AC0 /* ThreadList.cpp in Sources */,<br>
> 268900FB13353E6F00698AC0 /* ThreadPlan.cpp in Sources */,<br>
> 232CB619191E00CD00EF39FC /* NativeProcessProtocol.cpp in Sources */,<br>
> + 8CF02AEF19DD16B100B14BE0 /* InstrumentationRuntimeStopInfo.cpp in Sources */,<br>
> 268900FC13353E6F00698AC0 /* ThreadPlanBase.cpp in Sources */,<br>
> 268900FD13353E6F00698AC0 /* ThreadPlanCallFunction.cpp in Sources */,<br>
> 268900FE13353E6F00698AC0 /* ThreadPlanCallUserExpression.cpp in Sources */,<br>
><br>
> Modified: lldb/trunk/scripts/Python/interface/SBThread.i<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/scripts/Python/interface/SBThread.i?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/scripts/Python/interface/SBThread.i?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/scripts/Python/interface/SBThread.i (original)<br>
> +++ lldb/trunk/scripts/Python/interface/SBThread.i Fri Oct 10 18:43:03 2014<br>
> @@ -110,6 +110,14 @@ public:<br>
> ") GetStopReasonDataAtIndex;<br>
> uint64_t<br>
> GetStopReasonDataAtIndex(uint32_t idx);<br>
> +<br>
> + %feature("autodoc", "<br>
> + Collects a thread's stop reason extended information dictionary and prints it<br>
> + into the SBStream in a JSON format. The format of this JSON dictionary depends<br>
> + on the stop reason and is currently used only for instrumentation plugins.<br>
> + ") GetStopReasonExtendedInfoAsJSON;<br>
> + bool<br>
> + GetStopReasonExtendedInfoAsJSON (lldb::SBStream &stream);<br>
><br>
> %feature("autodoc", "<br>
> Pass only an (int)length and expect to get a Python string describing the<br>
><br>
> Modified: lldb/trunk/source/API/SBThread.cpp<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/API/SBThread.cpp?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/API/SBThread.cpp?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/API/SBThread.cpp (original)<br>
> +++ lldb/trunk/source/API/SBThread.cpp Fri Oct 10 18:43:03 2014<br>
> @@ -195,6 +195,7 @@ SBThread::GetStopReasonDataCount ()<br>
> case eStopReasonExec:<br>
> case eStopReasonPlanComplete:<br>
> case eStopReasonThreadExiting:<br>
> + case eStopReasonInstrumentation:<br>
> // There is no data for these stop reasons.<br>
> return 0;<br>
><br>
> @@ -255,6 +256,7 @@ SBThread::GetStopReasonDataAtIndex (uint<br>
> case eStopReasonExec:<br>
> case eStopReasonPlanComplete:<br>
> case eStopReasonThreadExiting:<br>
> + case eStopReasonInstrumentation:<br>
> // There is no data for these stop reasons.<br>
> return 0;<br>
><br>
> @@ -306,6 +308,26 @@ SBThread::GetStopReasonDataAtIndex (uint<br>
> return 0;<br>
> }<br>
><br>
> +bool<br>
> +SBThread::GetStopReasonExtendedInfoAsJSON (lldb::SBStream &stream)<br>
> +{<br>
> + Stream &strm = stream.ref();<br>
> +<br>
> + ExecutionContext exe_ctx (m_opaque_sp.get());<br>
> + if (! exe_ctx.HasThreadScope())<br>
> + return false;<br>
> +<br>
> +<br>
> + StopInfoSP stop_info = exe_ctx.GetThreadPtr()->GetStopInfo();<br>
> + StructuredData::ObjectSP info = stop_info->GetExtendedInfo();<br>
> + if (! info)<br>
> + return false;<br>
> +<br>
> + info->Dump(strm);<br>
> +<br>
> + return true;<br>
> +}<br>
> +<br>
> size_t<br>
> SBThread::GetStopDescription (char *dst, size_t dst_len)<br>
> {<br>
><br>
> Modified: lldb/trunk/source/Commands/CommandObjectThread.cpp<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Commands/CommandObjectThread.cpp?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Commands/CommandObjectThread.cpp?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Commands/CommandObjectThread.cpp (original)<br>
> +++ lldb/trunk/source/Commands/CommandObjectThread.cpp Fri Oct 10 18:43:03 2014<br>
> @@ -1431,7 +1431,8 @@ public:<br>
> void<br>
> OptionParsingStarting ()<br>
> {<br>
> - m_json = false;<br>
> + m_json_thread = false;<br>
> + m_json_stopinfo = false;<br>
> }<br>
><br>
> virtual<br>
> @@ -1448,10 +1449,14 @@ public:<br>
> switch (short_option)<br>
> {<br>
> case 'j':<br>
> - m_json = true;<br>
> + m_json_thread = true;<br>
> + break;<br>
> +<br>
> + case 's':<br>
> + m_json_stopinfo = true;<br>
> break;<br>
><br>
> - default:<br>
> + default:<br>
> return Error("invalid short option character '%c'", short_option);<br>
><br>
> }<br>
> @@ -1464,7 +1469,8 @@ public:<br>
> return g_option_table;<br>
> }<br>
><br>
> - bool m_json;<br>
> + bool m_json_thread;<br>
> + bool m_json_stopinfo;<br>
><br>
> static OptionDefinition g_option_table[];<br>
> };<br>
> @@ -1486,7 +1492,7 @@ public:<br>
> HandleOneThread (Thread &thread, CommandReturnObject &result)<br>
> {<br>
> Stream &strm = result.GetOutputStream();<br>
> - if (!thread.GetDescription (strm, eDescriptionLevelFull, m_options.m_json))<br>
> + if (!thread.GetDescription (strm, eDescriptionLevelFull, m_options.m_json_thread, m_options.m_json_stopinfo))<br>
> {<br>
> result.AppendErrorWithFormat ("error displaying info for thread: \"%d\"\n", thread.GetIndexID());<br>
> result.SetStatus (eReturnStatusFailed);<br>
> @@ -1503,6 +1509,7 @@ OptionDefinition<br>
> CommandObjectThreadInfo::CommandOptions::g_option_table[] =<br>
> {<br>
> { LLDB_OPT_SET_ALL, false, "json",'j', OptionParser::eNoArgument, NULL, NULL, 0, eArgTypeNone, "Display the thread info in JSON format."},<br>
> + { LLDB_OPT_SET_ALL, false, "stop-info",'s', OptionParser::eNoArgument, NULL, NULL, 0, eArgTypeNone, "Display the extended stop info in JSON format."},<br>
><br>
> { 0, false, NULL, 0, 0, NULL, NULL, 0, eArgTypeNone, NULL }<br>
> };<br>
><br>
> Modified: lldb/trunk/source/Core/Debugger.cpp<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Core/Debugger.cpp?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Core/Debugger.cpp?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Core/Debugger.cpp (original)<br>
> +++ lldb/trunk/source/Core/Debugger.cpp Fri Oct 10 18:43:03 2014<br>
> @@ -3234,6 +3234,7 @@ Debugger::HandleProcessEvent (const Even<br>
> case eStopReasonException:<br>
> case eStopReasonExec:<br>
> case eStopReasonThreadExiting:<br>
> + case eStopReasonInstrumentation:<br>
> if (!other_thread)<br>
> other_thread = thread;<br>
> break;<br>
><br>
> Modified: lldb/trunk/source/Core/PluginManager.cpp<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Core/PluginManager.cpp?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Core/PluginManager.cpp?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Core/PluginManager.cpp (original)<br>
> +++ lldb/trunk/source/Core/PluginManager.cpp Fri Oct 10 18:43:03 2014<br>
> @@ -2172,6 +2172,125 @@ PluginManager::GetMemoryHistoryCreateCal<br>
> return NULL;<br>
> }<br>
><br>
> +#pragma mark InstrumentationRuntime<br>
> +<br>
> +struct InstrumentationRuntimeInstance<br>
> +{<br>
> + InstrumentationRuntimeInstance() :<br>
> + name(),<br>
> + description(),<br>
> + create_callback(NULL)<br>
> + {<br>
> + }<br>
> +<br>
> + ConstString name;<br>
> + std::string description;<br>
> + InstrumentationRuntimeCreateInstance create_callback;<br>
> + InstrumentationRuntimeGetType get_type_callback;<br>
> +};<br>
> +<br>
> +typedef std::vector<InstrumentationRuntimeInstance> InstrumentationRuntimeInstances;<br>
> +<br>
> +static Mutex &<br>
> +GetInstrumentationRuntimeMutex ()<br>
> +{<br>
> + static Mutex g_instances_mutex (Mutex::eMutexTypeRecursive);<br>
> + return g_instances_mutex;<br>
> +}<br>
> +<br>
> +static InstrumentationRuntimeInstances &<br>
> +GetInstrumentationRuntimeInstances ()<br>
> +{<br>
> + static InstrumentationRuntimeInstances g_instances;<br>
> + return g_instances;<br>
> +}<br>
> +<br>
> +bool<br>
> +PluginManager::RegisterPlugin<br>
> +(<br>
> + const ConstString &name,<br>
> + const char *description,<br>
> + InstrumentationRuntimeCreateInstance create_callback,<br>
> + InstrumentationRuntimeGetType get_type_callback<br>
> + )<br>
> +{<br>
> + if (create_callback)<br>
> + {<br>
> + InstrumentationRuntimeInstance instance;<br>
> + assert ((bool)name);<br>
> + <a href="http://instance.name" target="_blank">instance.name</a> = name;<br>
> + if (description && description[0])<br>
> + instance.description = description;<br>
> + instance.create_callback = create_callback;<br>
> + instance.get_type_callback = get_type_callback;<br>
> + Mutex::Locker locker (GetInstrumentationRuntimeMutex ());<br>
> + GetInstrumentationRuntimeInstances ().push_back (instance);<br>
> + }<br>
> + return false;<br>
> +}<br>
> +<br>
> +bool<br>
> +PluginManager::UnregisterPlugin (InstrumentationRuntimeCreateInstance create_callback)<br>
> +{<br>
> + if (create_callback)<br>
> + {<br>
> + Mutex::Locker locker (GetInstrumentationRuntimeMutex ());<br>
> + InstrumentationRuntimeInstances &instances = GetInstrumentationRuntimeInstances ();<br>
> +<br>
> + InstrumentationRuntimeInstances::iterator pos, end = instances.end();<br>
> + for (pos = instances.begin(); pos != end; ++ pos)<br>
> + {<br>
> + if (pos->create_callback == create_callback)<br>
> + {<br>
> + instances.erase(pos);<br>
> + return true;<br>
> + }<br>
> + }<br>
> + }<br>
> + return false;<br>
> +}<br>
> +<br>
> +InstrumentationRuntimeGetType<br>
> +PluginManager::GetInstrumentationRuntimeGetTypeCallbackAtIndex (uint32_t idx)<br>
> +{<br>
> + Mutex::Locker locker (GetInstrumentationRuntimeMutex ());<br>
> + InstrumentationRuntimeInstances &instances = GetInstrumentationRuntimeInstances ();<br>
> + if (idx < instances.size())<br>
> + return instances[idx].get_type_callback;<br>
> + return NULL;<br>
> +}<br>
> +<br>
> +InstrumentationRuntimeCreateInstance<br>
> +PluginManager::GetInstrumentationRuntimeCreateCallbackAtIndex (uint32_t idx)<br>
> +{<br>
> + Mutex::Locker locker (GetInstrumentationRuntimeMutex ());<br>
> + InstrumentationRuntimeInstances &instances = GetInstrumentationRuntimeInstances ();<br>
> + if (idx < instances.size())<br>
> + return instances[idx].create_callback;<br>
> + return NULL;<br>
> +}<br>
> +<br>
> +<br>
> +InstrumentationRuntimeCreateInstance<br>
> +PluginManager::GetInstrumentationRuntimeCreateCallbackForPluginName (const ConstString &name)<br>
> +{<br>
> + if (name)<br>
> + {<br>
> + Mutex::Locker locker (GetInstrumentationRuntimeMutex ());<br>
> + InstrumentationRuntimeInstances &instances = GetInstrumentationRuntimeInstances ();<br>
> +<br>
> + InstrumentationRuntimeInstances::iterator pos, end = instances.end();<br>
> + for (pos = instances.begin(); pos != end; ++ pos)<br>
> + {<br>
> + if (name == pos->name)<br>
> + return pos->create_callback;<br>
> + }<br>
> + }<br>
> + return NULL;<br>
> +}<br>
> +<br>
> +#pragma mark PluginManager<br>
> +<br>
> void<br>
> PluginManager::DebuggerInitialize (Debugger &debugger)<br>
> {<br>
><br>
> Added: lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.cpp<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.cpp?rev=219546&view=auto" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.cpp?rev=219546&view=auto</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.cpp (added)<br>
> +++ lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.cpp Fri Oct 10 18:43:03 2014<br>
> @@ -0,0 +1,300 @@<br>
> +//===-- AddressSanitizerRuntime.cpp -----------------------------*- C++ -*-===//<br>
> +//<br>
> +// The LLVM Compiler Infrastructure<br>
> +//<br>
> +// This file is distributed under the University of Illinois Open Source<br>
> +// License. See LICENSE.TXT for details.<br>
> +//<br>
> +//===----------------------------------------------------------------------===//<br>
> +<br>
> +#include "AddressSanitizerRuntime.h"<br>
> +<br>
> +#include "lldb/Core/PluginInterface.h"<br>
> +#include "lldb/Core/PluginManager.h"<br>
> +#include "lldb/Core/ModuleList.h"<br>
> +#include "lldb/Symbol/SymbolContext.h"<br>
> +#include "lldb/Core/Module.h"<br>
> +#include "lldb/Target/Target.h"<br>
> +#include "lldb/Target/Thread.h"<br>
> +#include "lldb/Target/StopInfo.h"<br>
> +#include "lldb/Symbol/Symbol.h"<br>
> +#include "lldb/Breakpoint/StoppointCallbackContext.h"<br>
> +#include "lldb/Core/ValueObject.h"<br>
> +#include "lldb/Target/InstrumentationRuntimeStopInfo.h"<br>
> +#include "lldb/Core/Debugger.h"<br>
> +#include "lldb/Core/Stream.h"<br>
> +#include "lldb/Core/StreamFile.h"<br>
> +<br>
> +using namespace lldb;<br>
> +using namespace lldb_private;<br>
> +<br>
> +lldb::InstrumentationRuntimeSP<br>
> +AddressSanitizerRuntime::CreateInstance (const lldb::ProcessSP &process_sp)<br>
> +{<br>
> + return InstrumentationRuntimeSP(new AddressSanitizerRuntime(process_sp));<br>
> +}<br>
> +<br>
> +void<br>
> +AddressSanitizerRuntime::Initialize()<br>
> +{<br>
> + PluginManager::RegisterPlugin (GetPluginNameStatic(),<br>
> + "AddressSanitizer instrumentation runtime plugin.",<br>
> + CreateInstance,<br>
> + GetTypeStatic);<br>
> +}<br>
> +<br>
> +void<br>
> +AddressSanitizerRuntime::Terminate()<br>
> +{<br>
> + PluginManager::UnregisterPlugin (CreateInstance);<br>
> +}<br>
> +<br>
> +lldb_private::ConstString<br>
> +AddressSanitizerRuntime::GetPluginNameStatic()<br>
> +{<br>
> + return ConstString("AddressSanitizer");<br>
> +}<br>
> +<br>
> +lldb::InstrumentationRuntimeType<br>
> +AddressSanitizerRuntime::GetTypeStatic()<br>
> +{<br>
> + return eInstrumentationRuntimeTypeAddressSanitizer;<br>
> +}<br>
> +<br>
> +AddressSanitizerRuntime::AddressSanitizerRuntime(const ProcessSP &process_sp) :<br>
> + m_is_active(false),<br>
> + m_runtime_module(),<br>
> + m_process(process_sp),<br>
> + m_breakpoint_id(0)<br>
> +{<br>
> +}<br>
> +<br>
> +AddressSanitizerRuntime::~AddressSanitizerRuntime()<br>
> +{<br>
> + Deactivate();<br>
> +}<br>
> +<br>
> +bool ModuleContainsASanRuntime(Module * module)<br>
> +{<br>
> + SymbolContextList sc_list;<br>
> + const bool include_symbols = true;<br>
> + const bool append = true;<br>
> + const bool include_inlines = true;<br>
> +<br>
> + size_t num_matches = module->FindFunctions(ConstString("__asan_get_alloc_stack"), NULL, eFunctionNameTypeAuto, include_symbols, include_inlines, append, sc_list);<br>
> +<br>
> + return num_matches > 0;<br>
> +}<br>
> +<br>
> +void<br>
> +AddressSanitizerRuntime::ModulesDidLoad(lldb_private::ModuleList &module_list)<br>
> +{<br>
> + if (IsActive())<br>
> + return;<br>
> +<br>
> + if (m_runtime_module) {<br>
> + Activate();<br>
> + return;<br>
> + }<br>
> +<br>
> + Mutex::Locker modules_locker(module_list.GetMutex());<br>
> + const size_t num_modules = module_list.GetSize();<br>
> + for (size_t i = 0; i < num_modules; ++i)<br>
> + {<br>
> + Module *module_pointer = module_list.GetModulePointerAtIndexUnlocked(i);<br>
> + const FileSpec & file_spec = module_pointer->GetFileSpec();<br>
> + if (! file_spec)<br>
> + continue;<br>
> +<br>
> + static RegularExpression g_asan_runtime_regex("libclang_rt.asan_(.*)_dynamic\\.dylib");<br>
> + if (g_asan_runtime_regex.Execute (file_spec.GetFilename().GetCString()) || module_pointer->IsExecutable())<br>
> + {<br>
> + if (ModuleContainsASanRuntime(module_pointer))<br>
> + {<br>
> + m_runtime_module = module_pointer->shared_from_this();<br>
> + Activate();<br>
> + return;<br>
> + }<br>
> + }<br>
> + }<br>
> +}<br>
> +<br>
> +bool<br>
> +AddressSanitizerRuntime::IsActive()<br>
> +{<br>
> + return m_is_active;<br>
> +}<br>
> +<br>
> +#define RETRIEVE_REPORT_DATA_FUNCTION_TIMEOUT_USEC 2*1000*1000<br>
> +<br>
> +const char *<br>
> +address_sanitizer_retrieve_report_data_command = R"(<br>
> + struct {<br>
> + int present;<br>
> + void *pc, *bp, *sp, *address;<br>
> + int access_type;<br>
> + size_t access_size;<br>
> + const char *description;<br>
> + } t;<br>
> +<br>
> + t.present = ((int (*) ())__asan_report_present)();<br>
> + t.pc = ((void * (*) ())__asan_get_report_pc)();<br>
> + /* commented out because rdar://problem/18533301<br>
> + t.bp = ((void * (*) ())__asan_get_report_bp)();<br>
> + t.sp = ((void * (*) ())__asan_get_report_sp)();<br>
> + */<br>
> + t.address = ((void * (*) ())__asan_get_report_address)();<br>
> + t.description = ((const char * (*) ())__asan_get_report_description)();<br>
> + t.access_type = ((int (*) ())__asan_get_report_access_type)();<br>
> + t.access_size = ((size_t (*) ())__asan_get_report_access_size)();<br>
> +<br>
> + t;<br>
> +)";<br>
> +<br>
> +StructuredData::ObjectSP<br>
> +AddressSanitizerRuntime::RetrieveReportData()<br>
> +{<br>
> + ThreadSP thread_sp = m_process->GetThreadList().GetSelectedThread();<br>
> + StackFrameSP frame_sp = thread_sp->GetSelectedFrame();<br>
> +<br>
> + if (!frame_sp)<br>
> + return StructuredData::ObjectSP();<br>
> +<br>
> + EvaluateExpressionOptions options;<br>
> + options.SetUnwindOnError(true);<br>
> + options.SetTryAllThreads(true);<br>
> + options.SetStopOthers(true);<br>
> + options.SetIgnoreBreakpoints(true);<br>
> + options.SetTimeoutUsec(RETRIEVE_REPORT_DATA_FUNCTION_TIMEOUT_USEC);<br>
> +<br>
> + ValueObjectSP return_value_sp;<br>
> + if (m_process->GetTarget().EvaluateExpression(address_sanitizer_retrieve_report_data_command, frame_sp.get(), return_value_sp, options) != eExpressionCompleted)<br>
> + return StructuredData::ObjectSP();<br>
> +<br>
> + int present = return_value_sp->GetValueForExpressionPath(".present")->GetValueAsUnsigned(0);<br>
> + if (present != 1)<br>
> + return StructuredData::ObjectSP();<br>
> +<br>
> + addr_t pc = return_value_sp->GetValueForExpressionPath(".pc")->GetValueAsUnsigned(0);<br>
> + addr_t bp = return_value_sp->GetValueForExpressionPath(".bp")->GetValueAsUnsigned(0);<br>
> + addr_t sp = return_value_sp->GetValueForExpressionPath(".sp")->GetValueAsUnsigned(0);<br>
> + addr_t address = return_value_sp->GetValueForExpressionPath(".address")->GetValueAsUnsigned(0);<br>
> + addr_t access_type = return_value_sp->GetValueForExpressionPath(".access_type")->GetValueAsUnsigned(0);<br>
> + addr_t access_size = return_value_sp->GetValueForExpressionPath(".access_size")->GetValueAsUnsigned(0);<br>
> + addr_t description_ptr = return_value_sp->GetValueForExpressionPath(".description")->GetValueAsUnsigned(0);<br>
> + std::string description;<br>
> + Error error;<br>
> + m_process->ReadCStringFromMemory(description_ptr, description, error);<br>
> +<br>
> + StructuredData::Dictionary *dict = new StructuredData::Dictionary();<br>
> + dict->AddStringItem("instrumentation_class", "AddressSanitizer");<br>
> + dict->AddStringItem("stop_type", "fatal_error");<br>
> + dict->AddIntegerItem("pc", pc);<br>
> + dict->AddIntegerItem("bp", bp);<br>
> + dict->AddIntegerItem("sp", sp);<br>
> + dict->AddIntegerItem("address", address);<br>
> + dict->AddIntegerItem("access_type", access_type);<br>
> + dict->AddIntegerItem("access_size", access_size);<br>
> + dict->AddStringItem("description", description);<br>
> +<br>
> + return StructuredData::ObjectSP(dict);<br>
> +}<br>
> +<br>
> +std::string<br>
> +AddressSanitizerRuntime::FormatDescription(StructuredData::ObjectSP report)<br>
> +{<br>
> + std::string description = report->GetAsDictionary()->GetValueForKey("description")->GetAsString()->GetValue();<br>
> + if (description == "heap-use-after-free") {<br>
> + return "Use of deallocated memory detected";<br>
> + } else if (description == "heap-buffer-overflow") {<br>
> + return "Heap buffer overflow detected";<br>
> + } else if (description == "stack-buffer-underflow") {<br>
> + return "Stack buffer underflow detected";<br>
> + } else if (description == "initialization-order-fiasco") {<br>
> + return "Initialization order problem detected";<br>
> + } else if (description == "stack-buffer-overflow") {<br>
> + return "Stack buffer overflow detected";<br>
> + } else if (description == "stack-use-after-return") {<br>
> + return "Use of returned stack memory detected";<br>
> + } else if (description == "use-after-poison") {<br>
> + return "Use of poisoned memory detected";<br>
> + } else if (description == "container-overflow") {<br>
> + return "Container overflow detected";<br>
> + } else if (description == "stack-use-after-scope") {<br>
> + return "Use of out-of-scope stack memory detected";<br>
> + } else if (description == "global-buffer-overflow") {<br>
> + return "Global buffer overflow detected";<br>
> + } else if (description == "unknown-crash") {<br>
> + return "Invalid memory access detected";<br>
> + }<br>
> +<br>
> + // for unknown report codes just show the code<br>
> + return description;<br>
> +}<br>
> +<br>
> +bool<br>
> +AddressSanitizerRuntime::NotifyBreakpointHit(void *baton, StoppointCallbackContext *context, user_id_t break_id, user_id_t break_loc_id)<br>
> +{<br>
> + assert (baton && "null baton");<br>
> + if (!baton)<br>
> + return false;<br>
> +<br>
> + AddressSanitizerRuntime *const instance = static_cast<AddressSanitizerRuntime*>(baton);<br>
> +<br>
> + StructuredData::ObjectSP report = instance->RetrieveReportData();<br>
> + std::string description;<br>
> + if (report) {<br>
> + description = instance->FormatDescription(report);<br>
> + }<br>
> + ThreadSP thread = context->exe_ctx_ref.GetThreadSP();<br>
> + thread->SetStopInfo(InstrumentationRuntimeStopInfo::CreateStopReasonWithInstrumentationData(*thread, description.c_str(), report));<br>
> +<br>
> + instance->m_runtime_module->ReportWarning("AddressSanitizer report breakpoint hit. Use 'thread info -s' to get extended information about the report.\n");<br>
> +<br>
> + // Return true to stop the target, false to just let the target run.<br>
> + return true;<br>
> +}<br>
> +<br>
> +void<br>
> +AddressSanitizerRuntime::Activate()<br>
> +{<br>
> + if (m_is_active)<br>
> + return;<br>
> +<br>
> + ConstString symbol_name ("__asan::AsanDie()");<br>
> + const Symbol *symbol = m_runtime_module->FindFirstSymbolWithNameAndType (symbol_name, eSymbolTypeCode);<br>
> +<br>
> + if (symbol == NULL)<br>
> + return;<br>
> +<br>
> + if (!symbol->GetAddress().IsValid())<br>
> + return;<br>
> +<br>
> + Target &target = m_process->GetTarget();<br>
> + addr_t symbol_address = symbol->GetAddress().GetOpcodeLoadAddress(&target);<br>
> +<br>
> + if (symbol_address == LLDB_INVALID_ADDRESS)<br>
> + return;<br>
> +<br>
> + bool internal = true;<br>
> + bool hardware = false;<br>
> + Breakpoint *breakpoint = m_process->GetTarget().CreateBreakpoint(symbol_address, internal, hardware).get();<br>
> + breakpoint->SetCallback (AddressSanitizerRuntime::NotifyBreakpointHit, this, true);<br>
> + breakpoint->SetBreakpointKind ("address-sanitizer-report");<br>
> + m_breakpoint_id = breakpoint->GetID();<br>
> +<br>
> + m_runtime_module->ReportWarning("AddressSanitizer debugger support is active. Memory error breakpoint has been installed and you can now use the 'memory history' command.\n");<br>
> +<br>
> + m_is_active = true;<br>
> +}<br>
> +<br>
> +void<br>
> +AddressSanitizerRuntime::Deactivate()<br>
> +{<br>
> + if (m_breakpoint_id != LLDB_INVALID_BREAK_ID)<br>
> + {<br>
> + m_process->GetTarget().RemoveBreakpointByID(m_breakpoint_id);<br>
> + m_breakpoint_id = LLDB_INVALID_BREAK_ID;<br>
> + }<br>
> + m_is_active = false;<br>
> +}<br>
><br>
> Added: lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.h<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.h?rev=219546&view=auto" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.h?rev=219546&view=auto</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.h (added)<br>
> +++ lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.h Fri Oct 10 18:43:03 2014<br>
> @@ -0,0 +1,86 @@<br>
> +//===-- AddressSanitizerRuntime.h -------------------------------*- C++ -*-===//<br>
> +//<br>
> +// The LLVM Compiler Infrastructure<br>
> +//<br>
> +// This file is distributed under the University of Illinois Open Source<br>
> +// License. See LICENSE.TXT for details.<br>
> +//<br>
> +//===----------------------------------------------------------------------===//<br>
> +<br>
> +#ifndef liblldb_AddressSanitizerRuntime_h_<br>
> +#define liblldb_AddressSanitizerRuntime_h_<br>
> +<br>
> +#include "lldb/lldb-private.h"<br>
> +#include "lldb/Target/ABI.h"<br>
> +#include "lldb/Target/InstrumentationRuntime.h"<br>
> +#include "lldb/Target/Process.h"<br>
> +#include "lldb/Core/StructuredData.h"<br>
> +<br>
> +namespace lldb_private {<br>
> +<br>
> +class AddressSanitizerRuntime : public lldb_private::InstrumentationRuntime<br>
> +{<br>
> +public:<br>
> +<br>
> + static lldb::InstrumentationRuntimeSP<br>
> + CreateInstance (const lldb::ProcessSP &process_sp);<br>
> +<br>
> + static void<br>
> + Initialize();<br>
> +<br>
> + static void<br>
> + Terminate();<br>
> +<br>
> + static lldb_private::ConstString<br>
> + GetPluginNameStatic();<br>
> +<br>
> + static lldb::InstrumentationRuntimeType<br>
> + GetTypeStatic();<br>
> +<br>
> + virtual<br>
> + ~AddressSanitizerRuntime();<br>
> +<br>
> + virtual lldb_private::ConstString<br>
> + GetPluginName() { return GetPluginNameStatic(); }<br>
> +<br>
> + virtual lldb::InstrumentationRuntimeType<br>
> + GetType() { return GetTypeStatic(); }<br>
> +<br>
> + virtual uint32_t<br>
> + GetPluginVersion() { return 1; }<br>
> +<br>
> + virtual void<br>
> + ModulesDidLoad(lldb_private::ModuleList &module_list);<br>
> +<br>
> + virtual bool<br>
> + IsActive();<br>
> +<br>
> +private:<br>
> +<br>
> + AddressSanitizerRuntime(const lldb::ProcessSP &process_sp);<br>
> +<br>
> + void<br>
> + Activate();<br>
> +<br>
> + void<br>
> + Deactivate();<br>
> +<br>
> + static bool<br>
> + NotifyBreakpointHit(void *baton, StoppointCallbackContext *context, lldb::user_id_t break_id, lldb::user_id_t break_loc_id);<br>
> +<br>
> + StructuredData::ObjectSP<br>
> + RetrieveReportData();<br>
> +<br>
> + std::string<br>
> + FormatDescription(StructuredData::ObjectSP report);<br>
> +<br>
> + bool m_is_active;<br>
> + lldb::ModuleSP m_runtime_module;<br>
> + lldb::ProcessSP m_process;<br>
> + lldb::user_id_t m_breakpoint_id;<br>
> +<br>
> +};<br>
> +<br>
> +} // namespace lldb_private<br>
> +<br>
> +#endif // liblldb_InstrumentationRuntime_h_<br>
><br>
> Added: lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/CMakeLists.txt<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/CMakeLists.txt?rev=219546&view=auto" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/CMakeLists.txt?rev=219546&view=auto</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/CMakeLists.txt (added)<br>
> +++ lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/CMakeLists.txt Fri Oct 10 18:43:03 2014<br>
> @@ -0,0 +1,5 @@<br>
> +set(LLVM_NO_RTTI 1)<br>
> +<br>
> +add_lldb_library(lldbPluginInstrumentationRuntimeAddressSanitizer<br>
> + AddressSanitizerRuntime.cpp<br>
> + )<br>
><br>
> Added: lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/Makefile<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/Makefile?rev=219546&view=auto" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/Makefile?rev=219546&view=auto</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/Makefile (added)<br>
> +++ lldb/trunk/source/Plugins/InstrumentationRuntime/AddressSanitizer/Makefile Fri Oct 10 18:43:03 2014<br>
> @@ -0,0 +1,14 @@<br>
> +##===- source/Plugins/InstrumentationRuntime/AddressSanitizer Makefile -*-===##<br>
> +#<br>
> +# The LLVM Compiler Infrastructure<br>
> +#<br>
> +# This file is distributed under the University of Illinois Open Source<br>
> +# License. See LICENSE.TXT for details.<br>
> +#<br>
> +##===----------------------------------------------------------------------===##<br>
> +<br>
> +LLDB_LEVEL := ../../../..<br>
> +LIBRARYNAME := lldbPluginInstrumentationRuntimeAddressSanitizer<br>
> +BUILD_ARCHIVE = 1<br>
> +<br>
> +include $(LLDB_LEVEL)/Makefile<br>
><br>
> Added: lldb/trunk/source/Plugins/InstrumentationRuntime/CMakeLists.txt<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/InstrumentationRuntime/CMakeLists.txt?rev=219546&view=auto" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Plugins/InstrumentationRuntime/CMakeLists.txt?rev=219546&view=auto</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Plugins/InstrumentationRuntime/CMakeLists.txt (added)<br>
> +++ lldb/trunk/source/Plugins/InstrumentationRuntime/CMakeLists.txt Fri Oct 10 18:43:03 2014<br>
> @@ -0,0 +1 @@<br>
> +add_subdirectory(AddressSanitizer)<br>
><br>
> Modified: lldb/trunk/source/Target/CMakeLists.txt<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Target/CMakeLists.txt?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Target/CMakeLists.txt?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Target/CMakeLists.txt (original)<br>
> +++ lldb/trunk/source/Target/CMakeLists.txt Fri Oct 10 18:43:03 2014<br>
> @@ -9,6 +9,8 @@ add_lldb_library(lldbTarget<br>
> FileAction.cpp<br>
> JITLoader.cpp<br>
> JITLoaderList.cpp<br>
> + InstrumentationRuntime.cpp<br>
> + InstrumentationRuntimeStopInfo.cpp<br>
> LanguageRuntime.cpp<br>
> Memory.cpp<br>
> MemoryHistory.cpp<br>
><br>
> Added: lldb/trunk/source/Target/InstrumentationRuntime.cpp<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Target/InstrumentationRuntime.cpp?rev=219546&view=auto" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Target/InstrumentationRuntime.cpp?rev=219546&view=auto</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Target/InstrumentationRuntime.cpp (added)<br>
> +++ lldb/trunk/source/Target/InstrumentationRuntime.cpp Fri Oct 10 18:43:03 2014<br>
> @@ -0,0 +1,48 @@<br>
> +//===-- InstrumentationRuntime.cpp ------------------------------*- C++ -*-===//<br>
> +//<br>
> +// The LLVM Compiler Infrastructure<br>
> +//<br>
> +// This file is distributed under the University of Illinois Open Source<br>
> +// License. See LICENSE.TXT for details.<br>
> +//<br>
> +//===----------------------------------------------------------------------===//<br>
> +<br>
> +#include "lldb/lldb-private.h"<br>
> +#include "lldb/Target/Process.h"<br>
> +#include "lldb/Core/PluginManager.h"<br>
> +#include "lldb/Target/InstrumentationRuntime.h"<br>
> +<br>
> +using namespace lldb;<br>
> +using namespace lldb_private;<br>
> +<br>
> +void<br>
> +InstrumentationRuntime::ModulesDidLoad(lldb_private::ModuleList &module_list, lldb_private::Process *process, InstrumentationRuntimeCollection &runtimes)<br>
> +{<br>
> + InstrumentationRuntimeCreateInstance create_callback = NULL;<br>
> + InstrumentationRuntimeGetType get_type_callback;<br>
> + for (uint32_t idx = 0; ; ++idx)<br>
> + {<br>
> + create_callback = PluginManager::GetInstrumentationRuntimeCreateCallbackAtIndex(idx);<br>
> + if (create_callback == NULL)<br>
> + break;<br>
> + get_type_callback = PluginManager::GetInstrumentationRuntimeGetTypeCallbackAtIndex(idx);<br>
> + InstrumentationRuntimeType type = get_type_callback();<br>
> +<br>
> + InstrumentationRuntimeCollection::iterator pos;<br>
> + pos = runtimes.find (type);<br>
> + if (pos == runtimes.end()) {<br>
> + runtimes[type] = create_callback(process->shared_from_this());<br>
> + }<br>
> + }<br>
> +}<br>
> +<br>
> +void<br>
> +InstrumentationRuntime::ModulesDidLoad(lldb_private::ModuleList &module_list)<br>
> +{<br>
> +}<br>
> +<br>
> +bool<br>
> +InstrumentationRuntime::IsActive()<br>
> +{<br>
> + return false;<br>
> +}<br>
><br>
> Added: lldb/trunk/source/Target/InstrumentationRuntimeStopInfo.cpp<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Target/InstrumentationRuntimeStopInfo.cpp?rev=219546&view=auto" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Target/InstrumentationRuntimeStopInfo.cpp?rev=219546&view=auto</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Target/InstrumentationRuntimeStopInfo.cpp (added)<br>
> +++ lldb/trunk/source/Target/InstrumentationRuntimeStopInfo.cpp Fri Oct 10 18:43:03 2014<br>
> @@ -0,0 +1,36 @@<br>
> +//===-- InstrumentationRuntimeStopInfo.cpp ----------------------*- C++ -*-===//<br>
> +//<br>
> +// The LLVM Compiler Infrastructure<br>
> +//<br>
> +// This file is distributed under the University of Illinois Open Source<br>
> +// License. See LICENSE.TXT for details.<br>
> +//<br>
> +//===----------------------------------------------------------------------===//<br>
> +<br>
> +#include "lldb/Target/InstrumentationRuntimeStopInfo.h"<br>
> +<br>
> +#include "lldb/lldb-private.h"<br>
> +#include "lldb/Target/Process.h"<br>
> +#include "lldb/Target/InstrumentationRuntime.h"<br>
> +<br>
> +using namespace lldb;<br>
> +using namespace lldb_private;<br>
> +<br>
> +InstrumentationRuntimeStopInfo::InstrumentationRuntimeStopInfo(Thread &thread, std::string description, StructuredData::ObjectSP additional_data) :<br>
> + StopInfo(thread, 0)<br>
> +{<br>
> + m_extended_info = additional_data;<br>
> + m_description = description;<br>
> +}<br>
> +<br>
> +const char *<br>
> +InstrumentationRuntimeStopInfo::GetDescription ()<br>
> +{<br>
> + return m_description.c_str();<br>
> +}<br>
> +<br>
> +StopInfoSP<br>
> +InstrumentationRuntimeStopInfo::CreateStopReasonWithInstrumentationData (Thread &thread, std::string description, StructuredData::ObjectSP additionalData)<br>
> +{<br>
> + return StopInfoSP(new InstrumentationRuntimeStopInfo(thread, description, additionalData));<br>
> +}<br>
><br>
> Modified: lldb/trunk/source/Target/Process.cpp<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Target/Process.cpp?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Target/Process.cpp?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Target/Process.cpp (original)<br>
> +++ lldb/trunk/source/Target/Process.cpp Fri Oct 10 18:43:03 2014<br>
> @@ -48,6 +48,7 @@<br>
> #include "lldb/Target/Thread.h"<br>
> #include "lldb/Target/ThreadPlan.h"<br>
> #include "lldb/Target/ThreadPlanBase.h"<br>
> +#include "lldb/Target/InstrumentationRuntime.h"<br>
> #include "Plugins/Process/Utility/InferiorCallPOSIX.h"<br>
><br>
> using namespace lldb;<br>
> @@ -843,6 +844,7 @@ Process::Finalize()<br>
> m_memory_cache.Clear();<br>
> m_allocated_memory_cache.Clear();<br>
> m_language_runtimes.clear();<br>
> + m_instrumentation_runtimes.clear();<br>
> m_next_event_action_ap.reset();<br>
> //#ifdef LLDB_CONFIGURATION_DEBUG<br>
> // StreamFile s(stdout, false);<br>
> @@ -5983,6 +5985,7 @@ Process::DidExec ()<br>
> m_image_tokens.clear();<br>
> m_allocated_memory_cache.Clear();<br>
> m_language_runtimes.clear();<br>
> + m_instrumentation_runtimes.clear();<br>
> m_thread_list.DiscardThreadPlans();<br>
> m_memory_cache.Clear(true);<br>
> DoDidExec();<br>
> @@ -6033,13 +6036,24 @@ Process::ResolveIndirectFunction(const A<br>
> void<br>
> Process::ModulesDidLoad (ModuleList &module_list)<br>
> {<br>
> - SystemRuntime *sys_runtime = GetSystemRuntime();<br>
> - if (sys_runtime)<br>
> - {<br>
> - sys_runtime->ModulesDidLoad (module_list);<br>
> - }<br>
> + SystemRuntime *sys_runtime = GetSystemRuntime();<br>
> + if (sys_runtime)<br>
> + {<br>
> + sys_runtime->ModulesDidLoad (module_list);<br>
> + }<br>
> +<br>
> + GetJITLoaders().ModulesDidLoad (module_list);<br>
> +<br>
> + // Give runtimes a chance to be created.<br>
> + InstrumentationRuntime::ModulesDidLoad(module_list, this, m_instrumentation_runtimes);<br>
> +<br>
> + // Tell runtimes about new modules.<br>
> + for (auto pos = m_instrumentation_runtimes.begin(); pos != m_instrumentation_runtimes.end(); ++pos)<br>
> + {<br>
> + InstrumentationRuntimeSP runtime = pos->second;<br>
> + runtime->ModulesDidLoad(module_list);<br>
> + }<br>
><br>
> - GetJITLoaders().ModulesDidLoad (module_list);<br>
> }<br>
><br>
> ThreadCollectionSP<br>
><br>
> Modified: lldb/trunk/source/Target/StopInfo.cpp<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Target/StopInfo.cpp?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Target/StopInfo.cpp?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Target/StopInfo.cpp (original)<br>
> +++ lldb/trunk/source/Target/StopInfo.cpp Fri Oct 10 18:43:03 2014<br>
> @@ -40,7 +40,8 @@ StopInfo::StopInfo (Thread &thread, uint<br>
> m_resume_id (thread.GetProcess()->GetResumeID()),<br>
> m_value (value),<br>
> m_override_should_notify (eLazyBoolCalculate),<br>
> - m_override_should_stop (eLazyBoolCalculate)<br>
> + m_override_should_stop (eLazyBoolCalculate),<br>
> + m_extended_info()<br>
> {<br>
> }<br>
><br>
><br>
> Modified: lldb/trunk/source/Target/Thread.cpp<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Target/Thread.cpp?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/Target/Thread.cpp?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/Target/Thread.cpp (original)<br>
> +++ lldb/trunk/source/Target/Thread.cpp Fri Oct 10 18:43:03 2014<br>
> @@ -2086,6 +2086,7 @@ Thread::StopReasonAsCString (lldb::StopR<br>
> case eStopReasonExec: return "exec";<br>
> case eStopReasonPlanComplete: return "plan complete";<br>
> case eStopReasonThreadExiting: return "thread exiting";<br>
> + case eStopReasonInstrumentation: return "instrumentation break";<br>
> }<br>
><br>
><br>
> @@ -2165,17 +2166,28 @@ Thread::GetStatus (Stream &strm, uint32_<br>
> }<br>
><br>
> bool<br>
> -Thread::GetDescription (Stream &strm, lldb::DescriptionLevel level, bool print_json)<br>
> +Thread::GetDescription (Stream &strm, lldb::DescriptionLevel level, bool print_json_thread, bool print_json_stopinfo)<br>
> {<br>
> DumpUsingSettingsFormat (strm, 0);<br>
> strm.Printf("\n");<br>
><br>
> StructuredData::ObjectSP thread_info = GetExtendedInfo();<br>
> -<br>
> - if (thread_info && print_json)<br>
> + StructuredData::ObjectSP stop_info = m_stop_info_sp->GetExtendedInfo();<br>
> +<br>
> + if (print_json_thread || print_json_stopinfo)<br>
> {<br>
> - thread_info->Dump (strm);<br>
> - strm.Printf("\n");<br>
> + if (thread_info && print_json_thread)<br>
> + {<br>
> + thread_info->Dump (strm);<br>
> + strm.Printf("\n");<br>
> + }<br>
> +<br>
> + if (stop_info && print_json_stopinfo)<br>
> + {<br>
> + stop_info->Dump (strm);<br>
> + strm.Printf("\n");<br>
> + }<br>
> +<br>
> return true;<br>
> }<br>
><br>
><br>
> Modified: lldb/trunk/source/lldb.cpp<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/source/lldb.cpp?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/source/lldb.cpp?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/source/lldb.cpp (original)<br>
> +++ lldb/trunk/source/lldb.cpp Fri Oct 10 18:43:03 2014<br>
> @@ -93,6 +93,7 @@<br>
> #include "Plugins/Process/gdb-remote/ProcessGDBRemote.h"<br>
> #include "Plugins/DynamicLoader/Static/DynamicLoaderStatic.h"<br>
> #include "Plugins/MemoryHistory/asan/MemoryHistoryASan.h"<br>
> +#include "Plugins/InstrumentationRuntime/AddressSanitizer/AddressSanitizerRuntime.h"<br>
><br>
> using namespace lldb;<br>
> using namespace lldb_private;<br>
> @@ -156,6 +157,7 @@ lldb_private::Initialize ()<br>
> JITLoaderGDB::Initialize();<br>
> ProcessElfCore::Initialize();<br>
> MemoryHistoryASan::Initialize();<br>
> + AddressSanitizerRuntime::Initialize();<br>
><br>
> #if defined (__APPLE__)<br>
> //----------------------------------------------------------------------<br>
> @@ -247,6 +249,7 @@ lldb_private::Terminate ()<br>
> JITLoaderGDB::Terminate();<br>
> ProcessElfCore::Terminate();<br>
> MemoryHistoryASan::Terminate();<br>
> + AddressSanitizerRuntime::Terminate();<br>
><br>
> #if defined (__APPLE__)<br>
> DynamicLoaderMacOSXDYLD::Terminate();<br>
><br>
> Removed: lldb/trunk/test/functionalities/asan/TestAsan.py<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/test/functionalities/asan/TestAsan.py?rev=219545&view=auto" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/test/functionalities/asan/TestAsan.py?rev=219545&view=auto</a><br>
> ==============================================================================<br>
> --- lldb/trunk/test/functionalities/asan/TestAsan.py (original)<br>
> +++ lldb/trunk/test/functionalities/asan/TestAsan.py (removed)<br>
> @@ -1,117 +0,0 @@<br>
> -"""<br>
> -Test that ASan memory history provider returns correct stack traces<br>
> -"""<br>
> -<br>
> -import os, time<br>
> -import unittest2<br>
> -import lldb<br>
> -from lldbtest import *<br>
> -import lldbutil<br>
> -<br>
> -class AsanTestCase(TestBase):<br>
> -<br>
> - mydir = TestBase.compute_mydir(__file__)<br>
> -<br>
> - # The default compiler ("clang") may not support Address Sanitizer or it<br>
> - # may not have the debugging API which was recently added, so we're calling<br>
> - # self.useBuiltClang() to use clang from the llvm-build directory instead<br>
> -<br>
> - @unittest2.skipUnless(sys.platform.startswith("darwin"), "requires Darwin")<br>
> - @skipIfRemote<br>
> - @dsym_test<br>
> - def test_with_dsym (self):<br>
> - compiler = self.findBuiltClang ()<br>
> - self.buildDsym (None, compiler)<br>
> - self.asan_tests ()<br>
> -<br>
> - @skipIfFreeBSD # <a href="http://llvm.org/pr21136" target="_blank">llvm.org/pr21136</a> runtimes not yet available by default<br>
> - @skipIfRemote<br>
> - @dwarf_test<br>
> - def test_with_dwarf (self):<br>
> - compiler = self.findBuiltClang ()<br>
> - self.buildDwarf (None, compiler)<br>
> - self.asan_tests ()<br>
> -<br>
> - def setUp(self):<br>
> - # Call super's setUp().<br>
> - TestBase.setUp(self)<br>
> - self.line_malloc = line_number('main.c', '// malloc line')<br>
> - self.line_malloc2 = line_number('main.c', '// malloc2 line')<br>
> - self.line_free = line_number('main.c', '// free line')<br>
> - self.line_breakpoint = line_number('main.c', '// break line')<br>
> -<br>
> - def asan_tests (self):<br>
> - exe = os.path.join (os.getcwd(), "a.out")<br>
> - self.expect("file " + exe, patterns = [ "Current executable set to .*a.out" ])<br>
> -<br>
> - self.runCmd("breakpoint set -f main.c -l %d" % self.line_breakpoint)<br>
> -<br>
> - # "memory history" command should not work without a process<br>
> - self.expect("memory history 0",<br>
> - error = True,<br>
> - substrs = ["invalid process"])<br>
> -<br>
> - self.runCmd("run")<br>
> -<br>
> - # ASan will relaunch the process to insert its library.<br>
> - self.expect("thread list", "Process should be stopped due to exec.",<br>
> - substrs = ['stopped', 'stop reason = exec'])<br>
> -<br>
> - self.runCmd("continue")<br>
> -<br>
> - # the stop reason of the thread should be breakpoint.<br>
> - self.expect("thread list", STOPPED_DUE_TO_BREAKPOINT,<br>
> - substrs = ['stopped', 'stop reason = breakpoint'])<br>
> -<br>
> - # test that the ASan dylib is present<br>
> - self.expect("image lookup -n __asan_describe_address", "__asan_describe_address should be present",<br>
> - substrs = ['1 match found'])<br>
> -<br>
> - # test the 'memory history' command<br>
> - self.expect("memory history 'pointer'",<br>
> - substrs = [<br>
> - 'Memory allocated at', 'a.out`f1', 'main.c:%d' % self.line_malloc,<br>
> - 'Memory deallocated at', 'a.out`f2', 'main.c:%d' % self.line_free])<br>
> -<br>
> - # do the same using SB API<br>
> - process = self.dbg.GetSelectedTarget().process<br>
> - val = process.GetSelectedThread().GetSelectedFrame().EvaluateExpression("pointer")<br>
> - addr = val.GetValueAsUnsigned()<br>
> - threads = process.GetHistoryThreads(addr);<br>
> - self.assertEqual(threads.GetSize(), 2)<br>
> -<br>
> - history_thread = threads.GetThreadAtIndex(0)<br>
> - self.assertTrue(history_thread.num_frames >= 2)<br>
> - self.assertEqual(history_thread.frames[1].GetLineEntry().GetFileSpec().GetFilename(), "main.c")<br>
> - self.assertEqual(history_thread.frames[1].GetLineEntry().GetLine(), self.line_malloc)<br>
> -<br>
> - history_thread = threads.GetThreadAtIndex(1)<br>
> - self.assertTrue(history_thread.num_frames >= 2)<br>
> - self.assertEqual(history_thread.frames[1].GetLineEntry().GetFileSpec().GetFilename(), "main.c")<br>
> - self.assertEqual(history_thread.frames[1].GetLineEntry().GetLine(), self.line_free)<br>
> -<br>
> - # let's free the container (SBThreadCollection) and see if the SBThreads still live<br>
> - threads = None<br>
> - self.assertTrue(history_thread.num_frames >= 2)<br>
> - self.assertEqual(history_thread.frames[1].GetLineEntry().GetFileSpec().GetFilename(), "main.c")<br>
> - self.assertEqual(history_thread.frames[1].GetLineEntry().GetLine(), self.line_free)<br>
> -<br>
> - # now let's break when an ASan report occurs and try the API then<br>
> - self.runCmd("breakpoint set -n __asan_report_error")<br>
> -<br>
> - self.runCmd("continue")<br>
> -<br>
> - # the stop reason of the thread should be breakpoint.<br>
> - self.expect("thread list", STOPPED_DUE_TO_BREAKPOINT,<br>
> - substrs = ['stopped', 'stop reason = breakpoint'])<br>
> -<br>
> - # make sure the 'memory history' command still works even when we're generating a report now<br>
> - self.expect("memory history 'another_pointer'",<br>
> - substrs = [<br>
> - 'Memory allocated at', 'a.out`f1', 'main.c:%d' % self.line_malloc2])<br>
> -<br>
> -if __name__ == '__main__':<br>
> - import atexit<br>
> - lldb.SBDebugger.Initialize()<br>
> - atexit.register(lambda: lldb.SBDebugger.Terminate())<br>
> - unittest2.main()<br>
><br>
> Added: lldb/trunk/test/functionalities/asan/TestReportData.py<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/test/functionalities/asan/TestReportData.py?rev=219546&view=auto" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/test/functionalities/asan/TestReportData.py?rev=219546&view=auto</a><br>
> ==============================================================================<br>
> --- lldb/trunk/test/functionalities/asan/TestReportData.py (added)<br>
> +++ lldb/trunk/test/functionalities/asan/TestReportData.py Fri Oct 10 18:43:03 2014<br>
> @@ -0,0 +1,92 @@<br>
> +"""<br>
> +Test the AddressSanitizer runtime support for report breakpoint and data extraction.<br>
> +"""<br>
> +<br>
> +import os, time<br>
> +import unittest2<br>
> +import lldb<br>
> +from lldbtest import *<br>
> +import lldbutil<br>
> +import json<br>
> +<br>
> +class AsanTestReportDataCase(TestBase):<br>
> +<br>
> + mydir = TestBase.compute_mydir(__file__)<br>
> +<br>
> + # The default compiler ("clang") may not support Address Sanitizer or it<br>
> + # may not have the debugging API which was recently added, so we're calling<br>
> + # self.useBuiltClang() to use clang from the llvm-build directory instead<br>
> +<br>
> + @unittest2.skipUnless(sys.platform.startswith("darwin"), "requires Darwin")<br>
> + @skipIfRemote<br>
> + @dsym_test<br>
> + def test_with_dsym (self):<br>
> + compiler = self.findBuiltClang ()<br>
> + self.buildDsym (None, compiler)<br>
> + self.asan_tests ()<br>
> +<br>
> + @skipIfRemote<br>
> + @dwarf_test<br>
> + def test_with_dwarf (self):<br>
> + compiler = self.findBuiltClang ()<br>
> + self.buildDwarf (None, compiler)<br>
> + self.asan_tests ()<br>
> +<br>
> + def setUp(self):<br>
> + # Call super's setUp().<br>
> + TestBase.setUp(self)<br>
> + self.line_malloc = line_number('main.c', '// malloc line')<br>
> + self.line_malloc2 = line_number('main.c', '// malloc2 line')<br>
> + self.line_free = line_number('main.c', '// free line')<br>
> + self.line_breakpoint = line_number('main.c', '// break line')<br>
> + self.line_crash = line_number('main.c', '// BOOM line')<br>
> +<br>
> + def asan_tests (self):<br>
> + exe = os.path.join (os.getcwd(), "a.out")<br>
> + self.expect("file " + exe, patterns = [ "Current executable set to .*a.out" ])<br>
> + self.runCmd("run")<br>
> +<br>
> + # ASan will relaunch the process to insert its library.<br>
> + self.expect("thread list", "Process should be stopped due to exec.",<br>
> + substrs = ['stopped', 'stop reason = exec'])<br>
> +<br>
> + # no extended info when we have no ASan report<br>
> + thread = self.dbg.GetSelectedTarget().process.GetSelectedThread()<br>
> + s = lldb.SBStream()<br>
> + self.assertFalse(thread.GetStopReasonExtendedInfoAsJSON(s))<br>
> +<br>
> + self.runCmd("continue")<br>
> +<br>
> + self.expect("thread list", "Process should be stopped due to ASan report",<br>
> + substrs = ['stopped', 'stop reason = Use of deallocated memory detected'])<br>
> +<br>
> + self.assertEqual(self.dbg.GetSelectedTarget().process.GetSelectedThread().GetStopReason(), lldb.eStopReasonInstrumentation)<br>
> +<br>
> + self.expect("bt", "The backtrace should show the crashing line",<br>
> + substrs = ['main.c:%d' % self.line_crash])<br>
> +<br>
> + self.expect("thread info -s", "The extended stop info should contain the ASan provided fields",<br>
> + substrs = ["access_size", "access_type", "address", "pc", "description", "heap-use-after-free"])<br>
> +<br>
> + output_lines = self.res.GetOutput().split('\n')<br>
> + json_line = output_lines[2]<br>
> + data = json.loads(json_line)<br>
> + self.assertEqual(data["description"], "heap-use-after-free")<br>
> + self.assertEqual(data["instrumentation_class"], "AddressSanitizer")<br>
> + self.assertEqual(data["stop_type"], "fatal_error")<br>
> +<br>
> + # now let's try the SB API<br>
> + process = self.dbg.GetSelectedTarget().process<br>
> + thread = process.GetSelectedThread()<br>
> +<br>
> + s = lldb.SBStream()<br>
> + self.assertTrue(thread.GetStopReasonExtendedInfoAsJSON(s))<br>
> + s = s.GetData()<br>
> + data2 = json.loads(s)<br>
> + self.assertEqual(data, data2)<br>
> +<br>
> +if __name__ == '__main__':<br>
> + import atexit<br>
> + lldb.SBDebugger.Initialize()<br>
> + atexit.register(lambda: lldb.SBDebugger.Terminate())<br>
> + unittest2.main()<br>
><br>
> Modified: lldb/trunk/test/functionalities/asan/main.c<br>
> URL: <a href="http://llvm.org/viewvc/llvm-project/lldb/trunk/test/functionalities/asan/main.c?rev=219546&r1=219545&r2=219546&view=diff" target="_blank">http://llvm.org/viewvc/llvm-project/lldb/trunk/test/functionalities/asan/main.c?rev=219546&r1=219545&r2=219546&view=diff</a><br>
> ==============================================================================<br>
> --- lldb/trunk/test/functionalities/asan/main.c (original)<br>
> +++ lldb/trunk/test/functionalities/asan/main.c Fri Oct 10 18:43:03 2014<br>
> @@ -28,7 +28,7 @@ int main (int argc, char const *argv[])<br>
><br>
> printf("Hello world!\n"); // break line<br>
><br>
> - pointer[0] = 'A'; // BOOM<br>
> + pointer[0] = 'A'; // BOOM line<br>
><br>
> return 0;<br>
> }<br>
><br>
><br>
> _______________________________________________<br>
> lldb-commits mailing list<br>
> <a href="mailto:lldb-commits@cs.uiuc.edu">lldb-commits@cs.uiuc.edu</a><br>
> <a href="http://lists.cs.uiuc.edu/mailman/listinfo/lldb-commits" target="_blank">http://lists.cs.uiuc.edu/mailman/listinfo/lldb-commits</a><br>
</blockquote></div><br></div></div></div>