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>
2020-09-21 14:13:28 +08:00
# include <linux/iopoll.h>
2014-07-09 10:08:52 +09:00
# 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
/*
2018-07-03 19:46:57 +09:00
* - The V3 firmware is for almost all R - Car Gen3 ( except r8a7795 ES1 . x )
* - The V2 firmware is for r8a7795 ES1 . x .
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 ***/
2018-07-03 19:46:58 +09:00
# define RCAR_USB3_AXH_STA 0x104 /* AXI Host Control Status */
2014-07-09 10:08:52 +09:00
# 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 ***/
2018-07-03 19:46:58 +09:00
/* AXI Host Control Status */
# define RCAR_USB3_AXH_STA_B3_PLL_ACTIVE 0x00010000
# define RCAR_USB3_AXH_STA_B2_PLL_ACTIVE 0x00000001
# define RCAR_USB3_AXH_STA_PLL_ACTIVE_MASK (RCAR_USB3_AXH_STA_B3_PLL_ACTIVE | \
RCAR_USB3_AXH_STA_B2_PLL_ACTIVE )
2014-07-09 10:08:52 +09:00
/* 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 ,
} ,
2022-03-03 13:50:21 +01:00
{ /* sentinel */ }
2017-08-16 14:23:18 +03:00
} ;
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 " ) | |
2019-08-27 14:51:12 +02:00
of_device_is_compatible ( node , " renesas,rcar-gen2-xhci " ) ;
2016-04-22 13:17:12 +03:00
}
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 ;
2020-09-21 14:13:28 +08:00
int retval , index , j ;
2014-07-09 10:08:52 +09:00
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 ;
2021-08-27 15:32:27 +09:00
/*
* According to the datasheet , " Upon the completion of FW Download,
* there is no need to write or reload FW " .
*/
if ( readl ( regs + RCAR_USB3_DL_CTRL ) & RCAR_USB3_DL_CTRL_FW_SUCCESS )
return 0 ;
2017-08-16 14:23:18 +03:00
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 ) ;
2020-09-21 14:13:28 +08:00
retval = readl_poll_timeout_atomic ( regs + RCAR_USB3_DL_CTRL ,
val , ! ( val & RCAR_USB3_DL_CTRL_FW_SET_DATA0 ) ,
1 , 10000 ) ;
if ( retval < 0 )
2014-07-09 10:08:52 +09:00
break ;
}
temp = readl ( regs + RCAR_USB3_DL_CTRL ) ;
temp & = ~ RCAR_USB3_DL_CTRL_ENABLE ;
writel ( temp , regs + RCAR_USB3_DL_CTRL ) ;
2020-09-21 14:13:28 +08:00
retval = readl_poll_timeout_atomic ( ( regs + RCAR_USB3_DL_CTRL ) ,
val , val & RCAR_USB3_DL_CTRL_FW_SUCCESS , 1 , 10000 ) ;
2014-07-09 10:08:52 +09:00
release_firmware ( fw ) ;
return retval ;
}
2018-07-03 19:46:58 +09:00
static bool xhci_rcar_wait_for_pll_active ( struct usb_hcd * hcd )
{
2020-09-21 14:13:28 +08:00
int retval ;
2018-07-03 19:46:58 +09:00
u32 val , mask = RCAR_USB3_AXH_STA_PLL_ACTIVE_MASK ;
2020-09-21 14:13:28 +08:00
retval = readl_poll_timeout_atomic ( hcd - > regs + RCAR_USB3_AXH_STA ,
val , ( val & mask ) = = mask , 1 , 1000 ) ;
return ! retval ;
2018-07-03 19:46:58 +09:00
}
2014-07-09 10:08:52 +09:00
/* This function needs to initialize a "phy" of usb before */
int xhci_rcar_init_quirk ( struct usb_hcd * hcd )
{
/* If hcd->regs is NULL, we don't just call the following function */
if ( ! hcd - > regs )
return 0 ;
2018-07-03 19:46:58 +09:00
if ( ! xhci_rcar_wait_for_pll_active ( hcd ) )
return - ETIMEDOUT ;
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 ;
}