简单介绍 Rust 调用 C 常用的示例代码。
简介
使用的时候可以在 Rust 工程中,或者是独立的 C 库提供了 Rust 的封装。如下通过简单示例介绍,提供一个 libsysx
的系统库,通过 C 语言对底层的系统调用进行封装,然后由 Rust 调用。
目录结构如下。
├── Cargo.toml
├── README.md
├── libsysx/
│ ├── CMakeLists.txt
│ ├── include/
│ │ └── math.h
│ └── src/
│ └── math.c
├── build.rs
└── src/
└── main.rs
手动编译
对应内容为。
// math.h
#ifndef __LIBSYSX_H__
#define __LIBSYSX_H__
int add(int left, int right);
#endif
// math.c
int add(int left, int right) {
return left + right;
}
然后手动生成动态库和静态库,并复制到 target/{debug,release}/deps
目录下。
$ gcc -shared -fPIC -o libsysx.so -I include src/math.c
$ gcc -c -o math.o -I include src/math.c
$ ar rc libsysx.a math.o
对应的 Rust 代码如下,使用静态库链接,如果是动态库需要将路径添加到 LD_LIBRARY_PATH
环境变量中,然后通过 cargo run/build
运行即可。
//#[link(name = "math", kind = "dylib")]
#[link(name = "math", kind = "static")]
extern "C" {
fn add(left: isize, right: isize) -> isize;
}
fn main() {
println!("{}", unsafe { add(10, 20) });
}
CMake
可以通过 Makefile
管理,不过当前大部分的 C/CPP 工程会使用 CMake
管理,这里同时会生成动态和静态库,更多内容可以参考 CMake 使用简介 。
CMAKE_MINIMUM_REQUIRED(VERSION 3.9)
PROJECT(sysx)
SET(SRC_LIST src/math.c)
SET(LIB_STATIC_NAME ${PROJECT_NAME}-static)
SET(LIB_SHARED_NAME ${PROJECT_NAME}-shared)
ADD_LIBRARY(${LIB_STATIC_NAME} STATIC ${SRC_LIST})
SET_TARGET_PROPERTIES(${LIB_STATIC_NAME} PROPERTIES OUTPUT_NAME ${PROJECT_NAME})
ADD_LIBRARY(${LIB_SHARED_NAME} SHARED ${SRC_LIST})
SET_TARGET_PROPERTIES(${LIB_SHARED_NAME} PROPERTIES OUTPUT_NAME ${PROJECT_NAME})
INSTALL(TARGETS ${LIB_STATIC_NAME} DESTINATION .)
INSTALL(TARGETS ${LIB_SHARED_NAME} DESTINATION .)
此时常规的可以通过如下命令生成。
cd libsysx && make build && cd build
cmake ..
make
不过建议通过如下命令在 target/libsysx
目录下生成,可以直接在工程目录下执行如下命令。
cmake -S libsysx -B target/libsysx
cmake --build target/libsysx
上述命令也是后续 build.rs
中使用。
build.rs
这里直接使用命令行进行编译,对应代码如下。
use std::process::Command;
fn main() {
let libs = "libsysx";
let target = "target/libsysx";
std::fs::create_dir(target).unwrap();
Command::new("cmake")
.args(&["-S", libs, "-B", target])
.status()
.unwrap();
Command::new("cmake")
.args(&["--build", target])
.status()
.unwrap();
println!("cargo:rerun-if-changed=build.rs");
println!("cargo:rustc-link-search=native={}", target);
println!("cargo:rustc-link-lib=static=sysx");
}
使用示例
基础类型参数
最常见的是基础类型,例如 int
double
等,只需要注意两种不同类型语言之间的差异即可。
int add(int left, int right) {
return left + right;
}
use std::os::raw::c_int;
//#[link(name = "sysx", kind = "dylib")]
#[link(name = "sysx", kind = "static")]
extern "C" {
fn add(left: c_int, right: c_int) -> c_int;
}
fn main() {
println!("add(10, 20) = {}", unsafe { add(10, 20) });
}
同之前一样,需要使用 Rust 中的 std::os::raw
类型。
数组参数
通过指针+整数方式传递,指针指定的真实数据,而整数则是数组的大小,这里仅列出相关的代码。
// math.c
int sum01(const int *arr, int length) {
int total = 0;
for (int i = 0; i < length; i++) {
total += arr[i];
}
return total;
}
int sum02(const int arr[5]) {
int total = 0;
for (int i = 0; i < 5; i++) {
total += arr[i];
}
return total;
}
// main.rs
use std::os::raw::c_int;
//#[link(name = "sysx", kind = "dylib")]
#[link(name = "sysx", kind = "static")]
extern "C" {
fn sum01(arr: *const c_int, length: c_int) -> c_int;
fn sum02(arr: &[c_int; 5]) -> c_int;
}
fn main() {
let numbers: [c_int; 5] = [1, 2, 3, 4, 5];
unsafe {
let total = sum01(numbers.as_ptr(), numbers.len() as c_int);
println!("sum(1, 2, 3, 4, 5) = {}", total);
let total = sum02(&numbers);
println!("sum(1, 2, 3, 4, 5) = {}", total);
}
}
如果是可修改的,那么可以通过 as_mut_ptr()
替换原 as_ptr()
函数。还有一种场景是已经明确知道数组的长度,就可以通过 [c_int; 5]
类似的方式传入。
结构体参数
// math.c
#include <unistd.h>
struct Rectangle {
int width, height;
};
int area(const struct Rectangle *rect) {
if (rect == NULL) {
return 0;
}
return rect->height * rect->width;
}
use std::os::raw::c_int;
#[repr(C)]
pub struct Rectangle {
width: c_int,
height: c_int,
}
//#[link(name = "sysx", kind = "dylib")]
#[link(name = "sysx", kind = "static")]
extern "C" {
fn area(rect: &Rectangle) -> c_int;
}
fn main() {
unsafe {
let rect = Rectangle {
width: 10,
height: 20,
};
println!("area(Rectangle<10, 20>)={}", area(&rect));
}
}
上述是不可变指针,如果是可变指针可以通过 &mut T
方式传入。