7

我有一个项目依赖于 Hyper 和 Diesel,因此依赖于本地库 OpenSSL 和 libpq。该项目基于 nightly Rust 构建,因为它使用编译器插件。

我目前的尝试是在 Docker 容器上构建。我有 MUSL libc 和库make,并安装了前缀/usr/local/musl。我cargo使用以下命令运行:(不确定某些选项是否多余,我对编译器链不太精通,甚至不确定它们是否最终到达链接器,但我必须尝试,对.)

LDFLAGS="-static -L/usr/local/musl/lib" \
LD_LIBRARY_PATH=/usr/local/musl/lib:$LD_LIBRARY_PATH \
CFLAGS="-I/usr/local/musl/include" \
PKG_CONFIG_PATH=/usr/local/musl/lib/pkgconfig \
cargo build --release --target=x86_64-unknown-linux-musl

当我ldd得到结果文件时,它揭示了这一点:

$ ldd server
linux-vdso.so.1 (0x00007fffb878e000)
libpq.so.5 => /usr/local/musl/lib/libpq.so.5 (0x00007f4d730e7000)
libssl.so.1.0.0 => /usr/lib/x86_64-linux-gnu/libssl.so.1.0.0 (0x00007f4d72e82000)
libcrypto.so.1.0.0 => /usr/lib/x86_64-linux-gnu/libcrypto.so.1.0.0 (0x00007f4d72a85000)
libc.so => /usr/local/musl/lib/libc.so (0x00007f4d727f6000)
libdl.so.2 => /lib/x86_64-linux-gnu/libdl.so.2 (0x00007f4d725f2000)
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f4d72246000)
/lib/ld64.so.1 => /lib64/ld-linux-x86-64.so.2 (0x000055e2124a2000)

有所有动态链接的东西,有些甚至是“x86_64-linux-gnu”链!什么地方出了错?

我可以毫无问题地制作静态链接的简单纯 Rust 项目。ldd说它们是静态链接的,并且它们运行没有问题,与我遇到问题的可执行文件不同。

当我--verbose与 Cargo 一起使用时,我得到了以下rustc实际构建可执行文件的命令:http: //pastebin.com/ywv0zNBK(哎呀,那个有一个自定义outdir-Z print-link-args,由我添加)添加print-link-args标志,我得到以下链接器命令: http: //pastebin.com/Aw43qd7h

我如何获得cargorustc相信我想要一个静态二进制文件?

4

3 回答 3

13

问题在于,对于每个提供本地依赖的 crate(比如 OpenSSL),都有一个build.rs构建脚本负责将构建和链接选项与 Cargo 和rustc. (例如:他们打印出类似cargo:rustc-link-lib=static=sslCargo 读取并采取相应行动的内容。)

因此,仅设置“标准”GCC 环境变量几乎不会产生任何影响。您必须单独检查每一个,build.rs以了解如何强制该确切的板条箱运送货物。对于 OpenSSL,它的环境变量如OPENSSL_DIROPENSSL_STATIC

另一个障碍是,如果您使用编译器插件,它们也可能与目标三元组一起编译(至少是 docker_codegen)。另一方面,它们在编译过程中是动态链接的。这意味着不仅必须正确链接静态库,还必须具有目标主机种类的动态库,例如 Musl libc.so,并正确设置(LD_LIBRARY_PATH等)。

我制作了一个经过彻底注释的 Dockerfile,它使用一些本机依赖项静态构建我的项目。它可能对其他人也有帮助。

https://gitlab.com/rust_musl_docker/image

于 2016-11-20T07:26:12.227 回答
1

如果你想静态链接一个没有原生依赖的 Rust 程序,那要容易得多:

$ rustup target add x86_64-unknown-linux-musl
$ cargo build --release --target=x86_64-unknown-linux-musl
于 2018-11-15T08:56:02.927 回答
0

我对 ldd 和 GCC 有同样的问题。musl 目标是在不同的目录中生成的;不是在,target/release/...而是在target/x86_64-unknown-linux-musl/release/...

于 2021-08-04T13:35:53.610 回答