aboutsummaryrefslogtreecommitdiff
path: root/embassy-usb
diff options
context:
space:
mode:
authorDario Nieuwenhuis <[email protected]>2022-12-07 16:03:03 +0100
committerDario Nieuwenhuis <[email protected]>2022-12-13 16:43:25 +0100
commite9219405ca04e23b6543fb841fd97df54cf72f94 (patch)
tree32d9129286949fdb887898adad1076ab352e95e9 /embassy-usb
parentaaaf5f23a8a3a0df1ad2186802e2afc061a74b72 (diff)
usb/cdc-ncm: add embassy-net Device implementation.
Diffstat (limited to 'embassy-usb')
-rw-r--r--embassy-usb/Cargo.toml1
-rw-r--r--embassy-usb/src/class/cdc_ncm/embassy_net.rs449
-rw-r--r--embassy-usb/src/class/cdc_ncm/mod.rs (renamed from embassy-usb/src/class/cdc_ncm.rs)18
3 files changed, 468 insertions, 0 deletions
diff --git a/embassy-usb/Cargo.toml b/embassy-usb/Cargo.toml
index b59ba8a22..1e72ce682 100644
--- a/embassy-usb/Cargo.toml
+++ b/embassy-usb/Cargo.toml
@@ -19,6 +19,7 @@ default = ["usbd-hid"]
19embassy-futures = { version = "0.1.0", path = "../embassy-futures" } 19embassy-futures = { version = "0.1.0", path = "../embassy-futures" }
20embassy-usb-driver = { version = "0.1.0", path = "../embassy-usb-driver" } 20embassy-usb-driver = { version = "0.1.0", path = "../embassy-usb-driver" }
21embassy-sync = { version = "0.1.0", path = "../embassy-sync" } 21embassy-sync = { version = "0.1.0", path = "../embassy-sync" }
22embassy-net = { version = "0.1.0", path = "../embassy-net", optional = true }
22 23
23defmt = { version = "0.3", optional = true } 24defmt = { version = "0.3", optional = true }
24log = { version = "0.4.14", optional = true } 25log = { version = "0.4.14", optional = true }
diff --git a/embassy-usb/src/class/cdc_ncm/embassy_net.rs b/embassy-usb/src/class/cdc_ncm/embassy_net.rs
new file mode 100644
index 000000000..60bbfd8d4
--- /dev/null
+++ b/embassy-usb/src/class/cdc_ncm/embassy_net.rs
@@ -0,0 +1,449 @@
1use core::cell::RefCell;
2use core::mem::MaybeUninit;
3use core::task::Context;
4
5use embassy_futures::select::{select, Either};
6use embassy_net::device::{Device as DeviceTrait, DeviceCapabilities, LinkState, Medium};
7use embassy_sync::blocking_mutex::raw::NoopRawMutex;
8use embassy_sync::blocking_mutex::Mutex;
9use embassy_sync::waitqueue::WakerRegistration;
10use embassy_usb_driver::Driver;
11
12use super::{CdcNcmClass, Receiver, Sender};
13
14pub struct State<'d, const MTU: usize, const N_RX: usize, const N_TX: usize> {
15 rx: [PacketBuf<MTU>; N_RX],
16 tx: [PacketBuf<MTU>; N_TX],
17 inner: MaybeUninit<StateInner<'d, MTU>>,
18}
19
20impl<'d, const MTU: usize, const N_RX: usize, const N_TX: usize> State<'d, MTU, N_RX, N_TX> {
21 const NEW_PACKET: PacketBuf<MTU> = PacketBuf::new();
22
23 pub const fn new() -> Self {
24 Self {
25 rx: [Self::NEW_PACKET; N_RX],
26 tx: [Self::NEW_PACKET; N_TX],
27 inner: MaybeUninit::uninit(),
28 }
29 }
30}
31
32struct StateInner<'d, const MTU: usize> {
33 rx: zerocopy_channel::Channel<'d, NoopRawMutex, PacketBuf<MTU>>,
34 tx: zerocopy_channel::Channel<'d, NoopRawMutex, PacketBuf<MTU>>,
35 link_state: Mutex<NoopRawMutex, RefCell<LinkStateState>>,
36}
37
38/// State of the LinkState
39struct LinkStateState {
40 state: LinkState,
41 waker: WakerRegistration,
42}
43
44pub struct Runner<'d, D: Driver<'d>, const MTU: usize> {
45 tx_usb: Sender<'d, D>,
46 tx_chan: zerocopy_channel::Receiver<'d, NoopRawMutex, PacketBuf<MTU>>,
47 rx_usb: Receiver<'d, D>,
48 rx_chan: zerocopy_channel::Sender<'d, NoopRawMutex, PacketBuf<MTU>>,
49 link_state: &'d Mutex<NoopRawMutex, RefCell<LinkStateState>>,
50}
51
52impl<'d, D: Driver<'d>, const MTU: usize> Runner<'d, D, MTU> {
53 pub async fn run(mut self) -> ! {
54 let rx_fut = async move {
55 loop {
56 trace!("WAITING for connection");
57 self.link_state.lock(|s| {
58 let s = &mut *s.borrow_mut();
59 s.state = LinkState::Down;
60 s.waker.wake();
61 });
62
63 self.rx_usb.wait_connection().await.unwrap();
64
65 trace!("Connected");
66 self.link_state.lock(|s| {
67 let s = &mut *s.borrow_mut();
68 s.state = LinkState::Up;
69 s.waker.wake();
70 });
71
72 loop {
73 let p = self.rx_chan.send().await;
74 match self.rx_usb.read_packet(&mut p.buf).await {
75 Ok(n) => {
76 p.len = n;
77 self.rx_chan.send_done();
78 }
79 Err(e) => {
80 warn!("error reading packet: {:?}", e);
81 break;
82 }
83 };
84 }
85 }
86 };
87 let tx_fut = async move {
88 loop {
89 let p = self.tx_chan.recv().await;
90 if let Err(e) = self.tx_usb.write_packet(&p.buf[..p.len]).await {
91 warn!("Failed to TX packet: {:?}", e);
92 }
93 self.tx_chan.recv_done();
94 }
95 };
96 match select(rx_fut, tx_fut).await {
97 Either::First(x) => x,
98 Either::Second(x) => x,
99 }
100 }
101}
102
103impl<'d, D: Driver<'d>> CdcNcmClass<'d, D> {
104 pub fn into_embassy_net_device<const MTU: usize, const N_RX: usize, const N_TX: usize>(
105 self,
106 state: &'d mut State<'d, MTU, N_RX, N_TX>,
107 ethernet_address: [u8; 6],
108 ) -> (Runner<'d, D, MTU>, Device<'d, MTU>) {
109 let (tx_usb, rx_usb) = self.split();
110
111 let mut caps = DeviceCapabilities::default();
112 caps.max_transmission_unit = 1514; // 1500 IP + 14 ethernet header
113 caps.medium = Medium::Ethernet;
114
115 let state = state.inner.write(StateInner {
116 rx: zerocopy_channel::Channel::new(&mut state.rx[..]),
117 tx: zerocopy_channel::Channel::new(&mut state.tx[..]),
118 link_state: Mutex::new(RefCell::new(LinkStateState {
119 state: LinkState::Down,
120 waker: WakerRegistration::new(),
121 })),
122 });
123
124 let (rx_sender, rx_receiver) = state.rx.split();
125 let (tx_sender, tx_receiver) = state.tx.split();
126
127 (
128 Runner {
129 tx_usb,
130 tx_chan: tx_receiver,
131 rx_usb,
132 rx_chan: rx_sender,
133 link_state: &state.link_state,
134 },
135 Device {
136 caps,
137 ethernet_address,
138 link_state: &state.link_state,
139 rx: rx_receiver,
140 tx: tx_sender,
141 },
142 )
143 }
144}
145
146pub struct PacketBuf<const MTU: usize> {
147 len: usize,
148 buf: [u8; MTU],
149}
150
151impl<const MTU: usize> PacketBuf<MTU> {
152 pub const fn new() -> Self {
153 Self { len: 0, buf: [0; MTU] }
154 }
155}
156
157pub struct Device<'d, const MTU: usize> {
158 rx: zerocopy_channel::Receiver<'d, NoopRawMutex, PacketBuf<MTU>>,
159 tx: zerocopy_channel::Sender<'d, NoopRawMutex, PacketBuf<MTU>>,
160 link_state: &'d Mutex<NoopRawMutex, RefCell<LinkStateState>>,
161 caps: DeviceCapabilities,
162 ethernet_address: [u8; 6],
163}
164
165impl<'d, const MTU: usize> DeviceTrait for Device<'d, MTU> {
166 type RxToken<'a> = RxToken<'a, MTU> where Self: 'a ;
167 type TxToken<'a> = TxToken<'a, MTU> where Self: 'a ;
168
169 fn receive(&mut self, cx: &mut Context) -> Option<(Self::RxToken<'_>, Self::TxToken<'_>)> {
170 if self.rx.poll_recv(cx).is_ready() && self.tx.poll_send(cx).is_ready() {
171 Some((RxToken { rx: self.rx.borrow() }, TxToken { tx: self.tx.borrow() }))
172 } else {
173 None
174 }
175 }
176
177 /// Construct a transmit token.
178 fn transmit(&mut self, cx: &mut Context) -> Option<Self::TxToken<'_>> {
179 if self.tx.poll_send(cx).is_ready() {
180 Some(TxToken { tx: self.tx.borrow() })
181 } else {
182 None
183 }
184 }
185
186 /// Get a description of device capabilities.
187 fn capabilities(&self) -> DeviceCapabilities {
188 self.caps.clone()
189 }
190
191 fn ethernet_address(&self) -> [u8; 6] {
192 self.ethernet_address
193 }
194
195 fn link_state(&mut self, cx: &mut Context) -> LinkState {
196 self.link_state.lock(|s| {
197 let s = &mut *s.borrow_mut();
198 s.waker.register(cx.waker());
199 s.state
200 })
201 }
202}
203
204pub struct RxToken<'a, const MTU: usize> {
205 rx: zerocopy_channel::Receiver<'a, NoopRawMutex, PacketBuf<MTU>>,
206}
207
208impl<'a, const MTU: usize> embassy_net::device::RxToken for RxToken<'a, MTU> {
209 fn consume<R, F>(mut self, f: F) -> R
210 where
211 F: FnOnce(&mut [u8]) -> R,
212 {
213 // NOTE(unwrap): we checked the queue wasn't full when creating the token.
214 let pkt = unwrap!(self.rx.try_recv());
215 let r = f(&mut pkt.buf[..pkt.len]);
216 self.rx.recv_done();
217 r
218 }
219}
220
221pub struct TxToken<'a, const MTU: usize> {
222 tx: zerocopy_channel::Sender<'a, NoopRawMutex, PacketBuf<MTU>>,
223}
224
225impl<'a, const MTU: usize> embassy_net::device::TxToken for TxToken<'a, MTU> {
226 fn consume<R, F>(mut self, len: usize, f: F) -> R
227 where
228 F: FnOnce(&mut [u8]) -> R,
229 {
230 // NOTE(unwrap): we checked the queue wasn't full when creating the token.
231 let pkt = unwrap!(self.tx.try_send());
232 let r = f(&mut pkt.buf[..len]);
233 pkt.len = len;
234 self.tx.send_done();
235 r
236 }
237}
238
239mod zerocopy_channel {
240 use core::cell::RefCell;
241 use core::future::poll_fn;
242 use core::marker::PhantomData;
243 use core::task::{Context, Poll};
244
245 use embassy_sync::blocking_mutex::raw::RawMutex;
246 use embassy_sync::blocking_mutex::Mutex;
247 use embassy_sync::waitqueue::WakerRegistration;
248
249 pub struct Channel<'a, M: RawMutex, T> {
250 buf: *mut T,
251 phantom: PhantomData<&'a mut T>,
252 state: Mutex<M, RefCell<State>>,
253 }
254
255 impl<'a, M: RawMutex, T> Channel<'a, M, T> {
256 pub fn new(buf: &'a mut [T]) -> Self {
257 let len = buf.len();
258 assert!(len != 0);
259
260 Self {
261 buf: buf.as_mut_ptr(),
262 phantom: PhantomData,
263 state: Mutex::new(RefCell::new(State {
264 len,
265 front: 0,
266 back: 0,
267 full: false,
268 send_waker: WakerRegistration::new(),
269 recv_waker: WakerRegistration::new(),
270 })),
271 }
272 }
273
274 pub fn split(&mut self) -> (Sender<'_, M, T>, Receiver<'_, M, T>) {
275 (Sender { channel: self }, Receiver { channel: self })
276 }
277 }
278
279 pub struct Sender<'a, M: RawMutex, T> {
280 channel: &'a Channel<'a, M, T>,
281 }
282
283 impl<'a, M: RawMutex, T> Sender<'a, M, T> {
284 pub fn borrow(&mut self) -> Sender<'_, M, T> {
285 Sender { channel: self.channel }
286 }
287
288 pub fn try_send(&mut self) -> Option<&mut T> {
289 self.channel.state.lock(|s| {
290 let s = &mut *s.borrow_mut();
291 match s.push_index() {
292 Some(i) => Some(unsafe { &mut *self.channel.buf.add(i) }),
293 None => None,
294 }
295 })
296 }
297
298 pub fn poll_send(&mut self, cx: &mut Context) -> Poll<&mut T> {
299 self.channel.state.lock(|s| {
300 let s = &mut *s.borrow_mut();
301 match s.push_index() {
302 Some(i) => Poll::Ready(unsafe { &mut *self.channel.buf.add(i) }),
303 None => {
304 s.recv_waker.register(cx.waker());
305 Poll::Pending
306 }
307 }
308 })
309 }
310
311 pub async fn send(&mut self) -> &mut T {
312 let i = poll_fn(|cx| {
313 self.channel.state.lock(|s| {
314 let s = &mut *s.borrow_mut();
315 match s.push_index() {
316 Some(i) => Poll::Ready(i),
317 None => {
318 s.recv_waker.register(cx.waker());
319 Poll::Pending
320 }
321 }
322 })
323 })
324 .await;
325 unsafe { &mut *self.channel.buf.add(i) }
326 }
327
328 pub fn send_done(&mut self) {
329 self.channel.state.lock(|s| s.borrow_mut().push_done())
330 }
331 }
332 pub struct Receiver<'a, M: RawMutex, T> {
333 channel: &'a Channel<'a, M, T>,
334 }
335
336 impl<'a, M: RawMutex, T> Receiver<'a, M, T> {
337 pub fn borrow(&mut self) -> Receiver<'_, M, T> {
338 Receiver { channel: self.channel }
339 }
340
341 pub fn try_recv(&mut self) -> Option<&mut T> {
342 self.channel.state.lock(|s| {
343 let s = &mut *s.borrow_mut();
344 match s.pop_index() {
345 Some(i) => Some(unsafe { &mut *self.channel.buf.add(i) }),
346 None => None,
347 }
348 })
349 }
350
351 pub fn poll_recv(&mut self, cx: &mut Context) -> Poll<&mut T> {
352 self.channel.state.lock(|s| {
353 let s = &mut *s.borrow_mut();
354 match s.pop_index() {
355 Some(i) => Poll::Ready(unsafe { &mut *self.channel.buf.add(i) }),
356 None => {
357 s.send_waker.register(cx.waker());
358 Poll::Pending
359 }
360 }
361 })
362 }
363
364 pub async fn recv(&mut self) -> &mut T {
365 let i = poll_fn(|cx| {
366 self.channel.state.lock(|s| {
367 let s = &mut *s.borrow_mut();
368 match s.pop_index() {
369 Some(i) => Poll::Ready(i),
370 None => {
371 s.send_waker.register(cx.waker());
372 Poll::Pending
373 }
374 }
375 })
376 })
377 .await;
378 unsafe { &mut *self.channel.buf.add(i) }
379 }
380
381 pub fn recv_done(&mut self) {
382 self.channel.state.lock(|s| s.borrow_mut().pop_done())
383 }
384 }
385
386 struct State {
387 len: usize,
388
389 /// Front index. Always 0..=(N-1)
390 front: usize,
391 /// Back index. Always 0..=(N-1).
392 back: usize,
393
394 /// Used to distinguish "empty" and "full" cases when `front == back`.
395 /// May only be `true` if `front == back`, always `false` otherwise.
396 full: bool,
397
398 send_waker: WakerRegistration,
399 recv_waker: WakerRegistration,
400 }
401
402 impl State {
403 fn increment(&self, i: usize) -> usize {
404 if i + 1 == self.len {
405 0
406 } else {
407 i + 1
408 }
409 }
410
411 fn is_full(&self) -> bool {
412 self.full
413 }
414
415 fn is_empty(&self) -> bool {
416 self.front == self.back && !self.full
417 }
418
419 fn push_index(&mut self) -> Option<usize> {
420 match self.is_full() {
421 true => None,
422 false => Some(self.back),
423 }
424 }
425
426 fn push_done(&mut self) {
427 assert!(!self.is_full());
428 self.back = self.increment(self.back);
429 if self.back == self.front {
430 self.full = true;
431 }
432 self.send_waker.wake();
433 }
434
435 fn pop_index(&mut self) -> Option<usize> {
436 match self.is_empty() {
437 true => None,
438 false => Some(self.front),
439 }
440 }
441
442 fn pop_done(&mut self) {
443 assert!(!self.is_empty());
444 self.front = self.increment(self.front);
445 self.full = false;
446 self.recv_waker.wake();
447 }
448 }
449}
diff --git a/embassy-usb/src/class/cdc_ncm.rs b/embassy-usb/src/class/cdc_ncm/mod.rs
index a39b87e9b..2ee47f68c 100644
--- a/embassy-usb/src/class/cdc_ncm.rs
+++ b/embassy-usb/src/class/cdc_ncm/mod.rs
@@ -1,3 +1,18 @@
1/// CDC-NCM, aka Ethernet over USB.
2///
3/// # Compatibility
4///
5/// Windows: NOT supported in Windows 10. Supported in Windows 11.
6///
7/// Linux: Well-supported since forever.
8///
9/// Android: Support for CDC-NCM is spotty and varies across manufacturers.
10///
11/// - On Pixel 4a, it refused to work on Android 11, worked on Android 12.
12/// - if the host's MAC address has the "locally-administered" bit set (bit 1 of first byte),
13/// it doesn't work! The "Ethernet tethering" option in settings doesn't get enabled.
14/// This is due to regex spaghetti: https://android.googlesource.com/platform/frameworks/base/+/refs/tags/android-mainline-12.0.0_r84/core/res/res/values/config.xml#417
15/// and this nonsense in the linux kernel: https://github.com/torvalds/linux/blob/c00c5e1d157bec0ef0b0b59aa5482eb8dc7e8e49/drivers/net/usb/usbnet.c#L1751-L1757
1use core::intrinsics::copy_nonoverlapping; 16use core::intrinsics::copy_nonoverlapping;
2use core::mem::{size_of, MaybeUninit}; 17use core::mem::{size_of, MaybeUninit};
3 18
@@ -6,6 +21,9 @@ use crate::driver::{Driver, Endpoint, EndpointError, EndpointIn, EndpointOut};
6use crate::types::*; 21use crate::types::*;
7use crate::Builder; 22use crate::Builder;
8 23
24#[cfg(feature = "embassy-net")]
25pub mod embassy_net;
26
9/// This should be used as `device_class` when building the `UsbDevice`. 27/// This should be used as `device_class` when building the `UsbDevice`.
10pub const USB_CLASS_CDC: u8 = 0x02; 28pub const USB_CLASS_CDC: u8 = 0x02;
11 29