use super::rwhandle::{RWHandle, ReadableHandle as _, WritableHandle as _};
use fuchsia_zircon::{self as zx, AsHandleRef};
use futures::ready;
use std::fmt;
use std::future::Future;
use std::marker::PhantomData;
use std::mem::MaybeUninit;
use std::pin::Pin;
use std::task::{Context, Poll};
use zerocopy::{AsBytes, FromBytes, NoCell};
pub trait FifoEntry: AsBytes + FromBytes + NoCell {}
impl<O: AsBytes + FromBytes + NoCell> FifoEntry for O {}
pub unsafe trait FifoWriteBuffer<T> {
fn count(&self) -> usize;
fn as_bytes_ptr(&self) -> *const u8;
}
pub unsafe trait FifoReadBuffer<T> {
fn count(&self) -> usize;
fn as_bytes_ptr_mut(&mut self) -> *mut u8;
}
unsafe impl<T: FifoEntry> FifoWriteBuffer<T> for [T] {
fn count(&self) -> usize {
self.len()
}
fn as_bytes_ptr(&self) -> *const u8 {
self.as_bytes().as_ptr()
}
}
unsafe impl<T: FifoEntry> FifoReadBuffer<T> for [T] {
fn count(&self) -> usize {
self.len()
}
fn as_bytes_ptr_mut(&mut self) -> *mut u8 {
self.as_bytes_mut().as_mut_ptr()
}
}
unsafe impl<T: FifoEntry> FifoWriteBuffer<T> for T {
fn count(&self) -> usize {
1
}
fn as_bytes_ptr(&self) -> *const u8 {
self.as_bytes().as_ptr()
}
}
unsafe impl<T: FifoEntry> FifoReadBuffer<T> for T {
fn count(&self) -> usize {
1
}
fn as_bytes_ptr_mut(&mut self) -> *mut u8 {
self.as_bytes_mut().as_mut_ptr()
}
}
unsafe impl<T: FifoEntry> FifoReadBuffer<T> for MaybeUninit<T> {
fn count(&self) -> usize {
1
}
fn as_bytes_ptr_mut(&mut self) -> *mut u8 {
self.as_mut_ptr() as _
}
}
unsafe impl<T: FifoEntry> FifoReadBuffer<T> for [MaybeUninit<T>] {
fn count(&self) -> usize {
self.len()
}
fn as_bytes_ptr_mut(&mut self) -> *mut u8 {
self.as_mut_ptr() as _
}
}
struct OffsetWriteBuffer<'a, B: ?Sized, T> {
buffer: &'a B,
offset: usize,
marker: PhantomData<T>,
}
impl<'a, B: ?Sized + FifoWriteBuffer<T>, T: FifoEntry> OffsetWriteBuffer<'a, B, T> {
fn new(buffer: &'a B) -> Self {
Self { buffer, offset: 0, marker: PhantomData }
}
fn advance(mut self, len: usize) -> Option<Self> {
self.offset += len;
if self.offset == self.buffer.count() {
None
} else {
debug_assert!(self.offset < self.buffer.count());
Some(self)
}
}
}
unsafe impl<'a, T: FifoEntry, B: ?Sized + FifoWriteBuffer<T>> FifoWriteBuffer<T>
for OffsetWriteBuffer<'a, B, T>
{
fn count(&self) -> usize {
debug_assert!(self.offset <= self.buffer.count());
self.buffer.count() - self.offset
}
fn as_bytes_ptr(&self) -> *const u8 {
debug_assert!(self.offset <= self.buffer.count());
let buffer_bytes = self.buffer.as_bytes_ptr();
let count = self.offset * std::mem::size_of::<T>();
unsafe { buffer_bytes.add(count) }
}
}
pub trait FifoWritable<W: FifoEntry>
where
Self: Sized,
{
fn write_entries<'a, B: ?Sized + FifoWriteBuffer<W>>(
&'a self,
entries: &'a B,
) -> WriteEntries<'a, Self, B, W> {
WriteEntries::new(self, entries)
}
fn write<B: ?Sized + FifoWriteBuffer<W>>(
&self,
cx: &mut Context<'_>,
entries: &B,
) -> Poll<Result<usize, zx::Status>>;
}
pub trait FifoReadable<R: FifoEntry>
where
Self: Sized,
{
fn read_entries<'a, B: ?Sized + FifoReadBuffer<R>>(
&'a self,
entries: &'a mut B,
) -> ReadEntries<'a, Self, B, R> {
ReadEntries::new(self, entries)
}
fn read_entry<'a>(&'a self) -> ReadOne<'a, Self, R> {
ReadOne::new(self)
}
fn read<B: ?Sized + FifoReadBuffer<R>>(
&self,
cx: &mut Context<'_>,
entries: &mut B,
) -> Poll<Result<usize, zx::Status>>;
fn read_one(&self, cx: &mut Context<'_>) -> Poll<Result<R, zx::Status>> {
let mut entry = MaybeUninit::uninit();
self.read(cx, &mut entry).map_ok(|count| {
debug_assert_eq!(count, 1);
unsafe { entry.assume_init() }
})
}
}
pub struct Fifo<R, W = R> {
handle: RWHandle<zx::Fifo>,
read_marker: PhantomData<R>,
write_marker: PhantomData<W>,
}
impl<R, W> AsRef<zx::Fifo> for Fifo<R, W> {
fn as_ref(&self) -> &zx::Fifo {
self.handle.get_ref()
}
}
impl<R, W> AsHandleRef for Fifo<R, W> {
fn as_handle_ref(&self) -> zx::HandleRef<'_> {
self.handle.get_ref().as_handle_ref()
}
}
impl<R, W> From<Fifo<R, W>> for zx::Fifo {
fn from(fifo: Fifo<R, W>) -> zx::Fifo {
fifo.handle.into_inner()
}
}
impl<R: FifoEntry, W: FifoEntry> Fifo<R, W> {
pub fn from_fifo(fifo: zx::Fifo) -> Self {
Fifo { handle: RWHandle::new(fifo), read_marker: PhantomData, write_marker: PhantomData }
}
pub fn try_write<B: ?Sized + FifoWriteBuffer<W>>(
&self,
cx: &mut Context<'_>,
entries: &B,
) -> Poll<Result<usize, zx::Status>> {
ready!(self.handle.poll_writable(cx)?);
let elem_size = std::mem::size_of::<W>();
let bytes = entries.as_bytes_ptr();
let count = entries.count();
let fifo = self.as_ref();
loop {
let result = unsafe { fifo.write_ptr(elem_size, bytes, count) };
match result {
Err(zx::Status::SHOULD_WAIT) => ready!(self.handle.need_writable(cx)?),
Err(e) => return Poll::Ready(Err(e)),
Ok(count) => return Poll::Ready(Ok(count)),
}
}
}
pub fn try_read<B: ?Sized + FifoReadBuffer<R>>(
&self,
cx: &mut Context<'_>,
entries: &mut B,
) -> Poll<Result<usize, zx::Status>> {
ready!(self.handle.poll_readable(cx)?);
let elem_size = std::mem::size_of::<R>();
let bytes = entries.as_bytes_ptr_mut();
let count = entries.count();
let fifo = self.as_ref();
loop {
let result = unsafe { fifo.read_ptr(elem_size, bytes, count) };
match result {
Err(zx::Status::SHOULD_WAIT) => ready!(self.handle.need_readable(cx)?),
Err(e) => return Poll::Ready(Err(e)),
Ok(count) => return Poll::Ready(Ok(count)),
}
}
}
}
impl<R: FifoEntry, W: FifoEntry> FifoReadable<R> for Fifo<R, W> {
fn read<B: ?Sized + FifoReadBuffer<R>>(
&self,
cx: &mut Context<'_>,
entries: &mut B,
) -> Poll<Result<usize, zx::Status>> {
self.try_read(cx, entries)
}
}
impl<R: FifoEntry, W: FifoEntry> FifoWritable<W> for Fifo<R, W> {
fn write<B: ?Sized + FifoWriteBuffer<W>>(
&self,
cx: &mut Context<'_>,
entries: &B,
) -> Poll<Result<usize, zx::Status>> {
self.try_write(cx, entries)
}
}
impl<R, W> fmt::Debug for Fifo<R, W> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
self.handle.get_ref().fmt(f)
}
}
pub struct WriteEntries<'a, F, B: ?Sized, T> {
fifo: &'a F,
entries: Option<OffsetWriteBuffer<'a, B, T>>,
marker: PhantomData<T>,
}
impl<'a, F, B: ?Sized, T> Unpin for WriteEntries<'a, F, B, T> {}
impl<'a, T: FifoEntry, F: FifoWritable<T>, B: ?Sized + FifoWriteBuffer<T>>
WriteEntries<'a, F, B, T>
{
pub fn new(fifo: &'a F, entries: &'a B) -> Self {
WriteEntries { fifo, entries: Some(OffsetWriteBuffer::new(entries)), marker: PhantomData }
}
}
impl<'a, T: FifoEntry, F: FifoWritable<T>, B: ?Sized + FifoWriteBuffer<T>> Future
for WriteEntries<'a, F, B, T>
{
type Output = Result<(), zx::Status>;
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
let this = &mut *self;
while let Some(entries) = this.entries.as_ref() {
let advance = ready!(this.fifo.write(cx, entries)?);
this.entries = this.entries.take().unwrap().advance(advance);
}
Poll::Ready(Ok(()))
}
}
pub struct ReadEntries<'a, F, B: ?Sized, T> {
fifo: &'a F,
entries: &'a mut B,
marker: PhantomData<T>,
}
impl<'a, F, B: ?Sized, T> Unpin for ReadEntries<'a, F, B, T> {}
impl<'a, T: FifoEntry, F: FifoReadable<T>, B: ?Sized + FifoReadBuffer<T>> ReadEntries<'a, F, B, T> {
pub fn new(fifo: &'a F, entries: &'a mut B) -> Self {
ReadEntries { fifo, entries, marker: PhantomData }
}
}
impl<'a, T: FifoEntry, F: FifoReadable<T>, B: ?Sized + FifoReadBuffer<T>> Future
for ReadEntries<'a, F, B, T>
{
type Output = Result<usize, zx::Status>;
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
let this = &mut *self;
this.fifo.read(cx, this.entries)
}
}
pub struct ReadOne<'a, F, T> {
fifo: &'a F,
marker: PhantomData<T>,
}
impl<'a, F, T> Unpin for ReadOne<'a, F, T> {}
impl<'a, T: FifoEntry, F: FifoReadable<T>> ReadOne<'a, F, T> {
pub fn new(fifo: &'a F) -> Self {
ReadOne { fifo, marker: PhantomData }
}
}
impl<'a, T: FifoEntry, F: FifoReadable<T>> Future for ReadOne<'a, F, T> {
type Output = Result<T, zx::Status>;
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
let this = &mut *self;
this.fifo.read_one(cx)
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{DurationExt, TestExecutor, TimeoutExt, Timer};
use fuchsia_zircon::prelude::*;
use futures::future::try_join;
use futures::prelude::*;
use zerocopy::{FromZeros, NoCell};
#[derive(Copy, Clone, Debug, PartialEq, Eq, Default, AsBytes, FromZeros, FromBytes, NoCell)]
#[repr(C)]
struct entry {
a: u32,
b: u32,
}
#[derive(Clone, Debug, PartialEq, Eq, Default, AsBytes, FromZeros, FromBytes, NoCell)]
#[repr(C)]
struct wrong_entry {
a: u16,
}
#[test]
fn can_read_write() {
let mut exec = TestExecutor::new();
let element = entry { a: 10, b: 20 };
let (tx, rx) =
zx::Fifo::create(2, ::std::mem::size_of::<entry>()).expect("failed to create zx fifo");
let (tx, rx) = (Fifo::<entry>::from_fifo(tx), Fifo::<entry>::from_fifo(rx));
let mut buffer = entry::default();
let receive_future = rx.read_entries(&mut buffer).map_ok(|count| {
assert_eq!(count, 1);
});
let receiver = receive_future.on_timeout(300.millis().after_now(), || panic!("timeout"));
let sender = Timer::new(10.millis().after_now()).then(|()| tx.write_entries(&element));
let done = try_join(receiver, sender);
exec.run_singlethreaded(done).expect("failed to run receive future on executor");
assert_eq!(buffer, element);
}
#[test]
fn read_wrong_size() {
let mut exec = TestExecutor::new();
let elements = &[entry { a: 10, b: 20 }][..];
let (tx, rx) =
zx::Fifo::create(2, ::std::mem::size_of::<entry>()).expect("failed to create zx fifo");
let (tx, rx) = (Fifo::<entry>::from_fifo(tx), Fifo::<wrong_entry>::from_fifo(rx));
let mut buffer = wrong_entry::default();
let receive_future = rx
.read_entries(&mut buffer)
.map_ok(|count| panic!("read should have failed, got {}", count));
let receiver = receive_future.on_timeout(300.millis().after_now(), || panic!("timeout"));
let sender = Timer::new(10.millis().after_now()).then(|()| tx.write_entries(elements));
let done = try_join(receiver, sender);
let res = exec.run_singlethreaded(done);
match res {
Err(zx::Status::OUT_OF_RANGE) => (),
_ => panic!("did not get out-of-range error"),
}
}
#[test]
fn write_wrong_size() {
let mut exec = TestExecutor::new();
let elements = &[wrong_entry { a: 10 }][..];
let (tx, rx) =
zx::Fifo::create(2, ::std::mem::size_of::<entry>()).expect("failed to create zx fifo");
let (tx, _rx) = (Fifo::<wrong_entry>::from_fifo(tx), Fifo::<entry>::from_fifo(rx));
let sender = Timer::new(10.millis().after_now()).then(|()| tx.write_entries(elements));
let res = exec.run_singlethreaded(sender);
match res {
Err(zx::Status::OUT_OF_RANGE) => (),
_ => panic!("did not get out-of-range error"),
}
}
#[test]
fn write_into_full() {
use std::sync::atomic::{AtomicUsize, Ordering};
let mut exec = TestExecutor::new();
let elements =
&[entry { a: 10, b: 20 }, entry { a: 30, b: 40 }, entry { a: 50, b: 60 }][..];
let (tx, rx) =
zx::Fifo::create(2, ::std::mem::size_of::<entry>()).expect("failed to create zx fifo");
let (tx, rx) = (Fifo::<entry>::from_fifo(tx), Fifo::<entry>::from_fifo(rx));
let writes_completed = AtomicUsize::new(0);
let sender = async {
tx.write_entries(&elements[..2]).await?;
writes_completed.fetch_add(1, Ordering::SeqCst);
tx.write_entries(&elements[2..]).await?;
writes_completed.fetch_add(1, Ordering::SeqCst);
Ok::<(), zx::Status>(())
};
let receive_future = async {
Timer::new(10.millis().after_now()).await;
let mut buffer = entry::default();
let count = rx.read_entries(&mut buffer).await?;
assert_eq!(writes_completed.load(Ordering::SeqCst), 1);
assert_eq!(count, 1);
assert_eq!(buffer, elements[0]);
let count = rx.read_entries(&mut buffer).await?;
assert_eq!(count, 1);
assert_eq!(buffer, elements[1]);
let count = rx.read_entries(&mut buffer).await?;
assert_eq!(writes_completed.load(Ordering::SeqCst), 2);
assert_eq!(count, 1);
assert_eq!(buffer, elements[2]);
Ok::<(), zx::Status>(())
};
let receiver = receive_future.on_timeout(300.millis().after_now(), || panic!("timeout"));
let done = try_join(receiver, sender);
exec.run_singlethreaded(done).expect("failed to run receive future on executor");
}
#[test]
fn write_more_than_full() {
let mut exec = TestExecutor::new();
let elements =
&[entry { a: 10, b: 20 }, entry { a: 30, b: 40 }, entry { a: 50, b: 60 }][..];
let (tx, rx) =
zx::Fifo::create(2, ::std::mem::size_of::<entry>()).expect("failed to create zx fifo");
let (tx, rx) = (Fifo::<entry>::from_fifo(tx), Fifo::<entry>::from_fifo(rx));
let sender = tx.write_entries(elements);
let receive_future = async {
Timer::new(10.millis().after_now()).await;
for e in elements {
let mut buffer = [entry::default(); 1];
let count = rx.read_entries(&mut buffer[..]).await?;
assert_eq!(count, 1);
assert_eq!(&buffer[0], e);
}
Ok::<(), zx::Status>(())
};
let receiver = receive_future.on_timeout(300.millis().after_now(), || panic!("timeout"));
let done = try_join(receiver, sender);
exec.run_singlethreaded(done).expect("failed to run receive future on executor");
}
#[test]
fn read_multiple() {
let mut exec = TestExecutor::new();
let elements =
&[entry { a: 10, b: 20 }, entry { a: 30, b: 40 }, entry { a: 50, b: 60 }][..];
let (tx, rx) = zx::Fifo::create(elements.len(), ::std::mem::size_of::<entry>())
.expect("failed to create zx fifo");
let (tx, rx) = (Fifo::<entry>::from_fifo(tx), Fifo::<entry>::from_fifo(rx));
let write_fut = async {
tx.write_entries(&elements[..]).await.expect("failed write entries");
};
let read_fut = async {
let mut buffer = [entry::default(); 5];
let count = rx.read_entries(&mut buffer[..]).await.expect("failed to read entries");
assert_eq!(count, elements.len());
assert_eq!(&buffer[..count], &elements[..]);
};
let ((), ()) = exec.run_singlethreaded(futures::future::join(write_fut, read_fut));
}
#[test]
fn read_one() {
let mut exec = TestExecutor::new();
let elements =
&[entry { a: 10, b: 20 }, entry { a: 30, b: 40 }, entry { a: 50, b: 60 }][..];
let (tx, rx) = zx::Fifo::create(elements.len(), ::std::mem::size_of::<entry>())
.expect("failed to create zx fifo");
let (tx, rx) = (Fifo::<entry>::from_fifo(tx), Fifo::<entry>::from_fifo(rx));
let write_fut = async {
tx.write_entries(&elements[..]).await.expect("failed write entries");
};
let read_fut = async {
for e in elements {
let received = rx.read_entry().await.expect("failed to read entry");
assert_eq!(&received, e);
}
};
let ((), ()) = exec.run_singlethreaded(futures::future::join(write_fut, read_fut));
}
#[test]
fn maybe_uninit_single() {
let mut exec = TestExecutor::new();
let element = entry { a: 10, b: 20 };
let (tx, rx) =
zx::Fifo::create(1, ::std::mem::size_of::<entry>()).expect("failed to create zx fifo");
let (tx, rx) = (Fifo::<entry>::from_fifo(tx), Fifo::<entry>::from_fifo(rx));
let write_fut = async {
tx.write_entries(&element).await.expect("failed write entries");
};
let read_fut = async {
let mut buffer = MaybeUninit::<entry>::uninit();
let count = rx.read_entries(&mut buffer).await.expect("failed to read entries");
assert_eq!(count, 1);
let read = unsafe { buffer.assume_init() };
assert_eq!(read, element);
};
let ((), ()) = exec.run_singlethreaded(futures::future::join(write_fut, read_fut));
}
#[test]
fn maybe_uninit_slice() {
let mut exec = TestExecutor::new();
let elements =
&[entry { a: 10, b: 20 }, entry { a: 30, b: 40 }, entry { a: 50, b: 60 }][..];
let (tx, rx) = zx::Fifo::create(elements.len(), ::std::mem::size_of::<entry>())
.expect("failed to create zx fifo");
let (tx, rx) = (Fifo::<entry>::from_fifo(tx), Fifo::<entry>::from_fifo(rx));
let write_fut = async {
tx.write_entries(&elements[..]).await.expect("failed write entries");
};
let read_fut = async {
let mut buffer = [MaybeUninit::<entry>::uninit(); 15];
let count = rx.read_entries(&mut buffer[..]).await.expect("failed to read entries");
assert_eq!(count, elements.len());
let read = &mut buffer[..count];
for (i, v) in read.iter_mut().enumerate() {
let read = unsafe { v.assume_init_ref() };
assert_eq!(read, &elements[i]);
unsafe {
v.assume_init_drop();
}
}
};
let ((), ()) = exec.run_singlethreaded(futures::future::join(write_fut, read_fut));
}
}