blob: 7a014f76f0b4324a77d0868e23520ae3a48e6c8e [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_vgem.h"
#include "igt_debugfs.h"
#include "igt_sysfs.h"
#include <sys/mman.h>
#include <sys/stat.h>
#include <dirent.h>
IGT_TEST_DESCRIPTION("Basic sanity check of Virtual GEM module (vGEM).");
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 void test_sysfs_read(int fd)
{
int dir = igt_sysfs_open(fd, NULL);
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;
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 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_fixture {
fd = drm_open_driver(DRIVER_VGEM);
}
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_f("dmabuf-export")
test_dmabuf_export(fd);
igt_subtest_f("dmabuf-mmap")
test_dmabuf_mmap(fd);
}
igt_subtest("sysfs")
test_sysfs_read(fd);
igt_subtest("debugfs")
test_debugfs_read(fd);
igt_fixture {
close(fd);
}
}