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); 221