我正在尝试在ctypes中重做Misaka模块,但是当我尝试使用bufput时,我收到一个错误(参见第二个代码示例的结尾)。当我将指针传递给函数时,我使用pointer(b)
。这不起作用,byref(b)
也不起作用。
这是函数签名:
/* bufputs • appends a NUL-terminated string to a buffer */
void
bufputs(struct buf *, const char*);
这是我的代码:
>>> from ctypes import *
>>> sundown = cdll.LoadLibrary('./libsundown.so.1')
>>> sundown
<CDLL './libsundown.so.1', handle 1e2f190 at 1bea0d0>
>>> # OUT: <CDLL './libsundown.so.1', handle 2840d80 at 2797290>
>>> class buf(Structure):
... _fields_ = [
... ('data', c_char_p),
... ('size', c_size_t),
... ('asize', c_size_t),
... ('unit', c_size_t),
... ('ref', c_int)]
...
>>> sundown.bufnew.argtypes = [c_size_t]
>>> sundown.bufnew.restype = buf
>>> b = sundown.bufnew(c_size_t(1024))
>>> sundown.bufputs.argtypes = [POINTER(buf), c_char_p]
>>> s = c_char_p('this is a test')
>>> sundown.bufputs(pointer(b), s)
python2: malloc.c:3574: mremap_chunk: Assertion `((size + offset) & (mp_.pagesize-1)) == 0' failed.
Aborted
我无法弄清楚我做错了什么。
答案 0 :(得分:2)
OP的解决方案,最初发布在问题
中class buf(Structure):
_fields_ = [
('data', c_char_p),
('size', c_size_t),
('asize', c_size_t),
('unit', c_size_t),
('ref', c_int)
]
buf_p = POINTER(buf)
sundown.bufnew.argtypes = [c_size_t]
sundown.bufnew.restype = buf_p
sundown.bufgrow.argtypes = [buf_p, c_size_t]
sundown.bufputs.argtypes = [buf_p, c_char_p]
ib = buf()
# ctypes does this internally:
# memset(byref(ib), 0x0, sizeof(buf))
text = 'this is some text'
ib.data = text
ib.size = len(text)
ob = sundown.bufnew(128)
sundown.bufgrow(ob, int(math.ceil(ib.size * 1.4)))
sundown.bufputs(ob, 'test')