使用Rust FFI时如何初始化不透明的C结构?

EFT*_*FTH 9 ffi rust

这是我想用C代码做的事情:

#include <some_lib.h>
int main() {
    some_lib_struct_t x;
    some_lib_func(&x);
}
Run Code Online (Sandbox Code Playgroud)

如何在Rust中使用库?这是我到目前为止所得到的:

extern crate libc; // 0.2.51

struct some_lib_struct_t;

#[link(name = "some_lib")]
extern "C" {
    fn some_lib_func(x: *mut some_lib_struct_t);
}

fn main() {
    let mut x: some_lib_struct_t;
    unsafe {
        some_lib_func(&mut x);
    }
}
Run Code Online (Sandbox Code Playgroud)

编译时我(不是那么令人惊讶)得到错误:__CODE__.

编辑

在阅读Shepmaster的回答后,我仔细查看了图书馆的标题.就像他们说的那样,__CODE__只是一个指向的指针的typedef __CODE__.我做了以下更改:

error[E0381]: borrow of possibly uninitialized variable: `x`
  --> src/main.rs:13:23
   |
13 |         some_lib_func(&mut x);
   |                       ^^^^^^ use of possibly uninitialized `x`
Run Code Online (Sandbox Code Playgroud)

它的工作原理!但是我收到了警告__CODE__.这是我可以压制的东西还是我需要改变一些东西?

She*_*ter 8

最安全的答案是自己初始化结构:

let mut x: some_lib_struct_t = some_lib_struct_t;
unsafe {
    some_lib_func(&mut x);
}
Run Code Online (Sandbox Code Playgroud)

与C代码最接近的模拟是使用mem::uninitialized:

unsafe {
    let mut x: some_lib_struct_t = std::mem::uninitialized();
    some_lib_func(&mut x);
}
Run Code Online (Sandbox Code Playgroud)

必须确保some_lib_func完全初始化结构的所有成员,否则将不安全的外部泄漏unsafe块.

说到"结构的成员",我几乎可以保证你的代码不会做你想要的.您已定义some_lib_struct_t为零大小.这意味着不会为它分配任何堆栈空间,并且对它的引用将不是您的C代码所期望的.

您需要在Rust中镜像C结构的定义,以便可以分配适当的大小,填充和对齐.通常,这意味着使用repr(C).

很多时候,C库总是通过返回指向opaque类型的指针来避免暴露它们的内部结构表示: