Как передать (быстро обновляемую) переменную из кода ctypes C в Python? Передать по ссылке? - PullRequest
1 голос
/ 18 мая 2019

Я пытаюсь разработать код на C, который будет скомпилирован в DLL (или .SO для linux) с целью обработки некоторого внешнего ввода на высокой скорости и возврата результата в Python для последующей обработки.

Мой вопрос: каков наилучший способ регулярно возвращать значения (при> 1000 с времени в секунду) из функции C для использования в Python?

Я создал тестовый пример следующим образом:

//dummy_function.c
#include <stdio.h>
#include <Windows.h>

__declspec(dllexport) int runme() {
    int i;
    for (i=1; i<= 500; i++) {
        printf("int = %d \n", i);
        Sleep(100);  // Add some arbitrary delay for now
    }
}

Примечание: я импортирую Windows.h, чтобы использовать фиктивную задержку (имитирующую мою реальную проблему).Этот код может работать в Unix, используя вместо этого unistd.h (согласно: Каков правильный #include для функции 'sleep' в C? )

Этот код затем компилируется в.dll по

gcc -shared -o dummy_function.dll dummy_function.c

и импортируется в Python по:

import ctypes
libc = ctypes.CDLL('dummy_function.dll')
libc.runme()  # Start running

При выполнении этого кода Python он выводит растущие целые числа.Однако перехват печатного вывода из C с использованием Python и последующая его обработка не кажутся хорошим способом сделать это (не масштабируемым для высокой скорости).

Вместо этого мне интересно, есть ли способ для большеголегко передать переменные в Python из DLL.Я предполагаю, что не хочу использовать функцию return в C, так как она выходит из функции.

Я читал о передаче по ссылке в C, поэтому задаюсь вопросом, совместима ли она с ctypes.Могу ли я определить некоторую область памяти, в которую записывает код C, и затем Python опрашивает это?Или, лучше, какая-нибудь форма очереди / буфера, чтобы избежать пропущенных событий?

Буду признателен за любой совет, спасибо!

1 Ответ

1 голос
/ 18 мая 2019

Несмотря на то, что вы знакомы с понятиями, вот [Python 3.Docs]: ctypes - библиотека сторонних функций для Python .

Python код, опрашивающий область памяти во время выполнения кода C , подразумевает более одного потока.
Вот пример.

dll.c

#include <stdio.h>
#include <Windows.h>

#if defined(_WIN32)
#  define DLL_EXPORT __declspec(dllexport)
#else
#  define DLL_EXPORT
#endif

#define PRINT_MSG_2XI(ARG0, ARG1) printf("From C - [%s] (%d) - [%s]:  ARG0: 0x%016p, ARG1: %d\n", __FILE__, __LINE__, __FUNCTION__, ARG0, ARG1)


typedef struct Srtruct_ {
    int value;
    int sentinel;
} Struct;


DLL_EXPORT int func0(Struct *ptr) {
    int counter = 0;
    if (!ptr) {
        return -1;
    }
    while (ptr->sentinel) {
        ptr->value++;
        counter++;
        PRINT_MSG_2XI(ptr, ptr->value);
        Sleep(200);
    }
    return counter;
}

code.py

#!/usr/bin/env python3

import sys
import ctypes
import time
import threading


DLL_NAME = "./dll.dll"


class Struct(ctypes.Structure):
    _fields_ = [
        ("value", ctypes.c_int),
        ("sentinel", ctypes.c_int),
    ]


def thread_func(func, arg0):
    ret = func(arg0)
    print("\nFrom Python - Function returned {:d}".format(ret))


def main():
    dll = ctypes.CDLL(DLL_NAME)
    func0 = dll.func0
    func0.argtypes = [ctypes.POINTER(Struct)]
    func0.restype = ctypes.c_int

    data = Struct(30, 1)
    t = threading.Thread(target=thread_func, args=(func0, data,))
    t.start()
    time.sleep(1)
    print("From Python - Monitored value: {:d}".format(data.value))
    time.sleep(1)
    print("From Python - Monitored value: {:d}".format(data.value))
    data.sentinel = 0
    time.sleep(0.5)


if __name__ == "__main__":
    print("Python {:s} on {:s}\n".format(sys.version, sys.platform))
    main()
    print("\nDone.")

выход

[cfati@CFATI-5510-0:e:\Work\Dev\StackOverflow\q056197585]> sopr.bat
*** Set shorter prompt to better fit when pasted in StackOverflow (or other) pages ***

[prompt]> "c:\Install\x86\Microsoft\Visual Studio Community\2017\VC\Auxiliary\Build\vcvarsall.bat" x64
**********************************************************************
** Visual Studio 2017 Developer Command Prompt v15.9.11
** Copyright (c) 2017 Microsoft Corporation
**********************************************************************
[vcvarsall.bat] Environment initialized for: 'x64'

[prompt]> dir /b
code.py
dll.c

[prompt]> cl /nologo /DDLL dll.c  /link /NOLOGO /DLL /OUT:dll.dll
dll.c
   Creating library dll.lib and object dll.exp

[prompt]> dir /b
code.py
dll.c
dll.dll
dll.exp
dll.lib
dll.obj

[prompt]> "e:\Work\Dev\VEnvs\py_064_03.07.03_test0\Scripts\python.exe" code.py
Python 3.7.3 (v3.7.3:ef4ec6ed12, Mar 25 2019, 22:22:05) [MSC v.1916 64 bit (AMD64)] on win32

From C - [dll.c] (27) - [func0]:  ARG0: 0x00000152EB84CE90, ARG1: 31
From C - [dll.c] (27) - [func0]:  ARG0: 0x00000152EB84CE90, ARG1: 32
From C - [dll.c] (27) - [func0]:  ARG0: 0x00000152EB84CE90, ARG1: 33
From C - [dll.c] (27) - [func0]:  ARG0: 0x00000152EB84CE90, ARG1: 34
From C - [dll.c] (27) - [func0]:  ARG0: 0x00000152EB84CE90, ARG1: 35
From Python - Monitored value: 35
From C - [dll.c] (27) - [func0]:  ARG0: 0x00000152EB84CE90, ARG1: 36
From C - [dll.c] (27) - [func0]:  ARG0: 0x00000152EB84CE90, ARG1: 37
From C - [dll.c] (27) - [func0]:  ARG0: 0x00000152EB84CE90, ARG1: 38
From C - [dll.c] (27) - [func0]:  ARG0: 0x00000152EB84CE90, ARG1: 39
From C - [dll.c] (27) - [func0]:  ARG0: 0x00000152EB84CE90, ARG1: 40
From Python - Monitored value: 40

From Python - Function returned 10

Done.
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...