aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorPeter Zijlstra <peterz@infradead.org>2014-04-23 13:40:25 -0400
committerThomas Gleixner <tglx@linutronix.de>2015-07-27 08:06:23 -0400
commit304a0d699a3c6103b61d5ea18d56820e7d8e3116 (patch)
treedd0166235d2a1dc0c7a152e6faa825d2794cc3a0
parent658aa51459c2f5284183d35b6dd0beca0e0bfe2f (diff)
sparc: Provide atomic_{or,xor,and}
Implement atomic logic ops -- atomic_{or,xor,and}. These will replace the atomic_{set,clear}_mask functions that are available on some archs. Acked-by: David S. Miller <davem@davemloft.net> Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org> Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
-rw-r--r--arch/sparc/include/asm/atomic_32.h4
-rw-r--r--arch/sparc/include/asm/atomic_64.h6
-rw-r--r--arch/sparc/lib/atomic32.c22
-rw-r--r--arch/sparc/lib/atomic_64.S6
-rw-r--r--arch/sparc/lib/ksyms.c3
5 files changed, 38 insertions, 3 deletions
diff --git a/arch/sparc/include/asm/atomic_32.h b/arch/sparc/include/asm/atomic_32.h
index 0e69b7e7a439..e19d8880b146 100644
--- a/arch/sparc/include/asm/atomic_32.h
+++ b/arch/sparc/include/asm/atomic_32.h
@@ -17,10 +17,14 @@
17#include <asm/barrier.h> 17#include <asm/barrier.h>
18#include <asm-generic/atomic64.h> 18#include <asm-generic/atomic64.h>
19 19
20#define CONFIG_ARCH_HAS_ATOMIC_OR
20 21
21#define ATOMIC_INIT(i) { (i) } 22#define ATOMIC_INIT(i) { (i) }
22 23
23int atomic_add_return(int, atomic_t *); 24int atomic_add_return(int, atomic_t *);
25void atomic_and(int, atomic_t *);
26void atomic_or(int, atomic_t *);
27void atomic_xor(int, atomic_t *);
24int atomic_cmpxchg(atomic_t *, int, int); 28int atomic_cmpxchg(atomic_t *, int, int);
25int atomic_xchg(atomic_t *, int); 29int atomic_xchg(atomic_t *, int);
26int __atomic_add_unless(atomic_t *, int, int); 30int __atomic_add_unless(atomic_t *, int, int);
diff --git a/arch/sparc/include/asm/atomic_64.h b/arch/sparc/include/asm/atomic_64.h
index 4082749913ce..d6af27c93450 100644
--- a/arch/sparc/include/asm/atomic_64.h
+++ b/arch/sparc/include/asm/atomic_64.h
@@ -33,6 +33,12 @@ long atomic64_##op##_return(long, atomic64_t *);
33ATOMIC_OPS(add) 33ATOMIC_OPS(add)
34ATOMIC_OPS(sub) 34ATOMIC_OPS(sub)
35 35
36#define CONFIG_ARCH_HAS_ATOMIC_OR
37
38ATOMIC_OP(and)
39ATOMIC_OP(or)
40ATOMIC_OP(xor)
41
36#undef ATOMIC_OPS 42#undef ATOMIC_OPS
37#undef ATOMIC_OP_RETURN 43#undef ATOMIC_OP_RETURN
38#undef ATOMIC_OP 44#undef ATOMIC_OP
diff --git a/arch/sparc/lib/atomic32.c b/arch/sparc/lib/atomic32.c
index 71cd65ab200c..b9d63c0a7aab 100644
--- a/arch/sparc/lib/atomic32.c
+++ b/arch/sparc/lib/atomic32.c
@@ -27,22 +27,38 @@ static DEFINE_SPINLOCK(dummy);
27 27
28#endif /* SMP */ 28#endif /* SMP */
29 29
30#define ATOMIC_OP(op, cop) \ 30#define ATOMIC_OP_RETURN(op, c_op) \
31int atomic_##op##_return(int i, atomic_t *v) \ 31int atomic_##op##_return(int i, atomic_t *v) \
32{ \ 32{ \
33 int ret; \ 33 int ret; \
34 unsigned long flags; \ 34 unsigned long flags; \
35 spin_lock_irqsave(ATOMIC_HASH(v), flags); \ 35 spin_lock_irqsave(ATOMIC_HASH(v), flags); \
36 \ 36 \
37 ret = (v->counter cop i); \ 37 ret = (v->counter c_op i); \
38 \ 38 \
39 spin_unlock_irqrestore(ATOMIC_HASH(v), flags); \ 39 spin_unlock_irqrestore(ATOMIC_HASH(v), flags); \
40 return ret; \ 40 return ret; \
41} \ 41} \
42EXPORT_SYMBOL(atomic_##op##_return); 42EXPORT_SYMBOL(atomic_##op##_return);
43 43
44ATOMIC_OP(add, +=) 44#define ATOMIC_OP(op, c_op) \
45void atomic_##op(int i, atomic_t *v) \
46{ \
47 unsigned long flags; \
48 spin_lock_irqsave(ATOMIC_HASH(v), flags); \
49 \
50 v->counter c_op i; \
51 \
52 spin_unlock_irqrestore(ATOMIC_HASH(v), flags); \
53} \
54EXPORT_SYMBOL(atomic_##op);
55
56ATOMIC_OP_RETURN(add, +=)
57ATOMIC_OP(and, &=)
58ATOMIC_OP(or, |=)
59ATOMIC_OP(xor, ^=)
45 60
61#undef ATOMIC_OP_RETURN
46#undef ATOMIC_OP 62#undef ATOMIC_OP
47 63
48int atomic_xchg(atomic_t *v, int new) 64int atomic_xchg(atomic_t *v, int new)
diff --git a/arch/sparc/lib/atomic_64.S b/arch/sparc/lib/atomic_64.S
index 05dac43907d1..d6b0363f345b 100644
--- a/arch/sparc/lib/atomic_64.S
+++ b/arch/sparc/lib/atomic_64.S
@@ -47,6 +47,9 @@ ENDPROC(atomic_##op##_return);
47 47
48ATOMIC_OPS(add) 48ATOMIC_OPS(add)
49ATOMIC_OPS(sub) 49ATOMIC_OPS(sub)
50ATOMIC_OP(and)
51ATOMIC_OP(or)
52ATOMIC_OP(xor)
50 53
51#undef ATOMIC_OPS 54#undef ATOMIC_OPS
52#undef ATOMIC_OP_RETURN 55#undef ATOMIC_OP_RETURN
@@ -84,6 +87,9 @@ ENDPROC(atomic64_##op##_return);
84 87
85ATOMIC64_OPS(add) 88ATOMIC64_OPS(add)
86ATOMIC64_OPS(sub) 89ATOMIC64_OPS(sub)
90ATOMIC64_OP(and)
91ATOMIC64_OP(or)
92ATOMIC64_OP(xor)
87 93
88#undef ATOMIC64_OPS 94#undef ATOMIC64_OPS
89#undef ATOMIC64_OP_RETURN 95#undef ATOMIC64_OP_RETURN
diff --git a/arch/sparc/lib/ksyms.c b/arch/sparc/lib/ksyms.c
index 1d649a95660c..bb6005997268 100644
--- a/arch/sparc/lib/ksyms.c
+++ b/arch/sparc/lib/ksyms.c
@@ -111,6 +111,9 @@ EXPORT_SYMBOL(atomic64_##op##_return);
111 111
112ATOMIC_OPS(add) 112ATOMIC_OPS(add)
113ATOMIC_OPS(sub) 113ATOMIC_OPS(sub)
114ATOMIC_OP(and)
115ATOMIC_OP(or)
116ATOMIC_OP(xor)
114 117
115#undef ATOMIC_OPS 118#undef ATOMIC_OPS
116#undef ATOMIC_OP_RETURN 119#undef ATOMIC_OP_RETURN