mirror of
https://gitee.com/Vancouver2017/luban-lite-t3e-pro.git
synced 2025-12-14 18:38:55 +00:00
1334 lines
49 KiB
C
1334 lines
49 KiB
C
#include "usbh_core.h"
|
|
#include "usbh_hub.h"
|
|
#include "usb_ch32_usbfs_reg.h"
|
|
|
|
#if defined(CH581) || defined(CH582) || defined(CH583) || defined(CH571) || defined(CH572) || defined(CH573)
|
|
#undef USBFS_BASE
|
|
#undef USBFS_HOST
|
|
|
|
#ifndef USBFS_BASE
|
|
#pragma message "USB2 is used by default"
|
|
#define USBFS_BASE ((uint32_t)0x40008400u)
|
|
#endif
|
|
|
|
#define USBFS_HOST ((USB_FS_TypeDef *)USBFS_BASE)
|
|
|
|
#undef USBFS_UH_PRE_PID_EN
|
|
#undef USBFS_UH_SOF_EN
|
|
#define USBFS_UH_PRE_PID_EN 0x80 /*!< USB host PRE PID enable for low speed device via hub */
|
|
#define USBFS_UH_SOF_EN 0x40 /*!< USB host automatic SOF enable */
|
|
|
|
#undef USBFS_UH_R_AUTO_TOG
|
|
#undef USBFS_UH_R_TOG
|
|
#undef USBFS_UH_R_RES
|
|
#define USBFS_UH_R_AUTO_TOG 0x10 /*!< enable automatic toggle after successful transfer completion: 0=manual toggle, 1=automatic toggle */
|
|
#define USBFS_UH_R_TOG 0x80 /*!< expected data toggle flag of host receiving (IN): 0=DATA0, 1=DATA1 */
|
|
#define USBFS_UH_R_RES 0x04 /*!< prepared handshake response type for host receiving (IN): 0=ACK (ready), 1=no response, time out to device, for isochronous transactions */
|
|
|
|
#undef USBFS_UH_T_AUTO_TOG
|
|
#undef USBFS_UH_T_TOG
|
|
#undef USBFS_UH_T_RES
|
|
#define USBFS_UH_T_AUTO_TOG 0x10 /*!< enable automatic toggle after successful transfer completion: 0=manual toggle, 1=automatic toggle */
|
|
#define USBFS_UH_T_TOG 0x40 /*!< prepared data toggle flag of host transmittal (SETUP/OUT): 0=DATA0, 1=DATA1 */
|
|
#define USBFS_UH_T_RES 0x01 /*!< expected handshake response type for host transmittal (SETUP/OUT): 0=ACK (ready), 1=no response, time out from device, for isochronous transactions */
|
|
|
|
void USBH_IRQHandler(void) __attribute__((section(".highcode")));
|
|
#elif defined(CH32F203) || defined(CH32F205) || defined(CH32F207) || defined(CH32F208)
|
|
void USBH_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
|
|
#elif defined(CH32V203) || defined(CH32V303) || defined(CH32V305) || defined(CH32V307) || defined(CH32V208)
|
|
void USBH_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
|
|
#else
|
|
#error "Do not support"
|
|
#endif
|
|
|
|
#ifndef USBH_IRQHandler
|
|
#pragma message "Please make sure your platform interrupt name"
|
|
#define USBH_IRQHandler USB2_IRQHandler
|
|
#endif
|
|
|
|
#ifndef CONFIG_USBHOST_PIPE_NUM
|
|
#define CONFIG_USBHOST_PIPE_NUM 8
|
|
#endif
|
|
|
|
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;
|
|
|
|
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 ((USBFS_HOST->HOST_RX_CTRL & USBFS_UH_R_AUTO_TOG) != 0) {
|
|
/**
|
|
* USBFS_UH_R_TOG canot write
|
|
*/
|
|
if ((bit & USBFS_UH_R_TOG) != 0) {
|
|
/**
|
|
* bit contains USBFS_UH_R_TOG
|
|
*/
|
|
USBFS_HOST->HOST_RX_CTRL &= ~(USBFS_UH_R_AUTO_TOG);
|
|
USBFS_HOST->HOST_RX_CTRL |= (USBFS_UH_R_AUTO_TOG | bit);
|
|
} else {
|
|
USBFS_HOST->HOST_RX_CTRL |= bit;
|
|
}
|
|
} else {
|
|
USBFS_HOST->HOST_RX_CTRL |= bit;
|
|
}
|
|
}
|
|
|
|
static inline void SET_UH_TX_CTRL_BIT(uint8_t bit)
|
|
{
|
|
if ((USBFS_HOST->HOST_TX_CTRL & USBFS_UH_T_AUTO_TOG) != 0) {
|
|
/**
|
|
* USBFS_UH_T_TOG canot write
|
|
*/
|
|
if ((bit & USBFS_UH_T_TOG) != 0) {
|
|
/**
|
|
* bit contains USBFS_UH_T_TOG
|
|
*/
|
|
USBFS_HOST->HOST_TX_CTRL &= ~(USBFS_UH_T_AUTO_TOG);
|
|
USBFS_HOST->HOST_TX_CTRL |= (USBFS_UH_T_AUTO_TOG | bit);
|
|
} else {
|
|
USBFS_HOST->HOST_TX_CTRL |= bit;
|
|
}
|
|
} else {
|
|
USBFS_HOST->HOST_TX_CTRL |= bit;
|
|
}
|
|
}
|
|
|
|
static inline void CLEAR_UH_TX_CTRL_BIT(uint8_t bit)
|
|
{
|
|
if ((USBFS_HOST->HOST_TX_CTRL & USBFS_UH_T_AUTO_TOG) != 0) {
|
|
/**
|
|
* USBFS_UH_T_TOG canot write
|
|
*/
|
|
if ((bit & USBFS_UH_T_TOG) != 0) {
|
|
/**
|
|
* bit contains USBFS_UH_T_TOG
|
|
*/
|
|
USBFS_HOST->HOST_TX_CTRL &= ~(USBFS_UH_T_AUTO_TOG);
|
|
USBFS_HOST->HOST_TX_CTRL &= ~(bit);
|
|
USBFS_HOST->HOST_TX_CTRL |= USBFS_UH_T_AUTO_TOG;
|
|
} else {
|
|
USBFS_HOST->HOST_TX_CTRL &= ~(bit);
|
|
}
|
|
} else {
|
|
USBFS_HOST->HOST_TX_CTRL &= ~(bit);
|
|
}
|
|
}
|
|
|
|
static inline void CLEAR_UH_RX_CTRL_BIT(uint8_t bit)
|
|
{
|
|
if ((USBFS_HOST->HOST_RX_CTRL & USBFS_UH_R_AUTO_TOG) != 0) {
|
|
/**
|
|
* USBFS_UH_R_TOG can't write
|
|
*/
|
|
if ((bit & USBFS_UH_R_TOG) != 0) {
|
|
/**
|
|
* bit contains USBFS_UH_R_TOG
|
|
*/
|
|
USBFS_HOST->HOST_RX_CTRL &= ~(USBFS_UH_R_AUTO_TOG);
|
|
USBFS_HOST->HOST_RX_CTRL &= ~(bit);
|
|
USBFS_HOST->HOST_RX_CTRL |= USBFS_UH_R_AUTO_TOG;
|
|
} else {
|
|
USBFS_HOST->HOST_RX_CTRL &= ~(bit);
|
|
}
|
|
} else {
|
|
USBFS_HOST->HOST_RX_CTRL &= ~(bit);
|
|
}
|
|
}
|
|
|
|
static inline void SET_UH_RX_CTRL(uint8_t value)
|
|
{
|
|
USBFS_HOST->HOST_RX_CTRL = 0;
|
|
USBFS_HOST->HOST_RX_CTRL = value;
|
|
USB_LOG_DBG("USBFS_HOST->HOST_RX_CTRL:%02x\r\n", USBFS_HOST->HOST_RX_CTRL);
|
|
}
|
|
|
|
static inline void SET_UH_TX_CTRL(uint8_t value)
|
|
{
|
|
USBFS_HOST->HOST_TX_CTRL = 0;
|
|
USBFS_HOST->HOST_TX_CTRL = value;
|
|
USB_LOG_DBG("USBFS_HOST->HOST_TX_CTRL:%02x\r\n", USBFS_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) {
|
|
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;
|
|
}
|
|
|
|
USBFS_HOST->HOST_TX_DMA = (uint16_t)(uint32_t)data;
|
|
|
|
/*!< Record the data buffer address */
|
|
pipe->buffer = data;
|
|
|
|
if (len > pipe->ep_mps) {
|
|
len = pipe->ep_mps;
|
|
}
|
|
|
|
USBFS_HOST->HOST_TX_LEN = len;
|
|
USBFS_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) {
|
|
USBFS_HOST->HOST_TX_LEN = len;
|
|
USBFS_HOST->HOST_EP_PID = pid << 4 | (pipe->ep_addr & 0x0f);
|
|
return 0;
|
|
}
|
|
|
|
if ((uint32_t)data & 0x03) {
|
|
USB_LOG_INFO("OUT DMA address is not align \r\n");
|
|
return -3;
|
|
}
|
|
|
|
USBFS_HOST->HOST_TX_DMA = (uint16_t)(uint32_t)data;
|
|
|
|
/*!< Record the data buffer address */
|
|
pipe->buffer = data;
|
|
|
|
if (len > pipe->ep_mps) {
|
|
len = pipe->ep_mps;
|
|
}
|
|
|
|
USBFS_HOST->HOST_TX_LEN = len;
|
|
USBFS_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;
|
|
}
|
|
USBFS_HOST->HOST_RX_DMA = (uint16_t)(uint32_t)data;
|
|
pipe->buffer = data;
|
|
}
|
|
|
|
USBFS_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(USBFS_UH_T_AUTO_TOG);
|
|
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(USBFS_UH_R_AUTO_TOG | USBFS_UH_R_TOG);
|
|
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(USBFS_UH_R_AUTO_TOG | USBFS_UH_R_TOG);
|
|
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(USBFS_UH_T_AUTO_TOG | USBFS_UH_T_TOG);
|
|
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_addr & 0x80) {
|
|
/*!< IN */
|
|
g_chusb_hcd.current_token = USB_PID_IN;
|
|
if (pipe->data_pid == 1) {
|
|
SET_UH_RX_CTRL(USBFS_UH_R_AUTO_TOG | USBFS_UH_R_TOG);
|
|
} else {
|
|
SET_UH_RX_CTRL(USBFS_UH_R_AUTO_TOG);
|
|
}
|
|
} else {
|
|
/*!< OUT */
|
|
g_chusb_hcd.current_token = USB_PID_OUT;
|
|
if (pipe->data_pid == 1) {
|
|
SET_UH_TX_CTRL(USBFS_UH_T_AUTO_TOG | USBFS_UH_T_TOG);
|
|
} else {
|
|
SET_UH_TX_CTRL(USBFS_UH_T_AUTO_TOG);
|
|
}
|
|
}
|
|
|
|
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_addr & 0x80) {
|
|
/*!< IN */
|
|
g_chusb_hcd.current_token = USB_PID_IN;
|
|
if (pipe->data_pid == 1) {
|
|
SET_UH_RX_CTRL(USBFS_UH_R_AUTO_TOG | USBFS_UH_R_TOG);
|
|
} else {
|
|
SET_UH_RX_CTRL(USBFS_UH_R_AUTO_TOG);
|
|
}
|
|
} else {
|
|
/*!< OUT */
|
|
g_chusb_hcd.current_token = USB_PID_OUT;
|
|
if (pipe->data_pid == 1) {
|
|
SET_UH_TX_CTRL(USBFS_UH_T_AUTO_TOG | USBFS_UH_T_TOG);
|
|
} else {
|
|
SET_UH_TX_CTRL(USBFS_UH_T_AUTO_TOG);
|
|
}
|
|
}
|
|
|
|
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;
|
|
SET_UH_RX_CTRL(USBFS_UH_R_AUTO_TOG | USBFS_UH_R_RES);
|
|
} else {
|
|
/*!< OUT */
|
|
g_chusb_hcd.current_token = USB_PID_OUT;
|
|
SET_UH_TX_CTRL(USBFS_UH_T_AUTO_TOG | USBFS_UH_T_RES);
|
|
}
|
|
|
|
chusb_host_pipe_transfer(pipe, g_chusb_hcd.current_token, buffer, buflen);
|
|
}
|
|
|
|
static void chusbh_set_self_speed(uint8_t speed)
|
|
{
|
|
if (speed == USB_SPEED_HIGH) {
|
|
} else if (speed == USB_SPEED_FULL) {
|
|
USBFS_HOST->BASE_CTRL &= ~USBFS_UC_LOW_SPEED;
|
|
USBFS_HOST->HOST_CTRL &= ~USBFS_UH_LOW_SPEED;
|
|
USBFS_HOST->HOST_SETUP &= ~USBFS_UH_PRE_PID_EN;
|
|
} else {
|
|
USBFS_HOST->BASE_CTRL |= USBFS_UC_LOW_SPEED;
|
|
USBFS_HOST->HOST_CTRL |= USBFS_UH_LOW_SPEED;
|
|
USBFS_HOST->HOST_SETUP |= USBFS_UH_PRE_PID_EN;
|
|
}
|
|
}
|
|
|
|
static int usbh_reset_port(const uint8_t port)
|
|
{
|
|
/*!< Disable detect interrupt */
|
|
USBFS_HOST->INT_EN &= (~USBFS_UIE_DETECT);
|
|
USBFS_HOST->HOST_CTRL &= ~USBFS_UH_SOF_EN;
|
|
|
|
g_chusb_hcd.port_pe = 0;
|
|
/*!< Set dev add 0 */
|
|
USBFS_HOST->DEV_ADDR = (USBFS_HOST->DEV_ADDR & USBFS_UDA_GP_BIT) | (0x00 & USBFS_USB_ADDR_MASK);
|
|
chusbh_set_self_speed(USB_SPEED_FULL);
|
|
/*!< Close port */
|
|
USBFS_HOST->HOST_CTRL &= ~USBFS_UH_PORT_EN;
|
|
/*!< Start reset */
|
|
USBFS_HOST->HOST_CTRL |= USBFS_UH_BUS_RESET;
|
|
usb_osal_msleep(30);
|
|
/*!< Stop reset */
|
|
USBFS_HOST->HOST_CTRL &= ~USBFS_UH_BUS_RESET;
|
|
usb_osal_msleep(20);
|
|
|
|
if ((USBFS_HOST->HOST_CTRL & USBFS_UH_PORT_EN) == 0) {
|
|
volatile uint8_t speed = (USBFS_HOST->MIS_ST & USBFS_UMS_DM_LEVEL) ? 0 : 1;
|
|
if (speed == 0) {
|
|
/*!< Low speed */
|
|
USB_LOG_INFO("Dev USB_SPEED_LOW \r\n");
|
|
USBFS_HOST->HOST_CTRL |= USBFS_UH_LOW_SPEED;
|
|
g_chusb_hcd.dev_speed = USB_SPEED_LOW;
|
|
chusbh_set_self_speed(USB_SPEED_LOW);
|
|
} else {
|
|
/*!< Full speed */
|
|
USB_LOG_INFO("Dev USB_SPEED_FULL \r\n");
|
|
USBFS_HOST->HOST_CTRL &= ~USBFS_UH_LOW_SPEED;
|
|
g_chusb_hcd.dev_speed = USB_SPEED_FULL;
|
|
}
|
|
}
|
|
|
|
/*!< Enable HUB Port */
|
|
USBFS_HOST->HOST_CTRL |= USBFS_UH_PORT_EN;
|
|
// USBFS_HOST->HOST_SETUP |= USBFS_UH_SOF_EN;
|
|
USBFS_HOST->INT_EN |= USBFS_UIE_DETECT;
|
|
g_chusb_hcd.port_pe = 1;
|
|
return 0;
|
|
}
|
|
|
|
static uint8_t usbh_get_port_speed(const uint8_t port)
|
|
{
|
|
(void)port;
|
|
USBFS_HOST->HOST_SETUP |= USBFS_UH_SOF_EN;
|
|
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();
|
|
USBFS_HOST->BASE_CTRL = USBFS_UC_RESET_SIE | USBFS_UC_CLR_ALL;
|
|
static uint32_t wait_ct = 50000;
|
|
while (wait_ct--) {
|
|
}
|
|
USBFS_HOST->BASE_CTRL = 0;
|
|
|
|
USBFS_HOST->BASE_CTRL = USBFS_UC_HOST_MODE;
|
|
USBFS_HOST->HOST_CTRL = 0;
|
|
|
|
USBFS_HOST->DEV_ADDR = 0x00;
|
|
|
|
USBFS_HOST->HOST_EP_MOD = USBFS_UH_EP_TX_EN | USBFS_UH_EP_RX_EN;
|
|
|
|
USBFS_HOST->HOST_RX_CTRL = 0;
|
|
USBFS_HOST->HOST_TX_CTRL = 0;
|
|
|
|
USBFS_HOST->BASE_CTRL = USBFS_UC_HOST_MODE | USBFS_UC_INT_BUSY | USBFS_UC_DMA_EN;
|
|
// USBFS_HOST->HOST_SETUP = USBFS_UH_SOF_EN;
|
|
USBFS_HOST->INT_FG = 0xFF;
|
|
|
|
USBFS_HOST->INT_EN = USBFS_UIE_TRANSFER | USBFS_UIE_DETECT;
|
|
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 mult)
|
|
{
|
|
struct chusb_pipe *ppipe = (struct chusb_pipe *)pipe;
|
|
|
|
ppipe->dev_addr = dev_addr;
|
|
ppipe->ep_mps = ep_mps;
|
|
|
|
USBFS_HOST->DEV_ADDR = (USBFS_DEV_ADDR_OFFSET & USBFS_UDA_GP_BIT) | (dev_addr & USBFS_USB_ADDR_MASK);
|
|
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) {
|
|
} 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;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
// g_chusb_hcd.current_pipe_timeout = 0;
|
|
|
|
ret = urb->errorcode;
|
|
}
|
|
return ret;
|
|
errout_timeout:
|
|
pipe->waiter = false;
|
|
g_chusb_hcd.current_token = 0;
|
|
usbh_kill_urb(urb);
|
|
return ret;
|
|
}
|
|
|
|
int usbh_kill_urb(struct usbh_urb *urb)
|
|
{
|
|
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);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static int8_t chusb_outpipe_irq_handler(uint8_t res_state)
|
|
{
|
|
uint16_t current_tx_length = USBFS_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");
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
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");
|
|
} else {
|
|
USB_LOG_ERR("USB_PID_OUT STALL \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;
|
|
}
|
|
}
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
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;
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
return -3;
|
|
} else {
|
|
if (g_chusb_hcd.current_pipe->waiter == true) {
|
|
USB_LOG_DBG("Control 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 {
|
|
if (g_chusb_hcd.prv_set_zero == true) {
|
|
/**
|
|
* It is unlikely to run here,
|
|
* because the device can probably receive 0 length byte packets
|
|
*/
|
|
urb->errorcode = 0;
|
|
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");
|
|
urb->errorcode = -EAGAIN;
|
|
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("Your device seems unable to receive data\r\n");
|
|
urb->errorcode = -EBUSY;
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, false);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case USB_PID_ACK:
|
|
/**
|
|
* 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");
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
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;
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
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;
|
|
}
|
|
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
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");
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
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;
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
return -2;
|
|
case USB_PID_NAK:
|
|
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) {
|
|
urb->errorcode = -EAGAIN;
|
|
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;
|
|
}
|
|
} 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) {
|
|
/**
|
|
* Retry in
|
|
*/
|
|
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 {
|
|
urb->errorcode = -EIO;
|
|
USB_LOG_ERR("xferlen == 0//should get zero package\r\n");
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
return -1;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case USB_PID_ACK:
|
|
urb->errorcode = 0;
|
|
break;
|
|
case USB_PID_DATA0:
|
|
case USB_PID_DATA1:
|
|
if ((USBFS_HOST->INT_ST) & USBFS_UIS_TOG_OK) {
|
|
/*!< Data is OK */
|
|
rx_len = USBFS_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\r\n");
|
|
if (g_chusb_hcd.prv_get_zero == true) {
|
|
g_chusb_hcd.prv_get_zero = false;
|
|
}
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
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);
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
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");
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
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;
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
return -3;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
return 0;
|
|
}
|
|
|
|
void USBH_IRQHandler(void)
|
|
{
|
|
INT_PRE_HANDLER();
|
|
volatile uint8_t intflag = 0;
|
|
volatile uint8_t res = 0;
|
|
intflag = USBFS_HOST->INT_FG;
|
|
|
|
if (intflag & USBFS_UIF_TRANSFER) {
|
|
/*!< Check the equipment response status */
|
|
res = (USBFS_HOST->INT_ST) & USBFS_UIS_ENDP_MASK;
|
|
/*!< Stop this transmission after successful transmission */
|
|
USBFS_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:
|
|
USBFS_HOST->INT_FG = USBFS_UIF_TRANSFER;
|
|
break;
|
|
}
|
|
} else if (intflag & USBFS_UIF_DETECT) {
|
|
if (USBFS_HOST->MIS_ST & USBFS_UMS_DEV_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;
|
|
usbh_roothub_thread_wakeup(1);
|
|
} else {
|
|
USB_LOG_INFO("Dev remove \r\n");
|
|
/**
|
|
* Device remove
|
|
* Disable port and stop send sof
|
|
*/
|
|
USBFS_HOST->HOST_SETUP &= ~USBFS_UH_SOF_EN;
|
|
USBFS_HOST->HOST_CTRL &= ~USBFS_UH_PORT_EN;
|
|
#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);
|
|
}
|
|
USBFS_HOST->INT_FG = USBFS_UIF_DETECT;
|
|
} else {
|
|
USB_LOG_INFO("Unkonwn \r\n");
|
|
USBFS_HOST->INT_FG = intflag;
|
|
}
|
|
INT_POST_HANDLER();
|
|
return;
|
|
pipe_wait:
|
|
/**
|
|
* enerally, only errors can arrive here.
|
|
* After testing, most cases arrive here because of the problem of the DATA PID,
|
|
* but the transmission will be completed correctly next time.
|
|
*/
|
|
chusb_pipe_waitup(g_chusb_hcd.current_pipe, true);
|
|
INT_POST_HANDLER();
|
|
}
|