如何使用 glibc 而不是 musl 来使用 rustc 创建静态可执行文件?

8
我使用C、Go和Rust编写了简单的代码。
foo.c
#include <stdio.h>

int main()
{
    printf("hello\n");
    return 0;
}

foo.go

package main

import "fmt"

func main() {
    fmt.Println("hello");
}

foo.rs

fn main() {
    println!("hello");
}

然后我将它们全部构建完成。
$ gcc -static -o cfoo foo.c
$ go build -o gofoo foo.go
$ rustc -o rustfoo foo.rs

它们运行良好。

$ ./cfoo; ./gofoo; ./rustfoo
hello
hello
hello

与另外两个相比,Rust可执行文件的二进制代码太小了,所以我怀疑它不是一个静态可执行文件。

$ ls -l cfoo gofoo rustfoo
-rwxr-xr-x 1 lone lone  755744 Oct 23 21:17 cfoo
-rwxr-xr-x 1 lone lone 1906945 Oct 23 21:17 gofoo
-rwxr-xr-x 1 lone lone  253528 Oct 23 21:17 rustfoo

我确认 Rust 不会产生静态可执行文件。

$ ldd cfoo gofoo rustfoo
cfoo:
    not a dynamic executable
gofoo:
    not a dynamic executable
rustfoo:
    linux-vdso.so.1 (0x00007ffe6dfb7000)
    libdl.so.2 => /lib/x86_64-linux-gnu/libdl.so.2 (0x00007fd8d9b75000)
    librt.so.1 => /lib/x86_64-linux-gnu/librt.so.1 (0x00007fd8d9b6b000)
    libpthread.so.0 => /lib/x86_64-linux-gnu/libpthread.so.0 (0x00007fd8d9b4a000)
    libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007fd8d9b30000)
    libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fd8d996f000)
    /lib64/ld-linux-x86-64.so.2 (0x00007fd8d9bbb000)

有什么方法可以为Rust生成静态可执行文件吗?我查看了其他类似的答案,它们都提到使用musl。是否没有使用glibc生成静态可执行文件的方法?如果必须使用其他方式,你能否提供一步一步的命令来生成一个带有rustc的静态可执行文件?

1
您尝试过这个答案吗?https://dev59.com/olwZ5IYBdhLWcg3wJ9nN#44387312 - Boiethios
@FrenchBoiethios 这个答案是针对 Windows 的。 - Lone Learner
你的系统上是否有glibc的静态版本? - Shepmaster
@Shepmaster 我该如何检查系统上是否有glibc的静态版本?apt-cache search glibc | grep static没有结果。但是我的系统上有/usr/lib/x86_64-linux-gnu/libc.a。另外,正如我在问题中解释的那样,gcc -static -o cfoo foo.c会生成一个静态二进制文件。所以我猜测我的系统上有静态glibc。我是对的吗? - Lone Learner
2个回答

19

看起来自原回答以来情况可能已经发生了变化。

首先,您需要确保您的系统上有一个静态链接的glibc可用。我正在使用一个RHEL系统,所以我这样做:

sudo yum install glibc-static

我相信Ubuntu / Debian的等效项是:

sudo apt-get install libc6-dev

然后,在编译程序时,您需要向rustc传递选项-C target-feature=+crt-static

$ rustc -o rustfoo -C target-feature=+crt-static main.rs
$ ldd rustfoo
        not a dynamic executable

当然,几乎没有人手动运行rustc。您可以使用 RUSTFLAGS 环境变量指示cargo将此选项传递给rustc,像这样:

RUSTFLAGS="-C target-feature=+crt-static" cargo build --target x86_64-unknown-linux-gnu

即使你正在为主机平台构建,也必须添加--target选项的原因是,如果不这样做,则在构建编译时代码(例如过程宏等)时,通过RUSTFLAGS提供的选项将被应用,可能会导致这些代码无法编译。如果明确指定一个目标平台,则RUSTFLAGS仅在为目标平台编译代码时应用。另请参见此错误报告

如果您希望始终以这种方式静态构建,而不需要环境变量,则可以在项目顶级目录中创建文件.cargo/config.toml,并将以下内容放入其中:

[build]
rustflags = ["-C", "target-feature=+crt-static"]
target = "x86_64-unknown-linux-gnu"

参考资料:


2
太棒了!目前在处理proc-macro包时出现问题。解决方法是始终指定目标 RUSTFLAGS="-C target-feature=+crt-static" cargo build --target x86_64-unknown-linux-gnu --release。更多信息请参考github上的问题:https://github.com/rust-lang/rust/issues/78210 - michalhosna
1
很好的发现@michalhosna。我已经更新了答案,包括那个信息。 - harmic


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