Rust实践:Win10环境下的openssl交叉编译
nanshan 2025-07-21 16:35 5 浏览 0 评论
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交叉编译过程,如果遇到问题,欢迎在评论区留言。
相关推荐
- CentOS 7 搭建 Harbor2.4.1 Docker镜像仓库
-
上一篇文章我们使用了registry镜像来搭建Docker私有镜像仓库,但是使用体验不是很好,没有一个可管理的UI界面,管理很麻烦。本篇文章将介绍一个新的搭建Docker镜像仓库的工具叫做Har...
- 简单认识认识mqtt及mosquitto(mqtt报文解析)
-
某项目中使用了MQTT作为进程间的通信方式,之前没用过,这两篇笔记我们就来一起学习一下这种方式。MQTT的一些介绍以下介绍内容来自《[野火]《LwIP应用开发实战指南》MQTT协议全称是Messa...
- 全源码打造高性能 LNMP 架构: 实战教程(2025最新版)
-
适用场景:企业生产环境、自建Web服务、深度性能调优操作系统:CentOS7/8、RockyLinux、Debian、Ubuntu(本文以CentOSStream9为例)技术栈:N...
- Nacos3.0重磅来袭!全面拥抱AI,单机及集群模式安装详细教程!
-
之前和大家分享过JDK17的多版本管理及详细安装过程,然后在项目升级完jdk17后又发现之前的注册和配置中心nacos又用不了,原因是之前的nacos1.3版本的,版本太老了,已经无法适配当前新的JD...
- Ubuntu24.04.2 企业级MinIO存储系统部署指南
-
一、概要1.1MinIO架构解析MinIO是一款高性能的云原生对象存储系统,采用Golang开发并遵循ApacheLicensev2.0协议。其核心架构基于纠删码(ErasureCode)技...
- 从零打造自己的 国产鸿蒙(OpenHarmony)定制系统-完整可落地流程
-
适用版本:OpenHarmony4.0/5.0Standard目标人群:想在x86PC、RK3568开发板或自有硬件上裁剪、加品牌、预装应用并生成可刷机镜像的开发者/团队目录环境准...
- 一次暂未成功的dify安装经历(dify怎么安装)
-
前几天在阿里云买了一台机,这几天一直在尝试安装dify,到现在还没安装上我是按这个教程装的https://blog.csdn.net/2401_82469710/article/details/14...
- ZLMediaKit教程(五)支持webrtc(webrtc lib)
-
ZLMediaKit系列文章(共六篇):ZLMediaKit流媒体(一)编译安装ZLMediaKit教程(二)主程序和配置文件解析ZLMediaKit教程(三)URL规则ZLMediaKit教程...
- Linux程序安装与管理指南(linux程序安装命令大全)
-
在Linux系统中,安装和管理程序主要通过包管理器和手动编译安装两种主要方式实现。以下是详细的操作指南,涵盖常见发行版(如Ubuntu/Debian、CentOS/RHEL、Fedora等)的用法。一...
- 离线状态下安装 Nginx 各个模块?这篇攻略让你轻松搞定
-
你是不是也在为离线状态下安装Nginx各个模块而发愁?在互联网大厂后端开发工作中,我们常常会遇到一些特殊的网络环境,比如公司内部的离线服务器,或是处于隔离状态的测试环境。当需要在这些离线环境中安装...
- Rust实践:Win10环境下的openssl交叉编译
-
Rust支持跨平台,可以指定生成目标平台,交叉编译也是支持的。当然,想要交叉编译成功,还需要指定平台的编译器(如:msvc、gcc等)。openssl是C语言开发的库,如果在Rust代码中用到open...
- Linux下Blackwell架构显卡(RTX5070/5090)编译PaddlePaddle指南
-
Blackwell显卡架构如RTX5070\5090等显卡当前Paddle预编译版本中包含的GPU架构(即SM架构)是有限的,比如常见的SM75(T4)、SM86(A10)、SM89(...
- 突破操作系统界限,掌握Linux的必备指南
-
#头条创作挑战赛#简介Linux是一种开源的操作系统,它的核心思想是自由和开放。Linux以其稳定性、可靠性和安全性而闻名,被广泛用于服务器和嵌入式设备中。Linux创始人Linux安装在安装Linu...
- Linux日常高频使用的100条命令,强烈建议收藏
-
查看系统信息如何查看系统版本:uname-alsb_release-acat/etc/os-release如何查看系统内核信息:uname-r如何查看系统CPU信息:lscpucat...
- Linux文件系统结构全解析(linux文件结构详解)
-
对Linux新手而言,“一切皆文件”的设计哲学常让人既兴奋又困惑——打开终端输入ls/,看到的bin、etc、var等目录到底有什么用?如何快速定位关键文件?本文将从Linux文件系统的底层逻...
你 发表评论:
欢迎- 一周热门
-
-
UOS服务器操作系统防火墙设置(uos20关闭防火墙)
-
极空间如何无损移机,新Z4 Pro又有哪些升级?极空间Z4 Pro深度体验
-
手机如何设置与显示准确时间的详细指南
-
NAS:DS video/DS file/DS photo等群晖移动端APP远程访问的教程
-
如何在安装前及安装后修改黑群晖的Mac地址和Sn系列号
-
如何修复用户配置文件服务在 WINDOWS 上登录失败的问题
-
一加手机与电脑互传文件的便捷方法FileDash
-
日本海上自卫队的军衔制度(日本海上自卫队的军衔制度是什么)
-
10个免费文件中转服务站,分享文件简单方便,你知道几个?
-
爱折腾的特斯拉车主必看!手把手教你TESLAMATE的备份和恢复
-
- 最近发表
-
- CentOS 7 搭建 Harbor2.4.1 Docker镜像仓库
- 简单认识认识mqtt及mosquitto(mqtt报文解析)
- 全源码打造高性能 LNMP 架构: 实战教程(2025最新版)
- Nacos3.0重磅来袭!全面拥抱AI,单机及集群模式安装详细教程!
- Ubuntu24.04.2 企业级MinIO存储系统部署指南
- 从零打造自己的 国产鸿蒙(OpenHarmony)定制系统-完整可落地流程
- 一次暂未成功的dify安装经历(dify怎么安装)
- ZLMediaKit教程(五)支持webrtc(webrtc lib)
- Linux程序安装与管理指南(linux程序安装命令大全)
- 离线状态下安装 Nginx 各个模块?这篇攻略让你轻松搞定
- 标签列表
-
- linux 查询端口号 (58)
- docker映射容器目录到宿主机 (66)
- 杀端口 (60)
- yum更换阿里源 (62)
- internet explorer 增强的安全配置已启用 (65)
- linux自动挂载 (56)
- 禁用selinux (55)
- sysv-rc-conf (69)
- ubuntu防火墙状态查看 (64)
- windows server 2022激活密钥 (56)
- 无法与服务器建立安全连接是什么意思 (74)
- 443/80端口被占用怎么解决 (56)
- ping无法访问目标主机怎么解决 (58)
- fdatasync (59)
- 405 not allowed (56)
- 免备案虚拟主机zxhost (55)
- linux根据pid查看进程 (60)
- dhcp工具 (62)
- mysql 1045 (57)
- 宝塔远程工具 (56)
- ssh服务器拒绝了密码 请再试一次 (56)
- ubuntu卸载docker (56)
- linux查看nginx状态 (63)
- tomcat 乱码 (76)
- 2008r2激活序列号 (65)