Python / Ctypes - Ошибка сегментации при доступе к указателю (к массиву) - PullRequest
0 голосов
/ 24 апреля 2018

Это аналогично, за исключением того, что я не получаю единственное значение и вместо этого segfault на [0] или .contents

Как получить доступ к элементам массива из Python?

В C у меня есть что-то вроде

#pragma pack(push, 1)
struct Data {
    int i;
    int *array;
};
#pragma pack(pop)

typedef struct {
    bool (* const get)   (struct Data*, const char *source);
    void (* const print) (struct Data*);
} DataFn;
extern DataFn const DATAFUNC;

В Python у меня есть что-то вроде

class Data(ctypes.Structure):
    _fields_ = [
        ("i", ctypes.c_int),
        ("array",   ctypes.POINTER(ctypes.c_int))
    ]

class DATA_FN(ctypes.Structure):
    _fields_ = [
        ("get"   , ctypes.CFUNCTYPE(
                                          ctypes.c_bool,
                                          ctypes.POINTER(Data),
                                          ctypes.c_char_p)),
        ("print" , ctypes.CFUNCTYPE(None, 
                                          ctypes.POINTER(Data)))
    ]

Из питона я могу сделать что-то вроде ..

    with open("xx0c.exx", "rb") as f:
        fdata = f.read()
        dfn.get(ctypes.byref(data),f fdata)
        dfn.print(ctypes.byref(data))
        print(data)
        print(data.i)                #7, correct
        #print(data.array.contents)  #segfault
        #print(data.array)           #segfault
        ptr = ctypes.pointer(data)
        print(ptr.contents.array[0]) #segfault

Функция C, вызванная из python, напечатает .. что правильно

я = 7

массив = 0x6ffff360034

массив 0 = 20

массив 1 = 27a40

массив 2 = 127e20

массив 3 = 128ae0

массив 4 = 47e850

массив 5 = 57ec30

массив 6 = 580230

Я вижу, что массив в python не находится рядом с массивом в C.

Пример кода C в соответствии с запросом. Они статические, но к ним можно получить доступ через DATAFUNC. Программы на C, которые их используют, хороши.

static bool getdata(struct Data *d, const char *filedata) {
    bool good = false;
    if (d != NULL && filedata != NULL) {
        d -> i       = (int32_t)filedata[0];
        d -> array   = (uint32_t*)(filedata + sizeof(int32_t));
        if ( d->i > 0 ) {
          good = true;
        }
    }
    return good;
}

static void printdata(struct Data *d) {
    if (d == NULL) {
        return;
    }
    printf("i = %i\n", d -> i);
    printf("array = %p\n", d -> array);
    unsigned int i;
    for (i = 0; i < d -> i; i++) {
        printf("array %i = %x\n", i, d -> array[i]);
    }
}

1 Ответ

0 голосов
/ 25 апреля 2018

Я решил проблему, добавив pack в структуру python, как показано ниже.

class Data(ctypes.Structure):
    _pack_   = 1
    _fields_ = [
        ("i", ctypes.c_int),
        ("array",   ctypes.POINTER(ctypes.c_int))
    ]

Это все исправляет и выдает результат ... после добавления цикла для перебора массивов.

number of offsets = 7
offsets = 0x6ffff360034
offset 0 = 20
offset 1 = 27a40
offset 2 = 127e20
offset 3 = 128ae0
offset 4 = 47e850
offset 5 = 57ec30
offset 6 = 580230
<__main__.Data object at 0x6ffffd5f400>
7
<__main__.LP_c_int object at 0x6ffffd5f488>
0x20
0x27a40
0x127e20
0x128ae0
0x47e850
0x57ec30
0x580230
...