blob: 561c8a3f5bbbb0c174580fdba7e7782644ef0457 [file] [log] [blame]
/*
* arch/arm/mach-tegra/tegra3_usb_phy.c
*
* Copyright (c) 2012, NVIDIA CORPORATION. All rights reserved.
*
*
* 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/interrupt.h>
#include <linux/slab.h>
#include <linux/err.h>
#include <linux/platform_device.h>
#include <linux/io.h>
#include <linux/gpio.h>
#include <linux/interrupt.h>
#include <linux/clk.h>
#include <linux/regulator/consumer.h>
#include <linux/platform_data/tegra_usb.h>
#include <linux/clk/tegra.h>
#include <mach/pinmux.h>
#include <mach/pinmux-tegra30.h>
#include "tegra_usb_phy.h"
#include "../../../arch/arm/mach-tegra/gpio-names.h"
#include "../../../arch/arm/mach-tegra/fuse.h"
#include "../../../arch/arm/mach-tegra/clock.h"
/* HACK! This needs to come from DT */
#include "../../../arch/arm/mach-tegra/iomap.h"
#define USB_USBCMD 0x130
#define USB_USBCMD_RS (1 << 0)
#define USB_CMD_RESET (1<<1)
#define USB_USBSTS 0x134
#define USB_USBSTS_PCI (1 << 2)
#define USB_USBSTS_SRI (1 << 7)
#define USB_USBSTS_HCH (1 << 12)
#define USB_USBINTR 0x138
#define USB_TXFILLTUNING 0x154
#define USB_FIFO_TXFILL_THRES(x) (((x) & 0x1f) << 16)
#define USB_FIFO_TXFILL_MASK 0x1f0000
#define USB_ASYNCLISTADDR 0x148
#define ICUSB_CTRL 0x15c
#define USB_PORTSC 0x174
#define USB_PORTSC_WKOC (1 << 22)
#define USB_PORTSC_WKDS (1 << 21)
#define USB_PORTSC_WKCN (1 << 20)
#define USB_PORTSC_PTC(x) (((x) & 0xf) << 16)
#define USB_PORTSC_PP (1 << 12)
#define USB_PORTSC_LS(x) (((x) & 0x3) << 10)
#define USB_PORTSC_SUSP (1 << 7)
#define USB_PORTSC_RESUME (1 << 6)
#define USB_PORTSC_OCC (1 << 5)
#define USB_PORTSC_PEC (1 << 3)
#define USB_PORTSC_PE (1 << 2)
#define USB_PORTSC_CSC (1 << 1)
#define USB_PORTSC_CCS (1 << 0)
#define USB_PORTSC_RWC_BITS (USB_PORTSC_CSC | USB_PORTSC_PEC | USB_PORTSC_OCC)
#define HOSTPC1_DEVLC 0x1b4
#define HOSTPC1_DEVLC_PHCD (1 << 22)
#define HOSTPC1_DEVLC_PTS(x) (((x) & 0x7) << 29)
#define HOSTPC1_DEVLC_PTS_MASK 7
#define HOSTPC1_DEVLC_PTS_HSIC 4
#define HOSTPC1_DEVLC_STS (1 << 28)
#define HOSTPC1_DEVLC_PSPD(x) (((x) & 0x3) << 25)
#define HOSTPC1_DEVLC_PSPD_MASK 3
#define HOSTPC1_DEVLC_PSPD_HIGH_SPEED 2
#define USB_USBMODE 0x1f8
#define USB_USBMODE_MASK (3 << 0)
#define USB_USBMODE_HOST (3 << 0)
#define USB_USBMODE_DEVICE (2 << 0)
#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 USB_PHY_CLK_VALID_INT_ENB (1 << 9)
#define USB_PHY_CLK_VALID_INT_STS (1 << 8)
#define UTMIP_RESET (1 << 11)
#define UTMIP_PHY_ENABLE (1 << 12)
#define ULPI_PHY_ENABLE (1 << 13)
#define UHSIC_RESET (1 << 14)
#define USB_WAKEUP_DEBOUNCE_COUNT(x) (((x) & 0x7) << 16)
#define UHSIC_PHY_ENABLE (1 << 19)
#define ULPIS2S_SLV0_RESET (1 << 20)
#define ULPIS2S_SLV1_RESET (1 << 21)
#define ULPIS2S_LINE_RESET (1 << 22)
#define ULPI_PADS_RESET (1 << 23)
#define ULPI_PADS_CLKEN_RESET (1 << 24)
#define USB_PHY_VBUS_WAKEUP_ID 0x408
#define VDAT_DET_INT_EN (1 << 16)
#define VDAT_DET_CHG_DET (1 << 17)
#define VDAT_DET_STS (1 << 18)
#define USB_ID_STATUS (1 << 2)
#define ULPIS2S_CTRL 0x418
#define ULPIS2S_ENA (1 << 0)
#define ULPIS2S_SUPPORT_DISCONNECT (1 << 2)
#define ULPIS2S_PLLU_MASTER_BLASTER60 (1 << 3)
#define ULPIS2S_SPARE(x) (((x) & 0xF) << 8)
#define ULPIS2S_FORCE_ULPI_CLK_OUT (1 << 12)
#define ULPIS2S_DISCON_DONT_CHECK_SE0 (1 << 13)
#define ULPIS2S_SUPPORT_HS_KEEP_ALIVE (1 << 14)
#define ULPIS2S_DISABLE_STP_PU (1 << 15)
#define ULPIS2S_SLV0_CLAMP_XMIT (1 << 16)
#define ULPI_TIMING_CTRL_0 0x424
#define ULPI_CLOCK_OUT_DELAY(x) ((x) & 0x1F)
#define ULPI_OUTPUT_PINMUX_BYP (1 << 10)
#define ULPI_CLKOUT_PINMUX_BYP (1 << 11)
#define ULPI_SHADOW_CLK_LOOPBACK_EN (1 << 12)
#define ULPI_SHADOW_CLK_SEL (1 << 13)
#define ULPI_CORE_CLK_SEL (1 << 14)
#define ULPI_SHADOW_CLK_DELAY(x) (((x) & 0x1F) << 16)
#define ULPI_LBK_PAD_EN (1 << 26)
#define ULPI_LBK_PAD_E_INPUT_OR (1 << 27)
#define ULPI_CLK_OUT_ENA (1 << 28)
#define ULPI_CLK_PADOUT_ENA (1 << 29)
#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_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_LSBIAS_SEL (1 << 21)
#define UTMIP_XCVR_SETUP_MSB(x) (((x) & 0x7) << 22)
#define UTMIP_XCVR_HSSLEW_MSB(x) (((x) & 0x7f) << 25)
#define UTMIP_XCVR_MAX_OFFSET 2
#define UTMIP_XCVR_SETUP_MAX_VALUE 0x7f
#define UTMIP_XCVR_SETUP_MIN_VALUE 0
#define XCVR_SETUP_MSB_CALIB(x) ((x) >> 4)
#define UTMIP_BIAS_CFG0 0x80c
#define UTMIP_OTGPD (1 << 11)
#define UTMIP_BIASPD (1 << 10)
#define UTMIP_HSSQUELCH_LEVEL(x) (((x) & 0x3) << 0)
#define UTMIP_HSDISCON_LEVEL(x) (((x) & 0x3) << 2)
#define UTMIP_HSDISCON_LEVEL_MSB (1 << 24)
#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_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_ON_SINK_EN (1 << 2)
#define UTMIP_OP_SRC_EN (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)
#define UTMIP_BIAS_PDTRK_POWERDOWN (1 << 0)
#define UTMIP_BIAS_PDTRK_POWERUP (1 << 1)
#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 FORCE_PULLDN_DM (1 << 8)
#define FORCE_PULLDN_DP (1 << 9)
#define COMB_TERMS (1 << 0)
#define ALWAYS_FREE_RUNNING_TERMS (1 << 1)
#define UTMIP_SPARE_CFG0 0x834
#define FUSE_SETUP_SEL (1 << 3)
#define FUSE_ATERM_SEL (1 << 4)
#define UTMIP_PMC_WAKEUP0 0x84c
#define EVENT_INT_ENB (1 << 0)
#define UHSIC_PMC_WAKEUP0 0xc34
#define UTMIP_BIAS_STS0 0x840
#define UTMIP_RCTRL_VAL(x) (((x) & 0xffff) << 0)
#define UTMIP_TCTRL_VAL(x) (((x) & (0xffff << 16)) >> 16)
#define UHSIC_PLL_CFG1 0xc04
#define UHSIC_XTAL_FREQ_COUNT(x) (((x) & 0xfff) << 0)
#define UHSIC_PLLU_ENABLE_DLY_COUNT(x) (((x) & 0x1f) << 14)
#define UHSIC_HSRX_CFG0 0xc08
#define UHSIC_ELASTIC_UNDERRUN_LIMIT(x) (((x) & 0x1f) << 2)
#define UHSIC_ELASTIC_OVERRUN_LIMIT(x) (((x) & 0x1f) << 8)
#define UHSIC_IDLE_WAIT(x) (((x) & 0x1f) << 13)
#define UHSIC_HSRX_CFG1 0xc0c
#define UHSIC_HS_SYNC_START_DLY(x) (((x) & 0x1f) << 1)
#define UHSIC_TX_CFG0 0xc10
#define UHSIC_HS_READY_WAIT_FOR_VALID (1 << 9)
#define UHSIC_MISC_CFG0 0xc14
#define UHSIC_SUSPEND_EXIT_ON_EDGE (1 << 7)
#define UHSIC_DETECT_SHORT_CONNECT (1 << 8)
#define UHSIC_FORCE_XCVR_MODE (1 << 15)
#define UHSIC_DISABLE_BUSRESET (1 << 20)
#define UHSIC_MISC_CFG1 0xc18
#define UHSIC_PLLU_STABLE_COUNT(x) (((x) & 0xfff) << 2)
#define UHSIC_PADS_CFG0 0xc1c
#define UHSIC_TX_RTUNEN 0xf000
#define UHSIC_TX_RTUNE(x) (((x) & 0xf) << 12)
#define UHSIC_PADS_CFG1 0xc20
#define UHSIC_PD_BG (1 << 2)
#define UHSIC_PD_TX (1 << 3)
#define UHSIC_PD_TRK (1 << 4)
#define UHSIC_PD_RX (1 << 5)
#define UHSIC_PD_ZI (1 << 6)
#define UHSIC_RX_SEL (1 << 7)
#define UHSIC_RPD_DATA (1 << 9)
#define UHSIC_RPD_STROBE (1 << 10)
#define UHSIC_RPU_DATA (1 << 11)
#define UHSIC_RPU_STROBE (1 << 12)
#define UHSIC_CMD_CFG0 0xc24
#define UHSIC_PRETEND_CONNECT_DETECT (1 << 5)
#define UHSIC_STAT_CFG0 0xc28
#define UHSIC_CONNECT_DETECT (1 << 0)
#define PMC_USB_DEBOUNCE 0xec
#define UTMIP_LINE_DEB_CNT(x) (((x) & 0xf) << 16)
#define UHSIC_LINE_DEB_CNT(x) (((x) & 0xf) << 20)
#define PMC_USB_DEBOUNCE_VAL(x) ((x) & 0xffff)
#define PMC_USB_AO 0xf0
#define PMC_POWER_DOWN_MASK 0xffff
#define HSIC_RESERVED_P0 (3 << 14)
#define STROBE_VAL_PD_P0 (1 << 12)
#define DATA_VAL_PD_P0 (1 << 13)
#define USB_ID_PD(inst) (1 << ((4*(inst))+3))
#define VBUS_WAKEUP_PD(inst) (1 << ((4*(inst))+2))
#define USBON_VAL_PD(inst) (1 << ((4*(inst))+1))
#define USBON_VAL_PD_P2 (1 << 9)
#define USBON_VAL_PD_P1 (1 << 5)
#define USBON_VAL_PD_P0 (1 << 1)
#define USBOP_VAL_PD(inst) (1 << (4*(inst)))
#define USBOP_VAL_PD_P2 (1 << 8)
#define USBOP_VAL_PD_P1 (1 << 4)
#define USBOP_VAL_PD_P0 (1 << 0)
#define PMC_USB_AO_ID_PD_P0 (1 << 3)
#define PMC_USB_AO_VBUS_WAKEUP_PD_P0 (1 << 2)
#define PMC_TRIGGERS 0x1ec
#define UHSIC_CLR_WALK_PTR_P0 (1 << 3)
#define UTMIP_CLR_WALK_PTR(inst) (1 << (inst))
#define UTMIP_CLR_WALK_PTR_P2 (1 << 2)
#define UTMIP_CLR_WALK_PTR_P1 (1 << 1)
#define UTMIP_CLR_WALK_PTR_P0 (1 << 0)
#define UTMIP_CAP_CFG(inst) (1 << ((inst)+4))
#define UTMIP_CAP_CFG_P2 (1 << 6)
#define UTMIP_CAP_CFG_P1 (1 << 5)
#define UTMIP_CAP_CFG_P0 (1 << 4)
#define UTMIP_CLR_WAKE_ALARM(inst) (1 << ((inst)+12))
#define UHSIC_CLR_WAKE_ALARM_P0 (1 << 15)
#define UTMIP_CLR_WAKE_ALARM_P2 (1 << 14)
#define PMC_PAD_CFG (0x1f4)
#define PMC_UTMIP_TERM_PAD_CFG 0x1f8
#define PMC_TCTRL_VAL(x) (((x) & 0x1f) << 5)
#define PMC_RCTRL_VAL(x) (((x) & 0x1f) << 0)
#define PMC_SLEEP_CFG 0x1fc
#define UHSIC_MASTER_ENABLE (1 << 24)
#define UHSIC_WAKE_VAL(x) (((x) & 0xf) << 28)
#define WAKE_VAL_SD10 0x2
#define UTMIP_TCTRL_USE_PMC(inst) (1 << ((8*(inst))+3))
#define UTMIP_TCTRL_USE_PMC_P2 (1 << 19)
#define UTMIP_TCTRL_USE_PMC_P1 (1 << 11)
#define UTMIP_TCTRL_USE_PMC_P0 (1 << 3)
#define UTMIP_RCTRL_USE_PMC(inst) (1 << ((8*(inst))+2))
#define UTMIP_RCTRL_USE_PMC_P2 (1 << 18)
#define UTMIP_RCTRL_USE_PMC_P1 (1 << 10)
#define UTMIP_RCTRL_USE_PMC_P0 (1 << 2)
#define UTMIP_FSLS_USE_PMC(inst) (1 << ((8*(inst))+1))
#define UTMIP_FSLS_USE_PMC_P2 (1 << 17)
#define UTMIP_FSLS_USE_PMC_P1 (1 << 9)
#define UTMIP_FSLS_USE_PMC_P0 (1 << 1)
#define UTMIP_MASTER_ENABLE(inst) (1 << (8*(inst)))
#define UTMIP_MASTER_ENABLE_P2 (1 << 16)
#define UTMIP_MASTER_ENABLE_P1 (1 << 8)
#define UTMIP_MASTER_ENABLE_P0 (1 << 0)
#define UHSIC_MASTER_ENABLE_P0 (1 << 24)
#define UHSIC_WAKE_VAL_P0(x) (((x) & 0xf) << 28)
#define PMC_SLEEPWALK_CFG 0x200
#define UHSIC_WAKE_WALK_EN_P0 (1 << 30)
#define UHSIC_LINEVAL_WALK_EN (1 << 31)
#define UTMIP_LINEVAL_WALK_EN(inst) (1 << ((8*(inst))+7))
#define UTMIP_LINEVAL_WALK_EN_P2 (1 << 23)
#define UTMIP_LINEVAL_WALK_EN_P1 (1 << 15)
#define UTMIP_LINEVAL_WALK_EN_P0 (1 << 7)
#define UTMIP_WAKE_VAL(inst, x) (((x) & 0xf) << ((8*(inst))+4))
#define UTMIP_WAKE_VAL_P2(x) (((x) & 0xf) << 20)
#define UTMIP_WAKE_VAL_P1(x) (((x) & 0xf) << 12)
#define UTMIP_WAKE_VAL_P0(x) (((x) & 0xf) << 4)
#define WAKE_VAL_NONE 0xc
#define WAKE_VAL_ANY 0xF
#define WAKE_VAL_FSJ 0x2
#define WAKE_VAL_FSK 0x1
#define WAKE_VAL_SE0 0x0
#define PMC_SLEEPWALK_REG(inst) (0x204 + (4*(inst)))
#define UTMIP_USBOP_RPD_A (1 << 0)
#define UTMIP_USBON_RPD_A (1 << 1)
#define UTMIP_AP_A (1 << 4)
#define UTMIP_AN_A (1 << 5)
#define UTMIP_HIGHZ_A (1 << 6)
#define UTMIP_USBOP_RPD_B (1 << 8)
#define UTMIP_USBON_RPD_B (1 << 9)
#define UTMIP_AP_B (1 << 12)
#define UTMIP_AN_B (1 << 13)
#define UTMIP_HIGHZ_B (1 << 14)
#define UTMIP_USBOP_RPD_C (1 << 16)
#define UTMIP_USBON_RPD_C (1 << 17)
#define UTMIP_AP_C (1 << 20)
#define UTMIP_AN_C (1 << 21)
#define UTMIP_HIGHZ_C (1 << 22)
#define UTMIP_USBOP_RPD_D (1 << 24)
#define UTMIP_USBON_RPD_D (1 << 25)
#define UTMIP_AP_D (1 << 28)
#define UTMIP_AN_D (1 << 29)
#define UTMIP_HIGHZ_D (1 << 30)
#define PMC_SLEEPWALK_UHSIC 0x210
#define UHSIC_STROBE_RPD_A (1 << 0)
#define UHSIC_DATA_RPD_A (1 << 1)
#define UHSIC_STROBE_RPU_A (1 << 2)
#define UHSIC_DATA_RPU_A (1 << 3)
#define UHSIC_STROBE_RPD_B (1 << 8)
#define UHSIC_DATA_RPD_B (1 << 9)
#define UHSIC_STROBE_RPU_B (1 << 10)
#define UHSIC_DATA_RPU_B (1 << 11)
#define UHSIC_STROBE_RPD_C (1 << 16)
#define UHSIC_DATA_RPD_C (1 << 17)
#define UHSIC_STROBE_RPU_C (1 << 18)
#define UHSIC_DATA_RPU_C (1 << 19)
#define UHSIC_STROBE_RPD_D (1 << 24)
#define UHSIC_DATA_RPD_D (1 << 25)
#define UHSIC_STROBE_RPU_D (1 << 26)
#define UHSIC_DATA_RPU_D (1 << 27)
#define UTMIP_UHSIC_STATUS 0x214
#define UTMIP_USBOP_VAL(inst) (1 << ((2*(inst)) + 8))
#define UTMIP_USBOP_VAL_P2 (1 << 12)
#define UTMIP_USBOP_VAL_P1 (1 << 10)
#define UTMIP_USBOP_VAL_P0 (1 << 8)
#define UTMIP_USBON_VAL(inst) (1 << ((2*(inst)) + 9))
#define UTMIP_USBON_VAL_P2 (1 << 13)
#define UTMIP_USBON_VAL_P1 (1 << 11)
#define UTMIP_USBON_VAL_P0 (1 << 9)
#define UHSIC_WAKE_ALARM (1 << 19)
#define UTMIP_WAKE_ALARM(inst) (1 << ((inst) + 16))
#define UTMIP_WAKE_ALARM_P2 (1 << 18)
#define UTMIP_WAKE_ALARM_P1 (1 << 17)
#define UTMIP_WAKE_ALARM_P0 (1 << 16)
#define UHSIC_DATA_VAL_P0 (1 << 15)
#define UHSIC_STROBE_VAL_P0 (1 << 14)
#define UTMIP_WALK_PTR_VAL(inst) (0x3 << ((inst)*2))
#define UHSIC_WALK_PTR_VAL (0x3 << 6)
#define UTMIP_WALK_PTR(inst) (1 << ((inst)*2))
#define UTMIP_WALK_PTR_P2 (1 << 4)
#define UTMIP_WALK_PTR_P1 (1 << 2)
#define UTMIP_WALK_PTR_P0 (1 << 0)
#define USB1_PREFETCH_ID 6
#define USB2_PREFETCH_ID 18
#define USB3_PREFETCH_ID 17
#define PMC_UTMIP_UHSIC_FAKE 0x218
#define UHSIC_STROBE_VAL (1 << 12)
#define UHSIC_DATA_VAL (1 << 13)
#define UHSIC_STROBE_ENB (1 << 14)
#define UHSIC_DATA_ENB (1 << 15)
#define USBON_VAL(inst) (1 << ((4*(inst))+1))
#define USBON_VAL_P2 (1 << 9)
#define USBON_VAL_P1 (1 << 5)
#define USBON_VAL_P0 (1 << 1)
#define USBOP_VAL(inst) (1 << (4*(inst)))
#define USBOP_VAL_P2 (1 << 8)
#define USBOP_VAL_P1 (1 << 4)
#define USBOP_VAL_P0 (1 << 0)
#define PMC_UTMIP_BIAS_MASTER_CNTRL 0x30c
#define BIAS_MASTER_PROG_VAL (1 << 1)
#define PMC_UTMIP_MASTER_CONFIG 0x310
#define UTMIP_PWR(inst) (1 << (inst))
#define UHSIC_PWR (1 << 3)
#define FUSE_USB_CALIB_0 0x1F0
#define XCVR_SETUP(x) (((x) & 0x7F) << 0)
#define XCVR_SETUP_LSB_MASK 0xF
#define XCVR_SETUP_MSB_MASK 0x70
#define XCVR_SETUP_LSB_MAX_VAL 0xF
#define APB_MISC_GP_OBSCTRL_0 0x818
#define APB_MISC_GP_OBSDATA_0 0x81c
/* ULPI GPIO */
#define ULPI_STP TEGRA_GPIO_PY3
#define ULPI_DIR TEGRA_GPIO_PY1
#define ULPI_D0 TEGRA_GPIO_PO1
#define ULPI_D1 TEGRA_GPIO_PO2
/* These values (in milli second) are taken from the battery charging spec */
#define TDP_SRC_ON_MS 100
#define TDPSRC_CON_MS 40
#ifdef DEBUG
#define DBG(stuff...) pr_info("tegra3_usb_phy: " stuff)
#else
#define DBG(stuff...) do {} while (0)
#endif
#if 0
#define PHY_DBG(stuff...) pr_info("tegra3_usb_phy: " stuff)
#else
#define PHY_DBG(stuff...) do {} while (0)
#endif
/* define HSIC phy params */
#define HSIC_SYNC_START_DELAY 9
#define HSIC_IDLE_WAIT_DELAY 17
#define HSIC_ELASTIC_UNDERRUN_LIMIT 16
#define HSIC_ELASTIC_OVERRUN_LIMIT 16
static u32 utmip_rctrl_val, utmip_tctrl_val;
static DEFINE_SPINLOCK(utmip_pad_lock);
static int utmip_pad_count;
static struct tegra_xtal_freq utmip_freq_table[] = {
{
.freq = 12000000,
.enable_delay = 0x02,
.stable_count = 0x2F,
.active_delay = 0x04,
.xtal_freq_count = 0x76,
.debounce = 0x7530,
.pdtrk_count = 5,
},
{
.freq = 13000000,
.enable_delay = 0x02,
.stable_count = 0x33,
.active_delay = 0x05,
.xtal_freq_count = 0x7F,
.debounce = 0x7EF4,
.pdtrk_count = 5,
},
{
.freq = 19200000,
.enable_delay = 0x03,
.stable_count = 0x4B,
.active_delay = 0x06,
.xtal_freq_count = 0xBB,
.debounce = 0xBB80,
.pdtrk_count = 7,
},
{
.freq = 26000000,
.enable_delay = 0x04,
.stable_count = 0x66,
.active_delay = 0x09,
.xtal_freq_count = 0xFE,
.debounce = 0xFDE8,
.pdtrk_count = 9,
},
};
static struct tegra_xtal_freq uhsic_freq_table[] = {
{
.freq = 12000000,
.enable_delay = 0x02,
.stable_count = 0x2F,
.active_delay = 0x0,
.xtal_freq_count = 0x1CA,
},
{
.freq = 13000000,
.enable_delay = 0x02,
.stable_count = 0x33,
.active_delay = 0x0,
.xtal_freq_count = 0x1F0,
},
{
.freq = 19200000,
.enable_delay = 0x03,
.stable_count = 0x4B,
.active_delay = 0x0,
.xtal_freq_count = 0x2DD,
},
{
.freq = 26000000,
.enable_delay = 0x04,
.stable_count = 0x66,
.active_delay = 0x0,
.xtal_freq_count = 0x3E0,
},
};
static void usb_phy_fence_read(struct tegra_usb_phy *phy)
{
/* Fence read for coherency of AHB master intiated writes */
if (phy->inst == 0)
readb(IO_ADDRESS(IO_PPCS_PHYS + USB1_PREFETCH_ID));
else if (phy->inst == 1)
readb(IO_ADDRESS(IO_PPCS_PHYS + USB2_PREFETCH_ID));
else if (phy->inst == 2)
readb(IO_ADDRESS(IO_PPCS_PHYS + USB3_PREFETCH_ID));
return;
}
static void utmip_setup_pmc_wake_detect(struct tegra_usb_phy *phy)
{
unsigned long val, pmc_pad_cfg_val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
unsigned int inst = phy->inst;
void __iomem *base = phy->regs;
bool port_connected;
enum usb_phy_port_speed port_speed;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
/* check for port connect status */
val = readl(base + USB_PORTSC);
port_connected = val & USB_PORTSC_CCS;
if (!port_connected)
return;
port_speed = (readl(base + HOSTPC1_DEVLC) >> 25) &
HOSTPC1_DEVLC_PSPD_MASK;
/*Set PMC MASTER bits to do the following
* a. Take over the UTMI drivers
* b. set up such that it will take over resume
* if remote wakeup is detected
* Prepare PMC to take over suspend-wake detect-drive resume until USB
* controller ready
*/
/* disable master enable in PMC */
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UTMIP_MASTER_ENABLE(inst);
writel(val, pmc_base + PMC_SLEEP_CFG);
/* UTMIP_PWR_PX=1 for power savings mode */
val = readl(pmc_base + PMC_UTMIP_MASTER_CONFIG);
val |= UTMIP_PWR(inst);
writel(val, pmc_base + PMC_UTMIP_MASTER_CONFIG);
/* config debouncer */
val = readl(pmc_base + PMC_USB_DEBOUNCE);
val &= ~UTMIP_LINE_DEB_CNT(~0);
val |= UTMIP_LINE_DEB_CNT(1);
val |= PMC_USB_DEBOUNCE_VAL(2);
writel(val, pmc_base + PMC_USB_DEBOUNCE);
/* Make sure nothing is happening on the line with respect to PMC */
val = readl(pmc_base + PMC_UTMIP_UHSIC_FAKE);
val &= ~USBOP_VAL(inst);
val &= ~USBON_VAL(inst);
writel(val, pmc_base + PMC_UTMIP_UHSIC_FAKE);
/* Make sure wake value for line is none */
val = readl(pmc_base + PMC_SLEEPWALK_CFG);
val &= ~UTMIP_LINEVAL_WALK_EN(inst);
writel(val, pmc_base + PMC_SLEEPWALK_CFG);
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UTMIP_WAKE_VAL(inst, ~0);
val |= UTMIP_WAKE_VAL(inst, WAKE_VAL_NONE);
writel(val, pmc_base + PMC_SLEEP_CFG);
/* turn off pad detectors */
val = readl(pmc_base + PMC_USB_AO);
val |= (USBOP_VAL_PD(inst) | USBON_VAL_PD(inst));
writel(val, pmc_base + PMC_USB_AO);
/* Remove fake values and make synchronizers work a bit */
val = readl(pmc_base + PMC_UTMIP_UHSIC_FAKE);
val &= ~USBOP_VAL(inst);
val &= ~USBON_VAL(inst);
writel(val, pmc_base + PMC_UTMIP_UHSIC_FAKE);
/* Enable which type of event can trigger a walk,
* in this case usb_line_wake */
val = readl(pmc_base + PMC_SLEEPWALK_CFG);
val |= UTMIP_LINEVAL_WALK_EN(inst);
writel(val, pmc_base + PMC_SLEEPWALK_CFG);
/* Capture FS/LS pad configurations */
pmc_pad_cfg_val = readl(pmc_base + PMC_PAD_CFG);
val = readl(pmc_base + PMC_TRIGGERS);
val |= UTMIP_CAP_CFG(inst);
writel(val, pmc_base + PMC_TRIGGERS);
udelay(1);
pmc_pad_cfg_val = readl(pmc_base + PMC_PAD_CFG);
/* BIAS MASTER_ENABLE=0 */
val = readl(pmc_base + PMC_UTMIP_BIAS_MASTER_CNTRL);
val &= ~BIAS_MASTER_PROG_VAL;
writel(val, pmc_base + PMC_UTMIP_BIAS_MASTER_CNTRL);
/* program walk sequence, maintain a J, followed by a driven K
* to signal a resume once an wake event is detected */
val = readl(pmc_base + PMC_SLEEPWALK_REG(inst));
val &= ~UTMIP_AP_A;
val |= UTMIP_USBOP_RPD_A | UTMIP_USBON_RPD_A | UTMIP_AN_A |UTMIP_HIGHZ_A |
UTMIP_USBOP_RPD_B | UTMIP_USBON_RPD_B | UTMIP_AP_B | UTMIP_AN_B |
UTMIP_USBOP_RPD_C | UTMIP_USBON_RPD_C | UTMIP_AP_C | UTMIP_AN_C |
UTMIP_USBOP_RPD_D | UTMIP_USBON_RPD_D | UTMIP_AP_D | UTMIP_AN_D;
writel(val, pmc_base + PMC_SLEEPWALK_REG(inst));
if (port_speed == USB_PHY_PORT_SPEED_LOW) {
val = readl(pmc_base + PMC_SLEEPWALK_REG(inst));
val &= ~(UTMIP_AN_B | UTMIP_HIGHZ_B | UTMIP_AN_C |
UTMIP_HIGHZ_C | UTMIP_AN_D | UTMIP_HIGHZ_D);
writel(val, pmc_base + PMC_SLEEPWALK_REG(inst));
} else {
val = readl(pmc_base + PMC_SLEEPWALK_REG(inst));
val &= ~(UTMIP_AP_B | UTMIP_HIGHZ_B | UTMIP_AP_C |
UTMIP_HIGHZ_C | UTMIP_AP_D | UTMIP_HIGHZ_D);
writel(val, pmc_base + PMC_SLEEPWALK_REG(inst));
}
/* turn on pad detectors */
val = readl(pmc_base + PMC_USB_AO);
val &= ~(USBOP_VAL_PD(inst) | USBON_VAL_PD(inst));
writel(val, pmc_base + PMC_USB_AO);
/* Add small delay before usb detectors provide stable line values */
mdelay(1);
/* Program thermally encoded RCTRL_VAL, TCTRL_VAL into PMC space */
val = readl(pmc_base + PMC_UTMIP_TERM_PAD_CFG);
val = PMC_TCTRL_VAL(utmip_tctrl_val) | PMC_RCTRL_VAL(utmip_rctrl_val);
writel(val, pmc_base + PMC_UTMIP_TERM_PAD_CFG);
phy->pmc_remote_wakeup = false;
/* Turn over pad configuration to PMC for line wake events*/
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UTMIP_WAKE_VAL(inst, ~0);
val |= UTMIP_WAKE_VAL(inst, WAKE_VAL_ANY);
val |= UTMIP_RCTRL_USE_PMC(inst) | UTMIP_TCTRL_USE_PMC(inst);
val |= UTMIP_MASTER_ENABLE(inst) | UTMIP_FSLS_USE_PMC(inst);
writel(val, pmc_base + PMC_SLEEP_CFG);
val = readl(base + UTMIP_PMC_WAKEUP0);
val |= EVENT_INT_ENB;
writel(val, base + UTMIP_PMC_WAKEUP0);
PHY_DBG("%s ENABLE_PMC inst = %d\n", __func__, inst);
}
static void utmip_phy_disable_pmc_bus_ctrl(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
unsigned int inst = phy->inst;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UTMIP_WAKE_VAL(inst, 0xF);
val |= UTMIP_WAKE_VAL(inst, WAKE_VAL_NONE);
writel(val, pmc_base + PMC_SLEEP_CFG);
val = readl(base + UTMIP_PMC_WAKEUP0);
val &= ~EVENT_INT_ENB;
writel(val, base + UTMIP_PMC_WAKEUP0);
/* Disable PMC master mode by clearing MASTER_EN */
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~(UTMIP_RCTRL_USE_PMC(inst) | UTMIP_TCTRL_USE_PMC(inst) |
UTMIP_FSLS_USE_PMC(inst) | UTMIP_MASTER_ENABLE(inst));
writel(val, pmc_base + PMC_SLEEP_CFG);
val = readl(pmc_base + PMC_TRIGGERS);
val &= ~UTMIP_CAP_CFG(inst);
writel(val, pmc_base + PMC_TRIGGERS);
/* turn off pad detectors */
val = readl(pmc_base + PMC_USB_AO);
val |= (USBOP_VAL_PD(inst) | USBON_VAL_PD(inst));
writel(val, pmc_base + PMC_USB_AO);
val = readl(pmc_base + PMC_TRIGGERS);
val |= UTMIP_CLR_WALK_PTR(inst);
val |= UTMIP_CLR_WAKE_ALARM(inst);
writel(val, pmc_base + PMC_TRIGGERS);
phy->pmc_remote_wakeup = false;
PHY_DBG("%s DISABLE_PMC inst = %d\n", __func__, inst);
}
static bool utmi_phy_remotewake_detected(struct tegra_usb_phy *phy)
{
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
void __iomem *base = phy->regs;
unsigned int inst = phy->inst;
u32 val;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
val = readl(base + UTMIP_PMC_WAKEUP0);
if (val & EVENT_INT_ENB) {
val = readl(pmc_base + UTMIP_UHSIC_STATUS);
if (UTMIP_WAKE_ALARM(inst) & val) {
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UTMIP_WAKE_VAL(inst, 0xF);
val |= UTMIP_WAKE_VAL(inst, WAKE_VAL_NONE);
writel(val, pmc_base + PMC_SLEEP_CFG);
val = readl(pmc_base + PMC_TRIGGERS);
val |= UTMIP_CLR_WAKE_ALARM(inst);
writel(val, pmc_base + PMC_TRIGGERS);
val = readl(base + UTMIP_PMC_WAKEUP0);
val &= ~EVENT_INT_ENB;
writel(val, base + UTMIP_PMC_WAKEUP0);
phy->pmc_remote_wakeup = true;
return true;
}
}
return false;
}
static void utmi_phy_enable_trking_data(struct tegra_usb_phy *phy)
{
void __iomem *base = IO_ADDRESS(TEGRA_USB_BASE);
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
static bool init_done = false;
u32 val;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
/* Should be done only once after system boot */
if (init_done)
return;
tegra_clk_prepare_enable(phy->utmi_pad_clk);
/* Bias pad MASTER_ENABLE=1 */
val = readl(pmc_base + PMC_UTMIP_BIAS_MASTER_CNTRL);
val |= BIAS_MASTER_PROG_VAL;
writel(val, pmc_base + PMC_UTMIP_BIAS_MASTER_CNTRL);
/* Setting the tracking length time */
val = readl(base + UTMIP_BIAS_CFG1);
val &= ~UTMIP_BIAS_PDTRK_COUNT(~0);
val |= UTMIP_BIAS_PDTRK_COUNT(5);
writel(val, base + UTMIP_BIAS_CFG1);
/* Bias PDTRK is Shared and MUST be done from USB1 ONLY, PD_TRK=0 */
val = readl(base + UTMIP_BIAS_CFG1);
val &= ~UTMIP_BIAS_PDTRK_POWERDOWN;
writel(val, base + UTMIP_BIAS_CFG1);
val = readl(base + UTMIP_BIAS_CFG1);
val |= UTMIP_BIAS_PDTRK_POWERUP;
writel(val, base + UTMIP_BIAS_CFG1);
/* Wait for 25usec */
udelay(25);
/* Bias pad MASTER_ENABLE=0 */
val = readl(pmc_base + PMC_UTMIP_BIAS_MASTER_CNTRL);
val &= ~BIAS_MASTER_PROG_VAL;
writel(val, pmc_base + PMC_UTMIP_BIAS_MASTER_CNTRL);
/* Wait for 1usec */
udelay(1);
/* Bias pad MASTER_ENABLE=1 */
val = readl(pmc_base + PMC_UTMIP_BIAS_MASTER_CNTRL);
val |= BIAS_MASTER_PROG_VAL;
writel(val, pmc_base + PMC_UTMIP_BIAS_MASTER_CNTRL);
/* Read RCTRL and TCTRL from UTMIP space */
val = readl(base + UTMIP_BIAS_STS0);
utmip_rctrl_val = ffz(UTMIP_RCTRL_VAL(val));
utmip_tctrl_val = ffz(UTMIP_TCTRL_VAL(val));
/* PD_TRK=1 */
val = readl(base + UTMIP_BIAS_CFG1);
val |= UTMIP_BIAS_PDTRK_POWERDOWN;
writel(val, base + UTMIP_BIAS_CFG1);
/* Program thermally encoded RCTRL_VAL, TCTRL_VAL into PMC space */
val = readl(pmc_base + PMC_UTMIP_TERM_PAD_CFG);
val = PMC_TCTRL_VAL(utmip_tctrl_val) | PMC_RCTRL_VAL(utmip_rctrl_val);
writel(val, pmc_base + PMC_UTMIP_TERM_PAD_CFG);
tegra_clk_disable_unprepare(phy->utmi_pad_clk);
init_done = true;
}
static void utmip_powerdown_pmc_wake_detect(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
unsigned int inst = phy->inst;
/* power down UTMIP interfaces */
val = readl(pmc_base + PMC_UTMIP_MASTER_CONFIG);
val |= UTMIP_PWR(inst);
writel(val, pmc_base + PMC_UTMIP_MASTER_CONFIG);
/* setup sleep walk usb controller */
val = UTMIP_USBOP_RPD_A | UTMIP_USBON_RPD_A | UTMIP_HIGHZ_A |
UTMIP_USBOP_RPD_B | UTMIP_USBON_RPD_B | UTMIP_HIGHZ_B |
UTMIP_USBOP_RPD_C | UTMIP_USBON_RPD_C | UTMIP_HIGHZ_C |
UTMIP_USBOP_RPD_D | UTMIP_USBON_RPD_D | UTMIP_HIGHZ_D;
writel(val, pmc_base + PMC_SLEEPWALK_REG(inst));
/* Program thermally encoded RCTRL_VAL, TCTRL_VAL into PMC space */
val = readl(pmc_base + PMC_UTMIP_TERM_PAD_CFG);
val = PMC_TCTRL_VAL(utmip_tctrl_val) | PMC_RCTRL_VAL(utmip_rctrl_val);
writel(val, pmc_base + PMC_UTMIP_TERM_PAD_CFG);
/* Turn over pad configuration to PMC */
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UTMIP_WAKE_VAL(inst, ~0);
val |= UTMIP_WAKE_VAL(inst, WAKE_VAL_NONE) |
UTMIP_RCTRL_USE_PMC(inst) | UTMIP_TCTRL_USE_PMC(inst) |
UTMIP_FSLS_USE_PMC(inst) | UTMIP_MASTER_ENABLE(inst);
writel(val, pmc_base + PMC_SLEEP_CFG);
PHY_DBG("%s ENABLE_PMC inst = %d\n", __func__, inst);
}
static void utmip_powerup_pmc_wake_detect(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
unsigned int inst = phy->inst;
/* Disable PMC master mode by clearing MASTER_EN */
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~(UTMIP_RCTRL_USE_PMC(inst) | UTMIP_TCTRL_USE_PMC(inst) |
UTMIP_FSLS_USE_PMC(inst) | UTMIP_MASTER_ENABLE(inst));
writel(val, pmc_base + PMC_SLEEP_CFG);
mdelay(1);
PHY_DBG("%s DISABLE_PMC inst = %d\n", __func__, inst);
}
#ifdef KERNEL_WARNING
static void usb_phy_power_down_pmc(void)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
/* power down all 3 UTMIP interfaces */
val = readl(pmc_base + PMC_UTMIP_MASTER_CONFIG);
val |= UTMIP_PWR(0) | UTMIP_PWR(1) | UTMIP_PWR(2);
writel(val, pmc_base + PMC_UTMIP_MASTER_CONFIG);
/* turn on pad detectors */
writel(PMC_POWER_DOWN_MASK, pmc_base + PMC_USB_AO);
/* setup sleep walk fl all 3 usb controllers */
val = UTMIP_USBOP_RPD_A | UTMIP_USBON_RPD_A | UTMIP_HIGHZ_A |
UTMIP_USBOP_RPD_B | UTMIP_USBON_RPD_B | UTMIP_HIGHZ_B |
UTMIP_USBOP_RPD_C | UTMIP_USBON_RPD_C | UTMIP_HIGHZ_C |
UTMIP_USBOP_RPD_D | UTMIP_USBON_RPD_D | UTMIP_HIGHZ_D;
writel(val, pmc_base + PMC_SLEEPWALK_REG(0));
writel(val, pmc_base + PMC_SLEEPWALK_REG(1));
writel(val, pmc_base + PMC_SLEEPWALK_REG(2));
/* enable pull downs on HSIC PMC */
val = UHSIC_STROBE_RPD_A | UHSIC_DATA_RPD_A | UHSIC_STROBE_RPD_B |
UHSIC_DATA_RPD_B | UHSIC_STROBE_RPD_C | UHSIC_DATA_RPD_C |
UHSIC_STROBE_RPD_D | UHSIC_DATA_RPD_D;
writel(val, pmc_base + PMC_SLEEPWALK_UHSIC);
/* Turn over pad configuration to PMC */
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UTMIP_WAKE_VAL(0, ~0);
val &= ~UTMIP_WAKE_VAL(1, ~0);
val &= ~UTMIP_WAKE_VAL(2, ~0);
val &= ~UHSIC_WAKE_VAL_P0(~0);
val |= UTMIP_WAKE_VAL(0, WAKE_VAL_NONE) | UHSIC_WAKE_VAL_P0(WAKE_VAL_NONE) |
UTMIP_WAKE_VAL(1, WAKE_VAL_NONE) | UTMIP_WAKE_VAL(2, WAKE_VAL_NONE) |
UTMIP_RCTRL_USE_PMC(0) | UTMIP_RCTRL_USE_PMC(1) | UTMIP_RCTRL_USE_PMC(2) |
UTMIP_TCTRL_USE_PMC(0) | UTMIP_TCTRL_USE_PMC(1) | UTMIP_TCTRL_USE_PMC(2) |
UTMIP_FSLS_USE_PMC(0) | UTMIP_FSLS_USE_PMC(1) | UTMIP_FSLS_USE_PMC(2) |
UTMIP_MASTER_ENABLE(0) | UTMIP_MASTER_ENABLE(1) | UTMIP_MASTER_ENABLE(2) |
UHSIC_MASTER_ENABLE_P0;
writel(val, pmc_base + PMC_SLEEP_CFG);
}
#endif
static int usb_phy_bringup_host_controller(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
PHY_DBG("[%d] USB_USBSTS[0x%x] USB_PORTSC[0x%x] port_speed[%d]\n", __LINE__,
readl(base + USB_USBSTS), readl(base + USB_PORTSC),
phy->port_speed);
/* Device is plugged in when system is in LP0 */
/* Bring up the controller from LP0*/
val = readl(base + USB_USBCMD);
val |= USB_CMD_RESET;
writel(val, base + USB_USBCMD);
if (usb_phy_reg_status_wait(base + USB_USBCMD,
USB_CMD_RESET, 0, 2500) < 0) {
pr_err("%s: timeout waiting for reset\n", __func__);
}
val = readl(base + USB_USBMODE);
val &= ~USB_USBMODE_MASK;
val |= USB_USBMODE_HOST;
writel(val, base + USB_USBMODE);
val = readl(base + HOSTPC1_DEVLC);
val &= ~HOSTPC1_DEVLC_PTS(~0);
if (phy->pdata->phy_intf == TEGRA_USB_PHY_INTF_HSIC)
val |= HOSTPC1_DEVLC_PTS(HOSTPC1_DEVLC_PTS_HSIC);
else
val |= HOSTPC1_DEVLC_STS;
writel(val, base + HOSTPC1_DEVLC);
/* Enable Port Power */
val = readl(base + USB_PORTSC);
val |= USB_PORTSC_PP;
writel(val, base + USB_PORTSC);
udelay(10);
/* Check if the phy resume from LP0. When the phy resume from LP0
* USB register will be reset.to zero */
if (!readl(base + USB_ASYNCLISTADDR)) {
/* Program the field PTC based on the saved speed mode */
val = readl(base + USB_PORTSC);
val &= ~USB_PORTSC_PTC(~0);
if ((phy->port_speed == USB_PHY_PORT_SPEED_HIGH) ||
(phy->pdata->phy_intf == TEGRA_USB_PHY_INTF_HSIC))
val |= USB_PORTSC_PTC(5);
else if (phy->port_speed == USB_PHY_PORT_SPEED_FULL)
val |= USB_PORTSC_PTC(6);
else if (phy->port_speed == USB_PHY_PORT_SPEED_LOW)
val |= USB_PORTSC_PTC(7);
writel(val, base + USB_PORTSC);
udelay(10);
/* Disable test mode by setting PTC field to NORMAL_OP */
val = readl(base + USB_PORTSC);
val &= ~USB_PORTSC_PTC(~0);
writel(val, base + USB_PORTSC);
udelay(10);
}
/* Poll until CCS is enabled */
if (usb_phy_reg_status_wait(base + USB_PORTSC, USB_PORTSC_CCS,
USB_PORTSC_CCS, 2000)) {
pr_err("%s: timeout waiting for USB_PORTSC_CCS\n", __func__);
}
/* Poll until PE is enabled */
if (usb_phy_reg_status_wait(base + USB_PORTSC, USB_PORTSC_PE,
USB_PORTSC_PE, 2000)) {
pr_err("%s: timeout waiting for USB_PORTSC_PE\n", __func__);
}
/* Clear the PCI status, to avoid an interrupt taken upon resume */
val = readl(base + USB_USBSTS);
val |= USB_USBSTS_PCI;
writel(val, base + USB_USBSTS);
if (!phy->pmc_remote_wakeup) {
/* Put controller in suspend mode by writing 1 to SUSP bit of PORTSC */
val = readl(base + USB_PORTSC);
if ((val & USB_PORTSC_PP) && (val & USB_PORTSC_PE)) {
val |= USB_PORTSC_SUSP;
writel(val, base + USB_PORTSC);
/* Need a 4ms delay before the controller goes to suspend */
mdelay(4);
/* Wait until port suspend completes */
if (usb_phy_reg_status_wait(base + USB_PORTSC, USB_PORTSC_SUSP,
USB_PORTSC_SUSP, 1000)) {
pr_err("%s: timeout waiting for PORT_SUSPEND\n",
__func__);
}
}
}
PHY_DBG("[%d] USB_USBSTS[0x%x] USB_PORTSC[0x%x] port_speed[%d]\n", __LINE__,
readl(base + USB_USBSTS), readl(base + USB_PORTSC),
phy->port_speed);
DBG("USB_USBSTS[0x%x] USB_PORTSC[0x%x]\n",
readl(base + USB_USBSTS), readl(base + USB_PORTSC));
return 0;
}
static void usb_phy_wait_for_sof(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
val = readl(base + USB_USBSTS);
writel(val, base + USB_USBSTS);
udelay(20);
/* wait for two SOFs */
if (usb_phy_reg_status_wait(base + USB_USBSTS, USB_USBSTS_SRI,
USB_USBSTS_SRI, 2500))
pr_err("%s: timeout waiting for SOF\n", __func__);
val = readl(base + USB_USBSTS);
writel(val, base + USB_USBSTS);
if (usb_phy_reg_status_wait(base + USB_USBSTS, USB_USBSTS_SRI, 0, 2500))
pr_err("%s: timeout waiting for SOF\n", __func__);
if (usb_phy_reg_status_wait(base + USB_USBSTS, USB_USBSTS_SRI,
USB_USBSTS_SRI, 2500))
pr_err("%s: timeout waiting for SOF\n", __func__);
udelay(20);
}
static unsigned int utmi_phy_xcvr_setup_value(struct tegra_usb_phy *phy)
{
struct tegra_utmi_config *cfg = &phy->pdata->u_cfg.utmi;
signed long val;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
if (cfg->xcvr_use_fuses) {
val = XCVR_SETUP(tegra_fuse_readl(FUSE_USB_CALIB_0));
if (cfg->xcvr_use_lsb) {
val = min((unsigned int) ((val & XCVR_SETUP_LSB_MASK)
+ cfg->xcvr_setup_offset),
(unsigned int) XCVR_SETUP_LSB_MAX_VAL);
val |= (cfg->xcvr_setup & XCVR_SETUP_MSB_MASK);
} else {
if (cfg->xcvr_setup_offset <= UTMIP_XCVR_MAX_OFFSET)
val = val + cfg->xcvr_setup_offset;
if (val > UTMIP_XCVR_SETUP_MAX_VALUE) {
val = UTMIP_XCVR_SETUP_MAX_VALUE;
pr_info("%s: reset XCVR_SETUP to max value\n",
__func__);
} else if (val < UTMIP_XCVR_SETUP_MIN_VALUE) {
val = UTMIP_XCVR_SETUP_MIN_VALUE;
pr_info("%s: reset XCVR_SETUP to min value\n",
__func__);
}
}
} else {
val = cfg->xcvr_setup;
}
return (unsigned int) val;
}
static int utmi_phy_open(struct tegra_usb_phy *phy)
{
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
unsigned long parent_rate, val;
int i;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
phy->utmi_pad_clk = clk_get_sys("utmip-pad", NULL);
if (IS_ERR(phy->utmi_pad_clk)) {
pr_err("%s: can't get utmip pad clock\n", __func__);
return PTR_ERR(phy->utmi_pad_clk);
}
phy->utmi_xcvr_setup = utmi_phy_xcvr_setup_value(phy);
parent_rate = clk_get_rate(clk_get_parent(phy->pllu_clk));
for (i = 0; i < ARRAY_SIZE(utmip_freq_table); i++) {
if (utmip_freq_table[i].freq == parent_rate) {
phy->freq = &utmip_freq_table[i];
break;
}
}
if (!phy->freq) {
pr_err("invalid pll_u parent rate %ld\n", parent_rate);
return -EINVAL;
}
/* Power-up the VBUS detector for UTMIP PHY */
val = readl(pmc_base + PMC_USB_AO);
val &= ~(PMC_USB_AO_VBUS_WAKEUP_PD_P0 | PMC_USB_AO_ID_PD_P0);
writel(val, (pmc_base + PMC_USB_AO));
utmip_powerup_pmc_wake_detect(phy);
return 0;
}
static void utmi_phy_close(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
DBG("%s inst:[%d]\n", __func__, phy->inst);
/* Disable PHY clock valid interrupts while going into suspend*/
if (phy->hot_plug) {
val = readl(base + USB_SUSP_CTRL);
val &= ~USB_PHY_CLK_VALID_INT_ENB;
writel(val, base + USB_SUSP_CTRL);
}
val = readl(pmc_base + PMC_SLEEP_CFG);
if (val & UTMIP_MASTER_ENABLE(phy->inst))
utmip_phy_disable_pmc_bus_ctrl(phy);
clk_put(phy->utmi_pad_clk);
}
static int utmi_phy_pad_power_on(struct tegra_usb_phy *phy)
{
unsigned long val, flags;
void __iomem *pad_base = IO_ADDRESS(TEGRA_USB_BASE);
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
tegra_clk_prepare_enable(phy->utmi_pad_clk);
spin_lock_irqsave(&utmip_pad_lock, flags);
utmip_pad_count++;
val = readl(pad_base + UTMIP_BIAS_CFG0);
val &= ~(UTMIP_OTGPD | UTMIP_BIASPD);
val |= UTMIP_HSSQUELCH_LEVEL(0x2) | UTMIP_HSDISCON_LEVEL(0x1) |
UTMIP_HSDISCON_LEVEL_MSB;
writel(val, pad_base + UTMIP_BIAS_CFG0);
spin_unlock_irqrestore(&utmip_pad_lock, flags);
tegra_clk_disable_unprepare(phy->utmi_pad_clk);
return 0;
}
static int utmi_phy_pad_power_off(struct tegra_usb_phy *phy)
{
unsigned long val, flags;
void __iomem *pad_base = IO_ADDRESS(TEGRA_USB_BASE);
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
tegra_clk_prepare_enable(phy->utmi_pad_clk);
spin_lock_irqsave(&utmip_pad_lock, flags);
if (!utmip_pad_count) {
pr_err("%s: utmip pad already powered off\n", __func__);
goto out;
}
if (--utmip_pad_count == 0) {
val = readl(pad_base + UTMIP_BIAS_CFG0);
val |= UTMIP_OTGPD | UTMIP_BIASPD;
val &= ~(UTMIP_HSSQUELCH_LEVEL(~0) | UTMIP_HSDISCON_LEVEL(~0) |
UTMIP_HSDISCON_LEVEL_MSB);
writel(val, pad_base + UTMIP_BIAS_CFG0);
}
out:
spin_unlock_irqrestore(&utmip_pad_lock, flags);
tegra_clk_disable_unprepare(phy->utmi_pad_clk);
return 0;
}
static int utmi_phy_irq(struct tegra_usb_phy *phy)
{
void __iomem *base = phy->regs;
unsigned long val = 0;
bool remote_wakeup = false;
int irq_status = IRQ_HANDLED;
if (phy->phy_clk_on) {
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
DBG("USB_USBSTS[0x%x] USB_PORTSC[0x%x]\n",
readl(base + USB_USBSTS), readl(base + USB_PORTSC));
DBG("USB_USBMODE[0x%x] USB_USBCMD[0x%x]\n",
readl(base + USB_USBMODE), readl(base + USB_USBCMD));
}
usb_phy_fence_read(phy);
/* check if there is any remote wake event */
if (utmi_phy_remotewake_detected(phy)) {
pr_info("%s: utmip remote wake detected\n", __func__);
remote_wakeup = true;
}
if (phy->hot_plug) {
val = readl(base + USB_SUSP_CTRL);
if ((val & USB_PHY_CLK_VALID_INT_STS)) {
val &= ~USB_PHY_CLK_VALID_INT_ENB |
USB_PHY_CLK_VALID_INT_STS;
writel(val , (base + USB_SUSP_CTRL));
/* In case of remote wakeup PHY clock will not up
immediately, so should not access any controller
register but normal plug-in/plug-out should be
executed */
if (!remote_wakeup) {
val = readl(base + USB_USBSTS);
if (!(val & USB_USBSTS_PCI)) {
irq_status = IRQ_NONE;
goto exit;
}
val = readl(base + USB_PORTSC);
if (val & USB_PORTSC_CCS)
val &= ~USB_PORTSC_WKCN;
else
val &= ~USB_PORTSC_WKDS;
val &= ~USB_PORTSC_RWC_BITS;
writel(val , (base + USB_PORTSC));
}
} else if (!phy->phy_clk_on) {
if (remote_wakeup)
irq_status = IRQ_HANDLED;
else
irq_status = IRQ_NONE;
goto exit;
}
}
exit:
return irq_status;
}
static void utmi_phy_enable_obs_bus(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
/* (2LS WAR)is not required for LS and FS devices and is only for HS */
if ((phy->port_speed == USB_PHY_PORT_SPEED_LOW) ||
(phy->port_speed == USB_PHY_PORT_SPEED_FULL)) {
/* do not enable the OBS bus */
val = readl(base + UTMIP_MISC_CFG0);
val &= ~(UTMIP_DPDM_OBSERVE_SEL(~0));
writel(val, base + UTMIP_MISC_CFG0);
DBG("%s(%d) Disable OBS bus\n", __func__, __LINE__);
return;
}
/* Force DP/DM pulldown active for Host mode */
val = readl(base + UTMIP_MISC_CFG0);
val |= FORCE_PULLDN_DM | FORCE_PULLDN_DP |
COMB_TERMS | ALWAYS_FREE_RUNNING_TERMS;
writel(val, base + UTMIP_MISC_CFG0);
val = readl(base + UTMIP_MISC_CFG0);
val &= ~UTMIP_DPDM_OBSERVE_SEL(~0);
if (phy->port_speed == USB_PHY_PORT_SPEED_LOW)
val |= UTMIP_DPDM_OBSERVE_SEL_FS_J;
else
val |= UTMIP_DPDM_OBSERVE_SEL_FS_K;
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);
DBG("%s(%d) Enable OBS bus\n", __func__, __LINE__);
PHY_DBG("ENABLE_OBS_BUS\n");
}
static int utmi_phy_disable_obs_bus(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
unsigned long flags;
/* check if OBS bus is already enabled */
val = readl(base + UTMIP_MISC_CFG0);
if (val & UTMIP_DPDM_OBSERVE) {
PHY_DBG("DISABLE_OBS_BUS\n");
/* disable ALL interrupts on current CPU */
local_irq_save(flags);
/* Change the UTMIP OBS bus to drive SE0 */
val = readl(base + UTMIP_MISC_CFG0);
val &= ~UTMIP_DPDM_OBSERVE_SEL(~0);
val |= UTMIP_DPDM_OBSERVE_SEL_FS_SE0;
writel(val, base + UTMIP_MISC_CFG0);
/* Wait for 3us(2 LS bit times) */
udelay(3);
/* Release UTMIP OBS bus */
val = readl(base + UTMIP_MISC_CFG0);
val &= ~UTMIP_DPDM_OBSERVE;
writel(val, base + UTMIP_MISC_CFG0);
/* Release DP/DM pulldown for Host mode */
val = readl(base + UTMIP_MISC_CFG0);
val &= ~(FORCE_PULLDN_DM | FORCE_PULLDN_DP |
COMB_TERMS | ALWAYS_FREE_RUNNING_TERMS);
writel(val, base + UTMIP_MISC_CFG0);
val = readl(base + USB_USBCMD);
val |= USB_USBCMD_RS;
writel(val, base + USB_USBCMD);
/* restore ALL interrupts on current CPU */
local_irq_restore(flags);
if (usb_phy_reg_status_wait(base + USB_USBCMD, USB_USBCMD_RS,
USB_USBCMD_RS, 2000)) {
pr_err("%s: timeout waiting for USB_USBCMD_RS\n", __func__);
return -ETIMEDOUT;
}
}
return 0;
}
static int utmi_phy_post_resume(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
unsigned int inst = phy->inst;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
val = readl(pmc_base + PMC_SLEEP_CFG);
/* if PMC is not disabled by now then disable it */
if (val & UTMIP_MASTER_ENABLE(inst)) {
utmip_phy_disable_pmc_bus_ctrl(phy);
}
utmi_phy_disable_obs_bus(phy);
return 0;
}
static int phy_post_suspend(struct tegra_usb_phy *phy)
{
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
/* Need a 4ms delay for controller to suspend */
mdelay(4);
return 0;
}
static int utmi_phy_pre_resume(struct tegra_usb_phy *phy, bool remote_wakeup)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
void __iomem *base = phy->regs;
unsigned int inst = phy->inst;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
phy->port_speed = (readl(base + HOSTPC1_DEVLC) >> 25) &
HOSTPC1_DEVLC_PSPD_MASK;
if (phy->port_speed == USB_PHY_PORT_SPEED_HIGH) {
/* Disable interrupts */
writel(0, base + USB_USBINTR);
/* Clear the run bit to stop SOFs - 2LS WAR */
val = readl(base + USB_USBCMD);
val &= ~USB_USBCMD_RS;
writel(val, base + USB_USBCMD);
if (usb_phy_reg_status_wait(base + USB_USBSTS, USB_USBSTS_HCH,
USB_USBSTS_HCH, 2000)) {
pr_err("%s: timeout waiting for USB_USBSTS_HCH\n", __func__);
}
}
val = readl(pmc_base + PMC_SLEEP_CFG);
if (val & UTMIP_MASTER_ENABLE(inst)) {
if (!remote_wakeup)
utmip_phy_disable_pmc_bus_ctrl(phy);
} else {
utmi_phy_enable_obs_bus(phy);
}
return 0;
}
static int utmi_phy_power_off(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
PHY_DBG("%s(%d) inst:[%d] BEGIN\n", __func__, __LINE__, phy->inst);
if (!phy->phy_clk_on) {
PHY_DBG("%s(%d) inst:[%d] phy clk is already off\n",
__func__, __LINE__, phy->inst);
return 0;
}
if (phy->pdata->op_mode == TEGRA_USB_OPMODE_DEVICE) {
utmip_powerdown_pmc_wake_detect(phy);
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 + UTMIP_BAT_CHRG_CFG0);
val |= UTMIP_PD_CHRG;
writel(val, base + UTMIP_BAT_CHRG_CFG0);
} else {
phy->port_speed = (readl(base + HOSTPC1_DEVLC) >> 25) &
HOSTPC1_DEVLC_PSPD_MASK;
/* Disable interrupts */
writel(0, base + USB_USBINTR);
/* Clear the run bit to stop SOFs - 2LS WAR */
val = readl(base + USB_USBCMD);
val &= ~USB_USBCMD_RS;
writel(val, base + USB_USBCMD);
if (usb_phy_reg_status_wait(base + USB_USBSTS, USB_USBSTS_HCH,
USB_USBSTS_HCH, 2000)) {
pr_err("%s: timeout waiting for USB_USBSTS_HCH\n", __func__);
}
utmip_setup_pmc_wake_detect(phy);
val = readl(base + USB_SUSP_CTRL);
val &= ~USB_WAKE_ON_CNNT_EN_DEV;
writel(val, base + USB_SUSP_CTRL);
}
if (!phy->hot_plug) {
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);
val = readl(base + UTMIP_BIAS_CFG1);
val |= UTMIP_BIAS_PDTRK_COUNT(0x5);
writel(val, base + UTMIP_BIAS_CFG1);
utmi_phy_pad_power_off(phy);
if (phy->hot_plug) {
bool enable_hotplug = true;
/* if it is OTG port then make sure to enable hot-plug feature
only if host adaptor is connected, i.e id is low */
if (phy->pdata->port_otg) {
val = readl(base + USB_PHY_VBUS_WAKEUP_ID);
enable_hotplug = (val & USB_ID_STATUS) ? false : true;
}
if (enable_hotplug) {
/* Enable wakeup event of device plug-in/plug-out */
val = readl(base + USB_PORTSC);
if (val & USB_PORTSC_CCS)
val |= USB_PORTSC_WKDS;
else
val |= USB_PORTSC_WKCN;
writel(val, base + USB_PORTSC);
val = readl(base + USB_SUSP_CTRL);
val |= USB_PHY_CLK_VALID_INT_ENB;
writel(val, base + USB_SUSP_CTRL);
} else {
/* Disable PHY clock valid interrupts while going into suspend*/
val = readl(base + USB_SUSP_CTRL);
val &= ~USB_PHY_CLK_VALID_INT_ENB;
writel(val, base + USB_SUSP_CTRL);
}
}
/* Disable PHY clock */
val = readl(base + HOSTPC1_DEVLC);
val |= HOSTPC1_DEVLC_PHCD;
writel(val, base + HOSTPC1_DEVLC);
if (!phy->hot_plug) {
val = readl(base + USB_SUSP_CTRL);
val |= UTMIP_RESET;
writel(val, base + USB_SUSP_CTRL);
}
phy->phy_clk_on = false;
phy->hw_accessible = false;
PHY_DBG("%s(%d) inst:[%d] END\n", __func__, __LINE__, phy->inst);
return 0;
}
static int utmi_phy_power_on(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
struct tegra_utmi_config *config = &phy->pdata->u_cfg.utmi;
PHY_DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
if (phy->phy_clk_on) {
PHY_DBG("%s(%d) inst:[%d] phy clk is already On\n",
__func__, __LINE__, phy->inst);
return 0;
}
val = readl(base + USB_SUSP_CTRL);
val |= UTMIP_RESET;
writel(val, base + USB_SUSP_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);
if (phy->pdata->op_mode == TEGRA_USB_OPMODE_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);
val = readl(base + UTMIP_BAT_CHRG_CFG0);
val &= ~UTMIP_PD_CHRG;
writel(val, base + UTMIP_BAT_CHRG_CFG0);
} else {
val = readl(base + UTMIP_BAT_CHRG_CFG0);
val |= UTMIP_PD_CHRG;
writel(val, base + UTMIP_BAT_CHRG_CFG0);
}
utmi_phy_pad_power_on(phy);
val = readl(base + UTMIP_XCVR_CFG0);
val &= ~(UTMIP_XCVR_LSBIAS_SEL | 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(phy->utmi_xcvr_setup);
val |= UTMIP_XCVR_SETUP_MSB(XCVR_SETUP_MSB_CALIB(phy->utmi_xcvr_setup));
val |= UTMIP_XCVR_LSFSLEW(config->xcvr_lsfslew);
val |= UTMIP_XCVR_LSRSLEW(config->xcvr_lsrslew);
if (!config->xcvr_use_lsb)
val |= UTMIP_XCVR_HSSLEW_MSB(0x8);
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_BIAS_CFG1);
val &= ~UTMIP_BIAS_PDTRK_COUNT(~0);
val |= UTMIP_BIAS_PDTRK_COUNT(phy->freq->pdtrk_count);
writel(val, base + UTMIP_BIAS_CFG1);
val = readl(base + UTMIP_SPARE_CFG0);
val &= ~FUSE_SETUP_SEL;
val |= FUSE_ATERM_SEL;
writel(val, base + UTMIP_SPARE_CFG0);
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);
val = readl(base + HOSTPC1_DEVLC);
val &= ~HOSTPC1_DEVLC_PHCD;
writel(val, base + HOSTPC1_DEVLC);
if (usb_phy_reg_status_wait(base + USB_SUSP_CTRL,
USB_PHY_CLK_VALID, USB_PHY_CLK_VALID, 2500))
pr_warn("%s: timeout waiting for phy to stabilize\n", __func__);
utmi_phy_enable_trking_data(phy);
if (phy->inst == 2)
writel(0, base + ICUSB_CTRL);
val = readl(base + USB_USBMODE);
val &= ~USB_USBMODE_MASK;
if (phy->pdata->op_mode == TEGRA_USB_OPMODE_HOST)
val |= USB_USBMODE_HOST;
else
val |= USB_USBMODE_DEVICE;
writel(val, base + USB_USBMODE);
val = readl(base + HOSTPC1_DEVLC);
val &= ~HOSTPC1_DEVLC_PTS(~0);
val |= HOSTPC1_DEVLC_STS;
writel(val, base + HOSTPC1_DEVLC);
if (phy->pdata->op_mode == TEGRA_USB_OPMODE_DEVICE)
utmip_powerup_pmc_wake_detect(phy);
phy->phy_clk_on = true;
phy->hw_accessible = true;
PHY_DBG("%s(%d) End inst:[%d]\n", __func__, __LINE__, phy->inst);
return 0;
}
static void utmi_phy_restore_start(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
int inst = phy->inst;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
val = readl(pmc_base + UTMIP_UHSIC_STATUS);
/* Check whether we wake up from the remote resume.
For lp1 case, pmc is not responsible for waking the
system, it's the flow controller and hence
UTMIP_WALK_PTR_VAL(inst) will return 0.
Also, for lp1 case phy->pmc_remote_wakeup will already be set
to true by utmi_phy_irq() when the remote wakeup happens.
Hence change the logic in the else part to enter only
if phy->pmc_remote_wakeup is not set to true by the
utmi_phy_irq(). */
if (UTMIP_WALK_PTR_VAL(inst) & val) {
phy->pmc_remote_wakeup = true;
} else if (!phy->pmc_remote_wakeup) {
val = readl(pmc_base + PMC_SLEEP_CFG);
if (val & UTMIP_MASTER_ENABLE(inst))
utmip_phy_disable_pmc_bus_ctrl(phy);
}
utmi_phy_enable_obs_bus(phy);
}
static void utmi_phy_restore_end(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
int wait_time_us = 25000; /* FPR should be set by this time */
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
/* check whether we wake up from the remote resume */
if (phy->pmc_remote_wakeup) {
/* wait until SUSPEND and RESUME bit is cleared on remote resume */
do {
val = readl(base + USB_PORTSC);
udelay(1);
if (wait_time_us == 0) {
PHY_DBG("%s PMC FPR timeout val = 0x%lx ",
__func__, val);
utmip_phy_disable_pmc_bus_ctrl(phy);
utmi_phy_post_resume(phy);
return;
}
wait_time_us--;
} while (val & (USB_PORTSC_RESUME | USB_PORTSC_SUSP));
/* wait for 25 ms to port resume complete */
msleep(25);
/* disable PMC master control */
utmip_phy_disable_pmc_bus_ctrl(phy);
/* Clear PCI and SRI bits to avoid an interrupt upon resume */
val = readl(base + USB_USBSTS);
writel(val, base + USB_USBSTS);
/* wait to avoid SOF if there is any */
if (usb_phy_reg_status_wait(base + USB_USBSTS,
USB_USBSTS_SRI, USB_USBSTS_SRI, 2500) < 0) {
pr_err("%s: timeout waiting for SOF\n", __func__);
}
utmi_phy_post_resume(phy);
}
}
static int utmi_phy_resume(struct tegra_usb_phy *phy)
{
int status = 0;
unsigned long val;
int port_connected = 0;
int is_lp0;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
if (phy->pdata->op_mode == TEGRA_USB_OPMODE_HOST) {
if (readl(base + USB_ASYNCLISTADDR) &&
!phy->pdata->u_data.host.power_off_on_suspend)
return 0;
val = readl(base + USB_PORTSC);
port_connected = val & USB_PORTSC_CCS;
is_lp0 = !(readl(base + USB_ASYNCLISTADDR));
if ((phy->port_speed < USB_PHY_PORT_SPEED_UNKNOWN) &&
(port_connected ^ is_lp0)) {
utmi_phy_restore_start(phy);
usb_phy_bringup_host_controller(phy);
utmi_phy_restore_end(phy);
} else {
utmip_phy_disable_pmc_bus_ctrl(phy);
/* device is plugged in when system is in LP0 */
/* bring up the controller from LP0*/
val = readl(base + USB_USBCMD);
val |= USB_CMD_RESET;
writel(val, base + USB_USBCMD);
if (usb_phy_reg_status_wait(base + USB_USBCMD,
USB_CMD_RESET, 0, 2500) < 0) {
pr_err("%s: timeout waiting for reset\n", __func__);
}
val = readl(base + USB_USBMODE);
val &= ~USB_USBMODE_MASK;
val |= USB_USBMODE_HOST;
writel(val, base + USB_USBMODE);
val = readl(base + HOSTPC1_DEVLC);
val &= ~HOSTPC1_DEVLC_PTS(~0);
val |= HOSTPC1_DEVLC_STS;
writel(val, base + HOSTPC1_DEVLC);
writel(USB_USBCMD_RS, base + USB_USBCMD);
if (usb_phy_reg_status_wait(base + USB_USBCMD,
USB_USBCMD_RS, USB_USBCMD_RS, 2500) < 0) {
pr_err("%s: timeout waiting for run bit\n", __func__);
}
/* Enable Port Power */
val = readl(base + USB_PORTSC);
val |= USB_PORTSC_PP;
writel(val, base + USB_PORTSC);
udelay(10);
DBG("USB_USBSTS[0x%x] USB_PORTSC[0x%x]\n",
readl(base + USB_USBSTS), readl(base + USB_PORTSC));
}
}
return status;
}
static bool utmi_phy_charger_detect(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
bool status;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
if (phy->pdata->op_mode != TEGRA_USB_OPMODE_DEVICE) {
/* Charger detection is not there for ULPI
* return Charger not available */
return false;
}
/* Enable charger detection logic */
val = readl(base + UTMIP_BAT_CHRG_CFG0);
val |= UTMIP_OP_SRC_EN | UTMIP_ON_SINK_EN;
writel(val, base + UTMIP_BAT_CHRG_CFG0);
/* Source should be on for 100 ms as per USB charging spec */
msleep(TDP_SRC_ON_MS);
val = readl(base + USB_PHY_VBUS_WAKEUP_ID);
/* If charger is not connected disable the interrupt */
val &= ~VDAT_DET_INT_EN;
val |= VDAT_DET_CHG_DET;
writel(val, base + USB_PHY_VBUS_WAKEUP_ID);
val = readl(base + USB_PHY_VBUS_WAKEUP_ID);
if (val & VDAT_DET_STS)
status = true;
else
status = false;
/* Disable charger detection logic */
val = readl(base + UTMIP_BAT_CHRG_CFG0);
val &= ~(UTMIP_OP_SRC_EN | UTMIP_ON_SINK_EN);
writel(val, base + UTMIP_BAT_CHRG_CFG0);
/* Delay of 40 ms before we pull the D+ as per battery charger spec */
msleep(TDPSRC_CON_MS);
return status;
}
static void uhsic_powerup_pmc_wake_detect(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
/* turn on pad detectors for HSIC*/
val = readl(pmc_base + PMC_USB_AO);
val &= ~(HSIC_RESERVED_P0 | STROBE_VAL_PD_P0 | DATA_VAL_PD_P0);
writel(val, pmc_base + PMC_USB_AO);
/* Disable PMC master mode by clearing MASTER_EN */
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~(UHSIC_MASTER_ENABLE_P0);
writel(val, pmc_base + PMC_SLEEP_CFG);
mdelay(1);
}
static void uhsic_powerdown_pmc_wake_detect(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
DBG("%s:%d\n", __func__, __LINE__);
/* turn off pad detectors for HSIC*/
val = readl(pmc_base + PMC_USB_AO);
val |= (HSIC_RESERVED_P0 | STROBE_VAL_PD_P0 | DATA_VAL_PD_P0);
writel(val, pmc_base + PMC_USB_AO);
/* enable pull downs on HSIC PMC */
val = UHSIC_STROBE_RPD_A | UHSIC_DATA_RPD_A | UHSIC_STROBE_RPD_B |
UHSIC_DATA_RPD_B | UHSIC_STROBE_RPD_C | UHSIC_DATA_RPD_C |
UHSIC_STROBE_RPD_D | UHSIC_DATA_RPD_D;
writel(val, pmc_base + PMC_SLEEPWALK_UHSIC);
/* Turn over pad configuration to PMC */
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UHSIC_WAKE_VAL_P0(~0);
val |= UHSIC_WAKE_VAL_P0(WAKE_VAL_NONE) | UHSIC_MASTER_ENABLE_P0;
writel(val, pmc_base + PMC_SLEEP_CFG);
}
static void uhsic_setup_pmc_wake_detect(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
void __iomem *base = phy->regs;
bool port_connected;
DBG("%s:%d\n", __func__, __LINE__);
/* check for port connect status */
val = readl(base + USB_PORTSC);
port_connected = val & USB_PORTSC_CCS;
if (!port_connected)
return;
/*Set PMC MASTER bits to do the following
* a. Take over the hsic drivers
* b. set up such that it will take over resume
* if remote wakeup is detected
* Prepare PMC to take over suspend-wake detect-drive resume until USB
* controller ready
*/
/* disable master enable in PMC */
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UHSIC_MASTER_ENABLE_P0;
writel(val, pmc_base + PMC_SLEEP_CFG);
/* UTMIP_PWR_PX=1 for power savings mode */
val = readl(pmc_base + PMC_UTMIP_MASTER_CONFIG);
val |= UHSIC_PWR;
writel(val, pmc_base + PMC_UTMIP_MASTER_CONFIG);
/* config debouncer */
val = readl(pmc_base + PMC_USB_DEBOUNCE);
val |= PMC_USB_DEBOUNCE_VAL(2);
writel(val, pmc_base + PMC_USB_DEBOUNCE);
/* Make sure nothing is happening on the line with respect to PMC */
val = readl(pmc_base + PMC_UTMIP_UHSIC_FAKE);
val &= ~UHSIC_STROBE_VAL;
val &= ~UHSIC_DATA_VAL;
writel(val, pmc_base + PMC_UTMIP_UHSIC_FAKE);
/* Clear walk enable */
val = readl(pmc_base + PMC_SLEEPWALK_CFG);
val &= ~UHSIC_LINEVAL_WALK_EN;
writel(val, pmc_base + PMC_SLEEPWALK_CFG);
/* Make sure wake value for line is none */
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UHSIC_WAKE_VAL(WAKE_VAL_ANY);
val |= UHSIC_WAKE_VAL(WAKE_VAL_NONE);
writel(val, pmc_base + PMC_SLEEP_CFG);
/* turn on pad detectors */
val = readl(pmc_base + PMC_USB_AO);
val &= ~(STROBE_VAL_PD_P0 | DATA_VAL_PD_P0);
writel(val, pmc_base + PMC_USB_AO);
/* Add small delay before usb detectors provide stable line values */
udelay(1);
/* Enable which type of event can trigger a walk,
* in this case usb_line_wake */
val = readl(pmc_base + PMC_SLEEPWALK_CFG);
val |= UHSIC_LINEVAL_WALK_EN;
writel(val, pmc_base + PMC_SLEEPWALK_CFG);
/* program walk sequence, maintain a J, followed by a driven K
* to signal a resume once an wake event is detected */
val = readl(pmc_base + PMC_SLEEPWALK_UHSIC);
val &= ~UHSIC_DATA_RPU_A;
val |= UHSIC_DATA_RPD_A;
val &= ~UHSIC_STROBE_RPD_A;
val |= UHSIC_STROBE_RPU_A;
val &= ~UHSIC_DATA_RPD_B;
val |= UHSIC_DATA_RPU_B;
val &= ~UHSIC_STROBE_RPU_B;
val |= UHSIC_STROBE_RPD_B;
val &= ~UHSIC_DATA_RPD_C;
val |= UHSIC_DATA_RPU_C;
val &= ~UHSIC_STROBE_RPU_C;
val |= UHSIC_STROBE_RPD_C;
val &= ~UHSIC_DATA_RPD_D;
val |= UHSIC_DATA_RPU_D;
val &= ~UHSIC_STROBE_RPU_D;
val |= UHSIC_STROBE_RPD_D;
writel(val, pmc_base + PMC_SLEEPWALK_UHSIC);
phy->pmc_remote_wakeup = false;
/* Setting Wake event*/
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UHSIC_WAKE_VAL(WAKE_VAL_ANY);
val |= UHSIC_WAKE_VAL(WAKE_VAL_SD10);
writel(val, pmc_base + PMC_SLEEP_CFG);
/* Clear the walk pointers and wake alarm */
val = readl(pmc_base + PMC_TRIGGERS);
val |= UHSIC_CLR_WAKE_ALARM_P0 | UHSIC_CLR_WALK_PTR_P0;
writel(val, pmc_base + PMC_TRIGGERS);
/* Turn over pad configuration to PMC for line wake events*/
val = readl(pmc_base + PMC_SLEEP_CFG);
val |= UHSIC_MASTER_ENABLE;
writel(val, pmc_base + PMC_SLEEP_CFG);
val = readl(base + UHSIC_PMC_WAKEUP0);
val |= EVENT_INT_ENB;
writel(val, base + UHSIC_PMC_WAKEUP0);
PHY_DBG("%s ENABLE_PMC\n", __func__);
}
static void uhsic_phy_pmc_resume(struct tegra_usb_phy *phy, bool remote_wakeup)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
void __iomem *base = phy->regs;
bool port_connected;
DBG("%s:%d\n", __func__, __LINE__);
/* check for port connect status */
val = readl(base + USB_PORTSC);
port_connected = val & USB_PORTSC_CCS;
if (!port_connected)
return;
/* Make sure wake value for line is none */
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UHSIC_WAKE_VAL(WAKE_VAL_ANY);
val |= UHSIC_WAKE_VAL(WAKE_VAL_NONE);
writel(val, pmc_base + PMC_SLEEP_CFG);
/* turn on pad detectors */
val = readl(pmc_base + PMC_USB_AO);
val &= ~(STROBE_VAL_PD_P0 | DATA_VAL_PD_P0);
writel(val, pmc_base + PMC_USB_AO);
/* Add small delay before usb detectors provide stable line values */
udelay(1);
/* If it is during remote wakeup, set resume state on the BUS.
If it is during AP resume, set suspend state on the BUS. */
val = readl(pmc_base + PMC_SLEEPWALK_UHSIC);
if (remote_wakeup) {
/* Switching PMC from SUSPEND to resume.*/
val &= ~UHSIC_DATA_RPD_A;
val |= UHSIC_DATA_RPU_A;
val &= ~UHSIC_STROBE_RPU_A;
val |= UHSIC_STROBE_RPD_A;
} else {
val &= ~UHSIC_DATA_RPU_A;
val |= UHSIC_DATA_RPD_A;
val &= ~UHSIC_STROBE_RPD_A;
val |= UHSIC_STROBE_RPU_A;
}
writel(val, pmc_base + PMC_SLEEPWALK_UHSIC);
/* Turn over pad configuration to PMC for line wake events*/
val = readl(pmc_base + PMC_SLEEP_CFG);
val |= UHSIC_MASTER_ENABLE;
writel(val, pmc_base + PMC_SLEEP_CFG);
}
static void uhsic_phy_disable_pmc_bus_ctrl(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
void __iomem *base = phy->regs;
DBG("%s (%d)\n", __func__, __LINE__);
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UHSIC_WAKE_VAL(WAKE_VAL_ANY);
val |= UHSIC_WAKE_VAL(WAKE_VAL_NONE);
writel(val, pmc_base + PMC_SLEEP_CFG);
val = readl(base + UHSIC_PMC_WAKEUP0);
val &= ~EVENT_INT_ENB;
writel(val, base + UHSIC_PMC_WAKEUP0);
/* Disable PMC master mode by clearing MASTER_EN */
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~(UHSIC_MASTER_ENABLE);
writel(val, pmc_base + PMC_SLEEP_CFG);
/* turn off pad detectors */
val = readl(pmc_base + PMC_USB_AO);
val |= (STROBE_VAL_PD_P0 | DATA_VAL_PD_P0);
writel(val, pmc_base + PMC_USB_AO);
val = readl(pmc_base + PMC_TRIGGERS);
val |= (UHSIC_CLR_WALK_PTR_P0 | UHSIC_CLR_WAKE_ALARM_P0);
writel(val, pmc_base + PMC_TRIGGERS);
phy->pmc_remote_wakeup = false;
}
static bool uhsic_phy_remotewake_detected(struct tegra_usb_phy *phy)
{
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
void __iomem *base = phy->regs;
u32 val;
val = readl(base + UHSIC_PMC_WAKEUP0);
if (val & EVENT_INT_ENB) {
val = readl(pmc_base + UTMIP_UHSIC_STATUS);
if (UHSIC_WAKE_ALARM & val) {
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UHSIC_WAKE_VAL(WAKE_VAL_ANY);
val |= UHSIC_WAKE_VAL(WAKE_VAL_NONE);
writel(val, pmc_base + PMC_SLEEP_CFG);
val = readl(pmc_base + PMC_TRIGGERS);
val |= UHSIC_CLR_WAKE_ALARM_P0;
writel(val, pmc_base + PMC_TRIGGERS);
val = readl(base + UHSIC_PMC_WAKEUP0);
val &= ~EVENT_INT_ENB;
writel(val, base + UHSIC_PMC_WAKEUP0);
phy->pmc_remote_wakeup = true;
DBG("%s:PMC remote wakeup detected for HSIC\n", __func__);
return true;
}
}
return false;
}
static int uhsic_phy_pre_resume(struct tegra_usb_phy *phy, bool remote_wakeup)
{
void __iomem *base = phy->regs;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
unsigned long val;
DBG("%s(%d)\n", __func__, __LINE__);
/* Clear the run bit to stop SOFs - 2LS WAR */
val = readl(base + USB_USBCMD);
val &= ~USB_USBCMD_RS;
writel(val, base + USB_USBCMD);
if (usb_phy_reg_status_wait(base + USB_USBSTS, USB_USBSTS_HCH,
USB_USBSTS_HCH, 2000)) {
pr_err("%s: timeout waiting for USB_USBSTS_HCH\n", __func__);
}
/* disable USB interrupts */
writel(0, base + USB_USBINTR);
/* If PMC is not enabled, enable it. */
val = readl(pmc_base + PMC_SLEEP_CFG);
if (!(val & UHSIC_MASTER_ENABLE_P0)) {
/** set PMC lines in suspend and enable the master */
uhsic_phy_pmc_resume(phy, remote_wakeup);
} else {
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~UHSIC_WAKE_VAL(WAKE_VAL_ANY);
val |= UHSIC_WAKE_VAL(WAKE_VAL_NONE);
writel(val, pmc_base + PMC_SLEEP_CFG);
}
val = readl(base + UHSIC_PADS_CFG1);
val |= UHSIC_PD_TX;
writel(val, base + UHSIC_PADS_CFG1);
/* Driving Resume using PMC */
val = readl(pmc_base + PMC_SLEEPWALK_UHSIC);
val &= ~UHSIC_DATA_RPD_A;
val |= UHSIC_DATA_RPU_A;
val &= ~UHSIC_STROBE_RPU_A;
val |= UHSIC_STROBE_RPD_A;
writel(val, pmc_base + PMC_SLEEPWALK_UHSIC);
return 0;
}
static int uhsic_phy_post_resume(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
unsigned long flags;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
local_irq_save(flags);
val = readl(pmc_base + PMC_SLEEPWALK_UHSIC);
val &= ~(UHSIC_DATA_RPU_A | UHSIC_STROBE_RPD_A);
val |= (UHSIC_DATA_RPD_A | UHSIC_STROBE_RPU_A);
writel(val, pmc_base + PMC_SLEEPWALK_UHSIC);
udelay(3);
val = readl(base + UHSIC_PADS_CFG1);
val &= ~UHSIC_PD_TX;
writel(val, base + UHSIC_PADS_CFG1);
val = readl(pmc_base + PMC_SLEEP_CFG);
val &= ~(UHSIC_MASTER_ENABLE);
writel(val, pmc_base + PMC_SLEEP_CFG);
/* clear pending SRI, PCI interrupts. ehci will enable interrupts */
val = readl(base + USB_USBSTS);
writel(val, base + USB_USBSTS);
val = readl(base + USB_USBCMD);
val |= USB_USBCMD_RS;
writel(val, base + USB_USBCMD);
local_irq_restore(flags);
uhsic_phy_disable_pmc_bus_ctrl(phy);
val = readl(base + USB_TXFILLTUNING);
if ((val & USB_FIFO_TXFILL_MASK) != USB_FIFO_TXFILL_THRES(0x10)) {
val = USB_FIFO_TXFILL_THRES(0x10);
writel(val, base + USB_TXFILLTUNING);
}
return 0;
}
static void uhsic_phy_restore_start(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *pmc_base = IO_ADDRESS(TEGRA_PMC_BASE);
void __iomem *base = phy->regs;
val = readl(pmc_base + UTMIP_UHSIC_STATUS);
/* check whether we wake up from the remote resume */
if (UHSIC_WALK_PTR_VAL & val) {
phy->pmc_remote_wakeup = true;
} else {
DBG("%s(%d): setting pretend connect\n", __func__, __LINE__);
val = readl(base + UHSIC_CMD_CFG0);
val |= UHSIC_PRETEND_CONNECT_DETECT;
writel(val, base + UHSIC_CMD_CFG0);
}
}
static void uhsic_phy_restore_end(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
int wait_time_us = 25000; /* FPR should be set by this time */
DBG("%s(%d)\n", __func__, __LINE__);
/* check whether we wake up from the remote resume */
if (phy->pmc_remote_wakeup) {
/* wait until FPR bit is set automatically on remote resume */
do {
val = readl(base + USB_PORTSC);
udelay(1);
if (wait_time_us == 0) {
uhsic_phy_disable_pmc_bus_ctrl(phy);
uhsic_phy_post_resume(phy);
return;
}
wait_time_us--;
} while (val & (USB_PORTSC_RESUME | USB_PORTSC_SUSP));
/* Clear PCI and SRI bits to avoid an interrupt upon resume */
val = readl(base + USB_USBSTS);
writel(val, base + USB_USBSTS);
/* wait to avoid SOF if there is any */
if (usb_phy_reg_status_wait(base + USB_USBSTS,
USB_USBSTS_SRI, USB_USBSTS_SRI, 2500)) {
pr_warn("%s: timeout waiting for SOF\n", __func__);
}
uhsic_phy_post_resume(phy);
}
}
static int hsic_rail_enable(struct tegra_usb_phy *phy)
{
int ret;
if (phy->hsic_reg == NULL) {
phy->hsic_reg = regulator_get(&phy->pdev->dev, "avdd_hsic");
if (IS_ERR_OR_NULL(phy->hsic_reg)) {
pr_err("HSIC: Could not get regulator avdd_hsic\n");
ret = PTR_ERR(phy->hsic_reg);
phy->hsic_reg = NULL;
return ret;
}
}
ret = regulator_enable(phy->hsic_reg);
if (ret < 0) {
pr_err("%s avdd_hsic could not be enabled\n", __func__);
return ret;
}
return 0;
}
static int hsic_rail_disable(struct tegra_usb_phy *phy)
{
int ret;
if (phy->hsic_reg == NULL) {
pr_warn("%s: unbalanced disable\n", __func__);
return -EIO;
}
ret = regulator_disable(phy->hsic_reg);
if (ret < 0) {
pr_err("HSIC regulator avdd_hsic cannot be disabled\n");
return ret;
}
return 0;
}
static int uhsic_phy_open(struct tegra_usb_phy *phy)
{
unsigned long parent_rate;
int i;
int ret;
phy->hsic_reg = NULL;
ret = hsic_rail_enable(phy);
if (ret < 0) {
pr_err("%s avdd_hsic could not be enabled\n", __func__);
return ret;
}
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
parent_rate = clk_get_rate(clk_get_parent(phy->pllu_clk));
for (i = 0; i < ARRAY_SIZE(uhsic_freq_table); i++) {
if (uhsic_freq_table[i].freq == parent_rate) {
phy->freq = &uhsic_freq_table[i];
break;
}
}
if (!phy->freq) {
pr_err("invalid pll_u parent rate %ld\n", parent_rate);
return -EINVAL;
}
/* reset controller for reenumerating hsic device */
tegra_periph_reset_assert(phy->ctrlr_clk);
udelay(2);
tegra_periph_reset_deassert(phy->ctrlr_clk);
udelay(2);
uhsic_powerup_pmc_wake_detect(phy);
return 0;
}
static void uhsic_phy_close(struct tegra_usb_phy *phy)
{
int ret;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
uhsic_powerdown_pmc_wake_detect(phy);
ret = hsic_rail_disable(phy);
if (ret < 0)
pr_err("%s avdd_hsic could not be disabled\n", __func__);
}
static int uhsic_phy_irq(struct tegra_usb_phy *phy)
{
usb_phy_fence_read(phy);
/* check if there is any remote wake event */
if (uhsic_phy_remotewake_detected(phy))
pr_info("%s: uhsic remote wake detected\n", __func__);
return IRQ_HANDLED;
}
static int uhsic_phy_power_on(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
if (phy->phy_clk_on) {
DBG("%s(%d) inst:[%d] phy clk is already On\n",
__func__, __LINE__, phy->inst);
return 0;
}
val = readl(base + UHSIC_PADS_CFG1);
val &= ~(UHSIC_PD_BG | UHSIC_PD_TRK | UHSIC_PD_RX |
UHSIC_PD_ZI | UHSIC_RPD_DATA | UHSIC_RPD_STROBE);
val |= (UHSIC_RX_SEL | UHSIC_PD_TX);
writel(val, base + UHSIC_PADS_CFG1);
val = readl(base + USB_SUSP_CTRL);
val |= UHSIC_RESET;
writel(val, base + USB_SUSP_CTRL);
udelay(1);
val = readl(base + USB_SUSP_CTRL);
val |= UHSIC_PHY_ENABLE;
writel(val, base + USB_SUSP_CTRL);
val = readl(base + UHSIC_HSRX_CFG0);
val |= UHSIC_IDLE_WAIT(HSIC_IDLE_WAIT_DELAY);
val |= UHSIC_ELASTIC_UNDERRUN_LIMIT(HSIC_ELASTIC_UNDERRUN_LIMIT);
val |= UHSIC_ELASTIC_OVERRUN_LIMIT(HSIC_ELASTIC_OVERRUN_LIMIT);
writel(val, base + UHSIC_HSRX_CFG0);
val = readl(base + UHSIC_HSRX_CFG1);
val |= UHSIC_HS_SYNC_START_DLY(HSIC_SYNC_START_DELAY);
writel(val, base + UHSIC_HSRX_CFG1);
/* WAR HSIC TX */
val = readl(base + UHSIC_TX_CFG0);
val &= ~UHSIC_HS_READY_WAIT_FOR_VALID;
writel(val, base + UHSIC_TX_CFG0);
val = readl(base + UHSIC_MISC_CFG0);
val |= UHSIC_SUSPEND_EXIT_ON_EDGE;
/* Disable generic bus reset, to allow AP30 specific bus reset*/
val |= UHSIC_DISABLE_BUSRESET;
writel(val, base + UHSIC_MISC_CFG0);
val = readl(base + UHSIC_MISC_CFG1);
val |= UHSIC_PLLU_STABLE_COUNT(phy->freq->stable_count);
writel(val, base + UHSIC_MISC_CFG1);
val = readl(base + UHSIC_PLL_CFG1);
val |= UHSIC_PLLU_ENABLE_DLY_COUNT(phy->freq->enable_delay);
val |= UHSIC_XTAL_FREQ_COUNT(phy->freq->xtal_freq_count);
writel(val, base + UHSIC_PLL_CFG1);
val = readl(base + USB_SUSP_CTRL);
val &= ~(UHSIC_RESET);
writel(val, base + USB_SUSP_CTRL);
udelay(1);
val = readl(base + UHSIC_PADS_CFG1);
val &= ~(UHSIC_PD_TX);
writel(val, base + UHSIC_PADS_CFG1);
val = readl(base + USB_USBMODE);
val |= USB_USBMODE_HOST;
writel(val, base + USB_USBMODE);
/* Change the USB controller PHY type to HSIC */
val = readl(base + HOSTPC1_DEVLC);
val &= ~HOSTPC1_DEVLC_PTS(HOSTPC1_DEVLC_PTS_MASK);
val |= HOSTPC1_DEVLC_PTS(HOSTPC1_DEVLC_PTS_HSIC);
val &= ~HOSTPC1_DEVLC_STS;
writel(val, base + HOSTPC1_DEVLC);
val = readl(base + USB_TXFILLTUNING);
if ((val & USB_FIFO_TXFILL_MASK) != USB_FIFO_TXFILL_THRES(0x10)) {
val = USB_FIFO_TXFILL_THRES(0x10);
writel(val, base + USB_TXFILLTUNING);
}
val = readl(base + USB_PORTSC);
val &= ~(USB_PORTSC_WKOC | USB_PORTSC_WKDS | USB_PORTSC_WKCN);
writel(val, base + USB_PORTSC);
val = readl(base + UHSIC_PADS_CFG0);
val &= ~(UHSIC_TX_RTUNEN);
/* set Rtune impedance to 50 ohm */
val |= UHSIC_TX_RTUNE(8);
writel(val, base + UHSIC_PADS_CFG0);
if (usb_phy_reg_status_wait(base + USB_SUSP_CTRL,
USB_PHY_CLK_VALID, USB_PHY_CLK_VALID, 2500)) {
pr_err("%s: timeout waiting for phy to stabilize\n", __func__);
return -ETIMEDOUT;
}
phy->phy_clk_on = true;
phy->hw_accessible = true;
if (phy->pmc_sleepwalk) {
DBG("%s(%d) inst:[%d] restore phy\n", __func__, __LINE__,
phy->inst);
uhsic_phy_restore_start(phy);
usb_phy_bringup_host_controller(phy);
uhsic_phy_restore_end(phy);
phy->pmc_sleepwalk = false;
}
return 0;
}
static int uhsic_phy_power_off(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
if (!phy->phy_clk_on) {
DBG("%s(%d) inst:[%d] phy clk is already off\n",
__func__, __LINE__, phy->inst);
return 0;
}
/* Disable interrupts */
writel(0, base + USB_USBINTR);
if (phy->pmc_sleepwalk == false) {
uhsic_setup_pmc_wake_detect(phy);
phy->pmc_sleepwalk = true;
}
val = readl(base + HOSTPC1_DEVLC);
val |= HOSTPC1_DEVLC_PHCD;
writel(val, base + HOSTPC1_DEVLC);
/* Remove power downs for HSIC from PADS CFG1 register */
val = readl(base + UHSIC_PADS_CFG1);
val |= (UHSIC_PD_BG |UHSIC_PD_TRK | UHSIC_PD_RX |
UHSIC_PD_ZI | UHSIC_PD_TX);
writel(val, base + UHSIC_PADS_CFG1);
phy->phy_clk_on = false;
phy->hw_accessible = false;
return 0;
}
static int uhsic_phy_bus_port_power(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
val = readl(base + USB_USBMODE);
val |= USB_USBMODE_HOST;
writel(val, base + USB_USBMODE);
/* Change the USB controller PHY type to HSIC */
val = readl(base + HOSTPC1_DEVLC);
val &= ~(HOSTPC1_DEVLC_PTS(HOSTPC1_DEVLC_PTS_MASK));
val |= HOSTPC1_DEVLC_PTS(HOSTPC1_DEVLC_PTS_HSIC);
writel(val, base + HOSTPC1_DEVLC);
val = readl(base + UHSIC_MISC_CFG0);
val |= UHSIC_DETECT_SHORT_CONNECT;
writel(val, base + UHSIC_MISC_CFG0);
udelay(1);
val = readl(base + UHSIC_MISC_CFG0);
val |= UHSIC_FORCE_XCVR_MODE;
writel(val, base + UHSIC_MISC_CFG0);
val = readl(base + UHSIC_PADS_CFG1);
val &= ~UHSIC_RPD_STROBE;
writel(val, base + UHSIC_PADS_CFG1);
if (phy->pdata->ops && phy->pdata->ops->port_power)
phy->pdata->ops->port_power();
if (usb_phy_reg_status_wait(base + UHSIC_STAT_CFG0,
UHSIC_CONNECT_DETECT, UHSIC_CONNECT_DETECT, 25000)) {
pr_err("%s: timeout waiting for UHSIC_CONNECT_DETECT\n",
__func__);
return -ETIMEDOUT;
}
return 0;
}
static int uhsic_phy_bus_reset(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
/* Change the USB controller PHY type to HSIC */
val = readl(base + HOSTPC1_DEVLC);
val &= ~HOSTPC1_DEVLC_PTS(HOSTPC1_DEVLC_PTS_MASK);
val |= HOSTPC1_DEVLC_PTS(HOSTPC1_DEVLC_PTS_HSIC);
val &= ~HOSTPC1_DEVLC_STS;
writel(val, base + HOSTPC1_DEVLC);
/* wait here, otherwise HOSTPC1_DEVLC_PSPD will timeout */
mdelay(5);
val = readl(base + USB_PORTSC);
val |= USB_PORTSC_PTC(5);
writel(val, base + USB_PORTSC);
udelay(2);
val = readl(base + USB_PORTSC);
val &= ~(USB_PORTSC_PTC(~0));
writel(val, base + USB_PORTSC);
udelay(2);
if (usb_phy_reg_status_wait(base + USB_PORTSC, USB_PORTSC_LS(0),
0, 2000)) {
pr_err("%s: timeout waiting for USB_PORTSC_LS\n", __func__);
return -ETIMEDOUT;
}
/* Poll until CCS is enabled */
if (usb_phy_reg_status_wait(base + USB_PORTSC, USB_PORTSC_CCS,
USB_PORTSC_CCS, 2000)) {
pr_err("%s: timeout waiting for USB_PORTSC_CCS\n", __func__);
return -ETIMEDOUT;
}
if (usb_phy_reg_status_wait(base + HOSTPC1_DEVLC,
HOSTPC1_DEVLC_PSPD(2),
HOSTPC1_DEVLC_PSPD(2), 2000) < 0) {
pr_err("%s: timeout waiting hsic high speed configuration\n",
__func__);
return -ETIMEDOUT;
}
val = readl(base + USB_USBCMD);
val &= ~USB_USBCMD_RS;
writel(val, base + USB_USBCMD);
if (usb_phy_reg_status_wait(base + USB_USBSTS, USB_USBSTS_HCH,
USB_USBSTS_HCH, 2000)) {
pr_err("%s: timeout waiting for USB_USBSTS_HCH\n", __func__);
return -ETIMEDOUT;
}
val = readl(base + UHSIC_PADS_CFG1);
val &= ~UHSIC_RPU_STROBE;
val |= UHSIC_RPD_STROBE;
writel(val, base + UHSIC_PADS_CFG1);
mdelay(50);
val = readl(base + UHSIC_PADS_CFG1);
val &= ~UHSIC_RPD_STROBE;
val |= UHSIC_RPU_STROBE;
writel(val, base + UHSIC_PADS_CFG1);
val = readl(base + USB_USBCMD);
val |= USB_USBCMD_RS;
writel(val, base + USB_USBCMD);
val = readl(base + UHSIC_PADS_CFG1);
val &= ~UHSIC_RPU_STROBE;
writel(val, base + UHSIC_PADS_CFG1);
if (usb_phy_reg_status_wait(base + USB_USBCMD, USB_USBCMD_RS,
USB_USBCMD_RS, 2000)) {
pr_err("%s: timeout waiting for USB_USBCMD_RS\n", __func__);
return -ETIMEDOUT;
}
return 0;
}
int uhsic_phy_resume(struct tegra_usb_phy *phy)
{
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
return 0;
}
static void ulpi_set_trimmer(struct tegra_usb_phy *phy)
{
struct tegra_ulpi_config *config = &phy->pdata->u_cfg.ulpi;
void __iomem *base = phy->regs;
unsigned long val;
val = ULPI_DATA_TRIMMER_SEL(config->data_trimmer);
val |= ULPI_STPDIRNXT_TRIMMER_SEL(config->stpdirnxt_trimmer);
val |= ULPI_DIR_TRIMMER_SEL(config->dir_trimmer);
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);
}
static void reset_utmip_uhsic(void __iomem *base)
{
unsigned long val;
val = readl(base + USB_SUSP_CTRL);
val |= UHSIC_RESET;
writel(val, base + USB_SUSP_CTRL);
val = readl(base + USB_SUSP_CTRL);
val |= UTMIP_RESET;
writel(val, base + USB_SUSP_CTRL);
}
static void ulpi_set_host(void __iomem *base)
{
unsigned long val;
val = readl(base + USB_USBMODE);
val &= ~USB_USBMODE_MASK;
val |= USB_USBMODE_HOST;
writel(val, base + USB_USBMODE);
val = readl(base + HOSTPC1_DEVLC);
val |= HOSTPC1_DEVLC_PTS(2);
writel(val, base + HOSTPC1_DEVLC);
}
static inline void ulpi_pinmux_bypass(struct tegra_usb_phy *phy, bool enable)
{
unsigned long val;
void __iomem *base = phy->regs;
val = readl(base + ULPI_TIMING_CTRL_0);
if (enable)
val |= ULPI_OUTPUT_PINMUX_BYP;
else
val &= ~ULPI_OUTPUT_PINMUX_BYP;
writel(val, base + ULPI_TIMING_CTRL_0);
}
static inline void ulpi_null_phy_set_tristate(bool enable)
{
#ifndef CONFIG_ARCH_TEGRA_2x_SOC
int tristate = (enable) ? TEGRA_TRI_TRISTATE : TEGRA_TRI_NORMAL;
tegra_pinmux_set_tristate(TEGRA_PINGROUP_ULPI_DATA0, tristate);
tegra_pinmux_set_tristate(TEGRA_PINGROUP_ULPI_DATA1, tristate);
tegra_pinmux_set_tristate(TEGRA_PINGROUP_ULPI_DATA2, tristate);
tegra_pinmux_set_tristate(TEGRA_PINGROUP_ULPI_DATA3, tristate);
tegra_pinmux_set_tristate(TEGRA_PINGROUP_ULPI_DATA4, tristate);
tegra_pinmux_set_tristate(TEGRA_PINGROUP_ULPI_DATA5, tristate);
tegra_pinmux_set_tristate(TEGRA_PINGROUP_ULPI_DATA6, tristate);
tegra_pinmux_set_tristate(TEGRA_PINGROUP_ULPI_DATA7, tristate);
tegra_pinmux_set_tristate(TEGRA_PINGROUP_ULPI_NXT, tristate);
if (enable)
tegra_pinmux_set_tristate(TEGRA_PINGROUP_ULPI_DIR, tristate);
#endif
}
static void ulpi_null_phy_obs_read(void)
{
static void __iomem *apb_misc;
unsigned slv0_obs, s2s_obs;
if (!apb_misc)
apb_misc = ioremap(TEGRA_APB_MISC_BASE, TEGRA_APB_MISC_SIZE);
writel(0x80d1003c, apb_misc + APB_MISC_GP_OBSCTRL_0);
slv0_obs = readl(apb_misc + APB_MISC_GP_OBSDATA_0);
writel(0x80d10040, apb_misc + APB_MISC_GP_OBSCTRL_0);
s2s_obs = readl(apb_misc + APB_MISC_GP_OBSDATA_0);
pr_debug("slv0 obs: %08x\ns2s obs: %08x\n", slv0_obs, s2s_obs);
}
static const struct gpio ulpi_gpios[] = {
{ULPI_STP, GPIOF_IN, "ULPI_STP"},
{ULPI_DIR, GPIOF_OUT_INIT_LOW, "ULPI_DIR"},
{ULPI_D0, GPIOF_OUT_INIT_LOW, "ULPI_D0"},
{ULPI_D1, GPIOF_OUT_INIT_LOW, "ULPI_D1"},
};
static int ulpi_null_phy_open(struct tegra_usb_phy *phy)
{
struct tegra_ulpi_config *config = &phy->pdata->u_cfg.ulpi;
int ret;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
ret = gpio_request_array(ulpi_gpios, ARRAY_SIZE(ulpi_gpios));
if (ret)
return ret;
if (gpio_is_valid(config->phy_restore_gpio)) {
ret = gpio_request(config->phy_restore_gpio, "phy_restore");
if (ret)
goto err_gpio_free;
gpio_direction_input(config->phy_restore_gpio);
}
tegra_periph_reset_assert(phy->ctrlr_clk);
udelay(10);
tegra_periph_reset_deassert(phy->ctrlr_clk);
return 0;
err_gpio_free:
gpio_free_array(ulpi_gpios, ARRAY_SIZE(ulpi_gpios));
return ret;
}
static void ulpi_null_phy_close(struct tegra_usb_phy *phy)
{
struct tegra_ulpi_config *config = &phy->pdata->u_cfg.ulpi;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
if (gpio_is_valid(config->phy_restore_gpio))
gpio_free(config->phy_restore_gpio);
gpio_free_array(ulpi_gpios, ARRAY_SIZE(ulpi_gpios));
}
static int ulpi_null_phy_power_off(struct tegra_usb_phy *phy)
{
unsigned int val;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
if (!phy->phy_clk_on) {
DBG("%s(%d) inst:[%d] phy clk is already off\n", __func__,
__LINE__, phy->inst);
return 0;
}
phy->phy_clk_on = false;
phy->hw_accessible = false;
ulpi_null_phy_set_tristate(true);
val = readl(base + ULPIS2S_CTRL);
val &= ~ULPIS2S_PLLU_MASTER_BLASTER60;
writel(val, base + ULPIS2S_CTRL);
return 0;
}
/* NOTE: this function must be called before ehci reset */
static int ulpi_null_phy_init(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
val = readl(base + ULPIS2S_CTRL);
val |= ULPIS2S_SLV0_CLAMP_XMIT;
writel(val, base + ULPIS2S_CTRL);
val = readl(base + USB_SUSP_CTRL);
val |= ULPIS2S_SLV0_RESET;
writel(val, base + USB_SUSP_CTRL);
udelay(10);
return 0;
}
static int ulpi_null_phy_irq(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
usb_phy_fence_read(phy);
if (phy->bus_reseting){
val = readl(base + USB_USBCMD);
val |= USB_USBCMD_RS;
writel(val, base + USB_USBCMD);
phy->bus_reseting = false;
}
return IRQ_HANDLED;
}
/* NOTE: this function must be called after ehci reset */
static int ulpi_null_phy_cmd_reset(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
ulpi_set_host(base);
/* remove slave0 reset */
val = readl(base + USB_SUSP_CTRL);
val &= ~ULPIS2S_SLV0_RESET;
writel(val, base + USB_SUSP_CTRL);
val = readl(base + ULPIS2S_CTRL);
val &= ~ULPIS2S_SLV0_CLAMP_XMIT;
writel(val, base + ULPIS2S_CTRL);
udelay(10);
return 0;
}
static int ulpi_phy_bus_reset(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
/*DISABLE RUN BIT */
val = readl(base + USB_USBCMD);
val &= ~USB_USBCMD_RS;
writel(val, base + USB_USBCMD);
phy->bus_reseting = true;
return 0;
}
static int ulpi_null_phy_restore(struct tegra_usb_phy *phy)
{
struct tegra_ulpi_config *config = &phy->pdata->u_cfg.ulpi;
unsigned long timeout;
int ulpi_stp = ULPI_STP;
if (gpio_is_valid(config->phy_restore_gpio))
ulpi_stp = config->phy_restore_gpio;
/* disable ULPI pinmux bypass */
ulpi_pinmux_bypass(phy, false);
/* driving linstate by GPIO */
gpio_set_value(ULPI_D0, 0);
gpio_set_value(ULPI_D1, 0);
/* driving DIR high */
gpio_set_value(ULPI_DIR, 1);
/* remove ULPI tristate */
ulpi_null_phy_set_tristate(false);
/* wait for STP high */
timeout = jiffies + msecs_to_jiffies(25);
while (!gpio_get_value(ulpi_stp)) {
if (time_after(jiffies, timeout)) {
pr_warn("phy restore timeout\n");
return 1;
}
mdelay(1);
}
return 0;
}
static int ulpi_null_phy_lp0_resume(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
ulpi_null_phy_init(phy);
val = readl(base + USB_USBCMD);
val |= USB_CMD_RESET;
writel(val, base + USB_USBCMD);
if (usb_phy_reg_status_wait(base + USB_USBCMD,
USB_CMD_RESET, 0, 2500) < 0) {
pr_err("%s: timeout waiting for reset\n", __func__);
}
ulpi_null_phy_cmd_reset(phy);
val = readl(base + USB_USBCMD);
val |= USB_USBCMD_RS;
writel(val, base + USB_USBCMD);
if (usb_phy_reg_status_wait(base + USB_USBCMD, USB_USBCMD_RS,
USB_USBCMD_RS, 2000)) {
pr_err("%s: timeout waiting for USB_USBCMD_RS\n", __func__);
return -ETIMEDOUT;
}
/* Enable Port Power */
val = readl(base + USB_PORTSC);
val |= USB_PORTSC_PP;
writel(val, base + USB_PORTSC);
udelay(10);
ulpi_null_phy_restore(phy);
return 0;
}
static int ulpi_null_phy_power_on(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
struct tegra_ulpi_config *config = &phy->pdata->u_cfg.ulpi;
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
if (phy->phy_clk_on) {
DBG("%s(%d) inst:[%d] phy clk is already On\n", __func__,
__LINE__, phy->inst);
return 0;
}
reset_utmip_uhsic(base);
/* remove ULPI PADS CLKEN reset */
val = readl(base + USB_SUSP_CTRL);
val &= ~ULPI_PADS_CLKEN_RESET;
writel(val, base + USB_SUSP_CTRL);
udelay(10);
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);
udelay(10);
/* set timming parameters */
val = readl(base + ULPI_TIMING_CTRL_0);
val |= ULPI_SHADOW_CLK_LOOPBACK_EN;
val &= ~ULPI_SHADOW_CLK_SEL;
val &= ~ULPI_LBK_PAD_EN;
val |= ULPI_SHADOW_CLK_DELAY(config->shadow_clk_delay);
val |= ULPI_CLOCK_OUT_DELAY(config->clock_out_delay);
val |= ULPI_LBK_PAD_E_INPUT_OR;
writel(val, base + ULPI_TIMING_CTRL_0);
writel(0, base + ULPI_TIMING_CTRL_1);
udelay(10);
/* start internal 60MHz clock */
val = readl(base + ULPIS2S_CTRL);
val |= ULPIS2S_ENA;
val |= ULPIS2S_SUPPORT_DISCONNECT;
val |= ULPIS2S_SPARE((phy->pdata->op_mode == TEGRA_USB_OPMODE_HOST) ? 3 : 1);
val |= ULPIS2S_PLLU_MASTER_BLASTER60;
writel(val, base + ULPIS2S_CTRL);
/* select ULPI_CORE_CLK_SEL to SHADOW_CLK */
val = readl(base + ULPI_TIMING_CTRL_0);
val |= ULPI_CORE_CLK_SEL;
writel(val, base + ULPI_TIMING_CTRL_0);
udelay(10);
/* enable ULPI null phy clock - can't set the trimmers before this */
val = readl(base + ULPI_TIMING_CTRL_0);
val |= ULPI_CLK_OUT_ENA;
writel(val, base + ULPI_TIMING_CTRL_0);
udelay(10);
if (usb_phy_reg_status_wait(base + USB_SUSP_CTRL, USB_PHY_CLK_VALID,
USB_PHY_CLK_VALID, 2500)) {
pr_err("%s: timeout waiting for phy to stabilize\n", __func__);
return -ETIMEDOUT;
}
/* set ULPI trimmers */
ulpi_set_trimmer(phy);
ulpi_set_host(base);
/* remove slave0 reset */
val = readl(base + USB_SUSP_CTRL);
val &= ~ULPIS2S_SLV0_RESET;
writel(val, base + USB_SUSP_CTRL);
/* remove slave1 and line reset */
val = readl(base + USB_SUSP_CTRL);
val &= ~ULPIS2S_SLV1_RESET;
val &= ~ULPIS2S_LINE_RESET;
/* remove ULPI PADS reset */
val &= ~ULPI_PADS_RESET;
writel(val, base + USB_SUSP_CTRL);
if (!phy->ulpi_clk_padout_ena) {
val = readl(base + ULPI_TIMING_CTRL_0);
val |= ULPI_CLK_PADOUT_ENA;
writel(val, base + ULPI_TIMING_CTRL_0);
phy->ulpi_clk_padout_ena = true;
} else {
if (!readl(base + USB_ASYNCLISTADDR))
ulpi_null_phy_lp0_resume(phy);
}
udelay(10);
phy->bus_reseting = false;
phy->phy_clk_on = true;
phy->hw_accessible = true;
return 0;
}
static int ulpi_null_phy_pre_resume(struct tegra_usb_phy *phy, bool remote_wakeup)
{
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
ulpi_null_phy_obs_read();
usb_phy_wait_for_sof(phy);
ulpi_null_phy_obs_read();
return 0;
}
static int ulpi_null_phy_post_resume(struct tegra_usb_phy *phy)
{
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
ulpi_null_phy_obs_read();
return 0;
}
static int ulpi_null_phy_resume(struct tegra_usb_phy *phy)
{
unsigned long val;
void __iomem *base = phy->regs;
if (!readl(base + USB_ASYNCLISTADDR)) {
/* enable ULPI CLK output pad */
val = readl(base + ULPI_TIMING_CTRL_0);
val |= ULPI_CLK_PADOUT_ENA;
writel(val, base + ULPI_TIMING_CTRL_0);
/* enable ULPI pinmux bypass */
ulpi_pinmux_bypass(phy, true);
udelay(5);
#ifndef CONFIG_ARCH_TEGRA_2x_SOC
/* remove DIR tristate */
tegra_pinmux_set_tristate(TEGRA_PINGROUP_ULPI_DIR,
TEGRA_TRI_NORMAL);
#endif
}
return 0;
}
static struct tegra_usb_phy_ops utmi_phy_ops = {
.open = utmi_phy_open,
.close = utmi_phy_close,
.irq = utmi_phy_irq,
.power_on = utmi_phy_power_on,
.power_off = utmi_phy_power_off,
.pre_resume = utmi_phy_pre_resume,
.resume = utmi_phy_resume,
.post_resume = utmi_phy_post_resume,
.charger_detect = utmi_phy_charger_detect,
.post_suspend = phy_post_suspend,
};
static struct tegra_usb_phy_ops uhsic_phy_ops = {
.open = uhsic_phy_open,
.close = uhsic_phy_close,
.irq = uhsic_phy_irq,
.power_on = uhsic_phy_power_on,
.power_off = uhsic_phy_power_off,
.pre_resume = uhsic_phy_pre_resume,
.resume = uhsic_phy_resume,
.post_resume = uhsic_phy_post_resume,
.port_power = uhsic_phy_bus_port_power,
.bus_reset = uhsic_phy_bus_reset,
.post_suspend = phy_post_suspend,
};
static struct tegra_usb_phy_ops ulpi_null_phy_ops = {
.open = ulpi_null_phy_open,
.close = ulpi_null_phy_close,
.init = ulpi_null_phy_init,
.irq = ulpi_null_phy_irq,
.power_on = ulpi_null_phy_power_on,
.power_off = ulpi_null_phy_power_off,
.pre_resume = ulpi_null_phy_pre_resume,
.resume = ulpi_null_phy_resume,
.post_resume = ulpi_null_phy_post_resume,
.reset = ulpi_null_phy_cmd_reset,
.post_suspend = phy_post_suspend,
.bus_reset = ulpi_phy_bus_reset,
};
static struct tegra_usb_phy_ops ulpi_link_phy_ops;
static struct tegra_usb_phy_ops icusb_phy_ops;
static struct tegra_usb_phy_ops *phy_ops[] = {
[TEGRA_USB_PHY_INTF_UTMI] = &utmi_phy_ops,
[TEGRA_USB_PHY_INTF_ULPI_LINK] = &ulpi_link_phy_ops,
[TEGRA_USB_PHY_INTF_ULPI_NULL] = &ulpi_null_phy_ops,
[TEGRA_USB_PHY_INTF_HSIC] = &uhsic_phy_ops,
[TEGRA_USB_PHY_INTF_ICUSB] = &icusb_phy_ops,
};
int tegra3_usb_phy_init_ops(struct tegra_usb_phy *phy)
{
DBG("%s(%d) inst:[%d]\n", __func__, __LINE__, phy->inst);
phy->ops = phy_ops[phy->pdata->phy_intf];
/* FIXME: uncommenting below line to make USB host mode fail*/
/* usb_phy_power_down_pmc(); */
return 0;
}