use std::future::Future;
use std::ptr;
use std::sync::atomic::AtomicI32;
use std::sync::atomic::Ordering;
use std::sync::Arc;
use std::task::Context;
use std::task::Poll;
use futures::pin_mut;
use futures::task::waker_ref;
use futures::task::ArcWake;
const WAITING: i32 = 0x25de_74d1;
const WOKEN: i32 = 0x72d3_2c9f;
const FUTEX_WAIT_PRIVATE: libc::c_int = libc::FUTEX_WAIT | libc::FUTEX_PRIVATE_FLAG;
const FUTEX_WAKE_PRIVATE: libc::c_int = libc::FUTEX_WAKE | libc::FUTEX_PRIVATE_FLAG;
thread_local!(static PER_THREAD_WAKER: Arc<Waker> = Arc::new(Waker(AtomicI32::new(WAITING))));
#[repr(transparent)]
struct Waker(AtomicI32);
impl ArcWake for Waker {
    fn wake_by_ref(arc_self: &Arc<Self>) {
        let state = arc_self.0.swap(WOKEN, Ordering::Release);
        if state == WAITING {
            let res = unsafe {
                libc::syscall(
                    libc::SYS_futex,
                    &arc_self.0,
                    FUTEX_WAKE_PRIVATE,
                    libc::INT_MAX,                        ptr::null::<*const libc::timespec>(), ptr::null::<*const libc::c_int>(),    0_i32,                                )
            };
            if res < 0 {
                panic!(
                    "unexpected error from FUTEX_WAKE_PRIVATE: {}",
                    std::io::Error::last_os_error()
                );
            }
        }
    }
}
pub fn block_on<F: Future>(f: F) -> F::Output {
    pin_mut!(f);
    PER_THREAD_WAKER.with(|thread_waker| {
        let waker = waker_ref(thread_waker);
        let mut cx = Context::from_waker(&waker);
        loop {
            if let Poll::Ready(t) = f.as_mut().poll(&mut cx) {
                return t;
            }
            let state = thread_waker.0.swap(WAITING, Ordering::Acquire);
            if state == WAITING {
                let res = unsafe {
                    libc::syscall(
                        libc::SYS_futex,
                        &thread_waker.0,
                        FUTEX_WAIT_PRIVATE,
                        state,
                        ptr::null::<*const libc::timespec>(), ptr::null::<*const libc::c_int>(),    0_i32,                                )
                };
                if res < 0 {
                    let e = std::io::Error::last_os_error();
                    match e.raw_os_error() {
                        Some(libc::EAGAIN) | Some(libc::EINTR) => {}
                        _ => panic!("unexpected error from FUTEX_WAIT_PRIVATE: {}", e),
                    }
                }
                thread_waker.0.store(WAITING, Ordering::Release);
            }
        }
    })
}
#[cfg(test)]
mod test {
    use std::future::Future;
    use std::pin::Pin;
    use std::sync::mpsc::channel;
    use std::sync::mpsc::Sender;
    use std::sync::Arc;
    use std::task::Context;
    use std::task::Poll;
    use std::task::Waker;
    use std::thread;
    use std::time::Duration;
    use super::*;
    use crate::sync::SpinLock;
    struct TimerState {
        fired: bool,
        waker: Option<Waker>,
    }
    struct Timer {
        state: Arc<SpinLock<TimerState>>,
    }
    impl Future for Timer {
        type Output = ();
        fn poll(self: Pin<&mut Self>, cx: &mut Context) -> Poll<Self::Output> {
            let mut state = self.state.lock();
            if state.fired {
                return Poll::Ready(());
            }
            state.waker = Some(cx.waker().clone());
            Poll::Pending
        }
    }
    fn start_timer(dur: Duration, notify: Option<Sender<()>>) -> Timer {
        let state = Arc::new(SpinLock::new(TimerState {
            fired: false,
            waker: None,
        }));
        let thread_state = Arc::clone(&state);
        thread::spawn(move || {
            thread::sleep(dur);
            let mut ts = thread_state.lock();
            ts.fired = true;
            if let Some(waker) = ts.waker.take() {
                waker.wake();
            }
            drop(ts);
            if let Some(tx) = notify {
                tx.send(()).expect("Failed to send completion notification");
            }
        });
        Timer { state }
    }
    #[test]
    fn it_works() {
        block_on(start_timer(Duration::from_millis(100), None));
    }
    #[test]
    fn nested() {
        async fn inner() {
            block_on(start_timer(Duration::from_millis(100), None));
        }
        block_on(inner());
    }
    #[test]
    fn ready_before_poll() {
        let (tx, rx) = channel();
        let timer = start_timer(Duration::from_millis(50), Some(tx));
        rx.recv()
            .expect("Failed to receive completion notification");
        block_on(timer);
    }
}