Merge pull request #1788 from embassy-rs/esp-hosted-cleanup

net-esp-hosted: misc improvements.
This commit is contained in:
Dario Nieuwenhuis 2023-08-18 14:21:10 +00:00 committed by GitHub
commit d327b626e3
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 111 additions and 71 deletions

View File

@ -5,9 +5,12 @@ use heapless::String;
use crate::ioctl::Shared; use crate::ioctl::Shared;
use crate::proto::{self, CtrlMsg}; use crate::proto::{self, CtrlMsg};
#[derive(Debug)] #[derive(Copy, Clone, PartialEq, Eq, Debug)]
pub struct Error { #[cfg_attr(feature = "defmt", derive(defmt::Format))]
pub status: u32, pub enum Error {
Failed(u32),
Timeout,
Internal,
} }
pub struct Control<'a> { pub struct Control<'a> {
@ -23,58 +26,78 @@ enum WifiMode {
ApSta = 3, ApSta = 3,
} }
macro_rules! ioctl {
($self:ident, $req_variant:ident, $resp_variant:ident, $req:ident, $resp:ident) => {
let mut msg = proto::CtrlMsg {
msg_id: proto::CtrlMsgId::$req_variant as _,
msg_type: proto::CtrlMsgType::Req as _,
payload: Some(proto::CtrlMsgPayload::$req_variant($req)),
};
$self.ioctl(&mut msg).await?;
let Some(proto::CtrlMsgPayload::$resp_variant($resp)) = msg.payload else {
warn!("unexpected response variant");
return Err(Error::Internal);
};
if $resp.resp != 0 {
return Err(Error::Failed($resp.resp));
}
};
}
impl<'a> Control<'a> { impl<'a> Control<'a> {
pub(crate) fn new(state_ch: ch::StateRunner<'a>, shared: &'a Shared) -> Self { pub(crate) fn new(state_ch: ch::StateRunner<'a>, shared: &'a Shared) -> Self {
Self { state_ch, shared } Self { state_ch, shared }
} }
pub async fn init(&mut self) { pub async fn init(&mut self) -> Result<(), Error> {
debug!("wait for init event..."); debug!("wait for init event...");
self.shared.init_wait().await; self.shared.init_wait().await;
debug!("set wifi mode"); debug!("set heartbeat");
self.set_wifi_mode(WifiMode::Sta as _).await; self.set_heartbeat(10).await?;
let mac_addr = self.get_mac_addr().await; debug!("set wifi mode");
self.set_wifi_mode(WifiMode::Sta as _).await?;
let mac_addr = self.get_mac_addr().await?;
debug!("mac addr: {:02x}", mac_addr); debug!("mac addr: {:02x}", mac_addr);
self.state_ch.set_ethernet_address(mac_addr); self.state_ch.set_ethernet_address(mac_addr);
Ok(())
} }
pub async fn join(&mut self, ssid: &str, password: &str) { pub async fn connect(&mut self, ssid: &str, password: &str) -> Result<(), Error> {
let req = proto::CtrlMsg { let req = proto::CtrlMsgReqConnectAp {
msg_id: proto::CtrlMsgId::ReqConnectAp as _,
msg_type: proto::CtrlMsgType::Req as _,
payload: Some(proto::CtrlMsgPayload::ReqConnectAp(proto::CtrlMsgReqConnectAp {
ssid: String::from(ssid), ssid: String::from(ssid),
pwd: String::from(password), pwd: String::from(password),
bssid: String::new(), bssid: String::new(),
listen_interval: 3, listen_interval: 3,
is_wpa3_supported: false, is_wpa3_supported: false,
})),
}; };
let resp = self.ioctl(req).await; ioctl!(self, ReqConnectAp, RespConnectAp, req, resp);
let proto::CtrlMsgPayload::RespConnectAp(resp) = resp.payload.unwrap() else {
panic!("unexpected resp")
};
assert_eq!(resp.resp, 0);
self.state_ch.set_link_state(LinkState::Up); self.state_ch.set_link_state(LinkState::Up);
Ok(())
} }
async fn get_mac_addr(&mut self) -> [u8; 6] { pub async fn disconnect(&mut self) -> Result<(), Error> {
let req = proto::CtrlMsg { let req = proto::CtrlMsgReqGetStatus {};
msg_id: proto::CtrlMsgId::ReqGetMacAddress as _, ioctl!(self, ReqDisconnectAp, RespDisconnectAp, req, resp);
msg_type: proto::CtrlMsgType::Req as _, self.state_ch.set_link_state(LinkState::Up);
payload: Some(proto::CtrlMsgPayload::ReqGetMacAddress( Ok(())
proto::CtrlMsgReqGetMacAddress { }
/// duration in seconds, clamped to [10, 3600]
async fn set_heartbeat(&mut self, duration: u32) -> Result<(), Error> {
let req = proto::CtrlMsgReqConfigHeartbeat { enable: true, duration };
ioctl!(self, ReqConfigHeartbeat, RespConfigHeartbeat, req, resp);
Ok(())
}
async fn get_mac_addr(&mut self) -> Result<[u8; 6], Error> {
let req = proto::CtrlMsgReqGetMacAddress {
mode: WifiMode::Sta as _, mode: WifiMode::Sta as _,
},
)),
}; };
let resp = self.ioctl(req).await; ioctl!(self, ReqGetMacAddress, RespGetMacAddress, req, resp);
let proto::CtrlMsgPayload::RespGetMacAddress(resp) = resp.payload.unwrap() else {
panic!("unexpected resp")
};
assert_eq!(resp.resp, 0);
// WHY IS THIS A STRING? WHYYYY // WHY IS THIS A STRING? WHYYYY
fn nibble_from_hex(b: u8) -> u8 { fn nibble_from_hex(b: u8) -> u8 {
@ -88,32 +111,32 @@ impl<'a> Control<'a> {
let mac = resp.mac.as_bytes(); let mac = resp.mac.as_bytes();
let mut res = [0; 6]; let mut res = [0; 6];
assert_eq!(mac.len(), 17); if mac.len() != 17 {
warn!("unexpected MAC respnse length");
return Err(Error::Internal);
}
for (i, b) in res.iter_mut().enumerate() { for (i, b) in res.iter_mut().enumerate() {
*b = (nibble_from_hex(mac[i * 3]) << 4) | nibble_from_hex(mac[i * 3 + 1]) *b = (nibble_from_hex(mac[i * 3]) << 4) | nibble_from_hex(mac[i * 3 + 1])
} }
res Ok(res)
} }
async fn set_wifi_mode(&mut self, mode: u32) { async fn set_wifi_mode(&mut self, mode: u32) -> Result<(), Error> {
let req = proto::CtrlMsg { let req = proto::CtrlMsgReqSetMode { mode };
msg_id: proto::CtrlMsgId::ReqSetWifiMode as _, ioctl!(self, ReqSetWifiMode, RespSetWifiMode, req, resp);
msg_type: proto::CtrlMsgType::Req as _,
payload: Some(proto::CtrlMsgPayload::ReqSetWifiMode(proto::CtrlMsgReqSetMode { mode })), Ok(())
};
let resp = self.ioctl(req).await;
let proto::CtrlMsgPayload::RespSetWifiMode(resp) = resp.payload.unwrap() else {
panic!("unexpected resp")
};
assert_eq!(resp.resp, 0);
} }
async fn ioctl(&mut self, req: CtrlMsg) -> CtrlMsg { async fn ioctl(&mut self, msg: &mut CtrlMsg) -> Result<(), Error> {
debug!("ioctl req: {:?}", &req); debug!("ioctl req: {:?}", &msg);
let mut buf = [0u8; 128]; let mut buf = [0u8; 128];
let req_len = noproto::write(&req, &mut buf).unwrap(); let req_len = noproto::write(msg, &mut buf).map_err(|_| {
warn!("failed to serialize control request");
Error::Internal
})?;
struct CancelOnDrop<'a>(&'a Shared); struct CancelOnDrop<'a>(&'a Shared);
@ -135,9 +158,12 @@ impl<'a> Control<'a> {
ioctl.defuse(); ioctl.defuse();
let res = noproto::read(&buf[..resp_len]).unwrap(); *msg = noproto::read(&buf[..resp_len]).map_err(|_| {
debug!("ioctl resp: {:?}", &res); warn!("failed to serialize control request");
Error::Internal
})?;
debug!("ioctl resp: {:?}", msg);
res Ok(())
} }
} }

View File

@ -1,17 +1,15 @@
#![no_std] #![no_std]
use control::Control; use embassy_futures::select::{select4, Either4};
use embassy_futures::select::{select3, Either3};
use embassy_net_driver_channel as ch; use embassy_net_driver_channel as ch;
use embassy_net_driver_channel::driver::LinkState;
use embassy_time::{Duration, Instant, Timer}; use embassy_time::{Duration, Instant, Timer};
use embedded_hal::digital::{InputPin, OutputPin}; use embedded_hal::digital::{InputPin, OutputPin};
use embedded_hal_async::digital::Wait; use embedded_hal_async::digital::Wait;
use embedded_hal_async::spi::SpiDevice; use embedded_hal_async::spi::SpiDevice;
use ioctl::Shared;
use proto::CtrlMsg;
use crate::ioctl::PendingIoctl; use crate::ioctl::{PendingIoctl, Shared};
use crate::proto::CtrlMsgPayload; use crate::proto::{CtrlMsg, CtrlMsgPayload};
mod proto; mod proto;
@ -21,6 +19,8 @@ mod fmt;
mod control; mod control;
mod ioctl; mod ioctl;
pub use control::*;
const MTU: usize = 1514; const MTU: usize = 1514;
macro_rules! impl_bytes { macro_rules! impl_bytes {
@ -95,6 +95,7 @@ enum InterfaceType {
} }
const MAX_SPI_BUFFER_SIZE: usize = 1600; const MAX_SPI_BUFFER_SIZE: usize = 1600;
const HEARTBEAT_MAX_GAP: Duration = Duration::from_secs(20);
pub struct State { pub struct State {
shared: Shared, shared: Shared,
@ -129,12 +130,14 @@ where
let mut runner = Runner { let mut runner = Runner {
ch: ch_runner, ch: ch_runner,
state_ch,
shared: &state.shared, shared: &state.shared,
next_seq: 1, next_seq: 1,
handshake, handshake,
ready, ready,
reset, reset,
spi, spi,
heartbeat_deadline: Instant::now() + HEARTBEAT_MAX_GAP,
}; };
runner.init().await; runner.init().await;
@ -143,9 +146,11 @@ where
pub struct Runner<'a, SPI, IN, OUT> { pub struct Runner<'a, SPI, IN, OUT> {
ch: ch::Runner<'a, MTU>, ch: ch::Runner<'a, MTU>,
state_ch: ch::StateRunner<'a>,
shared: &'a Shared, shared: &'a Shared,
next_seq: u16, next_seq: u16,
heartbeat_deadline: Instant,
spi: SPI, spi: SPI,
handshake: IN, handshake: IN,
@ -177,9 +182,10 @@ where
let ioctl = self.shared.ioctl_wait_pending(); let ioctl = self.shared.ioctl_wait_pending();
let tx = self.ch.tx_buf(); let tx = self.ch.tx_buf();
let ev = async { self.ready.wait_for_high().await.unwrap() }; let ev = async { self.ready.wait_for_high().await.unwrap() };
let hb = Timer::at(self.heartbeat_deadline);
match select3(ioctl, tx, ev).await { match select4(ioctl, tx, ev, hb).await {
Either3::First(PendingIoctl { buf, req_len }) => { Either4::First(PendingIoctl { buf, req_len }) => {
tx_buf[12..24].copy_from_slice(b"\x01\x08\x00ctrlResp\x02"); tx_buf[12..24].copy_from_slice(b"\x01\x08\x00ctrlResp\x02");
tx_buf[24..26].copy_from_slice(&(req_len as u16).to_le_bytes()); tx_buf[24..26].copy_from_slice(&(req_len as u16).to_le_bytes());
tx_buf[26..][..req_len].copy_from_slice(&unsafe { &*buf }[..req_len]); tx_buf[26..][..req_len].copy_from_slice(&unsafe { &*buf }[..req_len]);
@ -198,7 +204,7 @@ where
header.checksum = checksum(&tx_buf[..26 + req_len]); header.checksum = checksum(&tx_buf[..26 + req_len]);
tx_buf[0..12].copy_from_slice(&header.to_bytes()); tx_buf[0..12].copy_from_slice(&header.to_bytes());
} }
Either3::Second(packet) => { Either4::Second(packet) => {
tx_buf[12..][..packet.len()].copy_from_slice(packet); tx_buf[12..][..packet.len()].copy_from_slice(packet);
let mut header = PayloadHeader { let mut header = PayloadHeader {
@ -217,9 +223,12 @@ where
self.ch.tx_done(); self.ch.tx_done();
} }
Either3::Third(()) => { Either4::Third(()) => {
tx_buf[..PayloadHeader::SIZE].fill(0); tx_buf[..PayloadHeader::SIZE].fill(0);
} }
Either4::Fourth(()) => {
panic!("heartbeat from esp32 stopped")
}
} }
if tx_buf[0] != 0 { if tx_buf[0] != 0 {
@ -308,7 +317,7 @@ where
} }
} }
fn handle_event(&self, data: &[u8]) { fn handle_event(&mut self, data: &[u8]) {
let Ok(event) = noproto::read::<CtrlMsg>(data) else { let Ok(event) = noproto::read::<CtrlMsg>(data) else {
warn!("failed to parse event"); warn!("failed to parse event");
return; return;
@ -323,6 +332,11 @@ where
match payload { match payload {
CtrlMsgPayload::EventEspInit(_) => self.shared.init_done(), CtrlMsgPayload::EventEspInit(_) => self.shared.init_done(),
CtrlMsgPayload::EventHeartbeat(_) => self.heartbeat_deadline = Instant::now() + HEARTBEAT_MAX_GAP,
CtrlMsgPayload::EventStationDisconnectFromAp(e) => {
info!("disconnected, code {}", e.resp);
self.state_ch.set_link_state(LinkState::Down);
}
_ => {} _ => {}
} }
} }

View File

@ -72,8 +72,8 @@ async fn main(spawner: Spawner) {
unwrap!(spawner.spawn(wifi_task(runner))); unwrap!(spawner.spawn(wifi_task(runner)));
control.init().await; unwrap!(control.init().await);
control.join(WIFI_NETWORK, WIFI_PASSWORD).await; unwrap!(control.connect(WIFI_NETWORK, WIFI_PASSWORD).await);
let config = embassy_net::Config::dhcpv4(Default::default()); let config = embassy_net::Config::dhcpv4(Default::default());
// let config = embassy_net::Config::ipv4_static(embassy_net::StaticConfigV4 { // let config = embassy_net::Config::ipv4_static(embassy_net::StaticConfigV4 {

View File

@ -73,8 +73,8 @@ async fn main(spawner: Spawner) {
unwrap!(spawner.spawn(wifi_task(runner))); unwrap!(spawner.spawn(wifi_task(runner)));
control.init().await; unwrap!(control.init().await);
control.join(WIFI_NETWORK, WIFI_PASSWORD).await; unwrap!(control.connect(WIFI_NETWORK, WIFI_PASSWORD).await);
// Generate random seed // Generate random seed
let mut rng = Rng::new(p.RNG, Irqs); let mut rng = Rng::new(p.RNG, Irqs);