Python ctypes - dll 函数接受结构体崩溃

4 投票
1 回答
2425 浏览
提问于 2025-04-16 02:49

我需要在Windows XP系统下访问一个POS终端。我使用的是Python 2.7。
我加载的DLL中有一个关键的函数负责处理支付,它需要两个指向结构体的指针,但在调用时出现崩溃,返回值是1(通信错误),而且没有其他的错误信息。
需要注意的是,当调用支付函数时,并不是所有的POSData结构体中的元素都有值。还有其他我尝试过的函数(比如GetVersion)是可以正常工作的。
以下是相关的规格说明和我的代码:

typedef struct
{
  char IPAddress[16]; //xxx.xxx.xxx.xxx
  int Port;
} TETHParameters;   
typedef struct
{
  char TerminalId[8+1];
  char AcquirerId[11+1];
  char TransactionType[3+1];
  char TransactionResult[2+1];
  char KODescription[24+1];
  char CardType[1+1];
  char STAN[6+1];
  char PAN[19+1];
  char AuthorizationCode[6+1];
  char OperationNumber[6+1];
  char DataTrs[7+1];
} TPOSData;

typedef struct
{
  char Amount[8+1];
  char ECRId[8+1];
  char PaymentType[1+1];
  char TerminalId[8+1];
} TECRData;

__declspec(dllexport) void IAE17_GetVersion(char *Version);
__declspec(dllexport) void IAE17_InitEth(TETHParameters *ETHParameters);

__declspec(dllexport) void IAE17_Free(void);

__declspec(dllexport) int IAE17_Payment(TECRData *ECRData, TPOSData *POSData);

from ctypes import *
#da python 3.x sara' configparser
import ConfigParser  
import logging
from time import  localtime,  strftime

    #STRUTTURE DATI
class TETHParameters(Structure):
    _fields_ =  [("IPAddress" , c_char_p), ("Port" , c_int )]


class TECRData(Structure):
    _fields_ = [("Amount" , c_char_p),
    ("ECRId", c_char_p),
    ("PaymentType", c_char_p),
    ("TerminalId", c_char_p),
    ("Contract", c_char_p),
    ("PreauthorizationCode", c_char_p),
    ("STAN", c_char_p),
    ("Ticket2Ecr", c_char_p)]


class TPOSData(Structure):
    _fields_ = [
    ("TerminalId" , c_char_p),
    ("AcquirerId" , c_char_p),
    ("TransactionType" , c_char_p),
    ("TransactionResult" , c_char_p),
    ("KODescription" , c_char_p),
    ("CardType" , c_char_p),
    ("STAN" , c_char_p),
    ("POSBalance" , c_char_p),
    ("BankBalance" , c_char_p),
    ("PAN" , c_char_p),
    ("AuthorizationCode" , c_char_p),
    ("OperationNumber" , c_char_p),
    ("AmountAuth" , c_char_p),
    ("PreauthorizationCode" , c_char_p),
    ("ActionCode" , c_char_p),
    ("DataTrs" , c_char_p),
    ("AmountEcho" , c_char_p),
    ("Ticket" , c_char_p)
    ] 

ECRData = TECRData( ECRId = c_char_p( '012345678' ), 
                    Amount  = c_char_p( '00000000')  , 
                    TerminalID = c_char_p( '01234567' ), 
                    PaymentType = c_char_p ("0")
                       )    


POSData = TPOSData( KODescription = c_char_p('                        '),
                            TerminalId = c_char_p('        '),  
                            AcquirerId = c_char_p('           '), 
                            TransactionType = c_char_p('   '), 
                            TransactionResult = c_char_p('   '),
                            CardType = c_char_p('  '), 
                            STAN = c_char_p('      '),
                            PAN = c_char_p('                   '), 
                            AuthorizationCode = c_char_p('      '),
                            OperationNumber = c_char_p('      '), 
                            DataTrs = c_char_p('       ')  
                            )   
ETHParameters = TETHParameters( IPAddress = c_char_p( '192.168.127.190' ) ,  Port = c_int(45119))                           
iae17 = windll.LoadLibrary('iae17')     
iae17.IAE17_InitEth( byref( ETHParameters) )   
result =  iae17.IAE17_Payment( byref(ECRData), byref(POSData))                      
print result

1 个回答

5

c_char_p 是 C 语言中 char * 的直接对应。也就是说,当你的 C 结构是

typedef struct
{
  char TerminalId[8+1];
  char AcquirerId[11+1];
  char TransactionType[3+1];

&c

而你在 ctypes 中创建的结构,实际上是等同于

typedef struct
{
  char* TerminalId;
  char* AcquirerId;
  char* TransactionType;

&c

这两者之间的差别非常大。你为什么不使用 ctypes 的 数组,而是用“指针”?

撰写回答