1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0+
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * bdc.h - header for the BRCM BDC USB3.0 device controller
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Copyright (C) 2014 Broadcom Corporation
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * Author: Ashwini Pahuja
8*4882a593Smuzhiyun */
9*4882a593Smuzhiyun
10*4882a593Smuzhiyun #ifndef __LINUX_BDC_H__
11*4882a593Smuzhiyun #define __LINUX_BDC_H__
12*4882a593Smuzhiyun
13*4882a593Smuzhiyun #include <linux/kernel.h>
14*4882a593Smuzhiyun #include <linux/usb.h>
15*4882a593Smuzhiyun #include <linux/device.h>
16*4882a593Smuzhiyun #include <linux/spinlock.h>
17*4882a593Smuzhiyun #include <linux/list.h>
18*4882a593Smuzhiyun #include <linux/dma-mapping.h>
19*4882a593Smuzhiyun #include <linux/mm.h>
20*4882a593Smuzhiyun #include <linux/debugfs.h>
21*4882a593Smuzhiyun #include <linux/usb/ch9.h>
22*4882a593Smuzhiyun #include <linux/usb/gadget.h>
23*4882a593Smuzhiyun #include <asm/unaligned.h>
24*4882a593Smuzhiyun
25*4882a593Smuzhiyun #define BRCM_BDC_NAME "bdc"
26*4882a593Smuzhiyun #define BRCM_BDC_DESC "Broadcom USB Device Controller driver"
27*4882a593Smuzhiyun
28*4882a593Smuzhiyun #define DMA_ADDR_INVALID (~(dma_addr_t)0)
29*4882a593Smuzhiyun
30*4882a593Smuzhiyun /* BDC command operation timeout in usec*/
31*4882a593Smuzhiyun #define BDC_CMD_TIMEOUT 1000
32*4882a593Smuzhiyun /* BDC controller operation timeout in usec*/
33*4882a593Smuzhiyun #define BDC_COP_TIMEOUT 500
34*4882a593Smuzhiyun
35*4882a593Smuzhiyun /*
36*4882a593Smuzhiyun * Maximum size of ep0 response buffer for ch9 requests,
37*4882a593Smuzhiyun * the set_sel request uses 6 so far, the max.
38*4882a593Smuzhiyun */
39*4882a593Smuzhiyun #define EP0_RESPONSE_BUFF 6
40*4882a593Smuzhiyun /* Start with SS as default */
41*4882a593Smuzhiyun #define EP0_MAX_PKT_SIZE 512
42*4882a593Smuzhiyun
43*4882a593Smuzhiyun /* 64 entries in a SRR */
44*4882a593Smuzhiyun #define NUM_SR_ENTRIES 64
45*4882a593Smuzhiyun
46*4882a593Smuzhiyun /* Num of bds per table */
47*4882a593Smuzhiyun #define NUM_BDS_PER_TABLE 64
48*4882a593Smuzhiyun
49*4882a593Smuzhiyun /* Num of tables in bd list for control,bulk and Int ep */
50*4882a593Smuzhiyun #define NUM_TABLES 2
51*4882a593Smuzhiyun
52*4882a593Smuzhiyun /* Num of tables in bd list for Isoch ep */
53*4882a593Smuzhiyun #define NUM_TABLES_ISOCH 6
54*4882a593Smuzhiyun
55*4882a593Smuzhiyun /* U1 Timeout default: 248usec */
56*4882a593Smuzhiyun #define U1_TIMEOUT 0xf8
57*4882a593Smuzhiyun
58*4882a593Smuzhiyun /* Interrupt coalescence in usec */
59*4882a593Smuzhiyun #define INT_CLS 500
60*4882a593Smuzhiyun
61*4882a593Smuzhiyun /* Register offsets */
62*4882a593Smuzhiyun /* Configuration and Capability registers */
63*4882a593Smuzhiyun #define BDC_BDCCFG0 0x00
64*4882a593Smuzhiyun #define BDC_BDCCFG1 0x04
65*4882a593Smuzhiyun #define BDC_BDCCAP0 0x08
66*4882a593Smuzhiyun #define BDC_BDCCAP1 0x0c
67*4882a593Smuzhiyun #define BDC_CMDPAR0 0x10
68*4882a593Smuzhiyun #define BDC_CMDPAR1 0x14
69*4882a593Smuzhiyun #define BDC_CMDPAR2 0x18
70*4882a593Smuzhiyun #define BDC_CMDSC 0x1c
71*4882a593Smuzhiyun #define BDC_USPC 0x20
72*4882a593Smuzhiyun #define BDC_USPPMS 0x28
73*4882a593Smuzhiyun #define BDC_USPPM2 0x2c
74*4882a593Smuzhiyun #define BDC_SPBBAL 0x38
75*4882a593Smuzhiyun #define BDC_SPBBAH 0x3c
76*4882a593Smuzhiyun #define BDC_BDCSC 0x40
77*4882a593Smuzhiyun #define BDC_XSFNTF 0x4c
78*4882a593Smuzhiyun
79*4882a593Smuzhiyun #define BDC_DVCSA 0x50
80*4882a593Smuzhiyun #define BDC_DVCSB 0x54
81*4882a593Smuzhiyun #define BDC_EPSTS0 0x60
82*4882a593Smuzhiyun #define BDC_EPSTS1 0x64
83*4882a593Smuzhiyun #define BDC_EPSTS2 0x68
84*4882a593Smuzhiyun #define BDC_EPSTS3 0x6c
85*4882a593Smuzhiyun #define BDC_EPSTS4 0x70
86*4882a593Smuzhiyun #define BDC_EPSTS5 0x74
87*4882a593Smuzhiyun #define BDC_EPSTS6 0x78
88*4882a593Smuzhiyun #define BDC_EPSTS7 0x7c
89*4882a593Smuzhiyun #define BDC_SRRBAL(n) (0x200 + (n * 0x10))
90*4882a593Smuzhiyun #define BDC_SRRBAH(n) (0x204 + (n * 0x10))
91*4882a593Smuzhiyun #define BDC_SRRINT(n) (0x208 + (n * 0x10))
92*4882a593Smuzhiyun #define BDC_INTCTLS(n) (0x20c + (n * 0x10))
93*4882a593Smuzhiyun
94*4882a593Smuzhiyun /* Extended capability regs */
95*4882a593Smuzhiyun #define BDC_FSCNOC 0xcd4
96*4882a593Smuzhiyun #define BDC_FSCNIC 0xce4
97*4882a593Smuzhiyun #define NUM_NCS(p) (p >> 28)
98*4882a593Smuzhiyun
99*4882a593Smuzhiyun /* Register bit fields and Masks */
100*4882a593Smuzhiyun /* BDC Configuration 0 */
101*4882a593Smuzhiyun #define BDC_PGS(p) (((p) & (0x7 << 8)) >> 8)
102*4882a593Smuzhiyun #define BDC_SPB(p) (p & 0x7)
103*4882a593Smuzhiyun
104*4882a593Smuzhiyun /* BDC Capability1 */
105*4882a593Smuzhiyun #define BDC_P64 (1 << 0)
106*4882a593Smuzhiyun
107*4882a593Smuzhiyun /* BDC Command register */
108*4882a593Smuzhiyun #define BDC_CMD_FH 0xe
109*4882a593Smuzhiyun #define BDC_CMD_DNC 0x6
110*4882a593Smuzhiyun #define BDC_CMD_EPO 0x4
111*4882a593Smuzhiyun #define BDC_CMD_BLA 0x3
112*4882a593Smuzhiyun #define BDC_CMD_EPC 0x2
113*4882a593Smuzhiyun #define BDC_CMD_DVC 0x1
114*4882a593Smuzhiyun #define BDC_CMD_CWS (0x1 << 5)
115*4882a593Smuzhiyun #define BDC_CMD_CST(p) (((p) & (0xf << 6))>>6)
116*4882a593Smuzhiyun #define BDC_CMD_EPN(p) ((p & 0x1f) << 10)
117*4882a593Smuzhiyun #define BDC_SUB_CMD_ADD (0x1 << 17)
118*4882a593Smuzhiyun #define BDC_SUB_CMD_FWK (0x4 << 17)
119*4882a593Smuzhiyun /* Reset sequence number */
120*4882a593Smuzhiyun #define BDC_CMD_EPO_RST_SN (0x1 << 16)
121*4882a593Smuzhiyun #define BDC_CMD_EP0_XSD (0x1 << 16)
122*4882a593Smuzhiyun #define BDC_SUB_CMD_ADD_EP (0x1 << 17)
123*4882a593Smuzhiyun #define BDC_SUB_CMD_DRP_EP (0x2 << 17)
124*4882a593Smuzhiyun #define BDC_SUB_CMD_EP_STP (0x2 << 17)
125*4882a593Smuzhiyun #define BDC_SUB_CMD_EP_STL (0x4 << 17)
126*4882a593Smuzhiyun #define BDC_SUB_CMD_EP_RST (0x1 << 17)
127*4882a593Smuzhiyun #define BDC_CMD_SRD (1 << 27)
128*4882a593Smuzhiyun
129*4882a593Smuzhiyun /* CMD completion status */
130*4882a593Smuzhiyun #define BDC_CMDS_SUCC 0x1
131*4882a593Smuzhiyun #define BDC_CMDS_PARA 0x3
132*4882a593Smuzhiyun #define BDC_CMDS_STAT 0x4
133*4882a593Smuzhiyun #define BDC_CMDS_FAIL 0x5
134*4882a593Smuzhiyun #define BDC_CMDS_INTL 0x6
135*4882a593Smuzhiyun #define BDC_CMDS_BUSY 0xf
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun /* CMDSC Param 2 shifts */
138*4882a593Smuzhiyun #define EPT_SHIFT 22
139*4882a593Smuzhiyun #define MP_SHIFT 10
140*4882a593Smuzhiyun #define MB_SHIFT 6
141*4882a593Smuzhiyun #define EPM_SHIFT 4
142*4882a593Smuzhiyun
143*4882a593Smuzhiyun /* BDC USPSC */
144*4882a593Smuzhiyun #define BDC_VBC (1 << 31)
145*4882a593Smuzhiyun #define BDC_PRC (1 << 30)
146*4882a593Smuzhiyun #define BDC_PCE (1 << 29)
147*4882a593Smuzhiyun #define BDC_CFC (1 << 28)
148*4882a593Smuzhiyun #define BDC_PCC (1 << 27)
149*4882a593Smuzhiyun #define BDC_PSC (1 << 26)
150*4882a593Smuzhiyun #define BDC_VBS (1 << 25)
151*4882a593Smuzhiyun #define BDC_PRS (1 << 24)
152*4882a593Smuzhiyun #define BDC_PCS (1 << 23)
153*4882a593Smuzhiyun #define BDC_PSP(p) (((p) & (0x7 << 20))>>20)
154*4882a593Smuzhiyun #define BDC_SCN (1 << 8)
155*4882a593Smuzhiyun #define BDC_SDC (1 << 7)
156*4882a593Smuzhiyun #define BDC_SWS (1 << 4)
157*4882a593Smuzhiyun
158*4882a593Smuzhiyun #define BDC_USPSC_RW (BDC_SCN|BDC_SDC|BDC_SWS|0xf)
159*4882a593Smuzhiyun #define BDC_PSP(p) (((p) & (0x7 << 20))>>20)
160*4882a593Smuzhiyun
161*4882a593Smuzhiyun #define BDC_SPEED_FS 0x1
162*4882a593Smuzhiyun #define BDC_SPEED_LS 0x2
163*4882a593Smuzhiyun #define BDC_SPEED_HS 0x3
164*4882a593Smuzhiyun #define BDC_SPEED_SS 0x4
165*4882a593Smuzhiyun
166*4882a593Smuzhiyun #define BDC_PST(p) (p & 0xf)
167*4882a593Smuzhiyun #define BDC_PST_MASK 0xf
168*4882a593Smuzhiyun
169*4882a593Smuzhiyun /* USPPMS */
170*4882a593Smuzhiyun #define BDC_U2E (0x1 << 31)
171*4882a593Smuzhiyun #define BDC_U1E (0x1 << 30)
172*4882a593Smuzhiyun #define BDC_U2A (0x1 << 29)
173*4882a593Smuzhiyun #define BDC_PORT_W1S (0x1 << 17)
174*4882a593Smuzhiyun #define BDC_U1T(p) ((p) & 0xff)
175*4882a593Smuzhiyun #define BDC_U2T(p) (((p) & 0xff) << 8)
176*4882a593Smuzhiyun #define BDC_U1T_MASK 0xff
177*4882a593Smuzhiyun
178*4882a593Smuzhiyun /* USBPM2 */
179*4882a593Smuzhiyun /* Hardware LPM Enable */
180*4882a593Smuzhiyun #define BDC_HLE (1 << 16)
181*4882a593Smuzhiyun
182*4882a593Smuzhiyun /* BDC Status and Control */
183*4882a593Smuzhiyun #define BDC_COP_RST (1 << 29)
184*4882a593Smuzhiyun #define BDC_COP_RUN (2 << 29)
185*4882a593Smuzhiyun #define BDC_COP_STP (4 << 29)
186*4882a593Smuzhiyun
187*4882a593Smuzhiyun #define BDC_COP_MASK (BDC_COP_RST|BDC_COP_RUN|BDC_COP_STP)
188*4882a593Smuzhiyun
189*4882a593Smuzhiyun #define BDC_COS (1 << 28)
190*4882a593Smuzhiyun #define BDC_CSTS(p) (((p) & (0x7 << 20)) >> 20)
191*4882a593Smuzhiyun #define BDC_MASK_MCW (1 << 7)
192*4882a593Smuzhiyun #define BDC_GIE (1 << 1)
193*4882a593Smuzhiyun #define BDC_GIP (1 << 0)
194*4882a593Smuzhiyun
195*4882a593Smuzhiyun #define BDC_HLT 1
196*4882a593Smuzhiyun #define BDC_NOR 2
197*4882a593Smuzhiyun #define BDC_OIP 7
198*4882a593Smuzhiyun
199*4882a593Smuzhiyun /* Buffer descriptor and Status report bit fields and masks */
200*4882a593Smuzhiyun #define BD_TYPE_BITMASK (0xf)
201*4882a593Smuzhiyun #define BD_CHAIN 0xf
202*4882a593Smuzhiyun
203*4882a593Smuzhiyun #define BD_TFS_SHIFT 4
204*4882a593Smuzhiyun #define BD_SOT (1 << 26)
205*4882a593Smuzhiyun #define BD_EOT (1 << 27)
206*4882a593Smuzhiyun #define BD_ISP (1 << 29)
207*4882a593Smuzhiyun #define BD_IOC (1 << 30)
208*4882a593Smuzhiyun #define BD_SBF (1 << 31)
209*4882a593Smuzhiyun
210*4882a593Smuzhiyun #define BD_INTR_TARGET(p) (((p) & 0x1f) << 27)
211*4882a593Smuzhiyun
212*4882a593Smuzhiyun #define BDC_SRR_RWS (1 << 4)
213*4882a593Smuzhiyun #define BDC_SRR_RST (1 << 3)
214*4882a593Smuzhiyun #define BDC_SRR_ISR (1 << 2)
215*4882a593Smuzhiyun #define BDC_SRR_IE (1 << 1)
216*4882a593Smuzhiyun #define BDC_SRR_IP (1 << 0)
217*4882a593Smuzhiyun #define BDC_SRR_EPI(p) (((p) & (0xff << 24)) >> 24)
218*4882a593Smuzhiyun #define BDC_SRR_DPI(p) (((p) & (0xff << 16)) >> 16)
219*4882a593Smuzhiyun #define BDC_SRR_DPI_MASK 0x00ff0000
220*4882a593Smuzhiyun
221*4882a593Smuzhiyun #define MARK_CHAIN_BD (BD_CHAIN|BD_EOT|BD_SOT)
222*4882a593Smuzhiyun
223*4882a593Smuzhiyun /* Control transfer BD specific fields */
224*4882a593Smuzhiyun #define BD_DIR_IN (1 << 25)
225*4882a593Smuzhiyun
226*4882a593Smuzhiyun #define BDC_PTC_MASK 0xf0000000
227*4882a593Smuzhiyun
228*4882a593Smuzhiyun /* status report defines */
229*4882a593Smuzhiyun #define SR_XSF 0
230*4882a593Smuzhiyun #define SR_USPC 4
231*4882a593Smuzhiyun #define SR_BD_LEN(p) (p & 0xffffff)
232*4882a593Smuzhiyun
233*4882a593Smuzhiyun #define XSF_SUCC 0x1
234*4882a593Smuzhiyun #define XSF_SHORT 0x3
235*4882a593Smuzhiyun #define XSF_BABB 0x4
236*4882a593Smuzhiyun #define XSF_SETUP_RECV 0x6
237*4882a593Smuzhiyun #define XSF_DATA_START 0x7
238*4882a593Smuzhiyun #define XSF_STATUS_START 0x8
239*4882a593Smuzhiyun
240*4882a593Smuzhiyun #define XSF_STS(p) (((p) >> 28) & 0xf)
241*4882a593Smuzhiyun
242*4882a593Smuzhiyun /* Transfer BD fields */
243*4882a593Smuzhiyun #define BD_LEN(p) ((p) & 0x1ffff)
244*4882a593Smuzhiyun #define BD_LTF (1 << 25)
245*4882a593Smuzhiyun #define BD_TYPE_DS 0x1
246*4882a593Smuzhiyun #define BD_TYPE_SS 0x2
247*4882a593Smuzhiyun
248*4882a593Smuzhiyun #define BDC_EP_ENABLED (1 << 0)
249*4882a593Smuzhiyun #define BDC_EP_STALL (1 << 1)
250*4882a593Smuzhiyun #define BDC_EP_STOP (1 << 2)
251*4882a593Smuzhiyun
252*4882a593Smuzhiyun /* One BD can transfer max 65536 bytes */
253*4882a593Smuzhiyun #define BD_MAX_BUFF_SIZE (1 << 16)
254*4882a593Smuzhiyun /* Maximum bytes in one XFR, Refer to BDC spec */
255*4882a593Smuzhiyun #define MAX_XFR_LEN 16777215
256*4882a593Smuzhiyun
257*4882a593Smuzhiyun /* defines for Force Header command */
258*4882a593Smuzhiyun #define DEV_NOTF_TYPE 6
259*4882a593Smuzhiyun #define FWK_SUBTYPE 1
260*4882a593Smuzhiyun #define TRA_PACKET 4
261*4882a593Smuzhiyun
262*4882a593Smuzhiyun #define to_bdc_ep(e) container_of(e, struct bdc_ep, usb_ep)
263*4882a593Smuzhiyun #define to_bdc_req(r) container_of(r, struct bdc_req, usb_req)
264*4882a593Smuzhiyun #define gadget_to_bdc(g) container_of(g, struct bdc, gadget)
265*4882a593Smuzhiyun
266*4882a593Smuzhiyun /* FUNCTION WAKE DEV NOTIFICATION interval, USB3 spec table 8.13 */
267*4882a593Smuzhiyun #define BDC_TNOTIFY 2500 /*in ms*/
268*4882a593Smuzhiyun /* Devstatus bitfields */
269*4882a593Smuzhiyun #define REMOTE_WAKEUP_ISSUED (1 << 16)
270*4882a593Smuzhiyun #define DEVICE_SUSPENDED (1 << 17)
271*4882a593Smuzhiyun #define FUNC_WAKE_ISSUED (1 << 18)
272*4882a593Smuzhiyun #define REMOTE_WAKE_ENABLE (1 << USB_DEVICE_REMOTE_WAKEUP)
273*4882a593Smuzhiyun
274*4882a593Smuzhiyun /* On disconnect, preserve these bits and clear rest */
275*4882a593Smuzhiyun #define DEVSTATUS_CLEAR (1 << USB_DEVICE_SELF_POWERED)
276*4882a593Smuzhiyun /* Hardware and software Data structures */
277*4882a593Smuzhiyun
278*4882a593Smuzhiyun /* Endpoint bd: buffer descriptor */
279*4882a593Smuzhiyun struct bdc_bd {
280*4882a593Smuzhiyun __le32 offset[4];
281*4882a593Smuzhiyun };
282*4882a593Smuzhiyun
283*4882a593Smuzhiyun /* Status report in Status report ring(srr) */
284*4882a593Smuzhiyun struct bdc_sr {
285*4882a593Smuzhiyun __le32 offset[4];
286*4882a593Smuzhiyun };
287*4882a593Smuzhiyun
288*4882a593Smuzhiyun /* bd_table: contiguous bd's in a table */
289*4882a593Smuzhiyun struct bd_table {
290*4882a593Smuzhiyun struct bdc_bd *start_bd;
291*4882a593Smuzhiyun /* dma address of start bd of table*/
292*4882a593Smuzhiyun dma_addr_t dma;
293*4882a593Smuzhiyun };
294*4882a593Smuzhiyun
295*4882a593Smuzhiyun /*
296*4882a593Smuzhiyun * Each endpoint has a bdl(buffer descriptor list), bdl consists of 1 or more bd
297*4882a593Smuzhiyun * table's chained to each other through a chain bd, every table has equal
298*4882a593Smuzhiyun * number of bds. the software uses bdi(bd index) to refer to particular bd in
299*4882a593Smuzhiyun * the list.
300*4882a593Smuzhiyun */
301*4882a593Smuzhiyun struct bd_list {
302*4882a593Smuzhiyun /* Array of bd table pointers*/
303*4882a593Smuzhiyun struct bd_table **bd_table_array;
304*4882a593Smuzhiyun /* How many tables chained to each other */
305*4882a593Smuzhiyun int num_tabs;
306*4882a593Smuzhiyun /* Max_bdi = num_tabs * num_bds_table - 1 */
307*4882a593Smuzhiyun int max_bdi;
308*4882a593Smuzhiyun /* current enq bdi from sw point of view */
309*4882a593Smuzhiyun int eqp_bdi;
310*4882a593Smuzhiyun /* current deq bdi from sw point of view */
311*4882a593Smuzhiyun int hwd_bdi;
312*4882a593Smuzhiyun /* numbers of bds per table */
313*4882a593Smuzhiyun int num_bds_table;
314*4882a593Smuzhiyun };
315*4882a593Smuzhiyun
316*4882a593Smuzhiyun struct bdc_req;
317*4882a593Smuzhiyun
318*4882a593Smuzhiyun /* Representation of a transfer, one transfer can have multiple bd's */
319*4882a593Smuzhiyun struct bd_transfer {
320*4882a593Smuzhiyun struct bdc_req *req;
321*4882a593Smuzhiyun /* start bd index */
322*4882a593Smuzhiyun int start_bdi;
323*4882a593Smuzhiyun /* this will be the next hw dqp when this transfer completes */
324*4882a593Smuzhiyun int next_hwd_bdi;
325*4882a593Smuzhiyun /* number of bds in this transfer */
326*4882a593Smuzhiyun int num_bds;
327*4882a593Smuzhiyun };
328*4882a593Smuzhiyun
329*4882a593Smuzhiyun /*
330*4882a593Smuzhiyun * Representation of a gadget request, every gadget request is contained
331*4882a593Smuzhiyun * by 1 bd_transfer.
332*4882a593Smuzhiyun */
333*4882a593Smuzhiyun struct bdc_req {
334*4882a593Smuzhiyun struct usb_request usb_req;
335*4882a593Smuzhiyun struct list_head queue;
336*4882a593Smuzhiyun struct bdc_ep *ep;
337*4882a593Smuzhiyun /* only one Transfer per request */
338*4882a593Smuzhiyun struct bd_transfer bd_xfr;
339*4882a593Smuzhiyun int epnum;
340*4882a593Smuzhiyun };
341*4882a593Smuzhiyun
342*4882a593Smuzhiyun /* scratchpad buffer needed by bdc hardware */
343*4882a593Smuzhiyun struct bdc_scratchpad {
344*4882a593Smuzhiyun dma_addr_t sp_dma;
345*4882a593Smuzhiyun void *buff;
346*4882a593Smuzhiyun u32 size;
347*4882a593Smuzhiyun };
348*4882a593Smuzhiyun
349*4882a593Smuzhiyun /* endpoint representation */
350*4882a593Smuzhiyun struct bdc_ep {
351*4882a593Smuzhiyun struct usb_ep usb_ep;
352*4882a593Smuzhiyun struct list_head queue;
353*4882a593Smuzhiyun struct bdc *bdc;
354*4882a593Smuzhiyun u8 ep_type;
355*4882a593Smuzhiyun u8 dir;
356*4882a593Smuzhiyun u8 ep_num;
357*4882a593Smuzhiyun const struct usb_ss_ep_comp_descriptor *comp_desc;
358*4882a593Smuzhiyun const struct usb_endpoint_descriptor *desc;
359*4882a593Smuzhiyun unsigned int flags;
360*4882a593Smuzhiyun char name[20];
361*4882a593Smuzhiyun /* endpoint bd list*/
362*4882a593Smuzhiyun struct bd_list bd_list;
363*4882a593Smuzhiyun /*
364*4882a593Smuzhiyun * HW generates extra event for multi bd tranfers, this flag helps in
365*4882a593Smuzhiyun * ignoring the extra event
366*4882a593Smuzhiyun */
367*4882a593Smuzhiyun bool ignore_next_sr;
368*4882a593Smuzhiyun };
369*4882a593Smuzhiyun
370*4882a593Smuzhiyun /* bdc cmmand parameter structure */
371*4882a593Smuzhiyun struct bdc_cmd_params {
372*4882a593Smuzhiyun u32 param2;
373*4882a593Smuzhiyun u32 param1;
374*4882a593Smuzhiyun u32 param0;
375*4882a593Smuzhiyun };
376*4882a593Smuzhiyun
377*4882a593Smuzhiyun /* status report ring(srr), currently one srr is supported for entire system */
378*4882a593Smuzhiyun struct srr {
379*4882a593Smuzhiyun struct bdc_sr *sr_bds;
380*4882a593Smuzhiyun u16 eqp_index;
381*4882a593Smuzhiyun u16 dqp_index;
382*4882a593Smuzhiyun dma_addr_t dma_addr;
383*4882a593Smuzhiyun };
384*4882a593Smuzhiyun
385*4882a593Smuzhiyun /* EP0 states */
386*4882a593Smuzhiyun enum bdc_ep0_state {
387*4882a593Smuzhiyun WAIT_FOR_SETUP = 0,
388*4882a593Smuzhiyun WAIT_FOR_DATA_START,
389*4882a593Smuzhiyun WAIT_FOR_DATA_XMIT,
390*4882a593Smuzhiyun WAIT_FOR_STATUS_START,
391*4882a593Smuzhiyun WAIT_FOR_STATUS_XMIT,
392*4882a593Smuzhiyun STATUS_PENDING
393*4882a593Smuzhiyun };
394*4882a593Smuzhiyun
395*4882a593Smuzhiyun /* Link states */
396*4882a593Smuzhiyun enum bdc_link_state {
397*4882a593Smuzhiyun BDC_LINK_STATE_U0 = 0x00,
398*4882a593Smuzhiyun BDC_LINK_STATE_U3 = 0x03,
399*4882a593Smuzhiyun BDC_LINK_STATE_RX_DET = 0x05,
400*4882a593Smuzhiyun BDC_LINK_STATE_RESUME = 0x0f
401*4882a593Smuzhiyun };
402*4882a593Smuzhiyun
403*4882a593Smuzhiyun /* representation of bdc */
404*4882a593Smuzhiyun struct bdc {
405*4882a593Smuzhiyun struct usb_gadget gadget;
406*4882a593Smuzhiyun struct usb_gadget_driver *gadget_driver;
407*4882a593Smuzhiyun struct device *dev;
408*4882a593Smuzhiyun /* device lock */
409*4882a593Smuzhiyun spinlock_t lock;
410*4882a593Smuzhiyun
411*4882a593Smuzhiyun /* generic phy */
412*4882a593Smuzhiyun struct phy **phys;
413*4882a593Smuzhiyun int num_phys;
414*4882a593Smuzhiyun /* num of endpoints for a particular instantiation of IP */
415*4882a593Smuzhiyun unsigned int num_eps;
416*4882a593Smuzhiyun /*
417*4882a593Smuzhiyun * Array of ep's, it uses the same index covention as bdc hw i.e.
418*4882a593Smuzhiyun * 1 for ep0, 2 for 1out,3 for 1in ....
419*4882a593Smuzhiyun */
420*4882a593Smuzhiyun struct bdc_ep **bdc_ep_array;
421*4882a593Smuzhiyun void __iomem *regs;
422*4882a593Smuzhiyun struct bdc_scratchpad scratchpad;
423*4882a593Smuzhiyun u32 sp_buff_size;
424*4882a593Smuzhiyun /* current driver supports 1 status ring */
425*4882a593Smuzhiyun struct srr srr;
426*4882a593Smuzhiyun /* Last received setup packet */
427*4882a593Smuzhiyun struct usb_ctrlrequest setup_pkt;
428*4882a593Smuzhiyun struct bdc_req ep0_req;
429*4882a593Smuzhiyun struct bdc_req status_req;
430*4882a593Smuzhiyun enum bdc_ep0_state ep0_state;
431*4882a593Smuzhiyun bool delayed_status;
432*4882a593Smuzhiyun bool zlp_needed;
433*4882a593Smuzhiyun bool reinit;
434*4882a593Smuzhiyun bool pullup;
435*4882a593Smuzhiyun /* Bits 0-15 are standard and 16-31 for proprietary information */
436*4882a593Smuzhiyun u32 devstatus;
437*4882a593Smuzhiyun int irq;
438*4882a593Smuzhiyun void *mem;
439*4882a593Smuzhiyun u32 dev_addr;
440*4882a593Smuzhiyun /* DMA pools */
441*4882a593Smuzhiyun struct dma_pool *bd_table_pool;
442*4882a593Smuzhiyun u8 test_mode;
443*4882a593Smuzhiyun /* array of callbacks for various status report handlers */
444*4882a593Smuzhiyun void (*sr_handler[2])(struct bdc *, struct bdc_sr *);
445*4882a593Smuzhiyun /* ep0 callback handlers */
446*4882a593Smuzhiyun void (*sr_xsf_ep0[3])(struct bdc *, struct bdc_sr *);
447*4882a593Smuzhiyun /* ep0 response buffer for ch9 requests like GET_STATUS and SET_SEL */
448*4882a593Smuzhiyun unsigned char ep0_response_buff[EP0_RESPONSE_BUFF];
449*4882a593Smuzhiyun /*
450*4882a593Smuzhiyun * Timer to check if host resumed transfer after bdc sent Func wake
451*4882a593Smuzhiyun * notification packet after a remote wakeup. if not, then resend the
452*4882a593Smuzhiyun * Func Wake packet every 2.5 secs. Refer to USB3 spec section 8.5.6.4
453*4882a593Smuzhiyun */
454*4882a593Smuzhiyun struct delayed_work func_wake_notify;
455*4882a593Smuzhiyun struct clk *clk;
456*4882a593Smuzhiyun };
457*4882a593Smuzhiyun
bdc_readl(void __iomem * base,u32 offset)458*4882a593Smuzhiyun static inline u32 bdc_readl(void __iomem *base, u32 offset)
459*4882a593Smuzhiyun {
460*4882a593Smuzhiyun return readl(base + offset);
461*4882a593Smuzhiyun }
462*4882a593Smuzhiyun
bdc_writel(void __iomem * base,u32 offset,u32 value)463*4882a593Smuzhiyun static inline void bdc_writel(void __iomem *base, u32 offset, u32 value)
464*4882a593Smuzhiyun {
465*4882a593Smuzhiyun writel(value, base + offset);
466*4882a593Smuzhiyun }
467*4882a593Smuzhiyun
468*4882a593Smuzhiyun /* Buffer descriptor list operations */
469*4882a593Smuzhiyun void bdc_notify_xfr(struct bdc *, u32);
470*4882a593Smuzhiyun void bdc_softconn(struct bdc *);
471*4882a593Smuzhiyun void bdc_softdisconn(struct bdc *);
472*4882a593Smuzhiyun int bdc_run(struct bdc *);
473*4882a593Smuzhiyun int bdc_stop(struct bdc *);
474*4882a593Smuzhiyun int bdc_reset(struct bdc *);
475*4882a593Smuzhiyun int bdc_udc_init(struct bdc *);
476*4882a593Smuzhiyun void bdc_udc_exit(struct bdc *);
477*4882a593Smuzhiyun int bdc_reinit(struct bdc *);
478*4882a593Smuzhiyun
479*4882a593Smuzhiyun /* Status report handlers */
480*4882a593Smuzhiyun /* Upstream port status change sr */
481*4882a593Smuzhiyun void bdc_sr_uspc(struct bdc *, struct bdc_sr *);
482*4882a593Smuzhiyun /* transfer sr */
483*4882a593Smuzhiyun void bdc_sr_xsf(struct bdc *, struct bdc_sr *);
484*4882a593Smuzhiyun /* EP0 XSF handlers */
485*4882a593Smuzhiyun void bdc_xsf_ep0_setup_recv(struct bdc *, struct bdc_sr *);
486*4882a593Smuzhiyun void bdc_xsf_ep0_data_start(struct bdc *, struct bdc_sr *);
487*4882a593Smuzhiyun void bdc_xsf_ep0_status_start(struct bdc *, struct bdc_sr *);
488*4882a593Smuzhiyun
489*4882a593Smuzhiyun #endif /* __LINUX_BDC_H__ */
490