Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux
1
fork

Configure Feed

Select the types of activity you want to include in your feed.

at v6.19-rc6 126 lines 3.2 kB view raw
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 7#include <stdint.h> 8#include <stdio.h> 9#include <sys/ioctl.h> 10#include <unistd.h> 11 12#include <libvfio.h> 13#include "kselftest_harness.h" 14 15static const char iommu_dev_path[] = "/dev/iommu"; 16static const char *cdev_path; 17 18static int vfio_device_bind_iommufd_ioctl(int cdev_fd, int iommufd) 19{ 20 struct vfio_device_bind_iommufd bind_args = { 21 .argsz = sizeof(bind_args), 22 .iommufd = iommufd, 23 }; 24 25 return ioctl(cdev_fd, VFIO_DEVICE_BIND_IOMMUFD, &bind_args); 26} 27 28static int vfio_device_get_info_ioctl(int cdev_fd) 29{ 30 struct vfio_device_info info_args = { .argsz = sizeof(info_args) }; 31 32 return ioctl(cdev_fd, VFIO_DEVICE_GET_INFO, &info_args); 33} 34 35static int vfio_device_ioas_alloc_ioctl(int iommufd, struct iommu_ioas_alloc *alloc_args) 36{ 37 *alloc_args = (struct iommu_ioas_alloc){ 38 .size = sizeof(struct iommu_ioas_alloc), 39 }; 40 41 return ioctl(iommufd, IOMMU_IOAS_ALLOC, alloc_args); 42} 43 44static int vfio_device_attach_iommufd_pt_ioctl(int cdev_fd, u32 pt_id) 45{ 46 struct vfio_device_attach_iommufd_pt attach_args = { 47 .argsz = sizeof(attach_args), 48 .pt_id = pt_id, 49 }; 50 51 return ioctl(cdev_fd, VFIO_DEVICE_ATTACH_IOMMUFD_PT, &attach_args); 52} 53 54static int vfio_device_detach_iommufd_pt_ioctl(int cdev_fd) 55{ 56 struct vfio_device_detach_iommufd_pt detach_args = { 57 .argsz = sizeof(detach_args), 58 }; 59 60 return ioctl(cdev_fd, VFIO_DEVICE_DETACH_IOMMUFD_PT, &detach_args); 61} 62 63FIXTURE(vfio_cdev) { 64 int cdev_fd; 65 int iommufd; 66}; 67 68FIXTURE_SETUP(vfio_cdev) 69{ 70 ASSERT_LE(0, (self->cdev_fd = open(cdev_path, O_RDWR, 0))); 71 ASSERT_LE(0, (self->iommufd = open(iommu_dev_path, O_RDWR, 0))); 72} 73 74FIXTURE_TEARDOWN(vfio_cdev) 75{ 76 ASSERT_EQ(0, close(self->cdev_fd)); 77 ASSERT_EQ(0, close(self->iommufd)); 78} 79 80TEST_F(vfio_cdev, bind) 81{ 82 ASSERT_EQ(0, vfio_device_bind_iommufd_ioctl(self->cdev_fd, self->iommufd)); 83 ASSERT_EQ(0, vfio_device_get_info_ioctl(self->cdev_fd)); 84} 85 86TEST_F(vfio_cdev, get_info_without_bind_fails) 87{ 88 ASSERT_NE(0, vfio_device_get_info_ioctl(self->cdev_fd)); 89} 90 91TEST_F(vfio_cdev, bind_bad_iommufd_fails) 92{ 93 ASSERT_NE(0, vfio_device_bind_iommufd_ioctl(self->cdev_fd, -2)); 94} 95 96TEST_F(vfio_cdev, repeated_bind_fails) 97{ 98 ASSERT_EQ(0, vfio_device_bind_iommufd_ioctl(self->cdev_fd, self->iommufd)); 99 ASSERT_NE(0, vfio_device_bind_iommufd_ioctl(self->cdev_fd, self->iommufd)); 100} 101 102TEST_F(vfio_cdev, attach_detatch_pt) 103{ 104 struct iommu_ioas_alloc alloc_args; 105 106 ASSERT_EQ(0, vfio_device_bind_iommufd_ioctl(self->cdev_fd, self->iommufd)); 107 ASSERT_EQ(0, vfio_device_ioas_alloc_ioctl(self->iommufd, &alloc_args)); 108 ASSERT_EQ(0, vfio_device_attach_iommufd_pt_ioctl(self->cdev_fd, alloc_args.out_ioas_id)); 109 ASSERT_EQ(0, vfio_device_detach_iommufd_pt_ioctl(self->cdev_fd)); 110} 111 112TEST_F(vfio_cdev, attach_invalid_pt_fails) 113{ 114 ASSERT_EQ(0, vfio_device_bind_iommufd_ioctl(self->cdev_fd, self->iommufd)); 115 ASSERT_NE(0, vfio_device_attach_iommufd_pt_ioctl(self->cdev_fd, UINT32_MAX)); 116} 117 118int main(int argc, char *argv[]) 119{ 120 const char *device_bdf = vfio_selftests_get_bdf(&argc, argv); 121 122 cdev_path = vfio_pci_get_cdev_path(device_bdf); 123 printf("Using cdev device %s\n", cdev_path); 124 125 return test_harness_run(argc, argv); 126}