Reputation: 41
This is similar except I do not get a single value and instead segfault on [0] or .contents
How can I access the elements in 'array' from Python?
In C I have something like
#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;
In Python I have something like
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)))
]
From python I can do something like ..
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 function called from python will print .. which is correct
i = 7
array = 0x6ffff360034
array 0 = 20
array 1 = 27a40
array 2 = 127e20
array 3 = 128ae0
array 4 = 47e850
array 5 = 57ec30
array 6 = 580230
I can see the array in python isn't anywhere near the array in C.
Sample C code as asked. These are static, but they can be accessed through DATAFUNC. The C programs that use these are fine.
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]);
}
}
Upvotes: 0
Views: 1016
Reputation: 41
I've resolved the issue by adding pack to the python struct like as follows.
class Data(ctypes.Structure):
_pack_ = 1
_fields_ = [
("i", ctypes.c_int),
("array", ctypes.POINTER(ctypes.c_int))
]
This fixes everything and gives the output of .. once a loop was added to iterate through arrays.
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
Upvotes: 1