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"))'] | [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] | [build] | ||||||
| target = "thumbv8m.main-none-eabihf" | 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-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-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-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 = "0.3" | ||||||
| defmt-rtt = "0.4" | 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 = { version = "0.7.6", features = ["inline-asm", "critical-section-single-core"] } | ||||||
| cortex-m-rt = "0.7.0" | cortex-m-rt = "0.7.0" | ||||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | 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] | [profile.release] | ||||||
| debug = 2 | debug = 2 | ||||||
|  | |||||||
| @ -1,5 +1,9 @@ | |||||||
| MEMORY | MEMORY | ||||||
| { | { | ||||||
|   FLASH                    : ORIGIN = 0x00000000, LENGTH = 1024K |     FLASH : ORIGIN = 0x00000000, LENGTH = 1024K | ||||||
|   RAM                      : ORIGIN = 0x20018000, LENGTH = 160K |     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