Add the rt linux 4.1.3-rt3 as base
[kvmfornfv.git] / kernel / drivers / usb / gadget / udc / bdc / bdc.h
diff --git a/kernel/drivers/usb/gadget/udc/bdc/bdc.h b/kernel/drivers/usb/gadget/udc/bdc/bdc.h
new file mode 100644 (file)
index 0000000..dc18a20
--- /dev/null
@@ -0,0 +1,490 @@
+/*
+ * bdc.h - header for the BRCM BDC USB3.0 device controller
+ *
+ * Copyright (C) 2014 Broadcom Corporation
+ *
+ * Author: Ashwini Pahuja
+ *
+ * This program is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License as published by the
+ * Free Software Foundation; either version 2 of the License, or (at your
+ * option) any later version.
+ *
+ */
+
+#ifndef        __LINUX_BDC_H__
+#define        __LINUX_BDC_H__
+
+#include <linux/kernel.h>
+#include <linux/usb.h>
+#include <linux/device.h>
+#include <linux/spinlock.h>
+#include <linux/list.h>
+#include <linux/dma-mapping.h>
+#include <linux/mm.h>
+#include <linux/debugfs.h>
+#include <linux/usb/ch9.h>
+#include <linux/usb/gadget.h>
+#include <asm/unaligned.h>
+
+#define BRCM_BDC_NAME "bdc_usb3"
+#define BRCM_BDC_DESC "BDC device controller driver"
+
+#define DMA_ADDR_INVALID        (~(dma_addr_t)0)
+
+/* BDC command operation timeout in usec*/
+#define BDC_CMD_TIMEOUT        1000
+/* BDC controller operation timeout in usec*/
+#define BDC_COP_TIMEOUT        500
+
+/*
+ * Maximum size of ep0 response buffer for ch9 requests,
+ * the set_sel request uses 6 so far, the max.
+*/
+#define EP0_RESPONSE_BUFF  6
+/* Start with SS as default */
+#define EP0_MAX_PKT_SIZE 512
+
+/* 64 entries in a SRR */
+#define NUM_SR_ENTRIES 64
+
+/* Num of bds per table */
+#define NUM_BDS_PER_TABLE      32
+
+/* Num of tables in bd list for control,bulk and Int ep */
+#define NUM_TABLES     2
+
+/* Num of tables in bd list for Isoch ep */
+#define NUM_TABLES_ISOCH       6
+
+/* U1 Timeout default: 248usec */
+#define U1_TIMEOUT     0xf8
+
+/* Interrupt coalescence in usec */
+#define        INT_CLS 500
+
+/* Register offsets */
+/* Configuration and Capability registers */
+#define BDC_BDCCFG0    0x00
+#define BDC_BDCCFG1    0x04
+#define BDC_BDCCAP0    0x08
+#define BDC_BDCCAP1    0x0c
+#define BDC_CMDPAR0    0x10
+#define BDC_CMDPAR1    0x14
+#define BDC_CMDPAR2    0x18
+#define BDC_CMDSC      0x1c
+#define BDC_USPC       0x20
+#define BDC_USPPMS     0x28
+#define BDC_USPPM2     0x2c
+#define BDC_SPBBAL     0x38
+#define BDC_SPBBAH     0x3c
+#define BDC_BDCSC      0x40
+#define BDC_XSFNTF     0x4c
+
+#define BDC_DVCSA      0x50
+#define BDC_DVCSB      0x54
+#define BDC_EPSTS0(n)  (0x60 + (n * 0x10))
+#define BDC_EPSTS1(n)  (0x64 + (n * 0x10))
+#define BDC_EPSTS2(n)  (0x68 + (n * 0x10))
+#define BDC_EPSTS3(n)  (0x6c + (n * 0x10))
+#define BDC_EPSTS4(n)  (0x70 + (n * 0x10))
+#define BDC_EPSTS5(n)  (0x74 + (n * 0x10))
+#define BDC_EPSTS6(n)  (0x78 + (n * 0x10))
+#define BDC_EPSTS7(n)  (0x7c + (n * 0x10))
+#define BDC_SRRBAL(n)  (0x200 + (n * 0x10))
+#define BDC_SRRBAH(n)  (0x204 + (n * 0x10))
+#define BDC_SRRINT(n)  (0x208 + (n * 0x10))
+#define BDC_INTCTLS(n) (0x20c + (n * 0x10))
+
+/* Extended capability regs */
+#define BDC_FSCNOC     0xcd4
+#define BDC_FSCNIC     0xce4
+#define NUM_NCS(p)     (p >> 28)
+
+/* Register bit fields and Masks */
+/* BDC Configuration 0 */
+#define BDC_PGS(p)     (((p) & (0x7 << 8)) >> 8)
+#define BDC_SPB(p)     (p & 0x7)
+
+/* BDC Capability1 */
+#define BDC_P64                (1 << 0)
+
+/* BDC Command register */
+#define BDC_CMD_FH     0xe
+#define BDC_CMD_DNC    0x6
+#define BDC_CMD_EPO    0x4
+#define BDC_CMD_BLA    0x3
+#define BDC_CMD_EPC    0x2
+#define BDC_CMD_DVC    0x1
+#define BDC_CMD_CWS            (0x1 << 5)
+#define BDC_CMD_CST(p)         (((p) & (0xf << 6))>>6)
+#define BDC_CMD_EPN(p)         ((p & 0x1f) << 10)
+#define BDC_SUB_CMD_ADD                (0x1 << 17)
+#define BDC_SUB_CMD_FWK                (0x4 << 17)
+/* Reset sequence number */
+#define BDC_CMD_EPO_RST_SN     (0x1 << 16)
+#define BDC_CMD_EP0_XSD                (0x1 << 16)
+#define BDC_SUB_CMD_ADD_EP     (0x1 << 17)
+#define BDC_SUB_CMD_DRP_EP     (0x2 << 17)
+#define BDC_SUB_CMD_EP_STP     (0x2 << 17)
+#define BDC_SUB_CMD_EP_STL     (0x4 << 17)
+#define BDC_SUB_CMD_EP_RST     (0x1 << 17)
+#define BDC_CMD_SRD            (1 << 27)
+
+/* CMD completion status */
+#define BDC_CMDS_SUCC  0x1
+#define BDC_CMDS_PARA  0x3
+#define BDC_CMDS_STAT  0x4
+#define BDC_CMDS_FAIL  0x5
+#define BDC_CMDS_INTL  0x6
+#define BDC_CMDS_BUSY  0xf
+
+/* CMDSC Param 2 shifts */
+#define EPT_SHIFT      22
+#define MP_SHIFT       10
+#define MB_SHIFT       6
+#define EPM_SHIFT      4
+
+/* BDC USPSC */
+#define BDC_VBC                (1 << 31)
+#define BDC_PRC                (1 << 30)
+#define BDC_PCE                (1 << 29)
+#define BDC_CFC                (1 << 28)
+#define BDC_PCC                (1 << 27)
+#define BDC_PSC                (1 << 26)
+#define BDC_VBS                (1 << 25)
+#define BDC_PRS                (1 << 24)
+#define BDC_PCS                (1 << 23)
+#define BDC_PSP(p)     (((p) & (0x7 << 20))>>20)
+#define BDC_SCN                (1 << 8)
+#define BDC_SDC                (1 << 7)
+#define BDC_SWS                (1 << 4)
+
+#define BDC_USPSC_RW   (BDC_SCN|BDC_SDC|BDC_SWS|0xf)
+#define BDC_PSP(p)     (((p) & (0x7 << 20))>>20)
+
+#define BDC_SPEED_FS   0x1
+#define BDC_SPEED_LS   0x2
+#define BDC_SPEED_HS   0x3
+#define BDC_SPEED_SS   0x4
+
+#define BDC_PST(p)     (p & 0xf)
+#define BDC_PST_MASK   0xf
+
+/* USPPMS */
+#define BDC_U2E                (0x1 << 31)
+#define BDC_U1E                (0x1 << 30)
+#define BDC_U2A                (0x1 << 29)
+#define BDC_PORT_W1S   (0x1 << 17)
+#define BDC_U1T(p)     ((p) & 0xff)
+#define BDC_U2T(p)     (((p) & 0xff) << 8)
+#define BDC_U1T_MASK   0xff
+
+/* USBPM2 */
+/* Hardware LPM Enable */
+#define BDC_HLE                (1 << 16)
+
+/* BDC Status and Control */
+#define BDC_COP_RST    (1 << 29)
+#define BDC_COP_RUN    (2 << 29)
+#define BDC_COP_STP    (4 << 29)
+
+#define BDC_COP_MASK (BDC_COP_RST|BDC_COP_RUN|BDC_COP_STP)
+
+#define BDC_COS                (1 << 28)
+#define BDC_CSTS(p)    (((p) & (0x7 << 20)) >> 20)
+#define BDC_MASK_MCW   (1 << 7)
+#define BDC_GIE                (1 << 1)
+#define BDC_GIP                (1 << 0)
+
+#define BDC_HLT        1
+#define BDC_NOR        2
+#define BDC_OIP        7
+
+/* Buffer descriptor and Status report bit fields and masks */
+#define BD_TYPE_BITMASK        (0xf)
+#define BD_CHAIN       0xf
+
+#define BD_TFS_SHIFT   4
+#define BD_SOT         (1 << 26)
+#define BD_EOT         (1 << 27)
+#define BD_ISP         (1 << 29)
+#define BD_IOC         (1 << 30)
+#define BD_SBF         (1 << 31)
+
+#define BD_INTR_TARGET(p)      (((p) & 0x1f) << 27)
+
+#define BDC_SRR_RWS            (1 << 4)
+#define BDC_SRR_RST            (1 << 3)
+#define BDC_SRR_ISR            (1 << 2)
+#define BDC_SRR_IE             (1 << 1)
+#define BDC_SRR_IP             (1 << 0)
+#define BDC_SRR_EPI(p) (((p) & (0xff << 24)) >> 24)
+#define BDC_SRR_DPI(p) (((p) & (0xff << 16)) >> 16)
+#define BDC_SRR_DPI_MASK       0x00ff0000
+
+#define MARK_CHAIN_BD  (BD_CHAIN|BD_EOT|BD_SOT)
+
+/* Control transfer BD specific fields */
+#define BD_DIR_IN              (1 << 25)
+
+#define BDC_PTC_MASK   0xf0000000
+
+/* status report defines */
+#define SR_XSF         0
+#define SR_USPC                4
+#define SR_BD_LEN(p)    (p & 0xffffff)
+
+#define XSF_SUCC       0x1
+#define XSF_SHORT      0x3
+#define XSF_BABB       0x4
+#define XSF_SETUP_RECV 0x6
+#define XSF_DATA_START 0x7
+#define XSF_STATUS_START 0x8
+
+#define XSF_STS(p) (((p) >> 28) & 0xf)
+
+/* Transfer BD fields */
+#define BD_LEN(p) ((p) & 0x1ffff)
+#define BD_LTF         (1 << 25)
+#define BD_TYPE_DS     0x1
+#define BD_TYPE_SS     0x2
+
+#define BDC_EP_ENABLED     (1 << 0)
+#define BDC_EP_STALL       (1 << 1)
+#define BDC_EP_STOP        (1 << 2)
+
+/* One BD can transfer max 65536 bytes */
+#define BD_MAX_BUFF_SIZE       (1 << 16)
+/* Maximum bytes in one XFR, Refer to BDC spec */
+#define MAX_XFR_LEN            16777215
+
+/* defines for Force Header command */
+#define DEV_NOTF_TYPE 6
+#define FWK_SUBTYPE  1
+#define TRA_PACKET   4
+
+#define to_bdc_ep(e)           container_of(e, struct bdc_ep, usb_ep)
+#define to_bdc_req(r)          container_of(r, struct bdc_req, usb_req)
+#define gadget_to_bdc(g)       container_of(g, struct bdc, gadget)
+
+/* FUNCTION WAKE DEV NOTIFICATION interval, USB3 spec table 8.13 */
+#define BDC_TNOTIFY 2500 /*in ms*/
+/* Devstatus bitfields */
+#define REMOTE_WAKEUP_ISSUED   (1 << 16)
+#define DEVICE_SUSPENDED       (1 << 17)
+#define FUNC_WAKE_ISSUED       (1 << 18)
+#define REMOTE_WAKE_ENABLE     (1 << USB_DEVICE_REMOTE_WAKEUP)
+
+/* On disconnect, preserve these bits and clear rest */
+#define DEVSTATUS_CLEAR                (1 << USB_DEVICE_SELF_POWERED)
+/* Hardware and software Data structures */
+
+/* Endpoint bd: buffer descriptor */
+struct bdc_bd {
+       __le32 offset[4];
+};
+
+/* Status report in Status report ring(srr) */
+struct bdc_sr {
+       __le32 offset[4];
+};
+
+/* bd_table: contigous bd's in a table */
+struct bd_table {
+       struct bdc_bd *start_bd;
+       /* dma address of start bd of table*/
+       dma_addr_t dma;
+};
+
+/*
+ * Each endpoint has a bdl(buffer descriptor list), bdl consists of 1 or more bd
+ * table's chained to each other through a chain bd, every table has equal
+ * number of bds. the software uses bdi(bd index) to refer to particular bd in
+ * the list.
+ */
+struct bd_list {
+       /* Array of bd table pointers*/
+       struct bd_table **bd_table_array;
+       /* How many tables chained to each other */
+       int num_tabs;
+       /* Max_bdi = num_tabs * num_bds_table - 1 */
+       int max_bdi;
+       /* current enq bdi from sw point of view */
+       int eqp_bdi;
+       /* current deq bdi from sw point of view */
+       int hwd_bdi;
+       /* numbers of bds per table */
+       int num_bds_table;
+};
+
+struct bdc_req;
+
+/* Representation of a transfer, one transfer can have multiple bd's */
+struct bd_transfer {
+       struct bdc_req *req;
+       /* start bd index */
+       int start_bdi;
+       /* this will be the next hw dqp when this transfer completes */
+       int next_hwd_bdi;
+       /* number of bds in this transfer */
+       int num_bds;
+};
+
+/*
+ * Representation of a gadget request, every gadget request is contained
+ * by 1 bd_transfer.
+ */
+struct bdc_req {
+       struct usb_request      usb_req;
+       struct list_head        queue;
+       struct bdc_ep           *ep;
+       /* only one Transfer per request */
+       struct bd_transfer bd_xfr;
+       int     epnum;
+};
+
+/* scratchpad buffer needed by bdc hardware */
+struct bdc_scratchpad {
+       dma_addr_t sp_dma;
+       void *buff;
+       u32 size;
+};
+
+/* endpoint representation */
+struct bdc_ep {
+       struct usb_ep   usb_ep;
+       struct list_head queue;
+       struct bdc *bdc;
+       u8  ep_type;
+       u8  dir;
+       u8  ep_num;
+       const struct usb_ss_ep_comp_descriptor  *comp_desc;
+       const struct usb_endpoint_descriptor    *desc;
+       unsigned int flags;
+       char name[20];
+       /* endpoint bd list*/
+       struct bd_list bd_list;
+       /*
+        * HW generates extra event for multi bd tranfers, this flag helps in
+        * ignoring the extra event
+        */
+       bool ignore_next_sr;
+};
+
+/* bdc cmmand parameter structure */
+struct bdc_cmd_params {
+       u32     param2;
+       u32     param1;
+       u32     param0;
+};
+
+/* status report ring(srr), currently one srr is supported for entire system */
+struct srr {
+       struct bdc_sr *sr_bds;
+       u16     eqp_index;
+       u16     dqp_index;
+       dma_addr_t      dma_addr;
+};
+
+/* EP0 states */
+enum bdc_ep0_state {
+       WAIT_FOR_SETUP = 0,
+       WAIT_FOR_DATA_START,
+       WAIT_FOR_DATA_XMIT,
+       WAIT_FOR_STATUS_START,
+       WAIT_FOR_STATUS_XMIT,
+       STATUS_PENDING
+};
+
+/* Link states */
+enum bdc_link_state {
+       BDC_LINK_STATE_U0       = 0x00,
+       BDC_LINK_STATE_U3       = 0x03,
+       BDC_LINK_STATE_RX_DET   = 0x05,
+       BDC_LINK_STATE_RESUME   = 0x0f
+};
+
+/* representation of bdc */
+struct bdc {
+       struct usb_gadget       gadget;
+       struct usb_gadget_driver        *gadget_driver;
+       struct device   *dev;
+       /* device lock */
+       spinlock_t      lock;
+
+       /* num of endpoints for a particular instantiation of IP */
+       unsigned int num_eps;
+       /*
+        * Array of ep's, it uses the same index covention as bdc hw i.e.
+        * 1 for ep0, 2 for 1out,3 for 1in ....
+        */
+       struct bdc_ep           **bdc_ep_array;
+       void __iomem            *regs;
+       struct bdc_scratchpad   scratchpad;
+       u32     sp_buff_size;
+       /* current driver supports 1 status ring */
+       struct srr      srr;
+       /* Last received setup packet */
+       struct  usb_ctrlrequest setup_pkt;
+       struct  bdc_req ep0_req;
+       struct  bdc_req status_req;
+       enum    bdc_ep0_state ep0_state;
+       bool    delayed_status;
+       bool    zlp_needed;
+       bool    reinit;
+       bool    pullup;
+       /* Bits 0-15 are standard and 16-31 for proprietary information */
+       u32     devstatus;
+       int     irq;
+       void    *mem;
+       u32     dev_addr;
+       /* DMA pools */
+       struct dma_pool *bd_table_pool;
+       u8              test_mode;
+       /* array of callbacks for various status report handlers */
+       void (*sr_handler[2])(struct bdc *, struct bdc_sr *);
+       /* ep0 callback handlers */
+       void (*sr_xsf_ep0[3])(struct bdc *, struct bdc_sr *);
+       /* ep0 response buffer for ch9 requests like GET_STATUS and SET_SEL */
+       unsigned char           ep0_response_buff[EP0_RESPONSE_BUFF];
+       /*
+        * Timer to check if host resumed transfer after bdc sent Func wake
+        * notification  packet after a remote wakeup. if not, then resend the
+        * Func Wake packet every 2.5 secs. Refer to USB3 spec section 8.5.6.4
+        */
+       struct delayed_work     func_wake_notify;
+};
+
+static inline u32 bdc_readl(void __iomem *base, u32 offset)
+{
+       return readl(base + offset);
+}
+
+static inline void bdc_writel(void __iomem *base, u32 offset, u32 value)
+{
+       writel(value, base + offset);
+}
+
+/* Buffer descriptor list operations */
+void bdc_notify_xfr(struct bdc *, u32);
+void bdc_softconn(struct bdc *);
+void bdc_softdisconn(struct bdc *);
+int bdc_run(struct bdc *);
+int bdc_stop(struct bdc *);
+int bdc_reset(struct bdc *);
+int bdc_udc_init(struct bdc *);
+void bdc_udc_exit(struct bdc *);
+int bdc_reinit(struct bdc *);
+
+/* Status report handlers */
+/* Upstream port status change sr */
+void bdc_sr_uspc(struct bdc *, struct bdc_sr *);
+/* transfer sr */
+void bdc_sr_xsf(struct bdc *, struct bdc_sr *);
+/* EP0 XSF handlers */
+void bdc_xsf_ep0_setup_recv(struct bdc *, struct bdc_sr *);
+void bdc_xsf_ep0_data_start(struct bdc *, struct bdc_sr *);
+void bdc_xsf_ep0_status_start(struct bdc *, struct bdc_sr *);
+
+#endif /* __LINUX_BDC_H__ */