1pub 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
28macro_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 {} };
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], 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], 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], 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], usbcon: ReadWrite<u32, Control::Register>,
103 usbsta: ReadOnly<u32, Status::Register>,
104 usbstaclr: WriteOnly<u32>,
105 usbstaset: WriteOnly<u32>,
106 _reserved4: [u8; 8],
107 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
237pub const N_ENDPOINTS: usize = 8;
240
241#[repr(C)]
243#[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 Reset,
272
273 Idle(Mode),
277
278 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 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(); 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() }
469
470 fn set_state(&self, state: State) {
471 self.state.set(state);
472 }
473
474 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 fn enable(&self, mode: Mode) {
497 match self.get_state() {
498 State::Reset => {
499 enable_clock(Clock::HSB(HSBClock::USBC));
504 enable_clock(Clock::PBB(PBBClock::USBC));
505
506 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 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 usbc_regs()
524 .udesc
525 .set(core::ptr::addr_of!(self.descriptors) as u32);
526
527 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 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 fn _disable(&self) {
558 if let State::Active(_) = self.get_state() {
560 self.detach();
561 }
562
563 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 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 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 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 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 usbc_regs()
622 .uerst
623 .set(usbc_regs().uerst.get() | (1 << endpoint));
624
625 self.endpoint_init(endpoint, endpoint_config);
626
627 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 self.map_state(|state| match *state {
636 State::Reset => {
637 client_err!("Not enabled");
638 }
639 State::Idle(Mode::Device { ref mut config, .. }) => {
640 config.endpoint_configs[endpoint] = Some(endpoint_config);
642 }
643 State::Active(Mode::Device { ref mut config, .. }) => {
644 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 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 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 }
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 endpoint_enable_interrupts(endpoint, EndpointControl::TXINE::SET);
709 *endpoint_state = EndpointState::BulkIn(BulkInState::Init);
710 }
711 _ => 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 endpoint_enable_interrupts(endpoint, EndpointControl::RXOUTE::SET);
727 *endpoint_state = EndpointState::BulkOut(BulkOutState::Init);
728 }
729 _ => 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 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 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 self.handle_requests();
774 }
775
776 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 debug1!("USB Bus Reset");
790
791 usbc_regs().udcon.modify(match speed {
793 Speed::Full => DeviceControl::LS::FullSpeed,
794 Speed::Low => DeviceControl::LS::LowSpeed,
795 });
796
797 *device_state = DeviceState::default();
799
800 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 self.client.map(|client| {
810 client.bus_reset();
811 });
812
813 usbc_regs().udintclr.write(DeviceInterrupt::EORST::SET);
815
816 return;
818 }
819
820 if udint.is_set(DeviceInterrupt::SUSP) {
821 usbc_regs().udinteset.write(DeviceInterrupt::WAKEUP::SET);
838
839 usbc_regs().udintclr.write(DeviceInterrupt::SUSP::SET);
841 }
842
843 if udint.is_set(DeviceInterrupt::WAKEUP) {
844 usbc_regs().udinteclr.write(DeviceInterrupt::WAKEUP::SET);
848
849 usbc_regs().udintclr.write(DeviceInterrupt::WAKEUP::SET);
851
852 }
854
855 if udint.is_set(DeviceInterrupt::SOF) {
856 usbc_regs().udintclr.write(DeviceInterrupt::SOF::SET);
858 }
859
860 if udint.is_set(DeviceInterrupt::EORSM) {
861 debug1!("UDINT EORSM");
863 }
864
865 if udint.is_set(DeviceInterrupt::UPRSM) {
866 debug1!("UDINT UPRSM");
868 }
869
870 for endpoint in 0..N_ENDPOINTS {
872 if udint.get() & (1 << (12 + endpoint)) == 0 {
873 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 usbc_regs().uestaclr[endpoint].write(EndpointStatus::STALLED::SET);
890 }
891
892 if status.is_set(EndpointStatus::RAMACER) {
893 debug1!("\tep{}: RAMACER", endpoint);
894
895 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 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 debug1!("\tep{}: RXSTP", endpoint);
941 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 endpoint_disable_interrupts(endpoint, EndpointControl::RXSTPE::SET);
958
959 if status.matches_all(EndpointStatus::CTRLDIR::In) {
960 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 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 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 }
1001 }
1002
1003 usbc_regs().uestaclr[endpoint].write(EndpointStatus::RXSTP::SET);
1005 }
1006 }
1007 CtrlState::ReadIn => {
1008 if status.is_set(EndpointStatus::NAKOUT) {
1012 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 endpoint_enable_interrupts(endpoint, EndpointControl::RXOUTE::SET);
1024
1025 usbc_regs().uestaclr[endpoint].write(EndpointStatus::NAKOUT::SET);
1027
1028 *ctrl_state = CtrlState::ReadStatus;
1029
1030 again = true;
1032 } else if status.is_set(EndpointStatus::TXIN) {
1033 debug1!("\tep{}: TXIN", endpoint);
1035
1036 let result = self.client.map(|c| {
1037 c.ctrl_in(endpoint)
1039 });
1040 match result {
1041 Some(hil::usb::CtrlInResult::Packet(
1042 packet_bytes,
1043 transfer_complete,
1044 )) => {
1045 let packet_size = if packet_bytes == 64 && transfer_complete {
1050 PacketSize::BYTE_COUNT.val(64) + PacketSize::AUTO_ZLP::Yes
1054 } else {
1055 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 if transfer_complete {
1073 endpoint_disable_interrupts(
1076 endpoint,
1077 EndpointControl::TXINE::SET,
1078 );
1079 } else {
1080 }
1082
1083 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 *ctrl_state = CtrlState::InDelay;
1095 }
1096 _ => {
1097 usbc_regs().ueconset[endpoint].write(EndpointControl::STALLRQ::SET);
1099
1100 debug1!("\tep{}: Client IN err => STALL", endpoint);
1101
1102 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 debug1!("\tep{}: RXOUT: End of Control Read transaction", endpoint);
1118
1119 endpoint_disable_interrupts(endpoint, EndpointControl::RXOUTE::SET);
1121 endpoint_enable_interrupts(endpoint, EndpointControl::RXSTPE::SET);
1122 *ctrl_state = CtrlState::Init;
1123
1124 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 debug1!("\tep{}: RXOUT: Received Control Write data", endpoint);
1135 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 usbc_regs().uestaclr[endpoint].write(EndpointStatus::RXOUT::SET);
1151 }
1152 Some(hil::usb::CtrlOutResult::Delay) => {
1153 endpoint_disable_interrupts(endpoint, EndpointControl::RXOUTE::SET);
1158 }
1159 _ => {
1160 usbc_regs().ueconset[endpoint].write(EndpointControl::STALLRQ::SET);
1163
1164 debug1!("\tep{}: Client OUT err => STALL", endpoint);
1165
1166 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 debug1!("\tep{}: NAKIN: Control Write -> Status stage", endpoint);
1179
1180 usbc_regs().uestaclr[endpoint].write(EndpointStatus::NAKIN::SET);
1182
1183 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 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 let bank = 0;
1206 self.descriptors[endpoint][bank]
1207 .packet_size
1208 .write(PacketSize::BYTE_COUNT.val(0));
1209
1210 usbc_regs().uestaclr[endpoint].write(EndpointStatus::TXIN::SET);
1212
1213 *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 endpoint_disable_interrupts(endpoint, EndpointControl::TXINE::SET);
1223 endpoint_enable_interrupts(endpoint, EndpointControl::RXSTPE::SET);
1224 *ctrl_state = CtrlState::Init;
1225
1226 self.client.map(|c| c.ctrl_status_complete(endpoint));
1228 }
1229 }
1230 CtrlState::InDelay => internal_err!("Not reached"),
1231 }
1232
1233 }
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 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 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 c.packet_out(TransferType::Bulk, endpoint, packet_bytes)
1267 });
1268 match result {
1269 Some(hil::usb::OutResult::Ok) => {
1270 usbc_regs().uestaclr[endpoint].write(EndpointStatus::RXOUT::SET);
1272
1273 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 }
1284 Some(hil::usb::OutResult::Delay) => {
1285 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 usbc_regs().ueconset[endpoint].write(EndpointControl::STALLRQ::SET);
1296
1297 }
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 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 let result = self.client.map(|c| {
1328 c.packet_in(TransferType::Bulk, endpoint)
1330 });
1331 match result {
1332 Some(hil::usb::InResult::Packet(packet_bytes)) => {
1333 usbc_regs().uestaclr[endpoint].write(EndpointStatus::TXIN::SET);
1335
1336 let bank = 0;
1338 self.descriptors[endpoint][bank]
1339 .packet_size
1340 .write(PacketSize::BYTE_COUNT.val(packet_bytes as u32));
1341
1342 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 }
1353 Some(hil::usb::InResult::Delay) => {
1354 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 usbc_regs().ueconset[endpoint].write(EndpointControl::STALLRQ::SET);
1365
1366 }
1370 }
1371 }
1372 }
1373 BulkInState::Delay => {
1374 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, 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 }
1425 }
1426 }
1427 _ => None,
1428 }
1429 }
1430
1431 }
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 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 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 self.handle_requests();
1580 }
1581}