本帖最后由 wuhanstudio 于 2020-4-3 11:58 编辑
更新地址: https://codimd.wuhanstudio.cc/s/1ZTNBKoD5
最近 Mozilla 的 Rust 和 Google 的 Golang 两门新语言非常受关注。
不过 Golang 除了 Docker 这个标志性的项目,在服务端配合 gRPC 也已经应用非常广泛了,可以说已经是相当很成熟了;然而相比之下 Rust 还是旁观的多,实际用的少,微软和 Facebook 也都只是宣称实验性地尝试用 Rust 实现一些新功能,毕竟 Facebook 也在研发自己的操作系统。
Rust 的官方主页标语就是 reliable and efficient,吉祥物也是硬壳的螃蟹,自然就有人想到把 Rust 应用到安全可靠的嵌入式。
嵌入式的特点就在于指令集 (ISA) 特别多,如果一门语言想要应用到嵌入式,首先自然是得能编译出兼容不同指令集的程序。Rust 早期编译器是 rustc 提供代理 (proxy),将编译任务传递到后端实际的编译器来进行编译,这样就可以编译出不同指令集的程序。但是在 Rust 1.28 之后,终于拥抱 LLVM,Golang 和 Swift 也都是积极拥抱 LLVM(LLVM 势不可挡)。LLVM 特点就是在于提供抽象的 IR (intermediate representation),先将源码翻译为 LLVM IR,再由后端生成不同指令集的程序,效率高,还能兼容不同平台。
所以在 Rust 积极拥抱 LLVM 之后,在嵌入式领域开发更加方便了,接下来就介绍一下如何在 STM32 (Cortex M3) 和 GD32 (RISC-V) 上用 Rust 开发。
首先自然是得搭建好 Rust 的开发环境,这在 Rust 的主页上已经介绍得很详细了,安装也很简单。
不过值得注意的是,在 Windows 下安装 Rust 会提示需要安装 Visual Studio,其实这并不是必须的,前面提到了 Rust 已经集成了 LLVM,所以本身就可以直接编译,不需要而外的编译器。当然,有些 rust 组件安装还是需要自己电脑上有编译器的,比如后面会提到的 cargo-binutils。不过在 Windows 安装 Rust 的时候暂且跳过 Visual Studio 也是完全没有问题的 ;)
如果安装好 Rust 的开发环境,重启一下控制台,应当能运行 rustup (工具链管理),rustc (编译器),cargo (软件包管理)。
这里简单介绍一下 Rust 的组成。
首先 rustc 是编译器,可以用 rustc 编译 rust 程序 (*.rs)生成二进制文件。但是一个个文件手动 rustc 编译是非常辛苦的,所以 rust 提供了 cargo 软件包管理器,可以 cargo new 生成项目,cargo build 编译,cargo run 运行,非常方便。
rustup 这个工具是用来管理 rust 工具链的,前面提到 rust 集成了 LLVM,但是程序要想在嵌入式环境运行,除了编译器还需要一个运行环境(Runtime),而 rustup target list 就可以看到不同的目标环境,这些环境会提供编译好的 rust-std,例如我安装好了 riscv 和 armv7 的环境。
rustup 还可以列出当前主机的编译器,我默认用的是 mysis 提供的 gnu-gcc:
rustup 也可以用来安装新的 rust 组件,例如 cargo 软件包管理器, LLVM 工具:
所以 rus 组成常用的就是 rustup 管理工具链,cargo 开发项目。
首先介绍一下如何在 STM32F103 上用 Rust 驱动经典的 SSD1306。
这里用到的是 ssd1306 这个 crate (Rust 的软件包叫 crate,发布在 crates.io)。
比如我们以这个 crate 的例程为例,首先把源码下载下来:
简单介绍一下 rust 项目的组成,Cargo.toml 这个文件里定义了项目的依赖,定义格式很简单,例如这里定义了项目依赖 cortex-m 的 runtime:
在 Cargo.toml 里面定义好依赖之后,编译的时候会自动下载依赖,非常方便。
在编译之前,首先当然是要确定目标开发板的硬件,比如这里要告诉 rust 我们希望生成 stm32f103 的目标(thumbv7em-none-eabi),而这个是在项目根目录 .cargo/config 文件里指定的,当然也可以编译的时候用命令行参数传进去 —target=thumbv7em-none-eabi
虽然现在指定了 MCU 的指令集架构,但是我们并没有告诉编译器开发板的 Flash,RAM 资源,为了让 rust 知道该如何链接,和 Keil 一样需要指定链接脚本,打开 memory.x 就可以看到了,这里我把 Flash 和 RAM 改成了 stm32f103rct6 的配置,默认是 stm32f103c8t6.
在编译之前,之前提到过需要为目标开发板提供预编译的运行环境 (runtime):
终于可以开始编译了,例如编译例程的 image_i2c:
前面提到,也可以在这里指定目标 MCU 的指令集:
编译完成就可以看到在 target/thumbv7m-none-eabi/release/examples 下多了,但是这个文件有 1M 多大小,很明显是包含了调试信息的 elf 文件,我们需要用 objcopy 生成最后上传到开发板的 bin 文件。
这里我用的是 arm-none-eabi-objcopy,可以在 GNU-Toolchain 下载。
最后把生成的 bin 文件用 st-link 或者其他工具上传到开发板就可以了 ;)
在 risc-v 的开发板上用 rust 开发其实也差不多,这里顺便以 GD32F103 的 Longan Naon 开发板为例,介绍一下 rust 依赖的组成。
大致是这样的组成:
不过这些都已经写好了,GD32F103的 Bumblebee 内核是 华中科技大学 对面芯来科技设计的,而 GD32F103 的 RUST HAL 实现刚好是华科本科生写的,恰好我也是华科的研究生,太巧了 ;)
所以用 rust 开发嵌入式不需要重复实现相同的功能,使用开源的 crate 软件包就可以了。
比如我们先下载下来项目:
项目结构和之前是一模一样的,这里就不重复介绍了,和之前一样,编译前需要添加 risc-v 的运行环境:
接下来我们就可以编译了:
同样的,编译之后我们需要用 objcopy 生成 bin 文件:
熟悉了这样的流程之后会发现其实挺顺手的,这里的工具链我是用的芯来科技官网的 RISC-V GNU Tool。
最后自然是用 dfu-utils 把 bin 文件上传到开发板了。
最后这里有一个没有解决的小问题,上面编译虽然用的都是 rust 工具链,但是连接其实还是用到了 arm-gcc 和 riscv-gcc。
rust 其实提供了 cargo-binutils 可以调用 LLVM 的 objcopy 来生成最终的二进制文件。
不过我用这个工具生成的 bin 传到开发板里没有正常运行,不知道有没有办法解决。
前天看到公众号说文档团队解散了。
标题党害人啊!