[cairo] [PATCH] glx-multithread-makecurrent-*: New tests for MESA_multithread_makecurrent
Eric Anholt
eric at anholt.net
Mon Feb 21 14:36:22 PST 2011
---
tests/all.tests | 4 +
tests/glx/CMakeLists.txt | 8 ++
tests/glx/glx-multithread-makecurrent-1.c | 183 +++++++++++++++++++++++++++++
tests/glx/glx-multithread-makecurrent-2.c | 183 +++++++++++++++++++++++++++++
tests/glx/glx-multithread-makecurrent-3.c | 175 +++++++++++++++++++++++++++
tests/glx/glx-multithread-makecurrent-4.c | 174 +++++++++++++++++++++++++++
6 files changed, 727 insertions(+), 0 deletions(-)
create mode 100644 tests/glx/glx-multithread-makecurrent-1.c
create mode 100644 tests/glx/glx-multithread-makecurrent-2.c
create mode 100644 tests/glx/glx-multithread-makecurrent-3.c
create mode 100644 tests/glx/glx-multithread-makecurrent-4.c
diff --git a/tests/all.tests b/tests/all.tests
index 10a384e..7c3df8a 100644
--- a/tests/all.tests
+++ b/tests/all.tests
@@ -828,6 +828,10 @@ glx = Group()
add_plain_test(glx, 'glx-destroycontext-1')
add_plain_test(glx, 'glx-destroycontext-2')
add_plain_test(glx, 'glx-multithread')
+add_plain_test(glx, 'glx-multithread-makecurrent-1')
+add_plain_test(glx, 'glx-multithread-makecurrent-2')
+add_plain_test(glx, 'glx-multithread-makecurrent-3')
+add_plain_test(glx, 'glx-multithread-makecurrent-4')
add_plain_test(glx, 'glx-shader-sharing')
add_plain_test(glx, 'glx-swap-exchange')
glx['glx-swap-event_event'] = PlainExecTest(['glx-swap-event', '-auto', '--event'])
diff --git a/tests/glx/CMakeLists.txt b/tests/glx/CMakeLists.txt
index 8516dc5..3f86450 100644
--- a/tests/glx/CMakeLists.txt
+++ b/tests/glx/CMakeLists.txt
@@ -27,6 +27,14 @@ IF(${CMAKE_SYSTEM_NAME} MATCHES "Linux")
add_executable (glx-swap-event glx-swap-event.c)
add_executable (glx-make-glxdrawable-current glx-make-glxdrawable-current.c)
target_link_libraries(glx-multithread pthread X11)
+ add_executable (glx-multithread-makecurrent-1 glx-multithread-makecurrent-1.c)
+ target_link_libraries(glx-multithread-makecurrent-1 pthread)
+ add_executable (glx-multithread-makecurrent-2 glx-multithread-makecurrent-2.c)
+ target_link_libraries(glx-multithread-makecurrent-2 pthread)
+ add_executable (glx-multithread-makecurrent-3 glx-multithread-makecurrent-3.c)
+ target_link_libraries(glx-multithread-makecurrent-3 pthread)
+ add_executable (glx-multithread-makecurrent-4 glx-multithread-makecurrent-4.c)
+ target_link_libraries(glx-multithread-makecurrent-4 pthread)
add_executable (glx-swap-exchange glx-swap-exchange.c)
target_link_libraries(glx-swap-exchange X11)
add_executable (glx-tfp glx-tfp.c)
diff --git a/tests/glx/glx-multithread-makecurrent-1.c b/tests/glx/glx-multithread-makecurrent-1.c
new file mode 100644
index 0000000..aa5b5a1
--- /dev/null
+++ b/tests/glx/glx-multithread-makecurrent-1.c
@@ -0,0 +1,183 @@
+/*
+ * Copyright © 2009-2011 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.
+ */
+
+/** @file glx-multithread-makecurrent-1.c
+ *
+ * First test of GLX_MESA_multithread_makecurrent: Bind one context into
+ * multiple threads and make sure that synchronized rendering from both
+ * threads works correctly.
+ */
+
+#include <unistd.h>
+#include "piglit-util.h"
+#include "piglit-glx-util.h"
+#include "pthread.h"
+
+int piglit_width = 70, piglit_height = 30;
+static Display *dpy;
+static Window win;
+static GLXContext ctx;
+static pthread_mutex_t mutex;
+static XVisualInfo *visinfo;
+static int current_step = 1;
+
+static void
+get_lock_for_step(int step)
+{
+ while (true) {
+ pthread_mutex_lock(&mutex);
+ if (step == current_step) {
+ current_step++;
+ return;
+ }
+ pthread_mutex_unlock(&mutex);
+ usleep(1);
+ }
+}
+
+static void *
+thread1_func(void *arg)
+{
+ get_lock_for_step(1);
+ glXMakeCurrent(dpy, win, ctx);
+ pthread_mutex_unlock(&mutex);
+
+ get_lock_for_step(3);
+ glColor4f(0.0, 1.0, 0.0, 0.0);
+ piglit_draw_rect(10, 10, 10, 10);
+ pthread_mutex_unlock(&mutex);
+
+ get_lock_for_step(5);
+ glXMakeCurrent(dpy, None, None);
+ pthread_mutex_unlock(&mutex);
+
+ return NULL;
+}
+
+static void *
+thread2_func(void *arg)
+{
+ get_lock_for_step(2);
+ glXMakeCurrent(dpy, win, ctx);
+ pthread_mutex_unlock(&mutex);
+
+ get_lock_for_step(4);
+ glColor4f(0.0, 0.0, 1.0, 0.0);
+ piglit_draw_rect(30, 10, 10, 10);
+ pthread_mutex_unlock(&mutex);
+
+ get_lock_for_step(6);
+ glXMakeCurrent(dpy, None, None);
+ pthread_mutex_unlock(&mutex);
+
+ return NULL;
+}
+
+enum piglit_result
+draw(Display *dpy)
+{
+ GLboolean pass = GL_TRUE;
+ float green[] = {0.0, 1.0, 0.0, 1.0};
+ float blue[] = {0.0, 0.0, 1.0, 1.0};
+ float gray[] = {0.5, 0.5, 0.5, 1.0};
+ pthread_t thread1, thread2;
+ void *retval;
+ int ret;
+ int x1 = 10, x2 = 30;
+
+ ctx = piglit_get_glx_context(dpy, visinfo);
+ glXMakeCurrent(dpy, win, ctx);
+
+ piglit_require_glx_extension(dpy, "MESA_multithread_makecurrent");
+
+ /* Clear background to gray */
+ glClearColor(0.5, 0.5, 0.5, 1.0);
+ glClear(GL_COLOR_BUFFER_BIT);
+
+ piglit_ortho_projection(piglit_width, piglit_height, GL_FALSE);
+
+ pthread_mutex_init(&mutex, NULL);
+
+ /* Now, spawn some threads that do some drawing, both with this
+ * context
+ */
+ pthread_create(&thread1, NULL, thread1_func, &x1);
+ pthread_create(&thread2, NULL, thread2_func, &x2);
+
+ ret = pthread_join(thread1, &retval);
+ assert(ret == 0);
+ ret = pthread_join(thread2, &retval);
+ assert(ret == 0);
+
+ pthread_mutex_destroy(&mutex);
+
+ glColor4f(0.0, 1.0, 0.0, 0.0);
+ piglit_draw_rect(50, 10, 10, 10);
+
+ pass &= piglit_probe_rect_rgba( 0, 10, 10, 10, gray);
+ pass &= piglit_probe_rect_rgba(10, 10, 10, 10, green);
+ pass &= piglit_probe_rect_rgba(20, 10, 10, 10, gray);
+ pass &= piglit_probe_rect_rgba(30, 10, 10, 10, blue);
+ pass &= piglit_probe_rect_rgba(40, 10, 10, 10, gray);
+ pass &= piglit_probe_rect_rgba(50, 10, 10, 10, green);
+ pass &= piglit_probe_rect_rgba(60, 10, 10, 10, gray);
+
+ pass &= piglit_probe_rect_rgba(0, 0, piglit_width, 10, gray);
+ pass &= piglit_probe_rect_rgba(0, 20, piglit_width, 10, gray);
+
+ glXSwapBuffers(dpy, win);
+
+ glXMakeCurrent(dpy, None, None);
+ glXDestroyContext(dpy, ctx);
+
+ return pass ? PIGLIT_SUCCESS : PIGLIT_FAILURE;
+}
+
+int
+main(int argc, char **argv)
+{
+ int i;
+
+ for(i = 1; i < argc; ++i) {
+ if (!strcmp(argv[i], "-auto"))
+ piglit_automatic = 1;
+ else
+ fprintf(stderr, "Unknown option: %s\n", argv[i]);
+ }
+
+ dpy = XOpenDisplay(NULL);
+ if (dpy == NULL) {
+ fprintf(stderr, "couldn't open display\n");
+ piglit_report_result(PIGLIT_FAILURE);
+ }
+ visinfo = piglit_get_glx_visual(dpy);
+ win = piglit_get_glx_window(dpy, visinfo);
+
+ XFree(visinfo);
+
+ XMapWindow(dpy, win);
+
+ piglit_glx_event_loop(dpy, draw);
+
+ return 0;
+}
diff --git a/tests/glx/glx-multithread-makecurrent-2.c b/tests/glx/glx-multithread-makecurrent-2.c
new file mode 100644
index 0000000..ae3276c
--- /dev/null
+++ b/tests/glx/glx-multithread-makecurrent-2.c
@@ -0,0 +1,183 @@
+/*
+ * Copyright © 2009-2011 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.
+ */
+
+/** @file glx-multithread-makecurrent-2.c
+ *
+ * Test of issue #1 of GLX_MESA_multithread_makecurrent: Bind one
+ * context into multiple threads, do rendering from both, then make
+ * sure that once we bind a new context to one thread that rendering
+ * from the second can continue.
+ */
+
+#include <unistd.h>
+#include "piglit-util.h"
+#include "piglit-glx-util.h"
+#include "pthread.h"
+
+int piglit_width = 70, piglit_height = 30;
+static Display *dpy;
+static Window win;
+static GLXContext ctx;
+static pthread_mutex_t mutex;
+static XVisualInfo *visinfo;
+static int current_step = 1;
+
+static void
+get_lock_for_step(int step)
+{
+ while (true) {
+ pthread_mutex_lock(&mutex);
+ if (step == current_step) {
+ current_step++;
+ return;
+ }
+ pthread_mutex_unlock(&mutex);
+ usleep(1);
+ }
+}
+
+static void *
+thread1_func(void *arg)
+{
+ GLXContext new_ctx;
+
+ get_lock_for_step(1);
+ glXMakeCurrent(dpy, win, ctx);
+ glColor4f(0.0, 1.0, 0.0, 0.0);
+ piglit_draw_rect(10, 10, 10, 10);
+ pthread_mutex_unlock(&mutex);
+
+ get_lock_for_step(3);
+ new_ctx = piglit_get_glx_context(dpy, visinfo);
+ glXMakeCurrent(dpy, win, new_ctx);
+ pthread_mutex_unlock(&mutex);
+
+ get_lock_for_step(5);
+ glXMakeCurrent(dpy, None, None);
+ glXDestroyContext(dpy, new_ctx);
+ pthread_mutex_unlock(&mutex);
+
+ return NULL;
+}
+
+static void *
+thread2_func(void *arg)
+{
+ get_lock_for_step(2);
+ glXMakeCurrent(dpy, win, ctx);
+ glColor4f(0.0, 1.0, 0.0, 0.0);
+ piglit_draw_rect(30, 10, 10, 10);
+ pthread_mutex_unlock(&mutex);
+
+ get_lock_for_step(4);
+ piglit_draw_rect(50, 10, 10, 10);
+ pthread_mutex_unlock(&mutex);
+
+ return NULL;
+}
+
+enum piglit_result
+draw(Display *dpy)
+{
+ GLboolean pass = GL_TRUE;
+ float green[] = {0.0, 1.0, 0.0, 1.0};
+ float gray[] = {0.5, 0.5, 0.5, 1.0};
+ pthread_t thread1, thread2;
+ void *retval;
+ int ret;
+
+ ctx = piglit_get_glx_context(dpy, visinfo);
+ glXMakeCurrent(dpy, win, ctx);
+
+ piglit_require_glx_extension(dpy, "MESA_multithread_makecurrent");
+
+ /* Clear background to gray */
+ glClearColor(0.5, 0.5, 0.5, 1.0);
+ glClear(GL_COLOR_BUFFER_BIT);
+
+ piglit_ortho_projection(piglit_width, piglit_height, GL_FALSE);
+
+ pthread_mutex_init(&mutex, NULL);
+
+ glFlush();
+ /* Now, spawn some threads that do some drawing, both with this
+ * context
+ */
+ pthread_create(&thread1, NULL, thread1_func, NULL);
+ pthread_create(&thread2, NULL, thread2_func, NULL);
+
+ ret = pthread_join(thread1, &retval);
+ assert(ret == 0);
+ ret = pthread_join(thread2, &retval);
+ assert(ret == 0);
+
+ pthread_mutex_destroy(&mutex);
+ glFlush();
+
+ pass &= piglit_probe_rect_rgba( 0, 10, 10, 10, gray);
+ pass &= piglit_probe_rect_rgba(10, 10, 10, 10, green);
+ pass &= piglit_probe_rect_rgba(20, 10, 10, 10, gray);
+ pass &= piglit_probe_rect_rgba(30, 10, 10, 10, green);
+ pass &= piglit_probe_rect_rgba(40, 10, 10, 10, gray);
+ pass &= piglit_probe_rect_rgba(50, 10, 10, 10, green);
+ pass &= piglit_probe_rect_rgba(60, 10, 10, 10, gray);
+
+ pass &= piglit_probe_rect_rgba(0, 0, piglit_width, 10, gray);
+ pass &= piglit_probe_rect_rgba(0, 20, piglit_width, 10, gray);
+
+ glXSwapBuffers(dpy, win);
+
+ glXMakeCurrent(dpy, None, None);
+ glXDestroyContext(dpy, ctx);
+
+ return pass ? PIGLIT_SUCCESS : PIGLIT_FAILURE;
+}
+
+int
+main(int argc, char **argv)
+{
+ int i;
+
+ for(i = 1; i < argc; ++i) {
+ if (!strcmp(argv[i], "-auto"))
+ piglit_automatic = 1;
+ else
+ fprintf(stderr, "Unknown option: %s\n", argv[i]);
+ }
+
+ dpy = XOpenDisplay(NULL);
+ if (dpy == NULL) {
+ fprintf(stderr, "couldn't open display\n");
+ piglit_report_result(PIGLIT_FAILURE);
+ }
+ visinfo = piglit_get_glx_visual(dpy);
+ win = piglit_get_glx_window(dpy, visinfo);
+
+ XMapWindow(dpy, win);
+
+ piglit_glx_event_loop(dpy, draw);
+
+ XFree(visinfo);
+
+ return 0;
+}
diff --git a/tests/glx/glx-multithread-makecurrent-3.c b/tests/glx/glx-multithread-makecurrent-3.c
new file mode 100644
index 0000000..c1f5540
--- /dev/null
+++ b/tests/glx/glx-multithread-makecurrent-3.c
@@ -0,0 +1,175 @@
+/*
+ * Copyright © 2009-2011 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.
+ */
+
+/** @file glx-multithread-makecurrent-3.c
+ *
+ * Test of issue #2 of GLX_MESA_multithread_makecurrent: Bind one
+ * context into multiple threads, do rendering from both, then make
+ * sure that once we unbind from one that rendering from the second
+ * can continue.
+ */
+
+#include <unistd.h>
+#include "piglit-util.h"
+#include "piglit-glx-util.h"
+#include "pthread.h"
+
+int piglit_width = 70, piglit_height = 30;
+static Display *dpy;
+static Window win;
+static GLXContext ctx;
+static pthread_mutex_t mutex;
+static XVisualInfo *visinfo;
+static int current_step = 1;
+
+static void
+get_lock_for_step(int step)
+{
+ while (true) {
+ pthread_mutex_lock(&mutex);
+ if (step == current_step) {
+ current_step++;
+ return;
+ }
+ pthread_mutex_unlock(&mutex);
+ usleep(1);
+ }
+}
+
+static void *
+thread1_func(void *arg)
+{
+ get_lock_for_step(1);
+ glXMakeCurrent(dpy, win, ctx);
+ glColor4f(0.0, 1.0, 0.0, 0.0);
+ piglit_draw_rect(10, 10, 10, 10);
+ pthread_mutex_unlock(&mutex);
+
+ get_lock_for_step(3);
+ glXMakeCurrent(dpy, None, None);
+ pthread_mutex_unlock(&mutex);
+
+ return NULL;
+}
+
+static void *
+thread2_func(void *arg)
+{
+ get_lock_for_step(2);
+ glXMakeCurrent(dpy, win, ctx);
+ glColor4f(0.0, 1.0, 0.0, 0.0);
+ piglit_draw_rect(30, 10, 10, 10);
+ pthread_mutex_unlock(&mutex);
+
+ get_lock_for_step(4);
+ piglit_draw_rect(50, 10, 10, 10);
+ pthread_mutex_unlock(&mutex);
+
+ return NULL;
+}
+
+enum piglit_result
+draw(Display *dpy)
+{
+ GLboolean pass = GL_TRUE;
+ float green[] = {0.0, 1.0, 0.0, 1.0};
+ float gray[] = {0.5, 0.5, 0.5, 1.0};
+ pthread_t thread1, thread2;
+ void *retval;
+ int ret;
+
+ ctx = piglit_get_glx_context(dpy, visinfo);
+ glXMakeCurrent(dpy, win, ctx);
+
+ piglit_require_glx_extension(dpy, "MESA_multithread_makecurrent");
+
+ /* Clear background to gray */
+ glClearColor(0.5, 0.5, 0.5, 1.0);
+ glClear(GL_COLOR_BUFFER_BIT);
+
+ piglit_ortho_projection(piglit_width, piglit_height, GL_FALSE);
+
+ pthread_mutex_init(&mutex, NULL);
+
+ glFlush();
+ /* Now, spawn some threads that do some drawing, both with this
+ * context
+ */
+ pthread_create(&thread1, NULL, thread1_func, NULL);
+ pthread_create(&thread2, NULL, thread2_func, NULL);
+
+ ret = pthread_join(thread1, &retval);
+ assert(ret == 0);
+ ret = pthread_join(thread2, &retval);
+ assert(ret == 0);
+
+ pthread_mutex_destroy(&mutex);
+ glFlush();
+
+ pass &= piglit_probe_rect_rgba( 0, 10, 10, 10, gray);
+ pass &= piglit_probe_rect_rgba(10, 10, 10, 10, green);
+ pass &= piglit_probe_rect_rgba(20, 10, 10, 10, gray);
+ pass &= piglit_probe_rect_rgba(30, 10, 10, 10, green);
+ pass &= piglit_probe_rect_rgba(40, 10, 10, 10, gray);
+ pass &= piglit_probe_rect_rgba(50, 10, 10, 10, green);
+ pass &= piglit_probe_rect_rgba(60, 10, 10, 10, gray);
+
+ pass &= piglit_probe_rect_rgba(0, 0, piglit_width, 10, gray);
+ pass &= piglit_probe_rect_rgba(0, 20, piglit_width, 10, gray);
+
+ glXSwapBuffers(dpy, win);
+
+ glXMakeCurrent(dpy, None, None);
+ glXDestroyContext(dpy, ctx);
+
+ return pass ? PIGLIT_SUCCESS : PIGLIT_FAILURE;
+}
+
+int
+main(int argc, char **argv)
+{
+ int i;
+
+ for(i = 1; i < argc; ++i) {
+ if (!strcmp(argv[i], "-auto"))
+ piglit_automatic = 1;
+ else
+ fprintf(stderr, "Unknown option: %s\n", argv[i]);
+ }
+
+ dpy = XOpenDisplay(NULL);
+ if (dpy == NULL) {
+ fprintf(stderr, "couldn't open display\n");
+ piglit_report_result(PIGLIT_FAILURE);
+ }
+ visinfo = piglit_get_glx_visual(dpy);
+ win = piglit_get_glx_window(dpy, visinfo);
+
+ XFree(visinfo);
+
+ XMapWindow(dpy, win);
+
+ piglit_glx_event_loop(dpy, draw);
+
+ return 0;
+}
diff --git a/tests/glx/glx-multithread-makecurrent-4.c b/tests/glx/glx-multithread-makecurrent-4.c
new file mode 100644
index 0000000..2118271
--- /dev/null
+++ b/tests/glx/glx-multithread-makecurrent-4.c
@@ -0,0 +1,174 @@
+/*
+ * Copyright © 2009-2011 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.
+ */
+
+/** @file glx-multithread-makecurrent-4.c
+ *
+ * Test of issue #3 of GLX_MESA_multithread_makecurrent: Bind one
+ * context into multiple threads, do rendering from both, then destroy
+ * the context in one and keep using it to render from the other.
+ */
+
+#include <unistd.h>
+#include "piglit-util.h"
+#include "piglit-glx-util.h"
+#include "pthread.h"
+
+int piglit_width = 70, piglit_height = 30;
+static Display *dpy;
+static Window win;
+static GLXContext ctx;
+static pthread_mutex_t mutex;
+static XVisualInfo *visinfo;
+static int current_step = 1;
+
+static void
+get_lock_for_step(int step)
+{
+ while (true) {
+ pthread_mutex_lock(&mutex);
+ if (step == current_step) {
+ current_step++;
+ return;
+ }
+ pthread_mutex_unlock(&mutex);
+ usleep(1);
+ }
+}
+
+static void *
+thread1_func(void *arg)
+{
+ get_lock_for_step(1);
+ glXMakeCurrent(dpy, win, ctx);
+ glColor4f(0.0, 1.0, 0.0, 0.0);
+ piglit_draw_rect(10, 10, 10, 10);
+ pthread_mutex_unlock(&mutex);
+
+ get_lock_for_step(3);
+ glXMakeCurrent(dpy, None, None);
+ glXDestroyContext(dpy, ctx);
+ pthread_mutex_unlock(&mutex);
+
+ return NULL;
+}
+
+static void *
+thread2_func(void *arg)
+{
+ get_lock_for_step(2);
+ glXMakeCurrent(dpy, win, ctx);
+ glColor4f(0.0, 1.0, 0.0, 0.0);
+ piglit_draw_rect(30, 10, 10, 10);
+ pthread_mutex_unlock(&mutex);
+
+ get_lock_for_step(4);
+ piglit_draw_rect(50, 10, 10, 10);
+ pthread_mutex_unlock(&mutex);
+
+ return NULL;
+}
+
+enum piglit_result
+draw(Display *dpy)
+{
+ GLboolean pass = GL_TRUE;
+ float green[] = {0.0, 1.0, 0.0, 1.0};
+ float gray[] = {0.5, 0.5, 0.5, 1.0};
+ pthread_t thread1, thread2;
+ void *retval;
+ int ret;
+
+ ctx = piglit_get_glx_context(dpy, visinfo);
+ glXMakeCurrent(dpy, win, ctx);
+
+ piglit_require_glx_extension(dpy, "MESA_multithread_makecurrent");
+
+ /* Clear background to gray */
+ glClearColor(0.5, 0.5, 0.5, 1.0);
+ glClear(GL_COLOR_BUFFER_BIT);
+
+ piglit_ortho_projection(piglit_width, piglit_height, GL_FALSE);
+
+ pthread_mutex_init(&mutex, NULL);
+
+ glFlush();
+ /* Now, spawn some threads that do some drawing, both with this
+ * context
+ */
+ pthread_create(&thread1, NULL, thread1_func, NULL);
+ pthread_create(&thread2, NULL, thread2_func, NULL);
+
+ ret = pthread_join(thread1, &retval);
+ assert(ret == 0);
+ ret = pthread_join(thread2, &retval);
+ assert(ret == 0);
+
+ pthread_mutex_destroy(&mutex);
+ glFlush();
+
+ pass &= piglit_probe_rect_rgba( 0, 10, 10, 10, gray);
+ pass &= piglit_probe_rect_rgba(10, 10, 10, 10, green);
+ pass &= piglit_probe_rect_rgba(20, 10, 10, 10, gray);
+ pass &= piglit_probe_rect_rgba(30, 10, 10, 10, green);
+ pass &= piglit_probe_rect_rgba(40, 10, 10, 10, gray);
+ pass &= piglit_probe_rect_rgba(50, 10, 10, 10, green);
+ pass &= piglit_probe_rect_rgba(60, 10, 10, 10, gray);
+
+ pass &= piglit_probe_rect_rgba(0, 0, piglit_width, 10, gray);
+ pass &= piglit_probe_rect_rgba(0, 20, piglit_width, 10, gray);
+
+ glXSwapBuffers(dpy, win);
+
+ glXMakeCurrent(dpy, None, None);
+
+ return pass ? PIGLIT_SUCCESS : PIGLIT_FAILURE;
+}
+
+int
+main(int argc, char **argv)
+{
+ int i;
+
+ for(i = 1; i < argc; ++i) {
+ if (!strcmp(argv[i], "-auto"))
+ piglit_automatic = 1;
+ else
+ fprintf(stderr, "Unknown option: %s\n", argv[i]);
+ }
+
+ dpy = XOpenDisplay(NULL);
+ if (dpy == NULL) {
+ fprintf(stderr, "couldn't open display\n");
+ piglit_report_result(PIGLIT_FAILURE);
+ }
+ visinfo = piglit_get_glx_visual(dpy);
+ win = piglit_get_glx_window(dpy, visinfo);
+
+ XFree(visinfo);
+
+ XMapWindow(dpy, win);
+
+ piglit_glx_event_loop(dpy, draw);
+
+ return 0;
+}
--
1.7.4.1
More information about the cairo
mailing list