mirror of
https://github.com/italicsjenga/rp-hal-boards.git
synced 2025-01-23 01:36:35 +11:00
Add the required synchronisation delays
As per 4.1.2.5.1, the access to the DPSRAM should "be considered asynchronous and not atomic". It is recommended to write to buffer control register in two steps. A first one to configure all bits but Available. Wait clk_sys/clk_usb (typically 125MHz/48MHz). Then set the available bit (if required).
This commit is contained in:
parent
69c2dd2c2b
commit
067f1396d9
1 changed files with 49 additions and 42 deletions
|
@ -212,10 +212,9 @@ impl Inner {
|
|||
.modify(|_, w| w.ep0_int_1buf().set_bit());
|
||||
// expect ctrl ep to receive on DATA first
|
||||
self.ctrl_dpram.ep_buffer_control[0].write(|w| w.pid_0().set_bit());
|
||||
self.ctrl_dpram.ep_buffer_control[1].write(|w| {
|
||||
w.available_0().set_bit();
|
||||
w.pid_0().set_bit()
|
||||
});
|
||||
self.ctrl_dpram.ep_buffer_control[1].write(|w| w.pid_0().set_bit());
|
||||
cortex_m::asm::delay(12);
|
||||
self.ctrl_dpram.ep_buffer_control[1].write(|w| w.available_0().set_bit());
|
||||
|
||||
for (index, ep) in itertools::interleave(
|
||||
self.in_endpoints.iter().skip(1), // skip control endpoint
|
||||
|
@ -246,10 +245,11 @@ impl Inner {
|
|||
buf_control.write(|w| w.pid_0().set_bit());
|
||||
} else {
|
||||
buf_control.write(|w| unsafe {
|
||||
w.available_0().set_bit();
|
||||
w.pid_0().clear_bit();
|
||||
w.length_0().bits(ep.max_packet_size)
|
||||
});
|
||||
cortex_m::asm::delay(12);
|
||||
buf_control.write(|w| w.available_0().set_bit());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -274,11 +274,12 @@ impl Inner {
|
|||
ep_buf[..buf.len()].copy_from_slice(buf);
|
||||
|
||||
buf_control.modify(|r, w| unsafe {
|
||||
w.available_0().set_bit();
|
||||
w.length_0().bits(buf.len() as u16);
|
||||
w.full_0().set_bit();
|
||||
w.pid_0().bit(!r.pid_0().bit())
|
||||
});
|
||||
cortex_m::asm::delay(12);
|
||||
buf_control.modify(|_, w| w.available_0().set_bit());
|
||||
|
||||
Ok(buf.len())
|
||||
}
|
||||
|
@ -295,62 +296,66 @@ impl Inner {
|
|||
let buf_control_val = buf_control.read();
|
||||
|
||||
let process_setup = index == 0 && self.read_setup;
|
||||
let (ep_buf, len) = if process_setup {
|
||||
if process_setup {
|
||||
// assume we want to read the setup request
|
||||
//
|
||||
// the OUT packet will be either data or a status zlp
|
||||
let len = 8;
|
||||
let ep_buf =
|
||||
unsafe { core::slice::from_raw_parts(USBCTRL_DPRAM::ptr() as *const u8, len) };
|
||||
if len > buf.len() {
|
||||
return Err(UsbError::BufferOverflow);
|
||||
}
|
||||
|
||||
buf[..len].copy_from_slice(&ep_buf[..len]);
|
||||
|
||||
// Next packet will be on DATA1 so clear pid_0 so it gets flipped by next buf config
|
||||
self.ctrl_dpram.ep_buffer_control[0].modify(|_, w| w.pid_0().clear_bit());
|
||||
// the OUT packet will be either data or a status zlp
|
||||
// clear setup request flag
|
||||
self.ctrl_reg.sie_status.write(|w| w.setup_rec().set_bit());
|
||||
(
|
||||
unsafe { core::slice::from_raw_parts(USBCTRL_DPRAM::ptr() as *const u8, 8) },
|
||||
8,
|
||||
)
|
||||
} else {
|
||||
if buf_control_val.full_0().bit_is_clear() {
|
||||
return Err(UsbError::WouldBlock);
|
||||
}
|
||||
let len: usize = buf_control_val.length_0().bits().into();
|
||||
(ep.get_buf(), len)
|
||||
};
|
||||
|
||||
if len > buf.len() {
|
||||
return Err(UsbError::BufferOverflow);
|
||||
}
|
||||
|
||||
buf[..len].copy_from_slice(&ep_buf[..len]);
|
||||
|
||||
if process_setup {
|
||||
self.read_setup = false;
|
||||
|
||||
// clear any out standing out flag e.g. in case a zlp got discarded
|
||||
self.ctrl_reg.buff_status.write(|w| unsafe { w.bits(2) });
|
||||
|
||||
let data_length = u16::from(buf[6]) | (u16::from(buf[7]) << 8);
|
||||
let is_in_request = (buf[0] & 0x80) == 0x80;
|
||||
let data_length = u16::from(buf[6]) | (u16::from(buf[7]) << 8);
|
||||
let expect_data_or_zlp = is_in_request || data_length != 0;
|
||||
|
||||
buf_control.modify(|_, w| unsafe {
|
||||
// enable if and only if a dataphase is expected.
|
||||
w.available_0().bit(expect_data_or_zlp);
|
||||
w.length_0().bits(ep.max_packet_size);
|
||||
w.full_0().clear_bit();
|
||||
w.pid_0().set_bit()
|
||||
});
|
||||
// enable if and only if a dataphase is expected.
|
||||
cortex_m::asm::delay(12);
|
||||
buf_control.modify(|_, w| w.available_0().bit(expect_data_or_zlp));
|
||||
|
||||
self.read_setup = false;
|
||||
Ok(len)
|
||||
} else {
|
||||
buf_control.modify(|r, w| unsafe {
|
||||
w.available_0().set_bit();
|
||||
w.length_0().bits(ep.max_packet_size);
|
||||
w.full_0().clear_bit();
|
||||
w.pid_0().bit(!r.pid_0().bit())
|
||||
});
|
||||
if buf_control_val.full_0().bit_is_clear() {
|
||||
return Err(UsbError::WouldBlock);
|
||||
}
|
||||
let len = buf_control_val.length_0().bits().into();
|
||||
if len > buf.len() {
|
||||
return Err(UsbError::BufferOverflow);
|
||||
}
|
||||
|
||||
buf[..len].copy_from_slice(&ep.get_buf()[..len]);
|
||||
// Clear OUT flag once it is read.
|
||||
self.ctrl_reg
|
||||
.buff_status
|
||||
.write(|w| unsafe { w.bits(1 << (index * 2 + 1)) });
|
||||
}
|
||||
|
||||
Ok(len)
|
||||
buf_control.modify(|r, w| unsafe {
|
||||
w.length_0().bits(ep.max_packet_size);
|
||||
w.full_0().clear_bit();
|
||||
w.pid_0().bit(!r.pid_0().bit())
|
||||
});
|
||||
cortex_m::asm::delay(12);
|
||||
buf_control.modify(|_, w| w.available_0().set_bit());
|
||||
Ok(len)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -548,10 +553,12 @@ impl UsbBusTrait for UsbBus {
|
|||
for i in 0..32u32 {
|
||||
let mask = 1 << i;
|
||||
if (buff_status & mask) == mask {
|
||||
if (i & 1) == 0 {
|
||||
ep_in_complete |= 1 << (i / 2);
|
||||
let is_in = (i & 1) == 0;
|
||||
let ep_idx = i / 2;
|
||||
if is_in {
|
||||
ep_in_complete |= 1 << ep_idx;
|
||||
} else {
|
||||
ep_out |= 1 << (i / 2);
|
||||
ep_out |= 1 << ep_idx;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue