亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定
已解決430363個問題,去搜搜看,總會有你想問的

如何處理 cgo 中打包結構中的 char *?

如何處理 cgo 中打包結構中的 char *?

Go
元芳怎么了 2022-06-21 16:42:17
由于 Go 不支持打包結構,我發現這篇很棒的文章通過示例解釋了如何在 go 中使用打包結構。https://medium.com/@liamkelly17/working-with-packed-c-structs-in-cgo-224a0a3b708b問題是當我嘗試 char * 代替 [10]char 時,它不起作用。我不確定這種轉換如何與 [10]char 而不是 char * 一起使用。這是取自上述文章并用 char * 修改的示例代碼。package main/*#include "stdio.h"#pragma pack(1)typedef struct{    unsigned char a;    char b;    int c;    unsigned int d;    char *e; // changed from char[10] to char *}packed;void PrintPacked(packed p){    printf("\nFrom C\na:%d\nb:%d\nc:%d\nd:%d\ne:%s\n", p.a, p.b, p.c, p.d, p.e);}*/import "C"import (    "bytes"    "encoding/binary")//GoPack is the go version of the c packed structuretype GoPack struct {    a uint8    b int8    c int32    d uint32    e [10]uint8}//Pack Produces a packed version of the go structfunc (g *GoPack) Pack(out *C.packed) {    buf := &bytes.Buffer{}    binary.Write(buf, binary.LittleEndian, g)    *out = *(*C.packed)(C.CBytes(buf.Bytes()))}func main() {    pack := &GoPack{1, 2, 3, 4, [10]byte{}}    copy(pack.e[:], "TEST123")    cpack := C.packed{} //just to allocate the memory, still under GC control    pack.Pack(&cpack)    C.PrintPacked(cpack)}我是第一次與 cgo 合作,所以如果我在任何時候錯了,請糾正我。
查看完整描述

1 回答

?
繁花不似錦

TA貢獻1851條經驗 獲得超4個贊

您正在將十個(零)字節GoPack.e寫入packed.ewhich 的類型為char *。這不起作用,因為指針將是 4 或 8 個字節,具體取決于您的系統,因此即使字節表示有效指針,您也會溢出分配的內存量。


如果要創建具有有效packed.e字段的有效結構,則需要在 C 堆中分配 10 個字節的內存,將字節復制到其中,然后指向packed.e此分配的內存。packed(當您釋放相應的結構時,您還需要釋放此內存)。你不能直接用binary.Write.


您可以以此為起點:


buf := &bytes.Buffer{}

binary.Write(buf, binary.LittleEndian, g.a)

binary.Write(buf, binary.LittleEndian, g.b)

binary.Write(buf, binary.LittleEndian, g.c)

binary.Write(buf, binary.LittleEndian, g.d)

binary.Write(buf, binary.LittleEndian, uintptr(C.CBytes(g.e))

*out = *(*C.packed)(C.CBytes(buf.Bytes()))

該函數在 C 堆中C.CBytes(b)分配len(b)字節,并將字節復制b到其中,返回一個unsafe.Pointer.


請注意,我已經*out = *(*C.packed)...從您的代碼中復制了您的行。這實際上會導致內存泄漏和不必要的副本。可能最好使用將字節直接寫入指向的內存的寫入器out。


也許這個?


const N = 10000 // should be sizeof(*out) or larger

buf := bytes.NewBuffer((*[N]byte)(unsafe.Pointer(out))[:])

這使得 abytes.Buffer直接寫入out結構而不經過任何中間內存。請注意,由于不安全的惡作劇,如果您寫入的數據字節多于out.


警告:這一切都非常討厭,并且容易出現您在 C 中發現的相同類型的問題,并且您需要檢查 cgo 指針規則以確保您不會受到垃圾收集交互的影響。一點建議:鑒于您說您“對指針和內存分配沒有太多經驗”,您可能應該避免編寫或包含這樣的代碼,因為它可能引入的問題是邪惡的并且可能不會立即顯而易見。


查看完整回答
反對 回復 2022-06-21
  • 1 回答
  • 0 關注
  • 364 瀏覽
慕課專欄
更多

添加回答

舉報

0/150
提交
取消
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號