CGO 如何从 C 函数中的内部 void * return []byte?

How to return []byte from internal void * in C function by CGO?

我正在用以下定义包装一个 C 函数:

int parser_shift(parser* parser, void* buffer, int length);

它从未解析字节的内部缓冲区中删除最多 length 个字节,将它们存储在给定的缓冲区中。

现在我想把它包装成一个 Go 函数,定义如下:

func (p *Parser) Shift() []byte {
    var buffer []byte
    // TODO:

    return buffer
}

用CGO完成上面的TODO,正确的写法是什么?

我尝试了以下方法,但它因错误而崩溃:“/path/to/my/program”中的错误:free():下一个大小无效(快速):0x00007f8fe0000aa0:

var buffer []byte
bufStr := C.CString(string(buffer))
defer C.free(unsafe.Pointer(bufStr))
C.parser_shift(p.Cparser, unsafe.Pointer(bufStr), C.int(8192))
buffer = []byte(C.GoString(bufStr))
return buffer

假设您的 parser_shift 函数 returns 实际存储在缓冲区中的字节数,您可以这样做:

func (p *Parser) Shift() []byte {
    var buffer [8192]byte
    parsed := int(C.parser_shift(p.Cparser, unsafe.Pointer(&buffer[0]), C.int(len(buffer))))
    return buffer[:parsed]
}

不需要与字符串相互转换,只需将它传递给您想要它写入的内存即可。

试试这个功能

// C pointer, length to Go []byte
func C.GoBytes(unsafe.Pointer, C.int) []byte