0

经过数小时的研究和主题分析,我找不到正确解决问题的解释和方法。如果您找到答案,非常感谢您提前。

我的问题摘要

我正在使用 python 3.8(64 位版本)和一个 dll(64 位版本)。而且,由于指针问题(据我所知),尽管有 ctypes 功能(argstype、restype..etc),但我无法将其正确包装到我的 python 代码中。

令我困扰的是,相同的 32 位 DLL 和 python 32 位在我的 python 代码中运行良好。

请在此之后找到:

  • C 原型
  • 包装 C 函数的 Python 代码
  • C函数在python代码中的用法

C 函数原型(来自 API 描述): WORD mpc_AddToScenarioPcd (BYTE, DWORD, DWORD, DWORD, DWORD, DWORD, DWORD, DWORD, POINTER(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, DWORD, DWORD, DWORD, DWORD, DWORD, DWORD, DWORD, 
                                 POINTER(BYTE), DWORD, POINTER(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(1),       
                     DWORD(1),        
                     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 位指针(或类型使用......)有什么问题,以及如何解决,我将不胜感激!!

再次非常感谢您在我的主题上的所有专业知识和工作,

PS:抱歉代码没有功能,无法重现异常

4

1 回答 1

2

ctypes这是适用于 32 位和 64 位 Python的示例 DLL 代码和最小代码。请注意以下几点:

  • 中有预定义的 Windows 类型ctypes.wintypes
  • .argtypes.restype设置正确时,会根据需要在 Python 和 C 之间检查和转换参数和类型的数量。通常不需要额外的强制转换和包装。
  • CDLL对应于 32 位__cdecl调用约定。 WinDLL对应于 32-bit __stdcall。在 64 位中,只有一种调用约定。它在 Linux 和 Windows 之间的实现方式不同,但仍然只有一个,所以两者都CDLL可以WinDLL工作,但使用正确的一个来实现 32 位可移植性。

测试.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;
}

测试.py

from ctypes import *
from ctypes.wintypes import *  # pre-defined Windows types

dll = CDLL('./test')
dll.mpc_AddToScenarioPcd.argtypes = BYTE,DWORD,DWORD,DWORD,DWORD,DWORD,DWORD,DWORD,LPBYTE,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,2,3,4,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
于 2021-01-20T17:57:53.187 回答