2005-07-29 12:18:03 -07:00
/*
* OHCI HCD ( Host Controller Driver ) for USB .
*
* ( C ) Copyright 1999 Roman Weissgaerber < weissg @ vienna . at >
* ( C ) Copyright 2000 - 2002 David Brownell < dbrownell @ users . sourceforge . net >
* ( C ) Copyright 2002 Hewlett - Packard Company
*
* USB Bus Glue for Samsung S3C2410
*
* Written by Christopher Hoover < ch @ hpl . hp . com >
2008-07-10 17:30:46 -07:00
* Based on fragments of previous driver by Russell King et al .
2005-07-29 12:18:03 -07:00
*
* Modified for S3C2410 from ohci - sa1111 . c , ohci - omap . c and ohci - lh7a40 . c
* by Ben Dooks , < ben @ simtec . co . uk >
* Copyright ( C ) 2004 Simtec Electronics
*
* Thanks to basprog @ mail . ru for updates to newer kernels
*
* This file is licenced under the GPL .
*/
2005-10-29 19:07:23 +01:00
# include <linux/platform_device.h>
2006-01-07 16:15:52 +00:00
# include <linux/clk.h>
2012-08-24 15:22:12 +02:00
# include <linux/platform_data/usb-ohci-s3c2410.h>
2005-07-29 12:18:03 -07:00
# define valid_port(idx) ((idx) == 1 || (idx) == 2)
/* clock device associated with the hcd */
static struct clk * clk ;
2006-04-02 01:45:00 +01:00
static struct clk * usb_clk ;
2005-07-29 12:18:03 -07:00
/* forward definitions */
static void s3c2410_hcd_oc ( struct s3c2410_hcd_info * info , int port_oc ) ;
/* conversion functions */
2006-03-21 22:54:47 +00:00
static struct s3c2410_hcd_info * to_s3c2410_info ( struct usb_hcd * hcd )
2005-07-29 12:18:03 -07:00
{
return hcd - > self . controller - > platform_data ;
}
static void s3c2410_start_hc ( struct platform_device * dev , struct usb_hcd * hcd )
{
struct s3c2410_hcd_info * info = dev - > dev . platform_data ;
dev_dbg ( & dev - > dev , " s3c2410_start_hc: \n " ) ;
2006-04-02 01:45:00 +01:00
clk_enable ( usb_clk ) ;
mdelay ( 2 ) ; /* let the bus clock stabilise */
2005-07-29 12:18:03 -07:00
clk_enable ( clk ) ;
if ( info ! = NULL ) {
info - > hcd = hcd ;
info - > report_oc = s3c2410_hcd_oc ;
2011-05-04 16:45:47 +09:00
if ( info - > enable_oc ! = NULL )
2005-07-29 12:18:03 -07:00
( info - > enable_oc ) ( info , 1 ) ;
}
}
static void s3c2410_stop_hc ( struct platform_device * dev )
{
struct s3c2410_hcd_info * info = dev - > dev . platform_data ;
dev_dbg ( & dev - > dev , " s3c2410_stop_hc: \n " ) ;
if ( info ! = NULL ) {
info - > report_oc = NULL ;
info - > hcd = NULL ;
2011-05-04 16:45:47 +09:00
if ( info - > enable_oc ! = NULL )
2005-07-29 12:18:03 -07:00
( info - > enable_oc ) ( info , 0 ) ;
}
clk_disable ( clk ) ;
2006-04-02 01:45:00 +01:00
clk_disable ( usb_clk ) ;
2005-07-29 12:18:03 -07:00
}
/* ohci_s3c2410_hub_status_data
*
* update the status data from the hub with anything that
* has been detected by our system
*/
static int
2011-05-04 16:45:47 +09:00
ohci_s3c2410_hub_status_data ( struct usb_hcd * hcd , char * buf )
2005-07-29 12:18:03 -07:00
{
struct s3c2410_hcd_info * info = to_s3c2410_info ( hcd ) ;
struct s3c2410_hcd_port * port ;
int orig ;
int portno ;
2011-05-04 16:45:47 +09:00
orig = ohci_hub_status_data ( hcd , buf ) ;
2005-07-29 12:18:03 -07:00
if ( info = = NULL )
return orig ;
port = & info - > port [ 0 ] ;
/* mark any changed port as changed */
for ( portno = 0 ; portno < 2 ; port + + , portno + + ) {
if ( port - > oc_changed = = 1 & &
port - > flags & S3C_HCDFLG_USED ) {
dev_dbg ( hcd - > self . controller ,
" oc change on port %d \n " , portno ) ;
if ( orig < 1 )
orig = 1 ;
buf [ 0 ] | = 1 < < ( portno + 1 ) ;
}
}
return orig ;
}
/* s3c2410_usb_set_power
*
* configure the power on a port , by calling the platform device
* routine registered with the platform device
*/
static void s3c2410_usb_set_power ( struct s3c2410_hcd_info * info ,
int port , int to )
{
if ( info = = NULL )
return ;
if ( info - > power_control ! = NULL ) {
info - > port [ port - 1 ] . power = to ;
2005-08-09 15:04:00 +01:00
( info - > power_control ) ( port - 1 , to ) ;
2005-07-29 12:18:03 -07:00
}
}
/* ohci_s3c2410_hub_control
*
* look at control requests to the hub , and see if we need
* to take any action or over - ride the results from the
* request .
*/
2011-05-04 16:45:47 +09:00
static int ohci_s3c2410_hub_control (
2005-07-29 12:18:03 -07:00
struct usb_hcd * hcd ,
u16 typeReq ,
u16 wValue ,
u16 wIndex ,
char * buf ,
u16 wLength )
{
struct s3c2410_hcd_info * info = to_s3c2410_info ( hcd ) ;
struct usb_hub_descriptor * desc ;
int ret = - EINVAL ;
u32 * data = ( u32 * ) buf ;
dev_dbg ( hcd - > self . controller ,
" s3c2410_hub_control(%p,0x%04x,0x%04x,0x%04x,%p,%04x) \n " ,
hcd , typeReq , wValue , wIndex , buf , wLength ) ;
2006-03-28 01:56:53 -08:00
/* if we are only an humble host without any special capabilities
2005-07-29 12:18:03 -07:00
* process the request straight away and exit */
if ( info = = NULL ) {
ret = ohci_hub_control ( hcd , typeReq , wValue ,
wIndex , buf , wLength ) ;
goto out ;
}
/* check the request to see if it needs handling */
switch ( typeReq ) {
case SetPortFeature :
if ( wValue = = USB_PORT_FEAT_POWER ) {
dev_dbg ( hcd - > self . controller , " SetPortFeat: POWER \n " ) ;
s3c2410_usb_set_power ( info , wIndex , 1 ) ;
goto out ;
}
break ;
case ClearPortFeature :
switch ( wValue ) {
case USB_PORT_FEAT_C_OVER_CURRENT :
dev_dbg ( hcd - > self . controller ,
" ClearPortFeature: C_OVER_CURRENT \n " ) ;
if ( valid_port ( wIndex ) ) {
info - > port [ wIndex - 1 ] . oc_changed = 0 ;
info - > port [ wIndex - 1 ] . oc_status = 0 ;
}
goto out ;
case USB_PORT_FEAT_OVER_CURRENT :
dev_dbg ( hcd - > self . controller ,
" ClearPortFeature: OVER_CURRENT \n " ) ;
2011-05-04 16:45:47 +09:00
if ( valid_port ( wIndex ) )
2005-07-29 12:18:03 -07:00
info - > port [ wIndex - 1 ] . oc_status = 0 ;
goto out ;
case USB_PORT_FEAT_POWER :
dev_dbg ( hcd - > self . controller ,
" ClearPortFeature: POWER \n " ) ;
if ( valid_port ( wIndex ) ) {
s3c2410_usb_set_power ( info , wIndex , 0 ) ;
return 0 ;
}
}
break ;
}
ret = ohci_hub_control ( hcd , typeReq , wValue , wIndex , buf , wLength ) ;
if ( ret )
goto out ;
switch ( typeReq ) {
case GetHubDescriptor :
/* update the hub's descriptor */
desc = ( struct usb_hub_descriptor * ) buf ;
if ( info - > power_control = = NULL )
return ret ;
dev_dbg ( hcd - > self . controller , " wHubCharacteristics 0x%04x \n " ,
desc - > wHubCharacteristics ) ;
/* remove the old configurations for power-switching, and
* over - current protection , and insert our new configuration
*/
desc - > wHubCharacteristics & = ~ cpu_to_le16 ( HUB_CHAR_LPSM ) ;
desc - > wHubCharacteristics | = cpu_to_le16 ( 0x0001 ) ;
if ( info - > enable_oc ) {
2011-05-04 16:45:47 +09:00
desc - > wHubCharacteristics & = ~ cpu_to_le16 (
HUB_CHAR_OCPM ) ;
desc - > wHubCharacteristics | = cpu_to_le16 (
0x0008 |
0x0001 ) ;
2005-07-29 12:18:03 -07:00
}
dev_dbg ( hcd - > self . controller , " wHubCharacteristics after 0x%04x \n " ,
desc - > wHubCharacteristics ) ;
return ret ;
case GetPortStatus :
/* check port status */
dev_dbg ( hcd - > self . controller , " GetPortStatus(%d) \n " , wIndex ) ;
if ( valid_port ( wIndex ) ) {
2011-05-04 16:45:47 +09:00
if ( info - > port [ wIndex - 1 ] . oc_changed )
2005-07-29 12:18:03 -07:00
* data | = cpu_to_le32 ( RH_PS_OCIC ) ;
2011-05-04 16:45:47 +09:00
if ( info - > port [ wIndex - 1 ] . oc_status )
2005-07-29 12:18:03 -07:00
* data | = cpu_to_le32 ( RH_PS_POCI ) ;
}
}
out :
return ret ;
}
/* s3c2410_hcd_oc
*
* handle an over - current report
*/
static void s3c2410_hcd_oc ( struct s3c2410_hcd_info * info , int port_oc )
{
struct s3c2410_hcd_port * port ;
struct usb_hcd * hcd ;
unsigned long flags ;
int portno ;
if ( info = = NULL )
return ;
port = & info - > port [ 0 ] ;
hcd = info - > hcd ;
local_irq_save ( flags ) ;
for ( portno = 0 ; portno < 2 ; port + + , portno + + ) {
if ( port_oc & ( 1 < < portno ) & &
port - > flags & S3C_HCDFLG_USED ) {
port - > oc_status = 1 ;
port - > oc_changed = 1 ;
/* ok, once over-current is detected,
the port needs to be powered down */
s3c2410_usb_set_power ( info , portno + 1 , 0 ) ;
}
}
local_irq_restore ( flags ) ;
}
/* may be called without controller electrically present */
/* may be called with controller, bus, and devices active */
/*
* usb_hcd_s3c2410_remove - shutdown processing for HCD
* @ dev : USB Host Controller being removed
* Context : ! in_interrupt ( )
*
* Reverses the effect of usb_hcd_3c2410_probe ( ) , first invoking
* the HCD ' s stop ( ) method . It is always called from a thread
* context , normally " rmmod " , " apmd " , or something similar .
*
*/
2006-03-21 22:54:47 +00:00
static void
2011-05-04 16:45:47 +09:00
usb_hcd_s3c2410_remove ( struct usb_hcd * hcd , struct platform_device * dev )
2005-07-29 12:18:03 -07:00
{
usb_remove_hcd ( hcd ) ;
s3c2410_stop_hc ( dev ) ;
usb_put_hcd ( hcd ) ;
}
/**
* usb_hcd_s3c2410_probe - initialize S3C2410 - based HCDs
* Context : ! in_interrupt ( )
*
* Allocates basic resources for this USB host controller , and
* then invokes the start ( ) method for the HCD associated with it
* through the hotplug entry ' s driver_data .
*
*/
2011-05-04 16:45:47 +09:00
static int usb_hcd_s3c2410_probe ( const struct hc_driver * driver ,
2006-03-21 22:54:47 +00:00
struct platform_device * dev )
2005-07-29 12:18:03 -07:00
{
struct usb_hcd * hcd = NULL ;
int retval ;
s3c2410_usb_set_power ( dev - > dev . platform_data , 1 , 1 ) ;
2005-08-09 15:04:00 +01:00
s3c2410_usb_set_power ( dev - > dev . platform_data , 2 , 1 ) ;
2005-07-29 12:18:03 -07:00
hcd = usb_create_hcd ( driver , & dev - > dev , " s3c24xx " ) ;
if ( hcd = = NULL )
return - ENOMEM ;
hcd - > rsrc_start = dev - > resource [ 0 ] . start ;
2011-04-14 21:09:16 +09:00
hcd - > rsrc_len = resource_size ( & dev - > resource [ 0 ] ) ;
2005-07-29 12:18:03 -07:00
2012-10-08 11:28:25 +09:00
hcd - > regs = devm_request_and_ioremap ( & dev - > dev , & dev - > resource [ 0 ] ) ;
if ( ! hcd - > regs ) {
dev_err ( & dev - > dev , " devm_request_and_ioremap failed \n " ) ;
retval = - ENOMEM ;
2006-04-02 01:45:00 +01:00
goto err_put ;
2005-07-29 12:18:03 -07:00
}
2012-10-08 11:28:25 +09:00
clk = devm_clk_get ( & dev - > dev , " usb-host " ) ;
2005-07-29 12:18:03 -07:00
if ( IS_ERR ( clk ) ) {
dev_err ( & dev - > dev , " cannot get usb-host clock \n " ) ;
2011-05-05 08:46:07 +09:00
retval = PTR_ERR ( clk ) ;
2012-10-08 11:28:25 +09:00
goto err_put ;
2006-04-02 01:45:00 +01:00
}
2012-10-08 11:28:25 +09:00
usb_clk = devm_clk_get ( & dev - > dev , " usb-bus-host " ) ;
2006-04-02 01:45:00 +01:00
if ( IS_ERR ( usb_clk ) ) {
2009-02-26 23:03:15 +00:00
dev_err ( & dev - > dev , " cannot get usb-bus-host clock \n " ) ;
2011-05-05 08:46:07 +09:00
retval = PTR_ERR ( usb_clk ) ;
2012-10-08 11:28:25 +09:00
goto err_put ;
2005-07-29 12:18:03 -07:00
}
s3c2410_start_hc ( dev , hcd ) ;
ohci_hcd_init ( hcd_to_ohci ( hcd ) ) ;
2011-09-07 16:10:52 +08:00
retval = usb_add_hcd ( hcd , dev - > resource [ 1 ] . start , 0 ) ;
2005-07-29 12:18:03 -07:00
if ( retval ! = 0 )
2006-04-02 01:45:00 +01:00
goto err_ioremap ;
2005-07-29 12:18:03 -07:00
return 0 ;
2006-04-02 01:45:00 +01:00
err_ioremap :
2005-07-29 12:18:03 -07:00
s3c2410_stop_hc ( dev ) ;
2006-04-02 01:45:00 +01:00
err_put :
2005-07-29 12:18:03 -07:00
usb_put_hcd ( hcd ) ;
return retval ;
}
/*-------------------------------------------------------------------------*/
static int
2011-05-04 16:45:47 +09:00
ohci_s3c2410_start ( struct usb_hcd * hcd )
2005-07-29 12:18:03 -07:00
{
2011-05-04 16:45:47 +09:00
struct ohci_hcd * ohci = hcd_to_ohci ( hcd ) ;
2005-07-29 12:18:03 -07:00
int ret ;
2011-05-04 16:45:47 +09:00
ret = ohci_init ( ohci ) ;
if ( ret < 0 )
2005-07-29 12:18:03 -07:00
return ret ;
2011-05-04 16:45:47 +09:00
ret = ohci_run ( ohci ) ;
if ( ret < 0 ) {
2012-04-27 11:24:43 -07:00
dev_err ( hcd - > self . controller , " can't start %s \n " ,
hcd - > self . bus_name ) ;
2011-05-04 16:45:47 +09:00
ohci_stop ( hcd ) ;
2005-07-29 12:18:03 -07:00
return ret ;
}
return 0 ;
}
static const struct hc_driver ohci_s3c2410_hc_driver = {
. description = hcd_name ,
. product_desc = " S3C24XX OHCI " ,
. hcd_priv_size = sizeof ( struct ohci_hcd ) ,
/*
* generic hardware linkage
*/
. irq = ohci_irq ,
. flags = HCD_USB11 | HCD_MEMORY ,
/*
* basic lifecycle operations
*/
. start = ohci_s3c2410_start ,
. stop = ohci_stop ,
2006-12-05 03:18:31 -08:00
. shutdown = ohci_shutdown ,
2005-07-29 12:18:03 -07:00
/*
* managing i / o requests and associated device resources
*/
. urb_enqueue = ohci_urb_enqueue ,
. urb_dequeue = ohci_urb_dequeue ,
. endpoint_disable = ohci_endpoint_disable ,
/*
* scheduling support
*/
. get_frame_number = ohci_get_frame ,
/*
* root hub support
*/
. hub_status_data = ohci_s3c2410_hub_status_data ,
. hub_control = ohci_s3c2410_hub_control ,
2005-09-13 19:59:11 -07:00
# ifdef CONFIG_PM
2005-10-13 17:08:02 -04:00
. bus_suspend = ohci_bus_suspend ,
. bus_resume = ohci_bus_resume ,
2005-07-29 12:18:03 -07:00
# endif
2005-09-22 22:32:11 -07:00
. start_port_reset = ohci_start_port_reset ,
2005-07-29 12:18:03 -07:00
} ;
/* device driver */
2011-04-14 21:09:37 +09:00
static int __devinit ohci_hcd_s3c2410_drv_probe ( struct platform_device * pdev )
2005-07-29 12:18:03 -07:00
{
return usb_hcd_s3c2410_probe ( & ohci_s3c2410_hc_driver , pdev ) ;
}
2011-04-14 21:09:37 +09:00
static int __devexit ohci_hcd_s3c2410_drv_remove ( struct platform_device * pdev )
2005-07-29 12:18:03 -07:00
{
2005-11-09 22:32:44 +00:00
struct usb_hcd * hcd = platform_get_drvdata ( pdev ) ;
2005-07-29 12:18:03 -07:00
usb_hcd_s3c2410_remove ( hcd , pdev ) ;
return 0 ;
}
2011-11-28 15:56:06 +09:00
# ifdef CONFIG_PM
static int ohci_hcd_s3c2410_drv_suspend ( struct device * dev )
{
struct usb_hcd * hcd = dev_get_drvdata ( dev ) ;
struct ohci_hcd * ohci = hcd_to_ohci ( hcd ) ;
struct platform_device * pdev = to_platform_device ( dev ) ;
unsigned long flags ;
int rc = 0 ;
/*
* Root hub was already suspended . Disable irq emission and
* mark HW unaccessible , bail out if RH has been resumed . Use
* the spinlock to properly synchronize with possible pending
* RH suspend or resume activity .
*/
spin_lock_irqsave ( & ohci - > lock , flags ) ;
if ( ohci - > rh_state ! = OHCI_RH_SUSPENDED ) {
rc = - EINVAL ;
goto bail ;
}
clear_bit ( HCD_FLAG_HW_ACCESSIBLE , & hcd - > flags ) ;
s3c2410_stop_hc ( pdev ) ;
bail :
spin_unlock_irqrestore ( & ohci - > lock , flags ) ;
return rc ;
}
static int ohci_hcd_s3c2410_drv_resume ( struct device * dev )
{
struct usb_hcd * hcd = dev_get_drvdata ( dev ) ;
struct platform_device * pdev = to_platform_device ( dev ) ;
s3c2410_start_hc ( pdev , hcd ) ;
2012-10-08 15:11:29 +02:00
ohci_resume ( hcd , false ) ;
2011-11-28 15:56:06 +09:00
return 0 ;
}
# else
# define ohci_hcd_s3c2410_drv_suspend NULL
# define ohci_hcd_s3c2410_drv_resume NULL
# endif
static const struct dev_pm_ops ohci_hcd_s3c2410_pm_ops = {
. suspend = ohci_hcd_s3c2410_drv_suspend ,
. resume = ohci_hcd_s3c2410_drv_resume ,
} ;
2005-11-09 22:32:44 +00:00
static struct platform_driver ohci_hcd_s3c2410_driver = {
2005-07-29 12:18:03 -07:00
. probe = ohci_hcd_s3c2410_drv_probe ,
2011-04-14 21:09:37 +09:00
. remove = __devexit_p ( ohci_hcd_s3c2410_drv_remove ) ,
2006-12-05 03:18:31 -08:00
. shutdown = usb_hcd_platform_shutdown ,
2005-11-09 22:32:44 +00:00
. driver = {
. owner = THIS_MODULE ,
. name = " s3c2410-ohci " ,
2011-11-28 15:56:06 +09:00
. pm = & ohci_hcd_s3c2410_pm_ops ,
2005-11-09 22:32:44 +00:00
} ,
2005-07-29 12:18:03 -07:00
} ;
2008-04-10 21:29:22 -07:00
MODULE_ALIAS ( " platform:s3c2410-ohci " ) ;