forked from OSchip/llvm-project
887 lines
32 KiB
C++
887 lines
32 KiB
C++
//===-- MICmnLLDBDebugger.cpp -----------------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Third party headers:
|
|
#include "lldb/API/SBTarget.h"
|
|
#include "lldb/API/SBThread.h"
|
|
#include "lldb/API/SBProcess.h"
|
|
#include "lldb/API/SBCommandInterpreter.h"
|
|
#include "lldb/API/SBTypeSummary.h"
|
|
#include "lldb/API/SBTypeCategory.h"
|
|
#include "lldb/API/SBTypeNameSpecifier.h"
|
|
#include "lldb/API/SBStream.h"
|
|
#include "lldb/API/SBType.h"
|
|
|
|
// In-house headers:
|
|
#include "MICmnLLDBDebugger.h"
|
|
#include "MICmnResources.h"
|
|
#include "MICmnLog.h"
|
|
#include "MIDriverBase.h"
|
|
#include "MICmnThreadMgrStd.h"
|
|
#include "MICmnLLDBDebuggerHandleEvents.h"
|
|
#include "MICmnLLDBDebugSessionInfo.h"
|
|
#include "MIUtilSingletonHelper.h"
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// MI private summary providers
|
|
static inline bool
|
|
MI_char_summary_provider(lldb::SBValue value, lldb::SBTypeSummaryOptions options, lldb::SBStream &stream)
|
|
{
|
|
if (!value.IsValid())
|
|
return false;
|
|
|
|
lldb::SBType value_type = value.GetType();
|
|
if(!value_type.IsValid())
|
|
return false;
|
|
|
|
lldb::BasicType type_code = value_type.GetBasicType();
|
|
if (type_code == lldb::eBasicTypeSignedChar)
|
|
stream.Printf("%d %s", (int)value.GetValueAsSigned(), value.GetValue());
|
|
else if (type_code == lldb::eBasicTypeUnsignedChar)
|
|
stream.Printf("%u %s", (unsigned)value.GetValueAsUnsigned(), value.GetValue());
|
|
else
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// MI summary helper routines
|
|
static inline bool
|
|
MI_add_summary(lldb::SBTypeCategory category, const char *typeName, lldb::SBTypeSummary::FormatCallback cb,
|
|
uint32_t options, bool regex = false)
|
|
{
|
|
#if defined(LLDB_DISABLE_PYTHON)
|
|
return false;
|
|
#else
|
|
lldb::SBTypeSummary summary = lldb::SBTypeSummary::CreateWithCallback(cb, options);
|
|
return summary.IsValid() ? category.AddTypeSummary(lldb::SBTypeNameSpecifier(typeName, regex), summary) : false;
|
|
#endif
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: CMICmnLLDBDebugger constructor.
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: None.
|
|
// Throws: None.
|
|
//--
|
|
CMICmnLLDBDebugger::CMICmnLLDBDebugger()
|
|
: m_constStrThisThreadId("MI debugger event")
|
|
{
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: CMICmnLLDBDebugger destructor.
|
|
// Type: Overridable.
|
|
// Args: None.
|
|
// Return: None.
|
|
// Throws: None.
|
|
//--
|
|
CMICmnLLDBDebugger::~CMICmnLLDBDebugger()
|
|
{
|
|
Shutdown();
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Initialize resources for *this debugger object.
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::Initialize()
|
|
{
|
|
m_clientUsageRefCnt++;
|
|
|
|
if (m_bInitialized)
|
|
return MIstatus::success;
|
|
|
|
bool bOk = MIstatus::success;
|
|
CMIUtilString errMsg;
|
|
ClrErrorDescription();
|
|
|
|
if (m_pClientDriver == nullptr)
|
|
{
|
|
bOk = false;
|
|
errMsg = MIRSRC(IDS_LLDBDEBUGGER_ERR_CLIENTDRIVER);
|
|
}
|
|
|
|
// Note initialization order is important here as some resources depend on previous
|
|
MI::ModuleInit<CMICmnLog>(IDS_MI_INIT_ERR_LOG, bOk, errMsg);
|
|
MI::ModuleInit<CMICmnResources>(IDS_MI_INIT_ERR_RESOURCES, bOk, errMsg);
|
|
MI::ModuleInit<CMICmnThreadMgrStd>(IDS_MI_INIT_ERR_THREADMGR, bOk, errMsg);
|
|
MI::ModuleInit<CMICmnLLDBDebuggerHandleEvents>(IDS_MI_INIT_ERR_OUTOFBANDHANDLER, bOk, errMsg);
|
|
MI::ModuleInit<CMICmnLLDBDebugSessionInfo>(IDS_MI_INIT_ERR_DEBUGSESSIONINFO, bOk, errMsg);
|
|
|
|
// Note order is important here!
|
|
if (bOk)
|
|
lldb::SBDebugger::Initialize();
|
|
if (bOk && !InitSBDebugger())
|
|
{
|
|
bOk = false;
|
|
if (!errMsg.empty())
|
|
errMsg += ", ";
|
|
errMsg += GetErrorDescription().c_str();
|
|
}
|
|
if (bOk && !InitSBListener())
|
|
{
|
|
bOk = false;
|
|
if (!errMsg.empty())
|
|
errMsg += ", ";
|
|
errMsg += GetErrorDescription().c_str();
|
|
}
|
|
bOk = bOk && InitStdStreams();
|
|
bOk = bOk && RegisterMISummaryProviders();
|
|
m_bInitialized = bOk;
|
|
|
|
if (!bOk && !HaveErrorDescription())
|
|
{
|
|
CMIUtilString strInitError(CMIUtilString::Format(MIRSRC(IDS_MI_INIT_ERR_LLDBDEBUGGER), errMsg.c_str()));
|
|
SetErrorDescription(strInitError);
|
|
}
|
|
|
|
return bOk;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Release resources for *this debugger object.
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::Shutdown()
|
|
{
|
|
if (--m_clientUsageRefCnt > 0)
|
|
return MIstatus::success;
|
|
|
|
if (!m_bInitialized)
|
|
return MIstatus::success;
|
|
|
|
m_bInitialized = false;
|
|
|
|
ClrErrorDescription();
|
|
|
|
bool bOk = MIstatus::success;
|
|
CMIUtilString errMsg;
|
|
|
|
// Explicitly delete the remote target in case MI needs to exit prematurely otherwise
|
|
// LLDB debugger may hang in its Destroy() fn waiting on events
|
|
lldb::SBTarget sbTarget = CMICmnLLDBDebugSessionInfo::Instance().GetTarget();
|
|
m_lldbDebugger.DeleteTarget(sbTarget);
|
|
|
|
// Debug: May need this but does seem to work without it so commented out the fudge 19/06/2014
|
|
// It appears we need to wait as hang does not occur when hitting a debug breakpoint here
|
|
// const std::chrono::milliseconds time( 1000 );
|
|
// std::this_thread::sleep_for( time );
|
|
|
|
lldb::SBDebugger::Destroy(m_lldbDebugger);
|
|
lldb::SBDebugger::Terminate();
|
|
m_pClientDriver = nullptr;
|
|
m_mapBroadcastClassNameToEventMask.clear();
|
|
m_mapIdToEventMask.clear();
|
|
|
|
// Note shutdown order is important here
|
|
MI::ModuleShutdown<CMICmnLLDBDebugSessionInfo>(IDS_MI_INIT_ERR_DEBUGSESSIONINFO, bOk, errMsg);
|
|
MI::ModuleShutdown<CMICmnLLDBDebuggerHandleEvents>(IDS_MI_INIT_ERR_OUTOFBANDHANDLER, bOk, errMsg);
|
|
MI::ModuleShutdown<CMICmnThreadMgrStd>(IDS_MI_INIT_ERR_THREADMGR, bOk, errMsg);
|
|
MI::ModuleShutdown<CMICmnResources>(IDS_MI_INIT_ERR_RESOURCES, bOk, errMsg);
|
|
MI::ModuleShutdown<CMICmnLog>(IDS_MI_INIT_ERR_LOG, bOk, errMsg);
|
|
|
|
if (!bOk)
|
|
{
|
|
SetErrorDescriptionn(MIRSRC(IDS_MI_SHTDWN_ERR_LLDBDEBUGGER), errMsg.c_str());
|
|
}
|
|
|
|
return MIstatus::success;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Return the LLDB debugger instance created for this debug session.
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: lldb::SBDebugger & - LLDB debugger object reference.
|
|
// Throws: None.
|
|
//--
|
|
lldb::SBDebugger &
|
|
CMICmnLLDBDebugger::GetTheDebugger()
|
|
{
|
|
return m_lldbDebugger;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Return the LLDB listener instance created for this debug session.
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: lldb::SBListener & - LLDB listener object reference.
|
|
// Throws: None.
|
|
//--
|
|
lldb::SBListener &
|
|
CMICmnLLDBDebugger::GetTheListener()
|
|
{
|
|
return m_lldbListener;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Set the client driver that wants to use *this LLDB debugger. Call this function
|
|
// prior to Initialize().
|
|
// Type: Method.
|
|
// Args: vClientDriver - (R) A driver.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::SetDriver(const CMIDriverBase &vClientDriver)
|
|
{
|
|
m_pClientDriver = const_cast<CMIDriverBase *>(&vClientDriver);
|
|
|
|
return MIstatus::success;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Get the client driver that is use *this LLDB debugger.
|
|
// Type: Method.
|
|
// Args: vClientDriver - (R) A driver.
|
|
// Return: CMIDriverBase & - A driver instance.
|
|
// Throws: None.
|
|
//--
|
|
CMIDriverBase &
|
|
CMICmnLLDBDebugger::GetDriver() const
|
|
{
|
|
return *m_pClientDriver;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Wait until all events have been handled.
|
|
// This function works in pair with CMICmnLLDBDebugger::MonitorSBListenerEvents
|
|
// that handles events from queue. When all events were handled and queue is
|
|
// empty the MonitorSBListenerEvents notifies this function that it's ready to
|
|
// go on. To synchronize them the m_mutexEventQueue and
|
|
// m_conditionEventQueueEmpty are used.
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: None.
|
|
// Throws: None.
|
|
//--
|
|
void
|
|
CMICmnLLDBDebugger::WaitForHandleEvent()
|
|
{
|
|
std::unique_lock<std::mutex> lock(m_mutexEventQueue);
|
|
|
|
lldb::SBEvent event;
|
|
if (ThreadIsActive() && m_lldbListener.PeekAtNextEvent(event))
|
|
m_conditionEventQueueEmpty.wait(lock);
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Check if need to rebroadcast stop event. This function will return true if
|
|
// debugger is in synchronouse mode. In such case the
|
|
// CMICmnLLDBDebugger::RebroadcastStopEvent should be called to rebroadcast
|
|
// a new stop event (if any).
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: bool - True = Need to rebroadcast stop event, false = otherwise.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::CheckIfNeedToRebroadcastStopEvent()
|
|
{
|
|
CMICmnLLDBDebugSessionInfo &rSessionInfo(CMICmnLLDBDebugSessionInfo::Instance());
|
|
if (!rSessionInfo.GetDebugger().GetAsync())
|
|
{
|
|
const bool include_expression_stops = false;
|
|
m_nLastStopId = CMICmnLLDBDebugSessionInfo::Instance().GetProcess().GetStopID(include_expression_stops);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Rebroadcast stop event if needed. This function should be called only if the
|
|
// CMICmnLLDBDebugger::CheckIfNeedToRebroadcastStopEvent() returned true.
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: None.
|
|
// Throws: None.
|
|
//--
|
|
void
|
|
CMICmnLLDBDebugger::RebroadcastStopEvent()
|
|
{
|
|
lldb::SBProcess process = CMICmnLLDBDebugSessionInfo::Instance().GetProcess();
|
|
const bool include_expression_stops = false;
|
|
const uint32_t nStopId = process.GetStopID(include_expression_stops);
|
|
if (m_nLastStopId != nStopId)
|
|
{
|
|
lldb::SBEvent event = process.GetStopEventForStopID(nStopId);
|
|
process.GetBroadcaster().BroadcastEvent(event);
|
|
}
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Initialize the LLDB Debugger object.
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::InitSBDebugger()
|
|
{
|
|
m_lldbDebugger = lldb::SBDebugger::Create(false);
|
|
if (!m_lldbDebugger.IsValid())
|
|
{
|
|
SetErrorDescription(MIRSRC(IDS_LLDBDEBUGGER_ERR_INVALIDDEBUGGER));
|
|
return MIstatus::failure;
|
|
}
|
|
|
|
m_lldbDebugger.GetCommandInterpreter().SetPromptOnQuit(false);
|
|
|
|
return MIstatus::success;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Set the LLDB Debugger's std in, err and out streams. (Not implemented left
|
|
// here for reference. Was called in the CMICmnLLDBDebugger::Initialize() )
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::InitStdStreams()
|
|
{
|
|
// This is not required when operating the MI driver's code as it has its own
|
|
// streams. Setting the Stdin for the lldbDebugger especially on LINUX will cause
|
|
// another thread to run and partially consume stdin data meant for MI stdin handler
|
|
// m_lldbDebugger.SetErrorFileHandle( m_pClientDriver->GetStderr(), false );
|
|
// m_lldbDebugger.SetOutputFileHandle( m_pClientDriver->GetStdout(), false );
|
|
// m_lldbDebugger.SetInputFileHandle( m_pClientDriver->GetStdin(), false );
|
|
|
|
return MIstatus::success;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Set up the events from the SBDebugger's we would like to listen to.
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::InitSBListener()
|
|
{
|
|
m_lldbListener = m_lldbDebugger.GetListener();
|
|
if (!m_lldbListener.IsValid())
|
|
{
|
|
SetErrorDescription(MIRSRC(IDS_LLDBDEBUGGER_ERR_INVALIDLISTENER));
|
|
return MIstatus::failure;
|
|
}
|
|
|
|
const CMIUtilString strDbgId("CMICmnLLDBDebugger1");
|
|
MIuint eventMask = lldb::SBTarget::eBroadcastBitBreakpointChanged | lldb::SBTarget::eBroadcastBitModulesLoaded |
|
|
lldb::SBTarget::eBroadcastBitModulesUnloaded | lldb::SBTarget::eBroadcastBitWatchpointChanged |
|
|
lldb::SBTarget::eBroadcastBitSymbolsLoaded;
|
|
bool bOk = RegisterForEvent(strDbgId, CMIUtilString(lldb::SBTarget::GetBroadcasterClassName()), eventMask);
|
|
|
|
eventMask = lldb::SBThread::eBroadcastBitStackChanged;
|
|
bOk = bOk && RegisterForEvent(strDbgId, CMIUtilString(lldb::SBThread::GetBroadcasterClassName()), eventMask);
|
|
|
|
eventMask = lldb::SBProcess::eBroadcastBitStateChanged | lldb::SBProcess::eBroadcastBitInterrupt |
|
|
lldb::SBProcess::eBroadcastBitSTDOUT | lldb::SBProcess::eBroadcastBitSTDERR | lldb::SBProcess::eBroadcastBitProfileData;
|
|
bOk = bOk && RegisterForEvent(strDbgId, CMIUtilString(lldb::SBProcess::GetBroadcasterClassName()), eventMask);
|
|
|
|
eventMask = lldb::SBCommandInterpreter::eBroadcastBitQuitCommandReceived | lldb::SBCommandInterpreter::eBroadcastBitThreadShouldExit |
|
|
lldb::SBCommandInterpreter::eBroadcastBitAsynchronousOutputData |
|
|
lldb::SBCommandInterpreter::eBroadcastBitAsynchronousErrorData;
|
|
bOk = bOk && RegisterForEvent(strDbgId, m_lldbDebugger.GetCommandInterpreter().GetBroadcaster(), eventMask);
|
|
|
|
return bOk;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Register with the debugger, the SBListener, the type of events you are interested
|
|
// in. Others, like commands, may have already set the mask.
|
|
// Type: Method.
|
|
// Args: vClientName - (R) ID of the client who wants these events set.
|
|
// vBroadcasterClass - (R) The SBBroadcaster's class name.
|
|
// vEventMask - (R) The mask of events to listen for.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::RegisterForEvent(const CMIUtilString &vClientName, const CMIUtilString &vBroadcasterClass, const MIuint vEventMask)
|
|
{
|
|
MIuint existingMask = 0;
|
|
if (!BroadcasterGetMask(vBroadcasterClass, existingMask))
|
|
return MIstatus::failure;
|
|
|
|
if (!ClientSaveMask(vClientName, vBroadcasterClass, vEventMask))
|
|
return MIstatus::failure;
|
|
|
|
const char *pBroadCasterName = vBroadcasterClass.c_str();
|
|
MIuint eventMask = vEventMask;
|
|
eventMask += existingMask;
|
|
const MIuint result = m_lldbListener.StartListeningForEventClass(m_lldbDebugger, pBroadCasterName, eventMask);
|
|
if (result == 0)
|
|
{
|
|
SetErrorDescription(CMIUtilString::Format(MIRSRC(IDS_LLDBDEBUGGER_ERR_STARTLISTENER), pBroadCasterName));
|
|
return MIstatus::failure;
|
|
}
|
|
|
|
return BroadcasterSaveMask(vBroadcasterClass, eventMask);
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Register with the debugger, the SBListener, the type of events you are interested
|
|
// in. Others, like commands, may have already set the mask.
|
|
// Type: Method.
|
|
// Args: vClientName - (R) ID of the client who wants these events set.
|
|
// vBroadcaster - (R) An SBBroadcaster's derived class.
|
|
// vEventMask - (R) The mask of events to listen for.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::RegisterForEvent(const CMIUtilString &vClientName, const lldb::SBBroadcaster &vBroadcaster, const MIuint vEventMask)
|
|
{
|
|
const char *pBroadcasterName = vBroadcaster.GetName();
|
|
if (pBroadcasterName == nullptr)
|
|
{
|
|
SetErrorDescription(CMIUtilString::Format(MIRSRC(IDS_LLDBDEBUGGER_ERR_BROADCASTER_NAME), MIRSRC(IDS_WORD_INVALIDNULLPTR)));
|
|
return MIstatus::failure;
|
|
}
|
|
CMIUtilString broadcasterName(pBroadcasterName);
|
|
if (broadcasterName.length() == 0)
|
|
{
|
|
SetErrorDescription(CMIUtilString::Format(MIRSRC(IDS_LLDBDEBUGGER_ERR_BROADCASTER_NAME), MIRSRC(IDS_WORD_INVALIDEMPTY)));
|
|
return MIstatus::failure;
|
|
}
|
|
|
|
MIuint existingMask = 0;
|
|
if (!BroadcasterGetMask(broadcasterName, existingMask))
|
|
return MIstatus::failure;
|
|
|
|
if (!ClientSaveMask(vClientName, broadcasterName, vEventMask))
|
|
return MIstatus::failure;
|
|
|
|
MIuint eventMask = vEventMask;
|
|
eventMask += existingMask;
|
|
const MIuint result = m_lldbListener.StartListeningForEvents(vBroadcaster, eventMask);
|
|
if (result == 0)
|
|
{
|
|
SetErrorDescription(CMIUtilString::Format(MIRSRC(IDS_LLDBDEBUGGER_ERR_STARTLISTENER), pBroadcasterName));
|
|
return MIstatus::failure;
|
|
}
|
|
|
|
return BroadcasterSaveMask(broadcasterName, eventMask);
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Unregister with the debugger, the SBListener, the type of events you are no
|
|
// longer interested in. Others, like commands, may still remain interested so
|
|
// an event may not necessarily be stopped.
|
|
// Type: Method.
|
|
// Args: vClientName - (R) ID of the client who no longer requires these events.
|
|
// vBroadcasterClass - (R) The SBBroadcaster's class name.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::UnregisterForEvent(const CMIUtilString &vClientName, const CMIUtilString &vBroadcasterClass)
|
|
{
|
|
MIuint clientsEventMask = 0;
|
|
if (!ClientGetTheirMask(vClientName, vBroadcasterClass, clientsEventMask))
|
|
return MIstatus::failure;
|
|
if (!ClientRemoveTheirMask(vClientName, vBroadcasterClass))
|
|
return MIstatus::failure;
|
|
|
|
const MIuint otherClientsEventMask = ClientGetMaskForAllClients(vBroadcasterClass);
|
|
MIuint newEventMask = 0;
|
|
for (MIuint i = 0; i < 32; i++)
|
|
{
|
|
const MIuint bit = 1 << i;
|
|
const MIuint clientBit = bit & clientsEventMask;
|
|
const MIuint othersBit = bit & otherClientsEventMask;
|
|
if ((clientBit != 0) && (othersBit == 0))
|
|
{
|
|
newEventMask += clientBit;
|
|
}
|
|
}
|
|
|
|
const char *pBroadCasterName = vBroadcasterClass.c_str();
|
|
if (!m_lldbListener.StopListeningForEventClass(m_lldbDebugger, pBroadCasterName, newEventMask))
|
|
{
|
|
SetErrorDescription(CMIUtilString::Format(MIRSRC(IDS_LLDBDEBUGGER_ERR_STOPLISTENER), vClientName.c_str(), pBroadCasterName));
|
|
return MIstatus::failure;
|
|
}
|
|
|
|
return BroadcasterSaveMask(vBroadcasterClass, otherClientsEventMask);
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Given the SBBroadcaster class name retrieve it's current event mask.
|
|
// Type: Method.
|
|
// Args: vBroadcasterClass - (R) The SBBroadcaster's class name.
|
|
// vEventMask - (W) The mask of events to listen for.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::BroadcasterGetMask(const CMIUtilString &vBroadcasterClass, MIuint &vwEventMask) const
|
|
{
|
|
vwEventMask = 0;
|
|
|
|
if (vBroadcasterClass.empty())
|
|
{
|
|
SetErrorDescription(CMIUtilString::Format(MIRSRC(IDS_LLDBDEBUGGER_ERR_INVALIDBROADCASTER), vBroadcasterClass.c_str()));
|
|
return MIstatus::failure;
|
|
}
|
|
|
|
const MapBroadcastClassNameToEventMask_t::const_iterator it = m_mapBroadcastClassNameToEventMask.find(vBroadcasterClass);
|
|
if (it != m_mapBroadcastClassNameToEventMask.end())
|
|
{
|
|
vwEventMask = (*it).second;
|
|
}
|
|
|
|
return MIstatus::success;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Remove the event mask for the specified SBBroadcaster class name.
|
|
// Type: Method.
|
|
// Args: vBroadcasterClass - (R) The SBBroadcaster's class name.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::BroadcasterRemoveMask(const CMIUtilString &vBroadcasterClass)
|
|
{
|
|
MapBroadcastClassNameToEventMask_t::const_iterator it = m_mapBroadcastClassNameToEventMask.find(vBroadcasterClass);
|
|
if (it != m_mapBroadcastClassNameToEventMask.end())
|
|
{
|
|
m_mapBroadcastClassNameToEventMask.erase(it);
|
|
}
|
|
|
|
return MIstatus::success;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Given the SBBroadcaster class name save it's current event mask.
|
|
// Type: Method.
|
|
// Args: vBroadcasterClass - (R) The SBBroadcaster's class name.
|
|
// vEventMask - (R) The mask of events to listen for.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::BroadcasterSaveMask(const CMIUtilString &vBroadcasterClass, const MIuint vEventMask)
|
|
{
|
|
if (vBroadcasterClass.empty())
|
|
{
|
|
SetErrorDescription(CMIUtilString::Format(MIRSRC(IDS_LLDBDEBUGGER_ERR_INVALIDBROADCASTER), vBroadcasterClass.c_str()));
|
|
return MIstatus::failure;
|
|
}
|
|
|
|
BroadcasterRemoveMask(vBroadcasterClass);
|
|
MapPairBroadcastClassNameToEventMask_t pr(vBroadcasterClass, vEventMask);
|
|
m_mapBroadcastClassNameToEventMask.insert(pr);
|
|
|
|
return MIstatus::success;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Iterate all the clients who have registered event masks against particular
|
|
// SBBroadcasters and build up the mask that is for all of them.
|
|
// Type: Method.
|
|
// Args: vBroadcasterClass - (R) The broadcaster to retrieve the mask for.
|
|
// Return: MIuint - Event mask.
|
|
// Throws: None.
|
|
//--
|
|
MIuint
|
|
CMICmnLLDBDebugger::ClientGetMaskForAllClients(const CMIUtilString &vBroadcasterClass) const
|
|
{
|
|
MIuint mask = 0;
|
|
MapIdToEventMask_t::const_iterator it = m_mapIdToEventMask.begin();
|
|
while (it != m_mapIdToEventMask.end())
|
|
{
|
|
const CMIUtilString &rId((*it).first);
|
|
if (rId.find(vBroadcasterClass) != std::string::npos)
|
|
{
|
|
const MIuint clientsMask = (*it).second;
|
|
mask |= clientsMask;
|
|
}
|
|
|
|
// Next
|
|
++it;
|
|
}
|
|
|
|
return mask;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Given the client save its particular event requirements.
|
|
// Type: Method.
|
|
// Args: vClientName - (R) The Client's unique ID.
|
|
// vBroadcasterClass - (R) The SBBroadcaster's class name targeted for the events.
|
|
// vEventMask - (R) The mask of events to listen for.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::ClientSaveMask(const CMIUtilString &vClientName, const CMIUtilString &vBroadcasterClass, const MIuint vEventMask)
|
|
{
|
|
if (vClientName.empty())
|
|
{
|
|
SetErrorDescription(CMIUtilString::Format(MIRSRC(IDS_LLDBDEBUGGER_ERR_INVALIDCLIENTNAME), vClientName.c_str()));
|
|
return MIstatus::failure;
|
|
}
|
|
|
|
CMIUtilString strId(vBroadcasterClass);
|
|
strId += vClientName;
|
|
|
|
ClientRemoveTheirMask(vClientName, vBroadcasterClass);
|
|
MapPairIdToEventMask_t pr(strId, vEventMask);
|
|
m_mapIdToEventMask.insert(pr);
|
|
|
|
return MIstatus::success;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Given the client remove it's particular event requirements.
|
|
// Type: Method.
|
|
// Args: vClientName - (R) The Client's unique ID.
|
|
// vBroadcasterClass - (R) The SBBroadcaster's class name.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::ClientRemoveTheirMask(const CMIUtilString &vClientName, const CMIUtilString &vBroadcasterClass)
|
|
{
|
|
if (vClientName.empty())
|
|
{
|
|
SetErrorDescription(CMIUtilString::Format(MIRSRC(IDS_LLDBDEBUGGER_ERR_INVALIDCLIENTNAME), vClientName.c_str()));
|
|
return MIstatus::failure;
|
|
}
|
|
|
|
CMIUtilString strId(vBroadcasterClass);
|
|
strId += vClientName;
|
|
|
|
const MapIdToEventMask_t::const_iterator it = m_mapIdToEventMask.find(strId);
|
|
if (it != m_mapIdToEventMask.end())
|
|
{
|
|
m_mapIdToEventMask.erase(it);
|
|
}
|
|
|
|
return MIstatus::success;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Retrieve the client's event mask used for on a particular SBBroadcaster.
|
|
// Type: Method.
|
|
// Args: vClientName - (R) The Client's unique ID.
|
|
// vBroadcasterClass - (R) The SBBroadcaster's class name.
|
|
// vwEventMask - (W) The client's mask.
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::ClientGetTheirMask(const CMIUtilString &vClientName, const CMIUtilString &vBroadcasterClass, MIuint &vwEventMask)
|
|
{
|
|
vwEventMask = 0;
|
|
|
|
if (vClientName.empty())
|
|
{
|
|
SetErrorDescription(CMIUtilString::Format(MIRSRC(IDS_LLDBDEBUGGER_ERR_INVALIDCLIENTNAME), vClientName.c_str()));
|
|
return MIstatus::failure;
|
|
}
|
|
|
|
const CMIUtilString strId(vBroadcasterClass + vClientName);
|
|
const MapIdToEventMask_t::const_iterator it = m_mapIdToEventMask.find(strId);
|
|
if (it != m_mapIdToEventMask.end())
|
|
{
|
|
vwEventMask = (*it).second;
|
|
}
|
|
|
|
SetErrorDescription(CMIUtilString::Format(MIRSRC(IDS_LLDBDEBUGGER_ERR_CLIENTNOTREGISTERED), vClientName.c_str()));
|
|
|
|
return MIstatus::failure;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Momentarily wait for an events being broadcast and inspect those that do
|
|
// come this way. Check if the target should exit event if so start shutting
|
|
// down this thread and the application. Any other events pass on to the
|
|
// Out-of-band handler to further determine what kind of event arrived.
|
|
// This function runs in the thread "MI debugger event".
|
|
// Type: Method.
|
|
// Args: vrbIsAlive - (W) False = yes exit event monitoring thread, true = continue.
|
|
// Return: MIstatus::success - Functional succeeded.
|
|
// MIstatus::failure - Functional failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::MonitorSBListenerEvents(bool &vrbIsAlive)
|
|
{
|
|
vrbIsAlive = true;
|
|
|
|
// Lock the mutex of event queue
|
|
// Note that it should be locked while we are in CMICmnLLDBDebugger::MonitorSBListenerEvents to
|
|
// avoid a race condition with CMICmnLLDBDebugger::WaitForHandleEvent
|
|
std::unique_lock<std::mutex> lock(m_mutexEventQueue);
|
|
|
|
lldb::SBEvent event;
|
|
const bool bGotEvent = m_lldbListener.GetNextEvent(event);
|
|
if (!bGotEvent)
|
|
{
|
|
// Notify that we are finished and unlock the mutex of event queue before sleeping
|
|
m_conditionEventQueueEmpty.notify_one();
|
|
lock.unlock();
|
|
|
|
// Wait a bit to reduce CPU load
|
|
const std::chrono::milliseconds time(1);
|
|
std::this_thread::sleep_for(time);
|
|
return MIstatus::success;
|
|
}
|
|
assert(event.IsValid());
|
|
assert(event.GetBroadcaster().IsValid());
|
|
|
|
// Debugging
|
|
m_pLog->WriteLog(CMIUtilString::Format("##### An event occurred: %s", event.GetBroadcasterClass()));
|
|
|
|
bool bHandledEvent = false;
|
|
bool bOk = false;
|
|
{
|
|
// Lock Mutex before handling events so that we don't disturb a running cmd
|
|
CMIUtilThreadLock lock(CMICmnLLDBDebugSessionInfo::Instance().GetSessionMutex());
|
|
bOk = CMICmnLLDBDebuggerHandleEvents::Instance().HandleEvent(event, bHandledEvent);
|
|
}
|
|
|
|
if (!bHandledEvent)
|
|
{
|
|
const CMIUtilString msg(CMIUtilString::Format(MIRSRC(IDS_LLDBDEBUGGER_WRN_UNKNOWN_EVENT), event.GetBroadcasterClass()));
|
|
m_pLog->WriteLog(msg);
|
|
}
|
|
|
|
if (!bOk)
|
|
m_pLog->WriteLog(CMICmnLLDBDebuggerHandleEvents::Instance().GetErrorDescription());
|
|
|
|
return MIstatus::success;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: The main worker method for this thread.
|
|
// Type: Method.
|
|
// Args: vrbIsAlive - (W) True = *this thread is working, false = thread has exited.
|
|
// Return: MIstatus::success - Functional succeeded.
|
|
// MIstatus::failure - Functional failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::ThreadRun(bool &vrbIsAlive)
|
|
{
|
|
return MonitorSBListenerEvents(vrbIsAlive);
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Let this thread clean up after itself.
|
|
// Type: Method.
|
|
// Args:
|
|
// Return: MIstatus::success - Functionality succeeded.
|
|
// MIstatus::failure - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::ThreadFinish()
|
|
{
|
|
return MIstatus::success;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Retrieve *this thread object's name.
|
|
// Type: Overridden.
|
|
// Args: None.
|
|
// Return: CMIUtilString & - Text.
|
|
// Throws: None.
|
|
//--
|
|
const CMIUtilString &
|
|
CMICmnLLDBDebugger::ThreadGetName() const
|
|
{
|
|
return m_constStrThisThreadId;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Loads lldb-mi formatters
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: true - Functionality succeeded.
|
|
// false - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::LoadMIFormatters(lldb::SBTypeCategory miCategory)
|
|
{
|
|
if (!MI_add_summary(miCategory, "char", MI_char_summary_provider,
|
|
lldb::eTypeOptionHideValue | lldb::eTypeOptionSkipPointers))
|
|
return false;
|
|
|
|
if (!MI_add_summary(miCategory, "unsigned char", MI_char_summary_provider,
|
|
lldb::eTypeOptionHideValue | lldb::eTypeOptionSkipPointers))
|
|
return false;
|
|
|
|
if (!MI_add_summary(miCategory, "signed char", MI_char_summary_provider,
|
|
lldb::eTypeOptionHideValue | lldb::eTypeOptionSkipPointers))
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
//++ ------------------------------------------------------------------------------------
|
|
// Details: Registers lldb-mi custom summary providers
|
|
// Type: Method.
|
|
// Args: None.
|
|
// Return: true - Functionality succeeded.
|
|
// false - Functionality failed.
|
|
// Throws: None.
|
|
//--
|
|
bool
|
|
CMICmnLLDBDebugger::RegisterMISummaryProviders()
|
|
{
|
|
static const char* miCategoryName = "lldb-mi";
|
|
lldb::SBTypeCategory miCategory = m_lldbDebugger.CreateCategory(miCategoryName);
|
|
if (!miCategory.IsValid())
|
|
return false;
|
|
|
|
if (!LoadMIFormatters(miCategory))
|
|
{
|
|
m_lldbDebugger.DeleteCategory(miCategoryName);
|
|
return false;
|
|
}
|
|
miCategory.SetEnabled(true);
|
|
return true;
|
|
}
|