涉及C和Go互操作性的问题,涉及C.free()函数。

13

我有一个Go函数,它包装了来自lib_proc.hproc_name(pid,...)函数。

这是完整的C原型


int proc_name(int pid, void * buffer, uint32_t buffersize) __OSX_AVAILABLE_STARTING(__MAC_10_5, __IPHONE_2_0);

这是我系统上可以找到 /Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX10.10.sdk/usr/include/libproc.h 文件的路径。

以下是 Go 代码:


package goproc

    /*
    #include "libproc.h"

    int call_proc_name(int pid, char *name, int name_size) {
      return proc_name(pid, name, name_size);
    }
    */
    import "C"
    import "unsafe"
    import "strings"

    type DarwinProcess struct {
        Process
    }

    func (DarwinProcess) NameOf(pid int) string {
        name := C.CString(strings.Repeat("\x00", 1024))
        defer C.free(unsafe.Pointer(name))
        nameLen := C.call_proc_name(C.int(pid), name, C.int(1024))
        var result string

        if (nameLen > 0) {
            result = C.GoString(name);
        } else {
            result = ""
        }

        return result;
    }

除非删除对C.free(unsafe.Pointer(...))的调用和import "unsafe"从代码中移除,否则此代码将无法编译。

DarwinProcess::NameOf(pid)方法仅适用于Mac OS X,如果从代码中删除C.free(...),它实际上是有效的。

go build之后的实际形式中,我收到以下错误消息:could not determine kind of name for C.free(这就是整个编译器输出内容)。

对我来说,删除C.CString()分配的内存是不可接受的,我必须找到正确的方法来释放内存。

我感到困惑,因为根据文档,一切都已经正确地完成了。我无法在这里或网络上找到解决方案。


2
你尝试在你的cgo注释中添加#include <stdlib.h>了吗? - Ainar-G
1个回答

21

libproc.h 没有包含 stdlib.h 文件,free() 在这里声明。因此,编译器无法解析该名称。在cgo注释块的开头添加 #include <stdlib.h> 后,您的代码在我的系统上成功构建。


+1 这个解决方案非常完美,回答了我的问题。说实话,我被文档误导了,因为它似乎暗示 C.free() 是特殊的 C 包的一部分。但实际上我们要使用的是标准的 stdlib::free() - gsscoder
就像其他运行时一样,Go库也缺乏用于管理已经运行的操作系统进程的API,因此我正在尝试编写一个。目前只是一个存根(https://github.com/gsscoder/goproc)。 - gsscoder

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接