1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18#ifndef __ASM_CMPXCHG_H
19#define __ASM_CMPXCHG_H
20
21#include <linux/bug.h>
22
23#include <asm/atomic.h>
24#include <asm/barrier.h>
25#include <asm/lse.h>
26
27
28
29
30
31
32#define __XCHG_CASE(w, sz, name, mb, nop_lse, acq, acq_lse, rel, cl) \
33static inline unsigned long __xchg_case_##name(unsigned long x, \
34 volatile void *ptr) \
35{ \
36 unsigned long ret, tmp; \
37 \
38 asm volatile(ARM64_LSE_ATOMIC_INSN( \
39 \
40 " prfm pstl1strm, %2\n" \
41 "1: ld" #acq "xr" #sz "\t%" #w "0, %2\n" \
42 " st" #rel "xr" #sz "\t%w1, %" #w "3, %2\n" \
43 " cbnz %w1, 1b\n" \
44 " " #mb, \
45 \
46 " nop\n" \
47 " nop\n" \
48 " swp" #acq_lse #rel #sz "\t%" #w "3, %" #w "0, %2\n" \
49 " nop\n" \
50 " " #nop_lse) \
51 : "=&r" (ret), "=&r" (tmp), "+Q" (*(u8 *)ptr) \
52 : "r" (x) \
53 : cl); \
54 \
55 return ret; \
56}
57
58__XCHG_CASE(w, b, 1, , , , , , )
59__XCHG_CASE(w, h, 2, , , , , , )
60__XCHG_CASE(w, , 4, , , , , , )
61__XCHG_CASE( , , 8, , , , , , )
62__XCHG_CASE(w, b, acq_1, , , a, a, , "memory")
63__XCHG_CASE(w, h, acq_2, , , a, a, , "memory")
64__XCHG_CASE(w, , acq_4, , , a, a, , "memory")
65__XCHG_CASE( , , acq_8, , , a, a, , "memory")
66__XCHG_CASE(w, b, rel_1, , , , , l, "memory")
67__XCHG_CASE(w, h, rel_2, , , , , l, "memory")
68__XCHG_CASE(w, , rel_4, , , , , l, "memory")
69__XCHG_CASE( , , rel_8, , , , , l, "memory")
70__XCHG_CASE(w, b, mb_1, dmb ish, nop, , a, l, "memory")
71__XCHG_CASE(w, h, mb_2, dmb ish, nop, , a, l, "memory")
72__XCHG_CASE(w, , mb_4, dmb ish, nop, , a, l, "memory")
73__XCHG_CASE( , , mb_8, dmb ish, nop, , a, l, "memory")
74
75#undef __XCHG_CASE
76
77#define __XCHG_GEN(sfx) \
78static inline unsigned long __xchg##sfx(unsigned long x, \
79 volatile void *ptr, \
80 int size) \
81{ \
82 switch (size) { \
83 case 1: \
84 return __xchg_case##sfx##_1(x, ptr); \
85 case 2: \
86 return __xchg_case##sfx##_2(x, ptr); \
87 case 4: \
88 return __xchg_case##sfx##_4(x, ptr); \
89 case 8: \
90 return __xchg_case##sfx##_8(x, ptr); \
91 default: \
92 BUILD_BUG(); \
93 } \
94 \
95 unreachable(); \
96}
97
98__XCHG_GEN()
99__XCHG_GEN(_acq)
100__XCHG_GEN(_rel)
101__XCHG_GEN(_mb)
102
103#undef __XCHG_GEN
104
105#define __xchg_wrapper(sfx, ptr, x) \
106({ \
107 __typeof__(*(ptr)) __ret; \
108 __ret = (__typeof__(*(ptr))) \
109 __xchg##sfx((unsigned long)(x), (ptr), sizeof(*(ptr))); \
110 __ret; \
111})
112
113
114#define xchg_relaxed(...) __xchg_wrapper( , __VA_ARGS__)
115#define xchg_acquire(...) __xchg_wrapper(_acq, __VA_ARGS__)
116#define xchg_release(...) __xchg_wrapper(_rel, __VA_ARGS__)
117#define xchg(...) __xchg_wrapper( _mb, __VA_ARGS__)
118
119#define __CMPXCHG_GEN(sfx) \
120static inline unsigned long __cmpxchg##sfx(volatile void *ptr, \
121 unsigned long old, \
122 unsigned long new, \
123 int size) \
124{ \
125 switch (size) { \
126 case 1: \
127 return __cmpxchg_case##sfx##_1(ptr, (u8)old, new); \
128 case 2: \
129 return __cmpxchg_case##sfx##_2(ptr, (u16)old, new); \
130 case 4: \
131 return __cmpxchg_case##sfx##_4(ptr, old, new); \
132 case 8: \
133 return __cmpxchg_case##sfx##_8(ptr, old, new); \
134 default: \
135 BUILD_BUG(); \
136 } \
137 \
138 unreachable(); \
139}
140
141__CMPXCHG_GEN()
142__CMPXCHG_GEN(_acq)
143__CMPXCHG_GEN(_rel)
144__CMPXCHG_GEN(_mb)
145
146#undef __CMPXCHG_GEN
147
148#define __cmpxchg_wrapper(sfx, ptr, o, n) \
149({ \
150 __typeof__(*(ptr)) __ret; \
151 __ret = (__typeof__(*(ptr))) \
152 __cmpxchg##sfx((ptr), (unsigned long)(o), \
153 (unsigned long)(n), sizeof(*(ptr))); \
154 __ret; \
155})
156
157
158#define cmpxchg_relaxed(...) __cmpxchg_wrapper( , __VA_ARGS__)
159#define cmpxchg_acquire(...) __cmpxchg_wrapper(_acq, __VA_ARGS__)
160#define cmpxchg_release(...) __cmpxchg_wrapper(_rel, __VA_ARGS__)
161#define cmpxchg(...) __cmpxchg_wrapper( _mb, __VA_ARGS__)
162#define cmpxchg_local cmpxchg_relaxed
163
164
165#define cmpxchg64_relaxed cmpxchg_relaxed
166#define cmpxchg64_acquire cmpxchg_acquire
167#define cmpxchg64_release cmpxchg_release
168#define cmpxchg64 cmpxchg
169#define cmpxchg64_local cmpxchg_local
170
171
172#define system_has_cmpxchg_double() 1
173
174#define __cmpxchg_double_check(ptr1, ptr2) \
175({ \
176 if (sizeof(*(ptr1)) != 8) \
177 BUILD_BUG(); \
178 VM_BUG_ON((unsigned long *)(ptr2) - (unsigned long *)(ptr1) != 1); \
179})
180
181#define cmpxchg_double(ptr1, ptr2, o1, o2, n1, n2) \
182({\
183 int __ret;\
184 __cmpxchg_double_check(ptr1, ptr2); \
185 __ret = !__cmpxchg_double_mb((unsigned long)(o1), (unsigned long)(o2), \
186 (unsigned long)(n1), (unsigned long)(n2), \
187 ptr1); \
188 __ret; \
189})
190
191#define cmpxchg_double_local(ptr1, ptr2, o1, o2, n1, n2) \
192({\
193 int __ret;\
194 __cmpxchg_double_check(ptr1, ptr2); \
195 __ret = !__cmpxchg_double((unsigned long)(o1), (unsigned long)(o2), \
196 (unsigned long)(n1), (unsigned long)(n2), \
197 ptr1); \
198 __ret; \
199})
200
201
202#define _protect_cmpxchg_local(pcp, o, n) \
203({ \
204 typeof(*raw_cpu_ptr(&(pcp))) __ret; \
205 preempt_disable(); \
206 __ret = cmpxchg_local(raw_cpu_ptr(&(pcp)), o, n); \
207 preempt_enable(); \
208 __ret; \
209})
210
211#define this_cpu_cmpxchg_1(ptr, o, n) _protect_cmpxchg_local(ptr, o, n)
212#define this_cpu_cmpxchg_2(ptr, o, n) _protect_cmpxchg_local(ptr, o, n)
213#define this_cpu_cmpxchg_4(ptr, o, n) _protect_cmpxchg_local(ptr, o, n)
214#define this_cpu_cmpxchg_8(ptr, o, n) _protect_cmpxchg_local(ptr, o, n)
215
216#define this_cpu_cmpxchg_double_8(ptr1, ptr2, o1, o2, n1, n2) \
217({ \
218 int __ret; \
219 preempt_disable(); \
220 __ret = cmpxchg_double_local( raw_cpu_ptr(&(ptr1)), \
221 raw_cpu_ptr(&(ptr2)), \
222 o1, o2, n1, n2); \
223 preempt_enable(); \
224 __ret; \
225})
226
227#endif
228