我正在为一位小型的solitär教练工作。我不知道为什么函数ReadProcessMemory不起作用。通常它返回False或True但在这种情况下没有。 GetlastError()给出了Errorcode 6。
#-*- coding: cp1252 -*-
import ctypes, win32ui, win32process ,win32api
PROCESS_ALL_ACCESS = 0x1F0FFF
HWND = win32ui.FindWindow(None,"Solitär").GetSafeHwnd()
print(HWND)
PID = win32process.GetWindowThreadProcessId(HWND)[1]
print(PID)
PROCESS = win32api.OpenProcess(PROCESS_ALL_ACCESS,0,PID).handle
rPM = ctypes.windll.kernel32.ReadProcessMemory
wPM = ctypes.windll.kernel32.WriteProcessMemory
ADDRESS1 = 0x00E97074
ADDRESS2 = ctypes.create_string_buffer(64)
pi = ctypes.pointer(ADDRESS2)
rPM(PROCESS,ADDRESS1,ADDRESS2,64,0)
print(ADDRESS2)
x=ctypes.windll.kernel32.GetLastError()
print(x)
答案 0 :(得分:12)
检查社区对MSDN ReadProcessMemory页面的评论,引用(原文如此):
W7不会运行读取进程内存
您可能需要检查“SE_DEBUG_NAME”对当前进程令牌的访问权限。如果未启用。启用它。这必须以管理员身份完成。
同样完全声明返回类型并使用use_last_error
参数,其中ctypes
将在调用后直接在内部缓存GetLastError()
值。否则,可能是不正确的。如果您使用的是64位系统,则SIZE_T和指针是64位值,因此ctypes需要知道为调用正确设置堆栈的类型。
...
from ctypes import wintypes
...
rPM = ctypes.WinDLL('kernel32',use_last_error=True).ReadProcessMemory
rPM.argtypes = [wintypes.HANDLE,wintypes.LPCVOID,wintypes.LPVOID,ctypes.c_size_t,ctypes.POINTER(ctypes.c_size_t)]
rPM.restype = wintypes.BOOL
wPM = ctypes.WinDLL('kernel32',use_last_error=True).WriteProcessMemory
wPM.argtypes = [wintypes.HANDLE,wintypes.LPVOID,wintypes.LPCVOID,ctypes.c_size_t,ctypes.POINTER(ctypes.c_size_t)]
wPM.restype = wintypes.BOOL
ADDRESS1 = 0x00E97074
ADDRESS2 = ctypes.create_string_buffer(64)
bytes_read = ctypes.c_size_t()
print(rPM(PROCESS,ADDRESS1,ADDRESS2,64,ctypes.byref(bytes_read)))
print(ctypes.get_last_error())
另外,仅供参考,即使有了所有修复,我也会得到相同的错误值,但我没有遇到启用SE_DEBUG_NAME
的问题。
<强>解决强>
以下问题是:
PROCESS = win32api.OpenProcess(PROCESS_ALL_ACCESS,0,PID).handle
win32api.OpenProcess
返回一个临时的PyHANDLE
,在检索到句柄后会被销毁并关闭句柄。
解决方案是使用:
PROCESS = win32api.OpenProcess(PROCESS_ALL_ACCESS,0,PID)
...
rPM(PROCESS.handle,ADDRESS1,ADDRESS2,64,0)
PROCESS
然后保存PyHANDLE
对象,句柄仍然有效。