2011-03-09 16:28:55 -08:00
/*
* Copyright ( C ) 2010 Google , Inc .
2013-05-16 19:43:02 +05:30
* Copyright ( C ) 2013 NVIDIA Corporation
2011-03-09 16:28:55 -08:00
*
* Author :
* Erik Gilling < konkers @ google . com >
* Benoit Goby < benoit @ android . com >
2013-05-16 19:43:02 +05:30
* Venu Byravarasu < vbyravarasu @ nvidia . com >
2011-03-09 16:28:55 -08:00
*
* This software is licensed under the terms of the GNU General Public
* License version 2 , as published by the Free Software Foundation , and
* may be copied , distributed , and modified under those terms .
*
* 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/resource.h>
# include <linux/delay.h>
# include <linux/slab.h>
# include <linux/err.h>
2012-03-02 15:58:42 -05:00
# include <linux/export.h>
2013-06-13 11:24:08 -06:00
# include <linux/module.h>
2011-03-09 16:28:55 -08:00
# include <linux/platform_device.h>
# include <linux/io.h>
# include <linux/gpio.h>
2013-01-16 03:30:20 +00:00
# include <linux/of.h>
2012-04-12 15:46:49 -06:00
# include <linux/of_gpio.h>
2011-03-09 16:28:55 -08:00
# include <linux/usb/otg.h>
# include <linux/usb/ulpi.h>
# include <asm/mach-types.h>
2012-09-05 18:50:23 +05:30
# include <linux/usb/tegra_usb_phy.h>
2013-05-16 19:43:02 +05:30
# include <linux/module.h>
2011-03-09 16:28:55 -08:00
# define ULPI_VIEWPORT 0x170
# define USB_SUSP_CTRL 0x400
# define USB_WAKE_ON_CNNT_EN_DEV (1 << 3)
# define USB_WAKE_ON_DISCON_EN_DEV (1 << 4)
# define USB_SUSP_CLR (1 << 5)
# define USB_PHY_CLK_VALID (1 << 7)
# define UTMIP_RESET (1 << 11)
# define UHSIC_RESET (1 << 11)
# define UTMIP_PHY_ENABLE (1 << 12)
# define ULPI_PHY_ENABLE (1 << 13)
# define USB_SUSP_SET (1 << 14)
# define USB_WAKEUP_DEBOUNCE_COUNT(x) (((x) & 0x7) << 16)
# define USB1_LEGACY_CTRL 0x410
# define USB1_NO_LEGACY_MODE (1 << 0)
# define USB1_VBUS_SENSE_CTL_MASK (3 << 1)
# define USB1_VBUS_SENSE_CTL_VBUS_WAKEUP (0 << 1)
# define USB1_VBUS_SENSE_CTL_AB_SESS_VLD_OR_VBUS_WAKEUP \
( 1 < < 1 )
# define USB1_VBUS_SENSE_CTL_AB_SESS_VLD (2 << 1)
# define USB1_VBUS_SENSE_CTL_A_SESS_VLD (3 << 1)
# define ULPI_TIMING_CTRL_0 0x424
# define ULPI_OUTPUT_PINMUX_BYP (1 << 10)
# define ULPI_CLKOUT_PINMUX_BYP (1 << 11)
# define ULPI_TIMING_CTRL_1 0x428
# define ULPI_DATA_TRIMMER_LOAD (1 << 0)
# define ULPI_DATA_TRIMMER_SEL(x) (((x) & 0x7) << 1)
# define ULPI_STPDIRNXT_TRIMMER_LOAD (1 << 16)
# define ULPI_STPDIRNXT_TRIMMER_SEL(x) (((x) & 0x7) << 17)
# define ULPI_DIR_TRIMMER_LOAD (1 << 24)
# define ULPI_DIR_TRIMMER_SEL(x) (((x) & 0x7) << 25)
# define UTMIP_PLL_CFG1 0x804
# define UTMIP_XTAL_FREQ_COUNT(x) (((x) & 0xfff) << 0)
# define UTMIP_PLLU_ENABLE_DLY_COUNT(x) (((x) & 0x1f) << 27)
# define UTMIP_XCVR_CFG0 0x808
# define UTMIP_XCVR_SETUP(x) (((x) & 0xf) << 0)
# define UTMIP_XCVR_LSRSLEW(x) (((x) & 0x3) << 8)
# define UTMIP_XCVR_LSFSLEW(x) (((x) & 0x3) << 10)
# define UTMIP_FORCE_PD_POWERDOWN (1 << 14)
# define UTMIP_FORCE_PD2_POWERDOWN (1 << 16)
# define UTMIP_FORCE_PDZI_POWERDOWN (1 << 18)
# define UTMIP_XCVR_HSSLEW_MSB(x) (((x) & 0x7f) << 25)
# define UTMIP_BIAS_CFG0 0x80c
# define UTMIP_OTGPD (1 << 11)
# define UTMIP_BIASPD (1 << 10)
# define UTMIP_HSRX_CFG0 0x810
# define UTMIP_ELASTIC_LIMIT(x) (((x) & 0x1f) << 10)
# define UTMIP_IDLE_WAIT(x) (((x) & 0x1f) << 15)
# define UTMIP_HSRX_CFG1 0x814
# define UTMIP_HS_SYNC_START_DLY(x) (((x) & 0x1f) << 1)
# define UTMIP_TX_CFG0 0x820
# define UTMIP_FS_PREABMLE_J (1 << 19)
# define UTMIP_HS_DISCON_DISABLE (1 << 8)
# define UTMIP_MISC_CFG0 0x824
# define UTMIP_DPDM_OBSERVE (1 << 26)
# define UTMIP_DPDM_OBSERVE_SEL(x) (((x) & 0xf) << 27)
# define UTMIP_DPDM_OBSERVE_SEL_FS_J UTMIP_DPDM_OBSERVE_SEL(0xf)
# define UTMIP_DPDM_OBSERVE_SEL_FS_K UTMIP_DPDM_OBSERVE_SEL(0xe)
# define UTMIP_DPDM_OBSERVE_SEL_FS_SE1 UTMIP_DPDM_OBSERVE_SEL(0xd)
# define UTMIP_DPDM_OBSERVE_SEL_FS_SE0 UTMIP_DPDM_OBSERVE_SEL(0xc)
# define UTMIP_SUSPEND_EXIT_ON_EDGE (1 << 22)
# define UTMIP_MISC_CFG1 0x828
# define UTMIP_PLL_ACTIVE_DLY_COUNT(x) (((x) & 0x1f) << 18)
# define UTMIP_PLLU_STABLE_COUNT(x) (((x) & 0xfff) << 6)
# define UTMIP_DEBOUNCE_CFG0 0x82c
# define UTMIP_BIAS_DEBOUNCE_A(x) (((x) & 0xffff) << 0)
# define UTMIP_BAT_CHRG_CFG0 0x830
# define UTMIP_PD_CHRG (1 << 0)
# define UTMIP_SPARE_CFG0 0x834
# define FUSE_SETUP_SEL (1 << 3)
# define UTMIP_XCVR_CFG1 0x838
# define UTMIP_FORCE_PDDISC_POWERDOWN (1 << 0)
# define UTMIP_FORCE_PDCHRP_POWERDOWN (1 << 2)
# define UTMIP_FORCE_PDDR_POWERDOWN (1 << 4)
# define UTMIP_XCVR_TERM_RANGE_ADJ(x) (((x) & 0xf) << 18)
# define UTMIP_BIAS_CFG1 0x83c
# define UTMIP_BIAS_PDTRK_COUNT(x) (((x) & 0x1f) << 3)
static DEFINE_SPINLOCK ( utmip_pad_lock ) ;
static int utmip_pad_count ;
struct tegra_xtal_freq {
int freq ;
u8 enable_delay ;
u8 stable_count ;
u8 active_delay ;
u8 xtal_freq_count ;
u16 debounce ;
} ;
static const struct tegra_xtal_freq tegra_freq_table [ ] = {
{
. freq = 12000000 ,
. enable_delay = 0x02 ,
. stable_count = 0x2F ,
. active_delay = 0x04 ,
. xtal_freq_count = 0x76 ,
. debounce = 0x7530 ,
} ,
{
. freq = 13000000 ,
. enable_delay = 0x02 ,
. stable_count = 0x33 ,
. active_delay = 0x05 ,
. xtal_freq_count = 0x7F ,
. debounce = 0x7EF4 ,
} ,
{
. freq = 19200000 ,
. enable_delay = 0x03 ,
. stable_count = 0x4B ,
. active_delay = 0x06 ,
. xtal_freq_count = 0xBB ,
. debounce = 0xBB80 ,
} ,
{
. freq = 26000000 ,
. enable_delay = 0x04 ,
. stable_count = 0x66 ,
. active_delay = 0x09 ,
. xtal_freq_count = 0xFE ,
. debounce = 0xFDE8 ,
} ,
} ;
static struct tegra_utmip_config utmip_default [ ] = {
[ 0 ] = {
. hssync_start_delay = 9 ,
. idle_wait_delay = 17 ,
. elastic_limit = 16 ,
. term_range_adj = 6 ,
. xcvr_setup = 9 ,
. xcvr_lsfslew = 1 ,
. xcvr_lsrslew = 1 ,
} ,
[ 2 ] = {
. hssync_start_delay = 9 ,
. idle_wait_delay = 17 ,
. elastic_limit = 16 ,
. term_range_adj = 6 ,
. xcvr_setup = 9 ,
. xcvr_lsfslew = 2 ,
. xcvr_lsrslew = 2 ,
} ,
} ;
static int utmip_pad_open ( struct tegra_usb_phy * phy )
{
2013-05-16 19:43:02 +05:30
phy - > pad_clk = devm_clk_get ( phy - > dev , " utmi-pads " ) ;
2011-03-09 16:28:55 -08:00
if ( IS_ERR ( phy - > pad_clk ) ) {
pr_err ( " %s: can't get utmip pad clock \n " , __func__ ) ;
return PTR_ERR ( phy - > pad_clk ) ;
}
return 0 ;
}
static void utmip_pad_power_on ( struct tegra_usb_phy * phy )
{
unsigned long val , flags ;
void __iomem * base = phy - > pad_regs ;
2012-06-05 09:59:35 +05:30
clk_prepare_enable ( phy - > pad_clk ) ;
2011-03-09 16:28:55 -08:00
spin_lock_irqsave ( & utmip_pad_lock , flags ) ;
if ( utmip_pad_count + + = = 0 ) {
val = readl ( base + UTMIP_BIAS_CFG0 ) ;
val & = ~ ( UTMIP_OTGPD | UTMIP_BIASPD ) ;
writel ( val , base + UTMIP_BIAS_CFG0 ) ;
}
spin_unlock_irqrestore ( & utmip_pad_lock , flags ) ;
2012-06-05 09:59:35 +05:30
clk_disable_unprepare ( phy - > pad_clk ) ;
2011-03-09 16:28:55 -08:00
}
static int utmip_pad_power_off ( struct tegra_usb_phy * phy )
{
unsigned long val , flags ;
void __iomem * base = phy - > pad_regs ;
if ( ! utmip_pad_count ) {
pr_err ( " %s: utmip pad already powered off \n " , __func__ ) ;
return - EINVAL ;
}
2012-06-05 09:59:35 +05:30
clk_prepare_enable ( phy - > pad_clk ) ;
2011-03-09 16:28:55 -08:00
spin_lock_irqsave ( & utmip_pad_lock , flags ) ;
if ( - - utmip_pad_count = = 0 ) {
val = readl ( base + UTMIP_BIAS_CFG0 ) ;
val | = UTMIP_OTGPD | UTMIP_BIASPD ;
writel ( val , base + UTMIP_BIAS_CFG0 ) ;
}
spin_unlock_irqrestore ( & utmip_pad_lock , flags ) ;
2012-06-05 09:59:35 +05:30
clk_disable_unprepare ( phy - > pad_clk ) ;
2011-03-09 16:28:55 -08:00
return 0 ;
}
static int utmi_wait_register ( void __iomem * reg , u32 mask , u32 result )
{
unsigned long timeout = 2000 ;
do {
if ( ( readl ( reg ) & mask ) = = result )
return 0 ;
udelay ( 1 ) ;
timeout - - ;
} while ( timeout ) ;
return - 1 ;
}
static void utmi_phy_clk_disable ( struct tegra_usb_phy * phy )
{
unsigned long val ;
void __iomem * base = phy - > regs ;
2013-01-16 03:30:20 +00:00
if ( phy - > is_legacy_phy ) {
2011-03-09 16:28:55 -08:00
val = readl ( base + USB_SUSP_CTRL ) ;
val | = USB_SUSP_SET ;
writel ( val , base + USB_SUSP_CTRL ) ;
udelay ( 10 ) ;
val = readl ( base + USB_SUSP_CTRL ) ;
val & = ~ USB_SUSP_SET ;
writel ( val , base + USB_SUSP_CTRL ) ;
2013-01-17 20:15:37 +00:00
} else
2013-05-16 19:43:02 +05:30
tegra_ehci_set_phcd ( & phy - > u_phy , true ) ;
2011-03-09 16:28:55 -08:00
if ( utmi_wait_register ( base + USB_SUSP_CTRL , USB_PHY_CLK_VALID , 0 ) < 0 )
pr_err ( " %s: timeout waiting for phy to stabilize \n " , __func__ ) ;
}
static void utmi_phy_clk_enable ( struct tegra_usb_phy * phy )
{
unsigned long val ;
void __iomem * base = phy - > regs ;
2013-01-16 03:30:20 +00:00
if ( phy - > is_legacy_phy ) {
2011-03-09 16:28:55 -08:00
val = readl ( base + USB_SUSP_CTRL ) ;
val | = USB_SUSP_CLR ;
writel ( val , base + USB_SUSP_CTRL ) ;
udelay ( 10 ) ;
val = readl ( base + USB_SUSP_CTRL ) ;
val & = ~ USB_SUSP_CLR ;
writel ( val , base + USB_SUSP_CTRL ) ;
2013-01-17 20:15:37 +00:00
} else
2013-05-16 19:43:02 +05:30
tegra_ehci_set_phcd ( & phy - > u_phy , false ) ;
2011-03-09 16:28:55 -08:00
if ( utmi_wait_register ( base + USB_SUSP_CTRL , USB_PHY_CLK_VALID ,
USB_PHY_CLK_VALID ) )
pr_err ( " %s: timeout waiting for phy to stabilize \n " , __func__ ) ;
}
static int utmi_phy_power_on ( struct tegra_usb_phy * phy )
{
unsigned long val ;
void __iomem * base = phy - > regs ;
struct tegra_utmip_config * config = phy - > config ;
val = readl ( base + USB_SUSP_CTRL ) ;
val | = UTMIP_RESET ;
writel ( val , base + USB_SUSP_CTRL ) ;
2013-01-16 03:30:20 +00:00
if ( phy - > is_legacy_phy ) {
2011-03-09 16:28:55 -08:00
val = readl ( base + USB1_LEGACY_CTRL ) ;
val | = USB1_NO_LEGACY_MODE ;
writel ( val , base + USB1_LEGACY_CTRL ) ;
}
val = readl ( base + UTMIP_TX_CFG0 ) ;
val & = ~ UTMIP_FS_PREABMLE_J ;
writel ( val , base + UTMIP_TX_CFG0 ) ;
val = readl ( base + UTMIP_HSRX_CFG0 ) ;
val & = ~ ( UTMIP_IDLE_WAIT ( ~ 0 ) | UTMIP_ELASTIC_LIMIT ( ~ 0 ) ) ;
val | = UTMIP_IDLE_WAIT ( config - > idle_wait_delay ) ;
val | = UTMIP_ELASTIC_LIMIT ( config - > elastic_limit ) ;
writel ( val , base + UTMIP_HSRX_CFG0 ) ;
val = readl ( base + UTMIP_HSRX_CFG1 ) ;
val & = ~ UTMIP_HS_SYNC_START_DLY ( ~ 0 ) ;
val | = UTMIP_HS_SYNC_START_DLY ( config - > hssync_start_delay ) ;
writel ( val , base + UTMIP_HSRX_CFG1 ) ;
val = readl ( base + UTMIP_DEBOUNCE_CFG0 ) ;
val & = ~ UTMIP_BIAS_DEBOUNCE_A ( ~ 0 ) ;
val | = UTMIP_BIAS_DEBOUNCE_A ( phy - > freq - > debounce ) ;
writel ( val , base + UTMIP_DEBOUNCE_CFG0 ) ;
val = readl ( base + UTMIP_MISC_CFG0 ) ;
val & = ~ UTMIP_SUSPEND_EXIT_ON_EDGE ;
writel ( val , base + UTMIP_MISC_CFG0 ) ;
val = readl ( base + UTMIP_MISC_CFG1 ) ;
val & = ~ ( UTMIP_PLL_ACTIVE_DLY_COUNT ( ~ 0 ) | UTMIP_PLLU_STABLE_COUNT ( ~ 0 ) ) ;
val | = UTMIP_PLL_ACTIVE_DLY_COUNT ( phy - > freq - > active_delay ) |
UTMIP_PLLU_STABLE_COUNT ( phy - > freq - > stable_count ) ;
writel ( val , base + UTMIP_MISC_CFG1 ) ;
val = readl ( base + UTMIP_PLL_CFG1 ) ;
val & = ~ ( UTMIP_XTAL_FREQ_COUNT ( ~ 0 ) | UTMIP_PLLU_ENABLE_DLY_COUNT ( ~ 0 ) ) ;
val | = UTMIP_XTAL_FREQ_COUNT ( phy - > freq - > xtal_freq_count ) |
UTMIP_PLLU_ENABLE_DLY_COUNT ( phy - > freq - > enable_delay ) ;
writel ( val , base + UTMIP_PLL_CFG1 ) ;
if ( phy - > mode = = TEGRA_USB_PHY_MODE_DEVICE ) {
val = readl ( base + USB_SUSP_CTRL ) ;
val & = ~ ( USB_WAKE_ON_CNNT_EN_DEV | USB_WAKE_ON_DISCON_EN_DEV ) ;
writel ( val , base + USB_SUSP_CTRL ) ;
}
utmip_pad_power_on ( phy ) ;
val = readl ( base + UTMIP_XCVR_CFG0 ) ;
val & = ~ ( UTMIP_FORCE_PD_POWERDOWN | UTMIP_FORCE_PD2_POWERDOWN |
UTMIP_FORCE_PDZI_POWERDOWN | UTMIP_XCVR_SETUP ( ~ 0 ) |
UTMIP_XCVR_LSFSLEW ( ~ 0 ) | UTMIP_XCVR_LSRSLEW ( ~ 0 ) |
UTMIP_XCVR_HSSLEW_MSB ( ~ 0 ) ) ;
val | = UTMIP_XCVR_SETUP ( config - > xcvr_setup ) ;
val | = UTMIP_XCVR_LSFSLEW ( config - > xcvr_lsfslew ) ;
val | = UTMIP_XCVR_LSRSLEW ( config - > xcvr_lsrslew ) ;
writel ( val , base + UTMIP_XCVR_CFG0 ) ;
val = readl ( base + UTMIP_XCVR_CFG1 ) ;
val & = ~ ( UTMIP_FORCE_PDDISC_POWERDOWN | UTMIP_FORCE_PDCHRP_POWERDOWN |
UTMIP_FORCE_PDDR_POWERDOWN | UTMIP_XCVR_TERM_RANGE_ADJ ( ~ 0 ) ) ;
val | = UTMIP_XCVR_TERM_RANGE_ADJ ( config - > term_range_adj ) ;
writel ( val , base + UTMIP_XCVR_CFG1 ) ;
val = readl ( base + UTMIP_BAT_CHRG_CFG0 ) ;
val & = ~ UTMIP_PD_CHRG ;
writel ( val , base + UTMIP_BAT_CHRG_CFG0 ) ;
val = readl ( base + UTMIP_BIAS_CFG1 ) ;
val & = ~ UTMIP_BIAS_PDTRK_COUNT ( ~ 0 ) ;
val | = UTMIP_BIAS_PDTRK_COUNT ( 0x5 ) ;
writel ( val , base + UTMIP_BIAS_CFG1 ) ;
2013-01-16 03:30:20 +00:00
if ( phy - > is_legacy_phy ) {
2011-03-09 16:28:55 -08:00
val = readl ( base + UTMIP_SPARE_CFG0 ) ;
if ( phy - > mode = = TEGRA_USB_PHY_MODE_DEVICE )
val & = ~ FUSE_SETUP_SEL ;
else
val | = FUSE_SETUP_SEL ;
writel ( val , base + UTMIP_SPARE_CFG0 ) ;
2013-01-16 03:30:20 +00:00
} else {
2011-03-09 16:28:55 -08:00
val = readl ( base + USB_SUSP_CTRL ) ;
val | = UTMIP_PHY_ENABLE ;
writel ( val , base + USB_SUSP_CTRL ) ;
}
val = readl ( base + USB_SUSP_CTRL ) ;
val & = ~ UTMIP_RESET ;
writel ( val , base + USB_SUSP_CTRL ) ;
2013-01-16 03:30:20 +00:00
if ( phy - > is_legacy_phy ) {
2011-03-09 16:28:55 -08:00
val = readl ( base + USB1_LEGACY_CTRL ) ;
val & = ~ USB1_VBUS_SENSE_CTL_MASK ;
val | = USB1_VBUS_SENSE_CTL_A_SESS_VLD ;
writel ( val , base + USB1_LEGACY_CTRL ) ;
val = readl ( base + USB_SUSP_CTRL ) ;
val & = ~ USB_SUSP_SET ;
writel ( val , base + USB_SUSP_CTRL ) ;
}
utmi_phy_clk_enable ( phy ) ;
2013-01-17 20:15:37 +00:00
if ( ! phy - > is_legacy_phy )
2013-05-16 19:43:02 +05:30
tegra_ehci_set_pts ( & phy - > u_phy , 0 ) ;
2011-03-09 16:28:55 -08:00
return 0 ;
}
2012-09-05 18:50:23 +05:30
static int utmi_phy_power_off ( struct tegra_usb_phy * phy )
2011-03-09 16:28:55 -08:00
{
unsigned long val ;
void __iomem * base = phy - > regs ;
utmi_phy_clk_disable ( phy ) ;
if ( phy - > mode = = TEGRA_USB_PHY_MODE_DEVICE ) {
val = readl ( base + USB_SUSP_CTRL ) ;
val & = ~ USB_WAKEUP_DEBOUNCE_COUNT ( ~ 0 ) ;
val | = USB_WAKE_ON_CNNT_EN_DEV | USB_WAKEUP_DEBOUNCE_COUNT ( 5 ) ;
writel ( val , base + USB_SUSP_CTRL ) ;
}
val = readl ( base + USB_SUSP_CTRL ) ;
val | = UTMIP_RESET ;
writel ( val , base + USB_SUSP_CTRL ) ;
val = readl ( base + UTMIP_BAT_CHRG_CFG0 ) ;
val | = UTMIP_PD_CHRG ;
writel ( val , base + UTMIP_BAT_CHRG_CFG0 ) ;
val = readl ( base + UTMIP_XCVR_CFG0 ) ;
val | = UTMIP_FORCE_PD_POWERDOWN | UTMIP_FORCE_PD2_POWERDOWN |
UTMIP_FORCE_PDZI_POWERDOWN ;
writel ( val , base + UTMIP_XCVR_CFG0 ) ;
val = readl ( base + UTMIP_XCVR_CFG1 ) ;
val | = UTMIP_FORCE_PDDISC_POWERDOWN | UTMIP_FORCE_PDCHRP_POWERDOWN |
UTMIP_FORCE_PDDR_POWERDOWN ;
writel ( val , base + UTMIP_XCVR_CFG1 ) ;
2012-09-05 18:50:23 +05:30
return utmip_pad_power_off ( phy ) ;
2011-03-09 16:28:55 -08:00
}
static void utmi_phy_preresume ( struct tegra_usb_phy * phy )
{
unsigned long val ;
void __iomem * base = phy - > regs ;
val = readl ( base + UTMIP_TX_CFG0 ) ;
val | = UTMIP_HS_DISCON_DISABLE ;
writel ( val , base + UTMIP_TX_CFG0 ) ;
}
static void utmi_phy_postresume ( struct tegra_usb_phy * phy )
{
unsigned long val ;
void __iomem * base = phy - > regs ;
val = readl ( base + UTMIP_TX_CFG0 ) ;
val & = ~ UTMIP_HS_DISCON_DISABLE ;
writel ( val , base + UTMIP_TX_CFG0 ) ;
}
static void utmi_phy_restore_start ( struct tegra_usb_phy * phy ,
enum tegra_usb_phy_port_speed port_speed )
{
unsigned long val ;
void __iomem * base = phy - > regs ;
val = readl ( base + UTMIP_MISC_CFG0 ) ;
val & = ~ UTMIP_DPDM_OBSERVE_SEL ( ~ 0 ) ;
if ( port_speed = = TEGRA_USB_PHY_PORT_SPEED_LOW )
val | = UTMIP_DPDM_OBSERVE_SEL_FS_K ;
else
val | = UTMIP_DPDM_OBSERVE_SEL_FS_J ;
writel ( val , base + UTMIP_MISC_CFG0 ) ;
udelay ( 1 ) ;
val = readl ( base + UTMIP_MISC_CFG0 ) ;
val | = UTMIP_DPDM_OBSERVE ;
writel ( val , base + UTMIP_MISC_CFG0 ) ;
udelay ( 10 ) ;
}
static void utmi_phy_restore_end ( struct tegra_usb_phy * phy )
{
unsigned long val ;
void __iomem * base = phy - > regs ;
val = readl ( base + UTMIP_MISC_CFG0 ) ;
val & = ~ UTMIP_DPDM_OBSERVE ;
writel ( val , base + UTMIP_MISC_CFG0 ) ;
udelay ( 10 ) ;
}
static int ulpi_phy_power_on ( struct tegra_usb_phy * phy )
{
int ret ;
unsigned long val ;
void __iomem * base = phy - > regs ;
2013-05-16 19:43:01 +05:30
ret = gpio_direction_output ( phy - > reset_gpio , 0 ) ;
if ( ret < 0 ) {
dev_err ( phy - > dev , " gpio %d not set to 0 \n " , phy - > reset_gpio ) ;
return ret ;
}
2011-03-09 16:28:55 -08:00
msleep ( 5 ) ;
2013-05-16 19:43:01 +05:30
ret = gpio_direction_output ( phy - > reset_gpio , 1 ) ;
if ( ret < 0 ) {
dev_err ( phy - > dev , " gpio %d not set to 1 \n " , phy - > reset_gpio ) ;
return ret ;
}
2011-03-09 16:28:55 -08:00
2012-06-05 09:59:35 +05:30
clk_prepare_enable ( phy - > clk ) ;
2011-03-09 16:28:55 -08:00
msleep ( 1 ) ;
val = readl ( base + USB_SUSP_CTRL ) ;
val | = UHSIC_RESET ;
writel ( val , base + USB_SUSP_CTRL ) ;
val = readl ( base + ULPI_TIMING_CTRL_0 ) ;
val | = ULPI_OUTPUT_PINMUX_BYP | ULPI_CLKOUT_PINMUX_BYP ;
writel ( val , base + ULPI_TIMING_CTRL_0 ) ;
val = readl ( base + USB_SUSP_CTRL ) ;
val | = ULPI_PHY_ENABLE ;
writel ( val , base + USB_SUSP_CTRL ) ;
val = 0 ;
writel ( val , base + ULPI_TIMING_CTRL_1 ) ;
val | = ULPI_DATA_TRIMMER_SEL ( 4 ) ;
val | = ULPI_STPDIRNXT_TRIMMER_SEL ( 4 ) ;
val | = ULPI_DIR_TRIMMER_SEL ( 4 ) ;
writel ( val , base + ULPI_TIMING_CTRL_1 ) ;
udelay ( 10 ) ;
val | = ULPI_DATA_TRIMMER_LOAD ;
val | = ULPI_STPDIRNXT_TRIMMER_LOAD ;
val | = ULPI_DIR_TRIMMER_LOAD ;
writel ( val , base + ULPI_TIMING_CTRL_1 ) ;
/* Fix VbusInvalid due to floating VBUS */
2012-02-13 13:24:18 +02:00
ret = usb_phy_io_write ( phy - > ulpi , 0x40 , 0x08 ) ;
2011-03-09 16:28:55 -08:00
if ( ret ) {
pr_err ( " %s: ulpi write failed \n " , __func__ ) ;
return ret ;
}
2012-02-13 13:24:18 +02:00
ret = usb_phy_io_write ( phy - > ulpi , 0x80 , 0x0B ) ;
2011-03-09 16:28:55 -08:00
if ( ret ) {
pr_err ( " %s: ulpi write failed \n " , __func__ ) ;
return ret ;
}
val = readl ( base + USB_SUSP_CTRL ) ;
val | = USB_SUSP_CLR ;
writel ( val , base + USB_SUSP_CTRL ) ;
udelay ( 100 ) ;
val = readl ( base + USB_SUSP_CTRL ) ;
val & = ~ USB_SUSP_CLR ;
writel ( val , base + USB_SUSP_CTRL ) ;
return 0 ;
}
2012-09-05 18:50:23 +05:30
static int ulpi_phy_power_off ( struct tegra_usb_phy * phy )
2011-03-09 16:28:55 -08:00
{
clk_disable ( phy - > clk ) ;
2013-05-16 19:43:00 +05:30
return gpio_direction_output ( phy - > reset_gpio , 0 ) ;
2012-09-05 18:50:23 +05:30
}
static void tegra_usb_phy_close ( struct usb_phy * x )
{
struct tegra_usb_phy * phy = container_of ( x , struct tegra_usb_phy , u_phy ) ;
clk_disable_unprepare ( phy - > pll_u ) ;
}
static int tegra_usb_phy_power_on ( struct tegra_usb_phy * phy )
{
2013-01-16 03:30:21 +00:00
if ( phy - > is_ulpi_phy )
2012-09-05 18:50:23 +05:30
return ulpi_phy_power_on ( phy ) ;
else
return utmi_phy_power_on ( phy ) ;
}
static int tegra_usb_phy_power_off ( struct tegra_usb_phy * phy )
{
2013-01-16 03:30:21 +00:00
if ( phy - > is_ulpi_phy )
2012-09-05 18:50:23 +05:30
return ulpi_phy_power_off ( phy ) ;
else
return utmi_phy_power_off ( phy ) ;
}
static int tegra_usb_phy_suspend ( struct usb_phy * x , int suspend )
{
struct tegra_usb_phy * phy = container_of ( x , struct tegra_usb_phy , u_phy ) ;
if ( suspend )
return tegra_usb_phy_power_off ( phy ) ;
else
return tegra_usb_phy_power_on ( phy ) ;
2011-03-09 16:28:55 -08:00
}
2013-05-16 19:43:02 +05:30
static int ulpi_open ( struct tegra_usb_phy * phy )
2011-03-09 16:28:55 -08:00
{
int err ;
2013-05-16 19:43:02 +05:30
phy - > clk = devm_clk_get ( phy - > dev , " ulpi-link " ) ;
if ( IS_ERR ( phy - > clk ) ) {
pr_err ( " %s: can't get ulpi clock \n " , __func__ ) ;
return PTR_ERR ( phy - > clk ) ;
}
2011-03-09 16:28:55 -08:00
2013-05-16 19:43:02 +05:30
err = devm_gpio_request ( phy - > dev , phy - > reset_gpio , " ulpi_phy_reset_b " ) ;
if ( err < 0 ) {
dev_err ( phy - > dev , " request failed for gpio: %d \n " ,
phy - > reset_gpio ) ;
return err ;
}
2011-03-09 16:28:55 -08:00
2013-05-16 19:43:02 +05:30
err = gpio_direction_output ( phy - > reset_gpio , 0 ) ;
2013-05-16 19:42:59 +05:30
if ( err < 0 ) {
2013-05-16 19:43:02 +05:30
dev_err ( phy - > dev , " gpio %d direction not set to output \n " ,
phy - > reset_gpio ) ;
return err ;
}
phy - > ulpi = otg_ulpi_create ( & ulpi_viewport_access_ops , 0 ) ;
if ( ! phy - > ulpi ) {
dev_err ( phy - > dev , " otg_ulpi_create returned NULL \n " ) ;
err = - ENOMEM ;
return err ;
}
phy - > ulpi - > io_priv = phy - > regs + ULPI_VIEWPORT ;
return 0 ;
}
static int tegra_usb_phy_init ( struct tegra_usb_phy * phy )
{
unsigned long parent_rate ;
int i ;
int err ;
if ( ! phy - > is_ulpi_phy ) {
if ( phy - > is_legacy_phy )
phy - > config = & utmip_default [ 0 ] ;
2013-05-16 19:42:59 +05:30
else
2013-05-16 19:43:02 +05:30
phy - > config = & utmip_default [ 2 ] ;
2011-03-09 16:28:55 -08:00
}
2013-05-16 19:43:02 +05:30
phy - > pll_u = devm_clk_get ( phy - > dev , " pll_u " ) ;
2011-03-09 16:28:55 -08:00
if ( IS_ERR ( phy - > pll_u ) ) {
pr_err ( " Can't get pll_u clock \n " ) ;
2013-05-16 19:43:02 +05:30
return PTR_ERR ( phy - > pll_u ) ;
2011-03-09 16:28:55 -08:00
}
2013-05-16 19:43:02 +05:30
err = clk_prepare_enable ( phy - > pll_u ) ;
if ( err )
return err ;
2011-03-09 16:28:55 -08:00
parent_rate = clk_get_rate ( clk_get_parent ( phy - > pll_u ) ) ;
for ( i = 0 ; i < ARRAY_SIZE ( tegra_freq_table ) ; i + + ) {
if ( tegra_freq_table [ i ] . freq = = parent_rate ) {
phy - > freq = & tegra_freq_table [ i ] ;
break ;
}
}
if ( ! phy - > freq ) {
pr_err ( " invalid pll_u parent rate %ld \n " , parent_rate ) ;
err = - EINVAL ;
2013-05-16 19:43:02 +05:30
goto fail ;
2011-03-09 16:28:55 -08:00
}
2013-05-16 19:43:02 +05:30
if ( phy - > is_ulpi_phy )
err = ulpi_open ( phy ) ;
else
err = utmip_pad_open ( phy ) ;
if ( err < 0 )
goto fail ;
2011-03-09 16:28:55 -08:00
2013-05-16 19:43:02 +05:30
return 0 ;
2011-03-09 16:28:55 -08:00
2013-05-16 19:43:02 +05:30
fail :
2012-06-05 09:59:35 +05:30
clk_disable_unprepare ( phy - > pll_u ) ;
2013-05-16 19:43:02 +05:30
return err ;
2011-03-09 16:28:55 -08:00
}
2013-01-24 15:57:03 +05:30
void tegra_usb_phy_preresume ( struct usb_phy * x )
2011-03-09 16:28:55 -08:00
{
2013-01-24 15:57:03 +05:30
struct tegra_usb_phy * phy = container_of ( x , struct tegra_usb_phy , u_phy ) ;
2013-01-16 03:30:21 +00:00
if ( ! phy - > is_ulpi_phy )
2011-03-09 16:28:55 -08:00
utmi_phy_preresume ( phy ) ;
}
2012-03-02 15:58:42 -05:00
EXPORT_SYMBOL_GPL ( tegra_usb_phy_preresume ) ;
2011-03-09 16:28:55 -08:00
2013-01-24 15:57:03 +05:30
void tegra_usb_phy_postresume ( struct usb_phy * x )
2011-03-09 16:28:55 -08:00
{
2013-01-24 15:57:03 +05:30
struct tegra_usb_phy * phy = container_of ( x , struct tegra_usb_phy , u_phy ) ;
2013-01-16 03:30:21 +00:00
if ( ! phy - > is_ulpi_phy )
2011-03-09 16:28:55 -08:00
utmi_phy_postresume ( phy ) ;
}
2012-03-02 15:58:42 -05:00
EXPORT_SYMBOL_GPL ( tegra_usb_phy_postresume ) ;
2011-03-09 16:28:55 -08:00
2013-01-24 15:57:03 +05:30
void tegra_ehci_phy_restore_start ( struct usb_phy * x ,
2011-03-09 16:28:55 -08:00
enum tegra_usb_phy_port_speed port_speed )
{
2013-01-24 15:57:03 +05:30
struct tegra_usb_phy * phy = container_of ( x , struct tegra_usb_phy , u_phy ) ;
2013-01-16 03:30:21 +00:00
if ( ! phy - > is_ulpi_phy )
2011-03-09 16:28:55 -08:00
utmi_phy_restore_start ( phy , port_speed ) ;
}
2012-03-02 15:58:42 -05:00
EXPORT_SYMBOL_GPL ( tegra_ehci_phy_restore_start ) ;
2011-03-09 16:28:55 -08:00
2013-01-24 15:57:03 +05:30
void tegra_ehci_phy_restore_end ( struct usb_phy * x )
2011-03-09 16:28:55 -08:00
{
2013-01-24 15:57:03 +05:30
struct tegra_usb_phy * phy = container_of ( x , struct tegra_usb_phy , u_phy ) ;
2013-01-16 03:30:21 +00:00
if ( ! phy - > is_ulpi_phy )
2011-03-09 16:28:55 -08:00
utmi_phy_restore_end ( phy ) ;
}
2012-03-02 15:58:42 -05:00
EXPORT_SYMBOL_GPL ( tegra_ehci_phy_restore_end ) ;
2011-03-09 16:28:55 -08:00
2013-05-16 19:43:02 +05:30
static int tegra_usb_phy_probe ( struct platform_device * pdev )
{
struct resource * res ;
struct tegra_usb_phy * tegra_phy = NULL ;
struct device_node * np = pdev - > dev . of_node ;
int err ;
tegra_phy = devm_kzalloc ( & pdev - > dev , sizeof ( * tegra_phy ) , GFP_KERNEL ) ;
if ( ! tegra_phy ) {
dev_err ( & pdev - > dev , " unable to allocate memory for USB2 PHY \n " ) ;
return - ENOMEM ;
}
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( ! res ) {
dev_err ( & pdev - > dev , " Failed to get I/O memory \n " ) ;
return - ENXIO ;
}
tegra_phy - > regs = devm_ioremap ( & pdev - > dev , res - > start ,
resource_size ( res ) ) ;
if ( ! tegra_phy - > regs ) {
dev_err ( & pdev - > dev , " Failed to remap I/O memory \n " ) ;
return - ENOMEM ;
}
tegra_phy - > is_legacy_phy =
of_property_read_bool ( np , " nvidia,has-legacy-mode " ) ;
err = of_property_match_string ( np , " phy_type " , " ulpi " ) ;
if ( err < 0 ) {
tegra_phy - > is_ulpi_phy = false ;
res = platform_get_resource ( pdev , IORESOURCE_MEM , 1 ) ;
if ( ! res ) {
dev_err ( & pdev - > dev , " Failed to get UTMI Pad regs \n " ) ;
return - ENXIO ;
}
tegra_phy - > pad_regs = devm_ioremap ( & pdev - > dev , res - > start ,
resource_size ( res ) ) ;
if ( ! tegra_phy - > regs ) {
dev_err ( & pdev - > dev , " Failed to remap UTMI Pad regs \n " ) ;
return - ENOMEM ;
}
} else {
tegra_phy - > is_ulpi_phy = true ;
tegra_phy - > reset_gpio =
of_get_named_gpio ( np , " nvidia,phy-reset-gpio " , 0 ) ;
if ( ! gpio_is_valid ( tegra_phy - > reset_gpio ) ) {
dev_err ( & pdev - > dev , " invalid gpio: %d \n " ,
tegra_phy - > reset_gpio ) ;
return tegra_phy - > reset_gpio ;
}
}
err = of_property_match_string ( np , " dr_mode " , " otg " ) ;
if ( err < 0 ) {
err = of_property_match_string ( np , " dr_mode " , " peripheral " ) ;
if ( err < 0 )
tegra_phy - > mode = TEGRA_USB_PHY_MODE_HOST ;
else
tegra_phy - > mode = TEGRA_USB_PHY_MODE_DEVICE ;
} else
tegra_phy - > mode = TEGRA_USB_PHY_MODE_OTG ;
tegra_phy - > dev = & pdev - > dev ;
err = tegra_usb_phy_init ( tegra_phy ) ;
if ( err < 0 )
return err ;
tegra_phy - > u_phy . shutdown = tegra_usb_phy_close ;
tegra_phy - > u_phy . set_suspend = tegra_usb_phy_suspend ;
dev_set_drvdata ( & pdev - > dev , tegra_phy ) ;
return 0 ;
}
static struct of_device_id tegra_usb_phy_id_table [ ] = {
{ . compatible = " nvidia,tegra20-usb-phy " , } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , tegra_usb_phy_id_table ) ;
static struct platform_driver tegra_usb_phy_driver = {
. probe = tegra_usb_phy_probe ,
. driver = {
. name = " tegra-phy " ,
. owner = THIS_MODULE ,
. of_match_table = of_match_ptr ( tegra_usb_phy_id_table ) ,
} ,
} ;
module_platform_driver ( tegra_usb_phy_driver ) ;
static int tegra_usb_phy_match ( struct device * dev , void * data )
{
struct tegra_usb_phy * tegra_phy = dev_get_drvdata ( dev ) ;
struct device_node * dn = data ;
return ( tegra_phy - > dev - > of_node = = dn ) ? 1 : 0 ;
}
struct usb_phy * tegra_usb_get_phy ( struct device_node * dn )
{
struct device * dev ;
struct tegra_usb_phy * tegra_phy ;
dev = driver_find_device ( & tegra_usb_phy_driver . driver , NULL , dn ,
tegra_usb_phy_match ) ;
if ( ! dev )
return ERR_PTR ( - EPROBE_DEFER ) ;
tegra_phy = dev_get_drvdata ( dev ) ;
return & tegra_phy - > u_phy ;
}
EXPORT_SYMBOL_GPL ( tegra_usb_get_phy ) ;
2013-06-13 11:24:08 -06:00
MODULE_DESCRIPTION ( " Tegra USB PHY driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;