[PATCH v2 2/4] PCI: designware: Add ARM64 support
Lorenzo Pieralisi
lorenzo.pieralisi at arm.com
Mon Jul 13 03:58:19 PDT 2015
On Tue, Jun 16, 2015 at 03:14:21PM +0100, Gabriele Paoloni wrote:
[...]
> > > + bus = pci_create_root_bus(pp->dev, pp->root_bus_nr, &dw_pcie_ops,
> > > + pp, &res);
I think you can't do that on ARM 32-bit platforms. They rely on pci_sys_data
in pcibios_msi_controller and pcibios_align_resource, if you do not
pass a pci_sys_data struct in the sysdata pointer this can't work and
unless you get lucky it will explode.
We have to remove pci_sys_data dependency on ARM, then this approach
will work.
Thanks,
Lorenzo
> > > + if (!bus)
> > > + return -ENOMEM;
> > > +
> > > +#ifdef CONFIG_GENERIC_MSI_IRQ_DOMAIN
> > > + bus->msi = container_of(&pp->irq_domain, struct msi_controller,
> > domain);
> > > +#else
> > > + bus->msi = &dw_pcie_msi_chip;
> > > #endif
> > >
> > > - dw_pci.nr_controllers = 1;
> > > - dw_pci.private_data = (void **)&pp;
> > > + pci_scan_child_bus(bus);
> > > + if (pp->ops->scan_bus)
> > > + pp->ops->scan_bus(pp);
> > >
> > > - pci_common_init_dev(pp->dev, &dw_pci);
> > > +#ifdef CONFIG_ARM
> > > + /* support old dtbs that incorrectly describe IRQs */
> > > + pci_fixup_irqs(pci_common_swizzle, of_irq_parse_and_map_pci);
> > > +#endif
> > > +
> > > + pci_assign_unassigned_bus_resources(bus);
> > > + pci_bus_add_devices(bus);
> > >
> > > return 0;
> > > }
> > > @@ -653,7 +665,7 @@ static int dw_pcie_valid_config(struct pcie_port
> > *pp,
> > > static int dw_pcie_rd_conf(struct pci_bus *bus, u32 devfn, int where,
> > > int size, u32 *val)
> > > {
> > > - struct pcie_port *pp = sys_to_pcie(bus->sysdata);
> > > + struct pcie_port *pp = bus->sysdata;
> > > int ret;
> > >
> > > if (dw_pcie_valid_config(pp, bus, PCI_SLOT(devfn)) == 0) {
> > > @@ -677,7 +689,7 @@ static int dw_pcie_rd_conf(struct pci_bus *bus,
> > u32 devfn, int where,
> > > static int dw_pcie_wr_conf(struct pci_bus *bus, u32 devfn,
> > > int where, int size, u32 val)
> > > {
> > > - struct pcie_port *pp = sys_to_pcie(bus->sysdata);
> > > + struct pcie_port *pp = bus->sysdata;
> > > int ret;
> > >
> > > if (dw_pcie_valid_config(pp, bus, PCI_SLOT(devfn)) == 0)
> > > @@ -701,64 +713,6 @@ static struct pci_ops dw_pcie_ops = {
> > > .write = dw_pcie_wr_conf,
> > > };
> > >
> > > -static int dw_pcie_setup(int nr, struct pci_sys_data *sys)
> > > -{
> > > - struct pcie_port *pp;
> > > -
> > > - pp = sys_to_pcie(sys);
> > > -
> > > - if (global_io_offset < SZ_1M && pp->io_size > 0) {
> > > - sys->io_offset = global_io_offset - pp->io_bus_addr;
> > > - pci_ioremap_io(global_io_offset, pp->io_base);
> > > - global_io_offset += SZ_64K;
> > > - pci_add_resource_offset(&sys->resources, &pp->io,
> > > - sys->io_offset);
> > > - }
> > > -
> > > - sys->mem_offset = pp->mem.start - pp->mem_bus_addr;
> > > - pci_add_resource_offset(&sys->resources, &pp->mem, sys-
> > >mem_offset);
> > > - pci_add_resource(&sys->resources, &pp->busn);
> > > -
> > > - return 1;
> > > -}
> > > -
> > > -static struct pci_bus *dw_pcie_scan_bus(int nr, struct pci_sys_data
> > *sys)
> > > -{
> > > - struct pci_bus *bus;
> > > - struct pcie_port *pp = sys_to_pcie(sys);
> > > -
> > > - pp->root_bus_nr = sys->busnr;
> > > - bus = pci_create_root_bus(pp->dev, sys->busnr,
> > > - &dw_pcie_ops, sys, &sys->resources);
> > > - if (!bus)
> > > - return NULL;
> > > -
> > > - pci_scan_child_bus(bus);
> > > -
> > > - if (bus && pp->ops->scan_bus)
> > > - pp->ops->scan_bus(pp);
> > > -
> > > - return bus;
> > > -}
> > > -
> > > -static int dw_pcie_map_irq(const struct pci_dev *dev, u8 slot, u8
> > pin)
> > > -{
> > > - struct pcie_port *pp = sys_to_pcie(dev->bus->sysdata);
> > > - int irq;
> > > -
> > > - irq = of_irq_parse_and_map_pci(dev, slot, pin);
> > > - if (!irq)
> > > - irq = pp->irq;
> > > -
> > > - return irq;
> > > -}
> > > -
> > > -static struct hw_pci dw_pci = {
> > > - .setup = dw_pcie_setup,
> > > - .scan = dw_pcie_scan_bus,
> > > - .map_irq = dw_pcie_map_irq,
> > > -};
> > > -
> > > void dw_pcie_setup_rc(struct pcie_port *pp)
> > > {
> > > u32 val;
> > > diff --git a/drivers/pci/host/pcie-designware.h
> > b/drivers/pci/host/pcie-designware.h
> > > index d0bbd27..ab78710 100644
> > > --- a/drivers/pci/host/pcie-designware.h
> > > +++ b/drivers/pci/host/pcie-designware.h
> > > @@ -34,7 +34,7 @@ struct pcie_port {
> > > u64 cfg1_mod_base;
> > > void __iomem *va_cfg1_base;
> > > u32 cfg1_size;
> > > - u64 io_base;
> > > + resource_size_t io_base;
> > > u64 io_mod_base;
> > > phys_addr_t io_bus_addr;
> > > u32 io_size;
> > > @@ -42,10 +42,10 @@ struct pcie_port {
> > > u64 mem_mod_base;
> > > phys_addr_t mem_bus_addr;
> > > u32 mem_size;
> > > - struct resource cfg;
> > > - struct resource io;
> > > - struct resource mem;
> > > - struct resource busn;
> > > + struct resource *cfg;
> > > + struct resource *io;
> > > + struct resource *mem;
> > > + struct resource *busn;
> > > int irq;
> > > u32 lanes;
> > > struct pcie_host_ops *ops;
> > > diff --git a/drivers/pci/host/pcie-spear13xx.c
> > b/drivers/pci/host/pcie-spear13xx.c
> > > index 020d788..e78ddf8 100644
> > > --- a/drivers/pci/host/pcie-spear13xx.c
> > > +++ b/drivers/pci/host/pcie-spear13xx.c
> > > @@ -287,7 +287,7 @@ static int spear13xx_add_pcie_port(struct
> > pcie_port *pp,
> > > return ret;
> > > }
> > >
> > > - pp->root_bus_nr = -1;
> > > + pp->root_bus_nr = 0;
> > > pp->ops = &spear13xx_pcie_host_ops;
> > >
> > > ret = dw_pcie_host_init(pp);
> > >
> > > --------------------------------------------------
> > > --------------------------------------------------
> > >
> > >> -----Original Message-----
> > >> From: Wangzhou (B)
> > >> Sent: Thursday, June 11, 2015 6:44 AM
> > >> To: Gabriele Paoloni
> > >> Cc: Lorenzo Pieralisi; James Morse; Bjorn Helgaas; Jingoo Han;
> > Pratyush
> > >> Anand; Arnd Bergmann; fabrice.gasnier at st.com; Liviu Dudau; linux-
> > >> pci at vger.kernel.org; linux-arm-kernel at lists.infradead.org;
> > >> devicetree at vger.kernel.org; Yuanzhichang; Zhudacai; zhangjukuo;
> > >> qiuzhenfa; Liguozhu (Kenneth)
> > >> Subject: Re: [PATCH v2 2/4] PCI: designware: Add ARM64 support
> > >>
> > >> On 2015/6/10 21:35, Gabriele Paoloni wrote:
> > >>> Hi Zhou Wang and all
> > >>>
> > >>> I have worked on a patch that unify ARM and ARM64. As you can see I
> > >> have removed the parser and now I use
> > >> "of_pci_get_host_bridge_resources" in order to retrieve the
> > resources
> > >> associated to each pci host bridge window.
> > >>> The resources now are not copied to the designware pcie_port pp,
> > >> instead they are passed by pointer.
> > >>>
> > >>> This patch is intended to replace entirely "[PATCH v2 2/4]"; so I
> > >> have also included all the changes about the other drivers
> > >>>
> > >>> Please find the patch inline below.
> > >>>
> > >>> Let me know if you think it is ok.
> > >>>
> > >>> Regards
> > >>>
> > >>> Gab
> > >>>
> > >>> ------------------------------------
> > >>> ------------------------------------
> > >>>
> > >>> This patch is a rework of
> > >>> "[PATCH v2 2/4] PCI: designware: Add ARM64 support" from Zhou Wang.
> > >>>
> > >>> The intention is to complete the unification between ARM and
> > >>> ARM64 now that the dependency on arch/arm/kernel/bios32.c is gone.
> > >>>
> > >>> I have also included the rework asked by James Morse:
> > >>> io_base declared as resource_size_t
> > >>>
> > >>> It compiles fine on ARCH arm and arm64.
> > >>
> > >> Hi Gabriele,
> > >>
> > >> I manually apply this patch on my series, as there are some format
> > >> errors
> > >> about this patch. And I think this patch is a experimental one, it
> > >> could not
> > >> be compiled successfully.
> > >>
> > >>>
> > >>> It needs to be tested.
> > >>>
> > >>> Signed-off-by: Gabriele Paoloni <gabriele.paoloni at huawei.com>
> > >>>
> > >>> [...]
> > >>>
> > >>> -int dw_pcie_host_init(struct pcie_port *pp)
> > >>> +int __init dw_pcie_host_init(struct pcie_port *pp)
> > >>> {
> > >>> struct device_node *np = pp->dev->of_node;
> > >>> struct platform_device *pdev = to_platform_device(pp->dev);
> > >>> - struct of_pci_range range;
> > >>> - struct of_pci_range_parser parser;
> > >>> + struct pci_bus *bus;
> > >>> struct resource *cfg_res;
> > >>> + LIST_HEAD(res);
> > >>> u32 val, na, ns;
> > >>> const __be32 *addrp;
> > >>> int i, index, ret;
> > >>> + int rlen;
> > >>> + struct pci_host_bridge_window *win;
> > >>
> > >> This struct is not in latest kernel, I think you should use struct
> > >> resource_entry.
> > >>
> > >> Thanks,
> > >> Zhou
> > >>
> > >>> + const __be32 *parser_range_end = of_get_property(np, "ranges",
> > >> &rlen);
> > >>> +
> > >>> + if (parser_range_end == NULL)
> > >>> + return -ENOENT;
> > >>> + parser_range_end += rlen/sizeof(__be32);
> > >>> +
> > >>>
> > >>> /* Find the address cell size and the number of cells in order to
> > >> get
> > >>> * the untranslated address.
> > >>> @@ -375,78 +375,67 @@ int dw_pcie_host_init(struct pcie_port *pp)
> > >>> dev_err(pp->dev, "missing *config* reg space\n");
> > >>> }
> > >>>
> > >>> - if (of_pci_range_parser_init(&parser, np)) {
> > >>> - dev_err(pp->dev, "missing ranges property\n");
> > >>> - return -EINVAL;
> > >>> - }
> > >>> + ret = of_pci_get_host_bridge_resources(np, 0, 0xff, &res, &pp-
> > >>> io_base);
> > >>> + if (ret)
> > >>> + return ret;
> > >>>
> > >>> /* Get the I/O and memory ranges from DT */
> > >>> - for_each_of_pci_range(&parser, &range) {
> > >>> - unsigned long restype = range.flags & IORESOURCE_TYPE_BITS;
> > >>> -
> > >>> - if (restype == IORESOURCE_IO) {
> > >>> - of_pci_range_to_resource(&range, np, &pp->io);
> > >>> - pp->io.name = "I/O";
> > >>> - pp->io.start = max_t(resource_size_t,
> > >>> - PCIBIOS_MIN_IO,
> > >>> - range.pci_addr + global_io_offset);
> > >>> - pp->io.end = min_t(resource_size_t,
> > >>> - IO_SPACE_LIMIT,
> > >>> - range.pci_addr + range.size
> > >>> - + global_io_offset - 1);
> > >>> - pp->io_size = resource_size(&pp->io);
> > >>> - pp->io_bus_addr = range.pci_addr;
> > >>> - pp->io_base = range.cpu_addr;
> > >>> -
> > >>> - /* Find the untranslated IO space address */
> > >>> - pp->io_mod_base = of_read_number(parser.range -
> > >>> - parser.np + na, ns);
> > >>> - }
> > >>> - if (restype == IORESOURCE_MEM) {
> > >>> - of_pci_range_to_resource(&range, np, &pp->mem);
> > >>> - pp->mem.name = "MEM";
> > >>> - pp->mem_size = resource_size(&pp->mem);
> > >>> - pp->mem_bus_addr = range.pci_addr;
> > >>> -
> > >>> - /* Find the untranslated MEM space address */
> > >>> - pp->mem_mod_base = of_read_number(parser.range -
> > >>> - parser.np + na, ns);
> > >>> - }
> > >>> - if (restype == 0) {
> > >>> - of_pci_range_to_resource(&range, np, &pp->cfg);
> > >>> - pp->cfg0_size = resource_size(&pp->cfg)/2;
> > >>> - pp->cfg1_size = resource_size(&pp->cfg)/2;
> > >>> - pp->cfg0_base = pp->cfg.start;
> > >>> - pp->cfg1_base = pp->cfg.start + pp->cfg0_size;
> > >>> + list_for_each_entry(win, &res, list) {
> > >>> + switch (resource_type(win->res)) {
> > >>> + case IORESOURCE_IO:
> > >>> + pp->io = win->res;
> > >>> + pp->io->name = "I/O";
> > >>> + pp->io_size = resource_size(pp->io);
> > >>> + pp->io_bus_addr = pp->io->start - win->offset;
> > >>> + /* magic 5 below comes from magic na and ns in
> > >>> + * of_pci_range_parser_init() */
> > >>> + pp->io_mod_base = of_read_number(parser_range_end -
> > >>> + of_n_addr_cells(np) - 5 + na, ns);
> > >>> + ret = pci_remap_iospace(pp->io, pp->io_base);
> > >>> + if (ret) {
> > >>> + dev_warn(pp->dev, "error %d: failed to map
> > >> resource %pR\n",
> > >>> + ret, pp->io);
> > >>> + continue;
> > >>> + }
> > >>> + break;
> > >>> + case IORESOURCE_MEM:
> > >>> + pp->mem = win->res;
> > >>> + pp->mem->name = "MEM";
> > >>> + pp->mem_size = resource_size(pp->mem);
> > >>> + pp->mem_bus_addr = pp->mem->start - win->offset;
> > >>> + pp->mem_mod_base = of_read_number(parser_range_end -
> > >>> + of_n_addr_cells(np) - 5 + na, ns);
> > >>> + break;
> > >>> + case 0:
> > >>> + pp->cfg = win->res;
> > >>> + pp->cfg0_size = resource_size(pp->cfg)/2;
> > >>> + pp->cfg1_size = resource_size(pp->cfg)/2;
> > >>> + pp->cfg0_base = pp->cfg->start;
> > >>> + pp->cfg1_base = pp->cfg->start + pp->cfg0_size;
> > >>>
> > >>> /* Find the untranslated configuration space address
> > >> */
> > >>> - pp->cfg0_mod_base = of_read_number(parser.range -
> > >>> - parser.np + na, ns);
> > >>> - pp->cfg1_mod_base = pp->cfg0_mod_base +
> > >>> - pp->cfg0_size;
> > >>> + pp->cfg0_mod_base = of_read_number(parser_range_end -
> > >>> + of_n_addr_cells(np) - 5 + na, ns);
> > >>> + pp->cfg1_mod_base = pp->cfg0_mod_base + pp->cfg0_size;
> > >>> + break;
> > >>> + case IORESOURCE_BUS:
> > >>> + pp->busn = win->res;
> > >>> + break;
> > >>> + default:
> > >>> + continue;
> > >>> }
> > >>> }
> > >>>
> > >>> - ret = of_pci_parse_bus_range(np, &pp->busn);
> > >>> - if (ret < 0) {
> > >>> - pp->busn.name = np->name;
> > >>> - pp->busn.start = 0;
> > >>> - pp->busn.end = 0xff;
> > >>> - pp->busn.flags = IORESOURCE_BUS;
> > >>> - dev_dbg(pp->dev, "failed to parse bus-range property: %d,
> > >> using default %pR\n",
> > >>> - ret, &pp->busn);
> > >>> - }
> > >>> -
> > >>> if (!pp->dbi_base) {
> > >>> - pp->dbi_base = devm_ioremap(pp->dev, pp->cfg.start,
> > >>> - resource_size(&pp->cfg));
> > >>> + pp->dbi_base = devm_ioremap(pp->dev, pp->cfg->start,
> > >>> + resource_size(pp->cfg));
> > >>> if (!pp->dbi_base) {
> > >>> dev_err(pp->dev, "error with ioremap\n");
> > >>> return -ENOMEM;
> > >>> }
> > >>> }
> > >>>
> > >>> - pp->mem_base = pp->mem.start;
> > >>> + pp->mem_base = pp->mem->start;
> > >>>
> > >>> if (!pp->va_cfg0_base) {
> > >>> pp->va_cfg0_base = devm_ioremap(pp->dev, pp->cfg0_base, @@
> > >> -493,24 +482,47 @@ int dw_pcie_host_init(struct pcie_port *pp)
> > >>> if (pp->ops->host_init)
> > >>> pp->ops->host_init(pp);
> > >>>
> > >>> - dw_pcie_wr_own_conf(pp, PCI_BASE_ADDRESS_0, 4, 0);
> > >>> + if (dw_pcie_wr_own_conf(pp, PCI_BASE_ADDRESS_0, 4, 0)
> > >>> + != PCIBIOS_SUCCESSFUL)
> > >>> + return -EINVAL;
> > >>>
> > >>> /* program correct class for RC */
> > >>> - dw_pcie_wr_own_conf(pp, PCI_CLASS_DEVICE, 2,
> > >> PCI_CLASS_BRIDGE_PCI);
> > >>> + if (dw_pcie_wr_own_conf(pp, PCI_CLASS_DEVICE, 2,
> > >> PCI_CLASS_BRIDGE_PCI)
> > >>> + != PCIBIOS_SUCCESSFUL)
> > >>> + return -EINVAL;
> > >>> +
> > >>> + if (dw_pcie_rd_own_conf(pp, PCIE_LINK_WIDTH_SPEED_CONTROL, 4,
> > >> &val)
> > >>> + != PCIBIOS_SUCCESSFUL)
> > >>> + return -EINVAL;
> > >>>
> > >>> - dw_pcie_rd_own_conf(pp, PCIE_LINK_WIDTH_SPEED_CONTROL, 4, &val);
> > >>> val |= PORT_LOGIC_SPEED_CHANGE;
> > >>> - dw_pcie_wr_own_conf(pp, PCIE_LINK_WIDTH_SPEED_CONTROL, 4, val);
> > >>>
> > >>> -#ifdef CONFIG_PCI_MSI
> > >>> - dw_pcie_msi_chip.dev = pp->dev;
> > >>> - dw_pci.msi_ctrl = &dw_pcie_msi_chip;
> > >>> + if (dw_pcie_wr_own_conf(pp, PCIE_LINK_WIDTH_SPEED_CONTROL, 4, val)
> > >>> + != PCIBIOS_SUCCESSFUL)
> > >>> + return -EINVAL;
> > >>> +
> > >>> + bus = pci_create_root_bus(pp->dev, pp->root_bus_nr, &dw_pcie_ops,
> > >>> + pp, &res);
> > >>> + if (!bus)
> > >>> + return -ENOMEM;
> > >>> +
> > >>> +#ifdef CONFIG_GENERIC_MSI_IRQ_DOMAIN
> > >>> + bus->msi = container_of(&pp->irq_domain, struct msi_controller,
> > >>> +domain); #else
> > >>> + bus->msi = &dw_pcie_msi_chip;
> > >>> #endif
> > >>>
> > >>> - dw_pci.nr_controllers = 1;
> > >>> - dw_pci.private_data = (void **)&pp;
> > >>> + pci_scan_child_bus(bus);
> > >>> + if (pp->ops->scan_bus)
> > >>> + pp->ops->scan_bus(pp);
> > >>>
> > >>> - pci_common_init_dev(pp->dev, &dw_pci);
> > >>> +#ifdef CONFIG_ARM
> > >>> + /* support old dtbs that incorrectly describe IRQs */
> > >>> + pci_fixup_irqs(pci_common_swizzle, of_irq_parse_and_map_pci);
> > >> #endif
> > >>> +
> > >>> + pci_assign_unassigned_bus_resources(bus);
> > >>> + pci_bus_add_devices(bus);
> > >>>
> > >>> return 0;
> > >>> }
> > >>> @@ -653,7 +665,7 @@ static int dw_pcie_valid_config(struct
> > pcie_port
> > >> *pp, static int dw_pcie_rd_conf(struct pci_bus *bus, u32 devfn, int
> > >> where,
> > >>> int size, u32 *val)
> > >>> {
> > >>> - struct pcie_port *pp = sys_to_pcie(bus->sysdata);
> > >>> + struct pcie_port *pp = bus->sysdata;
> > >>> int ret;
> > >>>
> > >>> if (dw_pcie_valid_config(pp, bus, PCI_SLOT(devfn)) == 0) { @@ -
> > >> 677,7 +689,7 @@ static int dw_pcie_rd_conf(struct pci_bus *bus, u32
> > >> devfn, int where, static int dw_pcie_wr_conf(struct pci_bus *bus,
> > u32
> > >> devfn,
> > >>> int where, int size, u32 val)
> > >>> {
> > >>> - struct pcie_port *pp = sys_to_pcie(bus->sysdata);
> > >>> + struct pcie_port *pp = bus->sysdata;
> > >>> int ret;
> > >>>
> > >>> if (dw_pcie_valid_config(pp, bus, PCI_SLOT(devfn)) == 0) @@ -
> > >> 701,64 +713,6 @@ static struct pci_ops dw_pcie_ops = {
> > >>> .write = dw_pcie_wr_conf,
> > >>> };
> > >>>
> > >>> -static int dw_pcie_setup(int nr, struct pci_sys_data *sys) -{
> > >>> - struct pcie_port *pp;
> > >>> -
> > >>> - pp = sys_to_pcie(sys);
> > >>> -
> > >>> - if (global_io_offset < SZ_1M && pp->io_size > 0) {
> > >>> - sys->io_offset = global_io_offset - pp->io_bus_addr;
> > >>> - pci_ioremap_io(global_io_offset, pp->io_base);
> > >>> - global_io_offset += SZ_64K;
> > >>> - pci_add_resource_offset(&sys->resources, &pp->io,
> > >>> - sys->io_offset);
> > >>> - }
> > >>> -
> > >>> - sys->mem_offset = pp->mem.start - pp->mem_bus_addr;
> > >>> - pci_add_resource_offset(&sys->resources, &pp->mem, sys-
> > >>> mem_offset);
> > >>> - pci_add_resource(&sys->resources, &pp->busn);
> > >>> -
> > >>> - return 1;
> > >>> -}
> > >>> -
> > >>> -static struct pci_bus *dw_pcie_scan_bus(int nr, struct
> > pci_sys_data
> > >> *sys) -{
> > >>> - struct pci_bus *bus;
> > >>> - struct pcie_port *pp = sys_to_pcie(sys);
> > >>> -
> > >>> - pp->root_bus_nr = sys->busnr;
> > >>> - bus = pci_create_root_bus(pp->dev, sys->busnr,
> > >>> - &dw_pcie_ops, sys, &sys->resources);
> > >>> - if (!bus)
> > >>> - return NULL;
> > >>> -
> > >>> - pci_scan_child_bus(bus);
> > >>> -
> > >>> - if (bus && pp->ops->scan_bus)
> > >>> - pp->ops->scan_bus(pp);
> > >>> -
> > >>> - return bus;
> > >>> -}
> > >>> -
> > >>> -static int dw_pcie_map_irq(const struct pci_dev *dev, u8 slot, u8
> > >> pin) -{
> > >>> - struct pcie_port *pp = sys_to_pcie(dev->bus->sysdata);
> > >>> - int irq;
> > >>> -
> > >>> - irq = of_irq_parse_and_map_pci(dev, slot, pin);
> > >>> - if (!irq)
> > >>> - irq = pp->irq;
> > >>> -
> > >>> - return irq;
> > >>> -}
> > >>> -
> > >>> -static struct hw_pci dw_pci = {
> > >>> - .setup = dw_pcie_setup,
> > >>> - .scan = dw_pcie_scan_bus,
> > >>> - .map_irq = dw_pcie_map_irq,
> > >>> -};
> > >>> -
> > >>> void dw_pcie_setup_rc(struct pcie_port *pp) {
> > >>> u32 val;
> > >>> diff --git a/drivers/pci/host/pcie-designware.h
> > >> b/drivers/pci/host/pcie-designware.h
> > >>> index d0bbd27..ab78710 100644
> > >>> --- a/drivers/pci/host/pcie-designware.h
> > >>> +++ b/drivers/pci/host/pcie-designware.h
> > >>> @@ -34,7 +34,7 @@ struct pcie_port {
> > >>> u64 cfg1_mod_base;
> > >>> void __iomem *va_cfg1_base;
> > >>> u32 cfg1_size;
> > >>> - u64 io_base;
> > >>> + resource_size_t io_base;
> > >>> u64 io_mod_base;
> > >>> phys_addr_t io_bus_addr;
> > >>> u32 io_size;
> > >>> @@ -42,10 +42,10 @@ struct pcie_port {
> > >>> u64 mem_mod_base;
> > >>> phys_addr_t mem_bus_addr;
> > >>> u32 mem_size;
> > >>> - struct resource cfg;
> > >>> - struct resource io;
> > >>> - struct resource mem;
> > >>> - struct resource busn;
> > >>> + struct resource *cfg;
> > >>> + struct resource *io;
> > >>> + struct resource *mem;
> > >>> + struct resource *busn;
> > >>> int irq;
> > >>> u32 lanes;
> > >>> struct pcie_host_ops *ops;
> > >>> diff --git a/drivers/pci/host/pcie-spear13xx.c
> > >> b/drivers/pci/host/pcie-spear13xx.c
> > >>> index 020d788..e78ddf8 100644
> > >>> --- a/drivers/pci/host/pcie-spear13xx.c
> > >>> +++ b/drivers/pci/host/pcie-spear13xx.c
> > >>> @@ -287,7 +287,7 @@ static int spear13xx_add_pcie_port(struct
> > >> pcie_port *pp,
> > >>> return ret;
> > >>> }
> > >>>
> > >>> - pp->root_bus_nr = -1;
> > >>> + pp->root_bus_nr = 0;
> > >>> pp->ops = &spear13xx_pcie_host_ops;
> > >>>
> > >>> ret = dw_pcie_host_init(pp);
> > >>> ------------------------------------
> > >>> ------------------------------------
> > >>>
> > >>>
> > >>>
> > >>>
> > >>>> -----Original Message-----
> > >>>> From: Lorenzo Pieralisi [mailto:lorenzo.pieralisi at arm.com]
> > >>>> Sent: Tuesday, June 09, 2015 12:15 PM
> > >>>> To: Wangzhou (B)
> > >>>> Cc: James Morse; Bjorn Helgaas; Jingoo Han; Pratyush Anand; Arnd
> > >>>> Bergmann; fabrice.gasnier at st.com; Liviu Dudau; linux-
> > >>>> pci at vger.kernel.org; linux-arm-kernel at lists.infradead.org;
> > >>>> devicetree at vger.kernel.org; Gabriele Paoloni; Yuanzhichang;
> > Zhudacai;
> > >>>> zhangjukuo; qiuzhenfa; Liguozhu (Kenneth)
> > >>>> Subject: Re: [PATCH v2 2/4] PCI: designware: Add ARM64 support
> > >>>>
> > >>>> On Fri, Jun 05, 2015 at 09:11:30AM +0100, Zhou Wang wrote:
> > >>>>
> > >>>> [...]
> > >>>>
> > >>>>>>> -int dw_pcie_host_init(struct pcie_port *pp)
> > >>>>>>> +int __init dw_pcie_host_init(struct pcie_port *pp)
> > >>>>>>> {
> > >>>>>>> struct device_node *np = pp->dev->of_node;
> > >>>>>>> struct platform_device *pdev = to_platform_device(pp->dev);
> > >>>>>>> struct of_pci_range range;
> > >>>>>>> struct of_pci_range_parser parser;
> > >>>>>>> + struct pci_bus *bus;
> > >>>>>>> struct resource *cfg_res;
> > >>>>>>> + LIST_HEAD(res);
> > >>>>>>> u32 val, na, ns;
> > >>>>>>> const __be32 *addrp;
> > >>>>>>> int i, index, ret;
> > >>>>>>> @@ -502,15 +498,49 @@ int dw_pcie_host_init(struct pcie_port
> > *pp)
> > >>>>>>> val |= PORT_LOGIC_SPEED_CHANGE;
> > >>>>>>> dw_pcie_wr_own_conf(pp, PCIE_LINK_WIDTH_SPEED_CONTROL, 4,
> > >>>> val);
> > >>>>>>>
> > >>>>>>> -#ifdef CONFIG_PCI_MSI
> > >>>>>>> - dw_pcie_msi_chip.dev = pp->dev;
> > >>>>>>> - dw_pci.msi_ctrl = &dw_pcie_msi_chip;
> > >>>>>>> +#ifdef CONFIG_ARM
> > >>>>>>> + /*
> > >>>>>>> + * FIXME: we should really be able to use
> > >>>>>>> + * of_pci_get_host_bridge_resources on arm32 as well,
> > >>>>>>> + * but the conversion needs some more testing
> > >>>>>>> + */
> > >>>>>>> + if (global_io_offset < SZ_1M && pp->io_size > 0) {
> > >>>>>>> + pci_ioremap_io(global_io_offset, pp->io_base);
> > >>>>>>> + global_io_offset += SZ_64K;
> > >>>>>>> + pci_add_resource_offset(&res, &pp->io,
> > >>>>>>> + global_io_offset - pp->io_bus_addr);
> > >>>>>>> + }
> > >>>>>>> + pci_add_resource_offset(&res, &pp->mem,
> > >>>>>>> + pp->mem.start - pp->mem_bus_addr);
> > >>>>>>> + pci_add_resource(&res, &pp->busn);
> > >>>>>>
> > >>>>>> I don't think this #ifdef is necessary. In the spirit of 'the
> > >>>>>> conversion needs some more testing', I removed it leaving just
> > the
> > >>>> below arm64 code.
> > >>>>>>
> > >>>>>> This worked on my Freescale i.MX6 Quad SABRE Lite Board, I went
> > as
> > >>>>>> far as scanning for wireless access points.
> > >>>>>>
> > >>>>>
> > >>>>> I think it depends on which kind of PCIe device you use, if we
> > use
> > >> a
> > >>>>> PCIe device with a I/O Bar, it may not work well without above
> > code.
> > >>>>> But so far, I have not met a PCIe device which must work with a
> > I/O
> > >>>> Bar.
> > >>>>
> > >>>> There are two problems here:
> > >>>>
> > >>>> 1) the io_base address you get from
> > of_pci_get_host_bridge_resources
> > >>>> must be mapped using pci_remap_iospace. You are not doing this,
> > >> so
> > >>>> even if you had a PCIe card with I/O bar it would not work on
> > >> arm64
> > >>>> as the code stands. Remember that I/O space on arm/arm64 works
> > as
> > >> a
> > >>>> memory access with offset from PCI_IOBASE and that's the value
> > >> you
> > >>>> get
> > >>>> in the I/O resource. See:
> > >>>>
> > >>>> drivers/pci/host/pci-host-generic.c
> > >>>> drivers/pci/host/pci-versatile.c
> > >>>> drivers/pci/host/pci-xgene.c
> > >>>>
> > >>>> 2) (1) above would sort out I/O space access for both arm and
> > arm64
> > >> so,
> > >>>> as James said, the #ifdef is useless.
> > >>>>
> > >>>> I hope this makes things clearer.
> > >>>>
> > >>>> Thanks,
> > >>>> Lorenzo
> > >>>>
> > >>>>>
> > >>>>>>> +#else
> > >>>>>>> + ret = of_pci_get_host_bridge_resources(np, 0, 0xff, &res,
> > >>>>>>> +&pp->io_base);
> > >>>>>>
> > >>>>>> of_pci_get_host_bridge_resources expects &pp->io_base to be a
> > >>>>>> resource_size_t*, but &io_base is u64*. This generates a warning
> > >> on
> > >>>>>> arm with the above change. Changing the the type in
> > >>>>>> drivers/pci/host/pcie-designware.h fixes this.
> > >>>>>>
> > >>>>>>
> > >>>>>> Thanks,
> > >>>>>>
> > >>>>>> James
> > >>>>>>
> > >>>>>
> > >>>>> OK, will modify this in next version.
> > >>>>>
> > >>>>> Best Regards and thanks again for your test, Zhou
> > >>>>>
> > >>>>>>> + if (ret)
> > >>>>>>> + return ret;
> > >>>>>>> +#endif
> > >>>>>>> +
> > >>>>>>> + bus = pci_create_root_bus(pp->dev, pp->root_bus_nr,
> > >>>> &dw_pcie_ops,
> > >>>>>>> + pp, &res);
> > >>>>>>> + if (!bus)
> > >>>>>>> + return -ENOMEM;
> > >>>>>>> +
> > >>>>>>> +#ifdef CONFIG_GENERIC_MSI_IRQ_DOMAIN
> > >>>>>>> + bus->msi = container_of(&pp->irq_domain, struct
> > >>>> msi_controller,
> > >>>>>>> +domain); #else
> > >>>>>>> + bus->msi = &dw_pcie_msi_chip;
> > >>>>>>> #endif
> > >>>>>>>
> > >>>>>>> - dw_pci.nr_controllers = 1;
> > >>>>>>> - dw_pci.private_data = (void **)&pp;
> > >>>>>>> + pci_scan_child_bus(bus);
> > >>>>>>> + if (pp->ops->scan_bus)
> > >>>>>>> + pp->ops->scan_bus(pp);
> > >>>>>>>
> > >>>>>>> - pci_common_init_dev(pp->dev, &dw_pci);
> > >>>>>>> +#ifdef CONFIG_ARM
> > >>>>>>> + /* support old dtbs that incorrectly describe IRQs */
> > >>>>>>> + pci_fixup_irqs(pci_common_swizzle,
> > >>>> of_irq_parse_and_map_pci);
> > >>>>>>> +#endif
> > >>>>>>> +
> > >>>>>>> + pci_assign_unassigned_bus_resources(bus);
> > >>>>>>> + pci_bus_add_devices(bus);
> > >>>>>>>
> > >>>>>>> return 0;
> > >>>>>>> }
> > >>>>>>> @@ -653,7 +683,7 @@ static int dw_pcie_valid_config(struct
> > >>>>>>> pcie_port *pp, static int dw_pcie_rd_conf(struct pci_bus *bus,
> > >>>> u32 devfn, int where,
> > >>>>>>> int size, u32 *val)
> > >>>>>>> {
> > >>>>>>> - struct pcie_port *pp = sys_to_pcie(bus->sysdata);
> > >>>>>>> + struct pcie_port *pp = bus->sysdata;
> > >>>>>>> int ret;
> > >>>>>>>
> > >>>>>>> if (dw_pcie_valid_config(pp, bus, PCI_SLOT(devfn)) == 0)
> > >>>> { @@
> > >>>>>>> -677,7 +707,7 @@ static int dw_pcie_rd_conf(struct pci_bus *bus,
> > >>>>>>> u32 devfn, int where, static int dw_pcie_wr_conf(struct
> > pci_bus
> > >>>> *bus, u32 devfn,
> > >>>>>>> int where, int size, u32 val)
> > >>>>>>> {
> > >>>>>>> - struct pcie_port *pp = sys_to_pcie(bus->sysdata);
> > >>>>>>> + struct pcie_port *pp = bus->sysdata;
> > >>>>>>> int ret;
> > >>>>>>>
> > >>>>>>> if (dw_pcie_valid_config(pp, bus, PCI_SLOT(devfn)) == 0) @@
> > >>>>>>> -701,64 +731,6 @@ static struct pci_ops dw_pcie_ops = {
> > >>>>>>> .write = dw_pcie_wr_conf,
> > >>>>>>> };
> > >>>>>>>
> > >>>>>>> -static int dw_pcie_setup(int nr, struct pci_sys_data *sys) -{
> > >>>>>>> - struct pcie_port *pp;
> > >>>>>>> -
> > >>>>>>> - pp = sys_to_pcie(sys);
> > >>>>>>> -
> > >>>>>>> - if (global_io_offset < SZ_1M && pp->io_size > 0) {
> > >>>>>>> - sys->io_offset = global_io_offset - pp->io_bus_addr;
> > >>>>>>> - pci_ioremap_io(global_io_offset, pp->io_base);
> > >>>>>>> - global_io_offset += SZ_64K;
> > >>>>>>> - pci_add_resource_offset(&sys->resources, &pp->io,
> > >>>>>>> - sys->io_offset);
> > >>>>>>> - }
> > >>>>>>> -
> > >>>>>>> - sys->mem_offset = pp->mem.start - pp->mem_bus_addr;
> > >>>>>>> - pci_add_resource_offset(&sys->resources, &pp->mem, sys-
> > >>>>> mem_offset);
> > >>>>>>> - pci_add_resource(&sys->resources, &pp->busn);
> > >>>>>>> -
> > >>>>>>> - return 1;
> > >>>>>>> -}
> > >>>>>>> -
> > >>>>>>> -static struct pci_bus *dw_pcie_scan_bus(int nr, struct
> > >>>>>>> pci_sys_data *sys) -{
> > >>>>>>> - struct pci_bus *bus;
> > >>>>>>> - struct pcie_port *pp = sys_to_pcie(sys);
> > >>>>>>> -
> > >>>>>>> - pp->root_bus_nr = sys->busnr;
> > >>>>>>> - bus = pci_create_root_bus(pp->dev, sys->busnr,
> > >>>>>>> - &dw_pcie_ops, sys, &sys->resources);
> > >>>>>>> - if (!bus)
> > >>>>>>> - return NULL;
> > >>>>>>> -
> > >>>>>>> - pci_scan_child_bus(bus);
> > >>>>>>> -
> > >>>>>>> - if (bus && pp->ops->scan_bus)
> > >>>>>>> - pp->ops->scan_bus(pp);
> > >>>>>>> -
> > >>>>>>> - return bus;
> > >>>>>>> -}
> > >>>>>>> -
> > >>>>>>> -static int dw_pcie_map_irq(const struct pci_dev *dev, u8 slot,
> > >> u8
> > >>>>>>> pin) -{
> > >>>>>>> - struct pcie_port *pp = sys_to_pcie(dev->bus->sysdata);
> > >>>>>>> - int irq;
> > >>>>>>> -
> > >>>>>>> - irq = of_irq_parse_and_map_pci(dev, slot, pin);
> > >>>>>>> - if (!irq)
> > >>>>>>> - irq = pp->irq;
> > >>>>>>> -
> > >>>>>>> - return irq;
> > >>>>>>> -}
> > >>>>>>> -
> > >>>>>>> -static struct hw_pci dw_pci = {
> > >>>>>>> - .setup = dw_pcie_setup,
> > >>>>>>> - .scan = dw_pcie_scan_bus,
> > >>>>>>> - .map_irq = dw_pcie_map_irq,
> > >>>>>>> -};
> > >>>>>>> -
> > >>>>>>> void dw_pcie_setup_rc(struct pcie_port *pp) {
> > >>>>>>> u32 val;
> > >>>>>>
> > >>>>>>
> > >>>>>> .
> > >>>>>>
> > >>>>>
> > >>>>>
> > >>>>> --
> > >>>>> To unsubscribe from this list: send the line "unsubscribe linux-
> > >> pci"
> > >>>>> in the body of a message to majordomo at vger.kernel.org More
> > >> majordomo
> > >>>>> info at http://vger.kernel.org/majordomo-info.html
> > >>>>>
> > >>>
> > >>> .
> > >>>
> > >>
> > >
> > >
> > > .
> > >
> >
>
More information about the linux-arm-kernel
mailing list