在Ubuntu上,将64位程序交叉编译到32位时缺少包含“bits/c++config.h”文件。

224
我正在运行Ubuntu 10.10的32位版本,并尝试交叉编译到64位目标。根据我的研究,我已经安装了g++-multilib软件包。
程序非常简单,只是一个“hello world”:
#include <iostream>

int main( int argc, char** argv )
{
  std::cout << "hello world" << std::endl;
  return 0;
}

编译:

g++ -m64 main.cpp

错误:

In file included from main.cpp:1:
/usr/include/c++/4.4/iostream:39: fatal error: bits/c++config.h: No such file or directory
compilation terminated.
我找到了一个名为 c++config.h 的文件,但它们位于 /usr/include/c++/4.4/ 目录下的 i486-linux-gnui686-linux-gnu 目录中。在 /usr/include/c++/bits 中没有 c++config.h 文件。任何想法我漏掉了什么吗?没有使用-m64 标志编译可以正常工作(创建并运行a.out)。
编辑感谢 @nightcracker 的提示,我对32位和64位系统的包含结构进行了更多的调查。 我在下面添加了一个答案来临时 "解决" 该问题,但我认为它会在下一次更新时出现问题。 基本上,我缺少一个名为 /usr/include/c++/4.4/i686-linux-gnu/64 的目录,其中应包含一个名为 bits 的子目录,其中有所缺失的包含文件。 有什么包应该负责这个呢?

2
哇...我也遇到了这个问题。我的ARMv7-a CubieTruck(Cortex-A7)上运行Ubuntu的GCC 4.8。奇怪的是(对我来说),安东尼的答案解决了它。不管怎样... - jww
10个回答

380

我添加这个答案部分是因为它解决了我遇到的同样问题,因此我可以将这个问题加入书签。

我通过以下方法解决了这个问题:

sudo apt-get install gcc-multilib g++-multilib
如果您安装了一个非默认的 gcc / g++ 版本(例如 lucid 上的 g++-4.8),则需要匹配相应的版本:
sudo apt-get install gcc-4.8-multilib g++-4.8-multilib

31
请确保安装在您系统上的gcc和g++版本匹配。在Ubuntu 14.04上,我安装了gcc-4.8g++-4.8,因此我安装了gcc-4.8-multilibg++-4.8-multilib代替。 - Zoltán
4
在64位机器上使用-m32编译时,这解决了我的问题。谢谢。 - nic
44
谢谢您的指点,但 sudo apt-get install gcc-multilib g++-multilib 看起来更好(它会自动解决您的gcc版本)。 - leesei
@Man 随意提出修改建议,当我最初发布时,我匹配了问题中要求的版本,但由于其他人的编辑,它已经发生了变化。 - anthony sottile
这也发生在没有apt-get的Windows上。 - Tomáš Zato
显示剩余4条评论

14

你尝试添加了-I/usr/include/c++/4.4/i486-linux-gnu或者-I/usr/include/c++/4.4/i686-linux-gnu吗?


确实可以工作。您有什么想法,为什么我只需要为64位执行此操作?我正在尝试设置此机器以帮助分布式64位构建,并希望避免过多的定制。 - Jesse Vogt
2
我很抱歉,我不知道,我只是提供了一个快速的hacky工作解决方案 :) - orlp
只是因为“bits/c++config.h”驻留在那里 :) - Ehsan Fathi

11

在RHEL 6.2 (x86_64)编译时,我安装了32位和64位的libstdc++-dev包,但出现了“c++config.h文件不存在”的问题。

解决方案:

缺少目录/usr/include/c++/4.4.6/x86_64-redhat-linux

我执行了以下操作:

cd /usr/include/c++/4.4.6/
mkdir x86_64-redhat-linux
cd x86_64-redhat-linux
ln -s ../i686-redhat-linux 32

我现在可以在64位操作系统上编译32位二进制文件。


1
在 OpenSUSE 上,我执行了 cd /usr/include/c++/4.6;ln -s x86_64-suse-linux i586-suse-linux - Julian

4

在gcc的包中似乎出现了一个拼写错误。解决方法:

mv /usr/include/c++/4.x/i486-linux-gnu /usr/include/c++/4.x/i686-linux-gnu/64

3

基本上,它用于HeapOverflows或其他反向类型的问题,即如果您想将64位ELF更改为32位ELF,并且在转换时显示错误。

您可以简单地运行以下命令:

apt-get install gcc-multilib g++-multilib

这将更新您的库

以下附加软件包将被安装: g++-8-multilib gcc-8-multilib lib32asan5 lib32atomic1 lib32gcc-8-dev lib32gomp1 lib32itm1 lib32mpx2 lib32quadmath0 lib32stdc++-8-dev lib32ubsan1 libc-dev-bin libc6 libc6-dbg libc6-dev libc6-dev-i386 libc6-dev-x32 libc6-i386 libc6-x32 libx32asan5 libx32atomic1 libx32gcc-8-dev libx32gcc1 libx32gomp1 libx32itm1 libx32quadmath0 libx32stdc++-8-dev libx32stdc++6 libx32ubsan1 建议安装以下软件包: lib32stdc++6-8-dbg libx32stdc++6-8-dbg glibc-doc 将安装以下新软件包: g++-8-multilib g++-multilib gcc-8-multilib gcc-multilib lib32asan5 lib32atomic1 lib32gcc-8-dev lib32gomp1 lib32itm1 lib32mpx2 lib32quadmath0 lib32stdc++-8-dev lib32ubsan1 libc6-dev-i386 libc6-dev-x32 libc6-x32 libx32asan5 libx32atomic1 libx32gcc-8-dev libx32gcc1 libx32gomp1 libx32itm1 libx32quadmath0 libx32stdc++-8-dev libx32stdc++6 libx32ubsan1

类似于此将显示在您的终端上。


在Debian 10上,我想要保留的一些软件包将被删除。以下软件包将被删除: g++-8-arm-linux-gnueabi、g++-8-arm-linux-gnueabihf、g++-arm-linux-gnueabi、g++-arm-linux-gnueabihf、gcc-8-aarch64-linux-gnu、gcc-8-arm-linux-gnueabi、gcc-8-arm-linux-gnueabihf、gcc-aarch64-linux-gnu、gcc-arm-linux-gnueabi和gcc-arm-linux-gnueabihf。 - Thoth

3
在我的64位系统上,我注意到存在以下目录:
/usr/include/c++/4.4/x86_64-linux-gnu/32/bits

那么,在我的32位系统上,如果已经设置为64位交叉编译,应该会有一个相应的目录,例如:

/usr/include/c++/4.4/i686-linux-gnu/64/bits

我仔细检查了一下,发现这个目录并不存在。使用带有详细参数的 g++ 命令运行后,编译器实际上是在这个位置寻找某些东西:

jesse@shalored:~/projects/test$ g++ -v -m64 main.cpp 
Using built-in specs.
Target: i686-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu/Linaro 4.4.4-14ubuntu5' --with-bugurl=file:///usr/share/doc/gcc-4.4/README.Bugs --enable-languages=c,c++,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.4 --enable-shared --enable-multiarch --enable-linker-build-id --with-system-zlib --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.4 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-objc-gc --enable-targets=all --disable-werror --with-arch-32=i686 --with-tune=generic --enable-checking=release --build=i686-linux-gnu --host=i686-linux-gnu --target=i686-linux-gnu
Thread model: posix
gcc version 4.4.5 (Ubuntu/Linaro 4.4.4-14ubuntu5) 
COLLECT_GCC_OPTIONS='-v' '-m64' '-shared-libgcc' '-mtune=generic'
 /usr/lib/gcc/i686-linux-gnu/4.4.5/cc1plus -quiet -v -imultilib 64 -D_GNU_SOURCE main.cpp -D_FORTIFY_SOURCE=2 -quiet -dumpbase main.cpp -m64 -mtune=generic -auxbase main -version -fstack-protector -o /tmp/ccMvIfFH.s
ignoring nonexistent directory "/usr/include/c++/4.4/i686-linux-gnu/64"
ignoring nonexistent directory "/usr/local/include/x86_64-linux-gnu"
ignoring nonexistent directory "/usr/lib/gcc/i686-linux-gnu/4.4.5/../../../../i686-linux-gnu/include"
#include "..." search starts here:
#include <...> search starts here:
 /usr/include/c++/4.4
 /usr/include/c++/4.4/backward
 /usr/local/include
 /usr/lib/gcc/i686-linux-gnu/4.4.5/include
 /usr/lib/gcc/i686-linux-gnu/4.4.5/include-fixed
 /usr/include/x86_64-linux-gnu
 /usr/include
End of search list.
GNU C++ (Ubuntu/Linaro 4.4.4-14ubuntu5) version 4.4.5 (i686-linux-gnu)
    compiled by GNU C version 4.4.5, GMP version 4.3.2, MPFR version 3.0.0-p3.
GGC heuristics: --param ggc-min-expand=98 --param ggc-min-heapsize=128197
Compiler executable checksum: 1fe36891f4a5f71e4a498e712867261c
In file included from main.cpp:1:
/usr/include/c++/4.4/iostream:39: fatal error: bits/c++config.h: No such file or directory
compilation terminated.

关于“忽略不存在的目录”的错误是线索。不幸的是,我仍然不知道需要安装哪个软件包才能显示此目录,因此我只是从我的64位机器上复制了/usr/include/c++/4.4/x86_64-linux-gnu/bits目录到我的32位机器上的/usr/include/c++/4.4/i686-linux-gnu/64/bits

现在只使用-m64进行编译就可以正常工作了。主要缺点是这仍然不是正确的做法,我猜想下次更新管理器安装g++的更新时可能会出现问题。

不错的调试技巧! - shellter

2

2

在x86机器上为arm32进行交叉编译时遇到了相同的问题,如果你正在进行arm的跨编译,请确保已安装最新的libstdc++软件包。

sudo apt install libstdc++-10-dev-armhf-cross

你可能会遇到另一个标题错误,即“asm / errno.h”文件未找到。只需安装gcc-multilib即可解决此问题。

sudo apt install gcc-multilib

谢谢。我也有同样的使用情况。我也不得不执行 sudo apt install g++-multilib libstdc++-9-dev-armhf-cross - Frak

1

根据我的经验,sudo apt-get install gcc-multilib g++-multilib 可以解决问题。但是我还有另一个问题,就是我忘记清理目录,所以我仍然得到相同的错误。这是我第一次使用clang或cmake。所以我只需删除原始目录并重新编译即可解决问题。希望能帮助像我一样的人。


0
在我的情况下,这只是一个路径问题。我缩短了源文件的路径长度,然后它就可以工作了...

“..源文件的路径长度”真的不应该成为问题。你是怎么出现了一个太长的src文件路径的?请分享一个例子。 - shellter
抱歉,我现在无法重现这个问题。它发生在一个Windows系统上,该系统可能具有(注意:并非所有情况都适用)260个字符的路径限制。 - Traummaennlein

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接