/**************************************************************************//** * * @copyright (C) 2020 Nuvoton Technology Corp. All rights reserved. * * SPDX-License-Identifier: Apache-2.0 * * Change Logs: * Date Author Notes * 2020-12-30 Wayne First version * ******************************************************************************/ #include #if defined(BSP_USING_USBH) #include #include #include "NuMicro.h" #include "usb.h" #include "usbh_lib.h" #define LOG_TAG "drv.usb.host" #define DBG_LVL DBG_INFO #include #if !defined(NU_USBHOST_HUB_POLLING_INTERVAL) #define NU_USBHOST_HUB_POLLING_INTERVAL (100) #endif #define NU_MAX_USBH_PORT 2 //2* USB2.0 port #define NU_MAX_USBH_PIPE 16 #define NU_USBH_THREAD_STACK_SIZE 2048 #define NU_MAX_USBH_HUB_PORT_DEV USB_HUB_PORT_NUM #define NU_USBHOST_HUB_POLLING_LOCK #if defined(NU_USBHOST_HUB_POLLING_LOCK) #define NU_USBHOST_MUTEX_INIT() { \ s_sUSBHDev.lock = rt_mutex_create("usbhost_lock", RT_IPC_FLAG_PRIO); \ RT_ASSERT(s_sUSBHDev.lock != RT_NULL); \ } #define NU_USBHOST_LOCK() { \ rt_err_t result = rt_mutex_take(s_sUSBHDev.lock, RT_WAITING_FOREVER); \ RT_ASSERT(result == RT_EOK); \ } #define NU_USBHOST_UNLOCK() { \ rt_err_t result = rt_mutex_release(s_sUSBHDev.lock); \ RT_ASSERT(result == RT_EOK); \ } #else #define NU_USBHOST_MUTEX_INIT() #define NU_USBHOST_LOCK() #define NU_USBHOST_UNLOCK() #endif /* Private typedef --------------------------------------------------------------*/ typedef struct nu_port_dev { rt_bool_t bRHParent; UDEV_T *pUDev; EP_INFO_T *apsEPInfo[NU_MAX_USBH_PIPE]; struct urequest asSetupReq[NU_MAX_USBH_PIPE]; uint32_t u32SentLength[NU_MAX_USBH_PIPE]; struct rt_completion utr_completion; int port_num; rt_bool_t bEnumDone; #if defined(BSP_USING_MMU) void *asPipePktBuf[NU_MAX_USBH_PIPE]; #endif } S_NU_PORT_DEV; typedef struct nu_port_ctrl { S_NU_PORT_DEV sRHPortDev; S_NU_PORT_DEV asHubPortDev[NU_MAX_USBH_HUB_PORT_DEV]; } S_NU_RH_PORT_CTRL; struct nu_usbh_dev { struct uhcd uhcd; E_SYS_IPRST rstidx; E_SYS_IPCLK clkidx; rt_thread_t polling_thread; rt_mutex_t lock; S_NU_RH_PORT_CTRL asPortCtrl[NU_MAX_USBH_PORT]; }; /* Private variables ------------------------------------------------------------*/ static struct nu_usbh_dev s_sUSBHDev = { .rstidx = USBHRST, .clkidx = USBHCKEN, }; static S_NU_RH_PORT_CTRL * GetRHPortControlFromPipe( upipe_t pipe) { uinst_t inst; int port; if (!pipe || !pipe->inst || !pipe->inst->parent_hub) return RT_NULL; if (pipe->inst->parent_hub->is_roothub) { //case: device ---> root hub inst = pipe->inst; port = inst->port; } else { //case: device ---> hub ---> root hub inst = pipe->inst->parent_hub->self; port = inst->port; } if (port > NU_MAX_USBH_PORT) { LOG_D("nu_open_pipe ERROR: port index over NU_MAX_USBH_PORT"); return RT_NULL; } return &s_sUSBHDev.asPortCtrl[port - 1];; } static S_NU_PORT_DEV * GetPortDevFromPipe( upipe_t pipe) { S_NU_RH_PORT_CTRL *psRHPortCtrl = GetRHPortControlFromPipe(pipe); int i; if (psRHPortCtrl == RT_NULL) return RT_NULL; if (pipe->inst->parent_hub->is_roothub) { //case: device ---> root hub return &psRHPortCtrl->sRHPortDev; } //case: device ---> hub ---> root hub for (i = 0 ; i < NU_MAX_USBH_HUB_PORT_DEV; i ++) { if (psRHPortCtrl->asHubPortDev[i].port_num == pipe->inst->port) break; } if (i >= NU_MAX_USBH_HUB_PORT_DEV) return RT_NULL; return &psRHPortCtrl->asHubPortDev[i]; } static rt_err_t nu_reset_port(rt_uint8_t port) { S_NU_RH_PORT_CTRL *psPortCtrl; if (port > NU_MAX_USBH_PORT) { LOG_D("%s ERROR: port index over NU_MAX_USBH_PORT", __func__); return -RT_EIO; } psPortCtrl = &s_sUSBHDev.asPortCtrl[port - 1]; if (psPortCtrl->sRHPortDev.pUDev == NULL) { LOG_D("%s ERROR: udev not found", __func__); return -RT_EIO; } usbh_reset_port(psPortCtrl->sRHPortDev.pUDev); return RT_EOK; } static EP_INFO_T *GetFreePipe( S_NU_RH_PORT_CTRL *psPortCtrl, S_NU_PORT_DEV *psPortDev, rt_uint8_t *pu8PipeIndex) { if (psPortCtrl != NULL) { int i; /* Find free Pipe */ for (i = 0; i < NU_MAX_USBH_PIPE; i ++) { if (psPortDev->apsEPInfo[i] == NULL) break; } if (i < NU_MAX_USBH_PIPE) { EP_INFO_T *psEPInfo = (EP_INFO_T *)rt_malloc_align(sizeof(EP_INFO_T), CACHE_LINE_SIZE); if (psEPInfo != RT_NULL) { #if defined(BSP_USING_MMU) psPortDev->apsEPInfo[i] = (EP_INFO_T *)((uint32_t)psEPInfo | NON_CACHE_MASK); #else psPortDev->apsEPInfo[i] = psEPInfo; #endif *pu8PipeIndex = i; return psPortDev->apsEPInfo[i]; } } } return RT_NULL; } static void FreePipe( S_NU_RH_PORT_CTRL *psPortCtrl, S_NU_PORT_DEV *psPortDev, rt_uint8_t u8PipeIndex) { if ((psPortCtrl != RT_NULL) && (u8PipeIndex < NU_MAX_USBH_PIPE) && (psPortDev->apsEPInfo[u8PipeIndex] != RT_NULL)) { EP_INFO_T *psEPInfo = psPortDev->apsEPInfo[u8PipeIndex]; #if defined(BSP_USING_MMU) psEPInfo = (EP_INFO_T *)((uint32_t)psEPInfo & ~NON_CACHE_MASK); #endif rt_free_align(psEPInfo); psPortDev->apsEPInfo[u8PipeIndex] = RT_NULL; } } static S_NU_PORT_DEV * AllocateNewUDev( S_NU_RH_PORT_CTRL *psRHPortCtrl) { if (psRHPortCtrl != RT_NULL) { int i; /* Find free Dev */ for (i = 0 ; i < NU_MAX_USBH_HUB_PORT_DEV; i ++) { if (psRHPortCtrl->asHubPortDev[i].pUDev == NULL) break; } if (i < NU_MAX_USBH_HUB_PORT_DEV) { psRHPortCtrl->asHubPortDev[i].pUDev = alloc_device(); if (psRHPortCtrl->asHubPortDev[i].pUDev == NULL) { return RT_NULL; } else { return &psRHPortCtrl->asHubPortDev[i]; } } } return RT_NULL; } static rt_err_t nu_open_pipe(upipe_t pipe) { S_NU_RH_PORT_CTRL *psPortCtrl; S_NU_PORT_DEV *psPortDev; psPortCtrl = GetRHPortControlFromPipe(pipe); if (psPortCtrl == RT_NULL) { LOG_D("%s ERROR: RHPort not found", __func__); goto exit_nu_open_pipe; } if (psPortCtrl->sRHPortDev.pUDev == NULL) { LOG_D("%s ERROR: udev not found", __func__); goto exit_nu_open_pipe; } psPortDev = GetPortDevFromPipe(pipe); if ((psPortDev == NULL) || (psPortDev->pUDev == NULL)) { //allocate new dev for hub device psPortDev = AllocateNewUDev(psPortCtrl); if (psPortDev == RT_NULL) { LOG_D("nu_open_pipe ERROR: udev allocate failed"); goto exit_nu_open_pipe; } if (pipe->inst->speed) { psPortDev->pUDev->speed = SPEED_FULL; } else { psPortDev->pUDev->speed = SPEED_HIGH; } psPortDev->pUDev->parent = NULL; psPortDev->pUDev->hc_driver = psPortCtrl->sRHPortDev.pUDev->hc_driver; psPortDev->port_num = pipe->inst->port; psPortDev->pUDev->port_num = pipe->inst->port; psPortDev->bEnumDone = FALSE; } //For ep0 control transfer if ((pipe->ep.bEndpointAddress & 0x7F) == 0) { pipe->pipe_index = 0; } else { int pksz; EP_INFO_T *psEPInfo = GetFreePipe(psPortCtrl, psPortDev, &pipe->pipe_index); if (psEPInfo == RT_NULL) { LOG_D("%s ERROR: get free pipe failed", __func__); goto exit_nu_open_pipe; } psEPInfo->bEndpointAddress = pipe->ep.bEndpointAddress; psEPInfo->bmAttributes = pipe->ep.bmAttributes; pksz = pipe->ep.wMaxPacketSize; pksz = (pksz & 0x07ff) * (1 + ((pksz >> 11) & 3)); psEPInfo->wMaxPacketSize = pksz; psEPInfo->bInterval = pipe->ep.bInterval; psEPInfo->hw_pipe = NULL; psEPInfo->bToggle = 0; } #if defined(BSP_USING_MMU) if (!psPortDev->asPipePktBuf[pipe->pipe_index]) { void *paddr = rt_malloc_align(512ul, CACHE_LINE_SIZE); RT_ASSERT(paddr != RT_NULL); psPortDev->asPipePktBuf[pipe->pipe_index] = (void *)((uint32_t)paddr | NON_CACHE_MASK); } #endif return RT_EOK; exit_nu_open_pipe: return -RT_ERROR; } static rt_err_t nu_close_pipe(upipe_t pipe) { S_NU_RH_PORT_CTRL *psPortCtrl; S_NU_PORT_DEV *psPortDev; psPortCtrl = GetRHPortControlFromPipe(pipe); if (psPortCtrl == RT_NULL) { return -RT_EIO; } psPortDev = GetPortDevFromPipe(pipe); //For ep0 control transfer if ((pipe->ep.bEndpointAddress & 0x7F) == 0) { if ((psPortDev) && (psPortDev->bRHParent == FALSE) && (psPortDev->bEnumDone == TRUE)) { if (psPortDev->pUDev) { int i; for (i = 0; i < NU_MAX_USBH_PIPE; i++) { if (psPortDev->apsEPInfo[i] != NULL) { usbh_quit_xfer(psPortDev->pUDev, psPortDev->apsEPInfo[i]); } } free_device(psPortDev->pUDev); psPortDev->pUDev = NULL; } } } if (psPortDev != NULL) { #if defined(BSP_USING_MMU) if (psPortDev->asPipePktBuf[pipe->pipe_index]) { void *paddr = psPortDev->asPipePktBuf[pipe->pipe_index]; paddr = (void *)((uint32_t)paddr & ~NON_CACHE_MASK); rt_free_align(paddr); psPortDev->asPipePktBuf[pipe->pipe_index] = RT_NULL; } #endif FreePipe(psPortCtrl, psPortDev, pipe->pipe_index); } return RT_EOK; } static int nu_ctrl_xfer( S_NU_PORT_DEV *psPortDev, struct urequest *psSetup, void *buffer, int timeouts) { uint32_t xfer_len = 0; int ret; ret = usbh_ctrl_xfer(psPortDev->pUDev, psSetup->request_type, psSetup->bRequest, psSetup->wValue, psSetup->wIndex, psSetup->wLength, buffer, &xfer_len, timeouts * 10); if (ret < 0) { LOG_D("nu_ctrl_xfer ERROR: xfer failed %d", ret); return ret; } if (xfer_len != psSetup->wLength) { LOG_D("nu_ctrl_xfer ERROR: xfer length %d %d\n", psSetup->wLength, xfer_len); } if ((psSetup->bRequest == USB_REQ_SET_ADDRESS) && ((psSetup->request_type & 0x60) == REQ_TYPE_STD_DEV)) psPortDev->pUDev->dev_num = psSetup->wValue; if ((psSetup->bRequest == USB_REQ_SET_CONFIGURATION) && ((psSetup->request_type & 0x60) == REQ_TYPE_STD_DEV)) { psPortDev->pUDev->cur_conf = psSetup->wValue; psPortDev->bEnumDone = TRUE; } return xfer_len; } static int nu_bulk_xfer( S_NU_PORT_DEV *psPortDev, UTR_T *psUTR, int timeouts) { #define TIMEOUT_RETRY 3 int retry = TIMEOUT_RETRY; int ret = usbh_bulk_xfer(psUTR); if (ret < 0) { rt_kprintf("usbh_bulk_xfer %x\n", ret); return ret; } while (retry > 0) { if (rt_completion_wait(&(psPortDev->utr_completion), timeouts) != 0) { rt_uint32_t level; rt_kprintf("Request %d Timeout in %d ms!!\n", psUTR->data_len, timeouts); rt_completion_init(&(psPortDev->utr_completion)); rt_thread_mdelay(1); // Workaround: To fix timeout case, this way is traveling qh's linking-list again. level = rt_hw_interrupt_disable(); extern void scan_asynchronous_list(); extern void iaad_remove_qh(); scan_asynchronous_list(); iaad_remove_qh(); rt_hw_interrupt_enable(level); } else break; retry--; } return (retry > 0) ? 0 : -1; } static int nu_int_xfer( upipe_t pipe, S_NU_PORT_DEV *psPortDev, UTR_T *psUTR, int timeouts) { int ret; while (1) { ret = usbh_int_xfer(psUTR); if (ret < 0) return ret; if (rt_completion_wait(&(psPortDev->utr_completion), timeouts) != 0) { LOG_D("Request %08x Timeout in %d ms\n", psUTR, timeouts); usbh_quit_utr(psUTR); rt_completion_init(&(psPortDev->utr_completion)); rt_thread_mdelay(1); } else { LOG_D("Transferring done %08x\n", psUTR); usbh_quit_utr(psUTR); break; } } return 0; } static void xfer_done_cb(UTR_T *psUTR) { S_NU_PORT_DEV *psPortDev = (S_NU_PORT_DEV *)psUTR->context; //transfer done, signal utr_completion rt_completion_done(&(psPortDev->utr_completion)); } static int nu_pipe_xfer(upipe_t pipe, rt_uint8_t token, void *buffer, int nbytes, int timeouts) { S_NU_RH_PORT_CTRL *psPortCtrl; S_NU_PORT_DEV *psPortDev; UTR_T *psUTR = NULL; int i32XferLen = -1; void *buffer_nonch = buffer; NU_USBHOST_LOCK(); psPortCtrl = GetRHPortControlFromPipe(pipe); if (psPortCtrl == RT_NULL) { goto exit_nu_pipe_xfer; } psPortDev = GetPortDevFromPipe(pipe); if (psPortDev->pUDev == NULL) { LOG_D("nu_pipe_xfer ERROR: udev not found"); goto exit_nu_pipe_xfer; } #if defined(BSP_USING_MMU) if (buffer_nonch && nbytes) { buffer_nonch = psPortDev->asPipePktBuf[pipe->pipe_index]; if ((pipe->ep.bEndpointAddress & USB_DIR_MASK) == USB_DIR_OUT) { rt_memcpy(buffer_nonch, buffer, nbytes); } } #endif //ctrl xfer if (pipe->ep.bmAttributes == USB_EP_ATTR_CONTROL) { int ret; if (token == USBH_PID_SETUP) { struct urequest *psSetup = (struct urequest *)buffer_nonch; RT_ASSERT(buffer_nonch != RT_NULL); psPortCtrl->asHubPortDev->u32SentLength[pipe->pipe_index] = 0; /* Read data from USB device. */ if (psSetup->request_type & USB_REQ_TYPE_DIR_IN) { //Store setup request rt_memcpy(&psPortCtrl->asHubPortDev->asSetupReq[pipe->pipe_index], psSetup, sizeof(struct urequest)); } else { /* Write data to USB device. */ //Trigger USBHostLib Ctrl_Xfer ret = nu_ctrl_xfer(psPortDev, psSetup, NULL, timeouts); if (ret != psSetup->wLength) goto exit_nu_pipe_xfer; } } else { //token == USBH_PID_DATA if (buffer_nonch && ((pipe->ep.bEndpointAddress & USB_DIR_MASK) == USB_DIR_IN)) { struct urequest *psSetup = &psPortCtrl->asHubPortDev->asSetupReq[pipe->pipe_index]; /* Read data from USB device. */ //Trigger USBHostLib Ctril_Xfer /* * Workaround: HCD driver can readback all bytes of setup.wLength, but not support single packet transferring. */ if (psPortCtrl->asHubPortDev->u32SentLength[pipe->pipe_index] == 0) { ret = nu_ctrl_xfer(psPortDev, psSetup, buffer_nonch, timeouts); psPortCtrl->asHubPortDev->u32SentLength[pipe->pipe_index] = ret; if (ret > 0) { rt_memcpy(buffer, buffer_nonch, ret); } } else { if (psPortCtrl->asHubPortDev->u32SentLength[pipe->pipe_index] < nbytes) { ret = 0; } else { psPortCtrl->asHubPortDev->u32SentLength[pipe->pipe_index] -= nbytes; ret = nbytes; } } if (ret <= 0) goto exit_nu_pipe_xfer; } else { LOG_D("%d == USBH_PID_DATA, nil buf-%d", token, nbytes); } } //else i32XferLen = nbytes; goto exit_nu_pipe_xfer2; } // if ( pipe->ep.bmAttributes == USB_EP_ATTR_CONTROL ) else { psUTR = alloc_utr(psPortDev->pUDev); if (!psUTR) { LOG_D("nu_pipe_xfer ERROR: unable alloc UTR"); goto exit_nu_pipe_xfer; } psUTR->ep = psPortDev->apsEPInfo[pipe->pipe_index]; psUTR->buff = buffer_nonch; psUTR->data_len = nbytes; psUTR->xfer_len = 0; psUTR->func = xfer_done_cb; psUTR->context = psPortDev; psUTR->bIsTransferDone = 0; psUTR->status = 0; //others xfer rt_completion_init(&(psPortDev->utr_completion)); if (pipe->ep.bmAttributes == USB_EP_ATTR_BULK) { if (nu_bulk_xfer(psPortDev, psUTR, timeouts) < 0) { LOG_D("nu_pipe_xfer ERROR: bulk transfer failed"); goto failreport_nu_pipe_xfer; } } else if (pipe->ep.bmAttributes == USB_EP_ATTR_INT) { if (nu_int_xfer(pipe, psPortDev, psUTR, timeouts) < 0) { LOG_D("nu_pipe_xfer ERROR: int transfer failed"); //goto exit_nu_pipe_xfer; } else { i32XferLen = nbytes; } goto exit_nu_pipe_xfer; } else if (pipe->ep.bmAttributes == USB_EP_ATTR_ISOC) { //TODO: ISO transfer LOG_D("nu_pipe_xfer ERROR: isoc transfer not support"); goto exit_nu_pipe_xfer; } } //else failreport_nu_pipe_xfer: if (psUTR->bIsTransferDone == 0) { //Timeout LOG_D("nu_pipe_xfer ERROR: timeout"); pipe->status = UPIPE_STATUS_ERROR; usbh_quit_utr(psUTR); } else { // Transfer Done. Get status if (psUTR->status == 0) { pipe->status = UPIPE_STATUS_OK; } else if (psUTR->status == USBH_ERR_STALL) { pipe->status = UPIPE_STATUS_STALL; } else { pipe->status = UPIPE_STATUS_ERROR; } } i32XferLen = psUTR->xfer_len; exit_nu_pipe_xfer: //Call callback if (pipe->callback != RT_NULL) { pipe->callback(pipe); } if (psUTR) free_utr(psUTR); #if defined(BSP_USING_MMU) if ((nbytes) && (buffer_nonch != buffer)) { if ((pipe->ep.bEndpointAddress & USB_DIR_MASK) == USB_DIR_IN) { rt_memcpy(buffer, buffer_nonch, nbytes); } } #endif exit_nu_pipe_xfer2: NU_USBHOST_UNLOCK(); return i32XferLen; } /* Polling USB root hub status task */ static void nu_usbh_rh_thread_entry(void *parameter) { while (1) { NU_USBHOST_LOCK(); usbh_polling_root_hubs(); NU_USBHOST_UNLOCK(); rt_thread_mdelay(NU_USBHOST_HUB_POLLING_INTERVAL); } } static void nu_hcd_connect_callback( struct udev_t *udev, int param) { int i; int port_index; S_NU_RH_PORT_CTRL *psPortCtrl; /* Igrone event if role is USBD*/ if (nu_sys_usb0_role() == USB0_ID_HOST) { SYS_UnlockReg(); outpw(REG_SYS_MISCFCR, (inpw(REG_SYS_MISCFCR) | (1 << 11))); /* Set USRHDSEN as 1; USB host/device role selection decided by USBID (SYS_PWRON[16]) */ outpw(REG_SYS_PWRON, (inpw(REG_SYS_PWRON) | (1 << 16))); /* Set USB port 0 used for Host */ SYS_LockReg(); } for (i = 0; i < NU_MAX_USBH_PORT; i++) { psPortCtrl = &s_sUSBHDev.asPortCtrl[i]; if (psPortCtrl->sRHPortDev.pUDev == NULL) break; } if (i >= NU_MAX_USBH_PORT) { LOG_D("ERROR: port connect slot is full"); return; } port_index = i + 1; psPortCtrl->sRHPortDev.pUDev = udev; psPortCtrl->sRHPortDev.bRHParent = TRUE; LOG_D("usb connected"); if (udev->speed == SPEED_HIGH) rt_usbh_root_hub_connect_handler(&s_sUSBHDev.uhcd, port_index, RT_TRUE); else rt_usbh_root_hub_connect_handler(&s_sUSBHDev.uhcd, port_index, RT_FALSE); } static void nu_hcd_disconnect_callback( struct udev_t *udev, int param) { int i; int port_index; S_NU_RH_PORT_CTRL *psPortCtrl; for (i = 0; i < NU_MAX_USBH_PORT; i++) { psPortCtrl = &s_sUSBHDev.asPortCtrl[i]; if (psPortCtrl->sRHPortDev.pUDev == udev) break; } if (i >= NU_MAX_USBH_PORT) { LOG_D("ERROR: udev not found"); return; } port_index = i + 1; for (i = 0; i < NU_MAX_USBH_PIPE; i++) { if (psPortCtrl->sRHPortDev.apsEPInfo[i] != NULL) { usbh_quit_xfer(psPortCtrl->sRHPortDev.pUDev, psPortCtrl->sRHPortDev.apsEPInfo[i]); } } psPortCtrl->sRHPortDev.pUDev = NULL; LOG_D("usb disconnect"); rt_usbh_root_hub_disconnect_handler(&s_sUSBHDev.uhcd, port_index); if (nu_sys_usb0_role() != USB0_ID_HOST) { SYS_UnlockReg(); outpw(REG_SYS_MISCFCR, (inpw(REG_SYS_MISCFCR) & (~(1 << 11)))); outpw(REG_SYS_PWRON, (inpw(REG_SYS_PWRON) & (~(1 << 16)))); SYS_LockReg(); } } /* USB host operations -----------------------------------------------------------*/ static struct uhcd_ops nu_uhcd_ops = { nu_reset_port, nu_pipe_xfer, nu_open_pipe, nu_close_pipe, }; static rt_err_t nu_hcd_init(rt_device_t device) { struct nu_usbh_dev *pNuUSBHDev = (struct nu_usbh_dev *)device; usbh_core_init(); //install connect/disconnect callback usbh_install_conn_callback(nu_hcd_connect_callback, nu_hcd_disconnect_callback); //create thread for polling usbh port status /* create usb hub thread */ pNuUSBHDev->polling_thread = rt_thread_create("usbh_drv", nu_usbh_rh_thread_entry, RT_NULL, NU_USBH_THREAD_STACK_SIZE, 8, 20); RT_ASSERT(pNuUSBHDev->polling_thread != RT_NULL); /* startup usb host thread */ rt_thread_startup(pNuUSBHDev->polling_thread); return RT_EOK; } /* global function for USB host library -----------------------------*/ uint32_t usbh_get_ticks(void) { return rt_tick_get(); } void usbh_delay_ms(int msec) { rt_thread_mdelay(msec); } uint32_t usbh_tick_from_millisecond(uint32_t msec) { return rt_tick_from_millisecond(msec); } #if defined(RT_USING_PM) /* device pm suspend() entry. */ static int usbhost_pm_suspend(const struct rt_device *device, rt_uint8_t mode) { rt_err_t result; struct nu_usbh_dev *pNuUSBHDev = (struct nu_usbh_dev *)device; RT_ASSERT(pNuUSBHDev != RT_NULL); switch (mode) { case PM_SLEEP_MODE_LIGHT: case PM_SLEEP_MODE_DEEP: RT_SCHED_CTX(pNuUSBHDev->polling_thread).stat = RT_THREAD_READY; result = rt_thread_suspend(pNuUSBHDev->polling_thread); RT_ASSERT(result == RT_EOK); break; default: break; } return (int)RT_EOK; } /* device pm resume() entry. */ static void usbhost_pm_resume(const struct rt_device *device, rt_uint8_t mode) { rt_err_t result; struct nu_usbh_dev *pNuUSBHDev = (struct nu_usbh_dev *)device; RT_ASSERT(pNuUSBHDev != RT_NULL); switch (mode) { case PM_SLEEP_MODE_LIGHT: case PM_SLEEP_MODE_DEEP: result = rt_thread_resume(pNuUSBHDev->polling_thread); RT_ASSERT(result == RT_EOK); break; default: break; } } static struct rt_device_pm_ops device_pm_ops = { .suspend = usbhost_pm_suspend, .resume = usbhost_pm_resume, .frequency_change = RT_NULL }; #endif int nu_usbh_register(void) { rt_err_t res; uhcd_t psUHCD; psUHCD = (uhcd_t)&s_sUSBHDev.uhcd; psUHCD->parent.type = RT_Device_Class_USBHost; psUHCD->parent.init = nu_hcd_init; psUHCD->parent.user_data = &s_sUSBHDev; psUHCD->ops = &nu_uhcd_ops; psUHCD->num_ports = NU_MAX_USBH_PORT; NU_USBHOST_MUTEX_INIT(); res = rt_device_register(&psUHCD->parent, "usbh", RT_DEVICE_FLAG_DEACTIVATE); RT_ASSERT(res == RT_EOK); nu_sys_ipclk_enable(s_sUSBHDev.clkidx); nu_sys_ip_reset(s_sUSBHDev.rstidx); /*initialize the usb host function */ res = rt_usb_host_init("usbh"); RT_ASSERT(res == RT_EOK); #if defined(RT_USING_PM) rt_pm_device_register(&psUHCD->parent, &device_pm_ops); #endif return 0; } INIT_APP_EXPORT(nu_usbh_register); #endif