我在 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。