2020-09-30 16:53:55 +02:00

2451 lines
67 KiB
C++

//----------------------------------------------------------------------------
//
// Engine interface code.
//
// Copyright (C) Microsoft Corporation, 1999-2002.
//
//----------------------------------------------------------------------------
#include "precomp.hxx"
#pragma hdrstop
#include <dbghelp.h>
#define DBG_CALLBACK 0
#define DBG_INPUT 0
// Windows that change behavior depending on the execution status.
#define UPDATE_EXEC_WINDOWS \
((1 << CPU_WINDOW) | \
(1 << DISASM_WINDOW) | \
(1 << CMD_WINDOW) | \
(1 << LOCALS_WINDOW) | \
(1 << WATCH_WINDOW) | \
(1 << MEM_WINDOW))
// Windows that use symbol information.
#define UPDATE_SYM_WINDOWS \
((1 << DOC_WINDOW) | \
(1 << WATCH_WINDOW) | \
(1 << LOCALS_WINDOW) | \
(1 << DISASM_WINDOW) | \
(1 << QUICKW_WINDOW) | \
(1 << CALLS_WINDOW) | \
(1 << EVENT_BIT) | \
(1 << BP_BIT))
// Symbol options that cause visible changes and
// therefore require a refresh. Note that this
// doesn't include options that would cause a visible
// change only after symbol reload as things will
// get refreshed when the load notifications come in.
#define REFRESH_SYMOPT \
(~(SYMOPT_CASE_INSENSITIVE | SYMOPT_DEFERRED_LOADS | SYMOPT_LOAD_LINES | \
SYMOPT_LOAD_ANYTHING | SYMOPT_IGNORE_CVREC | \
SYMOPT_NO_UNQUALIFIED_LOADS | SYMOPT_EXACT_SYMBOLS))
//
// Session initialization parameters.
//
// Turn on verbose output or not.
BOOL g_Verbose;
// Dump files to open.
PTSTR g_DumpFiles[MAX_DUMP_FILES];
ULONG g_NumDumpFiles;
PTSTR g_DumpInfoFiles[MAX_DUMP_FILES];
ULONG g_DumpInfoTypes[MAX_DUMP_FILES];
ULONG g_NumDumpInfoFiles;
// Process server to use.
PSTR g_ProcessServer;
// Full command line with exe name.
PSTR g_DebugCommandLine;
PSTR g_ProcessStartDir;
// Process creation flags.
ULONG g_DebugCreateFlags = DEBUG_ONLY_THIS_PROCESS;
// Process ID to attach to or zero.
ULONG g_PidToDebug;
// Process name to attach to or NULL.
PSTR g_ProcNameToDebug;
BOOL g_DetachOnExit;
ULONG g_AttachProcessFlags = DEBUG_ATTACH_DEFAULT;
// Kernel connection options.
ULONG g_AttachKernelFlags = DEBUG_ATTACH_KERNEL_CONNECTION;
PSTR g_KernelConnectOptions;
// Remoting options.
BOOL g_RemoteClient;
ULONG g_HistoryLines = 10000;
//
// Debug engine interfaces for the engine thread.
//
IDebugClient *g_pDbgClient;
IDebugClient2 *g_pDbgClient2;
IDebugControl *g_pDbgControl;
IDebugSymbols *g_pDbgSymbols;
IDebugSymbolGroup *g_pDbgWatchSymbolGroup;
IDebugSymbolGroup *g_pDbgLocalSymbolGroup = NULL;
IDebugRegisters *g_pDbgRegisters;
IDebugDataSpaces *g_pDbgData;
IDebugSystemObjects *g_pDbgSystem;
IDebugSystemObjects3 *g_pDbgSystem3;
//
// Debug engine interfaces for the UI thread.
//
IDebugClient *g_pUiClient;
IDebugControl *g_pUiControl;
IDebugControl3 *g_pUiControl3;
IDebugSymbols *g_pUiSymbols;
IDebugSymbols2 *g_pUiSymbols2;
IDebugSystemObjects *g_pUiSystem;
//
// Debug engine interfaces for private output capture.
//
IDebugClient *g_pOutCapClient;
IDebugControl *g_pOutCapControl;
IDebugSymbols *g_pOutCapSymbols;
//
// Debug engine interfaces for local source file lookup.
//
IDebugClient *g_pLocClient;
IDebugControl *g_pLocControl;
IDebugSymbols *g_pLocSymbols;
IDebugClient *g_pUiLocClient;
IDebugControl *g_pUiLocControl;
IDebugSymbols *g_pUiLocSymbols;
ULONG g_ActualProcType = IMAGE_FILE_MACHINE_UNKNOWN;
char g_ActualProcAbbrevName[32];
ULONG g_CommandSequence;
ULONG g_TargetClass = DEBUG_CLASS_UNINITIALIZED;
ULONG g_TargetClassQual;
BOOL g_Ptr64;
ULONG g_ExecStatus = DEBUG_STATUS_NO_DEBUGGEE;
ULONG g_EngOptModified;
ULONG g_EngineThreadId;
HANDLE g_EngineThread;
PSTR g_InitialCommand;
char g_PromptText[32];
BOOL g_WaitingForEvent;
ULONG g_NumberRadix;
BOOL g_CodeLevelLocked;
BOOL g_IgnoreFilterChange;
BOOL g_IgnoreCodeLevelChange;
BOOL g_IgnoreThreadChange;
ULONG g_LastProcessExitCode;
ULONG g_SymOptions;
ULONG g_TypeOptions;
BOOL g_InputStarted;
BOOL g_Invisible;
BOOL g_HoldWaitOutput;
BOOL g_ScopeChanged;
enum
{
ENDING_NONE,
ENDING_RESTART,
ENDING_STOP,
ENDING_EXIT
};
ULONG g_EndingSession = ENDING_NONE;
void SetLocalScope(PDEBUG_STACK_FRAME);
BOOL g_SessionActive;
void SessionActive(void);
void SessionInactive(void);
StateBuffer g_UiCommandBuffer(MAX_COMMAND_LEN);
StateBuffer g_UiOutputBuffer(128 * 1024);
//----------------------------------------------------------------------------
//
// Default output callbacks implementation, provides IUnknown for
// static classes.
//
//----------------------------------------------------------------------------
STDMETHODIMP
DefOutputCallbacks::QueryInterface(
THIS_
IN REFIID InterfaceId,
OUT PVOID* Interface
)
{
*Interface = NULL;
if (IsEqualIID(InterfaceId, IID_IUnknown) ||
IsEqualIID(InterfaceId, IID_IDebugOutputCallbacks))
{
*Interface = (IDebugOutputCallbacks *)this;
AddRef();
return S_OK;
}
else
{
return E_NOINTERFACE;
}
}
STDMETHODIMP_(ULONG)
DefOutputCallbacks::AddRef(
THIS
)
{
// This class is designed to be static so
// there's no true refcount.
return 1;
}
STDMETHODIMP_(ULONG)
DefOutputCallbacks::Release(
THIS
)
{
// This class is designed to be static so
// there's no true refcount.
return 0;
}
//----------------------------------------------------------------------------
//
// Command window output callbacks.
//
//----------------------------------------------------------------------------
class OutputCallbacks : public DefOutputCallbacks
{
public:
// IDebugOutputCallbacks.
STDMETHOD(Output)(
THIS_
IN ULONG Mask,
IN PCSTR Text
);
};
STDMETHODIMP
OutputCallbacks::Output(
THIS_
IN ULONG Mask,
IN PCSTR Text
)
{
LockUiBuffer(&g_UiOutputBuffer);
HRESULT Status;
ULONG Len;
PSTR DataStart;
Len = sizeof(Mask) + strlen(Text) + 1;
if ((DataStart = (PSTR)g_UiOutputBuffer.AddData(Len)) != NULL)
{
*(ULONG UNALIGNED *)DataStart = Mask;
DataStart += sizeof(Mask);
strcpy(DataStart, Text);
if (!g_HoldWaitOutput || !g_WaitingForEvent)
{
UpdateUi();
}
Status = S_OK;
}
else
{
Status = E_OUTOFMEMORY;
}
UnlockUiBuffer(&g_UiOutputBuffer);
return Status;
}
OutputCallbacks g_OutputCb;
//----------------------------------------------------------------------------
//
// Input callbacks.
//
//----------------------------------------------------------------------------
class InputCallbacks :
public IDebugInputCallbacks
{
public:
// IUnknown.
STDMETHOD(QueryInterface)(
THIS_
IN REFIID InterfaceId,
OUT PVOID* Interface
);
STDMETHOD_(ULONG, AddRef)(
THIS
);
STDMETHOD_(ULONG, Release)(
THIS
);
// IDebugInputCallbacks.
STDMETHOD(StartInput)(
THIS_
IN ULONG BufferSize
);
STDMETHOD(EndInput)(
THIS
);
};
STDMETHODIMP
InputCallbacks::QueryInterface(
THIS_
IN REFIID InterfaceId,
OUT PVOID* Interface
)
{
*Interface = NULL;
if (IsEqualIID(InterfaceId, IID_IUnknown) ||
IsEqualIID(InterfaceId, IID_IDebugInputCallbacks))
{
*Interface = (IDebugInputCallbacks *)this;
AddRef();
return S_OK;
}
else
{
return E_NOINTERFACE;
}
}
STDMETHODIMP_(ULONG)
InputCallbacks::AddRef(
THIS
)
{
// This class is designed to be static so
// there's no true refcount.
return 1;
}
STDMETHODIMP_(ULONG)
InputCallbacks::Release(
THIS
)
{
// This class is designed to be static so
// there's no true refcount.
return 0;
}
STDMETHODIMP
InputCallbacks::StartInput(
THIS_
IN ULONG BufferSize
)
{
HRESULT Status;
//
// Pull the first command input command out
// of the UI's command buffer and use it as input.
//
LockUiBuffer(&g_UiCommandBuffer);
UiCommandData* CmdData =
(UiCommandData*)g_UiCommandBuffer.GetDataBuffer();
UiCommandData* CmdEnd = (UiCommandData*)
((PBYTE)g_UiCommandBuffer.GetDataBuffer() +
g_UiCommandBuffer.GetDataLen());
while (CmdData < CmdEnd)
{
if (CmdData->Cmd == UIC_CMD_INPUT)
{
break;
}
CmdData = (UiCommandData*)((PBYTE)CmdData + CmdData->Len);
}
#if DBG_INPUT
DebugPrint("StartInput available %d\n", CmdData < CmdEnd);
#endif
if (CmdData < CmdEnd)
{
g_OutputCb.Output(DEBUG_OUTPUT_NORMAL, (PSTR)(CmdData + 1));
g_OutputCb.Output(DEBUG_OUTPUT_NORMAL, "\n");
Status = g_pUiControl->ReturnInput((PSTR)(CmdData + 1));
#if DBG_INPUT
DebugPrint(" ReturnInput status %X\n", Status);
#endif
g_UiCommandBuffer.
RemoveMiddle((ULONG)((PBYTE)CmdData -
(PBYTE)g_UiCommandBuffer.GetDataBuffer()),
CmdData->Len);
}
else
{
g_InputStarted = TRUE;
// Didn't find any input waiting.
// Let the command window know that input is needed.
UpdateBufferWindows(1 << CMD_WINDOW, UPDATE_INPUT_REQUIRED);
}
UnlockUiBuffer(&g_UiCommandBuffer);
return S_OK;
}
STDMETHODIMP
InputCallbacks::EndInput(
THIS
)
{
LockUiBuffer(&g_UiCommandBuffer);
#if DBG_INPUT
DebugPrint("EndInput started %d\n", g_InputStarted);
#endif
g_InputStarted = FALSE;
// Reset the command window's state to what it was.
UpdateBufferWindows(1 << CMD_WINDOW, UPDATE_EXEC);
UnlockUiBuffer(&g_UiCommandBuffer);
return S_OK;
}
InputCallbacks g_InputCb;
//----------------------------------------------------------------------------
//
// Event callbacks.
//
//----------------------------------------------------------------------------
// This is safe to do from the engine thread as
// it just sets a flag.
#define DIRTY_WORKSPACE(Flags) \
if (!g_RemoteClient && \
g_EndingSession == ENDING_NONE && !g_Invisible && g_Workspace != NULL) \
{ \
g_Workspace->AddDirty(Flags); \
}
STDMETHODIMP_(ULONG)
EventCallbacks::AddRef(
THIS
)
{
// This class is designed to be static so
// there's no true refcount.
return 1;
}
STDMETHODIMP_(ULONG)
EventCallbacks::Release(
THIS
)
{
// This class is designed to be static so
// there's no true refcount.
return 0;
}
STDMETHODIMP
EventCallbacks::GetInterestMask(
THIS_
OUT PULONG Mask
)
{
*Mask =
DEBUG_EVENT_CREATE_THREAD |
DEBUG_EVENT_EXIT_THREAD |
DEBUG_EVENT_CREATE_PROCESS |
DEBUG_EVENT_EXIT_PROCESS |
DEBUG_EVENT_SESSION_STATUS |
DEBUG_EVENT_CHANGE_DEBUGGEE_STATE |
DEBUG_EVENT_CHANGE_ENGINE_STATE |
DEBUG_EVENT_CHANGE_SYMBOL_STATE;
return S_OK;
}
STDMETHODIMP
EventCallbacks::CreateThread(
THIS_
IN ULONG64 Handle,
IN ULONG64 DataOffset,
IN ULONG64 StartOffset
)
{
ULONG InvFlags =
(1 << PROCESS_THREAD_WINDOW);
#if DBG_CALLBACK
DebugPrint(" CT\n");
#endif
// There's no need to update buffers when we're throwing
// everything away while shutting down a session.
if (g_EndingSession == ENDING_NONE && InvFlags)
{
InvalidateStateBuffers(InvFlags);
UpdateEngine();
}
return DEBUG_STATUS_NO_CHANGE;
}
STDMETHODIMP
EventCallbacks::ExitThread(
THIS_
IN ULONG ExitCode
)
{
ULONG InvFlags =
(1 << PROCESS_THREAD_WINDOW);
#if DBG_CALLBACK
DebugPrint(" ET\n");
#endif
// There's no need to update buffers when we're throwing
// everything away while shutting down a session.
if (g_EndingSession == ENDING_NONE && InvFlags)
{
InvalidateStateBuffers(InvFlags);
UpdateEngine();
}
return DEBUG_STATUS_NO_CHANGE;
}
STDMETHODIMP
EventCallbacks::CreateProcess(
THIS_
IN ULONG64 ImageFileHandle,
IN ULONG64 Handle,
IN ULONG64 BaseOffset,
IN ULONG ModuleSize,
IN PCSTR ModuleName,
IN PCSTR ImageName,
IN ULONG CheckSum,
IN ULONG TimeDateStamp,
IN ULONG64 InitialThreadHandle,
IN ULONG64 ThreadDataOffset,
IN ULONG64 StartOffset
)
{
ULONG InvFlags =
(1 << PROCESS_THREAD_WINDOW);
#if DBG_CALLBACK
DebugPrint("CPR\n");
#endif
// Use this opportunity to get initial insertion
// of any workspace breakpoints and process other workspace
// commands which may be queued.
ProcessEngineCommands(TRUE);
// There's no need to update buffers when we're throwing
// everything away while shutting down a session.
if (g_EndingSession == ENDING_NONE && InvFlags)
{
InvalidateStateBuffers(InvFlags);
UpdateEngine();
}
return DEBUG_STATUS_NO_CHANGE;
}
STDMETHODIMP
EventCallbacks::ExitProcess(
THIS_
IN ULONG ExitCode
)
{
ULONG InvFlags =
(1 << PROCESS_THREAD_WINDOW);
#if DBG_CALLBACK
DebugPrint("EPR\n");
#endif
// There's no need to update buffers when we're throwing
// everything away while shutting down a session.
if (g_EndingSession == ENDING_NONE && InvFlags)
{
InvalidateStateBuffers(InvFlags);
UpdateEngine();
}
g_LastProcessExitCode = ExitCode;
return DEBUG_STATUS_NO_CHANGE;
}
STDMETHODIMP
EventCallbacks::SessionStatus(
THIS_
IN ULONG Status
)
{
#if DBG_CALLBACK
DebugPrint(" SS %X\n", Status);
#endif
switch(Status)
{
case DEBUG_SESSION_ACTIVE:
SessionActive();
break;
case DEBUG_SESSION_END_SESSION_ACTIVE_TERMINATE:
case DEBUG_SESSION_END_SESSION_ACTIVE_DETACH:
case DEBUG_SESSION_END:
case DEBUG_SESSION_REBOOT:
case DEBUG_SESSION_HIBERNATE:
SessionInactive();
break;
}
return DEBUG_STATUS_NO_CHANGE;
}
STDMETHODIMP
EventCallbacks::ChangeDebuggeeState(
THIS_
IN ULONG Flags,
IN ULONG64 Argument
)
{
ULONG InvFlags =
(1 << WATCH_WINDOW) |
(1 << LOCALS_WINDOW) |
(1 << DISASM_WINDOW) |
(1 << QUICKW_WINDOW) |
(1 << CALLS_WINDOW);
// Invalidate everything that changed.
if (Flags & DEBUG_CDS_REGISTERS)
{
InvFlags |= (1 << EVENT_BIT) | (1 << CPU_WINDOW);
}
if (Flags & DEBUG_CDS_DATA)
{
InvFlags |=
(1 << MEM_WINDOW);
}
#if DBG_CALLBACK
DebugPrint("CDS %X, arg %I64X, inv %X\n", Flags, Argument, InvFlags);
#endif
// There's no need to update buffers when we're throwing
// everything away while shutting down a session.
if (g_EndingSession == ENDING_NONE)
{
InvalidateStateBuffers(InvFlags);
}
if (InvFlags != 0)
{
UpdateEngine();
}
return S_OK;
}
STDMETHODIMP
EventCallbacks::ChangeEngineState(
THIS_
IN ULONG Flags,
IN ULONG64 Argument
)
{
ULONG InvFlags = 0;
// If the current thread changed we need to get
// new context information for the thread.
if ((Flags & DEBUG_CES_CURRENT_THREAD) &&
!g_IgnoreThreadChange)
{
InvFlags |=
(1 << LOCALS_WINDOW) |
(1 << CPU_WINDOW) |
(1 << DISASM_WINDOW) |
(1 << CALLS_WINDOW) |
(1 << PROCESS_THREAD_WINDOW) |
(1 << EVENT_BIT) |
(1 << BP_BIT);
}
// If the effective processor changed we need to update
// anything related to processor information.
if (Flags & DEBUG_CES_EFFECTIVE_PROCESSOR)
{
InvFlags |=
(1 << CPU_WINDOW) |
(1 << DISASM_WINDOW) |
(1 << CALLS_WINDOW) |
(1 << BP_BIT);
}
// If breakpoints changed we need to update the breakpoint cache.
if (Flags & DEBUG_CES_BREAKPOINTS)
{
InvFlags |= (1 << BP_BIT);
// If it's a bulk edit it's coming from a thread or process exit
// or from a session shutdown rather than a user operation.
// We only want to remember user-driven changes in the workspace.
if (Argument != DEBUG_ANY_ID)
{
InvFlags |= (1 << BP_CMDS_BIT);
DIRTY_WORKSPACE(WSPF_DIRTY_BREAKPOINTS);
}
}
// If the code level changed we need to update the toolbar.
if (Flags & DEBUG_CES_CODE_LEVEL)
{
InvFlags |= (1 << BP_BIT);
// If this isn't a notification due to a change
// from windbg itself the user must have changed
// things via a command. If the user does
// change things from the command window lock
// the code level so that it isn't overridden
// automatically.
if (!g_Invisible && !g_IgnoreCodeLevelChange)
{
g_CodeLevelLocked = TRUE;
PostMessage(g_hwndFrame, WU_UPDATE,
UPDATE_BUFFER, (ULONG)Argument);
}
else
{
// Setting the source mode from the GUI enables
// the source setting to float along with whether
// the GUI can display source code or not.
g_CodeLevelLocked = FALSE;
}
}
if (Flags & DEBUG_CES_EXECUTION_STATUS)
{
// If this notification came from a wait completing
// we want to wake up things thread so that new
// commands can be processed. If it came from inside
// a wait we don't want to wake up as the engine
// may go back to running at any time.
if ((Argument & DEBUG_STATUS_INSIDE_WAIT) == 0 &&
(ULONG)Argument != g_ExecStatus)
{
g_ExecStatus = (ULONG)Argument;
UpdateBufferWindows(UPDATE_EXEC_WINDOWS, UPDATE_EXEC);
if (InvFlags == 0)
{
// Force the loop waiting in DispatchCallbacks to go around.
UpdateEngine();
}
}
}
// If the log file changed we need to update the workspace.
if (Flags & DEBUG_CES_LOG_FILE)
{
DIRTY_WORKSPACE(WSPF_DIRTY_LOG_FILE);
}
// If event filters changed we need to update the filter cache.
if ((Flags & DEBUG_CES_EVENT_FILTERS) &&
!g_IgnoreFilterChange)
{
InvFlags |= (1 << FILTER_BIT);
DIRTY_WORKSPACE(WSPF_DIRTY_FILTERS);
}
if (Flags & DEBUG_CES_RADIX)
{
g_NumberRadix = (ULONG)Argument;
InvFlags |=
(1 << WATCH_WINDOW) |
(1 << LOCALS_WINDOW) |
(1 << CPU_WINDOW);
}
if (Flags & DEBUG_CES_SYSTEMS)
{
InvFlags |=
(1 << PROCESS_THREAD_WINDOW);
}
if (Flags & DEBUG_CES_ASSEMBLY_OPTIONS)
{
InvFlags |=
(1 << DISASM_WINDOW);
if (g_Workspace != NULL)
{
g_Workspace->SetUlong(WSP_GLOBAL_ASSEMBLY_OPTIONS,
(ULONG)Argument);
}
}
if (Flags & DEBUG_CES_EXPRESSION_SYNTAX)
{
InvFlags |=
(1 << LOCALS_WINDOW) |
(1 << WATCH_WINDOW);
if (g_Workspace != NULL)
{
g_Workspace->SetUlong(WSP_GLOBAL_EXPRESSION_SYNTAX,
(ULONG)Argument);
}
}
if (Flags & DEBUG_CES_TEXT_REPLACEMENTS)
{
InvFlags |= (1 << ALIAS_BIT);
DIRTY_WORKSPACE(WSPF_DIRTY_ALIASES);
}
#if DBG_CALLBACK
DebugPrint("CES %X, arg %I64X, inv %X\n", Flags, Argument, InvFlags);
#endif
// There's no need to update buffers when we're throwing
// everything away while shutting down a session.
if (g_EndingSession == ENDING_NONE)
{
InvalidateStateBuffers(InvFlags);
}
if (InvFlags != 0)
{
UpdateEngine();
}
if (InvFlags & (1 << LOCALS_WINDOW))
{
g_ScopeChanged = TRUE;
}
return S_OK;
}
STDMETHODIMP
EventCallbacks::ChangeSymbolState(
THIS_
IN ULONG Flags,
IN ULONG64 Argument
)
{
ULONG InvFlags = 0;
// If module information changed we need to update
// everything that might display or depend on symbols.
if (Flags & (DEBUG_CSS_LOADS |
DEBUG_CSS_UNLOADS))
{
InvFlags |= UPDATE_SYM_WINDOWS | (1 << MODULE_BIT);
// On any generic module load/unload refresh
// windbg's module state. This helps catch stale
// source after explicit .reload commands and
// other global refresh changes.
if (!Argument)
{
UpdateBufferWindows((1 << DOC_WINDOW),
UPDATE_REFRESH_MODULES);
}
}
// If the scope changed we need to update scope-related windows.
if (Flags & DEBUG_CSS_SCOPE)
{
InvFlags |=
(1 << WATCH_WINDOW) |
(1 << LOCALS_WINDOW) |
(1 << CALLS_WINDOW);
}
// If paths changed we need to update
// the event state in case we can suddenly load source.
if (Flags & DEBUG_CSS_PATHS)
{
InvFlags |= (1 << EVENT_BIT);
DIRTY_WORKSPACE(WSPF_DIRTY_PATHS);
}
// If certain options changed we need to update
// everything that might display or depend on symbols.
if (Flags & DEBUG_CSS_SYMBOL_OPTIONS)
{
if ((g_SymOptions ^ (ULONG)Argument) & REFRESH_SYMOPT)
{
InvFlags |= UPDATE_SYM_WINDOWS;
}
g_SymOptions = (ULONG)Argument;
}
// If certain options changed we need to update
// everything that might display or depend on symbols.
if (Flags & DEBUG_CSS_TYPE_OPTIONS)
{
InvFlags |=
(1 << WATCH_WINDOW) |
(1 << LOCALS_WINDOW) |
(1 << CALLS_WINDOW);
if (g_pUiSymbols2 != NULL)
{
g_pUiSymbols2->GetTypeOptions( &g_TypeOptions );
if (g_Workspace != NULL)
{
g_Workspace->SetUlong(WSP_GLOBAL_TYPE_OPTIONS,
g_TypeOptions);
}
}
}
#if DBG_CALLBACK
DebugPrint("CSS %X, arg %I64X, inv %X\n", Flags, Argument, InvFlags);
#endif
// There's no need to update buffers when we're throwing
// everything away while shutting down a session.
if (g_EndingSession == ENDING_NONE)
{
InvalidateStateBuffers(InvFlags);
}
if (InvFlags != 0)
{
UpdateEngine();
}
if (InvFlags & (1 << LOCALS_WINDOW))
{
g_ScopeChanged = TRUE;
}
return S_OK;
}
EventCallbacks g_EventCb;
//----------------------------------------------------------------------------
//
// Inter-thread communication.
//
//----------------------------------------------------------------------------
#define COMMAND_OVERHEAD (sizeof(ULONG64) + sizeof(UiCommandData))
PVOID
StartCommand(UiCommand Cmd, ULONG Len)
{
UiCommandData* Data;
// Round length up to a multiple of ULONG64s for
// alignment.
Len = ((Len + sizeof(ULONG64) - 1) & ~(sizeof(ULONG64) - 1)) +
sizeof(UiCommandData);
if (Len > MAX_COMMAND_LEN)
{
return NULL;
}
LockUiBuffer(&g_UiCommandBuffer);
Data = (UiCommandData *)g_UiCommandBuffer.AddData(Len);
if (Data == NULL)
{
return Data;
}
Data->Cmd = Cmd;
Data->Len = Len;
return Data + 1;
}
void
FinishCommand(void)
{
UnlockUiBuffer(&g_UiCommandBuffer);
// Wake up the engine to process the command.
UpdateEngine();
}
BOOL
AddStringCommand(UiCommand Cmd, PCSTR Str)
{
ULONG StrLen = strlen(Str) + 1;
PSTR Data;
// If we're adding command input we may need
// to send it directly to the engine in response
// to an input request.
if (Cmd == UIC_CMD_INPUT)
{
LockUiBuffer(&g_UiCommandBuffer);
#if DBG_INPUT
DebugPrint("InputStarted started %d\n", g_InputStarted);
#endif
if (g_InputStarted)
{
HRESULT Status;
g_OutputCb.Output(DEBUG_OUTPUT_NORMAL, Str);
g_OutputCb.Output(DEBUG_OUTPUT_NORMAL, "\n");
Status = g_pUiControl->ReturnInput(Str);
#if DBG_INPUT
DebugPrint(" ReturnInput status %X\n", Status);
#endif
g_InputStarted = FALSE;
UnlockUiBuffer(&g_UiCommandBuffer);
return TRUE;
}
}
Data = (PSTR)StartCommand(Cmd, StrLen);
if (Cmd == UIC_CMD_INPUT)
{
UnlockUiBuffer(&g_UiCommandBuffer);
}
if (Data == NULL)
{
return FALSE;
}
memcpy(Data, Str, StrLen);
FinishCommand();
return TRUE;
}
BOOL
AddStringMultiCommand(UiCommand Cmd, PSTR Str, BOOL ForceSplit)
{
//
// Given a string with multiple commands separated
// by newlines, break the string into multiple
// commands, one per line. This allows arbitrarily
// large command strings without running into
// the MAX_COMMAND_LEN limit as long as each individual
// line fits within that limit.
//
while (*Str)
{
PSTR Scan, LastNl;
ULONG Len;
BOOL Status;
Scan = Str + 1;
Len = 1;
LastNl = NULL;
while (*Scan && Len < (MAX_COMMAND_LEN - COMMAND_OVERHEAD))
{
if (*Scan == '\n')
{
LastNl = Scan;
if (ForceSplit)
{
Scan++;
break;
}
}
Scan++;
Len++;
}
// If the rest of the command string doesn't fit
// within the limit it needs to be split.
// If there's no newline to break it at
// the command is too large to be processed.
if (*Scan && !LastNl)
{
return FALSE;
}
// Split if necessary.
if (*Scan)
{
*LastNl = 0;
}
// Add the head (which may be the whole remainder).
Status = AddStringCommand(Cmd, Str);
if (*Scan)
{
*LastNl = '\n';
if (!Status)
{
return FALSE;
}
Str = LastNl + 1;
}
else
{
return Status;
}
}
return TRUE;
}
BOOL __cdecl
PrintStringCommand(UiCommand Cmd, PCSTR Format, ...)
{
char Buf[MAX_COMMAND_LEN - COMMAND_OVERHEAD];
va_list Args;
va_start(Args, Format);
_vsnprintf(Buf, DIMA(Buf), Format, Args);
Buf[DIMA(Buf) - 1] = 0;
va_end(Args);
return AddStringCommand(Cmd, Buf);
}
void
WriteData(UIC_WRITE_DATA_DATA* WriteData)
{
ULONG Written;
switch(WriteData->Type)
{
default:
Assert(!"Unhandled condition");
break;
case PHYSICAL_MEM_TYPE:
g_pDbgData->WritePhysical(WriteData->Offset,
WriteData->Data,
WriteData->Length,
&Written
);
break;
case VIRTUAL_MEM_TYPE:
g_pDbgData->WriteVirtual(WriteData->Offset,
WriteData->Data,
WriteData->Length,
&Written
);
break;
case CONTROL_MEM_TYPE:
g_pDbgData->WriteControl(WriteData->Any.control.Processor,
WriteData->Offset,
WriteData->Data,
WriteData->Length,
&Written
);
break;
case IO_MEM_TYPE:
g_pDbgData->WriteIo(WriteData->Any.io.interface_type,
WriteData->Any.io.BusNumber,
WriteData->Any.io.AddressSpace,
WriteData->Offset,
WriteData->Data,
WriteData->Length,
&Written
);
break;
case MSR_MEM_TYPE:
Assert(WriteData->Length == sizeof(ULONG64));
g_pDbgData->WriteMsr((ULONG)WriteData->Offset,
*(PULONG64)WriteData->Data
);
break;
case BUS_MEM_TYPE:
g_pDbgData->WriteBusData(WriteData->Any.bus.bus_type,
WriteData->Any.bus.BusNumber,
WriteData->Any.bus.SlotNumber,
(ULONG)WriteData->Offset,
WriteData->Data,
WriteData->Length,
&Written
);
break;
}
}
void
ProcessWatchCommand(
UIC_SYMBOL_WIN_DATA *SymWinData
)
{
PDEBUG_SYMBOL_GROUP pSymbolGroup;
if (!SymWinData->pSymbolGroup ||
!(pSymbolGroup = *SymWinData->pSymbolGroup))
{
return;
}
switch (SymWinData->Type)
{
case ADD_SYMBOL_WIN:
if (strlen(SymWinData->u.Add.Name))
{
// add only non-empty symbols
pSymbolGroup->AddSymbol(SymWinData->u.Add.Name,
&SymWinData->u.Add.Index);
}
break;
case DEL_SYMBOL_WIN_INDEX:
pSymbolGroup->RemoveSymbolByIndex(SymWinData->u.DelIndex);
break;
case DEL_SYMBOL_WIN_NAME:
pSymbolGroup->RemoveSymbolByName(SymWinData->u.DelName);
break;
case QUERY_NUM_SYMBOL_WIN:
pSymbolGroup->GetNumberSymbols(SymWinData->u.NumWatch);
break;
case GET_NAME:
pSymbolGroup->GetSymbolName(SymWinData->u.GetName.Index,
SymWinData->u.GetName.Buffer,
SymWinData->u.GetName.BufferSize,
SymWinData->u.GetName.NameSize);
break;
case GET_PARAMS:
pSymbolGroup->
GetSymbolParameters(SymWinData->u.GetParams.Start,
SymWinData->u.GetParams.Count,
SymWinData->u.GetParams.SymbolParams);
break;
case EXPAND_SYMBOL:
pSymbolGroup->ExpandSymbol(SymWinData->u.ExpandSymbol.Index,
SymWinData->u.ExpandSymbol.Expand);
break;
case EDIT_SYMBOL:
pSymbolGroup->WriteSymbol(SymWinData->u.WriteSymbol.Index,
SymWinData->u.WriteSymbol.Value);
break;
case EDIT_TYPE:
pSymbolGroup->OutputAsType(SymWinData->u.OutputAsType.Index,
SymWinData->u.OutputAsType.Type);
break;
case DEL_SYMBOL_WIN_ALL:
{
ULONG nSyms = 0;
pSymbolGroup->GetNumberSymbols(&nSyms);
while (nSyms)
{
pSymbolGroup->RemoveSymbolByIndex(0);
pSymbolGroup->GetNumberSymbols(&nSyms);
}
}
}
}
void
ProcessCommand(UiCommandData* CmdData)
{
DEBUG_VALUE Val;
HRESULT Status;
switch(CmdData->Cmd)
{
case UIC_CMD_INPUT:
case UIC_EXECUTE:
PSTR Str;
ULONG StrLen;
// Make sure the command has a newline at the end.
Str = (PSTR)(CmdData + 1);
StrLen = strlen(Str);
if (StrLen > 0 && Str[StrLen - 1] == '\n')
{
// Trim existing newline as we're adding one.
Str[StrLen - 1] = 0;
}
if (g_RemoteClient)
{
// Identify self before command.
g_pDbgClient->OutputIdentity(DEBUG_OUTCTL_ALL_OTHER_CLIENTS,
DEBUG_OUTPUT_IDENTITY_DEFAULT,
"[%s] ");
}
g_pDbgControl->OutputPrompt(DEBUG_OUTCTL_ALL_CLIENTS, " %s\n", Str);
g_pDbgControl->Execute(DEBUG_OUTCTL_ALL_CLIENTS,
Str, DEBUG_EXECUTE_NOT_LOGGED);
break;
case UIC_SILENT_EXECUTE:
// Execute the command without displaying it.
g_pDbgControl->Execute(DEBUG_OUTCTL_IGNORE,
(PCSTR)(CmdData + 1),
DEBUG_EXECUTE_NOT_LOGGED |
DEBUG_EXECUTE_NO_REPEAT);
break;
case UIC_INVISIBLE_EXECUTE:
// Execute the command without displaying it and
// ignore any notifications.
g_Invisible = TRUE;
g_pDbgControl->Execute(DEBUG_OUTCTL_IGNORE,
(PCSTR)(CmdData + 1),
DEBUG_EXECUTE_NOT_LOGGED |
DEBUG_EXECUTE_NO_REPEAT);
g_Invisible = FALSE;
break;
case UIC_SET_REG:
UIC_SET_REG_DATA* SetRegData;
SetRegData = (UIC_SET_REG_DATA*)(CmdData + 1);
g_pDbgRegisters->SetValue(SetRegData->Reg, &SetRegData->Val);
break;
case UIC_RESTART:
if (g_RemoteClient)
{
g_pDbgControl->
Output(DEBUG_OUTPUT_ERROR,
"Only user-mode created processes may be restarted\n");
}
else
{
if ((Status = g_pDbgClient->
EndSession(DEBUG_END_ACTIVE_TERMINATE)) != S_OK)
{
InternalError(Status, "EndSession");
}
else
{
g_EndingSession = ENDING_RESTART;
}
}
break;
case UIC_END_SESSION:
ULONG OldEnding;
ULONG OldExec;
// Mark the session as ending to avoid workspace
// deadlock problems.
OldEnding = g_EndingSession;
OldExec = g_ExecStatus;
g_EndingSession = ENDING_STOP;
g_ExecStatus = DEBUG_STATUS_NO_DEBUGGEE;
if (!g_RemoteClient)
{
if ((Status = g_pDbgClient->
EndSession(DEBUG_END_ACTIVE_TERMINATE)) != S_OK)
{
InternalError(Status, "EndSession");
g_EndingSession = OldEnding;
g_ExecStatus = OldExec;
}
}
break;
case UIC_WRITE_DATA:
WriteData((UIC_WRITE_DATA_DATA*)(CmdData + 1));
break;
case UIC_SYMBOL_WIN:
ProcessWatchCommand((UIC_SYMBOL_WIN_DATA*) (CmdData + 1));
break;
case UIC_DISPLAY_CODE:
FillCodeBuffer(((UIC_DISPLAY_CODE_DATA*)(CmdData + 1))->Offset,
TRUE);
break;
case UIC_DISPLAY_CODE_EXPR:
if (g_pDbgControl->Evaluate((PSTR)(CmdData + 1), DEBUG_VALUE_INT64,
&Val, NULL) != S_OK)
{
Val.I64 = 0;
}
FillCodeBuffer(Val.I64, TRUE);
break;
case UIC_SET_SCOPE:
SetLocalScope(&(((UIC_SET_SCOPE_DATA *)(CmdData + 1))->StackFrame));
InvalidateStateBuffers(1 << LOCALS_WINDOW);
break;
case UIC_SET_FILTER:
UIC_SET_FILTER_DATA* SetFilter;
SetFilter = (UIC_SET_FILTER_DATA*)(CmdData + 1);
if (SetFilter->Index != 0xffffffff)
{
DEBUG_SPECIFIC_FILTER_PARAMETERS Params;
Params.ExecutionOption = SetFilter->Execution;
Params.ContinueOption = SetFilter->Continue;
g_pDbgControl->SetSpecificFilterParameters(SetFilter->Index, 1,
&Params);
}
else
{
DEBUG_EXCEPTION_FILTER_PARAMETERS Params;
Params.ExecutionOption = SetFilter->Execution;
Params.ContinueOption = SetFilter->Continue;
Params.ExceptionCode = SetFilter->Code;
g_pDbgControl->SetExceptionFilterParameters(1, &Params);
}
break;
case UIC_SET_FILTER_ARGUMENT:
UIC_SET_FILTER_ARGUMENT_DATA* SetFilterArg;
SetFilterArg = (UIC_SET_FILTER_ARGUMENT_DATA*)(CmdData + 1);
g_pDbgControl->SetSpecificFilterArgument(SetFilterArg->Index,
SetFilterArg->Argument);
break;
case UIC_SET_FILTER_COMMAND:
UIC_SET_FILTER_COMMAND_DATA* SetFilterCmd;
SetFilterCmd = (UIC_SET_FILTER_COMMAND_DATA*)(CmdData + 1);
if (SetFilterCmd->Which == 0)
{
g_pDbgControl->SetEventFilterCommand(SetFilterCmd->Index,
SetFilterCmd->Command);
}
else
{
g_pDbgControl->SetExceptionFilterSecondCommand
(SetFilterCmd->Index, SetFilterCmd->Command);
}
break;
case UIC_SET_IP:
// First, execute the corresponding command to set the
// register value.
g_pDbgControl->Execute(DEBUG_OUTCTL_IGNORE,
(PCSTR)(CmdData + 1),
DEBUG_EXECUTE_NOT_LOGGED |
DEBUG_EXECUTE_NO_REPEAT);
// Now, since we know that IP changed we need
// to force an update to the current scope.
g_pDbgSymbols->ResetScope();
break;
}
}
void
ProcessEngineCommands(BOOL Internal)
{
#if DBG_CALLBACK
DebugPrint("ProcessEngineCommands\n");
#endif
// Check for commands to execute. We do not
// want to hold the lock while doing so because
// the commands may include things that cause waits
// and we don't want to lock out the GUI.
LockUiBuffer(&g_UiCommandBuffer);
while (g_UiCommandBuffer.GetDataLen() > 0)
{
//
// Remove the first command from the buffer.
//
// Extra char is for forcing a newline on executes.
char CmdBuf[MAX_COMMAND_LEN + 1];
UiCommandData* CmdData;
// Copy command to local buffer.
CmdData = (UiCommandData*)g_UiCommandBuffer.GetDataBuffer();
memcpy(CmdBuf, CmdData, CmdData->Len);
CmdData = (UiCommandData*)CmdBuf;
// Remove command from queue and release the queue for
// the UI thread to use again.
g_UiCommandBuffer.RemoveHead(CmdData->Len);
UnlockUiBuffer(&g_UiCommandBuffer);
ProcessCommand(CmdData);
InterlockedIncrement((PLONG)&g_CommandSequence);
// Lock the buffer again for the next command retrieval.
LockUiBuffer(&g_UiCommandBuffer);
if (g_EndingSession != ENDING_NONE)
{
// If we're ending a session just throw away the rest
// of the commands.
g_UiCommandBuffer.Empty();
}
}
UnlockUiBuffer(&g_UiCommandBuffer);
if (!Internal && g_EndingSession == ENDING_NONE)
{
ReadStateBuffers();
}
}
//----------------------------------------------------------------------------
//
// Engine processing.
//
//----------------------------------------------------------------------------
HRESULT
InitializeEngineInterfaces(void)
{
HRESULT Hr;
if ((Hr = g_pUiClient->CreateClient(&g_pDbgClient)) != S_OK)
{
InternalError(Hr, "Engine CreateClient");
return Hr;
}
if ((Hr = g_pDbgClient->
QueryInterface(IID_IDebugControl,
(void **)&g_pDbgControl)) != S_OK ||
(Hr = g_pDbgClient->
QueryInterface(IID_IDebugSymbols,
(void **)&g_pDbgSymbols)) != S_OK ||
(Hr = g_pDbgClient->
QueryInterface(IID_IDebugRegisters,
(void **)&g_pDbgRegisters)) != S_OK ||
(Hr = g_pDbgClient->
QueryInterface(IID_IDebugDataSpaces,
(void **)&g_pDbgData)) != S_OK ||
(Hr = g_pDbgClient->
QueryInterface(IID_IDebugSystemObjects,
(void **)&g_pDbgSystem)) != S_OK)
{
if (Hr == RPC_E_VERSION_MISMATCH)
{
InformationBox(ERR_Remoting_Version_Mismatch);
}
else
{
InternalError(Hr, "Engine QueryInterface");
}
return Hr;
}
//
// Try and get higher version interfaces.
//
if (g_pDbgClient->
QueryInterface(IID_IDebugClient2,
(void **)&g_pDbgClient2) != S_OK)
{
g_pDbgClient2 = NULL;
}
if (g_pDbgClient->
QueryInterface(IID_IDebugSystemObjects3,
(void **)&g_pDbgSystem3) != S_OK)
{
g_pDbgSystem3 = NULL;
}
if (g_RemoteClient)
{
// Create a local client to do local source file lookups.
if ((Hr = g_pUiLocClient->CreateClient(&g_pLocClient)) != S_OK ||
(Hr = g_pLocClient->
QueryInterface(IID_IDebugControl,
(void **)&g_pLocControl)) != S_OK ||
(Hr = g_pLocClient->
QueryInterface(IID_IDebugSymbols,
(void **)&g_pLocSymbols)) != S_OK)
{
InternalError(Hr, "Engine local client");
return Hr;
}
}
else
{
g_pLocClient = g_pDbgClient;
g_pLocClient->AddRef();
g_pLocControl = g_pDbgControl;
g_pLocControl->AddRef();
g_pLocSymbols = g_pDbgSymbols;
g_pLocSymbols->AddRef();
}
// Create separate client for private output capture
// during state buffer filling. The output capture client
// sets its output mask to nothing so that it doesn't
// receive any normal input. During private output capture
// the output control is set to THIS_CLIENT | OVERRIDE_MASK to force
// output to just the output capture client.
if ((Hr = g_pDbgClient->CreateClient(&g_pOutCapClient)) != S_OK ||
(Hr = g_pOutCapClient->
QueryInterface(IID_IDebugControl,
(void **)&g_pOutCapControl)) != S_OK)
{
InternalError(Hr, "Engine output capture client");
return Hr;
}
// Set callbacks.
if ((Hr = g_pDbgClient->SetOutputCallbacks(&g_OutputCb)) != S_OK ||
(Hr = g_pDbgClient->SetInputCallbacks(&g_InputCb)) != S_OK ||
(Hr = g_pDbgClient->SetEventCallbacks(&g_EventCb)) != S_OK ||
(g_RemoteClient &&
(Hr = g_pLocClient->SetOutputCallbacks(&g_OutputCb))) != S_OK ||
(Hr = g_pOutCapClient->SetOutputMask(0)) != S_OK ||
(Hr = g_pOutCapClient->SetOutputCallbacks(&g_OutStateBuf)) != S_OK ||
(Hr = g_pOutCapClient->
QueryInterface(IID_IDebugSymbols,
(void **)&g_pOutCapSymbols)) != S_OK)
{
InternalError(Hr, "Engine callbacks");
return Hr;
}
// Create a watch window client
if ((Hr = g_pOutCapSymbols->
CreateSymbolGroup(&g_pDbgWatchSymbolGroup)) != S_OK)
{
InternalError(Hr, "Engine CreateSymbolGroup");
return Hr;
}
// Create a local window client
if ((Hr = g_pOutCapSymbols->
GetScopeSymbolGroup(DEBUG_SCOPE_GROUP_LOCALS,
NULL, &g_pDbgLocalSymbolGroup)) == E_NOTIMPL)
{
// Older version
Hr = g_pOutCapSymbols->
GetScopeSymbolGroup(DEBUG_SCOPE_GROUP_ALL,
NULL, &g_pDbgLocalSymbolGroup);
}
if (Hr != S_OK ||
(Hr = g_FilterTextBuffer->Update()) != S_OK)
{
InternalError(Hr, "Engine GetScopeSymbolGroup");
return Hr;
}
if ((Hr = g_pDbgControl->GetRadix(&g_NumberRadix)) != S_OK)
{
InternalError(Hr, "Engine GetRadix");
return Hr;
}
if (g_RemoteClient)
{
return S_OK;
}
//
// Set up initial state for things that are important
// when starting the debug session.
//
if (g_Verbose)
{
DWORD OutMask;
g_pDbgClient->GetOutputMask(&OutMask);
OutMask |= DEBUG_OUTPUT_VERBOSE;
g_pDbgClient->SetOutputMask(OutMask);
g_pDbgControl->SetLogMask(OutMask);
}
// Always load line numbers for source support.
g_pDbgSymbols->AddSymbolOptions(SYMOPT_LOAD_LINES);
// Set the source stepping mode
g_IgnoreCodeLevelChange = TRUE;
if (GetSrcMode_StatusBar())
{
g_pDbgControl->SetCodeLevel(DEBUG_LEVEL_SOURCE);
}
else
{
g_pDbgControl->SetCodeLevel(DEBUG_LEVEL_ASSEMBLY);
}
g_IgnoreCodeLevelChange = FALSE;
// If this is a user-mode debug session default to
// initial and final breaks. Don't override settings
// that were given on the command line, though.
g_IgnoreFilterChange = TRUE;
if (g_DebugCommandLine != NULL ||
g_PidToDebug != 0 ||
g_ProcNameToDebug != NULL)
{
g_pDbgControl->AddEngineOptions((DEBUG_ENGOPT_INITIAL_BREAK |
DEBUG_ENGOPT_FINAL_BREAK) &
~g_EngOptModified);
}
else
{
g_pDbgControl->RemoveEngineOptions((DEBUG_ENGOPT_INITIAL_BREAK |
DEBUG_ENGOPT_FINAL_BREAK) &
~g_EngOptModified);
}
g_IgnoreFilterChange = FALSE;
return S_OK;
}
void
DiscardEngineState(void)
{
LockUiBuffer(&g_UiOutputBuffer);
g_UiOutputBuffer.Empty();
UnlockUiBuffer(&g_UiOutputBuffer);
g_TargetClass = DEBUG_CLASS_UNINITIALIZED;
g_ExecStatus = DEBUG_STATUS_NO_DEBUGGEE;
}
void
ReleaseEngineInterfaces(void)
{
DiscardEngineState();
RELEASE(g_pLocControl);
RELEASE(g_pLocSymbols);
RELEASE(g_pLocClient);
RELEASE(g_pDbgWatchSymbolGroup);
RELEASE(g_pDbgLocalSymbolGroup);
RELEASE(g_pOutCapControl);
RELEASE(g_pOutCapSymbols);
RELEASE(g_pOutCapClient);
RELEASE(g_pDbgControl);
RELEASE(g_pDbgSymbols);
RELEASE(g_pDbgRegisters);
RELEASE(g_pDbgData);
RELEASE(g_pDbgSystem3);
RELEASE(g_pDbgSystem);
RELEASE(g_pDbgClient2);
RELEASE(g_pDbgClient);
}
BOOL
ExtractWspName(PSTR CommandLine, PSTR Buf, ULONG BufLen)
{
PSTR Scan = CommandLine;
PSTR Start;
while (isspace(*Scan))
{
Scan++;
}
if (!*Scan)
{
return FALSE;
}
else if (*Scan == '"')
{
Start = ++Scan;
// Look for closing quote.
while (*Scan && *Scan != '"')
{
Scan++;
}
}
else
{
// Look for whitespace.
Start = Scan++;
while (*Scan && !isspace(*Scan))
{
Scan++;
}
}
ULONG Len = (ULONG) (ULONG64) (Scan - Start);
if (Len == 0)
{
return FALSE;
}
if (Len >= BufLen)
{
Len = BufLen - 1;
}
memcpy(Buf, Start, Len);
Buf[Len] = 0;
return TRUE;
}
HRESULT
StartSession(void)
{
TCHAR WspName[MAX_PATH];
ULONG WspKey;
PTSTR WspValue;
HRESULT Hr;
ULONG i;
// Reset things to the default priority first.
// If necessary, priority will be increased in certain code
// paths later.
SetPriorityClass(GetCurrentProcess(), g_DefPriority);
if (!g_RemoteClient)
{
if (g_NumDumpFiles)
{
WspKey = WSP_NAME_DUMP;
WspValue = g_DumpFiles[0];
EngSwitchWorkspace(WspKey, WspValue);
if (g_NumDumpInfoFiles)
{
if (g_pDbgClient2 == NULL)
{
ErrorBox(NULL, 0, ERR_Cant_Add_Dump_Info_File);
Hr = E_NOINTERFACE;
goto ResetWorkspace;
}
for (i = 0; i < g_NumDumpInfoFiles; i++)
{
if ((Hr = g_pDbgClient2->AddDumpInformationFile
(g_DumpInfoFiles[i], g_DumpInfoTypes[i])) != S_OK)
{
ErrorBox(NULL, 0, ERR_Add_Dump_Info_File_Failed,
g_DumpInfoFiles[i], FormatStatusCode(Hr),
FormatStatus(Hr));
goto ResetWorkspace;
}
}
}
for (i = 0; i < g_NumDumpFiles; i++)
{
Hr = g_pDbgClient->OpenDumpFile(g_DumpFiles[i]);
if (Hr != S_OK)
{
if ((HRESULT_FACILITY(Hr)) == FACILITY_WIN32)
{
// Win32 errors on open generally mean some
// kind of file error.
ErrorBox(NULL, 0, ERR_Invalid_Dump_File_Name,
g_DumpFiles[i], FormatStatusCode(Hr),
FormatStatus(Hr));
}
else
{
ErrorBox(NULL, 0, ERR_Unable_To_Open_Dump,
g_DumpFiles[i], FormatStatusCode(Hr),
FormatStatus(Hr));
}
goto ResetWorkspace;
}
}
}
else if (g_DebugCommandLine != NULL ||
g_PidToDebug != 0 ||
g_ProcNameToDebug != NULL)
{
ULONG64 Server = 0;
ULONG Pid;
WspKey = WSP_NAME_USER;
WspValue = g_ProcessServer != NULL ?
g_ProcessServer : g_WorkspaceDefaultName;
if (g_DebugCommandLine != NULL)
{
if (ExtractWspName(g_DebugCommandLine,
WspName, sizeof(WspName)))
{
WspValue = WspName;
}
}
EngSwitchWorkspace(WspKey, WspValue);
if (g_ProcessServer != NULL)
{
Hr = g_pDbgClient->ConnectProcessServer(g_ProcessServer,
&Server);
if (Hr != S_OK)
{
ErrorBox(NULL, 0, ERR_Connect_Process_Server,
g_ProcessServer, FormatStatusCode(Hr),
FormatStatus(Hr));
goto ResetWorkspace;
}
// Default to not automatically bringing up a disassembly
// window as it is very expensive to remote all
// the virtual reads done for it.
g_WinOptions &= ~WOPT_AUTO_DISASM;
}
if (g_ProcNameToDebug != NULL)
{
Hr = g_pDbgClient->GetRunningProcessSystemIdByExecutableName
(Server, g_ProcNameToDebug, DEBUG_GET_PROC_ONLY_MATCH,
&Pid);
if (Hr != S_OK)
{
if (Hr == S_FALSE)
{
ErrorBox(NULL, 0, ERR_Ambiguous_Named_Process,
g_ProcNameToDebug);
}
else
{
ErrorBox(NULL, 0, ERR_Get_Named_Process,
g_ProcNameToDebug, FormatStatusCode(Hr),
FormatStatus(Hr));
}
goto ResetWorkspace;
}
}
else
{
Pid = g_PidToDebug;
}
if (g_DebugCommandLine != NULL &&
g_ProcessStartDir != NULL &&
g_ProcessStartDir[0] != 0)
{
sprintf(WspName, ".createdir \"%s\"", g_ProcessStartDir);
g_pDbgControl->Execute(DEBUG_OUTCTL_IGNORE,
WspName,
DEBUG_EXECUTE_NOT_LOGGED);
}
Hr = g_pDbgClient->CreateProcessAndAttach(Server,
g_DebugCommandLine,
g_DebugCreateFlags,
Pid,
g_AttachProcessFlags);
if (g_DebugCommandLine != NULL &&
g_ProcessStartDir != NULL &&
g_ProcessStartDir[0] != 0)
{
g_pDbgControl->Execute(DEBUG_OUTCTL_IGNORE,
".createdir \"\"",
DEBUG_EXECUTE_NOT_LOGGED);
}
if (Hr != S_OK)
{
if (g_DebugCommandLine != NULL)
{
ErrorBox(NULL, 0, ERR_Invalid_Process_Create,
g_DebugCommandLine, FormatStatusCode(Hr),
FormatStatus(Hr));
}
else
{
ErrorBox(NULL, 0, ERR_Invalid_Process_Attach,
Pid, FormatStatusCode(Hr), FormatStatus(Hr));
}
goto ResetWorkspace;
}
if (g_DetachOnExit &&
(Hr = g_pDbgClient->
AddProcessOptions(DEBUG_PROCESS_DETACH_ON_EXIT)) != S_OK)
{
ErrorBox(NULL, 0, ERR_No_Detach_On_Exit);
}
if (Server != 0)
{
g_pDbgClient->DisconnectProcessServer(Server);
}
// Bump up our priority so that the debugger stays responsive
// even when the debuggee is running.
SetPriorityClass(GetCurrentProcess(), HIGH_PRIORITY_CLASS);
}
else
{
// Default to not automatically bringing up a disassembly
// window as it is very expensive in kernel debugging.
g_WinOptions &= ~WOPT_AUTO_DISASM;
WspKey = WSP_NAME_KERNEL;
WspValue = g_WorkspaceDefaultName;
EngSwitchWorkspace(WspKey, WspValue);
Hr = g_pDbgClient->AttachKernel(g_AttachKernelFlags,
g_KernelConnectOptions);
if (Hr != S_OK)
{
if (g_AttachKernelFlags == DEBUG_ATTACH_LOCAL_KERNEL)
{
if (Hr == E_NOTIMPL)
{
ErrorBox(NULL, 0, ERR_No_Local_Kernel_Debugging);
}
else
{
ErrorBox(NULL, 0, ERR_Failed_Local_Kernel_Debugging,
FormatStatusCode(Hr), FormatStatus(Hr));
}
}
else
{
ErrorBox(NULL, 0, ERR_Invalid_Kernel_Attach,
g_KernelConnectOptions, FormatStatusCode(Hr),
FormatStatus(Hr));
}
goto ResetWorkspace;
}
}
}
else
{
WspKey = WSP_NAME_REMOTE;
WspValue = g_WorkspaceDefaultName;
EngSwitchWorkspace(WspKey, WspValue);
// Use a heuristic of 45 characters per line.
g_pDbgClient->ConnectSession(DEBUG_CONNECT_SESSION_DEFAULT,
g_HistoryLines * 45);
}
PostMessage(g_hwndFrame, WU_ENGINE_STARTED, 0, S_OK);
return S_OK;
ResetWorkspace:
// We just switched to this workspace but
// we're failing and we want to abandon it.
if (g_Workspace != NULL && !g_ExplicitWorkspace)
{
// Make sure this doesn't cause a popup.
g_Workspace->ClearDirty();
EngSwitchWorkspace(WSP_NAME_BASE,
g_WorkspaceDefaultName);
}
return Hr;
}
void
SetLocalScope(PDEBUG_STACK_FRAME pStackFrame)
{
DEBUG_STACK_FRAME LocalFrame;
if (!pStackFrame)
{
// Get and use the default scope
if (g_pDbgSymbols->ResetScope() != S_OK ||
g_pDbgSymbols->GetScope(NULL, &LocalFrame, NULL, 0) != S_OK)
{
return;
}
pStackFrame = &LocalFrame;
}
else if (FAILED(g_pDbgSymbols->SetScope(0, pStackFrame, NULL, 0)))
{
return;
}
}
void
SessionActive(void)
{
HRESULT Hr;
// This can get called twice if a remote client connects
// just as a session is becoming active.
if (g_SessionActive)
{
return;
}
g_SessionActive = TRUE;
if ((Hr = g_pDbgControl->
GetActualProcessorType(&g_ActualProcType)) != S_OK ||
FAILED(Hr = g_pDbgControl->
GetProcessorTypeNames(g_ActualProcType, NULL, 0, NULL,
g_ActualProcAbbrevName,
sizeof(g_ActualProcAbbrevName),
NULL)))
{
ErrorExit(g_pDbgClient,
"Debug target initialization failed, 0x%X\n", Hr);
}
// Put in a request for the actual processor register
// names right now.
GetRegisterNames(g_ActualProcType);
if (FAILED(Hr = g_pDbgControl->IsPointer64Bit()))
{
ErrorExit(g_pDbgClient,
"Unable to get debuggee pointer size, 0x%X\n", Hr);
}
g_Ptr64 = Hr == S_OK;
if (!g_RemoteClient &&
(g_ExplicitWorkspace || g_TargetClass == DEBUG_CLASS_KERNEL) &&
g_NumDumpFiles == 0)
{
if (g_ExplicitWorkspace)
{
// Reapply the workspace after a reboot/restart to get
// breakpoints and other engine state back.
// Don't restart the session when doing so.
if (g_Workspace != NULL)
{
Workspace* Wsp = g_Workspace;
g_Workspace = NULL;
Wsp->Apply(WSP_APPLY_AGAIN);
g_Workspace = Wsp;
}
}
else
{
// The same machine could theoretically debug many different
// processor types over kernel connections or via different
// processor dumps. Workspaces contain processor-related
// information, such as register maps, so allow for different
// workspaces based on the processor type. This is only
// done when a default workspace would otherwise be used,
// though to reduce workspace explosion.
EngSwitchWorkspace(WSP_NAME_KERNEL, g_ActualProcAbbrevName);
}
}
InvalidateStateBuffers(BUFFERS_ALL);
UpdateBufferWindows((1 << CPU_WINDOW) | (1 << DOC_WINDOW),
UPDATE_START_SESSION);
UpdateEngine();
}
void
SessionInactive(void)
{
if (!g_RemoteClient && g_EndingSession != ENDING_STOP)
{
EngSwitchWorkspace(WSP_NAME_BASE,
g_WorkspaceDefaultName);
}
g_SessionActive = FALSE;
g_ActualProcType = IMAGE_FILE_MACHINE_UNKNOWN;
InvalidateStateBuffers(BUFFERS_ALL);
UpdateBufferWindows((1 << CPU_WINDOW) | (1 << DOC_WINDOW) |
(1 << DISASM_WINDOW), UPDATE_END_SESSION);
UpdateEngine();
SetPriorityClass(GetCurrentProcess(), g_DefPriority);
}
void
StopOrEndDebugging(void)
{
//
// If the session was started from the command line
// assume that debugging is done and exit.
// If the session was started from the UI treat it
// like a stop debugging request.
//
if (g_CommandLineStart)
{
EngSwitchWorkspace(WSP_NAME_BASE,
g_WorkspaceDefaultName);
g_Exit = TRUE;
PostMessage(g_hwndFrame, WU_UPDATE, UPDATE_EXIT, 0);
}
else
{
PostMessage(g_hwndFrame, WM_COMMAND,
0xffff0000 | IDM_DEBUG_STOPDEBUGGING, 0);
g_EndingSession = ENDING_STOP;
}
}
DWORD
WINAPI
EngineLoop(LPVOID Param)
{
HRESULT Hr;
if ((Hr = InitializeEngineInterfaces()) != S_OK ||
(Hr = StartSession()) != S_OK)
{
ReleaseEngineInterfaces();
PostMessage(g_hwndFrame, WU_ENGINE_STARTED, 0, Hr);
return 0;
}
g_EngineThreadId = GetCurrentThreadId();
Hr = g_pDbgControl->GetDebuggeeType(&g_TargetClass, &g_TargetClassQual);
if (Hr != S_OK)
{
ErrorExit(g_pDbgClient, "Unable to get debuggee type, 0x%X\n", Hr);
}
// Set initial execution state.
if ((Hr = g_pDbgControl->GetExecutionStatus(&g_ExecStatus)) != S_OK)
{
ErrorExit(g_pDbgClient, "Unable to get execution status, 0x%X\n", Hr);
}
if (g_ExecStatus != DEBUG_STATUS_NO_DEBUGGEE)
{
// Session is already active.
SessionActive();
}
UpdateBufferWindows(UPDATE_EXEC_WINDOWS, UPDATE_EXEC);
if (g_RemoteClient)
{
// Request an initial read of everything.
InvalidateStateBuffers(BUFFERS_ALL);
ReadStateBuffers();
// The server may be in an input request, which
// we would have been notified of back during
// ConnectSession. If we're still in an input
// request switch to input mode.
if (g_InputStarted)
{
UpdateBufferWindows(1 << CMD_WINDOW, UPDATE_INPUT_REQUIRED);
}
}
for (;;)
{
if (!g_RemoteClient)
{
g_WaitingForEvent = TRUE;
Hr = g_pDbgControl->WaitForEvent(DEBUG_WAIT_DEFAULT, INFINITE);
g_WaitingForEvent = FALSE;
if (g_HoldWaitOutput)
{
UpdateUi();
}
if (FAILED(Hr))
{
// The debug session may have ended. If so,
// stop or end things based on how the session
// was started.
if (g_pDbgControl->GetExecutionStatus(&g_ExecStatus) == S_OK &&
g_ExecStatus == DEBUG_STATUS_NO_DEBUGGEE)
{
g_pDbgClient->EndSession(DEBUG_END_PASSIVE);
StopOrEndDebugging();
break;
}
// Inform the user of the failure and force
// command processing.
g_OutputCb.Output(DEBUG_OUTPUT_ERROR, "WaitForEvent failed\n");
g_ExecStatus = DEBUG_STATUS_BREAK;
}
if (g_TargetClass != DEBUG_CLASS_KERNEL ||
(g_TargetClassQual != DEBUG_DUMP_SMALL &&
g_TargetClassQual != DEBUG_DUMP_DEFAULT &&
g_TargetClassQual != DEBUG_DUMP_FULL))
{
g_pDbgControl->OutputCurrentState(DEBUG_OUTCTL_ALL_CLIENTS,
DEBUG_CURRENT_DEFAULT);
}
ReadStateBuffers();
}
while (!g_Exit &&
g_EndingSession == ENDING_NONE &&
(g_RemoteClient || g_ExecStatus == DEBUG_STATUS_BREAK))
{
if (!g_InputStarted)
{
// Tell the command window to display a prompt to
// indicate the engine is ready to process commands.
if (g_pDbgControl->GetPromptText(g_PromptText,
sizeof(g_PromptText),
NULL) != S_OK)
{
strcpy(g_PromptText, "?Err");
}
UpdateBufferWindows(1 << CMD_WINDOW, UPDATE_PROMPT_TEXT);
PostMessage(g_hwndFrame, WU_ENGINE_IDLE, 0, 0);
}
// Wait until engine processing is needed.
Hr = g_pDbgClient->DispatchCallbacks(INFINITE);
if (FAILED(Hr))
{
if (g_RemoteClient && HRESULT_FACILITY(Hr) == FACILITY_RPC)
{
// A remote client was unable to communicate
// with the server so shut down the session.
InformationBox(ERR_Client_Disconnect);
StopOrEndDebugging();
break;
}
else
{
// A failure here is a critical problem as
// something is seriously wrong with the engine
// if it can't do a normal DispatchCallbacks.
ErrorExit(g_pDbgClient,
"Engine thread wait failed, 0x%X\n", Hr);
}
}
if (!g_InputStarted)
{
// Take away the prompt while the engine is working.
g_PromptText[0] = 0;
UpdateBufferWindows(1 << CMD_WINDOW, UPDATE_PROMPT_TEXT);
}
ProcessEngineCommands(FALSE);
}
if (g_Exit)
{
g_EndingSession = ENDING_EXIT;
break;
}
if (g_EndingSession != ENDING_NONE)
{
// Force windows to display empty state.
InvalidateStateBuffers(BUFFERS_ALL);
UpdateBufferWindows(BUFFERS_ALL, UPDATE_BUFFER);
if (g_EndingSession == ENDING_RESTART)
{
if (StartSession() != S_OK)
{
// If we couldn't restart go into
// the stop-debugging state.
g_EndingSession = ENDING_STOP;
break;
}
g_EndingSession = ENDING_NONE;
}
else
{
break;
}
}
}
if (g_EndingSession == ENDING_NONE)
{
// Wake up the message pump for exit.
PostMessage(g_hwndFrame, WM_CLOSE, 0, 0);
}
ULONG Code;
if (!g_RemoteClient && g_DebugCommandLine != NULL)
{
// Return exit code of last process to exit.
Code = g_LastProcessExitCode;
}
else
{
Code = S_OK;
}
if (g_pDbgClient)
{
if (g_RemoteClient)
{
g_pDbgClient->EndSession(DEBUG_END_DISCONNECT);
}
else if (g_EndingSession != ENDING_STOP)
{
g_pDbgClient->EndSession(DEBUG_END_REENTRANT);
}
}
ReleaseEngineInterfaces();
g_EngineThreadId = 0;
if (g_EndingSession != ENDING_STOP)
{
//
// Wait for UI to finish up.
//
while (!g_Exit)
{
Sleep(50);
}
ExitDebugger(g_pDbgClient, Code);
}
else
{
g_EndingSession = ENDING_NONE;
}
return 0;
}
void
UpdateEngine(void)
{
if (g_pUiClient != NULL && g_pDbgClient != NULL)
{
g_pUiClient->ExitDispatch(g_pDbgClient);
}
}