mirror of
https://github.com/hathach/tinyusb.git
synced 2025-03-28 05:37:15 +00:00
improve nrf52 dcd, add fix for hw ACK issue with bulk
This commit is contained in:
parent
d1ecef4fc3
commit
2b85a8fd46
@ -66,13 +66,7 @@ typedef struct {
|
||||
}cdcd_interface_t;
|
||||
|
||||
// TODO multiple rhport
|
||||
#if CFG_TUSB_MCU == OPT_MCU_NRF5X
|
||||
// FIXME nrf52 OUT bug ( Controller ACK data even we didn't prepare transfer )
|
||||
CFG_TUSB_ATTR_USBRAM CFG_TUSB_MEM_ALIGN uint8_t _tmp_rx_buf[600];
|
||||
#else
|
||||
CFG_TUSB_ATTR_USBRAM CFG_TUSB_MEM_ALIGN uint8_t _tmp_rx_buf[64];
|
||||
#endif
|
||||
|
||||
CFG_TUSB_ATTR_USBRAM CFG_TUSB_MEM_ALIGN uint8_t _tmp_tx_buf[64];
|
||||
|
||||
FIFO_DEF(_rx_ff, CFG_TUD_CDC_BUFSIZE, uint8_t, true);
|
||||
|
@ -70,6 +70,9 @@ typedef struct
|
||||
uint16_t actual_len;
|
||||
uint8_t mps; // max packet size
|
||||
|
||||
// FIXME nrf52840 does not NAK OUT packet properly
|
||||
bool data_received;
|
||||
|
||||
} nom_xfer_t;
|
||||
|
||||
/*static*/ struct
|
||||
@ -163,7 +166,7 @@ static void edpt_dma_end(void)
|
||||
_dcd.dma_running = false;
|
||||
}
|
||||
|
||||
static void control_xact_start(void)
|
||||
static void xact_control_start(void)
|
||||
{
|
||||
// Each transaction is up to 64 bytes
|
||||
uint8_t const xact_len = min16_of(_dcd.control.total_len-_dcd.control.actual_len, MAX_PACKET_SIZE);
|
||||
@ -200,7 +203,7 @@ bool dcd_control_xfer (uint8_t rhport, tusb_dir_t dir, uint8_t * buffer, uint16_
|
||||
_dcd.control.buffer = buffer;
|
||||
_dcd.control.dir = (uint8_t) dir;
|
||||
|
||||
control_xact_start();
|
||||
xact_control_start();
|
||||
}else
|
||||
{
|
||||
// Status Phase
|
||||
@ -220,27 +223,55 @@ static inline nom_xfer_t* get_td(uint8_t epnum, uint8_t dir)
|
||||
return &_dcd.xfer[epnum-1][dir];
|
||||
}
|
||||
|
||||
static void normal_xact_start(uint8_t epnum, uint8_t dir)
|
||||
/*------------- Bulk/Int OUT transfer -------------*/
|
||||
|
||||
/**
|
||||
* Prepare Bulk/Int out transaction, Endpoint start to accept/ACK Data
|
||||
* @param epnum
|
||||
*/
|
||||
static void xact_out_prepare(uint8_t epnum)
|
||||
{
|
||||
if ( dir == TUSB_DIR_OUT )
|
||||
{
|
||||
// Overwrite size will allow hw to accept data
|
||||
NRF_USBD->SIZE.EPOUT[epnum] = 0;
|
||||
__ISB(); __DSB();
|
||||
}else
|
||||
{
|
||||
nom_xfer_t* xfer = get_td(epnum, dir);
|
||||
// Write any value to size will allow hw to ACK (accept data)
|
||||
NRF_USBD->SIZE.EPOUT[epnum] = 0;
|
||||
__ISB(); __DSB();
|
||||
}
|
||||
|
||||
// Each transaction is up to Max Packet Size
|
||||
uint8_t const xact_len = min16_of(xfer->total_len - xfer->actual_len, xfer->mps);
|
||||
static void xact_out_dma(uint8_t epnum)
|
||||
{
|
||||
nom_xfer_t* xfer = get_td(epnum, TUSB_DIR_OUT);
|
||||
|
||||
NRF_USBD->EPIN[epnum].PTR = (uint32_t) xfer->buffer;
|
||||
NRF_USBD->EPIN[epnum].MAXCNT = xact_len;
|
||||
uint8_t const xact_len = NRF_USBD->SIZE.EPOUT[epnum];
|
||||
|
||||
xfer->buffer += xact_len;
|
||||
// Trigger DMA move data from Endpoint -> SRAM
|
||||
NRF_USBD->EPOUT[epnum].PTR = (uint32_t) xfer->buffer;
|
||||
NRF_USBD->EPOUT[epnum].MAXCNT = xact_len;
|
||||
|
||||
edpt_dma_start(epnum, TUSB_DIR_IN);
|
||||
}
|
||||
edpt_dma_start(epnum, TUSB_DIR_OUT);
|
||||
|
||||
xfer->buffer += xact_len;
|
||||
xfer->actual_len += xact_len;
|
||||
}
|
||||
|
||||
|
||||
/*------------- Bulk/Int IN transfer -------------*/
|
||||
|
||||
/**
|
||||
* Prepare Bulk/Int in transaction, transfer data from Memory -> Endpoint
|
||||
* @param epnum
|
||||
*/
|
||||
static void xact_in_prepare(uint8_t epnum)
|
||||
{
|
||||
nom_xfer_t* xfer = get_td(epnum, TUSB_DIR_IN);
|
||||
|
||||
// Each transaction is up to Max Packet Size
|
||||
uint8_t const xact_len = min16_of(xfer->total_len - xfer->actual_len, xfer->mps);
|
||||
|
||||
NRF_USBD->EPIN[epnum].PTR = (uint32_t) xfer->buffer;
|
||||
NRF_USBD->EPIN[epnum].MAXCNT = xact_len;
|
||||
|
||||
xfer->buffer += xact_len;
|
||||
|
||||
edpt_dma_start(epnum, TUSB_DIR_IN);
|
||||
}
|
||||
|
||||
bool dcd_edpt_open (uint8_t rhport, tusb_desc_endpoint_t const * desc_edpt)
|
||||
@ -279,7 +310,23 @@ bool dcd_edpt_xfer (uint8_t rhport, uint8_t ep_addr, uint8_t * buffer, uint16_t
|
||||
xfer->total_len = total_bytes;
|
||||
xfer->actual_len = 0;
|
||||
|
||||
normal_xact_start(epnum, dir);
|
||||
if ( dir == TUSB_DIR_OUT )
|
||||
{
|
||||
if ( xfer->data_received )
|
||||
{
|
||||
xfer->data_received = false;
|
||||
|
||||
// FIXME nrf52840 does not NAK OUT packet properly
|
||||
// Data already received preivously
|
||||
xact_out_dma(epnum);
|
||||
}else
|
||||
{
|
||||
xact_out_prepare(epnum);
|
||||
}
|
||||
}else
|
||||
{
|
||||
xact_in_prepare(epnum);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
@ -382,7 +429,7 @@ void USBD_IRQHandler(void)
|
||||
// IN: data transferred from Endpoint -> Host
|
||||
if ( _dcd.control.actual_len < _dcd.control.total_len )
|
||||
{
|
||||
control_xact_start();
|
||||
xact_control_start();
|
||||
}else
|
||||
{
|
||||
// Control IN complete
|
||||
@ -396,7 +443,7 @@ void USBD_IRQHandler(void)
|
||||
{
|
||||
if ( _dcd.control.actual_len < _dcd.control.total_len )
|
||||
{
|
||||
control_xact_start();
|
||||
xact_control_start();
|
||||
}else
|
||||
{
|
||||
// Control OUT complete
|
||||
@ -423,7 +470,7 @@ void USBD_IRQHandler(void)
|
||||
if ( xfer->actual_len < xfer->total_len )
|
||||
{
|
||||
// more to xfer
|
||||
normal_xact_start(epnum, TUSB_DIR_IN);
|
||||
xact_in_prepare(epnum);
|
||||
} else
|
||||
{
|
||||
// BULK/INT IN complete
|
||||
@ -439,19 +486,17 @@ void USBD_IRQHandler(void)
|
||||
{
|
||||
nom_xfer_t* xfer = get_td(epnum, TUSB_DIR_OUT);
|
||||
|
||||
uint8_t const xact_len = NRF_USBD->SIZE.EPOUT[epnum];
|
||||
if (xfer->actual_len < xfer->total_len)
|
||||
{
|
||||
xact_out_dma(epnum);
|
||||
}else
|
||||
{
|
||||
// FIXME nrf52840 does not NAK OUT packet properly
|
||||
// It will always ACK next package although we haven't write to SIZE yet
|
||||
|
||||
// FIXME nrf52840 rev A does not NAK OUT packet properly
|
||||
TU_ASSERT(xfer->actual_len < xfer->total_len, );
|
||||
|
||||
// Trigger DMA move data from Endpoint -> SRAM
|
||||
NRF_USBD->EPOUT[epnum].PTR = (uint32_t) xfer->buffer;
|
||||
NRF_USBD->EPOUT[epnum].MAXCNT = xact_len;
|
||||
|
||||
edpt_dma_start(epnum, TUSB_DIR_OUT);
|
||||
|
||||
xfer->buffer += xact_len;
|
||||
xfer->actual_len += xact_len;
|
||||
// Mark this endpoint with data received
|
||||
xfer->data_received = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -469,7 +514,7 @@ void USBD_IRQHandler(void)
|
||||
if ( (xact_len == xfer->mps) && (xfer->actual_len < xfer->total_len) )
|
||||
{
|
||||
// Prepare for more data from Host -> Endpoint
|
||||
normal_xact_start(epnum, TUSB_DIR_OUT);
|
||||
xact_out_prepare(epnum);
|
||||
}else
|
||||
{
|
||||
xfer->total_len = xfer->actual_len;
|
||||
|
Loading…
x
Reference in New Issue
Block a user