summaryrefslogtreecommitdiff
path: root/tests/drm_import_export.c
diff options
context:
space:
mode:
authorMichał Winiarski <michal.winiarski@intel.com>2015-07-24 16:43:33 +0200
committerThomas Wood <thomas.wood@intel.com>2015-07-24 15:55:18 +0100
commitd49a868378f88685aadc601f9d5a9ae435565ece (patch)
treec68eab66fcf6bfc4c071dbe94f2570c51f2cf390 /tests/drm_import_export.c
parente14507ce987a0cfb0ba9dec39afcb08a9ebc0453 (diff)
tests/drm_import_export: Add tests for prime/flink sharing races
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. v2: more comments in source file, dropped extra whitespace Signed-off-by: Michał Winiarski <michal.winiarski@intel.com> Cc: Thomas Wood <thomas.wood@intel.com> Signed-off-by: Thomas Wood <thomas.wood@intel.com>
Diffstat (limited to 'tests/drm_import_export.c')
-rw-r--r--tests/drm_import_export.c112
1 files changed, 112 insertions, 0 deletions
diff --git a/tests/drm_import_export.c b/tests/drm_import_export.c
index 57b13ddb..e24e0dff 100644
--- a/tests/drm_import_export.c
+++ b/tests/drm_import_export.c
@@ -131,6 +131,108 @@ static void * test_thread(void * par)
return NULL;
}
+#define IMPORT_RACE_LOOPS 100000
+
+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
+ /* We take the lock right after 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);
+ continue;
+ }
+
+ 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;
+ unsigned int loops = IMPORT_RACE_LOOPS;
+ 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);
+
+ while (loops--) {
+ 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(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;
@@ -153,6 +255,16 @@ igt_main {
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;