Search code examples
pythonpointersdllctypeslibusb

Passing pointers to dll function with ctypes


I have a vendor-provided DLL for connecting to a piece of lab equipment. The DLL has a function

long search_devices(char **serial_number_list, uint32_t *dev_count)

that I'm trying to access from Python. What I've done is

import ctypes
usb_dllpath = r"path_to_dll"
usb_dll = ctypes.CDLL(usb_dllpath)

usb_dll.search_devices.argtypes = [ctypes.POINTER(ctypes.c_char_p), ctypes.POINTER(ctypes.c_uint32)]
usb_dll.search_devices.restype = ctypes.c_long

serial_list = ctypes.c_char_p()

num_devices = ctypes.c_uint32(99)
status_search = usb_dll.search_devices(ctypes.byref(serial_list), ctypes.byref(num_devices))

The result is that the first run gives an error OSError: exception: access violation writing 0x0000000000000000, doesn't change the number of devices found from 99, and leaves the status of the search as 0 (failures should return negative numbers per the DLL's source code), while also apparently locking up the instrument and making the device not appear in the example GUI. Subsequent runs of the script throw no errors but set the value of num_devices to 0 while still leaving the status as 0.

Is this a problem in my python script due to, for example, a null pointer as indicated by the error? Or should I assume there is something going wrong in the DLL?

The relevant C function in the DLL is below, with #defines replaced by their values (search_devices is just a wrapper for UsbSearchDevices). The function returns a long to describe whether an error occurred and modifies the list of serial numbers and number of found devices in memory using the pointers I pass in.

long search_devices(char **serial_number_list, uint32_t *dev_count)
{
return UsbSearchDevices(serial_number_list, dev_count);
}

long UsbSearchDevicesLV(char *serialNumberList, int *numberDevices)
{
    libusb_device_handle *deviceHandle = NULL;
    int deviceCount = 0;
    uint8_t string_usb[9];
    char *list;
    libusb_device **devList;
    libusb_context *ctx = NULL;
    struct libusb_device_descriptor desc;
    int r, i;
    int status = 0;
    ssize_t count;

    list = (char*)malloc(160);
    if (list == NULL) return 0;
    list[0] = '\0';
    r = libusb_init(&ctx);
    if (r < 0) return -12;

    count = libusb_get_device_list(ctx, &devList);
    if (count < 0)
    {
        libusb_free_device_list(devList, 1);
        return -2;
    }
    for (i = 0; i < count; i++)
    {
        status = libusb_get_device_descriptor(devList[i], &desc);
        if (desc.idVendor == 0x277C && desc.idProduct == 0x0026)
        {
            status = libusb_open(devList[i], &deviceHandle);
            if (status == 0)
            {
                libusb_get_string_descriptor_ascii(deviceHandle, desc.iSerialNumber, string_usb, sizeof(string_usb));
                strncat(list, (char *)string_usb, 8 + 1);
                libusb_close(deviceHandle);
                deviceCount++;
            }
        }
    }
    strcpy(serialNumberList, list);

    libusb_free_device_list(devList, 1);
    libusb_exit(ctx);
    free(list);

    *numberDevices = deviceCount;
    return 0;
}

The provided example C code that uses these functions has its relevant part below:

    device_list = (char**)malloc(sizeof(char*)*128); // 128ports to search
    for (i=0;i<128; i++)
        device_list[i] = (char*)malloc(sizeof(char)*8);

    status = search_devices(device_list, &num_of_devices);

Solution

  • The C usage example shows how to call the function. Below is a test function I used that fills out data as described:

    test.c

    #include <stdio.h>
    #include <stdint.h>
    
    #ifdef _WIN32
    #   define API __declspec(dllexport)
    #else
    #   define API
    #endif
    
    API long search_devices(char **serial_number_list, uint32_t *dev_count) {
        for(int i = 0; i < 5; ++i)
            sprintf_s(serial_number_list[i], 8, "SN%05d", i);
        *dev_count = 5;
        return 0;
    }
    

    Here is the code to call it that should work for your DLL as well:

    test.py

    import ctypes as ct
    
    dll = ct.CDLL('./test')
    
    dll.search_devices.argtypes = ct.POINTER(ct.c_char_p), ct.POINTER(ct.c_uint32)
    dll.search_devices.restype = ct.c_long
    
    serial_list = (ct.c_char_p * 128)()
    for i in range(128):
        serial_list[i] = ct.cast(ct.create_string_buffer(8), ct.c_char_p)
    
    num_devices = ct.c_uint32()
    status_search = dll.search_devices(serial_list, ct.byref(num_devices))
    devices = serial_list[:num_devices.value]
    print(devices)
    

    Output:

    [b'SN00000', b'SN00001', b'SN00002', b'SN00003', b'SN00004']