]> git.openfabrics.org - ~emulex/infiniband.git/commitdiff
iommu/vt-d: Free resources if failed to create domain for PCIe endpoint
authorJiang Liu <jiang.liu@linux.intel.com>
Wed, 19 Feb 2014 06:07:27 +0000 (14:07 +0800)
committerJoerg Roedel <joro@8bytes.org>
Tue, 4 Mar 2014 16:51:01 +0000 (17:51 +0100)
Enhance function get_domain_for_dev() to release allocated resources
if failed to create domain for PCIe endpoint, otherwise the allocated
resources will get lost.

Signed-off-by: Jiang Liu <jiang.liu@linux.intel.com>
Signed-off-by: Joerg Roedel <joro@8bytes.org>
drivers/iommu/intel-iommu.c

index 67b114e784bc74bcbfd7928bb6f6e3ba9f9a85e9..503cc739d4b145146da84402c06e63151c531665 100644 (file)
@@ -2015,7 +2015,7 @@ static int dmar_insert_dev_info(int segment, int bus, int devfn,
 /* domain is initialized */
 static struct dmar_domain *get_domain_for_dev(struct pci_dev *pdev, int gaw)
 {
-       struct dmar_domain *domain;
+       struct dmar_domain *domain, *free = NULL;
        struct intel_iommu *iommu;
        struct dmar_drhd_unit *drhd;
        struct pci_dev *dev_tmp;
@@ -2062,17 +2062,16 @@ static struct dmar_domain *get_domain_for_dev(struct pci_dev *pdev, int gaw)
                free_domain_mem(domain);
                goto error;
        }
-       if (domain_init(domain, gaw)) {
-               domain_exit(domain);
+       free = domain;
+       if (domain_init(domain, gaw))
                goto error;
-       }
 
        /* register pcie-to-pci device */
        if (dev_tmp) {
-               if (dmar_insert_dev_info(segment, bus, devfn, NULL, &domain)) {
-                       domain_exit(domain);
+               if (dmar_insert_dev_info(segment, bus, devfn, NULL, &domain))
                        goto error;
-               }
+               else
+                       free = NULL;
        }
 
 found_domain:
@@ -2080,6 +2079,8 @@ found_domain:
                                 pdev, &domain) == 0)
                return domain;
 error:
+       if (free)
+               domain_exit(free);
        /* recheck it here, maybe others set it */
        return find_domain(pdev);
 }