Merge pull request #405 from ithinuel/fix-usb-ep0-out

Fix usb missed ep0 out packet and add recommended sync delays
This commit is contained in:
Jan Niehusmann 2022-08-01 15:32:45 +02:00 committed by GitHub
commit 6dafcc1ac0
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -212,10 +212,9 @@ impl Inner {
.modify(|_, w| w.ep0_int_1buf().set_bit()); .modify(|_, w| w.ep0_int_1buf().set_bit());
// expect ctrl ep to receive on DATA first // 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[0].write(|w| w.pid_0().set_bit());
self.ctrl_dpram.ep_buffer_control[1].write(|w| { self.ctrl_dpram.ep_buffer_control[1].write(|w| w.pid_0().set_bit());
w.available_0().set_bit(); cortex_m::asm::delay(12);
w.pid_0().set_bit() self.ctrl_dpram.ep_buffer_control[1].write(|w| w.available_0().set_bit());
});
for (index, ep) in itertools::interleave( for (index, ep) in itertools::interleave(
self.in_endpoints.iter().skip(1), // skip control endpoint self.in_endpoints.iter().skip(1), // skip control endpoint
@ -246,10 +245,11 @@ impl Inner {
buf_control.write(|w| w.pid_0().set_bit()); buf_control.write(|w| w.pid_0().set_bit());
} else { } else {
buf_control.write(|w| unsafe { buf_control.write(|w| unsafe {
w.available_0().set_bit();
w.pid_0().clear_bit(); w.pid_0().clear_bit();
w.length_0().bits(ep.max_packet_size) 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); ep_buf[..buf.len()].copy_from_slice(buf);
buf_control.modify(|r, w| unsafe { buf_control.modify(|r, w| unsafe {
w.available_0().set_bit();
w.length_0().bits(buf.len() as u16); w.length_0().bits(buf.len() as u16);
w.full_0().set_bit(); w.full_0().set_bit();
w.pid_0().bit(!r.pid_0().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()) Ok(buf.len())
} }
@ -295,63 +296,71 @@ impl Inner {
let buf_control_val = buf_control.read(); let buf_control_val = buf_control.read();
let process_setup = index == 0 && self.read_setup; 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 // assume we want to read the setup request
//
// 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 // the OUT packet will be either data or a status zlp
// clear setup request flag let len = 8;
self.ctrl_reg.sie_status.write(|w| w.setup_rec().set_bit()); let ep_buf =
( unsafe { core::slice::from_raw_parts(USBCTRL_DPRAM::ptr() as *const u8, len) };
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() { if len > buf.len() {
return Err(UsbError::BufferOverflow); return Err(UsbError::BufferOverflow);
} }
buf[..len].copy_from_slice(&ep_buf[..len]); buf[..len].copy_from_slice(&ep_buf[..len]);
if process_setup { // Next packet will be on DATA1 so clear pid_0 so it gets flipped by next buf config
self.read_setup = false; self.ctrl_dpram.ep_buffer_control[0].modify(|_, w| w.pid_0().clear_bit());
// clear setup request flag
self.ctrl_reg.sie_status.write(|w| w.setup_rec().set_bit());
// clear any out standing out flag e.g. in case a zlp got discarded // 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) }); 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 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; let expect_data_or_zlp = is_in_request || data_length != 0;
buf_control.modify(|_, w| unsafe { 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.length_0().bits(ep.max_packet_size);
w.full_0().clear_bit(); w.full_0().clear_bit();
w.pid_0().set_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 { } else {
buf_control.modify(|r, w| unsafe { if buf_control_val.full_0().bit_is_clear() {
w.available_0().set_bit(); return Err(UsbError::WouldBlock);
w.length_0().bits(ep.max_packet_size); }
w.full_0().clear_bit(); let len = buf_control_val.length_0().bits().into();
w.pid_0().bit(!r.pid_0().bit()) if len > buf.len() {
}); return Err(UsbError::BufferOverflow);
}
buf[..len].copy_from_slice(&ep.get_buf()[..len]);
// Clear OUT flag once it is read. // Clear OUT flag once it is read.
self.ctrl_reg self.ctrl_reg
.buff_status .buff_status
.write(|w| unsafe { w.bits(1 << (index * 2 + 1)) }); .write(|w| unsafe { w.bits(1 << (index * 2 + 1)) });
}
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())
});
if index != 0 || len == ep.max_packet_size.into() {
// only mark as available on the control endpoint if and only if the packet was
// max_packet_size
cortex_m::asm::delay(12);
buf_control.modify(|_, w| w.available_0().set_bit());
}
Ok(len) Ok(len)
} }
}
} }
/// Usb bus /// Usb bus
@ -574,10 +583,12 @@ impl UsbBusTrait for UsbBus {
for i in 0..32u32 { for i in 0..32u32 {
let mask = 1 << i; let mask = 1 << i;
if (buff_status & mask) == mask { if (buff_status & mask) == mask {
if (i & 1) == 0 { let is_in = (i & 1) == 0;
ep_in_complete |= 1 << (i / 2); let ep_idx = i / 2;
if is_in {
ep_in_complete |= 1 << ep_idx;
} else { } else {
ep_out |= 1 << (i / 2); ep_out |= 1 << ep_idx;
} }
} }
} }