@@ -137,7 +137,8 @@ static int _test_cmd_hwpt_alloc(int fd, __u32 device_id, __u32 pt_id,
})
static int _test_cmd_hwpt_alloc_nested(int fd, __u32 device_id, __u32 parent_id,
- __u32 *hwpt_id)
+ __u32 event_fd, __u32 *hwpt_id,
+ __u32 *out_fault_fd)
{
struct iommu_hwpt_selftest data = {
.flags = IOMMU_TEST_FLAG_NESTED,
@@ -153,21 +154,34 @@ static int _test_cmd_hwpt_alloc_nested(int fd, __u32 device_id, __u32 parent_id,
};
int ret;
+ if (out_fault_fd) {
+ cmd.event_fd = event_fd;
+ cmd.flags |= (IOMMU_HWPT_ALLOC_FLAGS_IOPF_CAPABLE |
+ IOMMU_HWPT_ALLOC_FLAGS_USER_PASID_TABLE);
+ }
+
ret = ioctl(fd, IOMMU_HWPT_ALLOC, &cmd);
if (ret)
return ret;
if (hwpt_id)
*hwpt_id = cmd.out_hwpt_id;
+ if (out_fault_fd)
+ *out_fault_fd = cmd.out_fault_fd;
return 0;
}
#define test_cmd_hwpt_alloc_nested(device_id, parent_id, hwpt_id) \
ASSERT_EQ(0, _test_cmd_hwpt_alloc_nested(self->fd, device_id, \
- parent_id, hwpt_id))
+ parent_id, 0, hwpt_id, NULL))
+#define test_cmd_hwpt_alloc_iopf(device_id, parent_id, event_fd, \
+ hwpt_id, out_fault_fd) \
+ ASSERT_EQ(0, _test_cmd_hwpt_alloc_nested(self->fd, device_id, \
+ parent_id, event_fd, \
+ hwpt_id, out_fault_fd))
#define test_err_cmd_hwpt_alloc_nested(_errno, device_id, parent_id, hwpt_id) \
EXPECT_ERRNO(_errno, \
_test_cmd_hwpt_alloc_nested(self->fd, device_id, \
- parent_id, hwpt_id))
+ parent_id, 0, hwpt_id, NULL))
static int _test_cmd_hwpt_invalidate(int fd, __u32 hwpt_id)
{
@@ -294,7 +294,9 @@ TEST_F(iommufd_ioas, nested_hwpt_alloc)
{
uint32_t nested_hwpt_id[2] = {};
uint32_t parent_hwpt_id = 0;
+ uint32_t event_fd, fault_fd;
uint32_t test_hwpt_id = 0;
+ uint32_t iopf_hwpt_id = 0;
if (self->device_id) {
/* Negative tests */
@@ -316,6 +318,12 @@ TEST_F(iommufd_ioas, nested_hwpt_alloc)
test_cmd_hwpt_check_iotlb(nested_hwpt_id[1],
IOMMU_TEST_IOTLB_DEFAULT);
+ /* Allocate and destroy iopf capable nested hwpt */
+ event_fd = eventfd(0, EFD_CLOEXEC);
+ ASSERT_NE(-1, event_fd);
+ test_cmd_hwpt_alloc_iopf(self->device_id, parent_hwpt_id,
+ event_fd, &iopf_hwpt_id, &fault_fd);
+
/* Negative test: a nested hwpt on top of a nested hwpt */
test_err_cmd_hwpt_alloc_nested(EINVAL, self->device_id,
nested_hwpt_id[0],
@@ -344,9 +352,16 @@ TEST_F(iommufd_ioas, nested_hwpt_alloc)
_test_ioctl_destroy(self->fd, nested_hwpt_id[1]));
test_ioctl_destroy(nested_hwpt_id[0]);
+ /* Switch from nested_hwpt_id[1] to iopf hwpt */
+ test_cmd_mock_domain_replace(self->stdev_id, iopf_hwpt_id);
+ EXPECT_ERRNO(EBUSY,
+ _test_ioctl_destroy(self->fd, iopf_hwpt_id));
+ test_ioctl_destroy(nested_hwpt_id[1]);
+
/* Detach from nested_hwpt_id[1] and destroy it */
test_cmd_mock_domain_replace(self->stdev_id, parent_hwpt_id);
- test_ioctl_destroy(nested_hwpt_id[1]);
+ test_ioctl_destroy(iopf_hwpt_id);
+ close(event_fd);
/* Detach from the parent hw_pagetable and destroy it */
test_cmd_mock_domain_replace(self->stdev_id, self->ioas_id);
The coverage includes operations to allocate, destroy, and replace an iopf-capable nested HWPT. Signed-off-by: Lu Baolu <baolu.lu@linux.intel.com> --- tools/testing/selftests/iommu/iommufd_utils.h | 20 ++++++++++++++++--- tools/testing/selftests/iommu/iommufd.c | 17 +++++++++++++++- 2 files changed, 33 insertions(+), 4 deletions(-)