1e05fbacdSPaolo Bonzini // SPDX-License-Identifier: MIT 2e05fbacdSPaolo Bonzini 3e05fbacdSPaolo Bonzini //! Utility functions to deal with callbacks from C to Rust. 4e05fbacdSPaolo Bonzini 5e05fbacdSPaolo Bonzini use std::{mem, ptr::NonNull}; 6e05fbacdSPaolo Bonzini 7e05fbacdSPaolo Bonzini /// Trait for functions (types implementing [`Fn`]) that can be used as 8e05fbacdSPaolo Bonzini /// callbacks. These include both zero-capture closures and function pointers. 9e05fbacdSPaolo Bonzini /// 10e05fbacdSPaolo Bonzini /// In Rust, calling a function through the `Fn` trait normally requires a 11e05fbacdSPaolo Bonzini /// `self` parameter, even though for zero-sized functions (including function 12e05fbacdSPaolo Bonzini /// pointers) the type itself contains all necessary information to call the 13e05fbacdSPaolo Bonzini /// function. This trait provides a `call` function that doesn't require `self`, 14e05fbacdSPaolo Bonzini /// allowing zero-sized functions to be called using only their type. 15e05fbacdSPaolo Bonzini /// 16e05fbacdSPaolo Bonzini /// This enables zero-sized functions to be passed entirely through generic 17e05fbacdSPaolo Bonzini /// parameters and resolved at compile-time. A typical use is a function 18e05fbacdSPaolo Bonzini /// receiving an unused parameter of generic type `F` and calling it via 19e05fbacdSPaolo Bonzini /// `F::call` or passing it to another function via `func::<F>`. 20e05fbacdSPaolo Bonzini /// 21e05fbacdSPaolo Bonzini /// QEMU uses this trick to create wrappers to C callbacks. The wrappers 22e05fbacdSPaolo Bonzini /// are needed to convert an opaque `*mut c_void` into a Rust reference, 23e05fbacdSPaolo Bonzini /// but they only have a single opaque that they can use. The `FnCall` 24e05fbacdSPaolo Bonzini /// trait makes it possible to use that opaque for `self` or any other 25e05fbacdSPaolo Bonzini /// reference: 26e05fbacdSPaolo Bonzini /// 27e05fbacdSPaolo Bonzini /// ```ignore 28e05fbacdSPaolo Bonzini /// // The compiler creates a new `rust_bh_cb` wrapper for each function 29e05fbacdSPaolo Bonzini /// // passed to `qemu_bh_schedule_oneshot` below. 30e05fbacdSPaolo Bonzini /// unsafe extern "C" fn rust_bh_cb<T, F: for<'a> FnCall<(&'a T,)>>( 31e05fbacdSPaolo Bonzini /// opaque: *mut c_void, 32e05fbacdSPaolo Bonzini /// ) { 33e05fbacdSPaolo Bonzini /// // SAFETY: the opaque was passed as a reference to `T`. 34e05fbacdSPaolo Bonzini /// F::call((unsafe { &*(opaque.cast::<T>()) }, )) 35e05fbacdSPaolo Bonzini /// } 36e05fbacdSPaolo Bonzini /// 37e05fbacdSPaolo Bonzini /// // The `_f` parameter is unused but it helps the compiler build the appropriate `F`. 38e05fbacdSPaolo Bonzini /// // Using a reference allows usage in const context. 39e05fbacdSPaolo Bonzini /// fn qemu_bh_schedule_oneshot<T, F: for<'a> FnCall<(&'a T,)>>(_f: &F, opaque: &T) { 40e05fbacdSPaolo Bonzini /// let cb: unsafe extern "C" fn(*mut c_void) = rust_bh_cb::<T, F>; 41e05fbacdSPaolo Bonzini /// unsafe { 42e05fbacdSPaolo Bonzini /// bindings::qemu_bh_schedule_oneshot(cb, opaque as *const T as *const c_void as *mut c_void) 43e05fbacdSPaolo Bonzini /// } 44e05fbacdSPaolo Bonzini /// } 45e05fbacdSPaolo Bonzini /// ``` 46e05fbacdSPaolo Bonzini /// 47e05fbacdSPaolo Bonzini /// Each wrapper is a separate instance of `rust_bh_cb` and is therefore 48e05fbacdSPaolo Bonzini /// compiled to a separate function ("monomorphization"). If you wanted 49e05fbacdSPaolo Bonzini /// to pass `self` as the opaque value, the generic parameters would be 50e05fbacdSPaolo Bonzini /// `rust_bh_cb::<Self, F>`. 51e05fbacdSPaolo Bonzini /// 52e05fbacdSPaolo Bonzini /// `Args` is a tuple type whose types are the arguments of the function, 53e05fbacdSPaolo Bonzini /// while `R` is the returned type. 54e05fbacdSPaolo Bonzini /// 55e05fbacdSPaolo Bonzini /// # Examples 56e05fbacdSPaolo Bonzini /// 57e05fbacdSPaolo Bonzini /// ``` 58e05fbacdSPaolo Bonzini /// # use qemu_api::callbacks::FnCall; 59e05fbacdSPaolo Bonzini /// fn call_it<F: for<'a> FnCall<(&'a str,), String>>(_f: &F, s: &str) -> String { 60e05fbacdSPaolo Bonzini /// F::call((s,)) 61e05fbacdSPaolo Bonzini /// } 62e05fbacdSPaolo Bonzini /// 63e05fbacdSPaolo Bonzini /// let s: String = call_it(&str::to_owned, "hello world"); 64e05fbacdSPaolo Bonzini /// assert_eq!(s, "hello world"); 65e05fbacdSPaolo Bonzini /// ``` 66e05fbacdSPaolo Bonzini /// 67e05fbacdSPaolo Bonzini /// Note that the compiler will produce a different version of `call_it` for 68e05fbacdSPaolo Bonzini /// each function that is passed to it. Therefore the argument is not really 69e05fbacdSPaolo Bonzini /// used, except to decide what is `F` and what `F::call` does. 70e05fbacdSPaolo Bonzini /// 71e05fbacdSPaolo Bonzini /// Attempting to pass a non-zero-sized closure causes a compile-time failure: 72e05fbacdSPaolo Bonzini /// 73e05fbacdSPaolo Bonzini /// ```compile_fail 74e05fbacdSPaolo Bonzini /// # use qemu_api::callbacks::FnCall; 75e05fbacdSPaolo Bonzini /// # fn call_it<'a, F: FnCall<(&'a str,), String>>(_f: &F, s: &'a str) -> String { 76e05fbacdSPaolo Bonzini /// # F::call((s,)) 77e05fbacdSPaolo Bonzini /// # } 78e05fbacdSPaolo Bonzini /// let x: &'static str = "goodbye world"; 79e05fbacdSPaolo Bonzini /// call_it(&move |_| String::from(x), "hello workd"); 80e05fbacdSPaolo Bonzini /// ``` 81e05fbacdSPaolo Bonzini /// 82*66bcc554SPaolo Bonzini /// `()` can be used to indicate "no function": 83*66bcc554SPaolo Bonzini /// 84*66bcc554SPaolo Bonzini /// ``` 85*66bcc554SPaolo Bonzini /// # use qemu_api::callbacks::FnCall; 86*66bcc554SPaolo Bonzini /// fn optional<F: for<'a> FnCall<(&'a str,), String>>(_f: &F, s: &str) -> Option<String> { 87*66bcc554SPaolo Bonzini /// if F::IS_SOME { 88*66bcc554SPaolo Bonzini /// Some(F::call((s,))) 89*66bcc554SPaolo Bonzini /// } else { 90*66bcc554SPaolo Bonzini /// None 91*66bcc554SPaolo Bonzini /// } 92*66bcc554SPaolo Bonzini /// } 93*66bcc554SPaolo Bonzini /// 94*66bcc554SPaolo Bonzini /// assert!(optional(&(), "hello world").is_none()); 95*66bcc554SPaolo Bonzini /// ``` 96*66bcc554SPaolo Bonzini /// 97*66bcc554SPaolo Bonzini /// Invoking `F::call` will then be a run-time error. 98*66bcc554SPaolo Bonzini /// 99*66bcc554SPaolo Bonzini /// ```should_panic 100*66bcc554SPaolo Bonzini /// # use qemu_api::callbacks::FnCall; 101*66bcc554SPaolo Bonzini /// # fn call_it<F: for<'a> FnCall<(&'a str,), String>>(_f: &F, s: &str) -> String { 102*66bcc554SPaolo Bonzini /// # F::call((s,)) 103*66bcc554SPaolo Bonzini /// # } 104*66bcc554SPaolo Bonzini /// let s: String = call_it(&(), "hello world"); // panics 105*66bcc554SPaolo Bonzini /// ``` 106*66bcc554SPaolo Bonzini /// 107e05fbacdSPaolo Bonzini /// # Safety 108e05fbacdSPaolo Bonzini /// 109e05fbacdSPaolo Bonzini /// Because `Self` is a zero-sized type, all instances of the type are 110e05fbacdSPaolo Bonzini /// equivalent. However, in addition to this, `Self` must have no invariants 111e05fbacdSPaolo Bonzini /// that could be violated by creating a reference to it. 112e05fbacdSPaolo Bonzini /// 113e05fbacdSPaolo Bonzini /// This is always true for zero-capture closures and function pointers, as long 114e05fbacdSPaolo Bonzini /// as the code is able to name the function in the first place. 115e05fbacdSPaolo Bonzini pub unsafe trait FnCall<Args, R = ()>: 'static + Sync + Sized { 116e05fbacdSPaolo Bonzini /// Referring to this internal constant asserts that the `Self` type is 117e05fbacdSPaolo Bonzini /// zero-sized. Can be replaced by an inline const expression in 118e05fbacdSPaolo Bonzini /// Rust 1.79.0+. 119e05fbacdSPaolo Bonzini const ASSERT_ZERO_SIZED: () = { assert!(mem::size_of::<Self>() == 0) }; 120e05fbacdSPaolo Bonzini 121*66bcc554SPaolo Bonzini /// Referring to this constant asserts that the `Self` type is an actual 122*66bcc554SPaolo Bonzini /// function type, which can be used to catch incorrect use of `()` 123*66bcc554SPaolo Bonzini /// at compile time. 124*66bcc554SPaolo Bonzini /// 125*66bcc554SPaolo Bonzini /// # Examples 126*66bcc554SPaolo Bonzini /// 127*66bcc554SPaolo Bonzini /// ```compile_fail 128*66bcc554SPaolo Bonzini /// # use qemu_api::callbacks::FnCall; 129*66bcc554SPaolo Bonzini /// fn call_it<F: for<'a> FnCall<(&'a str,), String>>(_f: &F, s: &str) -> String { 130*66bcc554SPaolo Bonzini /// let _: () = F::ASSERT_IS_SOME; 131*66bcc554SPaolo Bonzini /// F::call((s,)) 132*66bcc554SPaolo Bonzini /// } 133*66bcc554SPaolo Bonzini /// 134*66bcc554SPaolo Bonzini /// let s: String = call_it((), "hello world"); // does not compile 135*66bcc554SPaolo Bonzini /// ``` 136*66bcc554SPaolo Bonzini /// 137*66bcc554SPaolo Bonzini /// Note that this can be more simply `const { assert!(F::IS_SOME) }` in 138*66bcc554SPaolo Bonzini /// Rust 1.79.0 or newer. 139*66bcc554SPaolo Bonzini const ASSERT_IS_SOME: () = { assert!(Self::IS_SOME) }; 140*66bcc554SPaolo Bonzini 141*66bcc554SPaolo Bonzini /// `true` if `Self` is an actual function type and not `()`. 142*66bcc554SPaolo Bonzini /// 143*66bcc554SPaolo Bonzini /// # Examples 144*66bcc554SPaolo Bonzini /// 145*66bcc554SPaolo Bonzini /// You can use `IS_SOME` to catch this at compile time: 146*66bcc554SPaolo Bonzini /// 147*66bcc554SPaolo Bonzini /// ```compile_fail 148*66bcc554SPaolo Bonzini /// # use qemu_api::callbacks::FnCall; 149*66bcc554SPaolo Bonzini /// fn call_it<F: for<'a> FnCall<(&'a str,), String>>(_f: &F, s: &str) -> String { 150*66bcc554SPaolo Bonzini /// const { assert!(F::IS_SOME) } 151*66bcc554SPaolo Bonzini /// F::call((s,)) 152*66bcc554SPaolo Bonzini /// } 153*66bcc554SPaolo Bonzini /// 154*66bcc554SPaolo Bonzini /// let s: String = call_it((), "hello world"); // does not compile 155*66bcc554SPaolo Bonzini /// ``` 156*66bcc554SPaolo Bonzini const IS_SOME: bool; 157*66bcc554SPaolo Bonzini 158*66bcc554SPaolo Bonzini /// `false` if `Self` is an actual function type, `true` if it is `()`. is_none() -> bool159*66bcc554SPaolo Bonzini fn is_none() -> bool { 160*66bcc554SPaolo Bonzini !Self::IS_SOME 161*66bcc554SPaolo Bonzini } 162*66bcc554SPaolo Bonzini 163*66bcc554SPaolo Bonzini /// `true` if `Self` is an actual function type, `false` if it is `()`. is_some() -> bool164*66bcc554SPaolo Bonzini fn is_some() -> bool { 165*66bcc554SPaolo Bonzini Self::IS_SOME 166*66bcc554SPaolo Bonzini } 167*66bcc554SPaolo Bonzini 168e05fbacdSPaolo Bonzini /// Call the function with the arguments in args. call(a: Args) -> R169e05fbacdSPaolo Bonzini fn call(a: Args) -> R; 170e05fbacdSPaolo Bonzini } 171e05fbacdSPaolo Bonzini 172*66bcc554SPaolo Bonzini /// `()` acts as a "null" callback. Using `()` and `function` is nicer 173*66bcc554SPaolo Bonzini /// than `None` and `Some(function)`, because the compiler is unable to 174*66bcc554SPaolo Bonzini /// infer the type of just `None`. Therefore, the trait itself acts as the 175*66bcc554SPaolo Bonzini /// option type, with functions [`FnCall::is_some`] and [`FnCall::is_none`]. 176*66bcc554SPaolo Bonzini unsafe impl<Args, R> FnCall<Args, R> for () { 177*66bcc554SPaolo Bonzini const IS_SOME: bool = false; 178*66bcc554SPaolo Bonzini 179*66bcc554SPaolo Bonzini /// Call the function with the arguments in args. call(_a: Args) -> R180*66bcc554SPaolo Bonzini fn call(_a: Args) -> R { 181*66bcc554SPaolo Bonzini panic!("callback not specified") 182*66bcc554SPaolo Bonzini } 183*66bcc554SPaolo Bonzini } 184*66bcc554SPaolo Bonzini 185e05fbacdSPaolo Bonzini macro_rules! impl_call { 186e05fbacdSPaolo Bonzini ($($args:ident,)* ) => ( 187e05fbacdSPaolo Bonzini // SAFETY: because each function is treated as a separate type, 188e05fbacdSPaolo Bonzini // accessing `FnCall` is only possible in code that would be 189e05fbacdSPaolo Bonzini // allowed to call the function. 190e05fbacdSPaolo Bonzini unsafe impl<F, $($args,)* R> FnCall<($($args,)*), R> for F 191e05fbacdSPaolo Bonzini where 192e05fbacdSPaolo Bonzini F: 'static + Sync + Sized + Fn($($args, )*) -> R, 193e05fbacdSPaolo Bonzini { 194*66bcc554SPaolo Bonzini const IS_SOME: bool = true; 195*66bcc554SPaolo Bonzini 196e05fbacdSPaolo Bonzini #[inline(always)] 197e05fbacdSPaolo Bonzini fn call(a: ($($args,)*)) -> R { 198e05fbacdSPaolo Bonzini let _: () = Self::ASSERT_ZERO_SIZED; 199e05fbacdSPaolo Bonzini 200e05fbacdSPaolo Bonzini // SAFETY: the safety of this method is the condition for implementing 201e05fbacdSPaolo Bonzini // `FnCall`. As to the `NonNull` idiom to create a zero-sized type, 202e05fbacdSPaolo Bonzini // see https://github.com/rust-lang/libs-team/issues/292. 203e05fbacdSPaolo Bonzini let f: &'static F = unsafe { &*NonNull::<Self>::dangling().as_ptr() }; 204e05fbacdSPaolo Bonzini let ($($args,)*) = a; 205e05fbacdSPaolo Bonzini f($($args,)*) 206e05fbacdSPaolo Bonzini } 207e05fbacdSPaolo Bonzini } 208e05fbacdSPaolo Bonzini ) 209e05fbacdSPaolo Bonzini } 210e05fbacdSPaolo Bonzini 211e05fbacdSPaolo Bonzini impl_call!(_1, _2, _3, _4, _5,); 212e05fbacdSPaolo Bonzini impl_call!(_1, _2, _3, _4,); 213e05fbacdSPaolo Bonzini impl_call!(_1, _2, _3,); 214e05fbacdSPaolo Bonzini impl_call!(_1, _2,); 215e05fbacdSPaolo Bonzini impl_call!(_1,); 216e05fbacdSPaolo Bonzini impl_call!(); 217e05fbacdSPaolo Bonzini 218e05fbacdSPaolo Bonzini #[cfg(test)] 219e05fbacdSPaolo Bonzini mod tests { 220e05fbacdSPaolo Bonzini use super::*; 221e05fbacdSPaolo Bonzini 222e05fbacdSPaolo Bonzini // The `_f` parameter is unused but it helps the compiler infer `F`. do_test_call<'a, F: FnCall<(&'a str,), String>>(_f: &F) -> String223e05fbacdSPaolo Bonzini fn do_test_call<'a, F: FnCall<(&'a str,), String>>(_f: &F) -> String { 224e05fbacdSPaolo Bonzini F::call(("hello world",)) 225e05fbacdSPaolo Bonzini } 226e05fbacdSPaolo Bonzini 227e05fbacdSPaolo Bonzini #[test] test_call()228e05fbacdSPaolo Bonzini fn test_call() { 229e05fbacdSPaolo Bonzini assert_eq!(do_test_call(&str::to_owned), "hello world") 230e05fbacdSPaolo Bonzini } 231*66bcc554SPaolo Bonzini 232*66bcc554SPaolo Bonzini // The `_f` parameter is unused but it helps the compiler infer `F`. do_test_is_some<'a, F: FnCall<(&'a str,), String>>(_f: &F)233*66bcc554SPaolo Bonzini fn do_test_is_some<'a, F: FnCall<(&'a str,), String>>(_f: &F) { 234*66bcc554SPaolo Bonzini assert!(F::is_some()); 235*66bcc554SPaolo Bonzini } 236*66bcc554SPaolo Bonzini 237*66bcc554SPaolo Bonzini #[test] test_is_some()238*66bcc554SPaolo Bonzini fn test_is_some() { 239*66bcc554SPaolo Bonzini do_test_is_some(&str::to_owned); 240*66bcc554SPaolo Bonzini } 241e05fbacdSPaolo Bonzini } 242