]> xenbits.xensource.com Git - people/jgross/linux.git/commitdiff
s390/pci: improve zpci_dev reference counting
authorNiklas Schnelle <schnelle@linux.ibm.com>
Mon, 20 Sep 2021 07:32:21 +0000 (09:32 +0200)
committerVasily Gorbik <gor@linux.ibm.com>
Sun, 27 Mar 2022 20:18:39 +0000 (22:18 +0200)
Currently zpci_dev uses kref based reference counting but only accounts
for one original reference plus one reference from an added pci_dev to
its underlying zpci_dev. Counting just the original reference worked
until the pci_dev reference was added in commit 2a671f77ee49 ("s390/pci:
fix use after free of zpci_dev") because once a zpci_dev goes away, i.e.
enters the reserved state, it would immediately get released. However
with the pci_dev reference this is no longer the case and the zpci_dev
may still appear in multiple availability events indicating that it was
reserved. This was solved by detecting when the zpci_dev is already on
its way out but still hanging around. This has however shown some light
on how unusual our zpci_dev reference counting is.

Improve upon this by modelling zpci_dev reference counting on pci_dev.
Analogous to pci_get_slot() increment the reference count in
get_zdev_by_fid(). Thus all users of get_zdev_by_fid() must drop the
reference once they are done with the zpci_dev.

Similar to pci_scan_single_device(), zpci_create_device() returns the
device with an initial count of 1 and the device added to the zpci_list
(analogous to the PCI bus' device_list). In turn users of
zpci_create_device() must only drop the reference once the device is
gone from the point of view of the zPCI subsystem, it might still be
referenced by the common PCI subsystem though.

Reviewed-by: Matthew Rosato <mjrosato@linux.ibm.com>
Signed-off-by: Niklas Schnelle <schnelle@linux.ibm.com>
Signed-off-by: Vasily Gorbik <gor@linux.ibm.com>
arch/s390/pci/pci.c
arch/s390/pci/pci_bus.h
arch/s390/pci/pci_clp.c
arch/s390/pci/pci_event.c

index 792f8e0f217895324d906953bbb33673362b2f05..5bcd9228db5fa406ee24d90569a961e80e2e9c5c 100644 (file)
@@ -69,6 +69,7 @@ struct zpci_dev *get_zdev_by_fid(u32 fid)
        list_for_each_entry(tmp, &zpci_list, entry) {
                if (tmp->fid == fid) {
                        zdev = tmp;
+                       zpci_zdev_get(zdev);
                        break;
                }
        }
index e359d2686178b8575dedc4a07021a70cc078246d..ecef3a9e16c0020a1a4f7c7be81260aa99e3f098 100644 (file)
@@ -19,7 +19,8 @@ void zpci_bus_remove_device(struct zpci_dev *zdev, bool set_error);
 void zpci_release_device(struct kref *kref);
 static inline void zpci_zdev_put(struct zpci_dev *zdev)
 {
-       kref_put(&zdev->kref, zpci_release_device);
+       if (zdev)
+               kref_put(&zdev->kref, zpci_release_device);
 }
 
 static inline void zpci_zdev_get(struct zpci_dev *zdev)
index 63f3e057c168f3067663f4a1159200326d569979..1057d7af4a551b7498a0a37f3856faa1ac52cd99 100644 (file)
@@ -23,6 +23,8 @@
 #include <asm/clp.h>
 #include <uapi/asm/clp.h>
 
+#include "pci_bus.h"
+
 bool zpci_unique_uid;
 
 void update_uid_checking(bool new)
@@ -404,8 +406,11 @@ static void __clp_add(struct clp_fh_list_entry *entry, void *data)
                return;
 
        zdev = get_zdev_by_fid(entry->fid);
-       if (!zdev)
-               zpci_create_device(entry->fid, entry->fh, entry->config_state);
+       if (zdev) {
+               zpci_zdev_put(zdev);
+               return;
+       }
+       zpci_create_device(entry->fid, entry->fh, entry->config_state);
 }
 
 int clp_scan_pci_devices(void)
index 2e3e5b2789257ea0735725659ead4e075a311d49..ea9db5cea64e30a8b7133b57b059f9375608a82e 100644 (file)
@@ -269,7 +269,7 @@ static void __zpci_event_error(struct zpci_ccdf_err *ccdf)
               pdev ? pci_name(pdev) : "n/a", ccdf->pec, ccdf->fid);
 
        if (!pdev)
-               return;
+               goto no_pdev;
 
        switch (ccdf->pec) {
        case 0x003a: /* Service Action or Error Recovery Successful */
@@ -286,6 +286,8 @@ static void __zpci_event_error(struct zpci_ccdf_err *ccdf)
                break;
        }
        pci_dev_put(pdev);
+no_pdev:
+       zpci_zdev_put(zdev);
 }
 
 void zpci_event_error(void *data)
@@ -314,6 +316,7 @@ static void zpci_event_hard_deconfigured(struct zpci_dev *zdev, u32 fh)
 static void __zpci_event_availability(struct zpci_ccdf_avail *ccdf)
 {
        struct zpci_dev *zdev = get_zdev_by_fid(ccdf->fid);
+       bool existing_zdev = !!zdev;
        enum zpci_state state;
 
        zpci_dbg(3, "avl fid:%x, fh:%x, pec:%x\n",
@@ -378,6 +381,8 @@ static void __zpci_event_availability(struct zpci_ccdf_avail *ccdf)
        default:
                break;
        }
+       if (existing_zdev)
+               zpci_zdev_put(zdev);
 }
 
 void zpci_event_availability(void *data)