diff options
author | Sergei Shtylyov <sergei.shtylyov@cogentembedded.com> | 2018-07-18 16:40:26 -0400 |
---|---|---|
committer | Bjorn Helgaas <helgaas@kernel.org> | 2018-07-18 16:40:26 -0400 |
commit | a5fb9fb023a1435f2b42bccd7f547560f3a21dc3 (patch) | |
tree | 136cf57a5d371c95263cc3249f47ee882b2469a2 | |
parent | a83a2173441698f7ac9867b93bcca21cf18a032d (diff) |
PCI: OF: Fix I/O space page leak
When testing the R-Car PCIe driver on the Condor board, if the PCIe PHY
driver was left disabled, the kernel crashed with this BUG:
kernel BUG at lib/ioremap.c:72!
Internal error: Oops - BUG: 0 [#1] PREEMPT SMP
Modules linked in:
CPU: 0 PID: 39 Comm: kworker/0:1 Not tainted 4.17.0-dirty #1092
Hardware name: Renesas Condor board based on r8a77980 (DT)
Workqueue: events deferred_probe_work_func
pstate: 80000005 (Nzcv daif -PAN -UAO)
pc : ioremap_page_range+0x370/0x3c8
lr : ioremap_page_range+0x40/0x3c8
sp : ffff000008da39e0
x29: ffff000008da39e0 x28: 00e8000000000f07
x27: ffff7dfffee00000 x26: 0140000000000000
x25: ffff7dfffef00000 x24: 00000000000fe100
x23: ffff80007b906000 x22: ffff000008ab8000
x21: ffff000008bb1d58 x20: ffff7dfffef00000
x19: ffff800009c30fb8 x18: 0000000000000001
x17: 00000000000152d0 x16: 00000000014012d0
x15: 0000000000000000 x14: 0720072007200720
x13: 0720072007200720 x12: 0720072007200720
x11: 0720072007300730 x10: 00000000000000ae
x9 : 0000000000000000 x8 : ffff7dffff000000
x7 : 0000000000000000 x6 : 0000000000000100
x5 : 0000000000000000 x4 : 000000007b906000
x3 : ffff80007c61a880 x2 : ffff7dfffeefffff
x1 : 0000000040000000 x0 : 00e80000fe100f07
Process kworker/0:1 (pid: 39, stack limit = 0x (ptrval))
Call trace:
ioremap_page_range+0x370/0x3c8
pci_remap_iospace+0x7c/0xac
pci_parse_request_of_pci_ranges+0x13c/0x190
rcar_pcie_probe+0x4c/0xb04
platform_drv_probe+0x50/0xbc
driver_probe_device+0x21c/0x308
__device_attach_driver+0x98/0xc8
bus_for_each_drv+0x54/0x94
__device_attach+0xc4/0x12c
device_initial_probe+0x10/0x18
bus_probe_device+0x90/0x98
deferred_probe_work_func+0xb0/0x150
process_one_work+0x12c/0x29c
worker_thread+0x200/0x3fc
kthread+0x108/0x134
ret_from_fork+0x10/0x18
Code: f9004ba2 54000080 aa0003fb 17ffff48 (d4210000)
It turned out that pci_remap_iospace() wasn't undone when the driver's
probe failed, and since devm_phy_optional_get() returned -EPROBE_DEFER,
the probe was retried, finally causing the BUG due to trying to remap
already remapped pages.
Introduce the devm_pci_remap_iospace() managed API and replace the
pci_remap_iospace() call with it to fix the bug.
Fixes: dbf9826d5797 ("PCI: generic: Convert to DT resource parsing API")
Signed-off-by: Sergei Shtylyov <sergei.shtylyov@cogentembedded.com>
[lorenzo.pieralisi@arm.com: split commit/updated the commit log]
Signed-off-by: Lorenzo Pieralisi <lorenzo.pieralisi@arm.com>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Reviewed-by: Linus Walleij <linus.walleij@linaro.org>
-rw-r--r-- | drivers/pci/of.c | 2 | ||||
-rw-r--r-- | drivers/pci/pci.c | 38 | ||||
-rw-r--r-- | include/linux/pci.h | 2 |
3 files changed, 41 insertions, 1 deletions
diff --git a/drivers/pci/of.c b/drivers/pci/of.c index d088c9147f10..69a60d6ebd73 100644 --- a/drivers/pci/of.c +++ b/drivers/pci/of.c | |||
@@ -612,7 +612,7 @@ int pci_parse_request_of_pci_ranges(struct device *dev, | |||
612 | 612 | ||
613 | switch (resource_type(res)) { | 613 | switch (resource_type(res)) { |
614 | case IORESOURCE_IO: | 614 | case IORESOURCE_IO: |
615 | err = pci_remap_iospace(res, iobase); | 615 | err = devm_pci_remap_iospace(dev, res, iobase); |
616 | if (err) { | 616 | if (err) { |
617 | dev_warn(dev, "error %d: failed to map resource %pR\n", | 617 | dev_warn(dev, "error %d: failed to map resource %pR\n", |
618 | err, res); | 618 | err, res); |
diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index 97acba712e4e..316496e99da9 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c | |||
@@ -3579,6 +3579,44 @@ void pci_unmap_iospace(struct resource *res) | |||
3579 | } | 3579 | } |
3580 | EXPORT_SYMBOL(pci_unmap_iospace); | 3580 | EXPORT_SYMBOL(pci_unmap_iospace); |
3581 | 3581 | ||
3582 | static void devm_pci_unmap_iospace(struct device *dev, void *ptr) | ||
3583 | { | ||
3584 | struct resource **res = ptr; | ||
3585 | |||
3586 | pci_unmap_iospace(*res); | ||
3587 | } | ||
3588 | |||
3589 | /** | ||
3590 | * devm_pci_remap_iospace - Managed pci_remap_iospace() | ||
3591 | * @dev: Generic device to remap IO address for | ||
3592 | * @res: Resource describing the I/O space | ||
3593 | * @phys_addr: physical address of range to be mapped | ||
3594 | * | ||
3595 | * Managed pci_remap_iospace(). Map is automatically unmapped on driver | ||
3596 | * detach. | ||
3597 | */ | ||
3598 | int devm_pci_remap_iospace(struct device *dev, const struct resource *res, | ||
3599 | phys_addr_t phys_addr) | ||
3600 | { | ||
3601 | const struct resource **ptr; | ||
3602 | int error; | ||
3603 | |||
3604 | ptr = devres_alloc(devm_pci_unmap_iospace, sizeof(*ptr), GFP_KERNEL); | ||
3605 | if (!ptr) | ||
3606 | return -ENOMEM; | ||
3607 | |||
3608 | error = pci_remap_iospace(res, phys_addr); | ||
3609 | if (error) { | ||
3610 | devres_free(ptr); | ||
3611 | } else { | ||
3612 | *ptr = res; | ||
3613 | devres_add(dev, ptr); | ||
3614 | } | ||
3615 | |||
3616 | return error; | ||
3617 | } | ||
3618 | EXPORT_SYMBOL(devm_pci_remap_iospace); | ||
3619 | |||
3582 | /** | 3620 | /** |
3583 | * devm_pci_remap_cfgspace - Managed pci_remap_cfgspace() | 3621 | * devm_pci_remap_cfgspace - Managed pci_remap_cfgspace() |
3584 | * @dev: Generic device to remap IO address for | 3622 | * @dev: Generic device to remap IO address for |
diff --git a/include/linux/pci.h b/include/linux/pci.h index 340029b2fb38..abd5d5e17aee 100644 --- a/include/linux/pci.h +++ b/include/linux/pci.h | |||
@@ -1240,6 +1240,8 @@ int pci_register_io_range(struct fwnode_handle *fwnode, phys_addr_t addr, | |||
1240 | unsigned long pci_address_to_pio(phys_addr_t addr); | 1240 | unsigned long pci_address_to_pio(phys_addr_t addr); |
1241 | phys_addr_t pci_pio_to_address(unsigned long pio); | 1241 | phys_addr_t pci_pio_to_address(unsigned long pio); |
1242 | int pci_remap_iospace(const struct resource *res, phys_addr_t phys_addr); | 1242 | int pci_remap_iospace(const struct resource *res, phys_addr_t phys_addr); |
1243 | int devm_pci_remap_iospace(struct device *dev, const struct resource *res, | ||
1244 | phys_addr_t phys_addr); | ||
1243 | void pci_unmap_iospace(struct resource *res); | 1245 | void pci_unmap_iospace(struct resource *res); |
1244 | void __iomem *devm_pci_remap_cfgspace(struct device *dev, | 1246 | void __iomem *devm_pci_remap_cfgspace(struct device *dev, |
1245 | resource_size_t offset, | 1247 | resource_size_t offset, |