Commit ab909509 authored by Niklas Schnelle's avatar Niklas Schnelle Committed by Vasily Gorbik

PCI: s390: Fix use-after-free of PCI resources with per-function hotplug

On s390 PCI functions may be hotplugged individually even when they
belong to a multi-function device. In particular on an SR-IOV device VFs
may be removed and later re-added.

In commit a50297cf ("s390/pci: separate zbus creation from
scanning") it was missed however that struct pci_bus and struct
zpci_bus's resource list retained a reference to the PCI functions MMIO
resources even though those resources are released and freed on
hot-unplug. These stale resources may subsequently be claimed when the
PCI function re-appears resulting in use-after-free.

One idea of fixing this use-after-free in s390 specific code that was
investigated was to simply keep resources around from the moment a PCI
function first appeared until the whole virtual PCI bus created for
a multi-function device disappears. The problem with this however is
that due to the requirement of artificial MMIO addreesses (address
cookies) extra logic is then needed to keep the address cookies
compatible on re-plug. At the same time the MMIO resources semantically
belong to the PCI function so tying their lifecycle to the function
seems more logical.

Instead a simpler approach is to remove the resources of an individually
hot-unplugged PCI function from the PCI bus's resource list while
keeping the resources of other PCI functions on the PCI bus untouched.

This is done by introducing pci_bus_remove_resource() to remove an
individual resource. Similarly the resource also needs to be removed
from the struct zpci_bus's resource list. It turns out however, that
there is really no need to add the MMIO resources to the struct
zpci_bus's resource list at all and instead we can simply use the
zpci_bar_struct's resource pointer directly.

Fixes: a50297cf ("s390/pci: separate zbus creation from scanning")
Signed-off-by: default avatarNiklas Schnelle <schnelle@linux.ibm.com>
Reviewed-by: default avatarMatthew Rosato <mjrosato@linux.ibm.com>
Acked-by: default avatarBjorn Helgaas <bhelgaas@google.com>
Link: https://lore.kernel.org/r/20230306151014.60913-2-schnelle@linux.ibm.comSigned-off-by: default avatarVasily Gorbik <gor@linux.ibm.com>
parent a52e5cdb
...@@ -544,8 +544,7 @@ static struct resource *__alloc_res(struct zpci_dev *zdev, unsigned long start, ...@@ -544,8 +544,7 @@ static struct resource *__alloc_res(struct zpci_dev *zdev, unsigned long start,
return r; return r;
} }
int zpci_setup_bus_resources(struct zpci_dev *zdev, int zpci_setup_bus_resources(struct zpci_dev *zdev)
struct list_head *resources)
{ {
unsigned long addr, size, flags; unsigned long addr, size, flags;
struct resource *res; struct resource *res;
...@@ -581,7 +580,6 @@ int zpci_setup_bus_resources(struct zpci_dev *zdev, ...@@ -581,7 +580,6 @@ int zpci_setup_bus_resources(struct zpci_dev *zdev,
return -ENOMEM; return -ENOMEM;
} }
zdev->bars[i].res = res; zdev->bars[i].res = res;
pci_add_resource(resources, res);
} }
zdev->has_resources = 1; zdev->has_resources = 1;
...@@ -590,17 +588,23 @@ int zpci_setup_bus_resources(struct zpci_dev *zdev, ...@@ -590,17 +588,23 @@ int zpci_setup_bus_resources(struct zpci_dev *zdev,
static void zpci_cleanup_bus_resources(struct zpci_dev *zdev) static void zpci_cleanup_bus_resources(struct zpci_dev *zdev)
{ {
struct resource *res;
int i; int i;
pci_lock_rescan_remove();
for (i = 0; i < PCI_STD_NUM_BARS; i++) { for (i = 0; i < PCI_STD_NUM_BARS; i++) {
if (!zdev->bars[i].size || !zdev->bars[i].res) res = zdev->bars[i].res;
if (!res)
continue; continue;
release_resource(res);
pci_bus_remove_resource(zdev->zbus->bus, res);
zpci_free_iomap(zdev, zdev->bars[i].map_idx); zpci_free_iomap(zdev, zdev->bars[i].map_idx);
release_resource(zdev->bars[i].res); zdev->bars[i].res = NULL;
kfree(zdev->bars[i].res); kfree(res);
} }
zdev->has_resources = 0; zdev->has_resources = 0;
pci_unlock_rescan_remove();
} }
int pcibios_device_add(struct pci_dev *pdev) int pcibios_device_add(struct pci_dev *pdev)
......
...@@ -41,9 +41,7 @@ static int zpci_nb_devices; ...@@ -41,9 +41,7 @@ static int zpci_nb_devices;
*/ */
static int zpci_bus_prepare_device(struct zpci_dev *zdev) static int zpci_bus_prepare_device(struct zpci_dev *zdev)
{ {
struct resource_entry *window, *n; int rc, i;
struct resource *res;
int rc;
if (!zdev_enabled(zdev)) { if (!zdev_enabled(zdev)) {
rc = zpci_enable_device(zdev); rc = zpci_enable_device(zdev);
...@@ -57,10 +55,10 @@ static int zpci_bus_prepare_device(struct zpci_dev *zdev) ...@@ -57,10 +55,10 @@ static int zpci_bus_prepare_device(struct zpci_dev *zdev)
} }
if (!zdev->has_resources) { if (!zdev->has_resources) {
zpci_setup_bus_resources(zdev, &zdev->zbus->resources); zpci_setup_bus_resources(zdev);
resource_list_for_each_entry_safe(window, n, &zdev->zbus->resources) { for (i = 0; i < PCI_STD_NUM_BARS; i++) {
res = window->res; if (zdev->bars[i].res)
pci_bus_add_resource(zdev->zbus->bus, res, 0); pci_bus_add_resource(zdev->zbus->bus, zdev->bars[i].res, 0);
} }
} }
......
...@@ -30,8 +30,7 @@ static inline void zpci_zdev_get(struct zpci_dev *zdev) ...@@ -30,8 +30,7 @@ static inline void zpci_zdev_get(struct zpci_dev *zdev)
int zpci_alloc_domain(int domain); int zpci_alloc_domain(int domain);
void zpci_free_domain(int domain); void zpci_free_domain(int domain);
int zpci_setup_bus_resources(struct zpci_dev *zdev, int zpci_setup_bus_resources(struct zpci_dev *zdev);
struct list_head *resources);
static inline struct zpci_dev *zdev_from_bus(struct pci_bus *bus, static inline struct zpci_dev *zdev_from_bus(struct pci_bus *bus,
unsigned int devfn) unsigned int devfn)
......
...@@ -76,6 +76,27 @@ struct resource *pci_bus_resource_n(const struct pci_bus *bus, int n) ...@@ -76,6 +76,27 @@ struct resource *pci_bus_resource_n(const struct pci_bus *bus, int n)
} }
EXPORT_SYMBOL_GPL(pci_bus_resource_n); EXPORT_SYMBOL_GPL(pci_bus_resource_n);
void pci_bus_remove_resource(struct pci_bus *bus, struct resource *res)
{
struct pci_bus_resource *bus_res, *tmp;
int i;
for (i = 0; i < PCI_BRIDGE_RESOURCE_NUM; i++) {
if (bus->resource[i] == res) {
bus->resource[i] = NULL;
return;
}
}
list_for_each_entry_safe(bus_res, tmp, &bus->resources, list) {
if (bus_res->res == res) {
list_del(&bus_res->list);
kfree(bus_res);
return;
}
}
}
void pci_bus_remove_resources(struct pci_bus *bus) void pci_bus_remove_resources(struct pci_bus *bus)
{ {
int i; int i;
......
...@@ -1438,6 +1438,7 @@ void pci_bus_add_resource(struct pci_bus *bus, struct resource *res, ...@@ -1438,6 +1438,7 @@ void pci_bus_add_resource(struct pci_bus *bus, struct resource *res,
unsigned int flags); unsigned int flags);
struct resource *pci_bus_resource_n(const struct pci_bus *bus, int n); struct resource *pci_bus_resource_n(const struct pci_bus *bus, int n);
void pci_bus_remove_resources(struct pci_bus *bus); void pci_bus_remove_resources(struct pci_bus *bus);
void pci_bus_remove_resource(struct pci_bus *bus, struct resource *res);
int devm_request_pci_bus_resources(struct device *dev, int devm_request_pci_bus_resources(struct device *dev,
struct list_head *resources); struct list_head *resources);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment