able to get 8 byte descriptor, but read(RCVBC) always return 0

- rename max3421e to max3421
- fix incorrect bitmask for HCTL, fix initial device connect
- fix bus reset cause connect IRQ
This commit is contained in:
hathach 2023-08-23 15:08:12 +07:00
parent 274578ff46
commit e3f3179924
No known key found for this signature in database
GPG Key ID: F5D50C6D51D17CBA
4 changed files with 208 additions and 104 deletions

View File

@ -93,7 +93,7 @@ TU_ATTR_UNUSED static void power_event_handler(nrfx_power_usb_evt_t event) {
} }
//------------- Host using MAX2341E -------------// //------------- Host using MAX2341E -------------//
#if CFG_TUH_ENABLED && defined(CFG_TUH_MAX3421E) && CFG_TUH_MAX3421E #if CFG_TUH_ENABLED && defined(CFG_TUH_MAX3421) && CFG_TUH_MAX3421
static nrfx_spim_t _spi = NRFX_SPIM_INSTANCE(0); static nrfx_spim_t _spi = NRFX_SPIM_INSTANCE(0);
void max3421e_int_handler(nrfx_gpiote_pin_t pin, nrf_gpiote_polarity_t action) { void max3421e_int_handler(nrfx_gpiote_pin_t pin, nrf_gpiote_polarity_t action) {
@ -102,19 +102,17 @@ void max3421e_int_handler(nrfx_gpiote_pin_t pin, nrf_gpiote_polarity_t action) {
tuh_int_handler(1); tuh_int_handler(1);
} }
static inline void max3421e_cs_assert(bool active) {
nrf_gpio_pin_write(MAX3421E_CS_PIN, active ? 0 : 1);
}
//--------------------------------------------------------------------+ //--------------------------------------------------------------------+
// API: SPI transfer with MAX3421E, must be implemented by application // API: SPI transfer with MAX3421E, must be implemented by application
//--------------------------------------------------------------------+ //--------------------------------------------------------------------+
bool tuh_max3421e_spi_xfer_api(uint8_t rhport, uint8_t const * tx_buf, size_t tx_len, void tuh_max3421_spi_cs_api(uint8_t rhport, bool active) {
uint8_t * rx_buf, size_t rx_len, bool keep_cs) {
(void) rhport; (void) rhport;
nrf_gpio_pin_write(MAX3421E_CS_PIN, active ? 0 : 1);
}
max3421e_cs_assert(true); bool tuh_max3421_spi_xfer_api(uint8_t rhport, uint8_t const * tx_buf, size_t tx_len, uint8_t * rx_buf, size_t rx_len) {
(void) rhport;
nrfx_spim_xfer_desc_t xfer = { nrfx_spim_xfer_desc_t xfer = {
.p_tx_buffer = tx_buf, .p_tx_buffer = tx_buf,
@ -123,11 +121,7 @@ bool tuh_max3421e_spi_xfer_api(uint8_t rhport, uint8_t const * tx_buf, size_t tx
.rx_length = rx_len, .rx_length = rx_len,
}; };
bool ret = (nrfx_spim_xfer(&_spi, &xfer, 0) == NRFX_SUCCESS); return (nrfx_spim_xfer(&_spi, &xfer, 0) == NRFX_SUCCESS);
if ( !keep_cs ) max3421e_cs_assert(false);
return ret;
} }
#endif #endif
@ -218,10 +212,10 @@ void board_init(void) {
if ( usb_reg & OUTPUTRDY_Msk ) tusb_hal_nrf_power_event(USB_EVT_READY); if ( usb_reg & OUTPUTRDY_Msk ) tusb_hal_nrf_power_event(USB_EVT_READY);
#endif #endif
#if CFG_TUH_ENABLED && defined(CFG_TUH_MAX3421E) && CFG_TUH_MAX3421E #if CFG_TUH_ENABLED && defined(CFG_TUH_MAX3421) && CFG_TUH_MAX3421
// manually manage CS // manually manage CS
nrf_gpio_cfg_output(MAX3421E_CS_PIN); nrf_gpio_cfg_output(MAX3421E_CS_PIN);
max3421e_cs_assert(false); tuh_max3421_spi_cs_api(0, false);
// USB host using max3421e usb controller via SPI // USB host using max3421e usb controller via SPI
nrfx_spim_config_t cfg = { nrfx_spim_config_t cfg = {

View File

@ -124,7 +124,7 @@ function(family_configure_example TARGET RTOS)
family_add_tinyusb(${TARGET} OPT_MCU_NRF5X ${RTOS}) family_add_tinyusb(${TARGET} OPT_MCU_NRF5X ${RTOS})
target_sources(${TARGET}-tinyusb PUBLIC target_sources(${TARGET}-tinyusb PUBLIC
${TOP}/src/portable/nordic/nrf5x/dcd_nrf5x.c ${TOP}/src/portable/nordic/nrf5x/dcd_nrf5x.c
${TOP}/src/portable/analog/max3421e/hcd_max3421e.c ${TOP}/src/portable/analog/max3421/hcd_max3421.c
) )
target_link_libraries(${TARGET}-tinyusb PUBLIC board_${BOARD}) target_link_libraries(${TARGET}-tinyusb PUBLIC board_${BOARD})

View File

@ -170,7 +170,7 @@ void hcd_device_close(uint8_t rhport, uint8_t dev_addr);
bool hcd_edpt_open(uint8_t rhport, uint8_t dev_addr, tusb_desc_endpoint_t const * ep_desc); bool hcd_edpt_open(uint8_t rhport, uint8_t dev_addr, tusb_desc_endpoint_t const * ep_desc);
// Submit a transfer, when complete hcd_event_xfer_complete() must be invoked // Submit a transfer, when complete hcd_event_xfer_complete() must be invoked
bool hcd_edpt_xfer(uint8_t rhport, uint8_t dev_addr, uint8_t ep_addr, uint8_t * buffer, uint16_t buflen); bool hcd_edpt_xfer(uint8_t rhport, uint8_t daddr, uint8_t ep_addr, uint8_t * buffer, uint16_t buflen);
// Abort a queued transfer. Note: it can only abort transfer that has not been started // Abort a queued transfer. Note: it can only abort transfer that has not been started
// Return true if a queued transfer is aborted, false if there is no transfer to abort // Return true if a queued transfer is aborted, false if there is no transfer to abort

View File

@ -26,7 +26,7 @@
#include "tusb_option.h" #include "tusb_option.h"
#if CFG_TUH_ENABLED && defined(CFG_TUH_MAX3421E) && CFG_TUH_MAX3421E #if CFG_TUH_ENABLED && defined(CFG_TUH_MAX3421) && CFG_TUH_MAX3421
#include "host/hcd.h" #include "host/hcd.h"
@ -110,14 +110,14 @@ enum {
}; };
enum { enum {
HCTL_BUSRST = 1u << 1, HCTL_BUSRST = 1u << 0,
HCTL_FRMRST = 1u << 2, HCTL_FRMRST = 1u << 1,
HCTL_SAMPLEBUS = 1u << 3, HCTL_SAMPLEBUS = 1u << 2,
HCTL_SIGRSM = 1u << 4, HCTL_SIGRSM = 1u << 3,
HCTL_RCVTOG0 = 1u << 5, HCTL_RCVTOG0 = 1u << 4,
HCTL_RCVTOG1 = 1u << 6, HCTL_RCVTOG1 = 1u << 5,
HCTL_SNDTOG0 = 1u << 7, HCTL_SNDTOG0 = 1u << 6,
HCTL_SNDTOG1 = 1u << 8, HCTL_SNDTOG1 = 1u << 7,
}; };
enum { enum {
@ -155,45 +155,73 @@ enum {
HRSL_BABBLE, HRSL_BABBLE,
}; };
enum {
DEFAULT_HIEN = HIRQ_CONDET_IRQ | HIRQ_FRAME_IRQ | HIRQ_HXFRDN_IRQ
};
//--------------------------------------------------------------------+ //--------------------------------------------------------------------+
// //
//--------------------------------------------------------------------+ //--------------------------------------------------------------------+
typedef struct { typedef struct {
uint8_t xfer_type;
uint8_t data_toggle;
uint16_t packet_size;
uint16_t total_len;
uint16_t xferred_len;
uint8_t* buf;
} hcd_ep_t;
typedef struct {
bool inited;
// cached register // cached register
uint8_t sndbc;
uint8_t mode; uint8_t mode;
uint8_t peraddr; uint8_t peraddr;
uint8_t hxfr; uint8_t hxfr;
volatile uint16_t frame_count; volatile uint16_t frame_count;
struct { hcd_ep_t ep[8][2];
uint16_t packet_size; } max2341_data_t;
uint16_t total_len;
uint8_t xfer_type;
}ep[8][2];
} max2341e_data_t;
static max2341e_data_t _hcd_data; static max2341_data_t _hcd_data;
//--------------------------------------------------------------------+ //--------------------------------------------------------------------+
// API: SPI transfer with MAX3421E, must be implemented by application // API: SPI transfer with MAX3421E, must be implemented by application
//--------------------------------------------------------------------+ //--------------------------------------------------------------------+
bool tuh_max3421e_spi_xfer_api(uint8_t rhport, uint8_t const * tx_buf, size_t tx_len, void tuh_max3421_spi_cs_api(uint8_t rhport, bool active);
uint8_t * rx_buf, size_t rx_len, bool keep_cs); bool tuh_max3421_spi_xfer_api(uint8_t rhport, uint8_t const * tx_buf, size_t tx_len, uint8_t * rx_buf, size_t rx_len);
//void tuh_max3421e_int_enable(uint8_t rhport, bool enabled); //void tuh_max3421e_int_enable(uint8_t rhport, bool enabled);
//--------------------------------------------------------------------+ //--------------------------------------------------------------------+
// //
//--------------------------------------------------------------------+ //--------------------------------------------------------------------+
static void fifo_write(uint8_t reg, uint8_t const * buffer, uint16_t len) {
reg |= CMDBYTE_WRITE;
tuh_max3421_spi_cs_api(0, true);
tuh_max3421_spi_xfer_api(0, &reg, 1, NULL, 0);
tuh_max3421_spi_xfer_api(0, buffer, len, NULL, 0);
tuh_max3421_spi_cs_api(0, false);
}
// return HIRQ register since we are in full-duplex mode // return HIRQ register since we are in full-duplex mode
static uint8_t reg_write(uint8_t reg, uint8_t data) { static uint8_t reg_write(uint8_t reg, uint8_t data) {
uint8_t tx_buf[2] = {reg | CMDBYTE_WRITE, data}; uint8_t tx_buf[2] = {reg | CMDBYTE_WRITE, data};
uint8_t rx_buf[2] = {0, 0}; uint8_t rx_buf[2] = {0, 0};
tuh_max3421e_spi_xfer_api(0, tx_buf, 2, rx_buf, 2, false);
tuh_max3421_spi_cs_api(0, true);
tuh_max3421_spi_xfer_api(0, tx_buf, 2, rx_buf, 2);
tuh_max3421_spi_cs_api(0, false);
TU_LOG2("HIRQ: %02X\r\n", rx_buf[0]); TU_LOG2("HIRQ: %02X\r\n", rx_buf[0]);
return rx_buf[0]; return rx_buf[0];
} }
@ -201,7 +229,12 @@ static uint8_t reg_write(uint8_t reg, uint8_t data) {
static uint8_t reg_read(uint8_t reg) { static uint8_t reg_read(uint8_t reg) {
uint8_t tx_buf[2] = {reg, 0}; uint8_t tx_buf[2] = {reg, 0};
uint8_t rx_buf[2] = {0, 0}; uint8_t rx_buf[2] = {0, 0};
return tuh_max3421e_spi_xfer_api(0, tx_buf, 2, rx_buf, 2, false) ? rx_buf[1] : 0;
tuh_max3421_spi_cs_api(0, true);
bool ret = tuh_max3421_spi_xfer_api(0, tx_buf, 2, rx_buf, 2);
tuh_max3421_spi_cs_api(0, false);
return ret ? rx_buf[1] : 0;
} }
static inline uint8_t mode_write(uint8_t data) { static inline uint8_t mode_write(uint8_t data) {
@ -221,10 +254,9 @@ static inline uint8_t hxfr_write(uint8_t data) {
return reg_write(HXFR_ADDR, data); return reg_write(HXFR_ADDR, data);
} }
static void fifo_write(uint8_t reg, uint8_t const * buffer, uint16_t len) { static inline uint8_t sndbc_write(uint8_t data) {
uint8_t tx_buf[1] = {reg | CMDBYTE_WRITE}; _hcd_data.sndbc = data;
tuh_max3421e_spi_xfer_api(0, tx_buf, 1, NULL, 0, true); return reg_write(SNDBC_ADDR, data);
tuh_max3421e_spi_xfer_api(0, buffer, len, NULL, 0, false);
} }
@ -290,8 +322,8 @@ bool hcd_init(uint8_t rhport) {
tu_memclr(&_hcd_data, sizeof(_hcd_data)); tu_memclr(&_hcd_data, sizeof(_hcd_data));
// full duplex, interrupt level (should be configurable) // full duplex, interrupt negative edge
reg_write(PINCTL_ADDR, PINCTL_FDUPSPI | PINCTL_INTLEVEL); reg_write(PINCTL_ADDR, PINCTL_FDUPSPI);
// reset // reset
reg_write(USBCTL_ADDR, USBCTL_CHIPRES); reg_write(USBCTL_ADDR, USBCTL_CHIPRES);
@ -303,19 +335,16 @@ bool hcd_init(uint8_t rhport) {
// Mode: Host and DP/DM pull down // Mode: Host and DP/DM pull down
mode_write(MODE_DPPULLDN | MODE_DMPULLDN | MODE_HOST); mode_write(MODE_DPPULLDN | MODE_DMPULLDN | MODE_HOST);
// bus reset, this will trigger CONDET IRQ if device is already connected
reg_write(HCTL_ADDR, HCTL_BUSRST);
// clear all previously pending IRQ
reg_write(HIRQ_ADDR, 0xff);
_hcd_data.inited = true;
// Enable Connection IRQ // Enable Connection IRQ
reg_write(HIEN_ADDR, HIRQ_CONDET_IRQ | HIRQ_FRAME_IRQ | HIRQ_HXFRDN_IRQ); reg_write(HIEN_ADDR, DEFAULT_HIEN);
#if 0
// Note: if device is already connected, CONDET IRQ may not be triggered.
// We need to detect it by sampling bus signal. FIXME not working
reg_write(HCTL_ADDR, HCTL_SAMPLEBUS);
while ( reg_read(HCTL_ADDR) & HCTL_SAMPLEBUS ) {}
if ( TUSB_SPEED_INVALID != handle_connect_irq(rhport) ) {
reg_write(HIRQ_ADDR, HIRQ_CONDET_IRQ); // clear connect irq
}
#endif
// Enable Interrupt pin // Enable Interrupt pin
reg_write(CPUCTL_ADDR, CPUCTL_IE); reg_write(CPUCTL_ADDR, CPUCTL_IE);
@ -353,6 +382,10 @@ bool hcd_port_connect_status(uint8_t rhport) {
// Some port would require hcd_port_reset_end() to be invoked after 10ms to complete the reset sequence. // Some port would require hcd_port_reset_end() to be invoked after 10ms to complete the reset sequence.
void hcd_port_reset(uint8_t rhport) { void hcd_port_reset(uint8_t rhport) {
(void) rhport; (void) rhport;
// Bus reset will also trigger CONDET IRQ, disable it
uint8_t hien = DEFAULT_HIEN & ~HIRQ_CONDET_IRQ;
reg_write(HIEN_ADDR, hien);
reg_write(HCTL_ADDR, HCTL_BUSRST); reg_write(HCTL_ADDR, HCTL_BUSRST);
} }
@ -360,6 +393,10 @@ void hcd_port_reset(uint8_t rhport) {
void hcd_port_reset_end(uint8_t rhport) { void hcd_port_reset_end(uint8_t rhport) {
(void) rhport; (void) rhport;
reg_write(HCTL_ADDR, 0); reg_write(HCTL_ADDR, 0);
// Bus reset will also trigger CONDET IRQ, clear and re-enable it after reset
reg_write(HIRQ_ADDR, HIRQ_CONDET_IRQ);
reg_write(HIEN_ADDR, DEFAULT_HIEN);
} }
// Get port link speed // Get port link speed
@ -393,14 +430,51 @@ bool hcd_edpt_open(uint8_t rhport, uint8_t dev_addr, tusb_desc_endpoint_t const
} }
// Submit a transfer, when complete hcd_event_xfer_complete() must be invoked // Submit a transfer, when complete hcd_event_xfer_complete() must be invoked
bool hcd_edpt_xfer(uint8_t rhport, uint8_t dev_addr, uint8_t ep_addr, uint8_t * buffer, uint16_t buflen) { bool hcd_edpt_xfer(uint8_t rhport, uint8_t daddr, uint8_t ep_addr, uint8_t * buffer, uint16_t buflen) {
(void) rhport; (void) rhport;
(void) dev_addr;
(void) ep_addr;
(void) buffer;
(void) buflen;
return false; uint8_t const ep_num = tu_edpt_number(ep_addr);
uint8_t const ep_dir = tu_edpt_dir(ep_addr);
hcd_ep_t* ep = &_hcd_data.ep[ep_num][ep_dir];
ep->buf = buffer;
ep->total_len = buflen;
ep->xferred_len = 0;
uint8_t hirq = peraddr_write(daddr);
uint8_t hctl = 0;
uint8_t hxfr = ep_num;
if ( ep_num == 0 ) {
ep->data_toggle = 1;
if ( buffer == NULL || buflen == 0 ) {
// ZLP for ACK stage
hxfr |= HXFR_HS;
}
} else if ( ep->xfer_type == TUSB_XFER_ISOCHRONOUS ) {
hxfr |= HXFR_ISO;
}
if ( 0 == ep_dir ) {
// Page 12: Programming BULK-OUT Transfers
TU_ASSERT(hirq & HIRQ_RCVDAV_IRQ);
uint8_t const xact_len = (uint8_t) tu_min16(buflen, ep->packet_size);
fifo_write(SNDFIFO_ADDR, buffer, xact_len);
reg_write(SNDBC_ADDR, xact_len);
hctl = (ep->data_toggle ? HCTL_SNDTOG1 : HCTL_SNDTOG0);
hxfr |= HXFR_OUT_NIN;
} else {
// Page 13: Programming BULK-IN Transfers
hctl = (ep->data_toggle ? HCTL_RCVTOG1 : HCTL_RCVTOG0);
}
reg_write(HCTL_ADDR, hctl);
hxfr_write(hxfr);
return true;
} }
// Abort a queued transfer. Note: it can only abort transfer that has not been started // Abort a queued transfer. Note: it can only abort transfer that has not been started
@ -416,10 +490,10 @@ bool hcd_edpt_abort_xfer(uint8_t rhport, uint8_t dev_addr, uint8_t ep_addr) {
// Submit a special transfer to send 8-byte Setup Packet, when complete hcd_event_xfer_complete() must be invoked // Submit a special transfer to send 8-byte Setup Packet, when complete hcd_event_xfer_complete() must be invoked
bool hcd_setup_send(uint8_t rhport, uint8_t daddr, uint8_t const setup_packet[8]) { bool hcd_setup_send(uint8_t rhport, uint8_t daddr, uint8_t const setup_packet[8]) {
(void) rhport; (void) rhport;
(void) daddr;
(void) setup_packet;
_hcd_data.ep[0][0].total_len = 8; hcd_ep_t* ep = &_hcd_data.ep[0][0];
ep->total_len = 8;
ep->xferred_len = 0;
peraddr_write(daddr); peraddr_write(daddr);
fifo_write(SUDFIFO_ADDR, setup_packet, 8); fifo_write(SUDFIFO_ADDR, setup_packet, 8);
@ -437,11 +511,79 @@ bool hcd_edpt_clear_stall(uint8_t rhport, uint8_t dev_addr, uint8_t ep_addr) {
return false; return false;
} }
static void handle_xfer_done(uint8_t rhport) {
(void) rhport;
uint8_t const hrsl = reg_read(HRSL_ADDR);
uint8_t const result = hrsl & HRSL_RESULT_MASK;
uint8_t xfer_result;
TU_LOG3("HRSL: %02X\r\n", hrsl);
switch(result) {
case HRSL_SUCCESS:
xfer_result = XFER_RESULT_SUCCESS;
break;
case HRSL_STALL:
xfer_result = XFER_RESULT_STALLED;
break;
case HRSL_BAD_REQ:
// occurred when initialized without any pending transfer. Skip for now
return;
default:
xfer_result = XFER_RESULT_FAILED;
break;
}
uint8_t ep_dir = 0;
uint8_t ep_num = _hcd_data.hxfr & HXFR_EPNUM_MASK;
uint8_t const xfer_type = _hcd_data.hxfr & 0xf0;
hcd_ep_t * ep;
if ( (xfer_type & HXFR_SETUP) || (xfer_type & HXFR_OUT_NIN) ) {
// SETUP or OUT transfer
ep_dir = 0;
ep = &_hcd_data.ep[ep_num][ep_dir];
uint8_t const xact_len = (xfer_type & HXFR_SETUP) ? 8 : _hcd_data.sndbc;
ep->xferred_len += xact_len;
if ( xact_len < ep->packet_size || ep->xferred_len >= ep->total_len ) {
hcd_event_xfer_complete(_hcd_data.peraddr, ep_num, ep->xferred_len, xfer_result, true);
}else {
// more to transfer
}
} else {
// IN transfer
ep_dir = 1;
ep = &_hcd_data.ep[ep_num][ep_dir];
uint8_t const xact_len = reg_read(RCVBC_ADDR);
ep->xferred_len += xact_len;
// short packet or all bytes transferred
if ( xact_len < ep->packet_size || ep->xferred_len >= ep->total_len ) {
hcd_event_xfer_complete(_hcd_data.peraddr, TUSB_DIR_IN_MASK | ep_num, ep->xferred_len, xfer_result, true);
}else {
// more to transfer
}
}
}
// Interrupt Handler // Interrupt Handler
void hcd_int_handler(uint8_t rhport) { void hcd_int_handler(uint8_t rhport) {
// not initialized, do nothing
if ( !_hcd_data.inited ) return;
uint8_t hirq = reg_read(HIRQ_ADDR); uint8_t hirq = reg_read(HIRQ_ADDR);
TU_LOG3_HEX(hirq); TU_LOG3_HEX(hirq);
if (hirq & HIRQ_FRAME_IRQ) {
_hcd_data.frame_count++;
}
if (hirq & HIRQ_CONDET_IRQ) { if (hirq & HIRQ_CONDET_IRQ) {
tusb_speed_t speed = handle_connect_irq(rhport); tusb_speed_t speed = handle_connect_irq(rhport);
@ -452,49 +594,17 @@ void hcd_int_handler(uint8_t rhport) {
} }
} }
if (hirq & HIRQ_FRAME_IRQ) {
_hcd_data.frame_count++;
}
if (hirq & HIRQ_HXFRDN_IRQ) { if (hirq & HIRQ_HXFRDN_IRQ) {
uint8_t const hrsl = reg_read(HRSL_ADDR); handle_xfer_done(rhport);
uint8_t const result = hrsl & HRSL_RESULT_MASK;
uint8_t xfer_result;
TU_LOG3("HRSL: %02X\r\n", hrsl);
switch(result) {
case HRSL_SUCCESS:
xfer_result = XFER_RESULT_SUCCESS;
break;
case HRSL_STALL:
xfer_result = XFER_RESULT_STALLED;
break;
default:
xfer_result = XFER_RESULT_FAILED;
break;
}
uint8_t ep_dir = 0;
uint8_t ep_num = _hcd_data.hxfr & HXFR_EPNUM_MASK;
uint8_t const xfer_type = _hcd_data.hxfr & 0xf0;
if ( xfer_type & HXFR_SETUP ) {
// SETUP transfer
ep_dir = 0;
}else if ( !(xfer_type & HXFR_OUT_NIN) ) {
// IN transfer
ep_dir = 1;
}
uint8_t const ep_addr = tu_edpt_addr(ep_num, ep_dir);
uint16_t xferred_len = _hcd_data.ep[ep_num][ep_dir].total_len;
hcd_event_xfer_complete(_hcd_data.peraddr, ep_addr, xferred_len, xfer_result, true);
} }
// clear all interrupt if ( hirq & HIRQ_RCVDAV_IRQ ) {
TU_LOG3("RCVDAV\r\n");
TU_LOG3_INT(reg_read(RCVBC_ADDR));
}
// clear all interrupt execept SNDBAV_IRQ
hirq &= ~HIRQ_SNDBAV_IRQ;
if ( hirq ) { if ( hirq ) {
reg_write(HIRQ_ADDR, hirq); reg_write(HIRQ_ADDR, hirq);
} }