Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/mason/linux...
[pandora-kernel.git] / drivers / usb / musb / ux500.c
1 /*
2  * Copyright (C) 2010 ST-Ericsson AB
3  * Mian Yousaf Kaukab <mian.yousaf.kaukab@stericsson.com>
4  *
5  * Based on omap2430.c
6  *
7  * This program is free software; you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation; either version 2 of the License, or
10  * (at your option) any later version.
11  *
12  * This program is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with this program; if not, write to the Free Software
19  * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
20  */
21
22 #include <linux/module.h>
23 #include <linux/kernel.h>
24 #include <linux/init.h>
25 #include <linux/clk.h>
26 #include <linux/err.h>
27 #include <linux/io.h>
28 #include <linux/platform_device.h>
29
30 #include "musb_core.h"
31
32 struct ux500_glue {
33         struct device           *dev;
34         struct platform_device  *musb;
35         struct clk              *clk;
36 };
37 #define glue_to_musb(g) platform_get_drvdata(g->musb)
38
39 static int ux500_musb_init(struct musb *musb)
40 {
41         musb->xceiv = usb_get_phy(USB_PHY_TYPE_USB2);
42         if (IS_ERR_OR_NULL(musb->xceiv)) {
43                 pr_err("HS USB OTG: no transceiver configured\n");
44                 return -ENODEV;
45         }
46
47         return 0;
48 }
49
50 static int ux500_musb_exit(struct musb *musb)
51 {
52         usb_put_phy(musb->xceiv);
53
54         return 0;
55 }
56
57 static const struct musb_platform_ops ux500_ops = {
58         .init           = ux500_musb_init,
59         .exit           = ux500_musb_exit,
60 };
61
62 static int __devinit ux500_probe(struct platform_device *pdev)
63 {
64         struct musb_hdrc_platform_data  *pdata = pdev->dev.platform_data;
65         struct platform_device          *musb;
66         struct ux500_glue               *glue;
67         struct clk                      *clk;
68
69         int                             ret = -ENOMEM;
70
71         glue = kzalloc(sizeof(*glue), GFP_KERNEL);
72         if (!glue) {
73                 dev_err(&pdev->dev, "failed to allocate glue context\n");
74                 goto err0;
75         }
76
77         musb = platform_device_alloc("musb-hdrc", -1);
78         if (!musb) {
79                 dev_err(&pdev->dev, "failed to allocate musb device\n");
80                 goto err1;
81         }
82
83         clk = clk_get(&pdev->dev, "usb");
84         if (IS_ERR(clk)) {
85                 dev_err(&pdev->dev, "failed to get clock\n");
86                 ret = PTR_ERR(clk);
87                 goto err2;
88         }
89
90         ret = clk_enable(clk);
91         if (ret) {
92                 dev_err(&pdev->dev, "failed to enable clock\n");
93                 goto err3;
94         }
95
96         musb->dev.parent                = &pdev->dev;
97         musb->dev.dma_mask              = pdev->dev.dma_mask;
98         musb->dev.coherent_dma_mask     = pdev->dev.coherent_dma_mask;
99
100         glue->dev                       = &pdev->dev;
101         glue->musb                      = musb;
102         glue->clk                       = clk;
103
104         pdata->platform_ops             = &ux500_ops;
105
106         platform_set_drvdata(pdev, glue);
107
108         ret = platform_device_add_resources(musb, pdev->resource,
109                         pdev->num_resources);
110         if (ret) {
111                 dev_err(&pdev->dev, "failed to add resources\n");
112                 goto err4;
113         }
114
115         ret = platform_device_add_data(musb, pdata, sizeof(*pdata));
116         if (ret) {
117                 dev_err(&pdev->dev, "failed to add platform_data\n");
118                 goto err4;
119         }
120
121         ret = platform_device_add(musb);
122         if (ret) {
123                 dev_err(&pdev->dev, "failed to register musb device\n");
124                 goto err4;
125         }
126
127         return 0;
128
129 err4:
130         clk_disable(clk);
131
132 err3:
133         clk_put(clk);
134
135 err2:
136         platform_device_put(musb);
137
138 err1:
139         kfree(glue);
140
141 err0:
142         return ret;
143 }
144
145 static int __devexit ux500_remove(struct platform_device *pdev)
146 {
147         struct ux500_glue       *glue = platform_get_drvdata(pdev);
148
149         platform_device_del(glue->musb);
150         platform_device_put(glue->musb);
151         clk_disable(glue->clk);
152         clk_put(glue->clk);
153         kfree(glue);
154
155         return 0;
156 }
157
158 #ifdef CONFIG_PM
159 static int ux500_suspend(struct device *dev)
160 {
161         struct ux500_glue       *glue = dev_get_drvdata(dev);
162         struct musb             *musb = glue_to_musb(glue);
163
164         usb_phy_set_suspend(musb->xceiv, 1);
165         clk_disable(glue->clk);
166
167         return 0;
168 }
169
170 static int ux500_resume(struct device *dev)
171 {
172         struct ux500_glue       *glue = dev_get_drvdata(dev);
173         struct musb             *musb = glue_to_musb(glue);
174         int                     ret;
175
176         ret = clk_enable(glue->clk);
177         if (ret) {
178                 dev_err(dev, "failed to enable clock\n");
179                 return ret;
180         }
181
182         usb_phy_set_suspend(musb->xceiv, 0);
183
184         return 0;
185 }
186
187 static const struct dev_pm_ops ux500_pm_ops = {
188         .suspend        = ux500_suspend,
189         .resume         = ux500_resume,
190 };
191
192 #define DEV_PM_OPS      (&ux500_pm_ops)
193 #else
194 #define DEV_PM_OPS      NULL
195 #endif
196
197 static struct platform_driver ux500_driver = {
198         .probe          = ux500_probe,
199         .remove         = __devexit_p(ux500_remove),
200         .driver         = {
201                 .name   = "musb-ux500",
202                 .pm     = DEV_PM_OPS,
203         },
204 };
205
206 MODULE_DESCRIPTION("UX500 MUSB Glue Layer");
207 MODULE_AUTHOR("Mian Yousaf Kaukab <mian.yousaf.kaukab@stericsson.com>");
208 MODULE_LICENSE("GPL v2");
209
210 static int __init ux500_init(void)
211 {
212         return platform_driver_register(&ux500_driver);
213 }
214 module_init(ux500_init);
215
216 static void __exit ux500_exit(void)
217 {
218         platform_driver_unregister(&ux500_driver);
219 }
220 module_exit(ux500_exit);