Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

4959 lines
108 KiB

//----------------------------------------------------------------------------
//
// IDebugControl implementation.
//
// Copyright (C) Microsoft Corporation, 1999-2002.
//
//----------------------------------------------------------------------------
#include "ntsdp.hpp"
ULONG g_LogMask = DEFAULT_OUT_MASK;
STDMETHODIMP
DebugClient::GetInterrupt(
THIS
)
{
// This method is reentrant.
return CheckUserInterrupt() ? S_OK : S_FALSE;
}
STDMETHODIMP
DebugClient::SetInterrupt(
THIS_
IN ULONG Flags
)
{
// This method is reentrant.
if (
#if DEBUG_INTERRUPT_ACTIVE > 0
Flags < DEBUG_INTERRUPT_ACTIVE ||
#endif
Flags > DEBUG_INTERRUPT_EXIT)
{
return E_INVALIDARG;
}
if (!g_Target)
{
return E_UNEXPECTED;
}
// If the debuggee isn't currently running
// we just need to set the operation-interrupt
// flag. If this is a passive interrupt that's
// all that's ever done.
if (Flags == DEBUG_INTERRUPT_PASSIVE ||
(IS_MACHINE_SET(g_Target) && g_CmdState == 'c'))
{
g_EngStatus |= ENG_STATUS_USER_INTERRUPT;
return S_OK;
}
// If this is an exit interrupt we don't want
// to actually interrupt the running debuggee,
// we just want to terminate the current wait.
if (Flags == DEBUG_INTERRUPT_EXIT)
{
g_EngStatus |= ENG_STATUS_EXIT_CURRENT_WAIT;
return S_OK;
}
//
// Force a break-in. Don't set user-interrupt in
// this case as that's just a marker for
// interrupting commands. Setting it can
// interfere with break-in processing since commands
// executed during break-in may be affected by it.
//
HRESULT Status = g_Target->RequestBreakIn();
if (Status == S_OK)
{
g_EngStatus |= ENG_STATUS_PENDING_BREAK_IN;
}
return Status;
}
STDMETHODIMP
DebugClient::GetInterruptTimeout(
THIS_
OUT PULONG Seconds
)
{
ENTER_ENGINE();
*Seconds = g_PendingBreakInTimeoutLimit;
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::SetInterruptTimeout(
THIS_
IN ULONG Seconds
)
{
if (Seconds < 1)
{
return E_INVALIDARG;
}
ENTER_ENGINE();
g_PendingBreakInTimeoutLimit = Seconds;
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::GetLogFile(
THIS_
OUT OPTIONAL PSTR Buffer,
IN ULONG BufferSize,
OUT OPTIONAL PULONG FileSize,
OUT PBOOL Append
)
{
HRESULT Status;
ENTER_ENGINE();
if (!g_OpenLogFileName[0])
{
Status = E_NOINTERFACE;
*Append = FALSE;
}
else
{
Status = FillStringBuffer(g_OpenLogFileName, 0,
Buffer, BufferSize, FileSize);
*Append = g_OpenLogFileAppended;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::OpenLogFile(
THIS_
IN PCSTR File,
IN BOOL Append
)
{
ENTER_ENGINE();
::OpenLogFile(File, Append);
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::CloseLogFile(
THIS
)
{
ENTER_ENGINE();
::CloseLogFile();
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::GetLogMask(
THIS_
OUT PULONG Mask
)
{
// This method is reentrant.
*Mask = g_LogMask;
return S_OK;
}
STDMETHODIMP
DebugClient::SetLogMask(
THIS_
IN ULONG Mask
)
{
// This method is reentrant.
g_LogMask = Mask;
return S_OK;
}
STDMETHODIMP
DebugClient::Input(
THIS_
OUT PSTR Buffer,
IN ULONG BufferSize,
OUT OPTIONAL PULONG InputSize
)
{
HRESULT Status;
if (BufferSize < 2)
{
// Must have space for at least a character and a terminator.
return E_INVALIDARG;
}
ENTER_ENGINE();
ULONG Size;
//
// XXX drewb.
// In condbg we needed a way to see how many clients
// were available for input. Rather than define a new
// interface and method just for this purpose we've added
// this hack where you pass in a magic combination of parameters.
// If a new IDebugControl ever gets defined this should be
// formalized and the hack removed.
//
if (!Buffer && BufferSize == DEBUG_ANY_ID)
{
DebugClient* Client;
Size = 0;
for (Client = g_Clients; Client != NULL; Client = Client->m_Next)
{
if (Client->m_InputCb != NULL)
{
Size++;
}
}
Status = S_OK;
}
else
{
Size = GetInput(NULL, Buffer, BufferSize, GETIN_DEFAULT);
if (Size == 0)
{
Status = E_FAIL;
goto Exit;
}
Status = Size > BufferSize ? S_FALSE : S_OK;
}
if (InputSize != NULL)
{
*InputSize = Size;
}
Exit:
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::ReturnInput(
THIS_
IN PCSTR Buffer
)
{
// This method is reentrant.
HRESULT Status;
ULONG Seq = (ULONG)InterlockedIncrement((PLONG)&g_InputSequence);
if (Seq == m_InputSequence)
{
ULONG CopyLen = strlen(Buffer) + 1;
CopyLen = min(CopyLen, INPUT_BUFFER_SIZE);
memcpy(g_InputBuffer, Buffer, CopyLen);
g_InputBuffer[INPUT_BUFFER_SIZE - 1] = 0;
SetEvent(g_InputEvent);
Status = S_OK;
}
else
{
Status = S_FALSE;
}
m_InputSequence = 0xffffffff;
return Status;
}
STDMETHODIMPV
DebugClient::Output(
THIS_
IN ULONG Mask,
IN PCSTR Format,
...
)
{
ENTER_ENGINE();
va_list Args;
va_start(Args, Format);
MaskOutVa(Mask, Format, Args, TRUE);
va_end(Args);
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::OutputVaList(
THIS_
IN ULONG Mask,
IN PCSTR Format,
IN va_list Args
)
{
ENTER_ENGINE();
MaskOutVa(Mask, Format, Args, TRUE);
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMPV
DebugClient::ControlledOutput(
THIS_
IN ULONG OutputControl,
IN ULONG Mask,
IN PCSTR Format,
...
)
{
HRESULT Status;
ENTER_ENGINE();
OutCtlSave OldCtl;
if (!PushOutCtl(OutputControl, this, &OldCtl))
{
Status = E_INVALIDARG;
}
else
{
va_list Args;
va_start(Args, Format);
MaskOutVa(Mask, Format, Args, TRUE);
va_end(Args);
Status = S_OK;
PopOutCtl(&OldCtl);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::ControlledOutputVaList(
THIS_
IN ULONG OutputControl,
IN ULONG Mask,
IN PCSTR Format,
IN va_list Args
)
{
HRESULT Status;
ENTER_ENGINE();
OutCtlSave OldCtl;
if (!PushOutCtl(OutputControl, this, &OldCtl))
{
Status = E_INVALIDARG;
}
else
{
MaskOutVa(Mask, Format, Args, TRUE);
Status = S_OK;
PopOutCtl(&OldCtl);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMPV
DebugClient::OutputPrompt(
THIS_
IN ULONG OutputControl,
IN OPTIONAL PCSTR Format,
...
)
{
HRESULT Status;
ENTER_ENGINE();
OutCtlSave OldCtl;
if (!PushOutCtl(OutputControl, this, &OldCtl))
{
Status = E_INVALIDARG;
}
else
{
va_list Args;
va_start(Args, Format);
::OutputPrompt(Format, Args);
va_end(Args);
Status = S_OK;
PopOutCtl(&OldCtl);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::OutputPromptVaList(
THIS_
IN ULONG OutputControl,
IN OPTIONAL PCSTR Format,
IN va_list Args
)
{
HRESULT Status;
ENTER_ENGINE();
OutCtlSave OldCtl;
if (!PushOutCtl(OutputControl, this, &OldCtl))
{
Status = E_INVALIDARG;
}
else
{
::OutputPrompt(Format, Args);
Status = S_OK;
PopOutCtl(&OldCtl);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetPromptText(
THIS_
OUT OPTIONAL PSTR Buffer,
IN ULONG BufferSize,
OUT OPTIONAL PULONG TextSize
)
{
HRESULT Status;
ENTER_ENGINE();
Status = ::GetPromptText(Buffer, BufferSize, TextSize);
LEAVE_ENGINE();
return Status;
}
#define CURRENT_ALL DEBUG_CURRENT_DEFAULT
STDMETHODIMP
DebugClient::OutputCurrentState(
THIS_
IN ULONG OutputControl,
IN ULONG Flags
)
{
if (Flags & ~CURRENT_ALL)
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
OutCtlSave OldCtl;
if (!IS_CUR_MACHINE_ACCESSIBLE())
{
Status = E_UNEXPECTED;
}
else if (!PushOutCtl(OutputControl, this, &OldCtl))
{
Status = E_INVALIDARG;
}
else
{
ULONG Oci = 0;
if (Flags & DEBUG_CURRENT_SYMBOL)
{
Oci |= OCI_SYMBOL;
}
if (Flags & DEBUG_CURRENT_DISASM)
{
Oci |= OCI_DISASM | OCI_ALLOW_EA;
}
if (Flags & DEBUG_CURRENT_REGISTERS)
{
Oci |= OCI_ALLOW_REG;
}
if (Flags & DEBUG_CURRENT_SOURCE_LINE)
{
Oci |= OCI_ALLOW_SOURCE;
}
OutCurInfo(Oci, g_Machine->m_AllMask, DEBUG_OUTPUT_PROMPT_REGISTERS);
Status = S_OK;
PopOutCtl(&OldCtl);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::OutputVersionInformation(
THIS_
IN ULONG OutputControl
)
{
HRESULT Status;
// This method is reentrant. It uses many pieces of
// global information, though, so try and get
// the engine lock.
Status = TRY_ENTER_ENGINE();
if (Status != S_OK)
{
return Status;
}
OutCtlSave OldCtl;
if (!PushOutCtl(OutputControl, this, &OldCtl))
{
return E_INVALIDARG;
}
::OutputVersionInformation(this);
PopOutCtl(&OldCtl);
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::GetNotifyEventHandle(
THIS_
OUT PULONG64 Handle
)
{
ENTER_ENGINE();
*Handle = (ULONG64)g_EventToSignal;
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::SetNotifyEventHandle(
THIS_
IN ULONG64 Handle
)
{
if ((ULONG64)(HANDLE)Handle != Handle)
{
return E_INVALIDARG;
}
ENTER_ENGINE();
g_EventToSignal = (HANDLE)Handle;
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::Assemble(
THIS_
IN ULONG64 Offset,
IN PCSTR Instr,
OUT PULONG64 EndOffset
)
{
HRESULT Status;
ENTER_ENGINE();
__try
{
if (!IS_CUR_CONTEXT_ACCESSIBLE())
{
Status = E_UNEXPECTED;
__leave;
}
ADDR Addr;
// Assume this is a code segment address so that assembly
// picks up the appropriate type of address.
g_Machine->FormAddr(SEGREG_CODE, Offset,
FORM_SEGREG | FORM_CODE, &Addr);
g_Machine->Assemble(g_Process, &Addr, (PSTR)Instr);
*EndOffset = Flat(Addr);
Status = S_OK;
}
__except(CommandExceptionFilter(GetExceptionInformation()))
{
Status = E_FAIL;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::Disassemble(
THIS_
IN ULONG64 Offset,
IN ULONG Flags,
OUT OPTIONAL PSTR Buffer,
IN ULONG BufferSize,
OUT OPTIONAL PULONG DisassemblySize,
OUT PULONG64 EndOffset
)
{
if (Flags & ~DEBUG_DISASM_EFFECTIVE_ADDRESS)
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
if (!IS_CUR_CONTEXT_ACCESSIBLE())
{
Status = E_UNEXPECTED;
}
else
{
ADDR Addr;
CHAR Disasm[MAX_DISASM_LEN];
// Assume this is a code segment address so that disassembly
// picks up the appropriate type of address.
g_Machine->FormAddr(SEGREG_CODE, Offset, FORM_SEGREG | FORM_CODE,
&Addr);
g_Machine->Disassemble(g_Process, &Addr, Disasm,
(Flags & DEBUG_DISASM_EFFECTIVE_ADDRESS));
Status = FillStringBuffer(Disasm, 0,
Buffer, BufferSize, DisassemblySize);
*EndOffset = Flat(Addr);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetDisassembleEffectiveOffset(
THIS_
OUT PULONG64 Offset
)
{
HRESULT Status;
ENTER_ENGINE();
if (!IS_CUR_CONTEXT_ACCESSIBLE())
{
Status = E_UNEXPECTED;
}
else
{
ADDR Addr;
ULONG Size;
g_Machine->GetEffectiveAddr(&Addr, &Size);
*Offset = Flat(Addr);
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
HRESULT
DoOutputDisassembly(PADDR Addr, ULONG Flags,
ULONG SkipLines, ULONG OutputLines,
PULONG LineCount)
{
ULONG Lines = 0;
CHAR Buffer[MAX_DISASM_LEN];
PCHAR FirstLine;
HRESULT Status;
if (Flags & DEBUG_DISASM_MATCHING_SYMBOLS)
{
ULONG64 Disp;
GetSymbol(Flat(*Addr), Buffer, sizeof(Buffer), &Disp);
if (Disp == 0)
{
if (OutputLines > 0)
{
if (SkipLines == 0)
{
StartOutLine(DEBUG_OUTPUT_NORMAL, OUT_LINE_NO_TIMESTAMP);
dprintf("%s:\n", Buffer);
OutputLines--;
}
else
{
SkipLines--;
}
}
Lines++;
}
}
FirstLine = Buffer;
if (!g_Machine->
Disassemble(g_Process, Addr, Buffer,
(Flags & DEBUG_DISASM_EFFECTIVE_ADDRESS) != 0))
{
// Return S_FALSE if the disassembly failed.
// Output will still be produced, such as "???".
// Update the address to the next potential instruction
// locations so that callers that are satisfied with "???"
// can just iterate.
g_Machine->IncrementBySmallestInstruction(Addr);
Lines++;
Status = S_FALSE;
if (SkipLines > 0 || OutputLines == 0)
{
*FirstLine = 0;
}
}
else
{
PSTR Nl = Buffer;
PSTR LastLine = Nl;
// Count lines in output and determine line positions.
while (*Nl)
{
Nl = strchr(Nl, '\n');
DBG_ASSERT(Nl != NULL);
Lines++;
Nl++;
if (SkipLines > 0)
{
FirstLine = Nl;
SkipLines--;
}
if (OutputLines > 0)
{
LastLine = Nl;
OutputLines--;
}
}
*LastLine = 0;
Status = S_OK;
}
if (*FirstLine)
{
StartOutLine(DEBUG_OUTPUT_NORMAL, OUT_LINE_NO_TIMESTAMP);
dprintf("%s", FirstLine);
}
if (LineCount != NULL)
{
*LineCount = Lines;
}
return Status;
}
#define ALL_DISASM_FLAGS \
(DEBUG_DISASM_EFFECTIVE_ADDRESS | DEBUG_DISASM_MATCHING_SYMBOLS)
STDMETHODIMP
DebugClient::OutputDisassembly(
THIS_
IN ULONG OutputControl,
IN ULONG64 Offset,
IN ULONG Flags,
OUT PULONG64 EndOffset
)
{
if (Flags & ~ALL_DISASM_FLAGS)
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
if (!IS_CUR_CONTEXT_ACCESSIBLE())
{
Status = E_UNEXPECTED;
goto Exit;
}
OutCtlSave OldCtl;
if (!PushOutCtl(OutputControl, this, &OldCtl))
{
Status = E_INVALIDARG;
goto Exit;
}
ADDR Addr;
// Assume this is a code segment address so that disassembly
// picks up the appropriate type of address.
g_Machine->FormAddr(SEGREG_CODE, Offset, FORM_SEGREG | FORM_CODE, &Addr);
Status = DoOutputDisassembly(&Addr, Flags, 0, 0xffffffff, NULL);
*EndOffset = Flat(Addr);
PopOutCtl(&OldCtl);
Exit:
LEAVE_ENGINE();
return Status;
}
ULONG
BackUpDisassemblyLines(ULONG Lines, PADDR Addr, ULONG Flags, PADDR PcAddr)
{
//
// There's no easy way to predict how many lines of
// output a particular disassembly will take so
// just iteratively back up by the minimum amount until
// the appropriate number of lines is reached.
//
ADDR BackAddr = *Addr;
// Limit things so that failure can be detected.
// Right now X86's maximum instruction length of 16
// is big enough for all platforms so use that.
ADDR LimitAddr = *Addr;
ULONG BackBytes = X86_MAX_INSTRUCTION_LEN * Lines;
if (BackBytes > LimitAddr.off)
{
LimitAddr.off = 0;
}
else
{
AddrSub(&LimitAddr, BackBytes);
}
ADDR TryAddr;
ULONG TryLines;
//
// Reverse disassembly is difficult on x86 due
// to the variable length instructions. First
// just locate the nearest symbol and disassemble
// from that since this has a better chance of
// producing a valid disassembly.
//
CHAR Buffer[MAX_DISASM_LEN];
ULONG64 Disp;
ADDR DisAddr, StartAddr;
GetSymbol(Flat(LimitAddr), Buffer, sizeof(Buffer), &Disp);
ADDRFLAT(&DisAddr, Disp);
if (!AddrEqu(LimitAddr, DisAddr) &&
Disp <= 16 * X86_MAX_INSTRUCTION_LEN) // valid symbol
{
BOOL DoOneMore = FALSE;
StartAddr = LimitAddr;
AddrSub(&StartAddr, Disp);
TryAddr = StartAddr;
TryLines = 0;
while (1)
{
ULONG DisLines;
ULONG DisFlags;
while (AddrLt(TryAddr, *Addr) && ((TryLines < Lines) || DoOneMore))
{
UCHAR MemTest;
// If we can't read memory at this address there's
// no chance of getting a valid disassembly so
// just stop the whole process.
if (fnotFlat(TryAddr) ||
g_Target->
ReadAllVirtual(g_Process, Flat(TryAddr),
&MemTest, sizeof(MemTest)) != S_OK)
{
TryAddr = *Addr;
break;
}
DisFlags = Flags;
if (!AddrEqu(TryAddr, *PcAddr))
{
DisFlags &= ~DEBUG_DISASM_EFFECTIVE_ADDRESS;
}
DoOutputDisassembly(&TryAddr, DisFlags, 0, 0, &DisLines);
TryLines += DisLines;
DoOneMore = FALSE;
}
if (TryLines >= Lines && AddrEqu(TryAddr, *Addr))
{
*Addr = StartAddr;
return TryLines;
}
else if (AddrLt(TryAddr, *Addr))
{
DisAddr = StartAddr;
// Increase StartAddr
DisFlags = Flags;
if (!AddrEqu(StartAddr, *PcAddr))
{
DisFlags &= ~DEBUG_DISASM_EFFECTIVE_ADDRESS;
}
DoOutputDisassembly(&StartAddr, DisFlags, 0, 0, &DisLines);
if ((DisLines == 1) || ((TryLines - DisLines) >= (Lines - 1)))
{
TryLines -= DisLines;
}
else
{
StartAddr = DisAddr;
DoOneMore = TRUE;
}
}
else
{
// couldn't find it
break;
}
}
}
//
// If we couldn't do something with symbols just
// try a brute-force search backwards. This
// has limited utility on variable-length instruction
// sets but sometimes it works.
//
while (AddrGt(BackAddr, LimitAddr))
{
g_Machine->DecrementBySmallestInstruction(&BackAddr);
TryAddr = BackAddr;
TryLines = 0;
while (AddrLt(TryAddr, *Addr))
{
UCHAR MemTest;
// If we can't read memory at this address there's
// no chance of getting a valid disassembly so
// just stop the whole process.
if (fnotFlat(TryAddr) ||
g_Target->
ReadAllVirtual(g_Process, Flat(TryAddr),
&MemTest, sizeof(MemTest)) != S_OK)
{
BackAddr = LimitAddr;
break;
}
ULONG DisLines;
ULONG DisFlags = Flags;
if (!AddrEqu(TryAddr, *PcAddr))
{
DisFlags &= ~DEBUG_DISASM_EFFECTIVE_ADDRESS;
}
DoOutputDisassembly(&TryAddr, DisFlags, 0, 0, &DisLines);
TryLines += DisLines;
}
if (TryLines >= Lines && AddrEqu(TryAddr, *Addr))
{
*Addr = BackAddr;
return TryLines;
}
}
// Couldn't find a disassembly that worked.
return 0;
}
STDMETHODIMP
DebugClient::OutputDisassemblyLines(
THIS_
IN ULONG OutputControl,
IN ULONG PreviousLines,
IN ULONG TotalLines,
IN ULONG64 Offset,
IN ULONG Flags,
OUT OPTIONAL PULONG OffsetLine,
OUT OPTIONAL PULONG64 StartOffset,
OUT OPTIONAL PULONG64 EndOffset,
OUT OPTIONAL /* size_is(TotalLines) */ PULONG64 LineOffsets
)
{
if ((Flags & ~ALL_DISASM_FLAGS) ||
TotalLines < 1 || PreviousLines > TotalLines)
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
if (!IS_CUR_CONTEXT_ACCESSIBLE())
{
Status = E_UNEXPECTED;
goto Exit;
}
OutCtlSave OldCtl;
if (!PushOutCtl(OutputControl, this, &OldCtl))
{
Status = E_INVALIDARG;
goto Exit;
}
ULONG i;
if (LineOffsets != NULL)
{
for (i = 0; i < TotalLines; i++)
{
LineOffsets[i] = DEBUG_INVALID_OFFSET;
}
}
ULONG Line, Lines, SkipLines;
ADDR Addr, PcAddr;
// Assume this is a code segment address so that disassembly
// picks up the appropriate type of address.
g_Machine->FormAddr(SEGREG_CODE, Offset, FORM_SEGREG | FORM_CODE, &Addr);
g_Machine->GetPC(&PcAddr);
Line = 0;
SkipLines = 0;
if (PreviousLines > 0)
{
Lines = BackUpDisassemblyLines(PreviousLines, &Addr, Flags, &PcAddr);
if (Lines == 0)
{
dprintf("No prior disassembly possible\n");
Line = 1;
Lines = 1;
TotalLines--;
}
else if (Lines > PreviousLines)
{
SkipLines = Lines - PreviousLines;
Lines = PreviousLines;
}
}
else
{
Lines = 0;
}
if (OffsetLine != NULL)
{
*OffsetLine = Lines;
}
if (StartOffset != NULL)
{
*StartOffset = Flat(Addr);
}
while (TotalLines > 0)
{
if (LineOffsets != NULL)
{
LineOffsets[Line] = Flat(Addr);
}
ULONG DisFlags = Flags;
if (!AddrEqu(Addr, PcAddr))
{
DisFlags &= ~DEBUG_DISASM_EFFECTIVE_ADDRESS;
}
DoOutputDisassembly(&Addr, DisFlags, SkipLines, TotalLines, &Lines);
Lines -= SkipLines;
if (TotalLines <= Lines)
{
break;
}
TotalLines -= Lines;
Line += Lines;
SkipLines = 0;
}
if (EndOffset != NULL)
{
*EndOffset = Flat(Addr);
}
Status = S_OK;
PopOutCtl(&OldCtl);
Exit:
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetNearInstruction(
THIS_
IN ULONG64 Offset,
IN LONG Delta,
OUT PULONG64 NearOffset
)
{
HRESULT Status;
ENTER_ENGINE();
if (!IS_CUR_MACHINE_ACCESSIBLE())
{
Status = E_UNEXPECTED;
goto Exit;
}
Status = S_OK;
switch(g_Target->m_EffMachineType)
{
case IMAGE_FILE_MACHINE_ARM:
// Each instruction is 32 bits.
Offset += (LONG64)Delta * 4;
break;
case IMAGE_FILE_MACHINE_IA64:
ULONG Instr;
// Each 128-bit bundle has three instructions.
if (Delta < 0)
{
while (Delta++ < 0)
{
Instr = (ULONG)(Offset & 0xf);
if (Instr == 0)
{
Offset -= 8;
}
else
{
Offset -= 4;
}
}
}
else
{
while (Delta-- > 0)
{
Instr = (ULONG)(Offset & 0xf);
if (Instr == 8)
{
Offset += 8;
}
else
{
Offset += 4;
}
}
}
break;
case IMAGE_FILE_MACHINE_I386:
case IMAGE_FILE_MACHINE_AMD64:
ADDR Addr;
CHAR Buffer[MAX_DISASM_LEN];
// Instructions are highly variable. There isn't any
// way to really know whether a particular disassembly
// of a stretch of code is valid or not, so this
// routine is inherently fragile.
g_Machine->FormAddr(SEGREG_CODE, Offset,
FORM_SEGREG | FORM_CODE, &Addr);
if (Delta < 0)
{
// Back up byte-by-byte and disassemble. If the
// post-disassembly offset matches the current offset,
// a good-enough instruction sequence has been found.
for (;;)
{
ADDR TryAddr;
LONG TryDelta;
AddrSub(&Addr, 1);
TryAddr = Addr;
TryDelta = 0;
while (Flat(TryAddr) < Offset)
{
if (!g_Machine->
Disassemble(g_Process,&TryAddr, Buffer, FALSE))
{
break;
}
TryDelta--;
}
if (Flat(TryAddr) == Offset &&
TryDelta == Delta)
{
break;
}
// Limit things just as a precaution.
if (Flat(Addr) < Offset + Delta * X86_MAX_INSTRUCTION_LEN)
{
Status = E_FAIL;
break;
}
}
}
else
{
while (Delta-- > 0)
{
if (!g_Machine->Disassemble(g_Process, &Addr, Buffer, FALSE))
{
Status = E_FAIL;
break;
}
}
}
Offset = Flat(Addr);
break;
default:
Status = E_UNEXPECTED;
break;
}
if (SUCCEEDED(Status))
{
*NearOffset = Offset;
}
Exit:
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetStackTrace(
THIS_
IN ULONG64 FrameOffset,
IN ULONG64 StackOffset,
IN ULONG64 InstructionOffset,
OUT PDEBUG_STACK_FRAME Frames,
IN ULONG FramesSize,
OUT OPTIONAL PULONG FramesFilled
)
{
HRESULT Status;
ENTER_ENGINE();
if (!IS_CUR_CONTEXT_ACCESSIBLE())
{
Status = E_UNEXPECTED;
goto Exit;
}
ULONG FramesRet;
ULONG64 ThreadData;
if (g_ExtThread != 0)
{
ThreadData = g_ExtThread;
g_ExtThread = 0;
}
else
{
ThreadData = 0;
}
ULONG PtrDef =
(!InstructionOffset ? STACK_INSTR_DEFAULT : 0) |
(!StackOffset ? STACK_STACK_DEFAULT : 0) |
(!FrameOffset ? STACK_FRAME_DEFAULT : 0);
FramesRet = StackTrace(this,
FrameOffset, StackOffset, InstructionOffset,
PtrDef, Frames, FramesSize, ThreadData, 0, FALSE);
if (FramesRet > 0)
{
Status = S_OK;
if (FramesFilled != NULL)
{
*FramesFilled = FramesRet;
}
}
else
{
Status = E_FAIL;
}
if (g_ExtThreadScopeSaved)
{
PopScope(&g_ExtThreadSavedScope);
g_ExtThreadScopeSaved = FALSE;
}
Exit:
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetReturnOffset(
THIS_
OUT PULONG64 Offset
)
{
HRESULT Status;
ENTER_ENGINE();
if (!IS_CUR_CONTEXT_ACCESSIBLE())
{
Status = E_UNEXPECTED;
}
else
{
ADDR Addr;
g_Machine->GetRetAddr(&Addr);
*Offset = Flat(Addr);
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::OutputStackTrace(
THIS_
IN ULONG OutputControl,
IN PDEBUG_STACK_FRAME Frames,
IN ULONG FramesSize,
IN ULONG Flags
)
{
if (Flags & ~(DEBUG_STACK_ARGUMENTS |
DEBUG_STACK_FUNCTION_INFO |
DEBUG_STACK_SOURCE_LINE |
DEBUG_STACK_FRAME_ADDRESSES |
DEBUG_STACK_COLUMN_NAMES |
DEBUG_STACK_NONVOLATILE_REGISTERS |
DEBUG_STACK_FRAME_NUMBERS |
DEBUG_STACK_PARAMETERS |
DEBUG_STACK_FRAME_ADDRESSES_RA_ONLY |
DEBUG_STACK_FRAME_MEMORY_USAGE))
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
if (!IS_CUR_CONTEXT_ACCESSIBLE())
{
Status = E_UNEXPECTED;
goto Exit;
}
OutCtlSave OldCtl;
if (!PushOutCtl(OutputControl, this, &OldCtl))
{
Status = E_INVALIDARG;
goto Exit;
}
// Currently only IA64 supports nonvolatile register output.
if (g_Target->m_EffMachineType != IMAGE_FILE_MACHINE_IA64)
{
Flags &= ~DEBUG_STACK_NONVOLATILE_REGISTERS;
}
PDEBUG_STACK_FRAME LocalFrames;
LocalFrames = NULL;
if (Frames == NULL)
{
ULONG FramesFilled;
ULONG64 ThreadData;
if (g_ExtThread != 0)
{
ThreadData = g_ExtThread;
g_ExtThread = 0;
}
else
{
ThreadData = 0;
}
LocalFrames = new DEBUG_STACK_FRAME[FramesSize];
if (LocalFrames == NULL)
{
ErrOut("Unable to allocate memory for stack trace\n");
Status = E_OUTOFMEMORY;
goto PopExit;
}
//
// StackTrace will generate output if any flags are
// passed in. The only time we really require that
// it produce output is when nonvolatile registers
// are requested as they can only be displayed when
// the context is available during stack walking.
// In order to simplify later logic, we only
// pass flags if we have the nonvolatile register flag.
//
FramesFilled = StackTrace(this,
0, 0, 0, STACK_ALL_DEFAULT,
LocalFrames, FramesSize, ThreadData,
(Flags & DEBUG_STACK_NONVOLATILE_REGISTERS) ?
Flags : 0, FALSE);
if (FramesFilled == 0)
{
delete [] LocalFrames;
goto PopExit;
}
Frames = LocalFrames;
FramesSize = FramesFilled;
}
else if (Flags & DEBUG_STACK_NONVOLATILE_REGISTERS)
{
// Can't dump nonvolatile registers without a full
// context so this is not an allowable options.
Status = E_INVALIDARG;
goto PopExit;
}
if (!(Flags & DEBUG_STACK_NONVOLATILE_REGISTERS))
{
PrintStackTrace(FramesSize, Frames, Flags);
}
Status = S_OK;
delete [] LocalFrames;
if (g_ExtThreadScopeSaved)
{
PopScope(&g_ExtThreadSavedScope);
g_ExtThreadScopeSaved = FALSE;
}
PopExit:
PopOutCtl(&OldCtl);
Exit:
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetDebuggeeType(
THIS_
OUT PULONG Class,
OUT PULONG Qualifier
)
{
HRESULT Status;
ENTER_ENGINE();
if (!g_Target)
{
*Class = DEBUG_CLASS_UNINITIALIZED;
*Qualifier = 0;
Status = S_OK;
}
else
{
*Class = g_Target->m_Class;
*Qualifier = g_Target->m_ClassQualifier;
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetActualProcessorType(
THIS_
OUT PULONG Type
)
{
HRESULT Status;
ENTER_ENGINE();
if (!IS_MACHINE_SET(g_Target))
{
Status = E_UNEXPECTED;
}
else
{
*Type = g_Target->m_MachineType;
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetExecutingProcessorType(
THIS_
OUT PULONG Type
)
{
HRESULT Status;
ENTER_ENGINE();
if (!g_EventMachine)
{
Status = E_UNEXPECTED;
}
else
{
*Type = g_EventMachine->m_ExecTypes[0];
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetNumberPossibleExecutingProcessorTypes(
THIS_
OUT PULONG Number
)
{
HRESULT Status;
ENTER_ENGINE();
if (!IS_MACHINE_SET(g_Target))
{
Status = E_UNEXPECTED;
}
else
{
MachineInfo* Machine =
MachineTypeInfo(g_Target, g_Target->m_MachineType);
if (Machine == NULL)
{
Status = E_INVALIDARG;
}
else
{
*Number = Machine->m_NumExecTypes;
Status = S_OK;
}
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetPossibleExecutingProcessorTypes(
THIS_
IN ULONG Start,
IN ULONG Count,
OUT PULONG Types
)
{
HRESULT Status;
ENTER_ENGINE();
if (!IS_MACHINE_SET(g_Target))
{
Status = E_UNEXPECTED;
}
else
{
MachineInfo* Machine =
MachineTypeInfo(g_Target, g_Target->m_MachineType);
if (Machine == NULL ||
Start >= Machine->m_NumExecTypes ||
Start + Count > Machine->m_NumExecTypes)
{
Status = E_INVALIDARG;
}
else
{
Status = S_OK;
while (Count-- > 0)
{
*Types++ = Machine->m_ExecTypes[Start++];
}
}
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetNumberProcessors(
THIS_
OUT PULONG Number
)
{
HRESULT Status;
ENTER_ENGINE();
if (!IS_MACHINE_SET(g_Target))
{
Status = E_UNEXPECTED;
}
else
{
*Number = g_Target->m_NumProcessors;
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetSystemVersion(
THIS_
OUT PULONG PlatformId,
OUT PULONG Major,
OUT PULONG Minor,
OUT OPTIONAL PSTR ServicePackString,
IN ULONG ServicePackStringSize,
OUT OPTIONAL PULONG ServicePackStringUsed,
OUT PULONG ServicePackNumber,
OUT OPTIONAL PSTR BuildString,
IN ULONG BuildStringSize,
OUT OPTIONAL PULONG BuildStringUsed
)
{
HRESULT Status;
ENTER_ENGINE();
// This is insufficient to distinguish
// the various system types supported but we don't
// want to publish identifiers for every possible
// system version family. PlatformId is as good
// as it gets.
if (!IS_MACHINE_SET(g_Target))
{
Status = E_UNEXPECTED;
}
else
{
*PlatformId = g_Target->m_PlatformId;
*Major = g_Target->m_CheckedBuild;
*Minor = g_Target->m_BuildNumber;
Status = FillStringBuffer(g_Target->m_ServicePackString, 0,
ServicePackString, ServicePackStringSize,
ServicePackStringUsed);
*ServicePackNumber = g_Target->m_ServicePackNumber;
if (FillStringBuffer(g_Target->m_BuildLabName, 0,
BuildString, BuildStringSize,
BuildStringUsed) == S_FALSE)
{
Status = S_FALSE;
}
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetPageSize(
THIS_
OUT PULONG Size
)
{
HRESULT Status;
ENTER_ENGINE();
if (!g_Machine)
{
Status = E_UNEXPECTED;
}
else
{
*Size = g_Machine->m_PageSize;
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::IsPointer64Bit(
THIS
)
{
HRESULT Status;
ENTER_ENGINE();
if (!g_Machine)
{
Status = E_UNEXPECTED;
}
else
{
Status = g_Machine->m_Ptr64 ? S_OK : S_FALSE;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::ReadBugCheckData(
THIS_
OUT PULONG Code,
OUT PULONG64 Arg1,
OUT PULONG64 Arg2,
OUT PULONG64 Arg3,
OUT PULONG64 Arg4
)
{
HRESULT Status;
ENTER_ENGINE();
if (!IS_KERNEL_TARGET(g_Target))
{
Status = E_UNEXPECTED;
}
else
{
ULONG64 Args[4];
Status = g_Target->ReadBugCheckData(Code, Args);
if (Status == S_OK)
{
*Arg1 = Args[0];
*Arg2 = Args[1];
*Arg3 = Args[2];
*Arg4 = Args[3];
}
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetNumberSupportedProcessorTypes(
THIS_
OUT PULONG Number
)
{
ENTER_ENGINE();
*Number = MACHIDX_COUNT;
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::GetSupportedProcessorTypes(
THIS_
IN ULONG Start,
IN ULONG Count,
OUT PULONG Types
)
{
if (Start >= MACHIDX_COUNT ||
Start + Count > MACHIDX_COUNT)
{
return E_INVALIDARG;
}
ENTER_ENGINE();
if (Count > 0)
{
while (Count-- > 0)
{
// First ExecTypes entry is the actual processor
// type so it's a convenient place to turn an
// index into a type.
*Types++ = g_PossibleProcessorTypes[Start++];
}
}
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::GetProcessorTypeNames(
THIS_
IN ULONG Type,
OUT OPTIONAL PSTR FullNameBuffer,
IN ULONG FullNameBufferSize,
OUT OPTIONAL PULONG FullNameSize,
OUT OPTIONAL PSTR AbbrevNameBuffer,
IN ULONG AbbrevNameBufferSize,
OUT OPTIONAL PULONG AbbrevNameSize
)
{
HRESULT Status;
ENTER_ENGINE();
MachineInfo* Machine = MachineTypeInfo(g_Target, Type);
if (Machine == NULL)
{
Status = E_INVALIDARG;
}
else
{
Status = FillStringBuffer(Machine->m_FullName, 0,
FullNameBuffer, FullNameBufferSize,
FullNameSize);
if (FillStringBuffer(Machine->m_AbbrevName, 0,
AbbrevNameBuffer, AbbrevNameBufferSize,
AbbrevNameSize) == S_FALSE)
{
Status = S_FALSE;
}
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetEffectiveProcessorType(
THIS_
OUT PULONG Type
)
{
HRESULT Status;
ENTER_ENGINE();
if (!g_Target)
{
Status = E_UNEXPECTED;
}
else
{
*Type = g_Target->m_EffMachineType;
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::SetEffectiveProcessorType(
THIS_
IN ULONG Type
)
{
HRESULT Status;
ENTER_ENGINE();
if (!g_Target)
{
Status = E_UNEXPECTED;
}
else
{
MachineIndex Index = MachineTypeIndex(Type);
if (Index == MACHIDX_COUNT)
{
Status = E_INVALIDARG;
}
else
{
g_Target->SetEffMachine(Type, TRUE);
Status = S_OK;
}
}
LEAVE_ENGINE();
return Status;
}
ULONG
GetExecutionStatus(void)
{
if ((g_EngStatus & ENG_STATUS_STOP_SESSION) ||
!AnyActiveProcesses(TRUE))
{
return DEBUG_STATUS_NO_DEBUGGEE;
}
else if (g_CmdState == 'p')
{
return DEBUG_STATUS_STEP_OVER;
}
else if (g_CmdState == 't')
{
return DEBUG_STATUS_STEP_INTO;
}
else if (g_CmdState == 'b')
{
return DEBUG_STATUS_STEP_BRANCH;
}
else if (g_CmdState == 'g' || g_CmdState == 'e')
{
return DEBUG_STATUS_GO;
}
else
{
return DEBUG_STATUS_BREAK;
}
}
STDMETHODIMP
DebugClient::GetExecutionStatus(
THIS_
OUT PULONG Status
)
{
// This method is reentrant.
*Status = ::GetExecutionStatus();
return S_OK;
}
HRESULT
SetExecutionStatus(ULONG Status)
{
// If there's an outstanding request for input don't
// allow the execution status of the engine to change
// as it could lead to a wait which cannot
// be carried out in this situation. It's better to fail
// this call and have the caller try again.
if (g_InputNesting >= 1 ||
!g_Machine)
{
return E_UNEXPECTED;
}
if (IS_RUNNING(g_CmdState))
{
// Already running.
return S_OK;
}
ADDR PcAddr;
g_Machine->GetPC(&PcAddr);
// Notifications are sent in the step/go functions.
if (Status >= DEBUG_STATUS_GO &&
Status <= DEBUG_STATUS_GO_NOT_HANDLED)
{
SetExecGo(Status, &PcAddr, NULL, FALSE, 0, NULL, NULL);
}
else
{
DBG_ASSERT(Status == DEBUG_STATUS_STEP_OVER ||
Status == DEBUG_STATUS_STEP_INTO ||
Status == DEBUG_STATUS_STEP_BRANCH);
if (g_Machine->IsStepStatusSupported(Status))
{
char StepType;
switch (Status)
{
case DEBUG_STATUS_STEP_INTO:
StepType = 't';
break;
case DEBUG_STATUS_STEP_OVER:
StepType = 'p';
break;
case DEBUG_STATUS_STEP_BRANCH:
StepType = 'b';
break;
}
SetExecStepTrace(&PcAddr, 1, NULL, FALSE, FALSE, StepType);
}
else
{
char* Mode;
switch (Status)
{
case DEBUG_STATUS_STEP_BRANCH:
Mode = "Taken branch trace";
break;
default:
Mode = "Trace";
break;
}
ErrOut("%s mode not supported\n", Mode);
return E_INVALIDARG;
}
}
return S_OK;
}
STDMETHODIMP
DebugClient::SetExecutionStatus(
THIS_
IN ULONG Status
)
{
if ((Status <= DEBUG_STATUS_NO_CHANGE || Status >= DEBUG_STATUS_BREAK) &&
Status != DEBUG_STATUS_STEP_BRANCH)
{
return E_INVALIDARG;
}
HRESULT RetStatus;
ENTER_ENGINE();
if (!IS_MACHINE_SET(g_Target))
{
RetStatus = E_UNEXPECTED;
}
else
{
RetStatus = ::SetExecutionStatus(Status);
}
LEAVE_ENGINE();
return RetStatus;
}
STDMETHODIMP
DebugClient::GetCodeLevel(
THIS_
OUT PULONG Level
)
{
ENTER_ENGINE();
if (g_SrcOptions & SRCOPT_STEP_SOURCE)
{
*Level = DEBUG_LEVEL_SOURCE;
}
else
{
*Level = DEBUG_LEVEL_ASSEMBLY;
}
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::SetCodeLevel(
THIS_
IN ULONG Level
)
{
ENTER_ENGINE();
HRESULT Status = S_OK;
ULONG OldSrcOpt = g_SrcOptions;
switch(Level)
{
case DEBUG_LEVEL_ASSEMBLY:
g_SrcOptions &= ~SRCOPT_STEP_SOURCE;
break;
case DEBUG_LEVEL_SOURCE:
g_SrcOptions |= SRCOPT_STEP_SOURCE;
break;
default:
Status = E_INVALIDARG;
break;
}
if ((OldSrcOpt ^ g_SrcOptions) & SRCOPT_STEP_SOURCE)
{
NotifyChangeEngineState(DEBUG_CES_CODE_LEVEL,
(g_SrcOptions & SRCOPT_STEP_SOURCE) ?
DEBUG_LEVEL_SOURCE : DEBUG_LEVEL_ASSEMBLY,
TRUE);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetEngineOptions(
THIS_
OUT PULONG Options
)
{
// This method is reentrant.
*Options = g_EngOptions;
return S_OK;
}
HRESULT
SetEngOptions(ULONG Options)
{
if (g_EngOptions != Options)
{
// Make sure allow and disallow network paths aren't both on.
if ((Options & ~DEBUG_ENGOPT_ALL) ||
((Options & DEBUG_ENGOPT_NETWORK_PATHS) ==
DEBUG_ENGOPT_NETWORK_PATHS))
{
return E_INVALIDARG;
}
// Security options cannot be disabled.
if ((g_EngOptions & DEBUG_ENGOPT_DISALLOW_SHELL_COMMANDS) &&
!(Options & DEBUG_ENGOPT_DISALLOW_SHELL_COMMANDS))
{
return E_INVALIDARG;
}
g_EngOptions = Options;
ULONG Notify = DEBUG_CES_ENGINE_OPTIONS;
ULONG64 Arg = Options;
if (SyncFiltersWithOptions())
{
Notify |= DEBUG_CES_EVENT_FILTERS;
Arg = DEBUG_ANY_ID;
}
// XXX drewb - Notification without any lock.
NotifyChangeEngineState(Notify, Arg, FALSE);
}
return S_OK;
}
STDMETHODIMP
DebugClient::AddEngineOptions(
THIS_
IN ULONG Options
)
{
// This method is reentrant.
return SetEngOptions(g_EngOptions | Options);
}
STDMETHODIMP
DebugClient::RemoveEngineOptions(
THIS_
IN ULONG Options
)
{
// This method is reentrant.
return SetEngOptions(g_EngOptions & ~Options);
}
STDMETHODIMP
DebugClient::SetEngineOptions(
THIS_
IN ULONG Options
)
{
// This method is reentrant.
return SetEngOptions(Options);
}
STDMETHODIMP
DebugClient::GetSystemErrorControl(
THIS_
OUT PULONG OutputLevel,
OUT PULONG BreakLevel
)
{
ENTER_ENGINE();
*OutputLevel = g_SystemErrorOutput;
*BreakLevel = g_SystemErrorBreak;
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::SetSystemErrorControl(
THIS_
IN ULONG OutputLevel,
IN ULONG BreakLevel
)
{
if (OutputLevel > SLE_WARNING ||
BreakLevel > SLE_WARNING)
{
return E_INVALIDARG;
}
ENTER_ENGINE();
g_SystemErrorOutput = OutputLevel;
g_SystemErrorBreak = BreakLevel;
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::GetTextMacro(
THIS_
IN ULONG Slot,
OUT OPTIONAL PSTR Buffer,
IN ULONG BufferSize,
OUT OPTIONAL PULONG MacroSize
)
{
if (Slot >= REG_USER_COUNT)
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
Status = FillStringBuffer(GetUserReg(REG_USER_FIRST + Slot), 0,
Buffer, BufferSize, MacroSize);
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::SetTextMacro(
THIS_
IN ULONG Slot,
IN PCSTR Macro
)
{
if (Slot >= REG_USER_COUNT)
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
Status = SetUserReg(REG_USER_FIRST + Slot, Macro) ? S_OK : E_OUTOFMEMORY;
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetRadix(
THIS_
OUT PULONG Radix
)
{
ENTER_ENGINE();
*Radix = g_DefaultRadix;
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::SetRadix(
THIS_
IN ULONG Radix
)
{
if (Radix != 8 && Radix != 10 && Radix != 16)
{
return E_INVALIDARG;
}
ENTER_ENGINE();
g_DefaultRadix = Radix;
NotifyChangeEngineState(DEBUG_CES_RADIX, g_DefaultRadix, TRUE);
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::Evaluate(
THIS_
IN PCSTR Expression,
IN ULONG DesiredType,
OUT PDEBUG_VALUE Value,
OUT OPTIONAL PULONG RemainderIndex
)
{
char Copy[MAX_COMMAND];
if (Expression == NULL ||
strlen(Expression) >= sizeof(Copy))
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
if (!IS_CUR_MACHINE_ACCESSIBLE())
{
Status = E_UNEXPECTED;
goto Exit;
}
PSTR SaveCommand;
PSTR SaveStart;
// This Evaluate may be coming from an extension invoked
// from a command so save all command state.
SaveCommand = g_CurCmd;
SaveStart = g_CommandStart;
g_DisableErrorPrint++;
// Copy const string to buffer to avoid read-only memory
// AVs as the command is modified during parsing.
strcpy(Copy, Expression);
g_CurCmd = Copy;
g_CommandStart = Copy;
RemoveDelChar(g_CurCmd);
ExpandUserRegs(Copy, DIMA(Copy));
__try
{
TypedData Result;
EvalExpression* Eval = GetCurEvaluator();
Eval->EvalCurrent(&Result);
ReleaseEvaluator(Eval);
if (Result.IsFloat())
{
if (Result.ConvertToF64())
{
Status = E_FAIL;
__leave;
}
Value->F64 = Result.m_F64;
Value->Type = DEBUG_VALUE_FLOAT64;
}
else
{
if (Result.ConvertToU64())
{
Status = E_FAIL;
__leave;
}
Value->I64 = Result.m_U64;
Value->Type = DEBUG_VALUE_INT64;
}
if (RemainderIndex != NULL)
{
*RemainderIndex = (ULONG)(g_CurCmd - g_CommandStart);
}
if (DesiredType != DEBUG_VALUE_INVALID &&
DesiredType != Value->Type)
{
DEBUG_VALUE Natural = *Value;
Status = CoerceValue(&Natural, DesiredType, Value);
}
else
{
Status = S_OK;
}
}
__except(CommandExceptionFilter(GetExceptionInformation()))
{
Status = E_FAIL;
}
g_DisableErrorPrint--;
g_CurCmd = SaveCommand;
g_CommandStart = SaveStart;
Exit:
LEAVE_ENGINE();
return Status;
}
#define IS_INT(Type) \
((Type) >= DEBUG_VALUE_INT8 && (Type) <= DEBUG_VALUE_INT64)
#define IS_FLOAT(Type) \
((Type) >= DEBUG_VALUE_FLOAT32 && (Type) <= DEBUG_VALUE_FLOAT128)
#define IS_VECTOR(Type) \
((Type) >= DEBUG_VALUE_VECTOR64 && (Type) <= DEBUG_VALUE_VECTOR128)
STDMETHODIMP
DebugClient::CoerceValue(
THIS_
IN PDEBUG_VALUE In,
IN ULONG OutType,
OUT PDEBUG_VALUE Out
)
{
if (In->Type < DEBUG_VALUE_INT8 || In->Type >= DEBUG_VALUE_TYPES ||
OutType < DEBUG_VALUE_INT8 || OutType >= DEBUG_VALUE_TYPES)
{
return E_INVALIDARG;
}
if (In->Type == OutType)
{
*Out = *In;
return S_OK;
}
ENTER_ENGINE();
ZeroMemory(Out, sizeof(*Out));
Out->Type = OutType;
DEBUG_VALUE Inter;
char FloatStr[64];
ZeroMemory(&Inter, sizeof(Inter));
// Convert the input type to the largest
// matching type for intermediate operations.
switch(In->Type)
{
case DEBUG_VALUE_INT8:
Inter.I64 = In->I8;
Inter.Nat = FALSE;
break;
case DEBUG_VALUE_INT16:
Inter.I64 = In->I16;
Inter.Nat = FALSE;
break;
case DEBUG_VALUE_INT32:
Inter.I64 = In->I32;
Inter.Nat = FALSE;
break;
case DEBUG_VALUE_INT64:
Inter.I64 = In->I64;
Inter.Nat = In->Nat;
break;
case DEBUG_VALUE_FLOAT32:
// XXX drewb - Use direct conversion.
PrintString(FloatStr, DIMA(FloatStr), "%10g", In->F32);
_atoldbl((_ULDOUBLE*)Inter.RawBytes, FloatStr);
break;
case DEBUG_VALUE_FLOAT64:
// XXX drewb - Use direct conversion.
PrintString(FloatStr, DIMA(FloatStr), "%10lg", In->F64);
_atoldbl((_ULDOUBLE*)Inter.RawBytes, FloatStr);
break;
case DEBUG_VALUE_FLOAT80:
memcpy(Inter.RawBytes, In->F80Bytes, sizeof(In->F80Bytes));
break;
case DEBUG_VALUE_FLOAT82:
FLOAT128 f82;
memcpy(&f82, &(In->F82Bytes),
min(sizeof(f82), sizeof(In->F82Bytes)));
double f;
f = Float82ToDouble(&f82);
PrintString(FloatStr, DIMA(FloatStr), "%10g", f);
_atoldbl((_ULDOUBLE*)Inter.RawBytes, FloatStr);
break;
case DEBUG_VALUE_FLOAT128:
// XXX drewb - What's the format? How should this be supported.
memcpy(Inter.RawBytes, In->F128Bytes, sizeof(In->F128Bytes));
break;
case DEBUG_VALUE_VECTOR64:
memcpy(Inter.RawBytes, In->RawBytes, 8);
break;
case DEBUG_VALUE_VECTOR128:
memcpy(Inter.RawBytes, In->RawBytes, 16);
break;
default:
DBG_ASSERT(FALSE);
break;
}
// Convert between float, int and vector. There's
// no way to know what kind of data is in a vector
// so the raw bytes are just used directly.
if (IS_INT(In->Type) &&
IS_FLOAT(OutType))
{
// XXX drewb - Use direct conversion.
PrintString(FloatStr, DIMA(FloatStr), "%I64u", Inter.I64);
_atoldbl((_ULDOUBLE*)Inter.RawBytes, FloatStr);
}
else if (IS_FLOAT(In->Type) &&
IS_INT(OutType))
{
double TmpDbl;
// XXX drewb - Use direct conversion.
_uldtoa((_ULDOUBLE*)Inter.RawBytes, sizeof(FloatStr), FloatStr);
if (sscanf(FloatStr, "%lg", &TmpDbl) != 1)
{
TmpDbl = 0.0;
}
Inter.I64 = (ULONG64)TmpDbl;
Inter.Nat = FALSE;
}
// Convert the intermediate value down to the
// appropriate output size.
switch(OutType)
{
case DEBUG_VALUE_INT8:
Out->I8 = (UCHAR)Inter.I64;
break;
case DEBUG_VALUE_INT16:
Out->I16 = (USHORT)Inter.I64;
break;
case DEBUG_VALUE_INT32:
Out->I32 = (ULONG)Inter.I64;
break;
case DEBUG_VALUE_INT64:
Out->I64 = Inter.I64;
Out->Nat = Inter.Nat;
break;
case DEBUG_VALUE_FLOAT32:
// XXX drewb - Use direct conversion.
_uldtoa((_ULDOUBLE*)Inter.RawBytes, sizeof(FloatStr), FloatStr);
if (sscanf(FloatStr, "%g", &Out->F32) != 1)
{
Out->F32 = 0.0f;
}
break;
case DEBUG_VALUE_FLOAT64:
// XXX drewb - Use direct conversion.
_uldtoa((_ULDOUBLE*)Inter.RawBytes, sizeof(FloatStr), FloatStr);
if (sscanf(FloatStr, "%lg", &Out->F64) != 1)
{
Out->F64 = 0.0;
}
break;
case DEBUG_VALUE_FLOAT80:
memcpy(Out->F80Bytes, Inter.RawBytes, sizeof(Out->F80Bytes));
break;
case DEBUG_VALUE_FLOAT82:
_uldtoa((_ULDOUBLE*)Inter.RawBytes, sizeof(FloatStr), FloatStr);
double f;
if (sscanf(FloatStr, "%lg", &f) != 1)
{
f = 0.0;
}
FLOAT128 f82;
DoubleToFloat82(f, &f82);
memcpy(&(Out->F82Bytes), &f82,
min(sizeof(Out->F82Bytes), sizeof(f82)));
break;
case DEBUG_VALUE_FLOAT128:
// XXX drewb - What's the format? How should this be supported.
memcpy(Out->F128Bytes, Inter.RawBytes, sizeof(Out->F128Bytes));
break;
case DEBUG_VALUE_VECTOR64:
memcpy(Out->RawBytes, Inter.RawBytes, 8);
break;
case DEBUG_VALUE_VECTOR128:
memcpy(Out->RawBytes, Inter.RawBytes, 16);
break;
default:
DBG_ASSERT(FALSE);
break;
}
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::CoerceValues(
THIS_
IN ULONG Count,
IN /* size_is(Count) */ PDEBUG_VALUE In,
IN /* size_is(Count) */ PULONG OutTypes,
OUT /* size_is(Count) */ PDEBUG_VALUE Out
)
{
ENTER_ENGINE();
ULONG i;
HRESULT Status, SingleStatus;
Status = S_OK;
for (i = 0; i < Count; i++)
{
SingleStatus = CoerceValue(In, *OutTypes, Out);
if (SingleStatus != S_OK)
{
// Accumulate error and mark failed value.
Status = SingleStatus;
Out->Type = DEBUG_VALUE_INVALID;
}
In++;
OutTypes++;
Out++;
}
LEAVE_ENGINE();
return Status;
}
HRESULT
Execute(DebugClient* Client, PCSTR Command, ULONG Flags)
{
char Copy[MAX_COMMAND];
ULONG Len = strlen(Command);
if (Len >= MAX_COMMAND)
{
return E_INVALIDARG;
}
BOOL AddNewLine = Len == 0 || Command[Len - 1] != '\n';
if (Flags & DEBUG_EXECUTE_ECHO)
{
dprintf("%s", Command);
if (AddNewLine)
{
dprintf("\n");
}
}
else if ((Flags & DEBUG_EXECUTE_NOT_LOGGED) == 0)
{
lprintf(Command);
if (AddNewLine)
{
lprintf("\n");
}
}
HRESULT Status;
PSTR SaveCommand;
PSTR SaveStart;
// This Execute may be coming from an extension invoked
// from a command so save all command state.
SaveCommand = g_CurCmd;
SaveStart = g_CommandStart;
// Copy const string to buffer to avoid read-only memory
// AVs as the command is modified during parsing.
strcpy(Copy, Command);
g_CurCmd = Copy;
g_CommandStart = Copy;
RemoveDelChar(g_CurCmd);
ExpandUserRegs(Copy, DIMA(Copy));
ReplaceAliases(g_CurCmd, MAX_COMMAND);
if ((Flags & DEBUG_EXECUTE_NO_REPEAT) == 0 &&
(g_EngOptions & DEBUG_ENGOPT_NO_EXECUTE_REPEAT) == 0)
{
if (Copy[0] == 0)
{
strcpy(Copy, g_LastCommand);
}
else
{
strcpy(g_LastCommand, Copy);
}
}
for (;;)
{
Status = ProcessCommandsAndCatch(Client);
// If we're switching processors (g_CmdState == 's')
// we have to wait to allow the switch to occur.
if (g_CmdState != 's' &&
(Status != S_FALSE ||
(g_EngStatus & ENG_STATUS_NO_AUTO_WAIT)))
{
break;
}
if ((g_CmdState != 's' ||
!IS_CONN_KERNEL_TARGET(g_Target) ||
((ConnLiveKernelTargetInfo*)g_Target)->
m_Transport->m_WaitingThread != 0) &&
GetCurrentThreadId() != g_SessionThread)
{
ErrOut("Non-primary client caused an implicit wait\n");
Status = E_FAIL;
break;
}
if ((Status =
RawWaitForEvent(DEBUG_WAIT_DEFAULT, INFINITE)) != S_OK)
{
break;
}
}
g_CurCmd = SaveCommand;
g_CommandStart = SaveStart;
return Status;
}
#define ALL_EXECUTE_FLAGS \
(DEBUG_EXECUTE_DEFAULT | \
DEBUG_EXECUTE_ECHO | \
DEBUG_EXECUTE_NOT_LOGGED | \
DEBUG_EXECUTE_NO_REPEAT)
STDMETHODIMP
DebugClient::Execute(
THIS_
IN ULONG OutputControl,
IN PCSTR Command,
IN ULONG Flags
)
{
if ((Flags & ~ALL_EXECUTE_FLAGS) ||
strlen(Command) >= MAX_COMMAND)
{
return E_INVALIDARG;
}
// We can't do a blanket IS_MACHINE_ACCESSIBLE check
// here as Execute's commands have a mix of requirements.
// Individual commands should check when necessary.
HRESULT Status;
ENTER_ENGINE();
OutCtlSave OldCtl;
if (!PushOutCtl(OutputControl, this, &OldCtl))
{
Status = E_INVALIDARG;
}
else
{
Status = ::Execute(this, Command, Flags);
PopOutCtl(&OldCtl);
}
LEAVE_ENGINE();
return Status;
}
HRESULT
ExecuteCommandFile(DebugClient* Client, PCSTR CommandFile, ULONG Flags)
{
HRESULT Status;
FILE* File;
File = fopen(CommandFile, "r");
if (File == NULL)
{
Status = HRESULT_FROM_WIN32(_doserrno);
}
else
{
char Command[MAX_COMMAND + 1];
va_list VaUnused;
// This value is only used as a placeholder so
// it doesn't really matter what it's initialized to.
ZeroMemory(&VaUnused, sizeof(VaUnused));
for (;;)
{
ULONG Len;
Command[sizeof(Command) - 2] = 0;
if (fgets(Command, sizeof(Command) - 1, File) == NULL)
{
if (feof(File))
{
Status = S_OK;
}
else
{
Status = E_FAIL;
}
break;
}
if (Command[sizeof(Command) - 2] != 0)
{
// Input line is too long.
Status = E_INVALIDARG;
break;
}
//
// If the line didn't end with a newline force it
// to have one for consistency.
//
Len = strlen(Command);
if (Len > 0 && Command[Len - 1] != '\n')
{
Command[Len] = '\n';
Command[Len + 1] = 0;
}
if (Flags & DEBUG_EXECUTE_ECHO)
{
OutputPrompt(" ", VaUnused);
// Command has a new-line built in.
dprintf("%s", Command);
}
else if ((Flags & DEBUG_EXECUTE_NOT_LOGGED) == 0)
{
ULONG OutCtl;
// Restrict output to the log only.
OutCtl = g_OutputControl;
g_OutputControl = (OutCtl & ~DEBUG_OUTCTL_SEND_MASK) |
DEBUG_OUTCTL_LOG_ONLY;
OutputPrompt(" ", VaUnused);
// Command has a new-line built in.
dprintf("%s", Command);
g_OutputControl = OutCtl;
}
// If a command exception is thrown we don't want to
// terminate execution of the script, so specifically
// check for the status code which indicates that
// an exception occurred and ignore it.
Status = Execute(Client, Command, DEBUG_EXECUTE_NOT_LOGGED |
(Flags & ~DEBUG_EXECUTE_ECHO));
if (Status != S_OK && Status != HR_PROCESS_EXCEPTION)
{
break;
}
}
fclose(File);
}
return Status;
}
STDMETHODIMP
DebugClient::ExecuteCommandFile(
THIS_
IN ULONG OutputControl,
IN PCSTR CommandFile,
IN ULONG Flags
)
{
if (Flags & ~ALL_EXECUTE_FLAGS)
{
return E_INVALIDARG;
}
// We can't do a blanket IS_MACHINE_ACCESSIBLE check
// here as Execute's commands have a mix of requirements.
// Individual commands should check when necessary.
HRESULT Status;
ENTER_ENGINE();
OutCtlSave OldCtl;
if (!PushOutCtl(OutputControl, this, &OldCtl))
{
return E_INVALIDARG;
}
else
{
Status = ::ExecuteCommandFile(this, CommandFile, Flags);
PopOutCtl(&OldCtl);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetNumberBreakpoints(
THIS_
OUT PULONG Number
)
{
HRESULT Status;
ENTER_ENGINE();
if (g_Process == NULL)
{
Status = E_UNEXPECTED;
}
else
{
*Number = g_Process->m_NumBreakpoints;
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetBreakpointByIndex(
THIS_
IN ULONG Index,
OUT PDEBUG_BREAKPOINT* RetBp
)
{
HRESULT Status;
ENTER_ENGINE();
if (g_Process == NULL)
{
Status = E_UNEXPECTED;
}
else
{
Breakpoint* Bp = ::GetBreakpointByIndex(this, Index);
if (Bp != NULL)
{
Bp->AddRef();
*RetBp = Bp;
Status = S_OK;
}
else
{
Status = E_NOINTERFACE;
}
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetBreakpointById(
THIS_
IN ULONG Id,
OUT PDEBUG_BREAKPOINT* RetBp
)
{
HRESULT Status;
ENTER_ENGINE();
if (g_Process == NULL)
{
Status = E_UNEXPECTED;
}
else
{
Breakpoint* Bp = ::GetBreakpointById(this, Id);
if (Bp != NULL)
{
Bp->AddRef();
*RetBp = Bp;
Status = S_OK;
}
else
{
Status = E_NOINTERFACE;
}
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetBreakpointParameters(
THIS_
IN ULONG Count,
IN OPTIONAL /* size_is(Count) */ PULONG Ids,
IN ULONG Start,
OUT /* size_is(Count) */ PDEBUG_BREAKPOINT_PARAMETERS Params
)
{
HRESULT Status;
ENTER_ENGINE();
if (g_Process == NULL)
{
Status = E_UNEXPECTED;
}
else
{
ULONG i;
Breakpoint* Bp;
Status = S_OK;
for (i = 0; i < Count; i++)
{
if (Ids != NULL)
{
Bp = ::GetBreakpointById(this, *Ids++);
}
else
{
Bp = ::GetBreakpointByIndex(this, Start++);
}
if (Bp == NULL)
{
ZeroMemory(Params, sizeof(*Params));
Params->Id = DEBUG_ANY_ID;
Status = S_FALSE;
}
else
{
Bp->GetParameters(Params);
}
Params++;
}
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::AddBreakpoint(
THIS_
IN ULONG Type,
IN ULONG DesiredId,
OUT PDEBUG_BREAKPOINT* Bp
)
{
if (
#if DEBUG_BREAKPOINT_CODE > 0
Type < DEBUG_BREAKPOINT_CODE ||
#endif
Type > DEBUG_BREAKPOINT_DATA)
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
if (!IS_MACHINE_SET(g_Target) || !g_Process)
{
Status = E_UNEXPECTED;
}
else
{
Status = ::AddBreakpoint(this, g_Target->m_EffMachine,
Type, DesiredId, (Breakpoint**)Bp);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::RemoveBreakpoint(
THIS_
IN PDEBUG_BREAKPOINT Bp
)
{
HRESULT Status;
ENTER_ENGINE();
if (g_Process == NULL)
{
Status = E_UNEXPECTED;
}
else
{
::RemoveBreakpoint((Breakpoint*)Bp);
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::AddExtension(
THIS_
IN PCSTR Path,
IN ULONG Flags,
OUT PULONG64 Handle
)
{
if (Flags & ~(DEBUG_EXTENSION_AT_ENGINE))
{
return E_INVALIDARG;
}
// Remote extensions aren't supported at the moment.
if (Flags != DEBUG_EXTENSION_AT_ENGINE)
{
return E_NOTIMPL;
}
HRESULT Status;
ENTER_ENGINE();
char* End;
EXTDLL* Ext;
if ((Ext = AddExtensionDll((PSTR)Path, TRUE, NULL, &End)) == NULL)
{
Status = E_OUTOFMEMORY;
}
else
{
*Handle = (ULONG64)Ext;
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::RemoveExtension(
THIS_
IN ULONG64 Handle
)
{
ENTER_ENGINE();
UnloadExtensionDll((EXTDLL*)Handle, FALSE);
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::GetExtensionByPath(
THIS_
IN PCSTR Path,
OUT PULONG64 Handle
)
{
HRESULT Status;
ENTER_ENGINE();
EXTDLL* Ext;
Status = E_NOINTERFACE;
for (Ext = g_ExtDlls; Ext != NULL; Ext = Ext->Next)
{
if ((!Ext->Target || Ext->Target == g_Target) &&
!_strcmpi(Path, Ext->Name))
{
Status = S_OK;
*Handle = (ULONG64)Ext;
break;
}
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::CallExtension(
THIS_
IN OPTIONAL ULONG64 Handle,
IN PCSTR Function,
IN OPTIONAL PCSTR Arguments
)
{
char LocalFunc[MAX_COMMAND];
ULONG Len;
// Copy function name to temp buffer because it is
// modified.
Len = strlen(Function) + 1;
if (Len > sizeof(LocalFunc))
{
return E_INVALIDARG;
}
memcpy(LocalFunc, Function, Len);
HRESULT Status;
ENTER_ENGINE();
if (!IS_CUR_MACHINE_ACCESSIBLE())
{
Status = E_UNEXPECTED;
}
else if (!CallAnyExtension(this, (EXTDLL*)Handle, LocalFunc, Arguments,
Handle != 0, TRUE, &Status))
{
Status = E_FAIL;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetExtensionFunction(
THIS_
IN ULONG64 Handle,
IN PCSTR FuncName,
OUT FARPROC* Function
)
{
HRESULT Status;
char ExpName[MAX_PATH + 16];
if (strlen(FuncName) >= MAX_PATH)
{
return E_INVALIDARG;
}
// Keep the namespace for extension function exports
// separate from the namespace for extension commands.
// Extension commands are exported under the same
// name as the command, so prefix extension functions
// to make them obviously different and to avoid
// name conflicts.
strcpy(ExpName, "_EFN_");
strcat(ExpName, FuncName);
ENTER_ENGINE();
EXTDLL* Ext;
FARPROC Routine;
Status = E_NOINTERFACE;
if (Handle != 0)
{
Ext = (EXTDLL*)(ULONG_PTR)Handle;
}
else
{
Ext = g_ExtDlls;
}
while (Ext != NULL)
{
if ((!Ext->Target || Ext->Target == g_Target) &&
LoadExtensionDll(g_Target, Ext))
{
Routine = GetProcAddress(Ext->Dll, ExpName);
if (Routine != NULL)
{
Status = S_OK;
*Function = Routine;
break;
}
}
// If the search was limited to a single extension stop looking.
if (Handle != 0)
{
break;
}
Ext = Ext->Next;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetWindbgExtensionApis32(
THIS_
IN OUT PWINDBG_EXTENSION_APIS32 Api
)
{
if (Api->nSize != sizeof(*Api))
{
return E_INVALIDARG;
}
*Api = g_WindbgExtensions32;
return S_OK;
}
STDMETHODIMP
DebugClient::GetWindbgExtensionApis64(
THIS_
IN OUT PWINDBG_EXTENSION_APIS64 Api
)
{
if (Api->nSize != sizeof(*Api))
{
return E_INVALIDARG;
}
*Api = g_WindbgExtensions64;
return S_OK;
}
STDMETHODIMP
DebugClient::GetNumberEventFilters(
THIS_
OUT PULONG SpecificEvents,
OUT PULONG SpecificExceptions,
OUT PULONG ArbitraryExceptions
)
{
ENTER_ENGINE();
*SpecificEvents = FILTER_SPECIFIC_LAST - FILTER_SPECIFIC_FIRST + 1;
*SpecificExceptions = FILTER_EXCEPTION_LAST - FILTER_EXCEPTION_FIRST + 1;
*ArbitraryExceptions = g_NumOtherExceptions;
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::GetEventFilterText(
THIS_
IN ULONG Index,
OUT OPTIONAL PSTR Buffer,
IN ULONG BufferSize,
OUT OPTIONAL PULONG TextSize
)
{
HRESULT Status;
ENTER_ENGINE();
if (Index >= FILTER_COUNT)
{
Status = E_NOINTERFACE;
}
else
{
Status = FillStringBuffer(g_EventFilters[Index].Name, 0,
Buffer, BufferSize, TextSize);
}
LEAVE_ENGINE();
return Status;
}
HRESULT
GetEitherEventFilterCommand(ULONG Index, ULONG Which,
PSTR Buffer, ULONG BufferSize, PULONG CommandSize)
{
EVENT_COMMAND* EventCommand;
if (Index < FILTER_COUNT)
{
EventCommand = &g_EventFilters[Index].Command;
}
else if ((Index - FILTER_COUNT) < g_NumOtherExceptions)
{
EventCommand = &g_OtherExceptionCommands[Index - FILTER_COUNT];
}
else
{
return E_NOINTERFACE;
}
return FillStringBuffer(EventCommand->Command[Which],
EventCommand->CommandSize[Which],
Buffer, BufferSize, CommandSize);
}
STDMETHODIMP
DebugClient::GetEventFilterCommand(
THIS_
IN ULONG Index,
OUT OPTIONAL PSTR Buffer,
IN ULONG BufferSize,
OUT OPTIONAL PULONG CommandSize
)
{
HRESULT Status;
ENTER_ENGINE();
Status = GetEitherEventFilterCommand(Index, 0, Buffer, BufferSize,
CommandSize);
LEAVE_ENGINE();
return Status;
}
HRESULT
SetEitherEventFilterCommand(DebugClient* Client, ULONG Index, ULONG Which,
PCSTR Command)
{
EVENT_COMMAND* EventCommand;
if (Index < FILTER_COUNT)
{
EventCommand = &g_EventFilters[Index].Command;
}
else if ((Index - FILTER_COUNT) < g_NumOtherExceptions)
{
EventCommand = &g_OtherExceptionCommands[Index - FILTER_COUNT];
}
else
{
return E_NOINTERFACE;
}
HRESULT Status;
Status = ChangeString(&EventCommand->Command[Which],
&EventCommand->CommandSize[Which],
Command);
if (Status == S_OK)
{
if (Index < FILTER_COUNT)
{
g_EventFilters[Index].Flags |= FILTER_CHANGED_COMMAND;
}
EventCommand->Client = Client;
NotifyChangeEngineState(DEBUG_CES_EVENT_FILTERS, Index, TRUE);
}
return Status;
}
STDMETHODIMP
DebugClient::SetEventFilterCommand(
THIS_
IN ULONG Index,
IN PCSTR Command
)
{
HRESULT Status;
ENTER_ENGINE();
Status = SetEitherEventFilterCommand(this, Index, 0, Command);
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetSpecificFilterParameters(
THIS_
IN ULONG Start,
IN ULONG Count,
OUT /* size_is(Count) */ PDEBUG_SPECIFIC_FILTER_PARAMETERS Params
)
{
if (
#if FILTER_SPECIFIC_FIRST > 0
Start < FILTER_SPECIFIC_FIRST ||
#endif
Start > FILTER_SPECIFIC_LAST ||
Start + Count > FILTER_SPECIFIC_LAST + 1)
{
return E_INVALIDARG;
}
ENTER_ENGINE();
EVENT_FILTER* Filter;
for (ULONG i = 0; i < Count; i++)
{
Filter = g_EventFilters + (Start + i);
Params[i].ExecutionOption = Filter->Params.ExecutionOption;
Params[i].ContinueOption = Filter->Params.ContinueOption;
Params[i].TextSize = strlen(Filter->Name) + 1;
Params[i].CommandSize = Filter->Command.CommandSize[0];
Params[i].ArgumentSize = Filter->Argument != NULL ?
strlen(Filter->Argument) + 1 : 0;
}
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::SetSpecificFilterParameters(
THIS_
IN ULONG Start,
IN ULONG Count,
IN /* size_is(Count) */ PDEBUG_SPECIFIC_FILTER_PARAMETERS Params
)
{
if (
#if FILTER_SPECIFIC_FIRST > 0
Start < FILTER_SPECIFIC_FIRST ||
#endif
Start > FILTER_SPECIFIC_LAST ||
Start + Count > FILTER_SPECIFIC_LAST + 1)
{
return E_INVALIDARG;
}
HRESULT Status = S_OK;
ULONG Set = 0;
ULONG SetIndex = 0;
ENTER_ENGINE();
for (ULONG i = 0; i < Count; i++)
{
if (
#if DEBUG_FILTER_BREAK > 0
Params[i].ExecutionOption >= DEBUG_FILTER_BREAK &&
#endif
Params[i].ExecutionOption <= DEBUG_FILTER_IGNORE &&
#if DEBUG_FILTER_GO_HANDLED > 0
Params[i].ContinueOption >= DEBUG_FILTER_GO_HANDLED &&
#endif
Params[i].ContinueOption <= DEBUG_FILTER_GO_NOT_HANDLED)
{
g_EventFilters[Start + i].Params.ExecutionOption =
Params[i].ExecutionOption;
g_EventFilters[Start + i].Params.ContinueOption =
Params[i].ContinueOption;
g_EventFilters[Start + i].Flags |=
FILTER_CHANGED_EXECUTION | FILTER_CHANGED_CONTINUE;
Set++;
SetIndex = i;
}
else
{
Status = E_INVALIDARG;
}
}
if (SyncOptionsWithFilters())
{
NotifyChangeEngineState(DEBUG_CES_EVENT_FILTERS |
DEBUG_CES_ENGINE_OPTIONS,
DEBUG_ANY_ID, TRUE);
}
else if (Set == 1)
{
NotifyChangeEngineState(DEBUG_CES_EVENT_FILTERS, SetIndex, TRUE);
}
else if (Set > 1)
{
NotifyChangeEngineState(DEBUG_CES_EVENT_FILTERS, DEBUG_ANY_ID, TRUE);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetSpecificFilterArgument(
THIS_
IN ULONG Index,
OUT OPTIONAL PSTR Buffer,
IN ULONG BufferSize,
OUT OPTIONAL PULONG ArgumentSize
)
{
if (
#if FILTER_SPECIFIC_FIRST > 0
Index < FILTER_SPECIFIC_FIRST ||
#endif
Index > FILTER_SPECIFIC_LAST ||
g_EventFilters[Index].Argument == NULL)
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
Status = FillStringBuffer(g_EventFilters[Index].Argument, 0,
Buffer, BufferSize, ArgumentSize);
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::SetSpecificFilterArgument(
THIS_
IN ULONG Index,
IN PCSTR Argument
)
{
ULONG Len;
if (Argument == NULL)
{
Len = 1;
}
else
{
Len = strlen(Argument) + 1;
}
if (
#if FILTER_SPECIFIC_FIRST > 0
Index < FILTER_SPECIFIC_FIRST ||
#endif
Index > FILTER_SPECIFIC_LAST ||
g_EventFilters[Index].Argument == NULL ||
Len > FILTER_MAX_ARGUMENT)
{
return E_INVALIDARG;
}
ENTER_ENGINE();
if (Argument == NULL)
{
g_EventFilters[Index].Argument[0] = 0;
}
else
{
memcpy(g_EventFilters[Index].Argument, Argument, Len);
}
if (Index == DEBUG_FILTER_UNLOAD_MODULE)
{
g_UnloadDllBase = EvalStringNumAndCatch(Argument);
}
NotifyChangeEngineState(DEBUG_CES_EVENT_FILTERS, Index, TRUE);
LEAVE_ENGINE();
return S_OK;
}
void
GetExFilterParams(PCSTR Text,
PDEBUG_EXCEPTION_FILTER_PARAMETERS InParams,
EVENT_COMMAND* InCommand,
PDEBUG_EXCEPTION_FILTER_PARAMETERS OutParams)
{
OutParams->ExecutionOption = InParams->ExecutionOption;
OutParams->ContinueOption = InParams->ContinueOption;
OutParams->TextSize = Text != NULL ? strlen(Text) + 1 : 0;
OutParams->CommandSize = InCommand->CommandSize[0];
OutParams->SecondCommandSize = InCommand->CommandSize[1];
OutParams->ExceptionCode = InParams->ExceptionCode;
}
STDMETHODIMP
DebugClient::GetExceptionFilterParameters(
THIS_
IN ULONG Count,
IN OPTIONAL /* size_is(Count) */ PULONG Codes,
IN ULONG Start,
OUT /* size_is(Count) */ PDEBUG_EXCEPTION_FILTER_PARAMETERS Params
)
{
HRESULT Status = S_OK;
ENTER_ENGINE();
ULONG i, Index;
EVENT_FILTER* Filter;
if (Codes != NULL)
{
for (i = 0; i < Count; i++)
{
// Is this a specific exception?
Filter = g_EventFilters + FILTER_EXCEPTION_FIRST;
for (Index = FILTER_EXCEPTION_FIRST;
Index <= FILTER_EXCEPTION_LAST;
Index++)
{
if (Filter->Params.ExceptionCode == Codes[i])
{
GetExFilterParams(Filter->Name, &Filter->Params,
&Filter->Command, Params + i);
break;
}
Filter++;
}
if (Index > FILTER_EXCEPTION_LAST)
{
// Is this an other exception?
for (Index = 0; Index < g_NumOtherExceptions; Index++)
{
if (g_OtherExceptionList[Index].ExceptionCode == Codes[i])
{
GetExFilterParams(NULL, g_OtherExceptionList + Index,
g_OtherExceptionCommands + Index,
Params + i);
break;
}
}
if (Index >= g_NumOtherExceptions)
{
memset(Params + i, 0xff, sizeof(*Params));
Status = E_NOINTERFACE;
}
}
}
}
else
{
for (i = 0; i < Count; i++)
{
Index = Start + i;
// Is this a specific exception?
if (Index >= FILTER_EXCEPTION_FIRST &&
Index <= FILTER_EXCEPTION_LAST)
{
Filter = g_EventFilters + Index;
GetExFilterParams(Filter->Name, &Filter->Params,
&Filter->Command, Params + i);
}
// Is this an other exception?
else if (Index >= FILTER_COUNT &&
Index < FILTER_COUNT + g_NumOtherExceptions)
{
GetExFilterParams(NULL, g_OtherExceptionList +
(Index - FILTER_COUNT),
g_OtherExceptionCommands +
(Index - FILTER_COUNT),
Params + i);
}
else
{
memset(Params + i, 0xff, sizeof(*Params));
Status = E_INVALIDARG;
}
}
}
LEAVE_ENGINE();
return Status;
}
HRESULT
SetExFilterParams(PDEBUG_EXCEPTION_FILTER_PARAMETERS InParams,
PDEBUG_EXCEPTION_FILTER_PARAMETERS OutParams)
{
if (
#if DEBUG_FILTER_BREAK > 0
InParams->ExecutionOption >= DEBUG_FILTER_BREAK &&
#endif
InParams->ExecutionOption <= DEBUG_FILTER_IGNORE &&
#if DEBUG_FILTER_GO_HANDLED > 0
InParams->ContinueOption >= DEBUG_FILTER_GO_HANDLED &&
#endif
InParams->ContinueOption <= DEBUG_FILTER_GO_NOT_HANDLED)
{
OutParams->ExecutionOption = InParams->ExecutionOption;
OutParams->ContinueOption = InParams->ContinueOption;
return S_OK;
}
else
{
return E_INVALIDARG;
}
}
HRESULT
SetOtherExFilterParams(PDEBUG_EXCEPTION_FILTER_PARAMETERS InParams,
ULONG OutIndex, PULONG Set, PULONG SetIndex)
{
HRESULT Status;
if (OutIndex < g_NumOtherExceptions)
{
if (InParams->ExecutionOption == DEBUG_FILTER_REMOVE)
{
RemoveOtherException(OutIndex);
*Set += 2;
Status = S_OK;
}
else
{
Status = SetExFilterParams(InParams,
g_OtherExceptionList + OutIndex);
if (Status == S_OK)
{
(*Set)++;
*SetIndex = OutIndex + FILTER_COUNT;
}
}
}
else
{
if (g_NumOtherExceptions == OTHER_EXCEPTION_LIST_MAX)
{
Status = E_OUTOFMEMORY;
}
else
{
OutIndex = g_NumOtherExceptions;
g_OtherExceptionList[OutIndex].ExceptionCode =
InParams->ExceptionCode;
Status = SetExFilterParams(InParams,
g_OtherExceptionList + OutIndex);
if (Status == S_OK)
{
ZeroMemory(&g_OtherExceptionCommands[OutIndex],
sizeof(g_OtherExceptionCommands[OutIndex]));
g_NumOtherExceptions++;
(*Set)++;
*SetIndex = OutIndex + FILTER_COUNT;
}
}
}
return Status;
}
STDMETHODIMP
DebugClient::SetExceptionFilterParameters(
THIS_
IN ULONG Count,
IN /* size_is(Count) */ PDEBUG_EXCEPTION_FILTER_PARAMETERS Params
)
{
HRESULT Status = S_OK;
ENTER_ENGINE();
ULONG i, Index;
EVENT_FILTER* Filter;
ULONG Set = 0;
ULONG SetIndex = 0;
for (i = 0; i < Count; i++)
{
// Is this a specific exception?
Filter = g_EventFilters + FILTER_EXCEPTION_FIRST;
for (Index = FILTER_EXCEPTION_FIRST;
Index <= FILTER_EXCEPTION_LAST;
Index++)
{
if (Filter->Params.ExceptionCode == Params[i].ExceptionCode)
{
Status = SetExFilterParams(Params + i, &Filter->Params);
if (Status == S_OK)
{
Filter->Flags |= FILTER_CHANGED_EXECUTION |
FILTER_CHANGED_CONTINUE;
Set++;
SetIndex = Index;
}
break;
}
Filter++;
}
if (Index > FILTER_EXCEPTION_LAST)
{
// Is this an other exception?
for (Index = 0; Index < g_NumOtherExceptions; Index++)
{
if (g_OtherExceptionList[Index].ExceptionCode ==
Params[i].ExceptionCode)
{
break;
}
}
Status = SetOtherExFilterParams(Params + i, Index,
&Set, &SetIndex);
}
}
if (Set == 1)
{
NotifyChangeEngineState(DEBUG_CES_EVENT_FILTERS, SetIndex, TRUE);
}
else if (Set > 1)
{
NotifyChangeEngineState(DEBUG_CES_EVENT_FILTERS, DEBUG_ANY_ID, TRUE);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetExceptionFilterSecondCommand(
THIS_
IN ULONG Index,
OUT OPTIONAL PSTR Buffer,
IN ULONG BufferSize,
OUT OPTIONAL PULONG CommandSize
)
{
HRESULT Status;
ENTER_ENGINE();
Status = GetEitherEventFilterCommand(Index, 1, Buffer, BufferSize,
CommandSize);
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::SetExceptionFilterSecondCommand(
THIS_
IN ULONG Index,
IN PCSTR Command
)
{
if (Index <= FILTER_SPECIFIC_LAST)
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
Status = SetEitherEventFilterCommand(this, Index, 1, Command);
LEAVE_ENGINE();
return Status;
}
HRESULT
WaitForAnyTarget(ULONG Flags, ULONG Timeout, ULONG EventPoss,
ULONG ElapsedTime, PULONG EventStatus)
{
HRESULT Status;
TargetInfo* Target;
if (EventPoss == 1)
{
// If there's only one source find it and do a simple wait on it.
ForAllLayersToTarget()
{
if (Target->m_EventPossible)
{
Status = Target->
WaitForEvent(Flags, Timeout,
ElapsedTime - Target->m_WaitTimeBase,
EventStatus);
Target->m_FirstWait = FALSE;
return Status;
}
}
return E_UNEXPECTED;
}
for (;;)
{
//
// Poll all the available sources for an event.
// A zero timeout is used to get through them as
// quickly as possible so events are picked up right away.
//
// Sanity check the number of possible sources to
// avoid an infinite loop if something gets confused.
EventPoss = 0;
ForAllLayersToTarget()
{
if (!Target->m_EventPossible)
{
continue;
}
EventPoss++;
Status = Target->
WaitForEvent(Flags, 0,
ElapsedTime - Target->m_WaitTimeBase,
EventStatus);
Target->m_FirstWait = FALSE;
if (Status != S_FALSE)
{
return Status;
}
// If we've exhausted the timeout period return
// with a no-event status.
if (Timeout == 0)
{
return S_FALSE;
}
//
// No source had an event, so sleep for a little while.
//
ULONG UseTimeout = 500;
if (Timeout != INFINITE && UseTimeout > Timeout)
{
UseTimeout = Timeout;
}
SUSPEND_ENGINE();
Sleep(UseTimeout);
RESUME_ENGINE();
if (Timeout != INFINITE)
{
Timeout -= UseTimeout;
// Let the loop poll for events one more time
// before checking for a completed timeout.
}
}
if (!EventPoss)
{
// For some reason there are no longer any
// event sources.
return E_UNEXPECTED;
}
}
}
HRESULT
RawWaitForEvent(ULONG Flags, ULONG Timeout)
{
HRESULT Status;
ULONG ContinueStatus;
Status = PrepareForWait(Flags, &ContinueStatus);
if ((g_EngStatus & ENG_STATUS_WAITING) == 0)
{
// An error occurred or some kind of synthesized
// event is being returned.
goto Exit;
}
EventOut("> Executing\n");
BOOL FirstLoop = TRUE;
ULONG ElapsedTime = 0;
for (;;)
{
EventOut(">> Waiting, %d elapsed\n", ElapsedTime);
TargetInfo* Target;
ULONG EventPoss = 0;
ULONG DesiredTimeout;
ULONG UseTimeout = Timeout;
ForAllLayersToTarget()
{
if (FirstLoop)
{
Target->m_WaitTimeBase = 0;
}
if ((Status = Target->
WaitInitialize(Flags, Timeout,
FirstLoop ? WINIT_FIRST : WINIT_NOT_FIRST,
&DesiredTimeout)) != S_OK)
{
goto Calls;
}
if (Target->m_EventPossible)
{
EventPoss++;
UseTimeout = min(UseTimeout, DesiredTimeout);
}
}
if (!EventPoss)
{
DiscardTargets(DEBUG_SESSION_END);
Status = E_UNEXPECTED;
goto Calls;
}
FirstLoop = FALSE;
g_EngStatus &= ~ENG_STATUS_SPECIAL_EXECUTION;
// Don't process deferred work if this is
// just a processor switch.
if (g_CmdState != 's')
{
ProcessDeferredWork(&ContinueStatus);
}
if (g_EventTarget)
{
EventOut(">> Continue with %X\n", ContinueStatus);
if ((Status = g_EventTarget->
ReleaseLastEvent(ContinueStatus)) != S_OK)
{
goto Calls;
}
Target = g_EventTarget;
}
DiscardLastEvent();
g_EventTarget = NULL;
g_EngStatus &= ~ENG_STATUS_WAIT_SUCCESSFUL;
ULONG EventStatus;
if (g_EngStatus & ENG_STATUS_SPECIAL_EXECUTION)
{
// If this is special, directed execution we can only wait on
// the target that needs special execution.
Status = Target->
WaitForEvent(Flags, UseTimeout,
ElapsedTime - Target->m_WaitTimeBase,
&EventStatus);
Target->m_FirstWait = FALSE;
}
else
{
Status = WaitForAnyTarget(Flags, UseTimeout, EventPoss,
ElapsedTime, &EventStatus);
}
if (UseTimeout != INFINITE)
{
ElapsedTime += UseTimeout;
}
if (Timeout != INFINITE)
{
Timeout -= UseTimeout;
}
if (Status == S_FALSE && Timeout > 0)
{
// Everything timed out but we were using a
// timeout smaller than the caller's overall
// timeout, so loop around and keep waiting.
continue;
}
if (Status != S_OK)
{
goto Calls;
}
if (EventStatus != DEBUG_STATUS_NO_DEBUGGEE)
{
g_EngStatus |= ENG_STATUS_WAIT_SUCCESSFUL;
}
// Successfully retrieved an event so reset the timeout count.
ElapsedTime = 0;
ForAllLayersToTarget()
{
Target->m_WaitTimeBase = 0;
}
EventOut(">>> Event status %X\n", EventStatus);
if (EventStatus == DEBUG_STATUS_NO_DEBUGGEE)
{
// Machine has rebooted or something else
// which breaks the connection. Forget the
// connection and go back to waiting.
ContinueStatus = DBG_CONTINUE;
}
else if (EventStatus == DEBUG_STATUS_BREAK ||
(g_EngStatus & ENG_STATUS_SPECIAL_EXECUTION))
{
// If the event handlers requested a break return
// to the caller. This path is also taken
// during special execution to guarantee that
// the target doesn't start running when the
// engine is just executing for internal purposes.
Status = S_OK;
goto Calls;
}
else
{
// We're resuming execution so reverse any
// command preparation that may have occurred
// while processing the event.
if ((Status = PrepareForExecution(EventStatus)) != S_OK)
{
goto Calls;
}
ContinueStatus = EventStatusToContinue(EventStatus);
}
}
Calls:
g_EngStatus &= ~ENG_STATUS_WAITING;
// Control is passing back to the caller so the engine must
// be ready for command processing.
PrepareForCalls(0);
// If we did switch processors automatically
// update the page directory for the new processor.
if (IS_KERNEL_TARGET(g_EventTarget) &&
(g_EngStatus & ENG_STATUS_SPECIAL_EXECUTION) &&
g_EventTarget)
{
if (g_EventTarget->m_Machine->
SetDefaultPageDirectories(g_EventThread, PAGE_DIR_ALL) != S_OK)
{
WarnOut("WARNING: Unable to reset page directories\n");
}
}
g_EngStatus &= ~ENG_STATUS_SPECIAL_EXECUTION;
Exit:
EventOut("> Wait returning %X\n", Status);
return Status;
}
STDMETHODIMP
DebugClient::WaitForEvent(
THIS_
IN ULONG Flags,
IN ULONG Timeout
)
{
if (Flags != DEBUG_WAIT_DEFAULT)
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
// If there's an outstanding request for input don't wait
// as it's very unlikely that the other thread
// can handle such an engine change.
if (g_InputNesting >= 1 ||
::GetCurrentThreadId() != g_SessionThread ||
(g_EngStatus & ENG_STATUS_STOP_SESSION))
{
Status = E_UNEXPECTED;
goto Exit;
}
// If the caller is trying to force the engine to
// stop waiting return immediately.
if (g_EngStatus & ENG_STATUS_EXIT_CURRENT_WAIT)
{
Status = E_PENDING;
goto Exit;
}
// This constitutes interesting activity.
m_LastActivity = time(NULL);
if (g_EngStatus & ENG_STATUS_WAITING)
{
Status = E_FAIL;
goto Exit;
}
Status = RawWaitForEvent(Flags, Timeout);
Exit:
g_EngStatus &= ~ENG_STATUS_EXIT_CURRENT_WAIT;
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetLastEventInformation(
THIS_
OUT PULONG Type,
OUT PULONG ProcessId,
OUT PULONG ThreadId,
OUT OPTIONAL PVOID ExtraInformation,
IN ULONG ExtraInformationSize,
OUT OPTIONAL PULONG ExtraInformationUsed,
OUT OPTIONAL PSTR Description,
IN ULONG DescriptionSize,
OUT OPTIONAL PULONG DescriptionUsed
)
{
HRESULT Status;
ENTER_ENGINE();
*Type = g_LastEventType;
if (g_EventProcess != NULL)
{
*ProcessId = g_EventProcess->m_UserId;
*ThreadId = g_EventThread->m_UserId;
}
else
{
*ProcessId = DEBUG_ANY_ID;
*ThreadId = DEBUG_ANY_ID;
}
Status = FillDataBuffer(g_LastEventExtraData, g_LastEventExtraDataSize,
ExtraInformation, ExtraInformationSize,
ExtraInformationUsed);
if (FillStringBuffer(g_LastEventDesc, 0,
Description, DescriptionSize,
DescriptionUsed) == S_FALSE)
{
Status = S_FALSE;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetCurrentTimeDate(
THIS_
OUT PULONG TimeDate
)
{
HRESULT Status;
ENTER_ENGINE();
if (!g_Target)
{
Status = E_UNEXPECTED;
}
else
{
*TimeDate =
FileTimeToTimeDateStamp(g_Target->GetCurrentTimeDateN());
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetCurrentSystemUpTime(
THIS_
OUT PULONG UpTime
)
{
HRESULT Status;
ENTER_ENGINE();
if (!g_Target)
{
Status = E_UNEXPECTED;
}
else
{
*UpTime =
FileTimeToTime(g_Target->GetCurrentSystemUpTimeN());
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetDumpFormatFlags(
THIS_
OUT PULONG FormatFlags
)
{
HRESULT Status;
ENTER_ENGINE();
if (!IS_DUMP_TARGET(g_Target))
{
Status = E_UNEXPECTED;
}
else
{
*FormatFlags = ((DumpTargetInfo*)g_Target)->m_FormatFlags;
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetNumberTextReplacements(
THIS_
OUT PULONG NumRepl
)
{
HRESULT Status;
ENTER_ENGINE();
*NumRepl = g_NumAliases;
Status = S_OK;
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetTextReplacement(
THIS_
IN OPTIONAL PCSTR SrcText,
IN ULONG Index,
OUT OPTIONAL PSTR SrcBuffer,
IN ULONG SrcBufferSize,
OUT OPTIONAL PULONG SrcSize,
OUT OPTIONAL PSTR DstBuffer,
IN ULONG DstBufferSize,
OUT OPTIONAL PULONG DstSize
)
{
HRESULT Status;
ENTER_ENGINE();
PALIAS Scan = g_AliasListHead;
while (Scan != NULL &&
((SrcText != NULL && strcmp(SrcText, Scan->Name)) ||
(SrcText == NULL && Index-- > 0)))
{
Scan = Scan->Next;
}
if (Scan != NULL)
{
Status = FillStringBuffer(Scan->Name, 0,
SrcBuffer, SrcBufferSize, SrcSize);
if (FillStringBuffer(Scan->Value, 0,
DstBuffer, DstBufferSize, DstSize) == S_FALSE)
{
Status = S_FALSE;
}
}
else
{
Status = E_NOINTERFACE;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::SetTextReplacement(
THIS_
IN PCSTR SrcText,
IN OPTIONAL PCSTR DstText
)
{
HRESULT Status;
ENTER_ENGINE();
if (DstText != NULL)
{
Status = SetAlias(SrcText, DstText);
}
else
{
Status = DeleteAlias(SrcText);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::RemoveTextReplacements(
THIS
)
{
HRESULT Status;
ENTER_ENGINE();
Status = DeleteAlias("*");
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::OutputTextReplacements(
THIS_
IN ULONG OutputControl,
IN ULONG Flags
)
{
HRESULT Status;
ENTER_ENGINE();
ListAliases();
Status = S_OK;
LEAVE_ENGINE();
return Status;
}
#define ASMOPT_ALL \
(DEBUG_ASMOPT_VERBOSE | \
DEBUG_ASMOPT_NO_CODE_BYTES | \
DEBUG_ASMOPT_IGNORE_OUTPUT_WIDTH)
STDMETHODIMP
DebugClient::GetAssemblyOptions(
THIS_
OUT PULONG Options
)
{
ENTER_ENGINE();
*Options = g_AsmOptions;
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::AddAssemblyOptions(
THIS_
IN ULONG Options
)
{
if (Options & ~ASMOPT_ALL)
{
return E_INVALIDARG;
}
ENTER_ENGINE();
g_AsmOptions |= Options;
NotifyChangeEngineState(DEBUG_CES_ASSEMBLY_OPTIONS, g_AsmOptions, TRUE);
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::RemoveAssemblyOptions(
THIS_
IN ULONG Options
)
{
if (Options & ~ASMOPT_ALL)
{
return E_INVALIDARG;
}
ENTER_ENGINE();
g_AsmOptions &= ~Options;
NotifyChangeEngineState(DEBUG_CES_ASSEMBLY_OPTIONS, g_AsmOptions, TRUE);
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::SetAssemblyOptions(
THIS_
IN ULONG Options
)
{
if (Options & ~ASMOPT_ALL)
{
return E_INVALIDARG;
}
ENTER_ENGINE();
g_AsmOptions = Options;
NotifyChangeEngineState(DEBUG_CES_ASSEMBLY_OPTIONS, g_AsmOptions, TRUE);
LEAVE_ENGINE();
return S_OK;
}
STDMETHODIMP
DebugClient::GetExpressionSyntax(
THIS_
OUT PULONG Flags
)
{
ENTER_ENGINE();
*Flags = g_EvalSyntax;
LEAVE_ENGINE();
return S_OK;
}
HRESULT
SetExprSyntax(ULONG Flags)
{
if (Flags >= EVAL_COUNT)
{
return E_INVALIDARG;
}
g_EvalSyntax = Flags;
NotifyChangeEngineState(DEBUG_CES_EXPRESSION_SYNTAX, Flags, TRUE);
return S_OK;
}
STDMETHODIMP
DebugClient::SetExpressionSyntax(
THIS_
IN ULONG Flags
)
{
HRESULT Status;
ENTER_ENGINE();
Status = ::SetExprSyntax(Flags);
LEAVE_ENGINE();
return Status;
}
HRESULT
SetExprSyntaxByName(PCSTR Name)
{
HRESULT Status;
EvalExpression* Eval;
if ((Status = GetEvaluatorByName(Name, TRUE, &Eval)) != S_OK)
{
return Status;
}
SetExprSyntax(Eval->m_Syntax);
ReleaseEvaluator(Eval);
return S_OK;
}
STDMETHODIMP
DebugClient::SetExpressionSyntaxByName(
THIS_
IN PCSTR AbbrevName
)
{
HRESULT Status;
ENTER_ENGINE();
Status = ::SetExprSyntaxByName(AbbrevName);
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetNumberExpressionSyntaxes(
THIS_
OUT PULONG Number
)
{
*Number = EVAL_COUNT;
return S_OK;
}
STDMETHODIMP
DebugClient::GetExpressionSyntaxNames(
THIS_
IN ULONG Index,
OUT OPTIONAL PSTR FullNameBuffer,
IN ULONG FullNameBufferSize,
OUT OPTIONAL PULONG FullNameSize,
OUT OPTIONAL PSTR AbbrevNameBuffer,
IN ULONG AbbrevNameBufferSize,
OUT OPTIONAL PULONG AbbrevNameSize
)
{
if (Index >= EVAL_COUNT)
{
return E_INVALIDARG;
}
HRESULT Status;
ENTER_ENGINE();
EvalExpression* Eval = GetEvaluator(Index, TRUE);
if (!Eval)
{
Status = E_OUTOFMEMORY;
}
else
{
Status = FillStringBuffer(Eval->m_FullName, 0,
FullNameBuffer, FullNameBufferSize,
FullNameSize);
if (FillStringBuffer(Eval->m_AbbrevName, 0,
AbbrevNameBuffer, AbbrevNameBufferSize,
AbbrevNameSize) == S_FALSE)
{
Status = S_FALSE;
}
ReleaseEvaluator(Eval);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetNumberEvents(
THIS_
OUT PULONG Events
)
{
HRESULT Status;
ENTER_ENGINE();
if (!g_Target)
{
Status = E_UNEXPECTED;
}
else
{
*Events = g_Target->m_NumEvents;
Status = g_Target->m_DynamicEvents ? S_FALSE : S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetEventIndexDescription(
THIS_
IN ULONG Index,
IN ULONG Which,
IN OPTIONAL PSTR Buffer,
IN ULONG BufferSize,
OUT OPTIONAL PULONG DescSize
)
{
HRESULT Status;
if (Which != DEBUG_EINDEX_NAME)
{
return E_INVALIDARG;
}
ENTER_ENGINE();
if (!g_Target)
{
Status = E_UNEXPECTED;
}
else if (Index > g_Target->m_NumEvents)
{
Status = E_INVALIDARG;
}
else
{
Status = g_Target->
GetEventIndexDescription(Index, Which,
Buffer, BufferSize, DescSize);
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::GetCurrentEventIndex(
THIS_
OUT PULONG Index
)
{
HRESULT Status;
ENTER_ENGINE();
if (!g_Target)
{
Status = E_UNEXPECTED;
}
else
{
*Index = g_Target->m_EventIndex;
Status = S_OK;
}
LEAVE_ENGINE();
return Status;
}
STDMETHODIMP
DebugClient::SetNextEventIndex(
THIS_
IN ULONG Relation,
IN ULONG Value,
OUT PULONG NextIndex
)
{
HRESULT Status;
ENTER_ENGINE();
if (!IS_MACHINE_SET(g_Target) ||
IS_RUNNING(g_CmdState))
{
Status = E_UNEXPECTED;
goto Exit;
}
switch(Relation)
{
case DEBUG_EINDEX_FROM_START:
// Value is start index.
if (Value >= g_Target->m_NumEvents)
{
Status = E_INVALIDARG;
goto Exit;
}
break;
case DEBUG_EINDEX_FROM_END:
// Value is end index.
if (Value >= g_Target->m_NumEvents)
{
Status = E_INVALIDARG;
goto Exit;
}
Value = g_Target->m_NumEvents - Value - 1;
break;
case DEBUG_EINDEX_FROM_CURRENT:
if ((LONG)Value < 0)
{
Value = -(LONG)Value;
if (Value > g_Target->m_EventIndex)
{
Status = E_INVALIDARG;
goto Exit;
}
Value = g_Target->m_EventIndex - Value;
}
else
{
if (Value >= g_Target->m_NumEvents - g_Target->m_EventIndex)
{
Status = E_INVALIDARG;
goto Exit;
}
Value = g_Target->m_EventIndex + Value;
}
break;
default:
Status = E_INVALIDARG;
goto Exit;
}
if (g_Target->m_EventIndex != Value)
{
g_Target->m_NextEventIndex = Value;
g_CmdState = 'e';
NotifyChangeEngineState(DEBUG_CES_EXECUTION_STATUS, DEBUG_STATUS_GO,
TRUE);
}
*NextIndex = Value;
Status = S_OK;
Exit:
LEAVE_ENGINE();
return Status;
}