Message ID | 1446567566-10952-1-git-send-email-phil.edworthy@renesas.com (mailing list archive) |
---|---|
State | Awaiting Upstream |
Delegated to: | Simon Horman |
Headers | show |
On Tue, Nov 03, 2015 at 04:19:26PM +0000, Phil Edworthy wrote: > If the dtb specifies dma-ranges, we use those values. Otherwise, we > default to the values that were previously hardcoded into the driver. > > Signed-off-by: Phil Edworthy <phil.edworthy@renesas.com> > --- > .../devicetree/bindings/pci/pci-rcar-gen2.txt | 6 ++ Acked-by: Rob Herring <robh@kernel.org> > drivers/pci/host/pci-rcar-gen2.c | 76 +++++++++++++++++++++- > 2 files changed, 79 insertions(+), 3 deletions(-) > > diff --git a/Documentation/devicetree/bindings/pci/pci-rcar-gen2.txt b/Documentation/devicetree/bindings/pci/pci-rcar-gen2.txt > index 7fab84b..891463c 100644 > --- a/Documentation/devicetree/bindings/pci/pci-rcar-gen2.txt > +++ b/Documentation/devicetree/bindings/pci/pci-rcar-gen2.txt > @@ -24,6 +24,11 @@ Required properties: > - interrupt-map-mask: standard property that helps to define the interrupt > mapping. > > +Optional properties: > +- dma-ranges: a single range for the inbound memory region. If not supplied, > + defaults to 1GiB at 0x40000000. Note there are hardware restrictions on the > + allowed combinations of address and size. > + > Example SoC configuration: > > pci0: pci@ee090000 { > @@ -38,6 +43,7 @@ Example SoC configuration: > #address-cells = <3>; > #size-cells = <2>; > #interrupt-cells = <1>; > + dma-ranges = <0x42000000 0 0x40000000 0 0x40000000 0 0x40000000>; > interrupt-map-mask = <0xff00 0 0 0x7>; > interrupt-map = <0x0000 0 0 1 &gic 0 108 IRQ_TYPE_LEVEL_HIGH > 0x0800 0 0 1 &gic 0 108 IRQ_TYPE_LEVEL_HIGH > diff --git a/drivers/pci/host/pci-rcar-gen2.c b/drivers/pci/host/pci-rcar-gen2.c > index c4f64bf..6295116 100644 > --- a/drivers/pci/host/pci-rcar-gen2.c > +++ b/drivers/pci/host/pci-rcar-gen2.c > @@ -15,6 +15,7 @@ > #include <linux/io.h> > #include <linux/kernel.h> > #include <linux/module.h> > +#include <linux/of_address.h> > #include <linux/of_pci.h> > #include <linux/pci.h> > #include <linux/platform_device.h> > @@ -102,6 +103,8 @@ struct rcar_pci_priv { > unsigned busnr; > int irq; > unsigned long window_size; > + unsigned long window_addr; > + unsigned long window_pci; > }; > > /* PCI configuration space operations */ > @@ -239,8 +242,8 @@ static int rcar_pci_setup(int nr, struct pci_sys_data *sys) > RCAR_PCI_ARBITER_PCIBP_MODE; > iowrite32(val, reg + RCAR_PCI_ARBITER_CTR_REG); > > - /* PCI-AHB mapping: 0x40000000 base */ > - iowrite32(0x40000000 | RCAR_PCIAHB_PREFETCH16, > + /* PCI-AHB mapping */ > + iowrite32(priv->window_addr | RCAR_PCIAHB_PREFETCH16, > reg + RCAR_PCIAHB_WIN1_CTR_REG); > > /* AHB-PCI mapping: OHCI/EHCI registers */ > @@ -251,7 +254,7 @@ static int rcar_pci_setup(int nr, struct pci_sys_data *sys) > iowrite32(RCAR_AHBPCI_WIN1_HOST | RCAR_AHBPCI_WIN_CTR_CFG, > reg + RCAR_AHBPCI_WIN1_CTR_REG); > /* Set PCI-AHB Window1 address */ > - iowrite32(0x40000000 | PCI_BASE_ADDRESS_MEM_PREFETCH, > + iowrite32(priv->window_pci | PCI_BASE_ADDRESS_MEM_PREFETCH, > reg + PCI_BASE_ADDRESS_1); > /* Set AHB-PCI bridge PCI communication area address */ > val = priv->cfg_res->start + RCAR_AHBPCI_PCICOM_OFFSET; > @@ -284,6 +287,64 @@ static struct pci_ops rcar_pci_ops = { > .write = pci_generic_config_write, > }; > > +static int pci_dma_range_parser_init(struct of_pci_range_parser *parser, > + struct device_node *node) > +{ > + const int na = 3, ns = 2; > + int rlen; > + > + parser->node = node; > + parser->pna = of_n_addr_cells(node); > + parser->np = parser->pna + na + ns; > + > + parser->range = of_get_property(node, "dma-ranges", &rlen); > + if (!parser->range) > + return -ENOENT; > + > + parser->end = parser->range + rlen / sizeof(__be32); > + return 0; > +} > + > +static int rcar_pci_parse_map_dma_ranges(struct rcar_pci_priv *pci, > + struct device_node *np) > +{ > + struct of_pci_range range; > + struct of_pci_range_parser parser; > + int index = 0; > + > + /* Failure to parse is ok as we fall back to defaults */ > + if (pci_dma_range_parser_init(&parser, np)) > + return 0; > + > + /* Get the dma-ranges from DT */ > + for_each_of_pci_range(&parser, &range) { > + /* Hardware only allows one inbound 32-bit range */ > + if (index) > + return -EINVAL; > + > + pci->window_addr = (unsigned long)range.cpu_addr; > + pci->window_pci = (unsigned long)range.pci_addr; > + pci->window_size = (unsigned long)range.size; > + > + /* Catch HW limitations */ > + if (!(range.flags & IORESOURCE_PREFETCH)) { > + dev_err(pci->dev, "window must be prefetchable\n"); > + return -EINVAL; > + } > + if (pci->window_addr) { > + u32 lowaddr = 1 << (ffs(pci->window_addr) - 1); > + > + if (lowaddr < pci->window_size) { > + dev_err(pci->dev, "invalid window size/addr\n"); > + return -EINVAL; > + } > + } > + index++; > + } > + > + return 0; > +} > + > static int rcar_pci_probe(struct platform_device *pdev) > { > struct resource *cfg_res, *mem_res; > @@ -329,6 +390,9 @@ static int rcar_pci_probe(struct platform_device *pdev) > return priv->irq; > } > > + /* default window addr and size if not specified in DT */ > + priv->window_addr = 0x40000000; > + priv->window_pci = 0x40000000; > priv->window_size = SZ_1G; > > if (pdev->dev.of_node) { > @@ -344,6 +408,12 @@ static int rcar_pci_probe(struct platform_device *pdev) > priv->busnr = busnr.start; > if (busnr.end != busnr.start) > dev_warn(&pdev->dev, "only one bus number supported\n"); > + > + ret = rcar_pci_parse_map_dma_ranges(priv, pdev->dev.of_node); > + if (ret < 0) { > + dev_err(&pdev->dev, "failed to parse dma-range\n"); > + return ret; > + } > } else { > priv->busnr = pdev->id; > } > -- > 1.9.1 > -- To unsubscribe from this list: send the line "unsubscribe linux-sh" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
On Thu, Nov 05, 2015 at 07:49:47AM -0600, Rob Herring wrote: > On Tue, Nov 03, 2015 at 04:19:26PM +0000, Phil Edworthy wrote: > > If the dtb specifies dma-ranges, we use those values. Otherwise, we > > default to the values that were previously hardcoded into the driver. > > > > Signed-off-by: Phil Edworthy <phil.edworthy@renesas.com> > > --- > > .../devicetree/bindings/pci/pci-rcar-gen2.txt | 6 ++ > > Acked-by: Rob Herring <robh@kernel.org> Bjorn, this looks good to me. Please feel free to apply. Acked-by: Simon Horman <horms+renesas@verge.net.au> -- To unsubscribe from this list: send the line "unsubscribe linux-sh" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
On Tue, Nov 03, 2015 at 04:19:26PM +0000, Phil Edworthy wrote: > If the dtb specifies dma-ranges, we use those values. Otherwise, we > default to the values that were previously hardcoded into the driver. > > Signed-off-by: Phil Edworthy <phil.edworthy@renesas.com> Applied with acks from Rob and Simon to pci/host-rcar for v4.5, thanks! > --- > .../devicetree/bindings/pci/pci-rcar-gen2.txt | 6 ++ > drivers/pci/host/pci-rcar-gen2.c | 76 +++++++++++++++++++++- > 2 files changed, 79 insertions(+), 3 deletions(-) > > diff --git a/Documentation/devicetree/bindings/pci/pci-rcar-gen2.txt b/Documentation/devicetree/bindings/pci/pci-rcar-gen2.txt > index 7fab84b..891463c 100644 > --- a/Documentation/devicetree/bindings/pci/pci-rcar-gen2.txt > +++ b/Documentation/devicetree/bindings/pci/pci-rcar-gen2.txt > @@ -24,6 +24,11 @@ Required properties: > - interrupt-map-mask: standard property that helps to define the interrupt > mapping. > > +Optional properties: > +- dma-ranges: a single range for the inbound memory region. If not supplied, > + defaults to 1GiB at 0x40000000. Note there are hardware restrictions on the > + allowed combinations of address and size. > + > Example SoC configuration: > > pci0: pci@ee090000 { > @@ -38,6 +43,7 @@ Example SoC configuration: > #address-cells = <3>; > #size-cells = <2>; > #interrupt-cells = <1>; > + dma-ranges = <0x42000000 0 0x40000000 0 0x40000000 0 0x40000000>; > interrupt-map-mask = <0xff00 0 0 0x7>; > interrupt-map = <0x0000 0 0 1 &gic 0 108 IRQ_TYPE_LEVEL_HIGH > 0x0800 0 0 1 &gic 0 108 IRQ_TYPE_LEVEL_HIGH > diff --git a/drivers/pci/host/pci-rcar-gen2.c b/drivers/pci/host/pci-rcar-gen2.c > index c4f64bf..6295116 100644 > --- a/drivers/pci/host/pci-rcar-gen2.c > +++ b/drivers/pci/host/pci-rcar-gen2.c > @@ -15,6 +15,7 @@ > #include <linux/io.h> > #include <linux/kernel.h> > #include <linux/module.h> > +#include <linux/of_address.h> > #include <linux/of_pci.h> > #include <linux/pci.h> > #include <linux/platform_device.h> > @@ -102,6 +103,8 @@ struct rcar_pci_priv { > unsigned busnr; > int irq; > unsigned long window_size; > + unsigned long window_addr; > + unsigned long window_pci; > }; > > /* PCI configuration space operations */ > @@ -239,8 +242,8 @@ static int rcar_pci_setup(int nr, struct pci_sys_data *sys) > RCAR_PCI_ARBITER_PCIBP_MODE; > iowrite32(val, reg + RCAR_PCI_ARBITER_CTR_REG); > > - /* PCI-AHB mapping: 0x40000000 base */ > - iowrite32(0x40000000 | RCAR_PCIAHB_PREFETCH16, > + /* PCI-AHB mapping */ > + iowrite32(priv->window_addr | RCAR_PCIAHB_PREFETCH16, > reg + RCAR_PCIAHB_WIN1_CTR_REG); > > /* AHB-PCI mapping: OHCI/EHCI registers */ > @@ -251,7 +254,7 @@ static int rcar_pci_setup(int nr, struct pci_sys_data *sys) > iowrite32(RCAR_AHBPCI_WIN1_HOST | RCAR_AHBPCI_WIN_CTR_CFG, > reg + RCAR_AHBPCI_WIN1_CTR_REG); > /* Set PCI-AHB Window1 address */ > - iowrite32(0x40000000 | PCI_BASE_ADDRESS_MEM_PREFETCH, > + iowrite32(priv->window_pci | PCI_BASE_ADDRESS_MEM_PREFETCH, > reg + PCI_BASE_ADDRESS_1); > /* Set AHB-PCI bridge PCI communication area address */ > val = priv->cfg_res->start + RCAR_AHBPCI_PCICOM_OFFSET; > @@ -284,6 +287,64 @@ static struct pci_ops rcar_pci_ops = { > .write = pci_generic_config_write, > }; > > +static int pci_dma_range_parser_init(struct of_pci_range_parser *parser, > + struct device_node *node) > +{ > + const int na = 3, ns = 2; > + int rlen; > + > + parser->node = node; > + parser->pna = of_n_addr_cells(node); > + parser->np = parser->pna + na + ns; > + > + parser->range = of_get_property(node, "dma-ranges", &rlen); > + if (!parser->range) > + return -ENOENT; > + > + parser->end = parser->range + rlen / sizeof(__be32); > + return 0; > +} > + > +static int rcar_pci_parse_map_dma_ranges(struct rcar_pci_priv *pci, > + struct device_node *np) > +{ > + struct of_pci_range range; > + struct of_pci_range_parser parser; > + int index = 0; > + > + /* Failure to parse is ok as we fall back to defaults */ > + if (pci_dma_range_parser_init(&parser, np)) > + return 0; > + > + /* Get the dma-ranges from DT */ > + for_each_of_pci_range(&parser, &range) { > + /* Hardware only allows one inbound 32-bit range */ > + if (index) > + return -EINVAL; > + > + pci->window_addr = (unsigned long)range.cpu_addr; > + pci->window_pci = (unsigned long)range.pci_addr; > + pci->window_size = (unsigned long)range.size; > + > + /* Catch HW limitations */ > + if (!(range.flags & IORESOURCE_PREFETCH)) { > + dev_err(pci->dev, "window must be prefetchable\n"); > + return -EINVAL; > + } > + if (pci->window_addr) { > + u32 lowaddr = 1 << (ffs(pci->window_addr) - 1); > + > + if (lowaddr < pci->window_size) { > + dev_err(pci->dev, "invalid window size/addr\n"); > + return -EINVAL; > + } > + } > + index++; > + } > + > + return 0; > +} > + > static int rcar_pci_probe(struct platform_device *pdev) > { > struct resource *cfg_res, *mem_res; > @@ -329,6 +390,9 @@ static int rcar_pci_probe(struct platform_device *pdev) > return priv->irq; > } > > + /* default window addr and size if not specified in DT */ > + priv->window_addr = 0x40000000; > + priv->window_pci = 0x40000000; > priv->window_size = SZ_1G; > > if (pdev->dev.of_node) { > @@ -344,6 +408,12 @@ static int rcar_pci_probe(struct platform_device *pdev) > priv->busnr = busnr.start; > if (busnr.end != busnr.start) > dev_warn(&pdev->dev, "only one bus number supported\n"); > + > + ret = rcar_pci_parse_map_dma_ranges(priv, pdev->dev.of_node); > + if (ret < 0) { > + dev_err(&pdev->dev, "failed to parse dma-range\n"); > + return ret; > + } > } else { > priv->busnr = pdev->id; > } > -- > 1.9.1 > > -- > To unsubscribe from this list: send the line "unsubscribe linux-pci" in > the body of a message to majordomo@vger.kernel.org > More majordomo info at http://vger.kernel.org/majordomo-info.html -- To unsubscribe from this list: send the line "unsubscribe linux-sh" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html
diff --git a/Documentation/devicetree/bindings/pci/pci-rcar-gen2.txt b/Documentation/devicetree/bindings/pci/pci-rcar-gen2.txt index 7fab84b..891463c 100644 --- a/Documentation/devicetree/bindings/pci/pci-rcar-gen2.txt +++ b/Documentation/devicetree/bindings/pci/pci-rcar-gen2.txt @@ -24,6 +24,11 @@ Required properties: - interrupt-map-mask: standard property that helps to define the interrupt mapping. +Optional properties: +- dma-ranges: a single range for the inbound memory region. If not supplied, + defaults to 1GiB at 0x40000000. Note there are hardware restrictions on the + allowed combinations of address and size. + Example SoC configuration: pci0: pci@ee090000 { @@ -38,6 +43,7 @@ Example SoC configuration: #address-cells = <3>; #size-cells = <2>; #interrupt-cells = <1>; + dma-ranges = <0x42000000 0 0x40000000 0 0x40000000 0 0x40000000>; interrupt-map-mask = <0xff00 0 0 0x7>; interrupt-map = <0x0000 0 0 1 &gic 0 108 IRQ_TYPE_LEVEL_HIGH 0x0800 0 0 1 &gic 0 108 IRQ_TYPE_LEVEL_HIGH diff --git a/drivers/pci/host/pci-rcar-gen2.c b/drivers/pci/host/pci-rcar-gen2.c index c4f64bf..6295116 100644 --- a/drivers/pci/host/pci-rcar-gen2.c +++ b/drivers/pci/host/pci-rcar-gen2.c @@ -15,6 +15,7 @@ #include <linux/io.h> #include <linux/kernel.h> #include <linux/module.h> +#include <linux/of_address.h> #include <linux/of_pci.h> #include <linux/pci.h> #include <linux/platform_device.h> @@ -102,6 +103,8 @@ struct rcar_pci_priv { unsigned busnr; int irq; unsigned long window_size; + unsigned long window_addr; + unsigned long window_pci; }; /* PCI configuration space operations */ @@ -239,8 +242,8 @@ static int rcar_pci_setup(int nr, struct pci_sys_data *sys) RCAR_PCI_ARBITER_PCIBP_MODE; iowrite32(val, reg + RCAR_PCI_ARBITER_CTR_REG); - /* PCI-AHB mapping: 0x40000000 base */ - iowrite32(0x40000000 | RCAR_PCIAHB_PREFETCH16, + /* PCI-AHB mapping */ + iowrite32(priv->window_addr | RCAR_PCIAHB_PREFETCH16, reg + RCAR_PCIAHB_WIN1_CTR_REG); /* AHB-PCI mapping: OHCI/EHCI registers */ @@ -251,7 +254,7 @@ static int rcar_pci_setup(int nr, struct pci_sys_data *sys) iowrite32(RCAR_AHBPCI_WIN1_HOST | RCAR_AHBPCI_WIN_CTR_CFG, reg + RCAR_AHBPCI_WIN1_CTR_REG); /* Set PCI-AHB Window1 address */ - iowrite32(0x40000000 | PCI_BASE_ADDRESS_MEM_PREFETCH, + iowrite32(priv->window_pci | PCI_BASE_ADDRESS_MEM_PREFETCH, reg + PCI_BASE_ADDRESS_1); /* Set AHB-PCI bridge PCI communication area address */ val = priv->cfg_res->start + RCAR_AHBPCI_PCICOM_OFFSET; @@ -284,6 +287,64 @@ static struct pci_ops rcar_pci_ops = { .write = pci_generic_config_write, }; +static int pci_dma_range_parser_init(struct of_pci_range_parser *parser, + struct device_node *node) +{ + const int na = 3, ns = 2; + int rlen; + + parser->node = node; + parser->pna = of_n_addr_cells(node); + parser->np = parser->pna + na + ns; + + parser->range = of_get_property(node, "dma-ranges", &rlen); + if (!parser->range) + return -ENOENT; + + parser->end = parser->range + rlen / sizeof(__be32); + return 0; +} + +static int rcar_pci_parse_map_dma_ranges(struct rcar_pci_priv *pci, + struct device_node *np) +{ + struct of_pci_range range; + struct of_pci_range_parser parser; + int index = 0; + + /* Failure to parse is ok as we fall back to defaults */ + if (pci_dma_range_parser_init(&parser, np)) + return 0; + + /* Get the dma-ranges from DT */ + for_each_of_pci_range(&parser, &range) { + /* Hardware only allows one inbound 32-bit range */ + if (index) + return -EINVAL; + + pci->window_addr = (unsigned long)range.cpu_addr; + pci->window_pci = (unsigned long)range.pci_addr; + pci->window_size = (unsigned long)range.size; + + /* Catch HW limitations */ + if (!(range.flags & IORESOURCE_PREFETCH)) { + dev_err(pci->dev, "window must be prefetchable\n"); + return -EINVAL; + } + if (pci->window_addr) { + u32 lowaddr = 1 << (ffs(pci->window_addr) - 1); + + if (lowaddr < pci->window_size) { + dev_err(pci->dev, "invalid window size/addr\n"); + return -EINVAL; + } + } + index++; + } + + return 0; +} + static int rcar_pci_probe(struct platform_device *pdev) { struct resource *cfg_res, *mem_res; @@ -329,6 +390,9 @@ static int rcar_pci_probe(struct platform_device *pdev) return priv->irq; } + /* default window addr and size if not specified in DT */ + priv->window_addr = 0x40000000; + priv->window_pci = 0x40000000; priv->window_size = SZ_1G; if (pdev->dev.of_node) { @@ -344,6 +408,12 @@ static int rcar_pci_probe(struct platform_device *pdev) priv->busnr = busnr.start; if (busnr.end != busnr.start) dev_warn(&pdev->dev, "only one bus number supported\n"); + + ret = rcar_pci_parse_map_dma_ranges(priv, pdev->dev.of_node); + if (ret < 0) { + dev_err(&pdev->dev, "failed to parse dma-range\n"); + return ret; + } } else { priv->busnr = pdev->id; }
If the dtb specifies dma-ranges, we use those values. Otherwise, we default to the values that were previously hardcoded into the driver. Signed-off-by: Phil Edworthy <phil.edworthy@renesas.com> --- .../devicetree/bindings/pci/pci-rcar-gen2.txt | 6 ++ drivers/pci/host/pci-rcar-gen2.c | 76 +++++++++++++++++++++- 2 files changed, 79 insertions(+), 3 deletions(-)