/* * Copyright © 2014 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. * * Authors: * Daniel Vetter */ #include "igt.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define DURATION 10 int fd; drm_intel_bufmgr *bufmgr; int fd1; drm_intel_bufmgr *bufmgr1; bool use_flink; static void new_buffers(void) { unsigned int *buf1; drm_intel_bo *bo1, *bo2; bo1 = drm_intel_bo_alloc(bufmgr, "buf1",16384, 4096); igt_assert(bo1); drm_intel_bo_map(bo1, 1); bo2 = drm_intel_bo_alloc(bufmgr, "buf2", 16384, 4096); igt_assert(bo2); drm_intel_bo_map(bo2, 1); buf1 = (unsigned int *)bo1->virtual; igt_assert(buf1); memset(buf1, 0, 16384); buf1[4000]=0x05000000; drm_intel_bo_exec(bo1, 16384, NULL, 0,0); drm_intel_bo_wait_rendering(bo1); drm_intel_bo_unmap( bo1 ); drm_intel_bo_unreference(bo1); drm_intel_bo_unmap( bo2 ); drm_intel_bo_unreference(bo2); } static void test_surfaces(drm_intel_bo *bo_shared) { drm_intel_bo * bo; int loop=2; while(loop--) { if (use_flink) { uint32_t name; drm_intel_bo_flink(bo_shared, &name); bo = drm_intel_bo_gem_create_from_name(bufmgr, "shared resource", name); } else { int prime_fd; drm_intel_bo_gem_export_to_prime(bo_shared, &prime_fd); bo = drm_intel_bo_gem_create_from_prime(bufmgr, prime_fd, 4096); close(prime_fd); } igt_assert(bo); new_buffers(); drm_intel_bo_unreference(bo); } } static void start_test(void) { igt_until_timeout(DURATION) { drm_intel_bo * bo_shared; bo_shared = drm_intel_bo_alloc(bufmgr1, "buf-shared",16384, 4096); test_surfaces(bo_shared); drm_intel_bo_unreference(bo_shared); } } static void * test_thread(void * par) { #ifdef __linux__ igt_debug("start %ld\n", (long) gettid()); #else igt_debug("start %ld\n", (long) pthread_self()); #endif start_test(); return NULL; } struct import_race_thread_data { int prime_fd; uint32_t flink_name; unsigned int stop; pthread_mutex_t mutex; }; /* * Attempt to import the bo. It is possible that GEM_CLOSE was already called * in different thread and from i915 point of view the handle is no longer * valid (thus create_from_prime/name should fail). */ static void *import_close_thread(void *data) { struct import_race_thread_data *t = (struct import_race_thread_data *)data; drm_intel_bo *bo; pthread_mutex_lock(&t->mutex); while (!t->stop) { pthread_mutex_unlock(&t->mutex); bo = NULL; if (use_flink) bo = drm_intel_bo_gem_create_from_name(bufmgr, "buf-shared", t->flink_name); else { pthread_mutex_lock(&t->mutex); if (t->prime_fd != -1) { bo = drm_intel_bo_gem_create_from_prime(bufmgr, t->prime_fd, 4096); pthread_mutex_unlock(&t->mutex); } else /* Lock should be held on entering the loop */ continue; } if (bo == NULL) { /* * If the bo is NULL it means that we've unreferenced in other * thread - therefore we should expect ENOENT */ igt_assert_eq(errno, ENOENT); } else { drm_intel_bo_unreference(bo); } pthread_mutex_lock(&t->mutex); } pthread_mutex_unlock(&t->mutex); return NULL; } /* * It is possible to race between unreference of the underlying BO and importing * it from prime_fd/name. Verify that the behaviour of libdrm is consistent for * prime/flink. */ static void test_import_close_race(void) { pthread_t t; drm_intel_bo *bo; struct import_race_thread_data t_data; memset(&t_data, 0, sizeof(t_data)); pthread_mutex_init(&t_data.mutex, NULL); t_data.prime_fd = -1; igt_assert_eq(pthread_create(&t, NULL, import_close_thread , &t_data), 0); igt_until_timeout(DURATION) { bo = drm_intel_bo_alloc(bufmgr, "buf-shared", 4096, 4096); igt_assert(bo != NULL); /* * We setup the test in such way, that create_from_* can race between * unreference. If we're using prime, prime_fd is always a valid fd. */ if (use_flink) igt_assert_eq(drm_intel_bo_flink(bo, &(t_data.flink_name)), 0); else { pthread_mutex_lock(&t_data.mutex); igt_assert_eq(drm_intel_bo_gem_export_to_prime(bo, &(t_data.prime_fd)), 0); igt_assert_neq(t_data.prime_fd, -1); pthread_mutex_unlock(&t_data.mutex); } drm_intel_bo_unreference(bo); pthread_mutex_lock(&t_data.mutex); close(t_data.prime_fd); t_data.prime_fd = -1; pthread_mutex_unlock(&t_data.mutex); } pthread_mutex_lock(&t_data.mutex); t_data.stop = 1; pthread_mutex_unlock(&t_data.mutex); pthread_join(t, NULL); pthread_mutex_destroy(&t_data.mutex); } pthread_t test_thread_id1; pthread_t test_thread_id2; pthread_t test_thread_id3; pthread_t test_thread_id4; igt_main { igt_fixture { fd1 = drm_open_driver(DRIVER_INTEL); igt_assert(fd1 >= 0); bufmgr1 = drm_intel_bufmgr_gem_init(fd1, 8 *1024); igt_assert(bufmgr1); drm_intel_bufmgr_gem_enable_reuse(bufmgr1); fd = drm_open_driver(DRIVER_INTEL); igt_assert(fd >= 0); bufmgr = drm_intel_bufmgr_gem_init(fd, 8 *1024); igt_assert(bufmgr); drm_intel_bufmgr_gem_enable_reuse(bufmgr); } igt_subtest("import-close-race-flink") { use_flink = true; test_import_close_race(); } igt_subtest("import-close-race-prime") { use_flink = false; test_import_close_race(); } igt_subtest("flink") { use_flink = true; pthread_create(&test_thread_id1, NULL, test_thread, NULL); pthread_create(&test_thread_id2, NULL, test_thread, NULL); pthread_create(&test_thread_id3, NULL, test_thread, NULL); pthread_create(&test_thread_id4, NULL, test_thread, NULL); pthread_join(test_thread_id1, NULL); pthread_join(test_thread_id2, NULL); pthread_join(test_thread_id3, NULL); pthread_join(test_thread_id4, NULL); } igt_subtest("prime") { use_flink = false; pthread_create(&test_thread_id1, NULL, test_thread, NULL); pthread_create(&test_thread_id2, NULL, test_thread, NULL); pthread_create(&test_thread_id3, NULL, test_thread, NULL); pthread_create(&test_thread_id4, NULL, test_thread, NULL); pthread_join(test_thread_id1, NULL); pthread_join(test_thread_id2, NULL); pthread_join(test_thread_id3, NULL); pthread_join(test_thread_id4, NULL); } }