Skip to content

Commit

Permalink
WIP timeouts
Browse files Browse the repository at this point in the history
  • Loading branch information
tlyu committed Mar 15, 2023
1 parent 04b958f commit f0c281b
Show file tree
Hide file tree
Showing 2 changed files with 124 additions and 6 deletions.
124 changes: 118 additions & 6 deletions cores/arduino/USBCore.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -133,6 +133,14 @@ void EPBuffer<L>::init(uint8_t ep)
this->reset();
this->rxWaiting = false;
this->txWaiting = false;
this->timedOut = false;
this->timeout = 100;
}

template<size_t L>
void EPBuffer<L>::setTimeout(uint16_t timeout)
{
this->timeout = timeout;
}

template<size_t L>
Expand Down Expand Up @@ -236,6 +244,9 @@ void EPBuffer<L>::flush()
// fall through
case USBD_CONFIGURED:
case USBD_SUSPENDED: {
if (this->timedOut) {
break;
}
// This will temporarily reenable and disable interrupts
auto canWrite = this->waitForWriteComplete();
if (canWrite) {
Expand All @@ -246,6 +257,7 @@ void EPBuffer<L>::flush()
// Only start the next transmission if the device hasn't been
// reset.
this->txWaiting = true;
this->startTime = millis();
usbd_ep_send(&USBCore().usbDev(), this->ep, (uint8_t *)this->buf, this->len());
USBCore().logEP('>', this->ep, '>', this->len());
}
Expand Down Expand Up @@ -287,6 +299,7 @@ template<size_t L>
void EPBuffer<L>::transcIn()
{
this->txWaiting = false;
this->timedOut = false;
}

// Unused?
Expand All @@ -302,23 +315,117 @@ uint8_t* EPBuffer<L>::ptr()
template<size_t L>
bool EPBuffer<L>::waitForWriteComplete()
{
// auto start = getCurrentMillis();
auto ok = true;
auto desc = *EPBuffers().desc(this->ep);
auto udev = &USBCore().usbDev();

/*
* For interrupt EPs, count from now instead of from previous send.
* Otherwise, if a send occurs while the bus is suspended, the packet will
* be cheated of its full timeout.
*/
if (desc.type() == USB_EP_ATTR_INT) {
this->startTime = millis();
}
do {
usb_enable_interrupts();
switch (USBCore().usbDev().cur_status) {
switch (udev->cur_status) {
case USBD_DEFAULT:
case USBD_ADDRESSED:
ok = false;
break;
default:
break;
}
// if (getCurrentMillis() - start > 5) {
// EPBuffers().buf(ep).transcIn();
// ok = false;
// }
usb_disable_interrupts();
if (this->txWaiting && millis() - this->startTime > this->timeout) {
USBCore().logEP('X', this->ep, '>', this->len());
/*
* For interrupt EPs, overwrite the previous packet. In the common
* case of a key press HID report initiating a remote wakeup,
* transmission of the key release event might be delayed, because
* the host can't poll while the bus is suspended. If the
* application sends a key release event before the key press event
* is transmitted, that might time out. If the key release event is
* discarded due to the timeout, the host will never receive the
* key release event, leading to a stuck or repeating key.
*/
if (desc.type() == USB_EP_ATTR_INT) {
/*
* Set the endpoint status to NAK, in an attempt to prevent
* the queued packet from being transmitted while being
* overwritten.
*
* This must be done in two steps. Unfortunately, the USBD
* peripheral's design doesn't seem to make it possible to
* atomically abort a queued transmission. Even if interrupts
* are disabled, the peripheral state machine continues to run.
*
* The endpoint status control bits are toggle-only (toggle on
* writing 1; unchanged on writing 0). If the TX state
* transitions from VALID to NAK during the read-modify-write,
* a toggle operation (XOR 0b01) intended to transition from
* VALID (0b11) to NAK (0b10) will set the status back to
* VALID. This will cause the previous packet to be sent again.
*
* Instead, set the status to DISABLED (0b00, via XOR 0b11),
* which will become STALL (0b01) if the peripheral transitions
* the status to NAK between the read and the write. This can
* happen if the transmission was in progress and the host ACKs
* exactly between the read and the write to update the status
* bits.
*
* Possible states after the first USBD_EP_TX_STAT_SET:
*
* - DISABLED (metastable if TX in progress)
* - STALL (stable; only possible if we lost the r-m-w race)
* - NAK (stable; only possible if TX completed after toggle)
*
* There is a small theoretical chance that the host will poll
* the endpoint again between the two updates to the status
* bits, which could lead to the host receiving a STALL
* handshake, which would put the data toggle bit out of sync.
*
* The second host poll cannot complete sooner than 34 bit times
* after the ACK from the first host poll. (2 bits inter-packet
* delay, 8 bits sync, 8 bits PID, 16 bits addr/endpoint/CRC,
* about 2.83 microseconds) The second status bit update will
* complete before then, assuming reasonable CPU clock speeds
* and no lengthy interrupt handler executions occurring.
*
* On the other hand, it's unknown when exactly the peripheral
* checks the endpoint status bits to decide which response to
* send to an IN poll. It might latch them at EOP, or SOP, or
* maybe PID, or it might wait until after checking the CRC. So
* the actual available time might be between 0 and 34 bit
* times.
*
* We could globally disable interrupts to mitigate that, if
* necessary.
*/
USBD_EP_TX_STAT_SET(this->ep, EPTX_DISABLED);
if (USBD_EPxCS(this->ep) & EPxCS_TX_STA == EPTX_DISABLED) {
/*
* If the endpoint is disabled, there might still be a
* transmission in progress. Wait for it to complete:
*
* 1 byte sync + 1 byte PID + 64 bytes data + 2 bytes CRC,
* 1 turn-around time,
* 1 byte sync + 1 byte PID (handshake)
* assuming maximum bit stuffing at full speed (12MHz).
*/
delayMicroseconds(56);
}
// In case TX was in progress and completed
USBD_EP_TX_ST_CLEAR(this->ep);
USBD_EP_TX_STAT_SET(this->ep, EPTX_NAK);
break;
} else {
ok = false;
this->timedOut = true;
break;
}
}
} while (ok && this->txWaiting);
return ok;
}
Expand Down Expand Up @@ -874,6 +981,11 @@ int USBCore_::flush(uint8_t ep)
return 0;
}

void USBCore_::setTimeout(uint8_t ep, uint16_t timeout)
{
EPBuffers().buf(ep).setTimeout(timeout);
}

void USBCore_::transcSetupHelper(usb_dev* usbd, uint8_t ep)
{
USBCore_* core = (USBCore_*)usbd->user_data;
Expand Down
6 changes: 6 additions & 0 deletions cores/arduino/USBCore.h
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,7 @@ class EPBuffer
void flush();
uint8_t* ptr();
void enableOutEndpoint();
void setTimeout(uint16_t timeout);

void transcIn();
void transcOut();
Expand Down Expand Up @@ -131,6 +132,10 @@ class EPBuffer
*/
volatile bool currentlyFlushing = false;

volatile uint32_t startTime;
uint16_t timeout;
volatile bool timedOut;

uint8_t ep;
};

Expand Down Expand Up @@ -174,6 +179,7 @@ class USBCore_
int recv(uint8_t ep);
int flush(uint8_t ep);
void setResetHook(void (*hook)());
void setTimeout(uint8_t ep, uint16_t timeout);

// Debug counters
volatile uint16_t nreset;
Expand Down

0 comments on commit f0c281b

Please sign in to comment.