mirror of https://gitee.com/openkylin/linux.git
491 lines
12 KiB
C
491 lines
12 KiB
C
/*
|
|
* 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: contiguous 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__ */
|