Merge pull request #1676 from tswan-quasi/master
dcd_lpc_ip3511: isochronous support and endpoint accidental write fix
This commit is contained in:
@@ -43,6 +43,7 @@ SRC_C += \
|
|||||||
$(MCU_DIR)/drivers/fsl_power.c \
|
$(MCU_DIR)/drivers/fsl_power.c \
|
||||||
$(MCU_DIR)/drivers/fsl_reset.c \
|
$(MCU_DIR)/drivers/fsl_reset.c \
|
||||||
$(SDK_DIR)/drivers/lpc_gpio/fsl_gpio.c \
|
$(SDK_DIR)/drivers/lpc_gpio/fsl_gpio.c \
|
||||||
|
$(SDK_DIR)/drivers/common/fsl_common_arm.c \
|
||||||
$(SDK_DIR)/drivers/flexcomm/fsl_flexcomm.c \
|
$(SDK_DIR)/drivers/flexcomm/fsl_flexcomm.c \
|
||||||
$(SDK_DIR)/drivers/flexcomm/fsl_usart.c \
|
$(SDK_DIR)/drivers/flexcomm/fsl_usart.c \
|
||||||
lib/sct_neopixel/sct_neopixel.c
|
lib/sct_neopixel/sct_neopixel.c
|
||||||
|
@@ -78,8 +78,10 @@ typedef struct {
|
|||||||
|
|
||||||
// Max nbytes for each control/bulk/interrupt transfer
|
// Max nbytes for each control/bulk/interrupt transfer
|
||||||
enum {
|
enum {
|
||||||
NBYTES_CBI_FULLSPEED_MAX = 64,
|
NBYTES_ISO_FS_MAX = 1023, // FS ISO
|
||||||
NBYTES_CBI_HIGHSPEED_MAX = 32767 // can be up to all 15-bit, but only tested with 4096
|
NBYTES_ISO_HS_MAX = 1024, // HS ISO
|
||||||
|
NBYTES_CBI_FS_MAX = 64, // FS control/bulk/interrupt
|
||||||
|
NBYTES_CBI_HS_MAX = 32767 // can be up to all 15-bit, but only tested with 4096
|
||||||
};
|
};
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
@@ -112,6 +114,8 @@ enum {
|
|||||||
typedef union TU_ATTR_PACKED
|
typedef union TU_ATTR_PACKED
|
||||||
{
|
{
|
||||||
// Full and High speed has different bit layout for buffer_offset and nbytes
|
// Full and High speed has different bit layout for buffer_offset and nbytes
|
||||||
|
// TODO FS/HS layout depends on the max speed of controller e.g
|
||||||
|
// lpc55s69 PORT0 is only FS but actually has the same layout as HS on port1
|
||||||
|
|
||||||
// Buffer (aligned 64) = DATABUFSTART [31:22] | buffer_offset [21:6]
|
// Buffer (aligned 64) = DATABUFSTART [31:22] | buffer_offset [21:6]
|
||||||
volatile struct {
|
volatile struct {
|
||||||
@@ -175,6 +179,9 @@ typedef struct
|
|||||||
// Use CFG_TUSB_MEM_SECTION to place it accordingly.
|
// Use CFG_TUSB_MEM_SECTION to place it accordingly.
|
||||||
CFG_TUSB_MEM_SECTION TU_ATTR_ALIGNED(256) static dcd_data_t _dcd;
|
CFG_TUSB_MEM_SECTION TU_ATTR_ALIGNED(256) static dcd_data_t _dcd;
|
||||||
|
|
||||||
|
// Dummy buffer to fix ZLPs overwriting the buffer (probably an USB/DMA controller bug)
|
||||||
|
CFG_TUSB_MEM_SECTION TU_ATTR_ALIGNED(64) static uint8_t dummy[8];
|
||||||
|
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
// Multiple Controllers
|
// Multiple Controllers
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
@@ -225,8 +232,36 @@ static inline uint8_t ep_addr2id(uint8_t ep_addr)
|
|||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
// CONTROLLER API
|
// CONTROLLER API
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
|
|
||||||
|
static void prepare_setup_packet(uint8_t rhport)
|
||||||
|
{
|
||||||
|
if (_dcd_controller[rhport].max_speed == TUSB_SPEED_FULL )
|
||||||
|
{
|
||||||
|
_dcd.ep[0][1].buffer_fs.offset = get_buf_offset(_dcd.setup_packet);
|
||||||
|
}else
|
||||||
|
{
|
||||||
|
_dcd.ep[0][1].buffer_hs.offset = get_buf_offset(_dcd.setup_packet);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void edpt_reset(uint8_t rhport, uint8_t ep_id)
|
||||||
|
{
|
||||||
|
(void) rhport;
|
||||||
|
tu_memclr(&_dcd.ep[ep_id], sizeof(_dcd.ep[ep_id]));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void edpt_reset_all(uint8_t rhport)
|
||||||
|
{
|
||||||
|
for (uint8_t ep_id = 0; ep_id < 2*_dcd_controller[rhport].ep_pairs; ++ep_id)
|
||||||
|
{
|
||||||
|
edpt_reset(rhport, ep_id);
|
||||||
|
}
|
||||||
|
prepare_setup_packet(rhport);
|
||||||
|
}
|
||||||
void dcd_init(uint8_t rhport)
|
void dcd_init(uint8_t rhport)
|
||||||
{
|
{
|
||||||
|
edpt_reset_all(rhport);
|
||||||
|
|
||||||
dcd_registers_t* dcd_reg = _dcd_controller[rhport].regs;
|
dcd_registers_t* dcd_reg = _dcd_controller[rhport].regs;
|
||||||
|
|
||||||
dcd_reg->EPLISTSTART = (uint32_t) _dcd.ep;
|
dcd_reg->EPLISTSTART = (uint32_t) _dcd.ep;
|
||||||
@@ -310,18 +345,13 @@ void dcd_edpt_clear_stall(uint8_t rhport, uint8_t ep_addr)
|
|||||||
|
|
||||||
bool dcd_edpt_open(uint8_t rhport, tusb_desc_endpoint_t const * p_endpoint_desc)
|
bool dcd_edpt_open(uint8_t rhport, tusb_desc_endpoint_t const * p_endpoint_desc)
|
||||||
{
|
{
|
||||||
(void) rhport;
|
|
||||||
|
|
||||||
// TODO not support ISO yet
|
|
||||||
TU_VERIFY(p_endpoint_desc->bmAttributes.xfer != TUSB_XFER_ISOCHRONOUS);
|
|
||||||
|
|
||||||
//------------- Prepare Queue Head -------------//
|
//------------- Prepare Queue Head -------------//
|
||||||
uint8_t ep_id = ep_addr2id(p_endpoint_desc->bEndpointAddress);
|
uint8_t ep_id = ep_addr2id(p_endpoint_desc->bEndpointAddress);
|
||||||
|
|
||||||
// Check if endpoint is available
|
// Check if endpoint is available
|
||||||
TU_ASSERT( _dcd.ep[ep_id][0].disable && _dcd.ep[ep_id][1].disable );
|
TU_ASSERT( _dcd.ep[ep_id][0].disable && _dcd.ep[ep_id][1].disable );
|
||||||
|
|
||||||
tu_memclr(_dcd.ep[ep_id], 2*sizeof(ep_cmd_sts_t));
|
edpt_reset(rhport, ep_id);
|
||||||
_dcd.ep[ep_id][0].is_iso = (p_endpoint_desc->bmAttributes.xfer == TUSB_XFER_ISOCHRONOUS);
|
_dcd.ep[ep_id][0].is_iso = (p_endpoint_desc->bmAttributes.xfer == TUSB_XFER_ISOCHRONOUS);
|
||||||
|
|
||||||
// Enable EP interrupt
|
// Enable EP interrupt
|
||||||
@@ -333,19 +363,20 @@ bool dcd_edpt_open(uint8_t rhport, tusb_desc_endpoint_t const * p_endpoint_desc)
|
|||||||
|
|
||||||
void dcd_edpt_close_all (uint8_t rhport)
|
void dcd_edpt_close_all (uint8_t rhport)
|
||||||
{
|
{
|
||||||
(void) rhport;
|
for (uint8_t ep_id = 0; ep_id < 2*_dcd_controller[rhport].ep_pairs; ++ep_id)
|
||||||
// TODO implement dcd_edpt_close_all()
|
{
|
||||||
|
_dcd.ep[ep_id][0].active = _dcd.ep[ep_id][0].active = 0; // TODO proper way is to EPSKIP then wait ep[][].active then write ep[][].disable (see table 778 in LPC55S69 Use Manual)
|
||||||
|
_dcd.ep[ep_id][0].disable = _dcd.ep[ep_id][1].disable = 1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void prepare_setup_packet(uint8_t rhport)
|
void dcd_edpt_close(uint8_t rhport, uint8_t ep_addr)
|
||||||
{
|
{
|
||||||
if (_dcd_controller[rhport].max_speed == TUSB_SPEED_FULL )
|
(void) rhport;
|
||||||
{
|
|
||||||
_dcd.ep[0][1].buffer_fs.offset = get_buf_offset(_dcd.setup_packet);;
|
uint8_t ep_id = ep_addr2id(ep_addr);
|
||||||
}else
|
_dcd.ep[ep_id][0].active = _dcd.ep[ep_id][0].active = 0; // TODO proper way is to EPSKIP then wait ep[][].active then write ep[][].disable (see table 778 in LPC55S69 Use Manual)
|
||||||
{
|
_dcd.ep[ep_id][0].disable = _dcd.ep[ep_id][1].disable = 1;
|
||||||
_dcd.ep[0][1].buffer_hs.offset = get_buf_offset(_dcd.setup_packet);;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void prepare_ep_xfer(uint8_t rhport, uint8_t ep_id, uint16_t buf_offset, uint16_t total_bytes)
|
static void prepare_ep_xfer(uint8_t rhport, uint8_t ep_id, uint16_t buf_offset, uint16_t total_bytes)
|
||||||
@@ -354,13 +385,12 @@ static void prepare_ep_xfer(uint8_t rhport, uint8_t ep_id, uint16_t buf_offset,
|
|||||||
|
|
||||||
if (_dcd_controller[rhport].max_speed == TUSB_SPEED_FULL )
|
if (_dcd_controller[rhport].max_speed == TUSB_SPEED_FULL )
|
||||||
{
|
{
|
||||||
// TODO ISO FullSpeed can have up to 1023 bytes
|
nbytes = tu_min16(total_bytes, _dcd.ep[ep_id][0].is_iso ? NBYTES_ISO_FS_MAX : NBYTES_CBI_FS_MAX);
|
||||||
nbytes = tu_min16(total_bytes, NBYTES_CBI_FULLSPEED_MAX);
|
|
||||||
_dcd.ep[ep_id][0].buffer_fs.offset = buf_offset;
|
_dcd.ep[ep_id][0].buffer_fs.offset = buf_offset;
|
||||||
_dcd.ep[ep_id][0].buffer_fs.nbytes = nbytes;
|
_dcd.ep[ep_id][0].buffer_fs.nbytes = nbytes;
|
||||||
}else
|
}else
|
||||||
{
|
{
|
||||||
nbytes = tu_min16(total_bytes, NBYTES_CBI_HIGHSPEED_MAX);
|
nbytes = tu_min16(total_bytes, NBYTES_CBI_HS_MAX);
|
||||||
_dcd.ep[ep_id][0].buffer_hs.offset = buf_offset;
|
_dcd.ep[ep_id][0].buffer_hs.offset = buf_offset;
|
||||||
_dcd.ep[ep_id][0].buffer_hs.nbytes = nbytes;
|
_dcd.ep[ep_id][0].buffer_hs.nbytes = nbytes;
|
||||||
}
|
}
|
||||||
@@ -372,13 +402,19 @@ static void prepare_ep_xfer(uint8_t rhport, uint8_t ep_id, uint16_t buf_offset,
|
|||||||
|
|
||||||
bool dcd_edpt_xfer(uint8_t rhport, uint8_t ep_addr, uint8_t* buffer, uint16_t total_bytes)
|
bool dcd_edpt_xfer(uint8_t rhport, uint8_t ep_addr, uint8_t* buffer, uint16_t total_bytes)
|
||||||
{
|
{
|
||||||
(void) rhport;
|
|
||||||
|
|
||||||
uint8_t const ep_id = ep_addr2id(ep_addr);
|
uint8_t const ep_id = ep_addr2id(ep_addr);
|
||||||
|
|
||||||
tu_memclr(&_dcd.dma[ep_id], sizeof(xfer_dma_t));
|
tu_memclr(&_dcd.dma[ep_id], sizeof(xfer_dma_t));
|
||||||
_dcd.dma[ep_id].total_bytes = total_bytes;
|
_dcd.dma[ep_id].total_bytes = total_bytes;
|
||||||
|
|
||||||
|
if (!buffer)
|
||||||
|
{
|
||||||
|
// Although having no data, ZLPs can cause buffer overwritten to zeroes.
|
||||||
|
// Probably due to USB/DMA controller side effect/bug.
|
||||||
|
// Assigned buffer offset to (valid) dummy to prevent overwriting to DATABUFSTART
|
||||||
|
buffer = (uint8_t*)(uint32_t)dummy;
|
||||||
|
}
|
||||||
|
|
||||||
prepare_ep_xfer(rhport, ep_id, get_buf_offset(buffer), total_bytes);
|
prepare_ep_xfer(rhport, ep_id, get_buf_offset(buffer), total_bytes);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
@@ -390,15 +426,14 @@ bool dcd_edpt_xfer(uint8_t rhport, uint8_t ep_addr, uint8_t* buffer, uint16_t to
|
|||||||
static void bus_reset(uint8_t rhport)
|
static void bus_reset(uint8_t rhport)
|
||||||
{
|
{
|
||||||
tu_memclr(&_dcd, sizeof(dcd_data_t));
|
tu_memclr(&_dcd, sizeof(dcd_data_t));
|
||||||
|
edpt_reset_all(rhport);
|
||||||
|
|
||||||
// disable all non-control endpoints on bus reset
|
// disable all endpoints as specified by LPC55S69 UM Table 778
|
||||||
for(uint8_t ep_id = 2; ep_id < 2*MAX_EP_PAIRS; ep_id++)
|
for(uint8_t ep_id = 0; ep_id < 2*MAX_EP_PAIRS; ep_id++)
|
||||||
{
|
{
|
||||||
_dcd.ep[ep_id][0].disable = _dcd.ep[ep_id][1].disable = 1;
|
_dcd.ep[ep_id][0].disable = _dcd.ep[ep_id][1].disable = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
prepare_setup_packet(rhport);
|
|
||||||
|
|
||||||
dcd_registers_t* dcd_reg = _dcd_controller[rhport].regs;
|
dcd_registers_t* dcd_reg = _dcd_controller[rhport].regs;
|
||||||
|
|
||||||
dcd_reg->EPINUSE = 0;
|
dcd_reg->EPINUSE = 0;
|
||||||
@@ -506,24 +541,16 @@ void dcd_int_handler(uint8_t rhport)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO support suspend & resume
|
|
||||||
if (cmd_stat & CMDSTAT_SUSPEND_CHANGE_MASK)
|
if (cmd_stat & CMDSTAT_SUSPEND_CHANGE_MASK)
|
||||||
{
|
{
|
||||||
if (cmd_stat & CMDSTAT_DEVICE_SUSPEND_MASK)
|
// suspend signal, bus idle for more than 3ms
|
||||||
{ // suspend signal, bus idle for more than 3ms
|
|
||||||
// Note: Host may delay more than 3 ms before and/or after bus reset before doing enumeration.
|
// Note: Host may delay more than 3 ms before and/or after bus reset before doing enumeration.
|
||||||
if (cmd_stat & CMDSTAT_DEVICE_ADDR_MASK)
|
if (cmd_stat & CMDSTAT_DEVICE_ADDR_MASK)
|
||||||
{
|
{
|
||||||
dcd_event_bus_signal(rhport, DCD_EVENT_SUSPEND, true);
|
dcd_event_bus_signal(rhport, (cmd_stat & CMDSTAT_DEVICE_SUSPEND_MASK) ? DCD_EVENT_SUSPEND : DCD_EVENT_RESUME, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// else
|
|
||||||
// { // resume signal
|
|
||||||
// dcd_event_bus_signal(rhport, DCD_EVENT_RESUME, true);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
}
|
|
||||||
|
|
||||||
// Setup Receive
|
// Setup Receive
|
||||||
if ( tu_bit_test(int_status, 0) && (cmd_stat & CMDSTAT_SETUP_RECEIVED_MASK) )
|
if ( tu_bit_test(int_status, 0) && (cmd_stat & CMDSTAT_SETUP_RECEIVED_MASK) )
|
||||||
|
Reference in New Issue
Block a user