在老旧的系统环境下使用Rust开发工具
笔者的一台硬件资源紧张的旧电脑安装了Ubuntu-12.04,希望在其上运行比较新的Rust开发工具,但却发现其因glibc较老,不能正常运行cargo工具,结果如下:
yejq@UNIX:~$ uname -a
Linux UNIX 3.13.0-32-generic #57~precise1-Ubuntu SMP Tue Jul 15 03:51:20 UTC 2014 x86_64 x86_64 x86_64 GNU/Linux
yejq@UNIX:~$ lsb_release -a
No LSB modules are available.
Distributor ID: Ubuntu
Description: Ubuntu 12.04.5 LTS
Release: 12.04
Codename: precise
yejq@UNIX:~$ cargo new --bin hello
cargo: /lib/x86_64-linux-gnu/libc.so.6: version `GLIBC_2.17' not found (required by cargo)
如上,cargo
可执行文件对glibc
最低版本要求是GLIBC_2.17
,而Ubuntu-12.04系统的glibc
版本较低,因此不能正常运行。一种可行的解决方法是使用docker容量在Ubuntu-12.04系统下使用新的Rust开发工具;但限于笔者对docker了解程度比较浅,并没有尝试该方法。笔者想到的方法也比较简单直接(此前的博客对该方法有相关的说明):在Ubuntu-12.04系统上安装Ubuntu-22.04系统的glibc
动态库(不替换原先的glibc库),之后修改Rust开发工具的可执行文件,强制其依赖新的glibc
库。
在Ubuntu-22.04下载并解压新的glibc动态库
首先,笔者需要另一台安装了较新版本Ubuntu的系统环境(Ubuntu-22.04),使用apt-get download
命令下载四个相关的deb
软件包:
yejq@ubuntu:~/x64_libs$ apt-get download libc6
Get:1 https://mirrors.tuna.tsinghua.edu.cn/ubuntu jammy-updates/main amd64 libc6 amd64 2.35-0ubuntu3.6 [3,236 kB]
Fetched 3,236 kB in 1s (2,870 kB/s)
yejq@ubuntu:~/x64_libs$ apt-get download libgcc-s1
Get:1 https://mirrors.tuna.tsinghua.edu.cn/ubuntu jammy-updates/main amd64 libgcc-s1 amd64 12.3.0-1ubuntu1~22.04 [53.9 kB]
Fetched 53.9 kB in 0s (123 kB/s)
yejq@ubuntu:~/x64_libs$ apt-get download libstdc++6
Get:1 https://mirrors.tuna.tsinghua.edu.cn/ubuntu jammy-updates/main amd64 libstdc++6 amd64 12.3.0-1ubuntu1~22.04 [699 kB]
Fetched 699 kB in 1s (1,158 kB/s)
yejq@ubuntu:~/x64_libs$ apt-get download zlib1g
Get:1 https://mirrors.tuna.tsinghua.edu.cn/ubuntu jammy-updates/main amd64 zlib1g amd64 1:1.2.11.dfsg-2ubuntu9.2 [58.4 kB]
Fetched 58.4 kB in 0s (145 kB/s)
yejq@ubuntu:~/x64_libs$ ls -lh *.deb
-rw-r--r-- 1 yejq yejq 3.1M Jan 10 20:40 libc6_2.35-0ubuntu3.6_amd64.deb
-rw-r--r-- 1 yejq yejq 53K Jun 29 2023 libgcc-s1_12.3.0-1ubuntu1~22.04_amd64.deb
-rw-r--r-- 1 yejq yejq 683K Jun 29 2023 libstdc++6_12.3.0-1ubuntu1~22.04_amd64.deb
-rw-r--r-- 1 yejq yejq 58K Oct 18 2022 zlib1g_1%3a1.2.11.dfsg-2ubuntu9.2_amd64.deb
之后再手动解压这四个软件包,其中包含了我们要在Ubuntu-12.04系统中安装的动态库。以libgcc-s1
软件包为例,下面是解压的操作:
yejq@ubuntu:~/x64_libs$ ls -lh libgcc-s1_12.3.0-1ubuntu1~22.04_amd64.deb
-rw-r--r-- 1 yejq yejq 53K Jun 29 2023 libgcc-s1_12.3.0-1ubuntu1~22.04_amd64.deb
yejq@ubuntu:~/x64_libs$ ar x libgcc-s1_12.3.0-1ubuntu1~22.04_amd64.deb
yejq@ubuntu:~/x64_libs$ ls -lh control.tar.zst data.tar.zst debian-binary
-rw-r--r-- 1 yejq yejq 1.7K Feb 17 20:15 control.tar.zst
-rw-r--r-- 1 yejq yejq 51K Feb 17 20:15 data.tar.zst
-rw-r--r-- 1 yejq yejq 4 Feb 17 20:15 debian-binary
yejq@ubuntu:~/x64_libs$ tar -axf data.tar.zst
yejq@ubuntu:~/x64_libs$ find ./lib
./lib
./lib/x86_64-linux-gnu
./lib/x86_64-linux-gnu/libgcc_s.so.1
yejq@ubuntu:~/x64_libs$ mkdir -p lib64
yejq@ubuntu:~/x64_libs$ mv ./lib/x86_64-linux-gnu/* ./lib64/
如上,使用ar x
命令解压deb
包可以获得三个文件:control.tar.zst/data.tar.zst/debian-binary
;之后再次解压data.tar.zst
,可得到./lib
文件夹,其中包含了我们想要的动态库。对另外三个deb
需要重复这个解压的操作。最后将所有的动态库全部都集中到./lib64
文件夹下,复制到Ubuntu-12.04系统中(注意软链接的复制)。笔者得到的lib64
文件夹中的内容有(多了一些ncurses的库):
yejq@ubuntu:~/x64_libs$ ls ./lib64
audit libmenu.so.6.3 libpthread.so.0
gconv libm.so.6 libresolv.so.2
ld-linux-x86-64.so.2 libmvec.so.1 librt.so.1
libanl.so.1 libncurses.so.6 libstdc++.so.6
libBrokenLocale.so.1 libncurses.so.6.3 libstdc++.so.6.0.30
libc_malloc_debug.so.0 libnsl.so.1 libthread_db.so.1
libc.so.6 libnss_compat.so.2 libtic.so.6
libdl.so.2 libnss_dns.so.2 libtic.so.6.3
libform.so.6 libnss_files.so.2 libtinfo.so.6
libform.so.6.3 libnss_hesiod.so.2 libtinfo.so.6.3
libgcc_s.so.1 libpanel.so.6 libutil.so.1
libmemusage.so libpanel.so.6.3 libz.so.1
libmenu.so.6 libpcprofile.so libz.so.1.2.11
在Ubuntu-12.04系统上修改新的glibc
笔者在Ubuntu-12.04系统下创建了/home/user/x64_libs
文件夹,并将lib64
文件夹复制到该目录。之后笔者通过modify.sh
脚本,直接修改了lib64
中的动态库:
root@UNIX:/home/user/x64_libs# ls
lib64 modify.sh
root@UNIX:/home/user/x64_libs# ./modify.sh
xxd -g 26 -l 26 -s 11712 lib64/gconv/gconv-modules.cache
INFO: processing [lib64/gconv/gconv-modules.cache], (/usr/lib/x86_64-linux-gnu/) replaced 1 time(s).
xxd -g 26 -l 26 -s 180311 lib64/ld-linux-x86-64.so.2
xxd -g 26 -l 26 -s 182135 lib64/ld-linux-x86-64.so.2
INFO: processing [lib64/ld-linux-x86-64.so.2], (/usr/lib/x86_64-linux-gnu/) replaced 2 time(s).
xxd -g 26 -l 26 -s 1951808 lib64/libc.so.6
xxd -g 26 -l 26 -s 1953808 lib64/libc.so.6
INFO: processing [lib64/libc.so.6], (/usr/lib/x86_64-linux-gnu/) replaced 2 time(s).
xxd -g 16 -l 16 -s 188955 lib64/ld-linux-x86-64.so.2
xxd -g 16 -l 16 -s 204139 lib64/ld-linux-x86-64.so.2
INFO: processing [lib64/ld-linux-x86-64.so.2], (/etc/ld.so.cache) replaced 2 time(s).
xxd -g 22 -l 22 -s 180288 lib64/ld-linux-x86-64.so.2
xxd -g 22 -l 22 -s 182112 lib64/ld-linux-x86-64.so.2
INFO: processing [lib64/ld-linux-x86-64.so.2], (/lib/x86_64-linux-gnu/) replaced 2 time(s).
xxd -g 9 -l 9 -s 180344 lib64/ld-linux-x86-64.so.2
xxd -g 9 -l 9 -s 182168 lib64/ld-linux-x86-64.so.2
INFO: processing [lib64/ld-linux-x86-64.so.2], (/usr/lib/) replaced 2 time(s).
xxd -g 5 -l 5 -s 26613 lib64/ld-linux-x86-64.so.2
xxd -g 5 -l 5 -s 44617 lib64/ld-linux-x86-64.so.2
xxd -g 5 -l 5 -s 180338 lib64/ld-linux-x86-64.so.2
xxd -g 5 -l 5 -s 182162 lib64/ld-linux-x86-64.so.2
INFO: processing [lib64/ld-linux-x86-64.so.2], (/lib/) replaced 4 time(s).
xxd -g 27 -l 27 -s 189539 lib64/ld-linux-x86-64.so.2
xxd -g 27 -l 27 -s 205047 lib64/ld-linux-x86-64.so.2
INFO: processing [lib64/ld-linux-x86-64.so.2], (/lib64/ld-linux-x86-64.so.2) replaced 2 time(s).
xxd -g 27 -l 27 -s 1982000 lib64/libc.so.6
INFO: processing [lib64/libc.so.6], (/lib64/ld-linux-x86-64.so.2) replaced 1 time(s).
`/lib64/ld-linux-x86-64.so.y' -> `/home/user/x64_libs/lib64/ld-linux-x86-64.so.2'
脚本modify.sh
调用了笔者编写的一个简单的hed
工具,用于批量替换二进制文件中的字符串或二进制数据(有需要可以联系笔者),该脚本的内容如下:
root@UNIX:/home/user/x64_libs# cat modify.sh
#!/bin/bash
REPSTR='/home/user/x64_libs/lib64/'
OLDSTR='/usr/lib/x86_64-linux-gnu/'
if [ $UID -ne 0 ] ; then
echo "Error, run again as root." 1>&2
exit 1
fi
hed 'lib64/gconv/gconv-modules.cache' \
'lib64/ld-linux-x86-64.so.2' \
'lib64/libc.so.6' -s \
"${OLDSTR}" "${REPSTR}"
sync ; sleep 1
# /lib/
# /usr/lib/
# /lib/x86_64-linux-gnu/
hed 'lib64/ld-linux-x86-64.so.2' -s \
'/etc/ld.so.cache' \
'/etc/ld.so.cachy' \
'/lib/x86_64-linux-gnu/' \
'/NON/x86_64-linux-gnu/' \
'/usr/lib/' \
'/usr/NON/' \
'/lib/' \
'/NON/'
sync ; sleep 1
hed 'lib64/ld-linux-x86-64.so.2' \
'lib64/libc.so.6' -s \
'/lib64/ld-linux-x86-64.so.2' \
'/lib64/ld-linux-x86-64.so.y'
ln -sv "$(readlink -f ./lib64/ld-linux-x86-64.so.2)" \
'/lib64/ld-linux-x86-64.so.y'
sync
该操作完成后,接下来就需要修改Rust开发工具了。
修改安装于Ubuntu-12.04系统上的Rust工具链
笔者将Rust开发工具链安装到了/opt/rust-lang
路径下(可参考笔者之前的相关博客);实际上,这个工具链是在Ubuntu-22.04下安装的,之后笔者将其打包复制到了Ubuntu-12.04系统中。执行以下命令,可以批量替换/opt/rust-lang
路径下的可执行文件使用的动态链接器(从而强制其使用在Ubuntu-22.04系统下载的新的glibc动态库):
root@UNIX:/opt/rust-lang# find ./ -type f -print0 | xargs -0 -i hed {} -s /lib64/ld-linux-x86-64.so.2 /lib64/ld-linux-x86-64.so.y
以上命令执行完成后,就可以在Ubuntu-12.04系统中正常使用新版本的Rust开发工具了:
yejq@UNIX:~$ cargo --version
cargo 1.72.1 (103a7ff2e 2023-08-15)
yejq@UNIX:~$ cargo new --bin hello
Created binary (application) `hello` package
yejq@UNIX:~$ cd hello/
yejq@UNIX:~/hello$ ls
Cargo.toml src
yejq@UNIX:~/hello$ cargo build --release -j1
Compiling hello v0.1.0 (/home/yejq/hello)
Finished release [optimized] target(s) in 1.00s
yejq@UNIX:~/hello$ ./target/release/hello
Hello, world!
这种方法虽然可行,但仍存在一个问题:Ubuntu-12.04系统安装的gcc版本太老了,它作为Rust在编译构建时的链接器,可能会存在链接失败的问题。不过针对这个潜在的问题,我们仍有解决方案:从此处下载新版本的gcc编译器,通过TARGET_CC
这一与Rust编译链接相关的环境变量强制其使用新版本的gcc作为链接器;这里笔者就不展开了。文章来源:https://www.toymoban.com/news/detail-829623.html
总结
这种修改、替换应用使用的动态链接器(从而间接指定使用的glibc动态库)的方法,可以方便地为老系统安装一些新的应用。在一些工作环境受限的条件下(如不能安装新的系统作为开发环境),能够让我们不受过多的限制,不影响我们对开发工具的选择。文章来源地址https://www.toymoban.com/news/detail-829623.html
到了这里,关于在Ubuntu-12.04环境下使用新的Rust开发工具的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!