blob: 526636dd17728d3e9914fa1c9f76df24268761ee [file] [log] [blame]
/*
* 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 <sys/mman.h>
#include <sys/poll.h>
#include <sys/stat.h>
#include <dirent.h>
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);
}
}