mirror of
https://gitee.com/Vancouver2017/luban-lite-t3e-pro.git
synced 2025-12-14 18:38:55 +00:00
1394 lines
52 KiB
C
1394 lines
52 KiB
C
#include "usbh_core.h"
|
|
#include "usbh_hub.h"
|
|
#include "usb_ch32_usbhs_reg.h"
|
|
|
|
#ifndef USBH_IRQHandler
|
|
#define USBH_IRQHandler USB2_IRQHandler
|
|
#endif
|
|
|
|
#ifndef CONFIG_USBHOST_PIPE_NUM
|
|
#define CONFIG_USBHOST_PIPE_NUM 8
|
|
#endif
|
|
|
|
#ifndef USBHS_MAX_PACKET_SIZE
|
|
#define USBHS_MAX_PACKET_SIZE 1024
|
|
#endif
|
|
|
|
#define HIGH_SP_HIGH_BAND_Pos 11
|
|
#define HIGH_SP_HIGH_BAND_Mask (0x03 << HIGH_SP_HIGH_BAND_Pos)
|
|
#define HIGH_SP_HIGH_BAND_1 (0x00 << HIGH_SP_HIGH_BAND_Pos)
|
|
#define HIGH_SP_HIGH_BAND_2 (0x01 << HIGH_SP_HIGH_BAND_Pos)
|
|
#define HIGH_SP_HIGH_BAND_3 (0x02 << HIGH_SP_HIGH_BAND_Pos)
|
|
|
|
typedef enum {
|
|
USB_EP0_STATE_SETUP = 0x0, /**< SETUP DATA */
|
|
USB_EP0_STATE_IN_DATA, /**< IN DATA */
|
|
USB_EP0_STATE_IN_STATUS, /**< IN status */
|
|
USB_EP0_STATE_OUT_DATA, /**< OUT DATA */
|
|
USB_EP0_STATE_OUT_STATUS, /**< OUT status */
|
|
} ep0_state_t;
|
|
|
|
struct chusb_pipe {
|
|
uint8_t dev_addr;
|
|
uint8_t ep_addr;
|
|
uint8_t ep_type;
|
|
uint8_t ep_interval;
|
|
uint8_t speed;
|
|
uint16_t ep_mps;
|
|
bool inuse;
|
|
uint32_t xfrd;
|
|
volatile bool waiter;
|
|
|
|
uint32_t xferlen;
|
|
uint8_t *buffer;
|
|
uint8_t data_pid;
|
|
uint8_t trans_num;
|
|
|
|
usb_osal_sem_t waitsem;
|
|
struct usbh_hubport *hport;
|
|
struct usbh_urb *urb;
|
|
};
|
|
|
|
struct chusb_hcd {
|
|
volatile bool port_csc;
|
|
volatile bool port_pec;
|
|
volatile bool port_pe;
|
|
volatile uint8_t current_token;
|
|
volatile uint8_t ep0_state;
|
|
volatile bool prv_get_zero;
|
|
volatile bool prv_set_zero;
|
|
volatile bool main_pipe_using;
|
|
// uint32_t current_pipe_timeout;
|
|
uint8_t dev_speed;
|
|
struct chusb_pipe *current_pipe;
|
|
struct chusb_pipe pipe_pool[CONFIG_USBHOST_PIPE_NUM][2]; /* Support Bidirectional ep */
|
|
} g_chusb_hcd;
|
|
|
|
static inline void SET_UH_RX_CTRL_BIT(uint8_t bit)
|
|
{
|
|
if ((USBHS_HOST->HOST_RX_CTRL & USBHS_UH_R_TOG_AUTO) != 0) {
|
|
/**
|
|
* USBHS_UH_R_TOG canot write
|
|
*/
|
|
if ((bit & (USBHS_UH_R_TOG_1 | USBHS_UH_R_TOG_2 | USBHS_UH_R_TOG_3)) != 0) {
|
|
/**
|
|
* bit contains USBHS_UH_R_TOG
|
|
*/
|
|
USBHS_HOST->HOST_RX_CTRL &= ~(USBHS_UH_R_TOG_AUTO);
|
|
USBHS_HOST->HOST_RX_CTRL |= (USBHS_UH_R_TOG_AUTO | bit);
|
|
} else {
|
|
USBHS_HOST->HOST_RX_CTRL |= bit;
|
|
}
|
|
} else {
|
|
USBHS_HOST->HOST_RX_CTRL |= bit;
|
|
}
|
|
}
|
|
|
|
static inline void SET_UH_TX_CTRL_BIT(uint8_t bit)
|
|
{
|
|
if ((USBHS_HOST->HOST_TX_CTRL & USBHS_UH_T_TOG_AUTO) != 0) {
|
|
/**
|
|
* USBFS_UH_T_TOG canot write
|
|
*/
|
|
if ((bit & (USBHS_UH_T_TOG_1 | USBHS_UH_T_TOG_2 | USBHS_UH_T_TOG_3)) != 0) {
|
|
/**
|
|
* bit contains USBFS_UH_T_TOG
|
|
*/
|
|
USBHS_HOST->HOST_TX_CTRL &= ~(USBHS_UH_T_TOG_AUTO);
|
|
USBHS_HOST->HOST_TX_CTRL |= (USBHS_UH_T_TOG_AUTO | bit);
|
|
} else {
|
|
USBHS_HOST->HOST_TX_CTRL |= bit;
|
|
}
|
|
} else {
|
|
USBHS_HOST->HOST_TX_CTRL |= bit;
|
|
}
|
|
}
|
|
|
|
static inline void CLEAR_UH_TX_CTRL_BIT(uint8_t bit)
|
|
{
|
|
if ((USBHS_HOST->HOST_TX_CTRL & USBHS_UH_T_TOG_AUTO) != 0) {
|
|
/**
|
|
* USBFS_UH_T_TOG canot write
|
|
*/
|
|
if ((bit & (USBHS_UH_T_TOG_1 | USBHS_UH_T_TOG_2 | USBHS_UH_T_TOG_3)) != 0) {
|
|
/**
|
|
* bit contains USBFS_UH_T_TOG
|
|
*/
|
|
USBHS_HOST->HOST_TX_CTRL &= ~(USBHS_UH_T_TOG_AUTO);
|
|
USBHS_HOST->HOST_TX_CTRL &= ~(bit);
|
|
USBHS_HOST->HOST_TX_CTRL |= USBHS_UH_T_TOG_AUTO;
|
|
} else {
|
|
USBHS_HOST->HOST_TX_CTRL &= ~(bit);
|
|
}
|
|
} else {
|
|
USBHS_HOST->HOST_TX_CTRL &= ~(bit);
|
|
}
|
|
}
|
|
|
|
static inline void CLEAR_UH_RX_CTRL_BIT(uint8_t bit)
|
|
{
|
|
if ((USBHS_HOST->HOST_RX_CTRL & USBHS_UH_R_TOG_AUTO) != 0) {
|
|
/**
|
|
* USBFS_UH_R_TOG canot write
|
|
*/
|
|
if ((bit & (USBHS_UH_R_TOG_1 | USBHS_UH_R_TOG_2 | USBHS_UH_R_TOG_3)) != 0) {
|
|
/**
|
|
* bit contains USBFS_UH_R_TOG
|
|
*/
|
|
USBHS_HOST->HOST_RX_CTRL &= ~(USBHS_UH_R_TOG_AUTO);
|
|
USBHS_HOST->HOST_RX_CTRL &= ~(bit);
|
|
USBHS_HOST->HOST_RX_CTRL |= USBHS_UH_R_TOG_AUTO;
|
|
} else {
|
|
USBHS_HOST->HOST_RX_CTRL &= ~(bit);
|
|
}
|
|
} else {
|
|
USBHS_HOST->HOST_RX_CTRL &= ~(bit);
|
|
}
|
|
}
|
|
|
|
static inline void SET_UH_RX_CTRL(uint8_t value)
|
|
{
|
|
USBHS_HOST->HOST_RX_CTRL = 0;
|
|
USBHS_HOST->HOST_RX_CTRL = value;
|
|
USB_LOG_DBG("USBHS_HOST->HOST_RX_CTRL:%02x\r\n", USBHS_HOST->HOST_RX_CTRL);
|
|
}
|
|
|
|
static inline void SET_UH_TX_CTRL(uint8_t value)
|
|
{
|
|
USBHS_HOST->HOST_TX_CTRL = 0;
|
|
USBHS_HOST->HOST_TX_CTRL = value;
|
|
USB_LOG_DBG("USBHS_HOST->HOST_TX_CTRL:%02x\r\n", USBHS_HOST->HOST_TX_CTRL);
|
|
}
|
|
|
|
static inline void INT_PRE_HANDLER(void)
|
|
{
|
|
#if defined(CH581) || defined(CH582) || defined(CH583) || defined(CH571) || defined(CH572) || defined(CH573)
|
|
#else
|
|
asm("csrrw sp,mscratch,sp");
|
|
extern void rt_interrupt_enter(void);
|
|
rt_interrupt_enter();
|
|
#endif
|
|
}
|
|
|
|
static inline void INT_POST_HANDLER(void)
|
|
{
|
|
#if defined(CH581) || defined(CH582) || defined(CH583) || defined(CH571) || defined(CH572) || defined(CH573)
|
|
#else
|
|
extern void rt_interrupt_leave(void);
|
|
rt_interrupt_leave();
|
|
asm("csrrw sp,mscratch,sp");
|
|
#endif
|
|
}
|
|
|
|
static int8_t chusb_host_pipe_transfer(struct chusb_pipe *pipe, uint8_t pid, uint8_t *data, uint32_t len)
|
|
{
|
|
/*!< Updata current transfer pid */
|
|
g_chusb_hcd.current_token = pid;
|
|
/*!< Updata curretn pipe */
|
|
g_chusb_hcd.current_pipe = pipe;
|
|
/*!< Updata curretn pipe timeout */
|
|
// g_chusb_hcd.current_pipe_timeout = pipe->urb->timeout;
|
|
/*!< Updata main pipe using flag */
|
|
// g_chusb_hcd.main_pipe_using = true;
|
|
|
|
if (data == NULL && len > 0) {
|
|
USB_LOG_ERR("Please give correct data and len parameters\r\n");
|
|
return -1;
|
|
}
|
|
|
|
if (pid == USB_PID_SETUP) {
|
|
/*!< Record the data len */
|
|
g_chusb_hcd.current_pipe->xferlen = len;
|
|
|
|
if ((uint32_t)data & 0x03) {
|
|
USB_LOG_INFO("SETUP DMA address is not align \r\n");
|
|
return -3;
|
|
}
|
|
|
|
USBHS_HOST->HOST_TX_DMA = (uint32_t)data;
|
|
|
|
/*!< Record the data buffer address */
|
|
pipe->buffer = data;
|
|
|
|
if (len > pipe->ep_mps) {
|
|
len = pipe->ep_mps;
|
|
}
|
|
|
|
USBHS_HOST->HOST_TX_LEN = len;
|
|
USBHS_HOST->HOST_EP_PID = pid << 4 | (pipe->ep_addr & 0x0f);
|
|
|
|
} else if (pid == USB_PID_OUT) {
|
|
/*!< Record the data len */
|
|
g_chusb_hcd.current_pipe->xferlen = len;
|
|
|
|
if (len == 0) {
|
|
USBHS_HOST->HOST_TX_LEN = len;
|
|
USBHS_HOST->HOST_EP_PID = pid << 4 | (pipe->ep_addr & 0x0f);
|
|
return 0;
|
|
}
|
|
|
|
if ((uint32_t)data & 0x03) {
|
|
USB_LOG_WRN("OUT DMA address is not align \r\n");
|
|
return -3;
|
|
}
|
|
|
|
USBHS_HOST->HOST_TX_DMA = (uint32_t)data;
|
|
|
|
/*!< Record the data buffer address */
|
|
pipe->buffer = data;
|
|
|
|
if (len > pipe->ep_mps) {
|
|
len = pipe->ep_mps;
|
|
}
|
|
|
|
USBHS_HOST->HOST_TX_LEN = len;
|
|
USBHS_HOST->HOST_EP_PID = pid << 4 | (pipe->ep_addr & 0x0f);
|
|
|
|
} else if (pid == USB_PID_IN) {
|
|
/*!< Record the data len */
|
|
g_chusb_hcd.current_pipe->xferlen = len;
|
|
|
|
if (len == 0) {
|
|
/*!< Want get 0 length data */
|
|
} else {
|
|
if ((uint32_t)data & 0x03) {
|
|
USB_LOG_INFO("IN DMA address is not align \r\n");
|
|
return -3;
|
|
}
|
|
USBHS_HOST->HOST_RX_DMA = (uint32_t)data;
|
|
pipe->buffer = data;
|
|
}
|
|
|
|
USBHS_HOST->HOST_EP_PID = pid << 4 | (pipe->ep_addr & 0x0f);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void chusb_control_pipe_init(struct chusb_pipe *pipe, struct usb_setup_packet *setup, uint8_t *buffer, uint32_t buflen)
|
|
{
|
|
if (g_chusb_hcd.ep0_state == USB_EP0_STATE_SETUP) {
|
|
/**
|
|
* Setup is distributed as DATA0
|
|
*/
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_AUTO);
|
|
pipe->data_pid = 0;
|
|
chusb_host_pipe_transfer(pipe, USB_PID_SETUP, (uint8_t *)setup, 8);
|
|
} else if (g_chusb_hcd.ep0_state == USB_EP0_STATE_IN_DATA) {
|
|
if (pipe->data_pid != 1) {
|
|
USB_LOG_ERR("IN_DATA PID Error\r\n");
|
|
}
|
|
SET_UH_RX_CTRL(USBHS_UH_R_TOG_AUTO | USBHS_UH_R_TOG_1);
|
|
chusb_host_pipe_transfer(pipe, USB_PID_IN, buffer, buflen);
|
|
} else if (g_chusb_hcd.ep0_state == USB_EP0_STATE_OUT_DATA) {
|
|
if (pipe->data_pid != 1) {
|
|
USB_LOG_ERR("OUT_DATA PID Error\r\n");
|
|
}
|
|
chusb_host_pipe_transfer(pipe, USB_PID_OUT, buffer, buflen);
|
|
} else if (g_chusb_hcd.ep0_state == USB_EP0_STATE_IN_STATUS) {
|
|
/**
|
|
* Status stage must be DATA1
|
|
*/
|
|
// SET_UH_RX_CTRL_BIT(USBFS_UH_R_TOG);
|
|
pipe->data_pid = 1;
|
|
SET_UH_RX_CTRL(USBHS_UH_R_TOG_AUTO | USBHS_UH_R_TOG_1);
|
|
chusb_host_pipe_transfer(pipe, USB_PID_IN, NULL, 0);
|
|
} else if (g_chusb_hcd.ep0_state == USB_EP0_STATE_OUT_STATUS) {
|
|
/**
|
|
* Status stage must be DATA1
|
|
*/
|
|
// SET_UH_TX_CTRL_BIT(USBFS_UH_T_TOG);
|
|
pipe->data_pid = 1;
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_AUTO | USBHS_UH_T_TOG_1);
|
|
chusb_host_pipe_transfer(pipe, USB_PID_OUT, NULL, 0);
|
|
}
|
|
}
|
|
|
|
static void chusb_bulk_pipe_init(struct chusb_pipe *pipe, uint8_t *buffer, uint32_t buflen)
|
|
{
|
|
if (pipe->ep_type != USB_ENDPOINT_TYPE_BULK) {
|
|
USB_LOG_ERR("Endpoint type is not BULK\r\n");
|
|
return;
|
|
}
|
|
|
|
if (pipe->ep_addr & 0x80) {
|
|
/*!< IN */
|
|
g_chusb_hcd.current_token = USB_PID_IN;
|
|
if (pipe->data_pid == 1) {
|
|
SET_UH_RX_CTRL(USBHS_UH_R_TOG_AUTO | USBHS_UH_R_TOG_1);
|
|
} else {
|
|
SET_UH_RX_CTRL(USBHS_UH_R_TOG_AUTO);
|
|
}
|
|
} else {
|
|
/*!< OUT */
|
|
g_chusb_hcd.current_token = USB_PID_OUT;
|
|
if (pipe->data_pid == 1) {
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_AUTO | USBHS_UH_T_TOG_1);
|
|
} else {
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_AUTO);
|
|
}
|
|
}
|
|
|
|
chusb_host_pipe_transfer(pipe, g_chusb_hcd.current_token, buffer, buflen);
|
|
}
|
|
|
|
static void chusb_intr_pipe_init(struct chusb_pipe *pipe, uint8_t *buffer, uint32_t buflen)
|
|
{
|
|
if (pipe->ep_type != USB_ENDPOINT_TYPE_INTERRUPT) {
|
|
USB_LOG_ERR("Endpoint type is not INTERRUPT\r\n");
|
|
return;
|
|
}
|
|
|
|
if (pipe->ep_addr & 0x80) {
|
|
/*!< IN */
|
|
g_chusb_hcd.current_token = USB_PID_IN;
|
|
|
|
if (pipe->trans_num > 1) {
|
|
USB_LOG_WRN("Not supported temporarily\r\n");
|
|
if (pipe->trans_num == 2) {
|
|
} else if (pipe->trans_num == 3) {
|
|
}
|
|
} else {
|
|
if (pipe->data_pid == 1) {
|
|
SET_UH_RX_CTRL(USBHS_UH_R_TOG_AUTO | USBHS_UH_R_TOG_1);
|
|
} else {
|
|
SET_UH_RX_CTRL(USBHS_UH_R_TOG_AUTO);
|
|
}
|
|
}
|
|
} else {
|
|
/*!< OUT */
|
|
g_chusb_hcd.current_token = USB_PID_OUT;
|
|
if (pipe->trans_num > 1) {
|
|
USB_LOG_WRN("Not supported temporarily\r\n");
|
|
if (pipe->trans_num == 2) {
|
|
} else if (pipe->trans_num == 3) {
|
|
}
|
|
} else {
|
|
if (pipe->data_pid == 1) {
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_AUTO | USBHS_UH_T_TOG_1);
|
|
} else {
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_AUTO);
|
|
}
|
|
}
|
|
}
|
|
|
|
chusb_host_pipe_transfer(pipe, g_chusb_hcd.current_token, buffer, buflen);
|
|
}
|
|
|
|
static void chusb_iso_pipe_init(struct chusb_pipe *pipe, uint8_t *buffer, uint32_t buflen)
|
|
{
|
|
if (pipe->ep_type != USB_ENDPOINT_TYPE_ISOCHRONOUS) {
|
|
USB_LOG_ERR("Endpoint type is not ISOCHRONOUS\r\n");
|
|
return;
|
|
}
|
|
|
|
if (pipe->ep_addr & 0x80) {
|
|
/*!< IN */
|
|
g_chusb_hcd.current_token = USB_PID_IN;
|
|
if (pipe->trans_num > 1) {
|
|
/**
|
|
* High speed and high bandwidth isochronous endpoint
|
|
*/
|
|
USB_LOG_WRN("Not supported temporarily\r\n");
|
|
#if 0
|
|
if (pipe->trans_num == 2) {
|
|
SET_UH_RX_CTRL(USBHS_UH_R_TOG_AUTO | USBHS_UH_R_TOG_1 | USBHS_UH_R_RES_NO);
|
|
} else if (pipe->trans_num == 3) {
|
|
SET_UH_RX_CTRL(USBHS_UH_R_TOG_AUTO | USBHS_UH_R_TOG_2 | USBHS_UH_R_RES_NO);
|
|
}
|
|
#endif
|
|
} else {
|
|
SET_UH_RX_CTRL(USBHS_UH_R_TOG_AUTO | USBHS_UH_R_RES_NO);
|
|
chusb_host_pipe_transfer(pipe, g_chusb_hcd.current_token, buffer, buflen);
|
|
return;
|
|
}
|
|
} else {
|
|
/*!< OUT */
|
|
g_chusb_hcd.current_token = USB_PID_OUT;
|
|
if (pipe->trans_num > 1) {
|
|
/**
|
|
* High speed and high bandwidth isochronous endpoint
|
|
*/
|
|
USB_LOG_WRN("Not supported temporarily\r\n");
|
|
#if 0
|
|
if (pipe->trans_num == 2) {
|
|
if (buflen < pipe->ep_mps) {
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_AUTO | USBHS_UH_T_RES_NO);
|
|
} else {
|
|
uint32_t send_len = buflen;
|
|
while (send_len > pipe->ep_mps) {
|
|
/**
|
|
* First send a MDATA
|
|
*/
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_3 | USBHS_UH_T_RES_NO);
|
|
chusb_host_pipe_transfer(pipe, g_chusb_hcd.current_token, buffer, send_len);
|
|
send_len -= pipe->ep_mps;
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_1 | USBHS_UH_T_RES_NO);
|
|
chusb_host_pipe_transfer(pipe, g_chusb_hcd.current_token, (buffer + pipe->ep_mps), send_len);
|
|
if (send_len > pipe->ep_mps) {
|
|
send_len -= pipe->ep_mps;
|
|
}
|
|
}
|
|
}
|
|
return;
|
|
} else if (pipe->trans_num == 3) {
|
|
if (buflen < pipe->ep_mps) {
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_AUTO | USBHS_UH_T_RES_NO);
|
|
} else if (buflen < (2 * pipe->ep_mps)) {
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_3 | USBHS_UH_T_RES_NO);
|
|
chusb_host_pipe_transfer(pipe, g_chusb_hcd.current_token, buffer, buflen);
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_1 | USBHS_UH_T_RES_NO);
|
|
chusb_host_pipe_transfer(pipe, g_chusb_hcd.current_token, (buffer + pipe->ep_mps), (buflen - pipe->ep_mps));
|
|
return;
|
|
} else {
|
|
uint32_t send_len = buflen;
|
|
while (send_len > pipe->ep_mps) {
|
|
/**
|
|
* First send a MDATA
|
|
*/
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_3 | USBHS_UH_T_RES_NO);
|
|
chusb_host_pipe_transfer(pipe, g_chusb_hcd.current_token, buffer, send_len);
|
|
send_len -= pipe->ep_mps;
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_3 | USBHS_UH_T_RES_NO);
|
|
chusb_host_pipe_transfer(pipe, g_chusb_hcd.current_token, (buffer + pipe->ep_mps), send_len);
|
|
send_len -= pipe->ep_mps;
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_2 | USBHS_UH_T_RES_NO);
|
|
chusb_host_pipe_transfer(pipe, g_chusb_hcd.current_token, (buffer + pipe->ep_mps), send_len);
|
|
if (send_len > pipe->ep_mps) {
|
|
send_len -= pipe->ep_mps;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
} else {
|
|
SET_UH_TX_CTRL(USBHS_UH_T_TOG_AUTO | USBHS_UH_T_RES_NO);
|
|
chusb_host_pipe_transfer(pipe, g_chusb_hcd.current_token, buffer, buflen);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void chusbh_set_self_speed(uint8_t speed)
|
|
{
|
|
if (speed == USB_SPEED_HIGH) {
|
|
USBHS_HOST->CONTROL = (USBHS_HOST->CONTROL & ~USBHS_SPEED_MASK) | USBHS_HIGH_SPEED;
|
|
} else if (speed == USB_SPEED_FULL) {
|
|
USBHS_HOST->CONTROL = (USBHS_HOST->CONTROL & ~USBHS_SPEED_MASK) | USBHS_FULL_SPEED;
|
|
} else {
|
|
USBHS_HOST->CONTROL = (USBHS_HOST->CONTROL & ~USBHS_SPEED_MASK) | USBHS_LOW_SPEED;
|
|
}
|
|
}
|
|
|
|
static int usbh_reset_port(const uint8_t port)
|
|
{
|
|
USB_LOG_INFO("Reset port \r\n");
|
|
/*!< Clear flags that may not have been cleared */
|
|
// g_chusb_hcd.main_pipe_using = false;
|
|
|
|
g_chusb_hcd.port_pe = 0;
|
|
/*!< Set dev add 0 */
|
|
USBHS_HOST->DEV_AD = (0x00 & 0x7F);
|
|
chusbh_set_self_speed(USB_SPEED_HIGH);
|
|
|
|
/*!< Start reset */
|
|
USBHS_HOST->HOST_CTRL |= USBHS_SEND_BUS_RESET;
|
|
usb_osal_msleep(11);
|
|
/*!< Stop reset */
|
|
USBHS_HOST->HOST_CTRL &= ~USBHS_SEND_BUS_RESET;
|
|
usb_osal_msleep(2);
|
|
g_chusb_hcd.port_pe = 1;
|
|
|
|
if (USBHS_HOST->MIS_ST & USBHS_ATTACH) {
|
|
volatile uint8_t speed = USBHS_HOST->SPEED_TYPE & USBSPEED_MASK;
|
|
if (speed == 0x02) {
|
|
USB_LOG_INFO("Dev USB_SPEED_LOW \r\n");
|
|
g_chusb_hcd.dev_speed = USB_SPEED_LOW;
|
|
chusbh_set_self_speed(USB_SPEED_LOW);
|
|
} else if (speed == 0x00) {
|
|
USB_LOG_INFO("Dev USB_SPEED_FULL \r\n");
|
|
g_chusb_hcd.dev_speed = USB_SPEED_FULL;
|
|
chusbh_set_self_speed(USB_SPEED_FULL);
|
|
} else if (speed == 0x01) {
|
|
USB_LOG_INFO("Dev USB_SPEED_HIGH \r\n");
|
|
g_chusb_hcd.dev_speed = USB_SPEED_HIGH;
|
|
chusbh_set_self_speed(USB_SPEED_HIGH);
|
|
}
|
|
}
|
|
|
|
USBHS_HOST->INT_EN |= USBHS_DETECT_EN;
|
|
return 0;
|
|
}
|
|
|
|
static uint8_t usbh_get_port_speed(const uint8_t port)
|
|
{
|
|
(void)port;
|
|
return g_chusb_hcd.dev_speed;
|
|
}
|
|
|
|
__WEAK void usb_hc_low_level_init(void)
|
|
{
|
|
}
|
|
|
|
int usb_hc_init(void)
|
|
{
|
|
memset(&g_chusb_hcd, 0, sizeof(struct chusb_hcd));
|
|
|
|
for (uint8_t i = 0; i < CONFIG_USBHOST_PIPE_NUM; i++) {
|
|
g_chusb_hcd.pipe_pool[i][0].waitsem = usb_osal_sem_create(0);
|
|
g_chusb_hcd.pipe_pool[i][1].waitsem = usb_osal_sem_create(0);
|
|
}
|
|
|
|
usb_hc_low_level_init();
|
|
|
|
/* Reset USB module */
|
|
USBHS_HOST->CONTROL = USBHS_ALL_CLR | USBHS_FORCE_RST;
|
|
static volatile uint32_t wait_ct = 50000;
|
|
while (wait_ct--) {
|
|
}
|
|
USBHS_HOST->CONTROL = 0;
|
|
|
|
/* Initialize USB host configuration */
|
|
USBHS_HOST->CONTROL = USBHS_HOST_MODE | USBHS_HIGH_SPEED | USBHS_INT_BUSY_EN | USBHS_DMA_EN;
|
|
USBHS_HOST->HOST_EP_CONFIG = USBHS_HOST_TX_EN | USBHS_HOST_RX_EN;
|
|
USBHS_HOST->HOST_CTRL = USBHS_PHY_SUSPENDM;
|
|
USBHS_HOST->HOST_RX_MAX_LEN = USBHS_MAX_PACKET_SIZE;
|
|
|
|
USBHS_HOST->HOST_RX_CTRL = 0;
|
|
USBHS_HOST->HOST_TX_CTRL = 0;
|
|
USBHS_HOST->INT_FG = 0xFF;
|
|
USBHS_HOST->INT_EN = USBHS_TRANSFER_EN | USBHS_DETECT_EN;
|
|
return 0;
|
|
}
|
|
|
|
int usbh_roothub_control(struct usb_setup_packet *setup, uint8_t *buf)
|
|
{
|
|
uint8_t nports;
|
|
uint8_t port;
|
|
uint32_t status;
|
|
|
|
nports = CONFIG_USBHOST_MAX_RHPORTS;
|
|
port = setup->wIndex;
|
|
if (setup->bmRequestType & USB_REQUEST_RECIPIENT_DEVICE) {
|
|
switch (setup->bRequest) {
|
|
case HUB_REQUEST_CLEAR_FEATURE:
|
|
switch (setup->wValue) {
|
|
case HUB_FEATURE_HUB_C_LOCALPOWER:
|
|
break;
|
|
case HUB_FEATURE_HUB_C_OVERCURRENT:
|
|
break;
|
|
default:
|
|
return -EPIPE;
|
|
}
|
|
break;
|
|
case HUB_REQUEST_SET_FEATURE:
|
|
switch (setup->wValue) {
|
|
case HUB_FEATURE_HUB_C_LOCALPOWER:
|
|
break;
|
|
case HUB_FEATURE_HUB_C_OVERCURRENT:
|
|
break;
|
|
default:
|
|
return -EPIPE;
|
|
}
|
|
break;
|
|
case HUB_REQUEST_GET_DESCRIPTOR:
|
|
break;
|
|
case HUB_REQUEST_GET_STATUS:
|
|
memset(buf, 0, 4);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
} else if (setup->bmRequestType & USB_REQUEST_RECIPIENT_OTHER) {
|
|
switch (setup->bRequest) {
|
|
case HUB_REQUEST_CLEAR_FEATURE:
|
|
if (!port || port > nports) {
|
|
return -EPIPE;
|
|
}
|
|
|
|
switch (setup->wValue) {
|
|
case HUB_PORT_FEATURE_ENABLE:
|
|
break;
|
|
case HUB_PORT_FEATURE_SUSPEND:
|
|
case HUB_PORT_FEATURE_C_SUSPEND:
|
|
break;
|
|
case HUB_PORT_FEATURE_POWER:
|
|
break;
|
|
case HUB_PORT_FEATURE_C_CONNECTION:
|
|
g_chusb_hcd.port_csc = 0;
|
|
break;
|
|
case HUB_PORT_FEATURE_C_ENABLE:
|
|
g_chusb_hcd.port_pec = 0;
|
|
break;
|
|
case HUB_PORT_FEATURE_C_OVER_CURREN:
|
|
break;
|
|
case HUB_PORT_FEATURE_C_RESET:
|
|
break;
|
|
default:
|
|
return -EPIPE;
|
|
}
|
|
break;
|
|
case HUB_REQUEST_SET_FEATURE:
|
|
if (!port || port > nports) {
|
|
return -EPIPE;
|
|
}
|
|
|
|
switch (setup->wValue) {
|
|
case HUB_PORT_FEATURE_SUSPEND:
|
|
break;
|
|
case HUB_PORT_FEATURE_POWER:
|
|
break;
|
|
case HUB_PORT_FEATURE_RESET:
|
|
usbh_reset_port(port);
|
|
break;
|
|
|
|
default:
|
|
return -EPIPE;
|
|
}
|
|
break;
|
|
case HUB_REQUEST_GET_STATUS:
|
|
if (!port || port > nports) {
|
|
return -EPIPE;
|
|
}
|
|
|
|
status = 0;
|
|
if (g_chusb_hcd.port_csc) {
|
|
status |= (1 << HUB_PORT_FEATURE_C_CONNECTION);
|
|
}
|
|
if (g_chusb_hcd.port_pec) {
|
|
status |= (1 << HUB_PORT_FEATURE_C_ENABLE);
|
|
}
|
|
|
|
if (g_chusb_hcd.port_pe) {
|
|
status |= (1 << HUB_PORT_FEATURE_CONNECTION);
|
|
status |= (1 << HUB_PORT_FEATURE_ENABLE);
|
|
if (usbh_get_port_speed(port) == USB_SPEED_LOW) {
|
|
status |= (1 << HUB_PORT_FEATURE_LOWSPEED);
|
|
} else if (usbh_get_port_speed(port) == USB_SPEED_HIGH) {
|
|
status |= (1 << HUB_PORT_FEATURE_HIGHSPEED);
|
|
}
|
|
}
|
|
|
|
memcpy(buf, &status, 4);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int usbh_ep_pipe_reconfigure(usbh_pipe_t pipe, uint8_t dev_addr, uint8_t ep_mps, uint8_t speed)
|
|
{
|
|
struct chusb_pipe *ppipe = (struct chusb_pipe *)pipe;
|
|
|
|
ppipe->dev_addr = dev_addr;
|
|
ppipe->ep_mps = ep_mps;
|
|
|
|
USBHS_HOST->DEV_AD = dev_addr & 0x7f;
|
|
return 0;
|
|
}
|
|
|
|
int usbh_pipe_alloc(usbh_pipe_t *pipe, const struct usbh_endpoint_cfg *ep_cfg)
|
|
{
|
|
struct chusb_pipe *ppipe;
|
|
uint8_t ep_idx;
|
|
usb_osal_sem_t waitsem;
|
|
|
|
ep_idx = ep_cfg->ep_addr & 0x7f;
|
|
|
|
if (ep_idx > CONFIG_USBHOST_PIPE_NUM) {
|
|
return -ENOMEM;
|
|
}
|
|
|
|
if (ep_cfg->ep_addr & 0x80) {
|
|
ppipe = &g_chusb_hcd.pipe_pool[ep_idx][1];
|
|
} else {
|
|
ppipe = &g_chusb_hcd.pipe_pool[ep_idx][0];
|
|
}
|
|
|
|
/* store variables */
|
|
waitsem = ppipe->waitsem;
|
|
|
|
memset(ppipe, 0, sizeof(struct chusb_pipe));
|
|
|
|
ppipe->ep_addr = ep_cfg->ep_addr;
|
|
ppipe->ep_type = ep_cfg->ep_type;
|
|
ppipe->ep_mps = ep_cfg->ep_mps;
|
|
ppipe->ep_interval = ep_cfg->ep_interval;
|
|
ppipe->speed = ep_cfg->hport->speed;
|
|
ppipe->dev_addr = ep_cfg->hport->dev_addr;
|
|
ppipe->hport = ep_cfg->hport;
|
|
|
|
if (ep_cfg->ep_type == USB_ENDPOINT_TYPE_CONTROL) {
|
|
} else {
|
|
if (ppipe->speed == USB_SPEED_HIGH) {
|
|
if ((ep_cfg->ep_type == USB_ENDPOINT_TYPE_ISOCHRONOUS) ||
|
|
(ep_cfg->ep_type == USB_ENDPOINT_TYPE_INTERRUPT)) {
|
|
if ((ppipe->ep_mps & HIGH_SP_HIGH_BAND_Mask) != 0) {
|
|
/**
|
|
* High speed and high bandwidth isochronous endpoint
|
|
*/
|
|
if ((ppipe->ep_mps & HIGH_SP_HIGH_BAND_Mask) == HIGH_SP_HIGH_BAND_2) {
|
|
ppipe->trans_num = 2;
|
|
} else if ((ppipe->ep_mps & HIGH_SP_HIGH_BAND_Mask) == HIGH_SP_HIGH_BAND_3) {
|
|
ppipe->trans_num = 3;
|
|
}
|
|
} else {
|
|
ppipe->trans_num = 1;
|
|
}
|
|
}
|
|
} else if (ppipe->speed == USB_SPEED_FULL) {
|
|
} else if (ppipe->speed == USB_SPEED_LOW) {
|
|
}
|
|
}
|
|
/* restore variable */
|
|
ppipe->inuse = true;
|
|
ppipe->waitsem = waitsem;
|
|
|
|
*pipe = (usbh_pipe_t)ppipe;
|
|
return 0;
|
|
}
|
|
|
|
int usbh_pipe_free(usbh_pipe_t pipe)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static inline void chusb_pipe_waitup(struct chusb_pipe *pipe, bool callback)
|
|
{
|
|
struct usbh_urb *urb;
|
|
|
|
urb = pipe->urb;
|
|
pipe->urb = NULL;
|
|
// g_chusb_hcd.main_pipe_using = false;
|
|
|
|
if (pipe->waiter) {
|
|
pipe->waiter = false;
|
|
usb_osal_sem_give(pipe->waitsem);
|
|
}
|
|
|
|
if (callback == true) {
|
|
if (urb->complete) {
|
|
if (urb->errorcode < 0) {
|
|
urb->complete(urb->arg, urb->errorcode);
|
|
} else {
|
|
urb->complete(urb->arg, urb->actual_length);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
int usbh_submit_urb(struct usbh_urb *urb)
|
|
{
|
|
struct chusb_pipe *pipe;
|
|
size_t flags;
|
|
int ret = 0;
|
|
|
|
if (!urb) {
|
|
USB_LOG_INFO("urb is null \r\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
pipe = urb->pipe;
|
|
|
|
if (!pipe) {
|
|
USB_LOG_INFO("pipe is null \r\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!pipe->hport->connected) {
|
|
USB_LOG_INFO("!pipe->hport->connected \r\n");
|
|
return -ENODEV;
|
|
}
|
|
|
|
if (pipe->urb) {
|
|
USB_LOG_INFO("pipe->urb is not null\r\n");
|
|
return -EBUSY;
|
|
}
|
|
|
|
#if 0
|
|
if (g_chusb_hcd.main_pipe_using == true) {
|
|
USB_LOG_INFO("usbh_submit_urb//main pipe is using\r\n");
|
|
return -EBUSY;
|
|
}
|
|
#endif
|
|
|
|
flags = usb_osal_enter_critical_section();
|
|
|
|
pipe->waiter = false;
|
|
pipe->xfrd = 0;
|
|
pipe->urb = urb;
|
|
urb->errorcode = -EBUSY;
|
|
urb->actual_length = 0;
|
|
|
|
if (urb->timeout > 0) {
|
|
pipe->waiter = true;
|
|
}
|
|
usb_osal_leave_critical_section(flags);
|
|
|
|
switch (pipe->ep_type) {
|
|
case USB_ENDPOINT_TYPE_CONTROL:
|
|
g_chusb_hcd.ep0_state = USB_EP0_STATE_SETUP;
|
|
chusb_control_pipe_init(pipe, urb->setup, urb->transfer_buffer, urb->transfer_buffer_length);
|
|
break;
|
|
case USB_ENDPOINT_TYPE_BULK:
|
|
chusb_bulk_pipe_init(pipe, urb->transfer_buffer, urb->transfer_buffer_length);
|
|
break;
|
|
case USB_ENDPOINT_TYPE_INTERRUPT:
|
|
chusb_intr_pipe_init(pipe, urb->transfer_buffer, urb->transfer_buffer_length);
|
|
break;
|
|
case USB_ENDPOINT_TYPE_ISOCHRONOUS:
|
|
chusb_iso_pipe_init(pipe, urb->transfer_buffer, urb->transfer_buffer_length);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
if (urb->timeout > 0) {
|
|
/* wait until timeout or sem give */
|
|
ret = usb_osal_sem_take(pipe->waitsem, urb->timeout);
|
|
if (ret < 0) {
|
|
goto errout_timeout;
|
|
}
|
|
|
|
ret = urb->errorcode;
|
|
}
|
|
return ret;
|
|
errout_timeout:
|
|
pipe->waiter = false;
|
|
g_chusb_hcd.current_token = 0;
|
|
// g_chusb_hcd.main_pipe_using = false;
|
|
chusb_pipe_waitup(pipe, false);
|
|
usbh_kill_urb(urb);
|
|
return ret;
|
|
}
|
|
|
|
int usbh_kill_urb(struct usbh_urb *urb)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static int8_t chusb_outpipe_irq_handler(uint8_t res_state)
|
|
{
|
|
uint16_t current_tx_length = USBHS_HOST->HOST_TX_LEN;
|
|
struct usbh_urb *urb;
|
|
urb = (g_chusb_hcd.current_pipe->urb);
|
|
|
|
if (g_chusb_hcd.current_pipe->ep_type != USB_ENDPOINT_TYPE_CONTROL) {
|
|
if ((g_chusb_hcd.current_pipe->ep_addr & 0x80) != 0) {
|
|
/* Error */
|
|
USB_LOG_ERR("ep_addr is not out add \r\n");
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
switch (res_state) {
|
|
case USB_PID_STALL:
|
|
urb->errorcode = -EPERM;
|
|
if (g_chusb_hcd.current_token == USB_PID_SETUP) {
|
|
USB_LOG_ERR("USB_PID_SETUP STALL \r\n");
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return -2;
|
|
} else {
|
|
if (g_chusb_hcd.current_pipe->waiter == true) {
|
|
USB_LOG_WRN("USB_PID_OUT STALL \r\n");
|
|
chusb_host_pipe_transfer(g_chusb_hcd.current_pipe, USB_PID_OUT,
|
|
g_chusb_hcd.current_pipe->buffer, g_chusb_hcd.current_pipe->xferlen);
|
|
} else {
|
|
USB_LOG_ERR("USB_PID_OUT STALL \r\n");
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return -2;
|
|
}
|
|
}
|
|
case USB_PID_NAK:
|
|
urb->errorcode = -EAGAIN;
|
|
if (g_chusb_hcd.current_pipe->ep_type == USB_ENDPOINT_TYPE_CONTROL) {
|
|
if (g_chusb_hcd.current_token == USB_PID_SETUP) {
|
|
/**
|
|
* Device must ack for setup package
|
|
*/
|
|
USB_LOG_ERR("Setup NAK \r\n");
|
|
g_chusb_hcd.current_token = 0;
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return -3;
|
|
} else {
|
|
if (g_chusb_hcd.current_pipe->waiter == true) {
|
|
USB_LOG_DBG("Control endpoint out nak and retry, length:%d\r\n",g_chusb_hcd.current_pipe->xferlen);
|
|
chusb_host_pipe_transfer(g_chusb_hcd.current_pipe, USB_PID_OUT,
|
|
g_chusb_hcd.current_pipe->buffer, g_chusb_hcd.current_pipe->xferlen);
|
|
}
|
|
}
|
|
} else {
|
|
urb->errorcode = 0;
|
|
if (g_chusb_hcd.prv_set_zero == true) {
|
|
/**
|
|
* It is unlikely to run here,
|
|
* because the device can probably receive 0 length byte packets
|
|
*/
|
|
g_chusb_hcd.prv_set_zero = false;
|
|
g_chusb_hcd.current_token = 0;
|
|
urb->actual_length = g_chusb_hcd.current_pipe->xfrd;
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, true);
|
|
} else {
|
|
if (g_chusb_hcd.current_pipe->waiter == true) {
|
|
USB_LOG_DBG("Normal endpoint out nak and retry\r\n");
|
|
chusb_host_pipe_transfer(g_chusb_hcd.current_pipe, USB_PID_OUT,
|
|
g_chusb_hcd.current_pipe->buffer, g_chusb_hcd.current_pipe->xferlen);
|
|
} else {
|
|
g_chusb_hcd.current_token = 0;
|
|
if (g_chusb_hcd.current_pipe->xfrd > 0) {
|
|
/**
|
|
* The device received data but did not receive it completely
|
|
*/
|
|
urb->errorcode = 0;
|
|
USB_LOG_WRN("The data is not sent completely, but the timeout is 0\r\n");
|
|
urb->actual_length = g_chusb_hcd.current_pipe->xfrd;
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, true);
|
|
} else {
|
|
/**
|
|
* The device may not be ready, make sure your device can receive data
|
|
*/
|
|
USB_LOG_WRN("The device may not be ready, make sure your device can receive data\r\n");
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, false);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case USB_PID_ACK:
|
|
case USB_PID_NYET:
|
|
/**
|
|
* The last out or setup package was sent
|
|
*/
|
|
urb->errorcode = 0;
|
|
|
|
if (g_chusb_hcd.current_pipe->ep_type == USB_ENDPOINT_TYPE_CONTROL) {
|
|
/*!< Ctrol endpoint */
|
|
if (g_chusb_hcd.current_token == USB_PID_SETUP) {
|
|
/*!< Setup package send successfully */
|
|
urb->actual_length += 8;
|
|
g_chusb_hcd.current_pipe->data_pid ^= 1;
|
|
if (g_chusb_hcd.ep0_state == USB_EP0_STATE_SETUP) {
|
|
if (urb->setup->wLength) {
|
|
if (urb->setup->bmRequestType & 0x80) {
|
|
g_chusb_hcd.ep0_state = USB_EP0_STATE_IN_DATA;
|
|
} else {
|
|
g_chusb_hcd.ep0_state = USB_EP0_STATE_OUT_DATA;
|
|
}
|
|
} else {
|
|
g_chusb_hcd.ep0_state = USB_EP0_STATE_IN_STATUS;
|
|
}
|
|
chusb_control_pipe_init(g_chusb_hcd.current_pipe, urb->setup,
|
|
urb->transfer_buffer, urb->transfer_buffer_length);
|
|
}
|
|
} else if (g_chusb_hcd.current_token == USB_PID_OUT) {
|
|
if (g_chusb_hcd.ep0_state == USB_EP0_STATE_OUT_DATA) {
|
|
USB_LOG_DBG("ep0 tx_len:%d\r\n", current_tx_length);
|
|
g_chusb_hcd.current_pipe->data_pid ^= 1;
|
|
g_chusb_hcd.current_pipe->xfrd += current_tx_length;
|
|
g_chusb_hcd.current_pipe->buffer += current_tx_length;
|
|
g_chusb_hcd.current_pipe->xferlen -= current_tx_length;
|
|
if (g_chusb_hcd.current_pipe->xferlen == 0) {
|
|
if (current_tx_length == g_chusb_hcd.current_pipe->ep_mps) {
|
|
/**
|
|
* Need send 0 length data
|
|
*/
|
|
chusb_host_pipe_transfer(g_chusb_hcd.current_pipe, USB_PID_OUT, NULL, 0);
|
|
} else {
|
|
/*!< Out package send successfully */
|
|
g_chusb_hcd.ep0_state = USB_EP0_STATE_IN_STATUS;
|
|
chusb_control_pipe_init(g_chusb_hcd.current_pipe, urb->setup,
|
|
urb->transfer_buffer, urb->transfer_buffer_length);
|
|
}
|
|
} else {
|
|
/*!< Start send next out package */
|
|
chusb_host_pipe_transfer(g_chusb_hcd.current_pipe, USB_PID_OUT,
|
|
g_chusb_hcd.current_pipe->buffer, g_chusb_hcd.current_pipe->xferlen);
|
|
}
|
|
} else if (g_chusb_hcd.ep0_state == USB_EP0_STATE_OUT_STATUS) {
|
|
g_chusb_hcd.ep0_state = USB_EP0_STATE_SETUP;
|
|
urb->actual_length += g_chusb_hcd.current_pipe->xfrd;
|
|
USB_LOG_DBG("S-> I-> O-status stage: In:%d\r\n", urb->actual_length - 8);
|
|
if (g_chusb_hcd.current_pipe->data_pid != 1) {
|
|
USB_LOG_ERR("S-> I-> O-status stage DATA PID Error\r\n");
|
|
} else {
|
|
g_chusb_hcd.current_pipe->data_pid ^= 1;
|
|
}
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, true);
|
|
}
|
|
}
|
|
} else {
|
|
if (g_chusb_hcd.current_token == USB_PID_OUT) {
|
|
USB_LOG_DBG("ep%d tx_len:%d\r\n", (g_chusb_hcd.current_pipe->ep_addr & 0x0f), current_tx_length);
|
|
g_chusb_hcd.current_pipe->data_pid ^= 1;
|
|
g_chusb_hcd.current_pipe->xfrd += current_tx_length;
|
|
g_chusb_hcd.current_pipe->buffer += current_tx_length;
|
|
g_chusb_hcd.current_pipe->xferlen -= current_tx_length;
|
|
if (g_chusb_hcd.current_pipe->xferlen == 0) {
|
|
if (current_tx_length == g_chusb_hcd.current_pipe->ep_mps) {
|
|
/**
|
|
* Need send 0 length data
|
|
*/
|
|
if (g_chusb_hcd.prv_set_zero == true) {
|
|
USB_LOG_ERR("g_chusb_hcd.prv_set_zero is always true\r\n");
|
|
}
|
|
g_chusb_hcd.prv_set_zero = true;
|
|
chusb_host_pipe_transfer(g_chusb_hcd.current_pipe, USB_PID_OUT, NULL, 0);
|
|
} else {
|
|
/*!< Out package send successfully */
|
|
g_chusb_hcd.current_token = 0;
|
|
if (g_chusb_hcd.prv_set_zero == true) {
|
|
g_chusb_hcd.prv_set_zero = false;
|
|
}
|
|
urb->actual_length = g_chusb_hcd.current_pipe->xfrd;
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, true);
|
|
}
|
|
} else {
|
|
/*!< Start send next out package */
|
|
chusb_host_pipe_transfer(g_chusb_hcd.current_pipe, USB_PID_OUT,
|
|
g_chusb_hcd.current_pipe->buffer, g_chusb_hcd.current_pipe->xferlen);
|
|
}
|
|
} else {
|
|
USB_LOG_ERR("Error token is not match \r\n");
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return -4;
|
|
}
|
|
}
|
|
break;
|
|
case 0:
|
|
if (g_chusb_hcd.current_pipe->ep_type != USB_ENDPOINT_TYPE_ISOCHRONOUS) {
|
|
if (g_chusb_hcd.current_token == USB_PID_SETUP) {
|
|
if ((g_chusb_hcd.ep0_state == USB_EP0_STATE_SETUP) && (g_chusb_hcd.current_pipe->waiter == true)) {
|
|
USB_LOG_WRN("Setup Timeout and retry\r\n");
|
|
chusb_control_pipe_init(g_chusb_hcd.current_pipe, urb->setup,
|
|
urb->transfer_buffer, urb->transfer_buffer_length);
|
|
} else {
|
|
USB_LOG_ERR("Setup Timeout\r\n");
|
|
urb->errorcode = -EIO;
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return -5;
|
|
}
|
|
} else {
|
|
USB_LOG_ERR("Out Timeout \r\n");
|
|
if (g_chusb_hcd.current_pipe->ep_type != USB_ENDPOINT_TYPE_CONTROL) {
|
|
/**
|
|
* Reset data pid
|
|
*/
|
|
g_chusb_hcd.current_pipe->data_pid = 0;
|
|
}
|
|
}
|
|
urb->errorcode = -EIO;
|
|
} else {
|
|
/**
|
|
* No response from isochronous endpoint out
|
|
*/
|
|
urb->errorcode = 0;
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, true);
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return 0;
|
|
}
|
|
|
|
static int8_t chusb_inpipe_irq_handler(uint8_t res_state)
|
|
{
|
|
uint16_t rx_len = 0;
|
|
struct usbh_urb *urb;
|
|
urb = (g_chusb_hcd.current_pipe->urb);
|
|
|
|
if (g_chusb_hcd.current_pipe->ep_type != USB_ENDPOINT_TYPE_CONTROL) {
|
|
if ((g_chusb_hcd.current_pipe->ep_addr & 0x80) == 0) {
|
|
/* Error */
|
|
USB_LOG_ERR("ep_addr is not in add\r\n");
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
switch (res_state) {
|
|
case USB_PID_STALL:
|
|
USB_LOG_ERR("USB_PID_IN USB_PID_STALL\r\n");
|
|
g_chusb_hcd.current_token = 0;
|
|
urb->errorcode = -EPERM;
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return -2;
|
|
case USB_PID_NAK:
|
|
urb->errorcode = -EAGAIN;
|
|
g_chusb_hcd.current_token = 0;
|
|
if (g_chusb_hcd.current_pipe->ep_type == USB_ENDPOINT_TYPE_CONTROL) {
|
|
if (g_chusb_hcd.current_pipe->waiter == true) {
|
|
chusb_host_pipe_transfer(g_chusb_hcd.current_pipe, USB_PID_IN,
|
|
g_chusb_hcd.current_pipe->buffer, g_chusb_hcd.current_pipe->xferlen);
|
|
}
|
|
} else {
|
|
urb->errorcode = 0;
|
|
if (g_chusb_hcd.prv_get_zero == true) {
|
|
g_chusb_hcd.prv_get_zero = false;
|
|
g_chusb_hcd.current_token = 0;
|
|
urb->actual_length = g_chusb_hcd.current_pipe->xfrd;
|
|
USB_LOG_DBG("Normal endpoint get zero length package\r\n");
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, true);
|
|
} else {
|
|
USB_LOG_DBG("Normal endpoint in nak\r\n");
|
|
if (g_chusb_hcd.current_pipe->xferlen > 0) {
|
|
/**
|
|
* The data has not been transmitted completely
|
|
*/
|
|
if (g_chusb_hcd.current_pipe->xfrd > 0) {
|
|
/**
|
|
* Data was transmitted last time, but this time NAK
|
|
*/
|
|
if (g_chusb_hcd.current_pipe->waiter == true) {
|
|
chusb_host_pipe_transfer(g_chusb_hcd.current_pipe, USB_PID_IN,
|
|
g_chusb_hcd.current_pipe->buffer, g_chusb_hcd.current_pipe->xferlen);
|
|
} else {
|
|
/**
|
|
* Some data has been transferred
|
|
*/
|
|
USB_LOG_WRN("The device has not finished sending all data, but the timeout is 0\r\n");
|
|
g_chusb_hcd.current_token = 0;
|
|
/**
|
|
* Update the actual send length
|
|
*/
|
|
urb->actual_length = g_chusb_hcd.current_pipe->xfrd;
|
|
urb->errorcode = 0;
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, true);
|
|
}
|
|
} else {
|
|
/**
|
|
* The device did not send any data.
|
|
*/
|
|
if (g_chusb_hcd.current_pipe->waiter == true) {
|
|
/**
|
|
* Try again
|
|
*/
|
|
USB_LOG_DBG("The device does not transmit data, try again\r\n");
|
|
chusb_host_pipe_transfer(g_chusb_hcd.current_pipe, USB_PID_IN,
|
|
g_chusb_hcd.current_pipe->buffer, g_chusb_hcd.current_pipe->xferlen);
|
|
} else {
|
|
/**
|
|
* g_chusb_hcd.current_pipe->waiter = false
|
|
* We do not need to call a callback
|
|
*/
|
|
USB_LOG_DBG("Do not need try again\r\n");
|
|
urb->errorcode = -EBUSY;
|
|
g_chusb_hcd.current_token = 0;
|
|
urb->actual_length = 0;
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, false);
|
|
}
|
|
}
|
|
} else {
|
|
USB_LOG_ERR("xferlen == 0//should get zero package\r\n");
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case USB_PID_ACK:
|
|
urb->errorcode = 0;
|
|
break;
|
|
case USB_PID_DATA0:
|
|
case USB_PID_DATA1:
|
|
case USB_PID_DATA2:
|
|
case USB_PID_MDATA:
|
|
if ((USBHS_HOST->INT_ST) & USBHS_DEV_UIS_TOG_OK) {
|
|
/*!< Data is OK */
|
|
rx_len = USBHS_HOST->RX_LEN;
|
|
|
|
urb->errorcode = 0;
|
|
|
|
g_chusb_hcd.current_pipe->xfrd += rx_len;
|
|
|
|
if (g_chusb_hcd.current_pipe->xferlen < rx_len) {
|
|
g_chusb_hcd.current_pipe->data_pid ^= 1;
|
|
USB_LOG_ERR("Please provide the correct data length parameter xferlen:%d rxlen:%d\r\n",g_chusb_hcd.current_pipe->xferlen, rx_len);
|
|
if (g_chusb_hcd.prv_get_zero == true) {
|
|
g_chusb_hcd.prv_get_zero = false;
|
|
}
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return -6;
|
|
}
|
|
|
|
g_chusb_hcd.current_pipe->xferlen -= rx_len;
|
|
|
|
if (g_chusb_hcd.current_pipe->ep_type == USB_ENDPOINT_TYPE_CONTROL) {
|
|
/*!< Ctrol endpoint */
|
|
/**
|
|
* Status stage
|
|
*
|
|
* Setup ---> out data ---> in status stage
|
|
*/
|
|
if ((g_chusb_hcd.ep0_state == USB_EP0_STATE_IN_STATUS) && (rx_len == 0)) {
|
|
g_chusb_hcd.ep0_state = USB_EP0_STATE_SETUP;
|
|
urb->actual_length += rx_len;
|
|
USB_LOG_DBG("S-> O-> In-status stage: Out:%d\r\n", urb->actual_length - 8);
|
|
|
|
if (g_chusb_hcd.current_pipe->data_pid != 1) {
|
|
USB_LOG_ERR("S-> O-> In-status stage DATA PID Error//PID:%d\r\n", g_chusb_hcd.current_pipe->data_pid);
|
|
} else {
|
|
g_chusb_hcd.current_pipe->data_pid ^= 1;
|
|
}
|
|
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, true);
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Setup ---> in data ---> need generate next out status stage
|
|
*/
|
|
if (g_chusb_hcd.ep0_state == USB_EP0_STATE_IN_DATA) {
|
|
USB_LOG_DBG("ep0 rx_len:%d\r\n", rx_len);
|
|
g_chusb_hcd.current_pipe->data_pid ^= 1;
|
|
if ((g_chusb_hcd.current_pipe->xfrd) > (urb->setup->wLength)) {
|
|
/**
|
|
* Error
|
|
*/
|
|
USB_LOG_ERR("xfrd > setup->wLength\r\n");
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return -3;
|
|
}
|
|
|
|
if (rx_len == 0 || (rx_len & (g_chusb_hcd.current_pipe->ep_mps - 1))) {
|
|
/**
|
|
* Receive a short package, in data has transfer completed
|
|
*/
|
|
g_chusb_hcd.current_token = 0;
|
|
g_chusb_hcd.ep0_state = USB_EP0_STATE_OUT_STATUS;
|
|
/**
|
|
* generate next out status stage
|
|
*/
|
|
chusb_control_pipe_init(g_chusb_hcd.current_pipe, urb->setup,
|
|
urb->transfer_buffer, urb->transfer_buffer_length);
|
|
} else {
|
|
/**
|
|
* Retry in
|
|
*/
|
|
g_chusb_hcd.current_pipe->buffer += rx_len;
|
|
chusb_host_pipe_transfer(g_chusb_hcd.current_pipe, USB_PID_IN,
|
|
g_chusb_hcd.current_pipe->buffer, g_chusb_hcd.current_pipe->xferlen);
|
|
}
|
|
}
|
|
} else {
|
|
USB_LOG_DBG("ep%d rx_len:%d\r\n", (g_chusb_hcd.current_pipe->ep_addr & 0x0f), rx_len);
|
|
g_chusb_hcd.current_pipe->data_pid ^= 1;
|
|
if (rx_len == 0 || (rx_len & (g_chusb_hcd.current_pipe->ep_mps - 1)) || (g_chusb_hcd.current_pipe->xfrd == g_chusb_hcd.current_pipe->urb->transfer_buffer_length)) {
|
|
/**
|
|
* Receive a short package, in data has transfer completed
|
|
*/
|
|
g_chusb_hcd.current_token = 0;
|
|
urb->actual_length = g_chusb_hcd.current_pipe->xfrd;
|
|
|
|
if (g_chusb_hcd.prv_get_zero == true) {
|
|
g_chusb_hcd.prv_get_zero = false;
|
|
}
|
|
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, true);
|
|
} else {
|
|
/**
|
|
* Initiate In again
|
|
*/
|
|
g_chusb_hcd.current_pipe->buffer += rx_len;
|
|
|
|
if (g_chusb_hcd.current_pipe->xferlen == 0) {
|
|
if (g_chusb_hcd.prv_get_zero == true) {
|
|
USB_LOG_ERR("g_chusb_hcd.prv_get_zero is always true\r\n");
|
|
g_chusb_hcd.prv_get_zero = false;
|
|
}
|
|
g_chusb_hcd.prv_get_zero = true;
|
|
}
|
|
|
|
chusb_host_pipe_transfer(g_chusb_hcd.current_pipe, USB_PID_IN,
|
|
g_chusb_hcd.current_pipe->buffer, g_chusb_hcd.current_pipe->xferlen);
|
|
}
|
|
}
|
|
} else {
|
|
/*!< Discard data out of synchronization */
|
|
USB_LOG_ERR("EP%dData out of sync %d\r\n",
|
|
(g_chusb_hcd.current_pipe->ep_addr & 0x0f), g_chusb_hcd.current_pipe->data_pid);
|
|
urb->errorcode = -EIO;
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return -3;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
return 0;
|
|
}
|
|
|
|
__attribute__((interrupt("WCH-Interrupt-fast"))) void USBH_IRQHandler(void)
|
|
{
|
|
INT_PRE_HANDLER();
|
|
volatile uint8_t intflag = 0;
|
|
volatile uint8_t res = 0;
|
|
intflag = USBHS_HOST->INT_FG;
|
|
|
|
if (intflag & USBHS_TRANSFER_FLAG) {
|
|
/*!< Check the equipment response status */
|
|
res = (USBHS_HOST->INT_ST) & MASK_UIS_H_RES;
|
|
USBHS_HOST->HOST_EP_PID = 0x00;
|
|
switch (g_chusb_hcd.current_token) {
|
|
case USB_PID_SETUP:
|
|
case USB_PID_OUT:
|
|
if (chusb_outpipe_irq_handler(res) < 0) {
|
|
goto pipe_wait;
|
|
}
|
|
break;
|
|
case USB_PID_IN:
|
|
if (chusb_inpipe_irq_handler(res) < 0) {
|
|
goto pipe_wait;
|
|
}
|
|
break;
|
|
default:
|
|
USBHS_HOST->INT_FG = USBHS_TRANSFER_FLAG;
|
|
break;
|
|
}
|
|
} else if (intflag & USBHS_DETECT_FLAG) {
|
|
if (USBHS_HOST->MIS_ST & USBHS_ATTACH) {
|
|
USB_LOG_INFO("Dev connect \r\n");
|
|
g_chusb_hcd.port_csc = 1;
|
|
g_chusb_hcd.port_pec = 1;
|
|
g_chusb_hcd.port_pe = 1;
|
|
USBHS_HOST->HOST_CTRL |= USBHS_SEND_SOF_EN;
|
|
usbh_roothub_thread_wakeup(1);
|
|
} else {
|
|
USB_LOG_INFO("Dev remove \r\n");
|
|
USBHS_HOST->HOST_EP_PID = 0x00;
|
|
#if 0
|
|
if (g_chusb_hcd.main_pipe_using) {
|
|
g_chusb_hcd.main_pipe_using = false;
|
|
}
|
|
#endif
|
|
g_chusb_hcd.port_csc = 1;
|
|
g_chusb_hcd.port_pec = 1;
|
|
g_chusb_hcd.port_pe = 0;
|
|
for (uint8_t index = 0; index < CONFIG_USBHOST_PIPE_NUM; index++) {
|
|
for (uint8_t j = 0; j < 2; j++) {
|
|
struct chusb_pipe *pipe = &g_chusb_hcd.pipe_pool[index][j];
|
|
struct usbh_urb *urb = pipe->urb;
|
|
if (pipe->waiter) {
|
|
pipe->waiter = false;
|
|
urb->errorcode = -ESHUTDOWN;
|
|
usb_osal_sem_give(pipe->waitsem);
|
|
}
|
|
}
|
|
}
|
|
usbh_roothub_thread_wakeup(1);
|
|
}
|
|
USBHS_HOST->INT_FG = USBHS_DETECT_FLAG;
|
|
} else {
|
|
USB_LOG_INFO("Unkonwn \r\n");
|
|
USBHS_HOST->INT_FG = intflag;
|
|
}
|
|
INT_POST_HANDLER();
|
|
return;
|
|
pipe_wait:
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, false);
|
|
INT_POST_HANDLER();
|
|
}
|