2011-03-09 16:28:56 -08:00
/*
* EHCI - compliant USB host controller driver for NVIDIA Tegra SoCs
*
* Copyright ( C ) 2010 Google , Inc .
2013-01-17 20:15:37 +00:00
* Copyright ( C ) 2009 - 2013 NVIDIA Corporation
2011-03-09 16:28:56 -08:00
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation ; either version 2 of the License , or ( at your
* option ) any later version .
*
* This program is distributed in the hope that it will be useful , but WITHOUT
* ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE . See the GNU General Public License for
* more details .
*
*/
# include <linux/clk.h>
2012-06-26 17:40:32 +05:30
# include <linux/err.h>
2011-03-09 16:28:56 -08:00
# include <linux/platform_device.h>
# include <linux/platform_data/tegra_usb.h>
# include <linux/irq.h>
# include <linux/usb/otg.h>
2011-11-04 09:12:40 +00:00
# include <linux/gpio.h>
# include <linux/of.h>
# include <linux/of_gpio.h>
2012-05-01 11:28:49 -04:00
# include <linux/pm_runtime.h>
2013-01-17 20:15:37 +00:00
# include <linux/usb/ehci_def.h>
2012-09-05 18:50:23 +05:30
# include <linux/usb/tegra_usb_phy.h>
2013-04-03 16:11:12 +05:30
# include <linux/clk/tegra.h>
2012-10-02 16:49:25 -06:00
# define TEGRA_USB_BASE 0xC5000000
# define TEGRA_USB2_BASE 0xC5004000
# define TEGRA_USB3_BASE 0xC5008000
2011-03-09 16:28:56 -08:00
2013-01-17 20:15:37 +00:00
/* PORTSC registers */
# define TEGRA_USB_PORTSC1 0x184
# define TEGRA_USB_PORTSC1_PTS(x) (((x) & 0x3) << 30)
# define TEGRA_USB_PORTSC1_PHCD (1 << 23)
2011-03-09 16:28:57 -08:00
# define TEGRA_USB_DMA_ALIGN 32
2011-03-09 16:28:56 -08:00
struct tegra_ehci_hcd {
struct ehci_hcd * ehci ;
struct tegra_usb_phy * phy ;
struct clk * clk ;
2012-02-13 13:24:02 +02:00
struct usb_phy * transceiver ;
2011-03-09 16:28:56 -08:00
int host_resumed ;
int port_resuming ;
2012-12-13 20:59:08 +00:00
bool needs_double_reset ;
2011-03-09 16:28:56 -08:00
enum tegra_usb_phy_port_speed port_speed ;
} ;
static void tegra_ehci_power_up ( struct usb_hcd * hcd )
{
struct tegra_ehci_hcd * tegra = dev_get_drvdata ( hcd - > self . controller ) ;
2012-06-05 09:59:38 +05:30
clk_prepare_enable ( tegra - > clk ) ;
2013-01-24 15:57:03 +05:30
usb_phy_set_suspend ( hcd - > phy , 0 ) ;
2011-03-09 16:28:56 -08:00
tegra - > host_resumed = 1 ;
}
static void tegra_ehci_power_down ( struct usb_hcd * hcd )
{
struct tegra_ehci_hcd * tegra = dev_get_drvdata ( hcd - > self . controller ) ;
tegra - > host_resumed = 0 ;
2013-01-24 15:57:03 +05:30
usb_phy_set_suspend ( hcd - > phy , 1 ) ;
2012-06-05 09:59:38 +05:30
clk_disable_unprepare ( tegra - > clk ) ;
2011-03-09 16:28:56 -08:00
}
2011-04-17 11:58:25 +03:00
static int tegra_ehci_internal_port_reset (
struct ehci_hcd * ehci ,
u32 __iomem * portsc_reg
)
{
u32 temp ;
unsigned long flags ;
int retval = 0 ;
int i , tries ;
u32 saved_usbintr ;
spin_lock_irqsave ( & ehci - > lock , flags ) ;
saved_usbintr = ehci_readl ( ehci , & ehci - > regs - > intr_enable ) ;
/* disable USB interrupt */
ehci_writel ( ehci , 0 , & ehci - > regs - > intr_enable ) ;
spin_unlock_irqrestore ( & ehci - > lock , flags ) ;
/*
* Here we have to do Port Reset at most twice for
* Port Enable bit to be set .
*/
for ( i = 0 ; i < 2 ; i + + ) {
temp = ehci_readl ( ehci , portsc_reg ) ;
temp | = PORT_RESET ;
ehci_writel ( ehci , temp , portsc_reg ) ;
mdelay ( 10 ) ;
temp & = ~ PORT_RESET ;
ehci_writel ( ehci , temp , portsc_reg ) ;
mdelay ( 1 ) ;
tries = 100 ;
do {
mdelay ( 1 ) ;
/*
* Up to this point , Port Enable bit is
* expected to be set after 2 ms waiting .
* USB1 usually takes extra 45 ms , for safety ,
* we take 100 ms as timeout .
*/
temp = ehci_readl ( ehci , portsc_reg ) ;
} while ( ! ( temp & PORT_PE ) & & tries - - ) ;
if ( temp & PORT_PE )
break ;
}
if ( i = = 2 )
retval = - ETIMEDOUT ;
/*
* Clear Connect Status Change bit if it ' s set .
* We can ' t clear PORT_PEC . It will also cause PORT_PE to be cleared .
*/
if ( temp & PORT_CSC )
ehci_writel ( ehci , PORT_CSC , portsc_reg ) ;
/*
* Write to clear any interrupt status bits that might be set
* during port reset .
*/
temp = ehci_readl ( ehci , & ehci - > regs - > status ) ;
ehci_writel ( ehci , temp , & ehci - > regs - > status ) ;
/* restore original interrupt enable bits */
ehci_writel ( ehci , saved_usbintr , & ehci - > regs - > intr_enable ) ;
return retval ;
}
2011-03-09 16:28:56 -08:00
static int tegra_ehci_hub_control (
struct usb_hcd * hcd ,
u16 typeReq ,
u16 wValue ,
u16 wIndex ,
char * buf ,
u16 wLength
)
{
struct ehci_hcd * ehci = hcd_to_ehci ( hcd ) ;
struct tegra_ehci_hcd * tegra = dev_get_drvdata ( hcd - > self . controller ) ;
u32 __iomem * status_reg ;
u32 temp ;
unsigned long flags ;
int retval = 0 ;
status_reg = & ehci - > regs - > port_status [ ( wIndex & 0xff ) - 1 ] ;
spin_lock_irqsave ( & ehci - > lock , flags ) ;
2012-04-18 15:32:46 -06:00
if ( typeReq = = GetPortStatus ) {
2011-03-09 16:28:56 -08:00
temp = ehci_readl ( ehci , status_reg ) ;
if ( tegra - > port_resuming & & ! ( temp & PORT_SUSPEND ) ) {
/* Resume completed, re-enable disconnect detection */
tegra - > port_resuming = 0 ;
2013-01-24 15:57:03 +05:30
tegra_usb_phy_postresume ( hcd - > phy ) ;
2011-03-09 16:28:56 -08:00
}
}
else if ( typeReq = = SetPortFeature & & wValue = = USB_PORT_FEAT_SUSPEND ) {
temp = ehci_readl ( ehci , status_reg ) ;
if ( ( temp & PORT_PE ) = = 0 | | ( temp & PORT_RESET ) ! = 0 ) {
retval = - EPIPE ;
goto done ;
}
2012-04-25 12:31:10 -06:00
temp & = ~ ( PORT_RWC_BITS | PORT_WKCONN_E ) ;
2011-03-09 16:28:56 -08:00
temp | = PORT_WKDISC_E | PORT_WKOC_E ;
ehci_writel ( ehci , temp | PORT_SUSPEND , status_reg ) ;
/*
* If a transaction is in progress , there may be a delay in
* suspending the port . Poll until the port is suspended .
*/
if ( handshake ( ehci , status_reg , PORT_SUSPEND ,
PORT_SUSPEND , 5000 ) )
pr_err ( " %s: timeout waiting for SUSPEND \n " , __func__ ) ;
set_bit ( ( wIndex & 0xff ) - 1 , & ehci - > suspended_ports ) ;
goto done ;
}
2011-04-17 11:58:25 +03:00
/* For USB1 port we need to issue Port Reset twice internally */
2012-12-13 20:59:08 +00:00
if ( tegra - > needs_double_reset & &
2011-04-17 11:58:25 +03:00
( typeReq = = SetPortFeature & & wValue = = USB_PORT_FEAT_RESET ) ) {
spin_unlock_irqrestore ( & ehci - > lock , flags ) ;
return tegra_ehci_internal_port_reset ( ehci , status_reg ) ;
}
2011-03-09 16:28:56 -08:00
/*
* Tegra host controller will time the resume operation to clear the bit
* when the port control state switches to HS or FS Idle . This behavior
* is different from EHCI where the host controller driver is required
* to set this bit to a zero after the resume duration is timed in the
* driver .
*/
else if ( typeReq = = ClearPortFeature & &
wValue = = USB_PORT_FEAT_SUSPEND ) {
temp = ehci_readl ( ehci , status_reg ) ;
if ( ( temp & PORT_RESET ) | | ! ( temp & PORT_PE ) ) {
retval = - EPIPE ;
goto done ;
}
if ( ! ( temp & PORT_SUSPEND ) )
goto done ;
/* Disable disconnect detection during port resume */
2013-01-24 15:57:03 +05:30
tegra_usb_phy_preresume ( hcd - > phy ) ;
2011-03-09 16:28:56 -08:00
ehci - > reset_done [ wIndex - 1 ] = jiffies + msecs_to_jiffies ( 25 ) ;
temp & = ~ ( PORT_RWC_BITS | PORT_WAKE_BITS ) ;
/* start resume signalling */
ehci_writel ( ehci , temp | PORT_RESUME , status_reg ) ;
2012-04-03 15:24:30 -04:00
set_bit ( wIndex - 1 , & ehci - > resuming_ports ) ;
2011-03-09 16:28:56 -08:00
spin_unlock_irqrestore ( & ehci - > lock , flags ) ;
msleep ( 20 ) ;
spin_lock_irqsave ( & ehci - > lock , flags ) ;
/* Poll until the controller clears RESUME and SUSPEND */
if ( handshake ( ehci , status_reg , PORT_RESUME , 0 , 2000 ) )
pr_err ( " %s: timeout waiting for RESUME \n " , __func__ ) ;
if ( handshake ( ehci , status_reg , PORT_SUSPEND , 0 , 2000 ) )
pr_err ( " %s: timeout waiting for SUSPEND \n " , __func__ ) ;
ehci - > reset_done [ wIndex - 1 ] = 0 ;
2012-04-03 15:24:30 -04:00
clear_bit ( wIndex - 1 , & ehci - > resuming_ports ) ;
2011-03-09 16:28:56 -08:00
tegra - > port_resuming = 1 ;
goto done ;
}
spin_unlock_irqrestore ( & ehci - > lock , flags ) ;
/* Handle the hub control events here */
return ehci_hub_control ( hcd , typeReq , wValue , wIndex , buf , wLength ) ;
done :
spin_unlock_irqrestore ( & ehci - > lock , flags ) ;
return retval ;
}
static void tegra_ehci_restart ( struct usb_hcd * hcd )
{
struct ehci_hcd * ehci = hcd_to_ehci ( hcd ) ;
ehci_reset ( ehci ) ;
/* setup the frame list and Async q heads */
ehci_writel ( ehci , ehci - > periodic_dma , & ehci - > regs - > frame_list ) ;
ehci_writel ( ehci , ( u32 ) ehci - > async - > qh_dma , & ehci - > regs - > async_next ) ;
/* setup the command register and set the controller in RUN mode */
ehci - > command & = ~ ( CMD_LRESET | CMD_IAAD | CMD_PSE | CMD_ASE | CMD_RESET ) ;
ehci - > command | = CMD_RUN ;
ehci_writel ( ehci , ehci - > command , & ehci - > regs - > command ) ;
down_write ( & ehci_cf_port_reset_rwsem ) ;
ehci_writel ( ehci , FLAG_CF , & ehci - > regs - > configured_flag ) ;
/* flush posted writes */
ehci_readl ( ehci , & ehci - > regs - > command ) ;
up_write ( & ehci_cf_port_reset_rwsem ) ;
}
static void tegra_ehci_shutdown ( struct usb_hcd * hcd )
{
struct tegra_ehci_hcd * tegra = dev_get_drvdata ( hcd - > self . controller ) ;
/* ehci_shutdown touches the USB controller registers, make sure
* controller has clocks to it */
if ( ! tegra - > host_resumed )
tegra_ehci_power_up ( hcd ) ;
ehci_shutdown ( hcd ) ;
}
static int tegra_ehci_setup ( struct usb_hcd * hcd )
{
struct ehci_hcd * ehci = hcd_to_ehci ( hcd ) ;
/* EHCI registers start at offset 0x100 */
ehci - > caps = hcd - > regs + 0x100 ;
/* switch to host mode */
hcd - > has_tt = 1 ;
2012-10-31 13:21:06 -04:00
return ehci_setup ( hcd ) ;
2011-03-09 16:28:56 -08:00
}
2012-04-05 11:25:30 +05:30
struct dma_aligned_buffer {
2011-03-09 16:28:57 -08:00
void * kmalloc_ptr ;
void * old_xfer_buffer ;
u8 data [ 0 ] ;
} ;
2012-04-05 11:25:30 +05:30
static void free_dma_aligned_buffer ( struct urb * urb )
2011-03-09 16:28:57 -08:00
{
2012-04-05 11:25:30 +05:30
struct dma_aligned_buffer * temp ;
2011-03-09 16:28:57 -08:00
if ( ! ( urb - > transfer_flags & URB_ALIGNED_TEMP_BUFFER ) )
return ;
2012-04-05 11:25:30 +05:30
temp = container_of ( urb - > transfer_buffer ,
struct dma_aligned_buffer , data ) ;
2011-03-09 16:28:57 -08:00
2012-04-05 11:25:30 +05:30
if ( usb_urb_dir_in ( urb ) )
2011-03-09 16:28:57 -08:00
memcpy ( temp - > old_xfer_buffer , temp - > data ,
urb - > transfer_buffer_length ) ;
urb - > transfer_buffer = temp - > old_xfer_buffer ;
kfree ( temp - > kmalloc_ptr ) ;
urb - > transfer_flags & = ~ URB_ALIGNED_TEMP_BUFFER ;
}
2012-04-05 11:25:30 +05:30
static int alloc_dma_aligned_buffer ( struct urb * urb , gfp_t mem_flags )
2011-03-09 16:28:57 -08:00
{
2012-04-05 11:25:30 +05:30
struct dma_aligned_buffer * temp , * kmalloc_ptr ;
2011-03-09 16:28:57 -08:00
size_t kmalloc_size ;
if ( urb - > num_sgs | | urb - > sg | |
urb - > transfer_buffer_length = = 0 | |
! ( ( uintptr_t ) urb - > transfer_buffer & ( TEGRA_USB_DMA_ALIGN - 1 ) ) )
return 0 ;
/* Allocate a buffer with enough padding for alignment */
kmalloc_size = urb - > transfer_buffer_length +
2012-04-05 11:25:30 +05:30
sizeof ( struct dma_aligned_buffer ) + TEGRA_USB_DMA_ALIGN - 1 ;
2011-03-09 16:28:57 -08:00
kmalloc_ptr = kmalloc ( kmalloc_size , mem_flags ) ;
if ( ! kmalloc_ptr )
return - ENOMEM ;
2012-04-05 11:25:30 +05:30
/* Position our struct dma_aligned_buffer such that data is aligned */
2011-03-09 16:28:57 -08:00
temp = PTR_ALIGN ( kmalloc_ptr + 1 , TEGRA_USB_DMA_ALIGN ) - 1 ;
temp - > kmalloc_ptr = kmalloc_ptr ;
temp - > old_xfer_buffer = urb - > transfer_buffer ;
2012-04-05 11:25:30 +05:30
if ( usb_urb_dir_out ( urb ) )
2011-03-09 16:28:57 -08:00
memcpy ( temp - > data , urb - > transfer_buffer ,
urb - > transfer_buffer_length ) ;
urb - > transfer_buffer = temp - > data ;
urb - > transfer_flags | = URB_ALIGNED_TEMP_BUFFER ;
return 0 ;
}
static int tegra_ehci_map_urb_for_dma ( struct usb_hcd * hcd , struct urb * urb ,
gfp_t mem_flags )
{
int ret ;
2012-04-05 11:25:30 +05:30
ret = alloc_dma_aligned_buffer ( urb , mem_flags ) ;
2011-03-09 16:28:57 -08:00
if ( ret )
return ret ;
ret = usb_hcd_map_urb_for_dma ( hcd , urb , mem_flags ) ;
if ( ret )
2012-04-05 11:25:30 +05:30
free_dma_aligned_buffer ( urb ) ;
2011-03-09 16:28:57 -08:00
return ret ;
}
static void tegra_ehci_unmap_urb_for_dma ( struct usb_hcd * hcd , struct urb * urb )
{
usb_hcd_unmap_urb_for_dma ( hcd , urb ) ;
2012-04-05 11:25:30 +05:30
free_dma_aligned_buffer ( urb ) ;
2011-03-09 16:28:57 -08:00
}
2011-03-09 16:28:56 -08:00
static const struct hc_driver tegra_ehci_hc_driver = {
. description = hcd_name ,
. product_desc = " Tegra EHCI Host Controller " ,
. hcd_priv_size = sizeof ( struct ehci_hcd ) ,
. flags = HCD_USB2 | HCD_MEMORY ,
2012-04-06 09:40:18 +05:30
/* standard ehci functions */
2011-03-09 16:28:56 -08:00
. irq = ehci_irq ,
. start = ehci_run ,
. stop = ehci_stop ,
. urb_enqueue = ehci_urb_enqueue ,
. urb_dequeue = ehci_urb_dequeue ,
. endpoint_disable = ehci_endpoint_disable ,
. endpoint_reset = ehci_endpoint_reset ,
. get_frame_number = ehci_get_frame ,
. hub_status_data = ehci_hub_status_data ,
. clear_tt_buffer_complete = ehci_clear_tt_buffer_complete ,
2012-04-06 09:40:18 +05:30
. relinquish_port = ehci_relinquish_port ,
. port_handed_over = ehci_port_handed_over ,
/* modified ehci functions for tegra */
. reset = tegra_ehci_setup ,
. shutdown = tegra_ehci_shutdown ,
. map_urb_for_dma = tegra_ehci_map_urb_for_dma ,
. unmap_urb_for_dma = tegra_ehci_unmap_urb_for_dma ,
. hub_control = tegra_ehci_hub_control ,
2011-03-09 16:28:56 -08:00
# ifdef CONFIG_PM
2012-05-01 11:28:49 -04:00
. bus_suspend = ehci_bus_suspend ,
. bus_resume = ehci_bus_resume ,
2011-03-09 16:28:56 -08:00
# endif
} ;
2012-03-16 16:06:07 -06:00
static int setup_vbus_gpio ( struct platform_device * pdev ,
struct tegra_ehci_platform_data * pdata )
2011-11-04 09:12:40 +00:00
{
int err = 0 ;
int gpio ;
2012-03-16 16:06:07 -06:00
gpio = pdata - > vbus_gpio ;
if ( ! gpio_is_valid ( gpio ) )
gpio = of_get_named_gpio ( pdev - > dev . of_node ,
" nvidia,vbus-gpio " , 0 ) ;
2011-11-04 09:12:40 +00:00
if ( ! gpio_is_valid ( gpio ) )
return 0 ;
err = gpio_request ( gpio , " vbus_gpio " ) ;
if ( err ) {
dev_err ( & pdev - > dev , " can't request vbus gpio %d " , gpio ) ;
return err ;
}
err = gpio_direction_output ( gpio , 1 ) ;
if ( err ) {
dev_err ( & pdev - > dev , " can't enable vbus \n " ) ;
return err ;
}
return err ;
}
2012-05-01 11:28:49 -04:00
# ifdef CONFIG_PM
static int controller_suspend ( struct device * dev )
{
struct tegra_ehci_hcd * tegra =
platform_get_drvdata ( to_platform_device ( dev ) ) ;
struct ehci_hcd * ehci = tegra - > ehci ;
struct usb_hcd * hcd = ehci_to_hcd ( ehci ) ;
struct ehci_regs __iomem * hw = ehci - > regs ;
unsigned long flags ;
if ( time_before ( jiffies , ehci - > next_statechange ) )
msleep ( 10 ) ;
2012-07-11 11:23:10 -04:00
ehci_halt ( ehci ) ;
2012-05-01 11:28:49 -04:00
2012-07-11 11:23:10 -04:00
spin_lock_irqsave ( & ehci - > lock , flags ) ;
2012-05-01 11:28:49 -04:00
tegra - > port_speed = ( readl ( & hw - > port_status [ 0 ] ) > > 26 ) & 0x3 ;
clear_bit ( HCD_FLAG_HW_ACCESSIBLE , & hcd - > flags ) ;
spin_unlock_irqrestore ( & ehci - > lock , flags ) ;
tegra_ehci_power_down ( hcd ) ;
return 0 ;
}
static int controller_resume ( struct device * dev )
{
struct tegra_ehci_hcd * tegra =
platform_get_drvdata ( to_platform_device ( dev ) ) ;
struct ehci_hcd * ehci = tegra - > ehci ;
struct usb_hcd * hcd = ehci_to_hcd ( ehci ) ;
struct ehci_regs __iomem * hw = ehci - > regs ;
unsigned long val ;
set_bit ( HCD_FLAG_HW_ACCESSIBLE , & hcd - > flags ) ;
tegra_ehci_power_up ( hcd ) ;
if ( tegra - > port_speed > TEGRA_USB_PHY_PORT_SPEED_HIGH ) {
/* Wait for the phy to detect new devices
* before we restart the controller */
msleep ( 10 ) ;
goto restart ;
}
/* Force the phy to keep data lines in suspend state */
2013-01-24 15:57:03 +05:30
tegra_ehci_phy_restore_start ( hcd - > phy , tegra - > port_speed ) ;
2012-05-01 11:28:49 -04:00
/* Enable host mode */
tdi_reset ( ehci ) ;
/* Enable Port Power */
val = readl ( & hw - > port_status [ 0 ] ) ;
val | = PORT_POWER ;
writel ( val , & hw - > port_status [ 0 ] ) ;
udelay ( 10 ) ;
/* Check if the phy resume from LP0. When the phy resume from LP0
* USB register will be reset . */
if ( ! readl ( & hw - > async_next ) ) {
/* Program the field PTC based on the saved speed mode */
val = readl ( & hw - > port_status [ 0 ] ) ;
val & = ~ PORT_TEST ( ~ 0 ) ;
if ( tegra - > port_speed = = TEGRA_USB_PHY_PORT_SPEED_HIGH )
val | = PORT_TEST_FORCE ;
else if ( tegra - > port_speed = = TEGRA_USB_PHY_PORT_SPEED_FULL )
val | = PORT_TEST ( 6 ) ;
else if ( tegra - > port_speed = = TEGRA_USB_PHY_PORT_SPEED_LOW )
val | = PORT_TEST ( 7 ) ;
writel ( val , & hw - > port_status [ 0 ] ) ;
udelay ( 10 ) ;
/* Disable test mode by setting PTC field to NORMAL_OP */
val = readl ( & hw - > port_status [ 0 ] ) ;
val & = ~ PORT_TEST ( ~ 0 ) ;
writel ( val , & hw - > port_status [ 0 ] ) ;
udelay ( 10 ) ;
}
/* Poll until CCS is enabled */
if ( handshake ( ehci , & hw - > port_status [ 0 ] , PORT_CONNECT ,
PORT_CONNECT , 2000 ) ) {
pr_err ( " %s: timeout waiting for PORT_CONNECT \n " , __func__ ) ;
goto restart ;
}
/* Poll until PE is enabled */
if ( handshake ( ehci , & hw - > port_status [ 0 ] , PORT_PE ,
PORT_PE , 2000 ) ) {
pr_err ( " %s: timeout waiting for USB_PORTSC1_PE \n " , __func__ ) ;
goto restart ;
}
/* Clear the PCI status, to avoid an interrupt taken upon resume */
val = readl ( & hw - > status ) ;
val | = STS_PCD ;
writel ( val , & hw - > status ) ;
/* Put controller in suspend mode by writing 1 to SUSP bit of PORTSC */
val = readl ( & hw - > port_status [ 0 ] ) ;
if ( ( val & PORT_POWER ) & & ( val & PORT_PE ) ) {
val | = PORT_SUSPEND ;
writel ( val , & hw - > port_status [ 0 ] ) ;
/* Wait until port suspend completes */
if ( handshake ( ehci , & hw - > port_status [ 0 ] , PORT_SUSPEND ,
PORT_SUSPEND , 1000 ) ) {
pr_err ( " %s: timeout waiting for PORT_SUSPEND \n " ,
__func__ ) ;
goto restart ;
}
}
2013-01-24 15:57:03 +05:30
tegra_ehci_phy_restore_end ( hcd - > phy ) ;
2012-05-01 11:28:49 -04:00
goto done ;
restart :
if ( tegra - > port_speed < = TEGRA_USB_PHY_PORT_SPEED_HIGH )
2013-01-24 15:57:03 +05:30
tegra_ehci_phy_restore_end ( hcd - > phy ) ;
2012-05-01 11:28:49 -04:00
tegra_ehci_restart ( hcd ) ;
done :
2013-01-24 15:57:03 +05:30
tegra_usb_phy_preresume ( hcd - > phy ) ;
2012-05-01 11:28:49 -04:00
tegra - > port_resuming = 1 ;
return 0 ;
}
static int tegra_ehci_suspend ( struct device * dev )
{
struct tegra_ehci_hcd * tegra =
platform_get_drvdata ( to_platform_device ( dev ) ) ;
struct usb_hcd * hcd = ehci_to_hcd ( tegra - > ehci ) ;
int rc = 0 ;
/*
* When system sleep is supported and USB controller wakeup is
* implemented : If the controller is runtime - suspended and the
* wakeup setting needs to be changed , call pm_runtime_resume ( ) .
*/
if ( HCD_HW_ACCESSIBLE ( hcd ) )
rc = controller_suspend ( dev ) ;
return rc ;
}
static int tegra_ehci_resume ( struct device * dev )
{
int rc ;
rc = controller_resume ( dev ) ;
if ( rc = = 0 ) {
pm_runtime_disable ( dev ) ;
pm_runtime_set_active ( dev ) ;
pm_runtime_enable ( dev ) ;
}
return rc ;
}
static int tegra_ehci_runtime_suspend ( struct device * dev )
{
return controller_suspend ( dev ) ;
}
static int tegra_ehci_runtime_resume ( struct device * dev )
{
return controller_resume ( dev ) ;
}
static const struct dev_pm_ops tegra_ehci_pm_ops = {
. suspend = tegra_ehci_suspend ,
. resume = tegra_ehci_resume ,
. runtime_suspend = tegra_ehci_runtime_suspend ,
. runtime_resume = tegra_ehci_runtime_resume ,
} ;
# endif
2013-01-17 20:15:37 +00:00
/* Bits of PORTSC1, which will get cleared by writing 1 into them */
# define TEGRA_PORTSC1_RWC_BITS (PORT_CSC | PORT_PEC | PORT_OCC)
2013-05-16 19:43:02 +05:30
void tegra_ehci_set_pts ( struct usb_phy * x , u8 pts_val )
2013-01-17 20:15:37 +00:00
{
unsigned long val ;
struct usb_hcd * hcd = bus_to_hcd ( x - > otg - > host ) ;
void __iomem * base = hcd - > regs ;
val = readl ( base + TEGRA_USB_PORTSC1 ) & ~ TEGRA_PORTSC1_RWC_BITS ;
val & = ~ TEGRA_USB_PORTSC1_PTS ( 3 ) ;
val | = TEGRA_USB_PORTSC1_PTS ( pts_val & 3 ) ;
writel ( val , base + TEGRA_USB_PORTSC1 ) ;
}
2013-05-16 19:43:02 +05:30
EXPORT_SYMBOL_GPL ( tegra_ehci_set_pts ) ;
2013-01-17 20:15:37 +00:00
2013-05-16 19:43:02 +05:30
void tegra_ehci_set_phcd ( struct usb_phy * x , bool enable )
2013-01-17 20:15:37 +00:00
{
unsigned long val ;
struct usb_hcd * hcd = bus_to_hcd ( x - > otg - > host ) ;
void __iomem * base = hcd - > regs ;
val = readl ( base + TEGRA_USB_PORTSC1 ) & ~ TEGRA_PORTSC1_RWC_BITS ;
if ( enable )
val | = TEGRA_USB_PORTSC1_PHCD ;
else
val & = ~ TEGRA_USB_PORTSC1_PHCD ;
writel ( val , base + TEGRA_USB_PORTSC1 ) ;
}
2013-05-16 19:43:02 +05:30
EXPORT_SYMBOL_GPL ( tegra_ehci_set_phcd ) ;
2013-01-17 20:15:37 +00:00
2011-03-09 16:28:56 -08:00
static int tegra_ehci_probe ( struct platform_device * pdev )
{
struct resource * res ;
struct usb_hcd * hcd ;
struct tegra_ehci_hcd * tegra ;
struct tegra_ehci_platform_data * pdata ;
int err = 0 ;
int irq ;
2013-05-16 19:43:02 +05:30
struct device_node * np_phy ;
2013-01-17 20:15:37 +00:00
struct usb_phy * u_phy ;
2011-03-09 16:28:56 -08:00
pdata = pdev - > dev . platform_data ;
if ( ! pdata ) {
dev_err ( & pdev - > dev , " Platform data missing \n " ) ;
return - EINVAL ;
}
2011-11-04 09:12:40 +00: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 .
*/
if ( ! pdev - > dev . dma_mask )
2013-05-07 16:53:52 -06:00
pdev - > dev . dma_mask = & pdev - > dev . coherent_dma_mask ;
if ( ! pdev - > dev . coherent_dma_mask )
pdev - > dev . coherent_dma_mask = DMA_BIT_MASK ( 32 ) ;
2011-11-04 09:12:40 +00:00
2012-03-16 16:06:07 -06:00
setup_vbus_gpio ( pdev , pdata ) ;
2011-11-04 09:12:40 +00:00
2012-07-30 16:43:41 +02:00
tegra = devm_kzalloc ( & pdev - > dev , sizeof ( struct tegra_ehci_hcd ) ,
GFP_KERNEL ) ;
2011-03-09 16:28:56 -08:00
if ( ! tegra )
return - ENOMEM ;
2012-07-30 16:43:41 +02:00
tegra - > clk = devm_clk_get ( & pdev - > dev , NULL ) ;
2011-03-09 16:28:56 -08:00
if ( IS_ERR ( tegra - > clk ) ) {
dev_err ( & pdev - > dev , " Can't get ehci clock \n " ) ;
2013-05-16 19:43:02 +05:30
return PTR_ERR ( tegra - > clk ) ;
2011-03-09 16:28:56 -08:00
}
2012-06-05 09:59:38 +05:30
err = clk_prepare_enable ( tegra - > clk ) ;
2011-03-09 16:28:56 -08:00
if ( err )
2013-05-16 19:43:02 +05:30
return err ;
2011-03-09 16:28:56 -08:00
2013-04-03 16:11:12 +05:30
tegra_periph_reset_assert ( tegra - > clk ) ;
udelay ( 1 ) ;
tegra_periph_reset_deassert ( tegra - > clk ) ;
2013-05-16 19:43:02 +05:30
np_phy = of_parse_phandle ( pdev - > dev . of_node , " nvidia,phy " , 0 ) ;
if ( ! np_phy ) {
err = - ENODEV ;
goto cleanup_clk ;
}
u_phy = tegra_usb_get_phy ( np_phy ) ;
if ( IS_ERR ( u_phy ) ) {
err = PTR_ERR ( u_phy ) ;
goto cleanup_clk ;
}
2012-12-13 20:59:08 +00:00
tegra - > needs_double_reset = of_property_read_bool ( pdev - > dev . of_node ,
" nvidia,needs-double-reset " ) ;
2013-05-16 19:43:02 +05:30
hcd = usb_create_hcd ( & tegra_ehci_hc_driver , & pdev - > dev ,
dev_name ( & pdev - > dev ) ) ;
if ( ! hcd ) {
dev_err ( & pdev - > dev , " Unable to create HCD \n " ) ;
err = - ENOMEM ;
goto cleanup_clk ;
}
hcd - > phy = u_phy ;
2011-03-09 16:28:56 -08:00
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( ! res ) {
dev_err ( & pdev - > dev , " Failed to get I/O memory \n " ) ;
err = - ENXIO ;
2013-05-16 19:43:02 +05:30
goto cleanup_hcd_create ;
2011-03-09 16:28:56 -08:00
}
hcd - > rsrc_start = res - > start ;
hcd - > rsrc_len = resource_size ( res ) ;
2012-07-30 16:43:41 +02:00
hcd - > regs = devm_ioremap ( & pdev - > dev , res - > start , resource_size ( res ) ) ;
2011-03-09 16:28:56 -08:00
if ( ! hcd - > regs ) {
dev_err ( & pdev - > dev , " Failed to remap I/O memory \n " ) ;
err = - ENOMEM ;
2013-05-16 19:43:02 +05:30
goto cleanup_hcd_create ;
2011-11-04 09:12:40 +00:00
}
2013-05-16 19:43:02 +05:30
err = usb_phy_init ( hcd - > phy ) ;
if ( err ) {
dev_err ( & pdev - > dev , " Failed to initialize phy \n " ) ;
goto cleanup_hcd_create ;
2011-03-09 16:28:56 -08:00
}
2013-01-17 20:15:37 +00:00
u_phy - > otg = devm_kzalloc ( & pdev - > dev , sizeof ( struct usb_otg ) ,
GFP_KERNEL ) ;
if ( ! u_phy - > otg ) {
dev_err ( & pdev - > dev , " Failed to alloc memory for otg \n " ) ;
err = - ENOMEM ;
2013-05-16 19:43:02 +05:30
goto cleanup_phy ;
2013-01-17 20:15:37 +00:00
}
u_phy - > otg - > host = hcd_to_bus ( hcd ) ;
2013-01-24 15:57:03 +05:30
err = usb_phy_set_suspend ( hcd - > phy , 0 ) ;
2011-03-09 16:28:56 -08:00
if ( err ) {
dev_err ( & pdev - > dev , " Failed to power on the phy \n " ) ;
2013-05-16 19:43:02 +05:30
goto cleanup_phy ;
2011-03-09 16:28:56 -08:00
}
tegra - > host_resumed = 1 ;
tegra - > ehci = hcd_to_ehci ( hcd ) ;
irq = platform_get_irq ( pdev , 0 ) ;
if ( ! irq ) {
dev_err ( & pdev - > dev , " Failed to get IRQ \n " ) ;
err = - ENODEV ;
2013-05-16 19:43:02 +05:30
goto cleanup_phy ;
2011-03-09 16:28:56 -08:00
}
if ( pdata - > operating_mode = = TEGRA_USB_OTG ) {
2012-07-30 16:43:41 +02:00
tegra - > transceiver =
devm_usb_get_phy ( & pdev - > dev , USB_PHY_TYPE_USB2 ) ;
2013-03-15 11:04:39 +02:00
if ( ! IS_ERR ( tegra - > transceiver ) )
2012-02-13 13:24:20 +02:00
otg_set_host ( tegra - > transceiver - > otg , & hcd - > self ) ;
2013-04-03 21:57:57 +02:00
} else {
tegra - > transceiver = ERR_PTR ( - ENODEV ) ;
2011-03-09 16:28:56 -08:00
}
2013-05-16 19:43:02 +05:30
platform_set_drvdata ( pdev , tegra ) ;
2011-09-07 16:10:52 +08:00
err = usb_add_hcd ( hcd , irq , IRQF_SHARED ) ;
2011-03-09 16:28:56 -08:00
if ( err ) {
dev_err ( & pdev - > dev , " Failed to add USB HCD \n " ) ;
2013-05-16 19:43:02 +05:30
goto cleanup_phy ;
2011-03-09 16:28:56 -08:00
}
2012-05-01 11:28:49 -04:00
pm_runtime_set_active ( & pdev - > dev ) ;
pm_runtime_get_noresume ( & pdev - > dev ) ;
/* Don't skip the pm_runtime_forbid call if wakeup isn't working */
/* if (!pdata->power_down_on_bus_suspend) */
pm_runtime_forbid ( & pdev - > dev ) ;
pm_runtime_enable ( & pdev - > dev ) ;
pm_runtime_put_sync ( & pdev - > dev ) ;
2011-03-09 16:28:56 -08:00
return err ;
2013-05-16 19:43:02 +05:30
cleanup_phy :
2013-03-15 11:04:39 +02:00
if ( ! IS_ERR ( tegra - > transceiver ) )
2012-02-13 13:24:20 +02:00
otg_set_host ( tegra - > transceiver - > otg , NULL ) ;
2013-05-16 19:43:02 +05:30
2013-01-24 15:57:03 +05:30
usb_phy_shutdown ( hcd - > phy ) ;
2013-05-16 19:43:02 +05:30
cleanup_hcd_create :
2011-03-09 16:28:56 -08:00
usb_put_hcd ( hcd ) ;
2013-05-16 19:43:02 +05:30
cleanup_clk :
clk_disable_unprepare ( tegra - > clk ) ;
2011-03-09 16:28:56 -08:00
return err ;
}
static int tegra_ehci_remove ( struct platform_device * pdev )
{
struct tegra_ehci_hcd * tegra = platform_get_drvdata ( pdev ) ;
struct usb_hcd * hcd = ehci_to_hcd ( tegra - > ehci ) ;
2012-05-01 11:28:49 -04:00
pm_runtime_get_sync ( & pdev - > dev ) ;
pm_runtime_disable ( & pdev - > dev ) ;
pm_runtime_put_noidle ( & pdev - > dev ) ;
2013-03-15 11:04:39 +02:00
if ( ! IS_ERR ( tegra - > transceiver ) )
2012-02-13 13:24:20 +02:00
otg_set_host ( tegra - > transceiver - > otg , NULL ) ;
2011-03-09 16:28:56 -08:00
2013-01-24 15:57:03 +05:30
usb_phy_shutdown ( hcd - > phy ) ;
2011-03-09 16:28:56 -08:00
usb_remove_hcd ( hcd ) ;
2012-08-10 11:42:43 +05:30
usb_put_hcd ( hcd ) ;
2012-06-05 09:59:38 +05:30
clk_disable_unprepare ( tegra - > clk ) ;
2011-03-09 16:28:56 -08:00
return 0 ;
}
static void tegra_ehci_hcd_shutdown ( struct platform_device * pdev )
{
struct tegra_ehci_hcd * tegra = platform_get_drvdata ( pdev ) ;
struct usb_hcd * hcd = ehci_to_hcd ( tegra - > ehci ) ;
if ( hcd - > driver - > shutdown )
hcd - > driver - > shutdown ( hcd ) ;
}
2012-11-19 13:24:34 -05:00
static struct of_device_id tegra_ehci_of_match [ ] = {
2011-11-04 09:12:40 +00:00
{ . compatible = " nvidia,tegra20-ehci " , } ,
{ } ,
} ;
2011-03-09 16:28:56 -08:00
static struct platform_driver tegra_ehci_driver = {
. probe = tegra_ehci_probe ,
. remove = tegra_ehci_remove ,
. shutdown = tegra_ehci_hcd_shutdown ,
. driver = {
. name = " tegra-ehci " ,
2011-11-04 09:12:40 +00:00
. of_match_table = tegra_ehci_of_match ,
2012-05-01 11:28:49 -04:00
# ifdef CONFIG_PM
. pm = & tegra_ehci_pm_ops ,
# endif
2011-03-09 16:28:56 -08:00
}
} ;