On Wed, 21 Nov 2007 17:16:19 +1100 Benjamin Herrenschmidt wrote: > This patch merges the 32 and 64 bits implementations of > pci_process_bridge_OF_ranges(). The new function is cleaner than both > the old ones supports 64 bits ranges on ppc32 which is necessary for > the 4xx port. > > It also adds some better (hopefully) output to the kernel log which > should help disagnose problems and makes better use of existing OF > parsing helpers (avoiding a few bugs of both implementations along > the way). > > There are still a few unfortunate ifdef's but there is no way around > these for now at least not until some other bits of the PCI code are > made common. > > Signed-off-by: Benjamin Herrenschmidt <[EMAIL PROTECTED]> Acked-by: Vitaly Bordug <[EMAIL PROTECTED]>
> --- > > Tested on a few pSeries, PowerMac G5, and a 32 bits PowerMacs and > a BriQ. Please let me know if it misbehaves anywhere else. > Tested (whole series but anyway) on FSL 83xx reference platforms, no issues to address. Good work! > arch/powerpc/kernel/pci-common.c | 176 > +++++++++++++++++++++++++++++++++++++++ > arch/powerpc/kernel/pci_32.c | 114 ------------------------- > arch/powerpc/kernel/pci_64.c | 93 -------------------- > include/asm-powerpc/pci-bridge.h | 1 4 files changed, 177 > insertions(+), 207 deletions(-) > > Index: linux-work/arch/powerpc/kernel/pci-common.c > =================================================================== > --- linux-work.orig/arch/powerpc/kernel/pci-common.c > 2007-11-13 14:15:43.000000000 +1100 +++ > linux-work/arch/powerpc/kernel/pci-common.c 2007-11-13 > 16:04:06.000000000 +1100 @@ -479,3 +479,179 @@ void > pci_resource_to_user(const struct p *start = rsrc->start - offset; > *end = rsrc->end - offset; } > + > +/** > + * pci_process_bridge_OF_ranges - Parse PCI bridge resources from > device tree > + * @hose: newly allocated pci_controller to be setup > + * @dev: device node of the host bridge > + * @primary: set if primary bus (32 bits only, soon to be deprecated) > + * > + * This function will parse the "ranges" property of a PCI host > bridge device > + * node and setup the resource mapping of a pci controller based on > its > + * content. > + * > + * Life would be boring if it wasn't for a few issues that we have > to deal > + * with here: > + * > + * - We can only cope with one IO space range and up to 3 Memory > space > + * ranges. However, some machines (thanks Apple !) tend to split > their > + * space into lots of small contiguous ranges. So we have to > coalesce. > + * > + * - We can only cope with all memory ranges having the same offset > + * between CPU addresses and PCI addresses. Unfortunately, some > bridges > + * are setup for a large 1:1 mapping along with a small "window" > which > + * maps PCI address 0 to some arbitrary high address of the CPU > space in > + * order to give access to the ISA memory hole. > + * The way out of here that I've chosen for now is to always set > the > + * offset based on the first resource found, then override it if > we > + * have a different offset and the previous was set by an ISA > hole. > + * > + * - Some busses have IO space not starting at 0, which causes > trouble with > + * the way we do our IO resource renumbering. The code somewhat > deals with > + * it for 64 bits but I would expect problems on 32 bits. > + * > + * - Some 32 bits platforms such as 4xx can have physical space > larger than > + * 32 bits so we need to use 64 bits values for the parsing > + */ > +void __devinit pci_process_bridge_OF_ranges(struct pci_controller > *hose, > + struct device_node *dev, > + int primary) > +{ > + const u32 *ranges; > + int rlen; > + int pna = of_n_addr_cells(dev); > + int np = pna + 5; > + int memno = 0, isa_hole = -1; > + u32 pci_space; > + unsigned long long pci_addr, cpu_addr, pci_next, cpu_next, > size; > + unsigned long long isa_mb = 0; > + struct resource *res; > + > + printk(KERN_INFO "PCI host bridge %s %s ranges:\n", > + dev->full_name, primary ? "(primary)" : ""); > + > + /* Get ranges property */ > + ranges = of_get_property(dev, "ranges", &rlen); > + if (ranges == NULL) > + return; > + > + /* Parse it */ > + while ((rlen -= np * 4) >= 0) { > + /* Read next ranges element */ > + pci_space = ranges[0]; > + pci_addr = of_read_number(ranges + 1, 2); > + cpu_addr = of_translate_address(dev, ranges + 3); > + size = of_read_number(ranges + pna + 3, 2); > + ranges += np; > + if (cpu_addr == OF_BAD_ADDR || size == 0) > + continue; > + > + /* Now consume following elements while they are > contiguous */ > + for (;rlen >= np * sizeof(u32); ranges += np, rlen > -= np * 4) { > + if (ranges[0] != pci_space) > + break; > + pci_next = of_read_number(ranges + 1, 2); > + cpu_next = of_translate_address(dev, ranges > + 3); > + if (pci_next != pci_addr + size || > + cpu_next != cpu_addr + size) > + break; > + size += of_read_number(ranges + pna + 3, 2); > + } > + > + /* Act based on address space type */ > + res = NULL; > + switch ((pci_space >> 24) & 0x3) { > + case 1: /* PCI IO space */ > + printk(KERN_INFO > + " IO 0x%016llx..0x%016llx -> > 0x%016llx\n", > + cpu_addr, cpu_addr + size - 1, > pci_addr); + > + /* We support only one IO range */ > + if (hose->pci_io_size) { > + printk(KERN_WARNING > + " \\--> Skipped (too > many) !\n"); > + continue; > + } > +#ifdef CONFIG_PPC32 > + /* On 32 bits, limit I/O space to 16MB */ > + if (size > 0x01000000) > + size = 0x01000000; > + > + /* 32 bits needs to map IOs here */ > + hose->io_base_virt = ioremap(cpu_addr, size); > + > + /* Expect trouble if pci_addr is not 0 */ > + if (primary) > + isa_io_base = > + (unsigned > long)hose->io_base_virt; +#endif /* CONFIG_PPC32 */ > + /* pci_io_size and io_base_phys always > represent IO > + * space starting at 0 so we factor in > pci_addr > + */ > + hose->pci_io_size = pci_addr + size; > + hose->io_base_phys = cpu_addr - pci_addr; > + > + /* Build resource */ > + res = &hose->io_resource; > + res->flags = IORESOURCE_IO; > + res->start = pci_addr; > + break; > + case 2: /* PCI Memory space */ > + printk(KERN_INFO > + " MEM 0x%016llx..0x%016llx -> > 0x%016llx %s\n", > + cpu_addr, cpu_addr + size - 1, > pci_addr, > + (pci_space & 0x40000000) ? > "Prefetch" : ""); + > + /* We support only 3 memory ranges */ > + if (memno >= 3) { > + printk(KERN_WARNING > + " \\--> Skipped (too > many) !\n"); > + continue; > + } > + /* Handles ISA memory hole space here */ > + if (pci_addr == 0) { > + isa_mb = cpu_addr; > + isa_hole = memno; > + if (primary || isa_mem_base == 0) > + isa_mem_base = cpu_addr; > + } > + > + /* We get the PCI/Mem offset from the first > range or the, > + * current one if the offset came from an > ISA hole. > + * If they don't match, bugger. > + */ > + if (memno == 0 || > + (isa_hole >= 0 && pci_addr != 0 && > + hose->pci_mem_offset == isa_mb)) > + hose->pci_mem_offset = cpu_addr - > pci_addr; > + else if (pci_addr != 0 && > + hose->pci_mem_offset != cpu_addr - > pci_addr) { > + printk(KERN_WARNING > + " \\--> Skipped (offset > mismatch) !\n"); > + continue; > + } > + > + /* Build resource */ > + res = &hose->mem_resources[memno++]; > + res->flags = IORESOURCE_MEM; > + if (pci_space & 0x40000000) > + res->flags |= IORESOURCE_PREFETCH; > + res->start = cpu_addr; > + break; > + } > + if (res != NULL) { > + res->name = dev->full_name; > + res->end = res->start + size - 1; > + res->parent = NULL; > + res->sibling = NULL; > + res->child = NULL; > + } > + } > + > + /* Out of paranoia, let's put the ISA hole last if any */ > + if (isa_hole >= 0 && memno > 0 && isa_hole != (memno-1)) { > + struct resource tmp = hose->mem_resources[isa_hole]; > + hose->mem_resources[isa_hole] = > hose->mem_resources[memno-1]; > + hose->mem_resources[memno-1] = tmp; > + } > +} > Index: linux-work/arch/powerpc/kernel/pci_32.c > =================================================================== > --- linux-work.orig/arch/powerpc/kernel/pci_32.c 2007-11-13 > 14:16:17.000000000 +1100 +++ > linux-work/arch/powerpc/kernel/pci_32.c 2007-11-13 > 14:16:24.000000000 +1100 @@ -842,120 +842,6 @@ > pci_device_from_OF_node(struct device_no } > 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 > */ > Index: linux-work/arch/powerpc/kernel/pci_64.c > =================================================================== > --- linux-work.orig/arch/powerpc/kernel/pci_64.c 2007-11-13 > 14:15:43.000000000 +1100 +++ > linux-work/arch/powerpc/kernel/pci_64.c 2007-11-13 > 14:16:24.000000000 +1100 @@ -592,99 +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 > > Index: linux-work/include/asm-powerpc/pci-bridge.h > =================================================================== > --- linux-work.orig/include/asm-powerpc/pci-bridge.h > 2007-11-13 14:15:43.000000000 +1100 +++ > linux-work/include/asm-powerpc/pci-bridge.h 2007-11-13 > 14:16:24.000000000 +1100 @@ -27,6 +27,7 @@ struct pci_controller { > void __iomem *io_base_virt; > resource_size_t io_base_phys; > + resource_size_t pci_io_size; > > /* Some machines (PReP) have a non 1:1 mapping of > * the PCI memory space in the CPU bus space > _______________________________________________ > Linuxppc-dev mailing list > Linuxppc-dev@ozlabs.org > https://ozlabs.org/mailman/listinfo/linuxppc-dev -- Sincerely, Vitaly _______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@ozlabs.org https://ozlabs.org/mailman/listinfo/linuxppc-dev