Rust实践:Win10环境下的openssl交叉编译

Rust实践:Win10环境下的openssl交叉编译

编码文章call10242025-07-29 18:18:533A+A-

Rust支持跨平台,可以指定生成目标平台,交叉编译也是支持的。当然,想要交叉编译成功,还需要指定平台的编译器(如:msvc、gcc等)。

openssl是C语言开发的库,如果在Rust代码中用到openssl,需要先对openssl进行交叉编译。

关于在Linux进行交叉编译的文章很多,这篇文章讲如何在windows 10环境下设置aarch64的交叉编译,并基于openssl实现一个aarch64架构 TCP 双向认证示例。

aarch64的GCC编译器下载及配置

想要在windows进行交叉编译,需要下载能够在windows运行的编译器。aarch64的编译器一般都是在Linux下运行,好在 Linaro 提供了可以在windows上运行的aarch64编译器。

下载完成后,将编译器解压到一个目录,再将bin目录添加到环境变量中,保证在命令行中可以访问到gcc:

写一个helloworld,使用指令 "aarch64-linux-gnu-gcc.exe hello.c -o hello"编译成可执行文件,就可以将其复制到arm板上运行了。

我手里没有arm板,在虚拟机中安装ubuntu,使用 qemu的用户模式 模拟 aarch64 运行。

在ubuntu下,想要运行aarch64程序,需要安装以下组件:

sudo apt install qemu qemu-user gcc-aarch64-linux-gnu

aarch64编译器用于提供aarch64程序的动态链接库。运行指令需要使用指令参数 -L /usr/aarch64-linux-gnu 指定aarch64库位置。

Rust配置交叉编译

rustc是Rust语言的前端编译器,它会将Rust源码编译成 LLVM IR,然后再调用链接器(link.exe、ld等)链接,最终形成可执文件/动态库。

rustc本身并不提供链接的功能,这也是为什么在windows下要安装msvc编译器、在linux要安装gcc的原因。

gcc在前面一小节已经装好,在Rust中,需要安装一下aarch64的编译目标架构组件。在Rust支持的编译目标中,比较符合aarch64架构的目标是 aarch64-unknown-linux-gnu

rustup target add aarch64-unknown-linux-gnu

使用cargo创建一个新项目,交叉编译需要进行一些额外的设置(非常重要)。在根目录下新建一个 .cargo 目录,并创建一个 config.toml 文件:

//  .cargo/config.toml文件
[build]
target = "aarch64-unknown-linux-gnu" # 编译目标

[target.aarch64-unknown-linux-gnu]
linker = "aarch64-linux-gnu-gcc" # 交叉编译编译器

使用 cargo build编译,成功编译成可执行文件,可执行文件位于
target/aarch64-unknown-linux-gnu\debug 下:

从可执行文件的信息来看,这是一个AArch64架构的程序。并且依赖thread、dl、c、gcc_s库,这些库都要求是aarch64的。因此,运行时需要提供库位置:

qemu-aarch64 -L /usr/aarch64-linux-gnu ./hello

openssl交叉编译

我们需要在程序中集成openssl的TLS功能。因此,我们只需要编译库文件就可以。编译openssl,最好有一个linux环境,由于我们是在windows下,可以使用Cygwin这种类Linux环境。这里使用msys2来编译openssl。

  • MSYS2安装:

下载MSYS2安装程序,双击安装。安装完成后打开 "msys2.exe"终端程序。

  • openssl编译:

下载openssl源码,msys终端跳转到openssl源码所在目录(这里将openssl源码放在 e:/opensource 目录下),解压openssl压缩包 => 编译配置 => 编译 => 安装:

# 将编译器的bin目录添加到 PATH 环境变量中
export PATH=$PATH:/d/software/gcc-linaro-7.5.0-2019.12-i686-mingw32_aarch64-linux-gnu/bin
cd /e/opensource
tar -zxvf openssl-3.5.1.tar.gz
cd openssl-3.5.1
# 编译配置,只编译静态库,不编程动态库和openssl可执行文件,并且不生成测试文件
./Configure linux-aarch64 --cross-compile-prefix=aarch64-linux-gnu- --prefix=installed no-asm no-async no-shared
make
make install

上面的编译配置中,只生成静态库文件。编译成功后,会在当前目录下的installed输出openssl的输出文件,包含头文件、库文件、帮助文档等。

集成到Rust中

在Cargo.toml配置中,添加openssl依赖。在环境变量中配置OPENSSL_DIR:

// Cargo.toml文件

# 服务端
[[bin]]
name = "tls_server"
path = "src/bin/server.rs"

# 客户端
[[bin]]
name = "tls_client"
path = "src/bin/client.rs"

#  openssl依赖
[dependencies]
openssl = "0"

// 环境变量配置
set OPENSSL_DIR=E:\opensource\openssl-3.5.1\installed

接下分别写一个服务端和客户端,测试一下openssl的双向认证功能:

// server.rs
use std::io::{Read, Write};
use std::net::{SocketAddrV4, TcpListener};
use std::str::FromStr;
use openssl::ssl::{SslAcceptor, SslFiletype, SslMethod, SslVerifyMode};

fn openssl_server_test(host: &str, port: u16) -> Result<(), Box<dyn std::error::Error>> {
    let mut builder = SslAcceptor::mozilla_intermediate(SslMethod::tls())?;
    builder.set_private_key_file("certs/server.key", SslFiletype::PEM)?;
    builder.set_certificate_chain_file("certs/server.crt")?;
    builder.set_ca_file("certs/ca.crt")?;
    // 设置客户端认证
    builder.set_verify(SslVerifyMode::PEER | SslVerifyMode::FAIL_IF_NO_PEER_CERT);
    let acceptor = builder.build();
    // 绑定地址
    let addr = SocketAddrV4::from_str(format!("{}:{}", host, port).as_str())?;
    let listener = TcpListener::bind(addr)?;
    // 等待客户端连接
    let (stream, addr) = listener.accept()?;
    // ssl握手
    let mut stream = acceptor.accept(stream)?;
    // 读到数据
    let mut buffer: [u8; 1024] = [0; 1024];
    let r = stream.read(&mut buffer)?;
    // 写数据
    stream.write(&buffer[0..r])?;
    stream.shutdown()?;
    Ok(())
}

fn main() -> Result<(), Box<dyn std::error::Error>> {
    openssl_server_test("127.0.0.1",8080)
}

// client.rs
use std::io::{Read, Write};
use std::net::TcpStream;
use openssl::ssl::{SslConnector, SslFiletype, SslMethod};

fn openssl_client_test(host: &str, port:i32) -> Result<(), Box<dyn std::error::Error>> {
    // 设置客户端证书
    let mut connector = SslConnector::builder(SslMethod::tls())?;
    connector.set_ca_file("certs/ca.crt")?;
    connector.set_private_key_file("certs/client.key", SslFiletype::PEM)?;
    connector.set_certificate_chain_file("certs/client.crt")?;
    let connector = connector.build();
    // TCP连接
    let stream = TcpStream::connect(format!("{}:{}",host,port))?;
    // SSL握手
    let mut stream = connector.connect(host, stream)?;
    // 写数据
    stream.write_all("Hello".as_bytes())?;
    //  读数据
    let mut response = vec![];
    stream.read_to_end(&mut response)?;
    println!("{}", std::str::from_utf8(&response)?);
    Ok(())
}

fn main() -> Result<(), Box<dyn std::error::Error>> {
    openssl_client_test("127.0.0.1",8080)
}

使用cargo build进行编译,会在 target\aarch64-unknown-linux-gnu\debug 目录下生成tls_server和tls_client两个文件。

从文件信息来看,生成的可执行文件格式是aarch64格式的。编译的openssl是静态库,最终生成的可执行文件自身就包含openssl,不再外部依赖openssl。

使用qemu-aarch64运行一下(证书生成参见《Rust中的TLS库深度对比:openssl、rustls 和 native-tls》):

总结

上面只演示了aarch64的交叉编译过程。总体来讲,Rust交叉编译分以下几步:

  • 配置编译目标的交叉编译环境
  • 安装与编译目标匹配的Rust目标架构组件
  • 如果Rust库依赖C库,需要使用交叉编译器先编译C库,再进行库配置
  • 在项目中配置交叉编译,关键步骤是配置.cargo/config.toml文件

以上就是基于windows的arrch64交叉编译过程,如果遇到问题,欢迎在评论区留言。

点击这里复制本文地址 以上内容由文彬编程网整理呈现,请务必在转载分享时注明本文地址!如对内容有疑问,请联系我们,谢谢!
qrcode

文彬编程网 © All Rights Reserved.  蜀ICP备2024111239号-4