我有一个byte.Buffer,我使用binary.Write()函数打包数据。然后我需要将此字节数组发送到C函数。使用Go 1.6,我没有成功解决这个问题。
buf := new(bytes.Buffer) //create my buffer
....
binary.Write(buf, binary.LittleEndian, data) //write my data to buffer here
addr := (*C.uchar)(unsafe.Pointer(&buf.Bytes()[0])) //convert buffers byte array to a C array
rc := C.the_function(addr, C.int(buf.Len())) //Fails here
在调用C函数的行上失败了:
panic: runtime error: cgo argument has Go pointer to Go pointer
C函数:
int the_function(const void *data, int nbytes);
我能够让以下工作,但是将字节数组转换为字符串感觉不对。有一个更好的方法吗?这种方法是否会对数据产生副作用?
addr := unsafe.Pointer(C.CString(string(buf.Bytes()[0]))
同样,这需要在Go 1.6下工作,后者引入了更严格的cgo指针规则。
谢谢。
答案 0 :(得分:8)
如果要使用第一种方法,则需要在函数调用参数之外创建切片,并避免临时分配的切片标头或参数中的外部结构,因此cgo
检查不会将它看作存储在Go中的指针。
b := buf.Bytes()
rc := C.the_function(unsafe.Pointer(&b[0]), C.int(buf.Len()))
C.CString
方法会更安全,因为数据被复制到C缓冲区,因此没有指向Go内存的指针,并且bytes.Buffer
后面的切片不可能被修改或超出范围。您将要转换整个字符串,而不仅仅是第一个字节。这种方法确实需要分配和复制两次,但是如果数据量很小,那么与cgo调用本身的开销相比,这可能不是一个问题。
str := buf.String()
p := unsafe.Pointer(C.CString(str))
defer C.free(p)
rc = C.the_function(p, C.int(len(str)))
如果在该解决方案中不接受2个数据副本,则有第三个选项,您自己对C缓冲区进行malloc,并将一个副本复制到该缓冲区中:
p := C.malloc(C.size_t(len(b)))
defer C.free(p)
// copy the data into the buffer, by converting it to a Go array
cBuf := (*[1 << 30]byte)(p)
copy(cBuf[:], b)
rc = C.the_function(p, C.int(buf.Len()))
但是对于后两种选择,不要忘记释放malloc指针。
答案 1 :(得分:0)
你的程序崩溃是因为在go1.6中将指针传递给C的规则发生了变化(详见https://tip.golang.org/doc/go1.6#cgo)。
我不知道你的程序崩溃的原因,所以我创建了Go问题https://github.com/golang/go/issues/14546。
但无论问题的答案如何,我都不会使用bytes.Buffer
的内部位(如你所做的那样)直接传入cgo。 bytes.Buffer
实现可能在将来发生变化,您的程序将开始神秘地破解。我只是将你需要的数据复制到任何合适的结构中,然后用它来传递给cgo。