Golang CGo:将union字段转换为Go类型

时间:2013-01-29 10:44:43

标签: go

我在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 */
};

我可以为每个union元素编写一个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
}

但是这会产生错误无法将ptr(类型uint64)转换为类型unsafe.Pointer

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

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


上下文

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

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

C代码来自gsnmp

4 个答案:

答案 0 :(得分: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)

答案 1 :(得分:2)

索尼娅已经回答了她自己的问题,我只是想说明为什么需要两种类型的转换。

来自unsafe.Pointer的文档:

  

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

     

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

     

3)uintptr可以转换为指针。

     

4)指针可以转换为uintptr。

由于var ptr uint64不是指针(类型uint64不是指针),因此无法使用规则1将ptr直接转换为unsafe.Pointer。因此,有必要首先将ptr转换为uintptr,然后按照规则3将uintptr转换为Pointer

答案 2 :(得分:2)

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

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

data

我第一次看到它时并没有完全理解这一点,但当我把它分解时它变得更加清晰:

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

归功于Alan Shen的article,它以最终对我有意义的方式描述了工会的cgo表示。

答案 3 :(得分:1)

来自CGO documentation

  

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

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

myUint32var := somePtrTo_GNetSnmpVarBind.union_guint32

用于访问somePtrTo_GNetSnmpVarBind

指向的struct的union的guint32成员