/*++ Copyright (c) 1999-2000 Microsoft Corporation Module Name: w64cpuex.cpp Abstract: Debugger extension DLL for debugging the CPU Author: 27-Sept-1999 BarryBo Revision History: --*/ #define _WOW64CPUDBGAPI_ #define DECLARE_CPU_DEBUGGER_INTERFACE #include #include #include #include #include #include #include #if defined _X86_ #define WOW64EXTS_386 #endif #include "wow64.h" #include "wow64cpu.h" #include "ia64cpu.h" // Safe release and NULL. #define EXT_RELEASE(Unk) \ ((Unk) != NULL ? ((Unk)->Release(), (Unk) = NULL) : NULL) #define CONTEXT_OFFSET FIELD_OFFSET(TEB64, TlsSlots[WOW64_TLS_CPURESERVED]) CPUCONTEXT LocalCpuContext; PWOW64GETCPUDATA CpuGetData; PDEBUG_ADVANCED g_ExtAdvanced; PDEBUG_CLIENT g_ExtClient; PDEBUG_CONTROL g_ExtControl; PDEBUG_DATA_SPACES g_ExtData; PDEBUG_REGISTERS g_ExtRegisters; PDEBUG_SYMBOLS g_ExtSymbols; PDEBUG_SYSTEM_OBJECTS g_ExtSystem; void ExtRelease(void) { g_ExtClient = NULL; EXT_RELEASE(g_ExtAdvanced); EXT_RELEASE(g_ExtControl); EXT_RELEASE(g_ExtData); EXT_RELEASE(g_ExtRegisters); EXT_RELEASE(g_ExtSymbols); EXT_RELEASE(g_ExtSystem); } // Queries for all debugger interfaces. HRESULT ExtQuery(PDEBUG_CLIENT Client) { HRESULT Status; if ((Status = Client->QueryInterface(__uuidof(IDebugAdvanced), (void **)&g_ExtAdvanced)) != S_OK) { goto Fail; } if ((Status = Client->QueryInterface(__uuidof(IDebugControl), (void **)&g_ExtControl)) != S_OK) { goto Fail; } if ((Status = Client->QueryInterface(__uuidof(IDebugDataSpaces), (void **)&g_ExtData)) != S_OK) { goto Fail; } if ((Status = Client->QueryInterface(__uuidof(IDebugRegisters), (void **)&g_ExtRegisters)) != S_OK) { goto Fail; } if ((Status = Client->QueryInterface(__uuidof(IDebugSymbols), (void **)&g_ExtSymbols)) != S_OK) { goto Fail; } if ((Status = Client->QueryInterface(__uuidof(IDebugSystemObjects), (void **)&g_ExtSystem)) != S_OK) { goto Fail; } g_ExtClient = Client; return S_OK; Fail: ExtRelease(); return Status; } // Normal output. void __cdecl ExtOut(PCSTR Format, ...) { va_list Args; va_start(Args, Format); g_ExtControl->OutputVaList(DEBUG_OUTPUT_NORMAL, Format, Args); va_end(Args); } // Error output. void __cdecl ExtErr(PCSTR Format, ...) { va_list Args; va_start(Args, Format); g_ExtControl->OutputVaList(DEBUG_OUTPUT_ERROR, Format, Args); va_end(Args); } // Warning output. void __cdecl ExtWarn(PCSTR Format, ...) { va_list Args; va_start(Args, Format); g_ExtControl->OutputVaList(DEBUG_OUTPUT_WARNING, Format, Args); va_end(Args); } // Verbose output. void __cdecl ExtVerb(PCSTR Format, ...) { va_list Args; va_start(Args, Format); g_ExtControl->OutputVaList(DEBUG_OUTPUT_VERBOSE, Format, Args); va_end(Args); } WOW64CPUDBGAPI VOID CpuDbgInitEngapi( PWOW64GETCPUDATA lpGetData ) { CpuGetData = lpGetData; } HRESULT EngGetContextThread( IN OUT PCONTEXT32 Context) /*++ Routine Description: This routine extract the context record of any thread. This is a generic routine. When entered, if the target thread isn't the current thread, then it should be guaranteed that the target thread is suspended at a proper CPU state. Arguments: Context - Context record to fill Return Value: HRESULT. --*/ { HRESULT hr; NTSTATUS NtStatus; CONTEXT ContextEM; ULONG64 CpuRemoteContext; ULONG64 Teb; CPUCONTEXT CpuContext; ContextEM.ContextFlags = CONTEXT_FULL; hr = g_ExtSystem->GetCurrentThreadTeb(&Teb); if (FAILED(hr)) { return hr; } hr = g_ExtData->ReadVirtual(Teb + CONTEXT_OFFSET, &CpuRemoteContext, sizeof(CpuRemoteContext), NULL); if (FAILED(hr)) { return hr; } hr = g_ExtData->ReadVirtual(CpuRemoteContext, &CpuContext, sizeof(CpuContext), NULL); if (FAILED(hr)) { return hr; } NtStatus = GetContextRecord(&CpuContext, Context); if (!NT_SUCCESS(NtStatus)) { return E_FAIL; } return hr; } WOW64CPUDBGAPI BOOL CpuDbgGetRemoteContext( PDEBUG_CLIENT Client, PVOID CpuData ) { BOOL bRet = FALSE; HRESULT hr; CONTEXT Context; hr = ExtQuery(Client); if (FAILED(hr)) { return FALSE; } LocalCpuContext.Context.ContextFlags = CONTEXT32_FULLFLOAT; hr = EngGetContextThread(&LocalCpuContext.Context); if (FAILED(hr)) { goto Done; } bRet = TRUE; Done: ExtRelease(); return bRet; } HRESULT EngSetContextThread( IN OUT PCONTEXT32 Context) /*++ Routine Description: This routine sets the context record of any thread. This is a generic routine. When entered, if the target thread isn't the currently executing thread, then it should be guaranteed that the target thread is suspended at a proper CPU state. Arguments: Context - Context record to set Return Value: HRESULT. --*/ { HRESULT hr; NTSTATUS NtStatus; CONTEXT ContextEM; ULONG64 CpuRemoteContext; ULONG64 Teb; CPUCONTEXT CpuContext; ContextEM.ContextFlags = CONTEXT_FULL; hr = g_ExtAdvanced->SetThreadContext(&ContextEM, sizeof(ContextEM)); hr = g_ExtSystem->GetCurrentThreadTeb(&Teb); if (FAILED(hr)) { return hr; } hr = g_ExtData->ReadVirtual(Teb + CONTEXT_OFFSET, &CpuRemoteContext, sizeof(CpuRemoteContext), NULL); if (FAILED(hr)) { return hr; } hr = g_ExtData->ReadVirtual(CpuRemoteContext, &CpuContext, sizeof(CpuContext), NULL); if (FAILED(hr)) { return hr; } NtStatus = SetContextRecord(&CpuContext, Context); if (!NT_SUCCESS(NtStatus)) { return E_FAIL; } hr = g_ExtData->WriteVirtual(CpuRemoteContext, &CpuContext, sizeof(CpuContext), NULL); return hr; } WOW64CPUDBGAPI BOOL CpuDbgSetRemoteContext( PDEBUG_CLIENT Client ) { BOOL bRet = FALSE; HRESULT hr; NTSTATUS Status; CONTEXT Context; hr = ExtQuery(Client); if (FAILED(hr)) { return FALSE; } LocalCpuContext.Context.ContextFlags = CONTEXT32_FULLFLOAT; hr = EngSetContextThread(&LocalCpuContext.Context); if (FAILED(hr)) { ExtOut("CpuDbgSetRemoteContext: Error %x writing CPU context\n", hr); goto Done; } bRet = TRUE; Done: ExtRelease(); return bRet; } WOW64CPUDBGAPI BOOL CpuDbgGetLocalContext( PDEBUG_CLIENT Client, PCONTEXT32 Context ) { return NT_SUCCESS(GetContextRecord(&LocalCpuContext, Context)); } WOW64CPUDBGAPI BOOL CpuDbgSetLocalContext( PDEBUG_CLIENT Client, PCONTEXT32 Context ) { return NT_SUCCESS(SetContextRecord(&LocalCpuContext, Context)); } WOW64CPUDBGAPI VOID CpuDbgFlushInstructionCacheWithHandle( HANDLE Process, PVOID Addr, DWORD Length ) { NtFlushInstructionCache((HANDLE)Process, Addr, Length); } WOW64CPUDBGAPI VOID CpuDbgFlushInstructionCache( PDEBUG_CLIENT Client, PVOID Addr, DWORD Length ) { HRESULT hr; ULONG64 Process; hr = ExtQuery(Client); if (FAILED(hr)) { return; } hr = g_ExtSystem->GetCurrentProcessHandle(&Process); if (FAILED(hr)) { ExtOut("CpuDbgFlushInstructionCache: failed to get Process Handle!\n"); return; } CpuDbgFlushInstructionCacheWithHandle((HANDLE)Process, Addr, Length); ExtRelease(); } VOID SetEax(ULONG ul) { LocalCpuContext.Context.Eax = ul; } VOID SetEbx(ULONG ul) { LocalCpuContext.Context.Ebx = ul; } VOID SetEcx(ULONG ul) { LocalCpuContext.Context.Ecx = ul; } VOID SetEdx(ULONG ul) { LocalCpuContext.Context.Edx = ul; } VOID SetEsi(ULONG ul) { LocalCpuContext.Context.Esi = ul; } VOID SetEdi(ULONG ul) { LocalCpuContext.Context.Edi = ul; } VOID SetEbp(ULONG ul) { LocalCpuContext.Context.Ebp = ul; } VOID SetEsp(ULONG ul) { LocalCpuContext.Context.Esp = ul; } VOID SetEip(ULONG ul) { LocalCpuContext.Context.Eip = ul; } VOID SetEfl(ULONG ul) { LocalCpuContext.Context.EFlags = ul; } ULONG GetEax(VOID) { return LocalCpuContext.Context.Eax; } ULONG GetEbx(VOID) { return LocalCpuContext.Context.Ebx; } ULONG GetEcx(VOID) { return LocalCpuContext.Context.Ecx; } ULONG GetEdx(VOID) { return LocalCpuContext.Context.Edx; } ULONG GetEsi(VOID) { return LocalCpuContext.Context.Esi; } ULONG GetEdi(VOID) { return LocalCpuContext.Context.Edi; } ULONG GetEbp(VOID) { return LocalCpuContext.Context.Ebp; } ULONG GetEsp(VOID) { return LocalCpuContext.Context.Esp; } ULONG GetEip(VOID) { return LocalCpuContext.Context.Eip; } ULONG GetEfl(VOID) { return LocalCpuContext.Context.EFlags; } CPUREGFUNCS CpuRegFuncs[] = { { "Eax", SetEax, GetEax }, { "Ebx", SetEbx, GetEbx }, { "Ecx", SetEcx, GetEcx }, { "Edx", SetEdx, GetEdx }, { "Esi", SetEsi, GetEsi }, { "Edi", SetEdi, GetEdi }, { "Ebp", SetEbp, GetEbp }, { "Esp", SetEsp, GetEsp }, { "Eip", SetEip, GetEip }, { "Efl", SetEfl, GetEfl }, { NULL, NULL, NULL} }; WOW64CPUDBGAPI PCPUREGFUNCS CpuDbgGetRegisterFuncs( void ) { return CpuRegFuncs; }