blob: 2d43a7804dd1f2a95e247a3b47be9c3ec1a40af6 [file] [log] [blame]
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301/* ehci-msm-hsic.c - HSUSB Host Controller Driver Implementation
2 *
Manu Gautam5143b252012-01-05 19:25:23 -08003 * Copyright (c) 2011-2012, Code Aurora Forum. 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>
28#include <linux/wakelock.h>
29#include <linux/pm_runtime.h>
30#include <linux/regulator/consumer.h>
31
32#include <linux/usb/msm_hsusb_hw.h>
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +053033#include <linux/usb/msm_hsusb.h>
34#include <linux/gpio.h>
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053035#include <mach/clk.h>
36#include <mach/msm_iomap.h>
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +053037#include <mach/msm_xo.h>
Vamsi Krishna34f01582011-12-14 19:54:42 -080038#include <linux/spinlock.h>
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053039
40#define MSM_USB_BASE (hcd->regs)
41
42struct msm_hsic_hcd {
43 struct ehci_hcd ehci;
44 struct device *dev;
45 struct clk *ahb_clk;
Manu Gautam5143b252012-01-05 19:25:23 -080046 struct clk *core_clk;
47 struct clk *alt_core_clk;
48 struct clk *phy_clk;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053049 struct clk *cal_clk;
50 struct regulator *hsic_vddcx;
51 bool async_int;
52 atomic_t in_lpm;
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +053053 struct msm_xo_voter *xo_handle;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053054 struct wake_lock wlock;
Vamsi Krishna34f01582011-12-14 19:54:42 -080055 int peripheral_status_irq;
Vamsi Krishna6921cbe2012-02-21 18:34:43 -080056 int wakeup_irq;
Jack Phamfe441ea2012-03-23 17:03:15 -070057 bool wakeup_irq_enabled;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053058};
59
60static inline struct msm_hsic_hcd *hcd_to_hsic(struct usb_hcd *hcd)
61{
62 return (struct msm_hsic_hcd *) (hcd->hcd_priv);
63}
64
65static inline struct usb_hcd *hsic_to_hcd(struct msm_hsic_hcd *mehci)
66{
67 return container_of((void *) mehci, struct usb_hcd, hcd_priv);
68}
69
70#define ULPI_IO_TIMEOUT_USEC (10 * 1000)
71
Vamsi Krishna45d88fa2011-11-02 13:28:42 -070072#define USB_PHY_VDD_DIG_VOL_SUSP_MIN 500000 /* uV */
73#define USB_PHY_VDD_DIG_VOL_MIN 1000000 /* uV */
74#define USB_PHY_VDD_DIG_VOL_MAX 1320000 /* uV */
75#define USB_PHY_VDD_DIG_LOAD 49360 /* uA */
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053076
Lena Salman8c8ba382012-02-14 15:59:31 +020077#define HSIC_DBG1_REG 0x38
78
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +053079static int msm_hsic_init_vddcx(struct msm_hsic_hcd *mehci, int init)
80{
81 int ret = 0;
82
83 if (!init)
84 goto disable_reg;
85
86 mehci->hsic_vddcx = regulator_get(mehci->dev, "HSIC_VDDCX");
87 if (IS_ERR(mehci->hsic_vddcx)) {
88 dev_err(mehci->dev, "unable to get hsic vddcx\n");
89 return PTR_ERR(mehci->hsic_vddcx);
90 }
91
92 ret = regulator_set_voltage(mehci->hsic_vddcx,
93 USB_PHY_VDD_DIG_VOL_MIN,
94 USB_PHY_VDD_DIG_VOL_MAX);
95 if (ret) {
96 dev_err(mehci->dev, "unable to set the voltage"
97 "for hsic vddcx\n");
98 goto reg_set_voltage_err;
99 }
100
101 ret = regulator_set_optimum_mode(mehci->hsic_vddcx,
102 USB_PHY_VDD_DIG_LOAD);
103 if (ret < 0) {
104 pr_err("%s: Unable to set optimum mode of the regulator:"
105 "VDDCX\n", __func__);
106 goto reg_optimum_mode_err;
107 }
108
109 ret = regulator_enable(mehci->hsic_vddcx);
110 if (ret) {
111 dev_err(mehci->dev, "unable to enable hsic vddcx\n");
112 goto reg_enable_err;
113 }
114
115 return 0;
116
117disable_reg:
118 regulator_disable(mehci->hsic_vddcx);
119reg_enable_err:
120 regulator_set_optimum_mode(mehci->hsic_vddcx, 0);
121reg_optimum_mode_err:
122 regulator_set_voltage(mehci->hsic_vddcx, 0,
123 USB_PHY_VDD_DIG_VOL_MIN);
124reg_set_voltage_err:
125 regulator_put(mehci->hsic_vddcx);
126
127 return ret;
128
129}
130
131static int ulpi_write(struct msm_hsic_hcd *mehci, u32 val, u32 reg)
132{
133 struct usb_hcd *hcd = hsic_to_hcd(mehci);
134 int cnt = 0;
135
136 /* initiate write operation */
137 writel_relaxed(ULPI_RUN | ULPI_WRITE |
138 ULPI_ADDR(reg) | ULPI_DATA(val),
139 USB_ULPI_VIEWPORT);
140
141 /* wait for completion */
142 while (cnt < ULPI_IO_TIMEOUT_USEC) {
143 if (!(readl_relaxed(USB_ULPI_VIEWPORT) & ULPI_RUN))
144 break;
145 udelay(1);
146 cnt++;
147 }
148
149 if (cnt >= ULPI_IO_TIMEOUT_USEC) {
150 dev_err(mehci->dev, "ulpi_write: timeout\n");
151 return -ETIMEDOUT;
152 }
153
154 return 0;
155}
156
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530157#define HSIC_HUB_VDD_VOL_MIN 1650000 /* uV */
158#define HSIC_HUB_VDD_VOL_MAX 1950000 /* uV */
159#define HSIC_HUB_VDD_LOAD 36000 /* uA */
160static int msm_hsic_config_hub(struct msm_hsic_hcd *mehci, int init)
161{
162 int ret = 0;
163 struct msm_hsic_host_platform_data *pdata;
164 static struct regulator *hsic_hub_reg;
165
166 pdata = mehci->dev->platform_data;
Lena Salman8c8ba382012-02-14 15:59:31 +0200167 if (!pdata || !pdata->hub_reset)
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530168 return ret;
169
170 if (!init)
171 goto disable_reg;
172
173 hsic_hub_reg = regulator_get(mehci->dev, "EXT_HUB_VDDIO");
174 if (IS_ERR(hsic_hub_reg)) {
175 dev_err(mehci->dev, "unable to get ext hub vddcx\n");
176 return PTR_ERR(hsic_hub_reg);
177 }
178
179 ret = gpio_request(pdata->hub_reset, "HSIC_HUB_RESET_GPIO");
180 if (ret < 0) {
181 dev_err(mehci->dev, "gpio request failed for GPIO%d\n",
182 pdata->hub_reset);
183 goto gpio_req_fail;
184 }
185
186 ret = regulator_set_voltage(hsic_hub_reg,
187 HSIC_HUB_VDD_VOL_MIN,
188 HSIC_HUB_VDD_VOL_MAX);
189 if (ret) {
190 dev_err(mehci->dev, "unable to set the voltage"
191 "for hsic hub reg\n");
192 goto reg_set_voltage_fail;
193 }
194
195 ret = regulator_set_optimum_mode(hsic_hub_reg,
196 HSIC_HUB_VDD_LOAD);
197 if (ret < 0) {
198 pr_err("%s: Unable to set optimum mode of the regulator:"
199 "VDDCX\n", __func__);
200 goto reg_optimum_mode_fail;
201 }
202
203 ret = regulator_enable(hsic_hub_reg);
204 if (ret) {
205 dev_err(mehci->dev, "unable to enable ext hub vddcx\n");
206 goto reg_enable_fail;
207 }
208
209 gpio_direction_output(pdata->hub_reset, 0);
210 /* Hub reset should be asserted for minimum 2usec before deasserting */
211 udelay(5);
212 gpio_direction_output(pdata->hub_reset, 1);
213
214 return 0;
215
216disable_reg:
217 regulator_disable(hsic_hub_reg);
218reg_enable_fail:
219 regulator_set_optimum_mode(hsic_hub_reg, 0);
220reg_optimum_mode_fail:
221 regulator_set_voltage(hsic_hub_reg, 0,
222 HSIC_HUB_VDD_VOL_MIN);
223reg_set_voltage_fail:
224 gpio_free(pdata->hub_reset);
225gpio_req_fail:
226 regulator_put(hsic_hub_reg);
227
228 return ret;
229
230}
231
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530232static int msm_hsic_config_gpios(struct msm_hsic_hcd *mehci, int gpio_en)
233{
234 int rc = 0;
235 struct msm_hsic_host_platform_data *pdata;
Vamsi Krishna34f01582011-12-14 19:54:42 -0800236 static int gpio_status;
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530237
238 pdata = mehci->dev->platform_data;
Vamsi Krishna34f01582011-12-14 19:54:42 -0800239
Lena Salman8c8ba382012-02-14 15:59:31 +0200240 if (!pdata || !pdata->strobe || !pdata->data)
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530241 return rc;
242
Vamsi Krishna34f01582011-12-14 19:54:42 -0800243 if (gpio_status == gpio_en)
244 return 0;
245
246 gpio_status = gpio_en;
247
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530248 if (!gpio_en)
249 goto free_gpio;
250
251 rc = gpio_request(pdata->strobe, "HSIC_STROBE_GPIO");
252 if (rc < 0) {
253 dev_err(mehci->dev, "gpio request failed for HSIC STROBE\n");
254 return rc;
255 }
256
257 rc = gpio_request(pdata->data, "HSIC_DATA_GPIO");
258 if (rc < 0) {
259 dev_err(mehci->dev, "gpio request failed for HSIC DATA\n");
260 goto free_strobe;
261 }
262
263 return 0;
264
265free_gpio:
266 gpio_free(pdata->data);
267free_strobe:
268 gpio_free(pdata->strobe);
269
270 return rc;
271}
272
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530273static int msm_hsic_phy_clk_reset(struct msm_hsic_hcd *mehci)
274{
275 int ret;
276
Manu Gautam28b1bac2012-01-30 16:43:06 +0530277 clk_prepare_enable(mehci->alt_core_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530278
Manu Gautam5143b252012-01-05 19:25:23 -0800279 ret = clk_reset(mehci->core_clk, CLK_RESET_ASSERT);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530280 if (ret) {
Manu Gautam28b1bac2012-01-30 16:43:06 +0530281 clk_disable_unprepare(mehci->alt_core_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530282 dev_err(mehci->dev, "usb phy clk assert failed\n");
283 return ret;
284 }
285 usleep_range(10000, 12000);
Manu Gautam28b1bac2012-01-30 16:43:06 +0530286 clk_disable_unprepare(mehci->alt_core_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530287
Manu Gautam5143b252012-01-05 19:25:23 -0800288 ret = clk_reset(mehci->core_clk, CLK_RESET_DEASSERT);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530289 if (ret)
290 dev_err(mehci->dev, "usb phy clk deassert failed\n");
291
292 return ret;
293}
294
295static int msm_hsic_phy_reset(struct msm_hsic_hcd *mehci)
296{
297 struct usb_hcd *hcd = hsic_to_hcd(mehci);
298 u32 val;
299 int ret;
300
301 ret = msm_hsic_phy_clk_reset(mehci);
302 if (ret)
303 return ret;
304
305 val = readl_relaxed(USB_PORTSC) & ~PORTSC_PTS_MASK;
306 writel_relaxed(val | PORTSC_PTS_ULPI, USB_PORTSC);
307
308 /* Ensure that RESET operation is completed before turning off clock */
309 mb();
310 dev_dbg(mehci->dev, "phy_reset: success\n");
311
312 return 0;
313}
314
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530315#define HSIC_GPIO150_PAD_CTL (MSM_TLMM_BASE+0x20C0)
316#define HSIC_GPIO151_PAD_CTL (MSM_TLMM_BASE+0x20C4)
317#define HSIC_CAL_PAD_CTL (MSM_TLMM_BASE+0x20C8)
318#define HSIC_LV_MODE 0x04
319#define HSIC_PAD_CALIBRATION 0xA8
320#define HSIC_GPIO_PAD_VAL 0x0A0AAA10
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530321#define LINK_RESET_TIMEOUT_USEC (250 * 1000)
322static int msm_hsic_reset(struct msm_hsic_hcd *mehci)
323{
324 struct usb_hcd *hcd = hsic_to_hcd(mehci);
325 int cnt = 0;
326 int ret;
Lena Salman8c8ba382012-02-14 15:59:31 +0200327 struct msm_hsic_host_platform_data *pdata = mehci->dev->platform_data;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530328
329 ret = msm_hsic_phy_reset(mehci);
330 if (ret) {
331 dev_err(mehci->dev, "phy_reset failed\n");
332 return ret;
333 }
334
335 writel_relaxed(USBCMD_RESET, USB_USBCMD);
336 while (cnt < LINK_RESET_TIMEOUT_USEC) {
337 if (!(readl_relaxed(USB_USBCMD) & USBCMD_RESET))
338 break;
339 udelay(1);
340 cnt++;
341 }
342 if (cnt >= LINK_RESET_TIMEOUT_USEC)
343 return -ETIMEDOUT;
344
Lena Salman8c8ba382012-02-14 15:59:31 +0200345 /* Reset PORTSC and select ULPI phy */
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530346 writel_relaxed(0x80000000, USB_PORTSC);
347
348 /* TODO: Need to confirm if HSIC PHY also requires delay after RESET */
349 msleep(100);
350
351 /* HSIC PHY Initialization */
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530352
Lena Salman8c8ba382012-02-14 15:59:31 +0200353 /* HSIC init sequence when HSIC signals (Strobe/Data) are
354 routed via GPIOs */
355 if (pdata && pdata->strobe && pdata->data) {
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530356
Lena Salman8c8ba382012-02-14 15:59:31 +0200357 /* Enable LV_MODE in HSIC_CAL_PAD_CTL register */
358 writel_relaxed(HSIC_LV_MODE, HSIC_CAL_PAD_CTL);
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530359
Lena Salman8c8ba382012-02-14 15:59:31 +0200360 /*set periodic calibration interval to ~2.048sec in
361 HSIC_IO_CAL_REG */
362 ulpi_write(mehci, 0xFF, 0x33);
363
364 /* Enable periodic IO calibration in HSIC_CFG register */
365 ulpi_write(mehci, HSIC_PAD_CALIBRATION, 0x30);
366
367 /* Configure GPIO 150/151 pins for HSIC functionality mode */
368 ret = msm_hsic_config_gpios(mehci, 1);
369 if (ret) {
370 dev_err(mehci->dev, " gpio configuarion failed\n");
371 return ret;
372 }
373 /* Set LV_MODE=0x1 and DCC=0x2 in HSIC_GPIO150/151_PAD_CTL
374 register */
375 writel_relaxed(HSIC_GPIO_PAD_VAL, HSIC_GPIO150_PAD_CTL);
376 writel_relaxed(HSIC_GPIO_PAD_VAL, HSIC_GPIO151_PAD_CTL);
377 /* Enable HSIC mode in HSIC_CFG register */
378 ulpi_write(mehci, 0x01, 0x31);
379 } else {
380 /* HSIC init sequence when HSIC signals (Strobe/Data) are routed
381 via dedicated I/O */
382
383 /* programmable length of connect signaling (33.2ns) */
384 ret = ulpi_write(mehci, 3, HSIC_DBG1_REG);
385 if (ret) {
386 pr_err("%s: Unable to program length of connect "
387 "signaling\n", __func__);
388 }
389
390 /*set periodic calibration interval to ~2.048sec in
391 HSIC_IO_CAL_REG */
392 ulpi_write(mehci, 0xFF, 0x33);
393
394 /* Enable HSIC mode in HSIC_CFG register */
395 ulpi_write(mehci, 0xA9, 0x30);
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530396 }
397
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530398 return 0;
399}
400
401#define PHY_SUSPEND_TIMEOUT_USEC (500 * 1000)
402#define PHY_RESUME_TIMEOUT_USEC (100 * 1000)
403
404#ifdef CONFIG_PM_SLEEP
405static int msm_hsic_suspend(struct msm_hsic_hcd *mehci)
406{
407 struct usb_hcd *hcd = hsic_to_hcd(mehci);
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530408 int cnt = 0, ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530409 u32 val;
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530410 struct msm_hsic_host_platform_data *pdata;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530411
412 if (atomic_read(&mehci->in_lpm)) {
413 dev_dbg(mehci->dev, "%s called in lpm\n", __func__);
414 return 0;
415 }
416
417 disable_irq(hcd->irq);
418 /*
419 * PHY may take some time or even fail to enter into low power
420 * mode (LPM). Hence poll for 500 msec and reset the PHY and link
421 * in failure case.
422 */
423 val = readl_relaxed(USB_PORTSC) | PORTSC_PHCD;
424 writel_relaxed(val, USB_PORTSC);
425 while (cnt < PHY_SUSPEND_TIMEOUT_USEC) {
426 if (readl_relaxed(USB_PORTSC) & PORTSC_PHCD)
427 break;
428 udelay(1);
429 cnt++;
430 }
431
432 if (cnt >= PHY_SUSPEND_TIMEOUT_USEC) {
433 dev_err(mehci->dev, "Unable to suspend PHY\n");
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530434 msm_hsic_config_gpios(mehci, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530435 msm_hsic_reset(mehci);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530436 }
437
438 /*
439 * PHY has capability to generate interrupt asynchronously in low
440 * power mode (LPM). This interrupt is level triggered. So USB IRQ
441 * line must be disabled till async interrupt enable bit is cleared
442 * in USBCMD register. Assert STP (ULPI interface STOP signal) to
443 * block data communication from PHY.
444 */
445 writel_relaxed(readl_relaxed(USB_USBCMD) | ASYNC_INTR_CTRL |
446 ULPI_STP_CTRL, USB_USBCMD);
447
448 /*
449 * Ensure that hardware is put in low power mode before
450 * clocks are turned OFF and VDD is allowed to minimize.
451 */
452 mb();
453
Manu Gautam28b1bac2012-01-30 16:43:06 +0530454 clk_disable_unprepare(mehci->core_clk);
455 clk_disable_unprepare(mehci->phy_clk);
456 clk_disable_unprepare(mehci->cal_clk);
457 clk_disable_unprepare(mehci->ahb_clk);
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530458 pdata = mehci->dev->platform_data;
Lena Salman8c8ba382012-02-14 15:59:31 +0200459 if (pdata && pdata->hub_reset) {
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530460 ret = msm_xo_mode_vote(mehci->xo_handle, MSM_XO_MODE_OFF);
461 if (ret)
462 pr_err("%s failed to devote for"
463 "TCXO D1 buffer%d\n", __func__, ret);
464 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530465
Vamsi Krishna45d88fa2011-11-02 13:28:42 -0700466 ret = regulator_set_voltage(mehci->hsic_vddcx,
467 USB_PHY_VDD_DIG_VOL_SUSP_MIN,
468 USB_PHY_VDD_DIG_VOL_MAX);
469 if (ret < 0)
470 dev_err(mehci->dev, "unable to set vddcx voltage: min:0.5v max:1.3v\n");
471
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530472 atomic_set(&mehci->in_lpm, 1);
473 enable_irq(hcd->irq);
474 wake_unlock(&mehci->wlock);
475
476 dev_info(mehci->dev, "HSIC-USB in low power mode\n");
477
478 return 0;
479}
480
481static int msm_hsic_resume(struct msm_hsic_hcd *mehci)
482{
483 struct usb_hcd *hcd = hsic_to_hcd(mehci);
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530484 int cnt = 0, ret;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530485 unsigned temp;
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530486 struct msm_hsic_host_platform_data *pdata;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530487
488 if (!atomic_read(&mehci->in_lpm)) {
489 dev_dbg(mehci->dev, "%s called in !in_lpm\n", __func__);
490 return 0;
491 }
492
493 wake_lock(&mehci->wlock);
494
Vamsi Krishna45d88fa2011-11-02 13:28:42 -0700495 ret = regulator_set_voltage(mehci->hsic_vddcx,
496 USB_PHY_VDD_DIG_VOL_MIN,
497 USB_PHY_VDD_DIG_VOL_MAX);
498 if (ret < 0)
499 dev_err(mehci->dev, "unable to set vddcx voltage: min:1v max:1.3v\n");
500
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530501 pdata = mehci->dev->platform_data;
Lena Salman8c8ba382012-02-14 15:59:31 +0200502 if (pdata && pdata->hub_reset) {
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530503 ret = msm_xo_mode_vote(mehci->xo_handle, MSM_XO_MODE_ON);
504 if (ret)
505 pr_err("%s failed to vote for"
506 "TCXO D1 buffer%d\n", __func__, ret);
507 }
Manu Gautam28b1bac2012-01-30 16:43:06 +0530508 clk_prepare_enable(mehci->core_clk);
509 clk_prepare_enable(mehci->phy_clk);
510 clk_prepare_enable(mehci->cal_clk);
511 clk_prepare_enable(mehci->ahb_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530512
513 temp = readl_relaxed(USB_USBCMD);
514 temp &= ~ASYNC_INTR_CTRL;
515 temp &= ~ULPI_STP_CTRL;
516 writel_relaxed(temp, USB_USBCMD);
517
518 if (!(readl_relaxed(USB_PORTSC) & PORTSC_PHCD))
519 goto skip_phy_resume;
520
521 temp = readl_relaxed(USB_PORTSC) & ~PORTSC_PHCD;
522 writel_relaxed(temp, USB_PORTSC);
523 while (cnt < PHY_RESUME_TIMEOUT_USEC) {
524 if (!(readl_relaxed(USB_PORTSC) & PORTSC_PHCD) &&
525 (readl_relaxed(USB_ULPI_VIEWPORT) & ULPI_SYNC_STATE))
526 break;
527 udelay(1);
528 cnt++;
529 }
530
531 if (cnt >= PHY_RESUME_TIMEOUT_USEC) {
532 /*
533 * This is a fatal error. Reset the link and
534 * PHY to make hsic working.
535 */
536 dev_err(mehci->dev, "Unable to resume USB. Reset the hsic\n");
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530537 msm_hsic_config_gpios(mehci, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530538 msm_hsic_reset(mehci);
539 }
540
541skip_phy_resume:
542
543 atomic_set(&mehci->in_lpm, 0);
544
545 if (mehci->async_int) {
546 mehci->async_int = false;
547 pm_runtime_put_noidle(mehci->dev);
548 enable_irq(hcd->irq);
549 }
550
551 dev_info(mehci->dev, "HSIC-USB exited from low power mode\n");
552
553 return 0;
554}
555#endif
556
557static irqreturn_t msm_hsic_irq(struct usb_hcd *hcd)
558{
559 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
560
561 if (atomic_read(&mehci->in_lpm)) {
562 disable_irq_nosync(hcd->irq);
563 mehci->async_int = true;
564 pm_runtime_get(mehci->dev);
565 return IRQ_HANDLED;
566 }
567
568 return ehci_irq(hcd);
569}
570
571static int ehci_hsic_reset(struct usb_hcd *hcd)
572{
573 struct ehci_hcd *ehci = hcd_to_ehci(hcd);
574 int retval;
575
576 ehci->caps = USB_CAPLENGTH;
577 ehci->regs = USB_CAPLENGTH +
578 HC_LENGTH(ehci, ehci_readl(ehci, &ehci->caps->hc_capbase));
579 dbg_hcs_params(ehci, "reset");
580 dbg_hcc_params(ehci, "reset");
581
582 /* cache the data to minimize the chip reads*/
583 ehci->hcs_params = ehci_readl(ehci, &ehci->caps->hcs_params);
584
585 hcd->has_tt = 1;
586 ehci->sbrn = HCD_USB2;
587
588 retval = ehci_halt(ehci);
589 if (retval)
590 return retval;
591
592 /* data structure init */
593 retval = ehci_init(hcd);
594 if (retval)
595 return retval;
596
597 retval = ehci_reset(ehci);
598 if (retval)
599 return retval;
600
601 /* bursts of unspecified length. */
602 writel_relaxed(0, USB_AHBBURST);
603 /* Use the AHB transactor */
604 writel_relaxed(0, USB_AHBMODE);
605 /* Disable streaming mode and select host mode */
606 writel_relaxed(0x13, USB_USBMODE);
607
608 ehci_port_power(ehci, 1);
609 return 0;
610}
611
612static struct hc_driver msm_hsic_driver = {
613 .description = hcd_name,
614 .product_desc = "Qualcomm EHCI Host Controller using HSIC",
615 .hcd_priv_size = sizeof(struct msm_hsic_hcd),
616
617 /*
618 * generic hardware linkage
619 */
620 .irq = msm_hsic_irq,
621 .flags = HCD_USB2 | HCD_MEMORY,
622
623 .reset = ehci_hsic_reset,
624 .start = ehci_run,
625
626 .stop = ehci_stop,
627 .shutdown = ehci_shutdown,
628
629 /*
630 * managing i/o requests and associated device resources
631 */
632 .urb_enqueue = ehci_urb_enqueue,
633 .urb_dequeue = ehci_urb_dequeue,
634 .endpoint_disable = ehci_endpoint_disable,
635 .endpoint_reset = ehci_endpoint_reset,
636 .clear_tt_buffer_complete = ehci_clear_tt_buffer_complete,
637
638 /*
639 * scheduling support
640 */
641 .get_frame_number = ehci_get_frame,
642
643 /*
644 * root hub support
645 */
646 .hub_status_data = ehci_hub_status_data,
647 .hub_control = ehci_hub_control,
648 .relinquish_port = ehci_relinquish_port,
649 .port_handed_over = ehci_port_handed_over,
650
651 /*
652 * PM support
653 */
654 .bus_suspend = ehci_bus_suspend,
655 .bus_resume = ehci_bus_resume,
656};
657
658static int msm_hsic_init_clocks(struct msm_hsic_hcd *mehci, u32 init)
659{
660 int ret = 0;
661
662 if (!init)
663 goto put_clocks;
664
Lena Salman8c8ba382012-02-14 15:59:31 +0200665 /*core_clk is required for LINK protocol engine
666 *clock rate appropriately set by target specific clock driver */
Manu Gautam5143b252012-01-05 19:25:23 -0800667 mehci->core_clk = clk_get(mehci->dev, "core_clk");
668 if (IS_ERR(mehci->core_clk)) {
669 dev_err(mehci->dev, "failed to get core_clk\n");
670 ret = PTR_ERR(mehci->core_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530671 return ret;
672 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530673
Lena Salman8c8ba382012-02-14 15:59:31 +0200674 /* alt_core_clk is for LINK to be used during PHY RESET
675 * clock rate appropriately set by target specific clock driver */
Manu Gautam5143b252012-01-05 19:25:23 -0800676 mehci->alt_core_clk = clk_get(mehci->dev, "alt_core_clk");
677 if (IS_ERR(mehci->alt_core_clk)) {
678 dev_err(mehci->dev, "failed to core_clk\n");
679 ret = PTR_ERR(mehci->alt_core_clk);
680 goto put_core_clk;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530681 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530682
Lena Salman8c8ba382012-02-14 15:59:31 +0200683 /* phy_clk is required for HSIC PHY operation
684 * clock rate appropriately set by target specific clock driver */
Manu Gautam5143b252012-01-05 19:25:23 -0800685 mehci->phy_clk = clk_get(mehci->dev, "phy_clk");
686 if (IS_ERR(mehci->phy_clk)) {
687 dev_err(mehci->dev, "failed to get phy_clk\n");
688 ret = PTR_ERR(mehci->phy_clk);
689 goto put_alt_core_clk;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530690 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530691
692 /* 10MHz cal_clk is required for calibration of I/O pads */
Manu Gautam5143b252012-01-05 19:25:23 -0800693 mehci->cal_clk = clk_get(mehci->dev, "cal_clk");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530694 if (IS_ERR(mehci->cal_clk)) {
Manu Gautam5143b252012-01-05 19:25:23 -0800695 dev_err(mehci->dev, "failed to get cal_clk\n");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530696 ret = PTR_ERR(mehci->cal_clk);
Manu Gautam5143b252012-01-05 19:25:23 -0800697 goto put_phy_clk;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530698 }
699 clk_set_rate(mehci->cal_clk, 10000000);
700
701 /* ahb_clk is required for data transfers */
Manu Gautam5143b252012-01-05 19:25:23 -0800702 mehci->ahb_clk = clk_get(mehci->dev, "iface_clk");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530703 if (IS_ERR(mehci->ahb_clk)) {
Manu Gautam5143b252012-01-05 19:25:23 -0800704 dev_err(mehci->dev, "failed to get iface_clk\n");
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530705 ret = PTR_ERR(mehci->ahb_clk);
706 goto put_cal_clk;
707 }
708
Manu Gautam28b1bac2012-01-30 16:43:06 +0530709 clk_prepare_enable(mehci->core_clk);
710 clk_prepare_enable(mehci->phy_clk);
711 clk_prepare_enable(mehci->cal_clk);
712 clk_prepare_enable(mehci->ahb_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530713
714 return 0;
715
716put_clocks:
Jack Phamfd193eb2012-02-22 15:38:08 -0800717 if (!atomic_read(&mehci->in_lpm)) {
718 clk_disable_unprepare(mehci->core_clk);
719 clk_disable_unprepare(mehci->phy_clk);
720 clk_disable_unprepare(mehci->cal_clk);
721 clk_disable_unprepare(mehci->ahb_clk);
722 }
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530723 clk_put(mehci->ahb_clk);
724put_cal_clk:
725 clk_put(mehci->cal_clk);
Manu Gautam5143b252012-01-05 19:25:23 -0800726put_phy_clk:
727 clk_put(mehci->phy_clk);
728put_alt_core_clk:
729 clk_put(mehci->alt_core_clk);
730put_core_clk:
731 clk_put(mehci->core_clk);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530732
733 return ret;
734}
Vamsi Krishna34f01582011-12-14 19:54:42 -0800735static irqreturn_t hsic_peripheral_status_change(int irq, void *dev_id)
736{
737 struct msm_hsic_hcd *mehci = dev_id;
738
Vamsi Krishna6921cbe2012-02-21 18:34:43 -0800739 pr_debug("%s: mehci:%p dev_id:%p\n", __func__, mehci, dev_id);
Vamsi Krishna34f01582011-12-14 19:54:42 -0800740
741 if (mehci)
742 msm_hsic_config_gpios(mehci, 0);
743
744 return IRQ_HANDLED;
745}
746
Vamsi Krishna6921cbe2012-02-21 18:34:43 -0800747static irqreturn_t msm_hsic_wakeup_irq(int irq, void *data)
748{
749 struct msm_hsic_hcd *mehci = data;
750
751 dev_dbg(mehci->dev, "%s: hsic remote wakeup interrupt\n", __func__);
752
Jack Phamfe441ea2012-03-23 17:03:15 -0700753 if (mehci->wakeup_irq_enabled) {
754 mehci->wakeup_irq_enabled = 0;
755 disable_irq_wake(irq);
756 disable_irq_nosync(irq);
757 }
758
Vamsi Krishna6921cbe2012-02-21 18:34:43 -0800759 return IRQ_HANDLED;
760}
761
762
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530763static int __devinit ehci_hsic_msm_probe(struct platform_device *pdev)
764{
765 struct usb_hcd *hcd;
766 struct resource *res;
767 struct msm_hsic_hcd *mehci;
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530768 struct msm_hsic_host_platform_data *pdata;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530769 int ret;
770
771 dev_dbg(&pdev->dev, "ehci_msm-hsic probe\n");
772
773 hcd = usb_create_hcd(&msm_hsic_driver, &pdev->dev,
774 dev_name(&pdev->dev));
775 if (!hcd) {
776 dev_err(&pdev->dev, "Unable to create HCD\n");
777 return -ENOMEM;
778 }
779
780 hcd->irq = platform_get_irq(pdev, 0);
781 if (hcd->irq < 0) {
782 dev_err(&pdev->dev, "Unable to get IRQ resource\n");
783 ret = hcd->irq;
784 goto put_hcd;
785 }
786
787 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
788 if (!res) {
789 dev_err(&pdev->dev, "Unable to get memory resource\n");
790 ret = -ENODEV;
791 goto put_hcd;
792 }
793
794 hcd->rsrc_start = res->start;
795 hcd->rsrc_len = resource_size(res);
796 hcd->regs = ioremap(hcd->rsrc_start, hcd->rsrc_len);
797 if (!hcd->regs) {
798 dev_err(&pdev->dev, "ioremap failed\n");
799 ret = -ENOMEM;
800 goto put_hcd;
801 }
802
803 mehci = hcd_to_hsic(hcd);
804 mehci->dev = &pdev->dev;
805
Vamsi Krishna34f01582011-12-14 19:54:42 -0800806 res = platform_get_resource_byname(pdev,
807 IORESOURCE_IRQ,
808 "peripheral_status_irq");
809 if (res)
810 mehci->peripheral_status_irq = res->start;
811
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530812 ret = msm_hsic_init_clocks(mehci, 1);
813 if (ret) {
814 dev_err(&pdev->dev, "unable to initialize clocks\n");
815 ret = -ENODEV;
816 goto unmap;
817 }
818
819 ret = msm_hsic_init_vddcx(mehci, 1);
820 if (ret) {
821 dev_err(&pdev->dev, "unable to initialize VDDCX\n");
822 ret = -ENODEV;
823 goto deinit_clocks;
824 }
825
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530826 pdata = mehci->dev->platform_data;
Lena Salman8c8ba382012-02-14 15:59:31 +0200827 if (pdata && pdata->hub_reset) {
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530828 mehci->xo_handle = msm_xo_get(MSM_XO_TCXO_D1, "hsic");
829 if (IS_ERR(mehci->xo_handle)) {
830 pr_err(" %s not able to get the handle"
831 "to vote for TCXO D1 buffer\n", __func__);
832 ret = PTR_ERR(mehci->xo_handle);
833 goto deinit_vddcx;
834 }
835
836 ret = msm_xo_mode_vote(mehci->xo_handle, MSM_XO_MODE_ON);
837 if (ret) {
838 pr_err("%s failed to vote for TCXO"
839 "D1 buffer%d\n", __func__, ret);
840 goto free_xo_handle;
841 }
842 }
843
844 ret = msm_hsic_config_hub(mehci, 1);
845 if (ret) {
846 dev_err(&pdev->dev, "unable to initialize hsic hub");
847 goto free_xo_handle;
848 }
849
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530850 ret = msm_hsic_reset(mehci);
851 if (ret) {
852 dev_err(&pdev->dev, "unable to initialize PHY\n");
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530853 goto deinit_hub;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530854 }
855
856 ret = usb_add_hcd(hcd, hcd->irq, IRQF_SHARED);
857 if (ret) {
858 dev_err(&pdev->dev, "unable to register HCD\n");
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530859 goto unconfig_gpio;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530860 }
861
862 device_init_wakeup(&pdev->dev, 1);
863 wake_lock_init(&mehci->wlock, WAKE_LOCK_SUSPEND, dev_name(&pdev->dev));
864 wake_lock(&mehci->wlock);
Vamsi Krishna34f01582011-12-14 19:54:42 -0800865
866 if (mehci->peripheral_status_irq) {
867 ret = request_threaded_irq(mehci->peripheral_status_irq,
868 NULL, hsic_peripheral_status_change,
869 IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING
870 | IRQF_SHARED,
871 "hsic_peripheral_status", mehci);
872 if (ret)
873 dev_err(&pdev->dev, "%s:request_irq:%d failed:%d",
874 __func__, mehci->peripheral_status_irq, ret);
875 }
876
Vamsi Krishna6921cbe2012-02-21 18:34:43 -0800877 /* configure wakeup irq */
878 ret = platform_get_irq(pdev, 2);
879 if (ret > 0) {
880 mehci->wakeup_irq = ret;
881 dev_dbg(&pdev->dev, "wakeup_irq: %d\n", mehci->wakeup_irq);
882 ret = request_irq(mehci->wakeup_irq, msm_hsic_wakeup_irq,
883 IRQF_TRIGGER_LOW,
884 "msm_hsic_wakeup", mehci);
885 if (!ret) {
886 disable_irq_nosync(mehci->wakeup_irq);
Vamsi Krishna6921cbe2012-02-21 18:34:43 -0800887 } else {
888 dev_err(&pdev->dev, "request_irq(%d) failed: %d\n",
889 mehci->wakeup_irq, ret);
890 mehci->wakeup_irq = 0;
891 }
892 }
893
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530894 /*
895 * This pdev->dev is assigned parent of root-hub by USB core,
896 * hence, runtime framework automatically calls this driver's
897 * runtime APIs based on root-hub's state.
898 */
899 pm_runtime_set_active(&pdev->dev);
900 pm_runtime_enable(&pdev->dev);
901
902 return 0;
903
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530904unconfig_gpio:
905 msm_hsic_config_gpios(mehci, 0);
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530906deinit_hub:
907 msm_hsic_config_hub(mehci, 0);
908free_xo_handle:
Lena Salman8c8ba382012-02-14 15:59:31 +0200909 if (pdata && pdata->hub_reset)
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530910 msm_xo_put(mehci->xo_handle);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530911deinit_vddcx:
912 msm_hsic_init_vddcx(mehci, 0);
913deinit_clocks:
914 msm_hsic_init_clocks(mehci, 0);
915unmap:
916 iounmap(hcd->regs);
917put_hcd:
918 usb_put_hcd(hcd);
919
920 return ret;
921}
922
923static int __devexit ehci_hsic_msm_remove(struct platform_device *pdev)
924{
925 struct usb_hcd *hcd = platform_get_drvdata(pdev);
926 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530927 struct msm_hsic_host_platform_data *pdata;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530928
Vamsi Krishna34f01582011-12-14 19:54:42 -0800929 if (mehci->peripheral_status_irq)
930 free_irq(mehci->peripheral_status_irq, mehci);
Jack Phamfe441ea2012-03-23 17:03:15 -0700931
Vamsi Krishna6921cbe2012-02-21 18:34:43 -0800932 if (mehci->wakeup_irq) {
Jack Phamfe441ea2012-03-23 17:03:15 -0700933 if (mehci->wakeup_irq_enabled)
934 disable_irq_wake(mehci->wakeup_irq);
Vamsi Krishna6921cbe2012-02-21 18:34:43 -0800935 free_irq(mehci->wakeup_irq, mehci);
936 }
Vamsi Krishna34f01582011-12-14 19:54:42 -0800937
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530938 device_init_wakeup(&pdev->dev, 0);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530939 pm_runtime_set_suspended(&pdev->dev);
940
941 usb_remove_hcd(hcd);
Vijayavardhan Vennapusae3316a12011-10-15 06:05:17 +0530942 msm_hsic_config_gpios(mehci, 0);
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530943 msm_hsic_config_hub(mehci, 0);
944 pdata = mehci->dev->platform_data;
Lena Salman8c8ba382012-02-14 15:59:31 +0200945 if (pdata && pdata->hub_reset)
Vijayavardhan Vennapusa2b592824f2011-11-02 19:51:32 +0530946 msm_xo_put(mehci->xo_handle);
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530947 msm_hsic_init_vddcx(mehci, 0);
948
949 msm_hsic_init_clocks(mehci, 0);
950 wake_lock_destroy(&mehci->wlock);
951 iounmap(hcd->regs);
952 usb_put_hcd(hcd);
953
954 return 0;
955}
956
957#ifdef CONFIG_PM_SLEEP
958static int msm_hsic_pm_suspend(struct device *dev)
959{
960 struct usb_hcd *hcd = dev_get_drvdata(dev);
961 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
962
963 dev_dbg(dev, "ehci-msm-hsic PM suspend\n");
964
965 if (device_may_wakeup(dev))
966 enable_irq_wake(hcd->irq);
967
968 return msm_hsic_suspend(mehci);
Jack Phamfe441ea2012-03-23 17:03:15 -0700969}
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530970
Jack Phamfe441ea2012-03-23 17:03:15 -0700971static int msm_hsic_pm_suspend_noirq(struct device *dev)
972{
973 struct usb_hcd *hcd = dev_get_drvdata(dev);
974 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
975
976 dev_dbg(dev, "ehci-msm-hsic PM suspend_noirq\n");
977
978 if (device_may_wakeup(dev) && !mehci->wakeup_irq_enabled) {
979 enable_irq(mehci->wakeup_irq);
980 enable_irq_wake(mehci->wakeup_irq);
981 mehci->wakeup_irq_enabled = 1;
982 }
983
984 return 0;
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +0530985}
986
987static int msm_hsic_pm_resume(struct device *dev)
988{
989 int ret;
990 struct usb_hcd *hcd = dev_get_drvdata(dev);
991 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
992
993 dev_dbg(dev, "ehci-msm-hsic PM resume\n");
994
995 if (device_may_wakeup(dev))
996 disable_irq_wake(hcd->irq);
997
Jack Phamfe441ea2012-03-23 17:03:15 -0700998 if (mehci->wakeup_irq_enabled) {
999 mehci->wakeup_irq_enabled = 0;
1000 disable_irq_wake(mehci->wakeup_irq);
1001 disable_irq_nosync(mehci->wakeup_irq);
1002 }
1003
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301004 ret = msm_hsic_resume(mehci);
1005 if (ret)
1006 return ret;
1007
1008 /* Bring the device to full powered state upon system resume */
1009 pm_runtime_disable(dev);
1010 pm_runtime_set_active(dev);
1011 pm_runtime_enable(dev);
1012
1013 return 0;
1014}
1015#endif
1016
1017#ifdef CONFIG_PM_RUNTIME
1018static int msm_hsic_runtime_idle(struct device *dev)
1019{
1020 dev_dbg(dev, "EHCI runtime idle\n");
1021
1022 return 0;
1023}
1024
1025static int msm_hsic_runtime_suspend(struct device *dev)
1026{
1027 struct usb_hcd *hcd = dev_get_drvdata(dev);
1028 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
1029
1030 dev_dbg(dev, "EHCI runtime suspend\n");
1031 return msm_hsic_suspend(mehci);
1032}
1033
1034static int msm_hsic_runtime_resume(struct device *dev)
1035{
1036 struct usb_hcd *hcd = dev_get_drvdata(dev);
1037 struct msm_hsic_hcd *mehci = hcd_to_hsic(hcd);
1038
1039 dev_dbg(dev, "EHCI runtime resume\n");
1040 return msm_hsic_resume(mehci);
1041}
1042#endif
1043
1044#ifdef CONFIG_PM
1045static const struct dev_pm_ops msm_hsic_dev_pm_ops = {
1046 SET_SYSTEM_SLEEP_PM_OPS(msm_hsic_pm_suspend, msm_hsic_pm_resume)
Jack Phamfe441ea2012-03-23 17:03:15 -07001047 .suspend_noirq = msm_hsic_pm_suspend_noirq,
Vijayavardhan Vennapusa39025fe2011-10-15 05:55:10 +05301048 SET_RUNTIME_PM_OPS(msm_hsic_runtime_suspend, msm_hsic_runtime_resume,
1049 msm_hsic_runtime_idle)
1050};
1051#endif
1052
1053static struct platform_driver ehci_msm_hsic_driver = {
1054 .probe = ehci_hsic_msm_probe,
1055 .remove = __devexit_p(ehci_hsic_msm_remove),
1056 .driver = {
1057 .name = "msm_hsic_host",
1058#ifdef CONFIG_PM
1059 .pm = &msm_hsic_dev_pm_ops,
1060#endif
1061 },
1062};