2017-11-03 11:28:30 +01:00
// SPDX-License-Identifier: GPL-2.0
2014-07-09 10:08:52 +09:00
/*
* xHCI host controller driver for R - Car SoCs
*
* Copyright ( C ) 2014 Renesas Electronics Corporation
*/
# include <linux/firmware.h>
# include <linux/module.h>
# include <linux/platform_device.h>
2016-04-22 13:17:12 +03:00
# include <linux/of.h>
2014-07-09 10:08:52 +09:00
# include <linux/usb/phy.h>
2017-08-16 14:23:18 +03:00
# include <linux/sys_soc.h>
2014-07-09 10:08:52 +09:00
# include "xhci.h"
2015-11-24 13:09:51 +02:00
# include "xhci-plat.h"
2014-07-09 10:08:52 +09:00
# include "xhci-rcar.h"
2015-11-24 13:09:53 +02:00
/*
2017-08-16 14:23:20 +03:00
* - The V3 firmware is for r8a7796 ( with good performance ) and r8a7795 es2 .0
* or later .
2016-11-11 15:13:26 +02:00
* - The V2 firmware can be used on both r8a7795 ( es1 . x ) and r8a7796 .
2015-11-24 13:09:53 +02:00
* - The V2 firmware is possible to use on R - Car Gen2 . However , the V2 causes
* performance degradation . So , this driver continues to use the V1 if R - Car
* Gen2 .
* - The V1 firmware is impossible to use on R - Car Gen3 .
*/
2015-11-24 13:09:51 +02:00
MODULE_FIRMWARE ( XHCI_RCAR_FIRMWARE_NAME_V1 ) ;
2015-11-24 13:09:53 +02:00
MODULE_FIRMWARE ( XHCI_RCAR_FIRMWARE_NAME_V2 ) ;
2016-11-11 15:13:26 +02:00
MODULE_FIRMWARE ( XHCI_RCAR_FIRMWARE_NAME_V3 ) ;
2014-07-09 10:08:52 +09:00
/*** Register Offset ***/
# define RCAR_USB3_INT_ENA 0x224 /* Interrupt Enable */
# define RCAR_USB3_DL_CTRL 0x250 /* FW Download Control & Status */
# define RCAR_USB3_FW_DATA0 0x258 /* FW Data0 */
# define RCAR_USB3_LCLK 0xa44 /* LCLK Select */
# define RCAR_USB3_CONF1 0xa48 /* USB3.0 Configuration1 */
# define RCAR_USB3_CONF2 0xa5c /* USB3.0 Configuration2 */
# define RCAR_USB3_CONF3 0xaa8 /* USB3.0 Configuration3 */
# define RCAR_USB3_RX_POL 0xab0 /* USB3.0 RX Polarity */
# define RCAR_USB3_TX_POL 0xab8 /* USB3.0 TX Polarity */
/*** Register Settings ***/
/* Interrupt Enable */
# define RCAR_USB3_INT_XHC_ENA 0x00000001
# define RCAR_USB3_INT_PME_ENA 0x00000002
# define RCAR_USB3_INT_HSE_ENA 0x00000004
# define RCAR_USB3_INT_ENA_VAL (RCAR_USB3_INT_XHC_ENA | \
RCAR_USB3_INT_PME_ENA | RCAR_USB3_INT_HSE_ENA )
/* FW Download Control & Status */
# define RCAR_USB3_DL_CTRL_ENABLE 0x00000001
# define RCAR_USB3_DL_CTRL_FW_SUCCESS 0x00000010
# define RCAR_USB3_DL_CTRL_FW_SET_DATA0 0x00000100
/* LCLK Select */
# define RCAR_USB3_LCLK_ENA_VAL 0x01030001
/* USB3.0 Configuration */
# define RCAR_USB3_CONF1_VAL 0x00030204
# define RCAR_USB3_CONF2_VAL 0x00030300
# define RCAR_USB3_CONF3_VAL 0x13802007
/* USB3.0 Polarity */
# define RCAR_USB3_RX_POL_VAL BIT(21)
# define RCAR_USB3_TX_POL_VAL BIT(4)
2017-08-16 14:23:18 +03:00
/* For soc_device_attribute */
# define RCAR_XHCI_FIRMWARE_V2 BIT(0) /* FIRMWARE V2 */
# define RCAR_XHCI_FIRMWARE_V3 BIT(1) /* FIRMWARE V3 */
static const struct soc_device_attribute rcar_quirks_match [ ] = {
{
. soc_id = " r8a7795 " , . revision = " ES1.* " ,
. data = ( void * ) RCAR_XHCI_FIRMWARE_V2 ,
} ,
2017-08-16 14:23:20 +03:00
{
. soc_id = " r8a7795 " ,
. data = ( void * ) RCAR_XHCI_FIRMWARE_V3 ,
} ,
2017-08-16 14:23:18 +03:00
{
. soc_id = " r8a7796 " ,
. data = ( void * ) RCAR_XHCI_FIRMWARE_V3 ,
} ,
{ /* sentinel */ } ,
} ;
2015-11-24 13:09:51 +02:00
static void xhci_rcar_start_gen2 ( struct usb_hcd * hcd )
{
/* LCLK Select */
writel ( RCAR_USB3_LCLK_ENA_VAL , hcd - > regs + RCAR_USB3_LCLK ) ;
/* USB3.0 Configuration */
writel ( RCAR_USB3_CONF1_VAL , hcd - > regs + RCAR_USB3_CONF1 ) ;
writel ( RCAR_USB3_CONF2_VAL , hcd - > regs + RCAR_USB3_CONF2 ) ;
writel ( RCAR_USB3_CONF3_VAL , hcd - > regs + RCAR_USB3_CONF3 ) ;
/* USB3.0 Polarity */
writel ( RCAR_USB3_RX_POL_VAL , hcd - > regs + RCAR_USB3_RX_POL ) ;
writel ( RCAR_USB3_TX_POL_VAL , hcd - > regs + RCAR_USB3_TX_POL ) ;
}
2016-04-22 13:17:12 +03:00
static int xhci_rcar_is_gen2 ( struct device * dev )
{
struct device_node * node = dev - > of_node ;
return of_device_is_compatible ( node , " renesas,xhci-r8a7790 " ) | |
of_device_is_compatible ( node , " renesas,xhci-r8a7791 " ) | |
of_device_is_compatible ( node , " renesas,xhci-r8a7793 " ) | |
of_device_is_compatible ( node , " renensas,rcar-gen2-xhci " ) ;
}
static int xhci_rcar_is_gen3 ( struct device * dev )
{
struct device_node * node = dev - > of_node ;
return of_device_is_compatible ( node , " renesas,xhci-r8a7795 " ) | |
2016-11-11 15:13:27 +02:00
of_device_is_compatible ( node , " renesas,xhci-r8a7796 " ) | |
2016-04-22 13:17:12 +03:00
of_device_is_compatible ( node , " renesas,rcar-gen3-xhci " ) ;
}
2014-07-09 10:08:52 +09:00
void xhci_rcar_start ( struct usb_hcd * hcd )
{
u32 temp ;
if ( hcd - > regs ! = NULL ) {
/* Interrupt Enable */
temp = readl ( hcd - > regs + RCAR_USB3_INT_ENA ) ;
temp | = RCAR_USB3_INT_ENA_VAL ;
writel ( temp , hcd - > regs + RCAR_USB3_INT_ENA ) ;
2016-04-22 13:17:12 +03:00
if ( xhci_rcar_is_gen2 ( hcd - > self . controller ) )
2015-11-24 13:09:51 +02:00
xhci_rcar_start_gen2 ( hcd ) ;
2014-07-09 10:08:52 +09:00
}
}
2015-11-24 13:09:51 +02:00
static int xhci_rcar_download_firmware ( struct usb_hcd * hcd )
2014-07-09 10:08:52 +09:00
{
2015-11-24 13:09:51 +02:00
struct device * dev = hcd - > self . controller ;
void __iomem * regs = hcd - > regs ;
struct xhci_plat_priv * priv = hcd_to_xhci_priv ( hcd ) ;
2014-07-09 10:08:52 +09:00
const struct firmware * fw ;
int retval , index , j , time ;
int timeout = 10000 ;
u32 data , val , temp ;
2017-08-16 14:23:18 +03:00
u32 quirks = 0 ;
const struct soc_device_attribute * attr ;
const char * firmware_name ;
attr = soc_device_match ( rcar_quirks_match ) ;
if ( attr )
quirks = ( uintptr_t ) attr - > data ;
if ( quirks & RCAR_XHCI_FIRMWARE_V2 )
firmware_name = XHCI_RCAR_FIRMWARE_NAME_V2 ;
else if ( quirks & RCAR_XHCI_FIRMWARE_V3 )
firmware_name = XHCI_RCAR_FIRMWARE_NAME_V3 ;
else
firmware_name = priv - > firmware_name ;
2014-07-09 10:08:52 +09:00
/* request R-Car USB3.0 firmware */
2017-08-16 14:23:18 +03:00
retval = request_firmware ( & fw , firmware_name , dev ) ;
2014-07-09 10:08:52 +09:00
if ( retval )
return retval ;
/* download R-Car USB3.0 firmware */
temp = readl ( regs + RCAR_USB3_DL_CTRL ) ;
temp | = RCAR_USB3_DL_CTRL_ENABLE ;
writel ( temp , regs + RCAR_USB3_DL_CTRL ) ;
for ( index = 0 ; index < fw - > size ; index + = 4 ) {
/* to avoid reading beyond the end of the buffer */
for ( data = 0 , j = 3 ; j > = 0 ; j - - ) {
if ( ( j + index ) < fw - > size )
data | = fw - > data [ index + j ] < < ( 8 * j ) ;
}
writel ( data , regs + RCAR_USB3_FW_DATA0 ) ;
temp = readl ( regs + RCAR_USB3_DL_CTRL ) ;
temp | = RCAR_USB3_DL_CTRL_FW_SET_DATA0 ;
writel ( temp , regs + RCAR_USB3_DL_CTRL ) ;
for ( time = 0 ; time < timeout ; time + + ) {
val = readl ( regs + RCAR_USB3_DL_CTRL ) ;
if ( ( val & RCAR_USB3_DL_CTRL_FW_SET_DATA0 ) = = 0 )
break ;
udelay ( 1 ) ;
}
if ( time = = timeout ) {
retval = - ETIMEDOUT ;
break ;
}
}
temp = readl ( regs + RCAR_USB3_DL_CTRL ) ;
temp & = ~ RCAR_USB3_DL_CTRL_ENABLE ;
writel ( temp , regs + RCAR_USB3_DL_CTRL ) ;
for ( time = 0 ; time < timeout ; time + + ) {
val = readl ( regs + RCAR_USB3_DL_CTRL ) ;
if ( val & RCAR_USB3_DL_CTRL_FW_SUCCESS ) {
retval = 0 ;
break ;
}
udelay ( 1 ) ;
}
if ( time = = timeout )
retval = - ETIMEDOUT ;
release_firmware ( fw ) ;
return retval ;
}
/* This function needs to initialize a "phy" of usb before */
int xhci_rcar_init_quirk ( struct usb_hcd * hcd )
{
2016-04-22 13:17:12 +03:00
struct xhci_hcd * xhci = hcd_to_xhci ( hcd ) ;
2014-07-09 10:08:52 +09:00
/* If hcd->regs is NULL, we don't just call the following function */
if ( ! hcd - > regs )
return 0 ;
2016-04-22 13:17:12 +03:00
/*
* On R - Car Gen2 and Gen3 , the AC64 bit ( bit 0 ) of HCCPARAMS1 is set
* to 1. However , these SoCs don ' t support 64 - bit address memory
* pointers . So , this driver clears the AC64 bit of xhci - > hcc_params
* to call dma_set_coherent_mask ( dev , DMA_BIT_MASK ( 32 ) ) in
* xhci_gen_setup ( ) .
*/
if ( xhci_rcar_is_gen2 ( hcd - > self . controller ) | |
xhci_rcar_is_gen3 ( hcd - > self . controller ) )
xhci - > quirks | = XHCI_NO_64BIT_SUPPORT ;
2015-11-24 13:09:51 +02:00
return xhci_rcar_download_firmware ( hcd ) ;
2014-07-09 10:08:52 +09:00
}
2017-04-19 16:55:48 +03:00
int xhci_rcar_resume_quirk ( struct usb_hcd * hcd )
{
int ret ;
ret = xhci_rcar_download_firmware ( hcd ) ;
if ( ! ret )
xhci_rcar_start ( hcd ) ;
return ret ;
}