[PATCH v2 2/4] PCI: designware: Add ARM64 support
Zhou Wang
wangzhou1 at hisilicon.com
Mon Jul 13 04:45:52 PDT 2015
On 2015/7/13 18:58, Lorenzo Pieralisi wrote:
> 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
>
Hi Lorenzo,
For pcibios_align_resource, 1/4 can be used to solve this problem.
For pcibios_msi_controller, bus->msi can be set in pcie-designware, then
we can get msi controller by dev->bus->msi in pci_msi_controller.
And the latest series is v3 version
"[PATCH v3 0/5] PCI: hisi: Add PCIe host support for Hisilicon Soc Hip05"
Best regards,
Zhou
>>>> + 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