blob: 08b607969a16978055f2b5e5cd1bf833fb8a4618 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
Ralf Baechle70342282013-01-22 12:59:30 +01002 * Atomic operations that C can't guarantee us. Useful for
Linus Torvalds1da177e2005-04-16 15:20:36 -07003 * resource counting etc..
4 *
5 * But use these as seldom as possible since they are much more slower
6 * than regular operations.
7 *
8 * This file is subject to the terms and conditions of the GNU General Public
9 * License. See the file "COPYING" in the main directory of this archive
10 * for more details.
11 *
Ralf Baechlee303e082006-11-30 01:14:50 +000012 * Copyright (C) 1996, 97, 99, 2000, 03, 04, 06 by Ralf Baechle
Linus Torvalds1da177e2005-04-16 15:20:36 -070013 */
Linus Torvalds1da177e2005-04-16 15:20:36 -070014#ifndef _ASM_ATOMIC_H
15#define _ASM_ATOMIC_H
16
Ralf Baechle192ef362006-07-07 14:07:18 +010017#include <linux/irqflags.h>
Matthew Wilcoxea4354672009-01-06 14:40:39 -080018#include <linux/types.h>
Ralf Baechle0004a9d2006-10-31 03:45:07 +000019#include <asm/barrier.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070020#include <asm/cpu-features.h>
David Howellsb81947c2012-03-28 18:30:02 +010021#include <asm/cmpxchg.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070022#include <asm/war.h>
23
Ralf Baechle70342282013-01-22 12:59:30 +010024#define ATOMIC_INIT(i) { (i) }
Linus Torvalds1da177e2005-04-16 15:20:36 -070025
26/*
27 * atomic_read - read atomic variable
28 * @v: pointer of type atomic_t
29 *
30 * Atomically reads the value of @v.
31 */
Anton Blanchardf3d46f92010-05-17 14:33:53 +100032#define atomic_read(v) (*(volatile int *)&(v)->counter)
Linus Torvalds1da177e2005-04-16 15:20:36 -070033
34/*
35 * atomic_set - set atomic variable
36 * @v: pointer of type atomic_t
37 * @i: required value
38 *
39 * Atomically sets the value of @v to @i.
40 */
Ralf Baechle21a151d2007-10-11 23:46:15 +010041#define atomic_set(v, i) ((v)->counter = (i))
Linus Torvalds1da177e2005-04-16 15:20:36 -070042
43/*
44 * atomic_add - add integer to atomic variable
45 * @i: integer value to add
46 * @v: pointer of type atomic_t
47 *
48 * Atomically adds @i to @v.
49 */
50static __inline__ void atomic_add(int i, atomic_t * v)
51{
David Daneyb791d112009-07-13 11:15:19 -070052 if (kernel_uses_llsc && R10000_LLSC_WAR) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +000053 int temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -070054
55 __asm__ __volatile__(
Maciej W. Rozyckic4559f62005-06-23 15:57:15 +000056 " .set mips3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -070057 "1: ll %0, %1 # atomic_add \n"
58 " addu %0, %2 \n"
59 " sc %0, %1 \n"
60 " beqzl %0, 1b \n"
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +000061 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -040062 : "=&r" (temp), "+m" (v->counter)
63 : "Ir" (i));
David Daneyb791d112009-07-13 11:15:19 -070064 } else if (kernel_uses_llsc) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +000065 int temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -070066
Ralf Baechle78373142010-10-29 19:08:24 +010067 do {
68 __asm__ __volatile__(
69 " .set mips3 \n"
70 " ll %0, %1 # atomic_add \n"
71 " addu %0, %2 \n"
72 " sc %0, %1 \n"
73 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -040074 : "=&r" (temp), "+m" (v->counter)
75 : "Ir" (i));
Ralf Baechle78373142010-10-29 19:08:24 +010076 } while (unlikely(!temp));
Linus Torvalds1da177e2005-04-16 15:20:36 -070077 } else {
78 unsigned long flags;
79
Ralf Baechle49edd092007-03-16 16:10:36 +000080 raw_local_irq_save(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -070081 v->counter += i;
Ralf Baechle49edd092007-03-16 16:10:36 +000082 raw_local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -070083 }
84}
85
86/*
87 * atomic_sub - subtract the atomic variable
88 * @i: integer value to subtract
89 * @v: pointer of type atomic_t
90 *
91 * Atomically subtracts @i from @v.
92 */
93static __inline__ void atomic_sub(int i, atomic_t * v)
94{
David Daneyb791d112009-07-13 11:15:19 -070095 if (kernel_uses_llsc && R10000_LLSC_WAR) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +000096 int temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -070097
98 __asm__ __volatile__(
Maciej W. Rozyckic4559f62005-06-23 15:57:15 +000099 " .set mips3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700100 "1: ll %0, %1 # atomic_sub \n"
101 " subu %0, %2 \n"
102 " sc %0, %1 \n"
103 " beqzl %0, 1b \n"
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000104 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400105 : "=&r" (temp), "+m" (v->counter)
106 : "Ir" (i));
David Daneyb791d112009-07-13 11:15:19 -0700107 } else if (kernel_uses_llsc) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000108 int temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700109
Ralf Baechle78373142010-10-29 19:08:24 +0100110 do {
111 __asm__ __volatile__(
112 " .set mips3 \n"
113 " ll %0, %1 # atomic_sub \n"
114 " subu %0, %2 \n"
115 " sc %0, %1 \n"
116 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400117 : "=&r" (temp), "+m" (v->counter)
118 : "Ir" (i));
Ralf Baechle78373142010-10-29 19:08:24 +0100119 } while (unlikely(!temp));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700120 } else {
121 unsigned long flags;
122
Ralf Baechle49edd092007-03-16 16:10:36 +0000123 raw_local_irq_save(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700124 v->counter -= i;
Ralf Baechle49edd092007-03-16 16:10:36 +0000125 raw_local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700126 }
127}
128
129/*
130 * Same as above, but return the result value
131 */
132static __inline__ int atomic_add_return(int i, atomic_t * v)
133{
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000134 int result;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700135
David Daneyf252ffd2010-01-08 17:17:43 -0800136 smp_mb__before_llsc();
Ralf Baechle0004a9d2006-10-31 03:45:07 +0000137
David Daneyb791d112009-07-13 11:15:19 -0700138 if (kernel_uses_llsc && R10000_LLSC_WAR) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000139 int temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700140
141 __asm__ __volatile__(
Maciej W. Rozyckic4559f62005-06-23 15:57:15 +0000142 " .set mips3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700143 "1: ll %1, %2 # atomic_add_return \n"
144 " addu %0, %1, %3 \n"
145 " sc %0, %2 \n"
146 " beqzl %0, 1b \n"
147 " addu %0, %1, %3 \n"
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000148 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400149 : "=&r" (result), "=&r" (temp), "+m" (v->counter)
150 : "Ir" (i));
David Daneyb791d112009-07-13 11:15:19 -0700151 } else if (kernel_uses_llsc) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000152 int temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700153
Ralf Baechle78373142010-10-29 19:08:24 +0100154 do {
155 __asm__ __volatile__(
156 " .set mips3 \n"
157 " ll %1, %2 # atomic_add_return \n"
158 " addu %0, %1, %3 \n"
159 " sc %0, %2 \n"
160 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400161 : "=&r" (result), "=&r" (temp), "+m" (v->counter)
162 : "Ir" (i));
Ralf Baechle78373142010-10-29 19:08:24 +0100163 } while (unlikely(!result));
164
165 result = temp + i;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700166 } else {
167 unsigned long flags;
168
Ralf Baechle49edd092007-03-16 16:10:36 +0000169 raw_local_irq_save(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700170 result = v->counter;
171 result += i;
172 v->counter = result;
Ralf Baechle49edd092007-03-16 16:10:36 +0000173 raw_local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700174 }
175
Ralf Baechle17099b12007-07-14 13:24:05 +0100176 smp_llsc_mb();
Ralf Baechle0004a9d2006-10-31 03:45:07 +0000177
Linus Torvalds1da177e2005-04-16 15:20:36 -0700178 return result;
179}
180
181static __inline__ int atomic_sub_return(int i, atomic_t * v)
182{
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000183 int result;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700184
David Daneyf252ffd2010-01-08 17:17:43 -0800185 smp_mb__before_llsc();
Ralf Baechle0004a9d2006-10-31 03:45:07 +0000186
David Daneyb791d112009-07-13 11:15:19 -0700187 if (kernel_uses_llsc && R10000_LLSC_WAR) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000188 int temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700189
190 __asm__ __volatile__(
Maciej W. Rozyckic4559f62005-06-23 15:57:15 +0000191 " .set mips3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700192 "1: ll %1, %2 # atomic_sub_return \n"
193 " subu %0, %1, %3 \n"
194 " sc %0, %2 \n"
195 " beqzl %0, 1b \n"
196 " subu %0, %1, %3 \n"
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000197 " .set mips0 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700198 : "=&r" (result), "=&r" (temp), "=m" (v->counter)
199 : "Ir" (i), "m" (v->counter)
200 : "memory");
Ralf Baechle78373142010-10-29 19:08:24 +0100201
202 result = temp - i;
David Daneyb791d112009-07-13 11:15:19 -0700203 } else if (kernel_uses_llsc) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000204 int temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700205
Ralf Baechle78373142010-10-29 19:08:24 +0100206 do {
207 __asm__ __volatile__(
208 " .set mips3 \n"
209 " ll %1, %2 # atomic_sub_return \n"
210 " subu %0, %1, %3 \n"
211 " sc %0, %2 \n"
212 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400213 : "=&r" (result), "=&r" (temp), "+m" (v->counter)
214 : "Ir" (i));
Ralf Baechle78373142010-10-29 19:08:24 +0100215 } while (unlikely(!result));
216
217 result = temp - i;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700218 } else {
219 unsigned long flags;
220
Ralf Baechle49edd092007-03-16 16:10:36 +0000221 raw_local_irq_save(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700222 result = v->counter;
223 result -= i;
224 v->counter = result;
Ralf Baechle49edd092007-03-16 16:10:36 +0000225 raw_local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700226 }
227
Ralf Baechle17099b12007-07-14 13:24:05 +0100228 smp_llsc_mb();
Ralf Baechle0004a9d2006-10-31 03:45:07 +0000229
Linus Torvalds1da177e2005-04-16 15:20:36 -0700230 return result;
231}
232
233/*
Arnaud Gierschf10d14d2005-11-13 00:38:18 +0100234 * atomic_sub_if_positive - conditionally subtract integer from atomic variable
235 * @i: integer value to subtract
Linus Torvalds1da177e2005-04-16 15:20:36 -0700236 * @v: pointer of type atomic_t
237 *
Arnaud Gierschf10d14d2005-11-13 00:38:18 +0100238 * Atomically test @v and subtract @i if @v is greater or equal than @i.
239 * The function returns the old value of @v minus @i.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700240 */
241static __inline__ int atomic_sub_if_positive(int i, atomic_t * v)
242{
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000243 int result;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700244
David Daneyf252ffd2010-01-08 17:17:43 -0800245 smp_mb__before_llsc();
Ralf Baechle0004a9d2006-10-31 03:45:07 +0000246
David Daneyb791d112009-07-13 11:15:19 -0700247 if (kernel_uses_llsc && R10000_LLSC_WAR) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000248 int temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700249
250 __asm__ __volatile__(
Maciej W. Rozyckic4559f62005-06-23 15:57:15 +0000251 " .set mips3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700252 "1: ll %1, %2 # atomic_sub_if_positive\n"
253 " subu %0, %1, %3 \n"
254 " bltz %0, 1f \n"
255 " sc %0, %2 \n"
Ralf Baechle92f22c12006-02-23 14:10:53 +0000256 " .set noreorder \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700257 " beqzl %0, 1b \n"
Ralf Baechle92f22c12006-02-23 14:10:53 +0000258 " subu %0, %1, %3 \n"
259 " .set reorder \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700260 "1: \n"
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000261 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400262 : "=&r" (result), "=&r" (temp), "+m" (v->counter)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700263 : "Ir" (i), "m" (v->counter)
264 : "memory");
David Daneyb791d112009-07-13 11:15:19 -0700265 } else if (kernel_uses_llsc) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000266 int temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700267
268 __asm__ __volatile__(
Maciej W. Rozyckic4559f62005-06-23 15:57:15 +0000269 " .set mips3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700270 "1: ll %1, %2 # atomic_sub_if_positive\n"
271 " subu %0, %1, %3 \n"
272 " bltz %0, 1f \n"
273 " sc %0, %2 \n"
Ralf Baechle92f22c12006-02-23 14:10:53 +0000274 " .set noreorder \n"
Ralf Baechle78373142010-10-29 19:08:24 +0100275 " beqz %0, 1b \n"
Ralf Baechle92f22c12006-02-23 14:10:53 +0000276 " subu %0, %1, %3 \n"
277 " .set reorder \n"
Ralf Baechle50952022008-07-03 23:28:35 +0100278 "1: \n"
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000279 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400280 : "=&r" (result), "=&r" (temp), "+m" (v->counter)
281 : "Ir" (i));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700282 } else {
283 unsigned long flags;
284
Ralf Baechle49edd092007-03-16 16:10:36 +0000285 raw_local_irq_save(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700286 result = v->counter;
287 result -= i;
288 if (result >= 0)
289 v->counter = result;
Ralf Baechle49edd092007-03-16 16:10:36 +0000290 raw_local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700291 }
292
Ralf Baechle17099b12007-07-14 13:24:05 +0100293 smp_llsc_mb();
Ralf Baechle0004a9d2006-10-31 03:45:07 +0000294
Linus Torvalds1da177e2005-04-16 15:20:36 -0700295 return result;
296}
297
Mathieu Desnoyerse12f6442007-05-08 00:34:24 -0700298#define atomic_cmpxchg(v, o, n) (cmpxchg(&((v)->counter), (o), (n)))
299#define atomic_xchg(v, new) (xchg(&((v)->counter), (new)))
Nick Piggin4a6dae62005-11-13 16:07:24 -0800300
Nick Piggin8426e1f2005-11-13 16:07:25 -0800301/**
Arun Sharmaf24219b2011-07-26 16:09:07 -0700302 * __atomic_add_unless - add unless the number is a given value
Nick Piggin8426e1f2005-11-13 16:07:25 -0800303 * @v: pointer of type atomic_t
304 * @a: the amount to add to v...
305 * @u: ...unless v is equal to u.
306 *
307 * Atomically adds @a to @v, so long as it was not @u.
Arun Sharmaf24219b2011-07-26 16:09:07 -0700308 * Returns the old value of @v.
Nick Piggin8426e1f2005-11-13 16:07:25 -0800309 */
Arun Sharmaf24219b2011-07-26 16:09:07 -0700310static __inline__ int __atomic_add_unless(atomic_t *v, int a, int u)
Mathieu Desnoyers2856f5e2007-05-08 00:34:38 -0700311{
312 int c, old;
313 c = atomic_read(v);
314 for (;;) {
315 if (unlikely(c == (u)))
316 break;
317 old = atomic_cmpxchg((v), c, c + (a));
318 if (likely(old == c))
319 break;
320 c = old;
321 }
Arun Sharmaf24219b2011-07-26 16:09:07 -0700322 return c;
Mathieu Desnoyers2856f5e2007-05-08 00:34:38 -0700323}
Nick Piggin8426e1f2005-11-13 16:07:25 -0800324
Ralf Baechle21a151d2007-10-11 23:46:15 +0100325#define atomic_dec_return(v) atomic_sub_return(1, (v))
326#define atomic_inc_return(v) atomic_add_return(1, (v))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700327
328/*
329 * atomic_sub_and_test - subtract value from variable and test result
330 * @i: integer value to subtract
331 * @v: pointer of type atomic_t
332 *
333 * Atomically subtracts @i from @v and returns
334 * true if the result is zero, or false for all
335 * other cases.
336 */
Ralf Baechle21a151d2007-10-11 23:46:15 +0100337#define atomic_sub_and_test(i, v) (atomic_sub_return((i), (v)) == 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700338
339/*
340 * atomic_inc_and_test - increment and test
341 * @v: pointer of type atomic_t
342 *
343 * Atomically increments @v by 1
344 * and returns true if the result is zero, or false for all
345 * other cases.
346 */
347#define atomic_inc_and_test(v) (atomic_inc_return(v) == 0)
348
349/*
350 * atomic_dec_and_test - decrement by 1 and test
351 * @v: pointer of type atomic_t
352 *
353 * Atomically decrements @v by 1 and
354 * returns true if the result is 0, or false for all other
355 * cases.
356 */
357#define atomic_dec_and_test(v) (atomic_sub_return(1, (v)) == 0)
358
359/*
360 * atomic_dec_if_positive - decrement by 1 if old value positive
361 * @v: pointer of type atomic_t
362 */
363#define atomic_dec_if_positive(v) atomic_sub_if_positive(1, v)
364
365/*
366 * atomic_inc - increment atomic variable
367 * @v: pointer of type atomic_t
368 *
369 * Atomically increments @v by 1.
370 */
Ralf Baechle21a151d2007-10-11 23:46:15 +0100371#define atomic_inc(v) atomic_add(1, (v))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700372
373/*
374 * atomic_dec - decrement and test
375 * @v: pointer of type atomic_t
376 *
377 * Atomically decrements @v by 1.
378 */
Ralf Baechle21a151d2007-10-11 23:46:15 +0100379#define atomic_dec(v) atomic_sub(1, (v))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700380
381/*
382 * atomic_add_negative - add and test if negative
383 * @v: pointer of type atomic_t
384 * @i: integer value to add
385 *
386 * Atomically adds @i to @v and returns true
387 * if the result is negative, or false when
388 * result is greater than or equal to zero.
389 */
Ralf Baechle21a151d2007-10-11 23:46:15 +0100390#define atomic_add_negative(i, v) (atomic_add_return(i, (v)) < 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700391
Ralf Baechle875d43e2005-09-03 15:56:16 -0700392#ifdef CONFIG_64BIT
Linus Torvalds1da177e2005-04-16 15:20:36 -0700393
Linus Torvalds1da177e2005-04-16 15:20:36 -0700394#define ATOMIC64_INIT(i) { (i) }
395
396/*
397 * atomic64_read - read atomic variable
398 * @v: pointer of type atomic64_t
399 *
400 */
Anton Blanchardf3d46f92010-05-17 14:33:53 +1000401#define atomic64_read(v) (*(volatile long *)&(v)->counter)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700402
403/*
404 * atomic64_set - set atomic variable
405 * @v: pointer of type atomic64_t
406 * @i: required value
407 */
Ralf Baechle21a151d2007-10-11 23:46:15 +0100408#define atomic64_set(v, i) ((v)->counter = (i))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700409
410/*
411 * atomic64_add - add integer to atomic variable
412 * @i: integer value to add
413 * @v: pointer of type atomic64_t
414 *
415 * Atomically adds @i to @v.
416 */
417static __inline__ void atomic64_add(long i, atomic64_t * v)
418{
David Daneyb791d112009-07-13 11:15:19 -0700419 if (kernel_uses_llsc && R10000_LLSC_WAR) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000420 long temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700421
422 __asm__ __volatile__(
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000423 " .set mips3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700424 "1: lld %0, %1 # atomic64_add \n"
David Daneyf2a68272010-07-22 11:59:27 -0700425 " daddu %0, %2 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700426 " scd %0, %1 \n"
427 " beqzl %0, 1b \n"
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000428 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400429 : "=&r" (temp), "+m" (v->counter)
430 : "Ir" (i));
David Daneyb791d112009-07-13 11:15:19 -0700431 } else if (kernel_uses_llsc) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000432 long temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700433
Ralf Baechle78373142010-10-29 19:08:24 +0100434 do {
435 __asm__ __volatile__(
436 " .set mips3 \n"
437 " lld %0, %1 # atomic64_add \n"
438 " daddu %0, %2 \n"
439 " scd %0, %1 \n"
440 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400441 : "=&r" (temp), "+m" (v->counter)
442 : "Ir" (i));
Ralf Baechle78373142010-10-29 19:08:24 +0100443 } while (unlikely(!temp));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700444 } else {
445 unsigned long flags;
446
Ralf Baechle49edd092007-03-16 16:10:36 +0000447 raw_local_irq_save(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700448 v->counter += i;
Ralf Baechle49edd092007-03-16 16:10:36 +0000449 raw_local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700450 }
451}
452
453/*
454 * atomic64_sub - subtract the atomic variable
455 * @i: integer value to subtract
456 * @v: pointer of type atomic64_t
457 *
458 * Atomically subtracts @i from @v.
459 */
460static __inline__ void atomic64_sub(long i, atomic64_t * v)
461{
David Daneyb791d112009-07-13 11:15:19 -0700462 if (kernel_uses_llsc && R10000_LLSC_WAR) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000463 long temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700464
465 __asm__ __volatile__(
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000466 " .set mips3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700467 "1: lld %0, %1 # atomic64_sub \n"
David Daneyf2a68272010-07-22 11:59:27 -0700468 " dsubu %0, %2 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700469 " scd %0, %1 \n"
470 " beqzl %0, 1b \n"
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000471 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400472 : "=&r" (temp), "+m" (v->counter)
473 : "Ir" (i));
David Daneyb791d112009-07-13 11:15:19 -0700474 } else if (kernel_uses_llsc) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000475 long temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700476
Ralf Baechle78373142010-10-29 19:08:24 +0100477 do {
478 __asm__ __volatile__(
479 " .set mips3 \n"
480 " lld %0, %1 # atomic64_sub \n"
481 " dsubu %0, %2 \n"
482 " scd %0, %1 \n"
483 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400484 : "=&r" (temp), "+m" (v->counter)
485 : "Ir" (i));
Ralf Baechle78373142010-10-29 19:08:24 +0100486 } while (unlikely(!temp));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700487 } else {
488 unsigned long flags;
489
Ralf Baechle49edd092007-03-16 16:10:36 +0000490 raw_local_irq_save(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700491 v->counter -= i;
Ralf Baechle49edd092007-03-16 16:10:36 +0000492 raw_local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700493 }
494}
495
496/*
497 * Same as above, but return the result value
498 */
499static __inline__ long atomic64_add_return(long i, atomic64_t * v)
500{
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000501 long result;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700502
David Daneyf252ffd2010-01-08 17:17:43 -0800503 smp_mb__before_llsc();
Ralf Baechle0004a9d2006-10-31 03:45:07 +0000504
David Daneyb791d112009-07-13 11:15:19 -0700505 if (kernel_uses_llsc && R10000_LLSC_WAR) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000506 long temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700507
508 __asm__ __volatile__(
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000509 " .set mips3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700510 "1: lld %1, %2 # atomic64_add_return \n"
David Daneyf2a68272010-07-22 11:59:27 -0700511 " daddu %0, %1, %3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700512 " scd %0, %2 \n"
513 " beqzl %0, 1b \n"
David Daneyf2a68272010-07-22 11:59:27 -0700514 " daddu %0, %1, %3 \n"
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000515 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400516 : "=&r" (result), "=&r" (temp), "+m" (v->counter)
517 : "Ir" (i));
David Daneyb791d112009-07-13 11:15:19 -0700518 } else if (kernel_uses_llsc) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000519 long temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700520
Ralf Baechle78373142010-10-29 19:08:24 +0100521 do {
522 __asm__ __volatile__(
523 " .set mips3 \n"
524 " lld %1, %2 # atomic64_add_return \n"
525 " daddu %0, %1, %3 \n"
526 " scd %0, %2 \n"
527 " .set mips0 \n"
528 : "=&r" (result), "=&r" (temp), "=m" (v->counter)
529 : "Ir" (i), "m" (v->counter)
530 : "memory");
531 } while (unlikely(!result));
532
533 result = temp + i;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700534 } else {
535 unsigned long flags;
536
Ralf Baechle49edd092007-03-16 16:10:36 +0000537 raw_local_irq_save(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700538 result = v->counter;
539 result += i;
540 v->counter = result;
Ralf Baechle49edd092007-03-16 16:10:36 +0000541 raw_local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700542 }
543
Ralf Baechle17099b12007-07-14 13:24:05 +0100544 smp_llsc_mb();
Ralf Baechle0004a9d2006-10-31 03:45:07 +0000545
Linus Torvalds1da177e2005-04-16 15:20:36 -0700546 return result;
547}
548
549static __inline__ long atomic64_sub_return(long i, atomic64_t * v)
550{
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000551 long result;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700552
David Daneyf252ffd2010-01-08 17:17:43 -0800553 smp_mb__before_llsc();
Ralf Baechle0004a9d2006-10-31 03:45:07 +0000554
David Daneyb791d112009-07-13 11:15:19 -0700555 if (kernel_uses_llsc && R10000_LLSC_WAR) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000556 long temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700557
558 __asm__ __volatile__(
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000559 " .set mips3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700560 "1: lld %1, %2 # atomic64_sub_return \n"
David Daneyf2a68272010-07-22 11:59:27 -0700561 " dsubu %0, %1, %3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700562 " scd %0, %2 \n"
563 " beqzl %0, 1b \n"
David Daneyf2a68272010-07-22 11:59:27 -0700564 " dsubu %0, %1, %3 \n"
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000565 " .set mips0 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700566 : "=&r" (result), "=&r" (temp), "=m" (v->counter)
567 : "Ir" (i), "m" (v->counter)
568 : "memory");
David Daneyb791d112009-07-13 11:15:19 -0700569 } else if (kernel_uses_llsc) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000570 long temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700571
Ralf Baechle78373142010-10-29 19:08:24 +0100572 do {
573 __asm__ __volatile__(
574 " .set mips3 \n"
575 " lld %1, %2 # atomic64_sub_return \n"
576 " dsubu %0, %1, %3 \n"
577 " scd %0, %2 \n"
578 " .set mips0 \n"
579 : "=&r" (result), "=&r" (temp), "=m" (v->counter)
580 : "Ir" (i), "m" (v->counter)
581 : "memory");
582 } while (unlikely(!result));
583
584 result = temp - i;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700585 } else {
586 unsigned long flags;
587
Ralf Baechle49edd092007-03-16 16:10:36 +0000588 raw_local_irq_save(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700589 result = v->counter;
590 result -= i;
591 v->counter = result;
Ralf Baechle49edd092007-03-16 16:10:36 +0000592 raw_local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700593 }
594
Ralf Baechle17099b12007-07-14 13:24:05 +0100595 smp_llsc_mb();
Ralf Baechle0004a9d2006-10-31 03:45:07 +0000596
Linus Torvalds1da177e2005-04-16 15:20:36 -0700597 return result;
598}
599
600/*
Arnaud Gierschf10d14d2005-11-13 00:38:18 +0100601 * atomic64_sub_if_positive - conditionally subtract integer from atomic variable
602 * @i: integer value to subtract
Linus Torvalds1da177e2005-04-16 15:20:36 -0700603 * @v: pointer of type atomic64_t
604 *
Arnaud Gierschf10d14d2005-11-13 00:38:18 +0100605 * Atomically test @v and subtract @i if @v is greater or equal than @i.
606 * The function returns the old value of @v minus @i.
Linus Torvalds1da177e2005-04-16 15:20:36 -0700607 */
608static __inline__ long atomic64_sub_if_positive(long i, atomic64_t * v)
609{
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000610 long result;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700611
David Daneyf252ffd2010-01-08 17:17:43 -0800612 smp_mb__before_llsc();
Ralf Baechle0004a9d2006-10-31 03:45:07 +0000613
David Daneyb791d112009-07-13 11:15:19 -0700614 if (kernel_uses_llsc && R10000_LLSC_WAR) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000615 long temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700616
617 __asm__ __volatile__(
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000618 " .set mips3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700619 "1: lld %1, %2 # atomic64_sub_if_positive\n"
620 " dsubu %0, %1, %3 \n"
621 " bltz %0, 1f \n"
622 " scd %0, %2 \n"
Ralf Baechle92f22c12006-02-23 14:10:53 +0000623 " .set noreorder \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700624 " beqzl %0, 1b \n"
Ralf Baechle92f22c12006-02-23 14:10:53 +0000625 " dsubu %0, %1, %3 \n"
626 " .set reorder \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700627 "1: \n"
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000628 " .set mips0 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700629 : "=&r" (result), "=&r" (temp), "=m" (v->counter)
630 : "Ir" (i), "m" (v->counter)
631 : "memory");
David Daneyb791d112009-07-13 11:15:19 -0700632 } else if (kernel_uses_llsc) {
Ralf Baechle915ec1e2009-01-12 00:52:18 +0000633 long temp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700634
635 __asm__ __volatile__(
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000636 " .set mips3 \n"
Linus Torvalds1da177e2005-04-16 15:20:36 -0700637 "1: lld %1, %2 # atomic64_sub_if_positive\n"
638 " dsubu %0, %1, %3 \n"
639 " bltz %0, 1f \n"
640 " scd %0, %2 \n"
Ralf Baechle92f22c12006-02-23 14:10:53 +0000641 " .set noreorder \n"
Ralf Baechle78373142010-10-29 19:08:24 +0100642 " beqz %0, 1b \n"
Ralf Baechle92f22c12006-02-23 14:10:53 +0000643 " dsubu %0, %1, %3 \n"
644 " .set reorder \n"
Ralf Baechle50952022008-07-03 23:28:35 +0100645 "1: \n"
Maciej W. Rozyckiaac8aa72005-06-14 17:35:03 +0000646 " .set mips0 \n"
Joshua Kinardb4f2a172012-06-24 21:01:34 -0400647 : "=&r" (result), "=&r" (temp), "+m" (v->counter)
648 : "Ir" (i));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700649 } else {
650 unsigned long flags;
651
Ralf Baechle49edd092007-03-16 16:10:36 +0000652 raw_local_irq_save(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700653 result = v->counter;
654 result -= i;
655 if (result >= 0)
656 v->counter = result;
Ralf Baechle49edd092007-03-16 16:10:36 +0000657 raw_local_irq_restore(flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700658 }
659
Ralf Baechle17099b12007-07-14 13:24:05 +0100660 smp_llsc_mb();
Ralf Baechle0004a9d2006-10-31 03:45:07 +0000661
Linus Torvalds1da177e2005-04-16 15:20:36 -0700662 return result;
663}
664
Mathieu Desnoyerse12f6442007-05-08 00:34:24 -0700665#define atomic64_cmpxchg(v, o, n) \
Atsushi Nemoto7b239bb2007-05-10 23:47:45 +0900666 ((__typeof__((v)->counter))cmpxchg(&((v)->counter), (o), (n)))
Mathieu Desnoyerse12f6442007-05-08 00:34:24 -0700667#define atomic64_xchg(v, new) (xchg(&((v)->counter), (new)))
668
669/**
670 * atomic64_add_unless - add unless the number is a given value
671 * @v: pointer of type atomic64_t
672 * @a: the amount to add to v...
673 * @u: ...unless v is equal to u.
674 *
675 * Atomically adds @a to @v, so long as it was not @u.
Arun Sharmaf24219b2011-07-26 16:09:07 -0700676 * Returns the old value of @v.
Mathieu Desnoyerse12f6442007-05-08 00:34:24 -0700677 */
Mathieu Desnoyers2856f5e2007-05-08 00:34:38 -0700678static __inline__ int atomic64_add_unless(atomic64_t *v, long a, long u)
679{
680 long c, old;
681 c = atomic64_read(v);
682 for (;;) {
683 if (unlikely(c == (u)))
684 break;
685 old = atomic64_cmpxchg((v), c, c + (a));
686 if (likely(old == c))
687 break;
688 c = old;
689 }
690 return c != (u);
691}
692
Mathieu Desnoyerse12f6442007-05-08 00:34:24 -0700693#define atomic64_inc_not_zero(v) atomic64_add_unless((v), 1, 0)
694
Ralf Baechle21a151d2007-10-11 23:46:15 +0100695#define atomic64_dec_return(v) atomic64_sub_return(1, (v))
696#define atomic64_inc_return(v) atomic64_add_return(1, (v))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700697
698/*
699 * atomic64_sub_and_test - subtract value from variable and test result
700 * @i: integer value to subtract
701 * @v: pointer of type atomic64_t
702 *
703 * Atomically subtracts @i from @v and returns
704 * true if the result is zero, or false for all
705 * other cases.
706 */
Ralf Baechle21a151d2007-10-11 23:46:15 +0100707#define atomic64_sub_and_test(i, v) (atomic64_sub_return((i), (v)) == 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700708
709/*
710 * atomic64_inc_and_test - increment and test
711 * @v: pointer of type atomic64_t
712 *
713 * Atomically increments @v by 1
714 * and returns true if the result is zero, or false for all
715 * other cases.
716 */
717#define atomic64_inc_and_test(v) (atomic64_inc_return(v) == 0)
718
719/*
720 * atomic64_dec_and_test - decrement by 1 and test
721 * @v: pointer of type atomic64_t
722 *
723 * Atomically decrements @v by 1 and
724 * returns true if the result is 0, or false for all other
725 * cases.
726 */
727#define atomic64_dec_and_test(v) (atomic64_sub_return(1, (v)) == 0)
728
729/*
730 * atomic64_dec_if_positive - decrement by 1 if old value positive
731 * @v: pointer of type atomic64_t
732 */
733#define atomic64_dec_if_positive(v) atomic64_sub_if_positive(1, v)
734
735/*
736 * atomic64_inc - increment atomic variable
737 * @v: pointer of type atomic64_t
738 *
739 * Atomically increments @v by 1.
740 */
Ralf Baechle21a151d2007-10-11 23:46:15 +0100741#define atomic64_inc(v) atomic64_add(1, (v))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700742
743/*
744 * atomic64_dec - decrement and test
745 * @v: pointer of type atomic64_t
746 *
747 * Atomically decrements @v by 1.
748 */
Ralf Baechle21a151d2007-10-11 23:46:15 +0100749#define atomic64_dec(v) atomic64_sub(1, (v))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700750
751/*
752 * atomic64_add_negative - add and test if negative
753 * @v: pointer of type atomic64_t
754 * @i: integer value to add
755 *
756 * Atomically adds @i to @v and returns true
757 * if the result is negative, or false when
758 * result is greater than or equal to zero.
759 */
Ralf Baechle21a151d2007-10-11 23:46:15 +0100760#define atomic64_add_negative(i, v) (atomic64_add_return(i, (v)) < 0)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700761
Ralf Baechle875d43e2005-09-03 15:56:16 -0700762#endif /* CONFIG_64BIT */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700763
764/*
765 * atomic*_return operations are serializing but not the non-*_return
766 * versions.
767 */
David Daneyf252ffd2010-01-08 17:17:43 -0800768#define smp_mb__before_atomic_dec() smp_mb__before_llsc()
Ralf Baechle17099b12007-07-14 13:24:05 +0100769#define smp_mb__after_atomic_dec() smp_llsc_mb()
David Daneyf252ffd2010-01-08 17:17:43 -0800770#define smp_mb__before_atomic_inc() smp_mb__before_llsc()
Ralf Baechle17099b12007-07-14 13:24:05 +0100771#define smp_mb__after_atomic_inc() smp_llsc_mb()
Linus Torvalds1da177e2005-04-16 15:20:36 -0700772
Linus Torvalds1da177e2005-04-16 15:20:36 -0700773#endif /* _ASM_ATOMIC_H */