Hi David, On Mon, Sep 28, 2015 at 05:56:24PM -0700, David Daney wrote: > From: David Daney <david.daney@xxxxxxxxxx> > > Early versions of the Cavium Thunder CN88XX processor are missing > Enhanced Allocation (EA) capabilities for the fixed BAR addresses used > by the on-SoC hardware blocks. > > Add config access functions that synthesize the missing EA > capabilities for versions that are missing that information. Since > this is a little hacky, gate the inclusion of the code with a new > Kconfig variable. > > Signed-off-by: David Daney <david.daney@xxxxxxxxxx> What about this one? Do we still need it? This version looks like it still has some debug code and it feels like a lot of hard-coding of config offsets; it'd be nice if it could be more table-driven. But maybe this isn't needed anymore anyway. Bjorn > --- > > As suggested by Bjorn Helgaas... It is RFC at this point, but this is > working well for me. > > Depends on: > > https://lkml.org/lkml/2015/9/28/796 > > drivers/pci/host/Kconfig | 9 + > drivers/pci/host/Makefile | 1 + > drivers/pci/host/pci-host-generic.c | 22 ++- > drivers/pci/host/thunder_ecam_config_io.c | 273 ++++++++++++++++++++++++++++++ > 4 files changed, 304 insertions(+), 1 deletion(-) > create mode 100644 drivers/pci/host/thunder_ecam_config_io.c > > diff --git a/drivers/pci/host/Kconfig b/drivers/pci/host/Kconfig > index d5e58ba..9f3a9cd 100644 > --- a/drivers/pci/host/Kconfig > +++ b/drivers/pci/host/Kconfig > @@ -58,6 +58,15 @@ config PCI_HOST_GENERIC > Say Y here if you want to support a simple generic PCI host > controller, such as the one emulated by kvmtool. > > +config PCI_HOST_THUNDER > + bool "Extensions to Generic PCI host controller for Cavium Thunder" > + depends on PCI_HOST_GENERIC && ARM64 > + help > + Say Y here to enable PCI config access methods needed by > + CN88XX Cavium Thunder SoCs. The access is standard ECAM, > + but Enhanced Allocation (EA) capability structures are > + synthesized for on-SoC devices with fixed BARs. > + > config PCIE_SPEAR13XX > bool "STMicroelectronics SPEAr PCIe controller" > depends on ARCH_SPEAR13XX > diff --git a/drivers/pci/host/Makefile b/drivers/pci/host/Makefile > index 140d66f..8b77d62 100644 > --- a/drivers/pci/host/Makefile > +++ b/drivers/pci/host/Makefile > @@ -7,6 +7,7 @@ obj-$(CONFIG_PCI_TEGRA) += pci-tegra.o > obj-$(CONFIG_PCI_RCAR_GEN2) += pci-rcar-gen2.o > obj-$(CONFIG_PCI_RCAR_GEN2_PCIE) += pcie-rcar.o > obj-$(CONFIG_PCI_HOST_GENERIC) += pci-host-generic.o > +obj-$(CONFIG_PCI_HOST_THUNDER) += thunder_ecam_config_io.o > obj-$(CONFIG_PCIE_SPEAR13XX) += pcie-spear13xx.o > obj-$(CONFIG_PCI_KEYSTONE) += pci-keystone-dw.o pci-keystone.o > obj-$(CONFIG_PCIE_XILINX) += pcie-xilinx.o > diff --git a/drivers/pci/host/pci-host-generic.c b/drivers/pci/host/pci-host-generic.c > index 6f12830..64558e5 100644 > --- a/drivers/pci/host/pci-host-generic.c > +++ b/drivers/pci/host/pci-host-generic.c > @@ -91,6 +91,21 @@ static struct gen_pci_cfg_bus_ops gen_pci_cfg_ecam_bus_ops = { > } > }; > > +#ifdef CONFIG_PCI_HOST_THUNDER > +int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn, > + int where, int size, u32 *val); > +int thunder_ecam_config_write(struct pci_bus *bus, unsigned int devfn, > + int where, int size, u32 val); > +static struct gen_pci_cfg_bus_ops gen_pci_cfg_thunder_ecam_bus_ops = { > + .bus_shift = 20, > + .ops = { > + .map_bus = gen_pci_map_cfg_bus_ecam, > + .read = thunder_ecam_config_read, > + .write = thunder_ecam_config_write, > + } > +}; > +#endif > + > static void __iomem *gen_pci_map_cfg_bus_thunder_pem(struct pci_bus *bus, > unsigned int devfn, > int where) > @@ -108,6 +123,7 @@ static void __iomem *gen_pci_map_cfg_bus_thunder_pem(struct pci_bus *bus, > return pci->cfg.win[idx] + ((devfn << 16) | where); > } > > +#ifdef CONFIG_PCI_HOST_THUNDER > static struct gen_pci_cfg_bus_ops gen_pci_cfg_thunder_pem_bus_ops = { > .bus_shift = 24, > .ops = { > @@ -116,6 +132,7 @@ static struct gen_pci_cfg_bus_ops gen_pci_cfg_thunder_pem_bus_ops = { > .write = pci_generic_config_write, > } > }; > +#endif > > static const struct of_device_id gen_pci_of_match[] = { > { .compatible = "pci-host-cam-generic", > @@ -126,7 +143,10 @@ static const struct of_device_id gen_pci_of_match[] = { > > { .compatible = "cavium,pci-host-thunder-pem", > .data = &gen_pci_cfg_thunder_pem_bus_ops }, > - > +#ifdef CONFIG_PCI_HOST_THUNDER > + { .compatible = "cavium,pci-host-thunder-ecam", > + .data = &gen_pci_cfg_thunder_ecam_bus_ops }, > +#endif > { }, > }; > MODULE_DEVICE_TABLE(of, gen_pci_of_match); > diff --git a/drivers/pci/host/thunder_ecam_config_io.c b/drivers/pci/host/thunder_ecam_config_io.c > new file mode 100644 > index 0000000..58c3109 > --- /dev/null > +++ b/drivers/pci/host/thunder_ecam_config_io.c > @@ -0,0 +1,273 @@ > +/* > + * > + * This file is subject to the terms and conditions of the GNU General Public > + * License. See the file "COPYING" in the main directory of this archive > + * for more details. > + * > + * Copyright (C) 2015 Cavium, Inc. > + * > + */ > + > +#include <linux/pci.h> > +#include <linux/ioport.h> > +#include <linux/printk.h> > + > +static void set_val(u32 v, int where, int size, u32 *val) > +{ > + int shift = (where & 3) * 8; > + > + pr_debug("set_val %04x: %08x\n", (unsigned)(where & ~3), v); > + v >>= shift; > + if (size == 1) > + v &= 0xff; > + else if (size == 2) > + v &= 0xffff; > + *val = v; > +} > + > +static int handle_ea_bar(u32 e0, int bar, struct pci_bus *bus, > + unsigned int devfn, int where, int size, u32 *val) > +{ > + void __iomem *addr; > + u32 v; > + /* > + * Each entry is 16-byte aligned bits[2,3] select which word > + * in the entry > + */ > + int where_a = where & 0xc; > + > + if (where_a == 0) { > + set_val(e0, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a == 0x4) { > + addr = bus->ops->map_bus(bus, devfn, bar); /* BAR 0 */ > + if (!addr) { > + *val = ~0; > + return PCIBIOS_DEVICE_NOT_FOUND; > + } > + v = readl(addr); > + v &= ~0xf; > + v |= 2; /* EA entry-1. Base-L */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a == 0x8) { > + u32 barl_orig; > + u32 barl_rb; > + > + addr = bus->ops->map_bus(bus, devfn, bar); /* BAR 0 */ > + if (!addr) { > + *val = ~0; > + return PCIBIOS_DEVICE_NOT_FOUND; > + } > + barl_orig = readl(addr + 0); > + writel(0xffffffff, addr + 0); > + barl_rb = readl(addr + 0); > + writel(barl_orig, addr + 0); > + /* zeros in unsettable bits. */ > + v = ~barl_rb & ~3; > + v |= 0xc; /* EA entry-2. Offset-L */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a == 0xc) { > + addr = bus->ops->map_bus(bus, devfn, bar + 4); /* BAR 1 */ > + if (!addr) { > + *val = ~0; > + return PCIBIOS_DEVICE_NOT_FOUND; > + } > + v = readl(addr); /* EA entry-3. Base-H */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + return PCIBIOS_DEVICE_NOT_FOUND; > +} > +int thunder_ecam_config_write(struct pci_bus *bus, unsigned int devfn, > + int where, int size, u32 val) > +{ > + /* > + * All BARs have fixed addresses, ignore BAR writes so they > + * don't get corrupted. > + */ > + if ((where >= 0x10 && where < 0x2c) || (where >= 0x1a4 && where < 0x1bc)) > + /* BAR or SRIOV BAR */ > + return PCIBIOS_SUCCESSFUL; > + > + return pci_generic_config_write(bus, devfn, where, size, val); > +} > + > +int thunder_ecam_config_read(struct pci_bus *bus, unsigned int devfn, > + int where, int size, u32 *val) > +{ > + u32 v; > + u32 vendor_device; > + void __iomem *addr; > + int cfg_type; > + int where_a = where & ~3; > + > + /* > + * All BARs have fixed addresses specified by the EA > + * capability, they must return zero on read. > + */ > + if ((where >= 0x10 && where < 0x2c) || (where >= 0x1a4 && where < 0x1bc)) { > + /* BAR or SRIOV BAR */ > + *val = 0; > + return PCIBIOS_SUCCESSFUL; > + } > + > + addr = bus->ops->map_bus(bus, devfn, 0); > + if (!addr) { > + *val = ~0; > + return PCIBIOS_DEVICE_NOT_FOUND; > + } > + > + vendor_device = readl(addr); > + if (vendor_device == 0xffffffff) > + goto no_emulation; > + > + addr = bus->ops->map_bus(bus, devfn, 8); > + if (!addr) { > + *val = ~0; > + return PCIBIOS_DEVICE_NOT_FOUND; > + } > + > + v = readl(addr); > + if (v == 0xffffffff) > + goto no_emulation; > + > + if ((v & 0xff) < 8) { > + pr_debug("%04x:%04x - Fix pass#: %08x, where: %03x, devfn: %03x\n", > + vendor_device & 0xffff, vendor_device >> 16, v, (unsigned) where, devfn); > + /* pass 1.x*/ > + } else { > + pr_debug("%04x:%04x - OK pass#: %08x, devfn: %03x\n", > + vendor_device & 0xffff, vendor_device >> 16, v, devfn); > + goto no_emulation; > + } > + > + addr = bus->ops->map_bus(bus, devfn, 0xc); > + if (!addr) { > + *val = ~0; > + return PCIBIOS_DEVICE_NOT_FOUND; > + } > + > + v = readl(addr); > + /* Check for non type-00 header. */ > + cfg_type = (v >> 16) & 0x7f; > + if (cfg_type == 0) { > + bool has_msix; > + bool is_nic = (vendor_device == 0xa01e177d); > + bool is_tns = (vendor_device == 0xa01f177d); > + > + addr = bus->ops->map_bus(bus, devfn, 0x70); > + if (!addr) { > + *val = ~0; > + return PCIBIOS_DEVICE_NOT_FOUND; > + } > + /* E_CAP */ > + v = readl(addr); > + has_msix = (v & 0xff00) != 0; > + > + if (!has_msix && where_a == 0x70) { > + v |= 0xbc00; /* next capability is EA at 0xbc */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a == 0xb0) { > + addr = bus->ops->map_bus(bus, devfn, where_a); > + if (!addr) { > + *val = ~0; > + return PCIBIOS_DEVICE_NOT_FOUND; > + } > + v = readl(addr); > + if (v & 0xff00) > + pr_err("Bad MSIX cap header: %08x\n", v); > + v |= 0xbc00; /* next capability is EA at 0xbc */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a == 0xbc) { > + if (is_nic) > + v = 0x40014; /* EA last in chain, 4 entries. */ > + else if (is_tns) > + v = 0x40014; /* EA last in chain, 3 entries. */ > + else if (has_msix) > + v = 0x20014; /* EA last in chain, 2 entries. */ > + else > + v = 0x10014; /* EA last in chain, 1 entry. */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a >= 0xc0 && where_a < 0xd0) > + return handle_ea_bar(0x80ff0003, /* EA entry-0. PP=0, BAR0 Size:3 */ > + 0x10, bus, devfn, where, size, val); > + if (where_a >= 0xd0 && where_a < 0xe0 && has_msix) > + return handle_ea_bar(0x80ff0043, /* EA entry-1. PP=0, BAR4 Size:3 */ > + 0x20, bus, devfn, where, size, val); > + if (where_a >= 0xe0 && where_a < 0xf0 && is_tns) > + return handle_ea_bar(0x80ff0023, /* EA entry-2. PP=0, BAR2, Size:3 */ > + 0x18, bus, devfn, where, size, val); > + if (where_a >= 0xe0 && where_a < 0xf0 && is_nic) > + return handle_ea_bar(0x80ff0493, /* EA entry-2. PP=4, VF_BAR0 (9), Size:3 */ > + 0x1a4, bus, devfn, where, size, val); > + if (where_a >= 0xf0 && where_a < 0x100 && is_nic) > + return handle_ea_bar(0x80ff04d3, /* EA entry-3. PP=4, VF_BAR4 (d), Size:3 */ > + 0x1b4, bus, devfn, where, size, val); > + } else if (cfg_type == 1) { > + if (where_a == 0x70) { > + addr = bus->ops->map_bus(bus, devfn, where_a); > + if (!addr) { > + *val = ~0; > + return PCIBIOS_DEVICE_NOT_FOUND; > + } > + v = readl(addr); > + if (v & 0xff00) > + pr_err("Bad PCIe cap header: %08x\n", v); > + v |= 0xbc00; /* next capability is EA at 0xbc */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a == 0xbc) { > + v = 0x10014; /* EA last in chain, 1 entry. */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a == 0xc0) { > + v = 0x0101; /* subordinate:secondary = 1:1 */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a == 0xc4) { > + v = 0x80ff0564; /* Enabled, not-Write, SP=ff, PP=05, BEI=6, ES=4 */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a == 0xc8) { > + v = 0x00000002; /* Base-L 64-bit */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a == 0xcc) { > + v = 0xfffffffe; /* MaxOffset-L 64-bit */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a == 0xd0) { > + if (devfn == 8) > + v = 0x000087e0; /* RSL Base-H */ > + else > + v = 0x00008430; /* NIC Base-H */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + if (where_a == 0xd4) { > + v = 0x0000000f; /* MaxOffset-H */ > + set_val(v, where, size, val); > + return PCIBIOS_SUCCESSFUL; > + } > + > + } > +no_emulation: > + return pci_generic_config_read(bus, devfn, where, size, val); > +} > -- > 1.9.1 > > -- > To unsubscribe from this list: send the line "unsubscribe linux-kernel" in > the body of a message to majordomo@xxxxxxxxxxxxxxx > More majordomo info at http://vger.kernel.org/majordomo-info.html > Please read the FAQ at http://www.tux.org/lkml/ -- To unsubscribe from this list: send the line "unsubscribe linux-pci" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html