11

我在 64 位平台上使用这个 C 结构,试图访问值联合中的ui32v字段:

struct _GNetSnmpVarBind {
  guint32       *oid;       /* name of the variable */
  gsize     oid_len;    /* length of the name */
  GNetSnmpVarBindType   type;       /* variable type / exception */
  union {
    gint32   i32;           /* 32 bit signed   */
    guint32  ui32;          /* 32 bit unsigned */
    gint64   i64;           /* 64 bit signed   */
    guint64  ui64;          /* 64 bit unsigned */
    guint8  *ui8v;          /*  8 bit unsigned vector */
    guint32 *ui32v;         /* 32 bit unsigned vector */
  }         value;      /* value of the variable */
  gsize     value_len;  /* length of a vector in bytes */
};

我可以为每个联合元素编写一个 C 包装函数,但出于教学目的,我宁愿在 Go 中工作。这是我尝试访问ui32v字段的方式:

func union_to_guint32_ptr(cbytes [8]byte) (result *_Ctype_guint32) {
  buf := bytes.NewBuffer(cbytes[:])
  var ptr uint64
  if err := binary.Read(buf, binary.LittleEndian, &ptr); err == nil {
    return (*_Ctype_guint32)(unsafe.Pointer(ptr))
  }
  return nil
}

然而这给出了一个错误cannot convert ptr (type uint64) to type unsafe.Pointer

那么如何uint64 转换为指向C guint32 的 Go 类型呢?我尝试了各种组合,将转换为 uintptr,然后转换为 *_Ctype_guint32,转换为 uintptr,然后使用 unsafe.Pointer,...

我的理由是:我传递了一个 8 个字节的数组。将其转换为 uint64,即内存地址。将其转换为指向 guint32 的指针(即 guint32 的 C 数组),并将其作为结果返回 - 即联合字段“值”作为 guint32 *。


语境

稍后我想使用 value_len 字段将 guint32 的 C 数组转换为字符串,使用我知道已经可以使用的函数:

guint32_star := union_to_guint32_ptr(data.value)
result += OidArrayToString(guint32_star, data.value_len)

C 代码来自gsnmp

4

4 回答 4

6

cgo 将联合公开为一个字节数组,其大小足以容纳联合的最大成员。在您的情况下,它是 64 位,即 8 个字节,[8]byte. 正如您所演示的,此数组的内容包含联合的内容,使用它是指针转换的问题。

但是,您可以使用数组的地址来大大简化该过程。对于一个C._GNetSnmpVarBind名为data,

guint32_star := *(**C.guint32)(unsafe.Pointer(&data.value[0]))

第一次看到的时候还没有完全理解,但是拆开之后就更清楚了:

var data C._GNetSnmpVarBind    // The C struct
var union [8]byte = data.value // The union, as eight contiguous bytes of memory

// The first magic. The address of the first element in that contiguous memory
// is the address of that memory. In other words, the address of that union.
var addr *byte = &union[0]

// The second magic. Instead of pointing to bytes of memory, we can point
// to some useful type, T, by changing the type of the pointer to *T using
// unsafe.Pointer. In this case we want to interpret the union as member
// `guint32 *ui32v`. That is, T = (*C.guint32) and *T = (**C.guint32).
var cast **C.guint32 = (**C.guint32)(unsafe.Pointer(addr))

// The final step. We wanted the contents of the union, not the address
// of the union. Dereference it!
var guint32_star *C.guint32 = *cast

归功于 Alan Shen 的文章,该文章以一种对我来说终于有意义的方式描述了联合的 cgo 表示。

于 2018-05-24T04:43:59.927 回答
5

解决方案是首先转换为uintptr,然后转换为unsafe.Pointer ,即两个单独的转换:

func union_to_guint32_ptr(cbytes [8]byte) (result *_Ctype_guint32) {
    buf := bytes.NewBuffer(cbytes[:])
    var ptr uint64
    if err := binary.Read(buf, binary.LittleEndian, &ptr); err == nil {
        uptr := uintptr(ptr)
        return (*_Ctype_guint32)(unsafe.Pointer(uptr))
    }   
    return nil 
}                

我通过将结果与命令行工具进行比较来检查这一点,它返回了正确的结果。


语境

// gsnmp._Ctype_gpointer -> *gsnmp._Ctype_GNetSnmpVarBind
data := (*C.GNetSnmpVarBind)(out.data)

switch VarBindType(data._type) {
case GNET_SNMP_VARBIND_TYPE_OBJECTID:
    result += "GNET_SNMP_VARBIND_TYPE_OBJECTID" + ":"
    guint32_star := union_to_guint32_ptr(data.value)
    result += OidArrayToString(guint32_star, data.value_len)
于 2013-01-29T13:24:42.303 回答
3

Sonia 已经回答了她自己的问题,我只想提供为什么需要进行两次类型转换的原因。

unsafe.Pointer的文档中:

1) 任何类型的指针值都可以转换为Pointer。

2) 指针可以转换为任何类型的指针值。

3) uintptr 可以转换为指针。

4) 指针可以转换为 uintptr。

由于var ptr uint64is not a pointer(因为 typeuint64不是指针),ptr不能直接转换为unsafe.Pointer使用规则 1。因此需要先转换ptruintptr,然后从转换uintptrPointer以下规则 3。

于 2015-08-08T23:32:58.750 回答
1

来自 CGO文档

要直接访问 struct、union 或 enum 类型,请在其前面加上 struct_、union_ 或 enum_,如 C.struct_stat 中所示。

所以我(未经测试)代码可能类似于:

myUint32var := somePtrTo_GNetSnmpVarBind.union_guint32

用于访问指向的结构的联合的 guint32 成员somePtrTo_GNetSnmpVarBind

于 2013-01-29T11:06:41.970 回答