Cleanup BufferedUarte
This commit is contained in:
		
							parent
							
								
									3a4dbfa52e
								
							
						
					
					
						commit
						607e67f51a
					
				| @ -33,12 +33,25 @@ enum RxState { | |||||||
|     ReceivingReady, |     ReceivingReady, | ||||||
|     Stopping, |     Stopping, | ||||||
| } | } | ||||||
|  | 
 | ||||||
| #[derive(Copy, Clone, Debug, PartialEq)] | #[derive(Copy, Clone, Debug, PartialEq)] | ||||||
| enum TxState { | enum TxState { | ||||||
|     Idle, |     Idle, | ||||||
|     Transmitting(usize), |     Transmitting(usize), | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | struct State<'a, T: Instance> { | ||||||
|  |     inner: T, | ||||||
|  | 
 | ||||||
|  |     rx: RingBuffer<'a>, | ||||||
|  |     rx_state: RxState, | ||||||
|  |     rx_waker: WakerRegistration, | ||||||
|  | 
 | ||||||
|  |     tx: RingBuffer<'a>, | ||||||
|  |     tx_state: TxState, | ||||||
|  |     tx_waker: WakerRegistration, | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /// Interface to a UARTE instance
 | /// Interface to a UARTE instance
 | ||||||
| ///
 | ///
 | ||||||
| /// This is a very basic interface that comes with the following limitations:
 | /// This is a very basic interface that comes with the following limitations:
 | ||||||
| @ -145,6 +158,10 @@ impl<'a, T: Instance> BufferedUarte<'a, T> { | |||||||
|             ), |             ), | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|  |     fn inner(self: Pin<&mut Self>) -> Pin<&mut PeripheralMutex<T::Interrupt, State<'a, T>>> { | ||||||
|  |         unsafe { Pin::new_unchecked(&mut self.get_unchecked_mut().inner) } | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl<'a, T: Instance> Drop for BufferedUarte<'a, T> { | impl<'a, T: Instance> Drop for BufferedUarte<'a, T> { | ||||||
| @ -156,51 +173,7 @@ impl<'a, T: Instance> Drop for BufferedUarte<'a, T> { | |||||||
| 
 | 
 | ||||||
| impl<'a, T: Instance> AsyncBufRead for BufferedUarte<'a, T> { | impl<'a, T: Instance> AsyncBufRead for BufferedUarte<'a, T> { | ||||||
|     fn poll_fill_buf(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Result<&[u8]>> { |     fn poll_fill_buf(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Result<&[u8]>> { | ||||||
|         let this = unsafe { self.get_unchecked_mut() }; |         self.inner().with(|_irq, state| { | ||||||
|         let reg = unsafe { Pin::new_unchecked(&mut this.inner) }; |  | ||||||
|         reg.with(|_irq, state| { |  | ||||||
|             let z: Poll<Result<&[u8]>> = state.poll_fill_buf(cx); |  | ||||||
|             let z: Poll<Result<&[u8]>> = unsafe { mem::transmute(z) }; |  | ||||||
|             z |  | ||||||
|         }) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     fn consume(self: Pin<&mut Self>, amt: usize) { |  | ||||||
|         let this = unsafe { self.get_unchecked_mut() }; |  | ||||||
|         let reg = unsafe { Pin::new_unchecked(&mut this.inner) }; |  | ||||||
|         reg.with(|irq, state| state.consume(irq, amt)) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<'a, T: Instance> AsyncWrite for BufferedUarte<'a, T> { |  | ||||||
|     fn poll_write(self: Pin<&mut Self>, cx: &mut Context<'_>, buf: &[u8]) -> Poll<Result<usize>> { |  | ||||||
|         let this = unsafe { self.get_unchecked_mut() }; |  | ||||||
|         let reg = unsafe { Pin::new_unchecked(&mut this.inner) }; |  | ||||||
|         reg.with(|irq, state| state.poll_write(irq, cx, buf)) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // ====================================
 |  | ||||||
| // ====================================
 |  | ||||||
| // ====================================
 |  | ||||||
| 
 |  | ||||||
| // public because it needs to be used in Instance trait, but
 |  | ||||||
| // should not be used outside the module
 |  | ||||||
| #[doc(hidden)] |  | ||||||
| pub struct State<'a, T: Instance> { |  | ||||||
|     inner: T, |  | ||||||
| 
 |  | ||||||
|     rx: RingBuffer<'a>, |  | ||||||
|     rx_state: RxState, |  | ||||||
|     rx_waker: WakerRegistration, |  | ||||||
| 
 |  | ||||||
|     tx: RingBuffer<'a>, |  | ||||||
|     tx_state: TxState, |  | ||||||
|     tx_waker: WakerRegistration, |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<'a, T: Instance> State<'a, T> { |  | ||||||
|     fn poll_fill_buf(&mut self, cx: &mut Context<'_>) -> Poll<Result<&[u8]>> { |  | ||||||
|             // Conservative compiler fence to prevent optimizations that do not
 |             // Conservative compiler fence to prevent optimizations that do not
 | ||||||
|             // take in to account actions by DMA. The fence has been placed here,
 |             // take in to account actions by DMA. The fence has been placed here,
 | ||||||
|             // before any DMA action has started
 |             // before any DMA action has started
 | ||||||
| @ -208,48 +181,51 @@ impl<'a, T: Instance> State<'a, T> { | |||||||
|             trace!("poll_read"); |             trace!("poll_read"); | ||||||
| 
 | 
 | ||||||
|             // We have data ready in buffer? Return it.
 |             // We have data ready in buffer? Return it.
 | ||||||
|         let buf = self.rx.pop_buf(); |             let buf = state.rx.pop_buf(); | ||||||
|             if buf.len() != 0 { |             if buf.len() != 0 { | ||||||
|                 trace!("  got {:?} {:?}", buf.as_ptr() as u32, buf.len()); |                 trace!("  got {:?} {:?}", buf.as_ptr() as u32, buf.len()); | ||||||
|  |                 let buf: &[u8] = buf; | ||||||
|  |                 let buf: &[u8] = unsafe { mem::transmute(buf) }; | ||||||
|                 return Poll::Ready(Ok(buf)); |                 return Poll::Ready(Ok(buf)); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             trace!("  empty"); |             trace!("  empty"); | ||||||
| 
 | 
 | ||||||
|         if self.rx_state == RxState::ReceivingReady { |             if state.rx_state == RxState::ReceivingReady { | ||||||
|                 trace!("  stopping"); |                 trace!("  stopping"); | ||||||
|             self.rx_state = RxState::Stopping; |                 state.rx_state = RxState::Stopping; | ||||||
|             self.inner.tasks_stoprx.write(|w| unsafe { w.bits(1) }); |                 state.inner.tasks_stoprx.write(|w| unsafe { w.bits(1) }); | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|         self.rx_waker.register(cx.waker()); |             state.rx_waker.register(cx.waker()); | ||||||
|         Poll::Pending |             Poll::<Result<&[u8]>>::Pending | ||||||
|  |         }) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     fn consume(&mut self, irq: &mut T::Interrupt, amt: usize) { |     fn consume(self: Pin<&mut Self>, amt: usize) { | ||||||
|  |         self.inner().with(|irq, state| { | ||||||
|             trace!("consume {:?}", amt); |             trace!("consume {:?}", amt); | ||||||
|         self.rx.pop(amt); |             state.rx.pop(amt); | ||||||
|             irq.pend(); |             irq.pend(); | ||||||
|  |         }) | ||||||
|  |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|     fn poll_write( | impl<'a, T: Instance> AsyncWrite for BufferedUarte<'a, T> { | ||||||
|         &mut self, |     fn poll_write(self: Pin<&mut Self>, cx: &mut Context<'_>, buf: &[u8]) -> Poll<Result<usize>> { | ||||||
|         irq: &mut T::Interrupt, |         self.inner().with(|irq, state| { | ||||||
|         cx: &mut Context<'_>, |  | ||||||
|         buf: &[u8], |  | ||||||
|     ) -> Poll<Result<usize>> { |  | ||||||
|             trace!("poll_write: {:?}", buf.len()); |             trace!("poll_write: {:?}", buf.len()); | ||||||
| 
 | 
 | ||||||
|         let tx_buf = self.tx.push_buf(); |             let tx_buf = state.tx.push_buf(); | ||||||
|             if tx_buf.len() == 0 { |             if tx_buf.len() == 0 { | ||||||
|                 trace!("poll_write: pending"); |                 trace!("poll_write: pending"); | ||||||
|             self.tx_waker.register(cx.waker()); |                 state.tx_waker.register(cx.waker()); | ||||||
|                 return Poll::Pending; |                 return Poll::Pending; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             let n = min(tx_buf.len(), buf.len()); |             let n = min(tx_buf.len(), buf.len()); | ||||||
|             tx_buf[..n].copy_from_slice(&buf[..n]); |             tx_buf[..n].copy_from_slice(&buf[..n]); | ||||||
|         self.tx.push(n); |             state.tx.push(n); | ||||||
| 
 | 
 | ||||||
|             trace!("poll_write: queued {:?}", n); |             trace!("poll_write: queued {:?}", n); | ||||||
| 
 | 
 | ||||||
| @ -261,6 +237,7 @@ impl<'a, T: Instance> State<'a, T> { | |||||||
|             irq.pend(); |             irq.pend(); | ||||||
| 
 | 
 | ||||||
|             Poll::Ready(Ok(n)) |             Poll::Ready(Ok(n)) | ||||||
|  |         }) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user