2021-03-23 12:14:33 +00:00
|
|
|
use core::slice;
|
|
|
|
|
|
|
|
use crate::{
|
|
|
|
error::Error,
|
|
|
|
micropython::{
|
|
|
|
map::{Map, MapElem},
|
|
|
|
obj::Obj,
|
2021-09-09 14:53:05 +00:00
|
|
|
runtime::raise_exception,
|
2021-03-23 12:14:33 +00:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2021-10-06 21:21:47 +00:00
|
|
|
/// Perform a call and convert errors into a raised MicroPython exception.
|
|
|
|
/// Should only called when returning from Rust to C. See `raise_exception` for
|
|
|
|
/// details.
|
2021-09-16 11:48:26 +00:00
|
|
|
pub unsafe fn try_or_raise<T>(func: impl FnOnce() -> Result<T, Error>) -> T {
|
|
|
|
func().unwrap_or_else(|err| unsafe {
|
|
|
|
raise_exception(err);
|
|
|
|
})
|
2021-03-23 12:14:33 +00:00
|
|
|
}
|
|
|
|
|
2021-09-16 11:48:26 +00:00
|
|
|
/// Extract kwargs from a C call and pass them into Rust. Raise exception if an
|
2021-10-06 21:21:47 +00:00
|
|
|
/// error occurs. Should only called when returning from Rust to C. See
|
|
|
|
/// `raise_exception` for details.
|
2021-09-16 11:48:26 +00:00
|
|
|
pub unsafe fn try_with_kwargs(
|
|
|
|
kwargs: *const Map,
|
|
|
|
func: impl FnOnce(&Map) -> Result<Obj, Error>,
|
|
|
|
) -> Obj {
|
|
|
|
let block = || {
|
2021-09-09 14:53:05 +00:00
|
|
|
let kwargs = unsafe { kwargs.as_ref() }.ok_or(Error::MissingKwargs)?;
|
2021-03-23 12:14:33 +00:00
|
|
|
|
|
|
|
func(kwargs)
|
2021-09-16 11:48:26 +00:00
|
|
|
};
|
|
|
|
unsafe { try_or_raise(block) }
|
2021-03-23 12:14:33 +00:00
|
|
|
}
|
|
|
|
|
2021-09-16 11:48:26 +00:00
|
|
|
/// Extract args and kwargs from a C call and pass them into Rust. Raise
|
2021-10-06 21:21:47 +00:00
|
|
|
/// exception if an error occurs. Should only called when returning from Rust to
|
|
|
|
/// C. See `raise_exception` for details.
|
2021-09-16 11:48:26 +00:00
|
|
|
pub unsafe fn try_with_args_and_kwargs(
|
2021-03-23 12:14:33 +00:00
|
|
|
n_args: usize,
|
|
|
|
args: *const Obj,
|
|
|
|
kwargs: *const Map,
|
|
|
|
func: impl FnOnce(&[Obj], &Map) -> Result<Obj, Error>,
|
|
|
|
) -> Obj {
|
2021-09-16 11:48:26 +00:00
|
|
|
let block = || {
|
2021-03-23 12:14:33 +00:00
|
|
|
let args = if args.is_null() {
|
|
|
|
&[]
|
|
|
|
} else {
|
|
|
|
unsafe { slice::from_raw_parts(args, n_args) }
|
|
|
|
};
|
2021-09-09 14:53:05 +00:00
|
|
|
let kwargs = unsafe { kwargs.as_ref() }.ok_or(Error::MissingKwargs)?;
|
2021-03-23 12:14:33 +00:00
|
|
|
|
|
|
|
func(args, kwargs)
|
2021-09-16 11:48:26 +00:00
|
|
|
};
|
|
|
|
unsafe { try_or_raise(block) }
|
2021-03-23 12:14:33 +00:00
|
|
|
}
|
|
|
|
|
2021-09-16 11:48:26 +00:00
|
|
|
/// Extract args and kwargs from a C call where args and kwargs are inlined, and
|
2021-10-06 21:21:47 +00:00
|
|
|
/// pass them into Rust. Raise exception if an error occurs. Should only called
|
|
|
|
/// when returning from Rust to C. See `raise_exception` for details.
|
2021-09-16 11:48:26 +00:00
|
|
|
pub unsafe fn try_with_args_and_kwargs_inline(
|
2021-03-23 12:14:33 +00:00
|
|
|
n_args: usize,
|
|
|
|
n_kw: usize,
|
|
|
|
args: *const Obj,
|
|
|
|
func: impl FnOnce(&[Obj], &Map) -> Result<Obj, Error>,
|
|
|
|
) -> Obj {
|
2021-09-16 11:48:26 +00:00
|
|
|
let block = || {
|
2021-03-23 12:14:33 +00:00
|
|
|
let args_slice: &[Obj];
|
|
|
|
let kwargs_slice: &[MapElem];
|
|
|
|
|
|
|
|
if args.is_null() {
|
|
|
|
args_slice = &[];
|
|
|
|
kwargs_slice = &[];
|
|
|
|
} else {
|
|
|
|
args_slice = unsafe { slice::from_raw_parts(args, n_args) };
|
|
|
|
kwargs_slice = unsafe { slice::from_raw_parts(args.add(n_args).cast(), n_kw) };
|
|
|
|
}
|
|
|
|
|
|
|
|
let kw_map = Map::from_fixed(kwargs_slice);
|
|
|
|
func(args_slice, &kw_map)
|
2021-09-16 11:48:26 +00:00
|
|
|
};
|
|
|
|
unsafe { try_or_raise(block) }
|
2021-03-23 12:14:33 +00:00
|
|
|
}
|