sel4_async_io/
lib.rs
1#![no_std]
11
12use core::future::{poll_fn, Future};
13use core::pin::{pin, Pin};
14use core::task::{Context, Poll};
15
16use embedded_io_async as eio;
17
18pub use embedded_io_async::{Error, ErrorKind, ErrorType};
19
20pub trait Read: ErrorType {
21 fn poll_read(
22 self: Pin<&mut Self>,
23 cx: &mut Context<'_>,
24 buf: &mut [u8],
25 ) -> Poll<Result<usize, Self::Error>>;
26}
27
28pub trait Write: ErrorType {
29 fn poll_write(
30 self: Pin<&mut Self>,
31 cx: &mut Context<'_>,
32 buf: &[u8],
33 ) -> Poll<Result<usize, Self::Error>>;
34
35 fn poll_flush(self: Pin<&mut Self>, _cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
36 Poll::Ready(Ok(()))
37 }
38}
39
40impl<T: Read + Unpin> Read for &mut T {
41 fn poll_read(
42 self: Pin<&mut Self>,
43 cx: &mut Context<'_>,
44 buf: &mut [u8],
45 ) -> Poll<Result<usize, Self::Error>> {
46 T::poll_read(Pin::new(*Pin::into_inner(self)), cx, buf)
47 }
48}
49
50impl<T: Write + Unpin> Write for &mut T {
51 fn poll_write(
52 self: Pin<&mut Self>,
53 cx: &mut Context<'_>,
54 buf: &[u8],
55 ) -> Poll<Result<usize, Self::Error>> {
56 T::poll_write(Pin::new(*Pin::into_inner(self)), cx, buf)
57 }
58
59 fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
60 T::poll_flush(Pin::new(*Pin::into_inner(self)), cx)
61 }
62}
63
64#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash)]
65pub struct EmbeddedIOAsyncAdapter<T>(pub T);
66
67impl<T: ErrorType> ErrorType for EmbeddedIOAsyncAdapter<T> {
68 type Error = T::Error;
69}
70
71impl<T> eio::Read for EmbeddedIOAsyncAdapter<T>
72where
73 T: Read + Unpin,
74{
75 async fn read(&mut self, buf: &mut [u8]) -> Result<usize, Self::Error> {
76 poll_fn(|cx| Pin::new(&mut self.0).poll_read(cx, buf)).await
77 }
78}
79
80impl<T> eio::Write for EmbeddedIOAsyncAdapter<T>
81where
82 T: Write + Unpin,
83{
84 async fn write(&mut self, buf: &[u8]) -> Result<usize, Self::Error> {
85 poll_fn(|cx| Pin::new(&mut self.0).poll_write(cx, buf)).await
86 }
87
88 async fn flush(&mut self) -> Result<(), Self::Error> {
89 poll_fn(|cx| Pin::new(&mut self.0).poll_flush(cx)).await
90 }
91}
92
93pub trait ReadCancelSafe: eio::Read {}
94pub trait FlushCancelSafe: eio::Write {}
95pub trait WriteCancelSafe: FlushCancelSafe {}
96
97impl<T: Read + Unpin> ReadCancelSafe for EmbeddedIOAsyncAdapter<T> {}
98impl<T: Write + Unpin> FlushCancelSafe for EmbeddedIOAsyncAdapter<T> {}
99impl<T: Write + Unpin> WriteCancelSafe for EmbeddedIOAsyncAdapter<T> {}
100
101impl<T> Read for EmbeddedIOAsyncAdapter<T>
102where
103 T: eio::Read + ReadCancelSafe + Unpin,
104{
105 fn poll_read(
106 mut self: Pin<&mut Self>,
107 cx: &mut Context<'_>,
108 buf: &mut [u8],
109 ) -> Poll<Result<usize, Self::Error>> {
110 pin!(self.0.read(buf)).poll(cx)
111 }
112}
113
114impl<T> Write for EmbeddedIOAsyncAdapter<T>
115where
116 T: eio::Write + WriteCancelSafe + Unpin,
117{
118 fn poll_write(
119 mut self: Pin<&mut Self>,
120 cx: &mut Context<'_>,
121 buf: &[u8],
122 ) -> Poll<Result<usize, Self::Error>> {
123 pin!(self.0.write(buf)).poll(cx)
124 }
125
126 fn poll_flush(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
127 pin!(self.0.flush()).poll(cx)
128 }
129}
130
131#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash)]
132pub struct EmbeddedIOAsyncAdapterUsingReady<T>(pub T);
133
134impl<T: ErrorType> ErrorType for EmbeddedIOAsyncAdapterUsingReady<T> {
135 type Error = T::Error;
136}
137
138impl<T> Read for EmbeddedIOAsyncAdapterUsingReady<T>
139where
140 T: eio::Read + eio::ReadReady + Unpin,
141{
142 fn poll_read(
143 mut self: Pin<&mut Self>,
144 cx: &mut Context<'_>,
145 buf: &mut [u8],
146 ) -> Poll<Result<usize, Self::Error>> {
147 if self.0.read_ready()? {
148 match pin!(self.0.read(buf)).poll(cx) {
149 Poll::Ready(r) => Poll::Ready(r),
150 Poll::Pending => unreachable!(),
151 }
152 } else {
153 Poll::Pending
154 }
155 }
156}
157
158impl<T> Write for EmbeddedIOAsyncAdapterUsingReady<T>
159where
160 T: eio::Write + eio::WriteReady + FlushCancelSafe + Unpin,
161{
162 fn poll_write(
163 mut self: Pin<&mut Self>,
164 cx: &mut Context<'_>,
165 buf: &[u8],
166 ) -> Poll<Result<usize, Self::Error>> {
167 if self.0.write_ready()? {
168 match pin!(self.0.write(buf)).poll(cx) {
169 Poll::Ready(r) => Poll::Ready(r),
170 Poll::Pending => unreachable!(),
171 }
172 } else {
173 Poll::Pending
174 }
175 }
176
177 fn poll_flush(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
178 pin!(self.0.flush()).poll(cx)
179 }
180}