编程语言
首页 > 编程语言> > Rust FFI 编程 - Rust导出共享库04

Rust FFI 编程 - Rust导出共享库04

作者:互联网

这节我们主要关注 Rust 导出共享库时的错误处理。主要涉及到:

错误对于软件来说是不可避免的,错误处理是保证程序健壮性的前提,编程语言一般都会有一些机制来处理出现错误的情况,大致分为两种:抛出异常和作为值返回。Rust 中没有异常,而是将错误作为值返回,并且通过将错误分成两个主要类别可恢复错误(Result<T, E>)和不可恢复错误(panic!)提供了 Rust 特色的错误处理机制。C 虽然错误处理机制简陋,但最常见也是将错误作为值返回,其中的 POSIX 风格就是函数返回一个int值,其中0表示成功,而负值表示错误。基于setjmp/longjmp的错误处理不属于此节的讨论范畴,如果有必要后面再做说明。

Option 和 Result 的处理

在 FFI 中允许使用任何T: SizedOption<&T>Option<&mut T>,代替显式地进行无效性(nullity )检查的指针。这是由于 Rust 保证了可空指针优化(nullable pointer optimization),在 C 端可以接受可空指针。C 端的NULL在 Rust 中被转换为None,而非空指针被封装在Some中。我们知道 Rust 中的Result <T,E>是用于返回和传播错误的类型,其实质是一个枚举,其中Ok(T)表示成功并包含一个值,而Err(E)表示错误并包含一个错误值。在设计 Rust 导出共享库时,我们可以使用返回值的错误处理机制,使 C 调用者可以通过检查返回值来检测何时发生了错误,并获得相关的错误信息。对于 Option 和 Result 的转换,我们一般采取以下一些方法:

本节我们采取简单的返回数值,示例如下:

#[no_mangle]
pub unsafe extern "C" fn handle_option(x: c_float, y: c_float) -> i32 {
   // The return value of the function is an option
   let result = divide(x, y);

   // Pattern match to retrieve the value
   match result {
       // The division was valid
       Some(_) => 0,
       // The division was invalid
       None    => -1,
   }
}

#[no_mangle]
pub unsafe extern "C" fn handle_result(s: *const c_char) -> i32 {
   if (s as *mut c_void).is_null() {
       return -1;
   }

   let vb = CStr::from_ptr(s).to_str().unwrap();
   let version = parse_version(vb);
   match version {
       Ok(_) => 0,
       Err(_) => -1,
   }
}

panic 的处理

同时跨越 FFI 边界的panic会导致未定义的行为(Undefined Behavior,UB),我们还需要确保我们的 FFI 绑定是异常安全(Exception Safety)的。也就是说如果 Rust 导出库的代码可能会出现panic,则需要有个处理机制。在 FFI 绑定时我们可以使用catch_unwind将其包含在 Rust 中,从而不跨越 FFI 边界。

use std::panic::catch_unwind;

fn may_panic() {
   if rand::random() {
       panic!("panic happens");
   }
}

#[no_mangle]
pub unsafe extern "C" fn no_panic() -> i32 {
   let result = catch_unwind(may_panic);
   match result {
       Ok(_) => 0,
       Err(_) => -1,
   }
}

请注意,catch_unwind只能捕获 Rust 中的展开(unwindingpanic,而不能处理 Rust 中的终止程序(abortpanic

当出现 panic 时,Rust 程序默认会开始展开,这意味着 Rust 会回溯栈并清理它遇到的每一个函数的数据,不过这个回溯并清理的过程有很多工作。另一种选择是直接终止,这会不清理数据就退出程序。那么程序所使用的内存需要由操作系统来清理。通过在 Cargo.toml 的 [profile] 部分增加 panic = 'abort',程序在panic时会由展开切换为终止。


标签:04,错误,FFI,错误处理,panic,Rust,Option
来源: https://blog.51cto.com/u_15127605/2763094