将Go [] byte转换为C*char

dan*_*obo 9 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
Run Code Online (Sandbox Code Playgroud)

它在调用C函数的行上失败了:

panic: runtime error: cgo argument has Go pointer to Go pointer
Run Code Online (Sandbox Code Playgroud)

C函数:

int the_function(const void *data, int nbytes);
Run Code Online (Sandbox Code Playgroud)

我能够得到以下工作,但是将字节数组转换为字符串感觉不对.有一个更好的方法吗?这种方法是否会对数据产生副作用?

addr := unsafe.Pointer(C.CString(string(buf.Bytes()[0]))
Run Code Online (Sandbox Code Playgroud)

再次,这需要在Go 1.6下工作,它引入了更严格的cgo指针规则.

谢谢.

Jim*_*imB 12

如果要使用第一种方法,则需要在函数调用参数之外创建切片,并避免临时分配的切片头或参数中的外部结构,因此cgo检查不会将其视为存储在Go中的指针.

b := buf.Bytes()
rc := C.the_function(unsafe.Pointer(&b[0]), C.int(buf.Len()))
Run Code Online (Sandbox Code Playgroud)

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)))
Run Code Online (Sandbox Code Playgroud)

如果在该解决方案中不接受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()))
Run Code Online (Sandbox Code Playgroud)

但是对于后两种选择,不要忘记释放malloc指针.