if (!dmar_find_matched_atsr_unit(pdev))
return NULL;
- info->iommu = iommu;
-
return info;
}
return NULL;
}
-static inline struct dmar_domain *
+static inline struct device_domain_info *
dmar_search_domain_by_dev_info(int segment, int bus, int devfn)
{
struct device_domain_info *info;
list_for_each_entry(info, &device_domain_list, global)
if (info->segment == segment && info->bus == bus &&
info->devfn == devfn)
- return info->domain;
+ return info;
return NULL;
}
-static struct dmar_domain *dmar_insert_dev_info(int segment, int bus, int devfn,
+static struct dmar_domain *dmar_insert_dev_info(struct intel_iommu *iommu,
+ int segment, int bus, int devfn,
struct device *dev,
struct dmar_domain *domain)
{
- struct dmar_domain *found;
+ struct dmar_domain *found = NULL;
struct device_domain_info *info;
unsigned long flags;
info->devfn = devfn;
info->dev = dev;
info->domain = domain;
+ info->iommu = iommu;
if (!dev)
domain->flags |= DOMAIN_FLAG_P2P_MULTIPLE_DEVICES;
spin_lock_irqsave(&device_domain_lock, flags);
if (dev)
found = find_domain(dev);
- else
- found = dmar_search_domain_by_dev_info(segment, bus, devfn);
+ else {
+ struct device_domain_info *info2;
+ info2 = dmar_search_domain_by_dev_info(segment, bus, devfn);
+ if (info2)
+ found = info2->domain;
+ }
if (found) {
spin_unlock_irqrestore(&device_domain_lock, flags);
free_devinfo_mem(info);
static struct dmar_domain *get_domain_for_dev(struct pci_dev *pdev, int gaw)
{
struct dmar_domain *domain, *free = NULL;
- struct intel_iommu *iommu;
+ struct intel_iommu *iommu = NULL;
+ struct device_domain_info *info;
struct dmar_drhd_unit *drhd;
struct pci_dev *dev_tmp;
unsigned long flags;
devfn = dev_tmp->devfn;
}
spin_lock_irqsave(&device_domain_lock, flags);
- domain = dmar_search_domain_by_dev_info(segment, bus, devfn);
+ info = dmar_search_domain_by_dev_info(segment, bus, devfn);
+ if (info) {
+ iommu = info->iommu;
+ domain = info->domain;
+ }
spin_unlock_irqrestore(&device_domain_lock, flags);
- /* pcie-pci bridge already has a domain, uses it */
- if (domain)
+ if (info)
goto found_domain;
}
/* register pcie-to-pci device */
if (dev_tmp) {
- domain = dmar_insert_dev_info(segment, bus, devfn, NULL, domain);
+ domain = dmar_insert_dev_info(iommu, segment, bus, devfn, NULL,
+ domain);
if (!domain)
goto error;
}
found_domain:
- domain = dmar_insert_dev_info(segment, pdev->bus->number, pdev->devfn,
- &pdev->dev, domain);
+ domain = dmar_insert_dev_info(iommu, segment, pdev->bus->number,
+ pdev->devfn, &pdev->dev, domain);
error:
if (free != domain)
domain_exit(free);
int translation)
{
struct dmar_domain *ndomain;
+ struct intel_iommu *iommu;
int ret;
- ndomain = dmar_insert_dev_info(pci_domain_nr(pdev->bus),
+ iommu = device_to_iommu(pci_domain_nr(pdev->bus),
+ pdev->bus->number, pdev->devfn);
+ if (!iommu)
+ return -ENODEV;
+
+ ndomain = dmar_insert_dev_info(iommu, pci_domain_nr(pdev->bus),
pdev->bus->number, pdev->devfn,
&pdev->dev, domain);
if (ndomain != domain)