diff options
| author | Linus Torvalds <[email protected]> | 2021-11-06 14:36:12 -0700 | 
|---|---|---|
| committer | Linus Torvalds <[email protected]> | 2021-11-06 14:36:12 -0700 | 
| commit | 0c5c62ddf88c34bc83b66e4ac9beb2bb0e1887d4 (patch) | |
| tree | f1043e124056c7f2a061f2d2e5240aa687534633 /drivers/pci/controller/dwc | |
| parent | 512b7931ad0561ffe14265f9ff554a3c081b476b (diff) | |
| parent | dda4b381f05d447a0ae31e2e44aeb35d313a311f (diff) | |
Merge tag 'pci-v5.16-changes' of git://git.kernel.org/pub/scm/linux/kernel/git/helgaas/pci
Pull pci updates from Bjorn Helgaas:
 "Enumeration:
   - Conserve IRQs by setting up portdrv IRQs only when there are users
     (Jan Kiszka)
   - Rework and simplify _OSC negotiation for control of PCIe features
     (Joerg Roedel)
   - Remove struct pci_dev.driver pointer since it's redundant with the
     struct device.driver pointer (Uwe Kleine-König)
  Resource management:
   - Coalesce contiguous host bridge apertures from _CRS to accommodate
     BARs that cover more than one aperture (Kai-Heng Feng)
  Sysfs:
   - Check CAP_SYS_ADMIN before parsing user input (Krzysztof
     Wilczyński)
   - Return -EINVAL consistently from "store" functions (Krzysztof
     Wilczyński)
   - Use sysfs_emit() in endpoint "show" functions to avoid buffer
     overruns (Kunihiko Hayashi)
  PCIe native device hotplug:
   - Ignore Link Down/Up caused by resets during error recovery so
     endpoint drivers can remain bound to the device (Lukas Wunner)
  Virtualization:
   - Avoid bus resets on Atheros QCA6174, where they hang the device
     (Ingmar Klein)
   - Work around Pericom PI7C9X2G switch packet drop erratum by using
     store and forward mode instead of cut-through (Nathan Rossi)
   - Avoid trying to enable AtomicOps on VFs; the PF setting applies to
     all VFs (Selvin Xavier)
  MSI:
   - Document that /sys/bus/pci/devices/.../irq contains the legacy INTx
     interrupt or the IRQ of the first MSI (not MSI-X) vector (Barry
     Song)
  VPD:
   - Add pci_read_vpd_any() and pci_write_vpd_any() to access anywhere
     in the possible VPD space; use these to simplify the cxgb3 driver
     (Heiner Kallweit)
  Peer-to-peer DMA:
   - Add (not subtract) the bus offset when calculating DMA address
     (Wang Lu)
  ASPM:
   - Re-enable LTR at Downstream Ports so they don't report Unsupported
     Requests when reset or hot-added devices send LTR messages
     (Mingchuang Qiao)
  Apple PCIe controller driver:
   - Add driver for Apple M1 PCIe controller (Alyssa Rosenzweig, Marc
     Zyngier)
  Cadence PCIe controller driver:
   - Return success when probe succeeds instead of falling into error
     path (Li Chen)
  HiSilicon Kirin PCIe controller driver:
   - Reorganize PHY logic and add support for external PHY drivers
     (Mauro Carvalho Chehab)
   - Support PERST# GPIOs for HiKey970 external PEX 8606 bridge (Mauro
     Carvalho Chehab)
   - Add Kirin 970 support (Mauro Carvalho Chehab)
   - Make driver removable (Mauro Carvalho Chehab)
  Intel VMD host bridge driver:
   - If IOMMU supports interrupt remapping, leave VMD MSI-X remapping
     enabled (Adrian Huang)
   - Number each controller so we can tell them apart in
     /proc/interrupts (Chunguang Xu)
   - Avoid building on UML because VMD depends on x86 bare metal APIs
     (Johannes Berg)
  Marvell Aardvark PCIe controller driver:
   - Define macros for PCI_EXP_DEVCTL_PAYLOAD_* (Pali Rohár)
   - Set Max Payload Size to 512 bytes per Marvell spec (Pali Rohár)
   - Downgrade PIO Response Status messages to debug level (Marek Behún)
   - Preserve CRS SV (Config Request Retry Software Visibility) bit in
     emulated Root Control register (Pali Rohár)
   - Fix issue in configuring reference clock (Pali Rohár)
   - Don't clear status bits for masked interrupts (Pali Rohár)
   - Don't mask unused interrupts (Pali Rohár)
   - Avoid code repetition in advk_pcie_rd_conf() (Marek Behún)
   - Retry config accesses on CRS response (Pali Rohár)
   - Simplify emulated Root Capabilities initialization (Pali Rohár)
   - Fix several link training issues (Pali Rohár)
   - Fix link-up checking via LTSSM (Pali Rohár)
   - Fix reporting of Data Link Layer Link Active (Pali Rohár)
   - Fix emulation of W1C bits (Marek Behún)
   - Fix MSI domain .alloc() method to return zero on success (Marek
     Behún)
   - Read entire 16-bit MSI vector in MSI handler, not just low 8 bits
     (Marek Behún)
   - Clear Root Port I/O Space, Memory Space, and Bus Master Enable bits
     at startup; PCI core will set those as necessary (Pali Rohár)
   - When operating as a Root Port, set class code to "PCI Bridge"
     instead of the default "Mass Storage Controller" (Pali Rohár)
   - Add emulation for PCI_BRIDGE_CTL_BUS_RESET since aardvark doesn't
     implement this per spec (Pali Rohár)
   - Add emulation of option ROM BAR since aardvark doesn't implement
     this per spec (Pali Rohár)
  MediaTek MT7621 PCIe controller driver:
   - Add MediaTek MT7621 PCIe host controller driver and DT binding
     (Sergio Paracuellos)
  Qualcomm PCIe controller driver:
   - Add SC8180x compatible string (Bjorn Andersson)
   - Add endpoint controller driver and DT binding (Manivannan
     Sadhasivam)
   - Restructure to use of_device_get_match_data() (Prasad Malisetty)
   - Add SC7280-specific pcie_1_pipe_clk_src handling (Prasad Malisetty)
  Renesas R-Car PCIe controller driver:
   - Remove unnecessary includes (Geert Uytterhoeven)
  Rockchip DesignWare PCIe controller driver:
   - Add DT binding (Simon Xue)
  Socionext UniPhier Pro5 controller driver:
   - Serialize INTx masking/unmasking (Kunihiko Hayashi)
  Synopsys DesignWare PCIe controller driver:
   - Run dwc .host_init() method before registering MSI interrupt
     handler so we can deal with pending interrupts left by bootloader
     (Bjorn Andersson)
   - Clean up Kconfig dependencies (Andy Shevchenko)
   - Export symbols to allow more modular drivers (Luca Ceresoli)
  TI DRA7xx PCIe controller driver:
   - Allow host and endpoint drivers to be modules (Luca Ceresoli)
   - Enable external clock if present (Luca Ceresoli)
  TI J721E PCIe driver:
   - Disable PHY when probe fails after initializing it (Christophe
     JAILLET)
  MicroSemi Switchtec management driver:
   - Return error to application when command execution fails because an
     out-of-band reset has cleared the device BARs, Memory Space Enable,
     etc (Kelvin Cao)
   - Fix MRPC error status handling issue (Kelvin Cao)
   - Mask out other bits when reading of management VEP instance ID
     (Kelvin Cao)
   - Return EOPNOTSUPP instead of ENOTSUPP from sysfs show functions
     (Kelvin Cao)
   - Add check of event support (Logan Gunthorpe)
  Miscellaneous:
   - Remove unused pci_pool wrappers, which have been replaced by
     dma_pool (Cai Huoqing)
   - Use 'unsigned int' instead of bare 'unsigned' (Krzysztof
     Wilczyński)
   - Use kstrtobool() directly, sans strtobool() wrapper (Krzysztof
     Wilczyński)
   - Fix some sscanf(), sprintf() format mismatches (Krzysztof
     Wilczyński)
   - Update PCI subsystem information in MAINTAINERS (Krzysztof
     Wilczyński)
   - Correct some misspellings (Krzysztof Wilczyński)"
* tag 'pci-v5.16-changes' of git://git.kernel.org/pub/scm/linux/kernel/git/helgaas/pci: (137 commits)
  PCI: Add ACS quirk for Pericom PI7C9X2G switches
  PCI: apple: Configure RID to SID mapper on device addition
  iommu/dart: Exclude MSI doorbell from PCIe device IOVA range
  PCI: apple: Implement MSI support
  PCI: apple: Add INTx and per-port interrupt support
  PCI: kirin: Allow removing the driver
  PCI: kirin: De-init the dwc driver
  PCI: kirin: Disable clkreq during poweroff sequence
  PCI: kirin: Move the power-off code to a common routine
  PCI: kirin: Add power_off support for Kirin 960 PHY
  PCI: kirin: Allow building it as a module
  PCI: kirin: Add MODULE_* macros
  PCI: kirin: Add Kirin 970 compatible
  PCI: kirin: Support PERST# GPIOs for HiKey970 external PEX 8606 bridge
  PCI: apple: Set up reference clocks when probing
  PCI: apple: Add initial hardware bring-up
  PCI: of: Allow matching of an interrupt-map local to a PCI device
  of/irq: Allow matching of an interrupt-map local to an interrupt controller
  irqdomain: Make of_phandle_args_to_fwspec() generally available
  PCI: Do not enable AtomicOps on VFs
  ...
Diffstat (limited to 'drivers/pci/controller/dwc')
| -rw-r--r-- | drivers/pci/controller/dwc/Kconfig | 30 | ||||
| -rw-r--r-- | drivers/pci/controller/dwc/Makefile | 1 | ||||
| -rw-r--r-- | drivers/pci/controller/dwc/pci-dra7xx.c | 22 | ||||
| -rw-r--r-- | drivers/pci/controller/dwc/pci-imx6.c | 2 | ||||
| -rw-r--r-- | drivers/pci/controller/dwc/pcie-designware-ep.c | 3 | ||||
| -rw-r--r-- | drivers/pci/controller/dwc/pcie-designware-host.c | 19 | ||||
| -rw-r--r-- | drivers/pci/controller/dwc/pcie-designware.c | 1 | ||||
| -rw-r--r-- | drivers/pci/controller/dwc/pcie-kirin.c | 646 | ||||
| -rw-r--r-- | drivers/pci/controller/dwc/pcie-qcom-ep.c | 721 | ||||
| -rw-r--r-- | drivers/pci/controller/dwc/pcie-qcom.c | 96 | ||||
| -rw-r--r-- | drivers/pci/controller/dwc/pcie-uniphier.c | 26 | ||||
| -rw-r--r-- | drivers/pci/controller/dwc/pcie-visconti.c | 5 | 
12 files changed, 1370 insertions, 202 deletions
| diff --git a/drivers/pci/controller/dwc/Kconfig b/drivers/pci/controller/dwc/Kconfig index 76c0a63a3f64..62ce3abf0f19 100644 --- a/drivers/pci/controller/dwc/Kconfig +++ b/drivers/pci/controller/dwc/Kconfig @@ -8,22 +8,20 @@ config PCIE_DW  config PCIE_DW_HOST  	bool -	depends on PCI_MSI_IRQ_DOMAIN  	select PCIE_DW  config PCIE_DW_EP  	bool -	depends on PCI_ENDPOINT  	select PCIE_DW  config PCI_DRA7XX -	bool +	tristate  config PCI_DRA7XX_HOST -	bool "TI DRA7xx PCIe controller Host Mode" +	tristate "TI DRA7xx PCIe controller Host Mode"  	depends on SOC_DRA7XX || COMPILE_TEST -	depends on PCI_MSI_IRQ_DOMAIN  	depends on OF && HAS_IOMEM && TI_PIPE3 +	depends on PCI_MSI_IRQ_DOMAIN  	select PCIE_DW_HOST  	select PCI_DRA7XX  	default y if SOC_DRA7XX @@ -36,10 +34,10 @@ config PCI_DRA7XX_HOST  	  This uses the DesignWare core.  config PCI_DRA7XX_EP -	bool "TI DRA7xx PCIe controller Endpoint Mode" +	tristate "TI DRA7xx PCIe controller Endpoint Mode"  	depends on SOC_DRA7XX || COMPILE_TEST -	depends on PCI_ENDPOINT  	depends on OF && HAS_IOMEM && TI_PIPE3 +	depends on PCI_ENDPOINT  	select PCIE_DW_EP  	select PCI_DRA7XX  	help @@ -55,7 +53,7 @@ config PCIE_DW_PLAT  config PCIE_DW_PLAT_HOST  	bool "Platform bus based DesignWare PCIe Controller - Host mode" -	depends on PCI && PCI_MSI_IRQ_DOMAIN +	depends on PCI_MSI_IRQ_DOMAIN  	select PCIE_DW_HOST  	select PCIE_DW_PLAT  	help @@ -138,8 +136,8 @@ config PCI_LAYERSCAPE  	bool "Freescale Layerscape PCIe controller - Host mode"  	depends on OF && (ARM || ARCH_LAYERSCAPE || COMPILE_TEST)  	depends on PCI_MSI_IRQ_DOMAIN -	select MFD_SYSCON  	select PCIE_DW_HOST +	select MFD_SYSCON  	help  	  Say Y here if you want to enable PCIe controller support on Layerscape  	  SoCs to work in Host mode. @@ -180,6 +178,16 @@ config PCIE_QCOM  	  PCIe controller uses the DesignWare core plus Qualcomm-specific  	  hardware wrappers. +config PCIE_QCOM_EP +	tristate "Qualcomm PCIe controller - Endpoint mode" +	depends on OF && (ARCH_QCOM || COMPILE_TEST) +	depends on PCI_ENDPOINT +	select PCIE_DW_EP +	help +	  Say Y here to enable support for the PCIe controllers on Qualcomm SoCs +	  to work in endpoint mode. The PCIe controller uses the DesignWare core +	  plus Qualcomm-specific hardware wrappers. +  config PCIE_ARMADA_8K  	bool "Marvell Armada-8K PCIe controller"  	depends on ARCH_MVEBU || COMPILE_TEST @@ -266,7 +274,7 @@ config PCIE_KEEMBAY_EP  config PCIE_KIRIN  	depends on OF && (ARM64 || COMPILE_TEST) -	bool "HiSilicon Kirin series SoCs PCIe controllers" +	tristate "HiSilicon Kirin series SoCs PCIe controllers"  	depends on PCI_MSI_IRQ_DOMAIN  	select PCIE_DW_HOST  	help @@ -283,8 +291,8 @@ config PCIE_HISI_STB  config PCI_MESON  	tristate "MESON PCIe controller" -	depends on PCI_MSI_IRQ_DOMAIN  	default m if ARCH_MESON +	depends on PCI_MSI_IRQ_DOMAIN  	select PCIE_DW_HOST  	help  	  Say Y here if you want to enable PCI controller support on Amlogic diff --git a/drivers/pci/controller/dwc/Makefile b/drivers/pci/controller/dwc/Makefile index 73244409792c..8ba7b67f5e50 100644 --- a/drivers/pci/controller/dwc/Makefile +++ b/drivers/pci/controller/dwc/Makefile @@ -12,6 +12,7 @@ obj-$(CONFIG_PCI_KEYSTONE) += pci-keystone.o  obj-$(CONFIG_PCI_LAYERSCAPE) += pci-layerscape.o  obj-$(CONFIG_PCI_LAYERSCAPE_EP) += pci-layerscape-ep.o  obj-$(CONFIG_PCIE_QCOM) += pcie-qcom.o +obj-$(CONFIG_PCIE_QCOM_EP) += pcie-qcom-ep.o  obj-$(CONFIG_PCIE_ARMADA_8K) += pcie-armada8k.o  obj-$(CONFIG_PCIE_ARTPEC6) += pcie-artpec6.o  obj-$(CONFIG_PCIE_ROCKCHIP_DW_HOST) += pcie-dw-rockchip.o diff --git a/drivers/pci/controller/dwc/pci-dra7xx.c b/drivers/pci/controller/dwc/pci-dra7xx.c index fbbb78f6885e..a4221f6f3629 100644 --- a/drivers/pci/controller/dwc/pci-dra7xx.c +++ b/drivers/pci/controller/dwc/pci-dra7xx.c @@ -7,6 +7,7 @@   * Authors: Kishon Vijay Abraham I <[email protected]>   */ +#include <linux/clk.h>  #include <linux/delay.h>  #include <linux/device.h>  #include <linux/err.h> @@ -14,7 +15,7 @@  #include <linux/irq.h>  #include <linux/irqdomain.h>  #include <linux/kernel.h> -#include <linux/init.h> +#include <linux/module.h>  #include <linux/of_device.h>  #include <linux/of_gpio.h>  #include <linux/of_pci.h> @@ -90,6 +91,7 @@ struct dra7xx_pcie {  	int			phy_count;	/* DT phy-names count */  	struct phy		**phy;  	struct irq_domain	*irq_domain; +	struct clk              *clk;  	enum dw_pcie_device_mode mode;  }; @@ -607,6 +609,7 @@ static const struct of_device_id of_dra7xx_pcie_match[] = {  	},  	{},  }; +MODULE_DEVICE_TABLE(of, of_dra7xx_pcie_match);  /*   * dra7xx_pcie_unaligned_memaccess: workaround for AM572x/AM571x Errata i870 @@ -740,6 +743,15 @@ static int dra7xx_pcie_probe(struct platform_device *pdev)  	if (!link)  		return -ENOMEM; +	dra7xx->clk = devm_clk_get_optional(dev, NULL); +	if (IS_ERR(dra7xx->clk)) +		return dev_err_probe(dev, PTR_ERR(dra7xx->clk), +				     "clock request failed"); + +	ret = clk_prepare_enable(dra7xx->clk); +	if (ret) +		return ret; +  	for (i = 0; i < phy_count; i++) {  		snprintf(name, sizeof(name), "pcie-phy%d", i);  		phy[i] = devm_phy_get(dev, name); @@ -925,6 +937,8 @@ static void dra7xx_pcie_shutdown(struct platform_device *pdev)  	pm_runtime_disable(dev);  	dra7xx_pcie_disable_phy(dra7xx); + +	clk_disable_unprepare(dra7xx->clk);  }  static const struct dev_pm_ops dra7xx_pcie_pm_ops = { @@ -943,4 +957,8 @@ static struct platform_driver dra7xx_pcie_driver = {  	},  	.shutdown = dra7xx_pcie_shutdown,  }; -builtin_platform_driver(dra7xx_pcie_driver); +module_platform_driver(dra7xx_pcie_driver); + +MODULE_AUTHOR("Kishon Vijay Abraham I <[email protected]>"); +MODULE_DESCRIPTION("PCIe controller driver for TI DRA7xx SoCs"); +MODULE_LICENSE("GPL v2"); diff --git a/drivers/pci/controller/dwc/pci-imx6.c b/drivers/pci/controller/dwc/pci-imx6.c index 80fc98acf097..26f49f797b0f 100644 --- a/drivers/pci/controller/dwc/pci-imx6.c +++ b/drivers/pci/controller/dwc/pci-imx6.c @@ -1132,7 +1132,7 @@ static int imx6_pcie_probe(struct platform_device *pdev)  	/* Limit link speed */  	pci->link_gen = 1; -	ret = of_property_read_u32(node, "fsl,max-link-speed", &pci->link_gen); +	of_property_read_u32(node, "fsl,max-link-speed", &pci->link_gen);  	imx6_pcie->vpcie = devm_regulator_get_optional(&pdev->dev, "vpcie");  	if (IS_ERR(imx6_pcie->vpcie)) { diff --git a/drivers/pci/controller/dwc/pcie-designware-ep.c b/drivers/pci/controller/dwc/pcie-designware-ep.c index 998b698f4085..0eda8236c125 100644 --- a/drivers/pci/controller/dwc/pcie-designware-ep.c +++ b/drivers/pci/controller/dwc/pcie-designware-ep.c @@ -83,6 +83,7 @@ void dw_pcie_ep_reset_bar(struct dw_pcie *pci, enum pci_barno bar)  	for (func_no = 0; func_no < funcs; func_no++)  		__dw_pcie_ep_reset_bar(pci, func_no, bar, 0);  } +EXPORT_SYMBOL_GPL(dw_pcie_ep_reset_bar);  static u8 __dw_pcie_ep_find_next_cap(struct dw_pcie_ep *ep, u8 func_no,  		u8 cap_ptr, u8 cap) @@ -485,6 +486,7 @@ int dw_pcie_ep_raise_legacy_irq(struct dw_pcie_ep *ep, u8 func_no)  	return -EINVAL;  } +EXPORT_SYMBOL_GPL(dw_pcie_ep_raise_legacy_irq);  int dw_pcie_ep_raise_msi_irq(struct dw_pcie_ep *ep, u8 func_no,  			     u8 interrupt_num) @@ -536,6 +538,7 @@ int dw_pcie_ep_raise_msi_irq(struct dw_pcie_ep *ep, u8 func_no,  	return 0;  } +EXPORT_SYMBOL_GPL(dw_pcie_ep_raise_msi_irq);  int dw_pcie_ep_raise_msix_irq_doorbell(struct dw_pcie_ep *ep, u8 func_no,  				       u16 interrupt_num) diff --git a/drivers/pci/controller/dwc/pcie-designware-host.c b/drivers/pci/controller/dwc/pcie-designware-host.c index d1d9b8344ec9..f4755f3a03be 100644 --- a/drivers/pci/controller/dwc/pcie-designware-host.c +++ b/drivers/pci/controller/dwc/pcie-designware-host.c @@ -335,6 +335,16 @@ int dw_pcie_host_init(struct pcie_port *pp)  	if (pci->link_gen < 1)  		pci->link_gen = of_pci_get_max_link_speed(np); +	/* Set default bus ops */ +	bridge->ops = &dw_pcie_ops; +	bridge->child_ops = &dw_child_pcie_ops; + +	if (pp->ops->host_init) { +		ret = pp->ops->host_init(pp); +		if (ret) +			return ret; +	} +  	if (pci_msi_enabled()) {  		pp->has_msi_ctrl = !(pp->ops->msi_host_init ||  				     of_property_read_bool(np, "msi-parent") || @@ -388,15 +398,6 @@ int dw_pcie_host_init(struct pcie_port *pp)  		}  	} -	/* Set default bus ops */ -	bridge->ops = &dw_pcie_ops; -	bridge->child_ops = &dw_child_pcie_ops; - -	if (pp->ops->host_init) { -		ret = pp->ops->host_init(pp); -		if (ret) -			goto err_free_msi; -	}  	dw_pcie_iatu_detect(pci);  	dw_pcie_setup_rc(pp); diff --git a/drivers/pci/controller/dwc/pcie-designware.c b/drivers/pci/controller/dwc/pcie-designware.c index a945f0c0e73d..850b4533f4ef 100644 --- a/drivers/pci/controller/dwc/pcie-designware.c +++ b/drivers/pci/controller/dwc/pcie-designware.c @@ -538,6 +538,7 @@ int dw_pcie_link_up(struct dw_pcie *pci)  	return ((val & PCIE_PORT_DEBUG1_LINK_UP) &&  		(!(val & PCIE_PORT_DEBUG1_LINK_IN_TRAINING)));  } +EXPORT_SYMBOL_GPL(dw_pcie_link_up);  void dw_pcie_upconfig_setup(struct dw_pcie *pci)  { diff --git a/drivers/pci/controller/dwc/pcie-kirin.c b/drivers/pci/controller/dwc/pcie-kirin.c index 026fd1e42a55..095afbccf9c1 100644 --- a/drivers/pci/controller/dwc/pcie-kirin.c +++ b/drivers/pci/controller/dwc/pcie-kirin.c @@ -8,16 +8,18 @@   * Author: Xiaowei Song <[email protected]>   */ -#include <linux/compiler.h>  #include <linux/clk.h> +#include <linux/compiler.h>  #include <linux/delay.h>  #include <linux/err.h>  #include <linux/gpio.h>  #include <linux/interrupt.h>  #include <linux/mfd/syscon.h>  #include <linux/of_address.h> +#include <linux/of_device.h>  #include <linux/of_gpio.h>  #include <linux/of_pci.h> +#include <linux/phy/phy.h>  #include <linux/pci.h>  #include <linux/pci_regs.h>  #include <linux/platform_device.h> @@ -28,26 +30,16 @@  #define to_kirin_pcie(x) dev_get_drvdata((x)->dev) -#define REF_CLK_FREQ			100000000 -  /* PCIe ELBI registers */  #define SOC_PCIECTRL_CTRL0_ADDR		0x000  #define SOC_PCIECTRL_CTRL1_ADDR		0x004 -#define SOC_PCIEPHY_CTRL2_ADDR		0x008 -#define SOC_PCIEPHY_CTRL3_ADDR		0x00c  #define PCIE_ELBI_SLV_DBI_ENABLE	(0x1 << 21)  /* info located in APB */  #define PCIE_APP_LTSSM_ENABLE	0x01c -#define PCIE_APB_PHY_CTRL0	0x0 -#define PCIE_APB_PHY_CTRL1	0x4  #define PCIE_APB_PHY_STATUS0	0x400  #define PCIE_LINKUP_ENABLE	(0x8020)  #define PCIE_LTSSM_ENABLE_BIT	(0x1 << 11) -#define PIPE_CLK_STABLE		(0x1 << 19) -#define PHY_REF_PAD_BIT		(0x1 << 8) -#define PHY_PWR_DOWN_BIT	(0x1 << 22) -#define PHY_RST_ACK_BIT		(0x1 << 16)  /* info located in sysctrl */  #define SCTRL_PCIE_CMOS_OFFSET	0x60 @@ -60,17 +52,70 @@  #define PCIE_DEBOUNCE_PARAM	0xF0F400  #define PCIE_OE_BYPASS		(0x3 << 28) +/* + * Max number of connected PCI slots at an external PCI bridge + * + * This is used on HiKey 970, which has a PEX 8606 bridge with 4 connected + * lanes (lane 0 upstream, and the other three lanes, one connected to an + * in-board Ethernet adapter and the other two connected to M.2 and mini + * PCI slots. + * + * Each slot has a different clock source and uses a separate PERST# pin. + */ +#define MAX_PCI_SLOTS		3 + +enum pcie_kirin_phy_type { +	PCIE_KIRIN_INTERNAL_PHY, +	PCIE_KIRIN_EXTERNAL_PHY +}; + +struct kirin_pcie { +	enum pcie_kirin_phy_type	type; + +	struct dw_pcie	*pci; +	struct regmap   *apb; +	struct phy	*phy; +	void		*phy_priv;	/* only for PCIE_KIRIN_INTERNAL_PHY */ + +	/* DWC PERST# */ +	int		gpio_id_dwc_perst; + +	/* Per-slot PERST# */ +	int		num_slots; +	int		gpio_id_reset[MAX_PCI_SLOTS]; +	const char	*reset_names[MAX_PCI_SLOTS]; + +	/* Per-slot clkreq */ +	int		n_gpio_clkreq; +	int		gpio_id_clkreq[MAX_PCI_SLOTS]; +	const char	*clkreq_names[MAX_PCI_SLOTS]; +}; + +/* + * Kirin 960 PHY. Can't be split into a PHY driver without changing the + * DT schema. + */ + +#define REF_CLK_FREQ			100000000 + +/* PHY info located in APB */ +#define PCIE_APB_PHY_CTRL0	0x0 +#define PCIE_APB_PHY_CTRL1	0x4 +#define PCIE_APB_PHY_STATUS0   0x400 +#define PIPE_CLK_STABLE		BIT(19) +#define PHY_REF_PAD_BIT		BIT(8) +#define PHY_PWR_DOWN_BIT	BIT(22) +#define PHY_RST_ACK_BIT		BIT(16) +  /* peri_crg ctrl */  #define CRGCTRL_PCIE_ASSERT_OFFSET	0x88  #define CRGCTRL_PCIE_ASSERT_BIT		0x8c000000  /* Time for delay */ -#define REF_2_PERST_MIN		20000 +#define REF_2_PERST_MIN		21000  #define REF_2_PERST_MAX		25000  #define PERST_2_ACCESS_MIN	10000  #define PERST_2_ACCESS_MAX	12000 -#define LINK_WAIT_MIN		900 -#define LINK_WAIT_MAX		1000  #define PIPE_CLK_WAIT_MIN	550  #define PIPE_CLK_WAIT_MAX	600  #define TIME_CMOS_MIN		100 @@ -78,118 +123,101 @@  #define TIME_PHY_PD_MIN		10  #define TIME_PHY_PD_MAX		11 -struct kirin_pcie { -	struct dw_pcie	*pci; -	void __iomem	*apb_base; -	void __iomem	*phy_base; +struct hi3660_pcie_phy { +	struct device	*dev; +	void __iomem	*base;  	struct regmap	*crgctrl;  	struct regmap	*sysctrl;  	struct clk	*apb_sys_clk;  	struct clk	*apb_phy_clk;  	struct clk	*phy_ref_clk; -	struct clk	*pcie_aclk; -	struct clk	*pcie_aux_clk; -	int		gpio_id_reset; +	struct clk	*aclk; +	struct clk	*aux_clk;  }; -/* Registers in PCIeCTRL */ -static inline void kirin_apb_ctrl_writel(struct kirin_pcie *kirin_pcie, -					 u32 val, u32 reg) -{ -	writel(val, kirin_pcie->apb_base + reg); -} - -static inline u32 kirin_apb_ctrl_readl(struct kirin_pcie *kirin_pcie, u32 reg) -{ -	return readl(kirin_pcie->apb_base + reg); -} -  /* Registers in PCIePHY */ -static inline void kirin_apb_phy_writel(struct kirin_pcie *kirin_pcie, +static inline void kirin_apb_phy_writel(struct hi3660_pcie_phy *hi3660_pcie_phy,  					u32 val, u32 reg)  { -	writel(val, kirin_pcie->phy_base + reg); +	writel(val, hi3660_pcie_phy->base + reg);  } -static inline u32 kirin_apb_phy_readl(struct kirin_pcie *kirin_pcie, u32 reg) +static inline u32 kirin_apb_phy_readl(struct hi3660_pcie_phy *hi3660_pcie_phy, +				      u32 reg)  { -	return readl(kirin_pcie->phy_base + reg); +	return readl(hi3660_pcie_phy->base + reg);  } -static long kirin_pcie_get_clk(struct kirin_pcie *kirin_pcie, -			       struct platform_device *pdev) +static int hi3660_pcie_phy_get_clk(struct hi3660_pcie_phy *phy)  { -	struct device *dev = &pdev->dev; +	struct device *dev = phy->dev; -	kirin_pcie->phy_ref_clk = devm_clk_get(dev, "pcie_phy_ref"); -	if (IS_ERR(kirin_pcie->phy_ref_clk)) -		return PTR_ERR(kirin_pcie->phy_ref_clk); +	phy->phy_ref_clk = devm_clk_get(dev, "pcie_phy_ref"); +	if (IS_ERR(phy->phy_ref_clk)) +		return PTR_ERR(phy->phy_ref_clk); -	kirin_pcie->pcie_aux_clk = devm_clk_get(dev, "pcie_aux"); -	if (IS_ERR(kirin_pcie->pcie_aux_clk)) -		return PTR_ERR(kirin_pcie->pcie_aux_clk); +	phy->aux_clk = devm_clk_get(dev, "pcie_aux"); +	if (IS_ERR(phy->aux_clk)) +		return PTR_ERR(phy->aux_clk); -	kirin_pcie->apb_phy_clk = devm_clk_get(dev, "pcie_apb_phy"); -	if (IS_ERR(kirin_pcie->apb_phy_clk)) -		return PTR_ERR(kirin_pcie->apb_phy_clk); +	phy->apb_phy_clk = devm_clk_get(dev, "pcie_apb_phy"); +	if (IS_ERR(phy->apb_phy_clk)) +		return PTR_ERR(phy->apb_phy_clk); -	kirin_pcie->apb_sys_clk = devm_clk_get(dev, "pcie_apb_sys"); -	if (IS_ERR(kirin_pcie->apb_sys_clk)) -		return PTR_ERR(kirin_pcie->apb_sys_clk); +	phy->apb_sys_clk = devm_clk_get(dev, "pcie_apb_sys"); +	if (IS_ERR(phy->apb_sys_clk)) +		return PTR_ERR(phy->apb_sys_clk); -	kirin_pcie->pcie_aclk = devm_clk_get(dev, "pcie_aclk"); -	if (IS_ERR(kirin_pcie->pcie_aclk)) -		return PTR_ERR(kirin_pcie->pcie_aclk); +	phy->aclk = devm_clk_get(dev, "pcie_aclk"); +	if (IS_ERR(phy->aclk)) +		return PTR_ERR(phy->aclk);  	return 0;  } -static long kirin_pcie_get_resource(struct kirin_pcie *kirin_pcie, -				    struct platform_device *pdev) +static int hi3660_pcie_phy_get_resource(struct hi3660_pcie_phy *phy)  { -	kirin_pcie->apb_base = -		devm_platform_ioremap_resource_byname(pdev, "apb"); -	if (IS_ERR(kirin_pcie->apb_base)) -		return PTR_ERR(kirin_pcie->apb_base); - -	kirin_pcie->phy_base = -		devm_platform_ioremap_resource_byname(pdev, "phy"); -	if (IS_ERR(kirin_pcie->phy_base)) -		return PTR_ERR(kirin_pcie->phy_base); - -	kirin_pcie->crgctrl = -		syscon_regmap_lookup_by_compatible("hisilicon,hi3660-crgctrl"); -	if (IS_ERR(kirin_pcie->crgctrl)) -		return PTR_ERR(kirin_pcie->crgctrl); - -	kirin_pcie->sysctrl = -		syscon_regmap_lookup_by_compatible("hisilicon,hi3660-sctrl"); -	if (IS_ERR(kirin_pcie->sysctrl)) -		return PTR_ERR(kirin_pcie->sysctrl); +	struct device *dev = phy->dev; +	struct platform_device *pdev; + +	/* registers */ +	pdev = container_of(dev, struct platform_device, dev); + +	phy->base = devm_platform_ioremap_resource_byname(pdev, "phy"); +	if (IS_ERR(phy->base)) +		return PTR_ERR(phy->base); + +	phy->crgctrl = syscon_regmap_lookup_by_compatible("hisilicon,hi3660-crgctrl"); +	if (IS_ERR(phy->crgctrl)) +		return PTR_ERR(phy->crgctrl); + +	phy->sysctrl = syscon_regmap_lookup_by_compatible("hisilicon,hi3660-sctrl"); +	if (IS_ERR(phy->sysctrl)) +		return PTR_ERR(phy->sysctrl);  	return 0;  } -static int kirin_pcie_phy_init(struct kirin_pcie *kirin_pcie) +static int hi3660_pcie_phy_start(struct hi3660_pcie_phy *phy)  { -	struct device *dev = kirin_pcie->pci->dev; +	struct device *dev = phy->dev;  	u32 reg_val; -	reg_val = kirin_apb_phy_readl(kirin_pcie, PCIE_APB_PHY_CTRL1); +	reg_val = kirin_apb_phy_readl(phy, PCIE_APB_PHY_CTRL1);  	reg_val &= ~PHY_REF_PAD_BIT; -	kirin_apb_phy_writel(kirin_pcie, reg_val, PCIE_APB_PHY_CTRL1); +	kirin_apb_phy_writel(phy, reg_val, PCIE_APB_PHY_CTRL1); -	reg_val = kirin_apb_phy_readl(kirin_pcie, PCIE_APB_PHY_CTRL0); +	reg_val = kirin_apb_phy_readl(phy, PCIE_APB_PHY_CTRL0);  	reg_val &= ~PHY_PWR_DOWN_BIT; -	kirin_apb_phy_writel(kirin_pcie, reg_val, PCIE_APB_PHY_CTRL0); +	kirin_apb_phy_writel(phy, reg_val, PCIE_APB_PHY_CTRL0);  	usleep_range(TIME_PHY_PD_MIN, TIME_PHY_PD_MAX); -	reg_val = kirin_apb_phy_readl(kirin_pcie, PCIE_APB_PHY_CTRL1); +	reg_val = kirin_apb_phy_readl(phy, PCIE_APB_PHY_CTRL1);  	reg_val &= ~PHY_RST_ACK_BIT; -	kirin_apb_phy_writel(kirin_pcie, reg_val, PCIE_APB_PHY_CTRL1); +	kirin_apb_phy_writel(phy, reg_val, PCIE_APB_PHY_CTRL1);  	usleep_range(PIPE_CLK_WAIT_MIN, PIPE_CLK_WAIT_MAX); -	reg_val = kirin_apb_phy_readl(kirin_pcie, PCIE_APB_PHY_STATUS0); +	reg_val = kirin_apb_phy_readl(phy, PCIE_APB_PHY_STATUS0);  	if (reg_val & PIPE_CLK_STABLE) {  		dev_err(dev, "PIPE clk is not stable\n");  		return -EINVAL; @@ -198,102 +226,274 @@ static int kirin_pcie_phy_init(struct kirin_pcie *kirin_pcie)  	return 0;  } -static void kirin_pcie_oe_enable(struct kirin_pcie *kirin_pcie) +static void hi3660_pcie_phy_oe_enable(struct hi3660_pcie_phy *phy)  {  	u32 val; -	regmap_read(kirin_pcie->sysctrl, SCTRL_PCIE_OE_OFFSET, &val); +	regmap_read(phy->sysctrl, SCTRL_PCIE_OE_OFFSET, &val);  	val |= PCIE_DEBOUNCE_PARAM;  	val &= ~PCIE_OE_BYPASS; -	regmap_write(kirin_pcie->sysctrl, SCTRL_PCIE_OE_OFFSET, val); +	regmap_write(phy->sysctrl, SCTRL_PCIE_OE_OFFSET, val);  } -static int kirin_pcie_clk_ctrl(struct kirin_pcie *kirin_pcie, bool enable) +static int hi3660_pcie_phy_clk_ctrl(struct hi3660_pcie_phy *phy, bool enable)  {  	int ret = 0;  	if (!enable)  		goto close_clk; -	ret = clk_set_rate(kirin_pcie->phy_ref_clk, REF_CLK_FREQ); +	ret = clk_set_rate(phy->phy_ref_clk, REF_CLK_FREQ);  	if (ret)  		return ret; -	ret = clk_prepare_enable(kirin_pcie->phy_ref_clk); +	ret = clk_prepare_enable(phy->phy_ref_clk);  	if (ret)  		return ret; -	ret = clk_prepare_enable(kirin_pcie->apb_sys_clk); +	ret = clk_prepare_enable(phy->apb_sys_clk);  	if (ret)  		goto apb_sys_fail; -	ret = clk_prepare_enable(kirin_pcie->apb_phy_clk); +	ret = clk_prepare_enable(phy->apb_phy_clk);  	if (ret)  		goto apb_phy_fail; -	ret = clk_prepare_enable(kirin_pcie->pcie_aclk); +	ret = clk_prepare_enable(phy->aclk);  	if (ret)  		goto aclk_fail; -	ret = clk_prepare_enable(kirin_pcie->pcie_aux_clk); +	ret = clk_prepare_enable(phy->aux_clk);  	if (ret)  		goto aux_clk_fail;  	return 0;  close_clk: -	clk_disable_unprepare(kirin_pcie->pcie_aux_clk); +	clk_disable_unprepare(phy->aux_clk);  aux_clk_fail: -	clk_disable_unprepare(kirin_pcie->pcie_aclk); +	clk_disable_unprepare(phy->aclk);  aclk_fail: -	clk_disable_unprepare(kirin_pcie->apb_phy_clk); +	clk_disable_unprepare(phy->apb_phy_clk);  apb_phy_fail: -	clk_disable_unprepare(kirin_pcie->apb_sys_clk); +	clk_disable_unprepare(phy->apb_sys_clk);  apb_sys_fail: -	clk_disable_unprepare(kirin_pcie->phy_ref_clk); +	clk_disable_unprepare(phy->phy_ref_clk);  	return ret;  } -static int kirin_pcie_power_on(struct kirin_pcie *kirin_pcie) +static int hi3660_pcie_phy_power_on(struct kirin_pcie *pcie)  { +	struct hi3660_pcie_phy *phy = pcie->phy_priv;  	int ret;  	/* Power supply for Host */ -	regmap_write(kirin_pcie->sysctrl, +	regmap_write(phy->sysctrl,  		     SCTRL_PCIE_CMOS_OFFSET, SCTRL_PCIE_CMOS_BIT);  	usleep_range(TIME_CMOS_MIN, TIME_CMOS_MAX); -	kirin_pcie_oe_enable(kirin_pcie); -	ret = kirin_pcie_clk_ctrl(kirin_pcie, true); +	hi3660_pcie_phy_oe_enable(phy); + +	ret = hi3660_pcie_phy_clk_ctrl(phy, true);  	if (ret)  		return ret;  	/* ISO disable, PCIeCtrl, PHY assert and clk gate clear */ -	regmap_write(kirin_pcie->sysctrl, +	regmap_write(phy->sysctrl,  		     SCTRL_PCIE_ISO_OFFSET, SCTRL_PCIE_ISO_BIT); -	regmap_write(kirin_pcie->crgctrl, +	regmap_write(phy->crgctrl,  		     CRGCTRL_PCIE_ASSERT_OFFSET, CRGCTRL_PCIE_ASSERT_BIT); -	regmap_write(kirin_pcie->sysctrl, +	regmap_write(phy->sysctrl,  		     SCTRL_PCIE_HPCLK_OFFSET, SCTRL_PCIE_HPCLK_BIT); -	ret = kirin_pcie_phy_init(kirin_pcie); +	ret = hi3660_pcie_phy_start(phy);  	if (ret) -		goto close_clk; +		goto disable_clks; -	/* perst assert Endpoint */ -	if (!gpio_request(kirin_pcie->gpio_id_reset, "pcie_perst")) { -		usleep_range(REF_2_PERST_MIN, REF_2_PERST_MAX); -		ret = gpio_direction_output(kirin_pcie->gpio_id_reset, 1); -		if (ret) -			goto close_clk; -		usleep_range(PERST_2_ACCESS_MIN, PERST_2_ACCESS_MAX); +	return 0; + +disable_clks: +	hi3660_pcie_phy_clk_ctrl(phy, false); +	return ret; +} + +static int hi3660_pcie_phy_init(struct platform_device *pdev, +				struct kirin_pcie *pcie) +{ +	struct device *dev = &pdev->dev; +	struct hi3660_pcie_phy *phy; +	int ret; +	phy = devm_kzalloc(dev, sizeof(*phy), GFP_KERNEL); +	if (!phy) +		return -ENOMEM; + +	pcie->phy_priv = phy; +	phy->dev = dev; + +	/* registers */ +	pdev = container_of(dev, struct platform_device, dev); + +	ret = hi3660_pcie_phy_get_clk(phy); +	if (ret) +		return ret; + +	return hi3660_pcie_phy_get_resource(phy); +} + +static int hi3660_pcie_phy_power_off(struct kirin_pcie *pcie) +{ +	struct hi3660_pcie_phy *phy = pcie->phy_priv; + +	/* Drop power supply for Host */ +	regmap_write(phy->sysctrl, SCTRL_PCIE_CMOS_OFFSET, 0x00); + +	hi3660_pcie_phy_clk_ctrl(phy, false); + +	return 0; +} + +/* + * The non-PHY part starts here + */ + +static const struct regmap_config pcie_kirin_regmap_conf = { +	.name = "kirin_pcie_apb", +	.reg_bits = 32, +	.val_bits = 32, +	.reg_stride = 4, +}; + +static int kirin_pcie_get_gpio_enable(struct kirin_pcie *pcie, +				      struct platform_device *pdev) +{ +	struct device *dev = &pdev->dev; +	struct device_node *np = dev->of_node; +	char name[32]; +	int ret, i; + +	/* This is an optional property */ +	ret = of_gpio_named_count(np, "hisilicon,clken-gpios"); +	if (ret < 0)  		return 0; + +	if (ret > MAX_PCI_SLOTS) { +		dev_err(dev, "Too many GPIO clock requests!\n"); +		return -EINVAL;  	} -close_clk: -	kirin_pcie_clk_ctrl(kirin_pcie, false); +	pcie->n_gpio_clkreq = ret; + +	for (i = 0; i < pcie->n_gpio_clkreq; i++) { +		pcie->gpio_id_clkreq[i] = of_get_named_gpio(dev->of_node, +						    "hisilicon,clken-gpios", i); +		if (pcie->gpio_id_clkreq[i] < 0) +			return pcie->gpio_id_clkreq[i]; + +		sprintf(name, "pcie_clkreq_%d", i); +		pcie->clkreq_names[i] = devm_kstrdup_const(dev, name, +							    GFP_KERNEL); +		if (!pcie->clkreq_names[i]) +			return -ENOMEM; +	} + +	return 0; +} + +static int kirin_pcie_parse_port(struct kirin_pcie *pcie, +				 struct platform_device *pdev, +				 struct device_node *node) +{ +	struct device *dev = &pdev->dev; +	struct device_node *parent, *child; +	int ret, slot, i; +	char name[32]; + +	for_each_available_child_of_node(node, parent) { +		for_each_available_child_of_node(parent, child) { +			i = pcie->num_slots; + +			pcie->gpio_id_reset[i] = of_get_named_gpio(child, +							"reset-gpios", 0); +			if (pcie->gpio_id_reset[i] < 0) +				continue; + +			pcie->num_slots++; +			if (pcie->num_slots > MAX_PCI_SLOTS) { +				dev_err(dev, "Too many PCI slots!\n"); +				ret = -EINVAL; +				goto put_node; +			} + +			ret = of_pci_get_devfn(child); +			if (ret < 0) { +				dev_err(dev, "failed to parse devfn: %d\n", ret); +				goto put_node; +			} + +			slot = PCI_SLOT(ret); + +			sprintf(name, "pcie_perst_%d", slot); +			pcie->reset_names[i] = devm_kstrdup_const(dev, name, +								GFP_KERNEL); +			if (!pcie->reset_names[i]) { +				ret = -ENOMEM; +				goto put_node; +			} +		} +	} + +	return 0; + +put_node: +	of_node_put(child); +	of_node_put(parent); +	return ret; +} + +static long kirin_pcie_get_resource(struct kirin_pcie *kirin_pcie, +				    struct platform_device *pdev) +{ +	struct device *dev = &pdev->dev; +	struct device_node *child, *node = dev->of_node; +	void __iomem *apb_base; +	int ret; + +	apb_base = devm_platform_ioremap_resource_byname(pdev, "apb"); +	if (IS_ERR(apb_base)) +		return PTR_ERR(apb_base); + +	kirin_pcie->apb = devm_regmap_init_mmio(dev, apb_base, +						&pcie_kirin_regmap_conf); +	if (IS_ERR(kirin_pcie->apb)) +		return PTR_ERR(kirin_pcie->apb); + +	/* pcie internal PERST# gpio */ +	kirin_pcie->gpio_id_dwc_perst = of_get_named_gpio(dev->of_node, +							  "reset-gpios", 0); +	if (kirin_pcie->gpio_id_dwc_perst == -EPROBE_DEFER) { +		return -EPROBE_DEFER; +	} else if (!gpio_is_valid(kirin_pcie->gpio_id_dwc_perst)) { +		dev_err(dev, "unable to get a valid gpio pin\n"); +		return -ENODEV; +	} + +	ret = kirin_pcie_get_gpio_enable(kirin_pcie, pdev); +	if (ret) +		return ret; + +	/* Parse OF children */ +	for_each_available_child_of_node(node, child) { +		ret = kirin_pcie_parse_port(kirin_pcie, pdev, child); +		if (ret) +			goto put_node; +	} + +	return 0; + +put_node: +	of_node_put(child);  	return ret;  } @@ -302,13 +502,13 @@ static void kirin_pcie_sideband_dbi_w_mode(struct kirin_pcie *kirin_pcie,  {  	u32 val; -	val = kirin_apb_ctrl_readl(kirin_pcie, SOC_PCIECTRL_CTRL0_ADDR); +	regmap_read(kirin_pcie->apb, SOC_PCIECTRL_CTRL0_ADDR, &val);  	if (on)  		val = val | PCIE_ELBI_SLV_DBI_ENABLE;  	else  		val = val & ~PCIE_ELBI_SLV_DBI_ENABLE; -	kirin_apb_ctrl_writel(kirin_pcie, val, SOC_PCIECTRL_CTRL0_ADDR); +	regmap_write(kirin_pcie->apb, SOC_PCIECTRL_CTRL0_ADDR, val);  }  static void kirin_pcie_sideband_dbi_r_mode(struct kirin_pcie *kirin_pcie, @@ -316,13 +516,13 @@ static void kirin_pcie_sideband_dbi_r_mode(struct kirin_pcie *kirin_pcie,  {  	u32 val; -	val = kirin_apb_ctrl_readl(kirin_pcie, SOC_PCIECTRL_CTRL1_ADDR); +	regmap_read(kirin_pcie->apb, SOC_PCIECTRL_CTRL1_ADDR, &val);  	if (on)  		val = val | PCIE_ELBI_SLV_DBI_ENABLE;  	else  		val = val & ~PCIE_ELBI_SLV_DBI_ENABLE; -	kirin_apb_ctrl_writel(kirin_pcie, val, SOC_PCIECTRL_CTRL1_ADDR); +	regmap_write(kirin_pcie->apb, SOC_PCIECTRL_CTRL1_ADDR, val);  }  static int kirin_pcie_rd_own_conf(struct pci_bus *bus, unsigned int devfn, @@ -351,9 +551,32 @@ static int kirin_pcie_wr_own_conf(struct pci_bus *bus, unsigned int devfn,  	return PCIBIOS_SUCCESSFUL;  } +static int kirin_pcie_add_bus(struct pci_bus *bus) +{ +	struct dw_pcie *pci = to_dw_pcie_from_pp(bus->sysdata); +	struct kirin_pcie *kirin_pcie = to_kirin_pcie(pci); +	int i, ret; + +	if (!kirin_pcie->num_slots) +		return 0; + +	/* Send PERST# to each slot */ +	for (i = 0; i < kirin_pcie->num_slots; i++) { +		ret = gpio_direction_output(kirin_pcie->gpio_id_reset[i], 1); +		if (ret) { +			dev_err(pci->dev, "PERST# %s error: %d\n", +				kirin_pcie->reset_names[i], ret); +		} +	} +	usleep_range(PERST_2_ACCESS_MIN, PERST_2_ACCESS_MAX); + +	return 0; +} +  static struct pci_ops kirin_pci_ops = {  	.read = kirin_pcie_rd_own_conf,  	.write = kirin_pcie_wr_own_conf, +	.add_bus = kirin_pcie_add_bus,  };  static u32 kirin_pcie_read_dbi(struct dw_pcie *pci, void __iomem *base, @@ -382,8 +605,9 @@ static void kirin_pcie_write_dbi(struct dw_pcie *pci, void __iomem *base,  static int kirin_pcie_link_up(struct dw_pcie *pci)  {  	struct kirin_pcie *kirin_pcie = to_kirin_pcie(pci); -	u32 val = kirin_apb_ctrl_readl(kirin_pcie, PCIE_APB_PHY_STATUS0); +	u32 val; +	regmap_read(kirin_pcie->apb, PCIE_APB_PHY_STATUS0, &val);  	if ((val & PCIE_LINKUP_ENABLE) == PCIE_LINKUP_ENABLE)  		return 1; @@ -395,8 +619,8 @@ static int kirin_pcie_start_link(struct dw_pcie *pci)  	struct kirin_pcie *kirin_pcie = to_kirin_pcie(pci);  	/* assert LTSSM enable */ -	kirin_apb_ctrl_writel(kirin_pcie, PCIE_LTSSM_ENABLE_BIT, -			      PCIE_APP_LTSSM_ENABLE); +	regmap_write(kirin_pcie->apb, PCIE_APP_LTSSM_ENABLE, +		     PCIE_LTSSM_ENABLE_BIT);  	return 0;  } @@ -408,6 +632,44 @@ static int kirin_pcie_host_init(struct pcie_port *pp)  	return 0;  } +static int kirin_pcie_gpio_request(struct kirin_pcie *kirin_pcie, +				   struct device *dev) +{ +	int ret, i; + +	for (i = 0; i < kirin_pcie->num_slots; i++) { +		if (!gpio_is_valid(kirin_pcie->gpio_id_reset[i])) { +			dev_err(dev, "unable to get a valid %s gpio\n", +				kirin_pcie->reset_names[i]); +			return -ENODEV; +		} + +		ret = devm_gpio_request(dev, kirin_pcie->gpio_id_reset[i], +					kirin_pcie->reset_names[i]); +		if (ret) +			return ret; +	} + +	for (i = 0; i < kirin_pcie->n_gpio_clkreq; i++) { +		if (!gpio_is_valid(kirin_pcie->gpio_id_clkreq[i])) { +			dev_err(dev, "unable to get a valid %s gpio\n", +				kirin_pcie->clkreq_names[i]); +			return -ENODEV; +		} + +		ret = devm_gpio_request(dev, kirin_pcie->gpio_id_clkreq[i], +					kirin_pcie->clkreq_names[i]); +		if (ret) +			return ret; + +		ret = gpio_direction_output(kirin_pcie->gpio_id_clkreq[i], 0); +		if (ret) +			return ret; +	} + +	return 0; +} +  static const struct dw_pcie_ops kirin_dw_pcie_ops = {  	.read_dbi = kirin_pcie_read_dbi,  	.write_dbi = kirin_pcie_write_dbi, @@ -419,8 +681,99 @@ static const struct dw_pcie_host_ops kirin_pcie_host_ops = {  	.host_init = kirin_pcie_host_init,  }; +static int kirin_pcie_power_off(struct kirin_pcie *kirin_pcie) +{ +	int i; + +	if (kirin_pcie->type == PCIE_KIRIN_INTERNAL_PHY) +		return hi3660_pcie_phy_power_off(kirin_pcie); + +	for (i = 0; i < kirin_pcie->n_gpio_clkreq; i++) +		gpio_direction_output(kirin_pcie->gpio_id_clkreq[i], 1); + +	phy_power_off(kirin_pcie->phy); +	phy_exit(kirin_pcie->phy); + +	return 0; +} + +static int kirin_pcie_power_on(struct platform_device *pdev, +			       struct kirin_pcie *kirin_pcie) +{ +	struct device *dev = &pdev->dev; +	int ret; + +	if (kirin_pcie->type == PCIE_KIRIN_INTERNAL_PHY) { +		ret = hi3660_pcie_phy_init(pdev, kirin_pcie); +		if (ret) +			return ret; + +		ret = hi3660_pcie_phy_power_on(kirin_pcie); +		if (ret) +			return ret; +	} else { +		kirin_pcie->phy = devm_of_phy_get(dev, dev->of_node, NULL); +		if (IS_ERR(kirin_pcie->phy)) +			return PTR_ERR(kirin_pcie->phy); + +		ret = kirin_pcie_gpio_request(kirin_pcie, dev); +		if (ret) +			return ret; + +		ret = phy_init(kirin_pcie->phy); +		if (ret) +			goto err; + +		ret = phy_power_on(kirin_pcie->phy); +		if (ret) +			goto err; +	} + +	/* perst assert Endpoint */ +	usleep_range(REF_2_PERST_MIN, REF_2_PERST_MAX); + +	if (!gpio_request(kirin_pcie->gpio_id_dwc_perst, "pcie_perst_bridge")) { +		ret = gpio_direction_output(kirin_pcie->gpio_id_dwc_perst, 1); +		if (ret) +			goto err; +	} + +	usleep_range(PERST_2_ACCESS_MIN, PERST_2_ACCESS_MAX); + +	return 0; +err: +	kirin_pcie_power_off(kirin_pcie); + +	return ret; +} + +static int __exit kirin_pcie_remove(struct platform_device *pdev) +{ +	struct kirin_pcie *kirin_pcie = platform_get_drvdata(pdev); + +	dw_pcie_host_deinit(&kirin_pcie->pci->pp); + +	kirin_pcie_power_off(kirin_pcie); + +	return 0; +} + +static const struct of_device_id kirin_pcie_match[] = { +	{ +		.compatible = "hisilicon,kirin960-pcie", +		.data = (void *)PCIE_KIRIN_INTERNAL_PHY +	}, +	{ +		.compatible = "hisilicon,kirin970-pcie", +		.data = (void *)PCIE_KIRIN_EXTERNAL_PHY +	}, +	{}, +}; +  static int kirin_pcie_probe(struct platform_device *pdev)  { +	enum pcie_kirin_phy_type phy_type; +	const struct of_device_id *of_id;  	struct device *dev = &pdev->dev;  	struct kirin_pcie *kirin_pcie;  	struct dw_pcie *pci; @@ -431,6 +784,14 @@ static int kirin_pcie_probe(struct platform_device *pdev)  		return -EINVAL;  	} +	of_id = of_match_device(kirin_pcie_match, dev); +	if (!of_id) { +		dev_err(dev, "OF data missing\n"); +		return -EINVAL; +	} + +	phy_type = (long)of_id->data; +  	kirin_pcie = devm_kzalloc(dev, sizeof(struct kirin_pcie), GFP_KERNEL);  	if (!kirin_pcie)  		return -ENOMEM; @@ -443,44 +804,33 @@ static int kirin_pcie_probe(struct platform_device *pdev)  	pci->ops = &kirin_dw_pcie_ops;  	pci->pp.ops = &kirin_pcie_host_ops;  	kirin_pcie->pci = pci; - -	ret = kirin_pcie_get_clk(kirin_pcie, pdev); -	if (ret) -		return ret; +	kirin_pcie->type = phy_type;  	ret = kirin_pcie_get_resource(kirin_pcie, pdev);  	if (ret)  		return ret; -	kirin_pcie->gpio_id_reset = of_get_named_gpio(dev->of_node, -						      "reset-gpios", 0); -	if (kirin_pcie->gpio_id_reset == -EPROBE_DEFER) { -		return -EPROBE_DEFER; -	} else if (!gpio_is_valid(kirin_pcie->gpio_id_reset)) { -		dev_err(dev, "unable to get a valid gpio pin\n"); -		return -ENODEV; -	} +	platform_set_drvdata(pdev, kirin_pcie); -	ret = kirin_pcie_power_on(kirin_pcie); +	ret = kirin_pcie_power_on(pdev, kirin_pcie);  	if (ret)  		return ret; -	platform_set_drvdata(pdev, kirin_pcie); -  	return dw_pcie_host_init(&pci->pp);  } -static const struct of_device_id kirin_pcie_match[] = { -	{ .compatible = "hisilicon,kirin960-pcie" }, -	{}, -}; -  static struct platform_driver kirin_pcie_driver = {  	.probe			= kirin_pcie_probe, +	.remove	        	= __exit_p(kirin_pcie_remove),  	.driver			= {  		.name			= "kirin-pcie", -		.of_match_table = kirin_pcie_match, -		.suppress_bind_attrs = true, +		.of_match_table		= kirin_pcie_match, +		.suppress_bind_attrs	= true,  	},  }; -builtin_platform_driver(kirin_pcie_driver); +module_platform_driver(kirin_pcie_driver); + +MODULE_DEVICE_TABLE(of, kirin_pcie_match); +MODULE_DESCRIPTION("PCIe host controller driver for Kirin Phone SoCs"); +MODULE_AUTHOR("Xiaowei Song <[email protected]>"); +MODULE_LICENSE("GPL v2"); diff --git a/drivers/pci/controller/dwc/pcie-qcom-ep.c b/drivers/pci/controller/dwc/pcie-qcom-ep.c new file mode 100644 index 000000000000..7b17da2f9b3f --- /dev/null +++ b/drivers/pci/controller/dwc/pcie-qcom-ep.c @@ -0,0 +1,721 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * Qualcomm PCIe Endpoint controller driver + * + * Copyright (c) 2020, The Linux Foundation. All rights reserved. + * Author: Siddartha Mohanadoss <[email protected] + * + * Copyright (c) 2021, Linaro Ltd. + * Author: Manivannan Sadhasivam <[email protected] + */ + +#include <linux/clk.h> +#include <linux/delay.h> +#include <linux/gpio/consumer.h> +#include <linux/mfd/syscon.h> +#include <linux/phy/phy.h> +#include <linux/platform_device.h> +#include <linux/pm_domain.h> +#include <linux/regmap.h> +#include <linux/reset.h> + +#include "pcie-designware.h" + +/* PARF registers */ +#define PARF_SYS_CTRL				0x00 +#define PARF_DB_CTRL				0x10 +#define PARF_PM_CTRL				0x20 +#define PARF_MHI_BASE_ADDR_LOWER		0x178 +#define PARF_MHI_BASE_ADDR_UPPER		0x17c +#define PARF_DEBUG_INT_EN			0x190 +#define PARF_AXI_MSTR_RD_HALT_NO_WRITES		0x1a4 +#define PARF_AXI_MSTR_WR_ADDR_HALT		0x1a8 +#define PARF_Q2A_FLUSH				0x1ac +#define PARF_LTSSM				0x1b0 +#define PARF_CFG_BITS				0x210 +#define PARF_INT_ALL_STATUS			0x224 +#define PARF_INT_ALL_CLEAR			0x228 +#define PARF_INT_ALL_MASK			0x22c +#define PARF_SLV_ADDR_MSB_CTRL			0x2c0 +#define PARF_DBI_BASE_ADDR			0x350 +#define PARF_DBI_BASE_ADDR_HI			0x354 +#define PARF_SLV_ADDR_SPACE_SIZE		0x358 +#define PARF_SLV_ADDR_SPACE_SIZE_HI		0x35c +#define PARF_ATU_BASE_ADDR			0x634 +#define PARF_ATU_BASE_ADDR_HI			0x638 +#define PARF_SRIS_MODE				0x644 +#define PARF_DEVICE_TYPE			0x1000 +#define PARF_BDF_TO_SID_CFG			0x2c00 + +/* PARF_INT_ALL_{STATUS/CLEAR/MASK} register fields */ +#define PARF_INT_ALL_LINK_DOWN			BIT(1) +#define PARF_INT_ALL_BME			BIT(2) +#define PARF_INT_ALL_PM_TURNOFF			BIT(3) +#define PARF_INT_ALL_DEBUG			BIT(4) +#define PARF_INT_ALL_LTR			BIT(5) +#define PARF_INT_ALL_MHI_Q6			BIT(6) +#define PARF_INT_ALL_MHI_A7			BIT(7) +#define PARF_INT_ALL_DSTATE_CHANGE		BIT(8) +#define PARF_INT_ALL_L1SUB_TIMEOUT		BIT(9) +#define PARF_INT_ALL_MMIO_WRITE			BIT(10) +#define PARF_INT_ALL_CFG_WRITE			BIT(11) +#define PARF_INT_ALL_BRIDGE_FLUSH_N		BIT(12) +#define PARF_INT_ALL_LINK_UP			BIT(13) +#define PARF_INT_ALL_AER_LEGACY			BIT(14) +#define PARF_INT_ALL_PLS_ERR			BIT(15) +#define PARF_INT_ALL_PME_LEGACY			BIT(16) +#define PARF_INT_ALL_PLS_PME			BIT(17) + +/* PARF_BDF_TO_SID_CFG register fields */ +#define PARF_BDF_TO_SID_BYPASS			BIT(0) + +/* PARF_DEBUG_INT_EN register fields */ +#define PARF_DEBUG_INT_PM_DSTATE_CHANGE		BIT(1) +#define PARF_DEBUG_INT_CFG_BUS_MASTER_EN	BIT(2) +#define PARF_DEBUG_INT_RADM_PM_TURNOFF		BIT(3) + +/* PARF_DEVICE_TYPE register fields */ +#define PARF_DEVICE_TYPE_EP			0x0 + +/* PARF_PM_CTRL register fields */ +#define PARF_PM_CTRL_REQ_EXIT_L1		BIT(1) +#define PARF_PM_CTRL_READY_ENTR_L23		BIT(2) +#define PARF_PM_CTRL_REQ_NOT_ENTR_L1		BIT(5) + +/* PARF_AXI_MSTR_RD_HALT_NO_WRITES register fields */ +#define PARF_AXI_MSTR_RD_HALT_NO_WRITE_EN	BIT(0) + +/* PARF_AXI_MSTR_WR_ADDR_HALT register fields */ +#define PARF_AXI_MSTR_WR_ADDR_HALT_EN		BIT(31) + +/* PARF_Q2A_FLUSH register fields */ +#define PARF_Q2A_FLUSH_EN			BIT(16) + +/* PARF_SYS_CTRL register fields */ +#define PARF_SYS_CTRL_AUX_PWR_DET		BIT(4) +#define PARF_SYS_CTRL_CORE_CLK_CGC_DIS		BIT(6) +#define PARF_SYS_CTRL_SLV_DBI_WAKE_DISABLE	BIT(11) + +/* PARF_DB_CTRL register fields */ +#define PARF_DB_CTRL_INSR_DBNCR_BLOCK		BIT(0) +#define PARF_DB_CTRL_RMVL_DBNCR_BLOCK		BIT(1) +#define PARF_DB_CTRL_DBI_WKP_BLOCK		BIT(4) +#define PARF_DB_CTRL_SLV_WKP_BLOCK		BIT(5) +#define PARF_DB_CTRL_MST_WKP_BLOCK		BIT(6) + +/* PARF_CFG_BITS register fields */ +#define PARF_CFG_BITS_REQ_EXIT_L1SS_MSI_LTR_EN	BIT(1) + +/* ELBI registers */ +#define ELBI_SYS_STTS				0x08 + +/* DBI registers */ +#define DBI_CON_STATUS				0x44 + +/* DBI register fields */ +#define DBI_CON_STATUS_POWER_STATE_MASK		GENMASK(1, 0) + +#define XMLH_LINK_UP				0x400 +#define CORE_RESET_TIME_US_MIN			1000 +#define CORE_RESET_TIME_US_MAX			1005 +#define WAKE_DELAY_US				2000 /* 2 ms */ + +#define to_pcie_ep(x)				dev_get_drvdata((x)->dev) + +enum qcom_pcie_ep_link_status { +	QCOM_PCIE_EP_LINK_DISABLED, +	QCOM_PCIE_EP_LINK_ENABLED, +	QCOM_PCIE_EP_LINK_UP, +	QCOM_PCIE_EP_LINK_DOWN, +}; + +static struct clk_bulk_data qcom_pcie_ep_clks[] = { +	{ .id = "cfg" }, +	{ .id = "aux" }, +	{ .id = "bus_master" }, +	{ .id = "bus_slave" }, +	{ .id = "ref" }, +	{ .id = "sleep" }, +	{ .id = "slave_q2a" }, +}; + +struct qcom_pcie_ep { +	struct dw_pcie pci; + +	void __iomem *parf; +	void __iomem *elbi; +	struct regmap *perst_map; +	struct resource *mmio_res; + +	struct reset_control *core_reset; +	struct gpio_desc *reset; +	struct gpio_desc *wake; +	struct phy *phy; + +	u32 perst_en; +	u32 perst_sep_en; + +	enum qcom_pcie_ep_link_status link_status; +	int global_irq; +	int perst_irq; +}; + +static int qcom_pcie_ep_core_reset(struct qcom_pcie_ep *pcie_ep) +{ +	struct dw_pcie *pci = &pcie_ep->pci; +	struct device *dev = pci->dev; +	int ret; + +	ret = reset_control_assert(pcie_ep->core_reset); +	if (ret) { +		dev_err(dev, "Cannot assert core reset\n"); +		return ret; +	} + +	usleep_range(CORE_RESET_TIME_US_MIN, CORE_RESET_TIME_US_MAX); + +	ret = reset_control_deassert(pcie_ep->core_reset); +	if (ret) { +		dev_err(dev, "Cannot de-assert core reset\n"); +		return ret; +	} + +	usleep_range(CORE_RESET_TIME_US_MIN, CORE_RESET_TIME_US_MAX); + +	return 0; +} + +/* + * Delatch PERST_EN and PERST_SEPARATION_ENABLE with TCSR to avoid + * device reset during host reboot and hibernation. The driver is + * expected to handle this situation. + */ +static void qcom_pcie_ep_configure_tcsr(struct qcom_pcie_ep *pcie_ep) +{ +	regmap_write(pcie_ep->perst_map, pcie_ep->perst_en, 0); +	regmap_write(pcie_ep->perst_map, pcie_ep->perst_sep_en, 0); +} + +static int qcom_pcie_dw_link_up(struct dw_pcie *pci) +{ +	struct qcom_pcie_ep *pcie_ep = to_pcie_ep(pci); +	u32 reg; + +	reg = readl_relaxed(pcie_ep->elbi + ELBI_SYS_STTS); + +	return reg & XMLH_LINK_UP; +} + +static int qcom_pcie_dw_start_link(struct dw_pcie *pci) +{ +	struct qcom_pcie_ep *pcie_ep = to_pcie_ep(pci); + +	enable_irq(pcie_ep->perst_irq); + +	return 0; +} + +static void qcom_pcie_dw_stop_link(struct dw_pcie *pci) +{ +	struct qcom_pcie_ep *pcie_ep = to_pcie_ep(pci); + +	disable_irq(pcie_ep->perst_irq); +} + +static int qcom_pcie_perst_deassert(struct dw_pcie *pci) +{ +	struct qcom_pcie_ep *pcie_ep = to_pcie_ep(pci); +	struct device *dev = pci->dev; +	u32 val, offset; +	int ret; + +	ret = clk_bulk_prepare_enable(ARRAY_SIZE(qcom_pcie_ep_clks), +				      qcom_pcie_ep_clks); +	if (ret) +		return ret; + +	ret = qcom_pcie_ep_core_reset(pcie_ep); +	if (ret) +		goto err_disable_clk; + +	ret = phy_init(pcie_ep->phy); +	if (ret) +		goto err_disable_clk; + +	ret = phy_power_on(pcie_ep->phy); +	if (ret) +		goto err_phy_exit; + +	/* Assert WAKE# to RC to indicate device is ready */ +	gpiod_set_value_cansleep(pcie_ep->wake, 1); +	usleep_range(WAKE_DELAY_US, WAKE_DELAY_US + 500); +	gpiod_set_value_cansleep(pcie_ep->wake, 0); + +	qcom_pcie_ep_configure_tcsr(pcie_ep); + +	/* Disable BDF to SID mapping */ +	val = readl_relaxed(pcie_ep->parf + PARF_BDF_TO_SID_CFG); +	val |= PARF_BDF_TO_SID_BYPASS; +	writel_relaxed(val, pcie_ep->parf + PARF_BDF_TO_SID_CFG); + +	/* Enable debug IRQ */ +	val = readl_relaxed(pcie_ep->parf + PARF_DEBUG_INT_EN); +	val |= PARF_DEBUG_INT_RADM_PM_TURNOFF | +	       PARF_DEBUG_INT_CFG_BUS_MASTER_EN | +	       PARF_DEBUG_INT_PM_DSTATE_CHANGE; +	writel_relaxed(val, pcie_ep->parf + PARF_DEBUG_INT_EN); + +	/* Configure PCIe to endpoint mode */ +	writel_relaxed(PARF_DEVICE_TYPE_EP, pcie_ep->parf + PARF_DEVICE_TYPE); + +	/* Allow entering L1 state */ +	val = readl_relaxed(pcie_ep->parf + PARF_PM_CTRL); +	val &= ~PARF_PM_CTRL_REQ_NOT_ENTR_L1; +	writel_relaxed(val, pcie_ep->parf + PARF_PM_CTRL); + +	/* Read halts write */ +	val = readl_relaxed(pcie_ep->parf + PARF_AXI_MSTR_RD_HALT_NO_WRITES); +	val &= ~PARF_AXI_MSTR_RD_HALT_NO_WRITE_EN; +	writel_relaxed(val, pcie_ep->parf + PARF_AXI_MSTR_RD_HALT_NO_WRITES); + +	/* Write after write halt */ +	val = readl_relaxed(pcie_ep->parf + PARF_AXI_MSTR_WR_ADDR_HALT); +	val |= PARF_AXI_MSTR_WR_ADDR_HALT_EN; +	writel_relaxed(val, pcie_ep->parf + PARF_AXI_MSTR_WR_ADDR_HALT); + +	/* Q2A flush disable */ +	val = readl_relaxed(pcie_ep->parf + PARF_Q2A_FLUSH); +	val &= ~PARF_Q2A_FLUSH_EN; +	writel_relaxed(val, pcie_ep->parf + PARF_Q2A_FLUSH); + +	/* Disable DBI Wakeup, core clock CGC and enable AUX power */ +	val = readl_relaxed(pcie_ep->parf + PARF_SYS_CTRL); +	val |= PARF_SYS_CTRL_SLV_DBI_WAKE_DISABLE | +	       PARF_SYS_CTRL_CORE_CLK_CGC_DIS | +	       PARF_SYS_CTRL_AUX_PWR_DET; +	writel_relaxed(val, pcie_ep->parf + PARF_SYS_CTRL); + +	/* Disable the debouncers */ +	val = readl_relaxed(pcie_ep->parf + PARF_DB_CTRL); +	val |= PARF_DB_CTRL_INSR_DBNCR_BLOCK | PARF_DB_CTRL_RMVL_DBNCR_BLOCK | +	       PARF_DB_CTRL_DBI_WKP_BLOCK | PARF_DB_CTRL_SLV_WKP_BLOCK | +	       PARF_DB_CTRL_MST_WKP_BLOCK; +	writel_relaxed(val, pcie_ep->parf + PARF_DB_CTRL); + +	/* Request to exit from L1SS for MSI and LTR MSG */ +	val = readl_relaxed(pcie_ep->parf + PARF_CFG_BITS); +	val |= PARF_CFG_BITS_REQ_EXIT_L1SS_MSI_LTR_EN; +	writel_relaxed(val, pcie_ep->parf + PARF_CFG_BITS); + +	dw_pcie_dbi_ro_wr_en(pci); + +	/* Set the L0s Exit Latency to 2us-4us = 0x6 */ +	offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); +	val = dw_pcie_readl_dbi(pci, offset + PCI_EXP_LNKCAP); +	val &= ~PCI_EXP_LNKCAP_L0SEL; +	val |= FIELD_PREP(PCI_EXP_LNKCAP_L0SEL, 0x6); +	dw_pcie_writel_dbi(pci, offset + PCI_EXP_LNKCAP, val); + +	/* Set the L1 Exit Latency to be 32us-64 us = 0x6 */ +	offset = dw_pcie_find_capability(pci, PCI_CAP_ID_EXP); +	val = dw_pcie_readl_dbi(pci, offset + PCI_EXP_LNKCAP); +	val &= ~PCI_EXP_LNKCAP_L1EL; +	val |= FIELD_PREP(PCI_EXP_LNKCAP_L1EL, 0x6); +	dw_pcie_writel_dbi(pci, offset + PCI_EXP_LNKCAP, val); + +	dw_pcie_dbi_ro_wr_dis(pci); + +	writel_relaxed(0, pcie_ep->parf + PARF_INT_ALL_MASK); +	val = PARF_INT_ALL_LINK_DOWN | PARF_INT_ALL_BME | +	      PARF_INT_ALL_PM_TURNOFF | PARF_INT_ALL_DSTATE_CHANGE | +	      PARF_INT_ALL_LINK_UP; +	writel_relaxed(val, pcie_ep->parf + PARF_INT_ALL_MASK); + +	ret = dw_pcie_ep_init_complete(&pcie_ep->pci.ep); +	if (ret) { +		dev_err(dev, "Failed to complete initialization: %d\n", ret); +		goto err_phy_power_off; +	} + +	/* +	 * The physical address of the MMIO region which is exposed as the BAR +	 * should be written to MHI BASE registers. +	 */ +	writel_relaxed(pcie_ep->mmio_res->start, +		       pcie_ep->parf + PARF_MHI_BASE_ADDR_LOWER); +	writel_relaxed(0, pcie_ep->parf + PARF_MHI_BASE_ADDR_UPPER); + +	dw_pcie_ep_init_notify(&pcie_ep->pci.ep); + +	/* Enable LTSSM */ +	val = readl_relaxed(pcie_ep->parf + PARF_LTSSM); +	val |= BIT(8); +	writel_relaxed(val, pcie_ep->parf + PARF_LTSSM); + +	return 0; + +err_phy_power_off: +	phy_power_off(pcie_ep->phy); +err_phy_exit: +	phy_exit(pcie_ep->phy); +err_disable_clk: +	clk_bulk_disable_unprepare(ARRAY_SIZE(qcom_pcie_ep_clks), +				   qcom_pcie_ep_clks); + +	return ret; +} + +static void qcom_pcie_perst_assert(struct dw_pcie *pci) +{ +	struct qcom_pcie_ep *pcie_ep = to_pcie_ep(pci); +	struct device *dev = pci->dev; + +	if (pcie_ep->link_status == QCOM_PCIE_EP_LINK_DISABLED) { +		dev_dbg(dev, "Link is already disabled\n"); +		return; +	} + +	phy_power_off(pcie_ep->phy); +	phy_exit(pcie_ep->phy); +	clk_bulk_disable_unprepare(ARRAY_SIZE(qcom_pcie_ep_clks), +				   qcom_pcie_ep_clks); +	pcie_ep->link_status = QCOM_PCIE_EP_LINK_DISABLED; +} + +/* Common DWC controller ops */ +static const struct dw_pcie_ops pci_ops = { +	.link_up = qcom_pcie_dw_link_up, +	.start_link = qcom_pcie_dw_start_link, +	.stop_link = qcom_pcie_dw_stop_link, +}; + +static int qcom_pcie_ep_get_io_resources(struct platform_device *pdev, +					 struct qcom_pcie_ep *pcie_ep) +{ +	struct device *dev = &pdev->dev; +	struct dw_pcie *pci = &pcie_ep->pci; +	struct device_node *syscon; +	struct resource *res; +	int ret; + +	pcie_ep->parf = devm_platform_ioremap_resource_byname(pdev, "parf"); +	if (IS_ERR(pcie_ep->parf)) +		return PTR_ERR(pcie_ep->parf); + +	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); +	pci->dbi_base2 = pci->dbi_base; + +	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "elbi"); +	pcie_ep->elbi = devm_pci_remap_cfg_resource(dev, res); +	if (IS_ERR(pcie_ep->elbi)) +		return PTR_ERR(pcie_ep->elbi); + +	pcie_ep->mmio_res = platform_get_resource_byname(pdev, IORESOURCE_MEM, +							 "mmio"); + +	syscon = of_parse_phandle(dev->of_node, "qcom,perst-regs", 0); +	if (!syscon) { +		dev_err(dev, "Failed to parse qcom,perst-regs\n"); +		return -EINVAL; +	} + +	pcie_ep->perst_map = syscon_node_to_regmap(syscon); +	of_node_put(syscon); +	if (IS_ERR(pcie_ep->perst_map)) +		return PTR_ERR(pcie_ep->perst_map); + +	ret = of_property_read_u32_index(dev->of_node, "qcom,perst-regs", +					 1, &pcie_ep->perst_en); +	if (ret < 0) { +		dev_err(dev, "No Perst Enable offset in syscon\n"); +		return ret; +	} + +	ret = of_property_read_u32_index(dev->of_node, "qcom,perst-regs", +					 2, &pcie_ep->perst_sep_en); +	if (ret < 0) { +		dev_err(dev, "No Perst Separation Enable offset in syscon\n"); +		return ret; +	} + +	return 0; +} + +static int qcom_pcie_ep_get_resources(struct platform_device *pdev, +				      struct qcom_pcie_ep *pcie_ep) +{ +	struct device *dev = &pdev->dev; +	int ret; + +	ret = qcom_pcie_ep_get_io_resources(pdev, pcie_ep); +	if (ret) { +		dev_err(&pdev->dev, "Failed to get io resources %d\n", ret); +		return ret; +	} + +	ret = devm_clk_bulk_get(dev, ARRAY_SIZE(qcom_pcie_ep_clks), +				qcom_pcie_ep_clks); +	if (ret) +		return ret; + +	pcie_ep->core_reset = devm_reset_control_get_exclusive(dev, "core"); +	if (IS_ERR(pcie_ep->core_reset)) +		return PTR_ERR(pcie_ep->core_reset); + +	pcie_ep->reset = devm_gpiod_get(dev, "reset", GPIOD_IN); +	if (IS_ERR(pcie_ep->reset)) +		return PTR_ERR(pcie_ep->reset); + +	pcie_ep->wake = devm_gpiod_get_optional(dev, "wake", GPIOD_OUT_LOW); +	if (IS_ERR(pcie_ep->wake)) +		return PTR_ERR(pcie_ep->wake); + +	pcie_ep->phy = devm_phy_optional_get(&pdev->dev, "pciephy"); +	if (IS_ERR(pcie_ep->phy)) +		ret = PTR_ERR(pcie_ep->phy); + +	return ret; +} + +/* TODO: Notify clients about PCIe state change */ +static irqreturn_t qcom_pcie_ep_global_irq_thread(int irq, void *data) +{ +	struct qcom_pcie_ep *pcie_ep = data; +	struct dw_pcie *pci = &pcie_ep->pci; +	struct device *dev = pci->dev; +	u32 status = readl_relaxed(pcie_ep->parf + PARF_INT_ALL_STATUS); +	u32 mask = readl_relaxed(pcie_ep->parf + PARF_INT_ALL_MASK); +	u32 dstate, val; + +	writel_relaxed(status, pcie_ep->parf + PARF_INT_ALL_CLEAR); +	status &= mask; + +	if (FIELD_GET(PARF_INT_ALL_LINK_DOWN, status)) { +		dev_dbg(dev, "Received Linkdown event\n"); +		pcie_ep->link_status = QCOM_PCIE_EP_LINK_DOWN; +	} else if (FIELD_GET(PARF_INT_ALL_BME, status)) { +		dev_dbg(dev, "Received BME event. Link is enabled!\n"); +		pcie_ep->link_status = QCOM_PCIE_EP_LINK_ENABLED; +	} else if (FIELD_GET(PARF_INT_ALL_PM_TURNOFF, status)) { +		dev_dbg(dev, "Received PM Turn-off event! Entering L23\n"); +		val = readl_relaxed(pcie_ep->parf + PARF_PM_CTRL); +		val |= PARF_PM_CTRL_READY_ENTR_L23; +		writel_relaxed(val, pcie_ep->parf + PARF_PM_CTRL); +	} else if (FIELD_GET(PARF_INT_ALL_DSTATE_CHANGE, status)) { +		dstate = dw_pcie_readl_dbi(pci, DBI_CON_STATUS) & +					   DBI_CON_STATUS_POWER_STATE_MASK; +		dev_dbg(dev, "Received D%d state event\n", dstate); +		if (dstate == 3) { +			val = readl_relaxed(pcie_ep->parf + PARF_PM_CTRL); +			val |= PARF_PM_CTRL_REQ_EXIT_L1; +			writel_relaxed(val, pcie_ep->parf + PARF_PM_CTRL); +		} +	} else if (FIELD_GET(PARF_INT_ALL_LINK_UP, status)) { +		dev_dbg(dev, "Received Linkup event. Enumeration complete!\n"); +		dw_pcie_ep_linkup(&pci->ep); +		pcie_ep->link_status = QCOM_PCIE_EP_LINK_UP; +	} else { +		dev_dbg(dev, "Received unknown event: %d\n", status); +	} + +	return IRQ_HANDLED; +} + +static irqreturn_t qcom_pcie_ep_perst_irq_thread(int irq, void *data) +{ +	struct qcom_pcie_ep *pcie_ep = data; +	struct dw_pcie *pci = &pcie_ep->pci; +	struct device *dev = pci->dev; +	u32 perst; + +	perst = gpiod_get_value(pcie_ep->reset); +	if (perst) { +		dev_dbg(dev, "PERST asserted by host. Shutting down the PCIe link!\n"); +		qcom_pcie_perst_assert(pci); +	} else { +		dev_dbg(dev, "PERST de-asserted by host. Starting link training!\n"); +		qcom_pcie_perst_deassert(pci); +	} + +	irq_set_irq_type(gpiod_to_irq(pcie_ep->reset), +			 (perst ? IRQF_TRIGGER_HIGH : IRQF_TRIGGER_LOW)); + +	return IRQ_HANDLED; +} + +static int qcom_pcie_ep_enable_irq_resources(struct platform_device *pdev, +					     struct qcom_pcie_ep *pcie_ep) +{ +	int irq, ret; + +	irq = platform_get_irq_byname(pdev, "global"); +	if (irq < 0) { +		dev_err(&pdev->dev, "Failed to get Global IRQ\n"); +		return irq; +	} + +	ret = devm_request_threaded_irq(&pdev->dev, irq, NULL, +					qcom_pcie_ep_global_irq_thread, +					IRQF_ONESHOT, +					"global_irq", pcie_ep); +	if (ret) { +		dev_err(&pdev->dev, "Failed to request Global IRQ\n"); +		return ret; +	} + +	pcie_ep->perst_irq = gpiod_to_irq(pcie_ep->reset); +	irq_set_status_flags(pcie_ep->perst_irq, IRQ_NOAUTOEN); +	ret = devm_request_threaded_irq(&pdev->dev, pcie_ep->perst_irq, NULL, +					qcom_pcie_ep_perst_irq_thread, +					IRQF_TRIGGER_HIGH | IRQF_ONESHOT, +					"perst_irq", pcie_ep); +	if (ret) { +		dev_err(&pdev->dev, "Failed to request PERST IRQ\n"); +		disable_irq(irq); +		return ret; +	} + +	return 0; +} + +static int qcom_pcie_ep_raise_irq(struct dw_pcie_ep *ep, u8 func_no, +				  enum pci_epc_irq_type type, u16 interrupt_num) +{ +	struct dw_pcie *pci = to_dw_pcie_from_ep(ep); + +	switch (type) { +	case PCI_EPC_IRQ_LEGACY: +		return dw_pcie_ep_raise_legacy_irq(ep, func_no); +	case PCI_EPC_IRQ_MSI: +		return dw_pcie_ep_raise_msi_irq(ep, func_no, interrupt_num); +	default: +		dev_err(pci->dev, "Unknown IRQ type\n"); +		return -EINVAL; +	} +} + +static const struct pci_epc_features qcom_pcie_epc_features = { +	.linkup_notifier = true, +	.core_init_notifier = true, +	.msi_capable = true, +	.msix_capable = false, +}; + +static const struct pci_epc_features * +qcom_pcie_epc_get_features(struct dw_pcie_ep *pci_ep) +{ +	return &qcom_pcie_epc_features; +} + +static void qcom_pcie_ep_init(struct dw_pcie_ep *ep) +{ +	struct dw_pcie *pci = to_dw_pcie_from_ep(ep); +	enum pci_barno bar; + +	for (bar = BAR_0; bar <= BAR_5; bar++) +		dw_pcie_ep_reset_bar(pci, bar); +} + +static struct dw_pcie_ep_ops pci_ep_ops = { +	.ep_init = qcom_pcie_ep_init, +	.raise_irq = qcom_pcie_ep_raise_irq, +	.get_features = qcom_pcie_epc_get_features, +}; + +static int qcom_pcie_ep_probe(struct platform_device *pdev) +{ +	struct device *dev = &pdev->dev; +	struct qcom_pcie_ep *pcie_ep; +	int ret; + +	pcie_ep = devm_kzalloc(dev, sizeof(*pcie_ep), GFP_KERNEL); +	if (!pcie_ep) +		return -ENOMEM; + +	pcie_ep->pci.dev = dev; +	pcie_ep->pci.ops = &pci_ops; +	pcie_ep->pci.ep.ops = &pci_ep_ops; +	platform_set_drvdata(pdev, pcie_ep); + +	ret = qcom_pcie_ep_get_resources(pdev, pcie_ep); +	if (ret) +		return ret; + +	ret = clk_bulk_prepare_enable(ARRAY_SIZE(qcom_pcie_ep_clks), +				      qcom_pcie_ep_clks); +	if (ret) +		return ret; + +	ret = qcom_pcie_ep_core_reset(pcie_ep); +	if (ret) +		goto err_disable_clk; + +	ret = phy_init(pcie_ep->phy); +	if (ret) +		goto err_disable_clk; + +	/* PHY needs to be powered on for dw_pcie_ep_init() */ +	ret = phy_power_on(pcie_ep->phy); +	if (ret) +		goto err_phy_exit; + +	ret = dw_pcie_ep_init(&pcie_ep->pci.ep); +	if (ret) { +		dev_err(dev, "Failed to initialize endpoint: %d\n", ret); +		goto err_phy_power_off; +	} + +	ret = qcom_pcie_ep_enable_irq_resources(pdev, pcie_ep); +	if (ret) +		goto err_phy_power_off; + +	return 0; + +err_phy_power_off: +	phy_power_off(pcie_ep->phy); +err_phy_exit: +	phy_exit(pcie_ep->phy); +err_disable_clk: +	clk_bulk_disable_unprepare(ARRAY_SIZE(qcom_pcie_ep_clks), +				   qcom_pcie_ep_clks); + +	return ret; +} + +static int qcom_pcie_ep_remove(struct platform_device *pdev) +{ +	struct qcom_pcie_ep *pcie_ep = platform_get_drvdata(pdev); + +	if (pcie_ep->link_status == QCOM_PCIE_EP_LINK_DISABLED) +		return 0; + +	phy_power_off(pcie_ep->phy); +	phy_exit(pcie_ep->phy); +	clk_bulk_disable_unprepare(ARRAY_SIZE(qcom_pcie_ep_clks), +				   qcom_pcie_ep_clks); + +	return 0; +} + +static const struct of_device_id qcom_pcie_ep_match[] = { +	{ .compatible = "qcom,sdx55-pcie-ep", }, +	{ } +}; + +static struct platform_driver qcom_pcie_ep_driver = { +	.probe	= qcom_pcie_ep_probe, +	.remove = qcom_pcie_ep_remove, +	.driver	= { +		.name = "qcom-pcie-ep", +		.of_match_table	= qcom_pcie_ep_match, +	}, +}; +builtin_platform_driver(qcom_pcie_ep_driver); + +MODULE_AUTHOR("Siddartha Mohanadoss <[email protected]>"); +MODULE_AUTHOR("Manivannan Sadhasivam <[email protected]>"); +MODULE_DESCRIPTION("Qualcomm PCIe Endpoint controller driver"); +MODULE_LICENSE("GPL v2"); diff --git a/drivers/pci/controller/dwc/pcie-qcom.c b/drivers/pci/controller/dwc/pcie-qcom.c index 8a7a300163e5..1c3d1116bb60 100644 --- a/drivers/pci/controller/dwc/pcie-qcom.c +++ b/drivers/pci/controller/dwc/pcie-qcom.c @@ -166,6 +166,9 @@ struct qcom_pcie_resources_2_7_0 {  	struct regulator_bulk_data supplies[2];  	struct reset_control *pci_reset;  	struct clk *pipe_clk; +	struct clk *pipe_clk_src; +	struct clk *phy_pipe_clk; +	struct clk *ref_clk_src;  };  union qcom_pcie_resources { @@ -189,6 +192,11 @@ struct qcom_pcie_ops {  	int (*config_sid)(struct qcom_pcie *pcie);  }; +struct qcom_pcie_cfg { +	const struct qcom_pcie_ops *ops; +	unsigned int pipe_clk_need_muxing:1; +}; +  struct qcom_pcie {  	struct dw_pcie *pci;  	void __iomem *parf;			/* DT parf */ @@ -197,6 +205,7 @@ struct qcom_pcie {  	struct phy *phy;  	struct gpio_desc *reset;  	const struct qcom_pcie_ops *ops; +	unsigned int pipe_clk_need_muxing:1;  };  #define to_qcom_pcie(x)		dev_get_drvdata((x)->dev) @@ -1167,6 +1176,20 @@ static int qcom_pcie_get_resources_2_7_0(struct qcom_pcie *pcie)  	if (ret < 0)  		return ret; +	if (pcie->pipe_clk_need_muxing) { +		res->pipe_clk_src = devm_clk_get(dev, "pipe_mux"); +		if (IS_ERR(res->pipe_clk_src)) +			return PTR_ERR(res->pipe_clk_src); + +		res->phy_pipe_clk = devm_clk_get(dev, "phy_pipe"); +		if (IS_ERR(res->phy_pipe_clk)) +			return PTR_ERR(res->phy_pipe_clk); + +		res->ref_clk_src = devm_clk_get(dev, "ref"); +		if (IS_ERR(res->ref_clk_src)) +			return PTR_ERR(res->ref_clk_src); +	} +  	res->pipe_clk = devm_clk_get(dev, "pipe");  	return PTR_ERR_OR_ZERO(res->pipe_clk);  } @@ -1185,6 +1208,10 @@ static int qcom_pcie_init_2_7_0(struct qcom_pcie *pcie)  		return ret;  	} +	/* Set TCXO as clock source for pcie_pipe_clk_src */ +	if (pcie->pipe_clk_need_muxing) +		clk_set_parent(res->pipe_clk_src, res->ref_clk_src); +  	ret = clk_bulk_prepare_enable(res->num_clks, res->clks);  	if (ret < 0)  		goto err_disable_regulators; @@ -1256,6 +1283,10 @@ static int qcom_pcie_post_init_2_7_0(struct qcom_pcie *pcie)  {  	struct qcom_pcie_resources_2_7_0 *res = &pcie->res.v2_7_0; +	/* Set pipe clock as clock source for pcie_pipe_clk_src */ +	if (pcie->pipe_clk_need_muxing) +		clk_set_parent(res->pipe_clk_src, res->phy_pipe_clk); +  	return clk_prepare_enable(res->pipe_clk);  } @@ -1456,6 +1487,39 @@ static const struct qcom_pcie_ops ops_1_9_0 = {  	.config_sid = qcom_pcie_config_sid_sm8250,  }; +static const struct qcom_pcie_cfg apq8084_cfg = { +	.ops = &ops_1_0_0, +}; + +static const struct qcom_pcie_cfg ipq8064_cfg = { +	.ops = &ops_2_1_0, +}; + +static const struct qcom_pcie_cfg msm8996_cfg = { +	.ops = &ops_2_3_2, +}; + +static const struct qcom_pcie_cfg ipq8074_cfg = { +	.ops = &ops_2_3_3, +}; + +static const struct qcom_pcie_cfg ipq4019_cfg = { +	.ops = &ops_2_4_0, +}; + +static const struct qcom_pcie_cfg sdm845_cfg = { +	.ops = &ops_2_7_0, +}; + +static const struct qcom_pcie_cfg sm8250_cfg = { +	.ops = &ops_1_9_0, +}; + +static const struct qcom_pcie_cfg sc7280_cfg = { +	.ops = &ops_1_9_0, +	.pipe_clk_need_muxing = true, +}; +  static const struct dw_pcie_ops dw_pcie_ops = {  	.link_up = qcom_pcie_link_up,  	.start_link = qcom_pcie_start_link, @@ -1467,6 +1531,7 @@ static int qcom_pcie_probe(struct platform_device *pdev)  	struct pcie_port *pp;  	struct dw_pcie *pci;  	struct qcom_pcie *pcie; +	const struct qcom_pcie_cfg *pcie_cfg;  	int ret;  	pcie = devm_kzalloc(dev, sizeof(*pcie), GFP_KERNEL); @@ -1488,7 +1553,14 @@ static int qcom_pcie_probe(struct platform_device *pdev)  	pcie->pci = pci; -	pcie->ops = of_device_get_match_data(dev); +	pcie_cfg = of_device_get_match_data(dev); +	if (!pcie_cfg || !pcie_cfg->ops) { +		dev_err(dev, "Invalid platform data\n"); +		return -EINVAL; +	} + +	pcie->ops = pcie_cfg->ops; +	pcie->pipe_clk_need_muxing = pcie_cfg->pipe_clk_need_muxing;  	pcie->reset = devm_gpiod_get_optional(dev, "perst", GPIOD_OUT_HIGH);  	if (IS_ERR(pcie->reset)) { @@ -1545,16 +1617,18 @@ err_pm_runtime_put:  }  static const struct of_device_id qcom_pcie_match[] = { -	{ .compatible = "qcom,pcie-apq8084", .data = &ops_1_0_0 }, -	{ .compatible = "qcom,pcie-ipq8064", .data = &ops_2_1_0 }, -	{ .compatible = "qcom,pcie-ipq8064-v2", .data = &ops_2_1_0 }, -	{ .compatible = "qcom,pcie-apq8064", .data = &ops_2_1_0 }, -	{ .compatible = "qcom,pcie-msm8996", .data = &ops_2_3_2 }, -	{ .compatible = "qcom,pcie-ipq8074", .data = &ops_2_3_3 }, -	{ .compatible = "qcom,pcie-ipq4019", .data = &ops_2_4_0 }, -	{ .compatible = "qcom,pcie-qcs404", .data = &ops_2_4_0 }, -	{ .compatible = "qcom,pcie-sdm845", .data = &ops_2_7_0 }, -	{ .compatible = "qcom,pcie-sm8250", .data = &ops_1_9_0 }, +	{ .compatible = "qcom,pcie-apq8084", .data = &apq8084_cfg }, +	{ .compatible = "qcom,pcie-ipq8064", .data = &ipq8064_cfg }, +	{ .compatible = "qcom,pcie-ipq8064-v2", .data = &ipq8064_cfg }, +	{ .compatible = "qcom,pcie-apq8064", .data = &ipq8064_cfg }, +	{ .compatible = "qcom,pcie-msm8996", .data = &msm8996_cfg }, +	{ .compatible = "qcom,pcie-ipq8074", .data = &ipq8074_cfg }, +	{ .compatible = "qcom,pcie-ipq4019", .data = &ipq4019_cfg }, +	{ .compatible = "qcom,pcie-qcs404", .data = &ipq4019_cfg }, +	{ .compatible = "qcom,pcie-sdm845", .data = &sdm845_cfg }, +	{ .compatible = "qcom,pcie-sm8250", .data = &sm8250_cfg }, +	{ .compatible = "qcom,pcie-sc8180x", .data = &sm8250_cfg }, +	{ .compatible = "qcom,pcie-sc7280", .data = &sc7280_cfg },  	{ }  }; diff --git a/drivers/pci/controller/dwc/pcie-uniphier.c b/drivers/pci/controller/dwc/pcie-uniphier.c index d842fd018129..d05be942956e 100644 --- a/drivers/pci/controller/dwc/pcie-uniphier.c +++ b/drivers/pci/controller/dwc/pcie-uniphier.c @@ -168,30 +168,21 @@ static void uniphier_pcie_irq_enable(struct uniphier_pcie_priv *priv)  	writel(PCL_RCV_INTX_ALL_ENABLE, priv->base + PCL_RCV_INTX);  } -static void uniphier_pcie_irq_ack(struct irq_data *d) -{ -	struct pcie_port *pp = irq_data_get_irq_chip_data(d); -	struct dw_pcie *pci = to_dw_pcie_from_pp(pp); -	struct uniphier_pcie_priv *priv = to_uniphier_pcie(pci); -	u32 val; - -	val = readl(priv->base + PCL_RCV_INTX); -	val &= ~PCL_RCV_INTX_ALL_STATUS; -	val |= BIT(irqd_to_hwirq(d) + PCL_RCV_INTX_STATUS_SHIFT); -	writel(val, priv->base + PCL_RCV_INTX); -} -  static void uniphier_pcie_irq_mask(struct irq_data *d)  {  	struct pcie_port *pp = irq_data_get_irq_chip_data(d);  	struct dw_pcie *pci = to_dw_pcie_from_pp(pp);  	struct uniphier_pcie_priv *priv = to_uniphier_pcie(pci); +	unsigned long flags;  	u32 val; +	raw_spin_lock_irqsave(&pp->lock, flags); +  	val = readl(priv->base + PCL_RCV_INTX); -	val &= ~PCL_RCV_INTX_ALL_MASK;  	val |= BIT(irqd_to_hwirq(d) + PCL_RCV_INTX_MASK_SHIFT);  	writel(val, priv->base + PCL_RCV_INTX); + +	raw_spin_unlock_irqrestore(&pp->lock, flags);  }  static void uniphier_pcie_irq_unmask(struct irq_data *d) @@ -199,17 +190,20 @@ static void uniphier_pcie_irq_unmask(struct irq_data *d)  	struct pcie_port *pp = irq_data_get_irq_chip_data(d);  	struct dw_pcie *pci = to_dw_pcie_from_pp(pp);  	struct uniphier_pcie_priv *priv = to_uniphier_pcie(pci); +	unsigned long flags;  	u32 val; +	raw_spin_lock_irqsave(&pp->lock, flags); +  	val = readl(priv->base + PCL_RCV_INTX); -	val &= ~PCL_RCV_INTX_ALL_MASK;  	val &= ~BIT(irqd_to_hwirq(d) + PCL_RCV_INTX_MASK_SHIFT);  	writel(val, priv->base + PCL_RCV_INTX); + +	raw_spin_unlock_irqrestore(&pp->lock, flags);  }  static struct irq_chip uniphier_pcie_irq_chip = {  	.name = "PCI", -	.irq_ack = uniphier_pcie_irq_ack,  	.irq_mask = uniphier_pcie_irq_mask,  	.irq_unmask = uniphier_pcie_irq_unmask,  }; diff --git a/drivers/pci/controller/dwc/pcie-visconti.c b/drivers/pci/controller/dwc/pcie-visconti.c index a88eab6829bb..50f80f07e4db 100644 --- a/drivers/pci/controller/dwc/pcie-visconti.c +++ b/drivers/pci/controller/dwc/pcie-visconti.c @@ -279,13 +279,10 @@ static int visconti_add_pcie_port(struct visconti_pcie *pcie,  {  	struct dw_pcie *pci = &pcie->pci;  	struct pcie_port *pp = &pci->pp; -	struct device *dev = &pdev->dev;  	pp->irq = platform_get_irq_byname(pdev, "intr"); -	if (pp->irq < 0) { -		dev_err(dev, "Interrupt intr is missing"); +	if (pp->irq < 0)  		return pp->irq; -	}  	pp->ops = &visconti_pcie_host_ops; |