Я просмотрел другие ответы, но не могу заставить это работать. Я пытаюсь вызвать функцию в DLL для связи с устройствами SMBus. Эта функция принимает указатель на структуру, которая имеет массив как одно из полей. так что...
В C:
typedef struct _SMB_REQUEST
{
unsigned char Address;
unsigned char Command;
unsigned char BlockLength;
unsigned char Data[SMB_MAX_DATA_SIZE];
} SMB_REQUEST;
Я думаю, что мне нужно установить значения для Address, Command и BlockLength, в то время как DLL заполняет массив данных. Функция, требующая этой структуры, принимает ее как указатель
SMBUS_API int SmBusReadByte( SMBUS_HANDLE handle, SMB_REQUEST *request );
Итак, я создал структуру в Python так:
class SMB_REQUEST(ctypes.Structure):
_fields_ = [("Address", c_char),
("Command", c_char),
("BlockLength", c_char),
("Data", type(create_string_buffer(SMB_MAX_DATA_SIZE))]
* Примечание. Я также попробовал ctypes.c_char * SMB_MAX_DATA_SIZE для типа данных *
Чтобы передать указатель на структуру этого типа на функцию, которую я попытался инициализировать сначала, выполните следующие действия:
data = create_string_buffer(SMB_MAX_DATA_SIZE)
smb_request = SMB_REQUEST('\x53', \x00', 1, data)
Это отвечает:
TypeError: expected string or Unicode object, c_char_Array_32 found
Если я попытаюсь оставить массив данных, вот так:
smb_request = SMB_REQUEST('\x53', \x00', 1)
Нет, ошибка. Однако, когда я пытаюсь передать это функции:
int_response = smbus_read_byte(smbus_handle, smb_request))
Я получаю:
ArgumentError: argument 2: <type 'exceptions.TypeError'>: expected LP_SMB_REQUES
T instance instead of SMB_REQUEST
Я пробовал передать его как указатель:
int_response = smbus_read_byte(smbus_handle, ctypes.POINTER(smb_request))
и я получаю:
----> 1
2
3
4
5
TypeError: must be a ctypes type
Вот как я создал типы искусства:
smbus_read_byte.argtypes = (ctypes.c_void_p, ctypes.POINTER(SMB_REQUEST))
Я пробовал кастинг, но все равно не пошел. Может ли кто-нибудь пролить свет на это для меня?
Update:
Если я сначала инициализирую структуру следующим образом:
smb_request = SMB_REQUEST('\xA6', '\x00', chr(1), 'a test string')
а затем бас по ссылке:
int_response = smbus_receive_byte(smbus_handle, ctypes.byref(smb_request))
Я не ошибаюсь. Однако функция возвращает -1, когда она должна возвращать "0" для успеха и не равна нулю для отказа. Проверка значения smb_request.Data возвращает "тестовую строку", поэтому никаких изменений нет. Любые предложения относительно того, что может происходить здесь, будут очень признательны.
Спасибо
UPDATE:
Так как у меня есть несколько запросов о том, правильно ли мой дескриптор, вот как я его использую. Заголовочный файл для DLL объявляет следующее:
typedef void *SMBUS_HANDLE;
//
// This function call initializes the SMBus, opens the driver and
// allocates the resources associated with the SMBus.
// All SMBus API calls are valid
// after making this call except to re-open the SMBus.
//
SMBUS_API SMBUS_HANDLE OpenSmbus(void);
Итак, как я это делаю в python:
smbus_handle = c_void_p() # NOTE: I have also tried it without this line but same result
open_smbus = CDLL('smbus.dll').OpenSmbus
smbus_handle = open_smbus()
print 'SMBUS_API SMBUS_HANDLE OpenSmbus(void): ' + str(smbus_handle)
Я вызываю это перед вызовом smbus_read_byte(). Я попытался установить open_smbus.restype = c_void_p()
, но я получаю сообщение об ошибке: TypeError: restype must be a type, a callable, or None