Add embassy-net-nrf91.
This commit is contained in:
		
							parent
							
								
									aff66b9695
								
							
						
					
					
						commit
						160e1c38ce
					
				
							
								
								
									
										37
									
								
								embassy-net-nrf91/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								embassy-net-nrf91/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,37 @@ | ||||
| [package] | ||||
| name = "embassy-net-nrf91" | ||||
| version = "0.1.0" | ||||
| edition = "2021" | ||||
| description = "embassy-net driver for Nordic nRF91-series cellular modems" | ||||
| keywords = ["embedded", "nrf91", "embassy-net", "cellular"] | ||||
| categories = ["embedded", "hardware-support", "no-std", "network-programming", "asynchronous"] | ||||
| license = "MIT OR Apache-2.0" | ||||
| repository = "https://github.com/embassy-rs/embassy" | ||||
| documentation = "https://docs.embassy.dev/embassy-net-nrf91" | ||||
| 
 | ||||
| [features] | ||||
| defmt = [ "dep:defmt", "heapless/defmt-03" ] | ||||
| log = [ "dep:log" ] | ||||
| 
 | ||||
| [dependencies] | ||||
| defmt = { version = "0.3", optional = true } | ||||
| log = { version = "0.4.14", optional = true } | ||||
| 
 | ||||
| nrf9160-pac = { version = "0.12.0" } | ||||
| 
 | ||||
| embassy-time = { version = "0.3.1", path = "../embassy-time" } | ||||
| embassy-sync = { version = "0.6.0", path = "../embassy-sync"} | ||||
| embassy-futures = { version = "0.1.0", path = "../embassy-futures"} | ||||
| embassy-net-driver-channel = { version = "0.2.0", path = "../embassy-net-driver-channel"} | ||||
| 
 | ||||
| heapless = "0.8" | ||||
| embedded-io = "0.6.1" | ||||
| 
 | ||||
| [package.metadata.embassy_docs] | ||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-net-nrf91-v$VERSION/embassy-net-nrf91/src/" | ||||
| src_base_git = "https://github.com/embassy-rs/embassy/blob/$COMMIT/embassy-net-nrf91/src/" | ||||
| target = "thumbv7em-none-eabi" | ||||
| features = ["defmt"] | ||||
| 
 | ||||
| [package.metadata.docs.rs] | ||||
| features = ["defmt"] | ||||
							
								
								
									
										9
									
								
								embassy-net-nrf91/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								embassy-net-nrf91/README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | ||||
| # nRF91 `embassy-net` integration | ||||
| 
 | ||||
| [`embassy-net`](https://crates.io/crates/embassy-net) driver for Nordic nRF91-series cellular modems. | ||||
| 
 | ||||
| See the [`examples`](https://github.com/embassy-rs/embassy/tree/main/examples/nrf9160) directory for usage examples with the nRF9160. | ||||
| 
 | ||||
| ## Interoperability | ||||
| 
 | ||||
| This crate can run on any executor. | ||||
							
								
								
									
										274
									
								
								embassy-net-nrf91/src/fmt.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										274
									
								
								embassy-net-nrf91/src/fmt.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,274 @@ | ||||
| #![macro_use] | ||||
| #![allow(unused)] | ||||
| 
 | ||||
| use core::fmt::{Debug, Display, LowerHex}; | ||||
| 
 | ||||
| #[cfg(all(feature = "defmt", feature = "log"))] | ||||
| compile_error!("You may not enable both `defmt` and `log` features."); | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! assert { | ||||
|     ($($x:tt)*) => { | ||||
|         { | ||||
|             #[cfg(not(feature = "defmt"))] | ||||
|             ::core::assert!($($x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::assert!($($x)*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! assert_eq { | ||||
|     ($($x:tt)*) => { | ||||
|         { | ||||
|             #[cfg(not(feature = "defmt"))] | ||||
|             ::core::assert_eq!($($x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::assert_eq!($($x)*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! assert_ne { | ||||
|     ($($x:tt)*) => { | ||||
|         { | ||||
|             #[cfg(not(feature = "defmt"))] | ||||
|             ::core::assert_ne!($($x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::assert_ne!($($x)*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! debug_assert { | ||||
|     ($($x:tt)*) => { | ||||
|         { | ||||
|             #[cfg(not(feature = "defmt"))] | ||||
|             ::core::debug_assert!($($x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::debug_assert!($($x)*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! debug_assert_eq { | ||||
|     ($($x:tt)*) => { | ||||
|         { | ||||
|             #[cfg(not(feature = "defmt"))] | ||||
|             ::core::debug_assert_eq!($($x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::debug_assert_eq!($($x)*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! debug_assert_ne { | ||||
|     ($($x:tt)*) => { | ||||
|         { | ||||
|             #[cfg(not(feature = "defmt"))] | ||||
|             ::core::debug_assert_ne!($($x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::debug_assert_ne!($($x)*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! todo { | ||||
|     ($($x:tt)*) => { | ||||
|         { | ||||
|             #[cfg(not(feature = "defmt"))] | ||||
|             ::core::todo!($($x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::todo!($($x)*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[cfg(not(feature = "defmt"))] | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! unreachable { | ||||
|     ($($x:tt)*) => { | ||||
|         ::core::unreachable!($($x)*) | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[cfg(feature = "defmt")] | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! unreachable { | ||||
|     ($($x:tt)*) => { | ||||
|         ::defmt::unreachable!($($x)*) | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! panic { | ||||
|     ($($x:tt)*) => { | ||||
|         { | ||||
|             #[cfg(not(feature = "defmt"))] | ||||
|             ::core::panic!($($x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::panic!($($x)*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! trace { | ||||
|     ($s:literal $(, $x:expr)* $(,)?) => { | ||||
|         { | ||||
|             #[cfg(feature = "log")] | ||||
|             ::log::trace!($s $(, $x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::trace!($s $(, $x)*); | ||||
|             #[cfg(not(any(feature = "log", feature="defmt")))] | ||||
|             let _ = ($( & $x ),*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! debug { | ||||
|     ($s:literal $(, $x:expr)* $(,)?) => { | ||||
|         { | ||||
|             #[cfg(feature = "log")] | ||||
|             ::log::debug!($s $(, $x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::debug!($s $(, $x)*); | ||||
|             #[cfg(not(any(feature = "log", feature="defmt")))] | ||||
|             let _ = ($( & $x ),*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! info { | ||||
|     ($s:literal $(, $x:expr)* $(,)?) => { | ||||
|         { | ||||
|             #[cfg(feature = "log")] | ||||
|             ::log::info!($s $(, $x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::info!($s $(, $x)*); | ||||
|             #[cfg(not(any(feature = "log", feature="defmt")))] | ||||
|             let _ = ($( & $x ),*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! warn { | ||||
|     ($s:literal $(, $x:expr)* $(,)?) => { | ||||
|         { | ||||
|             #[cfg(feature = "log")] | ||||
|             ::log::warn!($s $(, $x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::warn!($s $(, $x)*); | ||||
|             #[cfg(not(any(feature = "log", feature="defmt")))] | ||||
|             let _ = ($( & $x ),*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! error { | ||||
|     ($s:literal $(, $x:expr)* $(,)?) => { | ||||
|         { | ||||
|             #[cfg(feature = "log")] | ||||
|             ::log::error!($s $(, $x)*); | ||||
|             #[cfg(feature = "defmt")] | ||||
|             ::defmt::error!($s $(, $x)*); | ||||
|             #[cfg(not(any(feature = "log", feature="defmt")))] | ||||
|             let _ = ($( & $x ),*); | ||||
|         } | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[cfg(feature = "defmt")] | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! unwrap { | ||||
|     ($($x:tt)*) => { | ||||
|         ::defmt::unwrap!($($x)*) | ||||
|     }; | ||||
| } | ||||
| 
 | ||||
| #[cfg(not(feature = "defmt"))] | ||||
| #[collapse_debuginfo(yes)] | ||||
| macro_rules! unwrap { | ||||
|     ($arg:expr) => { | ||||
|         match $crate::fmt::Try::into_result($arg) { | ||||
|             ::core::result::Result::Ok(t) => t, | ||||
|             ::core::result::Result::Err(e) => { | ||||
|                 ::core::panic!("unwrap of `{}` failed: {:?}", ::core::stringify!($arg), e); | ||||
|             } | ||||
|         } | ||||
|     }; | ||||
|     ($arg:expr, $($msg:expr),+ $(,)? ) => { | ||||
|         match $crate::fmt::Try::into_result($arg) { | ||||
|             ::core::result::Result::Ok(t) => t, | ||||
|             ::core::result::Result::Err(e) => { | ||||
|                 ::core::panic!("unwrap of `{}` failed: {}: {:?}", ::core::stringify!($arg), ::core::format_args!($($msg,)*), e); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| #[derive(Debug, Copy, Clone, Eq, PartialEq)] | ||||
| pub struct NoneError; | ||||
| 
 | ||||
| pub trait Try { | ||||
|     type Ok; | ||||
|     type Error; | ||||
|     fn into_result(self) -> Result<Self::Ok, Self::Error>; | ||||
| } | ||||
| 
 | ||||
| impl<T> Try for Option<T> { | ||||
|     type Ok = T; | ||||
|     type Error = NoneError; | ||||
| 
 | ||||
|     #[inline] | ||||
|     fn into_result(self) -> Result<T, NoneError> { | ||||
|         self.ok_or(NoneError) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<T, E> Try for Result<T, E> { | ||||
|     type Ok = T; | ||||
|     type Error = E; | ||||
| 
 | ||||
|     #[inline] | ||||
|     fn into_result(self) -> Self { | ||||
|         self | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| pub(crate) struct Bytes<'a>(pub &'a [u8]); | ||||
| 
 | ||||
| impl<'a> Debug for Bytes<'a> { | ||||
|     fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { | ||||
|         write!(f, "{:#02x?}", self.0) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<'a> Display for Bytes<'a> { | ||||
|     fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { | ||||
|         write!(f, "{:#02x?}", self.0) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<'a> LowerHex for Bytes<'a> { | ||||
|     fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { | ||||
|         write!(f, "{:#02x?}", self.0) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| #[cfg(feature = "defmt")] | ||||
| impl<'a> defmt::Format for Bytes<'a> { | ||||
|     fn format(&self, fmt: defmt::Formatter) { | ||||
|         defmt::write!(fmt, "{:02x}", self.0) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										970
									
								
								embassy-net-nrf91/src/lib.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										970
									
								
								embassy-net-nrf91/src/lib.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,970 @@ | ||||
| #![no_std] | ||||
| #![doc = include_str!("../README.md")] | ||||
| #![warn(missing_docs)] | ||||
| 
 | ||||
| // must be first
 | ||||
| mod fmt; | ||||
| 
 | ||||
| use core::cell::RefCell; | ||||
| use core::future::poll_fn; | ||||
| use core::marker::PhantomData; | ||||
| use core::mem::{self, MaybeUninit}; | ||||
| use core::ptr::{self, addr_of, addr_of_mut, copy_nonoverlapping}; | ||||
| use core::slice; | ||||
| use core::sync::atomic::{compiler_fence, fence, Ordering}; | ||||
| use core::task::{Poll, Waker}; | ||||
| 
 | ||||
| use embassy_net_driver_channel as ch; | ||||
| use embassy_sync::waitqueue::{AtomicWaker, WakerRegistration}; | ||||
| use heapless::Vec; | ||||
| use nrf9160_pac as pac; | ||||
| use pac::NVIC; | ||||
| 
 | ||||
| const RX_SIZE: usize = 8 * 1024; | ||||
| const TRACE_SIZE: usize = 16 * 1024; | ||||
| const MTU: usize = 1500; | ||||
| 
 | ||||
| /// Network driver.
 | ||||
| ///
 | ||||
| /// This is the type you have to pass to `embassy-net` when creating the network stack.
 | ||||
| pub type NetDriver<'a> = ch::Device<'a, MTU>; | ||||
| 
 | ||||
| static WAKER: AtomicWaker = AtomicWaker::new(); | ||||
| 
 | ||||
| /// Call this function on IPC IRQ
 | ||||
| pub fn on_ipc_irq() { | ||||
|     let ipc = unsafe { &*pac::IPC_NS::ptr() }; | ||||
| 
 | ||||
|     trace!("irq"); | ||||
| 
 | ||||
|     ipc.inten.write(|w| w); | ||||
|     WAKER.wake(); | ||||
| } | ||||
| 
 | ||||
| struct Allocator<'a> { | ||||
|     start: *mut u8, | ||||
|     end: *mut u8, | ||||
|     _phantom: PhantomData<&'a mut u8>, | ||||
| } | ||||
| 
 | ||||
| impl<'a> Allocator<'a> { | ||||
|     fn alloc_bytes(&mut self, size: usize) -> &'a mut [MaybeUninit<u8>] { | ||||
|         // safety: both pointers come from the same allocation.
 | ||||
|         let available_size = unsafe { self.end.offset_from(self.start) } as usize; | ||||
|         if size > available_size { | ||||
|             panic!("out of memory") | ||||
|         } | ||||
| 
 | ||||
|         // safety: we've checked above this doesn't go out of bounds.
 | ||||
|         let p = self.start; | ||||
|         self.start = unsafe { p.add(size) }; | ||||
| 
 | ||||
|         // safety: we've checked the pointer is in-bounds.
 | ||||
|         unsafe { slice::from_raw_parts_mut(p as *mut _, size) } | ||||
|     } | ||||
| 
 | ||||
|     fn alloc<T>(&mut self) -> &'a mut MaybeUninit<T> { | ||||
|         let align = mem::align_of::<T>(); | ||||
|         let size = mem::size_of::<T>(); | ||||
| 
 | ||||
|         let align_size = match (self.start as usize) % align { | ||||
|             0 => 0, | ||||
|             n => align - n, | ||||
|         }; | ||||
| 
 | ||||
|         // safety: both pointers come from the same allocation.
 | ||||
|         let available_size = unsafe { self.end.offset_from(self.start) } as usize; | ||||
|         if align_size + size > available_size { | ||||
|             panic!("out of memory") | ||||
|         } | ||||
| 
 | ||||
|         // safety: we've checked above this doesn't go out of bounds.
 | ||||
|         let p = unsafe { self.start.add(align_size) }; | ||||
|         self.start = unsafe { p.add(size) }; | ||||
| 
 | ||||
|         // safety: we've checked the pointer is aligned and in-bounds.
 | ||||
|         unsafe { &mut *(p as *mut _) } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /// Create a new nRF91 embassy-net driver.
 | ||||
| pub async fn new<'a, TW: embedded_io::Write>( | ||||
|     state: &'a mut State, | ||||
|     shmem: &'a mut [MaybeUninit<u8>], | ||||
|     trace_writer: TW, | ||||
| ) -> (NetDriver<'a>, Control<'a>, Runner<'a, TW>) { | ||||
|     let shmem_len = shmem.len(); | ||||
|     let shmem_ptr = shmem.as_mut_ptr() as *mut u8; | ||||
| 
 | ||||
|     const SPU_REGION_SIZE: usize = 8192; // 8kb
 | ||||
|     assert!(shmem_len != 0); | ||||
|     assert!( | ||||
|         shmem_len % SPU_REGION_SIZE == 0, | ||||
|         "shmem length must be a multiple of 8kb" | ||||
|     ); | ||||
|     assert!( | ||||
|         (shmem_ptr as usize) % SPU_REGION_SIZE == 0, | ||||
|         "shmem length must be a multiple of 8kb" | ||||
|     ); | ||||
|     assert!( | ||||
|         (shmem_ptr as usize + shmem_len) < 0x2002_0000, | ||||
|         "shmem must be in the lower 128kb of RAM" | ||||
|     ); | ||||
| 
 | ||||
|     let spu = unsafe { &*pac::SPU_S::ptr() }; | ||||
|     debug!("Setting IPC RAM as nonsecure..."); | ||||
|     let region_start = (shmem_ptr as usize - 0x2000_0000) / SPU_REGION_SIZE; | ||||
|     let region_end = region_start + shmem_len / SPU_REGION_SIZE; | ||||
|     for i in region_start..region_end { | ||||
|         spu.ramregion[i].perm.write(|w| { | ||||
|             w.execute().set_bit(); | ||||
|             w.write().set_bit(); | ||||
|             w.read().set_bit(); | ||||
|             w.secattr().clear_bit(); | ||||
|             w.lock().clear_bit(); | ||||
|             w | ||||
|         }) | ||||
|     } | ||||
| 
 | ||||
|     spu.periphid[42].perm.write(|w| w.secattr().non_secure()); | ||||
| 
 | ||||
|     let mut alloc = Allocator { | ||||
|         start: shmem_ptr, | ||||
|         end: unsafe { shmem_ptr.add(shmem_len) }, | ||||
|         _phantom: PhantomData, | ||||
|     }; | ||||
| 
 | ||||
|     let ipc = unsafe { &*pac::IPC_NS::ptr() }; | ||||
|     let power = unsafe { &*pac::POWER_S::ptr() }; | ||||
| 
 | ||||
|     let cb: &mut ControlBlock = alloc.alloc().write(unsafe { mem::zeroed() }); | ||||
|     let rx = alloc.alloc_bytes(RX_SIZE); | ||||
|     let trace = alloc.alloc_bytes(TRACE_SIZE); | ||||
| 
 | ||||
|     cb.version = 0x00010000; | ||||
|     cb.rx_base = rx.as_mut_ptr() as _; | ||||
|     cb.rx_size = RX_SIZE; | ||||
|     cb.control_list_ptr = &mut cb.lists[0]; | ||||
|     cb.data_list_ptr = &mut cb.lists[1]; | ||||
|     cb.modem_info_ptr = &mut cb.modem_info; | ||||
|     cb.trace_ptr = &mut cb.trace; | ||||
|     cb.lists[0].len = LIST_LEN; | ||||
|     cb.lists[1].len = LIST_LEN; | ||||
|     cb.trace.base = trace.as_mut_ptr() as _; | ||||
|     cb.trace.size = TRACE_SIZE; | ||||
| 
 | ||||
|     ipc.gpmem[0].write(|w| unsafe { w.bits(cb as *mut _ as u32) }); | ||||
|     ipc.gpmem[1].write(|w| unsafe { w.bits(0) }); | ||||
| 
 | ||||
|     // connect task/event i to channel i
 | ||||
|     for i in 0..8 { | ||||
|         ipc.send_cnf[i].write(|w| unsafe { w.bits(1 << i) }); | ||||
|         ipc.receive_cnf[i].write(|w| unsafe { w.bits(1 << i) }); | ||||
|     } | ||||
| 
 | ||||
|     compiler_fence(Ordering::SeqCst); | ||||
| 
 | ||||
|     // POWER.LTEMODEM.STARTN = 0
 | ||||
|     // The reg is missing in the PAC??
 | ||||
|     let startn = unsafe { (power as *const _ as *mut u32).add(0x610 / 4) }; | ||||
|     unsafe { startn.write_volatile(0) } | ||||
| 
 | ||||
|     unsafe { NVIC::unmask(pac::Interrupt::IPC) }; | ||||
| 
 | ||||
|     let state_inner = &*state.inner.write(RefCell::new(StateInner { | ||||
|         init: false, | ||||
|         init_waker: WakerRegistration::new(), | ||||
|         cb, | ||||
|         requests: [const { None }; REQ_COUNT], | ||||
|         next_req_serial: 0x12345678, | ||||
| 
 | ||||
|         rx_control_list: ptr::null_mut(), | ||||
|         rx_data_list: ptr::null_mut(), | ||||
|         rx_seq_no: 0, | ||||
|         rx_check: PointerChecker { | ||||
|             start: rx.as_mut_ptr() as *mut u8, | ||||
|             end: (rx.as_mut_ptr() as *mut u8).wrapping_add(RX_SIZE), | ||||
|         }, | ||||
| 
 | ||||
|         tx_seq_no: 0, | ||||
|         tx_buf_used: [false; TX_BUF_COUNT], | ||||
| 
 | ||||
|         trace_chans: Vec::new(), | ||||
|         trace_check: PointerChecker { | ||||
|             start: trace.as_mut_ptr() as *mut u8, | ||||
|             end: (trace.as_mut_ptr() as *mut u8).wrapping_add(TRACE_SIZE), | ||||
|         }, | ||||
|     })); | ||||
| 
 | ||||
|     let control = Control { state: state_inner }; | ||||
| 
 | ||||
|     let (ch_runner, device) = ch::new(&mut state.ch, ch::driver::HardwareAddress::Ip); | ||||
|     let state_ch = ch_runner.state_runner(); | ||||
|     state_ch.set_link_state(ch::driver::LinkState::Up); | ||||
| 
 | ||||
|     let runner = Runner { | ||||
|         ch: ch_runner, | ||||
|         state: state_inner, | ||||
|         trace_writer, | ||||
|     }; | ||||
| 
 | ||||
|     (device, control, runner) | ||||
| } | ||||
| 
 | ||||
| /// Shared state for the drivver.
 | ||||
| pub struct State { | ||||
|     ch: ch::State<MTU, 4, 4>, | ||||
|     inner: MaybeUninit<RefCell<StateInner>>, | ||||
| } | ||||
| 
 | ||||
| impl State { | ||||
|     /// Create a new State.
 | ||||
|     pub const fn new() -> Self { | ||||
|         Self { | ||||
|             ch: ch::State::new(), | ||||
|             inner: MaybeUninit::uninit(), | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| const TX_BUF_COUNT: usize = 4; | ||||
| const TX_BUF_SIZE: usize = 1024; | ||||
| 
 | ||||
| struct TraceChannelInfo { | ||||
|     ptr: *mut TraceChannel, | ||||
|     start: *mut u8, | ||||
|     end: *mut u8, | ||||
| } | ||||
| 
 | ||||
| const REQ_COUNT: usize = 4; | ||||
| 
 | ||||
| struct PendingRequest { | ||||
|     req_serial: u32, | ||||
|     resp_msg: *mut Message, | ||||
|     waker: Waker, | ||||
| } | ||||
| 
 | ||||
| struct StateInner { | ||||
|     init: bool, | ||||
|     init_waker: WakerRegistration, | ||||
| 
 | ||||
|     cb: *mut ControlBlock, | ||||
|     requests: [Option<PendingRequest>; REQ_COUNT], | ||||
|     next_req_serial: u32, | ||||
| 
 | ||||
|     rx_control_list: *mut List, | ||||
|     rx_data_list: *mut List, | ||||
|     rx_seq_no: u16, | ||||
|     rx_check: PointerChecker, | ||||
| 
 | ||||
|     tx_seq_no: u16, | ||||
|     tx_buf_used: [bool; TX_BUF_COUNT], | ||||
| 
 | ||||
|     trace_chans: Vec<TraceChannelInfo, TRACE_CHANNEL_COUNT>, | ||||
|     trace_check: PointerChecker, | ||||
| } | ||||
| 
 | ||||
| impl StateInner { | ||||
|     fn poll(&mut self, trace_writer: &mut impl embedded_io::Write, ch: &mut ch::Runner<MTU>) { | ||||
|         trace!("poll!"); | ||||
|         let ipc = unsafe { &*pac::IPC_NS::ptr() }; | ||||
| 
 | ||||
|         if ipc.events_receive[0].read().bits() != 0 { | ||||
|             ipc.events_receive[0].reset(); | ||||
|             trace!("ipc 0"); | ||||
|         } | ||||
| 
 | ||||
|         if ipc.events_receive[2].read().bits() != 0 { | ||||
|             ipc.events_receive[2].reset(); | ||||
|             trace!("ipc 2"); | ||||
| 
 | ||||
|             if !self.init { | ||||
|                 let desc = unsafe { addr_of!((*self.cb).modem_info).read_volatile() }; | ||||
|                 assert_eq!(desc.version, 1); | ||||
| 
 | ||||
|                 self.rx_check.check_mut(desc.control_list_ptr); | ||||
|                 self.rx_check.check_mut(desc.data_list_ptr); | ||||
| 
 | ||||
|                 self.rx_control_list = desc.control_list_ptr; | ||||
|                 self.rx_data_list = desc.data_list_ptr; | ||||
|                 let rx_control_len = unsafe { addr_of!((*self.rx_control_list).len).read_volatile() }; | ||||
|                 let rx_data_len = unsafe { addr_of!((*self.rx_data_list).len).read_volatile() }; | ||||
|                 assert_eq!(rx_control_len, LIST_LEN); | ||||
|                 assert_eq!(rx_data_len, LIST_LEN); | ||||
|                 self.init = true; | ||||
| 
 | ||||
|                 debug!("IPC initialized OK!"); | ||||
|                 self.init_waker.wake(); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         if ipc.events_receive[4].read().bits() != 0 { | ||||
|             ipc.events_receive[4].reset(); | ||||
|             trace!("ipc 4"); | ||||
| 
 | ||||
|             loop { | ||||
|                 let list = unsafe { &mut *self.rx_control_list }; | ||||
|                 let control_work = self.process(list, true, ch); | ||||
|                 let list = unsafe { &mut *self.rx_data_list }; | ||||
|                 let data_work = self.process(list, false, ch); | ||||
|                 if !control_work && !data_work { | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         if ipc.events_receive[6].read().bits() != 0 { | ||||
|             ipc.events_receive[6].reset(); | ||||
|             trace!("ipc 6"); | ||||
|         } | ||||
| 
 | ||||
|         if ipc.events_receive[7].read().bits() != 0 { | ||||
|             ipc.events_receive[7].reset(); | ||||
|             trace!("ipc 7: trace"); | ||||
| 
 | ||||
|             let msg = unsafe { addr_of!((*self.cb).trace.rx_state).read_volatile() }; | ||||
|             if msg != 0 { | ||||
|                 trace!("trace msg {}", msg); | ||||
|                 match msg { | ||||
|                     0 => unreachable!(), | ||||
|                     1 => { | ||||
|                         let ctx = unsafe { addr_of!((*self.cb).trace.rx_ptr).read_volatile() } as *mut TraceContext; | ||||
|                         debug!("trace init: {:?}", ctx); | ||||
|                         self.trace_check.check(ctx); | ||||
|                         let chans = unsafe { addr_of!((*ctx).chans).read_volatile() }; | ||||
|                         for chan_ptr in chans { | ||||
|                             let chan = self.trace_check.check_read(chan_ptr); | ||||
|                             self.trace_check.check(chan.start); | ||||
|                             self.trace_check.check(chan.end); | ||||
|                             assert!(chan.start < chan.end); | ||||
|                             self.trace_chans | ||||
|                                 .push(TraceChannelInfo { | ||||
|                                     ptr: chan_ptr, | ||||
|                                     start: chan.start, | ||||
|                                     end: chan.end, | ||||
|                                 }) | ||||
|                                 .map_err(|_| ()) | ||||
|                                 .unwrap() | ||||
|                         } | ||||
|                     } | ||||
|                     2 => { | ||||
|                         for chan_info in &self.trace_chans { | ||||
|                             let read_ptr = unsafe { addr_of!((*chan_info.ptr).read_ptr).read_volatile() }; | ||||
|                             let write_ptr = unsafe { addr_of!((*chan_info.ptr).write_ptr).read_volatile() }; | ||||
|                             assert!(read_ptr >= chan_info.start && read_ptr <= chan_info.end); | ||||
|                             assert!(write_ptr >= chan_info.start && write_ptr <= chan_info.end); | ||||
|                             if read_ptr != write_ptr { | ||||
|                                 let id = unsafe { addr_of!((*chan_info.ptr).id).read_volatile() }; | ||||
|                                 fence(Ordering::SeqCst); // synchronize volatile accesses with the slice access.
 | ||||
|                                 if read_ptr < write_ptr { | ||||
|                                     Self::handle_trace(trace_writer, id, unsafe { | ||||
|                                         slice::from_raw_parts(read_ptr, write_ptr.offset_from(read_ptr) as _) | ||||
|                                     }); | ||||
|                                 } else { | ||||
|                                     Self::handle_trace(trace_writer, id, unsafe { | ||||
|                                         slice::from_raw_parts(read_ptr, chan_info.end.offset_from(read_ptr) as _) | ||||
|                                     }); | ||||
|                                     Self::handle_trace(trace_writer, id, unsafe { | ||||
|                                         slice::from_raw_parts( | ||||
|                                             chan_info.start, | ||||
|                                             write_ptr.offset_from(chan_info.start) as _, | ||||
|                                         ) | ||||
|                                     }); | ||||
|                                 } | ||||
|                                 fence(Ordering::SeqCst); // synchronize volatile accesses with the slice access.
 | ||||
|                                 unsafe { addr_of_mut!((*chan_info.ptr).read_ptr).write_volatile(write_ptr) }; | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                     _ => warn!("unknown trace msg {}", msg), | ||||
|                 } | ||||
|                 unsafe { addr_of_mut!((*self.cb).trace.rx_state).write_volatile(0) }; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         ipc.intenset.write(|w| { | ||||
|             w.receive0().set_bit(); | ||||
|             w.receive2().set_bit(); | ||||
|             w.receive4().set_bit(); | ||||
|             w.receive6().set_bit(); | ||||
|             w.receive7().set_bit(); | ||||
|             w | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     fn handle_trace(writer: &mut impl embedded_io::Write, id: u8, data: &[u8]) { | ||||
|         trace!("trace: {} {}", id, data.len()); | ||||
|         let mut header = [0u8; 5]; | ||||
|         header[0] = 0xEF; | ||||
|         header[1] = 0xBE; | ||||
|         header[2..4].copy_from_slice(&(data.len() as u16).to_le_bytes()); | ||||
|         header[4] = id; | ||||
|         writer.write_all(&header).unwrap(); | ||||
|         writer.write_all(data).unwrap(); | ||||
|     } | ||||
| 
 | ||||
|     fn process(&mut self, list: *mut List, is_control: bool, ch: &mut ch::Runner<MTU>) -> bool { | ||||
|         let mut did_work = false; | ||||
|         for i in 0..LIST_LEN { | ||||
|             let item_ptr = unsafe { addr_of_mut!((*list).items[i]) }; | ||||
|             let preamble = unsafe { addr_of!((*item_ptr).state).read_volatile() }; | ||||
|             if preamble & 0xFF == 0x01 && preamble >> 16 == self.rx_seq_no as u32 { | ||||
|                 let msg_ptr = unsafe { addr_of!((*item_ptr).message).read_volatile() }; | ||||
|                 let msg = self.rx_check.check_read(msg_ptr); | ||||
| 
 | ||||
|                 debug!("rx seq {} msg: {:?}", preamble >> 16, msg); | ||||
| 
 | ||||
|                 if is_control { | ||||
|                     self.handle_control(&msg); | ||||
|                 } else { | ||||
|                     self.handle_data(&msg, ch); | ||||
|                 } | ||||
| 
 | ||||
|                 unsafe { addr_of_mut!((*item_ptr).state).write_volatile(0x03) }; | ||||
|                 self.rx_seq_no = self.rx_seq_no.wrapping_add(1); | ||||
| 
 | ||||
|                 did_work = true; | ||||
|             } | ||||
|         } | ||||
|         did_work | ||||
|     } | ||||
| 
 | ||||
|     fn find_free_message(&mut self, ch: usize) -> Option<usize> { | ||||
|         for i in 0..LIST_LEN { | ||||
|             let preamble = unsafe { addr_of!((*self.cb).lists[ch].items[i].state).read_volatile() }; | ||||
|             if matches!(preamble & 0xFF, 0 | 3) { | ||||
|                 trace!("using tx msg idx {}", i); | ||||
|                 return Some(i); | ||||
|             } | ||||
|         } | ||||
|         return None; | ||||
|     } | ||||
| 
 | ||||
|     fn find_free_tx_buf(&mut self) -> Option<usize> { | ||||
|         for i in 0..TX_BUF_COUNT { | ||||
|             if !self.tx_buf_used[i] { | ||||
|                 trace!("using tx buf idx {}", i); | ||||
|                 return Some(i); | ||||
|             } | ||||
|         } | ||||
|         return None; | ||||
|     } | ||||
| 
 | ||||
|     fn send_message(&mut self, msg: &mut Message, data: &[u8]) { | ||||
|         if data.is_empty() { | ||||
|             msg.data = ptr::null_mut(); | ||||
|             msg.data_len = 0; | ||||
|         } else { | ||||
|             assert!(data.len() <= TX_BUF_SIZE); | ||||
|             let buf_idx = self.find_free_tx_buf().unwrap(); // TODO handle out of bufs
 | ||||
|             let buf = unsafe { addr_of_mut!((*self.cb).tx_bufs[buf_idx]) } as *mut u8; | ||||
|             unsafe { copy_nonoverlapping(data.as_ptr(), buf, data.len()) } | ||||
|             msg.data = buf; | ||||
|             msg.data_len = data.len(); | ||||
|             self.tx_buf_used[buf_idx] = true; | ||||
| 
 | ||||
|             fence(Ordering::SeqCst); // synchronize copy_nonoverlapping (non-volatile) with volatile writes below.
 | ||||
|         } | ||||
| 
 | ||||
|         // TODO free data buf if send_message_raw fails.
 | ||||
|         self.send_message_raw(msg); | ||||
|     } | ||||
| 
 | ||||
|     fn send_message_raw(&mut self, msg: &Message) { | ||||
|         let (ch, ipc_ch) = match msg.channel { | ||||
|             1 => (0, 1), // control
 | ||||
|             2 => (1, 3), // data
 | ||||
|             _ => unreachable!(), | ||||
|         }; | ||||
| 
 | ||||
|         // allocate a msg.
 | ||||
|         let idx = self.find_free_message(ch).unwrap(); // TODO handle list full
 | ||||
| 
 | ||||
|         debug!("tx seq {} msg: {:?}", self.tx_seq_no, msg); | ||||
| 
 | ||||
|         let msg_slot = unsafe { addr_of_mut!((*self.cb).msgs[ch][idx]) }; | ||||
|         unsafe { msg_slot.write_volatile(*msg) } | ||||
|         let list_item = unsafe { addr_of_mut!((*self.cb).lists[ch].items[idx]) }; | ||||
|         unsafe { addr_of_mut!((*list_item).message).write_volatile(msg_slot) } | ||||
|         unsafe { addr_of_mut!((*list_item).state).write_volatile((self.tx_seq_no as u32) << 16 | 0x01) } | ||||
|         self.tx_seq_no = self.tx_seq_no.wrapping_add(1); | ||||
| 
 | ||||
|         let ipc = unsafe { &*pac::IPC_NS::ptr() }; | ||||
|         ipc.tasks_send[ipc_ch].write(|w| unsafe { w.bits(1) }); | ||||
|     } | ||||
| 
 | ||||
|     fn handle_control(&mut self, msg: &Message) { | ||||
|         match msg.id >> 16 { | ||||
|             1 => debug!("control msg: modem ready"), | ||||
|             2 => self.handle_control_free(msg.data), | ||||
|             _ => warn!("unknown control message id {:08x}", msg.id), | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fn handle_control_free(&mut self, ptr: *mut u8) { | ||||
|         let base = unsafe { addr_of!((*self.cb).tx_bufs) } as usize; | ||||
|         let ptr = ptr as usize; | ||||
| 
 | ||||
|         if ptr < base { | ||||
|             warn!("control free bad pointer {:08x}", ptr); | ||||
|             return; | ||||
|         } | ||||
| 
 | ||||
|         let diff = ptr - base; | ||||
|         let idx = diff / TX_BUF_SIZE; | ||||
| 
 | ||||
|         if idx >= TX_BUF_COUNT || idx * TX_BUF_SIZE != diff { | ||||
|             warn!("control free bad pointer {:08x}", ptr); | ||||
|             return; | ||||
|         } | ||||
| 
 | ||||
|         trace!("control free pointer {:08x} idx {}", ptr, idx); | ||||
|         if !self.tx_buf_used[idx] { | ||||
|             warn!( | ||||
|                 "control free pointer {:08x} idx {}: buffer was already free??", | ||||
|                 ptr, idx | ||||
|             ); | ||||
|         } | ||||
|         self.tx_buf_used[idx] = false; | ||||
|     } | ||||
| 
 | ||||
|     fn handle_data(&mut self, msg: &Message, ch: &mut ch::Runner<MTU>) { | ||||
|         if !msg.data.is_null() { | ||||
|             self.rx_check.check_length(msg.data, msg.data_len); | ||||
|         } | ||||
| 
 | ||||
|         let freed = match msg.id & 0xFFFF { | ||||
|             // AT
 | ||||
|             3 => { | ||||
|                 match msg.id >> 16 { | ||||
|                     // AT request ack
 | ||||
|                     2 => false, | ||||
|                     // AT response
 | ||||
|                     3 => self.handle_resp(msg), | ||||
|                     // AT notification
 | ||||
|                     4 => false, | ||||
|                     x => { | ||||
|                         warn!("received unknown AT kind {}", x); | ||||
|                         false | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             // IP
 | ||||
|             4 => { | ||||
|                 match msg.id >> 28 { | ||||
|                     // IP response
 | ||||
|                     8 => self.handle_resp(msg), | ||||
|                     // IP notification
 | ||||
|                     9 => match (msg.id >> 16) & 0xFFF { | ||||
|                         // IP receive notification
 | ||||
|                         1 => { | ||||
|                             if let Some(buf) = ch.try_rx_buf() { | ||||
|                                 let mut len = msg.data_len; | ||||
|                                 if len > buf.len() { | ||||
|                                     warn!("truncating rx'd packet from {} to {} bytes", len, buf.len()); | ||||
|                                     len = buf.len(); | ||||
|                                 } | ||||
|                                 fence(Ordering::SeqCst); // synchronize volatile accesses with the nonvolatile copy_nonoverlapping.
 | ||||
|                                 unsafe { ptr::copy_nonoverlapping(msg.data, buf.as_mut_ptr(), len) } | ||||
|                                 fence(Ordering::SeqCst); // synchronize volatile accesses with the nonvolatile copy_nonoverlapping.
 | ||||
|                                 ch.rx_done(len); | ||||
|                             } | ||||
|                             false | ||||
|                         } | ||||
|                         _ => false, | ||||
|                     }, | ||||
|                     x => { | ||||
|                         warn!("received unknown IP kind {}", x); | ||||
|                         false | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             x => { | ||||
|                 warn!("received unknown kind {}", x); | ||||
|                 false | ||||
|             } | ||||
|         }; | ||||
| 
 | ||||
|         if !freed { | ||||
|             self.send_free(msg); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fn handle_resp(&mut self, msg: &Message) -> bool { | ||||
|         let req_serial = u32::from_le_bytes(msg.param[0..4].try_into().unwrap()); | ||||
|         if req_serial == 0 { | ||||
|             return false; | ||||
|         } | ||||
| 
 | ||||
|         for optr in &mut self.requests { | ||||
|             if let Some(r) = optr { | ||||
|                 if r.req_serial == req_serial { | ||||
|                     let r = optr.take().unwrap(); | ||||
|                     unsafe { r.resp_msg.write(*msg) } | ||||
|                     r.waker.wake(); | ||||
|                     *optr = None; | ||||
|                     return true; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         warn!( | ||||
|             "resp with id {} serial {} doesn't match any pending req", | ||||
|             msg.id, req_serial | ||||
|         ); | ||||
|         false | ||||
|     } | ||||
| 
 | ||||
|     fn send_free(&mut self, msg: &Message) { | ||||
|         if msg.data.is_null() { | ||||
|             return; | ||||
|         } | ||||
| 
 | ||||
|         let mut free_msg: Message = unsafe { mem::zeroed() }; | ||||
|         free_msg.channel = 1; // control
 | ||||
|         free_msg.id = 0x20001; // free
 | ||||
|         free_msg.data = msg.data; | ||||
|         free_msg.data_len = msg.data_len; | ||||
| 
 | ||||
|         self.send_message_raw(&free_msg); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| struct PointerChecker { | ||||
|     start: *mut u8, | ||||
|     end: *mut u8, | ||||
| } | ||||
| 
 | ||||
| impl PointerChecker { | ||||
|     // check the pointer is in bounds in the arena, panic otherwise.
 | ||||
|     fn check_length(&self, ptr: *const u8, len: usize) { | ||||
|         assert!(ptr as usize >= self.start as usize); | ||||
|         let end_ptr = (ptr as usize).checked_add(len).unwrap(); | ||||
|         assert!(end_ptr <= self.end as usize); | ||||
|     } | ||||
| 
 | ||||
|     // check the pointer is in bounds in the arena, panic otherwise.
 | ||||
|     fn check<T>(&self, ptr: *const T) { | ||||
|         assert!(ptr.is_aligned()); | ||||
|         self.check_length(ptr as *const u8, mem::size_of::<T>()); | ||||
|     } | ||||
| 
 | ||||
|     // check the pointer is in bounds in the arena, panic otherwise.
 | ||||
|     fn check_read<T>(&self, ptr: *const T) -> T { | ||||
|         self.check(ptr); | ||||
|         unsafe { ptr.read_volatile() } | ||||
|     } | ||||
| 
 | ||||
|     // check the pointer is in bounds in the arena, panic otherwise.
 | ||||
|     fn check_mut<T>(&self, ptr: *mut T) { | ||||
|         self.check(ptr as *const T) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /// Control handle for the driver.
 | ||||
| ///
 | ||||
| /// You can use this object to control the modem at runtime, such as running AT commands.
 | ||||
| pub struct Control<'a> { | ||||
|     state: &'a RefCell<StateInner>, | ||||
| } | ||||
| 
 | ||||
| impl<'a> Control<'a> { | ||||
|     /// Wait for modem IPC to be initialized.
 | ||||
|     pub async fn wait_init(&self) { | ||||
|         poll_fn(|cx| { | ||||
|             let mut state = self.state.borrow_mut(); | ||||
|             if state.init { | ||||
|                 return Poll::Ready(()); | ||||
|             } | ||||
|             state.init_waker.register(cx.waker()); | ||||
|             Poll::Pending | ||||
|         }) | ||||
|         .await | ||||
|     } | ||||
| 
 | ||||
|     async fn request(&self, msg: &mut Message, req_data: &[u8], resp_data: &mut [u8]) -> usize { | ||||
|         // get waker
 | ||||
|         let waker = poll_fn(|cx| Poll::Ready(cx.waker().clone())).await; | ||||
| 
 | ||||
|         // Send request
 | ||||
|         let mut state = self.state.borrow_mut(); | ||||
|         let mut req_serial = state.next_req_serial; | ||||
|         if msg.id & 0xFFFF == 3 { | ||||
|             // AT response seems to keep only the lower 8 bits. Others do keep the full 32 bits..??
 | ||||
|             req_serial &= 0xFF; | ||||
|         } | ||||
| 
 | ||||
|         // increment next_req_serial, skip zero because we use it as an "ignore" value.
 | ||||
|         // We have to skip when the *lowest byte* is zero because AT responses.
 | ||||
|         state.next_req_serial = state.next_req_serial.wrapping_add(1); | ||||
|         if state.next_req_serial & 0xFF == 0 { | ||||
|             state.next_req_serial = state.next_req_serial.wrapping_add(1); | ||||
|         } | ||||
| 
 | ||||
|         msg.param[0..4].copy_from_slice(&req_serial.to_le_bytes()); | ||||
|         state.send_message(msg, req_data); | ||||
| 
 | ||||
|         // Setup the pending request state.
 | ||||
|         let (req_slot_idx, req_slot) = state | ||||
|             .requests | ||||
|             .iter_mut() | ||||
|             .enumerate() | ||||
|             .find(|(_, x)| x.is_none()) | ||||
|             .unwrap(); | ||||
|         msg.id = 0; // zero out id, so when it becomes nonzero we know the req is done.
 | ||||
|         let msg_ptr: *mut Message = msg; | ||||
|         *req_slot = Some(PendingRequest { | ||||
|             req_serial, | ||||
|             resp_msg: msg_ptr, | ||||
|             waker, | ||||
|         }); | ||||
| 
 | ||||
|         drop(state); // don't borrow state across awaits.
 | ||||
| 
 | ||||
|         // On cancel, unregister the request slot.
 | ||||
|         let _drop = OnDrop::new(|| { | ||||
|             // Remove request slot.
 | ||||
|             let mut state = self.state.borrow_mut(); | ||||
|             let slot = &mut state.requests[req_slot_idx]; | ||||
|             if let Some(s) = slot { | ||||
|                 if s.req_serial == req_serial { | ||||
|                     *slot = None; | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
|             // If cancelation raced with actually receiving the response,
 | ||||
|             // we own the data, so we have to free it.
 | ||||
|             let msg = unsafe { &mut *msg_ptr }; | ||||
|             if msg.id != 0 { | ||||
|                 state.send_free(msg); | ||||
|             } | ||||
|         }); | ||||
|         // Wait for response.
 | ||||
|         poll_fn(|_| { | ||||
|             // we have to use the raw pointer and not the original reference `msg`
 | ||||
|             // because that'd invalidate the raw ptr that's still stored in `req_slot`.
 | ||||
|             if unsafe { (*msg_ptr).id } != 0 { | ||||
|                 Poll::Ready(()) | ||||
|             } else { | ||||
|                 Poll::Pending | ||||
|             } | ||||
|         }) | ||||
|         .await; | ||||
|         _drop.defuse(); | ||||
| 
 | ||||
|         if msg.data.is_null() { | ||||
|             // no response data.
 | ||||
|             return 0; | ||||
|         } | ||||
| 
 | ||||
|         // Copy response data out, if any.
 | ||||
|         // Pointer was validated in StateInner::handle_data().
 | ||||
|         let mut len = msg.data_len; | ||||
|         if len > resp_data.len() { | ||||
|             warn!("truncating response data from {} to {}", len, resp_data.len()); | ||||
|             len = resp_data.len(); | ||||
|         } | ||||
|         fence(Ordering::SeqCst); // synchronize volatile accesses with the nonvolatile copy_nonoverlapping.
 | ||||
|         unsafe { ptr::copy_nonoverlapping(msg.data, resp_data.as_mut_ptr(), len) } | ||||
|         fence(Ordering::SeqCst); // synchronize volatile accesses with the nonvolatile copy_nonoverlapping.
 | ||||
|         self.state.borrow_mut().send_free(msg); | ||||
|         len | ||||
|     } | ||||
| 
 | ||||
|     /// Run an AT command.
 | ||||
|     ///
 | ||||
|     /// The response is written in `resp` and its length returned.
 | ||||
|     pub async fn at_command(&self, req: &[u8], resp: &mut [u8]) -> usize { | ||||
|         let mut msg: Message = unsafe { mem::zeroed() }; | ||||
|         msg.channel = 2; // data
 | ||||
|         msg.id = 0x0001_0003; // AT command
 | ||||
|         msg.param_len = 4; | ||||
| 
 | ||||
|         self.request(&mut msg, req, resp).await | ||||
|     } | ||||
| 
 | ||||
|     /// Open the raw socket used for sending/receiving IP packets.
 | ||||
|     ///
 | ||||
|     /// This must be done after `AT+CFUN=1` (?)
 | ||||
|     pub async fn open_raw_socket(&self) { | ||||
|         let mut msg: Message = unsafe { mem::zeroed() }; | ||||
|         msg.channel = 2; // data
 | ||||
|         msg.id = 0x7001_0004; // open socket
 | ||||
|         msg.param_len = 20; | ||||
| 
 | ||||
|         let param = [ | ||||
|             0xFF, 0xFF, 0xFF, 0xFF, // req_serial
 | ||||
|             0xFF, 0xFF, 0xFF, 0xFF, // ???
 | ||||
|             0x05, 0x00, 0x00, 0x00, // family
 | ||||
|             0x03, 0x00, 0x00, 0x00, // type
 | ||||
|             0x00, 0x00, 0x00, 0x00, // protocol
 | ||||
|         ]; | ||||
|         msg.param[..param.len()].copy_from_slice(¶m); | ||||
| 
 | ||||
|         self.request(&mut msg, &[], &mut []).await; | ||||
| 
 | ||||
|         assert_eq!(msg.id, 0x80010004); | ||||
|         assert!(msg.param_len >= 12); | ||||
|         let status = u32::from_le_bytes(msg.param[8..12].try_into().unwrap()); | ||||
|         assert_eq!(status, 0); | ||||
|         assert_eq!(msg.param_len, 16); | ||||
|         let fd = u32::from_le_bytes(msg.param[12..16].try_into().unwrap()); | ||||
|         debug!("got FD: {}", fd); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| /// Background runner for the driver.
 | ||||
| pub struct Runner<'a, TW: embedded_io::Write> { | ||||
|     ch: ch::Runner<'a, MTU>, | ||||
|     state: &'a RefCell<StateInner>, | ||||
|     trace_writer: TW, | ||||
| } | ||||
| 
 | ||||
| impl<'a, TW: embedded_io::Write> Runner<'a, TW> { | ||||
|     /// Run the driver operation in the background.
 | ||||
|     ///
 | ||||
|     /// You must run this in a background task, concurrently with all network operations.
 | ||||
|     pub async fn run(mut self) -> ! { | ||||
|         poll_fn(|cx| { | ||||
|             WAKER.register(cx.waker()); | ||||
| 
 | ||||
|             let mut state = self.state.borrow_mut(); | ||||
|             state.poll(&mut self.trace_writer, &mut self.ch); | ||||
| 
 | ||||
|             if let Poll::Ready(buf) = self.ch.poll_tx_buf(cx) { | ||||
|                 let fd = 128u32; // TODO unhardcode
 | ||||
|                 let mut msg: Message = unsafe { mem::zeroed() }; | ||||
|                 msg.channel = 2; // data
 | ||||
|                 msg.id = 0x7006_0004; // IP send
 | ||||
|                 msg.param_len = 12; | ||||
|                 msg.param[4..8].copy_from_slice(&fd.to_le_bytes()); | ||||
|                 state.send_message(&mut msg, buf); | ||||
|                 self.ch.tx_done(); | ||||
|             } | ||||
| 
 | ||||
|             Poll::Pending | ||||
|         }) | ||||
|         .await | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| const LIST_LEN: usize = 16; | ||||
| 
 | ||||
| #[repr(C)] | ||||
| struct ControlBlock { | ||||
|     version: u32, | ||||
|     rx_base: *mut u8, | ||||
|     rx_size: usize, | ||||
|     control_list_ptr: *mut List, | ||||
|     data_list_ptr: *mut List, | ||||
|     modem_info_ptr: *mut ModemInfo, | ||||
|     trace_ptr: *mut Trace, | ||||
|     unk: u32, | ||||
| 
 | ||||
|     modem_info: ModemInfo, | ||||
|     trace: Trace, | ||||
| 
 | ||||
|     // 0 = control, 1 = data
 | ||||
|     lists: [List; 2], | ||||
|     msgs: [[Message; LIST_LEN]; 2], | ||||
| 
 | ||||
|     tx_bufs: [[u8; TX_BUF_SIZE]; TX_BUF_COUNT], | ||||
| } | ||||
| 
 | ||||
| #[repr(C)] | ||||
| struct ModemInfo { | ||||
|     version: u32, | ||||
|     control_list_ptr: *mut List, | ||||
|     data_list_ptr: *mut List, | ||||
|     padding: [u32; 5], | ||||
| } | ||||
| 
 | ||||
| #[repr(C)] | ||||
| struct Trace { | ||||
|     size: usize, | ||||
|     base: *mut u8, | ||||
|     tx_state: u32, | ||||
|     tx_ptr: *mut u8, | ||||
|     rx_state: u32, | ||||
|     rx_ptr: *mut u8, | ||||
|     unk1: u32, | ||||
|     unk2: u32, | ||||
| } | ||||
| 
 | ||||
| const TRACE_CHANNEL_COUNT: usize = 3; | ||||
| 
 | ||||
| #[repr(C)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| struct TraceContext { | ||||
|     unk1: u32, | ||||
|     unk2: u32, | ||||
|     len: u32, | ||||
|     chans: [*mut TraceChannel; TRACE_CHANNEL_COUNT], | ||||
| } | ||||
| 
 | ||||
| #[repr(C)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| struct TraceChannel { | ||||
|     id: u8, | ||||
|     unk1: u8, | ||||
|     unk2: u8, | ||||
|     unk3: u8, | ||||
|     write_ptr: *mut u8, | ||||
|     read_ptr: *mut u8, | ||||
|     start: *mut u8, | ||||
|     end: *mut u8, | ||||
| } | ||||
| 
 | ||||
| #[repr(C)] | ||||
| struct List { | ||||
|     len: usize, | ||||
|     items: [ListItem; LIST_LEN], | ||||
| } | ||||
| 
 | ||||
| #[repr(C)] | ||||
| struct ListItem { | ||||
|     /// top 16 bits: seqno
 | ||||
|     /// bottom 8 bits:
 | ||||
|     ///     0x01: sent
 | ||||
|     ///     0x02: held
 | ||||
|     ///     0x03: freed
 | ||||
|     state: u32, | ||||
|     message: *mut Message, | ||||
| } | ||||
| 
 | ||||
| #[repr(C)] | ||||
| #[derive(defmt::Format, Clone, Copy)] | ||||
| struct Message { | ||||
|     id: u32, | ||||
| 
 | ||||
|     /// 1 = control, 2 = data
 | ||||
|     channel: u8, | ||||
|     unk1: u8, | ||||
|     unk2: u8, | ||||
|     unk3: u8, | ||||
| 
 | ||||
|     data: *mut u8, | ||||
|     data_len: usize, | ||||
|     param_len: usize, | ||||
|     param: [u8; 44], | ||||
| } | ||||
| 
 | ||||
| struct OnDrop<F: FnOnce()> { | ||||
|     f: MaybeUninit<F>, | ||||
| } | ||||
| 
 | ||||
| impl<F: FnOnce()> OnDrop<F> { | ||||
|     pub fn new(f: F) -> Self { | ||||
|         Self { f: MaybeUninit::new(f) } | ||||
|     } | ||||
| 
 | ||||
|     pub fn defuse(self) { | ||||
|         mem::forget(self) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl<F: FnOnce()> Drop for OnDrop<F> { | ||||
|     fn drop(&mut self) { | ||||
|         unsafe { self.f.as_ptr().read()() } | ||||
|     } | ||||
| } | ||||
| @ -1,5 +1,6 @@ | ||||
| [target.'cfg(all(target_arch = "arm", target_os = "none"))'] | ||||
| runner = "probe-rs run --chip nRF9160_xxAA" | ||||
| # runner = "probe-rs run --chip nRF9160_xxAA" | ||||
| runner = [ "probe-rs", "run", "--chip=nRF9160_xxAA", "--always-print-stacktrace", "--log-format={t} {[{L}]%bold} {s}  {{c} {ff}:{l:1}%dimmed}" ] | ||||
| 
 | ||||
| [build] | ||||
| target = "thumbv8m.main-none-eabihf" | ||||
|  | ||||
| @ -8,6 +8,8 @@ license = "MIT OR Apache-2.0" | ||||
| embassy-executor = { version = "0.6.0", path = "../../embassy-executor", features = ["task-arena-size-32768", "arch-cortex-m", "executor-thread", "executor-interrupt", "defmt", "integrated-timers"] } | ||||
| embassy-time = { version = "0.3.2", path = "../../embassy-time", features = ["defmt", "defmt-timestamp-uptime"] } | ||||
| embassy-nrf = { version = "0.2.0", path = "../../embassy-nrf", features = ["defmt", "nrf9160-s", "time-driver-rtc1", "gpiote", "unstable-pac", "time"] } | ||||
| embassy-net-nrf91 = { version = "0.1.0", path = "../../embassy-net-nrf91", features = ["defmt"] } | ||||
| embassy-net = { version = "0.4.0", path = "../../embassy-net", features = ["defmt", "tcp", "proto-ipv4", "medium-ip"] } | ||||
| 
 | ||||
| defmt = "0.3" | ||||
| defmt-rtt = "0.4" | ||||
| @ -15,6 +17,9 @@ defmt-rtt = "0.4" | ||||
| cortex-m = { version = "0.7.6", features = ["inline-asm", "critical-section-single-core"] } | ||||
| cortex-m-rt = "0.7.0" | ||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||
| static_cell = { version = "2" } | ||||
| embedded-io = "0.6.1" | ||||
| embedded-io-async = { version = "0.6.1", features = ["defmt-03"] } | ||||
| 
 | ||||
| [profile.release] | ||||
| debug = 2 | ||||
|  | ||||
| @ -1,5 +1,9 @@ | ||||
| MEMORY | ||||
| { | ||||
|   FLASH                    : ORIGIN = 0x00000000, LENGTH = 1024K | ||||
|   RAM                      : ORIGIN = 0x20018000, LENGTH = 160K | ||||
|     FLASH : ORIGIN = 0x00000000, LENGTH = 1024K | ||||
|     RAM   : ORIGIN = 0x20010000, LENGTH = 192K | ||||
|     IPC   : ORIGIN = 0x20000000, LENGTH = 64K | ||||
| } | ||||
| 
 | ||||
| PROVIDE(__start_ipc = ORIGIN(IPC)); | ||||
| PROVIDE(__end_ipc   = ORIGIN(IPC) + LENGTH(IPC)); | ||||
|  | ||||
							
								
								
									
										250
									
								
								examples/nrf9160/src/bin/modem_tcp_client.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										250
									
								
								examples/nrf9160/src/bin/modem_tcp_client.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,250 @@ | ||||
| #![no_std] | ||||
| #![no_main] | ||||
| 
 | ||||
| use core::mem::MaybeUninit; | ||||
| use core::ptr::addr_of_mut; | ||||
| use core::str::FromStr; | ||||
| use core::{slice, str}; | ||||
| 
 | ||||
| use defmt::{assert, *}; | ||||
| use embassy_executor::Spawner; | ||||
| use embassy_net::{Ipv4Address, Ipv4Cidr, Stack, StackResources}; | ||||
| use embassy_net_nrf91::{Runner, State}; | ||||
| use embassy_nrf::buffered_uarte::{self, BufferedUarteTx}; | ||||
| use embassy_nrf::gpio::{AnyPin, Level, Output, OutputDrive, Pin}; | ||||
| use embassy_nrf::uarte::Baudrate; | ||||
| use embassy_nrf::{bind_interrupts, interrupt, peripherals, uarte}; | ||||
| use embassy_time::{Duration, Timer}; | ||||
| use embedded_io_async::Write; | ||||
| use static_cell::StaticCell; | ||||
| use {defmt_rtt as _, panic_probe as _}; | ||||
| 
 | ||||
| #[interrupt] | ||||
| fn IPC() { | ||||
|     embassy_net_nrf91::on_ipc_irq(); | ||||
| } | ||||
| 
 | ||||
| bind_interrupts!(struct Irqs { | ||||
|     UARTE0_SPIM0_SPIS0_TWIM0_TWIS0 => buffered_uarte::InterruptHandler<peripherals::SERIAL0>; | ||||
| }); | ||||
| 
 | ||||
| // embassy-net-nrf91 only supports blocking trace write for now.
 | ||||
| // We don't want to block packet processing with slow uart writes, so
 | ||||
| // we make an adapter that writes whatever fits in the buffer and drops
 | ||||
| // data if it's full.
 | ||||
| struct TraceWriter(BufferedUarteTx<'static, peripherals::SERIAL0>); | ||||
| 
 | ||||
| impl embedded_io::ErrorType for TraceWriter { | ||||
|     type Error = core::convert::Infallible; | ||||
| } | ||||
| 
 | ||||
| impl embedded_io::Write for TraceWriter { | ||||
|     fn write(&mut self, buf: &[u8]) -> Result<usize, Self::Error> { | ||||
|         let _ = self.0.try_write(buf); | ||||
|         Ok(buf.len()) | ||||
|     } | ||||
|     fn flush(&mut self) -> Result<(), Self::Error> { | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| #[embassy_executor::task] | ||||
| async fn modem_task(runner: Runner<'static, TraceWriter>) -> ! { | ||||
|     runner.run().await | ||||
| } | ||||
| 
 | ||||
| #[embassy_executor::task] | ||||
| async fn net_task(stack: &'static Stack<embassy_net_nrf91::NetDriver<'static>>) -> ! { | ||||
|     stack.run().await | ||||
| } | ||||
| 
 | ||||
| #[embassy_executor::task] | ||||
| async fn blink_task(pin: AnyPin) { | ||||
|     let mut led = Output::new(pin, Level::Low, OutputDrive::Standard); | ||||
|     loop { | ||||
|         led.set_high(); | ||||
|         Timer::after_millis(100).await; | ||||
|         led.set_low(); | ||||
|         Timer::after_millis(100).await; | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| extern "C" { | ||||
|     static __start_ipc: u8; | ||||
|     static __end_ipc: u8; | ||||
| } | ||||
| 
 | ||||
| #[embassy_executor::main] | ||||
| async fn main(spawner: Spawner) { | ||||
|     let p = embassy_nrf::init(Default::default()); | ||||
| 
 | ||||
|     info!("Hello World!"); | ||||
| 
 | ||||
|     unwrap!(spawner.spawn(blink_task(p.P0_02.degrade()))); | ||||
| 
 | ||||
|     let ipc_mem = unsafe { | ||||
|         let ipc_start = &__start_ipc as *const u8 as *mut MaybeUninit<u8>; | ||||
|         let ipc_end = &__end_ipc as *const u8 as *mut MaybeUninit<u8>; | ||||
|         let ipc_len = ipc_end.offset_from(ipc_start) as usize; | ||||
|         slice::from_raw_parts_mut(ipc_start, ipc_len) | ||||
|     }; | ||||
| 
 | ||||
|     static mut TRACE_BUF: [u8; 4096] = [0u8; 4096]; | ||||
|     let mut config = uarte::Config::default(); | ||||
|     config.baudrate = Baudrate::BAUD1M; | ||||
|     let trace_writer = TraceWriter(BufferedUarteTx::new( | ||||
|         //let trace_uart = BufferedUarteTx::new(
 | ||||
|         unsafe { peripherals::SERIAL0::steal() }, | ||||
|         Irqs, | ||||
|         unsafe { peripherals::P0_01::steal() }, | ||||
|         //unsafe { peripherals::P0_14::steal() },
 | ||||
|         config, | ||||
|         unsafe { &mut *addr_of_mut!(TRACE_BUF) }, | ||||
|     )); | ||||
| 
 | ||||
|     static STATE: StaticCell<State> = StaticCell::new(); | ||||
|     let (device, control, runner) = embassy_net_nrf91::new(STATE.init(State::new()), ipc_mem, trace_writer).await; | ||||
|     unwrap!(spawner.spawn(modem_task(runner))); | ||||
| 
 | ||||
|     let config = embassy_net::Config::default(); | ||||
| 
 | ||||
|     // Generate "random" seed. nRF91 has no RNG, TODO figure out something...
 | ||||
|     let seed = 123456; | ||||
| 
 | ||||
|     // Init network stack
 | ||||
|     static RESOURCES: StaticCell<StackResources<2>> = StaticCell::new(); | ||||
|     static STACK: StaticCell<Stack<embassy_net_nrf91::NetDriver<'static>>> = StaticCell::new(); | ||||
|     let stack = &*STACK.init(Stack::new( | ||||
|         device, | ||||
|         config, | ||||
|         RESOURCES.init(StackResources::<2>::new()), | ||||
|         seed, | ||||
|     )); | ||||
| 
 | ||||
|     unwrap!(spawner.spawn(net_task(stack))); | ||||
| 
 | ||||
|     control.wait_init().await; | ||||
|     info!("INIT OK"); | ||||
| 
 | ||||
|     let mut buf = [0u8; 256]; | ||||
| 
 | ||||
|     let n = control.at_command(b"AT+CFUN?", &mut buf).await; | ||||
|     info!("AT resp: '{}'", unsafe { str::from_utf8_unchecked(&buf[..n]) }); | ||||
| 
 | ||||
|     let n = control | ||||
|         .at_command(b"AT+CGDCONT=0,\"IP\",\"iot.nat.es\"", &mut buf) | ||||
|         .await; | ||||
|     info!("AT resp: '{}'", unsafe { str::from_utf8_unchecked(&buf[..n]) }); | ||||
|     let n = control | ||||
|         .at_command(b"AT+CGAUTH=0,1,\"orange\",\"orange\"", &mut buf) | ||||
|         .await; | ||||
|     info!("AT resp: '{}'", unsafe { str::from_utf8_unchecked(&buf[..n]) }); | ||||
| 
 | ||||
|     let n = control.at_command(b"AT+CFUN=1", &mut buf).await; | ||||
|     info!("AT resp: '{}'", unsafe { str::from_utf8_unchecked(&buf[..n]) }); | ||||
| 
 | ||||
|     info!("waiting for attach..."); | ||||
|     loop { | ||||
|         Timer::after_millis(500).await; | ||||
|         let n = control.at_command(b"AT+CGATT?", &mut buf).await; | ||||
|         let mut res = &buf[..n]; | ||||
|         pop_prefix(&mut res, b"+CGATT: "); | ||||
|         let res = split_field(&mut res); | ||||
|         info!("AT resp field: '{}'", unsafe { str::from_utf8_unchecked(res) }); | ||||
|         if res == b"1" { | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     let n = control.at_command(b"AT+CGPADDR=0", &mut buf).await; | ||||
|     let mut res = &buf[..n]; | ||||
|     pop_prefix(&mut res, b"+CGPADDR: 0,"); | ||||
|     let ip = split_field(&mut res); | ||||
|     let ip = Ipv4Address::from_str(unsafe { str::from_utf8_unchecked(ip) }).unwrap(); | ||||
|     info!("IP: '{}'", ip); | ||||
| 
 | ||||
|     info!("============== OPENING SOCKET"); | ||||
|     control.open_raw_socket().await; | ||||
| 
 | ||||
|     stack.set_config_v4(embassy_net::ConfigV4::Static(embassy_net::StaticConfigV4 { | ||||
|         address: Ipv4Cidr::new(ip, 32), | ||||
|         gateway: None, | ||||
|         dns_servers: Default::default(), | ||||
|     })); | ||||
| 
 | ||||
|     let mut rx_buffer = [0; 4096]; | ||||
|     let mut tx_buffer = [0; 4096]; | ||||
|     loop { | ||||
|         let mut socket = embassy_net::tcp::TcpSocket::new(stack, &mut rx_buffer, &mut tx_buffer); | ||||
|         socket.set_timeout(Some(Duration::from_secs(10))); | ||||
| 
 | ||||
|         info!("Connecting..."); | ||||
|         let host_addr = embassy_net::Ipv4Address::from_str("83.51.182.206").unwrap(); | ||||
|         if let Err(e) = socket.connect((host_addr, 8000)).await { | ||||
|             warn!("connect error: {:?}", e); | ||||
|             continue; | ||||
|         } | ||||
|         info!("Connected to {:?}", socket.remote_endpoint()); | ||||
| 
 | ||||
|         let msg = b"Hello world!\n"; | ||||
|         loop { | ||||
|             if let Err(e) = socket.write_all(msg).await { | ||||
|                 warn!("write error: {:?}", e); | ||||
|                 break; | ||||
|             } | ||||
|             info!("txd: {}", core::str::from_utf8(msg).unwrap()); | ||||
|             Timer::after_secs(1).await; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| fn is_whitespace(char: u8) -> bool { | ||||
|     match char { | ||||
|         b'\r' | b'\n' | b' ' => true, | ||||
|         _ => false, | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| fn is_separator(char: u8) -> bool { | ||||
|     match char { | ||||
|         b',' | b'\r' | b'\n' | b' ' => true, | ||||
|         _ => false, | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| fn split_field<'a>(data: &mut &'a [u8]) -> &'a [u8] { | ||||
|     while !data.is_empty() && is_whitespace(data[0]) { | ||||
|         *data = &data[1..]; | ||||
|     } | ||||
| 
 | ||||
|     if data.is_empty() { | ||||
|         return &[]; | ||||
|     } | ||||
| 
 | ||||
|     if data[0] == b'"' { | ||||
|         let data2 = &data[1..]; | ||||
|         let end = data2.iter().position(|&x| x == b'"').unwrap_or(data2.len()); | ||||
|         let field = &data2[..end]; | ||||
|         let mut rest = &data2[data2.len().min(end + 1)..]; | ||||
|         if rest.first() == Some(&b'\"') { | ||||
|             rest = &rest[1..]; | ||||
|         } | ||||
|         while !rest.is_empty() && is_separator(rest[0]) { | ||||
|             rest = &rest[1..]; | ||||
|         } | ||||
|         *data = rest; | ||||
|         field | ||||
|     } else { | ||||
|         let end = data.iter().position(|&x| is_separator(x)).unwrap_or(data.len()); | ||||
|         let field = &data[0..end]; | ||||
|         let rest = &data[data.len().min(end + 1)..]; | ||||
|         *data = rest; | ||||
|         field | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| fn pop_prefix(data: &mut &[u8], prefix: &[u8]) { | ||||
|     assert!(data.len() >= prefix.len()); | ||||
|     assert!(&data[..prefix.len()] == prefix); | ||||
|     *data = &data[prefix.len()..]; | ||||
| } | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user