Hi Suneel, On Tue, 29 Oct 2019 at 14:08, Suneel Garapati <suneelgli...@gmail.com> wrote: > > From: Suneel Garapati <sgarap...@marvell.com> > > Adds support for PCI ECAM/PEM controllers found on OcteonTX > or OcteonTX2 SoC platforms. > > Signed-off-by: Suneel Garapati <sgarap...@marvell.com> > --- > drivers/pci/Kconfig | 8 + > drivers/pci/Makefile | 1 + > drivers/pci/pci_octeontx.c | 538 +++++++++++++++++++++++++++++++++++++ > 3 files changed, 547 insertions(+) > create mode 100644 drivers/pci/pci_octeontx.c > > diff --git a/drivers/pci/Kconfig b/drivers/pci/Kconfig > index 752439afcf..e865a06749 100644 > --- a/drivers/pci/Kconfig > +++ b/drivers/pci/Kconfig > @@ -115,6 +115,14 @@ config PCI_TEGRA > with a total of 5 lanes. Some boards require this for Ethernet > support to work (e.g. beaver, jetson-tk1). > > +config PCI_OCTEONTX > + bool "OcteonTX PCI support" > + depends on (ARCH_OCTEONTX || ARCH_OCTEONTX2) > + help > + Enable support for the OcteonTX/TX2 SoC family ECAM/PEM controllers. > + These controllers provide PCI configuration access to all on-board > + peripherals so it should only be disabled for testing purposes > + > config PCI_XILINX > bool "Xilinx AXI Bridge for PCI Express" > depends on DM_PCI > diff --git a/drivers/pci/Makefile b/drivers/pci/Makefile > index b1d3dc8610..7119d8bcb2 100644 > --- a/drivers/pci/Makefile > +++ b/drivers/pci/Makefile > @@ -40,3 +40,4 @@ obj-$(CONFIG_PCI_XILINX) += pcie_xilinx.o > obj-$(CONFIG_PCIE_INTEL_FPGA) += pcie_intel_fpga.o > obj-$(CONFIG_PCI_KEYSTONE) += pcie_dw_ti.o > obj-$(CONFIG_PCIE_MEDIATEK) += pcie_mediatek.o > +obj-$(CONFIG_PCI_OCTEONTX) += pci_octeontx.o > diff --git a/drivers/pci/pci_octeontx.c b/drivers/pci/pci_octeontx.c > new file mode 100644 > index 0000000000..d754f98bbd > --- /dev/null > +++ b/drivers/pci/pci_octeontx.c > @@ -0,0 +1,538 @@ > +// SPDX-License-Identifier: GPL-2.0 > +/* > + * Copyright (C) 2018 Marvell International Ltd. > + * > + * https://spdx.org/licenses > + */ > + > +#include <common.h> > +#include <dm.h> > +#include <errno.h> > +#include <fdtdec.h> > +#include <malloc.h> > +#include <pci.h> > + > +#include <asm/io.h> > + > +#if defined(CONFIG_ARCH_OCTEONTX2)
No need for this. We cannot have #ifdefs in drivers. Instead, use a .data field in the compatible strings and do the code conditionally at runtime. e.g. enum { TYPE1, TYPE2, TYPE3, }; static const struct udevice_id pci_octeontx_ecam_ids[] = { { .compatible = "cavium,pci-host-thunder-ecam", TYPE1 }, { .compatible = "cavium,pci-host-octeontx-ecam",TYPE2 }, { .compatible = "pci-host-ecam-generic", TYPE3 }, { } }; ... priv->type = dev_get_driver_data(dev); if (priv->type == TYPE2) ... do something > + > +#define PEM_CFG_WR 0x18 > +#define PEM_CFG_RD 0x20 > + > +#define PCIERC_RASDP_DE_ME 0x440 > +#define PCIERC_RASDP_EP_CTL 0x420 > +#define PCIERC_RAS_EINJ_EN 0x348 > +#define PCIERC_RAS_EINJ_CTL6PE 0x3A4 > +#define PCIERC_RAS_EINJ_CTL6_CMPP0 0x364 > +#define PCIERC_RAS_EINJ_CTL6_CMPV0 0x374 > +#define PCIERC_RAS_EINJ_CTL6_CHGP1 0x388 > +#define PCIERC_RAS_EINJ_CTL6_CHGV1 0x398 > + > +#endif > + > +DECLARE_GLOBAL_DATA_PTR; > + > +struct octeontx_pci { comments! > + unsigned int type; > + > + struct fdt_resource cfg; > + struct fdt_resource bus; > +#if defined(CONFIG_ARCH_OCTEONTX2) Again please remove all #ifdefs in drivers > + struct fdt_resource pem; > +#endif > +}; > + > +static int pci_octeontx_ecam_read_config(struct udevice *bus, pci_dev_t bdf, > + uint offset, ulong *valuep, > + enum pci_size_t size) > +{ > + struct octeontx_pci *pcie = (void *)dev_get_priv(bus); > + struct pci_controller *hose = dev_get_uclass_priv(bus); > + uintptr_t address; > + u32 b, d, f; > + > + b = PCI_BUS(bdf) + pcie->bus.start - hose->first_busno; > + d = PCI_DEV(bdf); > + f = PCI_FUNC(bdf); > + > + address = (b << 20) | (d << 15) | (f << 12) | offset; Looks like this all repeats below, so should go in a function. Also please avoid single-char var names unless they are index variables. E.g. bus, devno, func > + > + address += pcie->cfg.start; > + > + switch (size) { > + case PCI_SIZE_8: > + *valuep = readb(address); > + break; > + case PCI_SIZE_16: > + *valuep = readw(address); > + break; > + case PCI_SIZE_32: > + *valuep = readl(address); > + break; > + }; > + > + debug("%02x.%02x.%02x: u%d %x -> %lx\n", > + b, d, f, size, offset, *valuep); blank line before return (globally please) > + return 0; > +} > + > +static int pci_octeontx_ecam_write_config(struct udevice *bus, pci_dev_t bdf, > + uint offset, ulong valuep, > + enum pci_size_t size) > +{ > + struct octeontx_pci *pcie = (void *)dev_get_priv(bus); > + struct pci_controller *hose = dev_get_uclass_priv(bus); > + uintptr_t address; > + u32 b, d, f; > + > + b = PCI_BUS(bdf) + pcie->bus.start - hose->first_busno; > + d = PCI_DEV(bdf); > + f = PCI_FUNC(bdf); > + > + address = (b << 20) | (d << 15) | (f << 12) | offset; > + > + address += pcie->cfg.start; > + > + switch (size) { > + case PCI_SIZE_8: > + writeb(valuep, address); > + break; > + case PCI_SIZE_16: > + writew(valuep, address); > + break; > + case PCI_SIZE_32: > + writel(valuep, address); > + break; > + }; > + > + debug("%02x.%02x.%02x: u%d %x <- %lx\n", > + b, d, f, size, offset, valuep); > + return 0; > +} > + > +static int pci_octeontx_pem_read_config(struct udevice *bus, pci_dev_t bdf, > + uint offset, ulong *valuep, > + enum pci_size_t size) > +{ > + struct octeontx_pci *pcie = (void *)dev_get_priv(bus); > + struct pci_controller *hose = dev_get_uclass_priv(bus); > + uintptr_t address; > + u32 b, d, f; > + u8 hdrtype; > + u8 pri_bus = pcie->bus.start + 1 - hose->first_busno; > + u32 bus_offs = (pri_bus << 16) | (pri_bus << 8) | (pri_bus << 0); > + > + b = PCI_BUS(bdf) + 1 - hose->first_busno; > + d = PCI_DEV(bdf); > + f = PCI_FUNC(bdf); > + > + address = (b << 24) | (d << 19) | (f << 16); > + > + address += pcie->cfg.start; > + > + *valuep = pci_conv_32_to_size(~0UL, offset, size); > + > + if (b == 1 && d > 0) > + return 0; > + > + switch (size) { > + case PCI_SIZE_8: > + *valuep = readb(address + offset); > + break; > + case PCI_SIZE_16: > + *valuep = readw(address + offset); > + break; > + case PCI_SIZE_32: > + *valuep = readl(address + offset); > + break; > + default: > + printf("Invalid size\n"); > + } > + > + hdrtype = readb(address + PCI_HEADER_TYPE); > + drop blank line...best to group code by related lines. comment here? > + if (hdrtype == PCI_HEADER_TYPE_BRIDGE && > + offset >= PCI_PRIMARY_BUS && > + offset <= PCI_SUBORDINATE_BUS && > + *valuep != pci_conv_32_to_size(~0UL, offset, size)) { > + *valuep -= pci_conv_32_to_size(bus_offs, offset, size); > + } > + debug("%02x.%02x.%02x: u%d %x (%lx) -> %lx\n", > + b, d, f, size, offset, address, *valuep); > + return 0; > +} > + > +static int pci_octeontx_pem_write_config(struct udevice *bus, pci_dev_t bdf, > + uint offset, ulong value, > + enum pci_size_t size) > +{ > + struct octeontx_pci *pcie = (void *)dev_get_priv(bus); > + struct pci_controller *hose = dev_get_uclass_priv(bus); > + uintptr_t address; > + u32 b, d, f; > + u8 hdrtype; > + u8 pri_bus = pcie->bus.start + 1 - hose->first_busno; > + u32 bus_offs = (pri_bus << 16) | (pri_bus << 8) | (pri_bus << 0); > + > + b = PCI_BUS(bdf) + 1 - hose->first_busno; > + d = PCI_DEV(bdf); > + f = PCI_FUNC(bdf); > + > + address = (b << 24) | (d << 19) | (f << 16); > + > + address += pcie->cfg.start; > + > + hdrtype = readb(address + PCI_HEADER_TYPE); > + > + if (hdrtype == PCI_HEADER_TYPE_BRIDGE && > + offset >= PCI_PRIMARY_BUS && > + offset <= PCI_SUBORDINATE_BUS && > + value != pci_conv_32_to_size(~0UL, offset, size)) { Please run this through patman/checkpatch to fix errors like this. > + value += pci_conv_32_to_size(bus_offs, offset, size); > + } > + > + if (b == 1 && d > 0) > + return 0; > + > + switch (size) { > + case PCI_SIZE_8: > + writeb(value, address + offset); > + break; > + case PCI_SIZE_16: > + writew(value, address + offset); > + break; > + case PCI_SIZE_32: > + writel(value, address + offset); > + break; > + default: > + printf("Invalid size\n"); > + } > + debug("%02x.%02x.%02x: u%d %x (%lx) <- %lx\n", > + b, d, f, size, offset, address, value); > + return 0; > +} > + > +static int pci_octeontx2_pem_read_config(struct udevice *bus, pci_dev_t bdf, > + uint offset, ulong *valuep, > + enum pci_size_t size) > +{ > + struct octeontx_pci *pcie = (void *)dev_get_priv(bus); > + struct pci_controller *hose = dev_get_uclass_priv(bus); > + uintptr_t address; > + u32 b, d, f; > + > + b = PCI_BUS(bdf) + 1 - hose->first_busno; > + d = PCI_DEV(bdf); > + f = PCI_FUNC(bdf); > + > + address = (b << 20) | (d << 15) | (f << 12); > + > + debug("bdf %x %02x.%02x.%02x: u%d %x (%lx)\n", > + bdf, b, d, f, size, offset, address); > + address += pcie->cfg.start; > + > + debug("%02x.%02x.%02x: u%d %x (%lx) %lx\n", > + b, d, f, size, offset, address, *valuep); > + *valuep = pci_conv_32_to_size(~0UL, offset, size); > + > + if (b == 1 && d > 0) > + return 0; > + > + switch (size) { > + case PCI_SIZE_8: > + debug("byte %lx\n", address + offset); > + *valuep = readb(address + offset); > + break; > + case PCI_SIZE_16: > + debug("word %lx\n", address + offset); > + *valuep = readw(address + offset); > + break; > + case PCI_SIZE_32: > + debug("long %lx\n", address + offset); > + *valuep = readl(address + offset); > + break; Instead of repeating this switch () each time, perhaps create: static ulong readl_size(uintptr_t addr, enum pci_size_t size) { switch () ... } > + default: > + printf("Invalid size\n"); > + } > + > + debug("%02x.%02x.%02x: u%d %x (%lx) -> %lx\n", > + b, d, f, size, offset, address, *valuep); > + > + return 0; > +} > + > +static void pci_octeontx2_pem_workaround(struct udevice *bus, uint offset, > + enum pci_size_t size) > +{ > +#if defined(CONFIG_ARCH_OCTEONTX2) > + struct octeontx_pci *pcie = (void *)dev_get_priv(bus); > + u64 rval, wval; > + u32 cfg_off, data; > + u64 raddr, waddr; > + u8 shift; > + > + raddr = pcie->pem.start + PEM_CFG_RD; > + waddr = pcie->pem.start + PEM_CFG_WR; > + > + debug("%s raddr %llx waddr %llx\n", __func__, raddr, waddr); > + cfg_off = PCIERC_RASDP_DE_ME; > + wval = cfg_off; > + debug("%s DE_ME raddr %llx wval %llx\n", __func__, raddr, wval); > + writeq(wval, raddr); > + rval = readq(raddr); > + debug("%s DE_ME raddr %llx rval %llx\n", __func__, raddr, rval); > + data = rval >> 32; > + if (data & 0x1) { > + data = (data & (~0x1)); > + wval |= ((u64)data << 32); > + debug("%s DE_ME waddr %llx wval %llx\n", __func__, waddr, wval); > + writeq(wval, waddr); > + } > + > + cfg_off = PCIERC_RAS_EINJ_CTL6_CMPP0; > + wval = cfg_off; > + data = 0xFE000000; > + wval |= ((u64)data << 32); > + debug("%s CMPP0 waddr %llx wval %llx\n", __func__, waddr, wval); > + writeq(wval, waddr); > + > + cfg_off = PCIERC_RAS_EINJ_CTL6_CMPV0; > + wval = cfg_off; > + data = 0x44000000; > + wval |= ((u64)data << 32); > + debug("%s CMPV0 waddr %llx wval %llx\n", __func__, waddr, wval); > + writeq(wval, waddr); > + > + cfg_off = PCIERC_RAS_EINJ_CTL6_CHGP1; > + wval = cfg_off; > + data = 0xFF; > + wval |= ((u64)data << 32); > + debug("%s CHGP1 waddr %llx wval %llx\n", __func__, waddr, wval); > + writeq(wval, waddr); > + > + cfg_off = PCIERC_RAS_EINJ_EN; > + wval = cfg_off; > + data = 0x40; > + wval |= ((u64)data << 32); > + debug("%s EINJ_EN waddr %llx wval %llx\n", __func__, waddr, wval); > + writeq(wval, waddr); > + > + cfg_off = PCIERC_RAS_EINJ_CTL6PE; > + wval = cfg_off; > + data = 0x1; > + wval |= ((u64)data << 32); Does this work? wval |= 1 << 32ULL; > + debug("%s EINJ_CTL6PE waddr %llx wval %llx\n", __func__, waddr, wval); > + writeq(wval, waddr); > + > + switch (size) { > + case PCI_SIZE_8: > + shift = offset % 4; > + data = (0x1 << shift); > + break; > + case PCI_SIZE_16: > + shift = (offset % 4) ? 2 : 0; > + data = (0x3 << shift); > + break; > + default: > + case PCI_SIZE_32: > + data = 0xF; Please use lower-case hex everywhere. > + break; > + } > + > + cfg_off = PCIERC_RAS_EINJ_CTL6_CHGV1; > + wval = cfg_off; > + wval |= ((u64)data << 32); > + debug("%s EINJ_CHGV1 waddr %llx <= wval %llx\n", __func__, waddr, > wval); > + writeq(wval, waddr); > + > + cfg_off = PCIERC_RASDP_EP_CTL; > + wval = cfg_off; > + wval |= ((u64)0x1 << 32); > + debug("%s EP_CTL waddr %llx <= wval %llx\n", __func__, waddr, wval); > + writeq(wval, waddr); > + > + wval = readq(waddr); > + debug("%s EP_CTL waddr %llx => wval %llx\n", __func__, waddr, wval); > +#endif > +} > + > +static int pci_octeontx2_pem_write_config(struct udevice *bus, pci_dev_t bdf, > + uint offset, ulong value, > + enum pci_size_t size) > +{ > + struct octeontx_pci *pcie = (void *)dev_get_priv(bus); > + struct pci_controller *hose = dev_get_uclass_priv(bus); > + uintptr_t address, addr; > + u32 b, d, f; > + u32 data; > + int tmp; > + > + b = PCI_BUS(bdf) + 1 - hose->first_busno; > + d = PCI_DEV(bdf); > + f = PCI_FUNC(bdf); > + > + address = (b << 20) | (d << 15) | (f << 12); > + > + debug("bdf %x %02x.%02x.%02x: u%d %x (%lx)\n", > + bdf, b, d, f, size, offset, address); > + address += pcie->cfg.start; > + > + debug("%02x.%02x.%02x: u%d %x (%lx) %lx\n", > + b, d, f, size, offset, address, value); > + > + if (b == 1 && d > 0) > + return 0; > + > + addr = (address + offset) & ~0x3UL; > + switch (size) { > + case PCI_SIZE_8: > + tmp = (address + offset) & 0x3; > + size = PCI_SIZE_32; > + data = readl(addr); > + debug("tmp 8 long %lx %x\n", addr, data); > + tmp *= 8; > + value = (data & ~(0xFFUL << tmp)) | ((value & 0xFF) << tmp); > + break; > + case PCI_SIZE_16: > + tmp = (address + offset) & 0x3; > + size = PCI_SIZE_32; > + data = readl(addr); > + debug("tmp 16 long %lx %x\n", addr, data); > + tmp *= 8; > + value = (data & 0xFFFF) | (value << tmp); > + break; > + case PCI_SIZE_32: Could you add a comment about why this code is empty? > + break; > + } > + debug("tmp long %lx %lx\n", addr, value); > + > + pci_octeontx2_pem_workaround(bus, offset, size); > + > + switch (size) { > + case PCI_SIZE_8: > + debug("byte %lx %lx\n", address + offset, value); > + writeb(value, address + offset); > + break; > + case PCI_SIZE_16: > + debug("word %lx %lx\n", address + offset, value); > + writew(value, address + offset); > + break; > + case PCI_SIZE_32: > + debug("long %lx %lx\n", addr, value); > + writel(value, addr); > + break; > + default: > + printf("Invalid size\n"); > + } > + > + debug("%02x.%02x.%02x: u%d %x (%lx) <- %lx\n", > + b, d, f, size, offset, addr, value); > + > + return 0; > +} > + > +static int pci_octeontx_ofdata_to_platdata(struct udevice *dev) > +{ > + return 0; > +} > + > +static int pci_octeontx_ecam_probe(struct udevice *dev) > +{ > + struct octeontx_pci *pcie = (void *)dev_get_priv(dev); > + int err; > + > + err = fdt_get_resource(gd->fdt_blob, dev->node.of_offset, "reg", 0, > + &pcie->cfg); > + > + if (err) { > + printf("Error reading resource: %s\n", fdt_strerror(err)); debug() > + return err; > + } > + > +#if defined(CONFIG_ARCH_OCTEONTX2) > + err = fdt_node_check_compatible(gd->fdt_blob, dev->node.of_offset, > + "marvell,pci-host-octeontx2-pem"); > + if (!err) { > + err = fdt_get_resource(gd->fdt_blob, dev->node.of_offset, > + "reg", 1, &pcie->pem); > + > + if (err) { > + printf("Error reading resource: %s\n", > + fdt_strerror(err)); > + return err; > + } > + } > +#endif > + err = fdtdec_get_pci_bus_range(gd->fdt_blob, dev->node.of_offset, > + &pcie->bus); > + > + if (err) { > + printf("Error reading resource: %s\n", fdt_strerror(err)); > + return err; > + } > + > + return 0; > +} > + > +static const struct dm_pci_ops pci_octeontx_ecam_ops = { > + .read_config = pci_octeontx_ecam_read_config, > + .write_config = pci_octeontx_ecam_write_config, > +}; > + > +static const struct udevice_id pci_octeontx_ecam_ids[] = { > + { .compatible = "cavium,pci-host-thunder-ecam" }, > + { .compatible = "cavium,pci-host-octeontx-ecam" }, > + { .compatible = "pci-host-ecam-generic" }, > + { } > +}; > + > +static const struct dm_pci_ops pci_octeontx_pem_ops = { > + .read_config = pci_octeontx_pem_read_config, > + .write_config = pci_octeontx_pem_write_config, > +}; > + > +static const struct udevice_id pci_octeontx_pem_ids[] = { > + { .compatible = "cavium,pci-host-thunder-pem" }, > + { } > +}; > + > +static const struct dm_pci_ops pci_octeontx2_pem_ops = { > + .read_config = pci_octeontx2_pem_read_config, > + .write_config = pci_octeontx2_pem_write_config, > +}; > + > +static const struct udevice_id pci_octeontx2_pem_ids[] = { > + { .compatible = "marvell,pci-host-octeontx2-pem" }, > + { } > +}; > + > +U_BOOT_DRIVER(pci_octeontx_ecam) = { > + .name = "pci_octeontx_ecam", > + .id = UCLASS_PCI, > + .of_match = pci_octeontx_ecam_ids, > + .ops = &pci_octeontx_ecam_ops, > + .ofdata_to_platdata = pci_octeontx_ofdata_to_platdata, > + .probe = pci_octeontx_ecam_probe, > + .priv_auto_alloc_size = sizeof(struct octeontx_pci), > + .flags = DM_FLAG_PRE_RELOC, > +}; > + > +U_BOOT_DRIVER(pci_octeontx2_pcie) = { > + .name = "pci_octeontx2_pem", > + .id = UCLASS_PCI, > + .of_match = pci_octeontx2_pem_ids, > + .ops = &pci_octeontx2_pem_ops, > + .ofdata_to_platdata = pci_octeontx_ofdata_to_platdata, > + .probe = pci_octeontx_ecam_probe, > + .priv_auto_alloc_size = sizeof(struct octeontx_pci), > +}; > + > +U_BOOT_DRIVER(pci_octeontx_pcie) = { > + .name = "pci_octeontx_pem", > + .id = UCLASS_PCI, > + .of_match = pci_octeontx_pem_ids, > + .ops = &pci_octeontx_pem_ops, > + .ofdata_to_platdata = pci_octeontx_ofdata_to_platdata, > + .probe = pci_octeontx_ecam_probe, > + .priv_auto_alloc_size = sizeof(struct octeontx_pci), > +}; > -- > 2.23.0 > Regards, Simon _______________________________________________ U-Boot mailing list U-Boot@lists.denx.de https://lists.denx.de/listinfo/u-boot