Каков правильный тип указателя для возврата из обратного вызова ctypes? - PullRequest
0 голосов
/ 12 мая 2018

У меня есть обратный вызов ctypes, который берет указатель на два двойных числа из dll и возвращает указатель на два двойных обратно в .dll.Первая часть работает правильно, но после многих испытаний и исследований я все еще не могу получить правильный указатель, чтобы вернуться к .dll после обратного вызова.

Вот код обратного вызова ctypes:

from scipy.integrate import dblquad
import ctypes

LibraryCB = ctypes.WINFUNCTYPE(ctypes.py_object, ctypes.POINTER(ctypes.c_double))

def LibraryCall(ptr):
    n = ctypes.cast(ptr,ctypes.POINTER(ctypes.c_double))
    x = n[0] #Value = 83.0
    y = n[1] #Value = 30.0
    area = dblquad(lambda x, y: x*y, 0, 0.5, lambda x: 0, lambda x: 1-2*x)
    return_val = area[0], area[1]
    return (return_val)

lib_call = LibraryCB(LibraryCall)

lib_call = ctypes.cast(lib_call,ctypes.POINTER(ctypes.c_longlong))

Я использовал ctypes.py_object в качестве типа указателя для возврата, поскольку все другие типы указателей возвращают ошибку, что это недопустимый тип указателя для обратного вызова,

Указатель возвращается в dll, но извлеченные значения не совпадают, что говорит о том, что у меня нет правильного типа указателя.

Я также попытался привести возвращаемое значение к указателю следующим образом: rv = ctypes.cast (return_val, ctypes.POINTER (ctypes.c_double)) и вернуть его вместо этого, но это не сработало.

Вот соответствующий код ctypes, который вызывает dll и обратный вызов:

CA_data1 = (ctypes.c_double * len(data1))(*data1)
CA_data2 = (ctypes.c_double * len(data2))(*data2)
hDLL = ctypes.WinDLL("C:/NASM_Test_Projects/SciPy_Test/SciPy_Test.dll")
CallName = hDLL.Main_Entry_fn
CallName.argtypes = [ctypes.POINTER(ctypes.c_double),ctypes.POINTER(ctypes.c_double),ctypes.POINTER(ctypes.c_double),ctypes.POINTER(ctypes.c_longlong)]
CallName.restype = ctypes.c_double

ret_ptr = CallName(CA_data1,CA_data2,length_array_out,lib_call)

Большое спасибо за любые идеи о том, как решить эту проблему.

1 Ответ

0 голосов
/ 15 мая 2018

Код DLL должен был бы иметь дело непосредственно с объектами Python. Вы не можете просто привести объект Python к указателю ctypes. Самый простой способ сделать то, что вы хотите, если вы можете изменить код DLL, это использовать параметр ввода / вывода:

test.c

#include <stdio.h>

typedef void (*CALLBACK)(double*);

__declspec(dllexport) void function(CALLBACK callback)
{
    double param[2] = {1.5, 2.5};
    callback(param);
    printf("%lf %lf\n",param[0],param[1]);
}

test.py

from ctypes import *

CALLBACK = PYFUNCTYPE(None,POINTER(c_double))

@CALLBACK
def LibraryCB(data):
    x = data[0]
    y = data[1]
    print(x,y)
    data[0] *= 2
    data[1] *= 2

dll = CDLL('test')
dll.function.argtypes = [CALLBACK]
dll.function.restype = None

cb = CALLBACK(LibraryCB)

dll.function(cb)

выход

1.5 2.5
3.000000 5.000000
...