2018-01-26 14:22:04 -06:00
// SPDX-License-Identifier: GPL-2.0+
2013-08-09 16:49:19 +02:00
/*
2013-08-09 16:49:24 +02:00
* PCIe host controller driver for Tegra SoCs
2013-08-09 16:49:19 +02:00
*
* Copyright ( c ) 2010 , CompuLab , Ltd .
* Author : Mike Rapoport < mike @ compulab . co . il >
*
* Based on NVIDIA PCIe driver
* Copyright ( c ) 2008 - 2009 , NVIDIA Corporation .
*
* Bits taken from arch / arm / mach - dove / pcie . c
*
2016-07-02 19:13:31 -04:00
* Author : Thierry Reding < treding @ nvidia . com >
2013-08-09 16:49:19 +02:00
*/
# include <linux/clk.h>
2014-07-22 12:30:46 -06:00
# include <linux/debugfs.h>
2013-08-09 16:49:19 +02:00
# include <linux/delay.h>
# include <linux/export.h>
# include <linux/interrupt.h>
2018-02-28 15:30:33 +05:30
# include <linux/iopoll.h>
2013-08-09 16:49:19 +02:00
# include <linux/irq.h>
# include <linux/irqdomain.h>
# include <linux/kernel.h>
2016-07-02 19:13:31 -04:00
# include <linux/init.h>
2018-02-28 15:30:33 +05:30
# include <linux/module.h>
2013-08-09 16:49:19 +02:00
# include <linux/msi.h>
# include <linux/of_address.h>
# include <linux/of_pci.h>
# include <linux/of_platform.h>
# include <linux/pci.h>
2014-08-26 17:11:38 +02:00
# include <linux/phy/phy.h>
2013-08-09 16:49:19 +02:00
# include <linux/platform_device.h>
2013-11-06 15:56:58 -07:00
# include <linux/reset.h>
2013-08-09 16:49:19 +02:00
# include <linux/sizes.h>
# include <linux/slab.h>
# include <linux/vmalloc.h>
# include <linux/regulator/consumer.h>
2014-07-17 13:17:24 +02:00
# include <soc/tegra/cpuidle.h>
2014-07-11 13:19:06 +02:00
# include <soc/tegra/pmc.h>
2014-07-17 13:17:24 +02:00
2018-05-11 12:15:30 -05:00
# include "../pci.h"
2013-08-09 16:49:19 +02:00
# define INT_PCI_MSI_NR (8 * 32)
/* register definitions */
# define AFI_AXI_BAR0_SZ 0x00
# define AFI_AXI_BAR1_SZ 0x04
# define AFI_AXI_BAR2_SZ 0x08
# define AFI_AXI_BAR3_SZ 0x0c
# define AFI_AXI_BAR4_SZ 0x10
# define AFI_AXI_BAR5_SZ 0x14
# define AFI_AXI_BAR0_START 0x18
# define AFI_AXI_BAR1_START 0x1c
# define AFI_AXI_BAR2_START 0x20
# define AFI_AXI_BAR3_START 0x24
# define AFI_AXI_BAR4_START 0x28
# define AFI_AXI_BAR5_START 0x2c
# define AFI_FPCI_BAR0 0x30
# define AFI_FPCI_BAR1 0x34
# define AFI_FPCI_BAR2 0x38
# define AFI_FPCI_BAR3 0x3c
# define AFI_FPCI_BAR4 0x40
# define AFI_FPCI_BAR5 0x44
# define AFI_CACHE_BAR0_SZ 0x48
# define AFI_CACHE_BAR0_ST 0x4c
# define AFI_CACHE_BAR1_SZ 0x50
# define AFI_CACHE_BAR1_ST 0x54
# define AFI_MSI_BAR_SZ 0x60
# define AFI_MSI_FPCI_BAR_ST 0x64
# define AFI_MSI_AXI_BAR_ST 0x68
# define AFI_MSI_VEC0 0x6c
# define AFI_MSI_VEC1 0x70
# define AFI_MSI_VEC2 0x74
# define AFI_MSI_VEC3 0x78
# define AFI_MSI_VEC4 0x7c
# define AFI_MSI_VEC5 0x80
# define AFI_MSI_VEC6 0x84
# define AFI_MSI_VEC7 0x88
# define AFI_MSI_EN_VEC0 0x8c
# define AFI_MSI_EN_VEC1 0x90
# define AFI_MSI_EN_VEC2 0x94
# define AFI_MSI_EN_VEC3 0x98
# define AFI_MSI_EN_VEC4 0x9c
# define AFI_MSI_EN_VEC5 0xa0
# define AFI_MSI_EN_VEC6 0xa4
# define AFI_MSI_EN_VEC7 0xa8
# define AFI_CONFIGURATION 0xac
# define AFI_CONFIGURATION_EN_FPCI (1 << 0)
# define AFI_FPCI_ERROR_MASKS 0xb0
# define AFI_INTR_MASK 0xb4
# define AFI_INTR_MASK_INT_MASK (1 << 0)
# define AFI_INTR_MASK_MSI_MASK (1 << 8)
# define AFI_INTR_CODE 0xb8
# define AFI_INTR_CODE_MASK 0xf
2014-08-26 17:11:38 +02:00
# define AFI_INTR_INI_SLAVE_ERROR 1
# define AFI_INTR_INI_DECODE_ERROR 2
2013-08-09 16:49:19 +02:00
# define AFI_INTR_TARGET_ABORT 3
# define AFI_INTR_MASTER_ABORT 4
# define AFI_INTR_INVALID_WRITE 5
# define AFI_INTR_LEGACY 6
# define AFI_INTR_FPCI_DECODE_ERROR 7
2014-08-26 17:11:38 +02:00
# define AFI_INTR_AXI_DECODE_ERROR 8
# define AFI_INTR_FPCI_TIMEOUT 9
# define AFI_INTR_PE_PRSNT_SENSE 10
# define AFI_INTR_PE_CLKREQ_SENSE 11
# define AFI_INTR_CLKCLAMP_SENSE 12
# define AFI_INTR_RDY4PD_SENSE 13
# define AFI_INTR_P2P_ERROR 14
2013-08-09 16:49:19 +02:00
# define AFI_INTR_SIGNATURE 0xbc
# define AFI_UPPER_FPCI_ADDRESS 0xc0
# define AFI_SM_INTR_ENABLE 0xc4
# define AFI_SM_INTR_INTA_ASSERT (1 << 0)
# define AFI_SM_INTR_INTB_ASSERT (1 << 1)
# define AFI_SM_INTR_INTC_ASSERT (1 << 2)
# define AFI_SM_INTR_INTD_ASSERT (1 << 3)
# define AFI_SM_INTR_INTA_DEASSERT (1 << 4)
# define AFI_SM_INTR_INTB_DEASSERT (1 << 5)
# define AFI_SM_INTR_INTC_DEASSERT (1 << 6)
# define AFI_SM_INTR_INTD_DEASSERT (1 << 7)
# define AFI_AFI_INTR_ENABLE 0xc8
# define AFI_INTR_EN_INI_SLVERR (1 << 0)
# define AFI_INTR_EN_INI_DECERR (1 << 1)
# define AFI_INTR_EN_TGT_SLVERR (1 << 2)
# define AFI_INTR_EN_TGT_DECERR (1 << 3)
# define AFI_INTR_EN_TGT_WRERR (1 << 4)
# define AFI_INTR_EN_DFPCI_DECERR (1 << 5)
# define AFI_INTR_EN_AXI_DECERR (1 << 6)
# define AFI_INTR_EN_FPCI_TIMEOUT (1 << 7)
2013-08-09 16:49:24 +02:00
# define AFI_INTR_EN_PRSNT_SENSE (1 << 8)
2013-08-09 16:49:19 +02:00
2018-02-28 15:30:33 +05:30
# define AFI_PCIE_PME 0xf0
2013-08-09 16:49:19 +02:00
# define AFI_PCIE_CONFIG 0x0f8
# define AFI_PCIE_CONFIG_PCIE_DISABLE(x) (1 << ((x) + 1))
# define AFI_PCIE_CONFIG_PCIE_DISABLE_ALL 0xe
# define AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_MASK (0xf << 20)
# define AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_SINGLE (0x0 << 20)
2013-08-09 16:49:24 +02:00
# define AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_420 (0x0 << 20)
2014-08-26 17:11:38 +02:00
# define AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_X2_X1 (0x0 << 20)
2017-09-27 17:28:35 +05:30
# define AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_401 (0x0 << 20)
2013-08-09 16:49:19 +02:00
# define AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_DUAL (0x1 << 20)
2013-08-09 16:49:24 +02:00
# define AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_222 (0x1 << 20)
2014-08-26 17:11:38 +02:00
# define AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_X4_X1 (0x1 << 20)
2017-09-27 17:28:35 +05:30
# define AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_211 (0x1 << 20)
2013-08-09 16:49:24 +02:00
# define AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_411 (0x2 << 20)
2017-09-27 17:28:35 +05:30
# define AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_111 (0x2 << 20)
2013-08-09 16:49:19 +02:00
# define AFI_FUSE 0x104
# define AFI_FUSE_PCIE_T0_GEN2_DIS (1 << 2)
# define AFI_PEX0_CTRL 0x110
# define AFI_PEX1_CTRL 0x118
2013-08-09 16:49:24 +02:00
# define AFI_PEX2_CTRL 0x128
2013-08-09 16:49:19 +02:00
# define AFI_PEX_CTRL_RST (1 << 0)
2013-08-09 16:49:24 +02:00
# define AFI_PEX_CTRL_CLKREQ_EN (1 << 1)
2013-08-09 16:49:19 +02:00
# define AFI_PEX_CTRL_REFCLK_EN (1 << 3)
2014-08-26 17:11:38 +02:00
# define AFI_PEX_CTRL_OVERRIDE_EN (1 << 4)
# define AFI_PLLE_CONTROL 0x160
# define AFI_PLLE_CONTROL_BYPASS_PADS2PLLE_CONTROL (1 << 9)
# define AFI_PLLE_CONTROL_PADS2PLLE_CONTROL_EN (1 << 1)
2013-08-09 16:49:19 +02:00
2013-08-09 16:49:24 +02:00
# define AFI_PEXBIAS_CTRL_0 0x168
2016-07-25 16:02:05 -05:00
# define RP_VEND_XP 0x00000f00
2013-08-09 16:49:19 +02:00
# define RP_VEND_XP_DL_UP (1 << 30)
2016-11-25 11:57:14 +01:00
# define RP_VEND_CTL2 0x00000fa8
# define RP_VEND_CTL2_PCA_ENABLE (1 << 7)
2016-07-25 16:02:05 -05:00
# define RP_PRIV_MISC 0x00000fe0
# define RP_PRIV_MISC_PRSNT_MAP_EP_PRSNT (0xe << 0)
# define RP_PRIV_MISC_PRSNT_MAP_EP_ABSNT (0xf << 0)
2014-08-26 17:11:38 +02:00
2013-08-09 16:49:19 +02:00
# define RP_LINK_CONTROL_STATUS 0x00000090
# define RP_LINK_CONTROL_STATUS_DL_LINK_ACTIVE 0x20000000
# define RP_LINK_CONTROL_STATUS_LINKSTAT_MASK 0x3fff0000
2016-07-25 16:02:05 -05:00
# define PADS_CTL_SEL 0x0000009c
2013-08-09 16:49:19 +02:00
2016-07-25 16:02:05 -05:00
# define PADS_CTL 0x000000a0
2013-08-09 16:49:19 +02:00
# define PADS_CTL_IDDQ_1L (1 << 0)
# define PADS_CTL_TX_DATA_EN_1L (1 << 6)
# define PADS_CTL_RX_DATA_EN_1L (1 << 10)
2016-07-25 16:02:05 -05:00
# define PADS_PLL_CTL_TEGRA20 0x000000b8
# define PADS_PLL_CTL_TEGRA30 0x000000b4
2013-08-09 16:49:19 +02:00
# define PADS_PLL_CTL_RST_B4SM (1 << 1)
# define PADS_PLL_CTL_LOCKDET (1 << 8)
# define PADS_PLL_CTL_REFCLK_MASK (0x3 << 16)
# define PADS_PLL_CTL_REFCLK_INTERNAL_CML (0 << 16)
# define PADS_PLL_CTL_REFCLK_INTERNAL_CMOS (1 << 16)
# define PADS_PLL_CTL_REFCLK_EXTERNAL (2 << 16)
# define PADS_PLL_CTL_TXCLKREF_MASK (0x1 << 20)
# define PADS_PLL_CTL_TXCLKREF_DIV10 (0 << 20)
# define PADS_PLL_CTL_TXCLKREF_DIV5 (1 << 20)
2013-08-09 16:49:24 +02:00
# define PADS_PLL_CTL_TXCLKREF_BUF_EN (1 << 22)
2016-07-25 16:02:05 -05:00
# define PADS_REFCLK_CFG0 0x000000c8
# define PADS_REFCLK_CFG1 0x000000cc
# define PADS_REFCLK_BIAS 0x000000d0
2013-08-09 16:49:19 +02:00
2013-08-09 16:49:25 +02:00
/*
* Fields in PADS_REFCLK_CFG * . Those registers form an array of 16 - bit
* entries , one entry per PCIe port . These field definitions and desired
* values aren ' t in the TRM , but do come from NVIDIA .
*/
# define PADS_REFCLK_CFG_TERM_SHIFT 2 /* 6:2 */
# define PADS_REFCLK_CFG_E_TERM_SHIFT 7
# define PADS_REFCLK_CFG_PREDI_SHIFT 8 /* 11:8 */
# define PADS_REFCLK_CFG_DRVI_SHIFT 12 /* 15:12 */
2018-02-28 15:30:33 +05:30
# define PME_ACK_TIMEOUT 10000
2013-08-09 16:49:19 +02:00
struct tegra_msi {
2014-11-11 17:45:45 -07:00
struct msi_controller chip ;
2013-08-09 16:49:19 +02:00
DECLARE_BITMAP ( used , INT_PCI_MSI_NR ) ;
struct irq_domain * domain ;
2017-10-09 12:29:35 +02:00
unsigned long pages ;
2013-08-09 16:49:19 +02:00
struct mutex lock ;
2017-05-04 22:10:31 +02:00
u64 phys ;
2013-08-09 16:49:19 +02:00
int irq ;
} ;
2013-08-09 16:49:24 +02:00
/* used to differentiate between Tegra SoC generations */
2018-02-28 15:30:33 +05:30
struct tegra_pcie_port_soc {
struct {
u8 turnoff_bit ;
u8 ack_bit ;
} pme ;
} ;
2016-08-15 17:31:31 +02:00
struct tegra_pcie_soc {
2013-08-09 16:49:24 +02:00
unsigned int num_ports ;
2018-02-28 15:30:33 +05:30
const struct tegra_pcie_port_soc * ports ;
2013-08-09 16:49:24 +02:00
unsigned int msi_base_shift ;
u32 pads_pll_ctl ;
u32 tx_ref_sel ;
2016-07-25 16:02:27 -05:00
u32 pads_refclk_cfg0 ;
u32 pads_refclk_cfg1 ;
2013-08-09 16:49:24 +02:00
bool has_pex_clkreq_en ;
bool has_pex_bias_ctrl ;
bool has_intr_prsnt_sense ;
bool has_cml_clk ;
2014-08-26 17:11:38 +02:00
bool has_gen2 ;
2016-11-25 11:57:14 +01:00
bool force_pca_enable ;
2017-09-27 17:28:35 +05:30
bool program_uphy ;
2013-08-09 16:49:24 +02:00
} ;
2014-11-11 17:45:45 -07:00
static inline struct tegra_msi * to_tegra_msi ( struct msi_controller * chip )
2013-08-09 16:49:19 +02:00
{
return container_of ( chip , struct tegra_msi , chip ) ;
}
struct tegra_pcie {
struct device * dev ;
void __iomem * pads ;
void __iomem * afi ;
2017-12-20 21:36:07 +01:00
void __iomem * cfg ;
2013-08-09 16:49:19 +02:00
int irq ;
2017-12-20 21:36:07 +01:00
struct resource cs ;
2013-08-09 16:49:19 +02:00
struct resource io ;
2014-11-27 09:54:09 +01:00
struct resource pio ;
2013-08-09 16:49:19 +02:00
struct resource mem ;
struct resource prefetch ;
struct resource busn ;
2016-02-09 15:52:32 +01:00
struct {
resource_size_t mem ;
resource_size_t io ;
} offset ;
2013-08-09 16:49:19 +02:00
struct clk * pex_clk ;
struct clk * afi_clk ;
struct clk * pll_e ;
2013-08-09 16:49:24 +02:00
struct clk * cml_clk ;
2013-08-09 16:49:19 +02:00
2013-11-06 15:56:58 -07:00
struct reset_control * pex_rst ;
struct reset_control * afi_rst ;
struct reset_control * pcie_xrst ;
2015-11-11 18:25:59 +01:00
bool legacy_phy ;
2014-08-26 17:11:38 +02:00
struct phy * phy ;
2013-08-09 16:49:19 +02:00
struct tegra_msi msi ;
struct list_head ports ;
u32 xbar_config ;
2014-05-28 16:49:13 +02:00
struct regulator_bulk_data * supplies ;
unsigned int num_supplies ;
2013-08-09 16:49:24 +02:00
2016-08-15 17:31:31 +02:00
const struct tegra_pcie_soc * soc ;
2014-07-22 12:30:46 -06:00
struct dentry * debugfs ;
2013-08-09 16:49:19 +02:00
} ;
struct tegra_pcie_port {
struct tegra_pcie * pcie ;
2015-11-11 18:25:59 +01:00
struct device_node * np ;
2013-08-09 16:49:19 +02:00
struct list_head list ;
struct resource regs ;
void __iomem * base ;
unsigned int index ;
unsigned int lanes ;
2015-11-11 18:25:59 +01:00
struct phy * * phys ;
2013-08-09 16:49:19 +02:00
} ;
struct tegra_pcie_bus {
struct list_head list ;
unsigned int nr ;
} ;
static inline void afi_writel ( struct tegra_pcie * pcie , u32 value ,
unsigned long offset )
{
writel ( value , pcie - > afi + offset ) ;
}
static inline u32 afi_readl ( struct tegra_pcie * pcie , unsigned long offset )
{
return readl ( pcie - > afi + offset ) ;
}
static inline void pads_writel ( struct tegra_pcie * pcie , u32 value ,
unsigned long offset )
{
writel ( value , pcie - > pads + offset ) ;
}
static inline u32 pads_readl ( struct tegra_pcie * pcie , unsigned long offset )
{
return readl ( pcie - > pads + offset ) ;
}
/*
* The configuration space mapping on Tegra is somewhat similar to the ECAM
* defined by PCIe . However it deviates a bit in how the 4 bits for extended
* register accesses are mapped :
*
* [ 27 : 24 ] extended register number
* [ 23 : 16 ] bus number
* [ 15 : 11 ] device number
* [ 10 : 8 ] function number
* [ 7 : 0 ] register number
*
* Mapping the whole extended configuration space would require 256 MiB of
* virtual address space , only a small part of which will actually be used .
*
2017-12-20 21:36:07 +01:00
* To work around this , a 4 KiB region is used to generate the required
* configuration transaction with relevant B : D : F and register offset values .
* This is achieved by dynamically programming base address and size of
* AFI_AXI_BAR used for end point config space mapping to make sure that the
* address ( access to which generates correct config transaction ) falls in
* this 4 KiB region .
2013-08-09 16:49:19 +02:00
*/
2017-12-20 21:36:07 +01:00
static unsigned int tegra_pcie_conf_offset ( u8 bus , unsigned int devfn ,
unsigned int where )
2013-08-09 16:49:19 +02:00
{
2017-12-20 21:36:07 +01:00
return ( ( where & 0xf00 ) < < 16 ) | ( bus < < 16 ) | ( PCI_SLOT ( devfn ) < < 11 ) |
( PCI_FUNC ( devfn ) < < 8 ) | ( where & 0xff ) ;
2013-08-09 16:49:19 +02:00
}
2016-02-09 15:30:48 +01:00
static void __iomem * tegra_pcie_map_bus ( struct pci_bus * bus ,
unsigned int devfn ,
int where )
2013-08-09 16:49:19 +02:00
{
2018-01-11 11:38:03 +05:30
struct tegra_pcie * pcie = bus - > sysdata ;
2013-08-09 16:49:19 +02:00
void __iomem * addr = NULL ;
if ( bus - > number = = 0 ) {
unsigned int slot = PCI_SLOT ( devfn ) ;
struct tegra_pcie_port * port ;
list_for_each_entry ( port , & pcie - > ports , list ) {
if ( port - > index + 1 = = slot ) {
addr = port - > base + ( where & ~ 3 ) ;
break ;
}
}
} else {
2017-12-20 21:36:07 +01:00
unsigned int offset ;
u32 base ;
2016-02-09 15:30:48 +01:00
2017-12-20 21:36:07 +01:00
offset = tegra_pcie_conf_offset ( bus - > number , devfn , where ) ;
2016-02-09 15:30:48 +01:00
2017-12-20 21:36:07 +01:00
/* move 4 KiB window to offset within the FPCI region */
base = 0xfe100000 + ( ( offset & ~ ( SZ_4K - 1 ) ) > > 8 ) ;
afi_writel ( pcie , base , AFI_FPCI_BAR0 ) ;
2013-08-09 16:49:19 +02:00
2017-12-20 21:36:07 +01:00
/* move to correct offset within the 4 KiB page */
addr = pcie - > cfg + ( offset & ( SZ_4K - 1 ) ) ;
2013-08-09 16:49:19 +02:00
}
return addr ;
}
2017-09-22 23:18:41 -07:00
static int tegra_pcie_config_read ( struct pci_bus * bus , unsigned int devfn ,
int where , int size , u32 * value )
{
if ( bus - > number = = 0 )
return pci_generic_config_read32 ( bus , devfn , where , size ,
value ) ;
return pci_generic_config_read ( bus , devfn , where , size , value ) ;
}
static int tegra_pcie_config_write ( struct pci_bus * bus , unsigned int devfn ,
int where , int size , u32 value )
{
if ( bus - > number = = 0 )
return pci_generic_config_write32 ( bus , devfn , where , size ,
value ) ;
return pci_generic_config_write ( bus , devfn , where , size , value ) ;
}
2013-08-09 16:49:19 +02:00
static struct pci_ops tegra_pcie_ops = {
2016-02-09 15:30:48 +01:00
. map_bus = tegra_pcie_map_bus ,
2017-09-22 23:18:41 -07:00
. read = tegra_pcie_config_read ,
. write = tegra_pcie_config_write ,
2013-08-09 16:49:19 +02:00
} ;
static unsigned long tegra_pcie_port_get_pex_ctrl ( struct tegra_pcie_port * port )
{
unsigned long ret = 0 ;
switch ( port - > index ) {
case 0 :
ret = AFI_PEX0_CTRL ;
break ;
case 1 :
ret = AFI_PEX1_CTRL ;
break ;
2013-08-09 16:49:24 +02:00
case 2 :
ret = AFI_PEX2_CTRL ;
break ;
2013-08-09 16:49:19 +02:00
}
return ret ;
}
static void tegra_pcie_port_reset ( struct tegra_pcie_port * port )
{
unsigned long ctrl = tegra_pcie_port_get_pex_ctrl ( port ) ;
unsigned long value ;
/* pulse reset signal */
value = afi_readl ( port - > pcie , ctrl ) ;
value & = ~ AFI_PEX_CTRL_RST ;
afi_writel ( port - > pcie , value , ctrl ) ;
usleep_range ( 1000 , 2000 ) ;
value = afi_readl ( port - > pcie , ctrl ) ;
value | = AFI_PEX_CTRL_RST ;
afi_writel ( port - > pcie , value , ctrl ) ;
}
static void tegra_pcie_port_enable ( struct tegra_pcie_port * port )
{
unsigned long ctrl = tegra_pcie_port_get_pex_ctrl ( port ) ;
2016-08-15 17:31:31 +02:00
const struct tegra_pcie_soc * soc = port - > pcie - > soc ;
2013-08-09 16:49:19 +02:00
unsigned long value ;
/* enable reference clock */
value = afi_readl ( port - > pcie , ctrl ) ;
value | = AFI_PEX_CTRL_REFCLK_EN ;
2013-08-09 16:49:24 +02:00
if ( soc - > has_pex_clkreq_en )
value | = AFI_PEX_CTRL_CLKREQ_EN ;
2014-08-26 17:11:38 +02:00
value | = AFI_PEX_CTRL_OVERRIDE_EN ;
2013-08-09 16:49:19 +02:00
afi_writel ( port - > pcie , value , ctrl ) ;
tegra_pcie_port_reset ( port ) ;
2016-11-25 11:57:14 +01:00
if ( soc - > force_pca_enable ) {
value = readl ( port - > base + RP_VEND_CTL2 ) ;
value | = RP_VEND_CTL2_PCA_ENABLE ;
writel ( value , port - > base + RP_VEND_CTL2 ) ;
}
2013-08-09 16:49:19 +02:00
}
static void tegra_pcie_port_disable ( struct tegra_pcie_port * port )
{
unsigned long ctrl = tegra_pcie_port_get_pex_ctrl ( port ) ;
2016-08-15 17:31:31 +02:00
const struct tegra_pcie_soc * soc = port - > pcie - > soc ;
2013-08-09 16:49:19 +02:00
unsigned long value ;
/* assert port reset */
value = afi_readl ( port - > pcie , ctrl ) ;
value & = ~ AFI_PEX_CTRL_RST ;
afi_writel ( port - > pcie , value , ctrl ) ;
/* disable reference clock */
value = afi_readl ( port - > pcie , ctrl ) ;
2014-08-26 17:11:35 +02:00
if ( soc - > has_pex_clkreq_en )
value & = ~ AFI_PEX_CTRL_CLKREQ_EN ;
2013-08-09 16:49:19 +02:00
value & = ~ AFI_PEX_CTRL_REFCLK_EN ;
afi_writel ( port - > pcie , value , ctrl ) ;
}
static void tegra_pcie_port_free ( struct tegra_pcie_port * port )
{
struct tegra_pcie * pcie = port - > pcie ;
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2013-08-09 16:49:19 +02:00
2016-10-06 13:43:04 -05:00
devm_iounmap ( dev , port - > base ) ;
devm_release_mem_region ( dev , port - > regs . start ,
2013-08-09 16:49:19 +02:00
resource_size ( & port - > regs ) ) ;
list_del ( & port - > list ) ;
2016-10-06 13:43:04 -05:00
devm_kfree ( dev , port ) ;
2013-08-09 16:49:19 +02:00
}
/* Tegra PCIE root complex wrongly reports device class */
static void tegra_pcie_fixup_class ( struct pci_dev * dev )
{
dev - > class = PCI_CLASS_BRIDGE_PCI < < 8 ;
}
DECLARE_PCI_FIXUP_EARLY ( PCI_VENDOR_ID_NVIDIA , 0x0bf0 , tegra_pcie_fixup_class ) ;
DECLARE_PCI_FIXUP_EARLY ( PCI_VENDOR_ID_NVIDIA , 0x0bf1 , tegra_pcie_fixup_class ) ;
2013-08-09 16:49:24 +02:00
DECLARE_PCI_FIXUP_EARLY ( PCI_VENDOR_ID_NVIDIA , 0x0e1c , tegra_pcie_fixup_class ) ;
DECLARE_PCI_FIXUP_EARLY ( PCI_VENDOR_ID_NVIDIA , 0x0e1d , tegra_pcie_fixup_class ) ;
2013-08-09 16:49:19 +02:00
/* Tegra PCIE requires relaxed ordering */
static void tegra_pcie_relax_enable ( struct pci_dev * dev )
{
pcie_capability_set_word ( dev , PCI_EXP_DEVCTL , PCI_EXP_DEVCTL_RELAX_EN ) ;
}
DECLARE_PCI_FIXUP_FINAL ( PCI_ANY_ID , PCI_ANY_ID , tegra_pcie_relax_enable ) ;
2016-11-25 11:57:12 +01:00
static int tegra_pcie_request_resources ( struct tegra_pcie * pcie )
2013-08-09 16:49:19 +02:00
{
2016-11-25 11:57:12 +01:00
struct pci_host_bridge * host = pci_host_bridge_from_priv ( pcie ) ;
struct list_head * windows = & host - > windows ;
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2014-08-01 14:15:11 +02:00
int err ;
2016-11-25 11:57:12 +01:00
pci_add_resource_offset ( windows , & pcie - > pio , pcie - > offset . io ) ;
pci_add_resource_offset ( windows , & pcie - > mem , pcie - > offset . mem ) ;
pci_add_resource_offset ( windows , & pcie - > prefetch , pcie - > offset . mem ) ;
pci_add_resource ( windows , & pcie - > busn ) ;
2016-02-09 15:52:32 +01:00
2016-11-25 11:57:12 +01:00
err = devm_request_pci_bus_resources ( dev , windows ) ;
2018-02-28 15:30:32 +05:30
if ( err < 0 ) {
pci_free_resource_list ( windows ) ;
2014-08-01 14:15:11 +02:00
return err ;
2018-02-28 15:30:32 +05:30
}
2014-08-01 14:15:11 +02:00
2016-11-25 11:57:12 +01:00
pci_remap_iospace ( & pcie - > pio , pcie - > io . start ) ;
2013-08-09 16:49:19 +02:00
2016-11-25 11:57:12 +01:00
return 0 ;
2013-08-09 16:49:19 +02:00
}
2018-02-28 15:30:32 +05:30
static void tegra_pcie_free_resources ( struct tegra_pcie * pcie )
{
struct pci_host_bridge * host = pci_host_bridge_from_priv ( pcie ) ;
struct list_head * windows = & host - > windows ;
pci_unmap_iospace ( & pcie - > pio ) ;
pci_free_resource_list ( windows ) ;
}
2013-08-09 16:49:19 +02:00
static int tegra_pcie_map_irq ( const struct pci_dev * pdev , u8 slot , u8 pin )
{
2018-01-11 11:38:03 +05:30
struct tegra_pcie * pcie = pdev - > bus - > sysdata ;
2014-04-16 10:24:32 -06:00
int irq ;
2013-08-09 16:49:19 +02:00
2013-05-06 14:19:19 -06:00
tegra_cpuidle_pcie_irqs_in_use ( ) ;
2014-04-16 10:24:32 -06:00
irq = of_irq_parse_and_map_pci ( pdev , slot , pin ) ;
if ( ! irq )
irq = pcie - > irq ;
return irq ;
2013-08-09 16:49:19 +02:00
}
static irqreturn_t tegra_pcie_isr ( int irq , void * arg )
{
const char * err_msg [ ] = {
" Unknown " ,
" AXI slave error " ,
" AXI decode error " ,
" Target abort " ,
" Master abort " ,
" Invalid write " ,
2014-08-26 17:11:38 +02:00
" Legacy interrupt " ,
2013-08-09 16:49:19 +02:00
" Response decoding error " ,
" AXI response decoding error " ,
" Transaction timeout " ,
2014-08-26 17:11:38 +02:00
" Slot present pin change " ,
" Slot clock request change " ,
" TMS clock ramp change " ,
" TMS ready for power down " ,
" Peer2Peer error " ,
2013-08-09 16:49:19 +02:00
} ;
struct tegra_pcie * pcie = arg ;
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2013-08-09 16:49:19 +02:00
u32 code , signature ;
code = afi_readl ( pcie , AFI_INTR_CODE ) & AFI_INTR_CODE_MASK ;
signature = afi_readl ( pcie , AFI_INTR_SIGNATURE ) ;
afi_writel ( pcie , 0 , AFI_INTR_CODE ) ;
if ( code = = AFI_INTR_LEGACY )
return IRQ_NONE ;
if ( code > = ARRAY_SIZE ( err_msg ) )
code = 0 ;
/*
* do not pollute kernel log with master abort reports since they
* happen a lot during enumeration
*/
if ( code = = AFI_INTR_MASTER_ABORT )
2016-10-06 13:43:04 -05:00
dev_dbg ( dev , " %s, signature: %08x \n " , err_msg [ code ] , signature ) ;
2013-08-09 16:49:19 +02:00
else
2016-10-06 13:43:04 -05:00
dev_err ( dev , " %s, signature: %08x \n " , err_msg [ code ] , signature ) ;
2013-08-09 16:49:19 +02:00
if ( code = = AFI_INTR_TARGET_ABORT | | code = = AFI_INTR_MASTER_ABORT | |
code = = AFI_INTR_FPCI_DECODE_ERROR ) {
u32 fpci = afi_readl ( pcie , AFI_UPPER_FPCI_ADDRESS ) & 0xff ;
u64 address = ( u64 ) fpci < < 32 | ( signature & 0xfffffffc ) ;
if ( code = = AFI_INTR_MASTER_ABORT )
2016-10-06 13:43:04 -05:00
dev_dbg ( dev , " FPCI address: %10llx \n " , address ) ;
2013-08-09 16:49:19 +02:00
else
2016-10-06 13:43:04 -05:00
dev_err ( dev , " FPCI address: %10llx \n " , address ) ;
2013-08-09 16:49:19 +02:00
}
return IRQ_HANDLED ;
}
/*
* FPCI map is as follows :
* - 0xfdfc000000 : I / O space
* - 0xfdfe000000 : type 0 configuration space
* - 0xfdff000000 : type 1 configuration space
* - 0xfe00000000 : type 0 extended configuration space
* - 0xfe10000000 : type 1 extended configuration space
*/
static void tegra_pcie_setup_translations ( struct tegra_pcie * pcie )
{
u32 fpci_bar , size , axi_address ;
/* Bar 0: type 1 extended configuration space */
2017-12-20 21:36:07 +01:00
size = resource_size ( & pcie - > cs ) ;
afi_writel ( pcie , pcie - > cs . start , AFI_AXI_BAR0_START ) ;
2013-08-09 16:49:19 +02:00
afi_writel ( pcie , size > > 12 , AFI_AXI_BAR0_SZ ) ;
/* Bar 1: downstream IO bar */
fpci_bar = 0xfdfc0000 ;
size = resource_size ( & pcie - > io ) ;
2014-11-27 09:54:09 +01:00
axi_address = pcie - > io . start ;
2013-08-09 16:49:19 +02:00
afi_writel ( pcie , axi_address , AFI_AXI_BAR1_START ) ;
afi_writel ( pcie , size > > 12 , AFI_AXI_BAR1_SZ ) ;
afi_writel ( pcie , fpci_bar , AFI_FPCI_BAR1 ) ;
/* Bar 2: prefetchable memory BAR */
fpci_bar = ( ( ( pcie - > prefetch . start > > 12 ) & 0x0fffffff ) < < 4 ) | 0x1 ;
size = resource_size ( & pcie - > prefetch ) ;
axi_address = pcie - > prefetch . start ;
afi_writel ( pcie , axi_address , AFI_AXI_BAR2_START ) ;
afi_writel ( pcie , size > > 12 , AFI_AXI_BAR2_SZ ) ;
afi_writel ( pcie , fpci_bar , AFI_FPCI_BAR2 ) ;
/* Bar 3: non prefetchable memory BAR */
fpci_bar = ( ( ( pcie - > mem . start > > 12 ) & 0x0fffffff ) < < 4 ) | 0x1 ;
size = resource_size ( & pcie - > mem ) ;
axi_address = pcie - > mem . start ;
afi_writel ( pcie , axi_address , AFI_AXI_BAR3_START ) ;
afi_writel ( pcie , size > > 12 , AFI_AXI_BAR3_SZ ) ;
afi_writel ( pcie , fpci_bar , AFI_FPCI_BAR3 ) ;
/* NULL out the remaining BARs as they are not used */
afi_writel ( pcie , 0 , AFI_AXI_BAR4_START ) ;
afi_writel ( pcie , 0 , AFI_AXI_BAR4_SZ ) ;
afi_writel ( pcie , 0 , AFI_FPCI_BAR4 ) ;
afi_writel ( pcie , 0 , AFI_AXI_BAR5_START ) ;
afi_writel ( pcie , 0 , AFI_AXI_BAR5_SZ ) ;
afi_writel ( pcie , 0 , AFI_FPCI_BAR5 ) ;
/* map all upstream transactions as uncached */
2016-02-09 15:52:33 +01:00
afi_writel ( pcie , 0 , AFI_CACHE_BAR0_ST ) ;
2013-08-09 16:49:19 +02:00
afi_writel ( pcie , 0 , AFI_CACHE_BAR0_SZ ) ;
afi_writel ( pcie , 0 , AFI_CACHE_BAR1_ST ) ;
afi_writel ( pcie , 0 , AFI_CACHE_BAR1_SZ ) ;
/* MSI translations are setup only when needed */
afi_writel ( pcie , 0 , AFI_MSI_FPCI_BAR_ST ) ;
afi_writel ( pcie , 0 , AFI_MSI_BAR_SZ ) ;
afi_writel ( pcie , 0 , AFI_MSI_AXI_BAR_ST ) ;
afi_writel ( pcie , 0 , AFI_MSI_BAR_SZ ) ;
}
2014-08-26 17:11:38 +02:00
static int tegra_pcie_pll_wait ( struct tegra_pcie * pcie , unsigned long timeout )
2013-08-09 16:49:19 +02:00
{
2016-08-15 17:31:31 +02:00
const struct tegra_pcie_soc * soc = pcie - > soc ;
2014-08-26 17:11:38 +02:00
u32 value ;
2013-08-09 16:49:19 +02:00
2014-08-26 17:11:38 +02:00
timeout = jiffies + msecs_to_jiffies ( timeout ) ;
2013-08-09 16:49:24 +02:00
2014-08-26 17:11:38 +02:00
while ( time_before ( jiffies , timeout ) ) {
value = pads_readl ( pcie , soc - > pads_pll_ctl ) ;
if ( value & PADS_PLL_CTL_LOCKDET )
return 0 ;
}
2013-08-09 16:49:19 +02:00
2014-08-26 17:11:38 +02:00
return - ETIMEDOUT ;
}
2013-08-09 16:49:19 +02:00
2014-08-26 17:11:38 +02:00
static int tegra_pcie_phy_enable ( struct tegra_pcie * pcie )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2016-08-15 17:31:31 +02:00
const struct tegra_pcie_soc * soc = pcie - > soc ;
2014-08-26 17:11:38 +02:00
u32 value ;
int err ;
2013-08-09 16:49:19 +02:00
2013-11-14 11:28:18 -07:00
/* initialize internal PHY, enable up to 16 PCIE lanes */
2013-08-09 16:49:19 +02:00
pads_writel ( pcie , 0x0 , PADS_CTL_SEL ) ;
/* override IDDQ to 1 on all 4 lanes */
value = pads_readl ( pcie , PADS_CTL ) ;
value | = PADS_CTL_IDDQ_1L ;
pads_writel ( pcie , value , PADS_CTL ) ;
/*
* Set up PHY PLL inputs select PLLE output as refclock ,
* set TX ref sel to div10 ( not div5 ) .
*/
2013-08-09 16:49:24 +02:00
value = pads_readl ( pcie , soc - > pads_pll_ctl ) ;
2013-08-09 16:49:19 +02:00
value & = ~ ( PADS_PLL_CTL_REFCLK_MASK | PADS_PLL_CTL_TXCLKREF_MASK ) ;
2013-08-09 16:49:24 +02:00
value | = PADS_PLL_CTL_REFCLK_INTERNAL_CML | soc - > tx_ref_sel ;
pads_writel ( pcie , value , soc - > pads_pll_ctl ) ;
2013-08-09 16:49:19 +02:00
2014-08-26 17:11:37 +02:00
/* reset PLL */
value = pads_readl ( pcie , soc - > pads_pll_ctl ) ;
value & = ~ PADS_PLL_CTL_RST_B4SM ;
pads_writel ( pcie , value , soc - > pads_pll_ctl ) ;
usleep_range ( 20 , 100 ) ;
2013-08-09 16:49:19 +02:00
/* take PLL out of reset */
2013-08-09 16:49:24 +02:00
value = pads_readl ( pcie , soc - > pads_pll_ctl ) ;
2013-08-09 16:49:19 +02:00
value | = PADS_PLL_CTL_RST_B4SM ;
2013-08-09 16:49:24 +02:00
pads_writel ( pcie , value , soc - > pads_pll_ctl ) ;
2013-08-09 16:49:19 +02:00
/* wait for the PLL to lock */
2014-08-26 17:11:38 +02:00
err = tegra_pcie_pll_wait ( pcie , 500 ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " PLL failed to lock: %d \n " , err ) ;
2014-08-26 17:11:38 +02:00
return err ;
}
2013-08-09 16:49:19 +02:00
/* turn off IDDQ override */
value = pads_readl ( pcie , PADS_CTL ) ;
value & = ~ PADS_CTL_IDDQ_1L ;
pads_writel ( pcie , value , PADS_CTL ) ;
/* enable TX/RX data */
value = pads_readl ( pcie , PADS_CTL ) ;
value | = PADS_CTL_TX_DATA_EN_1L | PADS_CTL_RX_DATA_EN_1L ;
pads_writel ( pcie , value , PADS_CTL ) ;
2014-08-26 17:11:38 +02:00
return 0 ;
}
2015-11-11 18:25:59 +01:00
static int tegra_pcie_phy_disable ( struct tegra_pcie * pcie )
{
2016-08-15 17:31:31 +02:00
const struct tegra_pcie_soc * soc = pcie - > soc ;
2015-11-11 18:25:59 +01:00
u32 value ;
/* disable TX/RX data */
value = pads_readl ( pcie , PADS_CTL ) ;
value & = ~ ( PADS_CTL_TX_DATA_EN_1L | PADS_CTL_RX_DATA_EN_1L ) ;
pads_writel ( pcie , value , PADS_CTL ) ;
/* override IDDQ */
value = pads_readl ( pcie , PADS_CTL ) ;
value | = PADS_CTL_IDDQ_1L ;
2016-10-05 16:04:13 -05:00
pads_writel ( pcie , value , PADS_CTL ) ;
2015-11-11 18:25:59 +01:00
/* reset PLL */
value = pads_readl ( pcie , soc - > pads_pll_ctl ) ;
value & = ~ PADS_PLL_CTL_RST_B4SM ;
pads_writel ( pcie , value , soc - > pads_pll_ctl ) ;
usleep_range ( 20 , 100 ) ;
return 0 ;
}
static int tegra_pcie_port_phy_power_on ( struct tegra_pcie_port * port )
{
struct device * dev = port - > pcie - > dev ;
unsigned int i ;
int err ;
for ( i = 0 ; i < port - > lanes ; i + + ) {
err = phy_power_on ( port - > phys [ i ] ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to power on PHY#%u: %d \n " , i , err ) ;
2015-11-11 18:25:59 +01:00
return err ;
}
}
return 0 ;
}
static int tegra_pcie_port_phy_power_off ( struct tegra_pcie_port * port )
{
struct device * dev = port - > pcie - > dev ;
unsigned int i ;
int err ;
for ( i = 0 ; i < port - > lanes ; i + + ) {
err = phy_power_off ( port - > phys [ i ] ) ;
if ( err < 0 ) {
dev_err ( dev , " failed to power off PHY#%u: %d \n " , i ,
err ) ;
return err ;
}
}
return 0 ;
}
static int tegra_pcie_phy_power_on ( struct tegra_pcie * pcie )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2016-08-15 17:31:31 +02:00
const struct tegra_pcie_soc * soc = pcie - > soc ;
2015-11-11 18:25:59 +01:00
struct tegra_pcie_port * port ;
int err ;
if ( pcie - > legacy_phy ) {
if ( pcie - > phy )
err = phy_power_on ( pcie - > phy ) ;
else
err = tegra_pcie_phy_enable ( pcie ) ;
if ( err < 0 )
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to power on PHY: %d \n " , err ) ;
2015-11-11 18:25:59 +01:00
return err ;
}
list_for_each_entry ( port , & pcie - > ports , list ) {
err = tegra_pcie_port_phy_power_on ( port ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev ,
2015-11-11 18:25:59 +01:00
" failed to power on PCIe port %u PHY: %d \n " ,
port - > index , err ) ;
return err ;
}
}
2016-07-25 16:02:21 -05:00
/* Configure the reference clock driver */
2016-07-25 16:02:27 -05:00
pads_writel ( pcie , soc - > pads_refclk_cfg0 , PADS_REFCLK_CFG0 ) ;
2016-07-25 16:02:21 -05:00
if ( soc - > num_ports > 2 )
2016-07-25 16:02:27 -05:00
pads_writel ( pcie , soc - > pads_refclk_cfg1 , PADS_REFCLK_CFG1 ) ;
2016-07-25 16:02:21 -05:00
2015-11-11 18:25:59 +01:00
return 0 ;
}
static int tegra_pcie_phy_power_off ( struct tegra_pcie * pcie )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2015-11-11 18:25:59 +01:00
struct tegra_pcie_port * port ;
int err ;
if ( pcie - > legacy_phy ) {
if ( pcie - > phy )
err = phy_power_off ( pcie - > phy ) ;
else
err = tegra_pcie_phy_disable ( pcie ) ;
if ( err < 0 )
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to power off PHY: %d \n " , err ) ;
2015-11-11 18:25:59 +01:00
return err ;
}
list_for_each_entry ( port , & pcie - > ports , list ) {
err = tegra_pcie_port_phy_power_off ( port ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev ,
2015-11-11 18:25:59 +01:00
" failed to power off PCIe port %u PHY: %d \n " ,
port - > index , err ) ;
return err ;
}
}
return 0 ;
}
2014-08-26 17:11:38 +02:00
static int tegra_pcie_enable_controller ( struct tegra_pcie * pcie )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2016-08-15 17:31:31 +02:00
const struct tegra_pcie_soc * soc = pcie - > soc ;
2014-08-26 17:11:38 +02:00
struct tegra_pcie_port * port ;
unsigned long value ;
int err ;
/* enable PLL power down */
if ( pcie - > phy ) {
value = afi_readl ( pcie , AFI_PLLE_CONTROL ) ;
value & = ~ AFI_PLLE_CONTROL_BYPASS_PADS2PLLE_CONTROL ;
value | = AFI_PLLE_CONTROL_PADS2PLLE_CONTROL_EN ;
afi_writel ( pcie , value , AFI_PLLE_CONTROL ) ;
}
/* power down PCIe slot clock bias pad */
if ( soc - > has_pex_bias_ctrl )
afi_writel ( pcie , 0 , AFI_PEXBIAS_CTRL_0 ) ;
/* configure mode and disable all ports */
value = afi_readl ( pcie , AFI_PCIE_CONFIG ) ;
value & = ~ AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_MASK ;
value | = AFI_PCIE_CONFIG_PCIE_DISABLE_ALL | pcie - > xbar_config ;
list_for_each_entry ( port , & pcie - > ports , list )
value & = ~ AFI_PCIE_CONFIG_PCIE_DISABLE ( port - > index ) ;
afi_writel ( pcie , value , AFI_PCIE_CONFIG ) ;
if ( soc - > has_gen2 ) {
value = afi_readl ( pcie , AFI_FUSE ) ;
value & = ~ AFI_FUSE_PCIE_T0_GEN2_DIS ;
afi_writel ( pcie , value , AFI_FUSE ) ;
} else {
value = afi_readl ( pcie , AFI_FUSE ) ;
value | = AFI_FUSE_PCIE_T0_GEN2_DIS ;
afi_writel ( pcie , value , AFI_FUSE ) ;
}
2017-09-27 17:28:35 +05:30
if ( soc - > program_uphy ) {
err = tegra_pcie_phy_power_on ( pcie ) ;
if ( err < 0 ) {
dev_err ( dev , " failed to power on PHY(s): %d \n " , err ) ;
return err ;
}
2014-08-26 17:11:38 +02:00
}
2013-08-09 16:49:19 +02:00
/* take the PCIe interface module out of reset */
2013-11-06 15:56:58 -07:00
reset_control_deassert ( pcie - > pcie_xrst ) ;
2013-08-09 16:49:19 +02:00
/* finally enable PCIe */
value = afi_readl ( pcie , AFI_CONFIGURATION ) ;
value | = AFI_CONFIGURATION_EN_FPCI ;
afi_writel ( pcie , value , AFI_CONFIGURATION ) ;
value = AFI_INTR_EN_INI_SLVERR | AFI_INTR_EN_INI_DECERR |
AFI_INTR_EN_TGT_SLVERR | AFI_INTR_EN_TGT_DECERR |
AFI_INTR_EN_TGT_WRERR | AFI_INTR_EN_DFPCI_DECERR ;
2013-08-09 16:49:24 +02:00
if ( soc - > has_intr_prsnt_sense )
value | = AFI_INTR_EN_PRSNT_SENSE ;
2013-08-09 16:49:19 +02:00
afi_writel ( pcie , value , AFI_AFI_INTR_ENABLE ) ;
afi_writel ( pcie , 0xffffffff , AFI_SM_INTR_ENABLE ) ;
/* don't enable MSI for now, only when needed */
afi_writel ( pcie , AFI_INTR_MASK_INT_MASK , AFI_INTR_MASK ) ;
/* disable all exceptions */
afi_writel ( pcie , 0 , AFI_FPCI_ERROR_MASKS ) ;
return 0 ;
}
2018-02-28 15:30:32 +05:30
static void tegra_pcie_disable_controller ( struct tegra_pcie * pcie )
2013-08-09 16:49:19 +02:00
{
int err ;
2018-02-28 15:30:32 +05:30
reset_control_assert ( pcie - > pcie_xrst ) ;
2013-08-09 16:49:19 +02:00
2018-02-28 15:30:32 +05:30
if ( pcie - > soc - > program_uphy ) {
2017-09-27 17:28:35 +05:30
err = tegra_pcie_phy_power_off ( pcie ) ;
if ( err < 0 )
2018-02-28 15:30:32 +05:30
dev_err ( pcie - > dev , " failed to power off PHY(s): %d \n " ,
err ) ;
2017-09-27 17:28:35 +05:30
}
2018-02-28 15:30:32 +05:30
}
static void tegra_pcie_power_off ( struct tegra_pcie * pcie )
{
struct device * dev = pcie - > dev ;
const struct tegra_pcie_soc * soc = pcie - > soc ;
int err ;
2014-08-26 17:11:38 +02:00
2013-11-06 15:56:58 -07:00
reset_control_assert ( pcie - > afi_rst ) ;
reset_control_assert ( pcie - > pex_rst ) ;
2013-08-09 16:49:19 +02:00
2018-02-28 15:30:32 +05:30
clk_disable_unprepare ( pcie - > pll_e ) ;
if ( soc - > has_cml_clk )
clk_disable_unprepare ( pcie - > cml_clk ) ;
clk_disable_unprepare ( pcie - > afi_clk ) ;
clk_disable_unprepare ( pcie - > pex_clk ) ;
2017-09-27 17:28:35 +05:30
if ( ! dev - > pm_domain )
tegra_powergate_power_off ( TEGRA_POWERGATE_PCIE ) ;
2013-08-09 16:49:19 +02:00
2014-05-28 16:49:13 +02:00
err = regulator_bulk_disable ( pcie - > num_supplies , pcie - > supplies ) ;
2013-08-09 16:49:19 +02:00
if ( err < 0 )
2016-10-06 13:43:04 -05:00
dev_warn ( dev , " failed to disable regulators: %d \n " , err ) ;
2013-08-09 16:49:19 +02:00
}
static int tegra_pcie_power_on ( struct tegra_pcie * pcie )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2016-08-15 17:31:31 +02:00
const struct tegra_pcie_soc * soc = pcie - > soc ;
2013-08-09 16:49:19 +02:00
int err ;
2013-11-06 15:56:58 -07:00
reset_control_assert ( pcie - > pcie_xrst ) ;
reset_control_assert ( pcie - > afi_rst ) ;
reset_control_assert ( pcie - > pex_rst ) ;
2013-08-09 16:49:19 +02:00
2017-09-27 17:28:35 +05:30
if ( ! dev - > pm_domain )
tegra_powergate_power_off ( TEGRA_POWERGATE_PCIE ) ;
2013-08-09 16:49:19 +02:00
/* enable regulators */
2014-05-28 16:49:13 +02:00
err = regulator_bulk_enable ( pcie - > num_supplies , pcie - > supplies ) ;
if ( err < 0 )
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to enable regulators: %d \n " , err ) ;
2013-08-09 16:49:24 +02:00
2017-09-27 17:28:35 +05:30
if ( dev - > pm_domain ) {
err = clk_prepare_enable ( pcie - > pex_clk ) ;
if ( err ) {
dev_err ( dev , " failed to enable PEX clock: %d \n " , err ) ;
return err ;
}
reset_control_deassert ( pcie - > pex_rst ) ;
} else {
err = tegra_powergate_sequence_power_up ( TEGRA_POWERGATE_PCIE ,
pcie - > pex_clk ,
pcie - > pex_rst ) ;
if ( err ) {
dev_err ( dev , " powerup sequence failed: %d \n " , err ) ;
return err ;
}
2013-08-09 16:49:19 +02:00
}
2013-11-06 15:56:58 -07:00
reset_control_deassert ( pcie - > afi_rst ) ;
2013-08-09 16:49:19 +02:00
err = clk_prepare_enable ( pcie - > afi_clk ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to enable AFI clock: %d \n " , err ) ;
2013-08-09 16:49:19 +02:00
return err ;
}
2013-08-09 16:49:24 +02:00
if ( soc - > has_cml_clk ) {
err = clk_prepare_enable ( pcie - > cml_clk ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to enable CML clock: %d \n " , err ) ;
2013-08-09 16:49:24 +02:00
return err ;
}
}
2013-08-09 16:49:19 +02:00
err = clk_prepare_enable ( pcie - > pll_e ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to enable PLLE clock: %d \n " , err ) ;
2013-08-09 16:49:19 +02:00
return err ;
}
return 0 ;
}
static int tegra_pcie_clocks_get ( struct tegra_pcie * pcie )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2016-08-15 17:31:31 +02:00
const struct tegra_pcie_soc * soc = pcie - > soc ;
2013-08-09 16:49:24 +02:00
2016-10-06 13:43:04 -05:00
pcie - > pex_clk = devm_clk_get ( dev , " pex " ) ;
2013-08-09 16:49:19 +02:00
if ( IS_ERR ( pcie - > pex_clk ) )
return PTR_ERR ( pcie - > pex_clk ) ;
2016-10-06 13:43:04 -05:00
pcie - > afi_clk = devm_clk_get ( dev , " afi " ) ;
2013-08-09 16:49:19 +02:00
if ( IS_ERR ( pcie - > afi_clk ) )
return PTR_ERR ( pcie - > afi_clk ) ;
2016-10-06 13:43:04 -05:00
pcie - > pll_e = devm_clk_get ( dev , " pll_e " ) ;
2013-08-09 16:49:19 +02:00
if ( IS_ERR ( pcie - > pll_e ) )
return PTR_ERR ( pcie - > pll_e ) ;
2013-08-09 16:49:24 +02:00
if ( soc - > has_cml_clk ) {
2016-10-06 13:43:04 -05:00
pcie - > cml_clk = devm_clk_get ( dev , " cml " ) ;
2013-08-09 16:49:24 +02:00
if ( IS_ERR ( pcie - > cml_clk ) )
return PTR_ERR ( pcie - > cml_clk ) ;
}
2013-08-09 16:49:19 +02:00
return 0 ;
}
2013-11-06 15:56:58 -07:00
static int tegra_pcie_resets_get ( struct tegra_pcie * pcie )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2017-07-19 17:25:57 +02:00
pcie - > pex_rst = devm_reset_control_get_exclusive ( dev , " pex " ) ;
2013-11-06 15:56:58 -07:00
if ( IS_ERR ( pcie - > pex_rst ) )
return PTR_ERR ( pcie - > pex_rst ) ;
2017-07-19 17:25:57 +02:00
pcie - > afi_rst = devm_reset_control_get_exclusive ( dev , " afi " ) ;
2013-11-06 15:56:58 -07:00
if ( IS_ERR ( pcie - > afi_rst ) )
return PTR_ERR ( pcie - > afi_rst ) ;
2017-07-19 17:25:57 +02:00
pcie - > pcie_xrst = devm_reset_control_get_exclusive ( dev , " pcie_x " ) ;
2013-11-06 15:56:58 -07:00
if ( IS_ERR ( pcie - > pcie_xrst ) )
return PTR_ERR ( pcie - > pcie_xrst ) ;
return 0 ;
}
2015-11-11 18:25:59 +01:00
static int tegra_pcie_phys_get_legacy ( struct tegra_pcie * pcie )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2015-11-11 18:25:59 +01:00
int err ;
2016-10-06 13:43:04 -05:00
pcie - > phy = devm_phy_optional_get ( dev , " pcie " ) ;
2015-11-11 18:25:59 +01:00
if ( IS_ERR ( pcie - > phy ) ) {
err = PTR_ERR ( pcie - > phy ) ;
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to get PHY: %d \n " , err ) ;
2015-11-11 18:25:59 +01:00
return err ;
}
err = phy_init ( pcie - > phy ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to initialize PHY: %d \n " , err ) ;
2015-11-11 18:25:59 +01:00
return err ;
}
pcie - > legacy_phy = true ;
return 0 ;
}
static struct phy * devm_of_phy_optional_get_index ( struct device * dev ,
struct device_node * np ,
const char * consumer ,
unsigned int index )
{
struct phy * phy ;
char * name ;
name = kasprintf ( GFP_KERNEL , " %s-%u " , consumer , index ) ;
if ( ! name )
return ERR_PTR ( - ENOMEM ) ;
phy = devm_of_phy_get ( dev , np , name ) ;
kfree ( name ) ;
if ( IS_ERR ( phy ) & & PTR_ERR ( phy ) = = - ENODEV )
phy = NULL ;
return phy ;
}
static int tegra_pcie_port_get_phys ( struct tegra_pcie_port * port )
{
struct device * dev = port - > pcie - > dev ;
struct phy * phy ;
unsigned int i ;
int err ;
port - > phys = devm_kcalloc ( dev , sizeof ( phy ) , port - > lanes , GFP_KERNEL ) ;
if ( ! port - > phys )
return - ENOMEM ;
for ( i = 0 ; i < port - > lanes ; i + + ) {
phy = devm_of_phy_optional_get_index ( dev , port - > np , " pcie " , i ) ;
if ( IS_ERR ( phy ) ) {
dev_err ( dev , " failed to get PHY#%u: %ld \n " , i ,
PTR_ERR ( phy ) ) ;
return PTR_ERR ( phy ) ;
}
err = phy_init ( phy ) ;
if ( err < 0 ) {
dev_err ( dev , " failed to initialize PHY#%u: %d \n " , i ,
err ) ;
return err ;
}
port - > phys [ i ] = phy ;
}
return 0 ;
}
static int tegra_pcie_phys_get ( struct tegra_pcie * pcie )
{
2016-08-15 17:31:31 +02:00
const struct tegra_pcie_soc * soc = pcie - > soc ;
2015-11-11 18:25:59 +01:00
struct device_node * np = pcie - > dev - > of_node ;
struct tegra_pcie_port * port ;
int err ;
if ( ! soc - > has_gen2 | | of_find_property ( np , " phys " , NULL ) ! = NULL )
return tegra_pcie_phys_get_legacy ( pcie ) ;
list_for_each_entry ( port , & pcie - > ports , list ) {
err = tegra_pcie_port_get_phys ( port ) ;
if ( err < 0 )
return err ;
}
return 0 ;
}
2018-02-28 15:30:32 +05:30
static void tegra_pcie_phys_put ( struct tegra_pcie * pcie )
{
struct tegra_pcie_port * port ;
struct device * dev = pcie - > dev ;
int err , i ;
if ( pcie - > legacy_phy ) {
err = phy_exit ( pcie - > phy ) ;
if ( err < 0 )
dev_err ( dev , " failed to teardown PHY: %d \n " , err ) ;
return ;
}
list_for_each_entry ( port , & pcie - > ports , list ) {
for ( i = 0 ; i < port - > lanes ; i + + ) {
err = phy_exit ( port - > phys [ i ] ) ;
if ( err < 0 )
dev_err ( dev , " failed to teardown PHY#%u: %d \n " ,
i , err ) ;
}
}
}
2013-08-09 16:49:19 +02:00
static int tegra_pcie_get_resources ( struct tegra_pcie * pcie )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
struct platform_device * pdev = to_platform_device ( dev ) ;
2013-08-09 16:49:19 +02:00
struct resource * pads , * afi , * res ;
2017-09-27 17:28:35 +05:30
const struct tegra_pcie_soc * soc = pcie - > soc ;
2013-08-09 16:49:19 +02:00
int err ;
err = tegra_pcie_clocks_get ( pcie ) ;
if ( err ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to get clocks: %d \n " , err ) ;
2013-08-09 16:49:19 +02:00
return err ;
}
2013-11-06 15:56:58 -07:00
err = tegra_pcie_resets_get ( pcie ) ;
if ( err ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to get resets: %d \n " , err ) ;
2013-11-06 15:56:58 -07:00
return err ;
}
2017-09-27 17:28:35 +05:30
if ( soc - > program_uphy ) {
err = tegra_pcie_phys_get ( pcie ) ;
if ( err < 0 ) {
dev_err ( dev , " failed to get PHYs: %d \n " , err ) ;
return err ;
}
2014-08-26 17:11:38 +02:00
}
2013-08-09 16:49:19 +02:00
pads = platform_get_resource_byname ( pdev , IORESOURCE_MEM , " pads " ) ;
2016-10-06 13:43:04 -05:00
pcie - > pads = devm_ioremap_resource ( dev , pads ) ;
2013-08-26 11:11:09 +02:00
if ( IS_ERR ( pcie - > pads ) ) {
err = PTR_ERR ( pcie - > pads ) ;
2018-02-28 15:30:34 +05:30
goto phys_put ;
2013-08-09 16:49:19 +02:00
}
afi = platform_get_resource_byname ( pdev , IORESOURCE_MEM , " afi " ) ;
2016-10-06 13:43:04 -05:00
pcie - > afi = devm_ioremap_resource ( dev , afi ) ;
2013-08-26 11:11:09 +02:00
if ( IS_ERR ( pcie - > afi ) ) {
err = PTR_ERR ( pcie - > afi ) ;
2018-02-28 15:30:34 +05:30
goto phys_put ;
2013-08-09 16:49:19 +02:00
}
2013-08-26 11:11:09 +02:00
/* request configuration space, but remap later, on demand */
2013-08-09 16:49:19 +02:00
res = platform_get_resource_byname ( pdev , IORESOURCE_MEM , " cs " ) ;
if ( ! res ) {
err = - EADDRNOTAVAIL ;
2018-02-28 15:30:34 +05:30
goto phys_put ;
2013-08-09 16:49:19 +02:00
}
2017-12-20 21:36:07 +01:00
pcie - > cs = * res ;
/* constrain configuration space to 4 KiB */
pcie - > cs . end = pcie - > cs . start + SZ_4K - 1 ;
pcie - > cfg = devm_ioremap_resource ( dev , & pcie - > cs ) ;
if ( IS_ERR ( pcie - > cfg ) ) {
err = PTR_ERR ( pcie - > cfg ) ;
2018-02-28 15:30:34 +05:30
goto phys_put ;
2013-08-09 16:49:19 +02:00
}
/* request interrupt */
err = platform_get_irq_byname ( pdev , " intr " ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to get IRQ: %d \n " , err ) ;
2018-02-28 15:30:34 +05:30
goto phys_put ;
2013-08-09 16:49:19 +02:00
}
pcie - > irq = err ;
err = request_irq ( pcie - > irq , tegra_pcie_isr , IRQF_SHARED , " PCIE " , pcie ) ;
if ( err ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to register IRQ: %d \n " , err ) ;
2018-02-28 15:30:34 +05:30
goto phys_put ;
2013-08-09 16:49:19 +02:00
}
return 0 ;
2018-02-28 15:30:32 +05:30
phys_put :
if ( soc - > program_uphy )
tegra_pcie_phys_put ( pcie ) ;
2013-08-09 16:49:19 +02:00
return err ;
}
static int tegra_pcie_put_resources ( struct tegra_pcie * pcie )
{
2017-09-27 17:28:35 +05:30
const struct tegra_pcie_soc * soc = pcie - > soc ;
2014-08-26 17:11:38 +02:00
2013-08-09 16:49:19 +02:00
if ( pcie - > irq > 0 )
free_irq ( pcie - > irq , pcie ) ;
2018-02-28 15:30:32 +05:30
if ( soc - > program_uphy )
tegra_pcie_phys_put ( pcie ) ;
2014-08-26 17:11:38 +02:00
2013-08-09 16:49:19 +02:00
return 0 ;
}
2018-02-28 15:30:33 +05:30
static void tegra_pcie_pme_turnoff ( struct tegra_pcie_port * port )
{
struct tegra_pcie * pcie = port - > pcie ;
const struct tegra_pcie_soc * soc = pcie - > soc ;
int err ;
u32 val ;
u8 ack_bit ;
val = afi_readl ( pcie , AFI_PCIE_PME ) ;
val | = ( 0x1 < < soc - > ports [ port - > index ] . pme . turnoff_bit ) ;
afi_writel ( pcie , val , AFI_PCIE_PME ) ;
ack_bit = soc - > ports [ port - > index ] . pme . ack_bit ;
err = readl_poll_timeout ( pcie - > afi + AFI_PCIE_PME , val ,
val & ( 0x1 < < ack_bit ) , 1 , PME_ACK_TIMEOUT ) ;
if ( err )
dev_err ( pcie - > dev , " PME Ack is not received on port: %d \n " ,
port - > index ) ;
usleep_range ( 10000 , 11000 ) ;
val = afi_readl ( pcie , AFI_PCIE_PME ) ;
val & = ~ ( 0x1 < < soc - > ports [ port - > index ] . pme . turnoff_bit ) ;
afi_writel ( pcie , val , AFI_PCIE_PME ) ;
}
2013-08-09 16:49:19 +02:00
static int tegra_msi_alloc ( struct tegra_msi * chip )
{
int msi ;
mutex_lock ( & chip - > lock ) ;
msi = find_first_zero_bit ( chip - > used , INT_PCI_MSI_NR ) ;
if ( msi < INT_PCI_MSI_NR )
set_bit ( msi , chip - > used ) ;
else
msi = - ENOSPC ;
mutex_unlock ( & chip - > lock ) ;
return msi ;
}
static void tegra_msi_free ( struct tegra_msi * chip , unsigned long irq )
{
struct device * dev = chip - > chip . dev ;
mutex_lock ( & chip - > lock ) ;
if ( ! test_bit ( irq , chip - > used ) )
dev_err ( dev , " trying to free unused MSI#%lu \n " , irq ) ;
else
clear_bit ( irq , chip - > used ) ;
mutex_unlock ( & chip - > lock ) ;
}
static irqreturn_t tegra_pcie_msi_irq ( int irq , void * data )
{
struct tegra_pcie * pcie = data ;
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2013-08-09 16:49:19 +02:00
struct tegra_msi * msi = & pcie - > msi ;
unsigned int i , processed = 0 ;
for ( i = 0 ; i < 8 ; i + + ) {
unsigned long reg = afi_readl ( pcie , AFI_MSI_VEC0 + i * 4 ) ;
while ( reg ) {
unsigned int offset = find_first_bit ( & reg , 32 ) ;
unsigned int index = i * 32 + offset ;
unsigned int irq ;
/* clear the interrupt */
afi_writel ( pcie , 1 < < offset , AFI_MSI_VEC0 + i * 4 ) ;
irq = irq_find_mapping ( msi - > domain , index ) ;
if ( irq ) {
if ( test_bit ( index , msi - > used ) )
generic_handle_irq ( irq ) ;
else
2016-10-06 13:43:04 -05:00
dev_info ( dev , " unhandled MSI \n " ) ;
2013-08-09 16:49:19 +02:00
} else {
/*
* that ' s weird who triggered this ?
* just clear it
*/
2016-10-06 13:43:04 -05:00
dev_info ( dev , " unexpected MSI \n " ) ;
2013-08-09 16:49:19 +02:00
}
/* see if there's any more pending in this vector */
reg = afi_readl ( pcie , AFI_MSI_VEC0 + i * 4 ) ;
processed + + ;
}
}
return processed > 0 ? IRQ_HANDLED : IRQ_NONE ;
}
2014-11-11 17:45:45 -07:00
static int tegra_msi_setup_irq ( struct msi_controller * chip ,
struct pci_dev * pdev , struct msi_desc * desc )
2013-08-09 16:49:19 +02:00
{
struct tegra_msi * msi = to_tegra_msi ( chip ) ;
struct msi_msg msg ;
unsigned int irq ;
int hwirq ;
hwirq = tegra_msi_alloc ( msi ) ;
if ( hwirq < 0 )
return hwirq ;
irq = irq_create_mapping ( msi - > domain , hwirq ) ;
2014-07-29 09:33:30 +08:00
if ( ! irq ) {
tegra_msi_free ( msi , hwirq ) ;
2013-08-09 16:49:19 +02:00
return - EINVAL ;
2014-07-29 09:33:30 +08:00
}
2013-08-09 16:49:19 +02:00
irq_set_msi_desc ( irq , desc ) ;
2017-05-04 22:10:31 +02:00
msg . address_lo = lower_32_bits ( msi - > phys ) ;
msg . address_hi = upper_32_bits ( msi - > phys ) ;
2013-08-09 16:49:19 +02:00
msg . data = hwirq ;
2014-11-09 23:10:34 +08:00
pci_write_msi_msg ( irq , & msg ) ;
2013-08-09 16:49:19 +02:00
return 0 ;
}
2014-11-11 17:45:45 -07:00
static void tegra_msi_teardown_irq ( struct msi_controller * chip ,
unsigned int irq )
2013-08-09 16:49:19 +02:00
{
struct tegra_msi * msi = to_tegra_msi ( chip ) ;
struct irq_data * d = irq_get_irq_data ( irq ) ;
2014-07-29 09:33:30 +08:00
irq_hw_number_t hwirq = irqd_to_hwirq ( d ) ;
2013-08-09 16:49:19 +02:00
2014-07-29 09:33:30 +08:00
irq_dispose_mapping ( irq ) ;
tegra_msi_free ( msi , hwirq ) ;
2013-08-09 16:49:19 +02:00
}
static struct irq_chip tegra_msi_irq_chip = {
. name = " Tegra PCIe MSI " ,
2014-11-23 12:23:20 +01:00
. irq_enable = pci_msi_unmask_irq ,
. irq_disable = pci_msi_mask_irq ,
. irq_mask = pci_msi_mask_irq ,
. irq_unmask = pci_msi_unmask_irq ,
2013-08-09 16:49:19 +02:00
} ;
static int tegra_msi_map ( struct irq_domain * domain , unsigned int irq ,
irq_hw_number_t hwirq )
{
irq_set_chip_and_handler ( irq , & tegra_msi_irq_chip , handle_simple_irq ) ;
irq_set_chip_data ( irq , domain - > host_data ) ;
2013-05-06 14:19:19 -06:00
tegra_cpuidle_pcie_irqs_in_use ( ) ;
2013-08-09 16:49:19 +02:00
return 0 ;
}
static const struct irq_domain_ops msi_domain_ops = {
. map = tegra_msi_map ,
} ;
2018-02-28 15:30:34 +05:30
static int tegra_pcie_msi_setup ( struct tegra_pcie * pcie )
2013-08-09 16:49:19 +02:00
{
2016-11-25 11:57:12 +01:00
struct pci_host_bridge * host = pci_host_bridge_from_priv ( pcie ) ;
struct platform_device * pdev = to_platform_device ( pcie - > dev ) ;
2013-08-09 16:49:19 +02:00
struct tegra_msi * msi = & pcie - > msi ;
2016-11-25 11:57:12 +01:00
struct device * dev = pcie - > dev ;
2013-08-09 16:49:19 +02:00
int err ;
mutex_init ( & msi - > lock ) ;
2016-10-06 13:43:04 -05:00
msi - > chip . dev = dev ;
2013-08-09 16:49:19 +02:00
msi - > chip . setup_irq = tegra_msi_setup_irq ;
msi - > chip . teardown_irq = tegra_msi_teardown_irq ;
2016-10-06 13:43:04 -05:00
msi - > domain = irq_domain_add_linear ( dev - > of_node , INT_PCI_MSI_NR ,
2013-08-09 16:49:19 +02:00
& msi_domain_ops , & msi - > chip ) ;
if ( ! msi - > domain ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to create IRQ domain \n " ) ;
2013-08-09 16:49:19 +02:00
return - ENOMEM ;
}
err = platform_get_irq_byname ( pdev , " msi " ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to get IRQ: %d \n " , err ) ;
2013-08-09 16:49:19 +02:00
goto err ;
}
msi - > irq = err ;
2015-12-10 21:18:20 +02:00
err = request_irq ( msi - > irq , tegra_pcie_msi_irq , IRQF_NO_THREAD ,
2013-08-09 16:49:19 +02:00
tegra_msi_irq_chip . name , pcie ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to request IRQ: %d \n " , err ) ;
2013-08-09 16:49:19 +02:00
goto err ;
}
2017-10-09 12:29:35 +02:00
/* setup AFI/FPCI range */
msi - > pages = __get_free_pages ( GFP_KERNEL , 0 ) ;
msi - > phys = virt_to_phys ( ( void * ) msi - > pages ) ;
2018-02-28 15:30:34 +05:30
host - > msi = & msi - > chip ;
return 0 ;
err :
irq_domain_remove ( msi - > domain ) ;
return err ;
}
static void tegra_pcie_enable_msi ( struct tegra_pcie * pcie )
{
const struct tegra_pcie_soc * soc = pcie - > soc ;
struct tegra_msi * msi = & pcie - > msi ;
u32 reg ;
2013-08-09 16:49:19 +02:00
2017-05-04 22:10:31 +02:00
afi_writel ( pcie , msi - > phys > > soc - > msi_base_shift , AFI_MSI_FPCI_BAR_ST ) ;
afi_writel ( pcie , msi - > phys , AFI_MSI_AXI_BAR_ST ) ;
2013-08-09 16:49:19 +02:00
/* this register is in 4K increments */
afi_writel ( pcie , 1 , AFI_MSI_BAR_SZ ) ;
/* enable all MSI vectors */
afi_writel ( pcie , 0xffffffff , AFI_MSI_EN_VEC0 ) ;
afi_writel ( pcie , 0xffffffff , AFI_MSI_EN_VEC1 ) ;
afi_writel ( pcie , 0xffffffff , AFI_MSI_EN_VEC2 ) ;
afi_writel ( pcie , 0xffffffff , AFI_MSI_EN_VEC3 ) ;
afi_writel ( pcie , 0xffffffff , AFI_MSI_EN_VEC4 ) ;
afi_writel ( pcie , 0xffffffff , AFI_MSI_EN_VEC5 ) ;
afi_writel ( pcie , 0xffffffff , AFI_MSI_EN_VEC6 ) ;
afi_writel ( pcie , 0xffffffff , AFI_MSI_EN_VEC7 ) ;
/* and unmask the MSI interrupt */
reg = afi_readl ( pcie , AFI_INTR_MASK ) ;
reg | = AFI_INTR_MASK_MSI_MASK ;
afi_writel ( pcie , reg , AFI_INTR_MASK ) ;
2018-02-28 15:30:34 +05:30
}
2013-08-09 16:49:19 +02:00
2018-02-28 15:30:34 +05:30
static void tegra_pcie_msi_teardown ( struct tegra_pcie * pcie )
{
struct tegra_msi * msi = & pcie - > msi ;
unsigned int i , irq ;
2016-11-25 11:57:12 +01:00
2018-02-28 15:30:34 +05:30
free_pages ( msi - > pages , 0 ) ;
if ( msi - > irq > 0 )
free_irq ( msi - > irq , pcie ) ;
for ( i = 0 ; i < INT_PCI_MSI_NR ; i + + ) {
irq = irq_find_mapping ( msi - > domain , i ) ;
if ( irq > 0 )
irq_dispose_mapping ( irq ) ;
}
2013-08-09 16:49:19 +02:00
irq_domain_remove ( msi - > domain ) ;
}
static int tegra_pcie_disable_msi ( struct tegra_pcie * pcie )
{
u32 value ;
/* mask the MSI interrupt */
value = afi_readl ( pcie , AFI_INTR_MASK ) ;
value & = ~ AFI_INTR_MASK_MSI_MASK ;
afi_writel ( pcie , value , AFI_INTR_MASK ) ;
/* disable all MSI vectors */
afi_writel ( pcie , 0 , AFI_MSI_EN_VEC0 ) ;
afi_writel ( pcie , 0 , AFI_MSI_EN_VEC1 ) ;
afi_writel ( pcie , 0 , AFI_MSI_EN_VEC2 ) ;
afi_writel ( pcie , 0 , AFI_MSI_EN_VEC3 ) ;
afi_writel ( pcie , 0 , AFI_MSI_EN_VEC4 ) ;
afi_writel ( pcie , 0 , AFI_MSI_EN_VEC5 ) ;
afi_writel ( pcie , 0 , AFI_MSI_EN_VEC6 ) ;
afi_writel ( pcie , 0 , AFI_MSI_EN_VEC7 ) ;
return 0 ;
}
static int tegra_pcie_get_xbar_config ( struct tegra_pcie * pcie , u32 lanes ,
u32 * xbar )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
struct device_node * np = dev - > of_node ;
2013-08-09 16:49:19 +02:00
2017-09-27 17:28:35 +05:30
if ( of_device_is_compatible ( np , " nvidia,tegra186-pcie " ) ) {
switch ( lanes ) {
case 0x010004 :
dev_info ( dev , " 4x1, 1x1 configuration \n " ) ;
* xbar = AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_401 ;
return 0 ;
case 0x010102 :
dev_info ( dev , " 2x1, 1X1, 1x1 configuration \n " ) ;
* xbar = AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_211 ;
return 0 ;
case 0x010101 :
dev_info ( dev , " 1x1, 1x1, 1x1 configuration \n " ) ;
* xbar = AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_111 ;
return 0 ;
default :
dev_info ( dev , " wrong configuration updated in DT, "
" switching to default 2x1, 1x1, 1x1 "
" configuration \n " ) ;
* xbar = AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_211 ;
return 0 ;
}
} else if ( of_device_is_compatible ( np , " nvidia,tegra124-pcie " ) | |
of_device_is_compatible ( np , " nvidia,tegra210-pcie " ) ) {
2014-08-26 17:11:38 +02:00
switch ( lanes ) {
case 0x0000104 :
2016-10-06 13:43:04 -05:00
dev_info ( dev , " 4x1, 1x1 configuration \n " ) ;
2014-08-26 17:11:38 +02:00
* xbar = AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_X4_X1 ;
return 0 ;
case 0x0000102 :
2016-10-06 13:43:04 -05:00
dev_info ( dev , " 2x1, 1x1 configuration \n " ) ;
2014-08-26 17:11:38 +02:00
* xbar = AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_X2_X1 ;
return 0 ;
}
} else if ( of_device_is_compatible ( np , " nvidia,tegra30-pcie " ) ) {
2013-08-09 16:49:24 +02:00
switch ( lanes ) {
case 0x00000204 :
2016-10-06 13:43:04 -05:00
dev_info ( dev , " 4x1, 2x1 configuration \n " ) ;
2013-08-09 16:49:24 +02:00
* xbar = AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_420 ;
return 0 ;
case 0x00020202 :
2016-10-06 13:43:04 -05:00
dev_info ( dev , " 2x3 configuration \n " ) ;
2013-08-09 16:49:24 +02:00
* xbar = AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_222 ;
return 0 ;
case 0x00010104 :
2016-10-06 13:43:04 -05:00
dev_info ( dev , " 4x1, 1x2 configuration \n " ) ;
2013-08-09 16:49:24 +02:00
* xbar = AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_411 ;
return 0 ;
}
} else if ( of_device_is_compatible ( np , " nvidia,tegra20-pcie " ) ) {
switch ( lanes ) {
case 0x00000004 :
2016-10-06 13:43:04 -05:00
dev_info ( dev , " single-mode configuration \n " ) ;
2013-08-09 16:49:24 +02:00
* xbar = AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_SINGLE ;
return 0 ;
case 0x00000202 :
2016-10-06 13:43:04 -05:00
dev_info ( dev , " dual-mode configuration \n " ) ;
2013-08-09 16:49:24 +02:00
* xbar = AFI_PCIE_CONFIG_SM2TMS0_XBAR_CONFIG_DUAL ;
return 0 ;
}
2013-08-09 16:49:19 +02:00
}
return - EINVAL ;
}
2014-05-28 16:49:13 +02:00
/*
* Check whether a given set of supplies is available in a device tree node .
* This is used to check whether the new or the legacy device tree bindings
* should be used .
*/
static bool of_regulator_bulk_available ( struct device_node * np ,
struct regulator_bulk_data * supplies ,
unsigned int num_supplies )
{
char property [ 32 ] ;
unsigned int i ;
for ( i = 0 ; i < num_supplies ; i + + ) {
snprintf ( property , 32 , " %s-supply " , supplies [ i ] . supply ) ;
if ( of_find_property ( np , property , NULL ) = = NULL )
return false ;
}
return true ;
}
/*
* Old versions of the device tree binding for this device used a set of power
* supplies that didn ' t match the hardware inputs . This happened to work for a
* number of cases but is not future proof . However to preserve backwards -
* compatibility with old device trees , this function will try to use the old
* set of supplies .
*/
static int tegra_pcie_get_legacy_regulators ( struct tegra_pcie * pcie )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
struct device_node * np = dev - > of_node ;
2014-05-28 16:49:13 +02:00
if ( of_device_is_compatible ( np , " nvidia,tegra30-pcie " ) )
pcie - > num_supplies = 3 ;
else if ( of_device_is_compatible ( np , " nvidia,tegra20-pcie " ) )
pcie - > num_supplies = 2 ;
if ( pcie - > num_supplies = = 0 ) {
2017-07-18 16:43:21 -05:00
dev_err ( dev , " device %pOF not supported in legacy mode \n " , np ) ;
2014-05-28 16:49:13 +02:00
return - ENODEV ;
}
2016-10-06 13:43:04 -05:00
pcie - > supplies = devm_kcalloc ( dev , pcie - > num_supplies ,
2014-05-28 16:49:13 +02:00
sizeof ( * pcie - > supplies ) ,
GFP_KERNEL ) ;
if ( ! pcie - > supplies )
return - ENOMEM ;
pcie - > supplies [ 0 ] . supply = " pex-clk " ;
pcie - > supplies [ 1 ] . supply = " vdd " ;
if ( pcie - > num_supplies > 2 )
pcie - > supplies [ 2 ] . supply = " avdd " ;
2016-10-06 13:43:04 -05:00
return devm_regulator_bulk_get ( dev , pcie - > num_supplies , pcie - > supplies ) ;
2014-05-28 16:49:13 +02:00
}
/*
* Obtains the list of regulators required for a particular generation of the
* IP block .
*
* This would ' ve been nice to do simply by providing static tables for use
* with the regulator_bulk_ * ( ) API , but unfortunately Tegra30 is a bit quirky
* in that it has two pairs or AVDD_PEX and VDD_PEX supplies ( PEXA and PEXB )
* and either seems to be optional depending on which ports are being used .
*/
static int tegra_pcie_get_regulators ( struct tegra_pcie * pcie , u32 lane_mask )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
struct device_node * np = dev - > of_node ;
2014-05-28 16:49:13 +02:00
unsigned int i = 0 ;
2017-09-27 17:28:35 +05:30
if ( of_device_is_compatible ( np , " nvidia,tegra186-pcie " ) ) {
pcie - > num_supplies = 4 ;
pcie - > supplies = devm_kcalloc ( pcie - > dev , pcie - > num_supplies ,
sizeof ( * pcie - > supplies ) ,
GFP_KERNEL ) ;
if ( ! pcie - > supplies )
return - ENOMEM ;
pcie - > supplies [ i + + ] . supply = " dvdd-pex " ;
pcie - > supplies [ i + + ] . supply = " hvdd-pex-pll " ;
pcie - > supplies [ i + + ] . supply = " hvdd-pex " ;
pcie - > supplies [ i + + ] . supply = " vddio-pexctl-aud " ;
} else if ( of_device_is_compatible ( np , " nvidia,tegra210-pcie " ) ) {
2016-11-25 11:57:15 +01:00
pcie - > num_supplies = 6 ;
pcie - > supplies = devm_kcalloc ( pcie - > dev , pcie - > num_supplies ,
sizeof ( * pcie - > supplies ) ,
GFP_KERNEL ) ;
if ( ! pcie - > supplies )
return - ENOMEM ;
pcie - > supplies [ i + + ] . supply = " avdd-pll-uerefe " ;
pcie - > supplies [ i + + ] . supply = " hvddio-pex " ;
pcie - > supplies [ i + + ] . supply = " dvddio-pex " ;
pcie - > supplies [ i + + ] . supply = " dvdd-pex-pll " ;
pcie - > supplies [ i + + ] . supply = " hvdd-pex-pll-e " ;
pcie - > supplies [ i + + ] . supply = " vddio-pex-ctl " ;
} else if ( of_device_is_compatible ( np , " nvidia,tegra124-pcie " ) ) {
2014-08-26 17:11:38 +02:00
pcie - > num_supplies = 7 ;
2016-10-06 13:43:04 -05:00
pcie - > supplies = devm_kcalloc ( dev , pcie - > num_supplies ,
2014-08-26 17:11:38 +02:00
sizeof ( * pcie - > supplies ) ,
GFP_KERNEL ) ;
if ( ! pcie - > supplies )
return - ENOMEM ;
pcie - > supplies [ i + + ] . supply = " avddio-pex " ;
pcie - > supplies [ i + + ] . supply = " dvddio-pex " ;
pcie - > supplies [ i + + ] . supply = " avdd-pex-pll " ;
pcie - > supplies [ i + + ] . supply = " hvdd-pex " ;
pcie - > supplies [ i + + ] . supply = " hvdd-pex-pll-e " ;
pcie - > supplies [ i + + ] . supply = " vddio-pex-ctl " ;
pcie - > supplies [ i + + ] . supply = " avdd-pll-erefe " ;
} else if ( of_device_is_compatible ( np , " nvidia,tegra30-pcie " ) ) {
2014-05-28 16:49:13 +02:00
bool need_pexa = false , need_pexb = false ;
/* VDD_PEXA and AVDD_PEXA supply lanes 0 to 3 */
if ( lane_mask & 0x0f )
need_pexa = true ;
/* VDD_PEXB and AVDD_PEXB supply lanes 4 to 5 */
if ( lane_mask & 0x30 )
need_pexb = true ;
pcie - > num_supplies = 4 + ( need_pexa ? 2 : 0 ) +
( need_pexb ? 2 : 0 ) ;
2016-10-06 13:43:04 -05:00
pcie - > supplies = devm_kcalloc ( dev , pcie - > num_supplies ,
2014-05-28 16:49:13 +02:00
sizeof ( * pcie - > supplies ) ,
GFP_KERNEL ) ;
if ( ! pcie - > supplies )
return - ENOMEM ;
pcie - > supplies [ i + + ] . supply = " avdd-pex-pll " ;
pcie - > supplies [ i + + ] . supply = " hvdd-pex " ;
pcie - > supplies [ i + + ] . supply = " vddio-pex-ctl " ;
pcie - > supplies [ i + + ] . supply = " avdd-plle " ;
if ( need_pexa ) {
pcie - > supplies [ i + + ] . supply = " avdd-pexa " ;
pcie - > supplies [ i + + ] . supply = " vdd-pexa " ;
}
if ( need_pexb ) {
pcie - > supplies [ i + + ] . supply = " avdd-pexb " ;
pcie - > supplies [ i + + ] . supply = " vdd-pexb " ;
}
} else if ( of_device_is_compatible ( np , " nvidia,tegra20-pcie " ) ) {
pcie - > num_supplies = 5 ;
2016-10-06 13:43:04 -05:00
pcie - > supplies = devm_kcalloc ( dev , pcie - > num_supplies ,
2014-05-28 16:49:13 +02:00
sizeof ( * pcie - > supplies ) ,
GFP_KERNEL ) ;
if ( ! pcie - > supplies )
return - ENOMEM ;
pcie - > supplies [ 0 ] . supply = " avdd-pex " ;
pcie - > supplies [ 1 ] . supply = " vdd-pex " ;
pcie - > supplies [ 2 ] . supply = " avdd-pex-pll " ;
pcie - > supplies [ 3 ] . supply = " avdd-plle " ;
pcie - > supplies [ 4 ] . supply = " vddio-pex-clk " ;
}
2016-10-06 13:43:04 -05:00
if ( of_regulator_bulk_available ( dev - > of_node , pcie - > supplies ,
2014-05-28 16:49:13 +02:00
pcie - > num_supplies ) )
2016-10-06 13:43:04 -05:00
return devm_regulator_bulk_get ( dev , pcie - > num_supplies ,
2014-05-28 16:49:13 +02:00
pcie - > supplies ) ;
/*
* If not all regulators are available for this new scheme , assume
* that the device tree complies with an older version of the device
* tree binding .
*/
2016-10-06 13:43:04 -05:00
dev_info ( dev , " using legacy DT binding for power supplies \n " ) ;
2014-05-28 16:49:13 +02:00
2016-10-06 13:43:04 -05:00
devm_kfree ( dev , pcie - > supplies ) ;
2014-05-28 16:49:13 +02:00
pcie - > num_supplies = 0 ;
return tegra_pcie_get_legacy_regulators ( pcie ) ;
}
2013-08-09 16:49:19 +02:00
static int tegra_pcie_parse_dt ( struct tegra_pcie * pcie )
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
struct device_node * np = dev - > of_node , * port ;
2016-08-15 17:31:31 +02:00
const struct tegra_pcie_soc * soc = pcie - > soc ;
2013-08-09 16:49:19 +02:00
struct of_pci_range_parser parser ;
struct of_pci_range range ;
2014-05-28 16:49:13 +02:00
u32 lanes = 0 , mask = 0 ;
unsigned int lane = 0 ;
2013-08-09 16:49:19 +02:00
struct resource res ;
int err ;
if ( of_pci_range_parser_init ( & parser , np ) ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " missing \" ranges \" property \n " ) ;
2013-08-09 16:49:19 +02:00
return - EINVAL ;
}
for_each_of_pci_range ( & parser , & range ) {
2014-09-29 15:29:25 +01:00
err = of_pci_range_to_resource ( & range , np , & res ) ;
if ( err < 0 )
return err ;
2013-08-09 16:49:19 +02:00
switch ( res . flags & IORESOURCE_TYPE_BITS ) {
case IORESOURCE_IO :
2016-02-09 15:52:32 +01:00
/* Track the bus -> CPU I/O mapping offset. */
pcie - > offset . io = res . start - range . pci_addr ;
2014-11-27 09:54:09 +01:00
memcpy ( & pcie - > pio , & res , sizeof ( res ) ) ;
pcie - > pio . name = np - > full_name ;
/*
* The Tegra PCIe host bridge uses this to program the
* mapping of the I / O space to the physical address ,
* so we override the . start and . end fields here that
* of_pci_range_to_resource ( ) converted to I / O space .
* We also set the IORESOURCE_MEM type to clarify that
* the resource is in the physical memory space .
*/
pcie - > io . start = range . cpu_addr ;
pcie - > io . end = range . cpu_addr + range . size - 1 ;
pcie - > io . flags = IORESOURCE_MEM ;
pcie - > io . name = " I/O " ;
memcpy ( & res , & pcie - > io , sizeof ( res ) ) ;
2013-08-09 16:49:19 +02:00
break ;
case IORESOURCE_MEM :
2016-02-09 15:52:32 +01:00
/*
* Track the bus - > CPU memory mapping offset . This
* assumes that the prefetchable and non - prefetchable
* regions will be the last of type IORESOURCE_MEM in
* the ranges property .
* */
pcie - > offset . mem = res . start - range . pci_addr ;
2013-08-09 16:49:19 +02:00
if ( res . flags & IORESOURCE_PREFETCH ) {
memcpy ( & pcie - > prefetch , & res , sizeof ( res ) ) ;
2014-08-01 14:15:11 +02:00
pcie - > prefetch . name = " prefetchable " ;
2013-08-09 16:49:19 +02:00
} else {
memcpy ( & pcie - > mem , & res , sizeof ( res ) ) ;
2014-08-01 14:15:11 +02:00
pcie - > mem . name = " non-prefetchable " ;
2013-08-09 16:49:19 +02:00
}
break ;
}
}
err = of_pci_parse_bus_range ( np , & pcie - > busn ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to parse ranges property: %d \n " , err ) ;
2013-08-09 16:49:19 +02:00
pcie - > busn . name = np - > name ;
pcie - > busn . start = 0 ;
pcie - > busn . end = 0xff ;
pcie - > busn . flags = IORESOURCE_BUS ;
}
/* parse root ports */
for_each_child_of_node ( np , port ) {
struct tegra_pcie_port * rp ;
unsigned int index ;
u32 value ;
err = of_pci_get_devfn ( port ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to parse address: %d \n " , err ) ;
2013-08-09 16:49:19 +02:00
return err ;
}
index = PCI_SLOT ( err ) ;
2013-08-09 16:49:24 +02:00
if ( index < 1 | | index > soc - > num_ports ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " invalid port number: %d \n " , index ) ;
2013-08-09 16:49:19 +02:00
return - EINVAL ;
}
index - - ;
err = of_property_read_u32 ( port , " nvidia,num-lanes " , & value ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to parse # of lanes: %d \n " ,
2013-08-09 16:49:19 +02:00
err ) ;
return err ;
}
if ( value > 16 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " invalid # of lanes: %u \n " , value ) ;
2013-08-09 16:49:19 +02:00
return - EINVAL ;
}
lanes | = value < < ( index < < 3 ) ;
2014-05-28 16:49:13 +02:00
if ( ! of_device_is_available ( port ) ) {
lane + = value ;
2013-08-09 16:49:19 +02:00
continue ;
2014-05-28 16:49:13 +02:00
}
mask | = ( ( 1 < < value ) - 1 ) < < lane ;
lane + = value ;
2013-08-09 16:49:19 +02:00
2016-10-06 13:43:04 -05:00
rp = devm_kzalloc ( dev , sizeof ( * rp ) , GFP_KERNEL ) ;
2013-08-09 16:49:19 +02:00
if ( ! rp )
return - ENOMEM ;
err = of_address_to_resource ( port , 0 , & rp - > regs ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to parse address: %d \n " , err ) ;
2013-08-09 16:49:19 +02:00
return err ;
}
INIT_LIST_HEAD ( & rp - > list ) ;
rp - > index = index ;
rp - > lanes = value ;
rp - > pcie = pcie ;
2015-11-11 18:25:59 +01:00
rp - > np = port ;
2013-08-09 16:49:19 +02:00
2017-04-19 17:49:06 +01:00
rp - > base = devm_pci_remap_cfg_resource ( dev , & rp - > regs ) ;
2013-08-26 11:11:09 +02:00
if ( IS_ERR ( rp - > base ) )
return PTR_ERR ( rp - > base ) ;
2013-08-09 16:49:19 +02:00
list_add_tail ( & rp - > list , & pcie - > ports ) ;
}
err = tegra_pcie_get_xbar_config ( pcie , lanes , & pcie - > xbar_config ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " invalid lane configuration \n " ) ;
2013-08-09 16:49:19 +02:00
return err ;
}
2014-05-28 16:49:13 +02:00
err = tegra_pcie_get_regulators ( pcie , mask ) ;
if ( err < 0 )
return err ;
2013-08-09 16:49:19 +02:00
return 0 ;
}
/*
* FIXME : If there are no PCIe cards attached , then calling this function
* can result in the increase of the bootup time as there are big timeout
* loops .
*/
# define TEGRA_PCIE_LINKUP_TIMEOUT 200 /* up to 1.2 seconds */
static bool tegra_pcie_port_check_link ( struct tegra_pcie_port * port )
{
2016-10-06 13:43:04 -05:00
struct device * dev = port - > pcie - > dev ;
2013-08-09 16:49:19 +02:00
unsigned int retries = 3 ;
unsigned long value ;
2014-08-26 17:11:38 +02:00
/* override presence detection */
value = readl ( port - > base + RP_PRIV_MISC ) ;
value & = ~ RP_PRIV_MISC_PRSNT_MAP_EP_ABSNT ;
value | = RP_PRIV_MISC_PRSNT_MAP_EP_PRSNT ;
writel ( value , port - > base + RP_PRIV_MISC ) ;
2013-08-09 16:49:19 +02:00
do {
unsigned int timeout = TEGRA_PCIE_LINKUP_TIMEOUT ;
do {
value = readl ( port - > base + RP_VEND_XP ) ;
if ( value & RP_VEND_XP_DL_UP )
break ;
usleep_range ( 1000 , 2000 ) ;
} while ( - - timeout ) ;
if ( ! timeout ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " link %u down, retrying \n " , port - > index ) ;
2013-08-09 16:49:19 +02:00
goto retry ;
}
timeout = TEGRA_PCIE_LINKUP_TIMEOUT ;
do {
value = readl ( port - > base + RP_LINK_CONTROL_STATUS ) ;
if ( value & RP_LINK_CONTROL_STATUS_DL_LINK_ACTIVE )
return true ;
usleep_range ( 1000 , 2000 ) ;
} while ( - - timeout ) ;
retry :
tegra_pcie_port_reset ( port ) ;
} while ( - - retries ) ;
return false ;
}
2016-11-25 11:57:12 +01:00
static void tegra_pcie_enable_ports ( struct tegra_pcie * pcie )
2013-08-09 16:49:19 +02:00
{
2016-10-06 13:43:04 -05:00
struct device * dev = pcie - > dev ;
2013-08-09 16:49:19 +02:00
struct tegra_pcie_port * port , * tmp ;
list_for_each_entry_safe ( port , tmp , & pcie - > ports , list ) {
2016-10-06 13:43:04 -05:00
dev_info ( dev , " probing port %u, using %u lanes \n " ,
2013-08-09 16:49:19 +02:00
port - > index , port - > lanes ) ;
tegra_pcie_port_enable ( port ) ;
if ( tegra_pcie_port_check_link ( port ) )
continue ;
2016-10-06 13:43:04 -05:00
dev_info ( dev , " link %u down, ignoring \n " , port - > index ) ;
2013-08-09 16:49:19 +02:00
tegra_pcie_port_disable ( port ) ;
tegra_pcie_port_free ( port ) ;
}
}
2018-02-28 15:30:32 +05:30
static void tegra_pcie_disable_ports ( struct tegra_pcie * pcie )
{
struct tegra_pcie_port * port , * tmp ;
2018-02-28 15:30:34 +05:30
list_for_each_entry_safe ( port , tmp , & pcie - > ports , list )
2018-02-28 15:30:32 +05:30
tegra_pcie_port_disable ( port ) ;
}
2018-02-28 15:30:33 +05:30
static const struct tegra_pcie_port_soc tegra20_pcie_ports [ ] = {
{ . pme . turnoff_bit = 0 , . pme . ack_bit = 5 } ,
{ . pme . turnoff_bit = 8 , . pme . ack_bit = 10 } ,
} ;
2016-08-15 17:31:31 +02:00
static const struct tegra_pcie_soc tegra20_pcie = {
2013-08-09 16:49:24 +02:00
. num_ports = 2 ,
2018-02-28 15:30:33 +05:30
. ports = tegra20_pcie_ports ,
2013-08-09 16:49:24 +02:00
. msi_base_shift = 0 ,
. pads_pll_ctl = PADS_PLL_CTL_TEGRA20 ,
. tx_ref_sel = PADS_PLL_CTL_TXCLKREF_DIV10 ,
2016-07-25 16:02:27 -05:00
. pads_refclk_cfg0 = 0xfa5cfa5c ,
2013-08-09 16:49:24 +02:00
. has_pex_clkreq_en = false ,
. has_pex_bias_ctrl = false ,
. has_intr_prsnt_sense = false ,
. has_cml_clk = false ,
2014-08-26 17:11:38 +02:00
. has_gen2 = false ,
2016-11-25 11:57:14 +01:00
. force_pca_enable = false ,
2017-09-27 17:28:35 +05:30
. program_uphy = true ,
2013-08-09 16:49:24 +02:00
} ;
2018-02-28 15:30:33 +05:30
static const struct tegra_pcie_port_soc tegra30_pcie_ports [ ] = {
{ . pme . turnoff_bit = 0 , . pme . ack_bit = 5 } ,
{ . pme . turnoff_bit = 8 , . pme . ack_bit = 10 } ,
{ . pme . turnoff_bit = 16 , . pme . ack_bit = 18 } ,
} ;
2016-08-15 17:31:31 +02:00
static const struct tegra_pcie_soc tegra30_pcie = {
2013-08-09 16:49:24 +02:00
. num_ports = 3 ,
2018-02-28 15:30:33 +05:30
. ports = tegra30_pcie_ports ,
2013-08-09 16:49:24 +02:00
. msi_base_shift = 8 ,
. pads_pll_ctl = PADS_PLL_CTL_TEGRA30 ,
. tx_ref_sel = PADS_PLL_CTL_TXCLKREF_BUF_EN ,
2016-07-25 16:02:27 -05:00
. pads_refclk_cfg0 = 0xfa5cfa5c ,
. pads_refclk_cfg1 = 0xfa5cfa5c ,
2013-08-09 16:49:24 +02:00
. has_pex_clkreq_en = true ,
. has_pex_bias_ctrl = true ,
. has_intr_prsnt_sense = true ,
. has_cml_clk = true ,
2014-08-26 17:11:38 +02:00
. has_gen2 = false ,
2016-11-25 11:57:14 +01:00
. force_pca_enable = false ,
2017-09-27 17:28:35 +05:30
. program_uphy = true ,
2014-08-26 17:11:38 +02:00
} ;
2016-08-15 17:31:31 +02:00
static const struct tegra_pcie_soc tegra124_pcie = {
2014-08-26 17:11:38 +02:00
. num_ports = 2 ,
2018-02-28 15:30:33 +05:30
. ports = tegra20_pcie_ports ,
2014-08-26 17:11:38 +02:00
. msi_base_shift = 8 ,
. pads_pll_ctl = PADS_PLL_CTL_TEGRA30 ,
. tx_ref_sel = PADS_PLL_CTL_TXCLKREF_BUF_EN ,
2016-07-25 16:02:27 -05:00
. pads_refclk_cfg0 = 0x44ac44ac ,
2014-08-26 17:11:38 +02:00
. has_pex_clkreq_en = true ,
. has_pex_bias_ctrl = true ,
. has_intr_prsnt_sense = true ,
. has_cml_clk = true ,
. has_gen2 = true ,
2016-11-25 11:57:14 +01:00
. force_pca_enable = false ,
2017-09-27 17:28:35 +05:30
. program_uphy = true ,
2013-08-09 16:49:24 +02:00
} ;
2016-11-25 11:57:15 +01:00
static const struct tegra_pcie_soc tegra210_pcie = {
. num_ports = 2 ,
2018-02-28 15:30:33 +05:30
. ports = tegra20_pcie_ports ,
2016-11-25 11:57:15 +01:00
. msi_base_shift = 8 ,
. pads_pll_ctl = PADS_PLL_CTL_TEGRA30 ,
. tx_ref_sel = PADS_PLL_CTL_TXCLKREF_BUF_EN ,
. pads_refclk_cfg0 = 0x90b890b8 ,
. has_pex_clkreq_en = true ,
. has_pex_bias_ctrl = true ,
. has_intr_prsnt_sense = true ,
. has_cml_clk = true ,
. has_gen2 = true ,
. force_pca_enable = true ,
2017-09-27 17:28:35 +05:30
. program_uphy = true ,
} ;
2018-02-28 15:30:33 +05:30
static const struct tegra_pcie_port_soc tegra186_pcie_ports [ ] = {
{ . pme . turnoff_bit = 0 , . pme . ack_bit = 5 } ,
{ . pme . turnoff_bit = 8 , . pme . ack_bit = 10 } ,
{ . pme . turnoff_bit = 12 , . pme . ack_bit = 14 } ,
} ;
2017-09-27 17:28:35 +05:30
static const struct tegra_pcie_soc tegra186_pcie = {
. num_ports = 3 ,
2018-02-28 15:30:33 +05:30
. ports = tegra186_pcie_ports ,
2017-09-27 17:28:35 +05:30
. msi_base_shift = 8 ,
. pads_pll_ctl = PADS_PLL_CTL_TEGRA30 ,
. tx_ref_sel = PADS_PLL_CTL_TXCLKREF_BUF_EN ,
. pads_refclk_cfg0 = 0x80b880b8 ,
. pads_refclk_cfg1 = 0x000480b8 ,
. has_pex_clkreq_en = true ,
. has_pex_bias_ctrl = true ,
. has_intr_prsnt_sense = true ,
. has_cml_clk = false ,
. has_gen2 = true ,
. force_pca_enable = false ,
. program_uphy = false ,
2016-11-25 11:57:15 +01:00
} ;
2013-08-09 16:49:24 +02:00
static const struct of_device_id tegra_pcie_of_match [ ] = {
2017-09-27 17:28:35 +05:30
{ . compatible = " nvidia,tegra186-pcie " , . data = & tegra186_pcie } ,
2016-11-25 11:57:15 +01:00
{ . compatible = " nvidia,tegra210-pcie " , . data = & tegra210_pcie } ,
2016-08-15 17:31:31 +02:00
{ . compatible = " nvidia,tegra124-pcie " , . data = & tegra124_pcie } ,
{ . compatible = " nvidia,tegra30-pcie " , . data = & tegra30_pcie } ,
{ . compatible = " nvidia,tegra20-pcie " , . data = & tegra20_pcie } ,
2013-08-09 16:49:24 +02:00
{ } ,
} ;
2014-07-22 12:30:46 -06:00
static void * tegra_pcie_ports_seq_start ( struct seq_file * s , loff_t * pos )
{
struct tegra_pcie * pcie = s - > private ;
if ( list_empty ( & pcie - > ports ) )
return NULL ;
seq_printf ( s , " Index Status \n " ) ;
return seq_list_start ( & pcie - > ports , * pos ) ;
}
static void * tegra_pcie_ports_seq_next ( struct seq_file * s , void * v , loff_t * pos )
{
struct tegra_pcie * pcie = s - > private ;
return seq_list_next ( v , & pcie - > ports , pos ) ;
}
static void tegra_pcie_ports_seq_stop ( struct seq_file * s , void * v )
{
}
static int tegra_pcie_ports_seq_show ( struct seq_file * s , void * v )
{
bool up = false , active = false ;
struct tegra_pcie_port * port ;
unsigned int value ;
port = list_entry ( v , struct tegra_pcie_port , list ) ;
value = readl ( port - > base + RP_VEND_XP ) ;
if ( value & RP_VEND_XP_DL_UP )
up = true ;
value = readl ( port - > base + RP_LINK_CONTROL_STATUS ) ;
if ( value & RP_LINK_CONTROL_STATUS_DL_LINK_ACTIVE )
active = true ;
seq_printf ( s , " %2u " , port - > index ) ;
if ( up )
seq_printf ( s , " up " ) ;
if ( active ) {
if ( up )
seq_printf ( s , " , " ) ;
seq_printf ( s , " active " ) ;
}
seq_printf ( s , " \n " ) ;
return 0 ;
}
static const struct seq_operations tegra_pcie_ports_seq_ops = {
. start = tegra_pcie_ports_seq_start ,
. next = tegra_pcie_ports_seq_next ,
. stop = tegra_pcie_ports_seq_stop ,
. show = tegra_pcie_ports_seq_show ,
} ;
static int tegra_pcie_ports_open ( struct inode * inode , struct file * file )
{
struct tegra_pcie * pcie = inode - > i_private ;
struct seq_file * s ;
int err ;
err = seq_open ( file , & tegra_pcie_ports_seq_ops ) ;
if ( err )
return err ;
s = file - > private_data ;
s - > private = pcie ;
return 0 ;
}
static const struct file_operations tegra_pcie_ports_ops = {
. owner = THIS_MODULE ,
. open = tegra_pcie_ports_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = seq_release ,
} ;
2018-02-28 15:30:33 +05:30
static void tegra_pcie_debugfs_exit ( struct tegra_pcie * pcie )
{
debugfs_remove_recursive ( pcie - > debugfs ) ;
pcie - > debugfs = NULL ;
}
2014-07-22 12:30:46 -06:00
static int tegra_pcie_debugfs_init ( struct tegra_pcie * pcie )
{
struct dentry * file ;
pcie - > debugfs = debugfs_create_dir ( " pcie " , NULL ) ;
if ( ! pcie - > debugfs )
return - ENOMEM ;
file = debugfs_create_file ( " ports " , S_IFREG | S_IRUGO , pcie - > debugfs ,
pcie , & tegra_pcie_ports_ops ) ;
if ( ! file )
goto remove ;
return 0 ;
remove :
2018-02-28 15:30:33 +05:30
tegra_pcie_debugfs_exit ( pcie ) ;
2014-07-22 12:30:46 -06:00
return - ENOMEM ;
}
2013-08-09 16:49:19 +02:00
static int tegra_pcie_probe ( struct platform_device * pdev )
{
2016-10-06 13:43:04 -05:00
struct device * dev = & pdev - > dev ;
2016-11-25 11:57:12 +01:00
struct pci_host_bridge * host ;
2013-08-09 16:49:19 +02:00
struct tegra_pcie * pcie ;
2016-11-25 11:57:12 +01:00
struct pci_bus * child ;
2013-08-09 16:49:19 +02:00
int err ;
2017-06-28 15:13:54 -05:00
host = devm_pci_alloc_host_bridge ( dev , sizeof ( * pcie ) ) ;
2016-11-25 11:57:12 +01:00
if ( ! host )
2013-08-09 16:49:19 +02:00
return - ENOMEM ;
2016-11-25 11:57:12 +01:00
pcie = pci_host_bridge_priv ( host ) ;
2018-01-11 11:38:03 +05:30
host - > sysdata = pcie ;
2018-02-28 15:30:33 +05:30
platform_set_drvdata ( pdev , pcie ) ;
2016-11-25 11:57:12 +01:00
2016-10-06 13:43:04 -05:00
pcie - > soc = of_device_get_match_data ( dev ) ;
2013-08-09 16:49:19 +02:00
INIT_LIST_HEAD ( & pcie - > ports ) ;
2016-10-06 13:43:04 -05:00
pcie - > dev = dev ;
2013-08-09 16:49:19 +02:00
err = tegra_pcie_parse_dt ( pcie ) ;
if ( err < 0 )
return err ;
err = tegra_pcie_get_resources ( pcie ) ;
if ( err < 0 ) {
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to request resources: %d \n " , err ) ;
2013-08-09 16:49:19 +02:00
return err ;
}
2018-02-28 15:30:34 +05:30
err = tegra_pcie_msi_setup ( pcie ) ;
if ( err < 0 ) {
dev_err ( dev , " failed to enable MSI support: %d \n " , err ) ;
2013-08-09 16:49:19 +02:00
goto put_resources ;
2018-02-28 15:30:34 +05:30
}
2013-08-09 16:49:19 +02:00
2018-02-28 15:30:34 +05:30
pm_runtime_enable ( pcie - > dev ) ;
err = pm_runtime_get_sync ( pcie - > dev ) ;
if ( err ) {
dev_err ( dev , " fail to enable pcie controller: %d \n " , err ) ;
goto teardown_msi ;
2013-08-09 16:49:19 +02:00
}
2018-02-28 15:30:34 +05:30
err = tegra_pcie_request_resources ( pcie ) ;
if ( err )
goto pm_runtime_put ;
2016-11-25 11:57:12 +01:00
host - > busnr = pcie - > busn . start ;
host - > dev . parent = & pdev - > dev ;
host - > ops = & tegra_pcie_ops ;
2017-06-28 15:14:05 -05:00
host - > map_irq = tegra_pcie_map_irq ;
host - > swizzle_irq = pci_common_swizzle ;
2016-11-25 11:57:12 +01:00
2017-06-28 15:13:55 -05:00
err = pci_scan_root_bus_bridge ( host ) ;
2013-08-09 16:49:19 +02:00
if ( err < 0 ) {
2016-11-25 11:57:12 +01:00
dev_err ( dev , " failed to register host: %d \n " , err ) ;
2018-02-28 15:30:34 +05:30
goto free_resources ;
2013-08-09 16:49:19 +02:00
}
2016-11-25 11:57:12 +01:00
pci_bus_size_bridges ( host - > bus ) ;
pci_bus_assign_resources ( host - > bus ) ;
list_for_each_entry ( child , & host - > bus - > children , node )
pcie_bus_configure_settings ( child ) ;
pci_bus_add_devices ( host - > bus ) ;
2014-07-22 12:30:46 -06:00
if ( IS_ENABLED ( CONFIG_DEBUG_FS ) ) {
err = tegra_pcie_debugfs_init ( pcie ) ;
if ( err < 0 )
2016-10-06 13:43:04 -05:00
dev_err ( dev , " failed to setup debugfs: %d \n " , err ) ;
2014-07-22 12:30:46 -06:00
}
2013-08-09 16:49:19 +02:00
return 0 ;
2018-02-28 15:30:32 +05:30
free_resources :
tegra_pcie_free_resources ( pcie ) ;
2018-02-28 15:30:34 +05:30
pm_runtime_put :
pm_runtime_put_sync ( pcie - > dev ) ;
pm_runtime_disable ( pcie - > dev ) ;
teardown_msi :
tegra_pcie_msi_teardown ( pcie ) ;
2013-08-09 16:49:19 +02:00
put_resources :
tegra_pcie_put_resources ( pcie ) ;
return err ;
}
2018-02-28 15:30:33 +05:30
static int tegra_pcie_remove ( struct platform_device * pdev )
{
struct tegra_pcie * pcie = platform_get_drvdata ( pdev ) ;
struct pci_host_bridge * host = pci_host_bridge_from_priv ( pcie ) ;
2018-02-28 15:30:34 +05:30
struct tegra_pcie_port * port , * tmp ;
2018-02-28 15:30:33 +05:30
if ( IS_ENABLED ( CONFIG_DEBUG_FS ) )
tegra_pcie_debugfs_exit ( pcie ) ;
pci_stop_root_bus ( host - > bus ) ;
pci_remove_root_bus ( host - > bus ) ;
2018-02-28 15:30:34 +05:30
tegra_pcie_free_resources ( pcie ) ;
pm_runtime_put_sync ( pcie - > dev ) ;
pm_runtime_disable ( pcie - > dev ) ;
if ( IS_ENABLED ( CONFIG_PCI_MSI ) )
tegra_pcie_msi_teardown ( pcie ) ;
tegra_pcie_put_resources ( pcie ) ;
list_for_each_entry_safe ( port , tmp , & pcie - > ports , list )
tegra_pcie_port_free ( port ) ;
return 0 ;
}
static int __maybe_unused tegra_pcie_pm_suspend ( struct device * dev )
{
struct tegra_pcie * pcie = dev_get_drvdata ( dev ) ;
struct tegra_pcie_port * port ;
2018-02-28 15:30:33 +05:30
list_for_each_entry ( port , & pcie - > ports , list )
tegra_pcie_pme_turnoff ( port ) ;
tegra_pcie_disable_ports ( pcie ) ;
if ( IS_ENABLED ( CONFIG_PCI_MSI ) )
tegra_pcie_disable_msi ( pcie ) ;
tegra_pcie_disable_controller ( pcie ) ;
2018-02-28 15:30:34 +05:30
tegra_pcie_power_off ( pcie ) ;
return 0 ;
}
static int __maybe_unused tegra_pcie_pm_resume ( struct device * dev )
{
struct tegra_pcie * pcie = dev_get_drvdata ( dev ) ;
int err ;
err = tegra_pcie_power_on ( pcie ) ;
if ( err ) {
dev_err ( dev , " tegra pcie power on fail: %d \n " , err ) ;
return err ;
}
err = tegra_pcie_enable_controller ( pcie ) ;
if ( err ) {
dev_err ( dev , " tegra pcie controller enable fail: %d \n " , err ) ;
goto poweroff ;
}
tegra_pcie_setup_translations ( pcie ) ;
if ( IS_ENABLED ( CONFIG_PCI_MSI ) )
tegra_pcie_enable_msi ( pcie ) ;
tegra_pcie_enable_ports ( pcie ) ;
2018-02-28 15:30:33 +05:30
return 0 ;
2018-02-28 15:30:34 +05:30
poweroff :
tegra_pcie_power_off ( pcie ) ;
return err ;
2018-02-28 15:30:33 +05:30
}
2018-02-28 15:30:34 +05:30
static const struct dev_pm_ops tegra_pcie_pm_ops = {
SET_RUNTIME_PM_OPS ( tegra_pcie_pm_suspend , tegra_pcie_pm_resume , NULL )
SET_NOIRQ_SYSTEM_SLEEP_PM_OPS ( tegra_pcie_pm_suspend ,
tegra_pcie_pm_resume )
} ;
2013-08-09 16:49:19 +02:00
static struct platform_driver tegra_pcie_driver = {
. driver = {
. name = " tegra-pcie " ,
. of_match_table = tegra_pcie_of_match ,
. suppress_bind_attrs = true ,
2018-02-28 15:30:34 +05:30
. pm = & tegra_pcie_pm_ops ,
2013-08-09 16:49:19 +02:00
} ,
. probe = tegra_pcie_probe ,
2018-02-28 15:30:33 +05:30
. remove = tegra_pcie_remove ,
2013-08-09 16:49:19 +02:00
} ;
2018-02-28 15:30:33 +05:30
module_platform_driver ( tegra_pcie_driver ) ;
MODULE_LICENSE ( " GPL " ) ;