PCI: cadence: Add host driver for Cadence PCIe controller
This patch adds support to the Cadence PCIe controller in host mode. Signed-off-by: Cyrille Pitchen <cyrille.pitchen@free-electrons.com> Signed-off-by: Lorenzo Pieralisi <lorenzo.pieralisi@arm.com>
This commit is contained in:
		
							parent
							
								
									2040fae4b2
								
							
						
					
					
						commit
						1b79c52844
					
				| @ -10402,6 +10402,13 @@ S:	Maintained | |||||||
| F:	Documentation/devicetree/bindings/pci/pci-armada8k.txt | F:	Documentation/devicetree/bindings/pci/pci-armada8k.txt | ||||||
| F:	drivers/pci/dwc/pcie-armada8k.c | F:	drivers/pci/dwc/pcie-armada8k.c | ||||||
| 
 | 
 | ||||||
|  | PCI DRIVER FOR CADENCE PCIE IP | ||||||
|  | M:	Alan Douglas <adouglas@cadence.com> | ||||||
|  | L:	linux-pci@vger.kernel.org | ||||||
|  | S:	Maintained | ||||||
|  | F:	Documentation/devicetree/bindings/pci/cdns,*.txt | ||||||
|  | F:	drivers/pci/host/pcie-cadence* | ||||||
|  | 
 | ||||||
| PCI DRIVER FOR FREESCALE LAYERSCAPE | PCI DRIVER FOR FREESCALE LAYERSCAPE | ||||||
| M:	Minghuan Lian <minghuan.Lian@freescale.com> | M:	Minghuan Lian <minghuan.Lian@freescale.com> | ||||||
| M:	Mingkai Hu <mingkai.hu@freescale.com> | M:	Mingkai Hu <mingkai.hu@freescale.com> | ||||||
|  | |||||||
| @ -210,6 +210,16 @@ config PCIE_TANGO_SMP8759 | |||||||
| 	  This can lead to data corruption if drivers perform concurrent | 	  This can lead to data corruption if drivers perform concurrent | ||||||
| 	  config and MMIO accesses. | 	  config and MMIO accesses. | ||||||
| 
 | 
 | ||||||
|  | config PCIE_CADENCE_HOST | ||||||
|  | 	bool "Cadence PCIe host controller" | ||||||
|  | 	depends on OF | ||||||
|  | 	depends on PCI | ||||||
|  | 	select IRQ_DOMAIN | ||||||
|  | 	help | ||||||
|  | 	  Say Y here if you want to support the Cadence PCIe controller in host | ||||||
|  | 	  mode. This PCIe controller may be embedded into many different vendors | ||||||
|  | 	  SoCs. | ||||||
|  | 
 | ||||||
| config VMD | config VMD | ||||||
| 	depends on PCI_MSI && X86_64 && SRCU | 	depends on PCI_MSI && X86_64 && SRCU | ||||||
| 	tristate "Intel Volume Management Device Driver" | 	tristate "Intel Volume Management Device Driver" | ||||||
|  | |||||||
| @ -22,6 +22,7 @@ obj-$(CONFIG_PCIE_ALTERA_MSI) += pcie-altera-msi.o | |||||||
| obj-$(CONFIG_PCIE_ROCKCHIP) += pcie-rockchip.o | obj-$(CONFIG_PCIE_ROCKCHIP) += pcie-rockchip.o | ||||||
| obj-$(CONFIG_PCIE_MEDIATEK) += pcie-mediatek.o | obj-$(CONFIG_PCIE_MEDIATEK) += pcie-mediatek.o | ||||||
| obj-$(CONFIG_PCIE_TANGO_SMP8759) += pcie-tango.o | obj-$(CONFIG_PCIE_TANGO_SMP8759) += pcie-tango.o | ||||||
|  | obj-$(CONFIG_PCIE_CADENCE_HOST) += pcie-cadence-host.o | ||||||
| obj-$(CONFIG_VMD) += vmd.o | obj-$(CONFIG_VMD) += vmd.o | ||||||
| 
 | 
 | ||||||
| # The following drivers are for devices that use the generic ACPI
 | # The following drivers are for devices that use the generic ACPI
 | ||||||
|  | |||||||
							
								
								
									
										397
									
								
								drivers/pci/host/pcie-cadence-host.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										397
									
								
								drivers/pci/host/pcie-cadence-host.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,397 @@ | |||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | // Copyright (c) 2017 Cadence
 | ||||||
|  | // Cadence PCIe host controller driver.
 | ||||||
|  | // Author: Cyrille Pitchen <cyrille.pitchen@free-electrons.com>
 | ||||||
|  | 
 | ||||||
|  | #include <linux/kernel.h> | ||||||
|  | #include <linux/of_address.h> | ||||||
|  | #include <linux/of_pci.h> | ||||||
|  | #include <linux/platform_device.h> | ||||||
|  | #include <linux/pm_runtime.h> | ||||||
|  | 
 | ||||||
|  | #include "pcie-cadence.h" | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * struct cdns_pcie_rc - private data for this PCIe Root Complex driver | ||||||
|  |  * @pcie: Cadence PCIe controller | ||||||
|  |  * @dev: pointer to PCIe device | ||||||
|  |  * @cfg_res: start/end offsets in the physical system memory to map PCI | ||||||
|  |  *           configuration space accesses | ||||||
|  |  * @bus_range: first/last buses behind the PCIe host controller | ||||||
|  |  * @cfg_base: IO mapped window to access the PCI configuration space of a | ||||||
|  |  *            single function at a time | ||||||
|  |  * @max_regions: maximum number of regions supported by the hardware | ||||||
|  |  * @no_bar_nbits: Number of bits to keep for inbound (PCIe -> CPU) address | ||||||
|  |  *                translation (nbits sets into the "no BAR match" register) | ||||||
|  |  * @vendor_id: PCI vendor ID | ||||||
|  |  * @device_id: PCI device ID | ||||||
|  |  */ | ||||||
|  | struct cdns_pcie_rc { | ||||||
|  | 	struct cdns_pcie	pcie; | ||||||
|  | 	struct device		*dev; | ||||||
|  | 	struct resource		*cfg_res; | ||||||
|  | 	struct resource		*bus_range; | ||||||
|  | 	void __iomem		*cfg_base; | ||||||
|  | 	u32			max_regions; | ||||||
|  | 	u32			no_bar_nbits; | ||||||
|  | 	u16			vendor_id; | ||||||
|  | 	u16			device_id; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void cdns_pcie_set_outbound_region(struct cdns_pcie *pcie, | ||||||
|  | 				   u32 r, bool is_io, | ||||||
|  | 				   u64 cpu_addr, u64 pci_addr, size_t size) | ||||||
|  | { | ||||||
|  | 	/*
 | ||||||
|  | 	 * roundup_pow_of_two() returns an unsigned long, which is not suited | ||||||
|  | 	 * for 64bit values. | ||||||
|  | 	 */ | ||||||
|  | 	u64 sz = 1ULL << fls64(size - 1); | ||||||
|  | 	int nbits = ilog2(sz); | ||||||
|  | 	u32 addr0, addr1, desc0, desc1; | ||||||
|  | 
 | ||||||
|  | 	if (nbits < 8) | ||||||
|  | 		nbits = 8; | ||||||
|  | 
 | ||||||
|  | 	/* Set the PCI address */ | ||||||
|  | 	addr0 = CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_NBITS(nbits) | | ||||||
|  | 		(lower_32_bits(pci_addr) & GENMASK(31, 8)); | ||||||
|  | 	addr1 = upper_32_bits(pci_addr); | ||||||
|  | 
 | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_OB_REGION_PCI_ADDR0(r), addr0); | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_OB_REGION_PCI_ADDR1(r), addr1); | ||||||
|  | 
 | ||||||
|  | 	/* Set the PCIe header descriptor */ | ||||||
|  | 	if (is_io) | ||||||
|  | 		desc0 = CDNS_PCIE_AT_OB_REGION_DESC0_TYPE_IO; | ||||||
|  | 	else | ||||||
|  | 		desc0 = CDNS_PCIE_AT_OB_REGION_DESC0_TYPE_MEM; | ||||||
|  | 	desc1 = 0; | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * Whatever Bit [23] is set or not inside DESC0 register of the outbound | ||||||
|  | 	 * PCIe descriptor, the PCI function number must be set into | ||||||
|  | 	 * Bits [26:24] of DESC0 anyway. | ||||||
|  | 	 * | ||||||
|  | 	 * In Root Complex mode, the function number is always 0 but in Endpoint | ||||||
|  | 	 * mode, the PCIe controller may support more than one function. This | ||||||
|  | 	 * function number needs to be set properly into the outbound PCIe | ||||||
|  | 	 * descriptor. | ||||||
|  | 	 * | ||||||
|  | 	 * Besides, setting Bit [23] is mandatory when in Root Complex mode: | ||||||
|  | 	 * then the driver must provide the bus, resp. device, number in | ||||||
|  | 	 * Bits [7:0] of DESC1, resp. Bits[31:27] of DESC0. Like the function | ||||||
|  | 	 * number, the device number is always 0 in Root Complex mode. | ||||||
|  | 	 */ | ||||||
|  | 	desc0 |= CDNS_PCIE_AT_OB_REGION_DESC0_HARDCODED_RID | | ||||||
|  | 		CDNS_PCIE_AT_OB_REGION_DESC0_DEVFN(0); | ||||||
|  | 	desc1 |= CDNS_PCIE_AT_OB_REGION_DESC1_BUS(pcie->bus); | ||||||
|  | 
 | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_OB_REGION_DESC0(r), desc0); | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_OB_REGION_DESC1(r), desc1); | ||||||
|  | 
 | ||||||
|  | 	/* Set the CPU address */ | ||||||
|  | 	cpu_addr -= pcie->mem_res->start; | ||||||
|  | 	addr0 = CDNS_PCIE_AT_OB_REGION_CPU_ADDR0_NBITS(nbits) | | ||||||
|  | 		(lower_32_bits(cpu_addr) & GENMASK(31, 8)); | ||||||
|  | 	addr1 = upper_32_bits(cpu_addr); | ||||||
|  | 
 | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_OB_REGION_CPU_ADDR0(r), addr0); | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_OB_REGION_CPU_ADDR1(r), addr1); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void __iomem *cdns_pci_map_bus(struct pci_bus *bus, unsigned int devfn, | ||||||
|  | 				      int where) | ||||||
|  | { | ||||||
|  | 	struct pci_host_bridge *bridge = pci_find_host_bridge(bus); | ||||||
|  | 	struct cdns_pcie_rc *rc = pci_host_bridge_priv(bridge); | ||||||
|  | 	struct cdns_pcie *pcie = &rc->pcie; | ||||||
|  | 	unsigned int busn = bus->number; | ||||||
|  | 	u32 addr0, desc0; | ||||||
|  | 
 | ||||||
|  | 	if (busn == rc->bus_range->start) { | ||||||
|  | 		/*
 | ||||||
|  | 		 * Only the root port (devfn == 0) is connected to this bus. | ||||||
|  | 		 * All other PCI devices are behind some bridge hence on another | ||||||
|  | 		 * bus. | ||||||
|  | 		 */ | ||||||
|  | 		if (devfn) | ||||||
|  | 			return NULL; | ||||||
|  | 
 | ||||||
|  | 		return pcie->reg_base + (where & 0xfff); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Update Output registers for AXI region 0. */ | ||||||
|  | 	addr0 = CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_NBITS(12) | | ||||||
|  | 		CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_DEVFN(devfn) | | ||||||
|  | 		CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_BUS(busn); | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_OB_REGION_PCI_ADDR0(0), addr0); | ||||||
|  | 
 | ||||||
|  | 	/* Configuration Type 0 or Type 1 access. */ | ||||||
|  | 	desc0 = CDNS_PCIE_AT_OB_REGION_DESC0_HARDCODED_RID | | ||||||
|  | 		CDNS_PCIE_AT_OB_REGION_DESC0_DEVFN(0); | ||||||
|  | 	/*
 | ||||||
|  | 	 * The bus number was already set once for all in desc1 by | ||||||
|  | 	 * cdns_pcie_host_init_address_translation(). | ||||||
|  | 	 */ | ||||||
|  | 	if (busn == rc->bus_range->start + 1) | ||||||
|  | 		desc0 |= CDNS_PCIE_AT_OB_REGION_DESC0_TYPE_CONF_TYPE0; | ||||||
|  | 	else | ||||||
|  | 		desc0 |= CDNS_PCIE_AT_OB_REGION_DESC0_TYPE_CONF_TYPE1; | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_OB_REGION_DESC0(0), desc0); | ||||||
|  | 
 | ||||||
|  | 	return rc->cfg_base + (where & 0xfff); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static struct pci_ops cdns_pcie_host_ops = { | ||||||
|  | 	.map_bus	= cdns_pci_map_bus, | ||||||
|  | 	.read		= pci_generic_config_read, | ||||||
|  | 	.write		= pci_generic_config_write, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static const struct of_device_id cdns_pcie_host_of_match[] = { | ||||||
|  | 	{ .compatible = "cdns,cdns-pcie-host" }, | ||||||
|  | 
 | ||||||
|  | 	{ }, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static int cdns_pcie_host_init_root_port(struct cdns_pcie_rc *rc) | ||||||
|  | { | ||||||
|  | 	struct cdns_pcie *pcie = &rc->pcie; | ||||||
|  | 	u32 value, ctrl; | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * Set the root complex BAR configuration register: | ||||||
|  | 	 * - disable both BAR0 and BAR1. | ||||||
|  | 	 * - enable Prefetchable Memory Base and Limit registers in type 1 | ||||||
|  | 	 *   config space (64 bits). | ||||||
|  | 	 * - enable IO Base and Limit registers in type 1 config | ||||||
|  | 	 *   space (32 bits). | ||||||
|  | 	 */ | ||||||
|  | 	ctrl = CDNS_PCIE_LM_BAR_CFG_CTRL_DISABLED; | ||||||
|  | 	value = CDNS_PCIE_LM_RC_BAR_CFG_BAR0_CTRL(ctrl) | | ||||||
|  | 		CDNS_PCIE_LM_RC_BAR_CFG_BAR1_CTRL(ctrl) | | ||||||
|  | 		CDNS_PCIE_LM_RC_BAR_CFG_PREFETCH_MEM_ENABLE | | ||||||
|  | 		CDNS_PCIE_LM_RC_BAR_CFG_PREFETCH_MEM_64BITS | | ||||||
|  | 		CDNS_PCIE_LM_RC_BAR_CFG_IO_ENABLE | | ||||||
|  | 		CDNS_PCIE_LM_RC_BAR_CFG_IO_32BITS; | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_LM_RC_BAR_CFG, value); | ||||||
|  | 
 | ||||||
|  | 	/* Set root port configuration space */ | ||||||
|  | 	if (rc->vendor_id != 0xffff) | ||||||
|  | 		cdns_pcie_rp_writew(pcie, PCI_VENDOR_ID, rc->vendor_id); | ||||||
|  | 	if (rc->device_id != 0xffff) | ||||||
|  | 		cdns_pcie_rp_writew(pcie, PCI_DEVICE_ID, rc->device_id); | ||||||
|  | 
 | ||||||
|  | 	cdns_pcie_rp_writeb(pcie, PCI_CLASS_REVISION, 0); | ||||||
|  | 	cdns_pcie_rp_writeb(pcie, PCI_CLASS_PROG, 0); | ||||||
|  | 	cdns_pcie_rp_writew(pcie, PCI_CLASS_DEVICE, PCI_CLASS_BRIDGE_PCI); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int cdns_pcie_host_init_address_translation(struct cdns_pcie_rc *rc) | ||||||
|  | { | ||||||
|  | 	struct cdns_pcie *pcie = &rc->pcie; | ||||||
|  | 	struct resource *cfg_res = rc->cfg_res; | ||||||
|  | 	struct resource *mem_res = pcie->mem_res; | ||||||
|  | 	struct resource *bus_range = rc->bus_range; | ||||||
|  | 	struct device *dev = rc->dev; | ||||||
|  | 	struct device_node *np = dev->of_node; | ||||||
|  | 	struct of_pci_range_parser parser; | ||||||
|  | 	struct of_pci_range range; | ||||||
|  | 	u32 addr0, addr1, desc1; | ||||||
|  | 	u64 cpu_addr; | ||||||
|  | 	int r, err; | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * Reserve region 0 for PCI configure space accesses: | ||||||
|  | 	 * OB_REGION_PCI_ADDR0 and OB_REGION_DESC0 are updated dynamically by | ||||||
|  | 	 * cdns_pci_map_bus(), other region registers are set here once for all. | ||||||
|  | 	 */ | ||||||
|  | 	addr1 = 0; /* Should be programmed to zero. */ | ||||||
|  | 	desc1 = CDNS_PCIE_AT_OB_REGION_DESC1_BUS(bus_range->start); | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_OB_REGION_PCI_ADDR1(0), addr1); | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_OB_REGION_DESC1(0), desc1); | ||||||
|  | 
 | ||||||
|  | 	cpu_addr = cfg_res->start - mem_res->start; | ||||||
|  | 	addr0 = CDNS_PCIE_AT_OB_REGION_CPU_ADDR0_NBITS(12) | | ||||||
|  | 		(lower_32_bits(cpu_addr) & GENMASK(31, 8)); | ||||||
|  | 	addr1 = upper_32_bits(cpu_addr); | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_OB_REGION_CPU_ADDR0(0), addr0); | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_OB_REGION_CPU_ADDR1(0), addr1); | ||||||
|  | 
 | ||||||
|  | 	err = of_pci_range_parser_init(&parser, np); | ||||||
|  | 	if (err) | ||||||
|  | 		return err; | ||||||
|  | 
 | ||||||
|  | 	r = 1; | ||||||
|  | 	for_each_of_pci_range(&parser, &range) { | ||||||
|  | 		bool is_io; | ||||||
|  | 
 | ||||||
|  | 		if (r >= rc->max_regions) | ||||||
|  | 			break; | ||||||
|  | 
 | ||||||
|  | 		if ((range.flags & IORESOURCE_TYPE_BITS) == IORESOURCE_MEM) | ||||||
|  | 			is_io = false; | ||||||
|  | 		else if ((range.flags & IORESOURCE_TYPE_BITS) == IORESOURCE_IO) | ||||||
|  | 			is_io = true; | ||||||
|  | 		else | ||||||
|  | 			continue; | ||||||
|  | 
 | ||||||
|  | 		cdns_pcie_set_outbound_region(pcie, r, is_io, | ||||||
|  | 					      range.cpu_addr, | ||||||
|  | 					      range.pci_addr, | ||||||
|  | 					      range.size); | ||||||
|  | 		r++; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * Set Root Port no BAR match Inbound Translation registers: | ||||||
|  | 	 * needed for MSI and DMA. | ||||||
|  | 	 * Root Port BAR0 and BAR1 are disabled, hence no need to set their | ||||||
|  | 	 * inbound translation registers. | ||||||
|  | 	 */ | ||||||
|  | 	addr0 = CDNS_PCIE_AT_IB_RP_BAR_ADDR0_NBITS(rc->no_bar_nbits); | ||||||
|  | 	addr1 = 0; | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_IB_RP_BAR_ADDR0(RP_NO_BAR), addr0); | ||||||
|  | 	cdns_pcie_writel(pcie, CDNS_PCIE_AT_IB_RP_BAR_ADDR1(RP_NO_BAR), addr1); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int cdns_pcie_host_init(struct device *dev, | ||||||
|  | 			       struct list_head *resources, | ||||||
|  | 			       struct cdns_pcie_rc *rc) | ||||||
|  | { | ||||||
|  | 	struct resource *bus_range = NULL; | ||||||
|  | 	int err; | ||||||
|  | 
 | ||||||
|  | 	/* Parse our PCI ranges and request their resources */ | ||||||
|  | 	err = pci_parse_request_of_pci_ranges(dev, resources, &bus_range); | ||||||
|  | 	if (err) | ||||||
|  | 		return err; | ||||||
|  | 
 | ||||||
|  | 	rc->bus_range = bus_range; | ||||||
|  | 	rc->pcie.bus = bus_range->start; | ||||||
|  | 
 | ||||||
|  | 	err = cdns_pcie_host_init_root_port(rc); | ||||||
|  | 	if (err) | ||||||
|  | 		goto err_out; | ||||||
|  | 
 | ||||||
|  | 	err = cdns_pcie_host_init_address_translation(rc); | ||||||
|  | 	if (err) | ||||||
|  | 		goto err_out; | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | 
 | ||||||
|  |  err_out: | ||||||
|  | 	pci_free_resource_list(resources); | ||||||
|  | 	return err; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 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; | ||||||
|  | 	struct list_head resources; | ||||||
|  | 	struct cdns_pcie_rc *rc; | ||||||
|  | 	struct cdns_pcie *pcie; | ||||||
|  | 	struct resource *res; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	bridge = devm_pci_alloc_host_bridge(dev, sizeof(*rc)); | ||||||
|  | 	if (!bridge) | ||||||
|  | 		return -ENOMEM; | ||||||
|  | 
 | ||||||
|  | 	rc = pci_host_bridge_priv(bridge); | ||||||
|  | 	rc->dev = dev; | ||||||
|  | 
 | ||||||
|  | 	pcie = &rc->pcie; | ||||||
|  | 
 | ||||||
|  | 	rc->max_regions = 32; | ||||||
|  | 	of_property_read_u32(np, "cdns,max-outbound-regions", &rc->max_regions); | ||||||
|  | 
 | ||||||
|  | 	rc->no_bar_nbits = 32; | ||||||
|  | 	of_property_read_u32(np, "cdns,no-bar-match-nbits", &rc->no_bar_nbits); | ||||||
|  | 
 | ||||||
|  | 	rc->vendor_id = 0xffff; | ||||||
|  | 	of_property_read_u16(np, "vendor-id", &rc->vendor_id); | ||||||
|  | 
 | ||||||
|  | 	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)) { | ||||||
|  | 		dev_err(dev, "missing \"reg\"\n"); | ||||||
|  | 		return PTR_ERR(pcie->reg_base); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "cfg"); | ||||||
|  | 	rc->cfg_base = devm_pci_remap_cfg_resource(dev, res); | ||||||
|  | 	if (IS_ERR(rc->cfg_base)) { | ||||||
|  | 		dev_err(dev, "missing \"cfg\"\n"); | ||||||
|  | 		return PTR_ERR(rc->cfg_base); | ||||||
|  | 	} | ||||||
|  | 	rc->cfg_res = res; | ||||||
|  | 
 | ||||||
|  | 	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "mem"); | ||||||
|  | 	if (!res) { | ||||||
|  | 		dev_err(dev, "missing \"mem\"\n"); | ||||||
|  | 		return -EINVAL; | ||||||
|  | 	} | ||||||
|  | 	pcie->mem_res = res; | ||||||
|  | 
 | ||||||
|  | 	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 = cdns_pcie_host_init(dev, &resources, rc); | ||||||
|  | 	if (ret) | ||||||
|  | 		goto err_init; | ||||||
|  | 
 | ||||||
|  | 	list_splice_init(&resources, &bridge->windows); | ||||||
|  | 	bridge->dev.parent = dev; | ||||||
|  | 	bridge->busnr = pcie->bus; | ||||||
|  | 	bridge->ops = &cdns_pcie_host_ops; | ||||||
|  | 	bridge->map_irq = of_irq_parse_and_map_pci; | ||||||
|  | 	bridge->swizzle_irq = pci_common_swizzle; | ||||||
|  | 
 | ||||||
|  | 	ret = pci_host_probe(bridge); | ||||||
|  | 	if (ret < 0) | ||||||
|  | 		goto err_host_probe; | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | 
 | ||||||
|  |  err_host_probe: | ||||||
|  | 	pci_free_resource_list(&resources); | ||||||
|  | 
 | ||||||
|  |  err_init: | ||||||
|  | 	pm_runtime_put_sync(dev); | ||||||
|  | 
 | ||||||
|  |  err_get_sync: | ||||||
|  | 	pm_runtime_disable(dev); | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static struct platform_driver cdns_pcie_host_driver = { | ||||||
|  | 	.driver = { | ||||||
|  | 		.name = "cdns-pcie-host", | ||||||
|  | 		.of_match_table = cdns_pcie_host_of_match, | ||||||
|  | 	}, | ||||||
|  | 	.probe = cdns_pcie_host_probe, | ||||||
|  | }; | ||||||
|  | builtin_platform_driver(cdns_pcie_host_driver); | ||||||
							
								
								
									
										189
									
								
								drivers/pci/host/pcie-cadence.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										189
									
								
								drivers/pci/host/pcie-cadence.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,189 @@ | |||||||
|  | // SPDX-License-Identifier: GPL-2.0
 | ||||||
|  | // Copyright (c) 2017 Cadence
 | ||||||
|  | // Cadence PCIe controller driver.
 | ||||||
|  | // Author: Cyrille Pitchen <cyrille.pitchen@free-electrons.com>
 | ||||||
|  | 
 | ||||||
|  | #ifndef _PCIE_CADENCE_H | ||||||
|  | #define _PCIE_CADENCE_H | ||||||
|  | 
 | ||||||
|  | #include <linux/kernel.h> | ||||||
|  | #include <linux/pci.h> | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Local Management Registers | ||||||
|  |  */ | ||||||
|  | #define CDNS_PCIE_LM_BASE	0x00100000 | ||||||
|  | 
 | ||||||
|  | /* Vendor ID Register */ | ||||||
|  | #define CDNS_PCIE_LM_ID		(CDNS_PCIE_LM_BASE + 0x0044) | ||||||
|  | #define  CDNS_PCIE_LM_ID_VENDOR_MASK	GENMASK(15, 0) | ||||||
|  | #define  CDNS_PCIE_LM_ID_VENDOR_SHIFT	0 | ||||||
|  | #define  CDNS_PCIE_LM_ID_VENDOR(vid) \ | ||||||
|  | 	(((vid) << CDNS_PCIE_LM_ID_VENDOR_SHIFT) & CDNS_PCIE_LM_ID_VENDOR_MASK) | ||||||
|  | #define  CDNS_PCIE_LM_ID_SUBSYS_MASK	GENMASK(31, 16) | ||||||
|  | #define  CDNS_PCIE_LM_ID_SUBSYS_SHIFT	16 | ||||||
|  | #define  CDNS_PCIE_LM_ID_SUBSYS(sub) \ | ||||||
|  | 	(((sub) << CDNS_PCIE_LM_ID_SUBSYS_SHIFT) & CDNS_PCIE_LM_ID_SUBSYS_MASK) | ||||||
|  | 
 | ||||||
|  | /* Root Port Requestor ID Register */ | ||||||
|  | #define CDNS_PCIE_LM_RP_RID	(CDNS_PCIE_LM_BASE + 0x0228) | ||||||
|  | #define  CDNS_PCIE_LM_RP_RID_MASK	GENMASK(15, 0) | ||||||
|  | #define  CDNS_PCIE_LM_RP_RID_SHIFT	0 | ||||||
|  | #define  CDNS_PCIE_LM_RP_RID_(rid) \ | ||||||
|  | 	(((rid) << CDNS_PCIE_LM_RP_RID_SHIFT) & CDNS_PCIE_LM_RP_RID_MASK) | ||||||
|  | 
 | ||||||
|  | /* Root Complex BAR Configuration Register */ | ||||||
|  | #define CDNS_PCIE_LM_RC_BAR_CFG	(CDNS_PCIE_LM_BASE + 0x0300) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_BAR0_APERTURE_MASK	GENMASK(5, 0) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_BAR0_APERTURE(a) \ | ||||||
|  | 	(((a) << 0) & CDNS_PCIE_LM_RC_BAR_CFG_BAR0_APERTURE_MASK) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_BAR0_CTRL_MASK		GENMASK(8, 6) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_BAR0_CTRL(c) \ | ||||||
|  | 	(((c) << 6) & CDNS_PCIE_LM_RC_BAR_CFG_BAR0_CTRL_MASK) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_BAR1_APERTURE_MASK	GENMASK(13, 9) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_BAR1_APERTURE(a) \ | ||||||
|  | 	(((a) << 9) & CDNS_PCIE_LM_RC_BAR_CFG_BAR1_APERTURE_MASK) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_BAR1_CTRL_MASK		GENMASK(16, 14) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_BAR1_CTRL(c) \ | ||||||
|  | 	(((c) << 14) & CDNS_PCIE_LM_RC_BAR_CFG_BAR1_CTRL_MASK) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_PREFETCH_MEM_ENABLE	BIT(17) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_PREFETCH_MEM_32BITS	0 | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_PREFETCH_MEM_64BITS	BIT(18) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_IO_ENABLE		BIT(19) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_IO_16BITS		0 | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_IO_32BITS		BIT(20) | ||||||
|  | #define  CDNS_PCIE_LM_RC_BAR_CFG_CHECK_ENABLE		BIT(31) | ||||||
|  | 
 | ||||||
|  | /* BAR control values applicable to both Endpoint Function and Root Complex */ | ||||||
|  | #define  CDNS_PCIE_LM_BAR_CFG_CTRL_DISABLED		0x0 | ||||||
|  | #define  CDNS_PCIE_LM_BAR_CFG_CTRL_IO_32BITS		0x1 | ||||||
|  | #define  CDNS_PCIE_LM_BAR_CFG_CTRL_MEM_32BITS		0x4 | ||||||
|  | #define  CDNS_PCIE_LM_BAR_CFG_CTRL_PREFETCH_MEM_32BITS	0x5 | ||||||
|  | #define  CDNS_PCIE_LM_BAR_CFG_CTRL_MEM_64BITS		0x6 | ||||||
|  | #define  CDNS_PCIE_LM_BAR_CFG_CTRL_PREFETCH_MEM_64BITS	0x7 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Root Port Registers (PCI configuration space for the root port function) | ||||||
|  |  */ | ||||||
|  | #define CDNS_PCIE_RP_BASE	0x00200000 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Address Translation Registers | ||||||
|  |  */ | ||||||
|  | #define CDNS_PCIE_AT_BASE	0x00400000 | ||||||
|  | 
 | ||||||
|  | /* Region r Outbound AXI to PCIe Address Translation Register 0 */ | ||||||
|  | #define CDNS_PCIE_AT_OB_REGION_PCI_ADDR0(r) \ | ||||||
|  | 	(CDNS_PCIE_AT_BASE + 0x0000 + ((r) & 0x1f) * 0x0020) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_NBITS_MASK	GENMASK(5, 0) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_NBITS(nbits) \ | ||||||
|  | 	(((nbits) - 1) & CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_NBITS_MASK) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_DEVFN_MASK	GENMASK(19, 12) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_DEVFN(devfn) \ | ||||||
|  | 	(((devfn) << 12) & CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_DEVFN_MASK) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_BUS_MASK	GENMASK(27, 20) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_BUS(bus) \ | ||||||
|  | 	(((bus) << 20) & CDNS_PCIE_AT_OB_REGION_PCI_ADDR0_BUS_MASK) | ||||||
|  | 
 | ||||||
|  | /* Region r Outbound AXI to PCIe Address Translation Register 1 */ | ||||||
|  | #define CDNS_PCIE_AT_OB_REGION_PCI_ADDR1(r) \ | ||||||
|  | 	(CDNS_PCIE_AT_BASE + 0x0004 + ((r) & 0x1f) * 0x0020) | ||||||
|  | 
 | ||||||
|  | /* Region r Outbound PCIe Descriptor Register 0 */ | ||||||
|  | #define CDNS_PCIE_AT_OB_REGION_DESC0(r) \ | ||||||
|  | 	(CDNS_PCIE_AT_BASE + 0x0008 + ((r) & 0x1f) * 0x0020) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_DESC0_TYPE_MASK		GENMASK(3, 0) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_DESC0_TYPE_MEM		0x2 | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_DESC0_TYPE_IO		0x6 | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_DESC0_TYPE_CONF_TYPE0	0xa | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_DESC0_TYPE_CONF_TYPE1	0xb | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_DESC0_TYPE_NORMAL_MSG	0xc | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_DESC0_TYPE_VENDOR_MSG	0xd | ||||||
|  | /* Bit 23 MUST be set in RC mode. */ | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_DESC0_HARDCODED_RID	BIT(23) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_DESC0_DEVFN_MASK	GENMASK(31, 24) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_DESC0_DEVFN(devfn) \ | ||||||
|  | 	(((devfn) << 24) & CDNS_PCIE_AT_OB_REGION_DESC0_DEVFN_MASK) | ||||||
|  | 
 | ||||||
|  | /* Region r Outbound PCIe Descriptor Register 1 */ | ||||||
|  | #define CDNS_PCIE_AT_OB_REGION_DESC1(r)	\ | ||||||
|  | 	(CDNS_PCIE_AT_BASE + 0x000c + ((r) & 0x1f) * 0x0020) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_DESC1_BUS_MASK	GENMASK(7, 0) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_DESC1_BUS(bus) \ | ||||||
|  | 	((bus) & CDNS_PCIE_AT_OB_REGION_DESC1_BUS_MASK) | ||||||
|  | 
 | ||||||
|  | /* Region r AXI Region Base Address Register 0 */ | ||||||
|  | #define CDNS_PCIE_AT_OB_REGION_CPU_ADDR0(r) \ | ||||||
|  | 	(CDNS_PCIE_AT_BASE + 0x0018 + ((r) & 0x1f) * 0x0020) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_CPU_ADDR0_NBITS_MASK	GENMASK(5, 0) | ||||||
|  | #define  CDNS_PCIE_AT_OB_REGION_CPU_ADDR0_NBITS(nbits) \ | ||||||
|  | 	(((nbits) - 1) & CDNS_PCIE_AT_OB_REGION_CPU_ADDR0_NBITS_MASK) | ||||||
|  | 
 | ||||||
|  | /* Region r AXI Region Base Address Register 1 */ | ||||||
|  | #define CDNS_PCIE_AT_OB_REGION_CPU_ADDR1(r) \ | ||||||
|  | 	(CDNS_PCIE_AT_BASE + 0x001c + ((r) & 0x1f) * 0x0020) | ||||||
|  | 
 | ||||||
|  | /* Root Port BAR Inbound PCIe to AXI Address Translation Register */ | ||||||
|  | #define CDNS_PCIE_AT_IB_RP_BAR_ADDR0(bar) \ | ||||||
|  | 	(CDNS_PCIE_AT_BASE + 0x0800 + (bar) * 0x0008) | ||||||
|  | #define  CDNS_PCIE_AT_IB_RP_BAR_ADDR0_NBITS_MASK	GENMASK(5, 0) | ||||||
|  | #define  CDNS_PCIE_AT_IB_RP_BAR_ADDR0_NBITS(nbits) \ | ||||||
|  | 	(((nbits) - 1) & CDNS_PCIE_AT_IB_RP_BAR_ADDR0_NBITS_MASK) | ||||||
|  | #define CDNS_PCIE_AT_IB_RP_BAR_ADDR1(bar) \ | ||||||
|  | 	(CDNS_PCIE_AT_BASE + 0x0804 + (bar) * 0x0008) | ||||||
|  | 
 | ||||||
|  | enum cdns_pcie_rp_bar { | ||||||
|  | 	RP_BAR0, | ||||||
|  | 	RP_BAR1, | ||||||
|  | 	RP_NO_BAR | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * struct cdns_pcie - private data for Cadence PCIe controller drivers | ||||||
|  |  * @reg_base: IO mapped register base | ||||||
|  |  * @mem_res: start/end offsets in the physical system memory to map PCI accesses | ||||||
|  |  * @bus: In Root Complex mode, the bus number | ||||||
|  |  */ | ||||||
|  | struct cdns_pcie { | ||||||
|  | 	void __iomem		*reg_base; | ||||||
|  | 	struct resource		*mem_res; | ||||||
|  | 	u8			bus; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* Register access */ | ||||||
|  | static inline void cdns_pcie_writeb(struct cdns_pcie *pcie, u32 reg, u8 value) | ||||||
|  | { | ||||||
|  | 	writeb(value, pcie->reg_base + reg); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void cdns_pcie_writew(struct cdns_pcie *pcie, u32 reg, u16 value) | ||||||
|  | { | ||||||
|  | 	writew(value, pcie->reg_base + reg); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void cdns_pcie_writel(struct cdns_pcie *pcie, u32 reg, u32 value) | ||||||
|  | { | ||||||
|  | 	writel(value, pcie->reg_base + reg); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline u32 cdns_pcie_readl(struct cdns_pcie *pcie, u32 reg) | ||||||
|  | { | ||||||
|  | 	return readl(pcie->reg_base + reg); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Root Port register access */ | ||||||
|  | static inline void cdns_pcie_rp_writeb(struct cdns_pcie *pcie, | ||||||
|  | 				       u32 reg, u8 value) | ||||||
|  | { | ||||||
|  | 	writeb(value, pcie->reg_base + CDNS_PCIE_RP_BASE + reg); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void cdns_pcie_rp_writew(struct cdns_pcie *pcie, | ||||||
|  | 				       u32 reg, u16 value) | ||||||
|  | { | ||||||
|  | 	writew(value, pcie->reg_base + CDNS_PCIE_RP_BASE + reg); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #endif /* _PCIE_CADENCE_H */ | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user