]> git.sesse.net Git - vlc/blob - src/misc/atomic.c
LGPL
[vlc] / src / misc / atomic.c
1 /*****************************************************************************
2  * atomic.c:
3  *****************************************************************************
4  * Copyright (C) 2010 RĂ©mi Denis-Courmont
5  *
6  * This program is free software; you can redistribute it and/or modify it
7  * under the terms of the GNU Lesser General Public License as published by
8  * the Free Software Foundation; either version 2.1 of the License, or
9  * (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14  * GNU Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public License
17  * along with this program; if not, write to the Free Software Foundation,
18  * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
19  *****************************************************************************/
20
21 #ifdef HAVE_CONFIG_H
22 # include <config.h>
23 #endif
24
25 #include <vlc_common.h>
26 #include <vlc_atomic.h>
27
28 #if defined (__GCC_HAVE_SYNC_COMPARE_AND_SWAP_4)
29 /* GCC intrinsics */
30
31 uintptr_t vlc_atomic_get (const vlc_atomic_t *atom)
32 {
33     __sync_synchronize ();
34     return atom->u;
35 }
36
37 uintptr_t vlc_atomic_set (vlc_atomic_t *atom, uintptr_t v)
38 {
39     atom->u = v;
40     __sync_synchronize ();
41     return v;
42 }
43
44 uintptr_t vlc_atomic_add (vlc_atomic_t *atom, uintptr_t v)
45 {
46     return __sync_add_and_fetch (&atom->u, v);
47 }
48
49 uintptr_t vlc_atomic_swap (vlc_atomic_t *atom, uintptr_t v)
50 {
51     /* grmbl, gcc does not provide an intrinsic for this! */
52     uintptr_t u;
53
54     do
55         u = vlc_atomic_get (atom);
56     while (vlc_atomic_compare_swap (atom, u, v) != u);
57
58     return u;
59 }
60
61 uintptr_t vlc_atomic_compare_swap (vlc_atomic_t *atom,
62                                    uintptr_t oldval, uintptr_t newval)
63 {
64     return __sync_val_compare_and_swap (&atom->u, oldval, newval);
65 }
66
67 #else
68 /* Worst-case fallback implementation with a mutex */
69
70 static vlc_mutex_t lock = VLC_STATIC_MUTEX;
71
72 uintptr_t vlc_atomic_get (const vlc_atomic_t *atom)
73 {
74     uintptr_t v;
75
76     vlc_mutex_lock (&lock);
77     v = atom->u;
78     vlc_mutex_unlock (&lock);
79     return v;
80 }
81
82 uintptr_t vlc_atomic_set (vlc_atomic_t *atom, uintptr_t v)
83 {
84     vlc_mutex_lock (&lock);
85     atom->u = v;
86     vlc_mutex_unlock (&lock);
87     return v;
88 }
89
90 uintptr_t vlc_atomic_add (vlc_atomic_t *atom, uintptr_t v)
91 {
92     vlc_mutex_lock (&lock);
93     atom->u += v;
94     v = atom->u;
95     vlc_mutex_unlock (&lock);
96     return v;
97 }
98
99 uintptr_t vlc_atomic_swap (vlc_atomic_t *atom, uintptr_t v)
100 {
101     uintptr_t u;
102
103     vlc_mutex_lock (&lock);
104     u = atom->u;
105     atom->u = v;
106     vlc_mutex_unlock (&lock);
107
108     return u;
109 }
110
111 uintptr_t vlc_atomic_compare_swap (vlc_atomic_t *atom,
112                                        uintptr_t oldval, uintptr_t newval)
113 {
114     uintptr_t u;
115
116     vlc_mutex_lock (&lock);
117     u = atom->u;
118     if (u == oldval)
119         atom->u = newval;
120     vlc_mutex_unlock (&lock);
121
122     return u;
123 }
124
125 #endif