blob: 1a0c0aa3757e9f6c99373d21f3b1f7bd740fb296 [file] [log] [blame]
Shimrit Malichi7cc54ba2013-03-12 15:31:18 +02001/* Copyright (c) 2013, The Linux Foundation. All rights reserved.
2 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 *
12 */
13
14#include <linux/kernel.h>
15#include <linux/usb/hbm.h>
16#include <mach/usb_bam.h>
17
18/**
19 * USB HBM Hardware registers.
20 *
21 */
22#define USB_OTG_HS_HBM_CFG (0x00000290)
23#define USB_OTG_HS_HBM_QH_MAP_PIPE(n) (0x00000294 + 4 * (n))
24#define USB_OTG_HS_HBM_PIPE_PRODUCER (0x00000314)
25#define USB_OTG_HS_HBM_PARK_MODE_DISABLE (0x00000318)
26#define USB_OTG_HS_HBM_PIPE_ZLT_DISABLE (0x0000031C)
27#define USB_OTG_HS_HBM_PIPE_EN (0x00000310)
28#define USB_OTG_HS_HBM_SW_RST (0x00000324)
29#define USB_OTG_HS_HBM_SB_SW_RST (0x00000320)
30#define USB_OTG_HS_USBCMD (0x00000140)
31#define USB_OTG_HS_USBSTS (0x00000144)
32
33/**
34 * USB HBM Hardware registers bitmask.
35 */
36#define HBM_EN 0x00000001
37#define ASE 0x20
38#define AS 0x8000
39#define PIPE_PRODUCER 1
40#define MAX_PIPE_NUM 16
41#define HBM_QH_MAP_PIPE 0xffffffc0
Shimrit Malichi6aad4162013-03-19 10:57:27 +020042#define QTD_CERR_MASK 0xfffff3ff
Shimrit Malichi7cc54ba2013-03-12 15:31:18 +020043
44struct hbm_msm {
45 u32 *base;
46 struct usb_hcd *hcd;
47};
48
49static struct hbm_msm *hbm_ctx;
50
51/**
52 * Read register masked field.
53 *
54 * @base - hbm base virtual address.
55 * @offset - register offset.
56 * @mask - register bitmask.
57 *
58 * @return u32
59 */
60static inline u32 hbm_msm_read_reg_field(void *base,
61 u32 offset, const u32 mask)
62{
63 u32 shift = find_first_bit((void *)&mask, 32);
64 u32 val = ioread32(base + offset);
65 val &= mask; /* clear other bits */
66 val >>= shift;
67 return val;
68}
69
70/**
71 * Write register field.
72 *
73 * @base - hbm base virtual address.
74 * @offset - register offset.
75 * @val - value to be written.
76 *
77 */
78static inline void hbm_msm_write_reg(void *base, u32 offset, u32 val)
79{
80 iowrite32(val, base + offset);
81}
82
83/**
84 * Write register masked field.
85 *
86 * @base - hbm base virtual address.
87 * @offset - register offset.
88 * @mask - register bitmask.
89 * @val - value to write.
90 *
91 */
92static inline void hbm_msm_write_reg_field(void *base, u32 offset,
93 const u32 mask, u32 val)
94{
95 u32 shift = find_first_bit((void *)&mask, 32);
96 u32 tmp = ioread32(base + offset);
97
98 tmp &= ~mask;
99 val = tmp | (val << shift);
100 iowrite32(val, base + offset);
101}
102
103/**
104 * Enable/disable park mode. Park mode enables executing up to 3 usb packets
105 * from each QH.
106 *
107 * @pipe_num - Connection index.
108 *
109 * @disable_park_mode - Enable/disable park mode.
110 *
111 */
112int set_disable_park_mode(u8 pipe_num, bool disable_park_mode)
113{
114 if (pipe_num >= MAX_PIPE_NUM) {
115 pr_err("%s: illegal pipe num %d", __func__, pipe_num);
116 return -EINVAL;
117 }
118
119 /* enable/disable park mode */
120 hbm_msm_write_reg_field(hbm_ctx->base,
121 USB_OTG_HS_HBM_PARK_MODE_DISABLE, 1 << pipe_num,
122 (disable_park_mode ? 1 : 0));
123 return 0;
124}
125
126/**
127 * Enable/disable zero length transfer.
128 *
129 * @pipe_num - Connection index.
130 *
131 * @disable_zlt - Enable/disable zlt.
132 *
133 */
134int set_disable_zlt(u8 pipe_num, bool disable_zlt)
135{
136 if (pipe_num >= MAX_PIPE_NUM) {
137 pr_err("%s: illegal pipe num %d", __func__, pipe_num);
138 return -EINVAL;
139 }
140
141 /* enable/disable zlt */
142 hbm_msm_write_reg_field(hbm_ctx->base,
143 USB_OTG_HS_HBM_PIPE_ZLT_DISABLE, 1 << pipe_num,
144 (disable_zlt ? 1 : 0));
145 return 0;
146}
147
148static void hbm_reset(bool reset)
149{
150 hbm_msm_write_reg_field(hbm_ctx->base, USB_OTG_HS_HBM_SW_RST, 1 << 0,
151 reset ? 1 : 0);
152}
153
154static void hbm_config(bool enable)
155{
156 hbm_msm_write_reg_field(hbm_ctx->base, USB_OTG_HS_HBM_CFG, HBM_EN,
157 enable ? 1 : 0);
158}
159
160int hbm_pipe_init(u32 QH_addr, u32 pipe_num, bool is_consumer)
161{
162 if (pipe_num >= MAX_PIPE_NUM) {
163 pr_err("%s: illegal pipe num %d", __func__, pipe_num);
164 return -EINVAL;
165 }
166
167 /* map QH(ep) <> pipe */
168 hbm_msm_write_reg(hbm_ctx->base,
169 USB_OTG_HS_HBM_QH_MAP_PIPE(pipe_num), QH_addr);
170
171 /* set pipe producer/consumer mode - (IN EP is producer) */
172 hbm_msm_write_reg_field(hbm_ctx->base,
173 USB_OTG_HS_HBM_PIPE_PRODUCER, 1 << pipe_num,
174 (is_consumer ? 0 : 1));
175
176 /* disable park mode as default */
177 set_disable_park_mode(pipe_num, true);
178
179 /* enable zlt as default*/
180 set_disable_zlt(pipe_num, false);
181
182 /* activate pipe */
183 hbm_msm_write_reg_field(hbm_ctx->base, USB_OTG_HS_HBM_PIPE_EN,
184 1 << pipe_num, 1);
185
186 return 0;
187}
188
189void hbm_init(struct usb_hcd *hcd)
190{
191 pr_info("%s\n", __func__);
192
193 hbm_ctx = kzalloc(sizeof(*hbm_ctx), GFP_KERNEL);
194 if (!hbm_ctx) {
195 pr_err("%s: hbm_ctx alloc failed\n", __func__);
196 return;
197 }
Shimrit Malichi7cc54ba2013-03-12 15:31:18 +0200198
199 hbm_ctx->base = hcd->regs;
200 hbm_ctx->hcd = hcd;
201
202 /* reset hbm */
203 hbm_reset(true);
204 /* delay was added to allow the reset process the end */
205 udelay(1000);
206 hbm_reset(false);
207 hbm_config(true);
208}
209
210void hbm_uninit(void)
211{
Shimrit Malichie8adf502013-03-13 18:16:30 +0200212 hbm_config(false);
Shimrit Malichi7cc54ba2013-03-12 15:31:18 +0200213 kfree(hbm_ctx);
214}
215
216static int hbm_submit_async(struct ehci_hcd *ehci, struct urb *urb,
217 struct list_head *qtd_list, gfp_t mem_flags)
218{
219 int epnum;
220 unsigned long flags;
221 struct ehci_qh *qh = NULL;
222 int rc;
223 struct usb_host_bam_type *bam =
224 (struct usb_host_bam_type *)urb->priv_data;
225
226 epnum = urb->ep->desc.bEndpointAddress;
227
228 spin_lock_irqsave(&ehci->lock, flags);
229 if (unlikely(!HCD_HW_ACCESSIBLE(ehci_to_hcd(ehci)))) {
230 rc = -ESHUTDOWN;
231 goto done;
232 }
233 rc = usb_hcd_link_urb_to_ep(ehci_to_hcd(ehci), urb);
234 if (unlikely(rc))
235 goto done;
236
237 qh = qh_append_tds(ehci, urb, qtd_list, epnum, &urb->ep->hcpriv);
238 if (unlikely(qh == NULL)) {
239 usb_hcd_unlink_urb_from_ep(ehci_to_hcd(ehci), urb);
240 rc = -ENOMEM;
241 goto done;
242 }
243
244 hbm_pipe_init(qh->qh_dma, bam->pipe_num, bam->dir);
245
246 if (likely(qh->qh_state == QH_STATE_IDLE))
247 qh_link_async(ehci, qh);
248
249done:
250 spin_unlock_irqrestore(&ehci->lock, flags);
251 if (unlikely(qh == NULL))
252 qtd_list_free(ehci, urb, qtd_list);
253 return rc;
254}
255
256int hbm_urb_enqueue(struct usb_hcd *hcd, struct urb *urb,
257 gfp_t mem_flags)
258{
259 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
260 struct list_head qtd_list;
Shimrit Malichi6aad4162013-03-19 10:57:27 +0200261 struct ehci_qtd *qtd;
Shimrit Malichi7cc54ba2013-03-12 15:31:18 +0200262
263 INIT_LIST_HEAD(&qtd_list);
264
265 if (usb_pipetype(urb->pipe) != PIPE_BULK) {
266 pr_err("%s pipe type is not BULK\n", __func__);
267 return -EINVAL;
268 }
269
270 /*no sg support*/
271 urb->transfer_buffer_length = 0;
272 urb->transfer_dma = 0;
273 urb->transfer_flags |= URB_NO_INTERRUPT;
274
275 if (!qh_urb_transaction(ehci, urb, &qtd_list, mem_flags))
276 return -ENOMEM;
Shimrit Malichi6aad4162013-03-19 10:57:27 +0200277
278 /* set err counter in qTD token to zero */
279 qtd = list_entry(qtd_list.next, struct ehci_qtd, qtd_list);
280 if (qtd != NULL)
281 qtd->hw_token &= QTD_CERR_MASK;
282
Shimrit Malichi7cc54ba2013-03-12 15:31:18 +0200283 return hbm_submit_async(ehci, urb, &qtd_list, mem_flags);
284}