iommu/vt-d: Don't register bus-notifier under dmar_global_lock
authorJoerg Roedel <jroedel@suse.de>
Fri, 6 Oct 2017 13:00:53 +0000 (15:00 +0200)
committerJoerg Roedel <jroedel@suse.de>
Fri, 6 Oct 2017 13:09:30 +0000 (15:09 +0200)
The notifier function will take the dmar_global_lock too, so
lockdep complains about inverse locking order when the
notifier is registered under the dmar_global_lock.

Reported-by: Jan Kiszka <jan.kiszka@siemens.com>
Fixes: 59ce0515cdaf ('iommu/vt-d: Update DRHD/RMRR/ATSR device scope caches when PCI hotplug happens')
Signed-off-by: Joerg Roedel <jroedel@suse.de>
drivers/iommu/dmar.c
drivers/iommu/intel-iommu.c
include/linux/dmar.h

index 57c920c..1ea7cd5 100644 (file)
@@ -801,13 +801,16 @@ int __init dmar_dev_scope_init(void)
                                dmar_free_pci_notify_info(info);
                        }
                }
-
-               bus_register_notifier(&pci_bus_type, &dmar_pci_bus_nb);
        }
 
        return dmar_dev_scope_status;
 }
 
+void dmar_register_bus_notifier(void)
+{
+       bus_register_notifier(&pci_bus_type, &dmar_pci_bus_nb);
+}
+
 
 int __init dmar_table_init(void)
 {
index 6784a05..934cef9 100644 (file)
@@ -4752,6 +4752,16 @@ int __init intel_iommu_init(void)
                goto out_free_dmar;
        }
 
+       up_write(&dmar_global_lock);
+
+       /*
+        * The bus notifier takes the dmar_global_lock, so lockdep will
+        * complain later when we register it under the lock.
+        */
+       dmar_register_bus_notifier();
+
+       down_write(&dmar_global_lock);
+
        if (no_iommu || dmar_disabled) {
                /*
                 * We exit the function here to ensure IOMMU's remapping and
index e8ffba1..e2433bc 100644 (file)
@@ -112,6 +112,7 @@ static inline bool dmar_rcu_check(void)
 
 extern int dmar_table_init(void);
 extern int dmar_dev_scope_init(void);
+extern void dmar_register_bus_notifier(void);
 extern int dmar_parse_dev_scope(void *start, void *end, int *cnt,
                                struct dmar_dev_scope **devices, u16 segment);
 extern void *dmar_alloc_dev_scope(void *start, void *end, int *cnt);