mirror of
https://github.com/embassy-rs/embassy.git
synced 2024-11-21 22:32:29 +00:00
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…
Reference in New Issue
Block a user