2018-09-12 00:19:09 +09:00
|
|
|
#![feature(unsized_locals)]
|
|
|
|
#![feature(unboxed_closures)]
|
2022-10-02 19:41:54 +00:00
|
|
|
#![feature(tuple_trait)]
|
2018-09-12 00:19:09 +09:00
|
|
|
|
2022-10-02 19:41:54 +00:00
|
|
|
pub trait FnOnce<Args: std::marker::Tuple> {
|
2018-09-12 00:19:09 +09:00
|
|
|
type Output;
|
|
|
|
extern "rust-call" fn call_once(self, args: Args) -> Self::Output;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct A;
|
|
|
|
|
|
|
|
impl FnOnce<()> for A {
|
|
|
|
type Output = String;
|
|
|
|
extern "rust-call" fn call_once(self, (): ()) -> Self::Output {
|
|
|
|
format!("hello")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct B(i32);
|
|
|
|
|
|
|
|
impl FnOnce<()> for B {
|
|
|
|
type Output = String;
|
|
|
|
extern "rust-call" fn call_once(self, (): ()) -> Self::Output {
|
|
|
|
format!("{}", self.0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct C(String);
|
|
|
|
|
|
|
|
impl FnOnce<()> for C {
|
|
|
|
type Output = String;
|
|
|
|
extern "rust-call" fn call_once(self, (): ()) -> Self::Output {
|
|
|
|
self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct D(Box<String>);
|
|
|
|
|
|
|
|
impl FnOnce<()> for D {
|
|
|
|
type Output = String;
|
|
|
|
extern "rust-call" fn call_once(self, (): ()) -> Self::Output {
|
|
|
|
*self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let x = *(Box::new(A) as Box<dyn FnOnce<(), Output = String>>);
|
|
|
|
assert_eq!(x.call_once(()), format!("hello"));
|
|
|
|
let x = *(Box::new(B(42)) as Box<dyn FnOnce<(), Output = String>>);
|
|
|
|
assert_eq!(x.call_once(()), format!("42"));
|
|
|
|
let x = *(Box::new(C(format!("jumping fox"))) as Box<dyn FnOnce<(), Output = String>>);
|
|
|
|
assert_eq!(x.call_once(()), format!("jumping fox"));
|
|
|
|
let x = *(Box::new(D(Box::new(format!("lazy dog")))) as Box<dyn FnOnce<(), Output = String>>);
|
|
|
|
assert_eq!(x.call_once(()), format!("lazy dog"));
|
|
|
|
}
|