blob: b1562b89a8c0eff6d4a6050be1f106491ca06286 [file] [log] [blame]
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301/* ehci-msm-hsic.c - HSUSB Host Controller Driver Implementation
2 *
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +05303 * Copyright (c) 2011-2013, Linux Foundation. All rights reserved.
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05304 *
5 * Partly derived from ehci-fsl.c and ehci-hcd.c
6 * Copyright (c) 2000-2004 by David Brownell
7 * Copyright (c) 2005 MontaVista Software
8 *
9 * All source code in this file is licensed under the following license except
10 * where indicated.
11 *
12 * This program is free software; you can redistribute it and/or modify it
13 * under the terms of the GNU General Public License version 2 as published
14 * by the Free Software Foundation.
15 *
16 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
19 *
20 * See the GNU General Public License for more details.
21 * You should have received a copy of the GNU General Public License
22 * along with this program; if not, you can find it at http://www.fsf.org
23 */
24
25#include <linux/platform_device.h>
26#include <linux/clk.h>
27#include <linux/err.h>
Hemant Kumare6275972012-02-29 20:06:21 -080028#include <linux/debugfs.h>
29#include <linux/seq_file.h>
Sudhir Sharma1673e302012-08-27 17:37:24 -070030#include <linux/wakelock.h>
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053031#include <linux/pm_runtime.h>
32#include <linux/regulator/consumer.h>
33
Manu Gautam863b74132012-11-21 14:30:04 +053034#include <linux/usb/ulpi.h>
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053035#include <linux/usb/msm_hsusb_hw.h>
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +053036#include <linux/usb/msm_hsusb.h>
37#include <linux/gpio.h>
Manu Gautam863b74132012-11-21 14:30:04 +053038#include <linux/of_gpio.h>
Amit Blayd6ea6102012-06-07 16:26:24 +030039#include <linux/spinlock.h>
Hemant Kumarbbcdcbc2012-08-30 14:23:38 -070040#include <linux/irq.h>
Vamsi Krishnafce1bd22012-08-17 17:12:26 -070041#include <linux/kthread.h>
42#include <linux/wait.h>
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +053043#include <linux/pm_qos.h>
Amit Blayd6ea6102012-06-07 16:26:24 +030044
45#include <mach/msm_bus.h>
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053046#include <mach/clk.h>
47#include <mach/msm_iomap.h>
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +053048#include <mach/msm_xo.h>
Vamsi Krishna34f01582011-12-14 19:54:42 -080049#include <linux/spinlock.h>
Hemant Kumar45d211b2012-05-31 17:58:43 -070050#include <linux/cpu.h>
Amit Blayd6ea6102012-06-07 16:26:24 +030051#include <mach/rpm-regulator.h>
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053052
53#define MSM_USB_BASE (hcd->regs)
Hemant Kumar105d07f2012-07-02 15:33:07 -070054#define USB_REG_START_OFFSET 0x90
55#define USB_REG_END_OFFSET 0x250
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053056
Hemant Kumar2309eaa2012-08-14 16:46:42 -070057static struct workqueue_struct *ehci_wq;
Vamsi Krishnafce1bd22012-08-17 17:12:26 -070058struct ehci_timer {
59#define GPT_LD(p) ((p) & 0x00FFFFFF)
60 u32 gptimer0_ld;
61#define GPT_RUN BIT(31)
62#define GPT_RESET BIT(30)
63#define GPT_MODE BIT(24)
64#define GPT_CNT(p) ((p) & 0x00FFFFFF)
65 u32 gptimer0_ctrl;
66
67 u32 gptimer1_ld;
68 u32 gptimer1_ctrl;
69};
Hemant Kumar2309eaa2012-08-14 16:46:42 -070070
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053071struct msm_hsic_hcd {
72 struct ehci_hcd ehci;
Hemant Kumard343c012012-09-06 19:57:14 -070073 spinlock_t wakeup_lock;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053074 struct device *dev;
75 struct clk *ahb_clk;
Manu Gautam5143b252012-01-05 19:25:23 -080076 struct clk *core_clk;
77 struct clk *alt_core_clk;
78 struct clk *phy_clk;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053079 struct clk *cal_clk;
80 struct regulator *hsic_vddcx;
Ofir Cohen4cc55372012-11-20 11:17:24 +020081 struct regulator *hsic_gdsc;
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +053082 atomic_t async_int;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053083 atomic_t in_lpm;
Sudhir Sharma1673e302012-08-27 17:37:24 -070084 struct wake_lock wlock;
Vamsi Krishna34f01582011-12-14 19:54:42 -080085 int peripheral_status_irq;
Vamsi Krishna6921cbe2012-02-21 18:34:43 -080086 int wakeup_irq;
Jack Phamfe441ea2012-03-23 17:03:15 -070087 bool wakeup_irq_enabled;
Sudhir Sharma1673e302012-08-27 17:37:24 -070088 atomic_t pm_usage_cnt;
Hemant Kumare6275972012-02-29 20:06:21 -080089 uint32_t bus_perf_client;
Hemant Kumar6fd65032012-05-23 13:02:24 -070090 uint32_t wakeup_int_cnt;
Amit Blayd6ea6102012-06-07 16:26:24 +030091 enum usb_vdd_type vdd_type;
Hemant Kumar2309eaa2012-08-14 16:46:42 -070092
93 struct work_struct bus_vote_w;
94 bool bus_vote;
Vamsi Krishnafce1bd22012-08-17 17:12:26 -070095
96 /* gp timer */
97 struct ehci_timer __iomem *timer;
98 struct completion gpt0_completion;
99 struct completion rt_completion;
100 int resume_status;
101 int resume_again;
Pavankumar Kondeti23720492013-01-29 14:41:58 +0530102 int bus_reset;
103 int reset_again;
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +0530104
105 struct pm_qos_request pm_qos_req_dma;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530106};
107
Hemant Kumar105d07f2012-07-02 15:33:07 -0700108struct msm_hsic_hcd *__mehci;
109
Hemant Kumare6275972012-02-29 20:06:21 -0800110static bool debug_bus_voting_enabled = true;
Ofir Cohen4cc55372012-11-20 11:17:24 +0200111static u64 ehci_msm_hsic_dma_mask = DMA_BIT_MASK(32);
112
Manu Gautam863b74132012-11-21 14:30:04 +0530113static struct platform_driver ehci_msm_hsic_driver;
Hemant Kumar45d211b2012-05-31 17:58:43 -0700114
Hemant Kumar4d50a432012-08-15 09:06:35 -0700115static unsigned int enable_payload_log = 1;
116module_param(enable_payload_log, uint, S_IRUGO | S_IWUSR);
Hemant Kumar45d211b2012-05-31 17:58:43 -0700117static unsigned int enable_dbg_log = 1;
118module_param(enable_dbg_log, uint, S_IRUGO | S_IWUSR);
119/*by default log ep0 and efs sync ep*/
120static unsigned int ep_addr_rxdbg_mask = 9;
121module_param(ep_addr_rxdbg_mask, uint, S_IRUGO | S_IWUSR);
122static unsigned int ep_addr_txdbg_mask = 9;
123module_param(ep_addr_txdbg_mask, uint, S_IRUGO | S_IWUSR);
124
125/* Maximum debug message length */
Hemant Kumar4d50a432012-08-15 09:06:35 -0700126#define DBG_MSG_LEN 128UL
Hemant Kumar45d211b2012-05-31 17:58:43 -0700127
128/* Maximum number of messages */
129#define DBG_MAX_MSG 256UL
130
131#define TIME_BUF_LEN 20
Hemant Kumar4d50a432012-08-15 09:06:35 -0700132#define HEX_DUMP_LEN 72
Hemant Kumar45d211b2012-05-31 17:58:43 -0700133
134enum event_type {
135 EVENT_UNDEF = -1,
136 URB_SUBMIT,
137 URB_COMPLETE,
138 EVENT_NONE,
139};
140
141#define EVENT_STR_LEN 5
142
Hemant Kumar45d211b2012-05-31 17:58:43 -0700143static enum event_type str_to_event(const char *name)
144{
145 if (!strncasecmp("S", name, EVENT_STR_LEN))
146 return URB_SUBMIT;
147 if (!strncasecmp("C", name, EVENT_STR_LEN))
148 return URB_COMPLETE;
149 if (!strncasecmp("", name, EVENT_STR_LEN))
150 return EVENT_NONE;
151
152 return EVENT_UNDEF;
153}
154
155/*log ep0 activity*/
156static struct {
157 char (buf[DBG_MAX_MSG])[DBG_MSG_LEN]; /* buffer */
158 unsigned idx; /* index */
159 rwlock_t lck; /* lock */
160} dbg_hsic_ctrl = {
161 .idx = 0,
162 .lck = __RW_LOCK_UNLOCKED(lck)
163};
164
165static struct {
166 char (buf[DBG_MAX_MSG])[DBG_MSG_LEN]; /* buffer */
167 unsigned idx; /* index */
168 rwlock_t lck; /* lock */
169} dbg_hsic_data = {
170 .idx = 0,
171 .lck = __RW_LOCK_UNLOCKED(lck)
172};
173
174/**
175 * dbg_inc: increments debug event index
176 * @idx: buffer index
177 */
178static void dbg_inc(unsigned *idx)
179{
180 *idx = (*idx + 1) & (DBG_MAX_MSG-1);
181}
182
183/*get_timestamp - returns time of day in us */
184static char *get_timestamp(char *tbuf)
185{
186 unsigned long long t;
187 unsigned long nanosec_rem;
188
189 t = cpu_clock(smp_processor_id());
190 nanosec_rem = do_div(t, 1000000000)/1000;
191 scnprintf(tbuf, TIME_BUF_LEN, "[%5lu.%06lu] ", (unsigned long)t,
192 nanosec_rem);
193 return tbuf;
194}
195
196static int allow_dbg_log(int ep_addr)
197{
198 int dir, num;
199
200 dir = ep_addr & USB_DIR_IN ? USB_DIR_IN : USB_DIR_OUT;
201 num = ep_addr & ~USB_DIR_IN;
202 num = 1 << num;
203
204 if ((dir == USB_DIR_IN) && (num & ep_addr_rxdbg_mask))
205 return 1;
206 if ((dir == USB_DIR_OUT) && (num & ep_addr_txdbg_mask))
207 return 1;
208
209 return 0;
210}
211
Hemant Kumar4d50a432012-08-15 09:06:35 -0700212static char *get_hex_data(char *dbuf, struct urb *urb, int event, int status)
213{
214 int ep_addr = urb->ep->desc.bEndpointAddress;
215 char *ubuf = urb->transfer_buffer;
216 size_t len = event ? \
217 urb->actual_length : urb->transfer_buffer_length;
218
219 if (status == -EINPROGRESS)
220 status = 0;
221
222 /*Only dump ep in completions and epout submissions*/
223 if (len && !status &&
224 (((ep_addr & USB_DIR_IN) && event) ||
225 (!(ep_addr & USB_DIR_IN) && !event))) {
226 if (len >= 32)
227 len = 32;
228 hex_dump_to_buffer(ubuf, len, 32, 4, dbuf, HEX_DUMP_LEN, 0);
229 } else {
230 dbuf = "";
231 }
232
233 return dbuf;
234}
235
Hemant Kumar45d211b2012-05-31 17:58:43 -0700236static void dbg_log_event(struct urb *urb, char * event, unsigned extra)
237{
238 unsigned long flags;
239 int ep_addr;
240 char tbuf[TIME_BUF_LEN];
Hemant Kumar4d50a432012-08-15 09:06:35 -0700241 char dbuf[HEX_DUMP_LEN];
Hemant Kumar45d211b2012-05-31 17:58:43 -0700242
243 if (!enable_dbg_log)
244 return;
245
246 if (!urb) {
247 write_lock_irqsave(&dbg_hsic_ctrl.lck, flags);
248 scnprintf(dbg_hsic_ctrl.buf[dbg_hsic_ctrl.idx], DBG_MSG_LEN,
Hemant Kumar4d50a432012-08-15 09:06:35 -0700249 "%s: %s : %u", get_timestamp(tbuf), event, extra);
Hemant Kumar45d211b2012-05-31 17:58:43 -0700250 dbg_inc(&dbg_hsic_ctrl.idx);
251 write_unlock_irqrestore(&dbg_hsic_ctrl.lck, flags);
252 return;
253 }
254
255 ep_addr = urb->ep->desc.bEndpointAddress;
256 if (!allow_dbg_log(ep_addr))
257 return;
258
259 if ((ep_addr & 0x0f) == 0x0) {
260 /*submit event*/
261 if (!str_to_event(event)) {
262 write_lock_irqsave(&dbg_hsic_ctrl.lck, flags);
263 scnprintf(dbg_hsic_ctrl.buf[dbg_hsic_ctrl.idx],
264 DBG_MSG_LEN, "%s: [%s : %p]:[%s] "
Hemant Kumar4d50a432012-08-15 09:06:35 -0700265 "%02x %02x %04x %04x %04x %u %d",
Hemant Kumar45d211b2012-05-31 17:58:43 -0700266 get_timestamp(tbuf), event, urb,
267 (ep_addr & USB_DIR_IN) ? "in" : "out",
268 urb->setup_packet[0], urb->setup_packet[1],
269 (urb->setup_packet[3] << 8) |
270 urb->setup_packet[2],
271 (urb->setup_packet[5] << 8) |
272 urb->setup_packet[4],
273 (urb->setup_packet[7] << 8) |
274 urb->setup_packet[6],
Hemant Kumar4d50a432012-08-15 09:06:35 -0700275 urb->transfer_buffer_length, extra);
Hemant Kumar45d211b2012-05-31 17:58:43 -0700276
277 dbg_inc(&dbg_hsic_ctrl.idx);
278 write_unlock_irqrestore(&dbg_hsic_ctrl.lck, flags);
279 } else {
280 write_lock_irqsave(&dbg_hsic_ctrl.lck, flags);
281 scnprintf(dbg_hsic_ctrl.buf[dbg_hsic_ctrl.idx],
Hemant Kumar4d50a432012-08-15 09:06:35 -0700282 DBG_MSG_LEN, "%s: [%s : %p]:[%s] %u %d",
Hemant Kumar45d211b2012-05-31 17:58:43 -0700283 get_timestamp(tbuf), event, urb,
284 (ep_addr & USB_DIR_IN) ? "in" : "out",
285 urb->actual_length, extra);
286
287 dbg_inc(&dbg_hsic_ctrl.idx);
288 write_unlock_irqrestore(&dbg_hsic_ctrl.lck, flags);
289 }
290 } else {
291 write_lock_irqsave(&dbg_hsic_data.lck, flags);
292 scnprintf(dbg_hsic_data.buf[dbg_hsic_data.idx], DBG_MSG_LEN,
Hemant Kumar4d50a432012-08-15 09:06:35 -0700293 "%s: [%s : %p]:ep%d[%s] %u %d %s",
Hemant Kumar45d211b2012-05-31 17:58:43 -0700294 get_timestamp(tbuf), event, urb, ep_addr & 0x0f,
295 (ep_addr & USB_DIR_IN) ? "in" : "out",
296 str_to_event(event) ? urb->actual_length :
Hemant Kumar4d50a432012-08-15 09:06:35 -0700297 urb->transfer_buffer_length, extra,
298 enable_payload_log ? get_hex_data(dbuf, urb,
299 str_to_event(event), extra) : "");
Hemant Kumar45d211b2012-05-31 17:58:43 -0700300
301 dbg_inc(&dbg_hsic_data.idx);
302 write_unlock_irqrestore(&dbg_hsic_data.lck, flags);
303 }
304}
305
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530306static inline struct msm_hsic_hcd *hcd_to_hsic(struct usb_hcd *hcd)
307{
308 return (struct msm_hsic_hcd *) (hcd->hcd_priv);
309}
310
311static inline struct usb_hcd *hsic_to_hcd(struct msm_hsic_hcd *mehci)
312{
313 return container_of((void *) mehci, struct usb_hcd, hcd_priv);
314}
315
Hemant Kumar105d07f2012-07-02 15:33:07 -0700316static void dump_hsic_regs(struct usb_hcd *hcd)
317{
318 int i;
319 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
320
321 if (atomic_read(&mehci->in_lpm))
322 return;
323
324 for (i = USB_REG_START_OFFSET; i <= USB_REG_END_OFFSET; i += 0x10)
325 pr_info("%p: %08x\t%08x\t%08x\t%08x\n", hcd->regs + i,
326 readl_relaxed(hcd->regs + i),
327 readl_relaxed(hcd->regs + i + 4),
328 readl_relaxed(hcd->regs + i + 8),
329 readl_relaxed(hcd->regs + i + 0xc));
330}
331
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530332#define ULPI_IO_TIMEOUT_USEC (10 * 1000)
333
Amit Blayd6ea6102012-06-07 16:26:24 +0300334#define USB_PHY_VDD_DIG_VOL_NONE 0 /*uV */
Hemant Kumar3b743cd2012-10-17 13:48:10 -0700335#define USB_PHY_VDD_DIG_VOL_MIN 945000 /* uV */
Vamsi Krishna45d88fa2011-11-02 13:28:42 -0700336#define USB_PHY_VDD_DIG_VOL_MAX 1320000 /* uV */
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530337
Lena Salman8c8ba382012-02-14 15:59:31 +0200338#define HSIC_DBG1_REG 0x38
339
Amit Blayd6ea6102012-06-07 16:26:24 +0300340static const int vdd_val[VDD_TYPE_MAX][VDD_VAL_MAX] = {
341 { /* VDD_CX CORNER Voting */
342 [VDD_NONE] = RPM_VREG_CORNER_NONE,
343 [VDD_MIN] = RPM_VREG_CORNER_NOMINAL,
344 [VDD_MAX] = RPM_VREG_CORNER_HIGH,
345 },
346 { /* VDD_CX Voltage Voting */
347 [VDD_NONE] = USB_PHY_VDD_DIG_VOL_NONE,
348 [VDD_MIN] = USB_PHY_VDD_DIG_VOL_MIN,
349 [VDD_MAX] = USB_PHY_VDD_DIG_VOL_MAX,
350 },
351};
352
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530353static int msm_hsic_init_vddcx(struct msm_hsic_hcd *mehci, int init)
354{
355 int ret = 0;
Amit Blayd6ea6102012-06-07 16:26:24 +0300356 int none_vol, min_vol, max_vol;
357
358 if (!mehci->hsic_vddcx) {
359 mehci->vdd_type = VDDCX_CORNER;
360 mehci->hsic_vddcx = devm_regulator_get(mehci->dev,
361 "hsic_vdd_dig");
362 if (IS_ERR(mehci->hsic_vddcx)) {
363 mehci->hsic_vddcx = devm_regulator_get(mehci->dev,
364 "HSIC_VDDCX");
365 if (IS_ERR(mehci->hsic_vddcx)) {
366 dev_err(mehci->dev, "unable to get hsic vddcx\n");
367 return PTR_ERR(mehci->hsic_vddcx);
368 }
369 mehci->vdd_type = VDDCX;
370 }
371 }
372
373 none_vol = vdd_val[mehci->vdd_type][VDD_NONE];
374 min_vol = vdd_val[mehci->vdd_type][VDD_MIN];
375 max_vol = vdd_val[mehci->vdd_type][VDD_MAX];
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530376
377 if (!init)
378 goto disable_reg;
379
Amit Blayd6ea6102012-06-07 16:26:24 +0300380 ret = regulator_set_voltage(mehci->hsic_vddcx, min_vol, max_vol);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530381 if (ret) {
382 dev_err(mehci->dev, "unable to set the voltage"
383 "for hsic vddcx\n");
Mayank Rana189ac052012-03-24 04:35:02 +0530384 return ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530385 }
386
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530387 ret = regulator_enable(mehci->hsic_vddcx);
388 if (ret) {
389 dev_err(mehci->dev, "unable to enable hsic vddcx\n");
390 goto reg_enable_err;
391 }
392
393 return 0;
394
395disable_reg:
396 regulator_disable(mehci->hsic_vddcx);
397reg_enable_err:
Amit Blayd6ea6102012-06-07 16:26:24 +0300398 regulator_set_voltage(mehci->hsic_vddcx, none_vol, max_vol);
399
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530400 return ret;
401
402}
403
Ofir Cohen4cc55372012-11-20 11:17:24 +0200404/* Global Distributed Switch Controller (GDSC) init */
405static int msm_hsic_init_gdsc(struct msm_hsic_hcd *mehci, int init)
406{
407 int ret = 0;
408
409 if (IS_ERR(mehci->hsic_gdsc))
410 return 0;
411
412 if (!mehci->hsic_gdsc) {
413 mehci->hsic_gdsc = devm_regulator_get(mehci->dev,
414 "HSIC_GDSC");
415 if (IS_ERR(mehci->hsic_gdsc))
416 return 0;
417 }
418
419 if (init) {
420 ret = regulator_enable(mehci->hsic_gdsc);
421 if (ret) {
422 dev_err(mehci->dev, "unable to enable hsic gdsc\n");
423 return ret;
424 }
425 } else {
426 regulator_disable(mehci->hsic_gdsc);
427 }
428
429 return 0;
430
431}
432
Pavankumar Kondeti23720492013-01-29 14:41:58 +0530433static int __maybe_unused ulpi_read(struct msm_hsic_hcd *mehci, u32 reg)
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -0700434{
435 struct usb_hcd *hcd = hsic_to_hcd(mehci);
Hemant Kumar8fdc3982012-08-21 12:22:27 -0700436 int cnt = 0;
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -0700437
438 /* initiate read operation */
439 writel_relaxed(ULPI_RUN | ULPI_READ | ULPI_ADDR(reg),
440 USB_ULPI_VIEWPORT);
441
442 /* wait for completion */
Hemant Kumar8fdc3982012-08-21 12:22:27 -0700443 while (cnt < ULPI_IO_TIMEOUT_USEC) {
444 if (!(readl_relaxed(USB_ULPI_VIEWPORT) & ULPI_RUN))
445 break;
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -0700446 udelay(1);
Hemant Kumar8fdc3982012-08-21 12:22:27 -0700447 cnt++;
448 }
449
450 if (cnt >= ULPI_IO_TIMEOUT_USEC) {
451 dev_err(mehci->dev, "ulpi_read: timeout ULPI_VIEWPORT: %08x\n",
452 readl_relaxed(USB_ULPI_VIEWPORT));
453 dev_err(mehci->dev, "PORTSC: %08x USBCMD: %08x FRINDEX: %08x\n",
454 readl_relaxed(USB_PORTSC),
455 readl_relaxed(USB_USBCMD),
456 readl_relaxed(USB_FRINDEX));
457
458 /*frame counter increments afte 125us*/
459 udelay(130);
460 dev_err(mehci->dev, "ulpi_read: FRINDEX: %08x\n",
461 readl_relaxed(USB_FRINDEX));
462 return -ETIMEDOUT;
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -0700463 }
464
465 return ULPI_DATA_READ(readl_relaxed(USB_ULPI_VIEWPORT));
466}
467
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530468static int ulpi_write(struct msm_hsic_hcd *mehci, u32 val, u32 reg)
469{
470 struct usb_hcd *hcd = hsic_to_hcd(mehci);
471 int cnt = 0;
472
473 /* initiate write operation */
474 writel_relaxed(ULPI_RUN | ULPI_WRITE |
475 ULPI_ADDR(reg) | ULPI_DATA(val),
476 USB_ULPI_VIEWPORT);
477
478 /* wait for completion */
479 while (cnt < ULPI_IO_TIMEOUT_USEC) {
480 if (!(readl_relaxed(USB_ULPI_VIEWPORT) & ULPI_RUN))
481 break;
482 udelay(1);
483 cnt++;
484 }
485
486 if (cnt >= ULPI_IO_TIMEOUT_USEC) {
Hemant Kumar8fdc3982012-08-21 12:22:27 -0700487 dev_err(mehci->dev, "ulpi_write: timeout ULPI_VIEWPORT: %08x\n",
488 readl_relaxed(USB_ULPI_VIEWPORT));
489 dev_err(mehci->dev, "PORTSC: %08x USBCMD: %08x FRINDEX: %08x\n",
490 readl_relaxed(USB_PORTSC),
491 readl_relaxed(USB_USBCMD),
492 readl_relaxed(USB_FRINDEX));
493
494 /*frame counter increments afte 125us*/
495 udelay(130);
496 dev_err(mehci->dev, "ulpi_write: FRINDEX: %08x\n",
497 readl_relaxed(USB_FRINDEX));
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530498 return -ETIMEDOUT;
499 }
500
501 return 0;
502}
503
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530504static int msm_hsic_config_gpios(struct msm_hsic_hcd *mehci, int gpio_en)
505{
506 int rc = 0;
507 struct msm_hsic_host_platform_data *pdata;
Vamsi Krishna34f01582011-12-14 19:54:42 -0800508 static int gpio_status;
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530509
510 pdata = mehci->dev->platform_data;
Vamsi Krishna34f01582011-12-14 19:54:42 -0800511
Lena Salman8c8ba382012-02-14 15:59:31 +0200512 if (!pdata || !pdata->strobe || !pdata->data)
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530513 return rc;
514
Vamsi Krishna34f01582011-12-14 19:54:42 -0800515 if (gpio_status == gpio_en)
516 return 0;
517
518 gpio_status = gpio_en;
519
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530520 if (!gpio_en)
521 goto free_gpio;
522
523 rc = gpio_request(pdata->strobe, "HSIC_STROBE_GPIO");
524 if (rc < 0) {
525 dev_err(mehci->dev, "gpio request failed for HSIC STROBE\n");
526 return rc;
527 }
528
529 rc = gpio_request(pdata->data, "HSIC_DATA_GPIO");
530 if (rc < 0) {
531 dev_err(mehci->dev, "gpio request failed for HSIC DATA\n");
532 goto free_strobe;
Hemant Kumar6fd65032012-05-23 13:02:24 -0700533 }
534
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530535 return 0;
536
537free_gpio:
538 gpio_free(pdata->data);
539free_strobe:
540 gpio_free(pdata->strobe);
541
542 return rc;
543}
544
Vamsi Krishna64b48612012-06-14 16:08:11 -0700545static void msm_hsic_clk_reset(struct msm_hsic_hcd *mehci)
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530546{
547 int ret;
548
Ofir Cohen07944aa2012-12-23 13:41:57 +0200549 /* alt_core_clk exists in targets that do not use asynchronous reset */
550 if (!IS_ERR(mehci->alt_core_clk)) {
551 ret = clk_reset(mehci->core_clk, CLK_RESET_ASSERT);
552 if (ret) {
553 dev_err(mehci->dev, "hsic clk assert failed:%d\n", ret);
554 return;
555 }
556
557 /* Since a hw bug, turn off the clock before complete reset */
558 clk_disable(mehci->core_clk);
559
560 ret = clk_reset(mehci->core_clk, CLK_RESET_DEASSERT);
561 if (ret)
562 dev_err(mehci->dev, "hsic clk deassert failed:%d\n",
563 ret);
564
565 usleep_range(10000, 12000);
566
567 clk_enable(mehci->core_clk);
568 } else {
569 /* Using asynchronous block reset to the hardware */
570 clk_disable_unprepare(mehci->core_clk);
571 clk_disable_unprepare(mehci->phy_clk);
572 clk_disable_unprepare(mehci->cal_clk);
573 clk_disable_unprepare(mehci->ahb_clk);
574
575 ret = clk_reset(mehci->core_clk, CLK_RESET_ASSERT);
576 if (ret) {
577 dev_err(mehci->dev, "hsic clk assert failed:%d\n", ret);
578 return;
579 }
580 usleep_range(10000, 12000);
581
582 ret = clk_reset(mehci->core_clk, CLK_RESET_DEASSERT);
583 if (ret)
584 dev_err(mehci->dev, "hsic clk deassert failed:%d\n",
585 ret);
586 /*
587 * Required delay between the deassertion and
588 * clock enablement.
589 */
590 ndelay(200);
591 clk_prepare_enable(mehci->core_clk);
592 clk_prepare_enable(mehci->phy_clk);
593 clk_prepare_enable(mehci->cal_clk);
594 clk_prepare_enable(mehci->ahb_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530595 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530596}
597
Vamsi Krishna64b48612012-06-14 16:08:11 -0700598#define HSIC_STROBE_GPIO_PAD_CTL (MSM_TLMM_BASE+0x20C0)
599#define HSIC_DATA_GPIO_PAD_CTL (MSM_TLMM_BASE+0x20C4)
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530600#define HSIC_CAL_PAD_CTL (MSM_TLMM_BASE+0x20C8)
601#define HSIC_LV_MODE 0x04
602#define HSIC_PAD_CALIBRATION 0xA8
603#define HSIC_GPIO_PAD_VAL 0x0A0AAA10
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530604#define LINK_RESET_TIMEOUT_USEC (250 * 1000)
Ofir Cohen5dddb152012-11-14 11:18:50 +0200605
606static void msm_hsic_phy_reset(struct msm_hsic_hcd *mehci)
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530607{
608 struct usb_hcd *hcd = hsic_to_hcd(mehci);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530609
Vamsi Krishna64b48612012-06-14 16:08:11 -0700610 msm_hsic_clk_reset(mehci);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530611
Vamsi Krishna64b48612012-06-14 16:08:11 -0700612 /* select ulpi phy */
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530613 writel_relaxed(0x80000000, USB_PORTSC);
Vamsi Krishna64b48612012-06-14 16:08:11 -0700614 mb();
Ofir Cohen5dddb152012-11-14 11:18:50 +0200615}
616
617static int msm_hsic_start(struct msm_hsic_hcd *mehci)
618{
619 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
620 int ret;
Manu Gautam863b74132012-11-21 14:30:04 +0530621 void __iomem *reg;
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530622
Lena Salman8c8ba382012-02-14 15:59:31 +0200623 /* HSIC init sequence when HSIC signals (Strobe/Data) are
624 routed via GPIOs */
625 if (pdata && pdata->strobe && pdata->data) {
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530626
Manu Gautam863b74132012-11-21 14:30:04 +0530627 if (!pdata->ignore_cal_pad_config) {
628 /* Enable LV_MODE in HSIC_CAL_PAD_CTL register */
629 writel_relaxed(HSIC_LV_MODE, HSIC_CAL_PAD_CTL);
630 mb();
631 }
Vamsi Krishna64b48612012-06-14 16:08:11 -0700632
Lena Salman8c8ba382012-02-14 15:59:31 +0200633 /*set periodic calibration interval to ~2.048sec in
634 HSIC_IO_CAL_REG */
635 ulpi_write(mehci, 0xFF, 0x33);
636
637 /* Enable periodic IO calibration in HSIC_CFG register */
638 ulpi_write(mehci, HSIC_PAD_CALIBRATION, 0x30);
639
Vamsi Krishna64b48612012-06-14 16:08:11 -0700640 /* Configure GPIO pins for HSIC functionality mode */
Lena Salman8c8ba382012-02-14 15:59:31 +0200641 ret = msm_hsic_config_gpios(mehci, 1);
642 if (ret) {
643 dev_err(mehci->dev, " gpio configuarion failed\n");
644 return ret;
645 }
Manu Gautam863b74132012-11-21 14:30:04 +0530646 if (pdata->strobe_pad_offset) {
647 /* Set CORE_CTL_EN in STROBE GPIO PAD_CTL register */
648 reg = MSM_TLMM_BASE + pdata->strobe_pad_offset;
649 writel_relaxed(readl_relaxed(reg) | 0x2000000, reg);
650 } else {
651 /* Set LV_MODE=0x1 and DCC=0x2 in STROBE GPIO PAD_CTL */
652 reg = HSIC_STROBE_GPIO_PAD_CTL;
653 writel_relaxed(HSIC_GPIO_PAD_VAL, reg);
654 }
655
656 if (pdata->data_pad_offset) {
657 /* Set CORE_CTL_EN in HSIC_DATA GPIO PAD_CTL register */
658 reg = MSM_TLMM_BASE + pdata->data_pad_offset;
659 writel_relaxed(readl_relaxed(reg) | 0x2000000, reg);
660 } else {
661 /* Set LV_MODE=0x1 and DCC=0x2 in STROBE GPIO PAD_CTL */
662 reg = HSIC_DATA_GPIO_PAD_CTL;
663 writel_relaxed(HSIC_GPIO_PAD_VAL, reg);
664 }
Vamsi Krishna64b48612012-06-14 16:08:11 -0700665
666 mb();
667
Lena Salman8c8ba382012-02-14 15:59:31 +0200668 /* Enable HSIC mode in HSIC_CFG register */
669 ulpi_write(mehci, 0x01, 0x31);
670 } else {
671 /* HSIC init sequence when HSIC signals (Strobe/Data) are routed
672 via dedicated I/O */
673
674 /* programmable length of connect signaling (33.2ns) */
675 ret = ulpi_write(mehci, 3, HSIC_DBG1_REG);
676 if (ret) {
677 pr_err("%s: Unable to program length of connect "
678 "signaling\n", __func__);
679 }
680
681 /*set periodic calibration interval to ~2.048sec in
682 HSIC_IO_CAL_REG */
683 ulpi_write(mehci, 0xFF, 0x33);
684
685 /* Enable HSIC mode in HSIC_CFG register */
686 ulpi_write(mehci, 0xA9, 0x30);
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530687 }
688
Hemant Kumar6fd65032012-05-23 13:02:24 -0700689 /*disable auto resume*/
690 ulpi_write(mehci, ULPI_IFC_CTRL_AUTORESUME, ULPI_CLR(ULPI_IFC_CTRL));
691
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530692 return 0;
693}
694
695#define PHY_SUSPEND_TIMEOUT_USEC (500 * 1000)
696#define PHY_RESUME_TIMEOUT_USEC (100 * 1000)
697
698#ifdef CONFIG_PM_SLEEP
Ofir Cohen5dddb152012-11-14 11:18:50 +0200699static int msm_hsic_reset(struct msm_hsic_hcd *mehci)
700{
701 /* reset HSIC phy */
702 msm_hsic_phy_reset(mehci);
703
704 /* HSIC init procedure (caliberation) */
705 return msm_hsic_start(mehci);
706}
707
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530708static int msm_hsic_suspend(struct msm_hsic_hcd *mehci)
709{
710 struct usb_hcd *hcd = hsic_to_hcd(mehci);
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530711 int cnt = 0, ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530712 u32 val;
Amit Blayd6ea6102012-06-07 16:26:24 +0300713 int none_vol, max_vol;
Hemant Kumarbc8bdf62012-10-17 12:29:51 -0700714 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530715
716 if (atomic_read(&mehci->in_lpm)) {
717 dev_dbg(mehci->dev, "%s called in lpm\n", __func__);
718 return 0;
719 }
720
721 disable_irq(hcd->irq);
Jack Phambe05fbb2012-05-16 10:56:26 -0700722
Sudhir Sharma1673e302012-08-27 17:37:24 -0700723 /* make sure we don't race against a remote wakeup */
724 if (test_bit(HCD_FLAG_WAKEUP_PENDING, &hcd->flags) ||
Jack Phambe05fbb2012-05-16 10:56:26 -0700725 readl_relaxed(USB_PORTSC) & PORT_RESUME) {
Sudhir Sharma1673e302012-08-27 17:37:24 -0700726 dev_dbg(mehci->dev, "wakeup pending, aborting suspend\n");
Jack Phambe05fbb2012-05-16 10:56:26 -0700727 enable_irq(hcd->irq);
728 return -EBUSY;
729 }
730
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530731 /*
732 * PHY may take some time or even fail to enter into low power
733 * mode (LPM). Hence poll for 500 msec and reset the PHY and link
734 * in failure case.
735 */
Hemant Kumar3dbc5b32012-05-09 15:36:11 -0700736 val = readl_relaxed(USB_PORTSC);
737 val &= ~PORT_RWC_BITS;
738 val |= PORTSC_PHCD;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530739 writel_relaxed(val, USB_PORTSC);
740 while (cnt < PHY_SUSPEND_TIMEOUT_USEC) {
741 if (readl_relaxed(USB_PORTSC) & PORTSC_PHCD)
742 break;
743 udelay(1);
744 cnt++;
745 }
746
747 if (cnt >= PHY_SUSPEND_TIMEOUT_USEC) {
748 dev_err(mehci->dev, "Unable to suspend PHY\n");
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530749 msm_hsic_config_gpios(mehci, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530750 msm_hsic_reset(mehci);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530751 }
752
753 /*
754 * PHY has capability to generate interrupt asynchronously in low
755 * power mode (LPM). This interrupt is level triggered. So USB IRQ
756 * line must be disabled till async interrupt enable bit is cleared
757 * in USBCMD register. Assert STP (ULPI interface STOP signal) to
Pavankumar Kondeti0c9f29c2013-01-28 21:37:59 +0530758 * block data communication from PHY. Enable asynchronous interrupt
759 * only when wakeup gpio IRQ is not present.
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530760 */
Pavankumar Kondeti0c9f29c2013-01-28 21:37:59 +0530761 if (mehci->wakeup_irq)
762 writel_relaxed(readl_relaxed(USB_USBCMD) |
763 ULPI_STP_CTRL, USB_USBCMD);
764 else
765 writel_relaxed(readl_relaxed(USB_USBCMD) | ASYNC_INTR_CTRL |
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530766 ULPI_STP_CTRL, USB_USBCMD);
767
768 /*
769 * Ensure that hardware is put in low power mode before
770 * clocks are turned OFF and VDD is allowed to minimize.
771 */
772 mb();
773
Manu Gautam28b1bac2012-01-30 16:43:06 +0530774 clk_disable_unprepare(mehci->core_clk);
775 clk_disable_unprepare(mehci->phy_clk);
776 clk_disable_unprepare(mehci->cal_clk);
777 clk_disable_unprepare(mehci->ahb_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530778
Amit Blayd6ea6102012-06-07 16:26:24 +0300779 none_vol = vdd_val[mehci->vdd_type][VDD_NONE];
780 max_vol = vdd_val[mehci->vdd_type][VDD_MAX];
781
782 ret = regulator_set_voltage(mehci->hsic_vddcx, none_vol, max_vol);
Vamsi Krishna45d88fa2011-11-02 13:28:42 -0700783 if (ret < 0)
Amit Blayd6ea6102012-06-07 16:26:24 +0300784 dev_err(mehci->dev, "unable to set vddcx voltage for VDD MIN\n");
Vamsi Krishna45d88fa2011-11-02 13:28:42 -0700785
Hemant Kumare6275972012-02-29 20:06:21 -0800786 if (mehci->bus_perf_client && debug_bus_voting_enabled) {
Hemant Kumar2309eaa2012-08-14 16:46:42 -0700787 mehci->bus_vote = false;
788 queue_work(ehci_wq, &mehci->bus_vote_w);
Hemant Kumare6275972012-02-29 20:06:21 -0800789 }
790
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530791 atomic_set(&mehci->in_lpm, 1);
792 enable_irq(hcd->irq);
Hemant Kumar6fd65032012-05-23 13:02:24 -0700793
794 mehci->wakeup_irq_enabled = 1;
795 enable_irq_wake(mehci->wakeup_irq);
796 enable_irq(mehci->wakeup_irq);
797
Hemant Kumarbc8bdf62012-10-17 12:29:51 -0700798 if (pdata && pdata->standalone_latency)
799 pm_qos_update_request(&mehci->pm_qos_req_dma,
800 PM_QOS_DEFAULT_VALUE);
801
Sudhir Sharma1673e302012-08-27 17:37:24 -0700802 wake_unlock(&mehci->wlock);
803
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530804 dev_info(mehci->dev, "HSIC-USB in low power mode\n");
805
806 return 0;
807}
808
809static int msm_hsic_resume(struct msm_hsic_hcd *mehci)
810{
811 struct usb_hcd *hcd = hsic_to_hcd(mehci);
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530812 int cnt = 0, ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530813 unsigned temp;
Amit Blayd6ea6102012-06-07 16:26:24 +0300814 int min_vol, max_vol;
Hemant Kumard343c012012-09-06 19:57:14 -0700815 unsigned long flags;
Hemant Kumarbc8bdf62012-10-17 12:29:51 -0700816 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530817
818 if (!atomic_read(&mehci->in_lpm)) {
819 dev_dbg(mehci->dev, "%s called in !in_lpm\n", __func__);
820 return 0;
821 }
822
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +0530823 /* Handles race with Async interrupt */
824 disable_irq(hcd->irq);
825
Hemant Kumarbc8bdf62012-10-17 12:29:51 -0700826 if (pdata && pdata->standalone_latency)
827 pm_qos_update_request(&mehci->pm_qos_req_dma,
828 pdata->standalone_latency + 1);
829
Hemant Kumard343c012012-09-06 19:57:14 -0700830 spin_lock_irqsave(&mehci->wakeup_lock, flags);
Hemant Kumar6fd65032012-05-23 13:02:24 -0700831 if (mehci->wakeup_irq_enabled) {
832 disable_irq_wake(mehci->wakeup_irq);
833 disable_irq_nosync(mehci->wakeup_irq);
834 mehci->wakeup_irq_enabled = 0;
835 }
Hemant Kumard343c012012-09-06 19:57:14 -0700836 spin_unlock_irqrestore(&mehci->wakeup_lock, flags);
Hemant Kumar6fd65032012-05-23 13:02:24 -0700837
Sudhir Sharma1673e302012-08-27 17:37:24 -0700838 wake_lock(&mehci->wlock);
839
Hemant Kumare6275972012-02-29 20:06:21 -0800840 if (mehci->bus_perf_client && debug_bus_voting_enabled) {
Hemant Kumar2309eaa2012-08-14 16:46:42 -0700841 mehci->bus_vote = true;
842 queue_work(ehci_wq, &mehci->bus_vote_w);
Hemant Kumare6275972012-02-29 20:06:21 -0800843 }
844
Amit Blayd6ea6102012-06-07 16:26:24 +0300845 min_vol = vdd_val[mehci->vdd_type][VDD_MIN];
846 max_vol = vdd_val[mehci->vdd_type][VDD_MAX];
847
848 ret = regulator_set_voltage(mehci->hsic_vddcx, min_vol, max_vol);
Vamsi Krishna45d88fa2011-11-02 13:28:42 -0700849 if (ret < 0)
Amit Blayd6ea6102012-06-07 16:26:24 +0300850 dev_err(mehci->dev, "unable to set nominal vddcx voltage (no VDD MIN)\n");
Vamsi Krishna45d88fa2011-11-02 13:28:42 -0700851
Manu Gautam28b1bac2012-01-30 16:43:06 +0530852 clk_prepare_enable(mehci->core_clk);
853 clk_prepare_enable(mehci->phy_clk);
854 clk_prepare_enable(mehci->cal_clk);
855 clk_prepare_enable(mehci->ahb_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530856
857 temp = readl_relaxed(USB_USBCMD);
858 temp &= ~ASYNC_INTR_CTRL;
859 temp &= ~ULPI_STP_CTRL;
860 writel_relaxed(temp, USB_USBCMD);
861
862 if (!(readl_relaxed(USB_PORTSC) & PORTSC_PHCD))
863 goto skip_phy_resume;
864
Hemant Kumar3dbc5b32012-05-09 15:36:11 -0700865 temp = readl_relaxed(USB_PORTSC);
866 temp &= ~(PORT_RWC_BITS | PORTSC_PHCD);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530867 writel_relaxed(temp, USB_PORTSC);
868 while (cnt < PHY_RESUME_TIMEOUT_USEC) {
869 if (!(readl_relaxed(USB_PORTSC) & PORTSC_PHCD) &&
870 (readl_relaxed(USB_ULPI_VIEWPORT) & ULPI_SYNC_STATE))
871 break;
872 udelay(1);
873 cnt++;
874 }
875
876 if (cnt >= PHY_RESUME_TIMEOUT_USEC) {
877 /*
878 * This is a fatal error. Reset the link and
879 * PHY to make hsic working.
880 */
881 dev_err(mehci->dev, "Unable to resume USB. Reset the hsic\n");
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530882 msm_hsic_config_gpios(mehci, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530883 msm_hsic_reset(mehci);
884 }
885
886skip_phy_resume:
887
Hemant Kumar6fd65032012-05-23 13:02:24 -0700888 usb_hcd_resume_root_hub(hcd);
889
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530890 atomic_set(&mehci->in_lpm, 0);
891
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +0530892 if (atomic_read(&mehci->async_int)) {
893 atomic_set(&mehci->async_int, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530894 pm_runtime_put_noidle(mehci->dev);
Jack Phamdd5ad792012-07-26 10:31:03 -0700895 enable_irq(hcd->irq);
Hemant Kumar6fd65032012-05-23 13:02:24 -0700896 }
897
Sudhir Sharma1673e302012-08-27 17:37:24 -0700898 if (atomic_read(&mehci->pm_usage_cnt)) {
899 atomic_set(&mehci->pm_usage_cnt, 0);
900 pm_runtime_put_noidle(mehci->dev);
901 }
Jack Phamdd5ad792012-07-26 10:31:03 -0700902
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +0530903 enable_irq(hcd->irq);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530904 dev_info(mehci->dev, "HSIC-USB exited from low power mode\n");
905
906 return 0;
907}
908#endif
909
Hemant Kumar2309eaa2012-08-14 16:46:42 -0700910static void ehci_hsic_bus_vote_w(struct work_struct *w)
911{
912 struct msm_hsic_hcd *mehci =
913 container_of(w, struct msm_hsic_hcd, bus_vote_w);
914 int ret;
915
916 ret = msm_bus_scale_client_update_request(mehci->bus_perf_client,
917 mehci->bus_vote);
918 if (ret)
919 dev_err(mehci->dev, "%s: Failed to vote for bus bandwidth %d\n",
920 __func__, ret);
921}
922
Pavankumar Kondeti23720492013-01-29 14:41:58 +0530923static int msm_hsic_reset_done(struct usb_hcd *hcd)
924{
925 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
926 u32 __iomem *status_reg = &ehci->regs->port_status[0];
927 int ret;
928
929 ehci_writel(ehci, ehci_readl(ehci, status_reg) & ~(PORT_RWC_BITS |
930 PORT_RESET), status_reg);
931
932 ret = handshake(ehci, status_reg, PORT_RESET, 0, 1 * 1000);
933
934 if (ret)
935 pr_err("reset handshake failed in %s\n", __func__);
936 else
937 ehci_writel(ehci, ehci_readl(ehci, &ehci->regs->command) |
938 CMD_RUN, &ehci->regs->command);
939
940 return ret;
941}
942
Vamsi Krishnafce1bd22012-08-17 17:12:26 -0700943#define STS_GPTIMER0_INTERRUPT BIT(24)
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530944static irqreturn_t msm_hsic_irq(struct usb_hcd *hcd)
945{
Vamsi Krishnafce1bd22012-08-17 17:12:26 -0700946 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530947 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
Vamsi Krishnafce1bd22012-08-17 17:12:26 -0700948 u32 status;
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +0530949 int ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530950
951 if (atomic_read(&mehci->in_lpm)) {
Hemant Kumar3dbc5b32012-05-09 15:36:11 -0700952 dev_dbg(mehci->dev, "phy async intr\n");
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +0530953 dbg_log_event(NULL, "Async IRQ", 0);
954 ret = pm_runtime_get(mehci->dev);
955 if ((ret == 1) || (ret == -EINPROGRESS)) {
956 pm_runtime_put_noidle(mehci->dev);
957 } else {
958 disable_irq_nosync(hcd->irq);
959 atomic_set(&mehci->async_int, 1);
960 }
961
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530962 return IRQ_HANDLED;
963 }
964
Vamsi Krishnafce1bd22012-08-17 17:12:26 -0700965 status = ehci_readl(ehci, &ehci->regs->status);
966
967 if (status & STS_GPTIMER0_INTERRUPT) {
968 int timeleft;
969
Pavankumar Kondeti23720492013-01-29 14:41:58 +0530970 dbg_log_event(NULL, "FPR: gpt0_isr", mehci->bus_reset);
Vamsi Krishnafce1bd22012-08-17 17:12:26 -0700971
972 timeleft = GPT_CNT(ehci_readl(ehci,
973 &mehci->timer->gptimer1_ctrl));
974 if (timeleft) {
Pavankumar Kondeti23720492013-01-29 14:41:58 +0530975 if (mehci->bus_reset) {
976 ret = msm_hsic_reset_done(hcd);
977 if (ret) {
978 mehci->reset_again = 1;
979 dbg_log_event(NULL, "RESET: fail", 0);
980 }
981 } else {
982 ehci_writel(ehci, ehci_readl(ehci,
983 &ehci->regs->command) | CMD_RUN,
984 &ehci->regs->command);
985 }
986 } else {
987 if (mehci->bus_reset)
988 mehci->reset_again = 1;
989 else
990 mehci->resume_again = 1;
991 }
Vamsi Krishnafce1bd22012-08-17 17:12:26 -0700992
993 dbg_log_event(NULL, "FPR: timeleft", timeleft);
994
995 complete(&mehci->gpt0_completion);
996 ehci_writel(ehci, STS_GPTIMER0_INTERRUPT, &ehci->regs->status);
997 }
998
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530999 return ehci_irq(hcd);
1000}
1001
1002static int ehci_hsic_reset(struct usb_hcd *hcd)
1003{
1004 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001005 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301006 int retval;
1007
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001008 mehci->timer = USB_HS_GPTIMER_BASE;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301009 ehci->caps = USB_CAPLENGTH;
1010 ehci->regs = USB_CAPLENGTH +
1011 HC_LENGTH(ehci, ehci_readl(ehci, &ehci->caps->hc_capbase));
1012 dbg_hcs_params(ehci, "reset");
1013 dbg_hcc_params(ehci, "reset");
1014
1015 /* cache the data to minimize the chip reads*/
1016 ehci->hcs_params = ehci_readl(ehci, &ehci->caps->hcs_params);
1017
1018 hcd->has_tt = 1;
1019 ehci->sbrn = HCD_USB2;
1020
1021 retval = ehci_halt(ehci);
1022 if (retval)
1023 return retval;
1024
1025 /* data structure init */
1026 retval = ehci_init(hcd);
1027 if (retval)
1028 return retval;
1029
1030 retval = ehci_reset(ehci);
1031 if (retval)
1032 return retval;
1033
1034 /* bursts of unspecified length. */
1035 writel_relaxed(0, USB_AHBBURST);
1036 /* Use the AHB transactor */
Vijayavardhan Vennapusa5f32d7a2012-03-14 16:30:26 +05301037 writel_relaxed(0x08, USB_AHBMODE);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301038 /* Disable streaming mode and select host mode */
1039 writel_relaxed(0x13, USB_USBMODE);
1040
1041 ehci_port_power(ehci, 1);
1042 return 0;
1043}
1044
Jack Phamc0d41732012-09-28 17:45:38 -07001045#ifdef CONFIG_PM
1046
Pavankumar Kondeti23720492013-01-29 14:41:58 +05301047#define RESET_RETRY_LIMIT 3
1048#define RESET_SIGNAL_TIME_SOF_USEC (50 * 1000)
1049#define RESET_SIGNAL_TIME_USEC (20 * 1000)
1050static void ehci_hsic_reset_sof_bug_handler(struct usb_hcd *hcd, u32 val)
1051{
1052 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
1053 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
1054 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
1055 u32 __iomem *status_reg = &ehci->regs->port_status[0];
1056 unsigned long flags;
1057 int retries = 0, ret, cnt = RESET_SIGNAL_TIME_USEC;
1058
1059 if (pdata && pdata->swfi_latency)
1060 pm_qos_update_request(&mehci->pm_qos_req_dma,
1061 pdata->swfi_latency + 1);
1062
1063 mehci->bus_reset = 1;
1064retry:
1065 retries++;
1066 dbg_log_event(NULL, "RESET: start", retries);
1067 pr_debug("reset begin %d\n", retries);
1068 mehci->reset_again = 0;
1069 spin_lock_irqsave(&ehci->lock, flags);
1070 ehci_writel(ehci, val, status_reg);
1071 ehci_writel(ehci, GPT_LD(RESET_SIGNAL_TIME_USEC - 1),
1072 &mehci->timer->gptimer0_ld);
1073 ehci_writel(ehci, GPT_RESET | GPT_RUN,
1074 &mehci->timer->gptimer0_ctrl);
1075 ehci_writel(ehci, INTR_MASK | STS_GPTIMER0_INTERRUPT,
1076 &ehci->regs->intr_enable);
1077
1078 ehci_writel(ehci, GPT_LD(RESET_SIGNAL_TIME_SOF_USEC - 1),
1079 &mehci->timer->gptimer1_ld);
1080 ehci_writel(ehci, GPT_RESET | GPT_RUN,
1081 &mehci->timer->gptimer1_ctrl);
1082
1083 spin_unlock_irqrestore(&ehci->lock, flags);
1084 wait_for_completion(&mehci->gpt0_completion);
1085
1086 if (!mehci->reset_again)
1087 goto done;
1088
1089 if (handshake(ehci, status_reg, PORT_RESET, 0, 10 * 1000)) {
1090 pr_err("reset handshake fatal error\n");
1091 dbg_log_event(NULL, "RESET: fatal", retries);
1092 goto fail;
1093 }
1094
1095 if (retries < RESET_RETRY_LIMIT)
1096 goto retry;
1097
1098 /* complete reset in tight loop */
1099 pr_info("RESET in tight loop\n");
1100 dbg_log_event(NULL, "RESET: tight", 0);
1101
1102 spin_lock_irqsave(&ehci->lock, flags);
1103 ehci_writel(ehci, val, status_reg);
1104 while (cnt--)
1105 udelay(1);
1106 ret = msm_hsic_reset_done(hcd);
1107 spin_unlock_irqrestore(&ehci->lock, flags);
1108 if (ret) {
1109 pr_err("RESET in tight loop failed\n");
1110 dbg_log_event(NULL, "RESET: tight failed", 0);
1111 goto fail;
1112 }
1113
1114done:
1115 dbg_log_event(NULL, "RESET: done", retries);
1116 pr_debug("reset completed\n");
1117fail:
1118 mehci->bus_reset = 0;
1119 if (pdata && pdata->swfi_latency)
1120 pm_qos_update_request(&mehci->pm_qos_req_dma,
1121 PM_QOS_DEFAULT_VALUE);
1122}
1123
Hemant Kumar45d211b2012-05-31 17:58:43 -07001124static int ehci_hsic_bus_suspend(struct usb_hcd *hcd)
1125{
Pavankumar Kondetia723f742012-09-14 14:02:36 +05301126 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
1127
1128 if (!(readl_relaxed(USB_PORTSC) & PORT_PE)) {
1129 dbg_log_event(NULL, "RH suspend attempt failed", 0);
1130 dev_dbg(mehci->dev, "%s:port is not enabled skip suspend\n",
1131 __func__);
1132 return -EAGAIN;
1133 }
1134
Hemant Kumar45d211b2012-05-31 17:58:43 -07001135 dbg_log_event(NULL, "Suspend RH", 0);
1136 return ehci_bus_suspend(hcd);
1137}
1138
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001139#define RESUME_RETRY_LIMIT 3
1140#define RESUME_SIGNAL_TIME_MS (21 * 999)
1141#define RESUME_SIGNAL_TIME_SOF_MS (23 * 999)
1142static int msm_hsic_resume_thread(void *data)
1143{
1144 struct msm_hsic_hcd *mehci = data;
1145 struct usb_hcd *hcd = hsic_to_hcd(mehci);
1146 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
1147 u32 temp;
1148 unsigned long resume_needed = 0;
1149 int retry_cnt = 0;
1150 int tight_resume = 0;
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05301151 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001152
1153 dbg_log_event(NULL, "Resume RH", 0);
1154
1155 /* keep delay between bus states */
1156 if (time_before(jiffies, ehci->next_statechange))
1157 usleep_range(5000, 5000);
1158
1159 spin_lock_irq(&ehci->lock);
1160 if (!HCD_HW_ACCESSIBLE(hcd)) {
1161 spin_unlock_irq(&ehci->lock);
1162 mehci->resume_status = -ESHUTDOWN;
1163 complete(&mehci->rt_completion);
1164 return 0;
1165 }
1166
1167 if (unlikely(ehci->debug)) {
1168 if (!dbgp_reset_prep())
1169 ehci->debug = NULL;
1170 else
1171 dbgp_external_startup();
1172 }
1173
1174 /* at least some APM implementations will try to deliver
1175 * IRQs right away, so delay them until we're ready.
1176 */
1177 ehci_writel(ehci, 0, &ehci->regs->intr_enable);
1178
1179 /* re-init operational registers */
1180 ehci_writel(ehci, 0, &ehci->regs->segment);
1181 ehci_writel(ehci, ehci->periodic_dma, &ehci->regs->frame_list);
1182 ehci_writel(ehci, (u32) ehci->async->qh_dma, &ehci->regs->async_next);
1183
1184 /*CMD_RUN will be set after, PORT_RESUME gets cleared*/
1185 if (ehci->resume_sof_bug)
1186 ehci->command &= ~CMD_RUN;
1187
1188 /* restore CMD_RUN, framelist size, and irq threshold */
1189 ehci_writel(ehci, ehci->command, &ehci->regs->command);
1190
1191 /* manually resume the ports we suspended during bus_suspend() */
1192resume_again:
1193 if (retry_cnt >= RESUME_RETRY_LIMIT) {
1194 pr_info("retry count(%d) reached max, resume in tight loop\n",
1195 retry_cnt);
1196 tight_resume = 1;
1197 }
1198
1199
1200 temp = ehci_readl(ehci, &ehci->regs->port_status[0]);
1201 temp &= ~(PORT_RWC_BITS | PORT_WAKE_BITS);
1202 if (test_bit(0, &ehci->bus_suspended) && (temp & PORT_SUSPEND)) {
1203 temp |= PORT_RESUME;
1204 set_bit(0, &resume_needed);
1205 }
1206 dbg_log_event(NULL, "FPR: Set", temp);
1207 ehci_writel(ehci, temp, &ehci->regs->port_status[0]);
1208
1209 /* HSIC controller has a h/w bug due to which it can try to send SOFs
1210 * (start of frames) during port resume resulting in phy lockup. HSIC hw
1211 * controller in MSM clears FPR bit after driving the resume signal for
1212 * 20ms. Workaround is to stop SOFs before driving resume and then start
1213 * sending SOFs immediately. Need to send SOFs within 3ms of resume
1214 * completion otherwise peripheral may enter undefined state. As
1215 * usleep_range does not gurantee exact sleep time, GPTimer is used to
1216 * to time the resume sequence. If driver exceeds allowable time SOFs,
1217 * repeat the resume process.
1218 */
1219 if (ehci->resume_sof_bug && resume_needed) {
1220 if (!tight_resume) {
1221 mehci->resume_again = 0;
1222 ehci_writel(ehci, GPT_LD(RESUME_SIGNAL_TIME_MS),
1223 &mehci->timer->gptimer0_ld);
1224 ehci_writel(ehci, GPT_RESET | GPT_RUN,
1225 &mehci->timer->gptimer0_ctrl);
1226 ehci_writel(ehci, INTR_MASK | STS_GPTIMER0_INTERRUPT,
1227 &ehci->regs->intr_enable);
1228
1229 ehci_writel(ehci, GPT_LD(RESUME_SIGNAL_TIME_SOF_MS),
1230 &mehci->timer->gptimer1_ld);
1231 ehci_writel(ehci, GPT_RESET | GPT_RUN,
1232 &mehci->timer->gptimer1_ctrl);
1233
1234 spin_unlock_irq(&ehci->lock);
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05301235 if (pdata && pdata->swfi_latency)
1236 pm_qos_update_request(&mehci->pm_qos_req_dma,
1237 pdata->swfi_latency + 1);
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001238 wait_for_completion(&mehci->gpt0_completion);
Hemant Kumarbc8bdf62012-10-17 12:29:51 -07001239 if (pdata && pdata->standalone_latency)
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05301240 pm_qos_update_request(&mehci->pm_qos_req_dma,
Hemant Kumarbc8bdf62012-10-17 12:29:51 -07001241 pdata->standalone_latency + 1);
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001242 spin_lock_irq(&ehci->lock);
1243 } else {
1244 dbg_log_event(NULL, "FPR: Tightloop", 0);
1245 /* do the resume in a tight loop */
1246 handshake(ehci, &ehci->regs->port_status[0],
1247 PORT_RESUME, 0, 22 * 1000);
1248 ehci_writel(ehci, ehci_readl(ehci,
1249 &ehci->regs->command) | CMD_RUN,
1250 &ehci->regs->command);
1251 }
1252
1253 if (mehci->resume_again) {
1254 int temp;
1255
1256 dbg_log_event(NULL, "FPR: Re-Resume", retry_cnt);
1257 pr_info("FPR: retry count: %d\n", retry_cnt);
1258 spin_unlock_irq(&ehci->lock);
1259 temp = ehci_readl(ehci, &ehci->regs->port_status[0]);
1260 temp &= ~PORT_RWC_BITS;
1261 temp |= PORT_SUSPEND;
1262 ehci_writel(ehci, temp, &ehci->regs->port_status[0]);
1263 /* Keep the bus idle for 5ms so that peripheral
1264 * can detect and initiate suspend
1265 */
1266 usleep_range(5000, 5000);
1267 dbg_log_event(NULL,
1268 "FPR: RResume",
1269 ehci_readl(ehci, &ehci->regs->port_status[0]));
1270 spin_lock_irq(&ehci->lock);
1271 mehci->resume_again = 0;
1272 retry_cnt++;
1273 goto resume_again;
1274 }
1275 }
1276
1277 dbg_log_event(NULL, "FPR: RT-Done", 0);
1278 mehci->resume_status = 1;
1279 spin_unlock_irq(&ehci->lock);
1280
1281 complete(&mehci->rt_completion);
1282
1283 return 0;
1284}
1285
Hemant Kumar45d211b2012-05-31 17:58:43 -07001286static int ehci_hsic_bus_resume(struct usb_hcd *hcd)
1287{
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001288 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
1289 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
1290 u32 temp;
1291 struct task_struct *resume_thread = NULL;
1292
1293 mehci->resume_status = 0;
1294 resume_thread = kthread_run(msm_hsic_resume_thread,
1295 mehci, "hsic_resume_thread");
1296 if (IS_ERR(resume_thread)) {
1297 pr_err("Error creating resume thread:%lu\n",
1298 PTR_ERR(resume_thread));
1299 return PTR_ERR(resume_thread);
1300 }
1301
1302 wait_for_completion(&mehci->rt_completion);
1303
1304 if (mehci->resume_status < 0)
1305 return mehci->resume_status;
1306
1307 dbg_log_event(NULL, "FPR: Wokeup", 0);
1308 spin_lock_irq(&ehci->lock);
1309 (void) ehci_readl(ehci, &ehci->regs->command);
1310
1311 temp = 0;
1312 if (ehci->async->qh_next.qh)
1313 temp |= CMD_ASE;
1314 if (ehci->periodic_sched)
1315 temp |= CMD_PSE;
1316 if (temp) {
1317 ehci->command |= temp;
1318 ehci_writel(ehci, ehci->command, &ehci->regs->command);
1319 }
1320
1321 ehci->next_statechange = jiffies + msecs_to_jiffies(5);
1322 hcd->state = HC_STATE_RUNNING;
1323 ehci->rh_state = EHCI_RH_RUNNING;
1324
1325 /* Now we can safely re-enable irqs */
1326 ehci_writel(ehci, INTR_MASK, &ehci->regs->intr_enable);
1327
1328 spin_unlock_irq(&ehci->lock);
1329
1330 return 0;
Hemant Kumar45d211b2012-05-31 17:58:43 -07001331}
1332
Jack Phamc0d41732012-09-28 17:45:38 -07001333#else
1334
1335#define ehci_hsic_bus_suspend NULL
1336#define ehci_hsic_bus_resume NULL
1337
1338#endif /* CONFIG_PM */
1339
Hemant Kumara3d4a6e2012-10-16 17:57:08 -07001340static void ehci_msm_set_autosuspend_delay(struct usb_device *dev)
1341{
1342 if (!dev->parent) /*for root hub no delay*/
1343 pm_runtime_set_autosuspend_delay(&dev->dev, 0);
1344 else
1345 pm_runtime_set_autosuspend_delay(&dev->dev, 200);
1346}
1347
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301348static struct hc_driver msm_hsic_driver = {
1349 .description = hcd_name,
1350 .product_desc = "Qualcomm EHCI Host Controller using HSIC",
1351 .hcd_priv_size = sizeof(struct msm_hsic_hcd),
1352
1353 /*
1354 * generic hardware linkage
1355 */
1356 .irq = msm_hsic_irq,
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -07001357 .flags = HCD_USB2 | HCD_MEMORY | HCD_OLD_ENUM,
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301358
1359 .reset = ehci_hsic_reset,
1360 .start = ehci_run,
1361
1362 .stop = ehci_stop,
1363 .shutdown = ehci_shutdown,
1364
1365 /*
1366 * managing i/o requests and associated device resources
1367 */
Hemant Kumar4d50a432012-08-15 09:06:35 -07001368 .urb_enqueue = ehci_urb_enqueue,
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301369 .urb_dequeue = ehci_urb_dequeue,
1370 .endpoint_disable = ehci_endpoint_disable,
1371 .endpoint_reset = ehci_endpoint_reset,
1372 .clear_tt_buffer_complete = ehci_clear_tt_buffer_complete,
1373
1374 /*
1375 * scheduling support
1376 */
1377 .get_frame_number = ehci_get_frame,
1378
1379 /*
1380 * root hub support
1381 */
1382 .hub_status_data = ehci_hub_status_data,
1383 .hub_control = ehci_hub_control,
1384 .relinquish_port = ehci_relinquish_port,
1385 .port_handed_over = ehci_port_handed_over,
1386
1387 /*
1388 * PM support
1389 */
Hemant Kumar45d211b2012-05-31 17:58:43 -07001390 .bus_suspend = ehci_hsic_bus_suspend,
1391 .bus_resume = ehci_hsic_bus_resume,
1392
Hemant Kumar4d50a432012-08-15 09:06:35 -07001393 .log_urb = dbg_log_event,
Hemant Kumar105d07f2012-07-02 15:33:07 -07001394 .dump_regs = dump_hsic_regs,
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -07001395
Hemant Kumara3d4a6e2012-10-16 17:57:08 -07001396 .set_autosuspend_delay = ehci_msm_set_autosuspend_delay,
Pavankumar Kondeti23720492013-01-29 14:41:58 +05301397 .reset_sof_bug_handler = ehci_hsic_reset_sof_bug_handler,
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301398};
1399
1400static int msm_hsic_init_clocks(struct msm_hsic_hcd *mehci, u32 init)
1401{
1402 int ret = 0;
1403
1404 if (!init)
1405 goto put_clocks;
1406
Lena Salman8c8ba382012-02-14 15:59:31 +02001407 /*core_clk is required for LINK protocol engine
1408 *clock rate appropriately set by target specific clock driver */
Manu Gautam5143b252012-01-05 19:25:23 -08001409 mehci->core_clk = clk_get(mehci->dev, "core_clk");
1410 if (IS_ERR(mehci->core_clk)) {
1411 dev_err(mehci->dev, "failed to get core_clk\n");
1412 ret = PTR_ERR(mehci->core_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301413 return ret;
1414 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301415
Ofir Cohen07944aa2012-12-23 13:41:57 +02001416 /* alt_core_clk is for LINK to be used during PHY RESET in
1417 * targets on which link does NOT use asynchronous reset methodology.
Lena Salman8c8ba382012-02-14 15:59:31 +02001418 * clock rate appropriately set by target specific clock driver */
Manu Gautam5143b252012-01-05 19:25:23 -08001419 mehci->alt_core_clk = clk_get(mehci->dev, "alt_core_clk");
Manu Gautam863b74132012-11-21 14:30:04 +05301420 if (IS_ERR(mehci->alt_core_clk))
1421 dev_dbg(mehci->dev, "failed to get alt_core_clk\n");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301422
Lena Salman8c8ba382012-02-14 15:59:31 +02001423 /* phy_clk is required for HSIC PHY operation
1424 * clock rate appropriately set by target specific clock driver */
Manu Gautam5143b252012-01-05 19:25:23 -08001425 mehci->phy_clk = clk_get(mehci->dev, "phy_clk");
1426 if (IS_ERR(mehci->phy_clk)) {
1427 dev_err(mehci->dev, "failed to get phy_clk\n");
1428 ret = PTR_ERR(mehci->phy_clk);
1429 goto put_alt_core_clk;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301430 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301431
1432 /* 10MHz cal_clk is required for calibration of I/O pads */
Manu Gautam5143b252012-01-05 19:25:23 -08001433 mehci->cal_clk = clk_get(mehci->dev, "cal_clk");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301434 if (IS_ERR(mehci->cal_clk)) {
Manu Gautam5143b252012-01-05 19:25:23 -08001435 dev_err(mehci->dev, "failed to get cal_clk\n");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301436 ret = PTR_ERR(mehci->cal_clk);
Manu Gautam5143b252012-01-05 19:25:23 -08001437 goto put_phy_clk;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301438 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301439
1440 /* ahb_clk is required for data transfers */
Manu Gautam5143b252012-01-05 19:25:23 -08001441 mehci->ahb_clk = clk_get(mehci->dev, "iface_clk");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301442 if (IS_ERR(mehci->ahb_clk)) {
Manu Gautam5143b252012-01-05 19:25:23 -08001443 dev_err(mehci->dev, "failed to get iface_clk\n");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301444 ret = PTR_ERR(mehci->ahb_clk);
1445 goto put_cal_clk;
1446 }
1447
Manu Gautam28b1bac2012-01-30 16:43:06 +05301448 clk_prepare_enable(mehci->core_clk);
1449 clk_prepare_enable(mehci->phy_clk);
1450 clk_prepare_enable(mehci->cal_clk);
1451 clk_prepare_enable(mehci->ahb_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301452
1453 return 0;
1454
1455put_clocks:
Jack Phamfd193eb2012-02-22 15:38:08 -08001456 if (!atomic_read(&mehci->in_lpm)) {
1457 clk_disable_unprepare(mehci->core_clk);
1458 clk_disable_unprepare(mehci->phy_clk);
1459 clk_disable_unprepare(mehci->cal_clk);
1460 clk_disable_unprepare(mehci->ahb_clk);
1461 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301462 clk_put(mehci->ahb_clk);
1463put_cal_clk:
1464 clk_put(mehci->cal_clk);
Manu Gautam5143b252012-01-05 19:25:23 -08001465put_phy_clk:
1466 clk_put(mehci->phy_clk);
1467put_alt_core_clk:
Manu Gautam863b74132012-11-21 14:30:04 +05301468 if (!IS_ERR(mehci->alt_core_clk))
1469 clk_put(mehci->alt_core_clk);
Manu Gautam5143b252012-01-05 19:25:23 -08001470 clk_put(mehci->core_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301471
1472 return ret;
1473}
Vamsi Krishna34f01582011-12-14 19:54:42 -08001474static irqreturn_t hsic_peripheral_status_change(int irq, void *dev_id)
1475{
1476 struct msm_hsic_hcd *mehci = dev_id;
1477
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001478 pr_debug("%s: mehci:%p dev_id:%p\n", __func__, mehci, dev_id);
Vamsi Krishna34f01582011-12-14 19:54:42 -08001479
1480 if (mehci)
1481 msm_hsic_config_gpios(mehci, 0);
1482
1483 return IRQ_HANDLED;
1484}
1485
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001486static irqreturn_t msm_hsic_wakeup_irq(int irq, void *data)
1487{
1488 struct msm_hsic_hcd *mehci = data;
Pavankumar Kondeti968cf272012-09-27 12:04:12 +05301489 int ret;
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001490
Hemant Kumar6fd65032012-05-23 13:02:24 -07001491 mehci->wakeup_int_cnt++;
Hemant Kumar45d211b2012-05-31 17:58:43 -07001492 dbg_log_event(NULL, "Remote Wakeup IRQ", mehci->wakeup_int_cnt);
Hemant Kumar6fd65032012-05-23 13:02:24 -07001493 dev_dbg(mehci->dev, "%s: hsic remote wakeup interrupt cnt: %u\n",
1494 __func__, mehci->wakeup_int_cnt);
1495
Sudhir Sharma1673e302012-08-27 17:37:24 -07001496 wake_lock(&mehci->wlock);
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001497
Hemant Kumard343c012012-09-06 19:57:14 -07001498 spin_lock(&mehci->wakeup_lock);
Jack Phamfe441ea2012-03-23 17:03:15 -07001499 if (mehci->wakeup_irq_enabled) {
1500 mehci->wakeup_irq_enabled = 0;
1501 disable_irq_wake(irq);
1502 disable_irq_nosync(irq);
1503 }
Hemant Kumard343c012012-09-06 19:57:14 -07001504 spin_unlock(&mehci->wakeup_lock);
Jack Phamfe441ea2012-03-23 17:03:15 -07001505
Sudhir Sharma1673e302012-08-27 17:37:24 -07001506 if (!atomic_read(&mehci->pm_usage_cnt)) {
Pavankumar Kondeti968cf272012-09-27 12:04:12 +05301507 ret = pm_runtime_get(mehci->dev);
1508 /*
1509 * HSIC runtime resume can race with us.
1510 * if we are active (ret == 1) or resuming
1511 * (ret == -EINPROGRESS), decrement the
1512 * PM usage counter before returning.
1513 */
1514 if ((ret == 1) || (ret == -EINPROGRESS))
1515 pm_runtime_put_noidle(mehci->dev);
1516 else
1517 atomic_set(&mehci->pm_usage_cnt, 1);
Sudhir Sharma1673e302012-08-27 17:37:24 -07001518 }
1519
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001520 return IRQ_HANDLED;
1521}
1522
Hemant Kumare6275972012-02-29 20:06:21 -08001523static int ehci_hsic_msm_bus_show(struct seq_file *s, void *unused)
1524{
1525 if (debug_bus_voting_enabled)
1526 seq_printf(s, "enabled\n");
1527 else
1528 seq_printf(s, "disabled\n");
1529
1530 return 0;
1531}
1532
1533static int ehci_hsic_msm_bus_open(struct inode *inode, struct file *file)
1534{
1535 return single_open(file, ehci_hsic_msm_bus_show, inode->i_private);
1536}
1537
1538static ssize_t ehci_hsic_msm_bus_write(struct file *file,
1539 const char __user *ubuf, size_t count, loff_t *ppos)
1540{
1541 char buf[8];
1542 int ret;
1543 struct seq_file *s = file->private_data;
1544 struct msm_hsic_hcd *mehci = s->private;
1545
1546 memset(buf, 0x00, sizeof(buf));
1547
1548 if (copy_from_user(&buf, ubuf, min_t(size_t, sizeof(buf) - 1, count)))
1549 return -EFAULT;
1550
1551 if (!strncmp(buf, "enable", 6)) {
1552 /* Do not vote here. Let hsic driver decide when to vote */
1553 debug_bus_voting_enabled = true;
1554 } else {
1555 debug_bus_voting_enabled = false;
1556 if (mehci->bus_perf_client) {
1557 ret = msm_bus_scale_client_update_request(
1558 mehci->bus_perf_client, 0);
1559 if (ret)
1560 dev_err(mehci->dev, "%s: Failed to devote "
1561 "for bus bw %d\n", __func__, ret);
1562 }
1563 }
1564
1565 return count;
1566}
1567
1568const struct file_operations ehci_hsic_msm_bus_fops = {
1569 .open = ehci_hsic_msm_bus_open,
1570 .read = seq_read,
1571 .write = ehci_hsic_msm_bus_write,
1572 .llseek = seq_lseek,
1573 .release = single_release,
1574};
1575
Hemant Kumar6fd65032012-05-23 13:02:24 -07001576static int ehci_hsic_msm_wakeup_cnt_show(struct seq_file *s, void *unused)
1577{
1578 struct msm_hsic_hcd *mehci = s->private;
1579
1580 seq_printf(s, "%u\n", mehci->wakeup_int_cnt);
1581
1582 return 0;
1583}
1584
1585static int ehci_hsic_msm_wakeup_cnt_open(struct inode *inode, struct file *f)
1586{
1587 return single_open(f, ehci_hsic_msm_wakeup_cnt_show, inode->i_private);
1588}
1589
1590const struct file_operations ehci_hsic_msm_wakeup_cnt_fops = {
1591 .open = ehci_hsic_msm_wakeup_cnt_open,
1592 .read = seq_read,
1593 .llseek = seq_lseek,
1594 .release = single_release,
1595};
1596
Hemant Kumar45d211b2012-05-31 17:58:43 -07001597static int ehci_hsic_msm_data_events_show(struct seq_file *s, void *unused)
1598{
1599 unsigned long flags;
1600 unsigned i;
1601
1602 read_lock_irqsave(&dbg_hsic_data.lck, flags);
1603
1604 i = dbg_hsic_data.idx;
1605 for (dbg_inc(&i); i != dbg_hsic_data.idx; dbg_inc(&i)) {
1606 if (!strnlen(dbg_hsic_data.buf[i], DBG_MSG_LEN))
1607 continue;
1608 seq_printf(s, "%s\n", dbg_hsic_data.buf[i]);
1609 }
1610
1611 read_unlock_irqrestore(&dbg_hsic_data.lck, flags);
1612
1613 return 0;
1614}
1615
1616static int ehci_hsic_msm_data_events_open(struct inode *inode, struct file *f)
1617{
1618 return single_open(f, ehci_hsic_msm_data_events_show, inode->i_private);
1619}
1620
1621const struct file_operations ehci_hsic_msm_dbg_data_fops = {
1622 .open = ehci_hsic_msm_data_events_open,
1623 .read = seq_read,
1624 .llseek = seq_lseek,
1625 .release = single_release,
1626};
1627
1628static int ehci_hsic_msm_ctrl_events_show(struct seq_file *s, void *unused)
1629{
1630 unsigned long flags;
1631 unsigned i;
1632
1633 read_lock_irqsave(&dbg_hsic_ctrl.lck, flags);
1634
1635 i = dbg_hsic_ctrl.idx;
1636 for (dbg_inc(&i); i != dbg_hsic_ctrl.idx; dbg_inc(&i)) {
1637 if (!strnlen(dbg_hsic_ctrl.buf[i], DBG_MSG_LEN))
1638 continue;
1639 seq_printf(s, "%s\n", dbg_hsic_ctrl.buf[i]);
1640 }
1641
1642 read_unlock_irqrestore(&dbg_hsic_ctrl.lck, flags);
1643
1644 return 0;
1645}
1646
1647static int ehci_hsic_msm_ctrl_events_open(struct inode *inode, struct file *f)
1648{
1649 return single_open(f, ehci_hsic_msm_ctrl_events_show, inode->i_private);
1650}
1651
1652const struct file_operations ehci_hsic_msm_dbg_ctrl_fops = {
1653 .open = ehci_hsic_msm_ctrl_events_open,
1654 .read = seq_read,
1655 .llseek = seq_lseek,
1656 .release = single_release,
1657};
1658
Hemant Kumare6275972012-02-29 20:06:21 -08001659static struct dentry *ehci_hsic_msm_dbg_root;
1660static int ehci_hsic_msm_debugfs_init(struct msm_hsic_hcd *mehci)
1661{
1662 struct dentry *ehci_hsic_msm_dentry;
1663
1664 ehci_hsic_msm_dbg_root = debugfs_create_dir("ehci_hsic_msm_dbg", NULL);
1665
1666 if (!ehci_hsic_msm_dbg_root || IS_ERR(ehci_hsic_msm_dbg_root))
1667 return -ENODEV;
1668
1669 ehci_hsic_msm_dentry = debugfs_create_file("bus_voting",
1670 S_IRUGO | S_IWUSR,
1671 ehci_hsic_msm_dbg_root, mehci,
1672 &ehci_hsic_msm_bus_fops);
1673
1674 if (!ehci_hsic_msm_dentry) {
1675 debugfs_remove_recursive(ehci_hsic_msm_dbg_root);
1676 return -ENODEV;
1677 }
1678
Hemant Kumar6fd65032012-05-23 13:02:24 -07001679 ehci_hsic_msm_dentry = debugfs_create_file("wakeup_cnt",
1680 S_IRUGO,
1681 ehci_hsic_msm_dbg_root, mehci,
1682 &ehci_hsic_msm_wakeup_cnt_fops);
1683
1684 if (!ehci_hsic_msm_dentry) {
1685 debugfs_remove_recursive(ehci_hsic_msm_dbg_root);
1686 return -ENODEV;
1687 }
1688
Hemant Kumar45d211b2012-05-31 17:58:43 -07001689 ehci_hsic_msm_dentry = debugfs_create_file("show_ctrl_events",
1690 S_IRUGO,
1691 ehci_hsic_msm_dbg_root, mehci,
1692 &ehci_hsic_msm_dbg_ctrl_fops);
1693
1694 if (!ehci_hsic_msm_dentry) {
1695 debugfs_remove_recursive(ehci_hsic_msm_dbg_root);
1696 return -ENODEV;
1697 }
1698
1699 ehci_hsic_msm_dentry = debugfs_create_file("show_data_events",
1700 S_IRUGO,
1701 ehci_hsic_msm_dbg_root, mehci,
1702 &ehci_hsic_msm_dbg_data_fops);
1703
1704 if (!ehci_hsic_msm_dentry) {
1705 debugfs_remove_recursive(ehci_hsic_msm_dbg_root);
1706 return -ENODEV;
1707 }
1708
Hemant Kumare6275972012-02-29 20:06:21 -08001709 return 0;
1710}
1711
1712static void ehci_hsic_msm_debugfs_cleanup(void)
1713{
1714 debugfs_remove_recursive(ehci_hsic_msm_dbg_root);
1715}
1716
Manu Gautam863b74132012-11-21 14:30:04 +05301717struct msm_hsic_host_platform_data *msm_hsic_dt_to_pdata(
1718 struct platform_device *pdev)
1719{
1720 struct device_node *node = pdev->dev.of_node;
1721 struct msm_hsic_host_platform_data *pdata;
Ido Shayevitzca5dc492013-01-09 14:49:59 +02001722 int res_gpio;
Manu Gautam863b74132012-11-21 14:30:04 +05301723
1724 pdata = devm_kzalloc(&pdev->dev, sizeof(*pdata), GFP_KERNEL);
1725 if (!pdata) {
1726 dev_err(&pdev->dev, "unable to allocate platform data\n");
1727 return NULL;
1728 }
Manu Gautam863b74132012-11-21 14:30:04 +05301729
Ido Shayevitzca5dc492013-01-09 14:49:59 +02001730 res_gpio = of_get_named_gpio(node, "hsic,strobe-gpio", 0);
1731 if (res_gpio < 0)
1732 res_gpio = 0;
1733 pdata->strobe = res_gpio;
1734
1735 res_gpio = of_get_named_gpio(node, "hsic,data-gpio", 0);
1736 if (res_gpio < 0)
1737 res_gpio = 0;
1738 pdata->data = res_gpio;
Manu Gautam863b74132012-11-21 14:30:04 +05301739
1740 pdata->ignore_cal_pad_config = of_property_read_bool(node,
1741 "hsic,ignore-cal-pad-config");
1742 of_property_read_u32(node, "hsic,strobe-pad-offset",
1743 &pdata->strobe_pad_offset);
1744 of_property_read_u32(node, "hsic,data-pad-offset",
1745 &pdata->data_pad_offset);
1746
1747 return pdata;
1748}
1749
1750
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301751static int __devinit ehci_hsic_msm_probe(struct platform_device *pdev)
1752{
1753 struct usb_hcd *hcd;
1754 struct resource *res;
1755 struct msm_hsic_hcd *mehci;
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +05301756 struct msm_hsic_host_platform_data *pdata;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301757 int ret;
1758
1759 dev_dbg(&pdev->dev, "ehci_msm-hsic probe\n");
1760
Manu Gautam863b74132012-11-21 14:30:04 +05301761 if (pdev->dev.of_node) {
1762 dev_dbg(&pdev->dev, "device tree enabled\n");
1763 pdev->dev.platform_data = msm_hsic_dt_to_pdata(pdev);
1764 dev_set_name(&pdev->dev, ehci_msm_hsic_driver.driver.name);
1765 }
1766 if (!pdev->dev.platform_data)
1767 dev_dbg(&pdev->dev, "No platform data given\n");
1768
Ofir Cohen4cc55372012-11-20 11:17:24 +02001769 if (!pdev->dev.dma_mask)
1770 pdev->dev.dma_mask = &ehci_msm_hsic_dma_mask;
1771 if (!pdev->dev.coherent_dma_mask)
1772 pdev->dev.coherent_dma_mask = DMA_BIT_MASK(32);
1773
Vijayavardhan Vennapusaafbbb8f2012-04-13 16:28:45 +05301774 /* After parent device's probe is executed, it will be put in suspend
1775 * mode. When child device's probe is called, driver core is not
1776 * resuming parent device due to which parent will be in suspend even
1777 * though child is active. Hence resume the parent device explicitly.
1778 */
1779 if (pdev->dev.parent)
1780 pm_runtime_get_sync(pdev->dev.parent);
1781
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301782 hcd = usb_create_hcd(&msm_hsic_driver, &pdev->dev,
1783 dev_name(&pdev->dev));
1784 if (!hcd) {
1785 dev_err(&pdev->dev, "Unable to create HCD\n");
1786 return -ENOMEM;
1787 }
1788
Pavankumar Kondeti6f199aa2012-09-18 17:52:51 +05301789 hcd_to_bus(hcd)->skip_resume = true;
1790
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301791 hcd->irq = platform_get_irq(pdev, 0);
1792 if (hcd->irq < 0) {
1793 dev_err(&pdev->dev, "Unable to get IRQ resource\n");
1794 ret = hcd->irq;
1795 goto put_hcd;
1796 }
1797
1798 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
1799 if (!res) {
1800 dev_err(&pdev->dev, "Unable to get memory resource\n");
1801 ret = -ENODEV;
1802 goto put_hcd;
1803 }
1804
1805 hcd->rsrc_start = res->start;
1806 hcd->rsrc_len = resource_size(res);
1807 hcd->regs = ioremap(hcd->rsrc_start, hcd->rsrc_len);
1808 if (!hcd->regs) {
1809 dev_err(&pdev->dev, "ioremap failed\n");
1810 ret = -ENOMEM;
1811 goto put_hcd;
1812 }
1813
1814 mehci = hcd_to_hsic(hcd);
1815 mehci->dev = &pdev->dev;
Hemant Kumar30d361c2012-08-20 14:44:40 -07001816 pdata = mehci->dev->platform_data;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301817
Hemant Kumard343c012012-09-06 19:57:14 -07001818 spin_lock_init(&mehci->wakeup_lock);
1819
Hemant Kumar38ce5d82012-05-29 13:00:58 -07001820 mehci->ehci.susp_sof_bug = 1;
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -07001821 mehci->ehci.reset_sof_bug = 1;
Hemant Kumar38ce5d82012-05-29 13:00:58 -07001822
Hemant Kumare4040492012-06-21 17:35:42 -07001823 mehci->ehci.resume_sof_bug = 1;
1824
Hemant Kumar30d361c2012-08-20 14:44:40 -07001825 if (pdata)
1826 mehci->ehci.log2_irq_thresh = pdata->log2_irq_thresh;
Hemant Kumar933e0402012-05-22 11:11:40 -07001827
Ofir Cohen4cc55372012-11-20 11:17:24 +02001828 ret = msm_hsic_init_gdsc(mehci, 1);
1829 if (ret) {
1830 dev_err(&pdev->dev, "unable to initialize GDSC\n");
1831 ret = -ENODEV;
1832 goto put_hcd;
1833 }
1834
Vamsi Krishna34f01582011-12-14 19:54:42 -08001835 res = platform_get_resource_byname(pdev,
1836 IORESOURCE_IRQ,
1837 "peripheral_status_irq");
1838 if (res)
1839 mehci->peripheral_status_irq = res->start;
1840
Jack Pham0cc75c42012-10-10 02:03:50 +02001841 res = platform_get_resource_byname(pdev, IORESOURCE_IRQ, "wakeup");
Hemant Kumar6fd65032012-05-23 13:02:24 -07001842 if (res) {
Jack Pham0cc75c42012-10-10 02:03:50 +02001843 mehci->wakeup_irq = res->start;
Hemant Kumar6fd65032012-05-23 13:02:24 -07001844 dev_dbg(mehci->dev, "wakeup_irq: %d\n", mehci->wakeup_irq);
1845 }
1846
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301847 ret = msm_hsic_init_clocks(mehci, 1);
1848 if (ret) {
1849 dev_err(&pdev->dev, "unable to initialize clocks\n");
1850 ret = -ENODEV;
1851 goto unmap;
1852 }
1853
1854 ret = msm_hsic_init_vddcx(mehci, 1);
1855 if (ret) {
1856 dev_err(&pdev->dev, "unable to initialize VDDCX\n");
1857 ret = -ENODEV;
1858 goto deinit_clocks;
1859 }
1860
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001861 init_completion(&mehci->rt_completion);
1862 init_completion(&mehci->gpt0_completion);
Ofir Cohen5dddb152012-11-14 11:18:50 +02001863
1864 msm_hsic_phy_reset(mehci);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301865
Hemant Kumar2309eaa2012-08-14 16:46:42 -07001866 ehci_wq = create_singlethread_workqueue("ehci_wq");
1867 if (!ehci_wq) {
1868 dev_err(&pdev->dev, "unable to create workqueue\n");
1869 ret = -ENOMEM;
1870 goto deinit_vddcx;
1871 }
1872
1873 INIT_WORK(&mehci->bus_vote_w, ehci_hsic_bus_vote_w);
1874
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301875 ret = usb_add_hcd(hcd, hcd->irq, IRQF_SHARED);
1876 if (ret) {
1877 dev_err(&pdev->dev, "unable to register HCD\n");
Ofir Cohen5dddb152012-11-14 11:18:50 +02001878 goto destroy_wq;
1879 }
1880
1881 ret = msm_hsic_start(mehci);
1882 if (ret) {
1883 dev_err(&pdev->dev, "unable to initialize PHY\n");
1884 goto destroy_wq;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301885 }
1886
1887 device_init_wakeup(&pdev->dev, 1);
Sudhir Sharma1673e302012-08-27 17:37:24 -07001888 wake_lock_init(&mehci->wlock, WAKE_LOCK_SUSPEND, dev_name(&pdev->dev));
1889 wake_lock(&mehci->wlock);
Vamsi Krishna34f01582011-12-14 19:54:42 -08001890
1891 if (mehci->peripheral_status_irq) {
1892 ret = request_threaded_irq(mehci->peripheral_status_irq,
1893 NULL, hsic_peripheral_status_change,
1894 IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING
1895 | IRQF_SHARED,
1896 "hsic_peripheral_status", mehci);
1897 if (ret)
1898 dev_err(&pdev->dev, "%s:request_irq:%d failed:%d",
1899 __func__, mehci->peripheral_status_irq, ret);
1900 }
1901
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001902 /* configure wakeup irq */
Hemant Kumar6fd65032012-05-23 13:02:24 -07001903 if (mehci->wakeup_irq) {
Hemant Kumarbbcdcbc2012-08-30 14:23:38 -07001904 /* In case if wakeup gpio is pulled high at this point
1905 * remote wakeup interrupt fires right after request_irq.
1906 * Remote wake up interrupt only needs to be enabled when
1907 * HSIC bus goes to suspend.
1908 */
1909 irq_set_status_flags(mehci->wakeup_irq, IRQ_NOAUTOEN);
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001910 ret = request_irq(mehci->wakeup_irq, msm_hsic_wakeup_irq,
Hemant Kumar6fd65032012-05-23 13:02:24 -07001911 IRQF_TRIGGER_HIGH,
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001912 "msm_hsic_wakeup", mehci);
Hemant Kumarbbcdcbc2012-08-30 14:23:38 -07001913 if (ret) {
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001914 dev_err(&pdev->dev, "request_irq(%d) failed: %d\n",
1915 mehci->wakeup_irq, ret);
1916 mehci->wakeup_irq = 0;
1917 }
1918 }
1919
Hemant Kumare6275972012-02-29 20:06:21 -08001920 ret = ehci_hsic_msm_debugfs_init(mehci);
1921 if (ret)
1922 dev_dbg(&pdev->dev, "mode debugfs file is"
1923 "not available\n");
1924
1925 if (pdata && pdata->bus_scale_table) {
1926 mehci->bus_perf_client =
1927 msm_bus_scale_register_client(pdata->bus_scale_table);
1928 /* Configure BUS performance parameters for MAX bandwidth */
1929 if (mehci->bus_perf_client) {
Hemant Kumar2309eaa2012-08-14 16:46:42 -07001930 mehci->bus_vote = true;
1931 queue_work(ehci_wq, &mehci->bus_vote_w);
Hemant Kumare6275972012-02-29 20:06:21 -08001932 } else {
1933 dev_err(&pdev->dev, "%s: Failed to register BUS "
1934 "scaling client!!\n", __func__);
1935 }
1936 }
1937
Hemant Kumar105d07f2012-07-02 15:33:07 -07001938 __mehci = mehci;
1939
Hemant Kumarbc8bdf62012-10-17 12:29:51 -07001940 if (pdata && pdata->standalone_latency)
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05301941 pm_qos_add_request(&mehci->pm_qos_req_dma,
Hemant Kumarbc8bdf62012-10-17 12:29:51 -07001942 PM_QOS_CPU_DMA_LATENCY, pdata->standalone_latency + 1);
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05301943
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301944 /*
1945 * This pdev->dev is assigned parent of root-hub by USB core,
1946 * hence, runtime framework automatically calls this driver's
1947 * runtime APIs based on root-hub's state.
1948 */
1949 pm_runtime_set_active(&pdev->dev);
1950 pm_runtime_enable(&pdev->dev);
Vijayavardhan Vennapusaafbbb8f2012-04-13 16:28:45 +05301951 /* Decrement the parent device's counter after probe.
1952 * As child is active, parent will not be put into
1953 * suspend mode.
1954 */
1955 if (pdev->dev.parent)
1956 pm_runtime_put_sync(pdev->dev.parent);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301957
1958 return 0;
1959
Ofir Cohen5dddb152012-11-14 11:18:50 +02001960destroy_wq:
Hemant Kumar2309eaa2012-08-14 16:46:42 -07001961 destroy_workqueue(ehci_wq);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301962deinit_vddcx:
1963 msm_hsic_init_vddcx(mehci, 0);
Ofir Cohen4cc55372012-11-20 11:17:24 +02001964 msm_hsic_init_gdsc(mehci, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301965deinit_clocks:
1966 msm_hsic_init_clocks(mehci, 0);
1967unmap:
1968 iounmap(hcd->regs);
1969put_hcd:
1970 usb_put_hcd(hcd);
1971
1972 return ret;
1973}
1974
1975static int __devexit ehci_hsic_msm_remove(struct platform_device *pdev)
1976{
1977 struct usb_hcd *hcd = platform_get_drvdata(pdev);
1978 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05301979 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
1980
Hemant Kumarbc8bdf62012-10-17 12:29:51 -07001981 if (pdata && pdata->standalone_latency)
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05301982 pm_qos_remove_request(&mehci->pm_qos_req_dma);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301983
Vamsi Krishna34f01582011-12-14 19:54:42 -08001984 if (mehci->peripheral_status_irq)
1985 free_irq(mehci->peripheral_status_irq, mehci);
Jack Phamfe441ea2012-03-23 17:03:15 -07001986
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001987 if (mehci->wakeup_irq) {
Jack Phamfe441ea2012-03-23 17:03:15 -07001988 if (mehci->wakeup_irq_enabled)
1989 disable_irq_wake(mehci->wakeup_irq);
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001990 free_irq(mehci->wakeup_irq, mehci);
1991 }
Vamsi Krishna34f01582011-12-14 19:54:42 -08001992
Hemant Kumar2309eaa2012-08-14 16:46:42 -07001993 /*
1994 * If the update request is called after unregister, the request will
1995 * fail. Results are undefined if unregister is called in the middle of
1996 * update request.
1997 */
1998 mehci->bus_vote = false;
1999 cancel_work_sync(&mehci->bus_vote_w);
2000
Hemant Kumare6275972012-02-29 20:06:21 -08002001 if (mehci->bus_perf_client)
2002 msm_bus_scale_unregister_client(mehci->bus_perf_client);
2003
2004 ehci_hsic_msm_debugfs_cleanup();
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302005 device_init_wakeup(&pdev->dev, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302006 pm_runtime_set_suspended(&pdev->dev);
2007
Hemant Kumar2309eaa2012-08-14 16:46:42 -07002008 destroy_workqueue(ehci_wq);
2009
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302010 usb_remove_hcd(hcd);
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +05302011 msm_hsic_config_gpios(mehci, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302012 msm_hsic_init_vddcx(mehci, 0);
Ofir Cohen4cc55372012-11-20 11:17:24 +02002013 msm_hsic_init_gdsc(mehci, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302014
2015 msm_hsic_init_clocks(mehci, 0);
Sudhir Sharma1673e302012-08-27 17:37:24 -07002016 wake_lock_destroy(&mehci->wlock);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302017 iounmap(hcd->regs);
2018 usb_put_hcd(hcd);
2019
2020 return 0;
2021}
2022
2023#ifdef CONFIG_PM_SLEEP
2024static int msm_hsic_pm_suspend(struct device *dev)
2025{
Jack Phambe05fbb2012-05-16 10:56:26 -07002026 int ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302027 struct usb_hcd *hcd = dev_get_drvdata(dev);
2028 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
2029
2030 dev_dbg(dev, "ehci-msm-hsic PM suspend\n");
2031
Hemant Kumar45d211b2012-05-31 17:58:43 -07002032 dbg_log_event(NULL, "PM Suspend", 0);
2033
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302034 if (device_may_wakeup(dev))
2035 enable_irq_wake(hcd->irq);
2036
Jack Phambe05fbb2012-05-16 10:56:26 -07002037 ret = msm_hsic_suspend(mehci);
2038
2039 if (ret && device_may_wakeup(dev))
2040 disable_irq_wake(hcd->irq);
2041
2042 return ret;
Jack Phamfe441ea2012-03-23 17:03:15 -07002043}
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302044
Jack Pham16b06f82012-08-14 20:03:59 -07002045static int msm_hsic_pm_suspend_noirq(struct device *dev)
2046{
2047 struct usb_hcd *hcd = dev_get_drvdata(dev);
2048 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
2049
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +05302050 if (atomic_read(&mehci->async_int)) {
Jack Pham16b06f82012-08-14 20:03:59 -07002051 dev_dbg(dev, "suspend_noirq: Aborting due to pending interrupt\n");
2052 return -EBUSY;
2053 }
2054
2055 return 0;
2056}
2057
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302058static int msm_hsic_pm_resume(struct device *dev)
2059{
2060 int ret;
2061 struct usb_hcd *hcd = dev_get_drvdata(dev);
2062 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
2063
Jack Pham16b06f82012-08-14 20:03:59 -07002064 dev_dbg(dev, "ehci-msm-hsic PM resume\n");
Hemant Kumar45d211b2012-05-31 17:58:43 -07002065 dbg_log_event(NULL, "PM Resume", 0);
2066
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302067 if (device_may_wakeup(dev))
2068 disable_irq_wake(hcd->irq);
2069
Pavankumar Kondeti6f199aa2012-09-18 17:52:51 +05302070 /*
2071 * Keep HSIC in Low Power Mode if system is resumed
2072 * by any other wakeup source. HSIC is resumed later
2073 * when remote wakeup is received or interface driver
2074 * start I/O.
2075 */
Pavankumar Kondeti41d004c2012-11-09 10:54:00 +05302076 if (!atomic_read(&mehci->pm_usage_cnt) &&
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +05302077 !atomic_read(&mehci->async_int) &&
Pavankumar Kondeti41d004c2012-11-09 10:54:00 +05302078 pm_runtime_suspended(dev))
Pavankumar Kondeti6f199aa2012-09-18 17:52:51 +05302079 return 0;
2080
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302081 ret = msm_hsic_resume(mehci);
2082 if (ret)
2083 return ret;
2084
2085 /* Bring the device to full powered state upon system resume */
2086 pm_runtime_disable(dev);
2087 pm_runtime_set_active(dev);
2088 pm_runtime_enable(dev);
2089
2090 return 0;
2091}
2092#endif
2093
2094#ifdef CONFIG_PM_RUNTIME
2095static int msm_hsic_runtime_idle(struct device *dev)
2096{
2097 dev_dbg(dev, "EHCI runtime idle\n");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302098 return 0;
2099}
2100
2101static int msm_hsic_runtime_suspend(struct device *dev)
2102{
2103 struct usb_hcd *hcd = dev_get_drvdata(dev);
2104 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
2105
2106 dev_dbg(dev, "EHCI runtime suspend\n");
Hemant Kumar45d211b2012-05-31 17:58:43 -07002107
2108 dbg_log_event(NULL, "Run Time PM Suspend", 0);
2109
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302110 return msm_hsic_suspend(mehci);
2111}
2112
2113static int msm_hsic_runtime_resume(struct device *dev)
2114{
2115 struct usb_hcd *hcd = dev_get_drvdata(dev);
2116 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
2117
2118 dev_dbg(dev, "EHCI runtime resume\n");
Hemant Kumar45d211b2012-05-31 17:58:43 -07002119
2120 dbg_log_event(NULL, "Run Time PM Resume", 0);
2121
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302122 return msm_hsic_resume(mehci);
2123}
2124#endif
2125
2126#ifdef CONFIG_PM
2127static const struct dev_pm_ops msm_hsic_dev_pm_ops = {
2128 SET_SYSTEM_SLEEP_PM_OPS(msm_hsic_pm_suspend, msm_hsic_pm_resume)
Jack Pham16b06f82012-08-14 20:03:59 -07002129 .suspend_noirq = msm_hsic_pm_suspend_noirq,
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302130 SET_RUNTIME_PM_OPS(msm_hsic_runtime_suspend, msm_hsic_runtime_resume,
2131 msm_hsic_runtime_idle)
2132};
2133#endif
Ofir Cohen4cc55372012-11-20 11:17:24 +02002134static const struct of_device_id hsic_host_dt_match[] = {
2135 { .compatible = "qcom,hsic-host",
2136 },
2137 {}
2138};
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302139static struct platform_driver ehci_msm_hsic_driver = {
2140 .probe = ehci_hsic_msm_probe,
2141 .remove = __devexit_p(ehci_hsic_msm_remove),
2142 .driver = {
2143 .name = "msm_hsic_host",
2144#ifdef CONFIG_PM
2145 .pm = &msm_hsic_dev_pm_ops,
2146#endif
Ofir Cohen4cc55372012-11-20 11:17:24 +02002147 .of_match_table = hsic_host_dt_match,
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302148 },
2149};