On Sat, Aug 25, 2007 at 01:29:47PM +0400, Vitaly Bordug wrote: > > We are having 2 different instances of pci_process_bridge_OF_ranges(), > which makes describing 64-bit physical addresses in non PPC64 case > impossible. > > This approach inherits pci space parsing, but has a new way to behave > equally good in both 32bit and 64bit environments. Currently validated > with 440EPx (sequoia) and mpc8349-eitx. > > Signed-off-by: Vitaly Bordug <[EMAIL PROTECTED]> > Signed-off-by: Stefan Roese <[EMAIL PROTECTED]>
I like the idea, but I don't think this implementation is adequate yet. > diff --git a/arch/powerpc/kernel/pci-common.c > b/arch/powerpc/kernel/pci-common.c > index 083cfbd..57cd039 100644 > --- a/arch/powerpc/kernel/pci-common.c > +++ b/arch/powerpc/kernel/pci-common.c > @@ -478,3 +478,162 @@ void pci_resource_to_user(const struct pci_dev *dev, > int bar, > *start = rsrc->start - offset; > *end = rsrc->end - offset; > } > + > +void __devinit pci_process_bridge_OF_ranges(struct pci_controller *hose, > + struct device_node *dev, int prim) > +{ > + static unsigned int static_lc_ranges[256]; > + const unsigned int *ranges; > + unsigned int *lc_ranges; > + unsigned int pci_space; > + unsigned long size = 0; size can be 64-bit on 32-bit systems, at least in theory. > + int rlen = 0; > + int orig_rlen, ranges_amnt, i; > + int memno = 0; > + struct resource *res; > + int np, na = of_n_addr_cells(dev); > + struct ranges_pci64_sz64 *ranges64 = NULL; > + struct ranges_pci32_sz64 *ranges32 = NULL; > + phys_addr_t pci_addr, This is wrong: the PCI binding defines the PCI addresses to be 64-bit, even if the CPU has 32-bit physical addresses. cpu_phys_addr; > + > + np = na + 5; > + > + /* From "PCI Binding to 1275" z> + * The ranges property is laid out as an array of elements, > + * each of which comprises: > + * cells 0 - 2: a PCI address > + * cells 3 or 3+4: a CPU physical address > + * (size depending on dev->n_addr_cells) > + * cells 4+5 or 5+6: the size of the range > + */ > + ranges = of_get_property(dev, "ranges", &rlen); > + if (ranges == NULL) > + return; if (!ranges) would be the usual idiom here. > + /* Sanity check, though hopefully that never happens */ > + if (rlen > sizeof(static_lc_ranges)) { > + printk(KERN_WARNING "OF ranges property too large !\n"); > + rlen = sizeof(static_lc_ranges); > + } > + > + /* Let's work on a copy of the "ranges" property instead > + * of damaging the device-tree image in memory > + */ > + lc_ranges = static_lc_ranges; > + memcpy(lc_ranges, ranges, rlen); > + orig_rlen = rlen; > + > + ranges = lc_ranges; You don't ever actually touch the ranges property in place, so there's no need for this copy stuff. > + /* Map ranges to struct according to spec. */ > + if (na >= 2) { > + ranges64 = (void *)ranges; > + ranges_amnt = rlen / sizeof(*ranges64); > + } else { > + ranges32 = (void *)ranges; > + ranges_amnt = rlen / sizeof(*ranges32); > + } > + > + hose->io_base_phys = 0; > + for (i = 0; i < ranges_amnt; i++) { > + res = NULL; > + if (ranges64) { > + if (ranges64[i].pci_space == 0) > + continue; > + > + pci_space = ranges64[i].pci_space; > + pci_addr = > + (u64) ranges64[i].pci_addr[0] << 32 | ranges64[i]. > + pci_addr[1]; Why not just define the pci_addr field in your structures as u64? You would have to define the structure with attribute((packed)), I guess. > + cpu_phys_addr = > + of_translate_address(dev, ranges64[i].phys_addr); > + /* > + * I haven't observed 2 significant size cells in kernel > + * code, so we're accounting only LSB of size part > + * from ranges. -vitb > + */ > + size = ranges64[i].size[1]; > +#ifdef CONFIG_PPC64 > + if (ranges64[i].size[0]) > + size |= ranges64[i].size[0]<<32; > +#endif > + DBG("Observed: pci %llx phys %llx size %x\n", pci_addr, > + cpu_phys_addr, size); > + } else { > + if (ranges32[i].pci_space == 0) > + continue; > + > + pci_space = (unsigned int)ranges32[i].pci_space; > + pci_addr = (unsigned int)ranges32[i].pci_addr[1]; > + cpu_phys_addr = (unsigned int)ranges32[i].phys_addr[0]; We should really be using of_translate_address() in all cases - that's what it's for, after all. > + size = ranges32[i].size[1]; > + > + DBG("Observed: pci %x phys %x size %x\n", > + (u32) pci_addr, (u32) cpu_phys_addr, size); > + } You don't have any equivalent of the code that exists in both pre-existing versions to coalesce contiguous ranges. You probably want to use the 64-bit version, since that doesn't need a local copy of the ranges. > + > + switch ((pci_space >> 24) & 0x3) { > + case 1: /* I/O space */ > +#ifdef CONFIG_PPC32 > + /* > + * check from ppc32 pci implementation. > + * not sure if it is needed. -vitb > + */ > + if (pci_addr != 0) > + break; > +#endif > + /* limit I/O space to 16MB */ > + if (size > 0x01000000) > + size = 0x01000000; > + > + hose->io_base_phys = cpu_phys_addr - pci_addr; > + /* handle from 0 to top of I/O window */ > +#ifdef CONFIG_PPC64 > + hose->pci_io_size = pci_addr + size; > +#endif > + hose->io_base_virt = ioremap(hose->io_base_phys, size); > + > + if (prim) > + isa_io_base = (unsigned long)hose->io_base_virt; The old 64-bit versions don't presently ioremap() or set isa_io_base. I'd be worried about changing this semantic, at least without a rather more widespread consolidation of the 32/64 bit PCI code. > + > + res = &hose->io_resource; > + res->flags = IORESOURCE_IO; > + res->start = pci_addr; > + DBG("phb%d: IO 0x%llx -> 0x%llx\n", hose->global_number, > + (u64) res->start, (u64) (res->start + size - 1)); > + DBG("IO phys %llx IO virt %p\n", > + (u64) hose->io_base_phys, hose->io_base_virt); > + break; > + case 2: /* memory space */ > + memno = 0; > +#ifdef CONFIG_PPC32 > + if ((pci_addr == 0) && (size <= (16 << 20))) { > + /* 1st 16MB, i.e. ISA memory area */ > + if (prim) > + isa_mem_base = cpu_phys_addr; > + memno = 1; > + } > +#endif > + while (memno < 3 && hose->mem_resources[memno].flags) > + ++memno; > + > + if (memno == 0) > + hose->pci_mem_offset = cpu_phys_addr - pci_addr; > + if (memno < 3) { > + res = &hose->mem_resources[memno]; > + res->flags = IORESOURCE_MEM; > + res->start = cpu_phys_addr; > + DBG("phb%d: MEM 0x%llx -> 0x%llx\n", > + hose->global_number, res->start, > + res->start + size - 1); > + } > + break; > + } > + if (res != NULL) { > + res->name = dev->full_name; > + res->end = res->start + size - 1; > + res->parent = NULL; > + res->sibling = NULL; > + res->child = NULL; > + } > + } > +} > + > diff --git a/arch/powerpc/kernel/pci_32.c b/arch/powerpc/kernel/pci_32.c > index 0e2bee4..dc519e1 100644 > --- a/arch/powerpc/kernel/pci_32.c > +++ b/arch/powerpc/kernel/pci_32.c > @@ -843,120 +843,6 @@ pci_device_from_OF_node(struct device_node* node, u8* > bus, u8* devfn) > } > EXPORT_SYMBOL(pci_device_from_OF_node); > > -void __init > -pci_process_bridge_OF_ranges(struct pci_controller *hose, > - struct device_node *dev, int primary) > -{ > - static unsigned int static_lc_ranges[256] __initdata; > - const unsigned int *dt_ranges; > - unsigned int *lc_ranges, *ranges, *prev, size; > - int rlen = 0, orig_rlen; > - int memno = 0; > - struct resource *res; > - int np, na = of_n_addr_cells(dev); > - np = na + 5; > - > - /* First we try to merge ranges to fix a problem with some pmacs > - * that can have more than 3 ranges, fortunately using contiguous > - * addresses -- BenH > - */ > - dt_ranges = of_get_property(dev, "ranges", &rlen); > - if (!dt_ranges) > - return; > - /* Sanity check, though hopefully that never happens */ > - if (rlen > sizeof(static_lc_ranges)) { > - printk(KERN_WARNING "OF ranges property too large !\n"); > - rlen = sizeof(static_lc_ranges); > - } > - lc_ranges = static_lc_ranges; > - memcpy(lc_ranges, dt_ranges, rlen); > - orig_rlen = rlen; > - > - /* Let's work on a copy of the "ranges" property instead of damaging > - * the device-tree image in memory > - */ > - ranges = lc_ranges; > - prev = NULL; > - while ((rlen -= np * sizeof(unsigned int)) >= 0) { > - if (prev) { > - if (prev[0] == ranges[0] && prev[1] == ranges[1] && > - (prev[2] + prev[na+4]) == ranges[2] && > - (prev[na+2] + prev[na+4]) == ranges[na+2]) { > - prev[na+4] += ranges[na+4]; > - ranges[0] = 0; > - ranges += np; > - continue; > - } > - } > - prev = ranges; > - ranges += np; > - } > - > - /* > - * The ranges property is laid out as an array of elements, > - * each of which comprises: > - * cells 0 - 2: a PCI address > - * cells 3 or 3+4: a CPU physical address > - * (size depending on dev->n_addr_cells) > - * cells 4+5 or 5+6: the size of the range > - */ > - ranges = lc_ranges; > - rlen = orig_rlen; > - while (ranges && (rlen -= np * sizeof(unsigned int)) >= 0) { > - res = NULL; > - size = ranges[na+4]; > - switch ((ranges[0] >> 24) & 0x3) { > - case 1: /* I/O space */ > - if (ranges[2] != 0) > - break; > - hose->io_base_phys = ranges[na+2]; > - /* limit I/O space to 16MB */ > - if (size > 0x01000000) > - size = 0x01000000; > - hose->io_base_virt = ioremap(ranges[na+2], size); > - if (primary) > - isa_io_base = (unsigned long) > hose->io_base_virt; > - res = &hose->io_resource; > - res->flags = IORESOURCE_IO; > - res->start = ranges[2]; > - DBG("PCI: IO 0x%llx -> 0x%llx\n", > - (u64)res->start, (u64)res->start + size - 1); > - break; > - case 2: /* memory space */ > - memno = 0; > - if (ranges[1] == 0 && ranges[2] == 0 > - && ranges[na+4] <= (16 << 20)) { > - /* 1st 16MB, i.e. ISA memory area */ > - if (primary) > - isa_mem_base = ranges[na+2]; > - memno = 1; > - } > - while (memno < 3 && hose->mem_resources[memno].flags) > - ++memno; > - if (memno == 0) > - hose->pci_mem_offset = ranges[na+2] - ranges[2]; > - if (memno < 3) { > - res = &hose->mem_resources[memno]; > - res->flags = IORESOURCE_MEM; > - if(ranges[0] & 0x40000000) > - res->flags |= IORESOURCE_PREFETCH; > - res->start = ranges[na+2]; > - DBG("PCI: MEM[%d] 0x%llx -> 0x%llx\n", memno, > - (u64)res->start, (u64)res->start + size - > 1); > - } > - break; > - } > - if (res != NULL) { > - res->name = dev->full_name; > - res->end = res->start + size - 1; > - res->parent = NULL; > - res->sibling = NULL; > - res->child = NULL; > - } > - ranges += np; > - } > -} > - > /* We create the "pci-OF-bus-map" property now so it appears in the > * /proc device tree > */ > diff --git a/arch/powerpc/kernel/pci_64.c b/arch/powerpc/kernel/pci_64.c > index 291ffbc..68bce38 100644 > --- a/arch/powerpc/kernel/pci_64.c > +++ b/arch/powerpc/kernel/pci_64.c > @@ -592,100 +592,6 @@ int pci_proc_domain(struct pci_bus *bus) > } > } > > -void __devinit pci_process_bridge_OF_ranges(struct pci_controller *hose, > - struct device_node *dev, int prim) > -{ > - const unsigned int *ranges; > - unsigned int pci_space; > - unsigned long size; > - int rlen = 0; > - int memno = 0; > - struct resource *res; > - int np, na = of_n_addr_cells(dev); > - unsigned long pci_addr, cpu_phys_addr; > - > - np = na + 5; > - > - /* From "PCI Binding to 1275" > - * The ranges property is laid out as an array of elements, > - * each of which comprises: > - * cells 0 - 2: a PCI address > - * cells 3 or 3+4: a CPU physical address > - * (size depending on dev->n_addr_cells) > - * cells 4+5 or 5+6: the size of the range > - */ > - ranges = of_get_property(dev, "ranges", &rlen); > - if (ranges == NULL) > - return; > - hose->io_base_phys = 0; > - while ((rlen -= np * sizeof(unsigned int)) >= 0) { > - res = NULL; > - pci_space = ranges[0]; > - pci_addr = ((unsigned long)ranges[1] << 32) | ranges[2]; > - cpu_phys_addr = of_translate_address(dev, &ranges[3]); > - size = ((unsigned long)ranges[na+3] << 32) | ranges[na+4]; > - ranges += np; > - if (size == 0) > - continue; > - > - /* Now consume following elements while they are contiguous */ > - while (rlen >= np * sizeof(unsigned int)) { > - unsigned long addr, phys; > - > - if (ranges[0] != pci_space) > - break; > - addr = ((unsigned long)ranges[1] << 32) | ranges[2]; > - phys = ranges[3]; > - if (na >= 2) > - phys = (phys << 32) | ranges[4]; > - if (addr != pci_addr + size || > - phys != cpu_phys_addr + size) > - break; > - > - size += ((unsigned long)ranges[na+3] << 32) > - | ranges[na+4]; > - ranges += np; > - rlen -= np * sizeof(unsigned int); > - } > - > - switch ((pci_space >> 24) & 0x3) { > - case 1: /* I/O space */ > - hose->io_base_phys = cpu_phys_addr - pci_addr; > - /* handle from 0 to top of I/O window */ > - hose->pci_io_size = pci_addr + size; > - > - res = &hose->io_resource; > - res->flags = IORESOURCE_IO; > - res->start = pci_addr; > - DBG("phb%d: IO 0x%lx -> 0x%lx\n", hose->global_number, > - res->start, res->start + size - 1); > - break; > - case 2: /* memory space */ > - memno = 0; > - while (memno < 3 && hose->mem_resources[memno].flags) > - ++memno; > - > - if (memno == 0) > - hose->pci_mem_offset = cpu_phys_addr - pci_addr; > - if (memno < 3) { > - res = &hose->mem_resources[memno]; > - res->flags = IORESOURCE_MEM; > - res->start = cpu_phys_addr; > - DBG("phb%d: MEM 0x%lx -> 0x%lx\n", > hose->global_number, > - res->start, res->start + size - 1); > - } > - break; > - } > - if (res != NULL) { > - res->name = dev->full_name; > - res->end = res->start + size - 1; > - res->parent = NULL; > - res->sibling = NULL; > - res->child = NULL; > - } > - } > -} > - > #ifdef CONFIG_HOTPLUG > > int pcibios_unmap_io_space(struct pci_bus *bus) > diff --git a/include/asm-powerpc/ppc-pci.h b/include/asm-powerpc/ppc-pci.h > index b847aa1..cd8ad87 100644 > --- a/include/asm-powerpc/ppc-pci.h > +++ b/include/asm-powerpc/ppc-pci.h > @@ -15,6 +15,22 @@ > #include <linux/pci.h> > #include <asm/pci-bridge.h> > > +/* Address-cells and size-cells 2 */ > +struct ranges_pci64_sz64 { > + unsigned int pci_space; > + unsigned int pci_addr[2]; > + unsigned int phys_addr[2]; > + unsigned int size[2]; > +}; > + > +/* Address-cells 1 */ > +struct ranges_pci32_sz64 { > + unsigned int pci_space; > + unsigned int pci_addr[2]; > + unsigned int phys_addr[1]; > + unsigned int size[2]; > +}; > + > extern unsigned long isa_io_base; > > extern void pci_setup_phb_io(struct pci_controller *hose, int primary); > > _______________________________________________ > Linuxppc-dev mailing list > Linuxppc-dev@ozlabs.org > https://ozlabs.org/mailman/listinfo/linuxppc-dev > -- David Gibson | I'll have my music baroque, and my code david AT gibson.dropbear.id.au | minimalist, thank you. NOT _the_ _other_ | _way_ _around_! http://www.ozlabs.org/~dgibson _______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@ozlabs.org https://ozlabs.org/mailman/listinfo/linuxppc-dev