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…
x
Reference in New Issue
Block a user