Skip to content

Commit

Permalink
iommu/vt-d: Identify default domains replaced with private
Browse files Browse the repository at this point in the history
When we put a device into an iommu group, the group's default
domain will be attached to the device. There are some corner
cases where the type (identity or dma) of the default domain
doesn't work for the device and the request of a new default
domain results in failure (e.x. multiple devices have already
existed in the group). In order to be compatible with the past,
we used a private domain. Mark the private domains and disallow
some iommu apis (map/unmap/iova_to_phys) on them.

Signed-off-by: Lu Baolu <[email protected]>
Signed-off-by: Joerg Roedel <[email protected]>
  • Loading branch information
LuBaolu authored and joergroedel committed May 28, 2019
1 parent fa954e6 commit 942067f
Showing 1 changed file with 63 additions and 1 deletion.
64 changes: 63 additions & 1 deletion drivers/iommu/intel-iommu.c
Original file line number Diff line number Diff line change
Expand Up @@ -305,6 +305,14 @@ static int hw_pass_through = 1;
/* si_domain contains mulitple devices */
#define DOMAIN_FLAG_STATIC_IDENTITY BIT(0)

/*
* This is a DMA domain allocated through the iommu domain allocation
* interface. But one or more devices belonging to this domain have
* been chosen to use a private domain. We should avoid to use the
* map/unmap/iova_to_phys APIs on it.
*/
#define DOMAIN_FLAG_LOSE_CHILDREN BIT(1)

#define for_each_domain_iommu(idx, domain) \
for (idx = 0; idx < g_num_of_iommus; idx++) \
if (domain->iommu_refcnt[idx])
Expand Down Expand Up @@ -4958,6 +4966,7 @@ static void domain_context_clear(struct intel_iommu *iommu, struct device *dev)

static void __dmar_remove_one_dev_info(struct device_domain_info *info)
{
struct dmar_domain *domain;
struct intel_iommu *iommu;
unsigned long flags;

Expand All @@ -4967,6 +4976,7 @@ static void __dmar_remove_one_dev_info(struct device_domain_info *info)
return;

iommu = info->iommu;
domain = info->domain;

if (info->dev) {
if (dev_is_pci(info->dev) && sm_supported(iommu))
Expand All @@ -4981,9 +4991,14 @@ static void __dmar_remove_one_dev_info(struct device_domain_info *info)
unlink_domain_info(info);

spin_lock_irqsave(&iommu->lock, flags);
domain_detach_iommu(info->domain, iommu);
domain_detach_iommu(domain, iommu);
spin_unlock_irqrestore(&iommu->lock, flags);

/* free the private domain */
if (domain->flags & DOMAIN_FLAG_LOSE_CHILDREN &&
!(domain->flags & DOMAIN_FLAG_STATIC_IDENTITY))
domain_exit(info->domain);

free_devinfo_mem(info);
}

Expand Down Expand Up @@ -5307,6 +5322,9 @@ static int intel_iommu_map(struct iommu_domain *domain,
int prot = 0;
int ret;

if (dmar_domain->flags & DOMAIN_FLAG_LOSE_CHILDREN)
return -EINVAL;

if (iommu_prot & IOMMU_READ)
prot |= DMA_PTE_READ;
if (iommu_prot & IOMMU_WRITE)
Expand Down Expand Up @@ -5348,6 +5366,8 @@ static size_t intel_iommu_unmap(struct iommu_domain *domain,
/* Cope with horrid API which requires us to unmap more than the
size argument if it happens to be a large-page mapping. */
BUG_ON(!pfn_to_dma_pte(dmar_domain, iova >> VTD_PAGE_SHIFT, &level));
if (dmar_domain->flags & DOMAIN_FLAG_LOSE_CHILDREN)
return 0;

if (size < VTD_PAGE_SIZE << level_to_offset_bits(level))
size = VTD_PAGE_SIZE << level_to_offset_bits(level);
Expand Down Expand Up @@ -5379,6 +5399,9 @@ static phys_addr_t intel_iommu_iova_to_phys(struct iommu_domain *domain,
int level = 0;
u64 phys = 0;

if (dmar_domain->flags & DOMAIN_FLAG_LOSE_CHILDREN)
return 0;

pte = pfn_to_dma_pte(dmar_domain, iova >> VTD_PAGE_SHIFT, &level);
if (pte)
phys = dma_pte_addr(pte);
Expand Down Expand Up @@ -5434,9 +5457,12 @@ static bool intel_iommu_capable(enum iommu_cap cap)

static int intel_iommu_add_device(struct device *dev)
{
struct dmar_domain *dmar_domain;
struct iommu_domain *domain;
struct intel_iommu *iommu;
struct iommu_group *group;
u8 bus, devfn;
int ret;

iommu = device_to_iommu(dev, &bus, &devfn);
if (!iommu)
Expand All @@ -5450,6 +5476,42 @@ static int intel_iommu_add_device(struct device *dev)
return PTR_ERR(group);

iommu_group_put(group);

domain = iommu_get_domain_for_dev(dev);
dmar_domain = to_dmar_domain(domain);
if (domain->type == IOMMU_DOMAIN_DMA) {
if (device_def_domain_type(dev, 1) == IOMMU_DOMAIN_IDENTITY) {
ret = iommu_request_dm_for_dev(dev);
if (ret) {
dmar_domain->flags |= DOMAIN_FLAG_LOSE_CHILDREN;
domain_add_dev_info(si_domain, dev);
dev_info(dev,
"Device uses a private identity domain.\n");
return 0;
}

return -ENODEV;
}
} else {
if (device_def_domain_type(dev, 1) == IOMMU_DOMAIN_DMA) {
ret = iommu_request_dma_domain_for_dev(dev);
if (ret) {
dmar_domain->flags |= DOMAIN_FLAG_LOSE_CHILDREN;
if (!get_valid_domain_for_dev(dev)) {
dev_warn(dev,
"Failed to get a private domain.\n");
return -ENOMEM;
}

dev_info(dev,
"Device uses a private dma domain.\n");
return 0;
}

return -ENODEV;
}
}

return 0;
}

Expand Down

0 comments on commit 942067f

Please sign in to comment.