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

Reply via email to