mirror of
https://github.com/hathach/tinyusb.git
synced 2025-01-31 05:52:55 +08:00
Implement dcd_edpt_iso_xfer() for dcd_esp32s2.c
This commit is contained in:
parent
9e2a1d2e6a
commit
59d6ed9ea4
@ -27,6 +27,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "tusb_option.h"
|
#include "tusb_option.h"
|
||||||
|
#include "common/tusb_fifo.h"
|
||||||
|
|
||||||
#if CFG_TUSB_MCU == OPT_MCU_ESP32S2 && TUSB_OPT_DEVICE_ENABLED
|
#if CFG_TUSB_MCU == OPT_MCU_ESP32S2 && TUSB_OPT_DEVICE_ENABLED
|
||||||
|
|
||||||
@ -59,6 +60,7 @@
|
|||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
uint8_t *buffer;
|
uint8_t *buffer;
|
||||||
|
tu_fifo_t * ff;
|
||||||
uint16_t total_len;
|
uint16_t total_len;
|
||||||
uint16_t queued_len;
|
uint16_t queued_len;
|
||||||
uint16_t max_size;
|
uint16_t max_size;
|
||||||
@ -354,6 +356,64 @@ bool dcd_edpt_xfer(uint8_t rhport, uint8_t ep_addr, uint8_t *buffer, uint16_t to
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool dcd_edpt_iso_xfer (uint8_t rhport, uint8_t ep_addr, tu_fifo_t * ff, uint16_t total_bytes)
|
||||||
|
{
|
||||||
|
(void)rhport;
|
||||||
|
|
||||||
|
// USB buffers always work in bytes so to avoid unnecessary divisions we demand item_size = 1
|
||||||
|
TU_ASSERT(ff->item_size == 1);
|
||||||
|
|
||||||
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
||||||
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
||||||
|
|
||||||
|
xfer_ctl_t * xfer = XFER_CTL_BASE(epnum, dir);
|
||||||
|
xfer->buffer = NULL;
|
||||||
|
xfer->ff = ff;
|
||||||
|
xfer->total_len = total_bytes;
|
||||||
|
xfer->queued_len = 0;
|
||||||
|
xfer->short_packet = false;
|
||||||
|
|
||||||
|
uint16_t num_packets = (total_bytes / xfer->max_size);
|
||||||
|
uint8_t short_packet_size = total_bytes % xfer->max_size;
|
||||||
|
|
||||||
|
// Zero-size packet is special case.
|
||||||
|
if (short_packet_size > 0 || (total_bytes == 0)) {
|
||||||
|
num_packets++;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set copy mode to constant address - required since data copied to or from the hardware USB FIFO needs to be written at a constant address
|
||||||
|
if (dir == TUSB_DIR_IN)
|
||||||
|
{
|
||||||
|
tu_fifo_set_copy_mode_write(ff, TU_FIFO_COPY_CST);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
tu_fifo_set_copy_mode_read(ff, TU_FIFO_COPY_CST);
|
||||||
|
}
|
||||||
|
|
||||||
|
ESP_LOGV(TAG, "Transfer <-> EP%i, %s, pkgs: %i, bytes: %i",
|
||||||
|
epnum, ((dir == TUSB_DIR_IN) ? "USB0.HOST (in)" : "HOST->DEV (out)"),
|
||||||
|
num_packets, total_bytes);
|
||||||
|
|
||||||
|
// IN and OUT endpoint xfers are interrupt-driven, we just schedule them
|
||||||
|
// here.
|
||||||
|
if (dir == TUSB_DIR_IN) {
|
||||||
|
// A full IN transfer (multiple packets, possibly) triggers XFRC.
|
||||||
|
USB0.in_ep_reg[epnum].dieptsiz = (num_packets << USB_D_PKTCNT0_S) | total_bytes;
|
||||||
|
USB0.in_ep_reg[epnum].diepctl |= USB_D_EPENA1_M | USB_D_CNAK1_M; // Enable | CNAK
|
||||||
|
|
||||||
|
// Enable fifo empty interrupt only if there are something to put in the fifo.
|
||||||
|
if(total_bytes != 0) {
|
||||||
|
USB0.dtknqr4_fifoemptymsk |= (1 << epnum);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Each complete packet for OUT xfers triggers XFRC.
|
||||||
|
USB0.out_ep_reg[epnum].doeptsiz |= USB_PKTCNT0_M | ((xfer->max_size & USB_XFERSIZE0_V) << USB_XFERSIZE0_S);
|
||||||
|
USB0.out_ep_reg[epnum].doepctl |= USB_EPENA0_M | USB_CNAK0_M;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
void dcd_edpt_stall(uint8_t rhport, uint8_t ep_addr)
|
void dcd_edpt_stall(uint8_t rhport, uint8_t ep_addr)
|
||||||
{
|
{
|
||||||
(void)rhport;
|
(void)rhport;
|
||||||
@ -462,36 +522,46 @@ static void receive_packet(xfer_ctl_t *xfer, /* usb_out_endpoint_t * out_ep, */
|
|||||||
to_recv_size = (xfer_size > xfer->max_size) ? xfer->max_size : xfer_size;
|
to_recv_size = (xfer_size > xfer->max_size) ? xfer->max_size : xfer_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t to_recv_rem = to_recv_size % 4;
|
// Common buffer read
|
||||||
uint16_t to_recv_size_aligned = to_recv_size - to_recv_rem;
|
if (xfer->buffer)
|
||||||
|
{
|
||||||
|
uint8_t to_recv_rem = to_recv_size % 4;
|
||||||
|
uint16_t to_recv_size_aligned = to_recv_size - to_recv_rem;
|
||||||
|
|
||||||
// Do not assume xfer buffer is aligned.
|
// Do not assume xfer buffer is aligned.
|
||||||
uint8_t *base = (xfer->buffer + xfer->queued_len);
|
uint8_t *base = (xfer->buffer + xfer->queued_len);
|
||||||
|
|
||||||
// This for loop always runs at least once- skip if less than 4 bytes
|
// This for loop always runs at least once- skip if less than 4 bytes
|
||||||
// to collect.
|
// to collect.
|
||||||
if (to_recv_size >= 4) {
|
if (to_recv_size >= 4) {
|
||||||
for (uint16_t i = 0; i < to_recv_size_aligned; i += 4) {
|
for (uint16_t i = 0; i < to_recv_size_aligned; i += 4) {
|
||||||
|
uint32_t tmp = (*rx_fifo);
|
||||||
|
base[i] = tmp & 0x000000FF;
|
||||||
|
base[i + 1] = (tmp & 0x0000FF00) >> 8;
|
||||||
|
base[i + 2] = (tmp & 0x00FF0000) >> 16;
|
||||||
|
base[i + 3] = (tmp & 0xFF000000) >> 24;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do not read invalid bytes from RX FIFO.
|
||||||
|
if (to_recv_rem != 0) {
|
||||||
uint32_t tmp = (*rx_fifo);
|
uint32_t tmp = (*rx_fifo);
|
||||||
base[i] = tmp & 0x000000FF;
|
uint8_t *last_32b_bound = base + to_recv_size_aligned;
|
||||||
base[i + 1] = (tmp & 0x0000FF00) >> 8;
|
|
||||||
base[i + 2] = (tmp & 0x00FF0000) >> 16;
|
last_32b_bound[0] = tmp & 0x000000FF;
|
||||||
base[i + 3] = (tmp & 0xFF000000) >> 24;
|
if (to_recv_rem > 1) {
|
||||||
|
last_32b_bound[1] = (tmp & 0x0000FF00) >> 8;
|
||||||
|
}
|
||||||
|
if (to_recv_rem > 2) {
|
||||||
|
last_32b_bound[2] = (tmp & 0x00FF0000) >> 16;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
else
|
||||||
// Do not read invalid bytes from RX FIFO.
|
{
|
||||||
if (to_recv_rem != 0) {
|
// Ring buffer
|
||||||
uint32_t tmp = (*rx_fifo);
|
tu_fifo_write_n(xfer->ff, (const void *) rx_fifo, to_recv_size);
|
||||||
uint8_t *last_32b_bound = base + to_recv_size_aligned;
|
|
||||||
|
|
||||||
last_32b_bound[0] = tmp & 0x000000FF;
|
|
||||||
if (to_recv_rem > 1) {
|
|
||||||
last_32b_bound[1] = (tmp & 0x0000FF00) >> 8;
|
|
||||||
}
|
|
||||||
if (to_recv_rem > 2) {
|
|
||||||
last_32b_bound[2] = (tmp & 0x00FF0000) >> 16;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
xfer->queued_len += xfer_size;
|
xfer->queued_len += xfer_size;
|
||||||
@ -510,37 +580,45 @@ static void transmit_packet(xfer_ctl_t *xfer, volatile usb_in_endpoint_t *in_ep,
|
|||||||
xfer->queued_len = xfer->total_len - remaining;
|
xfer->queued_len = xfer->total_len - remaining;
|
||||||
|
|
||||||
uint16_t to_xfer_size = (remaining > xfer->max_size) ? xfer->max_size : remaining;
|
uint16_t to_xfer_size = (remaining > xfer->max_size) ? xfer->max_size : remaining;
|
||||||
uint8_t to_xfer_rem = to_xfer_size % 4;
|
|
||||||
uint16_t to_xfer_size_aligned = to_xfer_size - to_xfer_rem;
|
|
||||||
|
|
||||||
// Buffer might not be aligned to 32b, so we need to force alignment
|
if (xfer->buffer)
|
||||||
// by copying to a temp var.
|
{
|
||||||
uint8_t *base = (xfer->buffer + xfer->queued_len);
|
uint8_t to_xfer_rem = to_xfer_size % 4;
|
||||||
|
uint16_t to_xfer_size_aligned = to_xfer_size - to_xfer_rem;
|
||||||
|
|
||||||
|
// Buffer might not be aligned to 32b, so we need to force alignment
|
||||||
|
// by copying to a temp var.
|
||||||
|
uint8_t *base = (xfer->buffer + xfer->queued_len);
|
||||||
|
|
||||||
|
// This for loop always runs at least once- skip if less than 4 bytes
|
||||||
|
// to send off.
|
||||||
|
if (to_xfer_size >= 4) {
|
||||||
|
for (uint16_t i = 0; i < to_xfer_size_aligned; i += 4) {
|
||||||
|
uint32_t tmp = base[i] | (base[i + 1] << 8) |
|
||||||
|
(base[i + 2] << 16) | (base[i + 3] << 24);
|
||||||
|
(*tx_fifo) = tmp;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do not read beyond end of buffer if not divisible by 4.
|
||||||
|
if (to_xfer_rem != 0) {
|
||||||
|
uint32_t tmp = 0;
|
||||||
|
uint8_t *last_32b_bound = base + to_xfer_size_aligned;
|
||||||
|
|
||||||
|
tmp |= last_32b_bound[0];
|
||||||
|
if (to_xfer_rem > 1) {
|
||||||
|
tmp |= (last_32b_bound[1] << 8);
|
||||||
|
}
|
||||||
|
if (to_xfer_rem > 2) {
|
||||||
|
tmp |= (last_32b_bound[2] << 16);
|
||||||
|
}
|
||||||
|
|
||||||
// This for loop always runs at least once- skip if less than 4 bytes
|
|
||||||
// to send off.
|
|
||||||
if (to_xfer_size >= 4) {
|
|
||||||
for (uint16_t i = 0; i < to_xfer_size_aligned; i += 4) {
|
|
||||||
uint32_t tmp = base[i] | (base[i + 1] << 8) |
|
|
||||||
(base[i + 2] << 16) | (base[i + 3] << 24);
|
|
||||||
(*tx_fifo) = tmp;
|
(*tx_fifo) = tmp;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else
|
||||||
// Do not read beyond end of buffer if not divisible by 4.
|
{
|
||||||
if (to_xfer_rem != 0) {
|
tu_fifo_read_n(xfer->ff, (void *) tx_fifo, to_xfer_size);
|
||||||
uint32_t tmp = 0;
|
|
||||||
uint8_t *last_32b_bound = base + to_xfer_size_aligned;
|
|
||||||
|
|
||||||
tmp |= last_32b_bound[0];
|
|
||||||
if (to_xfer_rem > 1) {
|
|
||||||
tmp |= (last_32b_bound[1] << 8);
|
|
||||||
}
|
|
||||||
if (to_xfer_rem > 2) {
|
|
||||||
tmp |= (last_32b_bound[2] << 16);
|
|
||||||
}
|
|
||||||
|
|
||||||
(*tx_fifo) = tmp;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user