简单协议缓冲程序在使用g ++而不是clang ++编译时可以正常工作

时间:2014-01-28 18:29:14

标签: c++ g++ protocol-buffers clang++

我正在尝试在一个简单的C ++程序中使用protocol buffers。当我用g++编译时,程序执行并正常退出。当我使用clang++进行编译时,程序失败,抱怨pointer being freed was not allocated

协议缓冲区消息定义

package test;

message Test {
    required int32 id = 1;
    required string name = 2;
}

主类

#include <iostream>
#include "test.pb.h"

int main(void) {
    // Build the original message
    test::Test original;
    original.set_id(0);
    original.set_name("original");

    // Serialize the original message
    int size = original.ByteSize();
    char data[size];
    original.SerializeToArray(data, size);

    // Deserialize the data into a previously initialized message
    test::Test success;
    success.set_id(1);
    success.set_name("success");
    success.ParseFromArray(data, size);
    std::cout << success.id() << ": " << success.name() << std::endl;

    // Deserialize the data into an uninitialized message
    test::Test failure;
    failure.ParseFromArray(data, size); // FAILS HERE WITH CLANG++
    std::cout << failure.id() << ": " << failure.name() << std::endl;
}

g ++输出

theisenp$ g++ test.pb.cc main.cpp -lprotobuf -o g++.out
theisenp$ ./g++.out 
0: original
0: original

clang ++输出

theisenp$ clang++ test.pb.cc main.cpp -L/usr/local/lib -lprotobuf -stdlib=libstdc++ -o clang++.out
theisenp$ ./clang++.out 
0: original
clang++.out(9948,0x7fff71195310) malloc: *** error for object 0x7fff72ed6330: pointer being freed was not allocated
*** set a breakpoint in malloc_error_break to debug
Abort trap: 6

我是Protocol Buffers和Clang的新手,所以我完全有可能错过一些明显的东西。有什么想法吗?

修改

对编译器版本的一些澄清。我正在运行OSX Mavericks(10.9.1)。默认情况下,Mavericks分别将呼叫映射到gccg++clangclang++。我已经独立安装了gcc 4.8.2并覆盖了默认行为。

g ++版本

theisenp$ g++ -v
Using built-in specs.
COLLECT_GCC=g++
COLLECT_LTO_WRAPPER=/usr/local/Cellar/gcc48/4.8.2/libexec/gcc/x86_64-apple-darwin13.0.2/4.8.2/lto-wrapper
Target: x86_64-apple-darwin13.0.2
Configured with: ../configure
    --build=x86_64-apple-darwin13.0.2
    --prefix=/usr/local/Cellar/gcc48/4.8.2
    --enable-languages=c,c++,objc,obj-c++
    --program-suffix=-4.8
    --with-gmp=/usr/local/opt/gmp4
    --with-mpfr=/usr/local/opt/mpfr2
    --with-mpc=/usr/local/opt/libmpc08
    --with-cloog=/usr/local/opt/cloog018
    --with-isl=/usr/local/opt/isl011
    --with-system-zlib
    --enable-version-specific-runtime-libs
    --enable-libstdcxx-time=yes
    --enable-stage1-checking
    --enable-checking=release
    --enable-lto
    --disable-werror
    --enable-plugin
    --disable-nls
    --disable-multilib
Thread model: posix
gcc version 4.8.2 (GCC) 

clang ++ Version

theisenp$ clang++ -v
Apple LLVM version 5.0 (clang-500.2.79) (based on LLVM 3.3svn)
Target: x86_64-apple-darwin13.0.2
Thread model: posix

2 个答案:

答案 0 :(得分:3)

正如评论中所讨论的那样,重要的是libprotobuf使用编译代码时使用的相同stdlib进行编译,因为libprotobuf在其接口中使用STL类型(尤其是std::string)。如果你已经安装了GCC 4.8,它将使用GCC的libstdc ++ 4.8。与此同时,Xcode附带的Clang将使用Xcode附带的任何stdlibs,所以即使你告诉它--stdlib=libstdc++,它也可能是一个不兼容的版本。

答案 1 :(得分:2)

我怀疑你的问题可能在这里:

int size = original.ByteSize();
char data[size];

可变长度数组不是有效的C ++(它们在C99中是允许的),它们是GCC extension

我建议使用动态分配的缓冲区,看看问题是否仍然存在。有关clang和VLA的讨论大约在2010年here