[PATCH 1/1] iommufd/selftest: Use right iommu_ops for mock device
From: Lu Baolu
Date: Thu Jan 11 2024 - 02:38:35 EST
In the iommu probe device path, __iommu_probe_device() gets the iommu_ops
for the device from dev->iommu->fwspec if this field has been initialized
before probing. Otherwise, it will lookup the global iommu device list
and use the iommu_ops of the first iommu device which has no
dev->iommu->fwspec. This causes the wrong iommu_ops to be used for the mock
device on x86 platforms where dev->iommu->fwspec is not used.
Preallocate the fwspec for the mock device so that the right iommu ops can
be used.
Fixes: 17de3f5fdd35 ("iommu: Retire bus ops")
Cc: Robin Murphy <robin.murphy@xxxxxxx>
Signed-off-by: Lu Baolu <baolu.lu@xxxxxxxxxxxxxxx>
---
drivers/iommu/iommufd/selftest.c | 22 ++++++++++++++++++++++
1 file changed, 22 insertions(+)
diff --git a/drivers/iommu/iommufd/selftest.c b/drivers/iommu/iommufd/selftest.c
index cf3e9fed039e..4eca67b8a5c6 100644
--- a/drivers/iommu/iommufd/selftest.c
+++ b/drivers/iommu/iommufd/selftest.c
@@ -611,6 +611,8 @@ static void mock_dev_release(struct device *dev)
static struct mock_dev *mock_dev_create(unsigned long dev_flags)
{
+ struct iommu_fwspec *fwspec;
+ struct dev_iommu *param;
struct mock_dev *mdev;
int rc;
@@ -621,10 +623,28 @@ static struct mock_dev *mock_dev_create(unsigned long dev_flags)
if (!mdev)
return ERR_PTR(-ENOMEM);
+ /* fwspec and param will be freed in the iommu core */
+ fwspec = kzalloc(sizeof(*fwspec), GFP_KERNEL);
+ if (!fwspec) {
+ kfree(mdev);
+ return ERR_PTR(-ENOMEM);
+ }
+ fwspec->ops = &mock_ops;
+
+ param = kzalloc(sizeof(*param), GFP_KERNEL);
+ if (!param) {
+ kfree(mdev);
+ kfree(fwspec);
+ return ERR_PTR(-ENOMEM);
+ }
+ mutex_init(¶m->lock);
+ param->fwspec = fwspec;
+
device_initialize(&mdev->dev);
mdev->flags = dev_flags;
mdev->dev.release = mock_dev_release;
mdev->dev.bus = &iommufd_mock_bus_type.bus;
+ mdev->dev.iommu = param;
rc = dev_set_name(&mdev->dev, "iommufd_mock%u",
atomic_inc_return(&mock_dev_num));
@@ -638,6 +658,8 @@ static struct mock_dev *mock_dev_create(unsigned long dev_flags)
err_put:
put_device(&mdev->dev);
+ kfree(param);
+ kfree(fwspec);
return ERR_PTR(rc);
}
--
2.34.1