vfio_iommufd_setup_test.c 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. // SPDX-License-Identifier: GPL-2.0
  2. #include <linux/limits.h>
  3. #include <linux/sizes.h>
  4. #include <linux/vfio.h>
  5. #include <linux/iommufd.h>
  6. #include <stdint.h>
  7. #include <stdio.h>
  8. #include <sys/ioctl.h>
  9. #include <unistd.h>
  10. #include <libvfio.h>
  11. #include "kselftest_harness.h"
  12. static const char iommu_dev_path[] = "/dev/iommu";
  13. static const char *cdev_path;
  14. static int vfio_device_bind_iommufd_ioctl(int cdev_fd, int iommufd)
  15. {
  16. struct vfio_device_bind_iommufd bind_args = {
  17. .argsz = sizeof(bind_args),
  18. .iommufd = iommufd,
  19. };
  20. return ioctl(cdev_fd, VFIO_DEVICE_BIND_IOMMUFD, &bind_args);
  21. }
  22. static int vfio_device_get_info_ioctl(int cdev_fd)
  23. {
  24. struct vfio_device_info info_args = { .argsz = sizeof(info_args) };
  25. return ioctl(cdev_fd, VFIO_DEVICE_GET_INFO, &info_args);
  26. }
  27. static int vfio_device_ioas_alloc_ioctl(int iommufd, struct iommu_ioas_alloc *alloc_args)
  28. {
  29. *alloc_args = (struct iommu_ioas_alloc){
  30. .size = sizeof(struct iommu_ioas_alloc),
  31. };
  32. return ioctl(iommufd, IOMMU_IOAS_ALLOC, alloc_args);
  33. }
  34. static int vfio_device_attach_iommufd_pt_ioctl(int cdev_fd, u32 pt_id)
  35. {
  36. struct vfio_device_attach_iommufd_pt attach_args = {
  37. .argsz = sizeof(attach_args),
  38. .pt_id = pt_id,
  39. };
  40. return ioctl(cdev_fd, VFIO_DEVICE_ATTACH_IOMMUFD_PT, &attach_args);
  41. }
  42. static int vfio_device_detach_iommufd_pt_ioctl(int cdev_fd)
  43. {
  44. struct vfio_device_detach_iommufd_pt detach_args = {
  45. .argsz = sizeof(detach_args),
  46. };
  47. return ioctl(cdev_fd, VFIO_DEVICE_DETACH_IOMMUFD_PT, &detach_args);
  48. }
  49. FIXTURE(vfio_cdev) {
  50. int cdev_fd;
  51. int iommufd;
  52. };
  53. FIXTURE_SETUP(vfio_cdev)
  54. {
  55. ASSERT_LE(0, (self->cdev_fd = open(cdev_path, O_RDWR, 0)));
  56. ASSERT_LE(0, (self->iommufd = open(iommu_dev_path, O_RDWR, 0)));
  57. }
  58. FIXTURE_TEARDOWN(vfio_cdev)
  59. {
  60. ASSERT_EQ(0, close(self->cdev_fd));
  61. ASSERT_EQ(0, close(self->iommufd));
  62. }
  63. TEST_F(vfio_cdev, bind)
  64. {
  65. ASSERT_EQ(0, vfio_device_bind_iommufd_ioctl(self->cdev_fd, self->iommufd));
  66. ASSERT_EQ(0, vfio_device_get_info_ioctl(self->cdev_fd));
  67. }
  68. TEST_F(vfio_cdev, get_info_without_bind_fails)
  69. {
  70. ASSERT_NE(0, vfio_device_get_info_ioctl(self->cdev_fd));
  71. }
  72. TEST_F(vfio_cdev, bind_bad_iommufd_fails)
  73. {
  74. ASSERT_NE(0, vfio_device_bind_iommufd_ioctl(self->cdev_fd, -2));
  75. }
  76. TEST_F(vfio_cdev, repeated_bind_fails)
  77. {
  78. ASSERT_EQ(0, vfio_device_bind_iommufd_ioctl(self->cdev_fd, self->iommufd));
  79. ASSERT_NE(0, vfio_device_bind_iommufd_ioctl(self->cdev_fd, self->iommufd));
  80. }
  81. TEST_F(vfio_cdev, attach_detatch_pt)
  82. {
  83. struct iommu_ioas_alloc alloc_args;
  84. ASSERT_EQ(0, vfio_device_bind_iommufd_ioctl(self->cdev_fd, self->iommufd));
  85. ASSERT_EQ(0, vfio_device_ioas_alloc_ioctl(self->iommufd, &alloc_args));
  86. ASSERT_EQ(0, vfio_device_attach_iommufd_pt_ioctl(self->cdev_fd, alloc_args.out_ioas_id));
  87. ASSERT_EQ(0, vfio_device_detach_iommufd_pt_ioctl(self->cdev_fd));
  88. }
  89. TEST_F(vfio_cdev, attach_invalid_pt_fails)
  90. {
  91. ASSERT_EQ(0, vfio_device_bind_iommufd_ioctl(self->cdev_fd, self->iommufd));
  92. ASSERT_NE(0, vfio_device_attach_iommufd_pt_ioctl(self->cdev_fd, UINT32_MAX));
  93. }
  94. int main(int argc, char *argv[])
  95. {
  96. const char *device_bdf = vfio_selftests_get_bdf(&argc, argv);
  97. cdev_path = vfio_pci_get_cdev_path(device_bdf);
  98. printf("Using cdev device %s\n", cdev_path);
  99. return test_harness_run(argc, argv);
  100. }