sam4l/usbc/
mod.rs

1// Licensed under the Apache License, Version 2.0 or the MIT License.
2// SPDX-License-Identifier: Apache-2.0 OR MIT
3// Copyright Tock Contributors 2022.
4
5//! SAM4L USB controller
6
7pub mod debug;
8
9#[allow(unused_imports)]
10use self::debug::{HexBuf, UdintFlags, UeconFlags, UestaFlags};
11use crate::pm;
12use crate::pm::{disable_clock, enable_clock, Clock, HSBClock, PBBClock};
13use crate::scif;
14use core::cell::Cell;
15use core::ptr;
16use core::slice;
17use kernel::debug as debugln;
18use kernel::hil;
19use kernel::hil::usb::TransferType;
20use kernel::utilities::cells::{OptionalCell, VolatileCell};
21use kernel::utilities::registers::interfaces::{ReadWriteable, Readable, Writeable};
22use kernel::utilities::registers::{
23    register_bitfields, FieldValue, InMemoryRegister, LocalRegisterCopy, ReadOnly, ReadWrite,
24    WriteOnly,
25};
26use kernel::utilities::StaticRef;
27
28// The following macros provide some diagnostics and panics(!)
29// while this module is experimental and should eventually be removed or
30// replaced with better error handling.
31
32macro_rules! client_warn {
33    [ $( $arg:expr ),+ ] => {
34        debugln!($( $arg ),+)
35    };
36}
37
38macro_rules! client_err {
39    [ $( $arg:expr ),+ ] => {
40        panic!($( $arg ),+)
41    };
42}
43
44macro_rules! debug1 {
45    [ $( $arg:expr ),+ ] => {
46        {} // debugln!($( $arg ),+)
47    };
48}
49
50macro_rules! internal_err {
51    [ $( $arg:expr ),+ ] => {
52        panic!($( $arg ),+)
53    };
54}
55
56#[repr(C)]
57struct UsbcRegisters {
58    udcon: ReadWrite<u32, DeviceControl::Register>,
59    udint: ReadOnly<u32, DeviceInterrupt::Register>,
60    udintclr: WriteOnly<u32, DeviceInterrupt::Register>,
61    udintset: WriteOnly<u32, DeviceInterrupt::Register>,
62    udinte: ReadOnly<u32, DeviceInterrupt::Register>,
63    udinteclr: WriteOnly<u32, DeviceInterrupt::Register>,
64    udinteset: WriteOnly<u32, DeviceInterrupt::Register>,
65    uerst: ReadWrite<u32>,
66    udfnum: ReadOnly<u32>,
67    _reserved0: [u8; 0xdc], // 220 bytes
68    // Note that the SAM4L supports only 8 endpoints, but the registers
69    // are laid out such that there is room for 12.
70    // 0x100
71    uecfg: [ReadWrite<u32, EndpointConfig::Register>; 12],
72    uesta: [ReadOnly<u32, EndpointStatus::Register>; 12],
73    uestaclr: [WriteOnly<u32, EndpointStatus::Register>; 12],
74    uestaset: [WriteOnly<u32, EndpointStatus::Register>; 12],
75    uecon: [ReadOnly<u32, EndpointControl::Register>; 12],
76    ueconset: [WriteOnly<u32, EndpointControl::Register>; 12],
77    ueconclr: [WriteOnly<u32, EndpointControl::Register>; 12],
78    _reserved1: [u8; 0x1b0], // 432 bytes
79    // 0x400 = 1024
80    uhcon: ReadWrite<u32>,
81    uhint: ReadOnly<u32>,
82    uhintclr: WriteOnly<u32>,
83    uhintset: WriteOnly<u32>,
84    uhinte: ReadOnly<u32>,
85    uhinteclr: WriteOnly<u32>,
86    uhinteset: WriteOnly<u32>,
87    uprst: ReadWrite<u32>,
88    uhfnum: ReadWrite<u32>,
89    uhsofc: ReadWrite<u32>,
90    _reserved2: [u8; 0xd8], // 216 bytes
91    // 0x500 = 1280
92    upcfg: [ReadWrite<u32>; 12],
93    upsta: [ReadOnly<u32>; 12],
94    upstaclr: [WriteOnly<u32>; 12],
95    upstaset: [WriteOnly<u32>; 12],
96    upcon: [ReadOnly<u32>; 12],
97    upconset: [WriteOnly<u32>; 12],
98    upconclr: [WriteOnly<u32>; 12],
99    upinrq: [ReadWrite<u32>; 12],
100    _reserved3: [u8; 0x180], // 384 bytes
101    // 0x800 = 2048
102    usbcon: ReadWrite<u32, Control::Register>,
103    usbsta: ReadOnly<u32, Status::Register>,
104    usbstaclr: WriteOnly<u32>,
105    usbstaset: WriteOnly<u32>,
106    _reserved4: [u8; 8],
107    // 0x818
108    uvers: ReadOnly<u32>,
109    ufeatures: ReadOnly<u32>,
110    uaddrsize: ReadOnly<u32>,
111    uname1: ReadOnly<u32>,
112    uname2: ReadOnly<u32>,
113    usbfsm: ReadOnly<u32>,
114    udesc: ReadWrite<u32>,
115}
116
117register_bitfields![u32,
118    Control [
119        UIMOD OFFSET(25) NUMBITS(1) [
120            HostMode = 0,
121            DeviceMode = 1
122        ],
123        USBE OFFSET(15) NUMBITS(1) [],
124        FRZCLK OFFSET(14) NUMBITS(1) []
125    ],
126    Status [
127        SUSPEND OFFSET(16) NUMBITS(1) [],
128        CLKUSABLE OFFSET(14) NUMBITS(1) [],
129        SPEED OFFSET(12) NUMBITS(2) [
130            SpeedFull = 0b00,
131            SpeedLow = 0b10
132        ],
133        VBUSRQ OFFSET(9) NUMBITS(1) []
134    ],
135    DeviceControl [
136        GNAK OFFSET(17) NUMBITS(1) [],
137        LS OFFSET(12) NUMBITS(1) [
138            FullSpeed = 0,
139            LowSpeed = 1
140        ],
141        RMWKUP OFFSET(9) NUMBITS(1) [],
142        DETACH OFFSET(8) NUMBITS(1) [],
143        ADDEN OFFSET(7) NUMBITS(1) [],
144        UADD OFFSET(0) NUMBITS(7) []
145    ],
146    DeviceInterrupt [
147        EPINT OFFSET(12) NUMBITS(8),
148        UPRSM OFFSET(6) NUMBITS(1),
149        EORSM OFFSET(5) NUMBITS(1),
150        WAKEUP OFFSET(4) NUMBITS(1),
151        EORST OFFSET(3) NUMBITS(1),
152        SOF OFFSET(2) NUMBITS(1),
153        SUSP OFFSET(0) NUMBITS(1)
154    ],
155    EndpointConfig [
156        REPNB OFFSET(16) NUMBITS(4) [
157            NotRedirected = 0
158        ],
159        EPTYPE OFFSET(11) NUMBITS(2) [
160            Control = 0,
161            Isochronous = 1,
162            Bulk = 2,
163            Interrupt = 3
164        ],
165        EPDIR OFFSET(8) NUMBITS(1) [
166            Out = 0,
167            In = 1
168        ],
169        EPSIZE OFFSET(4) NUMBITS(3) [
170            Bytes8 = 0,
171            Bytes16 = 1,
172            Bytes32 = 2,
173            Bytes64 = 3,
174            Bytes128 = 4,
175            Bytes256 = 5,
176            Bytes512 = 6,
177            Bytes1024 = 7
178        ],
179        EPBK OFFSET(2) NUMBITS(1) [
180            Single = 0,
181            Double = 1
182        ]
183    ],
184    EndpointStatus [
185        CTRLDIR OFFSET(17) NUMBITS(1) [
186            Out = 0,
187            In = 1
188        ],
189        CURRBK OFFSET(14) NUMBITS(2) [
190            Bank0 = 0,
191            Bank1 = 1
192        ],
193        NBUSYBK OFFSET(12) NUMBITS(2) [],
194        RAMACER OFFSET(11) NUMBITS(1) [],
195        DTSEQ OFFSET(8) NUMBITS(2) [
196            Data0 = 0,
197            Data1 = 1
198        ],
199        STALLED OFFSET(6) NUMBITS(1) [],
200        CRCERR OFFSET(6) NUMBITS(1) [],
201        NAKIN OFFSET(4) NUMBITS(1) [],
202        NAKOUT OFFSET(3) NUMBITS(1) [],
203        ERRORF OFFSET(2) NUMBITS(1) [],
204        RXSTP OFFSET(2) NUMBITS(1) [],
205        RXOUT OFFSET(1) NUMBITS(1) [],
206        TXIN OFFSET(0) NUMBITS(1) []
207    ],
208    EndpointControl [
209        BUSY1E 25,
210        BUSY0E 24,
211        STALLRQ 19,
212        RSTDT 18,
213        FIFOCON 14,
214        KILLBK 13,
215        NBUSYBKE 12,
216        RAMACERE 11,
217        NREPLY 8,
218        STALLEDE 6,
219        CRCERRE 6,
220        NAKINE 4,
221        NAKOUTE 3,
222        RXSTPE 2,
223        ERRORFE 2,
224        RXOUTE 1,
225        TXINE 0
226    ]
227];
228
229const USBC_BASE: StaticRef<UsbcRegisters> =
230    unsafe { StaticRef::new(0x400A5000 as *const UsbcRegisters) };
231
232#[inline]
233fn usbc_regs() -> &'static UsbcRegisters {
234    &USBC_BASE
235}
236
237// Datastructures for tracking USB controller state
238
239pub const N_ENDPOINTS: usize = 8;
240
241// This ensures the `descriptors` field is laid out first
242#[repr(C)]
243// This provides the required alignment for the `descriptors` field
244#[repr(align(8))]
245pub struct Usbc<'a> {
246    descriptors: [Endpoint; N_ENDPOINTS],
247    state: OptionalCell<State>,
248    requests: [Cell<Requests>; N_ENDPOINTS],
249    client: OptionalCell<&'a dyn hil::usb::Client<'a>>,
250    pm: &'a pm::PowerManager,
251}
252
253#[derive(Copy, Clone, Default, Debug)]
254pub struct Requests {
255    pub resume_in: bool,
256    pub resume_out: bool,
257}
258
259impl Requests {
260    pub const fn new() -> Self {
261        Requests {
262            resume_in: false,
263            resume_out: false,
264        }
265    }
266}
267
268#[derive(Copy, Clone, Debug)]
269pub enum State {
270    // Controller disabled
271    Reset,
272
273    // Controller enabled, detached from bus
274    // (We may go to this state when the Host
275    // controller suspends the bus.)
276    Idle(Mode),
277
278    // Controller enabled, attached to bus
279    Active(Mode),
280}
281
282#[derive(Copy, Clone, Debug)]
283pub enum Mode {
284    Host,
285    Device {
286        speed: Speed,
287        config: DeviceConfig,
288        state: DeviceState,
289    },
290}
291
292type EndpointConfigValue = LocalRegisterCopy<u32, EndpointConfig::Register>;
293type EndpointStatusValue = LocalRegisterCopy<u32, EndpointStatus::Register>;
294
295#[derive(Copy, Clone, Debug, Default)]
296pub struct DeviceConfig {
297    pub endpoint_configs: [Option<EndpointConfigValue>; N_ENDPOINTS],
298}
299
300#[derive(Copy, Clone, Debug, Default)]
301pub struct DeviceState {
302    pub endpoint_states: [EndpointState; N_ENDPOINTS],
303}
304
305#[derive(Copy, Clone, Debug, Default)]
306pub enum EndpointState {
307    #[default]
308    Disabled,
309    Ctrl(CtrlState),
310    BulkIn(BulkInState),
311    BulkOut(BulkOutState),
312}
313
314#[derive(Copy, Clone, PartialEq, Debug)]
315pub enum CtrlState {
316    Init,
317    ReadIn,
318    ReadStatus,
319    WriteOut,
320    WriteStatus,
321    WriteStatusWait,
322    InDelay,
323}
324
325#[derive(Copy, Clone, PartialEq, Debug)]
326pub enum BulkInState {
327    Init,
328    Delay,
329}
330
331#[derive(Copy, Clone, PartialEq, Debug)]
332pub enum BulkOutState {
333    Init,
334    Delay,
335}
336
337#[derive(Copy, Clone, PartialEq, Debug)]
338pub enum Speed {
339    Full,
340    Low,
341}
342
343pub enum BankIndex {
344    Bank0,
345    Bank1,
346}
347
348impl From<BankIndex> for usize {
349    fn from(bi: BankIndex) -> usize {
350        match bi {
351            BankIndex::Bank0 => 0,
352            BankIndex::Bank1 => 1,
353        }
354    }
355}
356
357pub struct EndpointIndex(u8);
358
359impl EndpointIndex {
360    pub fn new(index: usize) -> EndpointIndex {
361        EndpointIndex(index as u8 & 0xf)
362    }
363
364    pub fn to_u32(self) -> u32 {
365        self.0 as u32
366    }
367}
368
369impl From<EndpointIndex> for usize {
370    fn from(ei: EndpointIndex) -> usize {
371        ei.0 as usize
372    }
373}
374
375pub type Endpoint = [Bank; 2];
376
377pub const fn new_endpoint() -> Endpoint {
378    [Bank::new(), Bank::new()]
379}
380
381#[repr(C)]
382pub struct Bank {
383    addr: VolatileCell<*mut u8>,
384
385    // The following fields are not actually registers
386    // (they may be placed anywhere in memory),
387    // but the register interface provides the volatile
388    // read/writes and bitfields that we need.
389    pub packet_size: InMemoryRegister<u32, PacketSize::Register>,
390    pub control_status: InMemoryRegister<u32, ControlStatus::Register>,
391
392    _reserved: u32,
393}
394
395impl Bank {
396    pub const fn new() -> Bank {
397        Bank {
398            addr: VolatileCell::new(ptr::null_mut()),
399            packet_size: InMemoryRegister::new(0),
400            control_status: InMemoryRegister::new(0),
401            _reserved: 0,
402        }
403    }
404
405    pub fn set_addr(&self, addr: *mut u8) {
406        self.addr.set(addr);
407    }
408}
409
410register_bitfields![u32,
411    PacketSize [
412        AUTO_ZLP OFFSET(31) NUMBITS(1) [
413            No = 0,
414            Yes = 1
415        ],
416        MULTI_PACKET_SIZE OFFSET(16) NUMBITS(15) [],
417        BYTE_COUNT OFFSET(0) NUMBITS(15) []
418    ],
419    ControlStatus [
420        UNDERF 18,
421        OVERF 17,
422        CRCERR 16,
423        STALLRQ_NEXT 0
424    ]
425];
426
427impl<'a> Usbc<'a> {
428    pub const fn new(pm: &'a pm::PowerManager) -> Self {
429        Usbc {
430            client: OptionalCell::empty(),
431            state: OptionalCell::new(State::Reset),
432            descriptors: [
433                new_endpoint(),
434                new_endpoint(),
435                new_endpoint(),
436                new_endpoint(),
437                new_endpoint(),
438                new_endpoint(),
439                new_endpoint(),
440                new_endpoint(),
441            ],
442            requests: [
443                Cell::new(Requests::new()),
444                Cell::new(Requests::new()),
445                Cell::new(Requests::new()),
446                Cell::new(Requests::new()),
447                Cell::new(Requests::new()),
448                Cell::new(Requests::new()),
449                Cell::new(Requests::new()),
450                Cell::new(Requests::new()),
451            ],
452            pm,
453        }
454    }
455
456    fn map_state<F, R>(&self, closure: F) -> R
457    where
458        F: FnOnce(&mut State) -> R,
459    {
460        let mut state = self.state.take().unwrap(); // Unwrap fail = map_state: state value is in use
461        let result = closure(&mut state);
462        self.state.set(state);
463        result
464    }
465
466    fn get_state(&self) -> State {
467        self.state.unwrap_or_panic() // Unwrap fail = get_state: state value is in use
468    }
469
470    fn set_state(&self, state: State) {
471        self.state.set(state);
472    }
473
474    /// Provide a buffer for transfers in and out of the given endpoint
475    /// (The controller need not be enabled before calling this method.)
476    fn endpoint_bank_set_buffer(
477        &self,
478        endpoint: EndpointIndex,
479        bank: BankIndex,
480        buf: &[VolatileCell<u8>],
481    ) {
482        let e: usize = From::from(endpoint);
483        let b: usize = From::from(bank);
484        let p = buf.as_ptr() as *mut u8;
485
486        debug1!("Set Endpoint{}/Bank{} addr={:8?}", e, b, p);
487        self.descriptors[e][b].set_addr(p);
488        self.descriptors[e][b].packet_size.write(
489            PacketSize::BYTE_COUNT.val(0)
490                + PacketSize::MULTI_PACKET_SIZE.val(0)
491                + PacketSize::AUTO_ZLP::No,
492        );
493    }
494
495    /// Enable the controller's clocks and interrupt and transition to Idle state
496    fn enable(&self, mode: Mode) {
497        match self.get_state() {
498            State::Reset => {
499                // Are the USBC clocks enabled at reset?
500                //   10.7.4 says no, but 17.5.3 says yes
501                // Also, "Being in Idle state does not require the USB clocks to
502                //   be activated" (17.6.2)
503                enable_clock(Clock::HSB(HSBClock::USBC));
504                enable_clock(Clock::PBB(PBBClock::USBC));
505
506                // If we got to this state via disable() instead of chip reset,
507                // the values USBCON.FRZCLK, USBCON.UIMOD, UDCON.LS have *not* been
508                // reset to their default values.
509
510                if let Mode::Device { speed, .. } = mode {
511                    usbc_regs().udcon.modify(match speed {
512                        Speed::Full => DeviceControl::LS::FullSpeed,
513                        Speed::Low => DeviceControl::LS::LowSpeed,
514                    });
515                }
516
517                // Enable in device mode
518                usbc_regs().usbcon.modify(Control::UIMOD::DeviceMode);
519                usbc_regs().usbcon.modify(Control::FRZCLK::CLEAR);
520                usbc_regs().usbcon.modify(Control::USBE::SET);
521
522                // Set the pointer to the endpoint descriptors
523                usbc_regs()
524                    .udesc
525                    .set(core::ptr::addr_of!(self.descriptors) as u32);
526
527                // Clear pending device global interrupts
528                usbc_regs().udintclr.write(
529                    DeviceInterrupt::SUSP::SET
530                        + DeviceInterrupt::SOF::SET
531                        + DeviceInterrupt::EORST::SET
532                        + DeviceInterrupt::EORSM::SET
533                        + DeviceInterrupt::UPRSM::SET,
534                );
535
536                // Enable device global interrupts
537                // Note: SOF has been omitted as it is not presently used,
538                // Note: SUSP has been omitted as SUSPEND/WAKEUP is not yet
539                //   implemented.
540                // Note: SOF and SUSP may nevertheless be enabled here
541                //   without harm, but it makes debugging easier to omit them.
542                usbc_regs().udinteset.write(
543                    DeviceInterrupt::EORST::SET
544                        + DeviceInterrupt::EORSM::SET
545                        + DeviceInterrupt::UPRSM::SET,
546                );
547
548                debug1!("Enabled");
549
550                self.set_state(State::Idle(mode));
551            }
552            _ => internal_err!("Already enabled"),
553        }
554    }
555
556    /// Disable the controller, its interrupt, and its clocks
557    fn _disable(&self) {
558        // Detach if necessary
559        if let State::Active(_) = self.get_state() {
560            self.detach();
561        }
562
563        // Disable USBC and its clocks
564        match self.get_state() {
565            State::Idle(..) => {
566                usbc_regs().usbcon.modify(Control::USBE::CLEAR);
567
568                disable_clock(Clock::PBB(PBBClock::USBC));
569                disable_clock(Clock::HSB(HSBClock::USBC));
570
571                self.set_state(State::Reset);
572            }
573            _ => internal_err!("Disable called from wrong state"),
574        }
575    }
576
577    /// Attach to the USB bus after enabling USB bus clock
578    fn attach(&self) {
579        match self.get_state() {
580            State::Idle(mode) => {
581                if self.pm.get_system_frequency() != 48000000 {
582                    internal_err!("The system clock does not support USB");
583                }
584
585                // XX: not clear that this always results in a usable USB clock
586                scif::generic_clock_enable(scif::GenericClock::GCLK7, scif::ClockSource::CLK_HSB);
587
588                while !usbc_regs().usbsta.is_set(Status::CLKUSABLE) {}
589
590                usbc_regs().udcon.modify(DeviceControl::DETACH::CLEAR);
591
592                debug1!("Attached");
593
594                self.set_state(State::Active(mode));
595            }
596            _ => internal_err!("Attach called in wrong state"),
597        }
598    }
599
600    /// Detach from the USB bus.  Also disable USB bus clock to save energy.
601    fn detach(&self) {
602        match self.get_state() {
603            State::Active(mode) => {
604                usbc_regs().udcon.modify(DeviceControl::DETACH::SET);
605
606                scif::generic_clock_disable(scif::GenericClock::GCLK7);
607
608                self.set_state(State::Idle(mode));
609            }
610            _ => debug1!("Already detached"),
611        }
612    }
613
614    /// Configure and enable an endpoint
615    fn endpoint_enable(&self, endpoint: usize, endpoint_config: EndpointConfigValue) {
616        self.endpoint_record_config(endpoint, endpoint_config);
617        self.endpoint_write_config(endpoint, endpoint_config);
618
619        // Enable the endpoint (meaning the controller will respond to requests
620        // to this endpoint)
621        usbc_regs()
622            .uerst
623            .set(usbc_regs().uerst.get() | (1 << endpoint));
624
625        self.endpoint_init(endpoint, endpoint_config);
626
627        // Set EPnINTE, enabling interrupts for this endpoint
628        usbc_regs().udinteset.set(1 << (12 + endpoint));
629
630        debug1!("Enabled endpoint {}", endpoint);
631    }
632
633    fn endpoint_record_config(&self, endpoint: usize, endpoint_config: EndpointConfigValue) {
634        // Record config in case of later bus reset
635        self.map_state(|state| match *state {
636            State::Reset => {
637                client_err!("Not enabled");
638            }
639            State::Idle(Mode::Device { ref mut config, .. }) => {
640                // The endpoint will be active when we next attach
641                config.endpoint_configs[endpoint] = Some(endpoint_config);
642            }
643            State::Active(Mode::Device { ref mut config, .. }) => {
644                // The endpoint will be active immediately
645                config.endpoint_configs[endpoint] = Some(endpoint_config);
646            }
647            _ => client_err!("Not in Device mode"),
648        });
649    }
650
651    fn endpoint_write_config(&self, endpoint: usize, config: EndpointConfigValue) {
652        // This must be performed after each bus reset
653
654        // Configure the endpoint
655        usbc_regs().uecfg[endpoint].set(From::from(config));
656
657        debug1!("Configured endpoint {}", endpoint);
658    }
659
660    fn endpoint_init(&self, endpoint: usize, config: EndpointConfigValue) {
661        self.map_state(|state| match *state {
662            State::Idle(Mode::Device { ref mut state, .. }) => {
663                self.endpoint_init_with_device_state(state, endpoint, config);
664            }
665            State::Active(Mode::Device { ref mut state, .. }) => {
666                self.endpoint_init_with_device_state(state, endpoint, config);
667            }
668            _ => internal_err!("Not reached"),
669        });
670    }
671
672    fn endpoint_init_with_device_state(
673        &self,
674        state: &mut DeviceState,
675        endpoint: usize,
676        config: EndpointConfigValue,
677    ) {
678        // This must be performed after each bus reset (see 17.6.2.2)
679
680        endpoint_enable_interrupts(
681            endpoint,
682            EndpointControl::RAMACERE::SET + EndpointControl::STALLEDE::SET,
683        );
684
685        if config.matches_all(EndpointConfig::EPTYPE::Control) {
686            endpoint_enable_interrupts(endpoint, EndpointControl::RXSTPE::SET);
687            state.endpoint_states[endpoint] = EndpointState::Ctrl(CtrlState::Init);
688        } else if config.matches_all(EndpointConfig::EPTYPE::Bulk + EndpointConfig::EPDIR::In) {
689            endpoint_enable_interrupts(endpoint, EndpointControl::TXINE::SET);
690            state.endpoint_states[endpoint] = EndpointState::BulkIn(BulkInState::Init);
691        } else if config.matches_all(EndpointConfig::EPTYPE::Bulk + EndpointConfig::EPDIR::Out) {
692            endpoint_enable_interrupts(endpoint, EndpointControl::RXOUTE::SET);
693            state.endpoint_states[endpoint] = EndpointState::BulkOut(BulkOutState::Init);
694        } else {
695            // Other endpoint types unimplemented
696        }
697
698        debug1!("Initialized endpoint {}", endpoint);
699    }
700
701    fn endpoint_resume_in(&self, endpoint: usize) {
702        self.map_state(|state| match *state {
703            State::Active(Mode::Device { ref mut state, .. }) => {
704                let endpoint_state = &mut state.endpoint_states[endpoint];
705                match *endpoint_state {
706                    EndpointState::BulkIn(BulkInState::Delay) => {
707                        // Return to Init state
708                        endpoint_enable_interrupts(endpoint, EndpointControl::TXINE::SET);
709                        *endpoint_state = EndpointState::BulkIn(BulkInState::Init);
710                    }
711                    // XXX: Add support for EndpointState::Interrupt*
712                    _ => debugln!("Ignoring superfluous resume_in"),
713                }
714            }
715            _ => debugln!("Ignoring inappropriate resume_in"),
716        });
717    }
718
719    fn endpoint_resume_out(&self, endpoint: usize) {
720        self.map_state(|state| match *state {
721            State::Active(Mode::Device { ref mut state, .. }) => {
722                let endpoint_state = &mut state.endpoint_states[endpoint];
723                match *endpoint_state {
724                    EndpointState::BulkOut(BulkOutState::Delay) => {
725                        // Return to Init state
726                        endpoint_enable_interrupts(endpoint, EndpointControl::RXOUTE::SET);
727                        *endpoint_state = EndpointState::BulkOut(BulkOutState::Init);
728                    }
729                    // XXX: Add support for EndpointState::Interrupt*
730                    _ => debugln!("Ignoring superfluous resume_out"),
731                }
732            }
733            _ => debugln!("Ignoring inappropriate resume_out"),
734        });
735    }
736
737    fn handle_requests(&self) {
738        for endpoint in 0..N_ENDPOINTS {
739            let mut requests = self.requests[endpoint].get();
740
741            if requests.resume_in {
742                self.endpoint_resume_in(endpoint);
743                requests.resume_in = false;
744            }
745            if requests.resume_out {
746                self.endpoint_resume_out(endpoint);
747                requests.resume_out = false;
748            }
749
750            self.requests[endpoint].set(requests);
751        }
752    }
753
754    /// Handle an interrupt from the USBC
755    pub fn handle_interrupt(&self) {
756        self.map_state(|state| match *state {
757            State::Reset => internal_err!("Received interrupt in Reset"),
758            State::Idle(_) => {
759                // We might process WAKEUP here
760                debug1!("Received interrupt in Idle");
761            }
762            State::Active(ref mut mode) => match *mode {
763                Mode::Device {
764                    speed,
765                    ref config,
766                    ref mut state,
767                } => self.handle_device_interrupt(speed, config, state),
768                Mode::Host => internal_err!("Host mode unimplemented"),
769            },
770        });
771
772        // Client callbacks invoked above may have generated state-changing requests
773        self.handle_requests();
774    }
775
776    /// Handle an interrupt while in device mode
777    fn handle_device_interrupt(
778        &self,
779        speed: Speed,
780        device_config: &DeviceConfig,
781        device_state: &mut DeviceState,
782    ) {
783        let udint = usbc_regs().udint.extract();
784
785        debug1!("--> UDINT={:?}", UdintFlags(udint.get()));
786
787        if udint.is_set(DeviceInterrupt::EORST) {
788            // Bus reset
789            debug1!("USB Bus Reset");
790
791            // Reconfigure what has been reset in the USBC
792            usbc_regs().udcon.modify(match speed {
793                Speed::Full => DeviceControl::LS::FullSpeed,
794                Speed::Low => DeviceControl::LS::LowSpeed,
795            });
796
797            // Reset our record of the device state
798            *device_state = DeviceState::default();
799
800            // Reconfigure and initialize endpoints
801            for i in 0..N_ENDPOINTS {
802                if let Some(endpoint_config) = device_config.endpoint_configs[i] {
803                    self.endpoint_write_config(i, endpoint_config);
804                    self.endpoint_init_with_device_state(device_state, i, endpoint_config);
805                }
806            }
807
808            // Alert the client
809            self.client.map(|client| {
810                client.bus_reset();
811            });
812
813            // Acknowledge the interrupt
814            usbc_regs().udintclr.write(DeviceInterrupt::EORST::SET);
815
816            // Wait for the next interrupt before doing anything else
817            return;
818        }
819
820        if udint.is_set(DeviceInterrupt::SUSP) {
821            // The transceiver has been suspended due to the bus being idle for 3ms.
822            // This condition is over when WAKEUP is set.
823
824            // "To further reduce power consumption it is recommended to freeze the USB
825            // clock by writing a one to the Freeze USB Clock (FRZCLK) bit in USBCON when
826            // the USB bus is in suspend mode.
827            //
828            // To recover from the suspend mode, the user shall wait for the Wakeup
829            // (WAKEUP) interrupt bit, which is set when a non-idle event is detected, and
830            // then write a zero to FRZCLK.
831            //
832            // As the WAKEUP interrupt bit in UDINT is set when a non-idle event is
833            // detected, it can occur regardless of whether the controller is in the
834            // suspend mode or not."
835
836            // Subscribe to WAKEUP
837            usbc_regs().udinteset.write(DeviceInterrupt::WAKEUP::SET);
838
839            // Acknowledge the "suspend" event
840            usbc_regs().udintclr.write(DeviceInterrupt::SUSP::SET);
841        }
842
843        if udint.is_set(DeviceInterrupt::WAKEUP) {
844            // XX: If we were suspended: Unfreeze the clock (and unsleep the MCU)
845
846            // Unsubscribe from WAKEUP
847            usbc_regs().udinteclr.write(DeviceInterrupt::WAKEUP::SET);
848
849            // Acknowledge the interrupt
850            usbc_regs().udintclr.write(DeviceInterrupt::WAKEUP::SET);
851
852            // Continue processing, as WAKEUP is usually set
853        }
854
855        if udint.is_set(DeviceInterrupt::SOF) {
856            // Acknowledge Start of frame
857            usbc_regs().udintclr.write(DeviceInterrupt::SOF::SET);
858        }
859
860        if udint.is_set(DeviceInterrupt::EORSM) {
861            // Controller received End of Resume
862            debug1!("UDINT EORSM");
863        }
864
865        if udint.is_set(DeviceInterrupt::UPRSM) {
866            // Controller sent Upstream Resume
867            debug1!("UDINT UPRSM");
868        }
869
870        // Process per-endpoint interrupt flags
871        for endpoint in 0..N_ENDPOINTS {
872            if udint.get() & (1 << (12 + endpoint)) == 0 {
873                // No interrupts for this endpoint
874                continue;
875            }
876
877            self.handle_endpoint_interrupt(endpoint, &mut device_state.endpoint_states[endpoint]);
878        }
879    }
880
881    fn handle_endpoint_interrupt(&self, endpoint: usize, endpoint_state: &mut EndpointState) {
882        let status = usbc_regs().uesta[endpoint].extract();
883        debug1!("  UESTA{}={:?}", endpoint, UestaFlags(status.get()));
884
885        if status.is_set(EndpointStatus::STALLED) {
886            debug1!("\tep{}: STALLED/CRCERR", endpoint);
887
888            // Acknowledge
889            usbc_regs().uestaclr[endpoint].write(EndpointStatus::STALLED::SET);
890        }
891
892        if status.is_set(EndpointStatus::RAMACER) {
893            debug1!("\tep{}: RAMACER", endpoint);
894
895            // Acknowledge
896            usbc_regs().uestaclr[endpoint].write(EndpointStatus::RAMACER::SET);
897        }
898
899        match *endpoint_state {
900            EndpointState::Ctrl(ref mut ctrl_state) => {
901                self.handle_ctrl_endpoint_interrupt(endpoint, ctrl_state, status)
902            }
903            EndpointState::BulkIn(ref mut bulk_in_state) => {
904                self.handle_bulk_in_endpoint_interrupt(endpoint, bulk_in_state, status)
905            }
906            EndpointState::BulkOut(ref mut bulk_out_state) => {
907                self.handle_bulk_out_endpoint_interrupt(endpoint, bulk_out_state, status)
908            }
909            EndpointState::Disabled => {
910                debug1!("Ignoring interrupt for disabled endpoint {}", endpoint);
911            }
912        }
913    }
914
915    fn handle_ctrl_endpoint_interrupt(
916        &self,
917        endpoint: usize,
918        ctrl_state: &mut CtrlState,
919        status: EndpointStatusValue,
920    ) {
921        let mut again = true;
922        while again {
923            again = false;
924            // `again` may be set to true below when it would be
925            // advantageous to process more flags without waiting for
926            // another interrupt.
927
928            debug1!(
929                "  ep{}: Ctrl({:?})  UECON={:?}",
930                endpoint,
931                *ctrl_state,
932                UeconFlags(usbc_regs().uecon[endpoint].get())
933            );
934
935            match *ctrl_state {
936                CtrlState::Init => {
937                    if status.is_set(EndpointStatus::RXSTP) {
938                        // We received a SETUP transaction
939
940                        debug1!("\tep{}: RXSTP", endpoint);
941                        // self.debug_show_d0();
942
943                        let bank = 0;
944                        let packet_bytes = self.descriptors[endpoint][bank]
945                            .packet_size
946                            .read(PacketSize::BYTE_COUNT);
947                        let result = if packet_bytes == 8 {
948                            self.client.map(|c| c.ctrl_setup(endpoint))
949                        } else {
950                            Some(hil::usb::CtrlSetupResult::ErrBadLength)
951                        };
952
953                        match result {
954                            Some(hil::usb::CtrlSetupResult::Ok)
955                            | Some(hil::usb::CtrlSetupResult::OkSetAddress) => {
956                                // Unsubscribe from SETUP interrupts
957                                endpoint_disable_interrupts(endpoint, EndpointControl::RXSTPE::SET);
958
959                                if status.matches_all(EndpointStatus::CTRLDIR::In) {
960                                    // The following Data stage will be IN
961
962                                    // Wait until bank is clear to send (TXIN).
963                                    // Also wait for NAKOUT to signal end of IN
964                                    // stage (the datasheet incorrectly says
965                                    // NAKIN).
966                                    usbc_regs().uestaclr[endpoint]
967                                        .write(EndpointStatus::NAKOUT::SET);
968                                    endpoint_enable_interrupts(
969                                        endpoint,
970                                        EndpointControl::TXINE::SET + EndpointControl::NAKOUTE::SET,
971                                    );
972                                    *ctrl_state = CtrlState::ReadIn;
973                                } else {
974                                    // The following Data stage will be OUT
975
976                                    // Wait for OUT packets (RXOUT).  Also wait
977                                    // for NAKIN to signal end of OUT stage.
978                                    usbc_regs().uestaclr[endpoint]
979                                        .write(EndpointStatus::NAKIN::SET);
980                                    endpoint_enable_interrupts(
981                                        endpoint,
982                                        EndpointControl::RXOUTE::SET + EndpointControl::NAKINE::SET,
983                                    );
984                                    *ctrl_state = CtrlState::WriteOut;
985                                }
986                            }
987                            failure => {
988                                // Respond with STALL to any following
989                                // transactions in this request
990                                usbc_regs().ueconset[endpoint].write(EndpointControl::STALLRQ::SET);
991
992                                match failure {
993                                    None => debug1!("\tep{}: No client to handle Setup", endpoint),
994                                    Some(_err) => {
995                                        debug1!("\tep{}: Client err on Setup: {:?}", endpoint, _err)
996                                    }
997                                }
998
999                                // Remain in Init state for next SETUP
1000                            }
1001                        }
1002
1003                        // Acknowledge SETUP interrupt
1004                        usbc_regs().uestaclr[endpoint].write(EndpointStatus::RXSTP::SET);
1005                    }
1006                }
1007                CtrlState::ReadIn => {
1008                    // TODO: Handle Abort as described in 17.6.2.15
1009                    // (for Control and Isochronous only)
1010
1011                    if status.is_set(EndpointStatus::NAKOUT) {
1012                        // The host has completed the IN stage by sending an OUT token
1013
1014                        endpoint_disable_interrupts(
1015                            endpoint,
1016                            EndpointControl::TXINE::SET + EndpointControl::NAKOUTE::SET,
1017                        );
1018
1019                        debug1!("\tep{}: NAKOUT", endpoint);
1020                        self.client.map(|c| c.ctrl_status(endpoint));
1021
1022                        // Await end of Status stage
1023                        endpoint_enable_interrupts(endpoint, EndpointControl::RXOUTE::SET);
1024
1025                        // Acknowledge
1026                        usbc_regs().uestaclr[endpoint].write(EndpointStatus::NAKOUT::SET);
1027
1028                        *ctrl_state = CtrlState::ReadStatus;
1029
1030                        // Run handler again in case the RXOUT has already arrived
1031                        again = true;
1032                    } else if status.is_set(EndpointStatus::TXIN) {
1033                        // The data bank is ready to receive another IN payload
1034                        debug1!("\tep{}: TXIN", endpoint);
1035
1036                        let result = self.client.map(|c| {
1037                            // Allow client to write a packet payload to buffer
1038                            c.ctrl_in(endpoint)
1039                        });
1040                        match result {
1041                            Some(hil::usb::CtrlInResult::Packet(
1042                                packet_bytes,
1043                                transfer_complete,
1044                            )) => {
1045                                // Check if the entire buffer is full, and
1046                                // handle that case slightly differently. Note,
1047                                // this depends on the length of the buffer
1048                                // used. Right now, that is 64 bytes.
1049                                let packet_size = if packet_bytes == 64 && transfer_complete {
1050                                    // Send a complete final packet, and request
1051                                    // that the controller also send a zero-length
1052                                    // packet to signal the end of transfer
1053                                    PacketSize::BYTE_COUNT.val(64) + PacketSize::AUTO_ZLP::Yes
1054                                } else {
1055                                    // Send either a complete but not-final
1056                                    // packet, or a short and final packet (which
1057                                    // itself signals end of transfer)
1058                                    PacketSize::BYTE_COUNT.val(packet_bytes as u32)
1059                                };
1060                                let bank = 0;
1061                                self.descriptors[endpoint][bank]
1062                                    .packet_size
1063                                    .write(packet_size);
1064
1065                                debug1!(
1066                                    "\tep{}: Send CTRL IN packet ({} bytes)",
1067                                    endpoint,
1068                                    packet_bytes
1069                                );
1070                                // self.debug_show_d0();
1071
1072                                if transfer_complete {
1073                                    // IN data completely sent.  Unsubscribe from TXIN.
1074                                    // (Continue awaiting NAKOUT to indicate end of Data stage)
1075                                    endpoint_disable_interrupts(
1076                                        endpoint,
1077                                        EndpointControl::TXINE::SET,
1078                                    );
1079                                } else {
1080                                    // Continue waiting for next TXIN
1081                                }
1082
1083                                // Clear TXIN to signal to the controller that the IN payload is
1084                                // ready to send
1085                                usbc_regs().uestaclr[endpoint].write(EndpointStatus::TXIN::SET);
1086                            }
1087                            Some(hil::usb::CtrlInResult::Delay) => {
1088                                endpoint_disable_interrupts(endpoint, EndpointControl::TXINE::SET);
1089
1090                                debug1!("*** Client NAK");
1091
1092                                // XXX set busy bits?
1093
1094                                *ctrl_state = CtrlState::InDelay;
1095                            }
1096                            _ => {
1097                                // Respond with STALL to any following IN/OUT transactions
1098                                usbc_regs().ueconset[endpoint].write(EndpointControl::STALLRQ::SET);
1099
1100                                debug1!("\tep{}: Client IN err => STALL", endpoint);
1101
1102                                // Wait for next SETUP
1103                                endpoint_disable_interrupts(
1104                                    endpoint,
1105                                    EndpointControl::NAKOUTE::SET,
1106                                );
1107                                endpoint_enable_interrupts(endpoint, EndpointControl::RXSTPE::SET);
1108                                *ctrl_state = CtrlState::Init;
1109                            }
1110                        }
1111                    }
1112                }
1113                CtrlState::ReadStatus => {
1114                    if status.is_set(EndpointStatus::RXOUT) {
1115                        // Host has completed Status stage by sending an OUT packet
1116
1117                        debug1!("\tep{}: RXOUT: End of Control Read transaction", endpoint);
1118
1119                        // Wait for next SETUP
1120                        endpoint_disable_interrupts(endpoint, EndpointControl::RXOUTE::SET);
1121                        endpoint_enable_interrupts(endpoint, EndpointControl::RXSTPE::SET);
1122                        *ctrl_state = CtrlState::Init;
1123
1124                        // Acknowledge
1125                        usbc_regs().uestaclr[endpoint].write(EndpointStatus::RXOUT::SET);
1126
1127                        self.client.map(|c| c.ctrl_status_complete(endpoint));
1128                    }
1129                }
1130                CtrlState::WriteOut => {
1131                    if status.is_set(EndpointStatus::RXOUT) {
1132                        // Received data
1133
1134                        debug1!("\tep{}: RXOUT: Received Control Write data", endpoint);
1135                        // self.debug_show_d0();
1136
1137                        // Pass the data to the client and see how it reacts
1138                        let bank = 0;
1139                        let result = self.client.map(|c| {
1140                            c.ctrl_out(
1141                                endpoint,
1142                                self.descriptors[endpoint][bank]
1143                                    .packet_size
1144                                    .read(PacketSize::BYTE_COUNT),
1145                            )
1146                        });
1147                        match result {
1148                            Some(hil::usb::CtrlOutResult::Ok) => {
1149                                // Acknowledge
1150                                usbc_regs().uestaclr[endpoint].write(EndpointStatus::RXOUT::SET);
1151                            }
1152                            Some(hil::usb::CtrlOutResult::Delay) => {
1153                                // Don't acknowledge; hardware will have to send NAK
1154
1155                                // Unsubscribe from RXOUT until client says it is ready
1156                                // (But there is not yet any interface for that)
1157                                endpoint_disable_interrupts(endpoint, EndpointControl::RXOUTE::SET);
1158                            }
1159                            _ => {
1160                                // Respond with STALL to any following transactions
1161                                // in this request
1162                                usbc_regs().ueconset[endpoint].write(EndpointControl::STALLRQ::SET);
1163
1164                                debug1!("\tep{}: Client OUT err => STALL", endpoint);
1165
1166                                // Wait for next SETUP
1167                                endpoint_disable_interrupts(
1168                                    endpoint,
1169                                    EndpointControl::RXOUTE::SET + EndpointControl::NAKINE::SET,
1170                                );
1171                                endpoint_enable_interrupts(endpoint, EndpointControl::RXSTPE::SET);
1172                                *ctrl_state = CtrlState::Init;
1173                            }
1174                        }
1175                    }
1176                    if status.is_set(EndpointStatus::NAKIN) {
1177                        // The host has completed the Data stage by sending an IN token
1178                        debug1!("\tep{}: NAKIN: Control Write -> Status stage", endpoint);
1179
1180                        // Acknowledge
1181                        usbc_regs().uestaclr[endpoint].write(EndpointStatus::NAKIN::SET);
1182
1183                        // Wait for bank to be free so we can write ZLP to acknowledge transfer
1184                        endpoint_disable_interrupts(
1185                            endpoint,
1186                            EndpointControl::RXOUTE::SET + EndpointControl::NAKINE::SET,
1187                        );
1188                        endpoint_enable_interrupts(endpoint, EndpointControl::TXINE::SET);
1189                        *ctrl_state = CtrlState::WriteStatus;
1190
1191                        // Can probably send the ZLP immediately
1192                        again = true;
1193                    }
1194                }
1195                CtrlState::WriteStatus => {
1196                    if status.is_set(EndpointStatus::TXIN) {
1197                        debug1!(
1198                            "\tep{}: TXIN for Control Write Status (will send ZLP)",
1199                            endpoint
1200                        );
1201
1202                        self.client.map(|c| c.ctrl_status(endpoint));
1203
1204                        // Send zero-length packet to acknowledge transaction
1205                        let bank = 0;
1206                        self.descriptors[endpoint][bank]
1207                            .packet_size
1208                            .write(PacketSize::BYTE_COUNT.val(0));
1209
1210                        // Signal to the controller that the IN payload is ready to send
1211                        usbc_regs().uestaclr[endpoint].write(EndpointStatus::TXIN::SET);
1212
1213                        // Wait for TXIN again to confirm that IN payload has been sent
1214                        *ctrl_state = CtrlState::WriteStatusWait;
1215                    }
1216                }
1217                CtrlState::WriteStatusWait => {
1218                    if status.is_set(EndpointStatus::TXIN) {
1219                        debug1!("\tep{}: TXIN: Control Write Status Complete", endpoint);
1220
1221                        // Wait for next SETUP
1222                        endpoint_disable_interrupts(endpoint, EndpointControl::TXINE::SET);
1223                        endpoint_enable_interrupts(endpoint, EndpointControl::RXSTPE::SET);
1224                        *ctrl_state = CtrlState::Init;
1225
1226                        // for SetAddress, client must enable address after STATUS stage
1227                        self.client.map(|c| c.ctrl_status_complete(endpoint));
1228                    }
1229                }
1230                CtrlState::InDelay => internal_err!("Not reached"),
1231            }
1232
1233            // Uncomment the following line to run the above while loop only once per interrupt,
1234            // which can make debugging easier.
1235            //
1236            // again = false;
1237        }
1238    }
1239
1240    fn handle_bulk_out_endpoint_interrupt(
1241        &self,
1242        endpoint: usize,
1243        bulk_out_state: &mut BulkOutState,
1244        status: EndpointStatusValue,
1245    ) {
1246        match *bulk_out_state {
1247            BulkOutState::Init => {
1248                if status.is_set(EndpointStatus::RXOUT) {
1249                    // We got an OUT request from the host
1250
1251                    debug1!("\tep{}: RXOUT", endpoint);
1252
1253                    if !usbc_regs().uecon[endpoint].is_set(EndpointControl::FIFOCON) {
1254                        debugln!("Got RXOUT but not FIFOCON");
1255                        return;
1256                    }
1257                    // A bank is full of an OUT packet
1258
1259                    let bank = 0;
1260                    let packet_bytes = self.descriptors[endpoint][bank]
1261                        .packet_size
1262                        .read(PacketSize::BYTE_COUNT);
1263
1264                    let result = self.client.map(|c| {
1265                        // Allow client to consume the packet
1266                        c.packet_out(TransferType::Bulk, endpoint, packet_bytes)
1267                    });
1268                    match result {
1269                        Some(hil::usb::OutResult::Ok) => {
1270                            // Acknowledge
1271                            usbc_regs().uestaclr[endpoint].write(EndpointStatus::RXOUT::SET);
1272
1273                            // Clear FIFOCON to signal that the packet was consumed
1274                            usbc_regs().ueconclr[endpoint].write(EndpointControl::FIFOCON::SET);
1275
1276                            debug1!(
1277                                "\tep{}: Recv BULK OUT packet ({} bytes)",
1278                                endpoint,
1279                                packet_bytes
1280                            );
1281
1282                            // Remain in Init state
1283                        }
1284                        Some(hil::usb::OutResult::Delay) => {
1285                            // The client is not ready to consume data; wait for resume
1286
1287                            endpoint_disable_interrupts(endpoint, EndpointControl::RXOUTE::SET);
1288
1289                            *bulk_out_state = BulkOutState::Delay;
1290                        }
1291                        _ => {
1292                            debug1!("\tep{}: Client OUT err => STALL", endpoint);
1293
1294                            // Respond with STALL to any following IN/OUT transactions
1295                            usbc_regs().ueconset[endpoint].write(EndpointControl::STALLRQ::SET);
1296
1297                            // XXX: interrupts?
1298
1299                            // Remain in Init state?
1300                        }
1301                    }
1302                }
1303            }
1304            BulkOutState::Delay => internal_err!("Not reached"),
1305        }
1306    }
1307
1308    fn handle_bulk_in_endpoint_interrupt(
1309        &self,
1310        endpoint: usize,
1311        bulk_in_state: &mut BulkInState,
1312        status: EndpointStatusValue,
1313    ) {
1314        match *bulk_in_state {
1315            BulkInState::Init => {
1316                if status.is_set(EndpointStatus::TXIN) {
1317                    // We got an IN request from the host
1318
1319                    debug1!("\tep{}: TXIN", endpoint);
1320
1321                    if !usbc_regs().uecon[endpoint].is_set(EndpointControl::FIFOCON) {
1322                        debugln!("Got TXIN but not FIFOCON");
1323                        return;
1324                    }
1325                    // A bank is free to write an IN packet
1326
1327                    let result = self.client.map(|c| {
1328                        // Allow client to write a packet payload to the buffer
1329                        c.packet_in(TransferType::Bulk, endpoint)
1330                    });
1331                    match result {
1332                        Some(hil::usb::InResult::Packet(packet_bytes)) => {
1333                            // Acknowledge
1334                            usbc_regs().uestaclr[endpoint].write(EndpointStatus::TXIN::SET);
1335
1336                            // Tell the controller the size of the packet
1337                            let bank = 0;
1338                            self.descriptors[endpoint][bank]
1339                                .packet_size
1340                                .write(PacketSize::BYTE_COUNT.val(packet_bytes as u32));
1341
1342                            // Clear FIFOCON to signal data ready to send
1343                            usbc_regs().ueconclr[endpoint].write(EndpointControl::FIFOCON::SET);
1344
1345                            debug1!(
1346                                "\tep{}: Send BULK IN packet ({} bytes)",
1347                                endpoint,
1348                                packet_bytes
1349                            );
1350
1351                            // Remain in Init state
1352                        }
1353                        Some(hil::usb::InResult::Delay) => {
1354                            // The client is not ready to send data; wait for resume
1355
1356                            endpoint_disable_interrupts(endpoint, EndpointControl::TXINE::SET);
1357
1358                            *bulk_in_state = BulkInState::Delay;
1359                        }
1360                        _ => {
1361                            debug1!("\tep{}: Client IN err => STALL", endpoint);
1362
1363                            // Respond with STALL to any following IN/OUT transactions
1364                            usbc_regs().ueconset[endpoint].write(EndpointControl::STALLRQ::SET);
1365
1366                            // XXX: interrupts?
1367
1368                            // Remain in Init state?
1369                        }
1370                    }
1371                }
1372            }
1373            BulkInState::Delay => {
1374                // Endpoint interrupts should be handled already or disabled
1375                internal_err!("Not reached");
1376            }
1377        }
1378    }
1379
1380    #[allow(dead_code)]
1381    fn debug_show_d0(&self) {
1382        for bi in 0..1 {
1383            let b = &self.descriptors[0][bi];
1384            let addr = b.addr.get();
1385            let _buf = if addr.is_null() {
1386                None
1387            } else {
1388                unsafe {
1389                    Some(slice::from_raw_parts(
1390                        addr,
1391                        b.packet_size.read(PacketSize::BYTE_COUNT) as usize,
1392                    ))
1393                }
1394            };
1395
1396            debug1!(
1397                "B_0_{} \
1398                 \n     {:?}\
1399                 \n     {:?}\
1400                 \n     {:?}",
1401                bi, // core::ptr::addr_of!(b.addr), b.addr.get(),
1402                b.packet_size.get(),
1403                b.control_status.get(),
1404                _buf.map(HexBuf)
1405            );
1406        }
1407    }
1408
1409    pub fn mode(&self) -> Option<Mode> {
1410        match self.get_state() {
1411            State::Idle(mode) => Some(mode),
1412            State::Active(mode) => Some(mode),
1413            _ => None,
1414        }
1415    }
1416
1417    pub fn speed(&self) -> Option<Speed> {
1418        match self.mode() {
1419            Some(mode) => {
1420                match mode {
1421                    Mode::Device { speed, .. } => Some(speed),
1422                    Mode::Host => {
1423                        None // XX USBSTA.SPEED
1424                    }
1425                }
1426            }
1427            _ => None,
1428        }
1429    }
1430
1431    // TODO: Remote wakeup (Device -> Host, after receiving DEVICE_REMOTE_WAKEUP)
1432}
1433
1434#[inline]
1435fn endpoint_disable_interrupts(endpoint: usize, mask: FieldValue<u32, EndpointControl::Register>) {
1436    usbc_regs().ueconclr[endpoint].write(mask);
1437}
1438
1439#[inline]
1440fn endpoint_enable_interrupts(endpoint: usize, mask: FieldValue<u32, EndpointControl::Register>) {
1441    usbc_regs().ueconset[endpoint].write(mask);
1442}
1443
1444impl<'a> hil::usb::UsbController<'a> for Usbc<'a> {
1445    fn set_client(&self, client: &'a dyn hil::usb::Client<'a>) {
1446        self.client.set(client);
1447    }
1448
1449    fn endpoint_set_ctrl_buffer(&self, buf: &'a [VolatileCell<u8>]) {
1450        if buf.len() < 8 {
1451            client_err!("Bad endpoint buffer size");
1452        }
1453
1454        self.endpoint_bank_set_buffer(EndpointIndex::new(0), BankIndex::Bank0, buf);
1455    }
1456
1457    fn endpoint_set_in_buffer(&self, endpoint: usize, buf: &'a [VolatileCell<u8>]) {
1458        if buf.len() < 8 {
1459            client_err!("Bad endpoint buffer size");
1460        }
1461
1462        self.endpoint_bank_set_buffer(EndpointIndex::new(endpoint), BankIndex::Bank0, buf);
1463    }
1464
1465    fn endpoint_set_out_buffer(&self, endpoint: usize, buf: &'a [VolatileCell<u8>]) {
1466        if buf.len() < 8 {
1467            client_err!("Bad endpoint buffer size");
1468        }
1469
1470        // XXX: when implementing in_out endpoints, this should probably set a different slice than endpoint_set_in_buffer.
1471        self.endpoint_bank_set_buffer(EndpointIndex::new(endpoint), BankIndex::Bank0, buf);
1472    }
1473
1474    fn enable_as_device(&self, speed: hil::usb::DeviceSpeed) {
1475        let speed = match speed {
1476            hil::usb::DeviceSpeed::Full => Speed::Full,
1477            hil::usb::DeviceSpeed::Low => Speed::Low,
1478        };
1479
1480        match self.get_state() {
1481            State::Reset => self.enable(Mode::Device {
1482                speed,
1483                config: DeviceConfig::default(),
1484                state: DeviceState::default(),
1485            }),
1486            _ => client_err!("Already enabled"),
1487        }
1488    }
1489
1490    fn attach(&self) {
1491        match self.get_state() {
1492            State::Reset => client_warn!("Not enabled"),
1493            State::Active(_) => client_warn!("Already attached"),
1494            State::Idle(_) => self.attach(),
1495        }
1496    }
1497
1498    fn detach(&self) {
1499        match self.get_state() {
1500            State::Reset => client_warn!("Not enabled"),
1501            State::Idle(_) => client_warn!("Not attached"),
1502            State::Active(_) => self.detach(),
1503        }
1504    }
1505
1506    fn set_address(&self, addr: u16) {
1507        usbc_regs()
1508            .udcon
1509            .modify(DeviceControl::UADD.val(addr as u32));
1510
1511        debug1!("Set Address = {}", addr);
1512    }
1513
1514    fn enable_address(&self) {
1515        usbc_regs().udcon.modify(DeviceControl::ADDEN::SET);
1516
1517        debug1!(
1518            "Enable Address = {}",
1519            usbc_regs().udcon.read(DeviceControl::UADD)
1520        );
1521    }
1522
1523    fn endpoint_in_enable(&self, transfer_type: TransferType, endpoint: usize) {
1524        let endpoint_cfg = match transfer_type {
1525            TransferType::Control => {
1526                panic!("There is no IN control endpoint");
1527            }
1528            TransferType::Bulk => LocalRegisterCopy::new(From::from(
1529                EndpointConfig::EPTYPE::Bulk
1530                    + EndpointConfig::EPDIR::In
1531                    + EndpointConfig::EPSIZE::Bytes8
1532                    + EndpointConfig::EPBK::Single,
1533            )),
1534            TransferType::Interrupt | TransferType::Isochronous => unimplemented!(),
1535        };
1536
1537        self.endpoint_enable(endpoint, endpoint_cfg)
1538    }
1539
1540    fn endpoint_out_enable(&self, transfer_type: TransferType, endpoint: usize) {
1541        let endpoint_cfg = match transfer_type {
1542            TransferType::Control => LocalRegisterCopy::new(From::from(
1543                EndpointConfig::EPTYPE::Control
1544                    + EndpointConfig::EPDIR::Out
1545                    + EndpointConfig::EPSIZE::Bytes8
1546                    + EndpointConfig::EPBK::Single,
1547            )),
1548            TransferType::Bulk => LocalRegisterCopy::new(From::from(
1549                EndpointConfig::EPTYPE::Bulk
1550                    + EndpointConfig::EPDIR::Out
1551                    + EndpointConfig::EPSIZE::Bytes8
1552                    + EndpointConfig::EPBK::Single,
1553            )),
1554            TransferType::Interrupt | TransferType::Isochronous => unimplemented!(),
1555        };
1556
1557        self.endpoint_enable(endpoint, endpoint_cfg)
1558    }
1559
1560    fn endpoint_in_out_enable(&self, _transfer_type: TransferType, _endpoint: usize) {
1561        unimplemented!()
1562    }
1563
1564    fn endpoint_resume_in(&self, endpoint: usize) {
1565        let mut requests = self.requests[endpoint].get();
1566        requests.resume_in = true;
1567        self.requests[endpoint].set(requests);
1568
1569        // Immediately handle the request to resume the endpoint.
1570        self.handle_requests();
1571    }
1572
1573    fn endpoint_resume_out(&self, endpoint: usize) {
1574        let mut requests = self.requests[endpoint].get();
1575        requests.resume_out = true;
1576        self.requests[endpoint].set(requests);
1577
1578        // Immediately handle the request to resume the endpoint.
1579        self.handle_requests();
1580    }
1581}