Files
luban-lite/packages/third-party/cherryusb/port/mm32/usb_dc_mm32.c
刘可亮 564e22b32f v0.7.5
2023-08-28 09:48:01 +08:00

265 lines
8.0 KiB
C

#include "usbd_core.h"
#include "usb_mm32_reg.h"
#ifndef USBD_IRQHandler
#define USBD_IRQHandler USB_HP_CAN1_TX_IRQHandler //use actual usb irq name instead
#endif
#define USB_BASE ((uint32_t)0x40005C00)
#define USB ((USB_TypeDef *)USB_BASE)
#ifndef USB_NUM_BIDIR_ENDPOINTS
#define USB_NUM_BIDIR_ENDPOINTS 5
#endif
#define USB_GET_EPX_INT_STATE(ep_idx) (*(volatile uint32_t *)(&USB->rEP1_INT_STATE + (ep_idx - 1)))
#define USB_SET_EPX_INT_STATE(ep_idx, val) (*(volatile uint32_t *)(&USB->rEP1_INT_STATE + (ep_idx - 1)) = val)
#define USB_SET_EP_INT(ep_idx, val) (*(volatile uint32_t *)(&USB->rEP1_INT_EN + (ep_idx - 1)) = val)
#define USB_GET_EP_RX_CNT(ep_idx) (*(volatile uint32_t *)(&USB->rEP0_AVIL + ep_idx))
#define USB_GET_EP_CTRL(ep_idx) (*(volatile uint32_t *)(&USB->rEP0_CTRL + ep_idx))
#define USB_SET_EP_CTRL(ep_idx, val) (*(volatile uint32_t *)(&USB->rEP0_CTRL + ep_idx) = val)
#define USB_GET_EP_FIFO(ep_idx) (*(volatile uint32_t *)(&USB->rEP0_FIFO + ep_idx))
#define USB_SET_EP_FIFO(ep_idx, val) (*(volatile uint32_t *)(&USB->rEP0_FIFO + ep_idx) = val)
/* Endpoint state */
struct usb_dc_ep_state {
/** Endpoint max packet size */
uint16_t ep_mps;
/** Endpoint Transfer Type.
* May be Bulk, Interrupt, Control or Isochronous
*/
uint8_t ep_type;
uint8_t ep_stalled; /** Endpoint stall flag */
};
/* Driver state */
struct usb_dc_config_priv {
struct usb_dc_ep_state in_ep[USB_NUM_BIDIR_ENDPOINTS]; /*!< IN endpoint parameters*/
struct usb_dc_ep_state out_ep[USB_NUM_BIDIR_ENDPOINTS]; /*!< OUT endpoint parameters */
} usb_dc_cfg;
__WEAK void usb_dc_low_level_init(void)
{
}
__WEAK void usb_dc_low_level_deinit(void)
{
}
int usb_dc_init(void)
{
memset(&usb_dc_cfg, 0, sizeof(struct usb_dc_config_priv));
usb_dc_cfg.out_ep[0].ep_mps = USB_CTRL_EP_MPS;
usb_dc_cfg.out_ep[0].ep_type = 0x00;
usb_dc_cfg.in_ep[0].ep_mps = USB_CTRL_EP_MPS;
usb_dc_cfg.in_ep[0].ep_type = 0x00;
usb_dc_low_level_init();
USB->rTOP = USB_TOP_RESET; //reset usb
USB->rTOP &= ~USB_TOP_RESET;
USB->rTOP &= ~USB_TOP_CONNECT; //usb disconnect
USB->rINT_STATE |= 0;
USB->rEP_INT_STATE |= 0;
USB->rEP0_INT_STATE |= 0;
USB->rEP1_INT_STATE |= 0;
USB->rEP2_INT_STATE |= 0;
USB->rEP3_INT_STATE |= 0;
USB->rEP4_INT_STATE |= 0;
USB->rEP0_CTRL = 0;
USB->rEP1_CTRL = 0;
USB->rEP2_CTRL = 0;
USB->rEP3_CTRL = 0;
USB->rEP4_CTRL = 0;
USB->rINT_EN = USB_INT_EN_RSTIE | USB_INT_EN_SUSPENDIE | USB_INT_EN_RESUMIE | USB_INT_EN_EPINTIE; //enable rst、suppend、resume、ep global irq
USB->rEP0_INT_EN = EPn_INT_EN_SETUPIE | EPn_INT_EN_OUTACKIE | EPn_INT_EN_INACKIE | EPn_INT_EN_OUTSTALLIE | EPn_INT_EN_INSTALLIE; //enable ep0 setup、inack、outack irq
USB->rEP_INT_EN = EP_INT_EN_EP0IE; //enable ep0 irq
USB->rEP_EN = EP_EN_EP0EN; //enable ep0
USB->rADDR = 0; //set addr 0
USB->rTOP = USB_TOP_CONNECT | ((~USB_TOP_SPEED) & 0x01); //connect usb
USB->rPOWER = USB_POWER_SUSPEN | USB_POWER_SUSP;
return 0;
}
int usb_dc_deinit(void)
{
usb_dc_low_level_deinit();
return 0;
}
int usbd_set_address(const uint8_t addr)
{
return 0;
}
int usbd_ep_open(const struct usbd_endpoint_cfg *ep_cfg)
{
uint8_t ep_idx = USB_EP_GET_IDX(ep_cfg->ep_addr);
if (ep_idx == 0) {
return 0;
}
if (USB_EP_DIR_IS_OUT(ep_cfg->ep_addr)) {
usb_dc_cfg.out_ep[ep_idx].ep_mps = ep_cfg->ep_mps;
usb_dc_cfg.out_ep[ep_idx].ep_type = ep_cfg->ep_type;
USB_SET_EP_INT(ep_idx, EPn_INT_EN_OUTACKIE);
} else {
usb_dc_cfg.in_ep[ep_idx].ep_mps = ep_cfg->ep_mps;
usb_dc_cfg.in_ep[ep_idx].ep_type = ep_cfg->ep_type;
USB_SET_EP_INT(ep_idx, EPn_INT_EN_INACKIE);
}
USB->rEP_INT_EN |= (1 << ep_idx);
USB->rEP_EN |= (1 << ep_idx);
return 0;
}
int usbd_ep_close(const uint8_t ep)
{
return 0;
}
int usbd_ep_set_stall(const uint8_t ep)
{
USB->rEP_HALT |= (1 << (ep & 0x7f));
return 0;
}
int usbd_ep_clear_stall(const uint8_t ep)
{
USB->rEP_HALT &= ~(1 << (ep & 0x7f));
return 0;
}
int usbd_ep_is_stalled(const uint8_t ep, uint8_t *stalled)
{
return 0;
}
int usbd_ep_write(const uint8_t ep, const uint8_t *data, uint32_t data_len, uint32_t *ret_bytes)
{
uint8_t ep_idx = USB_EP_GET_IDX(ep);
uint8_t *pdata = (uint8_t *)data;
if (!data && data_len) {
return -1;
}
if (!data_len) {
while (USB_GET_EP_CTRL(ep_idx) & 0x80) {
}
USB_SET_EP_CTRL(ep_idx, 0x80);
return 0;
}
if (data_len > usb_dc_cfg.in_ep[ep_idx].ep_mps) {
data_len = usb_dc_cfg.in_ep[ep_idx].ep_mps;
}
while (USB_GET_EP_CTRL(ep_idx) & 0x80) {
}
for (uint8_t i = 0; i < data_len; i++) {
USB_SET_EP_FIFO(ep_idx, *pdata);
pdata++;
}
USB_SET_EP_CTRL(ep_idx, 0x80 | data_len);
if (ret_bytes) {
*ret_bytes = data_len;
}
return 0;
}
int usbd_ep_read(const uint8_t ep, uint8_t *data, uint32_t max_data_len, uint32_t *read_bytes)
{
uint8_t ep_idx = USB_EP_GET_IDX(ep);
uint32_t read_count;
uint8_t *pdata = (uint8_t *)data;
if (!data && max_data_len) {
return -1;
}
if (!max_data_len) {
return 0;
}
if ((ep_idx == 0) && (max_data_len == 8) && !read_bytes) {
for (uint8_t i = 0; i < 8; i++) {
*(pdata + i) = USB->rSETUP[i];
}
} else {
read_count = USB_GET_EP_RX_CNT(ep_idx);
read_count = MIN(read_count, max_data_len);
for (uint8_t i = 0; i < read_count; i++) {
*(pdata + i) = USB_GET_EP_FIFO(ep_idx);
}
}
if (read_bytes) {
*read_bytes = read_count;
}
return 0;
}
void USBD_IRQHandler(void)
{
uint32_t int_status;
uint32_t epindex;
uint32_t ep_int_status;
int_status = USB->rINT_STATE;
USB->rINT_STATE = int_status;
if (int_status & USB_INT_STATE_EPINTF) {
epindex = USB->rEP_INT_STATE; //read all ep interrupt status
USB->rEP_INT_STATE = epindex; //clear interrupt status
for (uint32_t i = 0; i < USB_NUM_BIDIR_ENDPOINTS; i++) {
if (epindex & (EP_INT_STATE_EP0F << i)) //read epx interrupt status
{
if (i == 0) {
ep_int_status = USB->rEP0_INT_STATE;
USB->rEP0_INT_STATE = ep_int_status;
if (ep_int_status & EPn_INT_STATE_SETUP) //setup interrupt status
{
usbd_event_notify_handler(USBD_EVENT_SETUP_NOTIFY, NULL);
}
if (ep_int_status & EPn_INT_STATE_OUTACK) //outack interrupt status
{
usbd_event_notify_handler(USBD_EVENT_EP0_OUT_NOTIFY, NULL);
}
if (ep_int_status & EPn_INT_STATE_INACK) //inack interrupt status
{
usbd_event_notify_handler(USBD_EVENT_EP0_IN_NOTIFY, NULL);
}
} else {
ep_int_status = USB_GET_EPX_INT_STATE(i);
USB_SET_EPX_INT_STATE(i, ep_int_status);
if (ep_int_status & EPn_INT_STATE_OUTACK) {
usbd_event_notify_handler(USBD_EVENT_EP_OUT_NOTIFY, (void *)(i & 0x7f));
}
if (ep_int_status & EPn_INT_STATE_INACK) {
usbd_event_notify_handler(USBD_EVENT_EP_IN_NOTIFY, (void *)(i | 0x80));
}
}
if (ep_int_status & EPn_INT_STATE_INSTALL) {
}
if (ep_int_status & EPn_INT_STATE_OUTSTALL) {
}
}
}
} else if (int_status & USB_INT_STATE_RSTF) {
USB->rTOP |= USB_TOP_RESET;
USB->rTOP &= ~USB_TOP_RESET;
usbd_event_notify_handler(USBD_EVENT_RESET, NULL);
} else if (int_status & USB_INT_STATE_SUSPENDF) {
} else if (int_status & USB_INT_STATE_RESUMF) {
}
}