问题描述
经过数小时的研究和主题分析,我找不到正确的解释和方法来正确解决我的问题。 如果您找到了答案,在此先非常感谢您。
我的问题摘要:
我正在使用 python 3.8(64 位版本)和 dll(也是 64 位版本)。而且,尽管由于指针问题(根据我的一点理解)具有 ctypes 功能(argstype、restype..etc),但我无法将其正确包装到我的 python 代码中。
让我烦恼的是,32 位的同一个 DLL 和 python 32 位在我的 python 代码中运行良好。
请在此之后找到:
C 函数原型(来自 API 描述): WORD mpc_AddToScenarioPcd (BYTE,DWORD,POINTER(BYTE),VOID*)
我的 Python 代码来包装这个(64 位版本和 32 位版本注释):
from ctypes import *
# DLL Loading ---------------------------------------------------------------------------------------
# 64 Bits -------------------------------------------------------------------------------------------
dllpath = '%s/MP300Com64.dll' % str(os.path.dirname(__file__))
MP300Dll = WinDLL(dllpath)
MP300Dll_CDECL = CDLL(dllpath)
# 64 Bits -------------------------------------------------------------------------------------------
# dllpath = '%s/MP300Com.dll' % str(os.path.dirname(__file__))
# MP300Dll = WinDLL(dllpath)
# MP300Dll_CDECL = CDLL(dllpath)
# Keyword Type deFinition ---------------------------------------------------------------------------
WORD = c_ushort
DWORD = c_ulong
BYTE = c_ubyte
# Function Wrapped ----------------------------------------------------------------------------------
mpc_AddToScenarioPcd = MP300Dll_CDECL.MPC_AddToScenarioPcd
mpc_AddToScenarioPcd.argtypes = [BYTE,POINTER(DWORD)]
mpc_AddToScenarioPcd.restype = WORD
# -- Creation of the First Pointer to the array of Bytes --------------------------------------------
array1 = (BYTE* len(5))(*[BYTE(1),BYTE(2),BYTE(3),BYTE(4),BYTE(5)])
array2 = (BYTE* len(5))(*[BYTE(10),BYTE(20),BYTE(30),BYTE(40),BYTE(50)])
# -- Pointer cast of the array1 and 2 ---------------------------------------------------------------
cast(array1,POINTER(BYTE))
cast(array2,POINTER(BYTE))
# VOID* obj creation --------------------------------------------------------------------------------
param = c_void_p()
# -- Function Usage through Python Code -------------------------------------------------------------
mpc_AddToScenarioPcd(BYTE(0),DWORD(0),(22),DWORD(1),DWORD(2),array1,DWORD(10),array2,param)
使用 Python 3.4(32 位版本)和 DLL 32 位:
=> 一切正常,没有错误。
使用 Python 3.8(64 位版本)和 DLL 64 位:
=> 我收到此异常:OSError:异常:访问冲突读取 0x000000005EC0B808
我需要使用 64 位版本,因此我被这个问题阻止了...如果有人可以向我解释我的指针(或类型用法..)在 64 位中有什么问题,并且,怎么解决,万分感谢!!
再次非常感谢您在我的主题上的所有专业知识和工作,
解决方法
以下是适用于 32 位和 64 位 Python 的示例 DLL 代码和最少的 ctypes
代码。请注意以下几点:
-
ctypes.wintypes
中有预定义的 Windows 类型。 - 当
.argtypes
和.restype
设置正确时,会根据需要在 Python 和 C 之间检查和转换参数和类型的数量。通常不需要额外的转换和包装。 -
CDLL
对应于 32 位__cdecl
调用约定。WinDLL
对应于 32 位__stdcall
。在 64 位中,只有一种调用约定。它在 Linux 和 Windows 之间的实现方式不同,但仍然只有一个,因此CDLL
和WinDLL
都可以工作,但使用正确的一个以实现 32 位可移植性。
test.c
#include <windows.h>
#include <stdio.h>
__declspec(dllexport)
WORD mpc_AddToScenarioPcd(BYTE b,DWORD d1,DWORD d2,DWORD d3,DWORD d4,DWORD d5,DWORD d6,DWORD d7,LPBYTE pb1,DWORD d8,LPBYTE pb2,LPVOID pv) {
printf("%hhu %u %u %u %u %u %u %u %p %u %p %p\n",b,d1,d2,d3,d4,d5,d6,d7,pb1,d8,pb2,pv);
for(int i = 0; i < 5; ++i)
printf("%hhu %hhu\n",pb1[i],pb2[i]);
return 1234;
}
test.py
from ctypes import *
from ctypes.wintypes import * # pre-defined Windows types
dll = CDLL('./test')
dll.mpc_AddToScenarioPcd.argtypes = BYTE,DWORD,LPBYTE,LPVOID
dll.mpc_AddToScenarioPcd.restype = WORD
b1 = (BYTE * 5)(1,2,3,4,5)
b2 = (BYTE * 5)(10,20,30,40,50)
print(f'{addressof(b1):016X} {addressof(b2):016X}')
result = dll.mpc_AddToScenarioPcd(100,1,5,6,7,b1,8,b2,None)
print(result)
输出
注意 Python 和 C 中打印的地址是相同的。
00000194D0447A90 00000194D0447C10
100 1 2 3 4 5 6 7 00000194D0447A90 8 00000194D0447C10 0000000000000000
1 10
2 20
3 30
4 40
5 50
1234