#![feature(unsized_locals)]
#![feature(unboxed_closures)]
#![feature(tuple_trait)]

pub trait FnOnce<Args: std::marker::Tuple> {
    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"));
}