Skip to content

Commit

Permalink
Merge branch 'master' into master
Browse files Browse the repository at this point in the history
  • Loading branch information
ryan-summers authored Feb 28, 2024
2 parents b96ccb9 + fbf903e commit 7c998d8
Show file tree
Hide file tree
Showing 7 changed files with 135 additions and 29 deletions.
8 changes: 8 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,14 @@ and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.

## [Unreleased]

### Added
* A new `log` feature can be enabled to provide logging and tracing information about the USB
interface.

### Changed
* Invalid LangIDs will default to `EN_US`
* Changed handling of EP0 state to eliminate unexpected issues with device enumeration

## [0.3.1] - 2023-11-15

### Added
Expand Down
4 changes: 4 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ repository = "https://github.com/rust-embedded-community/usb-device"
defmt = { version = "0.3", optional = true }
portable-atomic = { version = "1.2.0", default-features = false }
heapless = "0.8"
log = { version = "0.4", default-features = false, optional = true}

[dev-dependencies]
rusb = "0.9.1"
Expand All @@ -22,6 +23,9 @@ rand = "0.8.5"
# Use a 256 byte buffer for control transfers instead of 128.
control-buffer-256 = []

# Enable logging and tracing via the `log` crate
log = ["dep:log"]

# Use larger endpoint buffers for highspeed operation (default fullspeed)
#
# Note: usb-device doesn't truly support high speed enumeration yet, so setting this will make
Expand Down
3 changes: 2 additions & 1 deletion src/class.rs
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,8 @@ pub trait UsbClass<B: UsbBus> {
///
/// * `index` - A string index allocated earlier with
/// [`UsbAllocator`](crate::bus::UsbBusAllocator).
/// * `lang_id` - The language ID for the string to retrieve.
/// * `lang_id` - The language ID for the string to retrieve. If the requested lang_id is not
/// valid it will default to EN_US.
fn get_string(&self, index: StringIndex, lang_id: LangID) -> Option<&str> {
let _ = (index, lang_id);
None
Expand Down
57 changes: 46 additions & 11 deletions src/control_pipe.rs
Original file line number Diff line number Diff line change
Expand Up @@ -63,24 +63,26 @@ impl<B: UsbBus> ControlPipe<'_, B> {
}

pub fn reset(&mut self) {
usb_trace!("Control pipe reset");
self.state = ControlState::Idle;
}

pub fn handle_setup(&mut self) -> Option<Request> {
let count = match self.ep_out.read(&mut self.buf[..]) {
Ok(count) => count,
Ok(count) => {
usb_trace!("Read {} bytes on EP0-OUT: {:?}", count, &self.buf[..count]);
count
}
Err(UsbError::WouldBlock) => return None,
Err(_) => {
self.set_error();
return None;
}
};

let req = match Request::parse(&self.buf[0..count]) {
Ok(req) => req,
Err(_) => {
// Failed to parse SETUP packet
self.set_error();
// Failed to parse SETUP packet. We are supposed to silently ignore this.
return None;
}
};
Expand All @@ -89,6 +91,8 @@ impl<B: UsbBus> ControlPipe<'_, B> {
// a stalled state.
self.ep_out.unstall();

usb_debug!("EP0 request received: {:?}", req);

/*sprintln!("SETUP {:?} {:?} {:?} req:{} val:{} idx:{} len:{} {:?}",
req.direction, req.request_type, req.recipient,
req.request, req.value, req.index, req.length,
Expand All @@ -102,7 +106,6 @@ impl<B: UsbBus> ControlPipe<'_, B> {

if req.length as usize > self.buf.len() {
// Data stage won't fit in buffer
self.set_error();
return None;
}

Expand Down Expand Up @@ -141,9 +144,15 @@ impl<B: UsbBus> ControlPipe<'_, B> {
}
};

usb_trace!(
"Read {} bytes on EP0-OUT: {:?}",
count,
&self.buf[i..(i + count)]
);
self.i += count;

if self.i >= self.len {
usb_debug!("Request OUT complete: {:?}", req);
self.state = ControlState::CompleteOut;
return Some(req);
}
Expand All @@ -154,11 +163,19 @@ impl<B: UsbBus> ControlPipe<'_, B> {
| ControlState::DataInLast
| ControlState::DataInZlp
| ControlState::StatusOut => {
usb_debug!(
"Control transfer completed. Current state: {:?}",
self.state
);
let _ = self.ep_out.read(&mut []);
self.state = ControlState::Idle;
}
_ => {
// Discard the packet
usb_debug!(
"Discarding EP0 data due to unexpected state. Current state: {:?}",
self.state
);
let _ = self.ep_out.read(&mut []);

// Unexpected OUT packet
Expand All @@ -181,6 +198,7 @@ impl<B: UsbBus> ControlPipe<'_, B> {
return false;
}

usb_trace!("wrote EP0: ZLP");
self.state = ControlState::DataInLast;
}
ControlState::DataInLast => {
Expand All @@ -197,8 +215,9 @@ impl<B: UsbBus> ControlPipe<'_, B> {
// IN-complete status. Just ignore this indication.
}
_ => {
// Unexpected IN packet
self.set_error();
// If we get IN-COMPLETE indications in unexpected states, it's generally because
// of control flow in previous phases updating after our packet was successfully
// sent. Ignore these indications if they don't drive any further behavior.
}
};

Expand All @@ -213,9 +232,14 @@ impl<B: UsbBus> ControlPipe<'_, B> {
Ok(c) => c,
// There isn't much we can do if the write fails, except to wait for another poll or for
// the host to resend the request.
Err(_) => return,
Err(_err) => {
usb_debug!("Failed to write EP0: {:?}", _err);
return;
}
};

usb_trace!("wrote EP0: {:?}", &buffer[self.i..(self.i + count)]);

self.i += count;

if self.i >= self.len {
Expand All @@ -232,7 +256,10 @@ impl<B: UsbBus> ControlPipe<'_, B> {
pub fn accept_out(&mut self) -> Result<()> {
match self.state {
ControlState::CompleteOut => {}
_ => return Err(UsbError::InvalidState),
_ => {
usb_debug!("Cannot ACK, invalid state: {:?}", self.state);
return Err(UsbError::InvalidState);
}
};

let _ = self.ep_in.write(&[]);
Expand All @@ -243,7 +270,10 @@ impl<B: UsbBus> ControlPipe<'_, B> {
pub fn accept_in(&mut self, f: impl FnOnce(&mut [u8]) -> Result<usize>) -> Result<()> {
let req = match self.state {
ControlState::CompleteIn(req) => req,
_ => return Err(UsbError::InvalidState),
_ => {
usb_debug!("EP0-IN cannot ACK, invalid state: {:?}", self.state);
return Err(UsbError::InvalidState);
}
};

let len = f(&mut self.buf[..])?;
Expand All @@ -259,7 +289,10 @@ impl<B: UsbBus> ControlPipe<'_, B> {
pub fn accept_in_static(&mut self, data: &'static [u8]) -> Result<()> {
let req = match self.state {
ControlState::CompleteIn(req) => req,
_ => return Err(UsbError::InvalidState),
_ => {
usb_debug!("EP0-IN cannot ACK, invalid state: {:?}", self.state);
return Err(UsbError::InvalidState);
}
};

self.static_in_buf = Some(data);
Expand All @@ -277,6 +310,7 @@ impl<B: UsbBus> ControlPipe<'_, B> {
}

pub fn reject(&mut self) -> Result<()> {
usb_debug!("EP0 transfer rejected");
if !self.waiting_for_response() {
return Err(UsbError::InvalidState);
}
Expand All @@ -286,6 +320,7 @@ impl<B: UsbBus> ControlPipe<'_, B> {
}

fn set_error(&mut self) {
usb_debug!("EP0 stalled - error");
self.state = ControlState::Error;
self.ep_out.stall();
self.ep_in.stall();
Expand Down
Loading

0 comments on commit 7c998d8

Please sign in to comment.