embassy/embassy-net/src/device.rs

130 lines
3.3 KiB
Rust
Raw Normal View History

2021-03-02 20:20:00 +00:00
use core::task::Waker;
2022-06-12 20:15:44 +00:00
use smoltcp::phy::{Device as SmolDevice, DeviceCapabilities};
2021-02-03 04:09:37 +00:00
use smoltcp::time::Instant as SmolInstant;
use crate::packet_pool::PacketBoxExt;
2021-03-02 20:20:00 +00:00
use crate::{Packet, PacketBox, PacketBuf};
2021-02-03 04:09:37 +00:00
#[derive(PartialEq, Eq, Clone, Copy)]
pub enum LinkState {
Down,
Up,
}
2022-05-23 01:50:43 +00:00
// 'static required due to the "fake GAT" in smoltcp::phy::Device.
// https://github.com/smoltcp-rs/smoltcp/pull/572
2021-02-03 04:09:37 +00:00
pub trait Device {
fn is_transmit_ready(&mut self) -> bool;
fn transmit(&mut self, pkt: PacketBuf);
fn receive(&mut self) -> Option<PacketBuf>;
fn register_waker(&mut self, waker: &Waker);
2022-05-23 01:50:43 +00:00
fn capabilities(&self) -> DeviceCapabilities;
2021-02-03 04:09:37 +00:00
fn link_state(&mut self) -> LinkState;
fn ethernet_address(&self) -> [u8; 6];
2021-02-03 04:09:37 +00:00
}
2022-05-23 01:50:43 +00:00
impl<T: ?Sized + Device> Device for &'static mut T {
fn is_transmit_ready(&mut self) -> bool {
T::is_transmit_ready(self)
}
fn transmit(&mut self, pkt: PacketBuf) {
T::transmit(self, pkt)
}
fn receive(&mut self) -> Option<PacketBuf> {
T::receive(self)
}
fn register_waker(&mut self, waker: &Waker) {
T::register_waker(self, waker)
}
fn capabilities(&self) -> DeviceCapabilities {
T::capabilities(self)
}
fn link_state(&mut self) -> LinkState {
T::link_state(self)
}
fn ethernet_address(&self) -> [u8; 6] {
T::ethernet_address(self)
}
}
pub struct DeviceAdapter<D: Device> {
pub device: D,
2021-02-03 04:09:37 +00:00
caps: DeviceCapabilities,
}
2022-05-23 01:50:43 +00:00
impl<D: Device> DeviceAdapter<D> {
pub(crate) fn new(device: D) -> Self {
2021-02-03 04:09:37 +00:00
Self {
caps: device.capabilities(),
device,
}
}
}
2022-05-23 01:50:43 +00:00
impl<'a, D: Device + 'static> SmolDevice<'a> for DeviceAdapter<D> {
2021-02-03 04:09:37 +00:00
type RxToken = RxToken;
2022-05-23 01:50:43 +00:00
type TxToken = TxToken<'a, D>;
2021-02-03 04:09:37 +00:00
fn receive(&'a mut self) -> Option<(Self::RxToken, Self::TxToken)> {
let tx_pkt = PacketBox::new(Packet::new())?;
2021-02-03 04:09:37 +00:00
let rx_pkt = self.device.receive()?;
let rx_token = RxToken { pkt: rx_pkt };
let tx_token = TxToken {
2022-05-23 01:50:43 +00:00
device: &mut self.device,
2021-02-03 04:09:37 +00:00
pkt: tx_pkt,
};
Some((rx_token, tx_token))
}
/// Construct a transmit token.
fn transmit(&'a mut self) -> Option<Self::TxToken> {
if !self.device.is_transmit_ready() {
return None;
}
let tx_pkt = PacketBox::new(Packet::new())?;
Some(TxToken {
2022-05-23 01:50:43 +00:00
device: &mut self.device,
2021-02-03 04:09:37 +00:00
pkt: tx_pkt,
})
}
/// Get a description of device capabilities.
fn capabilities(&self) -> DeviceCapabilities {
self.caps.clone()
}
}
pub struct RxToken {
pkt: PacketBuf,
}
impl smoltcp::phy::RxToken for RxToken {
2022-05-04 18:48:37 +00:00
fn consume<R, F>(mut self, _timestamp: SmolInstant, f: F) -> smoltcp::Result<R>
2021-02-03 04:09:37 +00:00
where
2022-05-04 18:48:37 +00:00
F: FnOnce(&mut [u8]) -> smoltcp::Result<R>,
2021-02-03 04:09:37 +00:00
{
f(&mut self.pkt)
}
}
2022-05-23 01:50:43 +00:00
pub struct TxToken<'a, D: Device> {
device: &'a mut D,
2021-02-03 04:09:37 +00:00
pkt: PacketBox,
}
2022-05-23 01:50:43 +00:00
impl<'a, D: Device> smoltcp::phy::TxToken for TxToken<'a, D> {
2022-05-04 18:48:37 +00:00
fn consume<R, F>(self, _timestamp: SmolInstant, len: usize, f: F) -> smoltcp::Result<R>
2021-02-03 04:09:37 +00:00
where
2022-05-04 18:48:37 +00:00
F: FnOnce(&mut [u8]) -> smoltcp::Result<R>,
2021-02-03 04:09:37 +00:00
{
let mut buf = self.pkt.slice(0..len);
let r = f(&mut buf)?;
self.device.transmit(buf);
Ok(r)
}
}