mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-22 14:17:23 +08:00
2028 lines
53 KiB
C
2028 lines
53 KiB
C
|
|
#include <rthw.h>
|
|
#include <rtthread.h>
|
|
#include <rtdevice.h>
|
|
#include <drivers/usb_device.h>
|
|
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <string.h>
|
|
|
|
#include <x1000.h>
|
|
#include <mips_regs.h>
|
|
#include <cache.h>
|
|
|
|
#include "x1000_dwc.h"
|
|
|
|
|
|
//#define DWC_DEBUG
|
|
#ifdef DWC_DEBUG
|
|
#define DWC_DBG(fmt, args...) rt_kprintf(fmt ,##args)
|
|
#else
|
|
#define DWC_DBG(fmt, args...)
|
|
#endif
|
|
|
|
|
|
#define UdcID (('U' << 24) | ('D' << 16) | ('C' << 16) | (':' << 16))
|
|
#define IS_SLAVE_MODE 0
|
|
#define IS_INTERN_DMA 2
|
|
#define IS_EXTERN_DMA 1
|
|
|
|
const char *ep0_state_string[] =
|
|
{
|
|
"EP_SETUP",
|
|
"EP_DATA",
|
|
"EP_STATUS",
|
|
"EP_SETUP_PHASEDONE",
|
|
};
|
|
|
|
#if DWC_FORCE_SPEED_FULL
|
|
#define DEP_EP_MAXPKT(n) \
|
|
({ \
|
|
int v = 0; \
|
|
if (n) \
|
|
v = 64; \
|
|
else \
|
|
v = 64; \
|
|
v; \
|
|
})
|
|
#else
|
|
#define DEP_EP_MAXPKT(n) \
|
|
({ \
|
|
int v = 0; \
|
|
if (n) \
|
|
v = 512; \
|
|
else \
|
|
v = 64; \
|
|
v; \
|
|
})
|
|
#endif
|
|
|
|
#define MAX_PKT_CNT 1023
|
|
|
|
ALIGN(32)
|
|
//static uint32_t setup_packet[64] = {0, 0, 0, 0, 0};
|
|
static int sleep_flag = 0;
|
|
|
|
|
|
|
|
/*
|
|
* static functions
|
|
*/
|
|
static void dwc_otg_device_init(dwc_handle *dwc);
|
|
static void dwc_otg_core_reset(dwc_handle *dwc);
|
|
static void dwc_otg_core_init(dwc_handle *dwc,uint8_t dma_enable);
|
|
static void dwc_otg_phy_suspend(int suspend);
|
|
|
|
|
|
static void udelay(uint32_t x)
|
|
{
|
|
volatile uint32_t n = 1000;
|
|
|
|
while(x--)
|
|
{
|
|
for (n = 0; n < 1000; ++n);
|
|
}
|
|
}
|
|
|
|
static void mdelay(uint32_t x)
|
|
{
|
|
while(x--)
|
|
udelay(1000);
|
|
}
|
|
|
|
static int dwc_get_utmi_width(dwc_handle *dwc)
|
|
{
|
|
return (REG_GHW_CFG4 >> 14) & 0x3;
|
|
}
|
|
|
|
static void dwc_otg_select_phy_width(dwc_handle *dwc)
|
|
{
|
|
REG_GUSB_CFG &= ~USBCFG_TRDTIME_MASK;
|
|
REG_GUSB_CFG |= (1 << 3);
|
|
REG_GUSB_CFG |= USBCFG_TRDTIME_6;
|
|
REG_CPM_USBPCR1 |= (3 << 18);
|
|
}
|
|
|
|
|
|
static void dwc_otg_write_packet(dwc_handle *dwc, uint8_t epnum)
|
|
{
|
|
int i;
|
|
uint32_t dwords;
|
|
uint32_t byte_count;
|
|
dwc_ep *pep;
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_IN_OFS + epnum];
|
|
|
|
byte_count = pep->xfer_len - pep->xfer_count;
|
|
|
|
if (byte_count > DEP_EP_MAXPKT(epnum))
|
|
byte_count = DEP_EP_MAXPKT(epnum);
|
|
|
|
dwords = (byte_count + 3) / 4;
|
|
|
|
for (i = 0; i < dwords; i++)
|
|
{
|
|
REG_EP_FIFO(epnum) = REG32((uint32_t * )(pep->xfer_buff) + i);
|
|
}
|
|
|
|
pep->xfer_count += byte_count;
|
|
pep->xfer_buff += byte_count;
|
|
}
|
|
|
|
|
|
void dwc_read_ep_packet(dwc_handle *dwc, uint8_t epnum, uint32_t count)
|
|
{
|
|
int i;
|
|
int dwords = (count + 3) / 4;
|
|
dwc_ep *pep;
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_OUT_OFS + epnum];
|
|
|
|
for (i = 0; i < dwords; i++)
|
|
REG32((uint32_t *)(pep->xfer_buff + pep->xfer_count / 4) + i) = REG_EP_FIFO(epnum);
|
|
|
|
pep->xfer_count += count;
|
|
}
|
|
|
|
|
|
void dwc_write_ep_packet(dwc_handle *dwc,uint8_t epnum)
|
|
{
|
|
uint32_t xfersize, finish, insize;
|
|
uint32_t dwords;
|
|
uint32_t txstatus = REG_DIEP_TXFSTS(epnum & 0x0F);
|
|
dwc_ep *pep;
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_IN_OFS + epnum];
|
|
|
|
insize = pep->xfer_len;
|
|
if (pep->xfer_len > DEP_EP_MAXPKT(epnum))
|
|
xfersize = DEP_EP_MAXPKT(epnum);
|
|
else
|
|
xfersize = pep->xfer_len;
|
|
|
|
dwords = (xfersize + 3) / 4;
|
|
DWC_DBG("txstatus (%x) dwords (%x) length (%x) xfer_count (%x) \n", txstatus, dwords, pep->xfer_len, pep->xfer_count);
|
|
|
|
while ((txstatus > dwords) && (pep->xfer_len > 0) && (pep->xfer_count < pep->xfer_len) )
|
|
{
|
|
dwc_otg_write_packet(dwc, epnum);
|
|
xfersize = pep->xfer_len - pep->xfer_count;
|
|
if (xfersize > DEP_EP_MAXPKT(epnum))
|
|
xfersize = DEP_EP_MAXPKT(epnum);
|
|
dwords = (xfersize + 3) / 4;
|
|
txstatus = REG_DIEP_TXFSTS(epnum);
|
|
}
|
|
finish = pep->xfer_count;
|
|
|
|
if (insize > finish)
|
|
{
|
|
uint32_t intr = REG_DIEP_INT(epnum);
|
|
while (!(intr & DEP_TXFIFO_EMPTY))
|
|
{
|
|
intr = REG_DIEP_INT(epnum);
|
|
}
|
|
HW_SendPKT(dwc,epnum, pep->xfer_buff, insize - finish);
|
|
}
|
|
return;
|
|
}
|
|
|
|
void dwc_handle_ep_data_in_phase(dwc_handle *dwc, uint8_t epnum)
|
|
{
|
|
uint32_t pktcnt, xfersize;
|
|
uint32_t dma_addr, dma_len;
|
|
dwc_ep *pep;
|
|
|
|
DWC_DBG("%s %d\n",__func__,__LINE__);
|
|
DWC_DBG("epnum = %d\n",epnum);
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_IN_OFS + epnum];
|
|
|
|
xfersize = pep->xfer_len;
|
|
pktcnt = (xfersize + DEP_EP_MAXPKT(epnum) - 1) / DEP_EP_MAXPKT(epnum);
|
|
|
|
if (pktcnt > 1023)
|
|
{
|
|
DWC_DBG("WARNING...\n");
|
|
while (1) ;
|
|
}
|
|
|
|
if (epnum == 0)
|
|
{
|
|
REG_DIEP_SIZE(epnum) &= ~(0x1fffff);
|
|
REG_DIEP_SIZE(epnum) |= (pktcnt << 19) | xfersize;
|
|
}
|
|
else
|
|
{
|
|
REG_DIEP_SIZE(epnum) &= ~(0x1fffffff);
|
|
REG_DIEP_SIZE(epnum) |= (pktcnt << 19) | xfersize;
|
|
}
|
|
|
|
if (dwc->is_dma != 0)
|
|
{
|
|
dma_addr = (uint32_t)(pep->xfer_buff);
|
|
dma_len = (((pep->xfer_len + 7) >> 3) << 3);
|
|
|
|
//dump data...
|
|
DWC_DBG("IN:\n");
|
|
{
|
|
int i;
|
|
for (i = 0; i < dma_len; ++i)
|
|
{
|
|
DWC_DBG("%02x ", *(unsigned char *)(dma_addr+i));
|
|
if ((i + 1) % 16 == 0)
|
|
DWC_DBG("\n");
|
|
}
|
|
}
|
|
DWC_DBG("\n");
|
|
|
|
REG_DIEP_DMA(epnum) = PHYS(pep->xfer_buff);
|
|
REG_DIEP_CTL(epnum) |= (DEP_ENA_BIT | DEP_CLEAR_NAK);
|
|
}
|
|
else
|
|
{
|
|
REG_DIEP_CTL(epnum) |= (DEP_ENA_BIT | DEP_CLEAR_NAK);
|
|
REG_DIEP_EMPMSK |= (1 << epnum);
|
|
}
|
|
|
|
return ;
|
|
}
|
|
|
|
|
|
void dwc_handle_ep_status_in_phase(dwc_handle *dwc, uint8_t epnum)
|
|
{
|
|
dwc_ep *pep;
|
|
|
|
DWC_DBG("%s %d\n",__func__,__LINE__);
|
|
DWC_DBG("epnum = %d\n",epnum);
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_IN_OFS + epnum];
|
|
|
|
pep->xfer_len = 0;
|
|
pep->xfer_count = 0;
|
|
|
|
if (epnum == 0)
|
|
{
|
|
REG_DIEP_SIZE(epnum) &= ~(0x1fffff);
|
|
REG_DIEP_SIZE(epnum) |= DOEPSIZE0_PKTCNT_BIT | (pep->xfer_len); // pktcnt->1 xfersize->0
|
|
}
|
|
else
|
|
{
|
|
REG_DIEP_SIZE(epnum) &= ~(0x1FFFFFFF);
|
|
REG_DIEP_SIZE(epnum) |= DOEPSIZE0_PKTCNT_BIT | (pep->xfer_len); // pktcnt->1 xfersize->0
|
|
}
|
|
|
|
if (dwc->is_dma == IS_INTERN_DMA)
|
|
{
|
|
// pep->xfer_buff = (void *)0xFFFFFFFF;
|
|
// REG_DIEP_DMA(epnum) = PHYS(pep->xfer_buff);
|
|
REG_DIEP_DMA(epnum) = PHYS(0xFFFFFFFF);
|
|
REG_DIEP_CTL(epnum) |= DEP_ENA_BIT | DEP_CLEAR_NAK;
|
|
}
|
|
else
|
|
{
|
|
REG_DIEP_CTL(epnum) |= DEP_ENA_BIT | DEP_CLEAR_NAK;
|
|
}
|
|
return ;
|
|
}
|
|
|
|
|
|
|
|
void dwc_handle_ep_data_out_phase(dwc_handle *dwc,uint8_t epnum)
|
|
{
|
|
uint32_t dma_addr, dma_len;
|
|
uint32_t pktcnt;
|
|
dwc_ep *pep;
|
|
|
|
DWC_DBG("%s %d\n",__func__,__LINE__);
|
|
DWC_DBG("epnum = %d\n",epnum);
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_OUT_OFS + epnum];
|
|
|
|
if (epnum == 0)
|
|
{
|
|
if (dwc->is_dma == IS_INTERN_DMA)
|
|
{
|
|
dma_len = pep->maxpacket;
|
|
dma_addr = (uint32_t) (pep->xfer_buff);
|
|
rt_hw_dcache_flush_range(dma_addr,dma_len);
|
|
REG_DOEP_DMA(epnum) = PHYS(pep->xfer_buff);
|
|
}
|
|
REG_DOEP_SIZE(epnum) = DOEPSIZE0_SUPCNT_3 | DOEPSIZE0_PKTCNT_BIT | (pep->maxpacket);
|
|
REG_DOEP_CTL(epnum) |= DEP_ENA_BIT | DEP_CLEAR_NAK;
|
|
}
|
|
else
|
|
{
|
|
if (pep->xfer_len > 0)
|
|
{
|
|
if (pep->xfer_len > MAX_PKT_CNT * DEP_EP_MAXPKT(epnum))
|
|
pep->xfer_len = MAX_PKT_CNT * DEP_EP_MAXPKT(epnum);
|
|
pktcnt = (pep->xfer_len + DEP_EP_MAXPKT(epnum) - 1) / DEP_EP_MAXPKT(epnum);
|
|
if (pktcnt > 1023)
|
|
{
|
|
DWC_DBG("WARNING...\n");
|
|
while (1) ;
|
|
}
|
|
|
|
REG_DOEP_SIZE(epnum) &= ~(0x1fffffff);
|
|
REG_DOEP_SIZE(epnum) |= (pktcnt << 19) | (pep->xfer_len);
|
|
}
|
|
|
|
if (dwc->is_dma == IS_INTERN_DMA)
|
|
{
|
|
dma_len = (((pep->xfer_len + 7) >> 3) << 3);
|
|
dma_addr = (uint32_t)(pep->xfer_buff);
|
|
rt_hw_dcache_flush_range(dma_addr, dma_len);
|
|
REG_DOEP_DMA(epnum) = PHYS(pep->xfer_buff);
|
|
}
|
|
/* Program the DOEPCTLn Register with endpoint charateristics,
|
|
* and set the Endpoint Enable and Clear NAK bit */
|
|
REG_DOEP_CTL(epnum) |= DEP_ENA_BIT | DEP_CLEAR_NAK;
|
|
}
|
|
}
|
|
|
|
int HW_SendPKT(dwc_handle *dwc, uint8_t epnum, const uint8_t *buf, int size)
|
|
{
|
|
uint32_t dma_addr, dma_len;
|
|
dwc_ep *pep;
|
|
rt_base_t level;
|
|
|
|
DWC_DBG("HW_SendPKT addr = %02x,size = %d\n",epnum,size);
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_IN_OFS + epnum];
|
|
|
|
pep->xfer_len = size; /* number of bytes to transfer */
|
|
pep->xfer_count = 0; /* number of bytes transfered */
|
|
// pep->xfer_buff = (uint8_t *)buf; /* pointer to transfer buffer */
|
|
if(size > 0)
|
|
{
|
|
memcpy(pep->xfer_buff,buf,size);
|
|
rt_hw_dcache_flush_range((rt_uint32_t)pep->xfer_buff,(rt_uint32_t)size);
|
|
}
|
|
|
|
if (pep->xfer_len > MAX_PKT_CNT * DEP_EP_MAXPKT(epnum))
|
|
pep->xfer_len = MAX_PKT_CNT * DEP_EP_MAXPKT(epnum);
|
|
|
|
pep->xfer_count = 0;
|
|
|
|
switch (pep->type)
|
|
{
|
|
case DWC_OTG_EP_TYPE_CONTROL:
|
|
if (pep->xfer_len > 0)
|
|
pep->ep_state = EP_DATA;
|
|
else
|
|
pep->ep_state = EP_STATUS;
|
|
|
|
/* 2 Stage */
|
|
if (pep->ep_state == EP_STATUS && pep->xfer_len == 0) /*EP_SETUP 0 EP_DATA 1 EP_STATUS 2*/
|
|
{
|
|
DWC_DBG("%s %d ep_state = %s\n", __func__, __LINE__, ep0_state_string[pep->ep_state]);
|
|
|
|
dwc_handle_ep_status_in_phase(dwc, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* 3 Stage */
|
|
if (pep->ep_state == EP_DATA)
|
|
{
|
|
/* enable in data phase */
|
|
dwc_handle_ep_data_in_phase(dwc, epnum);
|
|
}
|
|
break;
|
|
case DWC_OTG_EP_TYPE_BULK:
|
|
if (pep->ep_state == EP_IDLE || pep->ep_state == EP_TRANSFERED)
|
|
{
|
|
pep->ep_state = EP_TRANSFERING;
|
|
if (pep->xfer_len == 0)
|
|
{
|
|
dwc_handle_ep_status_in_phase(dwc, epnum);
|
|
return 0;
|
|
}
|
|
dwc_handle_ep_data_in_phase(dwc, epnum);
|
|
}
|
|
break;
|
|
}
|
|
|
|
return pep->xfer_len;
|
|
}
|
|
|
|
int HW_GetPKT(dwc_handle *dwc, uint8_t epnum, uint8_t *buf,int size)
|
|
{
|
|
int i;
|
|
dwc_ep *pep;
|
|
|
|
DWC_DBG("HW_GetPKT:%d %d\n", epnum, dwc->is_dma);
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_OUT_OFS + epnum];
|
|
|
|
if ((size == 0) || (size > pep->xfer_count))
|
|
size = pep->xfer_count;
|
|
|
|
if (dwc->is_dma == IS_INTERN_DMA)
|
|
{
|
|
DWC_DBG("HW_GetPKT:%x %x \n", pep->ctrl_req_addr, UNCACHED(pep->xfer_buff));
|
|
memcpy((uint8_t*) buf, (uint8_t *) UNCACHED(pep->xfer_buff), size);
|
|
}
|
|
else
|
|
{
|
|
memcpy((uint8_t*) buf, (uint8_t *) (pep->xfer_buff), size);
|
|
}
|
|
|
|
return size;
|
|
}
|
|
|
|
static void dwc_otg_flush_rx_fifo(dwc_handle *dwc)
|
|
{
|
|
;
|
|
}
|
|
|
|
static void dwc_otg_flush_tx_fifo(dwc_handle *dwc,uint8_t epnum)
|
|
{
|
|
uint32_t gintsts;
|
|
uint32_t grstctl;
|
|
uint32_t cnt;
|
|
|
|
gintsts = REG_GINT_STS;
|
|
/* Step1: Check that GINTSTS.GinNakEff=0 if this
|
|
* bit is cleared then set Dctl.SGNPInNak = 1.
|
|
* Nak effective interrupt = H indicating the core
|
|
* is not reading from fifo*/
|
|
if ((gintsts & GINTSTS_GINNAK_EFF))
|
|
{
|
|
REG_OTG_DCTL |= DCTL_SGNPINNAK;
|
|
|
|
/* Step2: wait for GINTSTS.GINNakEff=1,which indicates
|
|
* the NAK setting has taken effect to all IN endpoints */
|
|
while (!(REG_GINT_STS & GINTSTS_GINNAK_EFF))
|
|
udelay(1);
|
|
}
|
|
|
|
/* Step3: wait for ahb master idle state */
|
|
while (!(REG_GRST_CTL & RSTCTL_AHB_IDLE))
|
|
udelay(1);
|
|
|
|
/* Step4: Check that GrstCtl.TxFFlsh=0, if it is 0, then write
|
|
* the TxFIFO number you want to flush to GrstCTL.TxFNum*/
|
|
grstctl = REG_GRST_CTL;
|
|
if (!(grstctl & RSTCTL_TXFIFO_FLUSH))
|
|
{
|
|
REG_GRST_CTL |= ((epnum & 0x0F) << 6);
|
|
}
|
|
|
|
/* Step5: Set GRSTCTL.TxFFlsh=1 and wait for it to clear */
|
|
REG_GRST_CTL |= RSTCTL_TXFIFO_FLUSH;
|
|
|
|
while (REG_GRST_CTL & RSTCTL_TXFIFO_FLUSH)
|
|
{
|
|
udelay(1);
|
|
}
|
|
|
|
/* Step6: Set the DCTL.GCNPinNak bit */
|
|
REG_OTG_DCTL |= DCTL_CLR_GNPINNAK;
|
|
}
|
|
|
|
static void dwc_set_in_nak(dwc_handle *dwc, int epnum)
|
|
{
|
|
int timeout = 5000;
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
|
|
REG_DIEP_CTL(epnum) |= DEP_SET_NAK;
|
|
do
|
|
{
|
|
udelay(1);
|
|
if (timeout < 2)
|
|
{
|
|
DWC_DBG("dwc set in nak timeout epnum %d\n", epnum);
|
|
}
|
|
} while ((!(REG_DIEP_INT(epnum) & DEP_INEP_NAKEFF)) && (--timeout > 0));
|
|
}
|
|
|
|
static void dwc_set_out_nak(dwc_handle *dwc,int epnum)
|
|
{
|
|
epnum &= DWC_EPNO_MASK;
|
|
REG_DOEP_CTL(epnum) |= DEP_SET_NAK;
|
|
}
|
|
|
|
static void dwc_disable_in_ep(dwc_handle *dwc,int epnum)
|
|
{
|
|
int timeout = 100000;
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
|
|
if (!(REG_DIEP_CTL(epnum) & DEP_ENA_BIT))
|
|
return ;
|
|
|
|
/*step 1 : set nak*/
|
|
dwc_set_in_nak(dwc,epnum);
|
|
|
|
/*step 2: disable endpoint*/
|
|
REG_DIEP_CTL(epnum) |= DEP_DISENA_BIT;
|
|
|
|
do
|
|
{
|
|
udelay(1);
|
|
if (timeout < 2)
|
|
{
|
|
DWC_DBG("dwc disable in ep timeout epnum : %d\n", epnum);
|
|
}
|
|
} while ( (!(REG_DIEP_INT(epnum) & DEP_EPDIS_INT)) && (--timeout > 0));
|
|
|
|
REG_DIEP_INT(epnum) = DEP_EPDIS_INT;
|
|
|
|
/*step 3: flush tx fifo*/
|
|
dwc_otg_flush_tx_fifo(dwc, epnum);
|
|
|
|
REG_DIEP_SIZE(epnum) = 0x0;
|
|
|
|
/*step 4: clear nak*/
|
|
if (epnum == 1)
|
|
REG_DIEP_CTL(1) |= DEP_CLEAR_NAK;
|
|
}
|
|
|
|
int dwc_enable_in_ep(dwc_handle *dwc,uint8_t epnum)
|
|
{
|
|
dwc_ep *pep = RT_NULL;
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_IN_OFS + epnum];
|
|
|
|
|
|
/* Program the endpoint register to configure them with the characteristics of valid endpoints */
|
|
REG_DIEP_CTL(epnum) &= ~DEP_PKTSIZE_MASK;
|
|
REG_DIEP_CTL(epnum) &= ~DEP_TYPE_MASK;
|
|
|
|
switch (dwc->speed)
|
|
{
|
|
case USB_SPEED_FULL:
|
|
case USB_SPEED_LOW:
|
|
REG_DIEP_CTL(epnum) |= DEP_FS_PKTSIZE;
|
|
break;
|
|
case USB_SPEED_HIGH:
|
|
REG_DIEP_CTL(epnum) |= DEP_HS_PKTSIZE;
|
|
break;
|
|
}
|
|
|
|
//tx fifo number
|
|
REG_DIEP_CTL(epnum) |= (epnum << 22);
|
|
|
|
//ep type
|
|
switch (pep->type)
|
|
{
|
|
case DWC_OTG_EP_TYPE_CONTROL:
|
|
REG_DIEP_CTL(epnum) |= USB_ACTIVE_EP | DEP_TYPE_CNTL;
|
|
break;
|
|
case DWC_OTG_EP_TYPE_ISOC:
|
|
REG_DIEP_CTL(epnum) |= USB_ACTIVE_EP | DEP_TYPE_ISO;
|
|
break;
|
|
case DWC_OTG_EP_TYPE_BULK:
|
|
REG_DIEP_CTL(epnum) |= USB_ACTIVE_EP | DEP_TYPE_BULK;
|
|
break;
|
|
case DWC_OTG_EP_TYPE_INTR:
|
|
REG_DIEP_CTL(epnum) |= USB_ACTIVE_EP | DEP_TYPE_INTR;
|
|
break;
|
|
}
|
|
|
|
/* DATA0 */
|
|
REG_DIEP_CTL(epnum) |= (1 << 28);
|
|
|
|
/* Enable EP INT */
|
|
REG_DAINT_MASK |= (0x01 << (DWC_EP_IN_OFS + epnum));
|
|
|
|
return 0;
|
|
}
|
|
|
|
int dwc_enable_out_ep(dwc_handle *dwc,uint8_t epnum)
|
|
{
|
|
uint32_t xfersize;
|
|
uint32_t dma_addr, dma_len, pktcnt;
|
|
dwc_ep *pep = RT_NULL;
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_OUT_OFS + epnum];
|
|
|
|
/* Program the endpoint register to configure them with the characteristics of valid endpoints */
|
|
REG_DOEP_CTL(epnum) &= ~DEP_PKTSIZE_MASK;
|
|
REG_DOEP_CTL(epnum) &= ~DEP_TYPE_MASK;
|
|
|
|
switch (dwc->speed)
|
|
{
|
|
case USB_SPEED_FULL:
|
|
case USB_SPEED_LOW:
|
|
REG_DOEP_CTL(epnum) |= DEP_FS_PKTSIZE;
|
|
break;
|
|
case USB_SPEED_HIGH:
|
|
REG_DOEP_CTL(epnum) |= DEP_HS_PKTSIZE;
|
|
break;
|
|
}
|
|
|
|
//ep type
|
|
switch (pep->type)
|
|
{
|
|
case DWC_OTG_EP_TYPE_CONTROL:
|
|
REG_DOEP_CTL(epnum) |= USB_ACTIVE_EP | DEP_TYPE_CNTL;
|
|
break;
|
|
case DWC_OTG_EP_TYPE_ISOC:
|
|
REG_DOEP_CTL(epnum) |= USB_ACTIVE_EP | DEP_TYPE_ISO;
|
|
break;
|
|
case DWC_OTG_EP_TYPE_BULK:
|
|
REG_DOEP_CTL(epnum) |= USB_ACTIVE_EP | DEP_TYPE_BULK;
|
|
break;
|
|
case DWC_OTG_EP_TYPE_INTR:
|
|
REG_DOEP_CTL(epnum) |= USB_ACTIVE_EP | DEP_TYPE_INTR;
|
|
break;
|
|
}
|
|
|
|
/* DATA0 */
|
|
REG_DOEP_CTL(epnum) |= (1 << 28);
|
|
|
|
/* Enable EP INT */
|
|
REG_DAINT_MASK |= (0x01 << (DWC_EP_OUT_OFS + epnum));
|
|
|
|
/* OUT-EP must init xfer buffer */
|
|
xfersize = pep->maxpacket * 2;
|
|
pktcnt = xfersize / DEP_EP_MAXPKT(epnum);
|
|
|
|
pep->xfer_len = xfersize;
|
|
pep->xfer_count = 0;
|
|
/* xfer_buffer has been initialized by up-layer */
|
|
// pep->xfer_buff = pep->xfer_buff;
|
|
|
|
DWC_DBG("%s %d xfer_buff: %x %x\n", __FUNCTION__, __LINE__, pep->xfer_buff, PHYS(pep->xfer_buff));
|
|
|
|
/* Program the DOEPSIZn register for the transfer size and corresponding packet count */
|
|
REG_DOEP_SIZE(epnum) &= ~(0x1fffffff);
|
|
REG_DOEP_SIZE(epnum) = (pktcnt << 19) | xfersize;
|
|
if (dwc->is_dma == IS_INTERN_DMA)
|
|
{
|
|
dma_addr = (uint32_t) (pep->xfer_buff);
|
|
dma_len = (((xfersize + 7) >> 3) << 3); //pep->xfer_len;
|
|
rt_hw_dcache_flush_range(dma_addr, dma_len);
|
|
/* Additionally, in DMA mode, program the DOEPDMAn register */
|
|
REG_DOEP_DMA(epnum) = PHYS(pep->xfer_buff);
|
|
}
|
|
|
|
/* Program the DOEPCTLn Register with endpoint charateristics,
|
|
* and set the Endpoint Enable and Clear NAK bit */
|
|
REG_DOEP_CTL(epnum) |= DEP_ENA_BIT | DEP_CLEAR_NAK;
|
|
|
|
return 0;
|
|
}
|
|
|
|
void dwc_set_address(dwc_handle *dwc,uint8_t address)
|
|
{
|
|
sleep_flag = 1;
|
|
REG_OTG_DCFG &= ~DCFG_DEV_ADDR_MASK;
|
|
REG_OTG_DCFG |= address << DCFG_DEV_ADDR_BIT;
|
|
}
|
|
|
|
|
|
void dwc_otg_ep0_out_start(dwc_handle *dwc)
|
|
{
|
|
dwc_ep *pep = dwc->dep[DWC_EP_OUT_OFS + 0];
|
|
|
|
DWC_DBG("%s %d\n",__func__,__LINE__);
|
|
pep->xfer_len = 64;
|
|
pep->xfer_count = 0;
|
|
pep->maxpacket = 64;
|
|
// pep->ctrl_req_addr = (uint32_t)(&setup_packet[0]);
|
|
pep->xfer_buff = pep->xfer_buff;
|
|
|
|
if (dwc->is_dma == IS_INTERN_DMA)
|
|
{
|
|
REG_DOEP_SIZE(0) = DOEPSIZE0_SUPCNT_3 | DOEPSIZE0_PKTCNT_BIT | (pep->maxpacket);
|
|
// REG_DOEP_DMA(0) = PHYS(pep->ctrl_req_addr);
|
|
REG_DOEP_DMA(0) = PHYS(pep->xfer_buff);
|
|
}
|
|
else
|
|
{
|
|
REG_DOEP_SIZE(0) = DOEPSIZE0_SUPCNT_3 | DOEPSIZE0_PKTCNT_BIT | (pep->maxpacket);
|
|
}
|
|
}
|
|
|
|
static void dwc_calculate_fifo_size(dwc_handle *dwc)
|
|
{
|
|
/*
|
|
* TODO: we are use "Dedicated FIFO Mode with No Thresholding"
|
|
* if need thresholding, the calculation algorithm may need change
|
|
*/
|
|
|
|
/**
|
|
* 3.2.1.1 FIFO SPRAM(Single-Port RAM) mapping:
|
|
*
|
|
* 1. One common RxFIFO, used in Host and Device modes
|
|
* 2. One common Periodic TxFIFO, used in Host mode
|
|
* 3. Separate IN endpoint transmit FIFO for each Device mode IN endpoints in Dedicated Transmit FIFO
|
|
* operation (OTG_EN_DED_TX_FIFO = 1)
|
|
* 4. The FIFO SPRAM is also used for storing some register values to save gates. In Scatter/Gather DMA
|
|
* mode, four SPRAM locations (four 35-bit words) are reserved for this. In DMA and Slave modes
|
|
* (non-Scatter/Gather mode), one SPRAM location (one 35-bit word) is used for storing the DMA epnum.
|
|
*
|
|
* NOTE: when the device is operating in Scatter/Gather mode, then the last
|
|
* locations of the SPRAM store the Base Descriptor epnum, Current
|
|
* Descriptor epnum, Current Buffer epnum and status quadlet
|
|
* information for each endpoint direction (4 locations per Endpoint).
|
|
* If an endpoint is bidirectional, then 4 locations will be used for IN,
|
|
* and another 4 for OUT
|
|
* 3.2.4.4 Endpoint Information Controller
|
|
* The last locations in the SPRAM are used to hold register values.
|
|
* Device Buffer DMA Mode:
|
|
* one location per endpoint direction is used in SPRAM to store the
|
|
* DIEPDMA and DOEPDMA value. The application writes data and then reads
|
|
* it from the same location
|
|
* For example, if there are ten bidirectional endpoints, then the last
|
|
* 20 SPRAM locations are reserved for storing the DMA epnum for IN
|
|
* and OUT endpoints
|
|
* Scatter/Gather DMA Mode:
|
|
* Four locations per endpoint direction are used in SPRAM to store the
|
|
* Base Descriptor epnum, Current Descriptor epnum, Current Buffer
|
|
* Pointer and the Status Quadlet.
|
|
* The application writes data to the base descriptor epnum.
|
|
* When the application reads the location where it wrote the base
|
|
* descriptor epnum, it receives the current descriptor epnum.
|
|
* For example, if there are ten bidirectional endpoints, then the last 80
|
|
* locations are reserved for storing these values.
|
|
*
|
|
* Figure 3-13
|
|
* ________________________
|
|
* | |
|
|
* | DI/OEPDMAn Register | Depends on the value of OTG_NUM_EPS
|
|
* | and Descriptor Status | and OTG_EP_DIRn, see not above
|
|
* | values |
|
|
* -------------------------
|
|
* | TxFIFO #n Packets | DIEPTXFn
|
|
* -------------------------
|
|
* | |
|
|
* | ................ |
|
|
* | |
|
|
* -------------------------
|
|
* | TxFIFO #1 Packets | DIEPTXF1
|
|
* -------------------------
|
|
* | TxFIFO #0 Packets |
|
|
* |( up to3 SETUP Packets)| GNPTXFSIZ
|
|
* ------------------------
|
|
* | |
|
|
* | Rx Packets | GRXFSIZ
|
|
* | |
|
|
* ------------------------- epnum = 0, Rx starting epnum fixed to 0
|
|
*
|
|
*/
|
|
|
|
/**
|
|
* Rx FIFO Allocation (rx_fifo_size)
|
|
*
|
|
* RAM for SETUP Packets: 4 * n + 6 locations must be Reserved in the receive FIFO to receive up to
|
|
* n SETUP packets on control endpoints, where n is the number of control endpoints the device
|
|
* core supports.
|
|
*
|
|
* One location for Global OUT NAK
|
|
*
|
|
* Status information is written to the FIFO along with each received packet. Therefore, a minimum
|
|
* space of (Largest Packet Size / 4) + 1 must be allotted to receive packets. If a high-bandwidth
|
|
* endpoint is enabled, or multiple isochronous endpoints are enabled, then at least two (Largest
|
|
* Packet Size / 4) + 1 spaces must be allotted to receive back-to-back packets. Typically, two
|
|
* (Largest Packet Size / 4) + 1 spaces are recommended so that when the previous packet is being
|
|
* transferred to AHB, the USB can receive the subsequent packet. If AHB latency is high, you must
|
|
* allocate enough space to receive multiple packets. This is critical to prevent dropping of any
|
|
* isochronous packets.
|
|
*
|
|
* Typically, one location for each OUT endpoint is recommended.
|
|
*
|
|
* one location for eatch endpoint for EPDisable is required
|
|
*/
|
|
|
|
/**
|
|
* Tx FIFO Allocation (tx_fifo_size[n])
|
|
*
|
|
* The minimum RAM space required for each IN Endpoint Transmit FIFO is the maximum packet size
|
|
* for that particular IN endpoint.
|
|
*
|
|
* More space allocated in the transmit IN Endpoint FIFO results in a better performance on the USB
|
|
*and can hide latencies on the AHB.
|
|
*/
|
|
uint32_t rx_fifo_size, i;
|
|
uint32_t np_txfifo_size = 0;
|
|
uint32_t tx_fifo_size;
|
|
uint16_t startaddr;
|
|
uint16_t fifocfg;
|
|
const int x = 1;
|
|
|
|
/* Step1: Recevice FIFO Size Register (GRXFSIZ) */
|
|
rx_fifo_size = (4 * 1 + 6) + (2) * (1024 / 4 + 1) + (2 * dwc->hwcfg2.b.num_dev_ep) + 1;
|
|
|
|
REG_GRXFIFO_SIZE = rx_fifo_size;
|
|
|
|
/* Step2: Program device in ep transmit fifo0 size register (GNPTXFSIZ) */
|
|
np_txfifo_size |= ((1 + 1) * (64 / 4) << 16); //depth
|
|
np_txfifo_size |= rx_fifo_size; //startaddr
|
|
REG_GNPTXFIFO_SIZE = np_txfifo_size;
|
|
|
|
#define DWC_TX_FIFO_SIZE ((1 + 1) * (512 / 4))
|
|
|
|
startaddr = ((1 + 1) * (64 / 4) << 16) + rx_fifo_size;
|
|
for (i=1; i<dwc->hwcfg4.b.num_in_eps; i++)
|
|
{
|
|
tx_fifo_size |= (DWC_TX_FIFO_SIZE << 16) | startaddr;
|
|
REG_GDIEP_TXF(i) = tx_fifo_size;
|
|
startaddr += DWC_TX_FIFO_SIZE;
|
|
}
|
|
|
|
/* Configure fifo start addr and depth for endpoint information controller */
|
|
REG_GDFIFO_CFG |= startaddr << 16;
|
|
fifocfg = REG_GHW_CFG3;
|
|
fifocfg = (fifocfg >> 16);
|
|
REG_GDFIFO_CFG |= fifocfg;
|
|
/* flush tx and rx fifo */
|
|
dwc_otg_flush_rx_fifo(dwc);
|
|
|
|
dwc_otg_flush_tx_fifo(dwc,0x10);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void dwc_handle_enum_done_intr(dwc_handle *dwc)
|
|
{
|
|
dwc_ep *pep = dwc->dep[0];
|
|
|
|
/* Step1: Read the DSTS register to determine the enumeration speed */
|
|
uint32_t dsts = REG_OTG_DSTS;
|
|
uint32_t diep0ctl = REG_DIEP_CTL(0);
|
|
diep0ctl &= ~(0x3);
|
|
|
|
switch (dsts & DSTS_ENUM_SPEED_MASK)
|
|
{
|
|
case DSTS_ENUM_SPEED_HIGH:
|
|
DWC_DBG("High speed.\n");
|
|
dwc->speed = USB_SPEED_HIGH;
|
|
pep->maxpacket = 64;
|
|
diep0ctl |= DEP_EP0_MPS_64;
|
|
REG_OTG_DCFG &= ~1;
|
|
break;
|
|
case DSTS_ENUM_SPEED_FULL_30OR60:
|
|
case DSTS_ENUM_SPEED_FULL_48:
|
|
DWC_DBG("Full speed.\n");
|
|
dwc->speed = USB_SPEED_FULL;
|
|
pep->maxpacket = 64;
|
|
diep0ctl |= DEP_EP0_MPS_64;
|
|
REG_OTG_DCFG |= 1;
|
|
break;
|
|
case DSTS_ENUM_SPEED_LOW:
|
|
DWC_DBG("Low speed.\n");
|
|
dwc->speed = USB_SPEED_LOW;
|
|
pep->maxpacket = 8;
|
|
diep0ctl |= DEP_EP0_MPS_8;
|
|
break;
|
|
default:
|
|
DWC_DBG("Fault speed enumration\n");
|
|
break;
|
|
}
|
|
REG_OTG_DCTL |= DCTL_CLR_GNPINNAK;
|
|
|
|
/* Step2: Program the DIEPCTL0.MPS to set the maximum packet size */
|
|
REG_DIEP_CTL(0) = diep0ctl;
|
|
|
|
/* Step3: In Dma mode program the DOEPCTL0 register
|
|
* to enable control ouctrl_req_addrt endpoint0 to receive setup
|
|
* packet .*/
|
|
// dwc_otg_ep0_out_start(dwc);
|
|
if (dwc->is_dma == IS_INTERN_DMA)
|
|
{
|
|
rt_hw_dcache_flush_all();
|
|
DWC_DBG("0 doepsize %x ctl %x\n", REG_DOEP_SIZE(0), REG_DOEP_CTL(0));
|
|
REG_DOEP_CTL(0) |= DEP_ENA_BIT | DEP_CLEAR_NAK;
|
|
}
|
|
else
|
|
{
|
|
REG_DOEP_CTL(0) |= DEP_ENA_BIT | DEP_CLEAR_NAK;
|
|
}
|
|
|
|
/* Step4: unmask the SOF interrupt */
|
|
REG_GINT_MASK |= GINTMSK_START_FRAM;
|
|
|
|
REG_GINT_STS = GINTSTS_ENUM_DONE;
|
|
// dump_global_dwcreg();
|
|
return;
|
|
}
|
|
|
|
static void dwc_handle_early_suspend_intr(dwc_handle *dwc)
|
|
{
|
|
DWC_DBG("Handle early suspend intr.\n");
|
|
|
|
REG_GINT_STS = GINTSTS_USB_EARLYSUSPEND;
|
|
|
|
if (REG_OTG_DSTS & DSTS_ERRATIC_ERROR)
|
|
{
|
|
REG_OTG_DCTL |= DCTL_SOFT_DISCONN;
|
|
mdelay(100);
|
|
dwc_otg_core_reset(dwc);
|
|
dwc_otg_core_init(dwc,1);
|
|
dwc_otg_device_init(dwc);
|
|
dwc_calculate_fifo_size(dwc);
|
|
}
|
|
}
|
|
|
|
static void dwc_handle_suspend_intr(dwc_handle *dwc)
|
|
{
|
|
DWC_DBG("Handle suspend intr.\n");
|
|
REG_GINT_STS = GINTSTS_USB_SUSPEND;
|
|
DWC_DBG("==>%s,sleep_flag = %d\n",__func__,sleep_flag);
|
|
#if 0
|
|
if(sleep_flag)
|
|
{
|
|
while(!(REG_OTG_DSTS & 1))
|
|
{
|
|
printf("REG_OTG_DSTS is 0x%x\n",REG_OTG_DSTS);
|
|
}
|
|
sleep_flag = 0;
|
|
enable_irq(IRQ_OTG);
|
|
jz_pm_sleep();
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static void dwc_handle_start_frame_intr(dwc_handle *dwc)
|
|
{
|
|
REG_GINT_STS = GINTSTS_START_FRAM;
|
|
}
|
|
|
|
static void dwc_handle_reset_intr(dwc_handle *dwc)
|
|
{
|
|
int i;
|
|
|
|
/* Step1: NAK OUT ep */
|
|
for (i=0; i<dwc->hwcfg2.b.num_dev_ep; i++)
|
|
{
|
|
REG_DOEP_CTL(i) |= DEP_SET_NAK;
|
|
}
|
|
|
|
/* Step2: unmask the following interrupt bits */
|
|
REG_DAINT_MASK = 0;
|
|
REG_DOEP_MASK = 0;
|
|
REG_DIEP_MASK = 0;
|
|
|
|
REG_DAINT_MASK |= (1 << 0) | (1 << 16); //inep0 outep0
|
|
REG_DOEP_MASK |= DEP_XFER_COMP | DEP_SETUP_PHASE_DONE | DEP_AHB_ERR; // xfercompl setupdone
|
|
REG_DIEP_MASK |= DEP_XFER_COMP | DEP_TIME_OUT | DEP_AHB_ERR; // xfercompl ahberr timeout
|
|
|
|
dwc->dep[0]->ep_state = EP_SETUP;
|
|
|
|
/* Step3: Device initalization */
|
|
dwc_otg_device_init(dwc);
|
|
|
|
/* Step4: Set up the data fifo ram for each of the fifo */
|
|
//dwc_calculate_fifo_size();
|
|
|
|
/* Step5: Reset Device Address */
|
|
REG_OTG_DCFG &= (~DCFG_DEV_ADDR_MASK);
|
|
|
|
/* Step6: setup EP0 to receive SETUP packets */
|
|
dwc_otg_ep0_out_start(dwc);
|
|
if (dwc->is_dma == IS_INTERN_DMA)
|
|
{
|
|
rt_hw_dcache_flush_all();
|
|
REG_DOEP_CTL(0) |= DEP_ENA_BIT | DEP_CLEAR_NAK;
|
|
}
|
|
else
|
|
{
|
|
REG_DOEP_CTL(0) |= DEP_ENA_BIT | DEP_CLEAR_NAK;
|
|
}
|
|
dwc_disable_in_ep(dwc,0);
|
|
|
|
REG_GINT_STS = GINTSTS_USB_RESET;
|
|
|
|
return;
|
|
}
|
|
|
|
void dwc_handle_rxfifo_nempty(dwc_handle *dwc)
|
|
{
|
|
dwc_ep *pep;
|
|
uint32_t *setup_buf;
|
|
uint32_t count;
|
|
uint32_t rxsts_pop = REG_GRXSTS_POP;
|
|
uint8_t epnum = (rxsts_pop & 0xf);
|
|
|
|
switch (rxsts_pop & GRXSTSP_PKSTS_MASK)
|
|
{
|
|
case GRXSTSP_PKSTS_GOUT_NAK:
|
|
DWC_DBG("GRXSTSP_PKSTS_GOUT_NAK.\n");
|
|
break;
|
|
case GRXSTSP_PKSTS_GOUT_RECV:
|
|
DWC_DBG("GRXSTSP_PKSTS_GOUT_RECV. - ");
|
|
|
|
count = (rxsts_pop & GRXSTSP_BYTE_CNT_MASK) >> GRXSTSP_BYTE_CNT_BIT;
|
|
if (count)
|
|
{
|
|
DWC_DBG("count:%d\n", count);
|
|
dwc_read_ep_packet(dwc,epnum, count);
|
|
}
|
|
|
|
break;
|
|
case GRXSTSP_PKSTS_TX_COMP:
|
|
DWC_DBG("GRXSTSP_PKSTS_TX_COMP.\n");
|
|
break;
|
|
case GRXSTSP_PKSTS_SETUP_COMP:
|
|
DWC_DBG("GRXSTSP_PKSTS_SETUP_COMP.\n");
|
|
break;
|
|
case GRXSTSP_PKSTS_SETUP_RECV:
|
|
DWC_DBG("GRXSTSP_PKSTS_SETUP_RECV. - ");
|
|
// setup_packet[0] = REG_EP_FIFO(0);
|
|
// setup_packet[1] = REG_EP_FIFO(0);
|
|
// DWC_DBG("%x %x\n", setup_packet[0], setup_packet[1]);
|
|
((uint8_t *)dwc->dep[0]->xfer_buff)[0] = REG_EP_FIFO(0);
|
|
((uint8_t *)dwc->dep[0]->xfer_buff)[1] = REG_EP_FIFO(1);
|
|
DWC_DBG("%x %x\n", ((uint8_t *)dwc->dep[0]->xfer_buff)[0], ((uint8_t *)dwc->dep[0]->xfer_buff)[1]);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
REG_GINT_STS = GINTSTS_RXFIFO_NEMPTY;
|
|
}
|
|
|
|
|
|
void dwc_ep0_in_intr(dwc_handle *dwc, uint8_t epnum)
|
|
{
|
|
uint32_t updated_size;
|
|
uint32_t dma_addr, dma_len;
|
|
uint8_t *ptr;
|
|
uint32_t intr = REG_DIEP_INT(epnum & 0x0F);
|
|
dwc_ep *pep ;
|
|
|
|
DWC_DBG("ep0 in intr:%x\n", intr);
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_IN_OFS + epnum];
|
|
udelay(1);
|
|
|
|
/* When the transfer size if 0 and the packet count is 0,
|
|
* the transfer complete interrupt for the endpoint is generated
|
|
* and the endpoint enable is cleared */
|
|
if (intr & DEP_XFER_COMP)
|
|
{
|
|
DWC_DBG("XFER_COMP\n");
|
|
REG_DIEP_INT(epnum) = DEP_XFER_COMP; // clear int
|
|
if (dwc->is_dma == IS_SLAVE_MODE)
|
|
REG_DIEP_EMPMSK &= ~(1 << epnum);
|
|
|
|
updated_size = (REG_DIEP_SIZE(epnum) & 0x7f);
|
|
pep->xfer_count = pep->xfer_len - updated_size; // number of bytes transfered
|
|
DWC_DBG("in xfer_count:%d xfer_len:%d updated_size:%d\n", pep->xfer_count, pep->xfer_len, updated_size);
|
|
|
|
if (pep->xfer_count != pep->xfer_len)
|
|
{
|
|
DWC_DBG("in xfer_count:%d xfer_len:%d updated_size:%d\n", pep->xfer_count, pep->xfer_len, updated_size);
|
|
pep->xfer_len -= pep->xfer_count;
|
|
ptr = (uint8_t *)pep->xfer_buff + pep->xfer_count;
|
|
HW_SendPKT(dwc, 0, ptr, pep->xfer_len);
|
|
return;
|
|
}
|
|
|
|
DWC_DBG("pep->ep_state = %s\n",ep0_state_string[pep->ep_state]);
|
|
|
|
switch(pep->ep_state)
|
|
{
|
|
case EP_DATA:
|
|
#if 1
|
|
/* 3 Stage */
|
|
if(pep->xfer_len == pep->maxpacket)
|
|
{
|
|
x1000_usbd_event_cb(0, USB_EVT_IN, RT_NULL);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
pep->ep_state = EP_STATUS;
|
|
dwc_handle_ep_data_out_phase(dwc,0);
|
|
}
|
|
break;
|
|
case EP_STATUS:
|
|
pep->ep_state = EP_SETUP;
|
|
dwc_handle_ep_data_out_phase(dwc,0);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (dwc->is_dma == IS_SLAVE_MODE)
|
|
{
|
|
if ((intr & DEP_TXFIFO_EMPTY) && (REG_DIEP_EMPMSK & (1 << epnum)))
|
|
{
|
|
if (pep->xfer_len)
|
|
{
|
|
dwc_write_ep_packet(dwc,epnum);
|
|
}
|
|
REG_DIEP_INT(epnum) = DEP_TXFIFO_EMPTY;
|
|
}
|
|
}
|
|
|
|
if (intr & DEP_AHB_ERR)
|
|
{
|
|
DWC_DBG("1 AHB ERR\n");
|
|
REG_DIEP_INT(epnum) = DEP_AHB_ERR;
|
|
}
|
|
|
|
|
|
if (intr & DEP_TIME_OUT)
|
|
{
|
|
DWC_DBG("IN TIME_OUT.\n");
|
|
REG_DIEP_INT(epnum) = DEP_TIME_OUT;
|
|
}
|
|
}
|
|
|
|
void dwc_epn_in_intr(dwc_handle *dwc, uint8_t epnum)
|
|
{
|
|
uint32_t intr = REG_DIEP_INT(epnum & 0x0F);
|
|
uint32_t updated_size;
|
|
|
|
/* When the transfer size if 0 and the packet count is 0,
|
|
* the transfer complete interrupt for the endpoint is generated
|
|
* and the endpoint enable is cleared */
|
|
dwc_ep *pep;
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_IN_OFS + epnum];
|
|
|
|
if (intr & DEP_XFER_COMP)
|
|
{
|
|
DWC_DBG("1 IN XFER_COMP. %x\n", REG_DIEP_SIZE(epnum));
|
|
REG_DIEP_INT(epnum) = DEP_XFER_COMP;
|
|
REG_DIEP_CTL(epnum) |= DEP_SET_NAK;
|
|
if (pep->ep_state == EP_TRANSFERING)
|
|
{
|
|
if (dwc->is_dma == IS_SLAVE_MODE)
|
|
REG_DIEP_EMPMSK &= ~(1 << epnum);
|
|
updated_size = (REG_DIEP_SIZE(epnum) & 0x7ffff);
|
|
pep->xfer_count = pep->xfer_len - updated_size;
|
|
pep->ep_state = EP_TRANSFERED;
|
|
|
|
// rt_kprintf("updated_size = %d,xfer_len = %d,xfer_count = %d\n",updated_size,pep->xfer_len, pep->xfer_count);
|
|
// BusNotify(arg, UDC_PROTAL_SEND_FINISH, NULL, 0);
|
|
x1000_usbd_event_cb(epnum,USB_EVT_IN,0);
|
|
}
|
|
}
|
|
if (dwc->is_dma == IS_SLAVE_MODE)
|
|
{
|
|
if ((intr & DEP_TXFIFO_EMPTY) && (REG_DIEP_EMPMSK & (1 << epnum)))
|
|
{
|
|
REG_DIEP_EMPMSK &= ~(1 << epnum);
|
|
// DWC_DBG("TX FIFO EMPTY intr.\n");
|
|
if (pep->xfer_len)
|
|
{
|
|
dwc_write_ep_packet(dwc,epnum);
|
|
}
|
|
REG_DIEP_INT(epnum) = DEP_TXFIFO_EMPTY;
|
|
}
|
|
}
|
|
if (intr & DEP_AHB_ERR)
|
|
{
|
|
DWC_DBG("1 AHB ERR\n");
|
|
REG_DIEP_INT(epnum) = DEP_AHB_ERR;
|
|
}
|
|
|
|
if (intr & DEP_TIME_OUT)
|
|
{
|
|
DWC_DBG("IN TIME_OUT.\n");
|
|
REG_DIEP_INT(epnum) = DEP_TIME_OUT;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* ep0 control transfer:
|
|
* 3 Stage:
|
|
* SetupPhase-------->IN DataPhase ---------> OUT StatusPhase
|
|
* Or 2 Stage:
|
|
* SetupPhase-------->IN StatusPhase
|
|
* */
|
|
|
|
typedef struct {
|
|
u8 bmRequestType;
|
|
u8 bRequest;
|
|
u16 wValue;
|
|
u16 wIndex;
|
|
u16 wLength;
|
|
} __attribute__ ((packed)) dwc_DeviceRequest;
|
|
|
|
int dwc_ep0_out_intr(dwc_handle *dwc, uint8_t epnum)
|
|
{
|
|
uint32_t intr, doep0size, dma_addr, dma_len;
|
|
uint8_t rem_supcnt, xfersize;
|
|
|
|
dwc_ep *pep = RT_NULL;
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_OUT_OFS + epnum];
|
|
|
|
intr = REG_DOEP_INT(epnum);
|
|
// printf("ep0 out intr:%x\n", intr);
|
|
udelay(1);
|
|
|
|
/* comp intrerrupt indeicates completion of the status out phase */
|
|
if (intr & DEP_XFER_COMP)
|
|
{
|
|
REG_DOEP_INT(epnum) = DEP_XFER_COMP;
|
|
|
|
DWC_DBG("pep->ep_state = %s\n",ep0_state_string[pep->ep_state]);
|
|
|
|
if (pep->ep_state == EP_STATUS)
|
|
{
|
|
pep->ep_state = EP_SETUP;
|
|
dwc_handle_ep_data_out_phase(dwc,0);
|
|
}
|
|
else if (pep->ep_state == EP_DATA)
|
|
{
|
|
DWC_DBG("*** EP0 DATA ***\n");
|
|
pep->xfer_count = pep->maxpacket - (REG_DOEP_SIZE(epnum) & 0x7ffff);
|
|
DWC_DBG("pep->xfer_count = %d\n",pep->xfer_count);
|
|
x1000_usbd_event_cb(0,USB_EVT_OUT,0);
|
|
}
|
|
else if (!(intr & (DEP_SETUP_PHASE_DONE | (1 << 15))))
|
|
{
|
|
DWC_DBG("error\n");
|
|
pep->ep_state = EP_SETUP;
|
|
dwc_handle_ep_data_out_phase(dwc, epnum);
|
|
}
|
|
else if (pep->ep_state != EP_SETUP)
|
|
{
|
|
DWC_DBG("ep0 state mismatch\n");
|
|
}
|
|
}
|
|
|
|
//IN Token
|
|
if (intr & DEP_INTOKEN_EPMISATCH)
|
|
{
|
|
if (pep->ep_state == EP_DATA)
|
|
{
|
|
pep->ep_state = EP_STATUS;
|
|
dwc_handle_ep_status_in_phase(dwc, epnum);
|
|
}
|
|
REG_DOEP_INT(epnum) = DEP_INTOKEN_EPMISATCH;
|
|
}
|
|
|
|
if (intr & DEP_AHB_ERR)
|
|
{
|
|
DWC_DBG("AHB ERR\n");
|
|
REG_DOEP_INT(0) = DEP_AHB_ERR;
|
|
}
|
|
|
|
if (intr & DEP_NAK_INT)
|
|
{
|
|
REG_DOEP_INT(0) = DEP_NAK_INT;
|
|
}
|
|
|
|
if (intr & (DEP_SETUP_PHASE_DONE | (1 << 15)))
|
|
{
|
|
DWC_DBG("SETUP_PHASE_DONE.\n");
|
|
|
|
/* read the DOEPTSIZn to determine the number of setup packets
|
|
* recevied and process the last recevied setup packet */
|
|
REG_DOEP_INT(epnum) = DEP_SETUP_PHASE_DONE | (1 << 15);
|
|
|
|
doep0size = REG_DOEP_SIZE(epnum);
|
|
xfersize = doep0size & 0x7ffff;
|
|
rem_supcnt = (doep0size & (0x3 << 29)) >> 29;
|
|
DWC_DBG("xfersize = %d,rem_supcnt = %d\n",xfersize,rem_supcnt);
|
|
|
|
if (intr & DEP_B2B_SETUP_RECV)
|
|
{
|
|
DWC_DBG("back to back setup recevie\n");
|
|
}
|
|
else
|
|
{
|
|
/* Read out the last packet from the rxfifo */
|
|
// rt_hw_dcache_invalidate_range((uint32_t)(pep->ctrl_req_addr), sizeof(dwc_DeviceRequest));
|
|
rt_hw_dcache_invalidate_range((uint32_t)(pep->xfer_buff), sizeof(dwc_DeviceRequest));
|
|
#if 0
|
|
{
|
|
dwc_DeviceRequest* device_req = (dwc_DeviceRequest *)(pep->ctrl_req_addr);
|
|
rt_kprintf("\n-------------\n");
|
|
rt_kprintf("bRequest: %x\n", device_req->bRequest);
|
|
rt_kprintf("bRequestType: %x\n", device_req->bmRequestType);
|
|
rt_kprintf("wIndex: %x\n", device_req->wIndex);
|
|
rt_kprintf("wLength: %x\n", device_req->wLength);
|
|
rt_kprintf("wValue: %x\n", device_req->wValue);
|
|
rt_kprintf("-------------\n");
|
|
}
|
|
#endif
|
|
|
|
/* At the end of the Setup stage, the appliaction must reporgram the
|
|
* DOEPTSIZn.SUPCnt field to 3 receive the next SETUP packet */
|
|
if (pep->ep_state == EP_SETUP)
|
|
{
|
|
if (dwc->is_dma == 2)
|
|
{
|
|
//printf("1 doepsize %x ctl %x\n", REG_DOEP_SIZE(0), REG_DOEP_CTL(0));
|
|
REG_DOEP_SIZE(epnum) = DOEPSIZE0_SUPCNT_3 | DOEPSIZE0_PKTCNT_BIT | (pep->maxpacket);
|
|
// REG_DOEP_DMA(epnum) = PHYS(pep->ctrl_req_addr);
|
|
REG_DOEP_DMA(epnum) = PHYS(pep->xfer_buff);
|
|
}
|
|
else
|
|
{
|
|
REG_DOEP_SIZE(epnum) = DOEPSIZE0_SUPCNT_3 | DOEPSIZE0_PKTCNT_BIT | (pep->maxpacket);
|
|
}
|
|
}
|
|
|
|
|
|
/* Setup Finish */
|
|
pep->xfer_count = sizeof(dwc_DeviceRequest);
|
|
// pep->xfer_buff = (void *) (pep->ctrl_req_addr);
|
|
// x1000_usbd_event_cb(0, USB_EVT_SETUP, (void *) (pep->ctrl_req_addr));
|
|
x1000_usbd_event_cb(0, USB_EVT_SETUP, pep->xfer_buff);
|
|
|
|
REG_DOEP_CTL(epnum) |= DEP_DISENA_BIT;
|
|
// REG_DOEP_CTL(epnum) |= DEP_SET_NAK;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int dwc_epn_out_intr(dwc_handle *dwc, uint8_t epnum)
|
|
{
|
|
uint32_t intr, updated_size;
|
|
dwc_ep *pep ;
|
|
|
|
epnum &= DWC_EPNO_MASK;
|
|
pep = dwc->dep[DWC_EP_OUT_OFS + epnum];
|
|
udelay(1);
|
|
DWC_DBG("ep%d out_intr\n",epnum);
|
|
|
|
intr = REG_DOEP_INT(epnum);
|
|
if (intr & DEP_XFER_COMP)
|
|
{
|
|
REG_DOEP_INT(epnum) = DEP_XFER_COMP;
|
|
updated_size = REG_DOEP_SIZE(epnum) & 0x7ffff;
|
|
pep->xfer_count = pep->xfer_len - updated_size;
|
|
|
|
DWC_DBG("xfer_count = %d\n",pep->xfer_count);
|
|
// BusNotify((uint32_t)arg,UDC_PROTAL_RECEIVE_FINISH, (uint8_t *)pep->xfer_buff, pep->xfer_count);
|
|
x1000_usbd_event_cb(epnum,USB_EVT_OUT,0);
|
|
|
|
#if 0
|
|
pep->xfer_len = pep->maxpacket; /* number of bytes to transfer */
|
|
pep->xfer_count = 0; /* number of bytes transfered */
|
|
// pep->xfer_buff = pep->xfer_buff; /* pointer to transfer buffer */
|
|
dwc_handle_ep_data_out_phase(dwc, epnum);
|
|
DWC_DBG("REG_DOEP_SIZE: %x \n", REG_DOEP_SIZE(epnum));
|
|
#endif
|
|
}
|
|
|
|
if (intr & DEP_AHB_ERR)
|
|
{
|
|
DWC_DBG("1 AHB ERR\n");
|
|
REG_DOEP_INT(epnum) = DEP_AHB_ERR;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void dwc_handle_inep_intr(dwc_handle *dwc)
|
|
{
|
|
uint32_t ep_intr;
|
|
uint8_t epnum = 0;
|
|
|
|
ep_intr = (REG_OTG_DAINT & 0xffff);
|
|
DWC_DBG("\n\nEp IN %x - \n", ep_intr);
|
|
|
|
while (ep_intr)
|
|
{
|
|
if (ep_intr & 0x01)
|
|
{
|
|
if (epnum == 0)
|
|
{
|
|
dwc_ep0_in_intr(dwc, epnum);
|
|
}
|
|
else
|
|
{
|
|
dwc_epn_in_intr(dwc, epnum);
|
|
}
|
|
}
|
|
epnum++;
|
|
ep_intr >>= 1;
|
|
}
|
|
REG_GINT_STS = GINTSTS_IEP_INTR;
|
|
return ;
|
|
}
|
|
|
|
static void dwc_handle_outep_intr(dwc_handle *dwc)
|
|
{
|
|
uint32_t ep_intr, epnum = 0;
|
|
ep_intr = (REG_OTG_DAINT & 0xffff0000) >> 16;
|
|
DWC_DBG("\n\nEp OUT %x - \n", ep_intr);
|
|
while (ep_intr)
|
|
{
|
|
if (ep_intr & 0x01)
|
|
{
|
|
if (epnum == 0)
|
|
{
|
|
dwc_ep0_out_intr(dwc, 0);
|
|
}
|
|
else
|
|
{
|
|
dwc_epn_out_intr(dwc, epnum);
|
|
}
|
|
}
|
|
epnum ++;
|
|
ep_intr >>= 1;
|
|
}
|
|
REG_GINT_STS = GINTSTS_OEP_INTR;
|
|
}
|
|
|
|
static void dwc_otg_intr(dwc_handle *dwc)
|
|
{
|
|
REG_GINT_STS = GINTSTS_OTG_INTR;
|
|
}
|
|
|
|
void dwc_common_intr(dwc_handle *dwc,uint32_t intsts)
|
|
{
|
|
if (intsts & GINTSTS_USB_EARLYSUSPEND)
|
|
{
|
|
dwc_handle_early_suspend_intr(dwc);
|
|
}
|
|
|
|
if (intsts & GINTSTS_USB_SUSPEND)
|
|
{
|
|
dwc_handle_suspend_intr(dwc);
|
|
}
|
|
|
|
if (intsts & GINTSTS_USB_RESET)
|
|
{
|
|
dwc_handle_reset_intr(dwc);
|
|
}
|
|
|
|
if (intsts & GINTSTS_ENUM_DONE)
|
|
{
|
|
dwc_handle_enum_done_intr(dwc);
|
|
}
|
|
|
|
if (intsts & GINTSTS_START_FRAM)
|
|
{
|
|
dwc_handle_start_frame_intr(dwc);
|
|
}
|
|
}
|
|
|
|
void dwc_handle_resume_intr(dwc_handle *dwc)
|
|
{
|
|
DWC_DBG("Handle resume intr.\n");
|
|
REG_GINT_STS = GINTSTS_RSUME_DETE;
|
|
|
|
// dwc_otg_phy_suspend(0);
|
|
}
|
|
|
|
static void dwc_irq_handler(int vector,void *arg)
|
|
{
|
|
dwc_handle *dwc = (dwc_handle *)arg;
|
|
|
|
RT_ASSERT(dwc != RT_NULL);
|
|
|
|
rt_hw_interrupt_mask(IRQ_OTG);
|
|
|
|
rt_sem_release(dwc->isr_sem);
|
|
}
|
|
|
|
static void dwc_otg_core_reset(dwc_handle *dwc)
|
|
{
|
|
uint32_t greset = 0;
|
|
uint32_t cnt = 0;
|
|
|
|
REG_GRST_CTL |= RSTCTL_CORE_RST;
|
|
do
|
|
{
|
|
greset = REG_GRST_CTL;
|
|
if (cnt++ > 100000)
|
|
{
|
|
DWC_DBG("GRESET wait reset timeout.\n");
|
|
return;
|
|
}
|
|
udelay(10);
|
|
} while (greset & RSTCTL_CORE_RST);
|
|
|
|
cnt = 0;
|
|
|
|
do
|
|
{
|
|
udelay(10);
|
|
greset = REG_GRST_CTL;
|
|
if (cnt++ > 100000)
|
|
{
|
|
DWC_DBG("GRESET wait IDLE timeout.\n");
|
|
return;
|
|
}
|
|
} while ((greset & RSTCTL_AHB_IDLE) == 0);
|
|
|
|
/* wait for 3 phy clocks */
|
|
udelay(100);
|
|
}
|
|
|
|
static int dwc_otg_phy_is_suspend(void)
|
|
{
|
|
return (!(cpm_test_bit(7, CPM_OPCR)));
|
|
}
|
|
|
|
static void dwc_otg_phy_suspend(int suspend)
|
|
{
|
|
if (!suspend && dwc_otg_phy_is_suspend())
|
|
{
|
|
DWC_DBG("EN PHY\n");
|
|
cpm_set_bit(7, CPM_OPCR);
|
|
udelay(45);
|
|
}
|
|
else if (suspend && !dwc_otg_phy_is_suspend())
|
|
{
|
|
DWC_DBG("DIS PHY\n");
|
|
cpm_clear_bit(7, CPM_OPCR);
|
|
udelay(5);
|
|
}
|
|
}
|
|
|
|
static void dwc_otg_device_init(dwc_handle *dwc)
|
|
{
|
|
uint32_t dcfg = 0;
|
|
uint32_t pcgcctl;
|
|
uint32_t rx_fifo_size;
|
|
|
|
/* Restart the phy clock */
|
|
if (REG_PCGC_CTL & 0x1)
|
|
{
|
|
DWC_DBG("<<<<<< pcgcctl %x >>>>>\n", REG_PCGC_CTL);
|
|
REG_PCGC_CTL &= ~(0x1 | (1 << 2) | (1 << 3));
|
|
}
|
|
|
|
/* In dma mode GINTMSK_NPTXFIFO_EMPTY , GINTMSK_RXFIFO_NEMPTY must be masked*/
|
|
if (dwc->is_dma == IS_INTERN_DMA)
|
|
{
|
|
if (REG_GINT_MASK & (GINTMSK_NPTXFIFO_EMPTY | GINTMSK_RXFIFO_NEMPTY))
|
|
{
|
|
REG_GINT_MASK &= ~(GINTMSK_NPTXFIFO_EMPTY | GINTMSK_RXFIFO_NEMPTY);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
REG_GINT_MASK |= (GINTMSK_NPTXFIFO_EMPTY | GINTMSK_RXFIFO_NEMPTY);
|
|
}
|
|
|
|
/* Program the DCFG register */
|
|
if (dwc->hwcfg4.b.desc_dma)
|
|
{
|
|
dcfg |= DCFG_DEV_DESC_DMA;
|
|
}
|
|
|
|
#if DWC_FORCE_SPEED_FULL
|
|
REG_OTG_DCFG |= 1; //dma buffer mode full speed
|
|
#else
|
|
REG_OTG_DCFG &= ~3; //dma buffer mode HIGH speed
|
|
#endif
|
|
|
|
/* Clear the DCTL.SftDiscon bit the core issues aconnect after ths bit is cleared */
|
|
REG_OTG_DCTL &= ~DCTL_SOFT_DISCONN;
|
|
|
|
REG_GINT_STS = 0xffffffff;
|
|
/* Program the GINTMSK */
|
|
REG_GINT_MASK |= GINTMSK_IEP_INTR | GINTMSK_OEP_INTR |GINTMSK_USB_RESET | GINTMSK_ENUM_DONE |GINTMSK_USB_EARLYSUSPEND | GINTMSK_USB_SUSPEND | (1 << 31);
|
|
}
|
|
|
|
static void dwc_otg_core_init(dwc_handle *dwc,uint8_t dma_enable)
|
|
{
|
|
uint32_t ahbcfg = 0, gusbcfg = 0, curmod = 0, tmp;
|
|
uint8_t arch;
|
|
|
|
DWC_DBG("Core Init...\n");
|
|
/* Step1: Read the GHWCFG1,2,3,4 to find the configuration parameters selected for DWC_otg core */
|
|
dwc->hwcfg1.d32 = REG_GHW_CFG1;
|
|
dwc->hwcfg2.d32 = REG_GHW_CFG2;
|
|
dwc->hwcfg3.d32 = REG_GHW_CFG3;
|
|
dwc->hwcfg4.d32 = REG_GHW_CFG4;
|
|
|
|
DWC_DBG("cfg1:%x 2:%x 3:%x 4:%x\n", dwc->hwcfg1, dwc->hwcfg2, dwc->hwcfg3, dwc->hwcfg4);
|
|
DWC_DBG("cfg2->arch %x\n", dwc->hwcfg2.b.architecture);
|
|
arch = dwc->hwcfg2.b.architecture;
|
|
switch (arch)
|
|
{
|
|
case IS_SLAVE_MODE:
|
|
dwc->is_dma = IS_SLAVE_MODE;
|
|
break;
|
|
case IS_EXTERN_DMA:
|
|
dwc->is_dma = IS_EXTERN_DMA;
|
|
break;
|
|
case IS_INTERN_DMA:
|
|
dwc->is_dma = IS_INTERN_DMA;
|
|
break;
|
|
}
|
|
/* Step2: Program the GAHBCFG register */
|
|
|
|
/* DMA Mode bit and Burst Length */
|
|
if (dwc->is_dma == IS_EXTERN_DMA)
|
|
{
|
|
DWC_DBG("DWC IS_EXTERN_DMA\n");
|
|
ahbcfg |= AHBCFG_DMA_ENA;
|
|
}
|
|
else if (dwc->is_dma == IS_INTERN_DMA)
|
|
{
|
|
if (dma_enable)
|
|
{
|
|
DWC_DBG("DWC IS_INTERN_DMA\n");
|
|
ahbcfg |= AHBCFG_DMA_ENA | (DWC_GAHBCFG_INT_DMA_BURST_INCR16 << 1);
|
|
}
|
|
else
|
|
{
|
|
ahbcfg |= AHBCFG_TXFE_LVL;
|
|
dwc->is_dma = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
DWC_DBG("DWC IS_SLAVE_MODE\n");
|
|
}
|
|
|
|
/* Step3: Program the GINTMSK register */
|
|
REG_GINT_MASK = 0;
|
|
|
|
/* Step4: Program the GUSBCFG register */
|
|
gusbcfg = REG_GUSB_CFG;
|
|
|
|
gusbcfg &= ~((1 << 4) | (1 << 6) | (1 << 8) | (1 << 9));
|
|
REG_GUSB_CFG = gusbcfg; // HNP SRP not support and select UTMI+
|
|
dwc_otg_select_phy_width(dwc);
|
|
|
|
dwc_otg_core_reset(dwc);
|
|
|
|
/* Global Interrupt Mask bit = 1 */
|
|
ahbcfg |= AHBCFG_GLOBLE_INTRMASK;
|
|
REG_GAHB_CFG = ahbcfg;
|
|
|
|
/* Step5: The software must unmask OTG Interrupt Mask bit ,
|
|
* MOde mismatch interrupt Mask bit in the GINTMSK */
|
|
REG_GINT_MASK |= (GINTMSK_MODE_MISMATCH | GINTMSK_OTG_INTR);
|
|
}
|
|
|
|
|
|
int dwc_set_config(dwc_handle *dwc)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int dwc_set_ep_stall(dwc_handle *dwc,uint8_t epnum)
|
|
{
|
|
depctl_data_t depctl;
|
|
daint_data_t daintmsk;
|
|
dwc_ep *pep = RT_NULL;
|
|
|
|
if(epnum & USB_DIR_IN)
|
|
{
|
|
pep = dwc->dep[epnum & 0x0F + DWC_EP_IN_OFS];
|
|
}
|
|
else
|
|
{
|
|
pep = dwc->dep[epnum & 0x0F + DWC_EP_OUT_OFS];
|
|
}
|
|
epnum &= DWC_EPNO_MASK;
|
|
|
|
if (pep->is_in)
|
|
{
|
|
depctl.d32 = REG_DIEP_CTL(epnum);
|
|
depctl.b.stall = 1;
|
|
REG_DIEP_CTL(epnum) = depctl.d32;
|
|
}
|
|
else
|
|
{
|
|
depctl.d32 = REG_DOEP_CTL(epnum);
|
|
depctl.b.stall = 1;
|
|
REG_DOEP_CTL(epnum) = depctl.d32;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int dwc_clr_ep_stall(dwc_handle *dwc,uint8_t epnum)
|
|
{
|
|
depctl_data_t depctl;
|
|
daint_data_t daintmsk;
|
|
dwc_ep *pep = RT_NULL;
|
|
if(epnum & USB_DIR_IN)
|
|
{
|
|
pep = dwc->dep[epnum & 0x0F + DWC_EP_IN_OFS];
|
|
}
|
|
else
|
|
{
|
|
pep = dwc->dep[epnum & 0x0F + DWC_EP_OUT_OFS];
|
|
}
|
|
epnum &= DWC_EPNO_MASK;
|
|
|
|
if (pep->is_in)
|
|
{
|
|
depctl.d32 = REG_DIEP_CTL(epnum);
|
|
depctl.b.stall = 0;
|
|
REG_DIEP_CTL(epnum) = depctl.d32;
|
|
}
|
|
else
|
|
{
|
|
depctl.d32 = REG_DOEP_CTL(epnum);
|
|
depctl.b.stall = 0;
|
|
REG_DOEP_CTL(epnum) = depctl.d32;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
int dwc_ep_disable(dwc_handle *dwc,uint8_t epnum)
|
|
{
|
|
depctl_data_t depctl;
|
|
daint_data_t daintmsk;
|
|
dwc_ep *pep = RT_NULL;
|
|
|
|
DWC_DBG("%s epnum = %02x \n",epnum);
|
|
|
|
if(epnum & USB_DIR_IN)
|
|
{
|
|
pep = dwc->dep[epnum & 0x0F + DWC_EP_IN_OFS];
|
|
}
|
|
else
|
|
{
|
|
pep = dwc->dep[epnum & 0x0F + DWC_EP_OUT_OFS];
|
|
}
|
|
epnum &= DWC_EPNO_MASK;
|
|
|
|
/* EP0 can not deactivate! */
|
|
if (epnum == 0)
|
|
return -1;
|
|
|
|
daintmsk.d32 = REG_DAINT_MASK;
|
|
if (pep->is_in)
|
|
{
|
|
depctl.d32 = REG_DIEP_CTL(epnum);
|
|
daintmsk.ep.in &= ~(1 << epnum);
|
|
}
|
|
else
|
|
{
|
|
depctl.d32 = REG_DOEP_CTL(epnum);
|
|
daintmsk.ep.out &= ~(1 << epnum);
|
|
}
|
|
if (!depctl.b.usbactep)
|
|
{
|
|
DWC_DBG("EP %d already deactivated\n", pep->num);
|
|
return 0;
|
|
}
|
|
|
|
depctl.b.usbactep = 0;
|
|
if (pep->is_in)
|
|
{
|
|
REG_DIEP_CTL(epnum) = depctl.d32;
|
|
}
|
|
else
|
|
{
|
|
REG_DOEP_CTL(epnum) = depctl.d32;
|
|
}
|
|
|
|
/* mask EP interrupts */
|
|
REG_DAINT_MASK = daintmsk.d32;
|
|
|
|
if(pep->is_in)
|
|
{/* Disable IN-EP */
|
|
|
|
}
|
|
else
|
|
{
|
|
/* Disable IN-EP */
|
|
}
|
|
|
|
DWC_DBG("EP %d deactivated\n", pep->num);
|
|
return 0;
|
|
}
|
|
|
|
|
|
static void x1000_usb_phy_switch(dwc_handle *dwc,uint8_t is_on)
|
|
{
|
|
uint32_t value;
|
|
if (is_on)
|
|
{
|
|
value = REG_CPM_OPCR;
|
|
REG_CPM_OPCR |= OPCR_OTGPHY0_ENABLE;
|
|
mdelay(1);
|
|
}
|
|
else
|
|
{
|
|
value = REG_CPM_OPCR;
|
|
REG_CPM_OPCR &= ~OPCR_OTGPHY0_ENABLE;
|
|
|
|
mdelay(1);
|
|
}
|
|
}
|
|
|
|
static void x1000_usb_set_device_only_mode(dwc_handle *dwc)
|
|
{
|
|
REG_CPM_USBPCR &= ~USBPCR_USB_MODE;
|
|
REG_CPM_USBPCR &= ~USBPCR_OTG_DISABLE;
|
|
}
|
|
|
|
static void x1000_usb_phy_init(dwc_handle *dwc)
|
|
{
|
|
REG_CPM_USBPCR |= USBPCR_POR;
|
|
mdelay(1);
|
|
|
|
REG_CPM_USBPCR &= ~USBPCR_POR;
|
|
mdelay(1);
|
|
}
|
|
|
|
|
|
|
|
/* usb device init */
|
|
static void dwc_gadget_init(dwc_handle *dwc)
|
|
{
|
|
uint32_t curmod;
|
|
int err;
|
|
// REG_CPM_CLKGR0 &= ~(1 << 2);
|
|
|
|
rt_hw_interrupt_mask(IRQ_OTG);
|
|
|
|
/* usb_cpm_init(); */
|
|
{
|
|
uint32_t ref_clk_div = 24 / 24; //24 / 24;
|
|
uint32_t usbpcr1;
|
|
|
|
/* select dwc otg */
|
|
REG_CPM_USBPCR1 |= USBPCR1_USB_SEL;
|
|
|
|
/* select utmi data bus width of port0 to 16bit/30M */
|
|
REG_CPM_USBPCR1 |= USBPCR1_WORD_IF0;
|
|
|
|
usbpcr1 = REG_CPM_USBPCR1;
|
|
usbpcr1 &= ~(0x3 << 24);
|
|
usbpcr1 |= (ref_clk_div << 24);
|
|
REG_CPM_USBPCR1 = usbpcr1;
|
|
|
|
/* fil */
|
|
REG_CPM_USBVBFIL = 0;
|
|
|
|
/* rdt */
|
|
REG_CPM_USBRDT = 0x96;
|
|
|
|
/* rdt - filload_en */
|
|
REG_CPM_USBRDT |= USBRDT_VBFIL_LD_EN;
|
|
|
|
/* TXRISETUNE & TXVREFTUNE. */
|
|
REG_CPM_USBPCR = 0x3f;
|
|
REG_CPM_USBPCR = 0x35;
|
|
|
|
#if 1
|
|
REG_CPM_USBPCR &= ~(1 << 31);
|
|
REG_CPM_USBPCR |= (1 << 23) | (1 << 24);
|
|
#endif
|
|
|
|
/* enable tx pre-emphasis */
|
|
REG_CPM_USBPCR |= USBPCR_TXPREEMPHTUNE;
|
|
|
|
/* OTGTUNE adjust */
|
|
REG_CPM_USBPCR = (7 << 14);
|
|
}
|
|
REG_CPM_USBPCR |= 1 << 20;
|
|
|
|
/* force usb device mode */
|
|
x1000_usb_set_device_only_mode(dwc);
|
|
|
|
x1000_usb_phy_init(dwc);
|
|
|
|
x1000_usb_phy_switch(dwc,1);
|
|
|
|
/* soft disconnect and soft reset */
|
|
REG_OTG_DCTL |= DCTL_SOFT_DISCONN;
|
|
udelay(3000);
|
|
|
|
/* reset dwc register */
|
|
dwc_otg_core_reset(dwc);
|
|
|
|
/* DWC OTG Core init */
|
|
dwc_otg_core_init(dwc,1);
|
|
|
|
/* Read Gintsts confirm the device or host mode */
|
|
curmod = REG_GINT_STS;
|
|
if (curmod & 0x1)
|
|
{
|
|
DWC_DBG("Curmod: Host Mode\n");
|
|
}
|
|
else
|
|
{
|
|
DWC_DBG("Curmod: Device Mode\n");
|
|
|
|
/* DWC OTG Device init */
|
|
dwc_otg_device_init(dwc);
|
|
|
|
/* DWC OTG Fifo init */
|
|
dwc_calculate_fifo_size(dwc);
|
|
}
|
|
|
|
/* End-point has been inited */
|
|
// dwc_init_endpoint(dwc);
|
|
|
|
}
|
|
|
|
static void x1000_usbd_isr_service(void *param)
|
|
{
|
|
dwc_handle *dwc = (dwc_handle *)param;
|
|
uint8_t err;
|
|
uint32_t intsts;
|
|
|
|
RT_ASSERT(dwc != RT_NULL);
|
|
|
|
while (1)
|
|
{
|
|
rt_sem_take(dwc->isr_sem, RT_WAITING_FOREVER);
|
|
|
|
intsts = REG_GINT_STS;
|
|
|
|
if (intsts & GINTSTS_OTG_INTR)
|
|
{
|
|
DWC_DBG("OTG_INTR\n");
|
|
dwc_otg_intr(dwc);
|
|
}
|
|
|
|
if ((intsts & GINTSTS_USB_EARLYSUSPEND)
|
|
|| (intsts & GINTSTS_USB_SUSPEND)
|
|
|| (intsts & GINTSTS_START_FRAM)
|
|
|| (intsts & GINTSTS_USB_RESET)
|
|
|| (intsts & GINTSTS_ENUM_DONE))
|
|
{
|
|
dwc_common_intr(dwc, intsts);
|
|
}
|
|
|
|
/* dwc in pio mode not dma mode */
|
|
if (intsts & GINTSTS_RXFIFO_NEMPTY)
|
|
{
|
|
DWC_DBG("GINTSTS_RXFIFO_NEMPTY!!\n");
|
|
if (dwc->is_dma == IS_SLAVE_MODE) dwc_handle_rxfifo_nempty(dwc);
|
|
|
|
REG_GINT_STS = GINTSTS_RXFIFO_NEMPTY;
|
|
}
|
|
|
|
if (intsts & GINTSTS_IEP_INTR)
|
|
{
|
|
DWC_DBG("IEP_INTR!!!\n");
|
|
dwc_handle_inep_intr(dwc);
|
|
}
|
|
|
|
if (intsts & GINTSTS_OEP_INTR)
|
|
{
|
|
DWC_DBG("OEP_INTR!!!\n");
|
|
dwc_handle_outep_intr(dwc);
|
|
}
|
|
|
|
if (intsts & GINTSTS_RSUME_DETE)
|
|
{
|
|
DWC_DBG("RESUME_INTR\n");
|
|
dwc_handle_resume_intr(dwc);
|
|
}
|
|
|
|
if (intsts & (1 << 31))
|
|
{
|
|
REG_GINT_STS = 1 << 31;
|
|
}
|
|
|
|
rt_hw_interrupt_umask(IRQ_OTG);
|
|
}
|
|
}
|
|
|
|
|
|
void x1000_usbd_init(dwc_handle *dwc)
|
|
{
|
|
uint32_t curmod = 0;
|
|
|
|
DWC_DBG("Init UDC %s %s\n",__DATE__,__TIME__);
|
|
|
|
if(dwc->isr_sem == RT_NULL)
|
|
{
|
|
dwc->isr_sem = rt_sem_create("dwcSem",0,RT_IPC_FLAG_FIFO);
|
|
if (!dwc->isr_sem)
|
|
{
|
|
DWC_DBG("%s %d sem create err\n", __func__, __LINE__);
|
|
while (1) ;
|
|
}
|
|
|
|
dwc->status.b.state = USB_CABLE_DISCONNECT;
|
|
dwc->status.b.event = 0;
|
|
}
|
|
|
|
dwc_gadget_init(dwc);
|
|
|
|
/* create a ISR service task */
|
|
{
|
|
rt_thread_t tid;
|
|
tid = rt_thread_create("dwcIntSv",
|
|
x1000_usbd_isr_service, (void *) dwc,
|
|
2048,
|
|
RT_THREAD_PRIORITY_MAX/5,
|
|
20);
|
|
if (tid != RT_NULL) rt_thread_startup(tid);
|
|
rt_kprintf("dwc interrupt service init done...\n");
|
|
}
|
|
|
|
/* request irq */
|
|
rt_hw_interrupt_install(IRQ_OTG,dwc_irq_handler,(void *)dwc,"otgISR");
|
|
rt_hw_interrupt_umask(IRQ_OTG);
|
|
DWC_DBG("[DWC] DWC request IRQ success %x\n", REG_GINT_MASK);
|
|
}
|