2017-08-25 49 views
1

我有大量的数据从TagLib库中读取并传递给GoLang(mpeg图像数据)。Golang CGO与大字符指针 - SEGSERV

这里是哪里取数据时:

void audiotags_mpeg_artwork(TagLib::MPEG::File *mpegFile, int id) { 
    TagLib::ID3v2::Tag *id3v2 = mpegFile->ID3v2Tag(false); 
    if (id3v2!=nullptr) { 
     const TagLib::ID3v2::FrameList frameList = id3v2->frameListMap()["APIC"]; 
     for(auto it = frameList.begin(); it != frameList.end(); it++) { 
      TagLib::ID3v2::AttachedPictureFrame * frame = (TagLib::ID3v2::AttachedPictureFrame *)(*it); 
      if (frame!=nullptr && frame->size() > 0) { 
       const auto &apicBase64 = frame->picture().toBase64(); 
       auto len = apicBase64.size(); 
       if (len > 0) { 
        // Generate memory for key 
        char* key = new char[5]; 
        memcpy(key, "APIC", 4); 
        key[4]='\0'; 

        // Generate memory for picture data 
        char* val = new char[len]; 
        memcpy (val, apicBase64.data(), len); 

        // Send to GoLang 
        go_map_audiotags(id, key, val); 

        // Free memory 
        delete[] key; 
        delete[] val; 
       } 
      } 
     } 
    } 
} 

在这一点上,go_map_autotags作品​​(我用类似的方法为其他数据)。这也适用于其他的图像数据,但是根据大小,这将与崩溃:

unexpected fault address 0x766a000

fatal error: fault

[signal SIGSEGV: segmentation violation code=0x1 addr=0x766a000 pc=0x404530b]

在GoLang,我有以下出口:

//export go_map_audiotags 
func go_map_audiotags(id C.int, key *C.char, val *C.char) { 
    m := maps[int(id)] 
    k := strings.ToLower(C.GoString(key)) 
    log.Println("go_map_audiotags k:", k) // <--- works 
    v := C.GoString(val) // <--- crashes 
    log.Println("go_map_audiotags v:", v) // <--- Does not reach 
    m[k] = v 
} 

是否有贝特办法,我应该运送这个数据?我认为发生的事情是: 1)C.char限制是正在达到 2)C++是,由于某种原因,设置V IN GoLang

+0

是否'apicBase64.data()'nul-terminated? – tkausl

+0

@tkausl如何识别此问题的最佳方法?据的TagLib文档: 帧 - >图像():http://taglib.org/api/classTagLib_1_1ID3v2_1_1AttachedPictureFrame.html#afd21599ba7fa9a237ec5f1f1cad78e24 ByteVector.data():http://taglib.org/api/classTagLib_1_1ByteVector.html# a55d40ed1e9b9749e3895c845f1f26901 – Stoyvo

+0

Nul-自己终止它,看看它是否仍然崩溃。 – tkausl

回答

1

存储在val数据之前回收的内存是不是空终止。在您的C代码中,当您使用memcpy进行复制时,不包括空终止符。在C代码中,将代码更改为:

// Generate memory for picture data 
char* val = new char[len+1]; 
memcpy (val, apicBase64.data(), len); 
val[len] = '\0';