r300g: fix a possible race when counting contexts
authorMarek Olšák <maraeo@gmail.com>
Fri, 18 Feb 2011 23:16:44 +0000 (00:16 +0100)
committerMarek Olšák <maraeo@gmail.com>
Fri, 18 Feb 2011 23:17:27 +0000 (00:17 +0100)
Atomics aren't sufficient here.

src/gallium/drivers/r300/r300_context.c
src/gallium/drivers/r300/r300_screen.c
src/gallium/drivers/r300/r300_screen.h

index a89bf7fac3176f138ceb18c3fd9c881f97b5a332..da6b0bb8aa7d02fb9835db7b4d488068449392b2 100644 (file)
 static void r300_update_num_contexts(struct r300_screen *r300screen,
                                      int diff)
 {
+    pipe_mutex_lock(r300screen->num_contexts_mutex);
     if (diff > 0) {
-        p_atomic_inc(&r300screen->num_contexts);
+        r300screen->num_contexts++;
 
         if (r300screen->num_contexts > 1)
             util_slab_set_thread_safety(&r300screen->pool_buffers,
                                         UTIL_SLAB_MULTITHREADED);
     } else {
-        p_atomic_dec(&r300screen->num_contexts);
+        r300screen->num_contexts--;
 
         if (r300screen->num_contexts <= 1)
             util_slab_set_thread_safety(&r300screen->pool_buffers,
                                         UTIL_SLAB_SINGLETHREADED);
     }
+    pipe_mutex_unlock(r300screen->num_contexts_mutex);
 }
 
 static void r300_release_referenced_objects(struct r300_context *r300)
index ed47315f42da2b12a1f707a675a6a7a55832c17d..77a9c6ad86f88db92a8cbecf77b1db5329b469a4 100644 (file)
@@ -398,6 +398,7 @@ static void r300_destroy_screen(struct pipe_screen* pscreen)
     struct r300_winsys_screen *rws = r300_winsys_screen(pscreen);
 
     util_slab_destroy(&r300screen->pool_buffers);
+    pipe_mutex_destroy(r300screen->num_contexts_mutex);
 
     if (rws)
       rws->destroy(rws);
@@ -459,6 +460,8 @@ struct pipe_screen* r300_screen_create(struct r300_winsys_screen *rws)
             r300screen->caps.is_r500 &&
             rws->get_value(rws, R300_VID_DRM_2_3_0);
 
+    pipe_mutex_init(r300screen->num_contexts_mutex);
+
     util_slab_create(&r300screen->pool_buffers,
                      sizeof(struct r300_resource), 64,
                      UTIL_SLAB_SINGLETHREADED);
index c935f55ccbfb414f759f9548d4fadd5892261326..576f9c1f4a96e65595b3ba35048b21fee03bb2d9 100644 (file)
@@ -52,6 +52,7 @@ struct r300_screen {
     /* The number of created contexts to know whether we have multiple
      * contexts or not. */
     int num_contexts;
+    pipe_mutex num_contexts_mutex;
 };