summaryrefslogtreecommitdiff
path: root/firmware/target/arm/usb-drv-arc.c
diff options
context:
space:
mode:
Diffstat (limited to 'firmware/target/arm/usb-drv-arc.c')
-rw-r--r--firmware/target/arm/usb-drv-arc.c897
1 files changed, 897 insertions, 0 deletions
diff --git a/firmware/target/arm/usb-drv-arc.c b/firmware/target/arm/usb-drv-arc.c
new file mode 100644
index 0000000000..3bf53fa431
--- /dev/null
+++ b/firmware/target/arm/usb-drv-arc.c
@@ -0,0 +1,897 @@
+/***************************************************************************
+ * __________ __ ___.
+ * Open \______ \ ____ ____ | | _\_ |__ _______ ___
+ * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
+ * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
+ * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
+ * \/ \/ \/ \/ \/
+ * $Id$
+ *
+ * Driver for ARC USBOTG Device Controller
+ *
+ * Copyright (C) 2007 by Björn Stenberg
+ *
+ * All files in this archive are subject to the GNU General Public License.
+ * See the file COPYING in the source tree root for full license agreement.
+ *
+ * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
+ * KIND, either express or implied.
+ *
+ ****************************************************************************/
+
+#include "system.h"
+#include "config.h"
+#include "string.h"
+#include "usb_ch9.h"
+#include "usb_core.h"
+#include "kernel.h"
+#include "panic.h"
+//#define LOGF_ENABLE
+#include "logf.h"
+
+#if CONFIG_CPU == IMX31L
+#include "avic-imx31.h"
+static void __attribute__((interrupt("IRQ"))) USB_OTG_HANDLER(void);
+#endif
+
+/* USB device mode registers (Little Endian) */
+
+#define REG_ID (*(volatile unsigned int *)(USB_BASE+0x000))
+#define REG_HWGENERAL (*(volatile unsigned int *)(USB_BASE+0x004))
+#define REG_HWHOST (*(volatile unsigned int *)(USB_BASE+0x008))
+#define REG_HWDEVICE (*(volatile unsigned int *)(USB_BASE+0x00c))
+#define REG_TXBUF (*(volatile unsigned int *)(USB_BASE+0x010))
+#define REG_RXBUF (*(volatile unsigned int *)(USB_BASE+0x014))
+#define REG_CAPLENGTH (*(volatile unsigned char*)(USB_BASE+0x100))
+#define REG_DCIVERSION (*(volatile unsigned int *)(USB_BASE+0x120))
+#define REG_DCCPARAMS (*(volatile unsigned int *)(USB_BASE+0x124))
+#define REG_USBCMD (*(volatile unsigned int *)(USB_BASE+0x140))
+#define REG_USBSTS (*(volatile unsigned int *)(USB_BASE+0x144))
+#define REG_USBINTR (*(volatile unsigned int *)(USB_BASE+0x148))
+#define REG_FRINDEX (*(volatile unsigned int *)(USB_BASE+0x14c))
+#define REG_DEVICEADDR (*(volatile unsigned int *)(USB_BASE+0x154))
+#define REG_ENDPOINTLISTADDR (*(volatile unsigned int *)(USB_BASE+0x158))
+#define REG_BURSTSIZE (*(volatile unsigned int *)(USB_BASE+0x160))
+#define REG_ULPI (*(volatile unsigned int *)(USB_BASE+0x170))
+#define REG_CONFIGFLAG (*(volatile unsigned int *)(USB_BASE+0x180))
+#define REG_PORTSC1 (*(volatile unsigned int *)(USB_BASE+0x184))
+#define REG_OTGSC (*(volatile unsigned int *)(USB_BASE+0x1a4))
+#define REG_USBMODE (*(volatile unsigned int *)(USB_BASE+0x1a8))
+#define REG_ENDPTSETUPSTAT (*(volatile unsigned int *)(USB_BASE+0x1ac))
+#define REG_ENDPTPRIME (*(volatile unsigned int *)(USB_BASE+0x1b0))
+#define REG_ENDPTFLUSH (*(volatile unsigned int *)(USB_BASE+0x1b4))
+#define REG_ENDPTSTATUS (*(volatile unsigned int *)(USB_BASE+0x1b8))
+#define REG_ENDPTCOMPLETE (*(volatile unsigned int *)(USB_BASE+0x1bc))
+#define REG_ENDPTCTRL0 (*(volatile unsigned int *)(USB_BASE+0x1c0))
+#define REG_ENDPTCTRL1 (*(volatile unsigned int *)(USB_BASE+0x1c4))
+#define REG_ENDPTCTRL2 (*(volatile unsigned int *)(USB_BASE+0x1c8))
+#define REG_ENDPTCTRL(_x_) (*(volatile unsigned int *)(USB_BASE+0x1c0+4*(_x_)))
+
+/* Frame Index Register Bit Masks */
+#define USB_FRINDEX_MASKS (0x3fff)
+
+/* USB CMD Register Bit Masks */
+#define USBCMD_RUN (0x00000001)
+#define USBCMD_CTRL_RESET (0x00000002)
+#define USBCMD_PERIODIC_SCHEDULE_EN (0x00000010)
+#define USBCMD_ASYNC_SCHEDULE_EN (0x00000020)
+#define USBCMD_INT_AA_DOORBELL (0x00000040)
+#define USBCMD_ASP (0x00000300)
+#define USBCMD_ASYNC_SCH_PARK_EN (0x00000800)
+#define USBCMD_SUTW (0x00002000)
+#define USBCMD_ATDTW (0x00004000)
+#define USBCMD_ITC (0x00FF0000)
+
+/* bit 15,3,2 are frame list size */
+#define USBCMD_FRAME_SIZE_1024 (0x00000000)
+#define USBCMD_FRAME_SIZE_512 (0x00000004)
+#define USBCMD_FRAME_SIZE_256 (0x00000008)
+#define USBCMD_FRAME_SIZE_128 (0x0000000C)
+#define USBCMD_FRAME_SIZE_64 (0x00008000)
+#define USBCMD_FRAME_SIZE_32 (0x00008004)
+#define USBCMD_FRAME_SIZE_16 (0x00008008)
+#define USBCMD_FRAME_SIZE_8 (0x0000800C)
+
+/* bit 9-8 are async schedule park mode count */
+#define USBCMD_ASP_00 (0x00000000)
+#define USBCMD_ASP_01 (0x00000100)
+#define USBCMD_ASP_10 (0x00000200)
+#define USBCMD_ASP_11 (0x00000300)
+#define USBCMD_ASP_BIT_POS (8)
+
+/* bit 23-16 are interrupt threshold control */
+#define USBCMD_ITC_NO_THRESHOLD (0x00000000)
+#define USBCMD_ITC_1_MICRO_FRM (0x00010000)
+#define USBCMD_ITC_2_MICRO_FRM (0x00020000)
+#define USBCMD_ITC_4_MICRO_FRM (0x00040000)
+#define USBCMD_ITC_8_MICRO_FRM (0x00080000)
+#define USBCMD_ITC_16_MICRO_FRM (0x00100000)
+#define USBCMD_ITC_32_MICRO_FRM (0x00200000)
+#define USBCMD_ITC_64_MICRO_FRM (0x00400000)
+#define USBCMD_ITC_BIT_POS (16)
+
+/* USB STS Register Bit Masks */
+#define USBSTS_INT (0x00000001)
+#define USBSTS_ERR (0x00000002)
+#define USBSTS_PORT_CHANGE (0x00000004)
+#define USBSTS_FRM_LST_ROLL (0x00000008)
+#define USBSTS_SYS_ERR (0x00000010) /* not used */
+#define USBSTS_IAA (0x00000020)
+#define USBSTS_RESET (0x00000040)
+#define USBSTS_SOF (0x00000080)
+#define USBSTS_SUSPEND (0x00000100)
+#define USBSTS_HC_HALTED (0x00001000)
+#define USBSTS_RCL (0x00002000)
+#define USBSTS_PERIODIC_SCHEDULE (0x00004000)
+#define USBSTS_ASYNC_SCHEDULE (0x00008000)
+
+/* USB INTR Register Bit Masks */
+#define USBINTR_INT_EN (0x00000001)
+#define USBINTR_ERR_INT_EN (0x00000002)
+#define USBINTR_PTC_DETECT_EN (0x00000004)
+#define USBINTR_FRM_LST_ROLL_EN (0x00000008)
+#define USBINTR_SYS_ERR_EN (0x00000010)
+#define USBINTR_ASYN_ADV_EN (0x00000020)
+#define USBINTR_RESET_EN (0x00000040)
+#define USBINTR_SOF_EN (0x00000080)
+#define USBINTR_DEVICE_SUSPEND (0x00000100)
+
+/* ULPI Register Bit Masks */
+#define ULPI_ULPIWU (0x80000000)
+#define ULPI_ULPIRUN (0x40000000)
+#define ULPI_ULPIRW (0x20000000)
+#define ULPI_ULPISS (0x08000000)
+#define ULPI_ULPIPORT (0x07000000)
+#define ULPI_ULPIADDR (0x00FF0000)
+#define ULPI_ULPIDATRD (0x0000FF00)
+#define ULPI_ULPIDATWR (0x000000FF)
+
+/* Device Address bit masks */
+#define USBDEVICEADDRESS_MASK (0xFE000000)
+#define USBDEVICEADDRESS_BIT_POS (25)
+
+/* endpoint list address bit masks */
+#define USB_EP_LIST_ADDRESS_MASK (0xfffff800)
+
+/* PORTSCX Register Bit Masks */
+#define PORTSCX_CURRENT_CONNECT_STATUS (0x00000001)
+#define PORTSCX_CONNECT_STATUS_CHANGE (0x00000002)
+#define PORTSCX_PORT_ENABLE (0x00000004)
+#define PORTSCX_PORT_EN_DIS_CHANGE (0x00000008)
+#define PORTSCX_OVER_CURRENT_ACT (0x00000010)
+#define PORTSCX_OVER_CURRENT_CHG (0x00000020)
+#define PORTSCX_PORT_FORCE_RESUME (0x00000040)
+#define PORTSCX_PORT_SUSPEND (0x00000080)
+#define PORTSCX_PORT_RESET (0x00000100)
+#define PORTSCX_LINE_STATUS_BITS (0x00000C00)
+#define PORTSCX_PORT_POWER (0x00001000)
+#define PORTSCX_PORT_INDICTOR_CTRL (0x0000C000)
+#define PORTSCX_PORT_TEST_CTRL (0x000F0000)
+#define PORTSCX_WAKE_ON_CONNECT_EN (0x00100000)
+#define PORTSCX_WAKE_ON_CONNECT_DIS (0x00200000)
+#define PORTSCX_WAKE_ON_OVER_CURRENT (0x00400000)
+#define PORTSCX_PHY_LOW_POWER_SPD (0x00800000)
+#define PORTSCX_PORT_FORCE_FULL_SPEED (0x01000000)
+#define PORTSCX_PORT_SPEED_MASK (0x0C000000)
+#define PORTSCX_PORT_WIDTH (0x10000000)
+#define PORTSCX_PHY_TYPE_SEL (0xC0000000)
+
+/* bit 11-10 are line status */
+#define PORTSCX_LINE_STATUS_SE0 (0x00000000)
+#define PORTSCX_LINE_STATUS_JSTATE (0x00000400)
+#define PORTSCX_LINE_STATUS_KSTATE (0x00000800)
+#define PORTSCX_LINE_STATUS_UNDEF (0x00000C00)
+#define PORTSCX_LINE_STATUS_BIT_POS (10)
+
+/* bit 15-14 are port indicator control */
+#define PORTSCX_PIC_OFF (0x00000000)
+#define PORTSCX_PIC_AMBER (0x00004000)
+#define PORTSCX_PIC_GREEN (0x00008000)
+#define PORTSCX_PIC_UNDEF (0x0000C000)
+#define PORTSCX_PIC_BIT_POS (14)
+
+/* bit 19-16 are port test control */
+#define PORTSCX_PTC_DISABLE (0x00000000)
+#define PORTSCX_PTC_JSTATE (0x00010000)
+#define PORTSCX_PTC_KSTATE (0x00020000)
+#define PORTSCX_PTC_SE0NAK (0x00030000)
+#define PORTSCX_PTC_PACKET (0x00040000)
+#define PORTSCX_PTC_FORCE_EN (0x00050000)
+#define PORTSCX_PTC_BIT_POS (16)
+
+/* bit 27-26 are port speed */
+#define PORTSCX_PORT_SPEED_FULL (0x00000000)
+#define PORTSCX_PORT_SPEED_LOW (0x04000000)
+#define PORTSCX_PORT_SPEED_HIGH (0x08000000)
+#define PORTSCX_PORT_SPEED_UNDEF (0x0C000000)
+#define PORTSCX_SPEED_BIT_POS (26)
+
+/* bit 28 is parallel transceiver width for UTMI interface */
+#define PORTSCX_PTW (0x10000000)
+#define PORTSCX_PTW_8BIT (0x00000000)
+#define PORTSCX_PTW_16BIT (0x10000000)
+
+/* bit 31-30 are port transceiver select */
+#define PORTSCX_PTS_UTMI (0x00000000)
+#define PORTSCX_PTS_CLASSIC (0x40000000)
+#define PORTSCX_PTS_ULPI (0x80000000)
+#define PORTSCX_PTS_FSLS (0xC0000000)
+#define PORTSCX_PTS_BIT_POS (30)
+
+/* USB MODE Register Bit Masks */
+#define USBMODE_CTRL_MODE_IDLE (0x00000000)
+#define USBMODE_CTRL_MODE_DEVICE (0x00000002)
+#define USBMODE_CTRL_MODE_HOST (0x00000003)
+#define USBMODE_CTRL_MODE_RSV (0x00000001)
+#define USBMODE_SETUP_LOCK_OFF (0x00000008)
+#define USBMODE_STREAM_DISABLE (0x00000010)
+
+/* Endpoint Flush Register */
+#define EPFLUSH_TX_OFFSET (0x00010000)
+#define EPFLUSH_RX_OFFSET (0x00000000)
+
+/* Endpoint Setup Status bit masks */
+#define EPSETUP_STATUS_MASK (0x0000003F)
+#define EPSETUP_STATUS_EP0 (0x00000001)
+
+/* ENDPOINTCTRLx Register Bit Masks */
+#define EPCTRL_TX_ENABLE (0x00800000)
+#define EPCTRL_TX_DATA_TOGGLE_RST (0x00400000) /* Not EP0 */
+#define EPCTRL_TX_DATA_TOGGLE_INH (0x00200000) /* Not EP0 */
+#define EPCTRL_TX_TYPE (0x000C0000)
+#define EPCTRL_TX_DATA_SOURCE (0x00020000) /* Not EP0 */
+#define EPCTRL_TX_EP_STALL (0x00010000)
+#define EPCTRL_RX_ENABLE (0x00000080)
+#define EPCTRL_RX_DATA_TOGGLE_RST (0x00000040) /* Not EP0 */
+#define EPCTRL_RX_DATA_TOGGLE_INH (0x00000020) /* Not EP0 */
+#define EPCTRL_RX_TYPE (0x0000000C)
+#define EPCTRL_RX_DATA_SINK (0x00000002) /* Not EP0 */
+#define EPCTRL_RX_EP_STALL (0x00000001)
+
+/* bit 19-18 and 3-2 are endpoint type */
+#define EPCTRL_EP_TYPE_CONTROL (0)
+#define EPCTRL_EP_TYPE_ISO (1)
+#define EPCTRL_EP_TYPE_BULK (2)
+#define EPCTRL_EP_TYPE_INTERRUPT (3)
+#define EPCTRL_TX_EP_TYPE_SHIFT (18)
+#define EPCTRL_RX_EP_TYPE_SHIFT (2)
+
+/* pri_ctrl Register Bit Masks */
+#define PRI_CTRL_PRI_LVL1 (0x0000000C)
+#define PRI_CTRL_PRI_LVL0 (0x00000003)
+
+/* si_ctrl Register Bit Masks */
+#define SI_CTRL_ERR_DISABLE (0x00000010)
+#define SI_CTRL_IDRC_DISABLE (0x00000008)
+#define SI_CTRL_RD_SAFE_EN (0x00000004)
+#define SI_CTRL_RD_PREFETCH_DISABLE (0x00000002)
+#define SI_CTRL_RD_PREFEFETCH_VAL (0x00000001)
+
+/* control Register Bit Masks */
+#define USB_CTRL_IOENB (0x00000004)
+#define USB_CTRL_ULPI_INT0EN (0x00000001)
+
+/* OTGSC Register Bit Masks */
+#define OTGSC_B_SESSION_VALID (0x00000800)
+
+#define QH_MULT_POS (30)
+#define QH_ZLT_SEL (0x20000000)
+#define QH_MAX_PKT_LEN_POS (16)
+#define QH_IOS (0x00008000)
+#define QH_NEXT_TERMINATE (0x00000001)
+#define QH_IOC (0x00008000)
+#define QH_MULTO (0x00000C00)
+#define QH_STATUS_HALT (0x00000040)
+#define QH_STATUS_ACTIVE (0x00000080)
+#define EP_QUEUE_CURRENT_OFFSET_MASK (0x00000FFF)
+#define EP_QUEUE_HEAD_NEXT_POINTER_MASK (0xFFFFFFE0)
+#define EP_QUEUE_FRINDEX_MASK (0x000007FF)
+#define EP_MAX_LENGTH_TRANSFER (0x4000)
+
+#define DTD_NEXT_TERMINATE (0x00000001)
+#define DTD_IOC (0x00008000)
+#define DTD_STATUS_ACTIVE (0x00000080)
+#define DTD_STATUS_HALTED (0x00000040)
+#define DTD_STATUS_DATA_BUFF_ERR (0x00000020)
+#define DTD_STATUS_TRANSACTION_ERR (0x00000008)
+#define DTD_RESERVED_FIELDS (0x80007300)
+#define DTD_ADDR_MASK (0xFFFFFFE0)
+#define DTD_PACKET_SIZE (0x7FFF0000)
+#define DTD_LENGTH_BIT_POS (16)
+#define DTD_ERROR_MASK (DTD_STATUS_HALTED | \
+ DTD_STATUS_DATA_BUFF_ERR | \
+ DTD_STATUS_TRANSACTION_ERR)
+
+#define DTD_RESERVED_LENGTH_MASK 0x0001ffff
+#define DTD_RESERVED_IN_USE 0x80000000
+#define DTD_RESERVED_PIPE_MASK 0x0ff00000
+#define DTD_RESERVED_PIPE_OFFSET 20
+/*-------------------------------------------------------------------------*/
+
+/* manual: 32.13.2 Endpoint Transfer Descriptor (dTD) */
+struct transfer_descriptor {
+ unsigned int next_td_ptr; /* Next TD pointer(31-5), T(0) set
+ indicate invalid */
+ unsigned int size_ioc_sts; /* Total bytes (30-16), IOC (15),
+ MultO(11-10), STS (7-0) */
+ unsigned int buff_ptr0; /* Buffer pointer Page 0 */
+ unsigned int buff_ptr1; /* Buffer pointer Page 1 */
+ unsigned int buff_ptr2; /* Buffer pointer Page 2 */
+ unsigned int buff_ptr3; /* Buffer pointer Page 3 */
+ unsigned int buff_ptr4; /* Buffer pointer Page 4 */
+ unsigned int reserved;
+} __attribute__ ((packed));
+
+static struct transfer_descriptor td_array[NUM_ENDPOINTS*2]
+ USBDEVBSS_ATTR __attribute__((aligned(32)));
+
+/* manual: 32.13.1 Endpoint Queue Head (dQH) */
+struct queue_head {
+ unsigned int max_pkt_length; /* Mult(31-30) , Zlt(29) , Max Pkt len
+ and IOS(15) */
+ unsigned int curr_dtd_ptr; /* Current dTD Pointer(31-5) */
+ struct transfer_descriptor dtd; /* dTD overlay */
+ unsigned int setup_buffer[2]; /* Setup data 8 bytes */
+ unsigned int reserved; /* for software use, pointer to the first TD */
+ unsigned int status; /* for software use, status of chain in progress */
+ unsigned int length; /* for software use, transfered bytes of chain in progress */
+ unsigned int wait; /* for softwate use, indicates if the transfer is blocking */
+} __attribute__((packed));
+
+static struct queue_head qh_array[NUM_ENDPOINTS*2]
+ USBDEVBSS_ATTR __attribute__((aligned (2048)));
+
+static struct wakeup transfer_completion_signal[NUM_ENDPOINTS*2]
+ SHAREDBSS_ATTR;
+
+static const unsigned int pipe2mask[] = {
+ 0x01, 0x010000,
+ 0x02, 0x020000,
+ 0x04, 0x040000,
+ 0x08, 0x080000,
+ 0x10, 0x100000,
+};
+
+static bool first_init = true;
+
+/*-------------------------------------------------------------------------*/
+static void transfer_completed(void);
+static void control_received(void);
+static int prime_transfer(int endpoint, void* ptr,
+ int len, bool send, bool wait);
+static void prepare_td(struct transfer_descriptor* td,
+ struct transfer_descriptor* previous_td,
+ void *ptr, int len,int pipe);
+static void bus_reset(void);
+static void init_control_queue_heads(void);
+static void init_bulk_queue_heads(void);
+static void init_endpoints(void);
+/*-------------------------------------------------------------------------*/
+
+bool usb_drv_powered(void)
+{
+ return (REG_OTGSC & OTGSC_B_SESSION_VALID) ? true : false;
+}
+
+/* manual: 32.14.1 Device Controller Initialization */
+void usb_drv_init(void)
+{
+ REG_USBCMD &= ~USBCMD_RUN;
+
+ if (first_init)
+ {
+ /* Initialize all the signal objects once */
+ int i;
+ for(i=0;i<NUM_ENDPOINTS*2;i++) {
+ wakeup_init(&transfer_completion_signal[i]);
+ }
+
+ first_init = false;
+ }
+
+ sleep(HZ/20);
+ REG_USBCMD |= USBCMD_CTRL_RESET;
+ while (REG_USBCMD & USBCMD_CTRL_RESET);
+
+
+ REG_USBMODE = USBMODE_CTRL_MODE_DEVICE;
+
+#if CONFIG_CPU == IMX31L
+ /* Set to ULPI */
+ REG_PORTSC1 = (REG_PORTSC1 & ~PORTSCX_PHY_TYPE_SEL) | PORTSCX_PTS_ULPI;
+#endif
+
+#ifndef USE_HIGH_SPEED
+ /* Force device to full speed */
+ /* See 32.9.5.9.2 */
+ REG_PORTSC1 |= PORTSCX_PORT_FORCE_FULL_SPEED;
+#endif
+
+ init_control_queue_heads();
+ memset(td_array, 0, sizeof td_array);
+
+ REG_ENDPOINTLISTADDR = (unsigned int)qh_array;
+ REG_DEVICEADDR = 0;
+
+ /* enable USB interrupts */
+ REG_USBINTR =
+ USBINTR_INT_EN |
+ USBINTR_ERR_INT_EN |
+ USBINTR_PTC_DETECT_EN |
+ USBINTR_RESET_EN |
+ USBINTR_SYS_ERR_EN;
+
+#if CONFIG_CPU == IMX31L
+ avic_enable_int(USB_OTG, IRQ, 7, USB_OTG_HANDLER);
+#else
+ /* enable USB IRQ in CPU */
+ CPU_INT_EN = USB_MASK;
+#endif
+
+ /* go go go */
+ REG_USBCMD |= USBCMD_RUN;
+
+
+ logf("usb_drv_init() finished");
+ logf("usb id %x", REG_ID);
+ logf("usb dciversion %x", REG_DCIVERSION);
+ logf("usb dccparams %x", REG_DCCPARAMS);
+
+ /* now a bus reset will occur. see bus_reset() */
+}
+
+void usb_drv_exit(void)
+{
+ /* disable interrupts */
+ REG_USBINTR = 0;
+
+ /* stop usb controller */
+ REG_USBCMD &= ~USBCMD_RUN;
+
+ /* TODO : is one of these needed to save power ?
+ REG_PORTSC1 |= PORTSCX_PHY_LOW_POWER_SPD;
+ REG_USBCMD |= USBCMD_CTRL_RESET;
+ */
+
+#if CONFIG_CPU == IMX31L
+ avic_disable_int(USB_OTG);
+#else
+ CPU_INT_CLR = USB_MASK;
+#endif
+
+ cancel_cpu_boost();
+}
+
+#if CONFIG_CPU == IMX31L
+static void __attribute__((interrupt("IRQ"))) USB_OTG_HANDLER(void)
+#else
+void usb_drv_int(void)
+#endif
+{
+ unsigned int status = REG_USBSTS;
+
+#if 0
+ if (status & USBSTS_INT) logf("int: usb ioc");
+ if (status & USBSTS_ERR) logf("int: usb err");
+ if (status & USBSTS_PORT_CHANGE) logf("int: portchange");
+ if (status & USBSTS_RESET) logf("int: reset");
+ if (status & USBSTS_SYS_ERR) logf("int: syserr");
+#endif
+
+ /* usb transaction interrupt */
+ if (status & USBSTS_INT) {
+ REG_USBSTS = USBSTS_INT;
+
+ /* a control packet? */
+ if (REG_ENDPTSETUPSTAT & EPSETUP_STATUS_EP0) {
+ control_received();
+ }
+
+ if (REG_ENDPTCOMPLETE)
+ transfer_completed();
+ }
+
+ /* error interrupt */
+ if (status & USBSTS_ERR) {
+ REG_USBSTS = USBSTS_ERR;
+ logf("usb error int");
+ }
+
+ /* reset interrupt */
+ if (status & USBSTS_RESET) {
+ REG_USBSTS = USBSTS_RESET;
+ bus_reset();
+ usb_core_bus_reset(); /* tell mom */
+ }
+
+ /* port change */
+ if (status & USBSTS_PORT_CHANGE) {
+ REG_USBSTS = USBSTS_PORT_CHANGE;
+ }
+}
+
+bool usb_drv_stalled(int endpoint,bool in)
+{
+ if(in) {
+ return ((REG_ENDPTCTRL(endpoint) & EPCTRL_TX_EP_STALL)!=0);
+ }
+ else {
+ return ((REG_ENDPTCTRL(endpoint) & EPCTRL_RX_EP_STALL)!=0);
+ }
+
+}
+void usb_drv_stall(int endpoint, bool stall,bool in)
+{
+ logf("%sstall %d", stall?"":"un", endpoint);
+
+ if(in) {
+ if (stall) {
+ REG_ENDPTCTRL(endpoint) |= EPCTRL_TX_EP_STALL;
+ }
+ else {
+ REG_ENDPTCTRL(endpoint) &= ~EPCTRL_TX_EP_STALL;
+ }
+ }
+ else {
+ if (stall) {
+ REG_ENDPTCTRL(endpoint) |= EPCTRL_RX_EP_STALL;
+ }
+ else {
+ REG_ENDPTCTRL(endpoint) &= ~EPCTRL_RX_EP_STALL;
+ }
+ }
+}
+
+int usb_drv_send_nonblocking(int endpoint, void* ptr, int length)
+{
+ return prime_transfer(endpoint, ptr, length, true, false);
+}
+
+int usb_drv_send(int endpoint, void* ptr, int length)
+{
+ return prime_transfer(endpoint, ptr, length, true, true);
+}
+
+int usb_drv_recv(int endpoint, void* ptr, int length)
+{
+ //logf("usbrecv(%x, %d)", ptr, length);
+ return prime_transfer(endpoint, ptr, length, false, false);
+}
+
+void usb_drv_wait(int endpoint, bool send)
+{
+ int pipe = endpoint * 2 + (send ? 1 : 0);
+ struct queue_head* qh = &qh_array[pipe];
+
+ while (qh->dtd.size_ioc_sts & QH_STATUS_ACTIVE) {
+ if (REG_USBSTS & USBSTS_RESET)
+ break;
+ }
+}
+
+int usb_drv_port_speed(void)
+{
+ return (REG_PORTSC1 & 0x08000000) ? 1 : 0;
+}
+
+bool usb_drv_connected(void)
+{
+ return ((REG_PORTSC1 & PORTSCX_CURRENT_CONNECT_STATUS) !=0);
+}
+
+void usb_drv_set_address(int address)
+{
+ REG_DEVICEADDR = address << USBDEVICEADDRESS_BIT_POS;
+ init_bulk_queue_heads();
+ init_endpoints();
+}
+
+void usb_drv_reset_endpoint(int endpoint, bool send)
+{
+ int pipe = endpoint * 2 + (send ? 1 : 0);
+ unsigned int mask = pipe2mask[pipe];
+ REG_ENDPTFLUSH = mask;
+ while (REG_ENDPTFLUSH & mask);
+}
+
+void usb_drv_set_test_mode(int mode)
+{
+ switch(mode){
+ case 0:
+ REG_PORTSC1 &= ~PORTSCX_PORT_TEST_CTRL;
+ break;
+ case 1:
+ REG_PORTSC1 |= PORTSCX_PTC_JSTATE;
+ break;
+ case 2:
+ REG_PORTSC1 |= PORTSCX_PTC_KSTATE;
+ break;
+ case 3:
+ REG_PORTSC1 |= PORTSCX_PTC_SE0NAK;
+ break;
+ case 4:
+ REG_PORTSC1 |= PORTSCX_PTC_PACKET;
+ break;
+ case 5:
+ REG_PORTSC1 |= PORTSCX_PTC_FORCE_EN;
+ break;
+ }
+ REG_USBCMD &= ~USBCMD_RUN;
+ sleep(HZ/20);
+ REG_USBCMD |= USBCMD_CTRL_RESET;
+ while (REG_USBCMD & USBCMD_CTRL_RESET);
+ REG_USBCMD |= USBCMD_RUN;
+}
+
+/*-------------------------------------------------------------------------*/
+
+/* manual: 32.14.5.2 */
+static int prime_transfer(int endpoint, void* ptr, int len, bool send, bool wait)
+{
+ int rc = 0;
+ int pipe = endpoint * 2 + (send ? 1 : 0);
+ unsigned int mask = pipe2mask[pipe];
+ struct queue_head* qh = &qh_array[pipe];
+ static long last_tick;
+ struct transfer_descriptor* new_td;
+
+/*
+ if (send && endpoint > EP_CONTROL) {
+ logf("usb: sent %d bytes", len);
+ }
+*/
+ qh->status = 0;
+ qh->length = 0;
+ qh->wait = wait;
+
+ new_td=&td_array[pipe];
+ prepare_td(new_td, 0, ptr, len,pipe);
+ //logf("starting ep %d %s",endpoint,send?"send":"receive");
+
+ qh->dtd.next_td_ptr = (unsigned int)new_td;
+ qh->dtd.size_ioc_sts &= ~(QH_STATUS_HALT | QH_STATUS_ACTIVE);
+
+ REG_ENDPTPRIME |= mask;
+
+ if(endpoint == EP_CONTROL && (REG_ENDPTSETUPSTAT & EPSETUP_STATUS_EP0)) {
+ /* 32.14.3.2.2 */
+ logf("new setup arrived");
+ rc = -4;
+ goto pt_error;
+ }
+
+ last_tick = current_tick;
+ while ((REG_ENDPTPRIME & mask)) {
+ if (REG_USBSTS & USBSTS_RESET) {
+ rc = -1;
+ goto pt_error;
+ }
+
+ if (TIME_AFTER(current_tick, last_tick + HZ/4)) {
+ logf("prime timeout");
+ rc = -2;
+ goto pt_error;
+ }
+ }
+
+ if (!(REG_ENDPTSTATUS & mask)) {
+ logf("no prime! %d %d %x", endpoint, pipe, qh->dtd.size_ioc_sts & 0xff );
+ rc = -3;
+ goto pt_error;
+ }
+ if(endpoint == EP_CONTROL && (REG_ENDPTSETUPSTAT & EPSETUP_STATUS_EP0)) {
+ /* 32.14.3.2.2 */
+ logf("new setup arrived");
+ rc = -4;
+ goto pt_error;
+ }
+
+ if (wait) {
+ /* wait for transfer to finish */
+ wakeup_wait(&transfer_completion_signal[pipe], TIMEOUT_BLOCK);
+ if(qh->status!=0) {
+ /* No need to cancel wait here since it was done and the signal
+ * came. */
+ return 5;
+ }
+ //logf("all tds done");
+ }
+
+pt_error:
+ /* Error status must make sure an abandoned wakeup signal isn't left */
+ if (rc < 0 && wait) {
+ /* Cancel wait */
+ qh->wait = 0;
+ /* Make sure to remove any signal if interrupt fired before we zeroed
+ * qh->wait. Could happen during a bus reset for example. */
+ wakeup_wait(&transfer_completion_signal[pipe], TIMEOUT_NOBLOCK);
+ }
+
+ return rc;
+}
+
+void usb_drv_cancel_all_transfers(void)
+{
+ int i;
+ REG_ENDPTFLUSH = ~0;
+ while (REG_ENDPTFLUSH);
+
+ memset(td_array, 0, sizeof td_array);
+ for(i=0;i<NUM_ENDPOINTS*2;i++) {
+ if(qh_array[i].wait) {
+ qh_array[i].wait=0;
+ qh_array[i].status=DTD_STATUS_HALTED;
+ wakeup_signal(&transfer_completion_signal[i]);
+ }
+ }
+}
+
+static void prepare_td(struct transfer_descriptor* td,
+ struct transfer_descriptor* previous_td,
+ void *ptr, int len,int pipe)
+{
+ //logf("adding a td : %d",len);
+ memset(td, 0, sizeof(struct transfer_descriptor));
+ td->next_td_ptr = DTD_NEXT_TERMINATE;
+ td->size_ioc_sts = (len<< DTD_LENGTH_BIT_POS) |
+ DTD_STATUS_ACTIVE | DTD_IOC;
+ td->buff_ptr0 = (unsigned int)ptr;
+ td->buff_ptr1 = ((unsigned int)ptr & 0xfffff000) + 0x1000;
+ td->buff_ptr2 = ((unsigned int)ptr & 0xfffff000) + 0x2000;
+ td->buff_ptr3 = ((unsigned int)ptr & 0xfffff000) + 0x3000;
+ td->buff_ptr4 = ((unsigned int)ptr & 0xfffff000) + 0x4000;
+ td->reserved |= DTD_RESERVED_LENGTH_MASK & len;
+ td->reserved |= DTD_RESERVED_IN_USE;
+ td->reserved |= (pipe << DTD_RESERVED_PIPE_OFFSET);
+
+ if (previous_td != 0) {
+ previous_td->next_td_ptr=(unsigned int)td;
+ }
+}
+
+static void control_received(void)
+{
+ int i;
+ /* copy setup data from packet */
+ static unsigned int tmp[2];
+ tmp[0] = qh_array[0].setup_buffer[0];
+ tmp[1] = qh_array[0].setup_buffer[1];
+
+ /* acknowledge packet recieved */
+ REG_ENDPTSETUPSTAT = EPSETUP_STATUS_EP0;
+
+ /* Stop pending control transfers */
+ for(i=0;i<2;i++) {
+ if(qh_array[i].wait) {
+ qh_array[i].wait=0;
+ qh_array[i].status=DTD_STATUS_HALTED;
+ wakeup_signal(&transfer_completion_signal[i]);
+ }
+ }
+
+ usb_core_control_request((struct usb_ctrlrequest*)tmp);
+}
+
+static void transfer_completed(void)
+{
+ int ep;
+ unsigned int mask = REG_ENDPTCOMPLETE;
+ REG_ENDPTCOMPLETE = mask;
+
+ for (ep=0; ep<NUM_ENDPOINTS; ep++) {
+ int dir;
+ for (dir=0; dir<2; dir++) {
+ int pipe = ep * 2 + dir;
+ if (mask & pipe2mask[pipe]) {
+ struct queue_head* qh = &qh_array[pipe];
+ struct transfer_descriptor *td = &td_array[pipe];
+
+ if(td->size_ioc_sts & DTD_STATUS_ACTIVE) {
+ /* TODO this shouldn't happen, but...*/
+ break;
+ }
+ if((td->size_ioc_sts & DTD_PACKET_SIZE) >> DTD_LENGTH_BIT_POS != 0 && dir==0) {
+ /* We got less data than we asked for. */
+ }
+ qh->length = (td->reserved & DTD_RESERVED_LENGTH_MASK) -
+ ((td->size_ioc_sts & DTD_PACKET_SIZE) >> DTD_LENGTH_BIT_POS);
+ if(td->size_ioc_sts & DTD_ERROR_MASK) {
+ logf("pipe %d err %x", pipe, td->size_ioc_sts & DTD_ERROR_MASK);
+ qh->status |= td->size_ioc_sts & DTD_ERROR_MASK;
+ /* TODO we need to handle this somehow. Flush the endpoint ? */
+ }
+ if(qh->wait) {
+ qh->wait=0;
+ wakeup_signal(&transfer_completion_signal[pipe]);
+ }
+ usb_core_transfer_complete(ep, dir, qh->status, qh->length);
+ }
+ }
+ }
+}
+
+/* manual: 32.14.2.1 Bus Reset */
+static void bus_reset(void)
+{
+ int i;
+ logf("usb bus_reset");
+
+ REG_DEVICEADDR = 0;
+ REG_ENDPTSETUPSTAT = REG_ENDPTSETUPSTAT;
+ REG_ENDPTCOMPLETE = REG_ENDPTCOMPLETE;
+
+ for (i=0; i<100; i++) {
+ if (!REG_ENDPTPRIME)
+ break;
+
+ if (REG_USBSTS & USBSTS_RESET) {
+ logf("usb: double reset");
+ return;
+ }
+#if CONFIG_CPU == IMX31L
+ int x;
+ for (x = 0; x < 30000; x++)
+ asm volatile ("");
+#else
+ udelay(100);
+#endif
+ }
+ if (REG_ENDPTPRIME) {
+ logf("usb: short reset timeout");
+ }
+
+ usb_drv_cancel_all_transfers();
+
+ if (!(REG_PORTSC1 & PORTSCX_PORT_RESET)) {
+ logf("usb: slow reset!");
+ }
+}
+
+/* manual: 32.14.4.1 Queue Head Initialization */
+static void init_control_queue_heads(void)
+{
+ memset(qh_array, 0, sizeof qh_array);
+
+ /*** control ***/
+ qh_array[EP_CONTROL].max_pkt_length = 64 << QH_MAX_PKT_LEN_POS | QH_IOS;
+ qh_array[EP_CONTROL].dtd.next_td_ptr = QH_NEXT_TERMINATE;
+ qh_array[EP_CONTROL+1].max_pkt_length = 64 << QH_MAX_PKT_LEN_POS;
+ qh_array[EP_CONTROL+1].dtd.next_td_ptr = QH_NEXT_TERMINATE;
+}
+/* manual: 32.14.4.1 Queue Head Initialization */
+static void init_bulk_queue_heads(void)
+{
+ int tx_packetsize;
+ int rx_packetsize;
+ int i;
+
+ if (usb_drv_port_speed()) {
+ rx_packetsize = 512;
+ tx_packetsize = 512;
+ }
+ else {
+ rx_packetsize = 64;
+ tx_packetsize = 64;
+ }
+
+ /*** bulk ***/
+ for(i=1;i<NUM_ENDPOINTS;i++) {
+ qh_array[i*2].max_pkt_length = rx_packetsize << QH_MAX_PKT_LEN_POS | QH_ZLT_SEL;
+ qh_array[i*2].dtd.next_td_ptr = QH_NEXT_TERMINATE;
+ qh_array[i*2+1].max_pkt_length = tx_packetsize << QH_MAX_PKT_LEN_POS | QH_ZLT_SEL;
+ qh_array[i*2+1].dtd.next_td_ptr = QH_NEXT_TERMINATE;
+ }
+}
+
+static void init_endpoints(void)
+{
+ int i;
+ /* bulk */
+ for(i=1;i<NUM_ENDPOINTS;i++) {
+ REG_ENDPTCTRL(i) =
+ EPCTRL_RX_DATA_TOGGLE_RST | EPCTRL_RX_ENABLE |
+ EPCTRL_TX_DATA_TOGGLE_RST | EPCTRL_TX_ENABLE |
+ (EPCTRL_EP_TYPE_BULK << EPCTRL_RX_EP_TYPE_SHIFT) |
+ (EPCTRL_EP_TYPE_BULK << EPCTRL_TX_EP_TYPE_SHIFT);
+ }
+}