blob: 4a085aa8d408237942cf9e74e68ff375fae1b6e7 [file] [log] [blame]
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301/* ehci-msm-hsic.c - HSUSB Host Controller Driver Implementation
2 *
Ido Shayevitz95c46882013-01-20 13:47:46 +02003 * Copyright (c) 2011-2013, The 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>
Shimrit Malichi82d7cbc2013-03-12 15:36:54 +020052#include "hbm.c"
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053053
54#define MSM_USB_BASE (hcd->regs)
Hemant Kumar105d07f2012-07-02 15:33:07 -070055#define USB_REG_START_OFFSET 0x90
56#define USB_REG_END_OFFSET 0x250
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053057
Hemant Kumar2309eaa2012-08-14 16:46:42 -070058static struct workqueue_struct *ehci_wq;
Vamsi Krishnafce1bd22012-08-17 17:12:26 -070059struct ehci_timer {
60#define GPT_LD(p) ((p) & 0x00FFFFFF)
61 u32 gptimer0_ld;
62#define GPT_RUN BIT(31)
63#define GPT_RESET BIT(30)
64#define GPT_MODE BIT(24)
65#define GPT_CNT(p) ((p) & 0x00FFFFFF)
66 u32 gptimer0_ctrl;
67
68 u32 gptimer1_ld;
69 u32 gptimer1_ctrl;
70};
Hemant Kumar2309eaa2012-08-14 16:46:42 -070071
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053072struct msm_hsic_hcd {
73 struct ehci_hcd ehci;
Hemant Kumard343c012012-09-06 19:57:14 -070074 spinlock_t wakeup_lock;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053075 struct device *dev;
76 struct clk *ahb_clk;
Manu Gautam5143b252012-01-05 19:25:23 -080077 struct clk *core_clk;
78 struct clk *alt_core_clk;
79 struct clk *phy_clk;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053080 struct clk *cal_clk;
81 struct regulator *hsic_vddcx;
Ofir Cohen4cc55372012-11-20 11:17:24 +020082 struct regulator *hsic_gdsc;
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +053083 atomic_t async_int;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053084 atomic_t in_lpm;
Sudhir Sharma1673e302012-08-27 17:37:24 -070085 struct wake_lock wlock;
Vamsi Krishna34f01582011-12-14 19:54:42 -080086 int peripheral_status_irq;
Vamsi Krishna6921cbe2012-02-21 18:34:43 -080087 int wakeup_irq;
Jack Phamfe441ea2012-03-23 17:03:15 -070088 bool wakeup_irq_enabled;
Manu Gautam3fb60ca2013-02-13 18:33:33 +053089 int async_irq;
90 uint32_t async_int_cnt;
Sudhir Sharma1673e302012-08-27 17:37:24 -070091 atomic_t pm_usage_cnt;
Hemant Kumare6275972012-02-29 20:06:21 -080092 uint32_t bus_perf_client;
Hemant Kumar6fd65032012-05-23 13:02:24 -070093 uint32_t wakeup_int_cnt;
Amit Blayd6ea6102012-06-07 16:26:24 +030094 enum usb_vdd_type vdd_type;
Hemant Kumar2309eaa2012-08-14 16:46:42 -070095
96 struct work_struct bus_vote_w;
97 bool bus_vote;
Vamsi Krishnafce1bd22012-08-17 17:12:26 -070098
99 /* gp timer */
100 struct ehci_timer __iomem *timer;
101 struct completion gpt0_completion;
102 struct completion rt_completion;
103 int resume_status;
104 int resume_again;
Pavankumar Kondeti23720492013-01-29 14:41:58 +0530105 int bus_reset;
106 int reset_again;
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +0530107
108 struct pm_qos_request pm_qos_req_dma;
Shimrit Malichi82d7cbc2013-03-12 15:36:54 +0200109 unsigned enable_hbm:1;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530110};
111
Hemant Kumar105d07f2012-07-02 15:33:07 -0700112struct msm_hsic_hcd *__mehci;
113
Hemant Kumare6275972012-02-29 20:06:21 -0800114static bool debug_bus_voting_enabled = true;
Ofir Cohen4cc55372012-11-20 11:17:24 +0200115static u64 ehci_msm_hsic_dma_mask = DMA_BIT_MASK(32);
116
Manu Gautam863b74132012-11-21 14:30:04 +0530117static struct platform_driver ehci_msm_hsic_driver;
Hemant Kumar45d211b2012-05-31 17:58:43 -0700118
Hemant Kumar4d50a432012-08-15 09:06:35 -0700119static unsigned int enable_payload_log = 1;
120module_param(enable_payload_log, uint, S_IRUGO | S_IWUSR);
Hemant Kumar45d211b2012-05-31 17:58:43 -0700121static unsigned int enable_dbg_log = 1;
122module_param(enable_dbg_log, uint, S_IRUGO | S_IWUSR);
123/*by default log ep0 and efs sync ep*/
124static unsigned int ep_addr_rxdbg_mask = 9;
125module_param(ep_addr_rxdbg_mask, uint, S_IRUGO | S_IWUSR);
126static unsigned int ep_addr_txdbg_mask = 9;
127module_param(ep_addr_txdbg_mask, uint, S_IRUGO | S_IWUSR);
128
129/* Maximum debug message length */
Hemant Kumar4d50a432012-08-15 09:06:35 -0700130#define DBG_MSG_LEN 128UL
Hemant Kumar45d211b2012-05-31 17:58:43 -0700131
132/* Maximum number of messages */
133#define DBG_MAX_MSG 256UL
134
135#define TIME_BUF_LEN 20
Hemant Kumar4d50a432012-08-15 09:06:35 -0700136#define HEX_DUMP_LEN 72
Hemant Kumar45d211b2012-05-31 17:58:43 -0700137
138enum event_type {
139 EVENT_UNDEF = -1,
140 URB_SUBMIT,
141 URB_COMPLETE,
142 EVENT_NONE,
143};
144
145#define EVENT_STR_LEN 5
146
Hemant Kumar45d211b2012-05-31 17:58:43 -0700147static enum event_type str_to_event(const char *name)
148{
149 if (!strncasecmp("S", name, EVENT_STR_LEN))
150 return URB_SUBMIT;
151 if (!strncasecmp("C", name, EVENT_STR_LEN))
152 return URB_COMPLETE;
153 if (!strncasecmp("", name, EVENT_STR_LEN))
154 return EVENT_NONE;
155
156 return EVENT_UNDEF;
157}
158
159/*log ep0 activity*/
160static struct {
161 char (buf[DBG_MAX_MSG])[DBG_MSG_LEN]; /* buffer */
162 unsigned idx; /* index */
163 rwlock_t lck; /* lock */
164} dbg_hsic_ctrl = {
165 .idx = 0,
166 .lck = __RW_LOCK_UNLOCKED(lck)
167};
168
169static struct {
170 char (buf[DBG_MAX_MSG])[DBG_MSG_LEN]; /* buffer */
171 unsigned idx; /* index */
172 rwlock_t lck; /* lock */
173} dbg_hsic_data = {
174 .idx = 0,
175 .lck = __RW_LOCK_UNLOCKED(lck)
176};
177
178/**
179 * dbg_inc: increments debug event index
180 * @idx: buffer index
181 */
182static void dbg_inc(unsigned *idx)
183{
184 *idx = (*idx + 1) & (DBG_MAX_MSG-1);
185}
186
187/*get_timestamp - returns time of day in us */
188static char *get_timestamp(char *tbuf)
189{
190 unsigned long long t;
191 unsigned long nanosec_rem;
192
193 t = cpu_clock(smp_processor_id());
194 nanosec_rem = do_div(t, 1000000000)/1000;
195 scnprintf(tbuf, TIME_BUF_LEN, "[%5lu.%06lu] ", (unsigned long)t,
196 nanosec_rem);
197 return tbuf;
198}
199
200static int allow_dbg_log(int ep_addr)
201{
202 int dir, num;
203
204 dir = ep_addr & USB_DIR_IN ? USB_DIR_IN : USB_DIR_OUT;
205 num = ep_addr & ~USB_DIR_IN;
206 num = 1 << num;
207
208 if ((dir == USB_DIR_IN) && (num & ep_addr_rxdbg_mask))
209 return 1;
210 if ((dir == USB_DIR_OUT) && (num & ep_addr_txdbg_mask))
211 return 1;
212
213 return 0;
214}
215
Hemant Kumar4d50a432012-08-15 09:06:35 -0700216static char *get_hex_data(char *dbuf, struct urb *urb, int event, int status)
217{
218 int ep_addr = urb->ep->desc.bEndpointAddress;
219 char *ubuf = urb->transfer_buffer;
220 size_t len = event ? \
221 urb->actual_length : urb->transfer_buffer_length;
222
223 if (status == -EINPROGRESS)
224 status = 0;
225
226 /*Only dump ep in completions and epout submissions*/
227 if (len && !status &&
228 (((ep_addr & USB_DIR_IN) && event) ||
229 (!(ep_addr & USB_DIR_IN) && !event))) {
230 if (len >= 32)
231 len = 32;
232 hex_dump_to_buffer(ubuf, len, 32, 4, dbuf, HEX_DUMP_LEN, 0);
233 } else {
234 dbuf = "";
235 }
236
237 return dbuf;
238}
239
Hemant Kumar45d211b2012-05-31 17:58:43 -0700240static void dbg_log_event(struct urb *urb, char * event, unsigned extra)
241{
242 unsigned long flags;
243 int ep_addr;
244 char tbuf[TIME_BUF_LEN];
Hemant Kumar4d50a432012-08-15 09:06:35 -0700245 char dbuf[HEX_DUMP_LEN];
Hemant Kumar45d211b2012-05-31 17:58:43 -0700246
247 if (!enable_dbg_log)
248 return;
249
250 if (!urb) {
251 write_lock_irqsave(&dbg_hsic_ctrl.lck, flags);
252 scnprintf(dbg_hsic_ctrl.buf[dbg_hsic_ctrl.idx], DBG_MSG_LEN,
Hemant Kumar4d50a432012-08-15 09:06:35 -0700253 "%s: %s : %u", get_timestamp(tbuf), event, extra);
Hemant Kumar45d211b2012-05-31 17:58:43 -0700254 dbg_inc(&dbg_hsic_ctrl.idx);
255 write_unlock_irqrestore(&dbg_hsic_ctrl.lck, flags);
256 return;
257 }
258
259 ep_addr = urb->ep->desc.bEndpointAddress;
260 if (!allow_dbg_log(ep_addr))
261 return;
262
263 if ((ep_addr & 0x0f) == 0x0) {
264 /*submit event*/
265 if (!str_to_event(event)) {
266 write_lock_irqsave(&dbg_hsic_ctrl.lck, flags);
267 scnprintf(dbg_hsic_ctrl.buf[dbg_hsic_ctrl.idx],
268 DBG_MSG_LEN, "%s: [%s : %p]:[%s] "
Hemant Kumar4d50a432012-08-15 09:06:35 -0700269 "%02x %02x %04x %04x %04x %u %d",
Hemant Kumar45d211b2012-05-31 17:58:43 -0700270 get_timestamp(tbuf), event, urb,
271 (ep_addr & USB_DIR_IN) ? "in" : "out",
272 urb->setup_packet[0], urb->setup_packet[1],
273 (urb->setup_packet[3] << 8) |
274 urb->setup_packet[2],
275 (urb->setup_packet[5] << 8) |
276 urb->setup_packet[4],
277 (urb->setup_packet[7] << 8) |
278 urb->setup_packet[6],
Hemant Kumar4d50a432012-08-15 09:06:35 -0700279 urb->transfer_buffer_length, extra);
Hemant Kumar45d211b2012-05-31 17:58:43 -0700280
281 dbg_inc(&dbg_hsic_ctrl.idx);
282 write_unlock_irqrestore(&dbg_hsic_ctrl.lck, flags);
283 } else {
284 write_lock_irqsave(&dbg_hsic_ctrl.lck, flags);
285 scnprintf(dbg_hsic_ctrl.buf[dbg_hsic_ctrl.idx],
Hemant Kumar4d50a432012-08-15 09:06:35 -0700286 DBG_MSG_LEN, "%s: [%s : %p]:[%s] %u %d",
Hemant Kumar45d211b2012-05-31 17:58:43 -0700287 get_timestamp(tbuf), event, urb,
288 (ep_addr & USB_DIR_IN) ? "in" : "out",
289 urb->actual_length, extra);
290
291 dbg_inc(&dbg_hsic_ctrl.idx);
292 write_unlock_irqrestore(&dbg_hsic_ctrl.lck, flags);
293 }
294 } else {
295 write_lock_irqsave(&dbg_hsic_data.lck, flags);
296 scnprintf(dbg_hsic_data.buf[dbg_hsic_data.idx], DBG_MSG_LEN,
Hemant Kumar4d50a432012-08-15 09:06:35 -0700297 "%s: [%s : %p]:ep%d[%s] %u %d %s",
Hemant Kumar45d211b2012-05-31 17:58:43 -0700298 get_timestamp(tbuf), event, urb, ep_addr & 0x0f,
299 (ep_addr & USB_DIR_IN) ? "in" : "out",
300 str_to_event(event) ? urb->actual_length :
Hemant Kumar4d50a432012-08-15 09:06:35 -0700301 urb->transfer_buffer_length, extra,
302 enable_payload_log ? get_hex_data(dbuf, urb,
303 str_to_event(event), extra) : "");
Hemant Kumar45d211b2012-05-31 17:58:43 -0700304
305 dbg_inc(&dbg_hsic_data.idx);
306 write_unlock_irqrestore(&dbg_hsic_data.lck, flags);
307 }
308}
309
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530310static inline struct msm_hsic_hcd *hcd_to_hsic(struct usb_hcd *hcd)
311{
312 return (struct msm_hsic_hcd *) (hcd->hcd_priv);
313}
314
315static inline struct usb_hcd *hsic_to_hcd(struct msm_hsic_hcd *mehci)
316{
317 return container_of((void *) mehci, struct usb_hcd, hcd_priv);
318}
319
Hemant Kumar105d07f2012-07-02 15:33:07 -0700320static void dump_hsic_regs(struct usb_hcd *hcd)
321{
322 int i;
323 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
324
325 if (atomic_read(&mehci->in_lpm))
326 return;
327
328 for (i = USB_REG_START_OFFSET; i <= USB_REG_END_OFFSET; i += 0x10)
329 pr_info("%p: %08x\t%08x\t%08x\t%08x\n", hcd->regs + i,
330 readl_relaxed(hcd->regs + i),
331 readl_relaxed(hcd->regs + i + 4),
332 readl_relaxed(hcd->regs + i + 8),
333 readl_relaxed(hcd->regs + i + 0xc));
334}
335
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530336#define ULPI_IO_TIMEOUT_USEC (10 * 1000)
337
Amit Blayd6ea6102012-06-07 16:26:24 +0300338#define USB_PHY_VDD_DIG_VOL_NONE 0 /*uV */
Hemant Kumar3b743cd2012-10-17 13:48:10 -0700339#define USB_PHY_VDD_DIG_VOL_MIN 945000 /* uV */
Vamsi Krishna45d88fa2011-11-02 13:28:42 -0700340#define USB_PHY_VDD_DIG_VOL_MAX 1320000 /* uV */
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530341
Lena Salman8c8ba382012-02-14 15:59:31 +0200342#define HSIC_DBG1_REG 0x38
343
Amit Blayd6ea6102012-06-07 16:26:24 +0300344static const int vdd_val[VDD_TYPE_MAX][VDD_VAL_MAX] = {
345 { /* VDD_CX CORNER Voting */
346 [VDD_NONE] = RPM_VREG_CORNER_NONE,
347 [VDD_MIN] = RPM_VREG_CORNER_NOMINAL,
348 [VDD_MAX] = RPM_VREG_CORNER_HIGH,
349 },
350 { /* VDD_CX Voltage Voting */
351 [VDD_NONE] = USB_PHY_VDD_DIG_VOL_NONE,
352 [VDD_MIN] = USB_PHY_VDD_DIG_VOL_MIN,
353 [VDD_MAX] = USB_PHY_VDD_DIG_VOL_MAX,
354 },
355};
356
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530357static int msm_hsic_init_vddcx(struct msm_hsic_hcd *mehci, int init)
358{
359 int ret = 0;
Amit Blayd6ea6102012-06-07 16:26:24 +0300360 int none_vol, min_vol, max_vol;
361
362 if (!mehci->hsic_vddcx) {
363 mehci->vdd_type = VDDCX_CORNER;
364 mehci->hsic_vddcx = devm_regulator_get(mehci->dev,
365 "hsic_vdd_dig");
366 if (IS_ERR(mehci->hsic_vddcx)) {
367 mehci->hsic_vddcx = devm_regulator_get(mehci->dev,
368 "HSIC_VDDCX");
369 if (IS_ERR(mehci->hsic_vddcx)) {
370 dev_err(mehci->dev, "unable to get hsic vddcx\n");
371 return PTR_ERR(mehci->hsic_vddcx);
372 }
373 mehci->vdd_type = VDDCX;
374 }
375 }
376
377 none_vol = vdd_val[mehci->vdd_type][VDD_NONE];
378 min_vol = vdd_val[mehci->vdd_type][VDD_MIN];
379 max_vol = vdd_val[mehci->vdd_type][VDD_MAX];
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530380
381 if (!init)
382 goto disable_reg;
383
Amit Blayd6ea6102012-06-07 16:26:24 +0300384 ret = regulator_set_voltage(mehci->hsic_vddcx, min_vol, max_vol);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530385 if (ret) {
386 dev_err(mehci->dev, "unable to set the voltage"
387 "for hsic vddcx\n");
Mayank Rana189ac052012-03-24 04:35:02 +0530388 return ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530389 }
390
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530391 ret = regulator_enable(mehci->hsic_vddcx);
392 if (ret) {
393 dev_err(mehci->dev, "unable to enable hsic vddcx\n");
394 goto reg_enable_err;
395 }
396
397 return 0;
398
399disable_reg:
400 regulator_disable(mehci->hsic_vddcx);
401reg_enable_err:
Amit Blayd6ea6102012-06-07 16:26:24 +0300402 regulator_set_voltage(mehci->hsic_vddcx, none_vol, max_vol);
403
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530404 return ret;
405
406}
407
Ofir Cohen4cc55372012-11-20 11:17:24 +0200408/* Global Distributed Switch Controller (GDSC) init */
409static int msm_hsic_init_gdsc(struct msm_hsic_hcd *mehci, int init)
410{
411 int ret = 0;
412
413 if (IS_ERR(mehci->hsic_gdsc))
414 return 0;
415
416 if (!mehci->hsic_gdsc) {
417 mehci->hsic_gdsc = devm_regulator_get(mehci->dev,
418 "HSIC_GDSC");
419 if (IS_ERR(mehci->hsic_gdsc))
420 return 0;
421 }
422
423 if (init) {
424 ret = regulator_enable(mehci->hsic_gdsc);
425 if (ret) {
426 dev_err(mehci->dev, "unable to enable hsic gdsc\n");
427 return ret;
428 }
429 } else {
430 regulator_disable(mehci->hsic_gdsc);
431 }
432
433 return 0;
434
435}
436
Pavankumar Kondeti23720492013-01-29 14:41:58 +0530437static int __maybe_unused ulpi_read(struct msm_hsic_hcd *mehci, u32 reg)
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -0700438{
439 struct usb_hcd *hcd = hsic_to_hcd(mehci);
Hemant Kumar8fdc3982012-08-21 12:22:27 -0700440 int cnt = 0;
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -0700441
442 /* initiate read operation */
443 writel_relaxed(ULPI_RUN | ULPI_READ | ULPI_ADDR(reg),
444 USB_ULPI_VIEWPORT);
445
446 /* wait for completion */
Hemant Kumar8fdc3982012-08-21 12:22:27 -0700447 while (cnt < ULPI_IO_TIMEOUT_USEC) {
448 if (!(readl_relaxed(USB_ULPI_VIEWPORT) & ULPI_RUN))
449 break;
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -0700450 udelay(1);
Hemant Kumar8fdc3982012-08-21 12:22:27 -0700451 cnt++;
452 }
453
454 if (cnt >= ULPI_IO_TIMEOUT_USEC) {
455 dev_err(mehci->dev, "ulpi_read: timeout ULPI_VIEWPORT: %08x\n",
456 readl_relaxed(USB_ULPI_VIEWPORT));
457 dev_err(mehci->dev, "PORTSC: %08x USBCMD: %08x FRINDEX: %08x\n",
458 readl_relaxed(USB_PORTSC),
459 readl_relaxed(USB_USBCMD),
460 readl_relaxed(USB_FRINDEX));
461
462 /*frame counter increments afte 125us*/
463 udelay(130);
464 dev_err(mehci->dev, "ulpi_read: FRINDEX: %08x\n",
465 readl_relaxed(USB_FRINDEX));
466 return -ETIMEDOUT;
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -0700467 }
468
469 return ULPI_DATA_READ(readl_relaxed(USB_ULPI_VIEWPORT));
470}
471
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530472static int ulpi_write(struct msm_hsic_hcd *mehci, u32 val, u32 reg)
473{
474 struct usb_hcd *hcd = hsic_to_hcd(mehci);
475 int cnt = 0;
476
477 /* initiate write operation */
478 writel_relaxed(ULPI_RUN | ULPI_WRITE |
479 ULPI_ADDR(reg) | ULPI_DATA(val),
480 USB_ULPI_VIEWPORT);
481
482 /* wait for completion */
483 while (cnt < ULPI_IO_TIMEOUT_USEC) {
484 if (!(readl_relaxed(USB_ULPI_VIEWPORT) & ULPI_RUN))
485 break;
486 udelay(1);
487 cnt++;
488 }
489
490 if (cnt >= ULPI_IO_TIMEOUT_USEC) {
Hemant Kumar8fdc3982012-08-21 12:22:27 -0700491 dev_err(mehci->dev, "ulpi_write: timeout ULPI_VIEWPORT: %08x\n",
492 readl_relaxed(USB_ULPI_VIEWPORT));
493 dev_err(mehci->dev, "PORTSC: %08x USBCMD: %08x FRINDEX: %08x\n",
494 readl_relaxed(USB_PORTSC),
495 readl_relaxed(USB_USBCMD),
496 readl_relaxed(USB_FRINDEX));
497
498 /*frame counter increments afte 125us*/
499 udelay(130);
500 dev_err(mehci->dev, "ulpi_write: FRINDEX: %08x\n",
501 readl_relaxed(USB_FRINDEX));
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530502 return -ETIMEDOUT;
503 }
504
505 return 0;
506}
507
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530508static int msm_hsic_config_gpios(struct msm_hsic_hcd *mehci, int gpio_en)
509{
510 int rc = 0;
511 struct msm_hsic_host_platform_data *pdata;
Vamsi Krishna34f01582011-12-14 19:54:42 -0800512 static int gpio_status;
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530513
514 pdata = mehci->dev->platform_data;
Vamsi Krishna34f01582011-12-14 19:54:42 -0800515
Lena Salman8c8ba382012-02-14 15:59:31 +0200516 if (!pdata || !pdata->strobe || !pdata->data)
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530517 return rc;
518
Vamsi Krishna34f01582011-12-14 19:54:42 -0800519 if (gpio_status == gpio_en)
520 return 0;
521
522 gpio_status = gpio_en;
523
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530524 if (!gpio_en)
525 goto free_gpio;
526
527 rc = gpio_request(pdata->strobe, "HSIC_STROBE_GPIO");
528 if (rc < 0) {
529 dev_err(mehci->dev, "gpio request failed for HSIC STROBE\n");
530 return rc;
531 }
532
533 rc = gpio_request(pdata->data, "HSIC_DATA_GPIO");
534 if (rc < 0) {
535 dev_err(mehci->dev, "gpio request failed for HSIC DATA\n");
536 goto free_strobe;
Hemant Kumar6fd65032012-05-23 13:02:24 -0700537 }
538
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530539 return 0;
540
541free_gpio:
542 gpio_free(pdata->data);
543free_strobe:
544 gpio_free(pdata->strobe);
545
546 return rc;
547}
548
Vamsi Krishna64b48612012-06-14 16:08:11 -0700549static void msm_hsic_clk_reset(struct msm_hsic_hcd *mehci)
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530550{
551 int ret;
552
Ofir Cohen07944aa2012-12-23 13:41:57 +0200553 /* alt_core_clk exists in targets that do not use asynchronous reset */
554 if (!IS_ERR(mehci->alt_core_clk)) {
555 ret = clk_reset(mehci->core_clk, CLK_RESET_ASSERT);
556 if (ret) {
557 dev_err(mehci->dev, "hsic clk assert failed:%d\n", ret);
558 return;
559 }
560
561 /* Since a hw bug, turn off the clock before complete reset */
562 clk_disable(mehci->core_clk);
563
564 ret = clk_reset(mehci->core_clk, CLK_RESET_DEASSERT);
565 if (ret)
566 dev_err(mehci->dev, "hsic clk deassert failed:%d\n",
567 ret);
568
569 usleep_range(10000, 12000);
570
571 clk_enable(mehci->core_clk);
572 } else {
573 /* Using asynchronous block reset to the hardware */
574 clk_disable_unprepare(mehci->core_clk);
575 clk_disable_unprepare(mehci->phy_clk);
576 clk_disable_unprepare(mehci->cal_clk);
577 clk_disable_unprepare(mehci->ahb_clk);
578
579 ret = clk_reset(mehci->core_clk, CLK_RESET_ASSERT);
580 if (ret) {
581 dev_err(mehci->dev, "hsic clk assert failed:%d\n", ret);
582 return;
583 }
584 usleep_range(10000, 12000);
585
586 ret = clk_reset(mehci->core_clk, CLK_RESET_DEASSERT);
587 if (ret)
588 dev_err(mehci->dev, "hsic clk deassert failed:%d\n",
589 ret);
590 /*
591 * Required delay between the deassertion and
592 * clock enablement.
593 */
594 ndelay(200);
595 clk_prepare_enable(mehci->core_clk);
596 clk_prepare_enable(mehci->phy_clk);
597 clk_prepare_enable(mehci->cal_clk);
598 clk_prepare_enable(mehci->ahb_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530599 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530600}
601
Vamsi Krishna64b48612012-06-14 16:08:11 -0700602#define HSIC_STROBE_GPIO_PAD_CTL (MSM_TLMM_BASE+0x20C0)
603#define HSIC_DATA_GPIO_PAD_CTL (MSM_TLMM_BASE+0x20C4)
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530604#define HSIC_CAL_PAD_CTL (MSM_TLMM_BASE+0x20C8)
605#define HSIC_LV_MODE 0x04
606#define HSIC_PAD_CALIBRATION 0xA8
607#define HSIC_GPIO_PAD_VAL 0x0A0AAA10
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530608#define LINK_RESET_TIMEOUT_USEC (250 * 1000)
Ofir Cohen5dddb152012-11-14 11:18:50 +0200609
610static void msm_hsic_phy_reset(struct msm_hsic_hcd *mehci)
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530611{
612 struct usb_hcd *hcd = hsic_to_hcd(mehci);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530613
Vamsi Krishna64b48612012-06-14 16:08:11 -0700614 msm_hsic_clk_reset(mehci);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530615
Vamsi Krishna64b48612012-06-14 16:08:11 -0700616 /* select ulpi phy */
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530617 writel_relaxed(0x80000000, USB_PORTSC);
Vamsi Krishna64b48612012-06-14 16:08:11 -0700618 mb();
Ofir Cohen5dddb152012-11-14 11:18:50 +0200619}
620
621static int msm_hsic_start(struct msm_hsic_hcd *mehci)
622{
623 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
624 int ret;
Manu Gautam863b74132012-11-21 14:30:04 +0530625 void __iomem *reg;
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530626
Vamsi Krishnabef91832013-03-11 14:10:20 -0700627 if (pdata && pdata->resume_gpio) {
628 ret = gpio_request(pdata->resume_gpio, "HSIC_RESUME_GPIO");
629 if (ret < 0) {
630 dev_err(mehci->dev,
631 "gpio req failed for hsic resume:%d\n", ret);
632 pdata->resume_gpio = 0;
633 }
634 }
635
Lena Salman8c8ba382012-02-14 15:59:31 +0200636 /* HSIC init sequence when HSIC signals (Strobe/Data) are
637 routed via GPIOs */
638 if (pdata && pdata->strobe && pdata->data) {
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530639
Manu Gautam863b74132012-11-21 14:30:04 +0530640 if (!pdata->ignore_cal_pad_config) {
641 /* Enable LV_MODE in HSIC_CAL_PAD_CTL register */
642 writel_relaxed(HSIC_LV_MODE, HSIC_CAL_PAD_CTL);
643 mb();
644 }
Vamsi Krishna64b48612012-06-14 16:08:11 -0700645
Lena Salman8c8ba382012-02-14 15:59:31 +0200646 /*set periodic calibration interval to ~2.048sec in
647 HSIC_IO_CAL_REG */
648 ulpi_write(mehci, 0xFF, 0x33);
649
650 /* Enable periodic IO calibration in HSIC_CFG register */
651 ulpi_write(mehci, HSIC_PAD_CALIBRATION, 0x30);
652
Vamsi Krishna64b48612012-06-14 16:08:11 -0700653 /* Configure GPIO pins for HSIC functionality mode */
Lena Salman8c8ba382012-02-14 15:59:31 +0200654 ret = msm_hsic_config_gpios(mehci, 1);
655 if (ret) {
656 dev_err(mehci->dev, " gpio configuarion failed\n");
Vamsi Krishnabef91832013-03-11 14:10:20 -0700657 goto free_resume_gpio;
Lena Salman8c8ba382012-02-14 15:59:31 +0200658 }
Manu Gautam863b74132012-11-21 14:30:04 +0530659 if (pdata->strobe_pad_offset) {
660 /* Set CORE_CTL_EN in STROBE GPIO PAD_CTL register */
661 reg = MSM_TLMM_BASE + pdata->strobe_pad_offset;
662 writel_relaxed(readl_relaxed(reg) | 0x2000000, reg);
663 } else {
664 /* Set LV_MODE=0x1 and DCC=0x2 in STROBE GPIO PAD_CTL */
665 reg = HSIC_STROBE_GPIO_PAD_CTL;
666 writel_relaxed(HSIC_GPIO_PAD_VAL, reg);
667 }
668
669 if (pdata->data_pad_offset) {
670 /* Set CORE_CTL_EN in HSIC_DATA GPIO PAD_CTL register */
671 reg = MSM_TLMM_BASE + pdata->data_pad_offset;
672 writel_relaxed(readl_relaxed(reg) | 0x2000000, reg);
673 } else {
674 /* Set LV_MODE=0x1 and DCC=0x2 in STROBE GPIO PAD_CTL */
675 reg = HSIC_DATA_GPIO_PAD_CTL;
676 writel_relaxed(HSIC_GPIO_PAD_VAL, reg);
677 }
Vamsi Krishna64b48612012-06-14 16:08:11 -0700678
679 mb();
680
Lena Salman8c8ba382012-02-14 15:59:31 +0200681 /* Enable HSIC mode in HSIC_CFG register */
682 ulpi_write(mehci, 0x01, 0x31);
683 } else {
684 /* HSIC init sequence when HSIC signals (Strobe/Data) are routed
685 via dedicated I/O */
686
687 /* programmable length of connect signaling (33.2ns) */
688 ret = ulpi_write(mehci, 3, HSIC_DBG1_REG);
689 if (ret) {
690 pr_err("%s: Unable to program length of connect "
691 "signaling\n", __func__);
692 }
693
694 /*set periodic calibration interval to ~2.048sec in
695 HSIC_IO_CAL_REG */
696 ulpi_write(mehci, 0xFF, 0x33);
697
698 /* Enable HSIC mode in HSIC_CFG register */
699 ulpi_write(mehci, 0xA9, 0x30);
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530700 }
701
Hemant Kumar6fd65032012-05-23 13:02:24 -0700702 /*disable auto resume*/
703 ulpi_write(mehci, ULPI_IFC_CTRL_AUTORESUME, ULPI_CLR(ULPI_IFC_CTRL));
704
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530705 return 0;
Vamsi Krishnabef91832013-03-11 14:10:20 -0700706
707free_resume_gpio:
708 if (pdata && pdata->resume_gpio)
709 gpio_free(pdata->resume_gpio);
710
711 return ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530712}
713
714#define PHY_SUSPEND_TIMEOUT_USEC (500 * 1000)
715#define PHY_RESUME_TIMEOUT_USEC (100 * 1000)
716
717#ifdef CONFIG_PM_SLEEP
Ofir Cohen5dddb152012-11-14 11:18:50 +0200718static int msm_hsic_reset(struct msm_hsic_hcd *mehci)
719{
720 /* reset HSIC phy */
721 msm_hsic_phy_reset(mehci);
722
723 /* HSIC init procedure (caliberation) */
724 return msm_hsic_start(mehci);
725}
726
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530727static int msm_hsic_suspend(struct msm_hsic_hcd *mehci)
728{
729 struct usb_hcd *hcd = hsic_to_hcd(mehci);
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530730 int cnt = 0, ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530731 u32 val;
Amit Blayd6ea6102012-06-07 16:26:24 +0300732 int none_vol, max_vol;
Hemant Kumarbc8bdf62012-10-17 12:29:51 -0700733 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530734
735 if (atomic_read(&mehci->in_lpm)) {
736 dev_dbg(mehci->dev, "%s called in lpm\n", __func__);
737 return 0;
738 }
739
740 disable_irq(hcd->irq);
Jack Phambe05fbb2012-05-16 10:56:26 -0700741
Sudhir Sharma1673e302012-08-27 17:37:24 -0700742 /* make sure we don't race against a remote wakeup */
743 if (test_bit(HCD_FLAG_WAKEUP_PENDING, &hcd->flags) ||
Jack Phambe05fbb2012-05-16 10:56:26 -0700744 readl_relaxed(USB_PORTSC) & PORT_RESUME) {
Sudhir Sharma1673e302012-08-27 17:37:24 -0700745 dev_dbg(mehci->dev, "wakeup pending, aborting suspend\n");
Jack Phambe05fbb2012-05-16 10:56:26 -0700746 enable_irq(hcd->irq);
747 return -EBUSY;
748 }
749
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530750 /*
751 * PHY may take some time or even fail to enter into low power
752 * mode (LPM). Hence poll for 500 msec and reset the PHY and link
753 * in failure case.
754 */
Hemant Kumar3dbc5b32012-05-09 15:36:11 -0700755 val = readl_relaxed(USB_PORTSC);
756 val &= ~PORT_RWC_BITS;
757 val |= PORTSC_PHCD;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530758 writel_relaxed(val, USB_PORTSC);
759 while (cnt < PHY_SUSPEND_TIMEOUT_USEC) {
760 if (readl_relaxed(USB_PORTSC) & PORTSC_PHCD)
761 break;
762 udelay(1);
763 cnt++;
764 }
765
766 if (cnt >= PHY_SUSPEND_TIMEOUT_USEC) {
767 dev_err(mehci->dev, "Unable to suspend PHY\n");
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530768 msm_hsic_config_gpios(mehci, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530769 msm_hsic_reset(mehci);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530770 }
771
772 /*
773 * PHY has capability to generate interrupt asynchronously in low
774 * power mode (LPM). This interrupt is level triggered. So USB IRQ
775 * line must be disabled till async interrupt enable bit is cleared
776 * in USBCMD register. Assert STP (ULPI interface STOP signal) to
Pavankumar Kondeti0c9f29c2013-01-28 21:37:59 +0530777 * block data communication from PHY. Enable asynchronous interrupt
778 * only when wakeup gpio IRQ is not present.
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530779 */
Pavankumar Kondeti0c9f29c2013-01-28 21:37:59 +0530780 if (mehci->wakeup_irq)
781 writel_relaxed(readl_relaxed(USB_USBCMD) |
782 ULPI_STP_CTRL, USB_USBCMD);
783 else
784 writel_relaxed(readl_relaxed(USB_USBCMD) | ASYNC_INTR_CTRL |
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530785 ULPI_STP_CTRL, USB_USBCMD);
786
787 /*
788 * Ensure that hardware is put in low power mode before
789 * clocks are turned OFF and VDD is allowed to minimize.
790 */
791 mb();
792
Manu Gautam28b1bac2012-01-30 16:43:06 +0530793 clk_disable_unprepare(mehci->core_clk);
794 clk_disable_unprepare(mehci->phy_clk);
795 clk_disable_unprepare(mehci->cal_clk);
796 clk_disable_unprepare(mehci->ahb_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530797
Amit Blayd6ea6102012-06-07 16:26:24 +0300798 none_vol = vdd_val[mehci->vdd_type][VDD_NONE];
799 max_vol = vdd_val[mehci->vdd_type][VDD_MAX];
800
801 ret = regulator_set_voltage(mehci->hsic_vddcx, none_vol, max_vol);
Vamsi Krishna45d88fa2011-11-02 13:28:42 -0700802 if (ret < 0)
Amit Blayd6ea6102012-06-07 16:26:24 +0300803 dev_err(mehci->dev, "unable to set vddcx voltage for VDD MIN\n");
Vamsi Krishna45d88fa2011-11-02 13:28:42 -0700804
Hemant Kumare6275972012-02-29 20:06:21 -0800805 if (mehci->bus_perf_client && debug_bus_voting_enabled) {
Hemant Kumar2309eaa2012-08-14 16:46:42 -0700806 mehci->bus_vote = false;
807 queue_work(ehci_wq, &mehci->bus_vote_w);
Hemant Kumare6275972012-02-29 20:06:21 -0800808 }
809
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530810 atomic_set(&mehci->in_lpm, 1);
811 enable_irq(hcd->irq);
Hemant Kumar6fd65032012-05-23 13:02:24 -0700812
Manu Gautam3fb60ca2013-02-13 18:33:33 +0530813 if (mehci->wakeup_irq) {
814 mehci->wakeup_irq_enabled = 1;
815 enable_irq_wake(mehci->wakeup_irq);
816 enable_irq(mehci->wakeup_irq);
817 }
Hemant Kumar6fd65032012-05-23 13:02:24 -0700818
Hemant Kumarbc8bdf62012-10-17 12:29:51 -0700819 if (pdata && pdata->standalone_latency)
820 pm_qos_update_request(&mehci->pm_qos_req_dma,
821 PM_QOS_DEFAULT_VALUE);
822
Sudhir Sharma1673e302012-08-27 17:37:24 -0700823 wake_unlock(&mehci->wlock);
824
Pavankumar Kondetiafbaf8e2013-02-22 11:58:44 +0530825 dev_dbg(mehci->dev, "HSIC-USB in low power mode\n");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530826
827 return 0;
828}
829
830static int msm_hsic_resume(struct msm_hsic_hcd *mehci)
831{
832 struct usb_hcd *hcd = hsic_to_hcd(mehci);
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530833 int cnt = 0, ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530834 unsigned temp;
Amit Blayd6ea6102012-06-07 16:26:24 +0300835 int min_vol, max_vol;
Hemant Kumard343c012012-09-06 19:57:14 -0700836 unsigned long flags;
Hemant Kumarbc8bdf62012-10-17 12:29:51 -0700837 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530838
839 if (!atomic_read(&mehci->in_lpm)) {
840 dev_dbg(mehci->dev, "%s called in !in_lpm\n", __func__);
841 return 0;
842 }
843
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +0530844 /* Handles race with Async interrupt */
845 disable_irq(hcd->irq);
846
Hemant Kumarbc8bdf62012-10-17 12:29:51 -0700847 if (pdata && pdata->standalone_latency)
848 pm_qos_update_request(&mehci->pm_qos_req_dma,
849 pdata->standalone_latency + 1);
850
Manu Gautam3fb60ca2013-02-13 18:33:33 +0530851 if (mehci->wakeup_irq) {
852 spin_lock_irqsave(&mehci->wakeup_lock, flags);
853 if (mehci->wakeup_irq_enabled) {
854 disable_irq_wake(mehci->wakeup_irq);
855 disable_irq_nosync(mehci->wakeup_irq);
856 mehci->wakeup_irq_enabled = 0;
857 }
858 spin_unlock_irqrestore(&mehci->wakeup_lock, flags);
Hemant Kumar6fd65032012-05-23 13:02:24 -0700859 }
860
Sudhir Sharma1673e302012-08-27 17:37:24 -0700861 wake_lock(&mehci->wlock);
862
Hemant Kumare6275972012-02-29 20:06:21 -0800863 if (mehci->bus_perf_client && debug_bus_voting_enabled) {
Hemant Kumar2309eaa2012-08-14 16:46:42 -0700864 mehci->bus_vote = true;
865 queue_work(ehci_wq, &mehci->bus_vote_w);
Hemant Kumare6275972012-02-29 20:06:21 -0800866 }
867
Amit Blayd6ea6102012-06-07 16:26:24 +0300868 min_vol = vdd_val[mehci->vdd_type][VDD_MIN];
869 max_vol = vdd_val[mehci->vdd_type][VDD_MAX];
870
871 ret = regulator_set_voltage(mehci->hsic_vddcx, min_vol, max_vol);
Vamsi Krishna45d88fa2011-11-02 13:28:42 -0700872 if (ret < 0)
Amit Blayd6ea6102012-06-07 16:26:24 +0300873 dev_err(mehci->dev, "unable to set nominal vddcx voltage (no VDD MIN)\n");
Vamsi Krishna45d88fa2011-11-02 13:28:42 -0700874
Manu Gautam28b1bac2012-01-30 16:43:06 +0530875 clk_prepare_enable(mehci->core_clk);
876 clk_prepare_enable(mehci->phy_clk);
877 clk_prepare_enable(mehci->cal_clk);
878 clk_prepare_enable(mehci->ahb_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530879
880 temp = readl_relaxed(USB_USBCMD);
881 temp &= ~ASYNC_INTR_CTRL;
882 temp &= ~ULPI_STP_CTRL;
883 writel_relaxed(temp, USB_USBCMD);
884
885 if (!(readl_relaxed(USB_PORTSC) & PORTSC_PHCD))
886 goto skip_phy_resume;
887
Hemant Kumar3dbc5b32012-05-09 15:36:11 -0700888 temp = readl_relaxed(USB_PORTSC);
889 temp &= ~(PORT_RWC_BITS | PORTSC_PHCD);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530890 writel_relaxed(temp, USB_PORTSC);
891 while (cnt < PHY_RESUME_TIMEOUT_USEC) {
892 if (!(readl_relaxed(USB_PORTSC) & PORTSC_PHCD) &&
893 (readl_relaxed(USB_ULPI_VIEWPORT) & ULPI_SYNC_STATE))
894 break;
895 udelay(1);
896 cnt++;
897 }
898
899 if (cnt >= PHY_RESUME_TIMEOUT_USEC) {
900 /*
901 * This is a fatal error. Reset the link and
902 * PHY to make hsic working.
903 */
904 dev_err(mehci->dev, "Unable to resume USB. Reset the hsic\n");
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530905 msm_hsic_config_gpios(mehci, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530906 msm_hsic_reset(mehci);
907 }
908
909skip_phy_resume:
910
Hemant Kumar6fd65032012-05-23 13:02:24 -0700911 usb_hcd_resume_root_hub(hcd);
912
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530913 atomic_set(&mehci->in_lpm, 0);
914
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +0530915 if (atomic_read(&mehci->async_int)) {
916 atomic_set(&mehci->async_int, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530917 pm_runtime_put_noidle(mehci->dev);
Jack Phamdd5ad792012-07-26 10:31:03 -0700918 enable_irq(hcd->irq);
Hemant Kumar6fd65032012-05-23 13:02:24 -0700919 }
920
Sudhir Sharma1673e302012-08-27 17:37:24 -0700921 if (atomic_read(&mehci->pm_usage_cnt)) {
922 atomic_set(&mehci->pm_usage_cnt, 0);
923 pm_runtime_put_noidle(mehci->dev);
924 }
Jack Phamdd5ad792012-07-26 10:31:03 -0700925
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +0530926 enable_irq(hcd->irq);
Pavankumar Kondetiafbaf8e2013-02-22 11:58:44 +0530927 dev_dbg(mehci->dev, "HSIC-USB exited from low power mode\n");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530928
929 return 0;
930}
931#endif
932
Hemant Kumar2309eaa2012-08-14 16:46:42 -0700933static void ehci_hsic_bus_vote_w(struct work_struct *w)
934{
935 struct msm_hsic_hcd *mehci =
936 container_of(w, struct msm_hsic_hcd, bus_vote_w);
937 int ret;
938
939 ret = msm_bus_scale_client_update_request(mehci->bus_perf_client,
940 mehci->bus_vote);
941 if (ret)
942 dev_err(mehci->dev, "%s: Failed to vote for bus bandwidth %d\n",
943 __func__, ret);
944}
945
Pavankumar Kondeti23720492013-01-29 14:41:58 +0530946static int msm_hsic_reset_done(struct usb_hcd *hcd)
947{
948 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
949 u32 __iomem *status_reg = &ehci->regs->port_status[0];
950 int ret;
951
952 ehci_writel(ehci, ehci_readl(ehci, status_reg) & ~(PORT_RWC_BITS |
953 PORT_RESET), status_reg);
954
955 ret = handshake(ehci, status_reg, PORT_RESET, 0, 1 * 1000);
956
957 if (ret)
958 pr_err("reset handshake failed in %s\n", __func__);
959 else
960 ehci_writel(ehci, ehci_readl(ehci, &ehci->regs->command) |
961 CMD_RUN, &ehci->regs->command);
962
963 return ret;
964}
965
Vamsi Krishnafce1bd22012-08-17 17:12:26 -0700966#define STS_GPTIMER0_INTERRUPT BIT(24)
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530967static irqreturn_t msm_hsic_irq(struct usb_hcd *hcd)
968{
Vamsi Krishnafce1bd22012-08-17 17:12:26 -0700969 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530970 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
Vamsi Krishnafce1bd22012-08-17 17:12:26 -0700971 u32 status;
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +0530972 int ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530973
974 if (atomic_read(&mehci->in_lpm)) {
Hemant Kumar3dbc5b32012-05-09 15:36:11 -0700975 dev_dbg(mehci->dev, "phy async intr\n");
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +0530976 dbg_log_event(NULL, "Async IRQ", 0);
977 ret = pm_runtime_get(mehci->dev);
978 if ((ret == 1) || (ret == -EINPROGRESS)) {
979 pm_runtime_put_noidle(mehci->dev);
980 } else {
981 disable_irq_nosync(hcd->irq);
982 atomic_set(&mehci->async_int, 1);
983 }
984
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530985 return IRQ_HANDLED;
986 }
987
Vamsi Krishnafce1bd22012-08-17 17:12:26 -0700988 status = ehci_readl(ehci, &ehci->regs->status);
989
990 if (status & STS_GPTIMER0_INTERRUPT) {
991 int timeleft;
992
Pavankumar Kondeti23720492013-01-29 14:41:58 +0530993 dbg_log_event(NULL, "FPR: gpt0_isr", mehci->bus_reset);
Vamsi Krishnafce1bd22012-08-17 17:12:26 -0700994
995 timeleft = GPT_CNT(ehci_readl(ehci,
996 &mehci->timer->gptimer1_ctrl));
997 if (timeleft) {
Pavankumar Kondeti23720492013-01-29 14:41:58 +0530998 if (mehci->bus_reset) {
999 ret = msm_hsic_reset_done(hcd);
1000 if (ret) {
1001 mehci->reset_again = 1;
1002 dbg_log_event(NULL, "RESET: fail", 0);
1003 }
1004 } else {
1005 ehci_writel(ehci, ehci_readl(ehci,
1006 &ehci->regs->command) | CMD_RUN,
1007 &ehci->regs->command);
1008 }
1009 } else {
1010 if (mehci->bus_reset)
1011 mehci->reset_again = 1;
1012 else
1013 mehci->resume_again = 1;
1014 }
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001015
1016 dbg_log_event(NULL, "FPR: timeleft", timeleft);
1017
1018 complete(&mehci->gpt0_completion);
1019 ehci_writel(ehci, STS_GPTIMER0_INTERRUPT, &ehci->regs->status);
1020 }
1021
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301022 return ehci_irq(hcd);
1023}
1024
1025static int ehci_hsic_reset(struct usb_hcd *hcd)
1026{
1027 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001028 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301029 int retval;
1030
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001031 mehci->timer = USB_HS_GPTIMER_BASE;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301032 ehci->caps = USB_CAPLENGTH;
1033 ehci->regs = USB_CAPLENGTH +
1034 HC_LENGTH(ehci, ehci_readl(ehci, &ehci->caps->hc_capbase));
1035 dbg_hcs_params(ehci, "reset");
1036 dbg_hcc_params(ehci, "reset");
1037
1038 /* cache the data to minimize the chip reads*/
1039 ehci->hcs_params = ehci_readl(ehci, &ehci->caps->hcs_params);
1040
1041 hcd->has_tt = 1;
1042 ehci->sbrn = HCD_USB2;
1043
1044 retval = ehci_halt(ehci);
1045 if (retval)
1046 return retval;
1047
1048 /* data structure init */
1049 retval = ehci_init(hcd);
1050 if (retval)
1051 return retval;
1052
1053 retval = ehci_reset(ehci);
1054 if (retval)
1055 return retval;
1056
1057 /* bursts of unspecified length. */
1058 writel_relaxed(0, USB_AHBBURST);
1059 /* Use the AHB transactor */
Vijayavardhan Vennapusa5f32d7a2012-03-14 16:30:26 +05301060 writel_relaxed(0x08, USB_AHBMODE);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301061 /* Disable streaming mode and select host mode */
1062 writel_relaxed(0x13, USB_USBMODE);
1063
1064 ehci_port_power(ehci, 1);
1065 return 0;
1066}
1067
Jack Phamc0d41732012-09-28 17:45:38 -07001068#ifdef CONFIG_PM
1069
Pavankumar Kondeti23720492013-01-29 14:41:58 +05301070#define RESET_RETRY_LIMIT 3
1071#define RESET_SIGNAL_TIME_SOF_USEC (50 * 1000)
1072#define RESET_SIGNAL_TIME_USEC (20 * 1000)
1073static void ehci_hsic_reset_sof_bug_handler(struct usb_hcd *hcd, u32 val)
1074{
1075 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
1076 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
1077 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
1078 u32 __iomem *status_reg = &ehci->regs->port_status[0];
Pavankumar Kondetifc7d1d82013-03-05 09:43:22 +05301079 u32 cmd;
Pavankumar Kondeti23720492013-01-29 14:41:58 +05301080 unsigned long flags;
1081 int retries = 0, ret, cnt = RESET_SIGNAL_TIME_USEC;
1082
1083 if (pdata && pdata->swfi_latency)
1084 pm_qos_update_request(&mehci->pm_qos_req_dma,
1085 pdata->swfi_latency + 1);
1086
1087 mehci->bus_reset = 1;
Pavankumar Kondetifc7d1d82013-03-05 09:43:22 +05301088
1089 /* Halt the controller */
1090 cmd = ehci_readl(ehci, &ehci->regs->command);
1091 cmd &= ~CMD_RUN;
1092 ehci_writel(ehci, cmd, &ehci->regs->command);
1093 ret = handshake(ehci, &ehci->regs->status, STS_HALT,
1094 STS_HALT, 16 * 125);
1095 if (ret) {
1096 pr_err("halt handshake fatal error\n");
1097 dbg_log_event(NULL, "HALT: fatal", 0);
1098 goto fail;
1099 }
1100
Pavankumar Kondeti23720492013-01-29 14:41:58 +05301101retry:
1102 retries++;
1103 dbg_log_event(NULL, "RESET: start", retries);
1104 pr_debug("reset begin %d\n", retries);
1105 mehci->reset_again = 0;
1106 spin_lock_irqsave(&ehci->lock, flags);
1107 ehci_writel(ehci, val, status_reg);
1108 ehci_writel(ehci, GPT_LD(RESET_SIGNAL_TIME_USEC - 1),
1109 &mehci->timer->gptimer0_ld);
1110 ehci_writel(ehci, GPT_RESET | GPT_RUN,
1111 &mehci->timer->gptimer0_ctrl);
1112 ehci_writel(ehci, INTR_MASK | STS_GPTIMER0_INTERRUPT,
1113 &ehci->regs->intr_enable);
1114
1115 ehci_writel(ehci, GPT_LD(RESET_SIGNAL_TIME_SOF_USEC - 1),
1116 &mehci->timer->gptimer1_ld);
1117 ehci_writel(ehci, GPT_RESET | GPT_RUN,
1118 &mehci->timer->gptimer1_ctrl);
1119
1120 spin_unlock_irqrestore(&ehci->lock, flags);
1121 wait_for_completion(&mehci->gpt0_completion);
1122
1123 if (!mehci->reset_again)
1124 goto done;
1125
1126 if (handshake(ehci, status_reg, PORT_RESET, 0, 10 * 1000)) {
1127 pr_err("reset handshake fatal error\n");
1128 dbg_log_event(NULL, "RESET: fatal", retries);
1129 goto fail;
1130 }
1131
1132 if (retries < RESET_RETRY_LIMIT)
1133 goto retry;
1134
1135 /* complete reset in tight loop */
1136 pr_info("RESET in tight loop\n");
1137 dbg_log_event(NULL, "RESET: tight", 0);
1138
1139 spin_lock_irqsave(&ehci->lock, flags);
1140 ehci_writel(ehci, val, status_reg);
1141 while (cnt--)
1142 udelay(1);
1143 ret = msm_hsic_reset_done(hcd);
1144 spin_unlock_irqrestore(&ehci->lock, flags);
1145 if (ret) {
1146 pr_err("RESET in tight loop failed\n");
1147 dbg_log_event(NULL, "RESET: tight failed", 0);
1148 goto fail;
1149 }
1150
1151done:
1152 dbg_log_event(NULL, "RESET: done", retries);
1153 pr_debug("reset completed\n");
1154fail:
1155 mehci->bus_reset = 0;
1156 if (pdata && pdata->swfi_latency)
1157 pm_qos_update_request(&mehci->pm_qos_req_dma,
1158 PM_QOS_DEFAULT_VALUE);
1159}
1160
Hemant Kumar45d211b2012-05-31 17:58:43 -07001161static int ehci_hsic_bus_suspend(struct usb_hcd *hcd)
1162{
Pavankumar Kondetia723f742012-09-14 14:02:36 +05301163 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
1164
1165 if (!(readl_relaxed(USB_PORTSC) & PORT_PE)) {
1166 dbg_log_event(NULL, "RH suspend attempt failed", 0);
1167 dev_dbg(mehci->dev, "%s:port is not enabled skip suspend\n",
1168 __func__);
1169 return -EAGAIN;
1170 }
1171
Hemant Kumar45d211b2012-05-31 17:58:43 -07001172 dbg_log_event(NULL, "Suspend RH", 0);
1173 return ehci_bus_suspend(hcd);
1174}
1175
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001176#define RESUME_RETRY_LIMIT 3
Pavankumar Kondetic4f817d2013-01-29 19:08:12 +05301177#define RESUME_SIGNAL_TIME_USEC (21 * 1000)
1178#define RESUME_SIGNAL_TIME_SOF_USEC (23 * 1000)
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001179static int msm_hsic_resume_thread(void *data)
1180{
1181 struct msm_hsic_hcd *mehci = data;
1182 struct usb_hcd *hcd = hsic_to_hcd(mehci);
1183 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
1184 u32 temp;
1185 unsigned long resume_needed = 0;
1186 int retry_cnt = 0;
1187 int tight_resume = 0;
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05301188 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001189
1190 dbg_log_event(NULL, "Resume RH", 0);
1191
1192 /* keep delay between bus states */
1193 if (time_before(jiffies, ehci->next_statechange))
1194 usleep_range(5000, 5000);
1195
1196 spin_lock_irq(&ehci->lock);
1197 if (!HCD_HW_ACCESSIBLE(hcd)) {
1198 spin_unlock_irq(&ehci->lock);
1199 mehci->resume_status = -ESHUTDOWN;
1200 complete(&mehci->rt_completion);
1201 return 0;
1202 }
1203
1204 if (unlikely(ehci->debug)) {
1205 if (!dbgp_reset_prep())
1206 ehci->debug = NULL;
1207 else
1208 dbgp_external_startup();
1209 }
1210
1211 /* at least some APM implementations will try to deliver
1212 * IRQs right away, so delay them until we're ready.
1213 */
1214 ehci_writel(ehci, 0, &ehci->regs->intr_enable);
1215
1216 /* re-init operational registers */
1217 ehci_writel(ehci, 0, &ehci->regs->segment);
1218 ehci_writel(ehci, ehci->periodic_dma, &ehci->regs->frame_list);
1219 ehci_writel(ehci, (u32) ehci->async->qh_dma, &ehci->regs->async_next);
1220
1221 /*CMD_RUN will be set after, PORT_RESUME gets cleared*/
1222 if (ehci->resume_sof_bug)
1223 ehci->command &= ~CMD_RUN;
1224
1225 /* restore CMD_RUN, framelist size, and irq threshold */
1226 ehci_writel(ehci, ehci->command, &ehci->regs->command);
1227
1228 /* manually resume the ports we suspended during bus_suspend() */
1229resume_again:
1230 if (retry_cnt >= RESUME_RETRY_LIMIT) {
1231 pr_info("retry count(%d) reached max, resume in tight loop\n",
1232 retry_cnt);
1233 tight_resume = 1;
1234 }
1235
1236
1237 temp = ehci_readl(ehci, &ehci->regs->port_status[0]);
1238 temp &= ~(PORT_RWC_BITS | PORT_WAKE_BITS);
1239 if (test_bit(0, &ehci->bus_suspended) && (temp & PORT_SUSPEND)) {
1240 temp |= PORT_RESUME;
1241 set_bit(0, &resume_needed);
1242 }
1243 dbg_log_event(NULL, "FPR: Set", temp);
1244 ehci_writel(ehci, temp, &ehci->regs->port_status[0]);
1245
1246 /* HSIC controller has a h/w bug due to which it can try to send SOFs
1247 * (start of frames) during port resume resulting in phy lockup. HSIC hw
1248 * controller in MSM clears FPR bit after driving the resume signal for
1249 * 20ms. Workaround is to stop SOFs before driving resume and then start
1250 * sending SOFs immediately. Need to send SOFs within 3ms of resume
1251 * completion otherwise peripheral may enter undefined state. As
1252 * usleep_range does not gurantee exact sleep time, GPTimer is used to
1253 * to time the resume sequence. If driver exceeds allowable time SOFs,
1254 * repeat the resume process.
1255 */
1256 if (ehci->resume_sof_bug && resume_needed) {
1257 if (!tight_resume) {
1258 mehci->resume_again = 0;
Pavankumar Kondetic4f817d2013-01-29 19:08:12 +05301259 ehci_writel(ehci, GPT_LD(RESUME_SIGNAL_TIME_USEC - 1),
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001260 &mehci->timer->gptimer0_ld);
1261 ehci_writel(ehci, GPT_RESET | GPT_RUN,
1262 &mehci->timer->gptimer0_ctrl);
1263 ehci_writel(ehci, INTR_MASK | STS_GPTIMER0_INTERRUPT,
1264 &ehci->regs->intr_enable);
1265
Pavankumar Kondetic4f817d2013-01-29 19:08:12 +05301266 ehci_writel(ehci, GPT_LD(
1267 RESUME_SIGNAL_TIME_SOF_USEC - 1),
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001268 &mehci->timer->gptimer1_ld);
1269 ehci_writel(ehci, GPT_RESET | GPT_RUN,
1270 &mehci->timer->gptimer1_ctrl);
1271
1272 spin_unlock_irq(&ehci->lock);
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05301273 if (pdata && pdata->swfi_latency)
1274 pm_qos_update_request(&mehci->pm_qos_req_dma,
1275 pdata->swfi_latency + 1);
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001276 wait_for_completion(&mehci->gpt0_completion);
Hemant Kumarbc8bdf62012-10-17 12:29:51 -07001277 if (pdata && pdata->standalone_latency)
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05301278 pm_qos_update_request(&mehci->pm_qos_req_dma,
Hemant Kumarbc8bdf62012-10-17 12:29:51 -07001279 pdata->standalone_latency + 1);
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001280 spin_lock_irq(&ehci->lock);
1281 } else {
1282 dbg_log_event(NULL, "FPR: Tightloop", 0);
1283 /* do the resume in a tight loop */
1284 handshake(ehci, &ehci->regs->port_status[0],
1285 PORT_RESUME, 0, 22 * 1000);
1286 ehci_writel(ehci, ehci_readl(ehci,
1287 &ehci->regs->command) | CMD_RUN,
1288 &ehci->regs->command);
1289 }
1290
1291 if (mehci->resume_again) {
1292 int temp;
1293
1294 dbg_log_event(NULL, "FPR: Re-Resume", retry_cnt);
1295 pr_info("FPR: retry count: %d\n", retry_cnt);
1296 spin_unlock_irq(&ehci->lock);
1297 temp = ehci_readl(ehci, &ehci->regs->port_status[0]);
1298 temp &= ~PORT_RWC_BITS;
1299 temp |= PORT_SUSPEND;
1300 ehci_writel(ehci, temp, &ehci->regs->port_status[0]);
1301 /* Keep the bus idle for 5ms so that peripheral
1302 * can detect and initiate suspend
1303 */
1304 usleep_range(5000, 5000);
1305 dbg_log_event(NULL,
1306 "FPR: RResume",
1307 ehci_readl(ehci, &ehci->regs->port_status[0]));
1308 spin_lock_irq(&ehci->lock);
1309 mehci->resume_again = 0;
1310 retry_cnt++;
1311 goto resume_again;
1312 }
1313 }
1314
1315 dbg_log_event(NULL, "FPR: RT-Done", 0);
1316 mehci->resume_status = 1;
1317 spin_unlock_irq(&ehci->lock);
1318
1319 complete(&mehci->rt_completion);
1320
1321 return 0;
1322}
1323
Hemant Kumar45d211b2012-05-31 17:58:43 -07001324static int ehci_hsic_bus_resume(struct usb_hcd *hcd)
1325{
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001326 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
1327 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
1328 u32 temp;
1329 struct task_struct *resume_thread = NULL;
Vamsi Krishnabef91832013-03-11 14:10:20 -07001330 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
1331
1332 if (pdata->resume_gpio)
1333 gpio_direction_output(pdata->resume_gpio, 1);
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001334
Manu Gautam3c598392013-03-22 16:59:10 +05301335 if (!mehci->ehci.resume_sof_bug) {
1336 ehci_bus_resume(hcd);
1337 } else {
1338 mehci->resume_status = 0;
1339 resume_thread = kthread_run(msm_hsic_resume_thread,
1340 mehci, "hsic_resume_thread");
1341 if (IS_ERR(resume_thread)) {
1342 pr_err("Error creating resume thread:%lu\n",
1343 PTR_ERR(resume_thread));
1344 return PTR_ERR(resume_thread);
1345 }
1346
1347 wait_for_completion(&mehci->rt_completion);
1348
1349 if (mehci->resume_status < 0)
1350 return mehci->resume_status;
1351
1352 dbg_log_event(NULL, "FPR: Wokeup", 0);
1353 spin_lock_irq(&ehci->lock);
1354 (void) ehci_readl(ehci, &ehci->regs->command);
1355
1356 temp = 0;
1357 if (ehci->async->qh_next.qh)
1358 temp |= CMD_ASE;
1359 if (ehci->periodic_sched)
1360 temp |= CMD_PSE;
1361 if (temp) {
1362 ehci->command |= temp;
1363 ehci_writel(ehci, ehci->command, &ehci->regs->command);
1364 }
1365
1366 ehci->next_statechange = jiffies + msecs_to_jiffies(5);
1367 hcd->state = HC_STATE_RUNNING;
1368 ehci->rh_state = EHCI_RH_RUNNING;
1369 ehci->command |= CMD_RUN;
1370
1371 /* Now we can safely re-enable irqs */
1372 ehci_writel(ehci, INTR_MASK, &ehci->regs->intr_enable);
1373
1374 spin_unlock_irq(&ehci->lock);
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001375 }
1376
Vamsi Krishnabef91832013-03-11 14:10:20 -07001377 if (pdata->resume_gpio)
1378 gpio_direction_output(pdata->resume_gpio, 0);
1379
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001380 return 0;
Hemant Kumar45d211b2012-05-31 17:58:43 -07001381}
1382
Jack Phamc0d41732012-09-28 17:45:38 -07001383#else
1384
1385#define ehci_hsic_bus_suspend NULL
1386#define ehci_hsic_bus_resume NULL
1387
1388#endif /* CONFIG_PM */
1389
Hemant Kumara3d4a6e2012-10-16 17:57:08 -07001390static void ehci_msm_set_autosuspend_delay(struct usb_device *dev)
1391{
1392 if (!dev->parent) /*for root hub no delay*/
1393 pm_runtime_set_autosuspend_delay(&dev->dev, 0);
1394 else
1395 pm_runtime_set_autosuspend_delay(&dev->dev, 200);
1396}
1397
Shimrit Malichi82d7cbc2013-03-12 15:36:54 +02001398static int ehci_msm_urb_enqueue(struct usb_hcd *hcd, struct urb *urb,
1399 gfp_t mem_flags)
1400{
1401 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
1402 struct usb_host_bam_type *usb_host_bam =
1403 (struct usb_host_bam_type *)urb->priv_data;
1404
1405 if (usb_host_bam && mehci && mehci->enable_hbm)
1406 return hbm_urb_enqueue(hcd, urb, mem_flags);
1407 return ehci_urb_enqueue(hcd, urb, mem_flags);
1408}
1409
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301410static struct hc_driver msm_hsic_driver = {
1411 .description = hcd_name,
1412 .product_desc = "Qualcomm EHCI Host Controller using HSIC",
1413 .hcd_priv_size = sizeof(struct msm_hsic_hcd),
1414
1415 /*
1416 * generic hardware linkage
1417 */
1418 .irq = msm_hsic_irq,
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -07001419 .flags = HCD_USB2 | HCD_MEMORY | HCD_OLD_ENUM,
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301420
1421 .reset = ehci_hsic_reset,
1422 .start = ehci_run,
1423
1424 .stop = ehci_stop,
1425 .shutdown = ehci_shutdown,
1426
1427 /*
1428 * managing i/o requests and associated device resources
1429 */
Shimrit Malichi82d7cbc2013-03-12 15:36:54 +02001430 .urb_enqueue = ehci_msm_urb_enqueue,
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301431 .urb_dequeue = ehci_urb_dequeue,
1432 .endpoint_disable = ehci_endpoint_disable,
1433 .endpoint_reset = ehci_endpoint_reset,
1434 .clear_tt_buffer_complete = ehci_clear_tt_buffer_complete,
1435
1436 /*
1437 * scheduling support
1438 */
1439 .get_frame_number = ehci_get_frame,
1440
1441 /*
1442 * root hub support
1443 */
1444 .hub_status_data = ehci_hub_status_data,
1445 .hub_control = ehci_hub_control,
1446 .relinquish_port = ehci_relinquish_port,
1447 .port_handed_over = ehci_port_handed_over,
1448
1449 /*
1450 * PM support
1451 */
Hemant Kumar45d211b2012-05-31 17:58:43 -07001452 .bus_suspend = ehci_hsic_bus_suspend,
1453 .bus_resume = ehci_hsic_bus_resume,
1454
Hemant Kumar4d50a432012-08-15 09:06:35 -07001455 .log_urb = dbg_log_event,
Hemant Kumar105d07f2012-07-02 15:33:07 -07001456 .dump_regs = dump_hsic_regs,
Vamsi Krishna8e6edcb2012-06-20 18:08:50 -07001457
Hemant Kumara3d4a6e2012-10-16 17:57:08 -07001458 .set_autosuspend_delay = ehci_msm_set_autosuspend_delay,
Pavankumar Kondeti23720492013-01-29 14:41:58 +05301459 .reset_sof_bug_handler = ehci_hsic_reset_sof_bug_handler,
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301460};
1461
1462static int msm_hsic_init_clocks(struct msm_hsic_hcd *mehci, u32 init)
1463{
1464 int ret = 0;
1465
1466 if (!init)
1467 goto put_clocks;
1468
Lena Salman8c8ba382012-02-14 15:59:31 +02001469 /*core_clk is required for LINK protocol engine
1470 *clock rate appropriately set by target specific clock driver */
Manu Gautam5143b252012-01-05 19:25:23 -08001471 mehci->core_clk = clk_get(mehci->dev, "core_clk");
1472 if (IS_ERR(mehci->core_clk)) {
1473 dev_err(mehci->dev, "failed to get core_clk\n");
1474 ret = PTR_ERR(mehci->core_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301475 return ret;
1476 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301477
Ofir Cohen07944aa2012-12-23 13:41:57 +02001478 /* alt_core_clk is for LINK to be used during PHY RESET in
1479 * targets on which link does NOT use asynchronous reset methodology.
Lena Salman8c8ba382012-02-14 15:59:31 +02001480 * clock rate appropriately set by target specific clock driver */
Manu Gautam5143b252012-01-05 19:25:23 -08001481 mehci->alt_core_clk = clk_get(mehci->dev, "alt_core_clk");
Manu Gautam863b74132012-11-21 14:30:04 +05301482 if (IS_ERR(mehci->alt_core_clk))
1483 dev_dbg(mehci->dev, "failed to get alt_core_clk\n");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301484
Lena Salman8c8ba382012-02-14 15:59:31 +02001485 /* phy_clk is required for HSIC PHY operation
1486 * clock rate appropriately set by target specific clock driver */
Manu Gautam5143b252012-01-05 19:25:23 -08001487 mehci->phy_clk = clk_get(mehci->dev, "phy_clk");
1488 if (IS_ERR(mehci->phy_clk)) {
1489 dev_err(mehci->dev, "failed to get phy_clk\n");
1490 ret = PTR_ERR(mehci->phy_clk);
1491 goto put_alt_core_clk;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301492 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301493
1494 /* 10MHz cal_clk is required for calibration of I/O pads */
Manu Gautam5143b252012-01-05 19:25:23 -08001495 mehci->cal_clk = clk_get(mehci->dev, "cal_clk");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301496 if (IS_ERR(mehci->cal_clk)) {
Manu Gautam5143b252012-01-05 19:25:23 -08001497 dev_err(mehci->dev, "failed to get cal_clk\n");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301498 ret = PTR_ERR(mehci->cal_clk);
Manu Gautam5143b252012-01-05 19:25:23 -08001499 goto put_phy_clk;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301500 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301501
1502 /* ahb_clk is required for data transfers */
Manu Gautam5143b252012-01-05 19:25:23 -08001503 mehci->ahb_clk = clk_get(mehci->dev, "iface_clk");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301504 if (IS_ERR(mehci->ahb_clk)) {
Manu Gautam5143b252012-01-05 19:25:23 -08001505 dev_err(mehci->dev, "failed to get iface_clk\n");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301506 ret = PTR_ERR(mehci->ahb_clk);
1507 goto put_cal_clk;
1508 }
1509
Manu Gautam28b1bac2012-01-30 16:43:06 +05301510 clk_prepare_enable(mehci->core_clk);
1511 clk_prepare_enable(mehci->phy_clk);
1512 clk_prepare_enable(mehci->cal_clk);
1513 clk_prepare_enable(mehci->ahb_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301514
1515 return 0;
1516
1517put_clocks:
Jack Phamfd193eb2012-02-22 15:38:08 -08001518 if (!atomic_read(&mehci->in_lpm)) {
1519 clk_disable_unprepare(mehci->core_clk);
1520 clk_disable_unprepare(mehci->phy_clk);
1521 clk_disable_unprepare(mehci->cal_clk);
1522 clk_disable_unprepare(mehci->ahb_clk);
1523 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301524 clk_put(mehci->ahb_clk);
1525put_cal_clk:
1526 clk_put(mehci->cal_clk);
Manu Gautam5143b252012-01-05 19:25:23 -08001527put_phy_clk:
1528 clk_put(mehci->phy_clk);
1529put_alt_core_clk:
Manu Gautam863b74132012-11-21 14:30:04 +05301530 if (!IS_ERR(mehci->alt_core_clk))
1531 clk_put(mehci->alt_core_clk);
Manu Gautam5143b252012-01-05 19:25:23 -08001532 clk_put(mehci->core_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301533
1534 return ret;
1535}
Vamsi Krishna34f01582011-12-14 19:54:42 -08001536static irqreturn_t hsic_peripheral_status_change(int irq, void *dev_id)
1537{
1538 struct msm_hsic_hcd *mehci = dev_id;
1539
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001540 pr_debug("%s: mehci:%p dev_id:%p\n", __func__, mehci, dev_id);
Vamsi Krishna34f01582011-12-14 19:54:42 -08001541
1542 if (mehci)
1543 msm_hsic_config_gpios(mehci, 0);
1544
1545 return IRQ_HANDLED;
1546}
1547
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001548static irqreturn_t msm_hsic_wakeup_irq(int irq, void *data)
1549{
1550 struct msm_hsic_hcd *mehci = data;
Pavankumar Kondeti968cf272012-09-27 12:04:12 +05301551 int ret;
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001552
Manu Gautam3fb60ca2013-02-13 18:33:33 +05301553 if (irq == mehci->async_irq) {
1554 mehci->async_int_cnt++;
1555 dbg_log_event(NULL, "Remote Wakeup (ASYNC) IRQ",
1556 mehci->async_int_cnt);
1557 } else {
1558 mehci->wakeup_int_cnt++;
1559 dbg_log_event(NULL, "Remote Wakeup IRQ", mehci->wakeup_int_cnt);
1560 }
1561 dev_dbg(mehci->dev, "%s: hsic remote wakeup interrupt %d cnt: %u, %u\n",
1562 __func__, irq, mehci->wakeup_int_cnt, mehci->async_int_cnt);
Hemant Kumar6fd65032012-05-23 13:02:24 -07001563
Sudhir Sharma1673e302012-08-27 17:37:24 -07001564 wake_lock(&mehci->wlock);
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001565
Manu Gautam3fb60ca2013-02-13 18:33:33 +05301566 if (mehci->wakeup_irq) {
1567 spin_lock(&mehci->wakeup_lock);
1568 if (mehci->wakeup_irq_enabled) {
1569 mehci->wakeup_irq_enabled = 0;
1570 disable_irq_wake(irq);
1571 disable_irq_nosync(irq);
1572 }
1573 spin_unlock(&mehci->wakeup_lock);
Jack Phamfe441ea2012-03-23 17:03:15 -07001574 }
1575
Sudhir Sharma1673e302012-08-27 17:37:24 -07001576 if (!atomic_read(&mehci->pm_usage_cnt)) {
Pavankumar Kondeti968cf272012-09-27 12:04:12 +05301577 ret = pm_runtime_get(mehci->dev);
1578 /*
1579 * HSIC runtime resume can race with us.
1580 * if we are active (ret == 1) or resuming
1581 * (ret == -EINPROGRESS), decrement the
1582 * PM usage counter before returning.
1583 */
1584 if ((ret == 1) || (ret == -EINPROGRESS))
1585 pm_runtime_put_noidle(mehci->dev);
1586 else
1587 atomic_set(&mehci->pm_usage_cnt, 1);
Sudhir Sharma1673e302012-08-27 17:37:24 -07001588 }
1589
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001590 return IRQ_HANDLED;
1591}
1592
Hemant Kumare6275972012-02-29 20:06:21 -08001593static int ehci_hsic_msm_bus_show(struct seq_file *s, void *unused)
1594{
1595 if (debug_bus_voting_enabled)
1596 seq_printf(s, "enabled\n");
1597 else
1598 seq_printf(s, "disabled\n");
1599
1600 return 0;
1601}
1602
1603static int ehci_hsic_msm_bus_open(struct inode *inode, struct file *file)
1604{
1605 return single_open(file, ehci_hsic_msm_bus_show, inode->i_private);
1606}
1607
1608static ssize_t ehci_hsic_msm_bus_write(struct file *file,
1609 const char __user *ubuf, size_t count, loff_t *ppos)
1610{
1611 char buf[8];
1612 int ret;
1613 struct seq_file *s = file->private_data;
1614 struct msm_hsic_hcd *mehci = s->private;
1615
1616 memset(buf, 0x00, sizeof(buf));
1617
1618 if (copy_from_user(&buf, ubuf, min_t(size_t, sizeof(buf) - 1, count)))
1619 return -EFAULT;
1620
1621 if (!strncmp(buf, "enable", 6)) {
1622 /* Do not vote here. Let hsic driver decide when to vote */
1623 debug_bus_voting_enabled = true;
1624 } else {
1625 debug_bus_voting_enabled = false;
1626 if (mehci->bus_perf_client) {
1627 ret = msm_bus_scale_client_update_request(
1628 mehci->bus_perf_client, 0);
1629 if (ret)
1630 dev_err(mehci->dev, "%s: Failed to devote "
1631 "for bus bw %d\n", __func__, ret);
1632 }
1633 }
1634
1635 return count;
1636}
1637
1638const struct file_operations ehci_hsic_msm_bus_fops = {
1639 .open = ehci_hsic_msm_bus_open,
1640 .read = seq_read,
1641 .write = ehci_hsic_msm_bus_write,
1642 .llseek = seq_lseek,
1643 .release = single_release,
1644};
1645
Hemant Kumar6fd65032012-05-23 13:02:24 -07001646static int ehci_hsic_msm_wakeup_cnt_show(struct seq_file *s, void *unused)
1647{
1648 struct msm_hsic_hcd *mehci = s->private;
1649
1650 seq_printf(s, "%u\n", mehci->wakeup_int_cnt);
1651
1652 return 0;
1653}
1654
1655static int ehci_hsic_msm_wakeup_cnt_open(struct inode *inode, struct file *f)
1656{
1657 return single_open(f, ehci_hsic_msm_wakeup_cnt_show, inode->i_private);
1658}
1659
1660const struct file_operations ehci_hsic_msm_wakeup_cnt_fops = {
1661 .open = ehci_hsic_msm_wakeup_cnt_open,
1662 .read = seq_read,
1663 .llseek = seq_lseek,
1664 .release = single_release,
1665};
1666
Hemant Kumar45d211b2012-05-31 17:58:43 -07001667static int ehci_hsic_msm_data_events_show(struct seq_file *s, void *unused)
1668{
1669 unsigned long flags;
1670 unsigned i;
1671
1672 read_lock_irqsave(&dbg_hsic_data.lck, flags);
1673
1674 i = dbg_hsic_data.idx;
1675 for (dbg_inc(&i); i != dbg_hsic_data.idx; dbg_inc(&i)) {
1676 if (!strnlen(dbg_hsic_data.buf[i], DBG_MSG_LEN))
1677 continue;
1678 seq_printf(s, "%s\n", dbg_hsic_data.buf[i]);
1679 }
1680
1681 read_unlock_irqrestore(&dbg_hsic_data.lck, flags);
1682
1683 return 0;
1684}
1685
1686static int ehci_hsic_msm_data_events_open(struct inode *inode, struct file *f)
1687{
1688 return single_open(f, ehci_hsic_msm_data_events_show, inode->i_private);
1689}
1690
1691const struct file_operations ehci_hsic_msm_dbg_data_fops = {
1692 .open = ehci_hsic_msm_data_events_open,
1693 .read = seq_read,
1694 .llseek = seq_lseek,
1695 .release = single_release,
1696};
1697
1698static int ehci_hsic_msm_ctrl_events_show(struct seq_file *s, void *unused)
1699{
1700 unsigned long flags;
1701 unsigned i;
1702
1703 read_lock_irqsave(&dbg_hsic_ctrl.lck, flags);
1704
1705 i = dbg_hsic_ctrl.idx;
1706 for (dbg_inc(&i); i != dbg_hsic_ctrl.idx; dbg_inc(&i)) {
1707 if (!strnlen(dbg_hsic_ctrl.buf[i], DBG_MSG_LEN))
1708 continue;
1709 seq_printf(s, "%s\n", dbg_hsic_ctrl.buf[i]);
1710 }
1711
1712 read_unlock_irqrestore(&dbg_hsic_ctrl.lck, flags);
1713
1714 return 0;
1715}
1716
1717static int ehci_hsic_msm_ctrl_events_open(struct inode *inode, struct file *f)
1718{
1719 return single_open(f, ehci_hsic_msm_ctrl_events_show, inode->i_private);
1720}
1721
1722const struct file_operations ehci_hsic_msm_dbg_ctrl_fops = {
1723 .open = ehci_hsic_msm_ctrl_events_open,
1724 .read = seq_read,
1725 .llseek = seq_lseek,
1726 .release = single_release,
1727};
1728
Hemant Kumare6275972012-02-29 20:06:21 -08001729static struct dentry *ehci_hsic_msm_dbg_root;
1730static int ehci_hsic_msm_debugfs_init(struct msm_hsic_hcd *mehci)
1731{
1732 struct dentry *ehci_hsic_msm_dentry;
1733
1734 ehci_hsic_msm_dbg_root = debugfs_create_dir("ehci_hsic_msm_dbg", NULL);
1735
1736 if (!ehci_hsic_msm_dbg_root || IS_ERR(ehci_hsic_msm_dbg_root))
1737 return -ENODEV;
1738
1739 ehci_hsic_msm_dentry = debugfs_create_file("bus_voting",
1740 S_IRUGO | S_IWUSR,
1741 ehci_hsic_msm_dbg_root, mehci,
1742 &ehci_hsic_msm_bus_fops);
1743
1744 if (!ehci_hsic_msm_dentry) {
1745 debugfs_remove_recursive(ehci_hsic_msm_dbg_root);
1746 return -ENODEV;
1747 }
1748
Hemant Kumar6fd65032012-05-23 13:02:24 -07001749 ehci_hsic_msm_dentry = debugfs_create_file("wakeup_cnt",
1750 S_IRUGO,
1751 ehci_hsic_msm_dbg_root, mehci,
1752 &ehci_hsic_msm_wakeup_cnt_fops);
1753
1754 if (!ehci_hsic_msm_dentry) {
1755 debugfs_remove_recursive(ehci_hsic_msm_dbg_root);
1756 return -ENODEV;
1757 }
1758
Hemant Kumar45d211b2012-05-31 17:58:43 -07001759 ehci_hsic_msm_dentry = debugfs_create_file("show_ctrl_events",
1760 S_IRUGO,
1761 ehci_hsic_msm_dbg_root, mehci,
1762 &ehci_hsic_msm_dbg_ctrl_fops);
1763
1764 if (!ehci_hsic_msm_dentry) {
1765 debugfs_remove_recursive(ehci_hsic_msm_dbg_root);
1766 return -ENODEV;
1767 }
1768
1769 ehci_hsic_msm_dentry = debugfs_create_file("show_data_events",
1770 S_IRUGO,
1771 ehci_hsic_msm_dbg_root, mehci,
1772 &ehci_hsic_msm_dbg_data_fops);
1773
1774 if (!ehci_hsic_msm_dentry) {
1775 debugfs_remove_recursive(ehci_hsic_msm_dbg_root);
1776 return -ENODEV;
1777 }
1778
Hemant Kumare6275972012-02-29 20:06:21 -08001779 return 0;
1780}
1781
1782static void ehci_hsic_msm_debugfs_cleanup(void)
1783{
1784 debugfs_remove_recursive(ehci_hsic_msm_dbg_root);
1785}
1786
Manu Gautam863b74132012-11-21 14:30:04 +05301787struct msm_hsic_host_platform_data *msm_hsic_dt_to_pdata(
1788 struct platform_device *pdev)
1789{
1790 struct device_node *node = pdev->dev.of_node;
1791 struct msm_hsic_host_platform_data *pdata;
Ido Shayevitzca5dc492013-01-09 14:49:59 +02001792 int res_gpio;
Manu Gautam863b74132012-11-21 14:30:04 +05301793
1794 pdata = devm_kzalloc(&pdev->dev, sizeof(*pdata), GFP_KERNEL);
1795 if (!pdata) {
1796 dev_err(&pdev->dev, "unable to allocate platform data\n");
1797 return NULL;
1798 }
Manu Gautam863b74132012-11-21 14:30:04 +05301799
Ido Shayevitzca5dc492013-01-09 14:49:59 +02001800 res_gpio = of_get_named_gpio(node, "hsic,strobe-gpio", 0);
1801 if (res_gpio < 0)
1802 res_gpio = 0;
1803 pdata->strobe = res_gpio;
1804
1805 res_gpio = of_get_named_gpio(node, "hsic,data-gpio", 0);
1806 if (res_gpio < 0)
1807 res_gpio = 0;
1808 pdata->data = res_gpio;
Manu Gautam863b74132012-11-21 14:30:04 +05301809
Vamsi Krishnabef91832013-03-11 14:10:20 -07001810 res_gpio = of_get_named_gpio(node, "hsic,resume-gpio", 0);
1811 if (res_gpio < 0)
1812 res_gpio = 0;
1813 pdata->resume_gpio = res_gpio;
1814
Manu Gautam3c598392013-03-22 16:59:10 +05301815 pdata->phy_sof_workaround = of_property_read_bool(node,
1816 "qcom,phy-sof-workaround");
Manu Gautam863b74132012-11-21 14:30:04 +05301817 pdata->ignore_cal_pad_config = of_property_read_bool(node,
1818 "hsic,ignore-cal-pad-config");
1819 of_property_read_u32(node, "hsic,strobe-pad-offset",
1820 &pdata->strobe_pad_offset);
1821 of_property_read_u32(node, "hsic,data-pad-offset",
1822 &pdata->data_pad_offset);
1823
Ido Shayevitz95c46882013-01-20 13:47:46 +02001824 pdata->bus_scale_table = msm_bus_cl_get_pdata(pdev);
1825
Shimrit Malichi36a0c852013-03-10 11:39:10 +02001826 pdata->pool_64_bit_align = of_property_read_bool(node,
1827 "qcom,pool-64-bit-align");
Shimrit Malichi82d7cbc2013-03-12 15:36:54 +02001828 pdata->enable_hbm = of_property_read_bool(node,
1829 "qcom,enable-hbm");
Shimrit Malichi36a0c852013-03-10 11:39:10 +02001830
Manu Gautam863b74132012-11-21 14:30:04 +05301831 return pdata;
1832}
1833
1834
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301835static int __devinit ehci_hsic_msm_probe(struct platform_device *pdev)
1836{
1837 struct usb_hcd *hcd;
1838 struct resource *res;
1839 struct msm_hsic_hcd *mehci;
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +05301840 struct msm_hsic_host_platform_data *pdata;
Manu Gautam16562792013-03-13 07:04:11 +05301841 unsigned long wakeup_irq_flags = 0;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301842 int ret;
1843
1844 dev_dbg(&pdev->dev, "ehci_msm-hsic probe\n");
1845
Manu Gautam863b74132012-11-21 14:30:04 +05301846 if (pdev->dev.of_node) {
1847 dev_dbg(&pdev->dev, "device tree enabled\n");
1848 pdev->dev.platform_data = msm_hsic_dt_to_pdata(pdev);
1849 dev_set_name(&pdev->dev, ehci_msm_hsic_driver.driver.name);
Manu Gautam16562792013-03-13 07:04:11 +05301850 } else {
1851 /* explicitly pass wakeup_irq flag for !DT */
1852 wakeup_irq_flags = IRQF_TRIGGER_HIGH;
Manu Gautam863b74132012-11-21 14:30:04 +05301853 }
1854 if (!pdev->dev.platform_data)
1855 dev_dbg(&pdev->dev, "No platform data given\n");
1856
Ofir Cohen4cc55372012-11-20 11:17:24 +02001857 if (!pdev->dev.dma_mask)
1858 pdev->dev.dma_mask = &ehci_msm_hsic_dma_mask;
1859 if (!pdev->dev.coherent_dma_mask)
1860 pdev->dev.coherent_dma_mask = DMA_BIT_MASK(32);
1861
Vijayavardhan Vennapusaafbbb8f2012-04-13 16:28:45 +05301862 /* After parent device's probe is executed, it will be put in suspend
1863 * mode. When child device's probe is called, driver core is not
1864 * resuming parent device due to which parent will be in suspend even
1865 * though child is active. Hence resume the parent device explicitly.
1866 */
1867 if (pdev->dev.parent)
1868 pm_runtime_get_sync(pdev->dev.parent);
1869
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301870 hcd = usb_create_hcd(&msm_hsic_driver, &pdev->dev,
1871 dev_name(&pdev->dev));
1872 if (!hcd) {
1873 dev_err(&pdev->dev, "Unable to create HCD\n");
1874 return -ENOMEM;
1875 }
1876
Pavankumar Kondeti6f199aa2012-09-18 17:52:51 +05301877 hcd_to_bus(hcd)->skip_resume = true;
1878
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301879 hcd->irq = platform_get_irq(pdev, 0);
1880 if (hcd->irq < 0) {
1881 dev_err(&pdev->dev, "Unable to get IRQ resource\n");
1882 ret = hcd->irq;
1883 goto put_hcd;
1884 }
1885
1886 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
1887 if (!res) {
1888 dev_err(&pdev->dev, "Unable to get memory resource\n");
1889 ret = -ENODEV;
1890 goto put_hcd;
1891 }
1892
1893 hcd->rsrc_start = res->start;
1894 hcd->rsrc_len = resource_size(res);
1895 hcd->regs = ioremap(hcd->rsrc_start, hcd->rsrc_len);
1896 if (!hcd->regs) {
1897 dev_err(&pdev->dev, "ioremap failed\n");
1898 ret = -ENOMEM;
1899 goto put_hcd;
1900 }
1901
1902 mehci = hcd_to_hsic(hcd);
1903 mehci->dev = &pdev->dev;
Hemant Kumar30d361c2012-08-20 14:44:40 -07001904 pdata = mehci->dev->platform_data;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301905
Hemant Kumard343c012012-09-06 19:57:14 -07001906 spin_lock_init(&mehci->wakeup_lock);
1907
Manu Gautam3c598392013-03-22 16:59:10 +05301908 if (pdata->phy_sof_workaround) {
1909 mehci->ehci.susp_sof_bug = 1;
1910 mehci->ehci.reset_sof_bug = 1;
1911 mehci->ehci.resume_sof_bug = 1;
1912 }
Hemant Kumar38ce5d82012-05-29 13:00:58 -07001913
Shimrit Malichi36a0c852013-03-10 11:39:10 +02001914 mehci->ehci.pool_64_bit_align = pdata->pool_64_bit_align;
Shimrit Malichi82d7cbc2013-03-12 15:36:54 +02001915 mehci->enable_hbm = pdata->enable_hbm;
Hemant Kumare4040492012-06-21 17:35:42 -07001916
Hemant Kumar30d361c2012-08-20 14:44:40 -07001917 if (pdata)
1918 mehci->ehci.log2_irq_thresh = pdata->log2_irq_thresh;
Hemant Kumar933e0402012-05-22 11:11:40 -07001919
Ofir Cohen4cc55372012-11-20 11:17:24 +02001920 ret = msm_hsic_init_gdsc(mehci, 1);
1921 if (ret) {
1922 dev_err(&pdev->dev, "unable to initialize GDSC\n");
1923 ret = -ENODEV;
1924 goto put_hcd;
1925 }
1926
Vamsi Krishna34f01582011-12-14 19:54:42 -08001927 res = platform_get_resource_byname(pdev,
1928 IORESOURCE_IRQ,
1929 "peripheral_status_irq");
1930 if (res)
1931 mehci->peripheral_status_irq = res->start;
1932
Jack Pham0cc75c42012-10-10 02:03:50 +02001933 res = platform_get_resource_byname(pdev, IORESOURCE_IRQ, "wakeup");
Hemant Kumar6fd65032012-05-23 13:02:24 -07001934 if (res) {
Jack Pham0cc75c42012-10-10 02:03:50 +02001935 mehci->wakeup_irq = res->start;
Hemant Kumar6fd65032012-05-23 13:02:24 -07001936 dev_dbg(mehci->dev, "wakeup_irq: %d\n", mehci->wakeup_irq);
1937 }
1938
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301939 ret = msm_hsic_init_clocks(mehci, 1);
1940 if (ret) {
1941 dev_err(&pdev->dev, "unable to initialize clocks\n");
1942 ret = -ENODEV;
1943 goto unmap;
1944 }
1945
1946 ret = msm_hsic_init_vddcx(mehci, 1);
1947 if (ret) {
1948 dev_err(&pdev->dev, "unable to initialize VDDCX\n");
1949 ret = -ENODEV;
1950 goto deinit_clocks;
1951 }
1952
Vamsi Krishnafce1bd22012-08-17 17:12:26 -07001953 init_completion(&mehci->rt_completion);
1954 init_completion(&mehci->gpt0_completion);
Ofir Cohen5dddb152012-11-14 11:18:50 +02001955
1956 msm_hsic_phy_reset(mehci);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301957
Hemant Kumar2309eaa2012-08-14 16:46:42 -07001958 ehci_wq = create_singlethread_workqueue("ehci_wq");
1959 if (!ehci_wq) {
1960 dev_err(&pdev->dev, "unable to create workqueue\n");
1961 ret = -ENOMEM;
1962 goto deinit_vddcx;
1963 }
1964
1965 INIT_WORK(&mehci->bus_vote_w, ehci_hsic_bus_vote_w);
1966
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301967 ret = usb_add_hcd(hcd, hcd->irq, IRQF_SHARED);
1968 if (ret) {
1969 dev_err(&pdev->dev, "unable to register HCD\n");
Ofir Cohen5dddb152012-11-14 11:18:50 +02001970 goto destroy_wq;
1971 }
1972
1973 ret = msm_hsic_start(mehci);
1974 if (ret) {
1975 dev_err(&pdev->dev, "unable to initialize PHY\n");
1976 goto destroy_wq;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301977 }
1978
1979 device_init_wakeup(&pdev->dev, 1);
Sudhir Sharma1673e302012-08-27 17:37:24 -07001980 wake_lock_init(&mehci->wlock, WAKE_LOCK_SUSPEND, dev_name(&pdev->dev));
1981 wake_lock(&mehci->wlock);
Vamsi Krishna34f01582011-12-14 19:54:42 -08001982
1983 if (mehci->peripheral_status_irq) {
1984 ret = request_threaded_irq(mehci->peripheral_status_irq,
1985 NULL, hsic_peripheral_status_change,
1986 IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING
1987 | IRQF_SHARED,
1988 "hsic_peripheral_status", mehci);
1989 if (ret)
1990 dev_err(&pdev->dev, "%s:request_irq:%d failed:%d",
1991 __func__, mehci->peripheral_status_irq, ret);
1992 }
1993
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08001994 /* configure wakeup irq */
Hemant Kumar6fd65032012-05-23 13:02:24 -07001995 if (mehci->wakeup_irq) {
Hemant Kumarbbcdcbc2012-08-30 14:23:38 -07001996 /* In case if wakeup gpio is pulled high at this point
1997 * remote wakeup interrupt fires right after request_irq.
1998 * Remote wake up interrupt only needs to be enabled when
1999 * HSIC bus goes to suspend.
2000 */
2001 irq_set_status_flags(mehci->wakeup_irq, IRQ_NOAUTOEN);
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08002002 ret = request_irq(mehci->wakeup_irq, msm_hsic_wakeup_irq,
Manu Gautam16562792013-03-13 07:04:11 +05302003 wakeup_irq_flags,
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08002004 "msm_hsic_wakeup", mehci);
Hemant Kumarbbcdcbc2012-08-30 14:23:38 -07002005 if (ret) {
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08002006 dev_err(&pdev->dev, "request_irq(%d) failed: %d\n",
2007 mehci->wakeup_irq, ret);
2008 mehci->wakeup_irq = 0;
2009 }
2010 }
2011
Manu Gautam3fb60ca2013-02-13 18:33:33 +05302012 mehci->async_irq = platform_get_irq_byname(pdev, "async_irq");
2013 if (mehci->async_irq < 0) {
2014 dev_dbg(&pdev->dev, "platform_get_irq for async_int failed\n");
2015 mehci->async_irq = 0;
2016 } else {
2017 ret = request_irq(mehci->async_irq, msm_hsic_wakeup_irq,
2018 IRQF_TRIGGER_RISING, "msm_hsic_async", mehci);
2019 if (ret) {
2020 dev_err(&pdev->dev, "request irq failed (ASYNC INT)\n");
2021 mehci->async_irq = 0;
Manu Gautam16562792013-03-13 07:04:11 +05302022 } else if (!mehci->wakeup_irq) {
2023 /* Async IRQ is used only in absence of dedicated irq */
Manu Gautam3fb60ca2013-02-13 18:33:33 +05302024 enable_irq_wake(mehci->async_irq);
2025 }
2026 }
2027
Hemant Kumare6275972012-02-29 20:06:21 -08002028 ret = ehci_hsic_msm_debugfs_init(mehci);
2029 if (ret)
2030 dev_dbg(&pdev->dev, "mode debugfs file is"
2031 "not available\n");
2032
2033 if (pdata && pdata->bus_scale_table) {
2034 mehci->bus_perf_client =
2035 msm_bus_scale_register_client(pdata->bus_scale_table);
2036 /* Configure BUS performance parameters for MAX bandwidth */
2037 if (mehci->bus_perf_client) {
Hemant Kumar2309eaa2012-08-14 16:46:42 -07002038 mehci->bus_vote = true;
2039 queue_work(ehci_wq, &mehci->bus_vote_w);
Hemant Kumare6275972012-02-29 20:06:21 -08002040 } else {
2041 dev_err(&pdev->dev, "%s: Failed to register BUS "
2042 "scaling client!!\n", __func__);
2043 }
2044 }
2045
Hemant Kumar105d07f2012-07-02 15:33:07 -07002046 __mehci = mehci;
2047
Hemant Kumarbc8bdf62012-10-17 12:29:51 -07002048 if (pdata && pdata->standalone_latency)
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05302049 pm_qos_add_request(&mehci->pm_qos_req_dma,
Hemant Kumarbc8bdf62012-10-17 12:29:51 -07002050 PM_QOS_CPU_DMA_LATENCY, pdata->standalone_latency + 1);
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05302051
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302052 /*
2053 * This pdev->dev is assigned parent of root-hub by USB core,
2054 * hence, runtime framework automatically calls this driver's
2055 * runtime APIs based on root-hub's state.
2056 */
2057 pm_runtime_set_active(&pdev->dev);
2058 pm_runtime_enable(&pdev->dev);
Vijayavardhan Vennapusaafbbb8f2012-04-13 16:28:45 +05302059 /* Decrement the parent device's counter after probe.
2060 * As child is active, parent will not be put into
2061 * suspend mode.
2062 */
2063 if (pdev->dev.parent)
2064 pm_runtime_put_sync(pdev->dev.parent);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302065
Shimrit Malichi82d7cbc2013-03-12 15:36:54 +02002066 if (mehci->enable_hbm)
2067 hbm_init(hcd);
2068
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302069 return 0;
2070
Ofir Cohen5dddb152012-11-14 11:18:50 +02002071destroy_wq:
Hemant Kumar2309eaa2012-08-14 16:46:42 -07002072 destroy_workqueue(ehci_wq);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302073deinit_vddcx:
2074 msm_hsic_init_vddcx(mehci, 0);
Ofir Cohen4cc55372012-11-20 11:17:24 +02002075 msm_hsic_init_gdsc(mehci, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302076deinit_clocks:
2077 msm_hsic_init_clocks(mehci, 0);
2078unmap:
2079 iounmap(hcd->regs);
2080put_hcd:
2081 usb_put_hcd(hcd);
2082
2083 return ret;
2084}
2085
2086static int __devexit ehci_hsic_msm_remove(struct platform_device *pdev)
2087{
2088 struct usb_hcd *hcd = platform_get_drvdata(pdev);
2089 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05302090 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
2091
Ido Shayevitz65943492013-02-20 18:55:59 +02002092 /* If the device was removed no need to call pm_runtime_disable */
2093 if (pdev->dev.power.power_state.event != PM_EVENT_INVALID)
2094 pm_runtime_disable(&pdev->dev);
2095
2096 pm_runtime_set_suspended(&pdev->dev);
2097
Shimrit Malichi82d7cbc2013-03-12 15:36:54 +02002098 if (mehci->enable_hbm)
2099 hbm_uninit();
2100
Pavankumar Kondeti21a840e2013-02-12 21:43:56 +05302101 /* Remove the HCD prior to releasing our resources. */
2102 usb_remove_hcd(hcd);
2103
Hemant Kumarbc8bdf62012-10-17 12:29:51 -07002104 if (pdata && pdata->standalone_latency)
Pavankumar Kondeti4f5dc3b2012-09-07 15:33:09 +05302105 pm_qos_remove_request(&mehci->pm_qos_req_dma);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302106
Vamsi Krishna34f01582011-12-14 19:54:42 -08002107 if (mehci->peripheral_status_irq)
2108 free_irq(mehci->peripheral_status_irq, mehci);
Jack Phamfe441ea2012-03-23 17:03:15 -07002109
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08002110 if (mehci->wakeup_irq) {
Jack Phamfe441ea2012-03-23 17:03:15 -07002111 if (mehci->wakeup_irq_enabled)
2112 disable_irq_wake(mehci->wakeup_irq);
Vamsi Krishna6921cbe2012-02-21 18:34:43 -08002113 free_irq(mehci->wakeup_irq, mehci);
2114 }
Vamsi Krishna34f01582011-12-14 19:54:42 -08002115
Manu Gautam3fb60ca2013-02-13 18:33:33 +05302116 if (mehci->async_irq) {
Manu Gautam1a1a9c62013-03-19 17:37:11 +05302117 /* Async IRQ is used only in absence of dedicated wakeup irq */
2118 if (!mehci->wakeup_irq)
2119 disable_irq_wake(mehci->async_irq);
Manu Gautam3fb60ca2013-02-13 18:33:33 +05302120 free_irq(mehci->async_irq, mehci);
2121 }
Hemant Kumar2309eaa2012-08-14 16:46:42 -07002122 /*
2123 * If the update request is called after unregister, the request will
2124 * fail. Results are undefined if unregister is called in the middle of
2125 * update request.
2126 */
2127 mehci->bus_vote = false;
2128 cancel_work_sync(&mehci->bus_vote_w);
2129
Hemant Kumare6275972012-02-29 20:06:21 -08002130 if (mehci->bus_perf_client)
2131 msm_bus_scale_unregister_client(mehci->bus_perf_client);
2132
2133 ehci_hsic_msm_debugfs_cleanup();
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302134 device_init_wakeup(&pdev->dev, 0);
Ido Shayevitze50d9142013-02-17 20:43:26 +02002135
Hemant Kumar2309eaa2012-08-14 16:46:42 -07002136 destroy_workqueue(ehci_wq);
2137
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +05302138 msm_hsic_config_gpios(mehci, 0);
Vamsi Krishnabef91832013-03-11 14:10:20 -07002139
2140 if (pdata && pdata->resume_gpio)
2141 gpio_free(pdata->resume_gpio);
2142
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302143 msm_hsic_init_vddcx(mehci, 0);
Ofir Cohen4cc55372012-11-20 11:17:24 +02002144 msm_hsic_init_gdsc(mehci, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302145
2146 msm_hsic_init_clocks(mehci, 0);
Sudhir Sharma1673e302012-08-27 17:37:24 -07002147 wake_lock_destroy(&mehci->wlock);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302148 iounmap(hcd->regs);
2149 usb_put_hcd(hcd);
2150
2151 return 0;
2152}
2153
2154#ifdef CONFIG_PM_SLEEP
2155static int msm_hsic_pm_suspend(struct device *dev)
2156{
Jack Phambe05fbb2012-05-16 10:56:26 -07002157 int ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302158 struct usb_hcd *hcd = dev_get_drvdata(dev);
2159 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
2160
2161 dev_dbg(dev, "ehci-msm-hsic PM suspend\n");
2162
Hemant Kumar45d211b2012-05-31 17:58:43 -07002163 dbg_log_event(NULL, "PM Suspend", 0);
2164
Manu Gautam16562792013-03-13 07:04:11 +05302165 if (device_may_wakeup(dev) && !mehci->async_irq)
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302166 enable_irq_wake(hcd->irq);
2167
Jack Phambe05fbb2012-05-16 10:56:26 -07002168 ret = msm_hsic_suspend(mehci);
2169
Manu Gautam16562792013-03-13 07:04:11 +05302170 if (ret && device_may_wakeup(dev) && !mehci->async_irq)
Jack Phambe05fbb2012-05-16 10:56:26 -07002171 disable_irq_wake(hcd->irq);
2172
2173 return ret;
Jack Phamfe441ea2012-03-23 17:03:15 -07002174}
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302175
Jack Pham16b06f82012-08-14 20:03:59 -07002176static int msm_hsic_pm_suspend_noirq(struct device *dev)
2177{
2178 struct usb_hcd *hcd = dev_get_drvdata(dev);
2179 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
2180
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +05302181 if (atomic_read(&mehci->async_int)) {
Jack Pham16b06f82012-08-14 20:03:59 -07002182 dev_dbg(dev, "suspend_noirq: Aborting due to pending interrupt\n");
2183 return -EBUSY;
2184 }
2185
2186 return 0;
2187}
2188
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302189static int msm_hsic_pm_resume(struct device *dev)
2190{
2191 int ret;
2192 struct usb_hcd *hcd = dev_get_drvdata(dev);
2193 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
2194
Jack Pham16b06f82012-08-14 20:03:59 -07002195 dev_dbg(dev, "ehci-msm-hsic PM resume\n");
Hemant Kumar45d211b2012-05-31 17:58:43 -07002196 dbg_log_event(NULL, "PM Resume", 0);
2197
Manu Gautam16562792013-03-13 07:04:11 +05302198 if (device_may_wakeup(dev) && !mehci->async_irq)
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302199 disable_irq_wake(hcd->irq);
2200
Pavankumar Kondeti6f199aa2012-09-18 17:52:51 +05302201 /*
2202 * Keep HSIC in Low Power Mode if system is resumed
2203 * by any other wakeup source. HSIC is resumed later
2204 * when remote wakeup is received or interface driver
2205 * start I/O.
2206 */
Pavankumar Kondeti41d004c2012-11-09 10:54:00 +05302207 if (!atomic_read(&mehci->pm_usage_cnt) &&
Pavankumar Kondeti822fbe62013-01-28 17:04:39 +05302208 !atomic_read(&mehci->async_int) &&
Pavankumar Kondeti41d004c2012-11-09 10:54:00 +05302209 pm_runtime_suspended(dev))
Pavankumar Kondeti6f199aa2012-09-18 17:52:51 +05302210 return 0;
2211
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302212 ret = msm_hsic_resume(mehci);
2213 if (ret)
2214 return ret;
2215
2216 /* Bring the device to full powered state upon system resume */
2217 pm_runtime_disable(dev);
2218 pm_runtime_set_active(dev);
2219 pm_runtime_enable(dev);
2220
2221 return 0;
2222}
2223#endif
2224
2225#ifdef CONFIG_PM_RUNTIME
2226static int msm_hsic_runtime_idle(struct device *dev)
2227{
2228 dev_dbg(dev, "EHCI runtime idle\n");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302229 return 0;
2230}
2231
2232static int msm_hsic_runtime_suspend(struct device *dev)
2233{
2234 struct usb_hcd *hcd = dev_get_drvdata(dev);
2235 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
2236
2237 dev_dbg(dev, "EHCI runtime suspend\n");
Hemant Kumar45d211b2012-05-31 17:58:43 -07002238
2239 dbg_log_event(NULL, "Run Time PM Suspend", 0);
2240
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302241 return msm_hsic_suspend(mehci);
2242}
2243
2244static int msm_hsic_runtime_resume(struct device *dev)
2245{
2246 struct usb_hcd *hcd = dev_get_drvdata(dev);
2247 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
2248
2249 dev_dbg(dev, "EHCI runtime resume\n");
Hemant Kumar45d211b2012-05-31 17:58:43 -07002250
2251 dbg_log_event(NULL, "Run Time PM Resume", 0);
2252
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302253 return msm_hsic_resume(mehci);
2254}
2255#endif
2256
2257#ifdef CONFIG_PM
2258static const struct dev_pm_ops msm_hsic_dev_pm_ops = {
2259 SET_SYSTEM_SLEEP_PM_OPS(msm_hsic_pm_suspend, msm_hsic_pm_resume)
Jack Pham16b06f82012-08-14 20:03:59 -07002260 .suspend_noirq = msm_hsic_pm_suspend_noirq,
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302261 SET_RUNTIME_PM_OPS(msm_hsic_runtime_suspend, msm_hsic_runtime_resume,
2262 msm_hsic_runtime_idle)
2263};
2264#endif
Ofir Cohen4cc55372012-11-20 11:17:24 +02002265static const struct of_device_id hsic_host_dt_match[] = {
2266 { .compatible = "qcom,hsic-host",
2267 },
2268 {}
2269};
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302270static struct platform_driver ehci_msm_hsic_driver = {
2271 .probe = ehci_hsic_msm_probe,
2272 .remove = __devexit_p(ehci_hsic_msm_remove),
2273 .driver = {
2274 .name = "msm_hsic_host",
2275#ifdef CONFIG_PM
2276 .pm = &msm_hsic_dev_pm_ops,
2277#endif
Ofir Cohen4cc55372012-11-20 11:17:24 +02002278 .of_match_table = hsic_host_dt_match,
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05302279 },
2280};