/* * Copyright © 2016 Intel Corporation * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice (including the next * paragraph) shall be included in all copies or substantial portions of the * Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS * IN THE SOFTWARE. */ #include "igt.h" #include "igt_kmod.h" #include "igt_vgem.h" #include "igt_debugfs.h" #include "igt_sysfs.h" #include #include #include #include IGT_TEST_DESCRIPTION("Basic sanity check of Virtual GEM module (vGEM)."); static int __gem_setversion(int fd, drm_set_version_t *sv) { int err; err = 0; if (igt_ioctl(fd, DRM_IOCTL_SET_VERSION, sv)) err = -errno; errno = 0; return err; } static void test_setversion(int fd) { drm_set_version_t sv; memset(&sv, 0, sizeof(sv)); sv.drm_di_major = 1; /* must be equal to DRM_IF_MAJOR */ sv.drm_di_minor = 4; /* must be less than DRM_IF_MINOR */ sv.drm_dd_major = -1; /* don't care */ sv.drm_dd_minor = -1; /* don't care */ igt_assert_eq(__gem_setversion(fd, &sv), 0); igt_info("vgem DRM interface v%d.%d, device v%d.%d\n", sv.drm_di_major, sv.drm_di_minor, sv.drm_dd_major, sv.drm_dd_minor); } static void test_client(int fd) { close(drm_open_driver(DRIVER_VGEM)); close(drm_open_driver_render(DRIVER_VGEM)); } static void test_create(int fd) { struct vgem_bo bo; bo.width = 0; bo.height = 0; bo.bpp = 0; igt_assert_eq(__vgem_create(fd, &bo), -EINVAL); bo.width = 1; bo.height = 1; bo.bpp = 1; vgem_create(fd, &bo); igt_assert_eq(bo.size, 4096); gem_close(fd, bo.handle); bo.width = 1024; bo.height = 1024; bo.bpp = 8; vgem_create(fd, &bo); igt_assert_eq(bo.size, 1<<20); gem_close(fd, bo.handle); bo.width = 1<<15; bo.height = 1<<15; bo.bpp = 16; vgem_create(fd, &bo); igt_assert_eq(bo.size, 1<<31); gem_close(fd, bo.handle); } static void test_mmap(int fd) { struct vgem_bo bo; uint32_t *ptr; bo.width = 1024; bo.height = 1024; bo.bpp = 32; vgem_create(fd, &bo); ptr = vgem_mmap(fd, &bo, PROT_WRITE); gem_close(fd, bo.handle); for (int page = 0; page < bo.size >> 12; page++) ptr[page] = 0; munmap(ptr, bo.size); } static bool has_prime_import(int fd) { uint64_t value; if (drmGetCap(fd, DRM_CAP_PRIME, &value)) return false; return value & DRM_PRIME_CAP_IMPORT; } static void test_dmabuf_export(int fd) { struct vgem_bo bo; uint32_t handle; int other; int dmabuf; other = drm_open_driver(DRIVER_ANY); igt_require(has_prime_import(other)); bo.width = 1024; bo.height = 1; bo.bpp = 32; vgem_create(fd, &bo); dmabuf = prime_handle_to_fd(fd, bo.handle); gem_close(fd, bo.handle); handle = prime_fd_to_handle(other, dmabuf); close(dmabuf); gem_close(other, handle); close(other); } static void test_dmabuf_mmap(int fd) { struct vgem_bo bo; uint32_t *ptr; int export; bo.width = 1024; bo.height = 1024; bo.bpp = 32; vgem_create(fd, &bo); export = prime_handle_to_fd_for_mmap(fd, bo.handle); ptr = mmap(NULL, bo.size, PROT_WRITE, MAP_SHARED, export, 0); close(export); igt_assert(ptr != MAP_FAILED); for (int page = 0; page < bo.size >> 12; page++) ptr[page] = page; munmap(ptr, bo.size); ptr = vgem_mmap(fd, &bo, PROT_READ); gem_close(fd, bo.handle); for (int page = 0; page < bo.size >> 12; page++) igt_assert_eq(ptr[page], page); munmap(ptr, bo.size); } static bool prime_busy(int fd, bool excl) { struct pollfd pfd = { .fd = fd, .events = excl ? POLLOUT : POLLIN }; return poll(&pfd, 1, 0) == 0; } static void test_dmabuf_fence(int fd) { struct vgem_bo bo; int dmabuf; uint32_t fence; bo.width = 1024; bo.height = 1; bo.bpp = 32; vgem_create(fd, &bo); /* export, then fence */ dmabuf = prime_handle_to_fd(fd, bo.handle); fence = vgem_fence_attach(fd, &bo, 0); igt_assert(!prime_busy(dmabuf, false)); igt_assert(prime_busy(dmabuf, true)); vgem_fence_signal(fd, fence); igt_assert(!prime_busy(dmabuf, false)); igt_assert(!prime_busy(dmabuf, true)); fence = vgem_fence_attach(fd, &bo, VGEM_FENCE_WRITE); igt_assert(prime_busy(dmabuf, false)); igt_assert(prime_busy(dmabuf, true)); vgem_fence_signal(fd, fence); igt_assert(!prime_busy(dmabuf, false)); igt_assert(!prime_busy(dmabuf, true)); close(dmabuf); gem_close(fd, bo.handle); } static void test_dmabuf_fence_before(int fd) { struct vgem_bo bo; int dmabuf; uint32_t fence; bo.width = 1024; bo.height = 1; bo.bpp = 32; vgem_create(fd, &bo); fence = vgem_fence_attach(fd, &bo, 0); dmabuf = prime_handle_to_fd(fd, bo.handle); igt_assert(!prime_busy(dmabuf, false)); igt_assert(prime_busy(dmabuf, true)); vgem_fence_signal(fd, fence); igt_assert(!prime_busy(dmabuf, false)); igt_assert(!prime_busy(dmabuf, true)); close(dmabuf); gem_close(fd, bo.handle); vgem_create(fd, &bo); fence = vgem_fence_attach(fd, &bo, VGEM_FENCE_WRITE); dmabuf = prime_handle_to_fd(fd, bo.handle); igt_assert(prime_busy(dmabuf, false)); igt_assert(prime_busy(dmabuf, true)); vgem_fence_signal(fd, fence); igt_assert(!prime_busy(dmabuf, false)); igt_assert(!prime_busy(dmabuf, true)); close(dmabuf); gem_close(fd, bo.handle); } static void test_sysfs_read(int fd) { int dir = igt_sysfs_open(fd); DIR *dirp = fdopendir(dir); struct dirent *de; while ((de = readdir(dirp))) { struct stat st; if (*de->d_name == '.') continue; if (fstatat(dir, de->d_name, &st, 0)) continue; if (S_ISDIR(st.st_mode)) continue; igt_debug("Reading %s\n", de->d_name); igt_set_timeout(1, "vgem sysfs read stalled"); free(igt_sysfs_get(dir, de->d_name)); igt_reset_timeout(); } closedir(dirp); close(dir); } static void test_debugfs_read(int fd) { int dir = igt_debugfs_dir(fd); DIR *dirp = fdopendir(dir); struct dirent *de; igt_assert(dirp); while ((de = readdir(dirp))) { struct stat st; if (*de->d_name == '.') continue; if (fstatat(dir, de->d_name, &st, 0)) continue; if (S_ISDIR(st.st_mode)) continue; igt_debug("Reading %s\n", de->d_name); igt_set_timeout(1, "vgem debugfs read stalled"); free(igt_sysfs_get(dir, de->d_name)); igt_reset_timeout(); } closedir(dirp); close(dir); } static int module_unload(void) { return igt_kmod_unload("vgem", 0); } static void test_unload(void) { struct vgem_bo bo; int vgem, dmabuf; uint32_t *ptr; /* Load and unload vgem just to make sure it exists */ vgem = __drm_open_driver(DRIVER_VGEM); igt_require(vgem != -1); close(vgem); igt_require(module_unload() == 0); vgem = __drm_open_driver(DRIVER_VGEM); igt_assert(vgem != -1); /* The driver should stop the module from unloading */ igt_assert_f(module_unload() != 0, "open(//dev/vgem) should keep the module alive\n"); bo.width = 1024; bo.height = 1; bo.bpp = 32; vgem_create(vgem, &bo); close(vgem); /* Closing the driver should clear all normal references */ igt_assert_f(module_unload() == 0, "No open(/dev/vgem), should be able to unload\n"); vgem = __drm_open_driver(DRIVER_VGEM); igt_assert(vgem != -1); bo.width = 1024; bo.height = 1; bo.bpp = 32; vgem_create(vgem, &bo); dmabuf = prime_handle_to_fd(vgem, bo.handle); close(vgem); /* A dmabuf should prevent module unload. */ igt_assert_f(module_unload() != 0, "A dmabuf should keep the module alive\n"); close(dmabuf); igt_assert_f(module_unload() == 0, "No open dmabuf, should be able to unload\n"); vgem = __drm_open_driver(DRIVER_VGEM); igt_assert(vgem != -1); bo.width = 1024; bo.height = 1; bo.bpp = 32; vgem_create(vgem, &bo); dmabuf = prime_handle_to_fd_for_mmap(vgem, bo.handle); close(vgem); ptr = mmap(NULL, bo.size, PROT_WRITE, MAP_SHARED, dmabuf, 0); igt_assert(ptr != MAP_FAILED); close(dmabuf); /* Although closed, the mmap should keep the dmabuf/module alive */ igt_assert_f(module_unload() == 0, "A mmap should not keep the module alive\n"); for (int page = 0; page < bo.size >> 12; page++) ptr[1024*page + page%1024] = page; /* And finally we should have no more uses on the module. */ munmap(ptr, bo.size); } static bool has_prime_export(int fd) { uint64_t value; if (drmGetCap(fd, DRM_CAP_PRIME, &value)) return false; return value & DRM_PRIME_CAP_EXPORT; } igt_main { int fd = -1; igt_subtest("unload") test_unload(); igt_fixture { fd = drm_open_driver(DRIVER_VGEM); } igt_subtest_f("setversion") test_setversion(fd); igt_subtest_f("second-client") test_client(fd); igt_subtest_f("create") test_create(fd); igt_subtest_f("mmap") test_mmap(fd); igt_subtest_group { igt_fixture { igt_require(has_prime_export(fd)); } igt_subtest("dmabuf-export") test_dmabuf_export(fd); igt_subtest("dmabuf-mmap") test_dmabuf_mmap(fd); igt_subtest_group { igt_fixture { igt_require(vgem_has_fences(fd)); } igt_subtest("dmabuf-fence") test_dmabuf_fence(fd); igt_subtest("dmabuf-fence-before") test_dmabuf_fence_before(fd); } } igt_subtest("sysfs") test_sysfs_read(fd); igt_subtest("debugfs") test_debugfs_read(fd); igt_fixture { close(fd); } }