【发布时间】:2013-07-04 11:01:35
【问题描述】:
我正在使用 Windows 7 64 位机器(我有管理员权限)。
我正在使用 Python 2.7(64 位)和 PyDev ctypes for Eclipse 来尝试读取与特定 PID 关联的所有线程中的寄存器值(尝试了在 64 位和 32 位模式下运行的进程的 PID),但是当我这样做时,寄存器的值全部归零。当我使用Wow64GetThreadContext 时,调用失败,GetLastError 返回 0x00000057(根据 MSDN 的“无效参数”)
我成功地附加到进程,枚举线程(通过CreateToolhelp32Snapshot),找到具有适当PID 的进程拥有的线程,并尝试获取线程上下文。这是我打开线程并获取线程上下文的代码:
打开一个线程:
def open_thread(self, thread_id):
h_thread = kernel32.OpenThread(THREAD_ALL_ACCESS, None, thread_id)
获取上下文:
def get_thread_context(self, thread_id = None, h_thread = None):
context = CONTEXT()
context.ContextFlags = CONTEXT_FULL | CONTEXT_DEBUG_REGISTERS
#alternatively, for 64
context64 = WOW64_CONTEXT()
context64.ContextFlags = CONTEXT_FULL | CONTEXT_DEBUG_REGISTERS
#Obtain a handle to the thread
if h_thread is None:
self.h_thread = self.open_thread(thread_id)
kernel32.SuspendThread(self.h_thread)
if kernel32.GetThreadContext(self.h_thread, byref(context)):
kernel32.ResumeThread(self.h_thread)
return context
else:
kernel32.ResumeThread(self.h_thread)
return False
我使用以下代码调用此代码:
debugger.attach(int(pid))
#debugger.run()
list = debugger.enumerate_threads()
for thread in list:
thread_context = debugger.get_thread_context(thread)
if thread_context == False:
print "[*] Thread context is false..."
else:
print "[*] Dumping registers for thread ID: 0x%08x" % thread
print "[**] Eip: 0x%016x" % thread_context.Eip
print "[**] Esp: 0x%016x" % thread_context.Esp
print "[**] Ebp: 0x%016x" % thread_context.Ebp
print "[**] Eax: 0x%016x" % thread_context.Eax
print "[**] Ebx: 0x%016x" % thread_context.Ebx
print "[**] Ecx: 0x%016x" % thread_context.Ecx
print "[**] Edx: 0x%016x" % thread_context.Edx
print "[*] End DUMP"
debugger.detach()
当我使用 GetThreadContext 和 CONTEXT 结构运行此代码时,我为每个线程取回了上下文对象,但寄存器值全为零。
我尝试将GetThreadContext 替换为Wow64GetThreadContext(以及分别将SuspendThread 替换为Wow64SuspendThread),但是当我这样做时,调用失败并出现错误“参数无效”。我提供给Wow64GetThreadContext 的参数与我提供给GetThreadContext 的参数相同,除了我提供的代码中的变量名称(这是因为当我在 WinNT.h 中查看它们的定义时,它们是等价的(除非我遗漏了什么)。我已经通过以下方式定义了这些结构:
class WOW64_CONTEXT(Structure):
_fields_ = [
("ContextFlags", DWORD),
("Dr0", DWORD),
("Dr1", DWORD),
("Dr2", DWORD),
("Dr3", DWORD),
("Dr6", DWORD),
("Dr7", DWORD),
("FloatSave", WOW64_FLOATING_SAVE_AREA),
("SegGs", DWORD),
("SegFs", DWORD),
("SegEs", DWORD),
("SegDs", DWORD),
("Edi", DWORD),
("Esi", DWORD),
("Ebx", DWORD),
("Edx", DWORD),
("Ecx", DWORD),
("Eax", DWORD),
("Ebp", DWORD),
("Eip", DWORD),
("SegCs", DWORD),
("EFlags", DWORD),
("Esp", DWORD),
("SegSs", DWORD),
("ExtendedRegisters", BYTE * 512),
]
class WOW64_FLOATING_SAVE_AREA(Structure):
_fields_ = [
("ControlWord", DWORD),
("StatusWord", DWORD),
("TagWord", DWORD),
("ErrorOffset", DWORD),
("ErrorSelector", DWORD),
("DataOffset", DWORD),
("DataSelector", DWORD),
("RegisterArea", BYTE * 80),
("Cr0NpxState", DWORD),
]
class CONTEXT(Structure):
_fields_ = [
("ContextFlags", DWORD),
("Dr0", DWORD),
("Dr1", DWORD),
("Dr2", DWORD),
("Dr3", DWORD),
("Dr6", DWORD),
("Dr7", DWORD),
("FloatSave", FLOATING_SAVE_AREA),
("SegGs", DWORD),
("SegFs", DWORD),
("SegEs", DWORD),
("SegDs", DWORD),
("Edi", DWORD),
("Esi", DWORD),
("Ebx", DWORD),
("Edx", DWORD),
("Ecx", DWORD),
("Eax", DWORD),
("Ebp", DWORD),
("Eip", DWORD),
("SegCs", DWORD),
("EFlags", DWORD),
("Esp", DWORD),
("SegSs", DWORD),
("ExtendedRegisters", BYTE * 512),
]
class FLOATING_SAVE_AREA(Structure):
_fields_ = [
("ControlWord", DWORD),
("StatusWord", DWORD),
("TagWord", DWORD),
("ErrorOffset", DWORD),
("ErrorSelector", DWORD),
("DataOffset", DWORD),
("DataSelector", DWORD),
("RegisterArea", BYTE * 80),
("Cr0NpxState", DWORD),
]
我已经在这个问题上进行了大量的谷歌搜索,并尝试了以下方法无济于事:
根据 MSDN 上的评论:
CONTEXT_FULL应该是CONTEXT_AMD64 | CONTEXT_CONTROL | CONTEXT_INTEGER | CONTEXT_FLOATING_POINT可以在 Win64 下正常使用。我尝试重命名我的 CONTEXT 和 WOW_64CONTEXT 中的寄存器 通过将寄存器名称中的“E”替换为“R”(Eax -> 拉克斯等)
有没有其他人使用 Python 和 ctypes 成功获取 Windows 上 64 位线程的上下文?
【问题讨论】:
-
@cghlke 是的,我只是尝试过,但无济于事。我仍然看到 0x57 错误代码。
-
go4answers.webhost4life.com/Example/… : google 结果 (0x57 = 87) 表明这不太可能与 Python 相关
-
我同意。有谁知道如何解决这个问题,或者上面发布的代码是否不正确/不完整?
标签: python windows 64-bit pydev threadcontext