diff options
Diffstat (limited to 'drivers/pci/controller')
22 files changed, 1356 insertions, 1149 deletions
| diff --git a/drivers/pci/controller/Kconfig b/drivers/pci/controller/Kconfig index 028b287466fb..6671946dbf66 100644 --- a/drivers/pci/controller/Kconfig +++ b/drivers/pci/controller/Kconfig @@ -9,12 +9,14 @@ config PCI_MVEBU  	depends on MVEBU_MBUS  	depends on ARM  	depends on OF +	select PCI_BRIDGE_EMUL  config PCI_AARDVARK  	bool "Aardvark PCIe controller"  	depends on (ARCH_MVEBU && ARM64) || COMPILE_TEST  	depends on OF  	depends on PCI_MSI_IRQ_DOMAIN +	select PCI_BRIDGE_EMUL  	help  	 Add support for Aardvark 64bit PCIe Host Controller. This  	 controller is part of the South Bridge of the Marvel Armada @@ -231,7 +233,7 @@ config PCIE_ROCKCHIP_EP  	  available to support GEN2 with 4 slots.  config PCIE_MEDIATEK -	bool "MediaTek PCIe controller" +	tristate "MediaTek PCIe controller"  	depends on ARCH_MEDIATEK || COMPILE_TEST  	depends on OF  	depends on PCI_MSI_IRQ_DOMAIN diff --git a/drivers/pci/controller/dwc/Makefile b/drivers/pci/controller/dwc/Makefile index 5d2ce72c7a52..fcf91eacfc63 100644 --- a/drivers/pci/controller/dwc/Makefile +++ b/drivers/pci/controller/dwc/Makefile @@ -7,7 +7,7 @@ obj-$(CONFIG_PCI_DRA7XX) += pci-dra7xx.o  obj-$(CONFIG_PCI_EXYNOS) += pci-exynos.o  obj-$(CONFIG_PCI_IMX6) += pci-imx6.o  obj-$(CONFIG_PCIE_SPEAR13XX) += pcie-spear13xx.o -obj-$(CONFIG_PCI_KEYSTONE) += pci-keystone-dw.o pci-keystone.o +obj-$(CONFIG_PCI_KEYSTONE) += pci-keystone.o  obj-$(CONFIG_PCI_LAYERSCAPE) += pci-layerscape.o  obj-$(CONFIG_PCIE_QCOM) += pcie-qcom.o  obj-$(CONFIG_PCIE_ARMADA_8K) += pcie-armada8k.o diff --git a/drivers/pci/controller/dwc/pci-dra7xx.c b/drivers/pci/controller/dwc/pci-dra7xx.c index ce9224a36f62..a32d6dde7a57 100644 --- a/drivers/pci/controller/dwc/pci-dra7xx.c +++ b/drivers/pci/controller/dwc/pci-dra7xx.c @@ -542,7 +542,7 @@ static const struct of_device_id of_dra7xx_pcie_match[] = {  };  /* - * dra7xx_pcie_ep_unaligned_memaccess: workaround for AM572x/AM571x Errata i870 + * dra7xx_pcie_unaligned_memaccess: workaround for AM572x/AM571x Errata i870   * @dra7xx: the dra7xx device where the workaround should be applied   *   * Access to the PCIe slave port that are not 32-bit aligned will result @@ -552,7 +552,7 @@ static const struct of_device_id of_dra7xx_pcie_match[] = {   *   * To avoid this issue set PCIE_SS1_AXI2OCP_LEGACY_MODE_ENABLE to 1.   */ -static int dra7xx_pcie_ep_unaligned_memaccess(struct device *dev) +static int dra7xx_pcie_unaligned_memaccess(struct device *dev)  {  	int ret;  	struct device_node *np = dev->of_node; @@ -704,6 +704,11 @@ static int __init dra7xx_pcie_probe(struct platform_device *pdev)  		dra7xx_pcie_writel(dra7xx, PCIECTRL_TI_CONF_DEVICE_TYPE,  				   DEVICE_TYPE_RC); + +		ret = dra7xx_pcie_unaligned_memaccess(dev); +		if (ret) +			dev_err(dev, "WA for Errata i870 not applied\n"); +  		ret = dra7xx_add_pcie_port(dra7xx, pdev);  		if (ret < 0)  			goto err_gpio; @@ -717,7 +722,7 @@ static int __init dra7xx_pcie_probe(struct platform_device *pdev)  		dra7xx_pcie_writel(dra7xx, PCIECTRL_TI_CONF_DEVICE_TYPE,  				   DEVICE_TYPE_EP); -		ret = dra7xx_pcie_ep_unaligned_memaccess(dev); +		ret = dra7xx_pcie_unaligned_memaccess(dev);  		if (ret)  			goto err_gpio; diff --git a/drivers/pci/controller/dwc/pci-imx6.c b/drivers/pci/controller/dwc/pci-imx6.c index 4a9a673b4777..2cbef2d7c207 100644 --- a/drivers/pci/controller/dwc/pci-imx6.c +++ b/drivers/pci/controller/dwc/pci-imx6.c @@ -50,6 +50,7 @@ struct imx6_pcie {  	struct regmap		*iomuxc_gpr;  	struct reset_control	*pciephy_reset;  	struct reset_control	*apps_reset; +	struct reset_control	*turnoff_reset;  	enum imx6_pcie_variants variant;  	u32			tx_deemph_gen1;  	u32			tx_deemph_gen2_3p5db; @@ -97,6 +98,16 @@ struct imx6_pcie {  #define PORT_LOGIC_SPEED_CHANGE		(0x1 << 17)  /* PHY registers (not memory-mapped) */ +#define PCIE_PHY_ATEOVRD			0x10 +#define  PCIE_PHY_ATEOVRD_EN			(0x1 << 2) +#define  PCIE_PHY_ATEOVRD_REF_CLKDIV_SHIFT	0 +#define  PCIE_PHY_ATEOVRD_REF_CLKDIV_MASK	0x1 + +#define PCIE_PHY_MPLL_OVRD_IN_LO		0x11 +#define  PCIE_PHY_MPLL_MULTIPLIER_SHIFT		2 +#define  PCIE_PHY_MPLL_MULTIPLIER_MASK		0x7f +#define  PCIE_PHY_MPLL_MULTIPLIER_OVRD		(0x1 << 9) +  #define PCIE_PHY_RX_ASIC_OUT 0x100D  #define PCIE_PHY_RX_ASIC_OUT_VALID	(1 << 0) @@ -508,6 +519,50 @@ static void imx6_pcie_init_phy(struct imx6_pcie *imx6_pcie)  			IMX6Q_GPR12_DEVICE_TYPE, PCI_EXP_TYPE_ROOT_PORT << 12);  } +static int imx6_setup_phy_mpll(struct imx6_pcie *imx6_pcie) +{ +	unsigned long phy_rate = clk_get_rate(imx6_pcie->pcie_phy); +	int mult, div; +	u32 val; + +	switch (phy_rate) { +	case 125000000: +		/* +		 * The default settings of the MPLL are for a 125MHz input +		 * clock, so no need to reconfigure anything in that case. +		 */ +		return 0; +	case 100000000: +		mult = 25; +		div = 0; +		break; +	case 200000000: +		mult = 25; +		div = 1; +		break; +	default: +		dev_err(imx6_pcie->pci->dev, +			"Unsupported PHY reference clock rate %lu\n", phy_rate); +		return -EINVAL; +	} + +	pcie_phy_read(imx6_pcie, PCIE_PHY_MPLL_OVRD_IN_LO, &val); +	val &= ~(PCIE_PHY_MPLL_MULTIPLIER_MASK << +		 PCIE_PHY_MPLL_MULTIPLIER_SHIFT); +	val |= mult << PCIE_PHY_MPLL_MULTIPLIER_SHIFT; +	val |= PCIE_PHY_MPLL_MULTIPLIER_OVRD; +	pcie_phy_write(imx6_pcie, PCIE_PHY_MPLL_OVRD_IN_LO, val); + +	pcie_phy_read(imx6_pcie, PCIE_PHY_ATEOVRD, &val); +	val &= ~(PCIE_PHY_ATEOVRD_REF_CLKDIV_MASK << +		 PCIE_PHY_ATEOVRD_REF_CLKDIV_SHIFT); +	val |= div << PCIE_PHY_ATEOVRD_REF_CLKDIV_SHIFT; +	val |= PCIE_PHY_ATEOVRD_EN; +	pcie_phy_write(imx6_pcie, PCIE_PHY_ATEOVRD, val); + +	return 0; +} +  static int imx6_pcie_wait_for_link(struct imx6_pcie *imx6_pcie)  {  	struct dw_pcie *pci = imx6_pcie->pci; @@ -542,6 +597,24 @@ static int imx6_pcie_wait_for_speed_change(struct imx6_pcie *imx6_pcie)  	return -EINVAL;  } +static void imx6_pcie_ltssm_enable(struct device *dev) +{ +	struct imx6_pcie *imx6_pcie = dev_get_drvdata(dev); + +	switch (imx6_pcie->variant) { +	case IMX6Q: +	case IMX6SX: +	case IMX6QP: +		regmap_update_bits(imx6_pcie->iomuxc_gpr, IOMUXC_GPR12, +				   IMX6Q_GPR12_PCIE_CTL_2, +				   IMX6Q_GPR12_PCIE_CTL_2); +		break; +	case IMX7D: +		reset_control_deassert(imx6_pcie->apps_reset); +		break; +	} +} +  static int imx6_pcie_establish_link(struct imx6_pcie *imx6_pcie)  {  	struct dw_pcie *pci = imx6_pcie->pci; @@ -560,11 +633,7 @@ static int imx6_pcie_establish_link(struct imx6_pcie *imx6_pcie)  	dw_pcie_writel_dbi(pci, PCIE_RC_LCR, tmp);  	/* Start LTSSM. */ -	if (imx6_pcie->variant == IMX7D) -		reset_control_deassert(imx6_pcie->apps_reset); -	else -		regmap_update_bits(imx6_pcie->iomuxc_gpr, IOMUXC_GPR12, -				   IMX6Q_GPR12_PCIE_CTL_2, 1 << 10); +	imx6_pcie_ltssm_enable(dev);  	ret = imx6_pcie_wait_for_link(imx6_pcie);  	if (ret) @@ -632,6 +701,7 @@ static int imx6_pcie_host_init(struct pcie_port *pp)  	imx6_pcie_assert_core_reset(imx6_pcie);  	imx6_pcie_init_phy(imx6_pcie);  	imx6_pcie_deassert_core_reset(imx6_pcie); +	imx6_setup_phy_mpll(imx6_pcie);  	dw_pcie_setup_rc(pp);  	imx6_pcie_establish_link(imx6_pcie); @@ -682,6 +752,94 @@ static const struct dw_pcie_ops dw_pcie_ops = {  	.link_up = imx6_pcie_link_up,  }; +#ifdef CONFIG_PM_SLEEP +static void imx6_pcie_ltssm_disable(struct device *dev) +{ +	struct imx6_pcie *imx6_pcie = dev_get_drvdata(dev); + +	switch (imx6_pcie->variant) { +	case IMX6SX: +	case IMX6QP: +		regmap_update_bits(imx6_pcie->iomuxc_gpr, IOMUXC_GPR12, +				   IMX6Q_GPR12_PCIE_CTL_2, 0); +		break; +	case IMX7D: +		reset_control_assert(imx6_pcie->apps_reset); +		break; +	default: +		dev_err(dev, "ltssm_disable not supported\n"); +	} +} + +static void imx6_pcie_pm_turnoff(struct imx6_pcie *imx6_pcie) +{ +	reset_control_assert(imx6_pcie->turnoff_reset); +	reset_control_deassert(imx6_pcie->turnoff_reset); + +	/* +	 * Components with an upstream port must respond to +	 * PME_Turn_Off with PME_TO_Ack but we can't check. +	 * +	 * The standard recommends a 1-10ms timeout after which to +	 * proceed anyway as if acks were received. +	 */ +	usleep_range(1000, 10000); +} + +static void imx6_pcie_clk_disable(struct imx6_pcie *imx6_pcie) +{ +	clk_disable_unprepare(imx6_pcie->pcie); +	clk_disable_unprepare(imx6_pcie->pcie_phy); +	clk_disable_unprepare(imx6_pcie->pcie_bus); + +	if (imx6_pcie->variant == IMX7D) { +		regmap_update_bits(imx6_pcie->iomuxc_gpr, IOMUXC_GPR12, +				   IMX7D_GPR12_PCIE_PHY_REFCLK_SEL, +				   IMX7D_GPR12_PCIE_PHY_REFCLK_SEL); +	} +} + +static int imx6_pcie_suspend_noirq(struct device *dev) +{ +	struct imx6_pcie *imx6_pcie = dev_get_drvdata(dev); + +	if (imx6_pcie->variant != IMX7D) +		return 0; + +	imx6_pcie_pm_turnoff(imx6_pcie); +	imx6_pcie_clk_disable(imx6_pcie); +	imx6_pcie_ltssm_disable(dev); + +	return 0; +} + +static int imx6_pcie_resume_noirq(struct device *dev) +{ +	int ret; +	struct imx6_pcie *imx6_pcie = dev_get_drvdata(dev); +	struct pcie_port *pp = &imx6_pcie->pci->pp; + +	if (imx6_pcie->variant != IMX7D) +		return 0; + +	imx6_pcie_assert_core_reset(imx6_pcie); +	imx6_pcie_init_phy(imx6_pcie); +	imx6_pcie_deassert_core_reset(imx6_pcie); +	dw_pcie_setup_rc(pp); + +	ret = imx6_pcie_establish_link(imx6_pcie); +	if (ret < 0) +		dev_info(dev, "pcie link is down after resume.\n"); + +	return 0; +} +#endif + +static const struct dev_pm_ops imx6_pcie_pm_ops = { +	SET_NOIRQ_SYSTEM_SLEEP_PM_OPS(imx6_pcie_suspend_noirq, +				      imx6_pcie_resume_noirq) +}; +  static int imx6_pcie_probe(struct platform_device *pdev)  {  	struct device *dev = &pdev->dev; @@ -776,6 +934,13 @@ static int imx6_pcie_probe(struct platform_device *pdev)  		break;  	} +	/* Grab turnoff reset */ +	imx6_pcie->turnoff_reset = devm_reset_control_get_optional_exclusive(dev, "turnoff"); +	if (IS_ERR(imx6_pcie->turnoff_reset)) { +		dev_err(dev, "Failed to get TURNOFF reset control\n"); +		return PTR_ERR(imx6_pcie->turnoff_reset); +	} +  	/* Grab GPR config register range */  	imx6_pcie->iomuxc_gpr =  		 syscon_regmap_lookup_by_compatible("fsl,imx6q-iomuxc-gpr"); @@ -848,6 +1013,7 @@ static struct platform_driver imx6_pcie_driver = {  		.name	= "imx6q-pcie",  		.of_match_table = imx6_pcie_of_match,  		.suppress_bind_attrs = true, +		.pm = &imx6_pcie_pm_ops,  	},  	.probe    = imx6_pcie_probe,  	.shutdown = imx6_pcie_shutdown, diff --git a/drivers/pci/controller/dwc/pci-keystone-dw.c b/drivers/pci/controller/dwc/pci-keystone-dw.c deleted file mode 100644 index 0682213328e9..000000000000 --- a/drivers/pci/controller/dwc/pci-keystone-dw.c +++ /dev/null @@ -1,484 +0,0 @@ -// SPDX-License-Identifier: GPL-2.0 -/* - * DesignWare application register space functions for Keystone PCI controller - * - * Copyright (C) 2013-2014 Texas Instruments., Ltd. - *		http://www.ti.com - * - * Author: Murali Karicheri <m-karicheri2@ti.com> - */ - -#include <linux/irq.h> -#include <linux/irqdomain.h> -#include <linux/irqreturn.h> -#include <linux/module.h> -#include <linux/of.h> -#include <linux/of_pci.h> -#include <linux/pci.h> -#include <linux/platform_device.h> - -#include "pcie-designware.h" -#include "pci-keystone.h" - -/* Application register defines */ -#define LTSSM_EN_VAL		        1 -#define LTSSM_STATE_MASK		0x1f -#define LTSSM_STATE_L0			0x11 -#define DBI_CS2_EN_VAL			0x20 -#define OB_XLAT_EN_VAL		        2 - -/* Application registers */ -#define CMD_STATUS			0x004 -#define CFG_SETUP			0x008 -#define OB_SIZE				0x030 -#define CFG_PCIM_WIN_SZ_IDX		3 -#define CFG_PCIM_WIN_CNT		32 -#define SPACE0_REMOTE_CFG_OFFSET	0x1000 -#define OB_OFFSET_INDEX(n)		(0x200 + (8 * n)) -#define OB_OFFSET_HI(n)			(0x204 + (8 * n)) - -/* IRQ register defines */ -#define IRQ_EOI				0x050 -#define IRQ_STATUS			0x184 -#define IRQ_ENABLE_SET			0x188 -#define IRQ_ENABLE_CLR			0x18c - -#define MSI_IRQ				0x054 -#define MSI0_IRQ_STATUS			0x104 -#define MSI0_IRQ_ENABLE_SET		0x108 -#define MSI0_IRQ_ENABLE_CLR		0x10c -#define IRQ_STATUS			0x184 -#define MSI_IRQ_OFFSET			4 - -/* Error IRQ bits */ -#define ERR_AER		BIT(5)	/* ECRC error */ -#define ERR_AXI		BIT(4)	/* AXI tag lookup fatal error */ -#define ERR_CORR	BIT(3)	/* Correctable error */ -#define ERR_NONFATAL	BIT(2)	/* Non-fatal error */ -#define ERR_FATAL	BIT(1)	/* Fatal error */ -#define ERR_SYS		BIT(0)	/* System (fatal, non-fatal, or correctable) */ -#define ERR_IRQ_ALL	(ERR_AER | ERR_AXI | ERR_CORR | \ -			 ERR_NONFATAL | ERR_FATAL | ERR_SYS) -#define ERR_FATAL_IRQ	(ERR_FATAL | ERR_AXI) -#define ERR_IRQ_STATUS_RAW		0x1c0 -#define ERR_IRQ_STATUS			0x1c4 -#define ERR_IRQ_ENABLE_SET		0x1c8 -#define ERR_IRQ_ENABLE_CLR		0x1cc - -/* Config space registers */ -#define DEBUG0				0x728 - -#define to_keystone_pcie(x)	dev_get_drvdata((x)->dev) - -static inline void update_reg_offset_bit_pos(u32 offset, u32 *reg_offset, -					     u32 *bit_pos) -{ -	*reg_offset = offset % 8; -	*bit_pos = offset >> 3; -} - -phys_addr_t ks_dw_pcie_get_msi_addr(struct pcie_port *pp) -{ -	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); -	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); - -	return ks_pcie->app.start + MSI_IRQ; -} - -static u32 ks_dw_app_readl(struct keystone_pcie *ks_pcie, u32 offset) -{ -	return readl(ks_pcie->va_app_base + offset); -} - -static void ks_dw_app_writel(struct keystone_pcie *ks_pcie, u32 offset, u32 val) -{ -	writel(val, ks_pcie->va_app_base + offset); -} - -void ks_dw_pcie_handle_msi_irq(struct keystone_pcie *ks_pcie, int offset) -{ -	struct dw_pcie *pci = ks_pcie->pci; -	struct pcie_port *pp = &pci->pp; -	struct device *dev = pci->dev; -	u32 pending, vector; -	int src, virq; - -	pending = ks_dw_app_readl(ks_pcie, MSI0_IRQ_STATUS + (offset << 4)); - -	/* -	 * MSI0 status bit 0-3 shows vectors 0, 8, 16, 24, MSI1 status bit -	 * shows 1, 9, 17, 25 and so forth -	 */ -	for (src = 0; src < 4; src++) { -		if (BIT(src) & pending) { -			vector = offset + (src << 3); -			virq = irq_linear_revmap(pp->irq_domain, vector); -			dev_dbg(dev, "irq: bit %d, vector %d, virq %d\n", -				src, vector, virq); -			generic_handle_irq(virq); -		} -	} -} - -void ks_dw_pcie_msi_irq_ack(int irq, struct pcie_port *pp) -{ -	u32 reg_offset, bit_pos; -	struct keystone_pcie *ks_pcie; -	struct dw_pcie *pci; - -	pci = to_dw_pcie_from_pp(pp); -	ks_pcie = to_keystone_pcie(pci); -	update_reg_offset_bit_pos(irq, ®_offset, &bit_pos); - -	ks_dw_app_writel(ks_pcie, MSI0_IRQ_STATUS + (reg_offset << 4), -			 BIT(bit_pos)); -	ks_dw_app_writel(ks_pcie, IRQ_EOI, reg_offset + MSI_IRQ_OFFSET); -} - -void ks_dw_pcie_msi_set_irq(struct pcie_port *pp, int irq) -{ -	u32 reg_offset, bit_pos; -	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); -	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); - -	update_reg_offset_bit_pos(irq, ®_offset, &bit_pos); -	ks_dw_app_writel(ks_pcie, MSI0_IRQ_ENABLE_SET + (reg_offset << 4), -			 BIT(bit_pos)); -} - -void ks_dw_pcie_msi_clear_irq(struct pcie_port *pp, int irq) -{ -	u32 reg_offset, bit_pos; -	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); -	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); - -	update_reg_offset_bit_pos(irq, ®_offset, &bit_pos); -	ks_dw_app_writel(ks_pcie, MSI0_IRQ_ENABLE_CLR + (reg_offset << 4), -			 BIT(bit_pos)); -} - -int ks_dw_pcie_msi_host_init(struct pcie_port *pp) -{ -	return dw_pcie_allocate_domains(pp); -} - -void ks_dw_pcie_enable_legacy_irqs(struct keystone_pcie *ks_pcie) -{ -	int i; - -	for (i = 0; i < PCI_NUM_INTX; i++) -		ks_dw_app_writel(ks_pcie, IRQ_ENABLE_SET + (i << 4), 0x1); -} - -void ks_dw_pcie_handle_legacy_irq(struct keystone_pcie *ks_pcie, int offset) -{ -	struct dw_pcie *pci = ks_pcie->pci; -	struct device *dev = pci->dev; -	u32 pending; -	int virq; - -	pending = ks_dw_app_readl(ks_pcie, IRQ_STATUS + (offset << 4)); - -	if (BIT(0) & pending) { -		virq = irq_linear_revmap(ks_pcie->legacy_irq_domain, offset); -		dev_dbg(dev, ": irq: irq_offset %d, virq %d\n", offset, virq); -		generic_handle_irq(virq); -	} - -	/* EOI the INTx interrupt */ -	ks_dw_app_writel(ks_pcie, IRQ_EOI, offset); -} - -void ks_dw_pcie_enable_error_irq(struct keystone_pcie *ks_pcie) -{ -	ks_dw_app_writel(ks_pcie, ERR_IRQ_ENABLE_SET, ERR_IRQ_ALL); -} - -irqreturn_t ks_dw_pcie_handle_error_irq(struct keystone_pcie *ks_pcie) -{ -	u32 status; - -	status = ks_dw_app_readl(ks_pcie, ERR_IRQ_STATUS_RAW) & ERR_IRQ_ALL; -	if (!status) -		return IRQ_NONE; - -	if (status & ERR_FATAL_IRQ) -		dev_err(ks_pcie->pci->dev, "fatal error (status %#010x)\n", -			status); - -	/* Ack the IRQ; status bits are RW1C */ -	ks_dw_app_writel(ks_pcie, ERR_IRQ_STATUS, status); -	return IRQ_HANDLED; -} - -static void ks_dw_pcie_ack_legacy_irq(struct irq_data *d) -{ -} - -static void ks_dw_pcie_mask_legacy_irq(struct irq_data *d) -{ -} - -static void ks_dw_pcie_unmask_legacy_irq(struct irq_data *d) -{ -} - -static struct irq_chip ks_dw_pcie_legacy_irq_chip = { -	.name = "Keystone-PCI-Legacy-IRQ", -	.irq_ack = ks_dw_pcie_ack_legacy_irq, -	.irq_mask = ks_dw_pcie_mask_legacy_irq, -	.irq_unmask = ks_dw_pcie_unmask_legacy_irq, -}; - -static int ks_dw_pcie_init_legacy_irq_map(struct irq_domain *d, -				unsigned int irq, irq_hw_number_t hw_irq) -{ -	irq_set_chip_and_handler(irq, &ks_dw_pcie_legacy_irq_chip, -				 handle_level_irq); -	irq_set_chip_data(irq, d->host_data); - -	return 0; -} - -static const struct irq_domain_ops ks_dw_pcie_legacy_irq_domain_ops = { -	.map = ks_dw_pcie_init_legacy_irq_map, -	.xlate = irq_domain_xlate_onetwocell, -}; - -/** - * ks_dw_pcie_set_dbi_mode() - Set DBI mode to access overlaid BAR mask - * registers - * - * Since modification of dbi_cs2 involves different clock domain, read the - * status back to ensure the transition is complete. - */ -static void ks_dw_pcie_set_dbi_mode(struct keystone_pcie *ks_pcie) -{ -	u32 val; - -	val = ks_dw_app_readl(ks_pcie, CMD_STATUS); -	ks_dw_app_writel(ks_pcie, CMD_STATUS, DBI_CS2_EN_VAL | val); - -	do { -		val = ks_dw_app_readl(ks_pcie, CMD_STATUS); -	} while (!(val & DBI_CS2_EN_VAL)); -} - -/** - * ks_dw_pcie_clear_dbi_mode() - Disable DBI mode - * - * Since modification of dbi_cs2 involves different clock domain, read the - * status back to ensure the transition is complete. - */ -static void ks_dw_pcie_clear_dbi_mode(struct keystone_pcie *ks_pcie) -{ -	u32 val; - -	val = ks_dw_app_readl(ks_pcie, CMD_STATUS); -	ks_dw_app_writel(ks_pcie, CMD_STATUS, ~DBI_CS2_EN_VAL & val); - -	do { -		val = ks_dw_app_readl(ks_pcie, CMD_STATUS); -	} while (val & DBI_CS2_EN_VAL); -} - -void ks_dw_pcie_setup_rc_app_regs(struct keystone_pcie *ks_pcie) -{ -	struct dw_pcie *pci = ks_pcie->pci; -	struct pcie_port *pp = &pci->pp; -	u32 start = pp->mem->start, end = pp->mem->end; -	int i, tr_size; -	u32 val; - -	/* Disable BARs for inbound access */ -	ks_dw_pcie_set_dbi_mode(ks_pcie); -	dw_pcie_writel_dbi(pci, PCI_BASE_ADDRESS_0, 0); -	dw_pcie_writel_dbi(pci, PCI_BASE_ADDRESS_1, 0); -	ks_dw_pcie_clear_dbi_mode(ks_pcie); - -	/* Set outbound translation size per window division */ -	ks_dw_app_writel(ks_pcie, OB_SIZE, CFG_PCIM_WIN_SZ_IDX & 0x7); - -	tr_size = (1 << (CFG_PCIM_WIN_SZ_IDX & 0x7)) * SZ_1M; - -	/* Using Direct 1:1 mapping of RC <-> PCI memory space */ -	for (i = 0; (i < CFG_PCIM_WIN_CNT) && (start < end); i++) { -		ks_dw_app_writel(ks_pcie, OB_OFFSET_INDEX(i), start | 1); -		ks_dw_app_writel(ks_pcie, OB_OFFSET_HI(i), 0); -		start += tr_size; -	} - -	/* Enable OB translation */ -	val = ks_dw_app_readl(ks_pcie, CMD_STATUS); -	ks_dw_app_writel(ks_pcie, CMD_STATUS, OB_XLAT_EN_VAL | val); -} - -/** - * ks_pcie_cfg_setup() - Set up configuration space address for a device - * - * @ks_pcie: ptr to keystone_pcie structure - * @bus: Bus number the device is residing on - * @devfn: device, function number info - * - * Forms and returns the address of configuration space mapped in PCIESS - * address space 0.  Also configures CFG_SETUP for remote configuration space - * access. - * - * The address space has two regions to access configuration - local and remote. - * We access local region for bus 0 (as RC is attached on bus 0) and remote - * region for others with TYPE 1 access when bus > 1.  As for device on bus = 1, - * we will do TYPE 0 access as it will be on our secondary bus (logical). - * CFG_SETUP is needed only for remote configuration access. - */ -static void __iomem *ks_pcie_cfg_setup(struct keystone_pcie *ks_pcie, u8 bus, -				       unsigned int devfn) -{ -	u8 device = PCI_SLOT(devfn), function = PCI_FUNC(devfn); -	struct dw_pcie *pci = ks_pcie->pci; -	struct pcie_port *pp = &pci->pp; -	u32 regval; - -	if (bus == 0) -		return pci->dbi_base; - -	regval = (bus << 16) | (device << 8) | function; - -	/* -	 * Since Bus#1 will be a virtual bus, we need to have TYPE0 -	 * access only. -	 * TYPE 1 -	 */ -	if (bus != 1) -		regval |= BIT(24); - -	ks_dw_app_writel(ks_pcie, CFG_SETUP, regval); -	return pp->va_cfg0_base; -} - -int ks_dw_pcie_rd_other_conf(struct pcie_port *pp, struct pci_bus *bus, -			     unsigned int devfn, int where, int size, u32 *val) -{ -	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); -	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); -	u8 bus_num = bus->number; -	void __iomem *addr; - -	addr = ks_pcie_cfg_setup(ks_pcie, bus_num, devfn); - -	return dw_pcie_read(addr + where, size, val); -} - -int ks_dw_pcie_wr_other_conf(struct pcie_port *pp, struct pci_bus *bus, -			     unsigned int devfn, int where, int size, u32 val) -{ -	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); -	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); -	u8 bus_num = bus->number; -	void __iomem *addr; - -	addr = ks_pcie_cfg_setup(ks_pcie, bus_num, devfn); - -	return dw_pcie_write(addr + where, size, val); -} - -/** - * ks_dw_pcie_v3_65_scan_bus() - keystone scan_bus post initialization - * - * This sets BAR0 to enable inbound access for MSI_IRQ register - */ -void ks_dw_pcie_v3_65_scan_bus(struct pcie_port *pp) -{ -	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); -	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); - -	/* Configure and set up BAR0 */ -	ks_dw_pcie_set_dbi_mode(ks_pcie); - -	/* Enable BAR0 */ -	dw_pcie_writel_dbi(pci, PCI_BASE_ADDRESS_0, 1); -	dw_pcie_writel_dbi(pci, PCI_BASE_ADDRESS_0, SZ_4K - 1); - -	ks_dw_pcie_clear_dbi_mode(ks_pcie); - -	 /* -	  * For BAR0, just setting bus address for inbound writes (MSI) should -	  * be sufficient.  Use physical address to avoid any conflicts. -	  */ -	dw_pcie_writel_dbi(pci, PCI_BASE_ADDRESS_0, ks_pcie->app.start); -} - -/** - * ks_dw_pcie_link_up() - Check if link up - */ -int ks_dw_pcie_link_up(struct dw_pcie *pci) -{ -	u32 val; - -	val = dw_pcie_readl_dbi(pci, DEBUG0); -	return (val & LTSSM_STATE_MASK) == LTSSM_STATE_L0; -} - -void ks_dw_pcie_initiate_link_train(struct keystone_pcie *ks_pcie) -{ -	u32 val; - -	/* Disable Link training */ -	val = ks_dw_app_readl(ks_pcie, CMD_STATUS); -	val &= ~LTSSM_EN_VAL; -	ks_dw_app_writel(ks_pcie, CMD_STATUS, LTSSM_EN_VAL | val); - -	/* Initiate Link Training */ -	val = ks_dw_app_readl(ks_pcie, CMD_STATUS); -	ks_dw_app_writel(ks_pcie, CMD_STATUS, LTSSM_EN_VAL | val); -} - -/** - * ks_dw_pcie_host_init() - initialize host for v3_65 dw hardware - * - * Ioremap the register resources, initialize legacy irq domain - * and call dw_pcie_v3_65_host_init() API to initialize the Keystone - * PCI host controller. - */ -int __init ks_dw_pcie_host_init(struct keystone_pcie *ks_pcie, -				struct device_node *msi_intc_np) -{ -	struct dw_pcie *pci = ks_pcie->pci; -	struct pcie_port *pp = &pci->pp; -	struct device *dev = pci->dev; -	struct platform_device *pdev = to_platform_device(dev); -	struct resource *res; - -	/* Index 0 is the config reg. space address */ -	res = platform_get_resource(pdev, IORESOURCE_MEM, 0); -	pci->dbi_base = devm_pci_remap_cfg_resource(dev, res); -	if (IS_ERR(pci->dbi_base)) -		return PTR_ERR(pci->dbi_base); - -	/* -	 * We set these same and is used in pcie rd/wr_other_conf -	 * functions -	 */ -	pp->va_cfg0_base = pci->dbi_base + SPACE0_REMOTE_CFG_OFFSET; -	pp->va_cfg1_base = pp->va_cfg0_base; - -	/* Index 1 is the application reg. space address */ -	res = platform_get_resource(pdev, IORESOURCE_MEM, 1); -	ks_pcie->va_app_base = devm_ioremap_resource(dev, res); -	if (IS_ERR(ks_pcie->va_app_base)) -		return PTR_ERR(ks_pcie->va_app_base); - -	ks_pcie->app = *res; - -	/* Create legacy IRQ domain */ -	ks_pcie->legacy_irq_domain = -			irq_domain_add_linear(ks_pcie->legacy_intc_np, -					PCI_NUM_INTX, -					&ks_dw_pcie_legacy_irq_domain_ops, -					NULL); -	if (!ks_pcie->legacy_irq_domain) { -		dev_err(dev, "Failed to add irq domain for legacy irqs\n"); -		return -EINVAL; -	} - -	return dw_pcie_host_init(pp); -} diff --git a/drivers/pci/controller/dwc/pci-keystone.c b/drivers/pci/controller/dwc/pci-keystone.c index e88bd221fffe..14f2b0b4ed5e 100644 --- a/drivers/pci/controller/dwc/pci-keystone.c +++ b/drivers/pci/controller/dwc/pci-keystone.c @@ -9,40 +9,510 @@   * Implementation based on pci-exynos.c and pcie-designware.c   */ -#include <linux/irqchip/chained_irq.h>  #include <linux/clk.h>  #include <linux/delay.h> +#include <linux/init.h>  #include <linux/interrupt.h> +#include <linux/irqchip/chained_irq.h>  #include <linux/irqdomain.h> -#include <linux/init.h> +#include <linux/mfd/syscon.h>  #include <linux/msi.h> -#include <linux/of_irq.h>  #include <linux/of.h> +#include <linux/of_irq.h>  #include <linux/of_pci.h> -#include <linux/platform_device.h>  #include <linux/phy/phy.h> +#include <linux/platform_device.h> +#include <linux/regmap.h>  #include <linux/resource.h>  #include <linux/signal.h>  #include "pcie-designware.h" -#include "pci-keystone.h" -#define DRIVER_NAME	"keystone-pcie" +#define PCIE_VENDORID_MASK	0xffff +#define PCIE_DEVICEID_SHIFT	16 + +/* Application registers */ +#define CMD_STATUS			0x004 +#define LTSSM_EN_VAL		        BIT(0) +#define OB_XLAT_EN_VAL		        BIT(1) +#define DBI_CS2				BIT(5) + +#define CFG_SETUP			0x008 +#define CFG_BUS(x)			(((x) & 0xff) << 16) +#define CFG_DEVICE(x)			(((x) & 0x1f) << 8) +#define CFG_FUNC(x)			((x) & 0x7) +#define CFG_TYPE1			BIT(24) + +#define OB_SIZE				0x030 +#define SPACE0_REMOTE_CFG_OFFSET	0x1000 +#define OB_OFFSET_INDEX(n)		(0x200 + (8 * (n))) +#define OB_OFFSET_HI(n)			(0x204 + (8 * (n))) +#define OB_ENABLEN			BIT(0) +#define OB_WIN_SIZE			8	/* 8MB */ + +/* IRQ register defines */ +#define IRQ_EOI				0x050 +#define IRQ_STATUS			0x184 +#define IRQ_ENABLE_SET			0x188 +#define IRQ_ENABLE_CLR			0x18c + +#define MSI_IRQ				0x054 +#define MSI0_IRQ_STATUS			0x104 +#define MSI0_IRQ_ENABLE_SET		0x108 +#define MSI0_IRQ_ENABLE_CLR		0x10c +#define IRQ_STATUS			0x184 +#define MSI_IRQ_OFFSET			4 + +#define ERR_IRQ_STATUS			0x1c4 +#define ERR_IRQ_ENABLE_SET		0x1c8 +#define ERR_AER				BIT(5)	/* ECRC error */ +#define ERR_AXI				BIT(4)	/* AXI tag lookup fatal error */ +#define ERR_CORR			BIT(3)	/* Correctable error */ +#define ERR_NONFATAL			BIT(2)	/* Non-fatal error */ +#define ERR_FATAL			BIT(1)	/* Fatal error */ +#define ERR_SYS				BIT(0)	/* System error */ +#define ERR_IRQ_ALL			(ERR_AER | ERR_AXI | ERR_CORR | \ +					 ERR_NONFATAL | ERR_FATAL | ERR_SYS) + +#define MAX_MSI_HOST_IRQS		8 +/* PCIE controller device IDs */ +#define PCIE_RC_K2HK			0xb008 +#define PCIE_RC_K2E			0xb009 +#define PCIE_RC_K2L			0xb00a +#define PCIE_RC_K2G			0xb00b + +#define to_keystone_pcie(x)		dev_get_drvdata((x)->dev) + +struct keystone_pcie { +	struct dw_pcie		*pci; +	/* PCI Device ID */ +	u32			device_id; +	int			num_legacy_host_irqs; +	int			legacy_host_irqs[PCI_NUM_INTX]; +	struct			device_node *legacy_intc_np; + +	int			num_msi_host_irqs; +	int			msi_host_irqs[MAX_MSI_HOST_IRQS]; +	int			num_lanes; +	u32			num_viewport; +	struct phy		**phy; +	struct device_link	**link; +	struct			device_node *msi_intc_np; +	struct irq_domain	*legacy_irq_domain; +	struct device_node	*np; + +	int error_irq; + +	/* Application register space */ +	void __iomem		*va_app_base;	/* DT 1st resource */ +	struct resource		app; +}; -/* DEV_STAT_CTRL */ -#define PCIE_CAP_BASE		0x70 +static inline void update_reg_offset_bit_pos(u32 offset, u32 *reg_offset, +					     u32 *bit_pos) +{ +	*reg_offset = offset % 8; +	*bit_pos = offset >> 3; +} -/* PCIE controller device IDs */ -#define PCIE_RC_K2HK		0xb008 -#define PCIE_RC_K2E		0xb009 -#define PCIE_RC_K2L		0xb00a +static phys_addr_t ks_pcie_get_msi_addr(struct pcie_port *pp) +{ +	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); +	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); + +	return ks_pcie->app.start + MSI_IRQ; +} + +static u32 ks_pcie_app_readl(struct keystone_pcie *ks_pcie, u32 offset) +{ +	return readl(ks_pcie->va_app_base + offset); +} + +static void ks_pcie_app_writel(struct keystone_pcie *ks_pcie, u32 offset, +			       u32 val) +{ +	writel(val, ks_pcie->va_app_base + offset); +} + +static void ks_pcie_handle_msi_irq(struct keystone_pcie *ks_pcie, int offset) +{ +	struct dw_pcie *pci = ks_pcie->pci; +	struct pcie_port *pp = &pci->pp; +	struct device *dev = pci->dev; +	u32 pending, vector; +	int src, virq; + +	pending = ks_pcie_app_readl(ks_pcie, MSI0_IRQ_STATUS + (offset << 4)); + +	/* +	 * MSI0 status bit 0-3 shows vectors 0, 8, 16, 24, MSI1 status bit +	 * shows 1, 9, 17, 25 and so forth +	 */ +	for (src = 0; src < 4; src++) { +		if (BIT(src) & pending) { +			vector = offset + (src << 3); +			virq = irq_linear_revmap(pp->irq_domain, vector); +			dev_dbg(dev, "irq: bit %d, vector %d, virq %d\n", +				src, vector, virq); +			generic_handle_irq(virq); +		} +	} +} + +static void ks_pcie_msi_irq_ack(int irq, struct pcie_port *pp) +{ +	u32 reg_offset, bit_pos; +	struct keystone_pcie *ks_pcie; +	struct dw_pcie *pci; + +	pci = to_dw_pcie_from_pp(pp); +	ks_pcie = to_keystone_pcie(pci); +	update_reg_offset_bit_pos(irq, ®_offset, &bit_pos); + +	ks_pcie_app_writel(ks_pcie, MSI0_IRQ_STATUS + (reg_offset << 4), +			   BIT(bit_pos)); +	ks_pcie_app_writel(ks_pcie, IRQ_EOI, reg_offset + MSI_IRQ_OFFSET); +} + +static void ks_pcie_msi_set_irq(struct pcie_port *pp, int irq) +{ +	u32 reg_offset, bit_pos; +	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); +	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); + +	update_reg_offset_bit_pos(irq, ®_offset, &bit_pos); +	ks_pcie_app_writel(ks_pcie, MSI0_IRQ_ENABLE_SET + (reg_offset << 4), +			   BIT(bit_pos)); +} + +static void ks_pcie_msi_clear_irq(struct pcie_port *pp, int irq) +{ +	u32 reg_offset, bit_pos; +	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); +	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); + +	update_reg_offset_bit_pos(irq, ®_offset, &bit_pos); +	ks_pcie_app_writel(ks_pcie, MSI0_IRQ_ENABLE_CLR + (reg_offset << 4), +			   BIT(bit_pos)); +} + +static int ks_pcie_msi_host_init(struct pcie_port *pp) +{ +	return dw_pcie_allocate_domains(pp); +} + +static void ks_pcie_enable_legacy_irqs(struct keystone_pcie *ks_pcie) +{ +	int i; + +	for (i = 0; i < PCI_NUM_INTX; i++) +		ks_pcie_app_writel(ks_pcie, IRQ_ENABLE_SET + (i << 4), 0x1); +} + +static void ks_pcie_handle_legacy_irq(struct keystone_pcie *ks_pcie, +				      int offset) +{ +	struct dw_pcie *pci = ks_pcie->pci; +	struct device *dev = pci->dev; +	u32 pending; +	int virq; + +	pending = ks_pcie_app_readl(ks_pcie, IRQ_STATUS + (offset << 4)); + +	if (BIT(0) & pending) { +		virq = irq_linear_revmap(ks_pcie->legacy_irq_domain, offset); +		dev_dbg(dev, ": irq: irq_offset %d, virq %d\n", offset, virq); +		generic_handle_irq(virq); +	} + +	/* EOI the INTx interrupt */ +	ks_pcie_app_writel(ks_pcie, IRQ_EOI, offset); +} + +static void ks_pcie_enable_error_irq(struct keystone_pcie *ks_pcie) +{ +	ks_pcie_app_writel(ks_pcie, ERR_IRQ_ENABLE_SET, ERR_IRQ_ALL); +} + +static irqreturn_t ks_pcie_handle_error_irq(struct keystone_pcie *ks_pcie) +{ +	u32 reg; +	struct device *dev = ks_pcie->pci->dev; + +	reg = ks_pcie_app_readl(ks_pcie, ERR_IRQ_STATUS); +	if (!reg) +		return IRQ_NONE; + +	if (reg & ERR_SYS) +		dev_err(dev, "System Error\n"); + +	if (reg & ERR_FATAL) +		dev_err(dev, "Fatal Error\n"); + +	if (reg & ERR_NONFATAL) +		dev_dbg(dev, "Non Fatal Error\n"); + +	if (reg & ERR_CORR) +		dev_dbg(dev, "Correctable Error\n"); + +	if (reg & ERR_AXI) +		dev_err(dev, "AXI tag lookup fatal Error\n"); + +	if (reg & ERR_AER) +		dev_err(dev, "ECRC Error\n"); + +	ks_pcie_app_writel(ks_pcie, ERR_IRQ_STATUS, reg); + +	return IRQ_HANDLED; +} + +static void ks_pcie_ack_legacy_irq(struct irq_data *d) +{ +} + +static void ks_pcie_mask_legacy_irq(struct irq_data *d) +{ +} + +static void ks_pcie_unmask_legacy_irq(struct irq_data *d) +{ +} + +static struct irq_chip ks_pcie_legacy_irq_chip = { +	.name = "Keystone-PCI-Legacy-IRQ", +	.irq_ack = ks_pcie_ack_legacy_irq, +	.irq_mask = ks_pcie_mask_legacy_irq, +	.irq_unmask = ks_pcie_unmask_legacy_irq, +}; + +static int ks_pcie_init_legacy_irq_map(struct irq_domain *d, +				       unsigned int irq, +				       irq_hw_number_t hw_irq) +{ +	irq_set_chip_and_handler(irq, &ks_pcie_legacy_irq_chip, +				 handle_level_irq); +	irq_set_chip_data(irq, d->host_data); + +	return 0; +} + +static const struct irq_domain_ops ks_pcie_legacy_irq_domain_ops = { +	.map = ks_pcie_init_legacy_irq_map, +	.xlate = irq_domain_xlate_onetwocell, +}; + +/** + * ks_pcie_set_dbi_mode() - Set DBI mode to access overlaid BAR mask + * registers + * + * Since modification of dbi_cs2 involves different clock domain, read the + * status back to ensure the transition is complete. + */ +static void ks_pcie_set_dbi_mode(struct keystone_pcie *ks_pcie) +{ +	u32 val; + +	val = ks_pcie_app_readl(ks_pcie, CMD_STATUS); +	val |= DBI_CS2; +	ks_pcie_app_writel(ks_pcie, CMD_STATUS, val); + +	do { +		val = ks_pcie_app_readl(ks_pcie, CMD_STATUS); +	} while (!(val & DBI_CS2)); +} + +/** + * ks_pcie_clear_dbi_mode() - Disable DBI mode + * + * Since modification of dbi_cs2 involves different clock domain, read the + * status back to ensure the transition is complete. + */ +static void ks_pcie_clear_dbi_mode(struct keystone_pcie *ks_pcie) +{ +	u32 val; + +	val = ks_pcie_app_readl(ks_pcie, CMD_STATUS); +	val &= ~DBI_CS2; +	ks_pcie_app_writel(ks_pcie, CMD_STATUS, val); + +	do { +		val = ks_pcie_app_readl(ks_pcie, CMD_STATUS); +	} while (val & DBI_CS2); +} + +static void ks_pcie_setup_rc_app_regs(struct keystone_pcie *ks_pcie) +{ +	u32 val; +	u32 num_viewport = ks_pcie->num_viewport; +	struct dw_pcie *pci = ks_pcie->pci; +	struct pcie_port *pp = &pci->pp; +	u64 start = pp->mem->start; +	u64 end = pp->mem->end; +	int i; + +	/* Disable BARs for inbound access */ +	ks_pcie_set_dbi_mode(ks_pcie); +	dw_pcie_writel_dbi(pci, PCI_BASE_ADDRESS_0, 0); +	dw_pcie_writel_dbi(pci, PCI_BASE_ADDRESS_1, 0); +	ks_pcie_clear_dbi_mode(ks_pcie); + +	val = ilog2(OB_WIN_SIZE); +	ks_pcie_app_writel(ks_pcie, OB_SIZE, val); + +	/* Using Direct 1:1 mapping of RC <-> PCI memory space */ +	for (i = 0; i < num_viewport && (start < end); i++) { +		ks_pcie_app_writel(ks_pcie, OB_OFFSET_INDEX(i), +				   lower_32_bits(start) | OB_ENABLEN); +		ks_pcie_app_writel(ks_pcie, OB_OFFSET_HI(i), +				   upper_32_bits(start)); +		start += OB_WIN_SIZE; +	} + +	val = ks_pcie_app_readl(ks_pcie, CMD_STATUS); +	val |= OB_XLAT_EN_VAL; +	ks_pcie_app_writel(ks_pcie, CMD_STATUS, val); +} + +static int ks_pcie_rd_other_conf(struct pcie_port *pp, struct pci_bus *bus, +				 unsigned int devfn, int where, int size, +				 u32 *val) +{ +	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); +	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); +	u32 reg; + +	reg = CFG_BUS(bus->number) | CFG_DEVICE(PCI_SLOT(devfn)) | +		CFG_FUNC(PCI_FUNC(devfn)); +	if (bus->parent->number != pp->root_bus_nr) +		reg |= CFG_TYPE1; +	ks_pcie_app_writel(ks_pcie, CFG_SETUP, reg); + +	return dw_pcie_read(pp->va_cfg0_base + where, size, val); +} + +static int ks_pcie_wr_other_conf(struct pcie_port *pp, struct pci_bus *bus, +				 unsigned int devfn, int where, int size, +				 u32 val) +{ +	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); +	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); +	u32 reg; + +	reg = CFG_BUS(bus->number) | CFG_DEVICE(PCI_SLOT(devfn)) | +		CFG_FUNC(PCI_FUNC(devfn)); +	if (bus->parent->number != pp->root_bus_nr) +		reg |= CFG_TYPE1; +	ks_pcie_app_writel(ks_pcie, CFG_SETUP, reg); + +	return dw_pcie_write(pp->va_cfg0_base + where, size, val); +} + +/** + * ks_pcie_v3_65_scan_bus() - keystone scan_bus post initialization + * + * This sets BAR0 to enable inbound access for MSI_IRQ register + */ +static void ks_pcie_v3_65_scan_bus(struct pcie_port *pp) +{ +	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); +	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); + +	/* Configure and set up BAR0 */ +	ks_pcie_set_dbi_mode(ks_pcie); + +	/* Enable BAR0 */ +	dw_pcie_writel_dbi(pci, PCI_BASE_ADDRESS_0, 1); +	dw_pcie_writel_dbi(pci, PCI_BASE_ADDRESS_0, SZ_4K - 1); + +	ks_pcie_clear_dbi_mode(ks_pcie); + +	 /* +	  * For BAR0, just setting bus address for inbound writes (MSI) should +	  * be sufficient.  Use physical address to avoid any conflicts. +	  */ +	dw_pcie_writel_dbi(pci, PCI_BASE_ADDRESS_0, ks_pcie->app.start); +} + +/** + * ks_pcie_link_up() - Check if link up + */ +static int ks_pcie_link_up(struct dw_pcie *pci) +{ +	u32 val; + +	val = dw_pcie_readl_dbi(pci, PCIE_PORT_DEBUG0); +	val &= PORT_LOGIC_LTSSM_STATE_MASK; +	return (val == PORT_LOGIC_LTSSM_STATE_L0); +} + +static void ks_pcie_initiate_link_train(struct keystone_pcie *ks_pcie) +{ +	u32 val; -#define to_keystone_pcie(x)	dev_get_drvdata((x)->dev) +	/* Disable Link training */ +	val = ks_pcie_app_readl(ks_pcie, CMD_STATUS); +	val &= ~LTSSM_EN_VAL; +	ks_pcie_app_writel(ks_pcie, CMD_STATUS, LTSSM_EN_VAL | val); -static void quirk_limit_mrrs(struct pci_dev *dev) +	/* Initiate Link Training */ +	val = ks_pcie_app_readl(ks_pcie, CMD_STATUS); +	ks_pcie_app_writel(ks_pcie, CMD_STATUS, LTSSM_EN_VAL | val); +} + +/** + * ks_pcie_dw_host_init() - initialize host for v3_65 dw hardware + * + * Ioremap the register resources, initialize legacy irq domain + * and call dw_pcie_v3_65_host_init() API to initialize the Keystone + * PCI host controller. + */ +static int __init ks_pcie_dw_host_init(struct keystone_pcie *ks_pcie) +{ +	struct dw_pcie *pci = ks_pcie->pci; +	struct pcie_port *pp = &pci->pp; +	struct device *dev = pci->dev; +	struct platform_device *pdev = to_platform_device(dev); +	struct resource *res; + +	/* Index 0 is the config reg. space address */ +	res = platform_get_resource(pdev, IORESOURCE_MEM, 0); +	pci->dbi_base = devm_pci_remap_cfg_resource(dev, res); +	if (IS_ERR(pci->dbi_base)) +		return PTR_ERR(pci->dbi_base); + +	/* +	 * We set these same and is used in pcie rd/wr_other_conf +	 * functions +	 */ +	pp->va_cfg0_base = pci->dbi_base + SPACE0_REMOTE_CFG_OFFSET; +	pp->va_cfg1_base = pp->va_cfg0_base; + +	/* Index 1 is the application reg. space address */ +	res = platform_get_resource(pdev, IORESOURCE_MEM, 1); +	ks_pcie->va_app_base = devm_ioremap_resource(dev, res); +	if (IS_ERR(ks_pcie->va_app_base)) +		return PTR_ERR(ks_pcie->va_app_base); + +	ks_pcie->app = *res; + +	/* Create legacy IRQ domain */ +	ks_pcie->legacy_irq_domain = +			irq_domain_add_linear(ks_pcie->legacy_intc_np, +					      PCI_NUM_INTX, +					      &ks_pcie_legacy_irq_domain_ops, +					      NULL); +	if (!ks_pcie->legacy_irq_domain) { +		dev_err(dev, "Failed to add irq domain for legacy irqs\n"); +		return -EINVAL; +	} + +	return dw_pcie_host_init(pp); +} + +static void ks_pcie_quirk(struct pci_dev *dev)  {  	struct pci_bus *bus = dev->bus; -	struct pci_dev *bridge = bus->self; +	struct pci_dev *bridge;  	static const struct pci_device_id rc_pci_devids[] = {  		{ PCI_DEVICE(PCI_VENDOR_ID_TI, PCIE_RC_K2HK),  		 .class = PCI_CLASS_BRIDGE_PCI << 8, .class_mask = ~0, }, @@ -50,11 +520,13 @@ static void quirk_limit_mrrs(struct pci_dev *dev)  		 .class = PCI_CLASS_BRIDGE_PCI << 8, .class_mask = ~0, },  		{ PCI_DEVICE(PCI_VENDOR_ID_TI, PCIE_RC_K2L),  		 .class = PCI_CLASS_BRIDGE_PCI << 8, .class_mask = ~0, }, +		{ PCI_DEVICE(PCI_VENDOR_ID_TI, PCIE_RC_K2G), +		 .class = PCI_CLASS_BRIDGE_PCI << 8, .class_mask = ~0, },  		{ 0, },  	};  	if (pci_is_root_bus(bus)) -		return; +		bridge = dev;  	/* look for the host bridge */  	while (!pci_is_root_bus(bus)) { @@ -62,43 +534,39 @@ static void quirk_limit_mrrs(struct pci_dev *dev)  		bus = bus->parent;  	} -	if (bridge) { -		/* -		 * Keystone PCI controller has a h/w limitation of -		 * 256 bytes maximum read request size.  It can't handle -		 * anything higher than this.  So force this limit on -		 * all downstream devices. -		 */ -		if (pci_match_id(rc_pci_devids, bridge)) { -			if (pcie_get_readrq(dev) > 256) { -				dev_info(&dev->dev, "limiting MRRS to 256\n"); -				pcie_set_readrq(dev, 256); -			} +	if (!bridge) +		return; + +	/* +	 * Keystone PCI controller has a h/w limitation of +	 * 256 bytes maximum read request size.  It can't handle +	 * anything higher than this.  So force this limit on +	 * all downstream devices. +	 */ +	if (pci_match_id(rc_pci_devids, bridge)) { +		if (pcie_get_readrq(dev) > 256) { +			dev_info(&dev->dev, "limiting MRRS to 256\n"); +			pcie_set_readrq(dev, 256);  		}  	}  } -DECLARE_PCI_FIXUP_ENABLE(PCI_ANY_ID, PCI_ANY_ID, quirk_limit_mrrs); +DECLARE_PCI_FIXUP_ENABLE(PCI_ANY_ID, PCI_ANY_ID, ks_pcie_quirk);  static int ks_pcie_establish_link(struct keystone_pcie *ks_pcie)  {  	struct dw_pcie *pci = ks_pcie->pci; -	struct pcie_port *pp = &pci->pp;  	struct device *dev = pci->dev; -	unsigned int retries; - -	dw_pcie_setup_rc(pp);  	if (dw_pcie_link_up(pci)) {  		dev_info(dev, "Link already up\n");  		return 0;  	} +	ks_pcie_initiate_link_train(ks_pcie); +  	/* check if the link is up or not */ -	for (retries = 0; retries < 5; retries++) { -		ks_dw_pcie_initiate_link_train(ks_pcie); -		if (!dw_pcie_wait_for_link(pci)) -			return 0; -	} +	if (!dw_pcie_wait_for_link(pci)) +		return 0;  	dev_err(dev, "phy link never came up\n");  	return -ETIMEDOUT; @@ -121,7 +589,7 @@ static void ks_pcie_msi_irq_handler(struct irq_desc *desc)  	 * ack operation.  	 */  	chained_irq_enter(chip, desc); -	ks_dw_pcie_handle_msi_irq(ks_pcie, offset); +	ks_pcie_handle_msi_irq(ks_pcie, offset);  	chained_irq_exit(chip, desc);  } @@ -150,7 +618,7 @@ static void ks_pcie_legacy_irq_handler(struct irq_desc *desc)  	 * ack operation.  	 */  	chained_irq_enter(chip, desc); -	ks_dw_pcie_handle_legacy_irq(ks_pcie, irq_offset); +	ks_pcie_handle_legacy_irq(ks_pcie, irq_offset);  	chained_irq_exit(chip, desc);  } @@ -222,7 +690,7 @@ static void ks_pcie_setup_interrupts(struct keystone_pcie *ks_pcie)  						 ks_pcie_legacy_irq_handler,  						 ks_pcie);  	} -	ks_dw_pcie_enable_legacy_irqs(ks_pcie); +	ks_pcie_enable_legacy_irqs(ks_pcie);  	/* MSI IRQ */  	if (IS_ENABLED(CONFIG_PCI_MSI)) { @@ -234,7 +702,7 @@ static void ks_pcie_setup_interrupts(struct keystone_pcie *ks_pcie)  	}  	if (ks_pcie->error_irq > 0) -		ks_dw_pcie_enable_error_irq(ks_pcie); +		ks_pcie_enable_error_irq(ks_pcie);  }  /* @@ -242,8 +710,8 @@ static void ks_pcie_setup_interrupts(struct keystone_pcie *ks_pcie)   * bus error instead of returning 0xffffffff. This handler always returns 0   * for this kind of faults.   */ -static int keystone_pcie_fault(unsigned long addr, unsigned int fsr, -				struct pt_regs *regs) +static int ks_pcie_fault(unsigned long addr, unsigned int fsr, +			 struct pt_regs *regs)  {  	unsigned long instr = *(unsigned long *) instruction_pointer(regs); @@ -257,59 +725,78 @@ static int keystone_pcie_fault(unsigned long addr, unsigned int fsr,  	return 0;  } +static int __init ks_pcie_init_id(struct keystone_pcie *ks_pcie) +{ +	int ret; +	unsigned int id; +	struct regmap *devctrl_regs; +	struct dw_pcie *pci = ks_pcie->pci; +	struct device *dev = pci->dev; +	struct device_node *np = dev->of_node; + +	devctrl_regs = syscon_regmap_lookup_by_phandle(np, "ti,syscon-pcie-id"); +	if (IS_ERR(devctrl_regs)) +		return PTR_ERR(devctrl_regs); + +	ret = regmap_read(devctrl_regs, 0, &id); +	if (ret) +		return ret; + +	dw_pcie_writew_dbi(pci, PCI_VENDOR_ID, id & PCIE_VENDORID_MASK); +	dw_pcie_writew_dbi(pci, PCI_DEVICE_ID, id >> PCIE_DEVICEID_SHIFT); + +	return 0; +} +  static int __init ks_pcie_host_init(struct pcie_port *pp)  {  	struct dw_pcie *pci = to_dw_pcie_from_pp(pp);  	struct keystone_pcie *ks_pcie = to_keystone_pcie(pci); -	u32 val; +	int ret; + +	dw_pcie_setup_rc(pp);  	ks_pcie_establish_link(ks_pcie); -	ks_dw_pcie_setup_rc_app_regs(ks_pcie); +	ks_pcie_setup_rc_app_regs(ks_pcie);  	ks_pcie_setup_interrupts(ks_pcie);  	writew(PCI_IO_RANGE_TYPE_32 | (PCI_IO_RANGE_TYPE_32 << 8),  			pci->dbi_base + PCI_IO_BASE); -	/* update the Vendor ID */ -	writew(ks_pcie->device_id, pci->dbi_base + PCI_DEVICE_ID); - -	/* update the DEV_STAT_CTRL to publish right mrrs */ -	val = readl(pci->dbi_base + PCIE_CAP_BASE + PCI_EXP_DEVCTL); -	val &= ~PCI_EXP_DEVCTL_READRQ; -	/* set the mrrs to 256 bytes */ -	val |= BIT(12); -	writel(val, pci->dbi_base + PCIE_CAP_BASE + PCI_EXP_DEVCTL); +	ret = ks_pcie_init_id(ks_pcie); +	if (ret < 0) +		return ret;  	/*  	 * PCIe access errors that result into OCP errors are caught by ARM as  	 * "External aborts"  	 */ -	hook_fault_code(17, keystone_pcie_fault, SIGBUS, 0, +	hook_fault_code(17, ks_pcie_fault, SIGBUS, 0,  			"Asynchronous external abort");  	return 0;  } -static const struct dw_pcie_host_ops keystone_pcie_host_ops = { -	.rd_other_conf = ks_dw_pcie_rd_other_conf, -	.wr_other_conf = ks_dw_pcie_wr_other_conf, +static const struct dw_pcie_host_ops ks_pcie_host_ops = { +	.rd_other_conf = ks_pcie_rd_other_conf, +	.wr_other_conf = ks_pcie_wr_other_conf,  	.host_init = ks_pcie_host_init, -	.msi_set_irq = ks_dw_pcie_msi_set_irq, -	.msi_clear_irq = ks_dw_pcie_msi_clear_irq, -	.get_msi_addr = ks_dw_pcie_get_msi_addr, -	.msi_host_init = ks_dw_pcie_msi_host_init, -	.msi_irq_ack = ks_dw_pcie_msi_irq_ack, -	.scan_bus = ks_dw_pcie_v3_65_scan_bus, +	.msi_set_irq = ks_pcie_msi_set_irq, +	.msi_clear_irq = ks_pcie_msi_clear_irq, +	.get_msi_addr = ks_pcie_get_msi_addr, +	.msi_host_init = ks_pcie_msi_host_init, +	.msi_irq_ack = ks_pcie_msi_irq_ack, +	.scan_bus = ks_pcie_v3_65_scan_bus,  }; -static irqreturn_t pcie_err_irq_handler(int irq, void *priv) +static irqreturn_t ks_pcie_err_irq_handler(int irq, void *priv)  {  	struct keystone_pcie *ks_pcie = priv; -	return ks_dw_pcie_handle_error_irq(ks_pcie); +	return ks_pcie_handle_error_irq(ks_pcie);  } -static int __init ks_add_pcie_port(struct keystone_pcie *ks_pcie, -			 struct platform_device *pdev) +static int __init ks_pcie_add_pcie_port(struct keystone_pcie *ks_pcie, +					struct platform_device *pdev)  {  	struct dw_pcie *pci = ks_pcie->pci;  	struct pcie_port *pp = &pci->pp; @@ -338,7 +825,7 @@ static int __init ks_add_pcie_port(struct keystone_pcie *ks_pcie,  	if (ks_pcie->error_irq <= 0)  		dev_info(dev, "no error IRQ defined\n");  	else { -		ret = request_irq(ks_pcie->error_irq, pcie_err_irq_handler, +		ret = request_irq(ks_pcie->error_irq, ks_pcie_err_irq_handler,  				  IRQF_SHARED, "pcie-error-irq", ks_pcie);  		if (ret < 0) {  			dev_err(dev, "failed to request error IRQ %d\n", @@ -347,8 +834,8 @@ static int __init ks_add_pcie_port(struct keystone_pcie *ks_pcie,  		}  	} -	pp->ops = &keystone_pcie_host_ops; -	ret = ks_dw_pcie_host_init(ks_pcie, ks_pcie->msi_intc_np); +	pp->ops = &ks_pcie_host_ops; +	ret = ks_pcie_dw_host_init(ks_pcie);  	if (ret) {  		dev_err(dev, "failed to initialize host\n");  		return ret; @@ -365,28 +852,62 @@ static const struct of_device_id ks_pcie_of_match[] = {  	{ },  }; -static const struct dw_pcie_ops dw_pcie_ops = { -	.link_up = ks_dw_pcie_link_up, +static const struct dw_pcie_ops ks_pcie_dw_pcie_ops = { +	.link_up = ks_pcie_link_up,  }; -static int __exit ks_pcie_remove(struct platform_device *pdev) +static void ks_pcie_disable_phy(struct keystone_pcie *ks_pcie)  { -	struct keystone_pcie *ks_pcie = platform_get_drvdata(pdev); +	int num_lanes = ks_pcie->num_lanes; -	clk_disable_unprepare(ks_pcie->clk); +	while (num_lanes--) { +		phy_power_off(ks_pcie->phy[num_lanes]); +		phy_exit(ks_pcie->phy[num_lanes]); +	} +} + +static int ks_pcie_enable_phy(struct keystone_pcie *ks_pcie) +{ +	int i; +	int ret; +	int num_lanes = ks_pcie->num_lanes; + +	for (i = 0; i < num_lanes; i++) { +		ret = phy_init(ks_pcie->phy[i]); +		if (ret < 0) +			goto err_phy; + +		ret = phy_power_on(ks_pcie->phy[i]); +		if (ret < 0) { +			phy_exit(ks_pcie->phy[i]); +			goto err_phy; +		} +	}  	return 0; + +err_phy: +	while (--i >= 0) { +		phy_power_off(ks_pcie->phy[i]); +		phy_exit(ks_pcie->phy[i]); +	} + +	return ret;  }  static int __init ks_pcie_probe(struct platform_device *pdev)  {  	struct device *dev = &pdev->dev; +	struct device_node *np = dev->of_node;  	struct dw_pcie *pci;  	struct keystone_pcie *ks_pcie; -	struct resource *res; -	void __iomem *reg_p; -	struct phy *phy; +	struct device_link **link; +	u32 num_viewport; +	struct phy **phy; +	u32 num_lanes; +	char name[10];  	int ret; +	int i;  	ks_pcie = devm_kzalloc(dev, sizeof(*ks_pcie), GFP_KERNEL);  	if (!ks_pcie) @@ -397,54 +918,99 @@ static int __init ks_pcie_probe(struct platform_device *pdev)  		return -ENOMEM;  	pci->dev = dev; -	pci->ops = &dw_pcie_ops; +	pci->ops = &ks_pcie_dw_pcie_ops; -	ks_pcie->pci = pci; +	ret = of_property_read_u32(np, "num-viewport", &num_viewport); +	if (ret < 0) { +		dev_err(dev, "unable to read *num-viewport* property\n"); +		return ret; +	} -	/* initialize SerDes Phy if present */ -	phy = devm_phy_get(dev, "pcie-phy"); -	if (PTR_ERR_OR_ZERO(phy) == -EPROBE_DEFER) -		return PTR_ERR(phy); +	ret = of_property_read_u32(np, "num-lanes", &num_lanes); +	if (ret) +		num_lanes = 1; -	if (!IS_ERR_OR_NULL(phy)) { -		ret = phy_init(phy); -		if (ret < 0) -			return ret; +	phy = devm_kzalloc(dev, sizeof(*phy) * num_lanes, GFP_KERNEL); +	if (!phy) +		return -ENOMEM; + +	link = devm_kzalloc(dev, sizeof(*link) * num_lanes, GFP_KERNEL); +	if (!link) +		return -ENOMEM; + +	for (i = 0; i < num_lanes; i++) { +		snprintf(name, sizeof(name), "pcie-phy%d", i); +		phy[i] = devm_phy_optional_get(dev, name); +		if (IS_ERR(phy[i])) { +			ret = PTR_ERR(phy[i]); +			goto err_link; +		} + +		if (!phy[i]) +			continue; + +		link[i] = device_link_add(dev, &phy[i]->dev, DL_FLAG_STATELESS); +		if (!link[i]) { +			ret = -EINVAL; +			goto err_link; +		}  	} -	/* index 2 is to read PCI DEVICE_ID */ -	res = platform_get_resource(pdev, IORESOURCE_MEM, 2); -	reg_p = devm_ioremap_resource(dev, res); -	if (IS_ERR(reg_p)) -		return PTR_ERR(reg_p); -	ks_pcie->device_id = readl(reg_p) >> 16; -	devm_iounmap(dev, reg_p); -	devm_release_mem_region(dev, res->start, resource_size(res)); +	ks_pcie->np = np; +	ks_pcie->pci = pci; +	ks_pcie->link = link; +	ks_pcie->num_lanes = num_lanes; +	ks_pcie->num_viewport = num_viewport; +	ks_pcie->phy = phy; -	ks_pcie->np = dev->of_node; -	platform_set_drvdata(pdev, ks_pcie); -	ks_pcie->clk = devm_clk_get(dev, "pcie"); -	if (IS_ERR(ks_pcie->clk)) { -		dev_err(dev, "Failed to get pcie rc clock\n"); -		return PTR_ERR(ks_pcie->clk); +	ret = ks_pcie_enable_phy(ks_pcie); +	if (ret) { +		dev_err(dev, "failed to enable phy\n"); +		goto err_link;  	} -	ret = clk_prepare_enable(ks_pcie->clk); -	if (ret) -		return ret;  	platform_set_drvdata(pdev, ks_pcie); +	pm_runtime_enable(dev); +	ret = pm_runtime_get_sync(dev); +	if (ret < 0) { +		dev_err(dev, "pm_runtime_get_sync failed\n"); +		goto err_get_sync; +	} -	ret = ks_add_pcie_port(ks_pcie, pdev); +	ret = ks_pcie_add_pcie_port(ks_pcie, pdev);  	if (ret < 0) -		goto fail_clk; +		goto err_get_sync;  	return 0; -fail_clk: -	clk_disable_unprepare(ks_pcie->clk); + +err_get_sync: +	pm_runtime_put(dev); +	pm_runtime_disable(dev); +	ks_pcie_disable_phy(ks_pcie); + +err_link: +	while (--i >= 0 && link[i]) +		device_link_del(link[i]);  	return ret;  } +static int __exit ks_pcie_remove(struct platform_device *pdev) +{ +	struct keystone_pcie *ks_pcie = platform_get_drvdata(pdev); +	struct device_link **link = ks_pcie->link; +	int num_lanes = ks_pcie->num_lanes; +	struct device *dev = &pdev->dev; + +	pm_runtime_put(dev); +	pm_runtime_disable(dev); +	ks_pcie_disable_phy(ks_pcie); +	while (num_lanes--) +		device_link_del(link[num_lanes]); + +	return 0; +} +  static struct platform_driver ks_pcie_driver __refdata = {  	.probe  = ks_pcie_probe,  	.remove = __exit_p(ks_pcie_remove), diff --git a/drivers/pci/controller/dwc/pci-keystone.h b/drivers/pci/controller/dwc/pci-keystone.h deleted file mode 100644 index 8a13da391543..000000000000 --- a/drivers/pci/controller/dwc/pci-keystone.h +++ /dev/null @@ -1,57 +0,0 @@ -/* SPDX-License-Identifier: GPL-2.0 */ -/* - * Keystone PCI Controller's common includes - * - * Copyright (C) 2013-2014 Texas Instruments., Ltd. - *		http://www.ti.com - * - * Author: Murali Karicheri <m-karicheri2@ti.com> - */ - -#define MAX_MSI_HOST_IRQS		8 - -struct keystone_pcie { -	struct dw_pcie		*pci; -	struct	clk		*clk; -	/* PCI Device ID */ -	u32			device_id; -	int			num_legacy_host_irqs; -	int			legacy_host_irqs[PCI_NUM_INTX]; -	struct			device_node *legacy_intc_np; - -	int			num_msi_host_irqs; -	int			msi_host_irqs[MAX_MSI_HOST_IRQS]; -	struct			device_node *msi_intc_np; -	struct irq_domain	*legacy_irq_domain; -	struct device_node	*np; - -	int error_irq; - -	/* Application register space */ -	void __iomem		*va_app_base;	/* DT 1st resource */ -	struct resource		app; -}; - -/* Keystone DW specific MSI controller APIs/definitions */ -void ks_dw_pcie_handle_msi_irq(struct keystone_pcie *ks_pcie, int offset); -phys_addr_t ks_dw_pcie_get_msi_addr(struct pcie_port *pp); - -/* Keystone specific PCI controller APIs */ -void ks_dw_pcie_enable_legacy_irqs(struct keystone_pcie *ks_pcie); -void ks_dw_pcie_handle_legacy_irq(struct keystone_pcie *ks_pcie, int offset); -void ks_dw_pcie_enable_error_irq(struct keystone_pcie *ks_pcie); -irqreturn_t ks_dw_pcie_handle_error_irq(struct keystone_pcie *ks_pcie); -int  ks_dw_pcie_host_init(struct keystone_pcie *ks_pcie, -			struct device_node *msi_intc_np); -int ks_dw_pcie_wr_other_conf(struct pcie_port *pp, struct pci_bus *bus, -		unsigned int devfn, int where, int size, u32 val); -int ks_dw_pcie_rd_other_conf(struct pcie_port *pp, struct pci_bus *bus, -		unsigned int devfn, int where, int size, u32 *val); -void ks_dw_pcie_setup_rc_app_regs(struct keystone_pcie *ks_pcie); -void ks_dw_pcie_initiate_link_train(struct keystone_pcie *ks_pcie); -void ks_dw_pcie_msi_irq_ack(int i, struct pcie_port *pp); -void ks_dw_pcie_msi_set_irq(struct pcie_port *pp, int irq); -void ks_dw_pcie_msi_clear_irq(struct pcie_port *pp, int irq); -void ks_dw_pcie_v3_65_scan_bus(struct pcie_port *pp); -int ks_dw_pcie_msi_host_init(struct pcie_port *pp); -int ks_dw_pcie_link_up(struct dw_pcie *pci); diff --git a/drivers/pci/controller/dwc/pcie-designware.h b/drivers/pci/controller/dwc/pcie-designware.h index 9f1a5e399b70..0989d880ac46 100644 --- a/drivers/pci/controller/dwc/pcie-designware.h +++ b/drivers/pci/controller/dwc/pcie-designware.h @@ -36,6 +36,10 @@  #define PORT_LINK_MODE_4_LANES		(0x7 << 16)  #define PORT_LINK_MODE_8_LANES		(0xf << 16) +#define PCIE_PORT_DEBUG0		0x728 +#define PORT_LOGIC_LTSSM_STATE_MASK	0x1f +#define PORT_LOGIC_LTSSM_STATE_L0	0x11 +  #define PCIE_LINK_WIDTH_SPEED_CONTROL	0x80C  #define PORT_LOGIC_SPEED_CHANGE		(0x1 << 17)  #define PORT_LOGIC_LINK_WIDTH_MASK	(0x1f << 8) diff --git a/drivers/pci/controller/dwc/pcie-kirin.c b/drivers/pci/controller/dwc/pcie-kirin.c index 5352e0c3be82..9b599296205d 100644 --- a/drivers/pci/controller/dwc/pcie-kirin.c +++ b/drivers/pci/controller/dwc/pcie-kirin.c @@ -467,8 +467,8 @@ static int kirin_pcie_add_msi(struct dw_pcie *pci,  	return 0;  } -static int __init kirin_add_pcie_port(struct dw_pcie *pci, -				      struct platform_device *pdev) +static int kirin_add_pcie_port(struct dw_pcie *pci, +			       struct platform_device *pdev)  {  	int ret; diff --git a/drivers/pci/controller/dwc/pcie-qcom.c b/drivers/pci/controller/dwc/pcie-qcom.c index 4352c1cb926d..d185ea5fe996 100644 --- a/drivers/pci/controller/dwc/pcie-qcom.c +++ b/drivers/pci/controller/dwc/pcie-qcom.c @@ -1089,7 +1089,6 @@ static int qcom_pcie_host_init(struct pcie_port *pp)  	struct qcom_pcie *pcie = to_qcom_pcie(pci);  	int ret; -	pm_runtime_get_sync(pci->dev);  	qcom_ep_reset_assert(pcie);  	ret = pcie->ops->init(pcie); @@ -1126,7 +1125,6 @@ err_disable_phy:  	phy_power_off(pcie->phy);  err_deinit:  	pcie->ops->deinit(pcie); -	pm_runtime_put(pci->dev);  	return ret;  } @@ -1216,6 +1214,12 @@ static int qcom_pcie_probe(struct platform_device *pdev)  		return -ENOMEM;  	pm_runtime_enable(dev); +	ret = pm_runtime_get_sync(dev); +	if (ret < 0) { +		pm_runtime_disable(dev); +		return ret; +	} +  	pci->dev = dev;  	pci->ops = &dw_pcie_ops;  	pp = &pci->pp; @@ -1225,44 +1229,56 @@ static int qcom_pcie_probe(struct platform_device *pdev)  	pcie->ops = of_device_get_match_data(dev);  	pcie->reset = devm_gpiod_get_optional(dev, "perst", GPIOD_OUT_LOW); -	if (IS_ERR(pcie->reset)) -		return PTR_ERR(pcie->reset); +	if (IS_ERR(pcie->reset)) { +		ret = PTR_ERR(pcie->reset); +		goto err_pm_runtime_put; +	}  	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "parf");  	pcie->parf = devm_ioremap_resource(dev, res); -	if (IS_ERR(pcie->parf)) -		return PTR_ERR(pcie->parf); +	if (IS_ERR(pcie->parf)) { +		ret = PTR_ERR(pcie->parf); +		goto err_pm_runtime_put; +	}  	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "dbi");  	pci->dbi_base = devm_pci_remap_cfg_resource(dev, res); -	if (IS_ERR(pci->dbi_base)) -		return PTR_ERR(pci->dbi_base); +	if (IS_ERR(pci->dbi_base)) { +		ret = PTR_ERR(pci->dbi_base); +		goto err_pm_runtime_put; +	}  	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "elbi");  	pcie->elbi = devm_ioremap_resource(dev, res); -	if (IS_ERR(pcie->elbi)) -		return PTR_ERR(pcie->elbi); +	if (IS_ERR(pcie->elbi)) { +		ret = PTR_ERR(pcie->elbi); +		goto err_pm_runtime_put; +	}  	pcie->phy = devm_phy_optional_get(dev, "pciephy"); -	if (IS_ERR(pcie->phy)) -		return PTR_ERR(pcie->phy); +	if (IS_ERR(pcie->phy)) { +		ret = PTR_ERR(pcie->phy); +		goto err_pm_runtime_put; +	}  	ret = pcie->ops->get_resources(pcie);  	if (ret) -		return ret; +		goto err_pm_runtime_put;  	pp->ops = &qcom_pcie_dw_ops;  	if (IS_ENABLED(CONFIG_PCI_MSI)) {  		pp->msi_irq = platform_get_irq_byname(pdev, "msi"); -		if (pp->msi_irq < 0) -			return pp->msi_irq; +		if (pp->msi_irq < 0) { +			ret = pp->msi_irq; +			goto err_pm_runtime_put; +		}  	}  	ret = phy_init(pcie->phy);  	if (ret) {  		pm_runtime_disable(&pdev->dev); -		return ret; +		goto err_pm_runtime_put;  	}  	platform_set_drvdata(pdev, pcie); @@ -1271,10 +1287,16 @@ static int qcom_pcie_probe(struct platform_device *pdev)  	if (ret) {  		dev_err(dev, "cannot initialize host\n");  		pm_runtime_disable(&pdev->dev); -		return ret; +		goto err_pm_runtime_put;  	}  	return 0; + +err_pm_runtime_put: +	pm_runtime_put(dev); +	pm_runtime_disable(dev); + +	return ret;  }  static const struct of_device_id qcom_pcie_match[] = { diff --git a/drivers/pci/controller/pci-aardvark.c b/drivers/pci/controller/pci-aardvark.c index 6b4555ff2548..750081c1cb48 100644 --- a/drivers/pci/controller/pci-aardvark.c +++ b/drivers/pci/controller/pci-aardvark.c @@ -20,12 +20,16 @@  #include <linux/of_pci.h>  #include "../pci.h" +#include "../pci-bridge-emul.h"  /* PCIe core registers */ +#define PCIE_CORE_DEV_ID_REG					0x0  #define PCIE_CORE_CMD_STATUS_REG				0x4  #define     PCIE_CORE_CMD_IO_ACCESS_EN				BIT(0)  #define     PCIE_CORE_CMD_MEM_ACCESS_EN				BIT(1)  #define     PCIE_CORE_CMD_MEM_IO_REQ_EN				BIT(2) +#define PCIE_CORE_DEV_REV_REG					0x8 +#define PCIE_CORE_PCIEXP_CAP					0xc0  #define PCIE_CORE_DEV_CTRL_STATS_REG				0xc8  #define     PCIE_CORE_DEV_CTRL_STATS_RELAX_ORDER_DISABLE	(0 << 4)  #define     PCIE_CORE_DEV_CTRL_STATS_MAX_PAYLOAD_SZ_SHIFT	5 @@ -41,7 +45,10 @@  #define     PCIE_CORE_ERR_CAPCTL_ECRC_CHK_TX_EN			BIT(6)  #define     PCIE_CORE_ERR_CAPCTL_ECRC_CHCK			BIT(7)  #define     PCIE_CORE_ERR_CAPCTL_ECRC_CHCK_RCV			BIT(8) - +#define     PCIE_CORE_INT_A_ASSERT_ENABLE			1 +#define     PCIE_CORE_INT_B_ASSERT_ENABLE			2 +#define     PCIE_CORE_INT_C_ASSERT_ENABLE			3 +#define     PCIE_CORE_INT_D_ASSERT_ENABLE			4  /* PIO registers base address and register offsets */  #define PIO_BASE_ADDR				0x4000  #define PIO_CTRL				(PIO_BASE_ADDR + 0x0) @@ -93,7 +100,9 @@  #define     PCIE_CORE_CTRL2_STRICT_ORDER_ENABLE	BIT(5)  #define     PCIE_CORE_CTRL2_OB_WIN_ENABLE	BIT(6)  #define     PCIE_CORE_CTRL2_MSI_ENABLE		BIT(10) +#define PCIE_MSG_LOG_REG			(CONTROL_BASE_ADDR + 0x30)  #define PCIE_ISR0_REG				(CONTROL_BASE_ADDR + 0x40) +#define PCIE_MSG_PM_PME_MASK			BIT(7)  #define PCIE_ISR0_MASK_REG			(CONTROL_BASE_ADDR + 0x44)  #define     PCIE_ISR0_MSI_INT_PENDING		BIT(24)  #define     PCIE_ISR0_INTX_ASSERT(val)		BIT(16 + (val)) @@ -189,6 +198,7 @@ struct advk_pcie {  	struct mutex msi_used_lock;  	u16 msi_msg;  	int root_bus_nr; +	struct pci_bridge_emul bridge;  };  static inline void advk_writel(struct advk_pcie *pcie, u32 val, u64 reg) @@ -390,6 +400,109 @@ static int advk_pcie_wait_pio(struct advk_pcie *pcie)  	return -ETIMEDOUT;  } + +static pci_bridge_emul_read_status_t +advk_pci_bridge_emul_pcie_conf_read(struct pci_bridge_emul *bridge, +				    int reg, u32 *value) +{ +	struct advk_pcie *pcie = bridge->data; + + +	switch (reg) { +	case PCI_EXP_SLTCTL: +		*value = PCI_EXP_SLTSTA_PDS << 16; +		return PCI_BRIDGE_EMUL_HANDLED; + +	case PCI_EXP_RTCTL: { +		u32 val = advk_readl(pcie, PCIE_ISR0_MASK_REG); +		*value = (val & PCIE_MSG_PM_PME_MASK) ? PCI_EXP_RTCTL_PMEIE : 0; +		return PCI_BRIDGE_EMUL_HANDLED; +	} + +	case PCI_EXP_RTSTA: { +		u32 isr0 = advk_readl(pcie, PCIE_ISR0_REG); +		u32 msglog = advk_readl(pcie, PCIE_MSG_LOG_REG); +		*value = (isr0 & PCIE_MSG_PM_PME_MASK) << 16 | (msglog >> 16); +		return PCI_BRIDGE_EMUL_HANDLED; +	} + +	case PCI_CAP_LIST_ID: +	case PCI_EXP_DEVCAP: +	case PCI_EXP_DEVCTL: +	case PCI_EXP_LNKCAP: +	case PCI_EXP_LNKCTL: +		*value = advk_readl(pcie, PCIE_CORE_PCIEXP_CAP + reg); +		return PCI_BRIDGE_EMUL_HANDLED; +	default: +		return PCI_BRIDGE_EMUL_NOT_HANDLED; +	} + +} + +static void +advk_pci_bridge_emul_pcie_conf_write(struct pci_bridge_emul *bridge, +				     int reg, u32 old, u32 new, u32 mask) +{ +	struct advk_pcie *pcie = bridge->data; + +	switch (reg) { +	case PCI_EXP_DEVCTL: +	case PCI_EXP_LNKCTL: +		advk_writel(pcie, new, PCIE_CORE_PCIEXP_CAP + reg); +		break; + +	case PCI_EXP_RTCTL: +		new = (new & PCI_EXP_RTCTL_PMEIE) << 3; +		advk_writel(pcie, new, PCIE_ISR0_MASK_REG); +		break; + +	case PCI_EXP_RTSTA: +		new = (new & PCI_EXP_RTSTA_PME) >> 9; +		advk_writel(pcie, new, PCIE_ISR0_REG); +		break; + +	default: +		break; +	} +} + +struct pci_bridge_emul_ops advk_pci_bridge_emul_ops = { +	.read_pcie = advk_pci_bridge_emul_pcie_conf_read, +	.write_pcie = advk_pci_bridge_emul_pcie_conf_write, +}; + +/* + * Initialize the configuration space of the PCI-to-PCI bridge + * associated with the given PCIe interface. + */ +static void advk_sw_pci_bridge_init(struct advk_pcie *pcie) +{ +	struct pci_bridge_emul *bridge = &pcie->bridge; + +	bridge->conf.vendor = advk_readl(pcie, PCIE_CORE_DEV_ID_REG) & 0xffff; +	bridge->conf.device = advk_readl(pcie, PCIE_CORE_DEV_ID_REG) >> 16; +	bridge->conf.class_revision = +		advk_readl(pcie, PCIE_CORE_DEV_REV_REG) & 0xff; + +	/* Support 32 bits I/O addressing */ +	bridge->conf.iobase = PCI_IO_RANGE_TYPE_32; +	bridge->conf.iolimit = PCI_IO_RANGE_TYPE_32; + +	/* Support 64 bits memory pref */ +	bridge->conf.pref_mem_base = PCI_PREF_RANGE_TYPE_64; +	bridge->conf.pref_mem_limit = PCI_PREF_RANGE_TYPE_64; + +	/* Support interrupt A for MSI feature */ +	bridge->conf.intpin = PCIE_CORE_INT_A_ASSERT_ENABLE; + +	bridge->has_pcie = true; +	bridge->data = pcie; +	bridge->ops = &advk_pci_bridge_emul_ops; + +	pci_bridge_emul_init(bridge); + +} +  static bool advk_pcie_valid_device(struct advk_pcie *pcie, struct pci_bus *bus,  				  int devfn)  { @@ -411,6 +524,10 @@ static int advk_pcie_rd_conf(struct pci_bus *bus, u32 devfn,  		return PCIBIOS_DEVICE_NOT_FOUND;  	} +	if (bus->number == pcie->root_bus_nr) +		return pci_bridge_emul_conf_read(&pcie->bridge, where, +						 size, val); +  	/* Start PIO */  	advk_writel(pcie, 0, PIO_START);  	advk_writel(pcie, 1, PIO_ISR); @@ -418,7 +535,7 @@ static int advk_pcie_rd_conf(struct pci_bus *bus, u32 devfn,  	/* Program the control register */  	reg = advk_readl(pcie, PIO_CTRL);  	reg &= ~PIO_CTRL_TYPE_MASK; -	if (bus->number ==  pcie->root_bus_nr) +	if (bus->primary ==  pcie->root_bus_nr)  		reg |= PCIE_CONFIG_RD_TYPE0;  	else  		reg |= PCIE_CONFIG_RD_TYPE1; @@ -463,6 +580,10 @@ static int advk_pcie_wr_conf(struct pci_bus *bus, u32 devfn,  	if (!advk_pcie_valid_device(pcie, bus, devfn))  		return PCIBIOS_DEVICE_NOT_FOUND; +	if (bus->number == pcie->root_bus_nr) +		return pci_bridge_emul_conf_write(&pcie->bridge, where, +						  size, val); +  	if (where % size)  		return PCIBIOS_SET_FAILED; @@ -473,7 +594,7 @@ static int advk_pcie_wr_conf(struct pci_bus *bus, u32 devfn,  	/* Program the control register */  	reg = advk_readl(pcie, PIO_CTRL);  	reg &= ~PIO_CTRL_TYPE_MASK; -	if (bus->number == pcie->root_bus_nr) +	if (bus->primary == pcie->root_bus_nr)  		reg |= PCIE_CONFIG_WR_TYPE0;  	else  		reg |= PCIE_CONFIG_WR_TYPE1; @@ -875,6 +996,8 @@ static int advk_pcie_probe(struct platform_device *pdev)  	advk_pcie_setup_hw(pcie); +	advk_sw_pci_bridge_init(pcie); +  	ret = advk_pcie_init_irq_domain(pcie);  	if (ret) {  		dev_err(dev, "Failed to initialize irq\n"); diff --git a/drivers/pci/controller/pci-host-common.c b/drivers/pci/controller/pci-host-common.c index d8f10451f273..c742881b5061 100644 --- a/drivers/pci/controller/pci-host-common.c +++ b/drivers/pci/controller/pci-host-common.c @@ -58,9 +58,7 @@ err_out:  int pci_host_common_probe(struct platform_device *pdev,  			  struct pci_ecam_ops *ops)  { -	const char *type;  	struct device *dev = &pdev->dev; -	struct device_node *np = dev->of_node;  	struct pci_host_bridge *bridge;  	struct pci_config_window *cfg;  	struct list_head resources; @@ -70,12 +68,6 @@ int pci_host_common_probe(struct platform_device *pdev,  	if (!bridge)  		return -ENOMEM; -	type = of_get_property(np, "device_type", NULL); -	if (!type || strcmp(type, "pci")) { -		dev_err(dev, "invalid \"device_type\" %s\n", type); -		return -EINVAL; -	} -  	of_pci_check_probe_only();  	/* Parse and map our Configuration Space windows */ diff --git a/drivers/pci/controller/pci-mvebu.c b/drivers/pci/controller/pci-mvebu.c index a41d79b8d46a..fa0fc46edb0c 100644 --- a/drivers/pci/controller/pci-mvebu.c +++ b/drivers/pci/controller/pci-mvebu.c @@ -22,6 +22,7 @@  #include <linux/of_platform.h>  #include "../pci.h" +#include "../pci-bridge-emul.h"  /*   * PCIe unit register offsets. @@ -63,61 +64,6 @@  #define PCIE_DEBUG_CTRL         0x1a60  #define  PCIE_DEBUG_SOFT_RESET		BIT(20) -enum { -	PCISWCAP = PCI_BRIDGE_CONTROL + 2, -	PCISWCAP_EXP_LIST_ID	= PCISWCAP + PCI_CAP_LIST_ID, -	PCISWCAP_EXP_DEVCAP	= PCISWCAP + PCI_EXP_DEVCAP, -	PCISWCAP_EXP_DEVCTL	= PCISWCAP + PCI_EXP_DEVCTL, -	PCISWCAP_EXP_LNKCAP	= PCISWCAP + PCI_EXP_LNKCAP, -	PCISWCAP_EXP_LNKCTL	= PCISWCAP + PCI_EXP_LNKCTL, -	PCISWCAP_EXP_SLTCAP	= PCISWCAP + PCI_EXP_SLTCAP, -	PCISWCAP_EXP_SLTCTL	= PCISWCAP + PCI_EXP_SLTCTL, -	PCISWCAP_EXP_RTCTL	= PCISWCAP + PCI_EXP_RTCTL, -	PCISWCAP_EXP_RTSTA	= PCISWCAP + PCI_EXP_RTSTA, -	PCISWCAP_EXP_DEVCAP2	= PCISWCAP + PCI_EXP_DEVCAP2, -	PCISWCAP_EXP_DEVCTL2	= PCISWCAP + PCI_EXP_DEVCTL2, -	PCISWCAP_EXP_LNKCAP2	= PCISWCAP + PCI_EXP_LNKCAP2, -	PCISWCAP_EXP_LNKCTL2	= PCISWCAP + PCI_EXP_LNKCTL2, -	PCISWCAP_EXP_SLTCAP2	= PCISWCAP + PCI_EXP_SLTCAP2, -	PCISWCAP_EXP_SLTCTL2	= PCISWCAP + PCI_EXP_SLTCTL2, -}; - -/* PCI configuration space of a PCI-to-PCI bridge */ -struct mvebu_sw_pci_bridge { -	u16 vendor; -	u16 device; -	u16 command; -	u16 status; -	u16 class; -	u8 interface; -	u8 revision; -	u8 bist; -	u8 header_type; -	u8 latency_timer; -	u8 cache_line_size; -	u32 bar[2]; -	u8 primary_bus; -	u8 secondary_bus; -	u8 subordinate_bus; -	u8 secondary_latency_timer; -	u8 iobase; -	u8 iolimit; -	u16 secondary_status; -	u16 membase; -	u16 memlimit; -	u16 iobaseupper; -	u16 iolimitupper; -	u32 romaddr; -	u8 intline; -	u8 intpin; -	u16 bridgectrl; - -	/* PCI express capability */ -	u32 pcie_sltcap; -	u16 pcie_devctl; -	u16 pcie_rtctl; -}; -  struct mvebu_pcie_port;  /* Structure representing all PCIe interfaces */ @@ -153,7 +99,7 @@ struct mvebu_pcie_port {  	struct clk *clk;  	struct gpio_desc *reset_gpio;  	char *reset_name; -	struct mvebu_sw_pci_bridge bridge; +	struct pci_bridge_emul bridge;  	struct device_node *dn;  	struct mvebu_pcie *pcie;  	struct mvebu_pcie_window memwin; @@ -415,11 +361,12 @@ static void mvebu_pcie_set_window(struct mvebu_pcie_port *port,  static void mvebu_pcie_handle_iobase_change(struct mvebu_pcie_port *port)  {  	struct mvebu_pcie_window desired = {}; +	struct pci_bridge_emul_conf *conf = &port->bridge.conf;  	/* Are the new iobase/iolimit values invalid? */ -	if (port->bridge.iolimit < port->bridge.iobase || -	    port->bridge.iolimitupper < port->bridge.iobaseupper || -	    !(port->bridge.command & PCI_COMMAND_IO)) { +	if (conf->iolimit < conf->iobase || +	    conf->iolimitupper < conf->iobaseupper || +	    !(conf->command & PCI_COMMAND_IO)) {  		mvebu_pcie_set_window(port, port->io_target, port->io_attr,  				      &desired, &port->iowin);  		return; @@ -438,11 +385,11 @@ static void mvebu_pcie_handle_iobase_change(struct mvebu_pcie_port *port)  	 * specifications. iobase is the bus address, port->iowin_base  	 * is the CPU address.  	 */ -	desired.remap = ((port->bridge.iobase & 0xF0) << 8) | -			(port->bridge.iobaseupper << 16); +	desired.remap = ((conf->iobase & 0xF0) << 8) | +			(conf->iobaseupper << 16);  	desired.base = port->pcie->io.start + desired.remap; -	desired.size = ((0xFFF | ((port->bridge.iolimit & 0xF0) << 8) | -			 (port->bridge.iolimitupper << 16)) - +	desired.size = ((0xFFF | ((conf->iolimit & 0xF0) << 8) | +			 (conf->iolimitupper << 16)) -  			desired.remap) +  		       1; @@ -453,10 +400,11 @@ static void mvebu_pcie_handle_iobase_change(struct mvebu_pcie_port *port)  static void mvebu_pcie_handle_membase_change(struct mvebu_pcie_port *port)  {  	struct mvebu_pcie_window desired = {.remap = MVEBU_MBUS_NO_REMAP}; +	struct pci_bridge_emul_conf *conf = &port->bridge.conf;  	/* Are the new membase/memlimit values invalid? */ -	if (port->bridge.memlimit < port->bridge.membase || -	    !(port->bridge.command & PCI_COMMAND_MEMORY)) { +	if (conf->memlimit < conf->membase || +	    !(conf->command & PCI_COMMAND_MEMORY)) {  		mvebu_pcie_set_window(port, port->mem_target, port->mem_attr,  				      &desired, &port->memwin);  		return; @@ -468,130 +416,32 @@ static void mvebu_pcie_handle_membase_change(struct mvebu_pcie_port *port)  	 * window to setup, according to the PCI-to-PCI bridge  	 * specifications.  	 */ -	desired.base = ((port->bridge.membase & 0xFFF0) << 16); -	desired.size = (((port->bridge.memlimit & 0xFFF0) << 16) | 0xFFFFF) - +	desired.base = ((conf->membase & 0xFFF0) << 16); +	desired.size = (((conf->memlimit & 0xFFF0) << 16) | 0xFFFFF) -  		       desired.base + 1;  	mvebu_pcie_set_window(port, port->mem_target, port->mem_attr, &desired,  			      &port->memwin);  } -/* - * Initialize the configuration space of the PCI-to-PCI bridge - * associated with the given PCIe interface. - */ -static void mvebu_sw_pci_bridge_init(struct mvebu_pcie_port *port) -{ -	struct mvebu_sw_pci_bridge *bridge = &port->bridge; - -	memset(bridge, 0, sizeof(struct mvebu_sw_pci_bridge)); - -	bridge->class = PCI_CLASS_BRIDGE_PCI; -	bridge->vendor = PCI_VENDOR_ID_MARVELL; -	bridge->device = mvebu_readl(port, PCIE_DEV_ID_OFF) >> 16; -	bridge->revision = mvebu_readl(port, PCIE_DEV_REV_OFF) & 0xff; -	bridge->header_type = PCI_HEADER_TYPE_BRIDGE; -	bridge->cache_line_size = 0x10; - -	/* We support 32 bits I/O addressing */ -	bridge->iobase = PCI_IO_RANGE_TYPE_32; -	bridge->iolimit = PCI_IO_RANGE_TYPE_32; - -	/* Add capabilities */ -	bridge->status = PCI_STATUS_CAP_LIST; -} - -/* - * Read the configuration space of the PCI-to-PCI bridge associated to - * the given PCIe interface. - */ -static int mvebu_sw_pci_bridge_read(struct mvebu_pcie_port *port, -				  unsigned int where, int size, u32 *value) +static pci_bridge_emul_read_status_t +mvebu_pci_bridge_emul_pcie_conf_read(struct pci_bridge_emul *bridge, +				     int reg, u32 *value)  { -	struct mvebu_sw_pci_bridge *bridge = &port->bridge; - -	switch (where & ~3) { -	case PCI_VENDOR_ID: -		*value = bridge->device << 16 | bridge->vendor; -		break; - -	case PCI_COMMAND: -		*value = bridge->command | bridge->status << 16; -		break; - -	case PCI_CLASS_REVISION: -		*value = bridge->class << 16 | bridge->interface << 8 | -			 bridge->revision; -		break; - -	case PCI_CACHE_LINE_SIZE: -		*value = bridge->bist << 24 | bridge->header_type << 16 | -			 bridge->latency_timer << 8 | bridge->cache_line_size; -		break; - -	case PCI_BASE_ADDRESS_0 ... PCI_BASE_ADDRESS_1: -		*value = bridge->bar[((where & ~3) - PCI_BASE_ADDRESS_0) / 4]; -		break; - -	case PCI_PRIMARY_BUS: -		*value = (bridge->secondary_latency_timer << 24 | -			  bridge->subordinate_bus         << 16 | -			  bridge->secondary_bus           <<  8 | -			  bridge->primary_bus); -		break; - -	case PCI_IO_BASE: -		if (!mvebu_has_ioport(port)) -			*value = bridge->secondary_status << 16; -		else -			*value = (bridge->secondary_status << 16 | -				  bridge->iolimit          <<  8 | -				  bridge->iobase); -		break; - -	case PCI_MEMORY_BASE: -		*value = (bridge->memlimit << 16 | bridge->membase); -		break; - -	case PCI_PREF_MEMORY_BASE: -		*value = 0; -		break; - -	case PCI_IO_BASE_UPPER16: -		*value = (bridge->iolimitupper << 16 | bridge->iobaseupper); -		break; - -	case PCI_CAPABILITY_LIST: -		*value = PCISWCAP; -		break; - -	case PCI_ROM_ADDRESS1: -		*value = 0; -		break; +	struct mvebu_pcie_port *port = bridge->data; -	case PCI_INTERRUPT_LINE: -		/* LINE PIN MIN_GNT MAX_LAT */ -		*value = 0; -		break; - -	case PCISWCAP_EXP_LIST_ID: -		/* Set PCIe v2, root port, slot support */ -		*value = (PCI_EXP_TYPE_ROOT_PORT << 4 | 2 | -			  PCI_EXP_FLAGS_SLOT) << 16 | PCI_CAP_ID_EXP; -		break; - -	case PCISWCAP_EXP_DEVCAP: +	switch (reg) { +	case PCI_EXP_DEVCAP:  		*value = mvebu_readl(port, PCIE_CAP_PCIEXP + PCI_EXP_DEVCAP);  		break; -	case PCISWCAP_EXP_DEVCTL: +	case PCI_EXP_DEVCTL:  		*value = mvebu_readl(port, PCIE_CAP_PCIEXP + PCI_EXP_DEVCTL) &  				 ~(PCI_EXP_DEVCTL_URRE | PCI_EXP_DEVCTL_FERE |  				   PCI_EXP_DEVCTL_NFERE | PCI_EXP_DEVCTL_CERE); -		*value |= bridge->pcie_devctl;  		break; -	case PCISWCAP_EXP_LNKCAP: +	case PCI_EXP_LNKCAP:  		/*  		 * PCIe requires the clock power management capability to be  		 * hard-wired to zero for downstream ports @@ -600,176 +450,140 @@ static int mvebu_sw_pci_bridge_read(struct mvebu_pcie_port *port,  			 ~PCI_EXP_LNKCAP_CLKPM;  		break; -	case PCISWCAP_EXP_LNKCTL: +	case PCI_EXP_LNKCTL:  		*value = mvebu_readl(port, PCIE_CAP_PCIEXP + PCI_EXP_LNKCTL);  		break; -	case PCISWCAP_EXP_SLTCAP: -		*value = bridge->pcie_sltcap; -		break; - -	case PCISWCAP_EXP_SLTCTL: +	case PCI_EXP_SLTCTL:  		*value = PCI_EXP_SLTSTA_PDS << 16;  		break; -	case PCISWCAP_EXP_RTCTL: -		*value = bridge->pcie_rtctl; -		break; - -	case PCISWCAP_EXP_RTSTA: +	case PCI_EXP_RTSTA:  		*value = mvebu_readl(port, PCIE_RC_RTSTA);  		break; -	/* PCIe requires the v2 fields to be hard-wired to zero */ -	case PCISWCAP_EXP_DEVCAP2: -	case PCISWCAP_EXP_DEVCTL2: -	case PCISWCAP_EXP_LNKCAP2: -	case PCISWCAP_EXP_LNKCTL2: -	case PCISWCAP_EXP_SLTCAP2: -	case PCISWCAP_EXP_SLTCTL2:  	default: -		/* -		 * PCI defines configuration read accesses to reserved or -		 * unimplemented registers to read as zero and complete -		 * normally. -		 */ -		*value = 0; -		return PCIBIOS_SUCCESSFUL; +		return PCI_BRIDGE_EMUL_NOT_HANDLED;  	} -	if (size == 2) -		*value = (*value >> (8 * (where & 3))) & 0xffff; -	else if (size == 1) -		*value = (*value >> (8 * (where & 3))) & 0xff; - -	return PCIBIOS_SUCCESSFUL; +	return PCI_BRIDGE_EMUL_HANDLED;  } -/* Write to the PCI-to-PCI bridge configuration space */ -static int mvebu_sw_pci_bridge_write(struct mvebu_pcie_port *port, -				     unsigned int where, int size, u32 value) +static void +mvebu_pci_bridge_emul_base_conf_write(struct pci_bridge_emul *bridge, +				      int reg, u32 old, u32 new, u32 mask)  { -	struct mvebu_sw_pci_bridge *bridge = &port->bridge; -	u32 mask, reg; -	int err; - -	if (size == 4) -		mask = 0x0; -	else if (size == 2) -		mask = ~(0xffff << ((where & 3) * 8)); -	else if (size == 1) -		mask = ~(0xff << ((where & 3) * 8)); -	else -		return PCIBIOS_BAD_REGISTER_NUMBER; - -	err = mvebu_sw_pci_bridge_read(port, where & ~3, 4, ®); -	if (err) -		return err; +	struct mvebu_pcie_port *port = bridge->data; +	struct pci_bridge_emul_conf *conf = &bridge->conf; -	value = (reg & mask) | value << ((where & 3) * 8); - -	switch (where & ~3) { +	switch (reg) {  	case PCI_COMMAND:  	{ -		u32 old = bridge->command; -  		if (!mvebu_has_ioport(port)) -			value &= ~PCI_COMMAND_IO; +			conf->command &= ~PCI_COMMAND_IO; -		bridge->command = value & 0xffff; -		if ((old ^ bridge->command) & PCI_COMMAND_IO) +		if ((old ^ new) & PCI_COMMAND_IO)  			mvebu_pcie_handle_iobase_change(port); -		if ((old ^ bridge->command) & PCI_COMMAND_MEMORY) +		if ((old ^ new) & PCI_COMMAND_MEMORY)  			mvebu_pcie_handle_membase_change(port); -		break; -	} -	case PCI_BASE_ADDRESS_0 ... PCI_BASE_ADDRESS_1: -		bridge->bar[((where & ~3) - PCI_BASE_ADDRESS_0) / 4] = value;  		break; +	}  	case PCI_IO_BASE:  		/* -		 * We also keep bit 1 set, it is a read-only bit that +		 * We keep bit 1 set, it is a read-only bit that  		 * indicates we support 32 bits addressing for the  		 * I/O  		 */ -		bridge->iobase = (value & 0xff) | PCI_IO_RANGE_TYPE_32; -		bridge->iolimit = ((value >> 8) & 0xff) | PCI_IO_RANGE_TYPE_32; +		conf->iobase |= PCI_IO_RANGE_TYPE_32; +		conf->iolimit |= PCI_IO_RANGE_TYPE_32;  		mvebu_pcie_handle_iobase_change(port);  		break;  	case PCI_MEMORY_BASE: -		bridge->membase = value & 0xffff; -		bridge->memlimit = value >> 16;  		mvebu_pcie_handle_membase_change(port);  		break;  	case PCI_IO_BASE_UPPER16: -		bridge->iobaseupper = value & 0xffff; -		bridge->iolimitupper = value >> 16;  		mvebu_pcie_handle_iobase_change(port);  		break;  	case PCI_PRIMARY_BUS: -		bridge->primary_bus             = value & 0xff; -		bridge->secondary_bus           = (value >> 8) & 0xff; -		bridge->subordinate_bus         = (value >> 16) & 0xff; -		bridge->secondary_latency_timer = (value >> 24) & 0xff; -		mvebu_pcie_set_local_bus_nr(port, bridge->secondary_bus); +		mvebu_pcie_set_local_bus_nr(port, conf->secondary_bus); +		break; + +	default:  		break; +	} +} + +static void +mvebu_pci_bridge_emul_pcie_conf_write(struct pci_bridge_emul *bridge, +				      int reg, u32 old, u32 new, u32 mask) +{ +	struct mvebu_pcie_port *port = bridge->data; -	case PCISWCAP_EXP_DEVCTL: +	switch (reg) { +	case PCI_EXP_DEVCTL:  		/*  		 * Armada370 data says these bits must always  		 * be zero when in root complex mode.  		 */ -		value &= ~(PCI_EXP_DEVCTL_URRE | PCI_EXP_DEVCTL_FERE | -			   PCI_EXP_DEVCTL_NFERE | PCI_EXP_DEVCTL_CERE); - -		/* -		 * If the mask is 0xffff0000, then we only want to write -		 * the device control register, rather than clearing the -		 * RW1C bits in the device status register.  Mask out the -		 * status register bits. -		 */ -		if (mask == 0xffff0000) -			value &= 0xffff; +		new &= ~(PCI_EXP_DEVCTL_URRE | PCI_EXP_DEVCTL_FERE | +			 PCI_EXP_DEVCTL_NFERE | PCI_EXP_DEVCTL_CERE); -		mvebu_writel(port, value, PCIE_CAP_PCIEXP + PCI_EXP_DEVCTL); +		mvebu_writel(port, new, PCIE_CAP_PCIEXP + PCI_EXP_DEVCTL);  		break; -	case PCISWCAP_EXP_LNKCTL: +	case PCI_EXP_LNKCTL:  		/*  		 * If we don't support CLKREQ, we must ensure that the  		 * CLKREQ enable bit always reads zero.  Since we haven't  		 * had this capability, and it's dependent on board wiring,  		 * disable it for the time being.  		 */ -		value &= ~PCI_EXP_LNKCTL_CLKREQ_EN; +		new &= ~PCI_EXP_LNKCTL_CLKREQ_EN; -		/* -		 * If the mask is 0xffff0000, then we only want to write -		 * the link control register, rather than clearing the -		 * RW1C bits in the link status register.  Mask out the -		 * RW1C status register bits. -		 */ -		if (mask == 0xffff0000) -			value &= ~((PCI_EXP_LNKSTA_LABS | -				    PCI_EXP_LNKSTA_LBMS) << 16); - -		mvebu_writel(port, value, PCIE_CAP_PCIEXP + PCI_EXP_LNKCTL); +		mvebu_writel(port, new, PCIE_CAP_PCIEXP + PCI_EXP_LNKCTL);  		break; -	case PCISWCAP_EXP_RTSTA: -		mvebu_writel(port, value, PCIE_RC_RTSTA); +	case PCI_EXP_RTSTA: +		mvebu_writel(port, new, PCIE_RC_RTSTA);  		break; +	} +} -	default: -		break; +struct pci_bridge_emul_ops mvebu_pci_bridge_emul_ops = { +	.write_base = mvebu_pci_bridge_emul_base_conf_write, +	.read_pcie = mvebu_pci_bridge_emul_pcie_conf_read, +	.write_pcie = mvebu_pci_bridge_emul_pcie_conf_write, +}; + +/* + * Initialize the configuration space of the PCI-to-PCI bridge + * associated with the given PCIe interface. + */ +static void mvebu_pci_bridge_emul_init(struct mvebu_pcie_port *port) +{ +	struct pci_bridge_emul *bridge = &port->bridge; + +	bridge->conf.vendor = PCI_VENDOR_ID_MARVELL; +	bridge->conf.device = mvebu_readl(port, PCIE_DEV_ID_OFF) >> 16; +	bridge->conf.class_revision = +		mvebu_readl(port, PCIE_DEV_REV_OFF) & 0xff; + +	if (mvebu_has_ioport(port)) { +		/* We support 32 bits I/O addressing */ +		bridge->conf.iobase = PCI_IO_RANGE_TYPE_32; +		bridge->conf.iolimit = PCI_IO_RANGE_TYPE_32;  	} -	return PCIBIOS_SUCCESSFUL; +	bridge->has_pcie = true; +	bridge->data = port; +	bridge->ops = &mvebu_pci_bridge_emul_ops; + +	pci_bridge_emul_init(bridge);  }  static inline struct mvebu_pcie *sys_to_pcie(struct pci_sys_data *sys) @@ -789,8 +603,8 @@ static struct mvebu_pcie_port *mvebu_pcie_find_port(struct mvebu_pcie *pcie,  		if (bus->number == 0 && port->devfn == devfn)  			return port;  		if (bus->number != 0 && -		    bus->number >= port->bridge.secondary_bus && -		    bus->number <= port->bridge.subordinate_bus) +		    bus->number >= port->bridge.conf.secondary_bus && +		    bus->number <= port->bridge.conf.subordinate_bus)  			return port;  	} @@ -811,7 +625,8 @@ static int mvebu_pcie_wr_conf(struct pci_bus *bus, u32 devfn,  	/* Access the emulated PCI-to-PCI bridge */  	if (bus->number == 0) -		return mvebu_sw_pci_bridge_write(port, where, size, val); +		return pci_bridge_emul_conf_write(&port->bridge, where, +						  size, val);  	if (!mvebu_pcie_link_up(port))  		return PCIBIOS_DEVICE_NOT_FOUND; @@ -839,7 +654,8 @@ static int mvebu_pcie_rd_conf(struct pci_bus *bus, u32 devfn, int where,  	/* Access the emulated PCI-to-PCI bridge */  	if (bus->number == 0) -		return mvebu_sw_pci_bridge_read(port, where, size, val); +		return pci_bridge_emul_conf_read(&port->bridge, where, +						 size, val);  	if (!mvebu_pcie_link_up(port)) {  		*val = 0xffffffff; @@ -1297,7 +1113,7 @@ static int mvebu_pcie_probe(struct platform_device *pdev)  		mvebu_pcie_setup_hw(port);  		mvebu_pcie_set_local_dev_nr(port, 1); -		mvebu_sw_pci_bridge_init(port); +		mvebu_pci_bridge_emul_init(port);  	}  	pcie->nports = i; diff --git a/drivers/pci/controller/pcie-cadence-ep.c b/drivers/pci/controller/pcie-cadence-ep.c index 9e87dd7f9ac3..c3a088910f48 100644 --- a/drivers/pci/controller/pcie-cadence-ep.c +++ b/drivers/pci/controller/pcie-cadence-ep.c @@ -258,7 +258,6 @@ static void cdns_pcie_ep_assert_intx(struct cdns_pcie_ep *ep, u8 fn,  				     u8 intx, bool is_asserted)  {  	struct cdns_pcie *pcie = &ep->pcie; -	u32 r = ep->max_regions - 1;  	u32 offset;  	u16 status;  	u8 msg_code; @@ -268,8 +267,8 @@ static void cdns_pcie_ep_assert_intx(struct cdns_pcie_ep *ep, u8 fn,  	/* Set the outbound region if needed. */  	if (unlikely(ep->irq_pci_addr != CDNS_PCIE_EP_IRQ_PCI_ADDR_LEGACY ||  		     ep->irq_pci_fn != fn)) { -		/* Last region was reserved for IRQ writes. */ -		cdns_pcie_set_outbound_region_for_normal_msg(pcie, fn, r, +		/* First region was reserved for IRQ writes. */ +		cdns_pcie_set_outbound_region_for_normal_msg(pcie, fn, 0,  							     ep->irq_phys_addr);  		ep->irq_pci_addr = CDNS_PCIE_EP_IRQ_PCI_ADDR_LEGACY;  		ep->irq_pci_fn = fn; @@ -347,8 +346,8 @@ static int cdns_pcie_ep_send_msi_irq(struct cdns_pcie_ep *ep, u8 fn,  	/* Set the outbound region if needed. */  	if (unlikely(ep->irq_pci_addr != (pci_addr & ~pci_addr_mask) ||  		     ep->irq_pci_fn != fn)) { -		/* Last region was reserved for IRQ writes. */ -		cdns_pcie_set_outbound_region(pcie, fn, ep->max_regions - 1, +		/* First region was reserved for IRQ writes. */ +		cdns_pcie_set_outbound_region(pcie, fn, 0,  					      false,  					      ep->irq_phys_addr,  					      pci_addr & ~pci_addr_mask, @@ -356,7 +355,7 @@ static int cdns_pcie_ep_send_msi_irq(struct cdns_pcie_ep *ep, u8 fn,  		ep->irq_pci_addr = (pci_addr & ~pci_addr_mask);  		ep->irq_pci_fn = fn;  	} -	writew(data, ep->irq_cpu_addr + (pci_addr & pci_addr_mask)); +	writel(data, ep->irq_cpu_addr + (pci_addr & pci_addr_mask));  	return 0;  } @@ -517,6 +516,8 @@ static int cdns_pcie_ep_probe(struct platform_device *pdev)  		goto free_epc_mem;  	}  	ep->irq_pci_addr = CDNS_PCIE_EP_IRQ_PCI_ADDR_NONE; +	/* Reserve region 0 for IRQs */ +	set_bit(0, &ep->ob_region_map);  	return 0; diff --git a/drivers/pci/controller/pcie-cadence-host.c b/drivers/pci/controller/pcie-cadence-host.c index ec394f6a19c8..97e251090b4f 100644 --- a/drivers/pci/controller/pcie-cadence-host.c +++ b/drivers/pci/controller/pcie-cadence-host.c @@ -235,7 +235,6 @@ static int cdns_pcie_host_init(struct device *dev,  static int cdns_pcie_host_probe(struct platform_device *pdev)  { -	const char *type;  	struct device *dev = &pdev->dev;  	struct device_node *np = dev->of_node;  	struct pci_host_bridge *bridge; @@ -268,12 +267,6 @@ static int cdns_pcie_host_probe(struct platform_device *pdev)  	rc->device_id = 0xffff;  	of_property_read_u16(np, "device-id", &rc->device_id); -	type = of_get_property(np, "device_type", NULL); -	if (!type || strcmp(type, "pci")) { -		dev_err(dev, "invalid \"device_type\" %s\n", type); -		return -EINVAL; -	} -  	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "reg");  	pcie->reg_base = devm_ioremap_resource(dev, res);  	if (IS_ERR(pcie->reg_base)) { diff --git a/drivers/pci/controller/pcie-cadence.c b/drivers/pci/controller/pcie-cadence.c index 975bcdd6b5c0..cd795f6fc1e2 100644 --- a/drivers/pci/controller/pcie-cadence.c +++ b/drivers/pci/controller/pcie-cadence.c @@ -190,14 +190,16 @@ int cdns_pcie_init_phy(struct device *dev, struct cdns_pcie *pcie)  	for (i = 0; i < phy_count; i++) {  		of_property_read_string_index(np, "phy-names", i, &name); -		phy[i] = devm_phy_optional_get(dev, name); -		if (IS_ERR(phy)) -			return PTR_ERR(phy); - +		phy[i] = devm_phy_get(dev, name); +		if (IS_ERR(phy[i])) { +			ret = PTR_ERR(phy[i]); +			goto err_phy; +		}  		link[i] = device_link_add(dev, &phy[i]->dev, DL_FLAG_STATELESS);  		if (!link[i]) { +			devm_phy_put(dev, phy[i]);  			ret = -EINVAL; -			goto err_link; +			goto err_phy;  		}  	} @@ -207,13 +209,15 @@ int cdns_pcie_init_phy(struct device *dev, struct cdns_pcie *pcie)  	ret =  cdns_pcie_enable_phy(pcie);  	if (ret) -		goto err_link; +		goto err_phy;  	return 0; -err_link: -	while (--i >= 0) +err_phy: +	while (--i >= 0) {  		device_link_del(link[i]); +		devm_phy_put(dev, phy[i]); +	}  	return ret;  } diff --git a/drivers/pci/controller/pcie-iproc.c b/drivers/pci/controller/pcie-iproc.c index 3160e9342a2f..c20fd6bd68fd 100644 --- a/drivers/pci/controller/pcie-iproc.c +++ b/drivers/pci/controller/pcie-iproc.c @@ -630,14 +630,6 @@ static void __iomem *iproc_pcie_map_cfg_bus(struct iproc_pcie *pcie,  			return (pcie->base + offset);  	} -	/* -	 * PAXC is connected to an internally emulated EP within the SoC.  It -	 * allows only one device. -	 */ -	if (pcie->ep_is_internal) -		if (slot > 0) -			return NULL; -  	return iproc_pcie_map_ep_cfg_reg(pcie, busno, slot, fn, where);  } diff --git a/drivers/pci/controller/pcie-mediatek.c b/drivers/pci/controller/pcie-mediatek.c index 861dda69f366..d069a76cbb95 100644 --- a/drivers/pci/controller/pcie-mediatek.c +++ b/drivers/pci/controller/pcie-mediatek.c @@ -15,6 +15,7 @@  #include <linux/irqdomain.h>  #include <linux/kernel.h>  #include <linux/msi.h> +#include <linux/module.h>  #include <linux/of_address.h>  #include <linux/of_pci.h>  #include <linux/of_platform.h> @@ -162,6 +163,7 @@ struct mtk_pcie_soc {   * @phy: pointer to PHY control block   * @lane: lane count   * @slot: port slot + * @irq: GIC irq   * @irq_domain: legacy INTx IRQ domain   * @inner_domain: inner IRQ domain   * @msi_domain: MSI IRQ domain @@ -182,6 +184,7 @@ struct mtk_pcie_port {  	struct phy *phy;  	u32 lane;  	u32 slot; +	int irq;  	struct irq_domain *irq_domain;  	struct irq_domain *inner_domain;  	struct irq_domain *msi_domain; @@ -225,10 +228,8 @@ static void mtk_pcie_subsys_powerdown(struct mtk_pcie *pcie)  	clk_disable_unprepare(pcie->free_ck); -	if (dev->pm_domain) { -		pm_runtime_put_sync(dev); -		pm_runtime_disable(dev); -	} +	pm_runtime_put_sync(dev); +	pm_runtime_disable(dev);  }  static void mtk_pcie_port_free(struct mtk_pcie_port *port) @@ -337,6 +338,17 @@ static struct mtk_pcie_port *mtk_pcie_find_port(struct pci_bus *bus,  {  	struct mtk_pcie *pcie = bus->sysdata;  	struct mtk_pcie_port *port; +	struct pci_dev *dev = NULL; + +	/* +	 * Walk the bus hierarchy to get the devfn value +	 * of the port in the root bus. +	 */ +	while (bus && bus->number) { +		dev = bus->self; +		bus = dev->bus; +		devfn = dev->devfn; +	}  	list_for_each_entry(port, &pcie->ports, list)  		if (port->slot == PCI_SLOT(devfn)) @@ -383,75 +395,6 @@ static struct pci_ops mtk_pcie_ops_v2 = {  	.write = mtk_pcie_config_write,  }; -static int mtk_pcie_startup_port_v2(struct mtk_pcie_port *port) -{ -	struct mtk_pcie *pcie = port->pcie; -	struct resource *mem = &pcie->mem; -	const struct mtk_pcie_soc *soc = port->pcie->soc; -	u32 val; -	size_t size; -	int err; - -	/* MT7622 platforms need to enable LTSSM and ASPM from PCIe subsys */ -	if (pcie->base) { -		val = readl(pcie->base + PCIE_SYS_CFG_V2); -		val |= PCIE_CSR_LTSSM_EN(port->slot) | -		       PCIE_CSR_ASPM_L1_EN(port->slot); -		writel(val, pcie->base + PCIE_SYS_CFG_V2); -	} - -	/* Assert all reset signals */ -	writel(0, port->base + PCIE_RST_CTRL); - -	/* -	 * Enable PCIe link down reset, if link status changed from link up to -	 * link down, this will reset MAC control registers and configuration -	 * space. -	 */ -	writel(PCIE_LINKDOWN_RST_EN, port->base + PCIE_RST_CTRL); - -	/* De-assert PHY, PE, PIPE, MAC and configuration reset	*/ -	val = readl(port->base + PCIE_RST_CTRL); -	val |= PCIE_PHY_RSTB | PCIE_PERSTB | PCIE_PIPE_SRSTB | -	       PCIE_MAC_SRSTB | PCIE_CRSTB; -	writel(val, port->base + PCIE_RST_CTRL); - -	/* Set up vendor ID and class code */ -	if (soc->need_fix_class_id) { -		val = PCI_VENDOR_ID_MEDIATEK; -		writew(val, port->base + PCIE_CONF_VEND_ID); - -		val = PCI_CLASS_BRIDGE_HOST; -		writew(val, port->base + PCIE_CONF_CLASS_ID); -	} - -	/* 100ms timeout value should be enough for Gen1/2 training */ -	err = readl_poll_timeout(port->base + PCIE_LINK_STATUS_V2, val, -				 !!(val & PCIE_PORT_LINKUP_V2), 20, -				 100 * USEC_PER_MSEC); -	if (err) -		return -ETIMEDOUT; - -	/* Set INTx mask */ -	val = readl(port->base + PCIE_INT_MASK); -	val &= ~INTX_MASK; -	writel(val, port->base + PCIE_INT_MASK); - -	/* Set AHB to PCIe translation windows */ -	size = mem->end - mem->start; -	val = lower_32_bits(mem->start) | AHB2PCIE_SIZE(fls(size)); -	writel(val, port->base + PCIE_AHB_TRANS_BASE0_L); - -	val = upper_32_bits(mem->start); -	writel(val, port->base + PCIE_AHB_TRANS_BASE0_H); - -	/* Set PCIe to AXI translation memory space.*/ -	val = fls(0xffffffff) | WIN_ENABLE; -	writel(val, port->base + PCIE_AXI_WINDOW0); - -	return 0; -} -  static void mtk_compose_msi_msg(struct irq_data *data, struct msi_msg *msg)  {  	struct mtk_pcie_port *port = irq_data_get_irq_chip_data(data); @@ -590,6 +533,27 @@ static void mtk_pcie_enable_msi(struct mtk_pcie_port *port)  	writel(val, port->base + PCIE_INT_MASK);  } +static void mtk_pcie_irq_teardown(struct mtk_pcie *pcie) +{ +	struct mtk_pcie_port *port, *tmp; + +	list_for_each_entry_safe(port, tmp, &pcie->ports, list) { +		irq_set_chained_handler_and_data(port->irq, NULL, NULL); + +		if (port->irq_domain) +			irq_domain_remove(port->irq_domain); + +		if (IS_ENABLED(CONFIG_PCI_MSI)) { +			if (port->msi_domain) +				irq_domain_remove(port->msi_domain); +			if (port->inner_domain) +				irq_domain_remove(port->inner_domain); +		} + +		irq_dispose_mapping(port->irq); +	} +} +  static int mtk_pcie_intx_map(struct irq_domain *domain, unsigned int irq,  			     irq_hw_number_t hwirq)  { @@ -628,8 +592,6 @@ static int mtk_pcie_init_irq_domain(struct mtk_pcie_port *port,  		ret = mtk_pcie_allocate_msi_domains(port);  		if (ret)  			return ret; - -		mtk_pcie_enable_msi(port);  	}  	return 0; @@ -682,7 +644,7 @@ static int mtk_pcie_setup_irq(struct mtk_pcie_port *port,  	struct mtk_pcie *pcie = port->pcie;  	struct device *dev = pcie->dev;  	struct platform_device *pdev = to_platform_device(dev); -	int err, irq; +	int err;  	err = mtk_pcie_init_irq_domain(port, node);  	if (err) { @@ -690,8 +652,81 @@ static int mtk_pcie_setup_irq(struct mtk_pcie_port *port,  		return err;  	} -	irq = platform_get_irq(pdev, port->slot); -	irq_set_chained_handler_and_data(irq, mtk_pcie_intr_handler, port); +	port->irq = platform_get_irq(pdev, port->slot); +	irq_set_chained_handler_and_data(port->irq, +					 mtk_pcie_intr_handler, port); + +	return 0; +} + +static int mtk_pcie_startup_port_v2(struct mtk_pcie_port *port) +{ +	struct mtk_pcie *pcie = port->pcie; +	struct resource *mem = &pcie->mem; +	const struct mtk_pcie_soc *soc = port->pcie->soc; +	u32 val; +	size_t size; +	int err; + +	/* MT7622 platforms need to enable LTSSM and ASPM from PCIe subsys */ +	if (pcie->base) { +		val = readl(pcie->base + PCIE_SYS_CFG_V2); +		val |= PCIE_CSR_LTSSM_EN(port->slot) | +		       PCIE_CSR_ASPM_L1_EN(port->slot); +		writel(val, pcie->base + PCIE_SYS_CFG_V2); +	} + +	/* Assert all reset signals */ +	writel(0, port->base + PCIE_RST_CTRL); + +	/* +	 * Enable PCIe link down reset, if link status changed from link up to +	 * link down, this will reset MAC control registers and configuration +	 * space. +	 */ +	writel(PCIE_LINKDOWN_RST_EN, port->base + PCIE_RST_CTRL); + +	/* De-assert PHY, PE, PIPE, MAC and configuration reset	*/ +	val = readl(port->base + PCIE_RST_CTRL); +	val |= PCIE_PHY_RSTB | PCIE_PERSTB | PCIE_PIPE_SRSTB | +	       PCIE_MAC_SRSTB | PCIE_CRSTB; +	writel(val, port->base + PCIE_RST_CTRL); + +	/* Set up vendor ID and class code */ +	if (soc->need_fix_class_id) { +		val = PCI_VENDOR_ID_MEDIATEK; +		writew(val, port->base + PCIE_CONF_VEND_ID); + +		val = PCI_CLASS_BRIDGE_PCI; +		writew(val, port->base + PCIE_CONF_CLASS_ID); +	} + +	/* 100ms timeout value should be enough for Gen1/2 training */ +	err = readl_poll_timeout(port->base + PCIE_LINK_STATUS_V2, val, +				 !!(val & PCIE_PORT_LINKUP_V2), 20, +				 100 * USEC_PER_MSEC); +	if (err) +		return -ETIMEDOUT; + +	/* Set INTx mask */ +	val = readl(port->base + PCIE_INT_MASK); +	val &= ~INTX_MASK; +	writel(val, port->base + PCIE_INT_MASK); + +	if (IS_ENABLED(CONFIG_PCI_MSI)) +		mtk_pcie_enable_msi(port); + +	/* Set AHB to PCIe translation windows */ +	size = mem->end - mem->start; +	val = lower_32_bits(mem->start) | AHB2PCIE_SIZE(fls(size)); +	writel(val, port->base + PCIE_AHB_TRANS_BASE0_L); + +	val = upper_32_bits(mem->start); +	writel(val, port->base + PCIE_AHB_TRANS_BASE0_H); + +	/* Set PCIe to AXI translation memory space.*/ +	val = fls(0xffffffff) | WIN_ENABLE; +	writel(val, port->base + PCIE_AXI_WINDOW0);  	return 0;  } @@ -987,10 +1022,8 @@ static int mtk_pcie_subsys_powerup(struct mtk_pcie *pcie)  		pcie->free_ck = NULL;  	} -	if (dev->pm_domain) { -		pm_runtime_enable(dev); -		pm_runtime_get_sync(dev); -	} +	pm_runtime_enable(dev); +	pm_runtime_get_sync(dev);  	/* enable top level clock */  	err = clk_prepare_enable(pcie->free_ck); @@ -1002,10 +1035,8 @@ static int mtk_pcie_subsys_powerup(struct mtk_pcie *pcie)  	return 0;  err_free_ck: -	if (dev->pm_domain) { -		pm_runtime_put_sync(dev); -		pm_runtime_disable(dev); -	} +	pm_runtime_put_sync(dev); +	pm_runtime_disable(dev);  	return err;  } @@ -1109,36 +1140,10 @@ static int mtk_pcie_request_resources(struct mtk_pcie *pcie)  	if (err < 0)  		return err; -	devm_pci_remap_iospace(dev, &pcie->pio, pcie->io.start); - -	return 0; -} - -static int mtk_pcie_register_host(struct pci_host_bridge *host) -{ -	struct mtk_pcie *pcie = pci_host_bridge_priv(host); -	struct pci_bus *child; -	int err; - -	host->busnr = pcie->busn.start; -	host->dev.parent = pcie->dev; -	host->ops = pcie->soc->ops; -	host->map_irq = of_irq_parse_and_map_pci; -	host->swizzle_irq = pci_common_swizzle; -	host->sysdata = pcie; - -	err = pci_scan_root_bus_bridge(host); -	if (err < 0) +	err = devm_pci_remap_iospace(dev, &pcie->pio, pcie->io.start); +	if (err)  		return err; -	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); -  	return 0;  } @@ -1168,7 +1173,14 @@ static int mtk_pcie_probe(struct platform_device *pdev)  	if (err)  		goto put_resources; -	err = mtk_pcie_register_host(host); +	host->busnr = pcie->busn.start; +	host->dev.parent = pcie->dev; +	host->ops = pcie->soc->ops; +	host->map_irq = of_irq_parse_and_map_pci; +	host->swizzle_irq = pci_common_swizzle; +	host->sysdata = pcie; + +	err = pci_host_probe(host);  	if (err)  		goto put_resources; @@ -1181,6 +1193,80 @@ put_resources:  	return err;  } + +static void mtk_pcie_free_resources(struct mtk_pcie *pcie) +{ +	struct pci_host_bridge *host = pci_host_bridge_from_priv(pcie); +	struct list_head *windows = &host->windows; + +	pci_free_resource_list(windows); +} + +static int mtk_pcie_remove(struct platform_device *pdev) +{ +	struct mtk_pcie *pcie = platform_get_drvdata(pdev); +	struct pci_host_bridge *host = pci_host_bridge_from_priv(pcie); + +	pci_stop_root_bus(host->bus); +	pci_remove_root_bus(host->bus); +	mtk_pcie_free_resources(pcie); + +	mtk_pcie_irq_teardown(pcie); + +	mtk_pcie_put_resources(pcie); + +	return 0; +} + +static int __maybe_unused mtk_pcie_suspend_noirq(struct device *dev) +{ +	struct mtk_pcie *pcie = dev_get_drvdata(dev); +	struct mtk_pcie_port *port; + +	if (list_empty(&pcie->ports)) +		return 0; + +	list_for_each_entry(port, &pcie->ports, list) { +		clk_disable_unprepare(port->pipe_ck); +		clk_disable_unprepare(port->obff_ck); +		clk_disable_unprepare(port->axi_ck); +		clk_disable_unprepare(port->aux_ck); +		clk_disable_unprepare(port->ahb_ck); +		clk_disable_unprepare(port->sys_ck); +		phy_power_off(port->phy); +		phy_exit(port->phy); +	} + +	clk_disable_unprepare(pcie->free_ck); + +	return 0; +} + +static int __maybe_unused mtk_pcie_resume_noirq(struct device *dev) +{ +	struct mtk_pcie *pcie = dev_get_drvdata(dev); +	struct mtk_pcie_port *port, *tmp; + +	if (list_empty(&pcie->ports)) +		return 0; + +	clk_prepare_enable(pcie->free_ck); + +	list_for_each_entry_safe(port, tmp, &pcie->ports, list) +		mtk_pcie_enable_port(port); + +	/* In case of EP was removed while system suspend. */ +	if (list_empty(&pcie->ports)) +		clk_disable_unprepare(pcie->free_ck); + +	return 0; +} + +static const struct dev_pm_ops mtk_pcie_pm_ops = { +	SET_NOIRQ_SYSTEM_SLEEP_PM_OPS(mtk_pcie_suspend_noirq, +				      mtk_pcie_resume_noirq) +}; +  static const struct mtk_pcie_soc mtk_pcie_soc_v1 = {  	.ops = &mtk_pcie_ops,  	.startup = mtk_pcie_startup_port, @@ -1209,10 +1295,13 @@ static const struct of_device_id mtk_pcie_ids[] = {  static struct platform_driver mtk_pcie_driver = {  	.probe = mtk_pcie_probe, +	.remove = mtk_pcie_remove,  	.driver = {  		.name = "mtk-pcie",  		.of_match_table = mtk_pcie_ids,  		.suppress_bind_attrs = true, +		.pm = &mtk_pcie_pm_ops,  	},  }; -builtin_platform_driver(mtk_pcie_driver); +module_platform_driver(mtk_pcie_driver); +MODULE_LICENSE("GPL v2"); diff --git a/drivers/pci/controller/pcie-mobiveil.c b/drivers/pci/controller/pcie-mobiveil.c index a939e8d31735..77052a0712d0 100644 --- a/drivers/pci/controller/pcie-mobiveil.c +++ b/drivers/pci/controller/pcie-mobiveil.c @@ -301,13 +301,6 @@ static int mobiveil_pcie_parse_dt(struct mobiveil_pcie *pcie)  	struct platform_device *pdev = pcie->pdev;  	struct device_node *node = dev->of_node;  	struct resource *res; -	const char *type; - -	type = of_get_property(node, "device_type", NULL); -	if (!type || strcmp(type, "pci")) { -		dev_err(dev, "invalid \"device_type\" %s\n", type); -		return -EINVAL; -	}  	/* map config resource */  	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, diff --git a/drivers/pci/controller/pcie-xilinx-nwl.c b/drivers/pci/controller/pcie-xilinx-nwl.c index fb32840ce8e6..81538d77f790 100644 --- a/drivers/pci/controller/pcie-xilinx-nwl.c +++ b/drivers/pci/controller/pcie-xilinx-nwl.c @@ -777,16 +777,7 @@ static int nwl_pcie_parse_dt(struct nwl_pcie *pcie,  			     struct platform_device *pdev)  {  	struct device *dev = pcie->dev; -	struct device_node *node = dev->of_node;  	struct resource *res; -	const char *type; - -	/* Check for device type */ -	type = of_get_property(node, "device_type", NULL); -	if (!type || strcmp(type, "pci")) { -		dev_err(dev, "invalid \"device_type\" %s\n", type); -		return -EINVAL; -	}  	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "breg");  	pcie->breg_base = devm_ioremap_resource(dev, res); diff --git a/drivers/pci/controller/pcie-xilinx.c b/drivers/pci/controller/pcie-xilinx.c index 7b1389d8e2a5..9bd1a35cd5d8 100644 --- a/drivers/pci/controller/pcie-xilinx.c +++ b/drivers/pci/controller/pcie-xilinx.c @@ -574,15 +574,8 @@ static int xilinx_pcie_parse_dt(struct xilinx_pcie_port *port)  	struct device *dev = port->dev;  	struct device_node *node = dev->of_node;  	struct resource regs; -	const char *type;  	int err; -	type = of_get_property(node, "device_type", NULL); -	if (!type || strcmp(type, "pci")) { -		dev_err(dev, "invalid \"device_type\" %s\n", type); -		return -EINVAL; -	} -  	err = of_address_to_resource(node, 0, ®s);  	if (err) {  		dev_err(dev, "missing \"reg\" property\n"); diff --git a/drivers/pci/controller/vmd.c b/drivers/pci/controller/vmd.c index fd2dbd7eed7b..e50b0b5815ff 100644 --- a/drivers/pci/controller/vmd.c +++ b/drivers/pci/controller/vmd.c @@ -404,12 +404,10 @@ static int vmd_dma_supported(struct device *dev, u64 mask)  	return vmd_dma_ops(dev)->dma_supported(to_vmd_dev(dev), mask);  } -#ifdef ARCH_HAS_DMA_GET_REQUIRED_MASK  static u64 vmd_get_required_mask(struct device *dev)  {  	return vmd_dma_ops(dev)->get_required_mask(to_vmd_dev(dev));  } -#endif  static void vmd_teardown_dma_ops(struct vmd_dev *vmd)  { @@ -450,9 +448,7 @@ static void vmd_setup_dma_ops(struct vmd_dev *vmd)  	ASSIGN_VMD_DMA_OPS(source, dest, sync_sg_for_device);  	ASSIGN_VMD_DMA_OPS(source, dest, mapping_error);  	ASSIGN_VMD_DMA_OPS(source, dest, dma_supported); -#ifdef ARCH_HAS_DMA_GET_REQUIRED_MASK  	ASSIGN_VMD_DMA_OPS(source, dest, get_required_mask); -#endif  	add_dma_domain(domain);  }  #undef ASSIGN_VMD_DMA_OPS @@ -813,12 +809,12 @@ static void vmd_remove(struct pci_dev *dev)  {  	struct vmd_dev *vmd = pci_get_drvdata(dev); -	vmd_detach_resources(vmd);  	sysfs_remove_link(&vmd->dev->dev.kobj, "domain");  	pci_stop_root_bus(vmd->bus);  	pci_remove_root_bus(vmd->bus);  	vmd_cleanup_srcu(vmd);  	vmd_teardown_dma_ops(vmd); +	vmd_detach_resources(vmd);  	irq_domain_remove(vmd->irq_domain);  } | 
