| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830 |
- /* #include "mu3d_hal_osal.h" */
- /* #define _MTK_USB_DRV_EXT_ */
- #include "mu3d_hal_usb_drv.h"
- /* #undef _MTK_USB_DRV_EXT_ */
- #include "mu3d_hal_hw.h"
- #include "mu3d_hal_qmu_drv.h"
- /* #include "./new/mu3d_hal_comm.h" */
- /* #include <linux/mu3phy/mtk-phy.h> */
- #include "ssusb_io.h"
- static struct usb_ep_setting g_u3d_ep_setting[2 * MAX_EP_NUM + 1];
- static u32 g_TxFIFOadd;
- static u32 g_RxFIFOadd;
- int wait_for_value(void __iomem *base, int addr, int msk, int value, int ms_intvl, int count)
- {
- u32 i;
- for (i = 0; i < count; i++) {
- if ((mu3d_readl(base, addr) & msk) == value)
- return 0;
- mdelay(ms_intvl);
- }
- return -EBUSY;
- }
- /**
- * mu3d_hal_check_clk_sts - check sys125,u3 mac,u2 mac clock status
- *
- */
- int mu3d_hal_check_clk_sts(struct musb *musb)
- {
- int ret;
- mu3d_dbg(K_ERR, "%s\n", __func__);
- ret = wait_for_value(musb->sif_base, U3D_SSUSB_IP_PW_STS1, SSUSB_SYS125_RST_B_STS,
- SSUSB_SYS125_RST_B_STS, 1, 10);
- if (ret)
- mu3d_dbg(K_ERR, "SSUSB_SYS125_RST_B_STS NG\n");
- /* do not check when SSUSB_U2_PORT_PDN = 1, because U2 port stays in reset state */
- if (!(mu3d_readl(musb->sif_base, U3D_SSUSB_U2_CTRL_0P) & SSUSB_U2_PORT_PDN)) {
- ret =
- wait_for_value(musb->sif_base, U3D_SSUSB_IP_PW_STS2, SSUSB_U2_MAC_SYS_RST_B_STS,
- SSUSB_U2_MAC_SYS_RST_B_STS, 1, 10);
- if (ret)
- mu3d_dbg(K_ERR, "SSUSB_U2_MAC_SYS_RST_B_STS NG\n");
- }
- #ifdef SUPPORT_U3
- /* do not check when SSUSB_U3_PORT_PDN = 1, because U3 port stays in reset state */
- if (!(mu3d_readl(musb->sif_base, U3D_SSUSB_U3_CTRL_0P) & SSUSB_U3_PORT_PDN)) {
- ret = wait_for_value(musb->sif_base, U3D_SSUSB_IP_PW_STS1, SSUSB_U3_MAC_RST_B_STS,
- SSUSB_U3_MAC_RST_B_STS, 1, 10);
- if (ret)
- mu3d_dbg(K_ERR, "SSUSB_U3_MAC_RST_B_STS NG\n");
- }
- #endif
- mu3d_dbg(K_CRIT, "check clk pass!!\n");
- return 0;
- }
- /**
- * mu3d_hal_ssusb_en - disable ssusb power down & enable u2/u3 ports
- *
- */
- void mu3d_hal_ssusb_en(struct musb *musb)
- {
- void __iomem *sif_base = musb->sif_base;
- mu3d_dbg(K_INFO, "%s\n", __func__);
- mu3d_clrmsk(sif_base, U3D_SSUSB_IP_PW_CTRL0, SSUSB_IP_SW_RST);
- mu3d_clrmsk(sif_base, U3D_SSUSB_IP_PW_CTRL2, SSUSB_IP_DEV_PDN);
- #ifdef SUPPORT_U3
- mu3d_clrmsk(sif_base, U3D_SSUSB_U3_CTRL_0P,
- (SSUSB_U3_PORT_DIS | SSUSB_U3_PORT_PDN | SSUSB_U3_PORT_HOST_SEL));
- #endif
- mu3d_clrmsk(sif_base, U3D_SSUSB_U2_CTRL_0P,
- (SSUSB_U2_PORT_DIS | SSUSB_U2_PORT_PDN | SSUSB_U2_PORT_HOST_SEL));
- mu3d_dbg(K_INFO, "%s 0x:%x = 0x%x\n", __func__, U3D_SSUSB_U2_CTRL_0P,
- mu3d_readl(sif_base, U3D_SSUSB_U2_CTRL_0P));
- mu3d_setmsk(sif_base, U3D_SSUSB_REF_CK_CTRL,
- (SSUSB_REF_MAC_CK_GATE_EN | SSUSB_REF_PHY_CK_GATE_EN | SSUSB_REF_CK_GATE_EN |
- SSUSB_REF_MAC3_CK_GATE_EN));
- /* check U3D sys125,u3 mac,u2 mac clock status. */
- mu3d_hal_check_clk_sts(musb);
- }
- void mu3d_hal_ssusb_dis(struct musb *musb)
- {
- void __iomem *sif_base = musb->sif_base;
- mu3d_dbg(K_INFO, "%s\n", __func__);
- #ifdef SUPPORT_U3
- mu3d_setmsk(sif_base, U3D_SSUSB_U3_CTRL_0P, (SSUSB_U3_PORT_DIS | SSUSB_U3_PORT_PDN));
- #endif
- mu3d_setmsk(sif_base, U3D_SSUSB_U2_CTRL_0P, (SSUSB_U2_PORT_DIS | SSUSB_U2_PORT_PDN));
- mu3d_setmsk(sif_base, U3D_SSUSB_IP_PW_CTRL2, SSUSB_IP_DEV_PDN);
- mu3d_dbg(K_INFO, "%s 0x:%x = 0x%x\n", __func__, U3D_SSUSB_U2_CTRL_0P,
- mu3d_readl(sif_base, U3D_SSUSB_U2_CTRL_0P));
- }
- /**
- * mu3d_hal_rst_dev - reset all device modules
- *
- */
- void mu3d_hal_rst_dev(struct musb *musb)
- {
- void __iomem *sif_base = musb->sif_base;
- int ret;
- mu3d_dbg(K_ERR, "%s\n", __func__);
- mu3d_writel(sif_base, U3D_SSUSB_DEV_RST_CTRL, SSUSB_DEV_SW_RST);
- mu3d_writel(sif_base, U3D_SSUSB_DEV_RST_CTRL, 0);
- mu3d_hal_check_clk_sts(musb);
- ret = wait_for_value(sif_base, U3D_SSUSB_IP_PW_STS1, SSUSB_DEV_QMU_RST_B_STS,
- SSUSB_DEV_QMU_RST_B_STS, 1, 10);
- if (ret)
- mu3d_dbg(K_ERR, "SSUSB_DEV_QMU_RST_B_STS NG\n");
- ret = wait_for_value(sif_base, U3D_SSUSB_IP_PW_STS1, SSUSB_DEV_BMU_RST_B_STS,
- SSUSB_DEV_BMU_RST_B_STS, 1, 10);
- if (ret)
- mu3d_dbg(K_ERR, "SSUSB_DEV_BMU_RST_B_STS NG\n");
- ret = wait_for_value(sif_base, U3D_SSUSB_IP_PW_STS1, SSUSB_DEV_RST_B_STS,
- SSUSB_DEV_RST_B_STS, 1, 10);
- if (ret)
- mu3d_dbg(K_ERR, "SSUSB_DEV_RST_B_STS NG\n");
- }
- #ifdef CONFIG_MTK_FPGA
- /**
- * mu3d_hal_link_up - u3d link up
- *
- */
- int mu3d_hal_link_up(struct musb *musb, int latch_val)
- {
- mu3d_hal_ssusb_en(musb);
- mu3d_hal_rst_dev(musb);
- mdelay(50);
- mu3d_writel(musb->mac_base, U3D_USB3_CONFIG, USB3_EN);
- mu3d_writel(musb->mac_base, U3D_PIPE_LATCH_SELECT, latch_val); /* set tx/rx latch sel */
- return 0;
- }
- #endif
- /**
- * mu3d_hal_initr_dis - disable all interrupts
- *
- */
- void mu3d_hal_initr_dis(struct musb *musb)
- {
- void __iomem *mbase = musb->mac_base;
- /* Disable Level 1 interrupts */
- mu3d_writel(mbase, U3D_LV1IECR, 0xFFFFFFFF);
- /* Disable Endpoint interrupts */
- mu3d_writel(mbase, U3D_EPIECR, 0xFFFFFFFF);
- /* Disable DMA interrupts */
- mu3d_writel(mbase, U3D_DMAIECR, 0xFFFFFFFF);
- }
- void mu3d_hal_clear_intr(struct musb *musb)
- {
- void __iomem *mbase = musb->mac_base;
- mu3d_dbg(K_ERR, "%s\n", __func__);
- /* Clear EP0 and Tx/Rx EPn interrupts status */
- mu3d_writel(mbase, U3D_EPISR, 0xFFFFFFFF);
- /* Clear EP0 and Tx/Rx EPn DMA interrupts status */
- mu3d_writel(mbase, U3D_DMAISR, 0xFFFFFFFF);
- /* Clear speed change event */
- mu3d_writel(mbase, U3D_DEV_LINK_INTR, 0xFFFFFFFF);
- /* Clear U2 USB common interrupt status */
- mu3d_writel(mbase, U3D_COMMON_USB_INTR, 0xFFFFFFFF);
- /* Clear U3 LTSSM interrupt status */
- mu3d_writel(mbase, U3D_LTSSM_INTR, 0xFFFFFFFF);
- }
- /**
- * mu3d_hal_system_intr_en - enable system global interrupt
- *
- */
- void mu3d_hal_system_intr_en(struct musb *musb)
- {
- void __iomem *mbase = musb->mac_base;
- u32 int_en;
- #ifdef SUPPORT_U3
- u32 ltssm_int_en;
- #endif
- mu3d_dbg(K_ERR, "%s\n", __func__);
- /* Disable All endpoint interrupt */
- mu3d_writel(mbase, U3D_EPIECR, mu3d_readl(mbase, U3D_EPIER));
- /* Disable All DMA interrput */
- mu3d_writel(mbase, U3D_DMAIECR, mu3d_readl(mbase, U3D_DMAIER));
- /* Disable U2 common USB interrupts */
- mu3d_writel(mbase, U3D_COMMON_USB_INTR_ENABLE, 0x00);
- /* Clear U2 common USB interrupts status */
- mu3d_writel(mbase, U3D_COMMON_USB_INTR, mu3d_readl(mbase, U3D_COMMON_USB_INTR));
- /* Enable U2 common USB interrupt */
- int_en = SUSPEND_INTR_EN | RESUME_INTR_EN | RESET_INTR_EN | CONN_INTR_EN | DISCONN_INTR_EN
- | VBUSERR_INTR_EN | LPM_INTR_EN | LPM_RESUME_INTR_EN;
- mu3d_writel(mbase, U3D_COMMON_USB_INTR_ENABLE, int_en);
- #ifdef SUPPORT_U3
- /* Disable U3 LTSSM interrupts */
- mu3d_writel(mbase, U3D_LTSSM_INTR_ENABLE, 0x00);
- mu3d_dbg(K_ERR, "U3D_LTSSM_INTR: %x\n", mu3d_readl(mbase, U3D_LTSSM_INTR));
- /* Clear U3 LTSSM interrupts */
- mu3d_writel(mbase, U3D_LTSSM_INTR, mu3d_readl(mbase, U3D_LTSSM_INTR));
- /* Enable U3 LTSSM interrupts */
- ltssm_int_en =
- SS_INACTIVE_INTR_EN | SS_DISABLE_INTR_EN | COMPLIANCE_INTR_EN | LOOPBACK_INTR_EN |
- HOT_RST_INTR_EN | WARM_RST_INTR_EN | RECOVERY_INTR_EN | ENTER_U0_INTR_EN |
- ENTER_U1_INTR_EN | ENTER_U2_INTR_EN | ENTER_U3_INTR_EN | EXIT_U1_INTR_EN |
- EXIT_U2_INTR_EN | EXIT_U3_INTR_EN | RXDET_SUCCESS_INTR_EN | VBUS_RISE_INTR_EN |
- VBUS_FALL_INTR_EN | U3_LFPS_TMOUT_INTR_EN | U3_RESUME_INTR_EN;
- mu3d_writel(mbase, U3D_LTSSM_INTR_ENABLE, ltssm_int_en);
- #endif
- if (need_vbus_chg_int(musb)) {
- mu3d_writel(musb->sif_base, U3D_SSUSB_OTG_INT_EN, 0x0);
- mu3d_dbg(K_ERR, "U3D_SSUSB_OTG_STS: %x\n",
- mu3d_readl(musb->sif_base, U3D_SSUSB_OTG_STS));
- mu3d_writel(musb->sif_base, U3D_SSUSB_OTG_STS_CLR,
- mu3d_readl(musb->sif_base, U3D_SSUSB_OTG_STS));
- mu3d_writel(musb->sif_base, U3D_SSUSB_OTG_INT_EN, SSUSB_VBUS_CHG_INT_B_EN);
- }
- #ifdef USE_SSUSB_QMU
- /* Enable QMU interrupt. */
- mu3d_writel(mbase, U3D_QIESR1, TXQ_EMPTY_IESR | TXQ_CSERR_IESR | TXQ_LENERR_IESR |
- RXQ_EMPTY_IESR | RXQ_CSERR_IESR | RXQ_LENERR_IESR | RXQ_ZLPERR_IESR);
- #endif
- /* Enable EP0 DMA interrupt */
- mu3d_writel(mbase, U3D_DMAIESR, EP0DMAIESR);
- /* Enable speed change interrupt */
- mu3d_writel(mbase, U3D_DEV_LINK_INTR_ENABLE, SSUSB_DEV_SPEED_CHG_INTR_EN);
- }
- /**
- * mu3d_hal_u2dev_connect - u2 device softconnect
- *
- */
- void mu3d_hal_u2dev_connect(struct musb *musb)
- {
- mu3d_setmsk(musb->mac_base, U3D_POWER_MANAGEMENT, SOFT_CONN);
- mu3d_setmsk(musb->mac_base, U3D_POWER_MANAGEMENT, SUSPENDM_ENABLE);
- mu3d_dbg(K_INFO, "SOFTCONN = 1\n");
- }
- void mu3d_hal_u2dev_disconn(struct musb *musb)
- {
- mu3d_clrmsk(musb->mac_base, U3D_POWER_MANAGEMENT, SOFT_CONN);
- mu3d_clrmsk(musb->mac_base, U3D_POWER_MANAGEMENT, SUSPENDM_ENABLE);
- mu3d_dbg(K_INFO, "SOFTCONN = 0\n");
- }
- /**
- * mu3d_hal_u3dev_en - enable U3D SS dev
- *
- */
- void mu3d_hal_u3dev_en(struct musb *musb)
- {
- /*
- * Enable super speed function.
- */
- mu3d_writel(musb->mac_base, U3D_USB3_CONFIG, USB3_EN);
- mu3d_dbg(K_INFO, "USB3_EN = 1\n");
- }
- /**
- * mu3d_hal_u3dev_dis - disable U3D SS dev
- *
- */
- void mu3d_hal_u3dev_dis(struct musb *musb)
- {
- /*
- * If usb3_en =0 => LTSSM will go to SS.Disable state.
- */
- mu3d_writel(musb->mac_base, U3D_USB3_CONFIG, 0);
- mu3d_dbg(K_INFO, "USB3_EN = 0\n");
- }
- /**
- * mu3d_hal_pdn_cg_en - enable U2/U3 pdn & cg
- *@args -
- */
- /* void mu3d_hal_pdn_cg_en(void) */
- /* { */
- /* } */
- /* void mu3d_hal_pdn_ip_port(u8 on, u8 touch_dis, u8 u3, u8 u2) */
- /* { */
- /* } */
- /**
- * mu3d_hal_write_fifo - pio write one packet
- *@args - arg1: ep number, arg2: data length, arg3: data buffer, arg4: max packet size
- */
- int mu3d_hal_write_fifo(struct musb_hw_ep *hw_ep, int ep_num, int length, u8 *buf, int maxp)
- {
- struct musb *musb = hw_ep->musb;
- void __iomem *fifo = hw_ep->fifo;
- void __iomem *mbase = musb->mac_base;
- u32 residue;
- u32 count;
- u32 temp;
- mu3d_dbg(K_DEBUG, "%s epnum=%d, len=%d, buf=%p, maxp=%d\n", __func__, ep_num, length, buf,
- maxp);
- count = residue = length;
- while (residue > 0) {
- if (residue == 1) {
- temp = ((*buf) & 0xFF);
- mu3d_writeb(fifo, 0, temp);
- buf += 1;
- residue -= 1;
- } else if (residue == 2) {
- temp = ((*buf) & 0xFF) + (((*(buf + 1)) << 8) & 0xFF00);
- mu3d_writew(fifo, 0, temp);
- buf += 2;
- residue -= 2;
- } else if (residue == 3) {
- temp = ((*buf) & 0xFF) + (((*(buf + 1)) << 8) & 0xFF00);
- mu3d_writew(fifo, 0, temp);
- buf += 2;
- temp = ((*buf) & 0xFF);
- mu3d_writeb(fifo, 0, temp);
- buf += 1;
- residue -= 3;
- } else {
- temp =
- ((*buf) & 0xFF) + (((*(buf + 1)) << 8) & 0xFF00) +
- (((*(buf + 2)) << 16) & 0xFF0000) + (((*(buf + 3)) << 24) & 0xFF000000);
- mu3d_writel(fifo, 0, temp);
- buf += 4;
- residue -= 4;
- }
- }
- if (ep_num == 0) {
- if (count == 0) {
- mu3d_dbg(K_DEBUG, "USB_EP0_DATAEND %8X+\n", mu3d_readl(mbase, U3D_EP0CSR));
- mu3d_setmsk(mbase, U3D_EP0CSR, EP0_DATAEND | EP0_TXPKTRDY);
- mu3d_dbg(K_DEBUG, "USB_EP0_DATAEND %8X-\n", mu3d_readl(mbase, U3D_EP0CSR));
- } else {
- #ifdef AUTOSET
- if (count < maxp) {
- mu3d_setmsk(mbase, U3D_EP0CSR, EP0_TXPKTRDY);
- mu3d_dbg(K_DEBUG, "EP0_TXPKTRDY\n");
- }
- #else
- mu3d_setmsk(mbase, U3D_EP0CSR, EP0_TXPKTRDY);
- #endif
- }
- } else {
- if (count == 0) {
- mu3d_setmsk(hw_ep->addr_txcsr0, 0, TX_TXPKTRDY);
- } else {
- #ifdef AUTOSET
- if (count < maxp) {
- mu3d_setmsk(hw_ep->addr_txcsr0, 0, TX_TXPKTRDY);
- mu3d_dbg(K_DEBUG, "short packet\n");
- }
- #else
- mu3d_setmsk(hw_ep->addr_txcsr0, 0, TX_TXPKTRDY);
- #endif
- }
- }
- return count;
- }
- /**
- * mu3d_hal_read_fifo - pio read one packet
- *@args - arg1: ep number, arg2: data buffer
- */
- /* NOTE: The function should not be used to read SETUP */
- int mu3d_hal_read_fifo(struct musb_hw_ep *hw_ep, int ep_num, u8 *buf)
- {
- struct musb *musb = hw_ep->musb;
- void __iomem *mbase = musb->mac_base;
- u16 count, residue;
- u32 temp;
- u8 *bp = buf;
- if (ep_num == 0)
- residue = count = mu3d_readl(mbase, U3D_RXCOUNT0);
- else
- residue = count = (mu3d_readl(hw_ep->addr_rxcsr3, 0) >> EP_RX_COUNT_OFST);
- mu3d_dbg(K_DEBUG, "%s, req->buf=%p, cnt=%d\n", __func__, buf, count);
- while (residue > 0) {
- temp = mu3d_readl(hw_ep->fifo, 0);
- /*Store the first byte */
- *bp = temp & 0xFF;
- /*Store the 2nd byte, If have */
- if (residue > 1)
- *(bp + 1) = (temp >> 8) & 0xFF;
- /*Store the 3rd byte, If have */
- if (residue > 2)
- *(bp + 2) = (temp >> 16) & 0xFF;
- /*Store the 4th byte, If have */
- if (residue > 3)
- *(bp + 3) = (temp >> 24) & 0xFF;
- if (residue > 4) {
- bp = bp + 4;
- residue = residue - 4;
- } else {
- residue = 0;
- }
- }
- #ifdef AUTOCLEAR
- if (ep_num == 0) {
- if (!count) {
- mu3d_setmsk(mbase, U3D_EP0CSR, EP0_RXPKTRDY);
- mu3d_dbg(K_DEBUG, "EP0_RXPKTRDY\n");
- }
- } else {
- if (!count) {
- mu3d_setmsk(hw_ep->addr_rxcsr0, 0, RX_RXPKTRDY);
- mu3d_dbg(K_ALET, "ZLP\n");
- }
- }
- #else
- if (ep_num == 0) {
- mu3d_setmsk(mbase, U3D_EP0CSR, EP0_RXPKTRDY);
- mu3d_dbg(K_DEBUG, "EP0_RXPKTRDY\n");
- } else {
- mu3d_setmsk(hw_ep->addr_rxcsr0, 0, RX_RXPKTRDY);
- mu3d_dbg(K_DEBUG, "RX_RXPKTRDY\n");
- }
- #endif
- return count;
- }
- /**
- * mu3d_hal_unfigured_ep -
- *@args -
- */
- void mu3d_hal_unfigured_ep(struct musb *musb)
- {
- void __iomem *mbase = musb->mac_base;
- u32 i, tx_ep_num, rx_ep_num;
- struct usb_ep_setting *ep_setting;
- u32 tmp;
- g_TxFIFOadd = USB_TX_FIFO_START_ADDRESS;
- g_RxFIFOadd = USB_RX_FIFO_START_ADDRESS;
- #ifdef HARDCODE_EP
- tx_ep_num = MAX_QMU_EP; /* os_readl(U3D_CAP_EPINFO) & CAP_TX_EP_NUM; */
- rx_ep_num = MAX_QMU_EP; /* (os_readl(U3D_CAP_EPINFO) & CAP_RX_EP_NUM) >> 8; */
- #else
- tx_ep_num = mu3d_readl(mbase, U3D_CAP_EPINFO) & CAP_TX_EP_NUM;
- rx_ep_num = (mu3d_readl(mbase, U3D_CAP_EPINFO) & CAP_RX_EP_NUM) >> 8;
- #endif
- for (i = 1; i <= tx_ep_num; i++) {
- tmp = mu3d_xcsr_readl(mbase, U3D_TX1CSR0, i);
- tmp &= ~TX_TXMAXPKTSZ;
- mu3d_xcsr_writel(mbase, U3D_TX1CSR0, i, tmp);
- mu3d_xcsr_writel(mbase, U3D_TX1CSR1, i, 0);
- mu3d_xcsr_writel(mbase, U3D_TX1CSR2, i, 0);
- ep_setting = &g_u3d_ep_setting[i];
- ep_setting->fifoaddr = 0;
- ep_setting->enabled = 0;
- }
- for (i = 1; i <= rx_ep_num; i++) {
- tmp = mu3d_xcsr_readl(mbase, U3D_RX1CSR0, i);
- tmp &= ~RX_RXMAXPKTSZ;
- mu3d_xcsr_writel(mbase, U3D_RX1CSR0, i, tmp);
- mu3d_xcsr_writel(mbase, U3D_RX1CSR1, i, 0);
- mu3d_xcsr_writel(mbase, U3D_RX1CSR2, i, 0);
- ep_setting = &g_u3d_ep_setting[i + MAX_EP_NUM];
- ep_setting->fifoaddr = 0;
- ep_setting->enabled = 0;
- }
- }
- void mu3d_hal_unfigured_ep_num(struct musb *musb, int ep_num, USB_DIR dir)
- {
- void __iomem *mbase = musb->mac_base;
- struct usb_ep_setting *ep_setting;
- u32 tmp;
- mu3d_dbg(K_INFO, "%s %d\n", __func__, ep_num);
- if (dir == USB_TX) {
- tmp = mu3d_xcsr_readl(mbase, U3D_TX1CSR0, ep_num);
- tmp &= ~TX_TXMAXPKTSZ;
- mu3d_xcsr_writel(mbase, U3D_TX1CSR0, ep_num, tmp);
- mu3d_xcsr_writel(mbase, U3D_TX1CSR1, ep_num, 0);
- mu3d_xcsr_writel(mbase, U3D_TX1CSR2, ep_num, 0);
- ep_setting = &g_u3d_ep_setting[ep_num];
- /* ep_setting->fifoaddr = 0; */
- ep_setting->enabled = 0;
- } else {
- tmp = mu3d_xcsr_readl(mbase, U3D_RX1CSR0, ep_num);
- tmp &= ~RX_RXMAXPKTSZ;
- mu3d_xcsr_writel(mbase, U3D_RX1CSR0, ep_num, tmp);
- mu3d_xcsr_writel(mbase, U3D_RX1CSR1, ep_num, 0);
- mu3d_xcsr_writel(mbase, U3D_RX1CSR2, ep_num, 0);
- ep_setting = &g_u3d_ep_setting[ep_num + MAX_EP_NUM];
- /* ep_setting->fifoaddr = 0; */
- ep_setting->enabled = 0;
- }
- }
- /**
- * mu3d_hal_ep_enable - configure ep
- *@args - arg1: ep number, arg2: dir, arg3: transfer type, arg4: max packet size, arg5: interval, arg6: slot, arg7: burst, arg8: mult
- */
- void mu3d_hal_ep_enable(struct musb *musb, int ep_num, USB_DIR dir, TRANSFER_TYPE type, int maxp,
- int interval, int slot, int burst, int mult)
- {
- int ep_index = 0;
- int used_before;
- u8 fifosz = 0, max_pkt, binterval;
- int csr0, csr1, csr2;
- struct usb_ep_setting *ep_setting;
- void __iomem *mbase = musb->mac_base;
- u8 update_FIFOadd = 0;
- mu3d_dbg(K_INFO, "%s\n", __func__);
- /*TODO: Enable in future. */
- /*Enable Burst, NumP=0, EoB */
- /* os_writel( U3D_USB3_EPCTRL_CAP, os_readl(U3D_USB3_EPCTRL_CAP) | TX_NUMP_0_EN | SET_EOB_EN | TX_BURST_EN); */
- /*
- * Default value of U3D_USB3_EPCTRL_CAP
- * 1. tx_burst_en 1'b1
- * 2. set_eob_en 1'b0
- * 3. usb3_iso_crc_chk_dis 1'b1
- * 4. send_stall_clr_pp_en 1'b1
- * 5. tx_nump_0_en 1'b0
- */
- if (slot > MAX_SLOT) {
- mu3d_dbg(K_ALET, "[ERROR]Configure wrong slot number(MAX=%d, Not=%d)\n", MAX_SLOT,
- slot);
- slot = MAX_SLOT;
- }
- if (type == USB_CTRL) {
- ep_setting = &g_u3d_ep_setting[0];
- ep_setting->fifosz = maxp;
- ep_setting->maxp = maxp;
- csr0 = mu3d_readl(mbase, U3D_EP0CSR) & EP0_W1C_BITS;
- csr0 |= maxp;
- mu3d_writel(mbase, U3D_EP0CSR, csr0);
- mu3d_setmsk(mbase, U3D_USB2_RX_EP_DATAERR_INTR, BIT16); /* EP0 data error interrupt */
- return;
- }
- if (dir == USB_TX)
- ep_index = ep_num;
- else if (dir == USB_RX)
- ep_index = ep_num + MAX_EP_NUM;
- else
- BUG_ON(1);
- ep_setting = &g_u3d_ep_setting[ep_index];
- used_before = ep_setting->fifoaddr;
- if (ep_setting->enabled)
- return;
- binterval = interval;
- if (dir == USB_TX) {
- if ((g_TxFIFOadd + maxp * (slot + 1) > mu3d_readl(mbase, U3D_CAP_EPNTXFFSZ))
- && (!used_before)) {
- mu3d_dbg(K_ALET, "mu3d_hal_ep_enable, FAILED: sram exhausted\n");
- mu3d_dbg(K_ALET, "g_FIFOadd :%x\n", g_TxFIFOadd);
- mu3d_dbg(K_ALET, "maxp :%d\n", maxp);
- mu3d_dbg(K_ALET, "mult :%d\n", slot);
- BUG_ON(1);
- }
- } else {
- if ((g_RxFIFOadd + maxp * (slot + 1) > mu3d_readl(mbase, U3D_CAP_EPNRXFFSZ))
- && (!used_before)) {
- mu3d_dbg(K_ALET, "mu3d_hal_ep_enable, FAILED: sram exhausted\n");
- mu3d_dbg(K_ALET, "g_FIFOadd :%x\n", g_RxFIFOadd);
- mu3d_dbg(K_ALET, "maxp :%d\n", maxp);
- mu3d_dbg(K_ALET, "mult :%d\n", slot);
- BUG_ON(1);
- }
- }
- ep_setting->transfer_type = type;
- if (dir == USB_TX) {
- if (!ep_setting->fifoaddr) {
- ep_setting->fifoaddr = g_TxFIFOadd;
- update_FIFOadd = 1;
- }
- } else {
- if (!ep_setting->fifoaddr) {
- ep_setting->fifoaddr = g_RxFIFOadd;
- update_FIFOadd = 1;
- }
- }
- ep_setting->fifosz = maxp;
- ep_setting->maxp = maxp;
- ep_setting->dir = dir;
- ep_setting->enabled = 1;
- if (maxp <= 16)
- fifosz = USB_FIFOSZ_SIZE_16;
- else if (maxp <= 32)
- fifosz = USB_FIFOSZ_SIZE_32;
- else if (maxp <= 64)
- fifosz = USB_FIFOSZ_SIZE_64;
- else if (maxp <= 128)
- fifosz = USB_FIFOSZ_SIZE_128;
- else if (maxp <= 256)
- fifosz = USB_FIFOSZ_SIZE_256;
- else if (maxp <= 512)
- fifosz = USB_FIFOSZ_SIZE_512;
- else if (maxp <= 32768)
- fifosz = USB_FIFOSZ_SIZE_1024;
- else {
- mu3d_dbg(K_ERR, "%s Wrong MAXP\n", __func__);
- fifosz = USB_FIFOSZ_SIZE_1024;
- }
- if (dir == USB_TX) {
- /* CSR0 */
- csr0 = mu3d_xcsr_readl(mbase, U3D_TX1CSR0, ep_num);
- csr0 &= ~TX_TXMAXPKTSZ;
- csr0 |= (maxp & TX_TXMAXPKTSZ);
- #ifdef USE_SSUSB_QMU
- /* from SSUSB Device Programming Guide(Revision 0.1) page26
- * TX EP Setting of QMU
- * TXnCSR0.TxMaxPktSz
- * TXnCSR1.SS_BURST/TxType/TxSlot
- * TXnCSR1.Tx_Max_Pkt/Tx_Mult (ISO Only)
- * TXnCSR2.TxFIFOAddr/TxFIFOSegSize
- * TXnCSR2.TxBInterval (SS ISO/INT Only)
- * TxnCSR0.AutoSet = 0
- * TxnCSR0.DMARegEn = 1
- */
- csr0 &= ~TX_AUTOSET;
- csr0 |= TX_DMAREQEN;
- #else
- #ifdef AUTOSET
- csr0 |= TX_AUTOSET;
- #endif
- /*Disable DMA, Use PIO mode */
- csr0 &= ~TX_DMAREQEN;
- #endif
- /* CSR1 */
- max_pkt = (burst + 1) * (mult + 1) - 1;
- csr1 = (burst & SS_TX_BURST);
- csr1 |= (slot << TX_SLOT_OFST) & TX_SLOT;
- csr1 |= (max_pkt << TX_MAX_PKT_OFST) & TX_MAX_PKT;
- csr1 |= (mult << TX_MULT_OFST) & TX_MULT;
- /* CSR2 */
- csr2 = (ep_setting->fifoaddr >> 4) & TXFIFOADDR;
- csr2 |= (fifosz << TXFIFOSEGSIZE_OFST) & TXFIFOSEGSIZE;
- if (type == USB_BULK) {
- csr1 |= TYPE_BULK;
- } else if (type == USB_INTR) {
- csr1 |= TYPE_INT;
- csr2 |= (binterval << TXBINTERVAL_OFST) & TXBINTERVAL;
- } else if (type == USB_ISO) {
- csr1 |= TYPE_ISO;
- csr2 |= (binterval << TXBINTERVAL_OFST) & TXBINTERVAL;
- }
- #ifdef USE_SSUSB_QMU
- /*Disable Endpoint interrupt */
- mu3d_setmsk(mbase, U3D_EPIECR, (BIT0 << ep_num));
- /* Enable QMU */
- mu3d_setmsk(mbase, U3D_QGCSR, QMU_TX_EN(ep_num));
- /* Enable QMU Done interrupt */
- mu3d_setmsk(mbase, U3D_QIESR0, QMU_TX_EN(ep_num));
- #endif
- mu3d_xcsr_writel(mbase, U3D_TX1CSR0, ep_num, csr0);
- mu3d_xcsr_writel(mbase, U3D_TX1CSR1, ep_num, csr1);
- mu3d_xcsr_writel(mbase, U3D_TX1CSR2, ep_num, csr2);
- mu3d_dbg(K_DEBUG, "[CSR]U3D_TX%dCSR0 :%x\n", ep_num,
- mu3d_xcsr_readl(mbase, U3D_TX1CSR0, ep_num));
- mu3d_dbg(K_DEBUG, "[CSR]U3D_TX%dCSR1 :%x\n", ep_num,
- mu3d_xcsr_readl(mbase, U3D_TX1CSR1, ep_num));
- mu3d_dbg(K_DEBUG, "[CSR]U3D_TX%dCSR2 :%x\n", ep_num,
- mu3d_xcsr_readl(mbase, U3D_TX1CSR2, ep_num));
- } else if (dir == USB_RX) {
- /* CSR0 */
- csr0 = mu3d_xcsr_readl(mbase, U3D_RX1CSR0, ep_num);
- csr0 &= ~RX_RXMAXPKTSZ;
- csr0 |= (maxp & RX_RXMAXPKTSZ);
- #ifdef USE_SSUSB_QMU
- /* from SSUSB Device Programming Guide(Revision 0.1) page32
- * RX EP Setting of QMU
- * RXnCSR0.RxMaxPktSz
- * RXnCSR1.SS_BURST/RxType/RxSlot
- * RXnCSR1.Rx_Max_Pkt/Rx_Mult (ISO Only)
- * RXnCSR2.RxFIFOAddr/RxFIFOSegSize
- * RXnCSR2.RxBInterval (SS ISO/INT Only)
- * RxnCSR0.AutoClear = 0
- * RxnCSR0.DMARegEn = 1
- */
- csr0 &= ~RX_AUTOCLEAR;
- csr0 |= RX_DMAREQEN;
- #else
- #ifdef AUTOCLEAR
- csr0 |= RX_AUTOCLEAR;
- #endif
- /*Disable DMA, Use PIO mode */
- csr0 &= ~RX_DMAREQEN;
- #endif
- /* CSR1 */
- max_pkt = (burst + 1) * (mult + 1) - 1;
- csr1 = (burst & SS_RX_BURST);
- csr1 |= (slot << RX_SLOT_OFST) & RX_SLOT;
- csr1 |= (max_pkt << RX_MAX_PKT_OFST) & RX_MAX_PKT;
- csr1 |= (mult << RX_MULT_OFST) & RX_MULT;
- /* CSR2 */
- csr2 = (ep_setting->fifoaddr >> 4) & RXFIFOADDR;
- csr2 |= (fifosz << RXFIFOSEGSIZE_OFST) & RXFIFOSEGSIZE;
- if (type == USB_BULK) {
- csr1 |= TYPE_BULK;
- } else if (type == USB_INTR) {
- csr1 |= TYPE_INT;
- csr2 |= (binterval << RXBINTERVAL_OFST) & RXBINTERVAL;
- } else if (type == USB_ISO) {
- csr1 |= TYPE_ISO;
- csr2 |= (binterval << RXBINTERVAL_OFST) & RXBINTERVAL;
- }
- #ifdef USE_SSUSB_QMU
- /*Disable Endpoint interrupt */
- mu3d_setmsk(mbase, U3D_EPIECR, (BIT16 << ep_num));
- /* Enable QMU */
- mu3d_setmsk(mbase, U3D_QGCSR, QMU_TX_EN(ep_num));
- /*Enable QMU Done interrupt */
- mu3d_setmsk(mbase, U3D_QIESR0, QMU_RX_EN(ep_num));
- #endif
- mu3d_xcsr_writel(mbase, U3D_RX1CSR0, ep_num, csr0);
- mu3d_xcsr_writel(mbase, U3D_RX1CSR1, ep_num, csr1);
- mu3d_xcsr_writel(mbase, U3D_RX1CSR2, ep_num, csr2);
- mu3d_dbg(K_DEBUG, "[CSR]U3D_RX%dCSR0 :%x\n", ep_num,
- mu3d_xcsr_readl(mbase, U3D_RX1CSR0, ep_num));
- mu3d_dbg(K_DEBUG, "[CSR]U3D_RX%dCSR1 :%x\n", ep_num,
- mu3d_xcsr_readl(mbase, U3D_RX1CSR1, ep_num));
- mu3d_dbg(K_DEBUG, "[CSR]U3D_RX%dCSR2 :%x\n", ep_num,
- mu3d_xcsr_readl(mbase, U3D_RX1CSR2, ep_num));
- mu3d_setmsk(mbase, U3D_USB2_RX_EP_DATAERR_INTR, BIT16 << ep_num); /* EPn data error interrupt */
- } else {
- mu3d_dbg(K_ERR, "WHAT THE DIRECTION IS?!?!\n");
- BUG_ON(1);
- }
- if (update_FIFOadd == 1) {
- /* The minimum unit of FIFO address is _16_ bytes.
- * So let the offset of each EP fifo address aligns _16_ bytes.*/
- int fifo_offset = 0;
- if ((maxp & 0xF))
- fifo_offset = (maxp + 16) & (~0xF);
- else
- fifo_offset = maxp;
- if (dir == USB_TX)
- g_TxFIFOadd += (fifo_offset * (slot + 1));
- else
- g_RxFIFOadd += (fifo_offset * (slot + 1));
- }
- }
|