345 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			345 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|  | #include "usb_config.h"
 | |||
|  | #include "usb/scsi.h"
 | |||
|  | #include "irq.h"
 | |||
|  | #include "init.h"
 | |||
|  | #include "gpio.h"
 | |||
|  | #include "timer.h"
 | |||
|  | #include "app_config.h"
 | |||
|  | #include "lbuf.h"
 | |||
|  | 
 | |||
|  | #ifdef CONFIG_ADAPTER_ENABLE
 | |||
|  | #include "adapter_usb_hid.h"
 | |||
|  | #endif//CONFIG_ADAPTER_ENABLE
 | |||
|  | 
 | |||
|  | #define LOG_TAG_CONST       USB
 | |||
|  | #define LOG_TAG             "[USB]"
 | |||
|  | #define LOG_ERROR_ENABLE
 | |||
|  | #define LOG_DEBUG_ENABLE
 | |||
|  | #define LOG_INFO_ENABLE
 | |||
|  | /* #define LOG_DUMP_ENABLE */ | |||
|  | #define LOG_CLI_ENABLE
 | |||
|  | #include "debug.h"
 | |||
|  | 
 | |||
|  | #define     SET_INTERRUPT   ___interrupt
 | |||
|  | 
 | |||
|  | 
 | |||
|  | #define     MAX_EP_TX   5
 | |||
|  | #define     MAX_EP_RX   5
 | |||
|  | 
 | |||
|  | static usb_interrupt usb_interrupt_tx[USB_MAX_HW_NUM][MAX_EP_TX];// SEC(.usb_g_bss);
 | |||
|  | static usb_interrupt usb_interrupt_rx[USB_MAX_HW_NUM][MAX_EP_RX];// SEC(.usb_h_bss);
 | |||
|  | 
 | |||
|  | static u8 ep0_dma_buffer[EP0_SETUP_LEN]     __attribute__((aligned(4))) SEC(.usb_ep0)    ; | |||
|  | 
 | |||
|  | #if TCFG_USB_SLAVE_MSD_ENABLE
 | |||
|  | #define     MSD_DMA_SIZE (64*2)
 | |||
|  | #else
 | |||
|  | #define     MSD_DMA_SIZE 0
 | |||
|  | #endif
 | |||
|  | 
 | |||
|  | #if TCFG_USB_SLAVE_HID_ENABLE
 | |||
|  | #define     HID_DMA_SIZE    64
 | |||
|  | #else
 | |||
|  | #define     HID_DMA_SIZE    0
 | |||
|  | #endif
 | |||
|  | 
 | |||
|  | #if TCFG_USB_SLAVE_AUDIO_ENABLE
 | |||
|  | #define     AUDIO_DMA_SIZE  256+192
 | |||
|  | #else
 | |||
|  | #define     AUDIO_DMA_SIZE  0
 | |||
|  | #endif
 | |||
|  | 
 | |||
|  | #if TCFG_USB_SLAVE_CDC_ENABLE
 | |||
|  | #if CDC_INTR_EP_ENABLE
 | |||
|  | #define     CDC_DMA_SIZE    (64 + 64 + 64*2)
 | |||
|  | #else
 | |||
|  | #define     CDC_DMA_SIZE    (64 + 64*2)
 | |||
|  | #endif
 | |||
|  | #else
 | |||
|  | #define     CDC_DMA_SIZE    0
 | |||
|  | #endif
 | |||
|  | 
 | |||
|  | struct usb_config_var_t { | |||
|  |     u8 usb_setup_buffer[USB_SETUP_SIZE]; | |||
|  |     struct usb_ep_addr_t usb_ep_addr; | |||
|  |     struct usb_setup_t usb_setup; | |||
|  | }; | |||
|  | static struct usb_config_var_t *usb_config_var = {NULL}; | |||
|  | 
 | |||
|  | #if USB_MALLOC_ENABLE
 | |||
|  | #else
 | |||
|  | static struct usb_config_var_t _usb_config_var SEC(.usb_config_var); | |||
|  | #endif
 | |||
|  | 
 | |||
|  | 
 | |||
|  | #define USB_DMA_BUF_ALIGN	(8)
 | |||
|  | #ifndef USB_DMA_BUF_MAX_SIZE
 | |||
|  | #define USB_DMA_BUF_MAX_SIZE (HID_DMA_SIZE +USB_DMA_BUF_ALIGN+ AUDIO_DMA_SIZE +USB_DMA_BUF_ALIGN+ MSD_DMA_SIZE + USB_DMA_BUF_ALIGN + 100)
 | |||
|  | #endif//USB_DMA_BUF_MAX_SIZE
 | |||
|  | 
 | |||
|  | static u8 usb_dma_buf[USB_DMA_BUF_MAX_SIZE] SEC(.usb_msd_dma) __attribute__((aligned(8))); | |||
|  | static u8 cdc_dma_buffer[CDC_DMA_SIZE]     __attribute__((aligned(8))) SEC(.usb_cdc_dma); | |||
|  | struct lbuff_head *usb_dma_lbuf = NULL; | |||
|  | void usb_memory_init() | |||
|  | { | |||
|  |     usb_dma_lbuf = lbuf_init(usb_dma_buf, sizeof(usb_dma_buf), USB_DMA_BUF_ALIGN, 0); | |||
|  |     log_info("%s() total dma size %x @%x", __func__, sizeof(usb_dma_buf), usb_dma_buf); | |||
|  | } | |||
|  | 
 | |||
|  | __attribute__((always_inline_when_const_args)) | |||
|  | void *usb_alloc_ep_dmabuffer(const usb_dev usb_id, u32 ep, u32 dma_size) | |||
|  | { | |||
|  |     u8 *ep_buffer = NULL; | |||
|  |     u32 _ep = ep & 0xf; | |||
|  |     if (ep & USB_DIR_IN) { | |||
|  |         switch (_ep) { | |||
|  |         case 0: | |||
|  |             ep_buffer = ep0_dma_buffer; | |||
|  |             break; | |||
|  |         case CDC_DATA_EP_IN: | |||
|  |             ep_buffer = cdc_dma_buffer + MAXP_SIZE_CDC_BULKIN * 2; | |||
|  |             break; | |||
|  |         default : | |||
|  |             ep_buffer = lbuf_alloc(usb_dma_lbuf, dma_size); | |||
|  |             break; | |||
|  |         } | |||
|  |     } else { | |||
|  |         switch (_ep) { | |||
|  |         case 0: | |||
|  |             ep_buffer = ep0_dma_buffer; | |||
|  |             break; | |||
|  |         case CDC_DATA_EP_OUT: | |||
|  |             ep_buffer = cdc_dma_buffer; | |||
|  |             break; | |||
|  |         default : | |||
|  |             ep_buffer = lbuf_alloc(usb_dma_lbuf, dma_size); | |||
|  |             break; | |||
|  |         } | |||
|  |     } | |||
|  |     ASSERT(ep_buffer, "usb_alloc_ep_dmabuffer ep_buffer = NULL!!!, _ep = %x, dma_size = %d\n", ep, dma_size); | |||
|  | 
 | |||
|  |     log_info("ep_buffer = %x, ep = %x, dma_size = %d\n", ep_buffer, ep, dma_size); | |||
|  | 
 | |||
|  |     return ep_buffer; | |||
|  | } | |||
|  | 
 | |||
|  | static void usb_resume_sign(void *priv) | |||
|  | { | |||
|  |     usb_dev usb_id = usb_device2id(priv); | |||
|  |     u32 reg = usb_read_power(usb_id); | |||
|  |     usb_write_power(usb_id, reg | BIT(2));//send resume
 | |||
|  |     os_time_dly(2);//10ms~20ms
 | |||
|  |     usb_write_power(usb_id, reg);//clean resume
 | |||
|  | } | |||
|  | 
 | |||
|  | void usb_remote_wakeup(const usb_dev usb_id) | |||
|  | { | |||
|  |     struct usb_device_t *usb_device = usb_id2device(usb_id); | |||
|  |     if (usb_device->bRemoteWakup) { | |||
|  |         sys_timeout_add(usb_device, usb_resume_sign, 1); | |||
|  |     } | |||
|  | } | |||
|  | void usb_phy_resume(const usb_dev usb_id) | |||
|  | { | |||
|  |     usb_iomode(0); | |||
|  | 
 | |||
|  |     struct usb_device_t *usb_device = usb_id2device(usb_id); | |||
|  |     usb_write_faddr(usb_id, usb_device->baddr); | |||
|  | 
 | |||
|  |     if (usb_device->baddr == 0) { | |||
|  |         usb_device->bDeviceStates = USB_DEFAULT; | |||
|  |     } else { | |||
|  |         usb_device->bDeviceStates = USB_CONFIGURED; | |||
|  |     } | |||
|  | 
 | |||
|  |     usb_otg_resume(usb_id); | |||
|  | } | |||
|  | 
 | |||
|  | void usb_phy_suspend(const usb_dev usb_id) | |||
|  | { | |||
|  |     gpio_set_pull_up(IO_PORT_DP, 1); | |||
|  |     gpio_set_pull_down(IO_PORT_DP, 0); | |||
|  |     gpio_set_direction(IO_PORT_DP, 1); | |||
|  | 
 | |||
|  |     usb_iomode(1); | |||
|  |     /* musb_read_usb(0, MUSB_INTRUSB); */ | |||
|  | 
 | |||
|  |     usb_otg_suspend(usb_id, OTG_KEEP_STATE); | |||
|  | } | |||
|  | void usb_isr(const usb_dev usb_id) | |||
|  | { | |||
|  |     u32 intr_usb, intr_usbe; | |||
|  |     u32 intr_tx, intr_txe; | |||
|  |     u32 intr_rx, intr_rxe; | |||
|  | 
 | |||
|  |     __asm__ volatile("ssync"); | |||
|  |     usb_read_intr(usb_id, &intr_usb, &intr_tx, &intr_rx); | |||
|  |     usb_read_intre(usb_id, &intr_usbe, &intr_txe, &intr_rxe); | |||
|  |     struct usb_device_t *usb_device = usb_id2device(usb_id); | |||
|  | 
 | |||
|  |     intr_usb &= intr_usbe; | |||
|  |     intr_tx &= intr_txe; | |||
|  |     intr_rx &= intr_rxe; | |||
|  | 
 | |||
|  |     if (intr_usb & INTRUSB_SUSPEND) { | |||
|  |         log_error("usb suspend"); | |||
|  | #if USB_SUSPEND_RESUME
 | |||
|  |         usb_phy_suspend(usb_id); | |||
|  | #endif
 | |||
|  |     } | |||
|  |     if (intr_usb & INTRUSB_RESET_BABBLE) { | |||
|  |         log_error("usb reset"); | |||
|  |         usb_reset_interface(usb_device); | |||
|  | 
 | |||
|  | #if USB_SUSPEND_RESUME
 | |||
|  |         u32 reg = usb_read_power(usb_id); | |||
|  |         usb_write_power(usb_id, (reg | INTRUSB_SUSPEND | INTRUSB_RESUME));//enable suspend resume
 | |||
|  | #endif
 | |||
|  |     } | |||
|  | 
 | |||
|  |     if (intr_usb & INTRUSB_RESUME) { | |||
|  |         log_error("usb resume"); | |||
|  | #if USB_SUSPEND_RESUME
 | |||
|  |         usb_phy_resume(usb_id); | |||
|  | #endif
 | |||
|  |     } | |||
|  | 
 | |||
|  |     if (intr_tx & BIT(0)) { | |||
|  |         if (usb_interrupt_rx[usb_id][0]) { | |||
|  |             usb_interrupt_rx[usb_id][0](usb_device, 0); | |||
|  |         } else { | |||
|  |             usb_control_transfer(usb_device); | |||
|  |         } | |||
|  |     } | |||
|  | 
 | |||
|  |     for (int i = 1; i < MAX_EP_TX; i++) { | |||
|  |         if (intr_tx & BIT(i)) { | |||
|  |             if (usb_interrupt_tx[usb_id][i]) { | |||
|  |                 usb_interrupt_tx[usb_id][i](usb_device, i); | |||
|  |             } | |||
|  |         } | |||
|  |     } | |||
|  | 
 | |||
|  |     for (int i = 1; i < MAX_EP_RX; i++) { | |||
|  |         if (intr_rx & BIT(i)) { | |||
|  |             if (usb_interrupt_rx[usb_id][i]) { | |||
|  |                 usb_interrupt_rx[usb_id][i](usb_device, i); | |||
|  |             } | |||
|  |         } | |||
|  |     } | |||
|  |     __asm__ volatile("csync"); | |||
|  | } | |||
|  | 
 | |||
|  | void usb_sof_isr(const usb_dev usb_id) | |||
|  | { | |||
|  |     usb_sof_clr_pnd(usb_id); | |||
|  |     static u32 sof_count = 0; | |||
|  |     if ((sof_count++ % 1000) == 0) { | |||
|  |         log_d("sof 1s isr frame:%d", usb_read_sofframe(usb_id)); | |||
|  |     } | |||
|  | } | |||
|  | void usb_suspend_check(void *p) | |||
|  | { | |||
|  |     usb_dev usb_id = (usb_dev)p; | |||
|  | 
 | |||
|  |     static u16 sof_frame = 0; | |||
|  |     u16 frame = usb_read_sofframe(usb_id);// sof frame 不更新,则usb进入断开或者suspend状态
 | |||
|  |     if (frame == sof_frame) { | |||
|  |         usb_phy_suspend(usb_id); | |||
|  |     } | |||
|  |     sof_frame = frame; | |||
|  | } | |||
|  | 
 | |||
|  | SET_INTERRUPT | |||
|  | void usb0_g_isr() | |||
|  | { | |||
|  |     usb_isr(0); | |||
|  | } | |||
|  | SET_INTERRUPT | |||
|  | void usb0_sof_isr() | |||
|  | { | |||
|  |     usb_sof_isr(0); | |||
|  | } | |||
|  | 
 | |||
|  | #if USB_MAX_HW_NUM == 2
 | |||
|  | SET_INTERRUPT | |||
|  | void usb1_g_isr() | |||
|  | { | |||
|  |     usb_isr(1); | |||
|  | } | |||
|  | SET_INTERRUPT | |||
|  | void usb1_sof_isr() | |||
|  | { | |||
|  |     usb_sof_isr(1); | |||
|  | } | |||
|  | #endif
 | |||
|  | __attribute__((always_inline_when_const_args)) | |||
|  | u32 usb_g_set_intr_hander(const usb_dev usb_id, u32 ep, usb_interrupt hander) | |||
|  | { | |||
|  |     if (ep & USB_DIR_IN) { | |||
|  |         usb_interrupt_tx[usb_id][ep & 0xf] = hander; | |||
|  |     } else { | |||
|  |         usb_interrupt_rx[usb_id][ep] = hander; | |||
|  |     } | |||
|  |     return 0; | |||
|  | } | |||
|  | void usb_g_isr_reg(const usb_dev usb_id, u8 priority, u8 cpu_id) | |||
|  | { | |||
|  |     if (usb_id == 0) { | |||
|  |         request_irq(IRQ_USB_CTRL_IDX, priority, usb0_g_isr, cpu_id); | |||
|  |     } else { | |||
|  | #if USB_MAX_HW_NUM == 2
 | |||
|  |         request_irq(IRQ_USB1_CTRL_IDX, priority, usb1_g_isr, cpu_id); | |||
|  | #endif
 | |||
|  |     } | |||
|  | } | |||
|  | void usb_sof_isr_reg(const usb_dev usb_id, u8 priority, u8 cpu_id) | |||
|  | { | |||
|  |     if (usb_id == 0) { | |||
|  |         request_irq(IRQ_USB_SOF_IDX, priority, usb0_sof_isr, cpu_id); | |||
|  |     } else { | |||
|  | #if USB_MAX_HW_NUM == 2
 | |||
|  |         request_irq(IRQ_USB1_SOF_IDX, priority, usb1_sof_isr, cpu_id); | |||
|  | #endif
 | |||
|  |     } | |||
|  | } | |||
|  | u32 usb_config(const usb_dev usb_id) | |||
|  | { | |||
|  |     memset(usb_interrupt_rx[usb_id], 0, sizeof(usb_interrupt_rx[usb_id])); | |||
|  |     memset(usb_interrupt_tx[usb_id], 0, sizeof(usb_interrupt_tx[usb_id])); | |||
|  | 
 | |||
|  |     if (!usb_config_var) { | |||
|  | #if USB_MALLOC_ENABLE
 | |||
|  |         usb_config_var = (struct usb_config_var_t *)zalloc(sizeof(struct usb_config_var_t)); | |||
|  |         if (!usb_config_var) { | |||
|  |             return -1; | |||
|  |         } | |||
|  | #else
 | |||
|  |         memset(&_usb_config_var, 0, sizeof(_usb_config_var)); | |||
|  |         usb_config_var = &_usb_config_var; | |||
|  | #endif
 | |||
|  |     } | |||
|  |     log_debug("zalloc: usb_config_var = %x\n", usb_config_var); | |||
|  | 
 | |||
|  |     usb_var_init(usb_id, &(usb_config_var->usb_ep_addr)); | |||
|  |     usb_setup_init(usb_id, &(usb_config_var->usb_setup), usb_config_var->usb_setup_buffer); | |||
|  |     return 0; | |||
|  | } | |||
|  | 
 | |||
|  | u32 usb_release(const usb_dev usb_id) | |||
|  | { | |||
|  |     log_debug("free zalloc: usb_config_var = %x\n", usb_id, usb_config_var); | |||
|  |     usb_var_init(usb_id, NULL); | |||
|  |     usb_setup_init(usb_id, NULL, NULL); | |||
|  | #if USB_MALLOC_ENABLE
 | |||
|  |     if (usb_config_var) { | |||
|  |         log_debug("free: usb_config_var = %x\n", usb_config_var); | |||
|  |         free(usb_config_var); | |||
|  |     } | |||
|  | #endif
 | |||
|  | 
 | |||
|  |     usb_config_var = NULL; | |||
|  | 
 | |||
|  |     return 0; | |||
|  | } |