]> git.sesse.net Git - vlc/commitdiff
Add atomic swap and compare-and-swap
authorRémi Denis-Courmont <remi@remlab.net>
Wed, 13 Oct 2010 15:57:58 +0000 (18:57 +0300)
committerRémi Denis-Courmont <remi@remlab.net>
Wed, 13 Oct 2010 16:20:29 +0000 (19:20 +0300)
include/vlc_atomic.h
src/libvlccore.sym
src/misc/atomic.c
src/win32/atomic.c

index 51acccc726e3306ea33dff71a008b1447bed6f2f..770e47aae6ba373f59e3bc68450c61b0edf958e6 100644 (file)
@@ -47,4 +47,7 @@ static inline uintptr_t vlc_atomic_dec (vlc_atomic_t *atom)
     return vlc_atomic_sub (atom, 1);
 }
 
+VLC_EXPORT(uintptr_t, vlc_atomic_swap, (vlc_atomic_t *, uintptr_t));
+VLC_EXPORT(uintptr_t, vlc_atomic_compare_swap, (vlc_atomic_t *, uintptr_t, uintptr_t));
+
 #endif
index 413ab79bc4bc15a70f98d097d23884291fc37cb8..5632223aeb4c122f6c9c1b7d327f8e97df57f369 100644 (file)
@@ -503,6 +503,8 @@ VLC_Compiler
 vlc_atomic_get
 vlc_atomic_set
 vlc_atomic_add
+vlc_atomic_swap
+vlc_atomic_compare_swap
 vlc_cond_broadcast
 vlc_cond_destroy
 vlc_cond_init
index 1f97f599e1578ed0811484ebe3158023a1beab11..e0319f51e1c95f445927f2d7908417ab94a74ce3 100644 (file)
@@ -46,6 +46,24 @@ uintptr_t vlc_atomic_add (vlc_atomic_t *atom, uintptr_t v)
     return __sync_add_and_fetch (&atom->u, v);
 }
 
+uintptr_t vlc_atomic_swap (vlc_atomic_t *atom, uintptr_t v)
+{
+    /* grmbl, gcc does not provide an intrinsic for this! */
+    uintptr_t u;
+
+    do
+        u = vlc_atomic_get (atom);
+    while (vlc_atomic_compare_swap (atom, u, v) != u);
+
+    return u;
+}
+
+uintptr_t vlc_atomic_compare_swap (vlc_atomic_t *atom,
+                                   uintptr_t oldval, uintptr_t newval)
+{
+    return __sync_val_compare_and_swap (&atom->u, oldval, newval);
+}
+
 #else
 /* Worst-case fallback implementation with a mutex */
 
@@ -78,4 +96,30 @@ uintptr_t vlc_atomic_add (vlc_atomic_t *atom, uintptr_t v)
     return v;
 }
 
+uintptr_t vlc_atomic_swap (vlc_atomic_t *atom, uintptr_t v)
+{
+    uintptr_t u;
+
+    vlc_mutex_lock (&lock);
+    u = atom->u;
+    atom->u = v;
+    vlc_mutex_unlock (&lock);
+
+    return u;
+}
+
+uintptr_t vlc_atomic_compare_and_swap (vlc_atomic_t *atom,
+                                       uintptr_t oldval, uintptr_t newval)
+{
+    uintptr_t u;
+
+    vlc_mutex_lock (&lock);
+    u = atom->u;
+    if (u == oldval)
+        atom->u = newval;
+    vlc_mutex_unlock (&lock);
+
+    return u;
+}
+
 #endif
index abadf894ff956e789f3203151284d24a74c67dda..14687e665f11116160535b041eaf2399a83cb2e8 100644 (file)
@@ -50,3 +50,22 @@ uintptr_t vlc_atomic_add (vlc_atomic_t *atom, uintptr_t v)
     return InterlockedExchangeAdd (&atom->s, v) + v;
 #endif
 }
+
+uintptr_t vlc_atomic_swap (vlc_atomic_t *atom, uintptr_t v)
+{
+#if defined (WIN64)
+    return InterlockedExchange64 (&atom->s, v);
+#else
+    return InterlockedExchange (&atom->s, v);
+#endif
+}
+
+uintptr_t vlc_atomic_compare_swap (vlc_atomic_t *atom,
+                                   uintptr_t oldval, uintptr_t newval)
+{
+#if defined (WIN64)
+    return InterlockedCompareExchange64 (&atom->s, newval, oldval);
+#else
+    return InterlockedCompareExchange (&atom->s, newval, oldval);
+#endif
+}