2010-11-10 12:03:18 +03:00
/*
* OHCI HCD ( Host Controller Driver ) for USB .
*
* Copyright ( C ) 2010 ST Microelectronics .
* Deepak Sikri < deepak . sikri @ st . com >
*
* Based on various ohci - * . c drivers
*
* This file is licensed under the terms of the GNU General Public
* License version 2. This program is licensed " as is " without any
* warranty of any kind , whether express or implied .
*/
# include <linux/clk.h>
2013-09-21 15:08:41 +04:00
# include <linux/dma-mapping.h>
# include <linux/io.h>
# include <linux/kernel.h>
# include <linux/module.h>
2012-04-16 07:38:08 +04:00
# include <linux/of.h>
2013-09-21 15:08:41 +04:00
# include <linux/platform_device.h>
# include <linux/signal.h>
# include <linux/usb.h>
# include <linux/usb/hcd.h>
# include "ohci.h"
2010-11-10 12:03:18 +03:00
2013-09-21 15:08:41 +04:00
# define DRIVER_DESC "OHCI SPEAr driver"
static const char hcd_name [ ] = " SPEAr-ohci " ;
2010-11-10 12:03:18 +03:00
struct spear_ohci {
struct clk * clk ;
} ;
2013-09-21 15:08:41 +04:00
# define to_spear_ohci(hcd) (struct spear_ohci *)(hcd_to_ohci(hcd)->priv)
2010-11-10 12:03:18 +03:00
2013-09-21 15:08:41 +04:00
static struct hc_driver __read_mostly ohci_spear_hc_driver ;
2010-11-10 12:03:18 +03:00
static int spear_ohci_hcd_drv_probe ( struct platform_device * pdev )
{
const struct hc_driver * driver = & ohci_spear_hc_driver ;
2013-09-21 15:08:41 +04:00
struct ohci_hcd * ohci ;
2010-11-10 12:03:18 +03:00
struct usb_hcd * hcd = NULL ;
struct clk * usbh_clk ;
2013-09-21 15:08:41 +04:00
struct spear_ohci * sohci_p ;
2010-11-10 12:03:18 +03:00
struct resource * res ;
int retval , irq ;
irq = platform_get_irq ( pdev , 0 ) ;
if ( irq < 0 ) {
retval = irq ;
2012-11-08 19:07:59 +04:00
goto fail ;
2010-11-10 12:03:18 +03:00
}
2012-04-16 07:38:08 +04:00
/*
* Right now device - tree probed devices don ' t get dma_mask set .
* Since shared usb code relies on it , set it here for now .
* Once we have dma capability bindings this can go away .
*/
2013-06-27 15:36:37 +04:00
retval = dma_coerce_mask_and_coherent ( & pdev - > dev , DMA_BIT_MASK ( 32 ) ) ;
2013-06-10 19:28:49 +04:00
if ( retval )
goto fail ;
2012-04-16 07:38:08 +04:00
2012-11-08 19:07:59 +04:00
usbh_clk = devm_clk_get ( & pdev - > dev , NULL ) ;
2010-11-10 12:03:18 +03:00
if ( IS_ERR ( usbh_clk ) ) {
dev_err ( & pdev - > dev , " Error getting interface clock \n " ) ;
retval = PTR_ERR ( usbh_clk ) ;
2012-11-08 19:07:59 +04:00
goto fail ;
2010-11-10 12:03:18 +03:00
}
hcd = usb_create_hcd ( driver , & pdev - > dev , dev_name ( & pdev - > dev ) ) ;
if ( ! hcd ) {
retval = - ENOMEM ;
2012-11-08 19:07:59 +04:00
goto fail ;
2010-11-10 12:03:18 +03:00
}
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2013-12-11 11:29:15 +04:00
hcd - > regs = devm_ioremap_resource ( & pdev - > dev , res ) ;
if ( IS_ERR ( hcd - > regs ) ) {
retval = PTR_ERR ( hcd - > regs ) ;
2012-11-08 19:07:59 +04:00
goto err_put_hcd ;
2010-11-10 12:03:18 +03:00
}
2014-11-04 05:21:19 +03:00
hcd - > rsrc_start = pdev - > resource [ 0 ] . start ;
hcd - > rsrc_len = resource_size ( res ) ;
2013-09-21 15:08:41 +04:00
sohci_p = to_spear_ohci ( hcd ) ;
sohci_p - > clk = usbh_clk ;
clk_prepare_enable ( sohci_p - > clk ) ;
ohci = hcd_to_ohci ( hcd ) ;
2010-11-10 12:03:18 +03:00
2011-09-07 12:10:52 +04:00
retval = usb_add_hcd ( hcd , platform_get_irq ( pdev , 0 ) , 0 ) ;
2013-11-05 06:46:02 +04:00
if ( retval = = 0 ) {
device_wakeup_enable ( hcd - > self . controller ) ;
2010-11-10 12:03:18 +03:00
return retval ;
2013-11-05 06:46:02 +04:00
}
2010-11-10 12:03:18 +03:00
2013-09-21 15:08:41 +04:00
clk_disable_unprepare ( sohci_p - > clk ) ;
2012-11-08 19:07:59 +04:00
err_put_hcd :
2010-11-10 12:03:18 +03:00
usb_put_hcd ( hcd ) ;
2012-11-08 19:07:59 +04:00
fail :
2010-11-10 12:03:18 +03:00
dev_err ( & pdev - > dev , " init fail, %d \n " , retval ) ;
return retval ;
}
static int spear_ohci_hcd_drv_remove ( struct platform_device * pdev )
{
struct usb_hcd * hcd = platform_get_drvdata ( pdev ) ;
2013-09-21 15:08:41 +04:00
struct spear_ohci * sohci_p = to_spear_ohci ( hcd ) ;
2010-11-10 12:03:18 +03:00
usb_remove_hcd ( hcd ) ;
2013-09-21 15:08:41 +04:00
if ( sohci_p - > clk )
clk_disable_unprepare ( sohci_p - > clk ) ;
2010-11-10 12:03:18 +03:00
usb_put_hcd ( hcd ) ;
return 0 ;
}
# if defined(CONFIG_PM)
2013-11-13 16:10:21 +04:00
static int spear_ohci_hcd_drv_suspend ( struct platform_device * pdev ,
2010-11-10 12:03:18 +03:00
pm_message_t message )
{
2013-11-13 16:10:21 +04:00
struct usb_hcd * hcd = platform_get_drvdata ( pdev ) ;
2010-11-10 12:03:18 +03:00
struct ohci_hcd * ohci = hcd_to_ohci ( hcd ) ;
2013-09-21 15:08:41 +04:00
struct spear_ohci * sohci_p = to_spear_ohci ( hcd ) ;
2013-11-13 16:10:21 +04:00
bool do_wakeup = device_may_wakeup ( & pdev - > dev ) ;
int ret ;
2010-11-10 12:03:18 +03:00
if ( time_before ( jiffies , ohci - > next_statechange ) )
msleep ( 5 ) ;
ohci - > next_statechange = jiffies ;
2013-11-13 16:10:21 +04:00
ret = ohci_suspend ( hcd , do_wakeup ) ;
if ( ret )
return ret ;
2013-09-21 15:08:41 +04:00
clk_disable_unprepare ( sohci_p - > clk ) ;
2013-11-13 16:10:21 +04:00
return ret ;
2010-11-10 12:03:18 +03:00
}
static int spear_ohci_hcd_drv_resume ( struct platform_device * dev )
{
struct usb_hcd * hcd = platform_get_drvdata ( dev ) ;
struct ohci_hcd * ohci = hcd_to_ohci ( hcd ) ;
2013-09-21 15:08:41 +04:00
struct spear_ohci * sohci_p = to_spear_ohci ( hcd ) ;
2010-11-10 12:03:18 +03:00
if ( time_before ( jiffies , ohci - > next_statechange ) )
msleep ( 5 ) ;
ohci - > next_statechange = jiffies ;
2013-09-21 15:08:41 +04:00
clk_prepare_enable ( sohci_p - > clk ) ;
2012-10-08 17:11:29 +04:00
ohci_resume ( hcd , false ) ;
2010-11-10 12:03:18 +03:00
return 0 ;
}
# endif
2014-06-18 08:38:10 +04:00
static const struct of_device_id spear_ohci_id_table [ ] = {
2012-04-16 07:38:08 +04:00
{ . compatible = " st,spear600-ohci " , } ,
{ } ,
} ;
2010-11-10 12:03:18 +03:00
/* Driver definition to register with the platform bus */
static struct platform_driver spear_ohci_hcd_driver = {
. probe = spear_ohci_hcd_drv_probe ,
. remove = spear_ohci_hcd_drv_remove ,
# ifdef CONFIG_PM
. suspend = spear_ohci_hcd_drv_suspend ,
. resume = spear_ohci_hcd_drv_resume ,
# endif
. driver = {
. owner = THIS_MODULE ,
. name = " spear-ohci " ,
2013-05-21 15:47:20 +04:00
. of_match_table = spear_ohci_id_table ,
2010-11-10 12:03:18 +03:00
} ,
} ;
2013-09-21 15:08:41 +04:00
static const struct ohci_driver_overrides spear_overrides __initconst = {
. extra_priv_size = sizeof ( struct spear_ohci ) ,
} ;
static int __init ohci_spear_init ( void )
{
if ( usb_disabled ( ) )
return - ENODEV ;
pr_info ( " %s: " DRIVER_DESC " \n " , hcd_name ) ;
ohci_init_driver ( & ohci_spear_hc_driver , & spear_overrides ) ;
return platform_driver_register ( & spear_ohci_hcd_driver ) ;
}
module_init ( ohci_spear_init ) ;
static void __exit ohci_spear_cleanup ( void )
{
platform_driver_unregister ( & spear_ohci_hcd_driver ) ;
}
module_exit ( ohci_spear_cleanup ) ;
MODULE_DESCRIPTION ( DRIVER_DESC ) ;
MODULE_AUTHOR ( " Deepak Sikri " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
2010-11-10 12:03:18 +03:00
MODULE_ALIAS ( " platform:spear-ohci " ) ;