1083 lines
28 KiB
C++
1083 lines
28 KiB
C++
//----------------------------------------------------------------------------
|
|
//
|
|
// Console input and output.
|
|
//
|
|
// Copyright (C) Microsoft Corporation, 1999-2002.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
#include "pch.cpp"
|
|
#pragma hdrstop
|
|
|
|
#include <stdarg.h>
|
|
#include <process.h>
|
|
|
|
#include "conio.hpp"
|
|
#include "engine.hpp"
|
|
#include "main.hpp"
|
|
|
|
#define CONTROL_A 1
|
|
#define CONTROL_B 2
|
|
#define CONTROL_D 4
|
|
#define CONTROL_E 5
|
|
#define CONTROL_F 6
|
|
#define CONTROL_K 11
|
|
#define CONTROL_P 16
|
|
#define CONTROL_R 18
|
|
#define CONTROL_V 22
|
|
#define CONTROL_W 23
|
|
#define CONTROL_X 24
|
|
|
|
HANDLE g_ConInput, g_ConOutput;
|
|
HANDLE g_PromptInput;
|
|
HANDLE g_AllowInput;
|
|
|
|
ConInputCallbacks g_ConInputCb;
|
|
ConOutputCallbacks g_ConOutputCb;
|
|
|
|
BOOL g_IoInitialized;
|
|
BOOL g_ConInitialized;
|
|
char g_Buffer[MAX_COMMAND];
|
|
LONG g_Lines;
|
|
|
|
HANDLE g_PipeWrite;
|
|
OVERLAPPED g_PipeWriteOverlapped;
|
|
|
|
CRITICAL_SECTION g_InputLock;
|
|
BOOL g_InputStarted;
|
|
|
|
// Input thread interfaces for direct input thread calls.
|
|
IDebugClient* g_ConClient;
|
|
IDebugControl* g_ConControl;
|
|
|
|
//----------------------------------------------------------------------------
|
|
//
|
|
// Default input callbacks implementation, provides IUnknown.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
DefInputCallbacks::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)
|
|
DefInputCallbacks::AddRef(
|
|
THIS
|
|
)
|
|
{
|
|
// This class is designed to be static so
|
|
// there's no true refcount.
|
|
return 1;
|
|
}
|
|
|
|
STDMETHODIMP_(ULONG)
|
|
DefInputCallbacks::Release(
|
|
THIS
|
|
)
|
|
{
|
|
// This class is designed to be static so
|
|
// there's no true refcount.
|
|
return 0;
|
|
}
|
|
|
|
//----------------------------------------------------------------------------
|
|
//
|
|
// Console input callbacks.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
ConInputCallbacks::StartInput(
|
|
THIS_
|
|
IN ULONG BufferSize
|
|
)
|
|
{
|
|
if (!g_IoInitialized || g_IoMode == IO_NONE)
|
|
{
|
|
// Ignore input requests.
|
|
return S_OK;
|
|
}
|
|
|
|
EnterCriticalSection(&g_InputLock);
|
|
|
|
if (g_ConControl == NULL)
|
|
{
|
|
// If we're not remoted we aren't running a separate input
|
|
// thread so we need to block here until we get some input.
|
|
while (!ConIn(g_Buffer, sizeof(g_Buffer), TRUE))
|
|
{
|
|
; // Wait.
|
|
}
|
|
g_DbgControl->ReturnInput(g_Buffer);
|
|
}
|
|
else if (ConIn(g_Buffer, sizeof(g_Buffer), FALSE))
|
|
{
|
|
g_ConControl->ReturnInput(g_Buffer);
|
|
}
|
|
else
|
|
{
|
|
g_InputStarted = TRUE;
|
|
|
|
#ifndef KERNEL
|
|
// Wake up the input thread if necessary.
|
|
SetEvent(g_AllowInput);
|
|
#endif
|
|
}
|
|
|
|
LeaveCriticalSection(&g_InputLock);
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP
|
|
ConInputCallbacks::EndInput(
|
|
THIS
|
|
)
|
|
{
|
|
g_InputStarted = FALSE;
|
|
return S_OK;
|
|
}
|
|
|
|
//----------------------------------------------------------------------------
|
|
//
|
|
// Default output callbacks implementation, provides IUnknown.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
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;
|
|
}
|
|
|
|
//----------------------------------------------------------------------------
|
|
//
|
|
// Console output callbacks.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDMETHODIMP
|
|
ConOutputCallbacks::Output(
|
|
THIS_
|
|
IN ULONG Mask,
|
|
IN PCSTR Text
|
|
)
|
|
{
|
|
ConOutStr(Text);
|
|
return S_OK;
|
|
}
|
|
|
|
//----------------------------------------------------------------------------
|
|
//
|
|
// Functions
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
void
|
|
InitializeIo(PCSTR InputFile)
|
|
{
|
|
__try
|
|
{
|
|
InitializeCriticalSection(&g_InputLock);
|
|
}
|
|
__except(EXCEPTION_EXECUTE_HANDLER)
|
|
{
|
|
ErrorExit("Unable to initialize lock\n");
|
|
}
|
|
|
|
// The input file may not exist so there's no
|
|
// check for failure.
|
|
g_InputFile = fopen(InputFile, "r");
|
|
|
|
g_IoInitialized = TRUE;
|
|
}
|
|
|
|
void
|
|
CreateConsole(void)
|
|
{
|
|
if (g_ConInitialized)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Set this early to prevent an init call from Exit in
|
|
// case an Exit call is made inside this routine.
|
|
g_ConInitialized = TRUE;
|
|
|
|
#ifdef INHERIT_CONSOLE
|
|
|
|
g_ConInput = GetStdHandle(STD_INPUT_HANDLE);
|
|
g_ConOutput = GetStdHandle(STD_OUTPUT_HANDLE);
|
|
|
|
#else
|
|
|
|
SECURITY_ATTRIBUTES Security;
|
|
|
|
if (!AllocConsole())
|
|
{
|
|
ErrorExit("AllocConsole failed, %d\n", GetLastError());
|
|
}
|
|
|
|
ZeroMemory(&Security, sizeof(Security));
|
|
Security.nLength = sizeof(Security);
|
|
Security.bInheritHandle = TRUE;
|
|
|
|
g_ConInput = CreateFile( "CONIN$",
|
|
GENERIC_READ,
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
&Security,
|
|
OPEN_EXISTING,
|
|
0,
|
|
NULL
|
|
);
|
|
if (g_ConInput == INVALID_HANDLE_VALUE)
|
|
{
|
|
ErrorExit("Create CONIN$ failed, %d\n", GetLastError());
|
|
}
|
|
|
|
g_ConOutput = CreateFile( "CONOUT$",
|
|
GENERIC_WRITE | GENERIC_READ,
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
&Security,
|
|
OPEN_EXISTING,
|
|
0,
|
|
NULL
|
|
);
|
|
if (g_ConOutput == INVALID_HANDLE_VALUE)
|
|
{
|
|
ErrorExit("Create CONOUT$ failed, %d\n", GetLastError());
|
|
}
|
|
|
|
#endif
|
|
|
|
g_PromptInput = g_ConInput;
|
|
}
|
|
|
|
void
|
|
ReadPromptInputChars(PSTR Buffer, ULONG BufferSize)
|
|
{
|
|
ULONG Len;
|
|
ULONG Read;
|
|
|
|
// Reading from another source. Read character by
|
|
// character until a line is read.
|
|
Len = 0;
|
|
while (Len < BufferSize)
|
|
{
|
|
if (!ReadFile(g_PromptInput, &Buffer[Len], sizeof(Buffer[0]),
|
|
&Read, NULL) ||
|
|
Read != sizeof(Buffer[0]))
|
|
{
|
|
OutputDebugString("Unable to read input\n");
|
|
ExitDebugger(E_FAIL);
|
|
}
|
|
|
|
if (Buffer[Len] == '\n')
|
|
{
|
|
InterlockedDecrement(&g_Lines);
|
|
break;
|
|
}
|
|
|
|
// Ignore carriage returns.
|
|
if (Buffer[Len] != '\r')
|
|
{
|
|
// Prevent buffer overflow.
|
|
if (Len == BufferSize - 1)
|
|
{
|
|
break;
|
|
}
|
|
|
|
Len++;
|
|
}
|
|
}
|
|
|
|
Buffer[Len] = '\0';
|
|
}
|
|
|
|
BOOL
|
|
CheckForControlCommands(PDEBUG_CLIENT Client, PDEBUG_CONTROL Control,
|
|
char Char)
|
|
{
|
|
HRESULT Hr;
|
|
ULONG OutMask;
|
|
PCHAR DebugAction;
|
|
ULONG EngOptions;
|
|
|
|
switch(Char)
|
|
{
|
|
case CONTROL_B:
|
|
case CONTROL_X:
|
|
if (Client)
|
|
{
|
|
// Tell server about disconnect or
|
|
// force servers to get cleaned up.
|
|
Client->EndSession(g_RemoteClient ?
|
|
DEBUG_END_DISCONNECT : DEBUG_END_REENTRANT);
|
|
}
|
|
ExitProcess(S_OK);
|
|
|
|
case CONTROL_F:
|
|
//
|
|
// Force a breakin like Ctrl-C would do.
|
|
// The advantage is this will work when kd is being debugged.
|
|
//
|
|
|
|
Control->SetInterrupt(DEBUG_INTERRUPT_ACTIVE);
|
|
return TRUE;
|
|
|
|
case CONTROL_P:
|
|
// Launch cdb on this debugger.
|
|
char PidStr[32];
|
|
sprintf(PidStr, "\"cdb -p %d\"", GetCurrentProcessId());
|
|
_spawnlp(_P_NOWAIT,
|
|
"cmd.exe", "/c", "start",
|
|
"remote", "/s", PidStr, "cdb_pipe",
|
|
NULL);
|
|
return TRUE;
|
|
|
|
case CONTROL_V:
|
|
Client->GetOtherOutputMask(g_DbgClient, &OutMask);
|
|
OutMask ^= DEBUG_OUTPUT_VERBOSE;
|
|
Client->SetOtherOutputMask(g_DbgClient, OutMask);
|
|
Control->SetLogMask(OutMask);
|
|
ConOut("Verbose mode %s.\n",
|
|
(OutMask & DEBUG_OUTPUT_VERBOSE) ? "ON" : "OFF");
|
|
return TRUE;
|
|
|
|
case CONTROL_W:
|
|
Hr = Control->OutputVersionInformation(DEBUG_OUTCTL_AMBIENT);
|
|
if (Hr == HRESULT_FROM_WIN32(ERROR_BUSY))
|
|
{
|
|
ConOut("Engine is busy, try again\n");
|
|
}
|
|
else if (Hr != S_OK)
|
|
{
|
|
ConOut("Unable to show version information, 0x%X\n", Hr);
|
|
}
|
|
return TRUE;
|
|
|
|
#ifdef KERNEL
|
|
|
|
case CONTROL_A:
|
|
Client->SetKernelConnectionOptions("cycle_speed");
|
|
return TRUE;
|
|
|
|
case CONTROL_D:
|
|
Client->GetOtherOutputMask(g_DbgClient, &OutMask);
|
|
OutMask ^= DEBUG_IOUTPUT_KD_PROTOCOL;
|
|
Client->SetOtherOutputMask(g_DbgClient, OutMask);
|
|
Control->SetLogMask(OutMask);
|
|
return TRUE;
|
|
|
|
case CONTROL_K:
|
|
//
|
|
// Toggle between the following possibilities-
|
|
//
|
|
// (0) no breakin
|
|
// (1) -b style (same as Control-C up the wire)
|
|
// (2) -d style (stop on first dll load).
|
|
//
|
|
// NB -b and -d could both be on the command line
|
|
// but become mutually exclusive via this method.
|
|
// (Maybe should be a single enum type).
|
|
//
|
|
|
|
Control->GetEngineOptions(&EngOptions);
|
|
if (EngOptions & DEBUG_ENGOPT_INITIAL_BREAK)
|
|
{
|
|
//
|
|
// Was type 1, go to type 2.
|
|
//
|
|
|
|
EngOptions |= DEBUG_ENGOPT_INITIAL_MODULE_BREAK;
|
|
EngOptions &= ~DEBUG_ENGOPT_INITIAL_BREAK;
|
|
|
|
DebugAction = "breakin on first symbol load";
|
|
}
|
|
else if (EngOptions & DEBUG_ENGOPT_INITIAL_MODULE_BREAK)
|
|
{
|
|
//
|
|
// Was type 2, go to type 0.
|
|
//
|
|
|
|
EngOptions &= ~DEBUG_ENGOPT_INITIAL_MODULE_BREAK;
|
|
DebugAction = "NOT breakin";
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Was type 0, go to type 1.
|
|
//
|
|
|
|
EngOptions |= DEBUG_ENGOPT_INITIAL_BREAK;
|
|
DebugAction = "request initial breakpoint";
|
|
}
|
|
Control->SetEngineOptions(EngOptions);
|
|
ConOut("Will %s at next boot.\n", DebugAction);
|
|
return TRUE;
|
|
|
|
case CONTROL_R:
|
|
Client->SetKernelConnectionOptions("resync");
|
|
return TRUE;
|
|
|
|
#endif // #ifdef KERNEL
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL
|
|
ConIn(PSTR Buffer, ULONG BufferSize, BOOL Wait)
|
|
{
|
|
if (g_InitialCommand != NULL)
|
|
{
|
|
ConOut("%s: Reading initial command '%s'\n",
|
|
g_DebuggerName, g_InitialCommand);
|
|
CopyString(Buffer, g_InitialCommand, BufferSize);
|
|
g_InitialCommand = NULL;
|
|
return TRUE;
|
|
}
|
|
|
|
while (g_InputFile && g_InputFile != stdin)
|
|
{
|
|
if (fgets(Buffer, BufferSize, g_InputFile))
|
|
{
|
|
ULONG Len = strlen(Buffer);
|
|
|
|
ConOut("%s", Buffer);
|
|
if (Len > 0 && Buffer[Len - 1] == '\n')
|
|
{
|
|
Buffer[Len - 1] = 0;
|
|
}
|
|
else
|
|
{
|
|
ConOut("\n");
|
|
}
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
fclose(g_InputFile);
|
|
if (g_NextOldInputFile > 0)
|
|
{
|
|
g_InputFile = g_OldInputFiles[--g_NextOldInputFile];
|
|
}
|
|
else
|
|
{
|
|
g_InputFile = stdin;
|
|
}
|
|
}
|
|
}
|
|
if (g_InputFile == NULL)
|
|
{
|
|
g_InputFile = stdin;
|
|
}
|
|
|
|
switch(g_IoMode)
|
|
{
|
|
case IO_NONE:
|
|
return FALSE;
|
|
|
|
case IO_DEBUG:
|
|
case IO_DEBUG_DEFER:
|
|
if (!Wait)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
g_NtDllCalls.DbgPrompt("", Buffer,
|
|
min(BufferSize, MAX_DBG_PROMPT_COMMAND));
|
|
break;
|
|
|
|
case IO_CONSOLE:
|
|
ULONG Len;
|
|
|
|
if (g_PromptInput == g_ConInput)
|
|
{
|
|
if (!Wait)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
// Reading from the console so we can assume we'll
|
|
// read a line.
|
|
for (;;)
|
|
{
|
|
if (!ReadFile(g_PromptInput, Buffer, BufferSize, &Len, NULL))
|
|
{
|
|
OutputDebugString("Unable to read input\n");
|
|
ExitDebugger(E_FAIL);
|
|
}
|
|
|
|
// At a minimum a read should have CRLF. If it
|
|
// doesn't assume that something weird happened
|
|
// and ignore the read.
|
|
if (Len >= 2)
|
|
{
|
|
break;
|
|
}
|
|
|
|
Sleep(50);
|
|
}
|
|
|
|
// Remove CR LF.
|
|
Len -= 2;
|
|
Buffer[Len] = '\0';
|
|
|
|
// Edit out any special characters.
|
|
for (ULONG i = 0; i < Len; i++)
|
|
{
|
|
if (CheckForControlCommands(g_DbgClient, g_DbgControl,
|
|
Buffer[i]))
|
|
{
|
|
Buffer[i] = ' ';
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
#ifndef KERNEL
|
|
if (g_Lines == 0)
|
|
{
|
|
// Allow the input thread to read the console.
|
|
SetEvent(g_AllowInput);
|
|
}
|
|
#endif
|
|
|
|
while (g_Lines == 0)
|
|
{
|
|
if (!Wait)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
// Wait for the input thread to notify us that
|
|
// a line of input is available. While we're waiting,
|
|
// let the engine process callbacks provoked by
|
|
// other clients.
|
|
HRESULT Hr = g_DbgClient->DispatchCallbacks(INFINITE);
|
|
if (Hr != S_OK)
|
|
{
|
|
OutputDebugString("Unable to dispatch callbacks\n");
|
|
ExitDebugger(Hr);
|
|
}
|
|
|
|
// Some other client may have started execution in
|
|
// which case we want to stop waiting for input.
|
|
if (g_ExecStatus != DEBUG_STATUS_BREAK)
|
|
{
|
|
#ifndef KERNEL
|
|
// XXX drewb - Need a way to turn input off.
|
|
#endif
|
|
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
ReadPromptInputChars(Buffer, BufferSize);
|
|
}
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
#define KD_OUT_LIMIT 510
|
|
|
|
void
|
|
ConOutStr(PCSTR Str)
|
|
{
|
|
int Len;
|
|
|
|
switch(g_IoMode)
|
|
{
|
|
case IO_NONE:
|
|
// Throw it away.
|
|
break;
|
|
|
|
case IO_DEBUG:
|
|
case IO_DEBUG_DEFER:
|
|
//
|
|
// Send the output to the kernel debugger but note that we
|
|
// want any control C processing to be done locally rather
|
|
// than in the kernel.
|
|
//
|
|
// The kernel silently truncates DbgPrints longer than
|
|
// 512 characters so use multiple calls if necessary.
|
|
//
|
|
|
|
Len = strlen(Str);
|
|
if (Len > KD_OUT_LIMIT)
|
|
{
|
|
while (Len > 0)
|
|
{
|
|
if (g_NtDllCalls.DbgPrint("%.*s", KD_OUT_LIMIT,
|
|
Str) == STATUS_BREAKPOINT &&
|
|
g_DbgControl != NULL)
|
|
{
|
|
g_DbgControl->SetInterrupt(DEBUG_INTERRUPT_PASSIVE);
|
|
}
|
|
|
|
Len -= KD_OUT_LIMIT;
|
|
Str += KD_OUT_LIMIT;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (g_NtDllCalls.DbgPrint("%s", Str) == STATUS_BREAKPOINT &&
|
|
g_DbgControl != NULL)
|
|
{
|
|
g_DbgControl->SetInterrupt(DEBUG_INTERRUPT_PASSIVE);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case IO_CONSOLE:
|
|
if (g_ConOutput != NULL)
|
|
{
|
|
ULONG Written;
|
|
WriteFile(g_ConOutput, Str, strlen(Str), &Written, NULL);
|
|
}
|
|
else
|
|
{
|
|
OutputDebugString(Str);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
ConOut(PCSTR Format, ...)
|
|
{
|
|
va_list Args;
|
|
|
|
// If no attempt has been made to create a console
|
|
// go ahead and try now.
|
|
if (g_IoMode == IO_CONSOLE && !g_ConInitialized)
|
|
{
|
|
CreateConsole();
|
|
}
|
|
|
|
va_start(Args, Format);
|
|
_vsnprintf(g_Buffer, DIMA(g_Buffer), Format, Args);
|
|
g_Buffer[DIMA(g_Buffer) - 1] = 0;
|
|
va_end(Args);
|
|
|
|
ConOutStr(g_Buffer);
|
|
}
|
|
|
|
void
|
|
ConClear(void)
|
|
{
|
|
if (g_IoMode != IO_CONSOLE)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// If no attempt has been made to create a console
|
|
// go ahead and try now.
|
|
if (g_IoMode == IO_CONSOLE && !g_ConInitialized)
|
|
{
|
|
CreateConsole();
|
|
}
|
|
|
|
HANDLE Console;
|
|
CONSOLE_SCREEN_BUFFER_INFO ConsoleScreenInfo;
|
|
DWORD Done;
|
|
DWORD Chars;
|
|
|
|
Console = GetStdHandle(STD_OUTPUT_HANDLE);
|
|
if (!GetConsoleScreenBufferInfo(Console, &ConsoleScreenInfo))
|
|
{
|
|
return;
|
|
}
|
|
|
|
ConsoleScreenInfo.dwCursorPosition.X = 0;
|
|
ConsoleScreenInfo.dwCursorPosition.Y = 0;
|
|
|
|
Chars = ConsoleScreenInfo.dwSize.Y * ConsoleScreenInfo.dwSize.X;
|
|
|
|
FillConsoleOutputCharacterA(Console, ' ', Chars,
|
|
ConsoleScreenInfo.dwCursorPosition,
|
|
&Done);
|
|
FillConsoleOutputAttribute(Console, ConsoleScreenInfo.wAttributes,
|
|
Chars, ConsoleScreenInfo.dwCursorPosition,
|
|
&Done);
|
|
SetConsoleCursorPosition(Console, ConsoleScreenInfo.dwCursorPosition);
|
|
}
|
|
|
|
void
|
|
ExitDebugger(ULONG Code)
|
|
{
|
|
if (g_DbgClient != NULL)
|
|
{
|
|
if (g_RemoteClient)
|
|
{
|
|
// Disconnect from server.
|
|
g_DbgClient->EndSession(DEBUG_END_DISCONNECT);
|
|
}
|
|
else
|
|
{
|
|
g_DbgClient->EndSession(DEBUG_END_PASSIVE);
|
|
// Force servers to get cleaned up.
|
|
g_DbgClient->EndSession(DEBUG_END_REENTRANT);
|
|
}
|
|
}
|
|
|
|
ExitProcess(Code);
|
|
}
|
|
|
|
void
|
|
ErrorExit(PCSTR Format, ...)
|
|
{
|
|
if (Format != NULL)
|
|
{
|
|
DWORD Len;
|
|
va_list Args;
|
|
|
|
// If no attempt has been made to create a console
|
|
// go ahead and try now.
|
|
if (g_IoRequested == IO_CONSOLE && !g_ConInitialized)
|
|
{
|
|
CreateConsole();
|
|
}
|
|
|
|
va_start(Args, Format);
|
|
Len = _vsnprintf(g_Buffer, DIMA(g_Buffer), Format, Args);
|
|
va_end(Args);
|
|
g_Buffer[DIMA(g_Buffer) - 1] = 0;
|
|
if ((int)Len < 0 || Len == DIMA(g_Buffer))
|
|
{
|
|
Len = DIMA(g_Buffer) - 1;
|
|
}
|
|
|
|
if (g_ConOutput != NULL)
|
|
{
|
|
WriteFile(g_ConOutput, g_Buffer, Len, &Len, NULL);
|
|
}
|
|
else
|
|
{
|
|
OutputDebugString(g_Buffer);
|
|
}
|
|
}
|
|
|
|
#ifndef INHERIT_CONSOLE
|
|
if (g_IoRequested == IO_CONSOLE)
|
|
{
|
|
ConOut("%s: exiting - press enter ---", g_DebuggerName);
|
|
g_InitialCommand = NULL;
|
|
g_InputFile = NULL;
|
|
ConIn(g_Buffer, sizeof(g_Buffer), TRUE);
|
|
}
|
|
#endif
|
|
|
|
ExitDebugger(E_FAIL);
|
|
}
|
|
|
|
DWORD WINAPI
|
|
InputThreadLoop(PVOID Param)
|
|
{
|
|
DWORD Read;
|
|
BOOL Status;
|
|
UCHAR Char;
|
|
BOOL NewLine = TRUE;
|
|
BOOL SpecialChar = FALSE;
|
|
HANDLE ConIn = g_ConInput;
|
|
HRESULT Hr;
|
|
BOOL ShowInputError = TRUE;
|
|
BOOL PipeInput;
|
|
|
|
// Create interfaces usable on this thread.
|
|
if ((Hr = g_DbgClient->CreateClient(&g_ConClient)) != S_OK ||
|
|
(Hr = g_ConClient->QueryInterface(IID_IDebugControl,
|
|
(void **)&g_ConControl)) != S_OK)
|
|
{
|
|
ConOut("%s: Unable to create input thread interfaces, 0x%X\n",
|
|
g_DebuggerName, Hr);
|
|
// Force servers to get cleaned up or disconnect from server.
|
|
g_DbgClient->EndSession(g_RemoteClient ?
|
|
DEBUG_END_DISCONNECT : DEBUG_END_REENTRANT);
|
|
ExitProcess(E_FAIL);
|
|
}
|
|
|
|
PipeInput = GetFileType(ConIn) == FILE_TYPE_PIPE;
|
|
|
|
//
|
|
// Capture all typed input immediately.
|
|
// Stuff the characters into an anonymous pipe, from which
|
|
// ConIn will read them.
|
|
//
|
|
|
|
for (;;)
|
|
{
|
|
#ifndef KERNEL
|
|
// The debugger should only read the console when the
|
|
// debuggee isn't running to avoid eating up input
|
|
// intended for the debuggee.
|
|
if (!g_RemoteClient && NewLine)
|
|
{
|
|
if (WaitForSingleObject(g_AllowInput,
|
|
INFINITE) != WAIT_OBJECT_0)
|
|
{
|
|
ConOut("%s: Failed to wait for input window, %d\n",
|
|
GetLastError());
|
|
}
|
|
|
|
NewLine = FALSE;
|
|
}
|
|
#endif
|
|
|
|
//
|
|
// The CRT does GetFileType's on all standard I/O handles
|
|
// when initializing. GetFileType counts as a synchronous
|
|
// I/O so if the handle happens to be a pipe and the pipe
|
|
// already has a sync I/O on it the GetFileType will block
|
|
// until the first I/O is satisfied.
|
|
//
|
|
// When this separate I/O thread is running it's normally
|
|
// blocked in a ReadFile, causing any GetFileType call
|
|
// on the handle to also block until there's some input.
|
|
// In order to avoid this we detect that the input is
|
|
// a pipe and use PeekNamedPipe to delay the ReadFile
|
|
// until there's some input available.
|
|
//
|
|
|
|
if (PipeInput)
|
|
{
|
|
for (;;)
|
|
{
|
|
ULONG Avail;
|
|
|
|
if (PeekNamedPipe(ConIn, NULL, 0, NULL, &Avail, NULL) &&
|
|
Avail > 0)
|
|
{
|
|
break;
|
|
}
|
|
|
|
Sleep(10);
|
|
}
|
|
}
|
|
|
|
Status = ReadFile(ConIn, &Char, sizeof(Char), &Read, NULL);
|
|
if (!Status || Read != sizeof(Char))
|
|
{
|
|
if (ShowInputError &&
|
|
GetLastError() != ERROR_OPERATION_ABORTED &&
|
|
GetLastError() != ERROR_IO_PENDING)
|
|
{
|
|
ConOut("%s: Could not read from console, %d\n",
|
|
g_DebuggerName, GetLastError());
|
|
ShowInputError = FALSE;
|
|
}
|
|
|
|
// The most common cause of a console read failure
|
|
// is killing remote with @K. Give things some
|
|
// time to kill this process.
|
|
// If this is a remote server it's possible that
|
|
// the debugger was run without a valid console
|
|
// and is just being accessed via remoting.
|
|
// Sleep longer in that case since errors will
|
|
// probably always occur.
|
|
Sleep(!g_RemoteClient && g_RemoteOptions != NULL ?
|
|
1000 : 50);
|
|
continue;
|
|
}
|
|
|
|
// We successfully got some input so if it
|
|
// fails later we should show a fresh error.
|
|
ShowInputError = TRUE;
|
|
|
|
if (CheckForControlCommands(g_ConClient, g_ConControl, Char))
|
|
{
|
|
SpecialChar = TRUE;
|
|
continue;
|
|
}
|
|
|
|
if (SpecialChar && Char == '\r')
|
|
{
|
|
// If we get a CR immediately after a special
|
|
// char turn it into a space so that it doesn't cause
|
|
// a command repeat.
|
|
Char = ' ';
|
|
}
|
|
|
|
SpecialChar = FALSE;
|
|
|
|
ULONG Len;
|
|
Status = WriteFile(g_PipeWrite, &Char, sizeof(Char), &Len,
|
|
&g_PipeWriteOverlapped);
|
|
if (!Status && GetLastError() != ERROR_IO_PENDING)
|
|
{
|
|
ConOut("%s: Could not write to pipe, %d\n",
|
|
g_DebuggerName, GetLastError());
|
|
}
|
|
else if (Char == '\n')
|
|
{
|
|
EnterCriticalSection(&g_InputLock);
|
|
|
|
InterlockedIncrement(&g_Lines);
|
|
|
|
// If input is needed send it directly
|
|
// to the engine.
|
|
if (g_InputStarted)
|
|
{
|
|
ReadPromptInputChars(g_Buffer, sizeof(g_Buffer));
|
|
g_ConControl->ReturnInput(g_Buffer);
|
|
g_InputStarted = FALSE;
|
|
}
|
|
else
|
|
{
|
|
// Wake up the engine thread when a line of
|
|
// input is present.
|
|
g_ConClient->ExitDispatch(g_DbgClient);
|
|
}
|
|
|
|
LeaveCriticalSection(&g_InputLock);
|
|
|
|
NewLine = TRUE;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
CreateInputThread(void)
|
|
{
|
|
HANDLE Thread;
|
|
DWORD ThreadId;
|
|
CHAR PipeName[256];
|
|
|
|
if (g_PipeWrite != NULL)
|
|
{
|
|
// Input thread already exists.
|
|
return;
|
|
}
|
|
|
|
#ifndef KERNEL
|
|
g_AllowInput = CreateEvent(NULL, FALSE, FALSE, NULL);
|
|
if (g_AllowInput == NULL)
|
|
{
|
|
ErrorExit("Unable to create input event, %d\n", GetLastError());
|
|
}
|
|
#endif
|
|
|
|
_snprintf(PipeName,
|
|
sizeof(PipeName),
|
|
"\\\\.\\pipe\\Dbg%d",
|
|
GetCurrentProcessId());
|
|
PipeName[sizeof(PipeName) - 1] = 0;
|
|
|
|
g_PipeWrite = CreateNamedPipe(PipeName,
|
|
PIPE_ACCESS_DUPLEX |
|
|
FILE_FLAG_OVERLAPPED,
|
|
PIPE_TYPE_BYTE | PIPE_READMODE_BYTE |
|
|
PIPE_WAIT,
|
|
2,
|
|
2000,
|
|
2000,
|
|
NMPWAIT_WAIT_FOREVER,
|
|
NULL);
|
|
if (g_PipeWrite == INVALID_HANDLE_VALUE)
|
|
{
|
|
ErrorExit("Failed to create input pipe, %d\n",
|
|
GetLastError());
|
|
}
|
|
|
|
g_PromptInput = CreateFile(PipeName,
|
|
GENERIC_READ,
|
|
FILE_SHARE_READ,
|
|
NULL,
|
|
OPEN_EXISTING,
|
|
FILE_ATTRIBUTE_NORMAL,
|
|
NULL);
|
|
if (g_PromptInput == INVALID_HANDLE_VALUE)
|
|
{
|
|
ErrorExit("Failed to create read pipe, %d\n",
|
|
GetLastError());
|
|
}
|
|
|
|
Thread = CreateThread(NULL,
|
|
16000, // THREAD_STACK_SIZE
|
|
InputThreadLoop,
|
|
NULL,
|
|
THREAD_SET_INFORMATION,
|
|
&ThreadId);
|
|
if (Thread == NULL)
|
|
{
|
|
ErrorExit("Failed to create input thread, %d\n",
|
|
GetLastError());
|
|
}
|
|
else
|
|
{
|
|
if (!SetThreadPriority(Thread, THREAD_PRIORITY_ABOVE_NORMAL))
|
|
{
|
|
ErrorExit("Failed to raise the input thread priority, %d\n",
|
|
GetLastError());
|
|
}
|
|
}
|
|
|
|
CloseHandle(Thread);
|
|
|
|
// Wait for thread initialization. Callbacks are
|
|
// already registered so we need to dispatch them
|
|
// while waiting.
|
|
while (g_ConControl == NULL)
|
|
{
|
|
g_DbgClient->DispatchCallbacks(50);
|
|
}
|
|
}
|