将Go [] byte转换为C * char

时间:2016-02-27 17:42:50

标签: go cgo

我有一个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指针规则。

谢谢。

2 个答案:

答案 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。

相关问题