这个问题与我提出的上一个问题有关。即this one如果有人有兴趣的话。基本上,我想要做的是使用包裹在Py_buffer
- 对象中的memoryview
将C数组公开给Python。我已经使用PyBuffer_FillInfo
工作了(work =我可以在Python中操作数据并将其写入C中的stdout),但如果我尝试滚动自己的缓冲区,我会在<之后得到段错误 / em> C函数返回。
我需要创建自己的缓冲区,因为PyBuffer_FillInfo假定格式为char,使itemsize字段为1.我需要能够提供大小为1,2,4和8的项目。
一些代码,这是一个有效的例子:
Py_buffer *buf = (Py_buffer *) malloc(sizeof(*buf));
int r = PyBuffer_FillInfo(buf, NULL, malloc(sizeof(char) * 4), 4, 0, PyBUF_CONTIG);
PyObject *mv = PyMemoryView_FromBuffer(buf);
//Pack the memoryview object into an argument list and call the Python function
for (blah)
printf("%c\n", *buf->buf++); //this prints the values i set in the Python function
查看PyBuffer_FillInfo
的实现,这非常简单,我推出了自己的函数,以便能够提供自定义项目:
//buffer creation function
Py_buffer *getReadWriteBuffer(int nitems, int itemsize, char *fmt) {
Py_buffer *buf = (Py_buffer *) malloc(sizeof(*buf));
buf->obj = NULL
buf->buf = malloc(nitems * itemsize);
buf->len = nitems * itemsize;
buf->readonly = 0;
buf->itemsize = itemsize;
buf->format = fmt;
buf->ndim = 1;
buf->shape = NULL;
buf->strides = NULL;
buf->suboffsets = NULL;
buf->internal = NULL;
return buf;
}
我如何使用它:
Py_buffer *buf = getReadWriteBuffer(32, 2, "h");
PyObject *mv = PyMemoryView_FromBuffer(buf);
// pack the memoryview into an argument list and call the Python function as before
for (blah)
printf("%d\n", *buf->buf); //this prints all zeroes even though i modify the array in Python
return 0;
//the segfault happens somewhere after here
使用我自己的缓冲区对象的结果是C函数返回后的段错误。我真的不明白为什么会发生这种情况。任何帮助都将非常感激。
修改
根据我之前未能找到的this question,itemize&gt;甚至可能根本不支持1。这使得这个问题更加有趣。也许我可以使用PyBuffer_FillInfo
一个足够大的内存块来保存我想要的东西(例如32 C浮点数)。在这种情况下,问题更多的是如何将Python浮点数分配给Python函数中的memoryview
对象。问题问题。
答案 0 :(得分:3)
所以,由于缺乏答案,我决定采取另一种方法,而不是我原先的意图。将此留在这里以防其他人遇到同样的问题。
基本上,不是在C中创建缓冲区(或bytearray,equiv。)而是将其传递给Python以供扩展用户修改。我只是稍微重新设计了代码,以便用户从Python回调函数返回一个bytearray(或任何支持缓冲区接口的类型)。这样我甚至不用担心项目的大小,因为在我的情况下,返回的对象所做的所有C代码都是提取缓冲区并将其复制到另一个带有简单memcpy
的缓冲区。
代码:
PYGILSTATE_ACQUIRE; //a macro i made
PyObject *result = PyEval_CallObject(python_callback, NULL);
if (!PyObject_CheckBuffer(result))
; //raise exception
Py_buffer *view = (Py_buffer *) malloc(sizeof(*view));
int error = PyObject_GetBuffer(result, view, PyBUF_SIMPLE);
if (error)
; //raise exception
memcpy(my_other_buffer, view->buf, view->len);
PyBuffer_Release(view);
Py_DECREF(result);
PYGILSTATE_RELEASE; //another macro
我希望这有助于某人。