writeback: safer lock nesting
[linux/fpc-iii.git] / include / asm-generic / percpu.h
blob1817a8415a5e82a7bc526a38361cea88566e62bb
1 /* SPDX-License-Identifier: GPL-2.0 */
2 #ifndef _ASM_GENERIC_PERCPU_H_
3 #define _ASM_GENERIC_PERCPU_H_
5 #include <linux/compiler.h>
6 #include <linux/threads.h>
7 #include <linux/percpu-defs.h>
9 #ifdef CONFIG_SMP
12 * per_cpu_offset() is the offset that has to be added to a
13 * percpu variable to get to the instance for a certain processor.
15 * Most arches use the __per_cpu_offset array for those offsets but
16 * some arches have their own ways of determining the offset (x86_64, s390).
18 #ifndef __per_cpu_offset
19 extern unsigned long __per_cpu_offset[NR_CPUS];
21 #define per_cpu_offset(x) (__per_cpu_offset[x])
22 #endif
25 * Determine the offset for the currently active processor.
26 * An arch may define __my_cpu_offset to provide a more effective
27 * means of obtaining the offset to the per cpu variables of the
28 * current processor.
30 #ifndef __my_cpu_offset
31 #define __my_cpu_offset per_cpu_offset(raw_smp_processor_id())
32 #endif
33 #ifdef CONFIG_DEBUG_PREEMPT
34 #define my_cpu_offset per_cpu_offset(smp_processor_id())
35 #else
36 #define my_cpu_offset __my_cpu_offset
37 #endif
40 * Arch may define arch_raw_cpu_ptr() to provide more efficient address
41 * translations for raw_cpu_ptr().
43 #ifndef arch_raw_cpu_ptr
44 #define arch_raw_cpu_ptr(ptr) SHIFT_PERCPU_PTR(ptr, __my_cpu_offset)
45 #endif
47 #ifdef CONFIG_HAVE_SETUP_PER_CPU_AREA
48 extern void setup_per_cpu_areas(void);
49 #endif
51 #endif /* SMP */
53 #ifndef PER_CPU_BASE_SECTION
54 #ifdef CONFIG_SMP
55 #define PER_CPU_BASE_SECTION ".data..percpu"
56 #else
57 #define PER_CPU_BASE_SECTION ".data"
58 #endif
59 #endif
61 #ifndef PER_CPU_ATTRIBUTES
62 #define PER_CPU_ATTRIBUTES
63 #endif
65 #ifndef PER_CPU_DEF_ATTRIBUTES
66 #define PER_CPU_DEF_ATTRIBUTES
67 #endif
69 #define raw_cpu_generic_read(pcp) \
70 ({ \
71 *raw_cpu_ptr(&(pcp)); \
74 #define raw_cpu_generic_to_op(pcp, val, op) \
75 do { \
76 *raw_cpu_ptr(&(pcp)) op val; \
77 } while (0)
79 #define raw_cpu_generic_add_return(pcp, val) \
80 ({ \
81 typeof(&(pcp)) __p = raw_cpu_ptr(&(pcp)); \
83 *__p += val; \
84 *__p; \
87 #define raw_cpu_generic_xchg(pcp, nval) \
88 ({ \
89 typeof(&(pcp)) __p = raw_cpu_ptr(&(pcp)); \
90 typeof(pcp) __ret; \
91 __ret = *__p; \
92 *__p = nval; \
93 __ret; \
96 #define raw_cpu_generic_cmpxchg(pcp, oval, nval) \
97 ({ \
98 typeof(&(pcp)) __p = raw_cpu_ptr(&(pcp)); \
99 typeof(pcp) __ret; \
100 __ret = *__p; \
101 if (__ret == (oval)) \
102 *__p = nval; \
103 __ret; \
106 #define raw_cpu_generic_cmpxchg_double(pcp1, pcp2, oval1, oval2, nval1, nval2) \
107 ({ \
108 typeof(&(pcp1)) __p1 = raw_cpu_ptr(&(pcp1)); \
109 typeof(&(pcp2)) __p2 = raw_cpu_ptr(&(pcp2)); \
110 int __ret = 0; \
111 if (*__p1 == (oval1) && *__p2 == (oval2)) { \
112 *__p1 = nval1; \
113 *__p2 = nval2; \
114 __ret = 1; \
116 (__ret); \
119 #define __this_cpu_generic_read_nopreempt(pcp) \
120 ({ \
121 typeof(pcp) __ret; \
122 preempt_disable_notrace(); \
123 __ret = READ_ONCE(*raw_cpu_ptr(&(pcp))); \
124 preempt_enable_notrace(); \
125 __ret; \
128 #define __this_cpu_generic_read_noirq(pcp) \
129 ({ \
130 typeof(pcp) __ret; \
131 unsigned long __flags; \
132 raw_local_irq_save(__flags); \
133 __ret = raw_cpu_generic_read(pcp); \
134 raw_local_irq_restore(__flags); \
135 __ret; \
138 #define this_cpu_generic_read(pcp) \
139 ({ \
140 typeof(pcp) __ret; \
141 if (__native_word(pcp)) \
142 __ret = __this_cpu_generic_read_nopreempt(pcp); \
143 else \
144 __ret = __this_cpu_generic_read_noirq(pcp); \
145 __ret; \
148 #define this_cpu_generic_to_op(pcp, val, op) \
149 do { \
150 unsigned long __flags; \
151 raw_local_irq_save(__flags); \
152 raw_cpu_generic_to_op(pcp, val, op); \
153 raw_local_irq_restore(__flags); \
154 } while (0)
157 #define this_cpu_generic_add_return(pcp, val) \
158 ({ \
159 typeof(pcp) __ret; \
160 unsigned long __flags; \
161 raw_local_irq_save(__flags); \
162 __ret = raw_cpu_generic_add_return(pcp, val); \
163 raw_local_irq_restore(__flags); \
164 __ret; \
167 #define this_cpu_generic_xchg(pcp, nval) \
168 ({ \
169 typeof(pcp) __ret; \
170 unsigned long __flags; \
171 raw_local_irq_save(__flags); \
172 __ret = raw_cpu_generic_xchg(pcp, nval); \
173 raw_local_irq_restore(__flags); \
174 __ret; \
177 #define this_cpu_generic_cmpxchg(pcp, oval, nval) \
178 ({ \
179 typeof(pcp) __ret; \
180 unsigned long __flags; \
181 raw_local_irq_save(__flags); \
182 __ret = raw_cpu_generic_cmpxchg(pcp, oval, nval); \
183 raw_local_irq_restore(__flags); \
184 __ret; \
187 #define this_cpu_generic_cmpxchg_double(pcp1, pcp2, oval1, oval2, nval1, nval2) \
188 ({ \
189 int __ret; \
190 unsigned long __flags; \
191 raw_local_irq_save(__flags); \
192 __ret = raw_cpu_generic_cmpxchg_double(pcp1, pcp2, \
193 oval1, oval2, nval1, nval2); \
194 raw_local_irq_restore(__flags); \
195 __ret; \
198 #ifndef raw_cpu_read_1
199 #define raw_cpu_read_1(pcp) raw_cpu_generic_read(pcp)
200 #endif
201 #ifndef raw_cpu_read_2
202 #define raw_cpu_read_2(pcp) raw_cpu_generic_read(pcp)
203 #endif
204 #ifndef raw_cpu_read_4
205 #define raw_cpu_read_4(pcp) raw_cpu_generic_read(pcp)
206 #endif
207 #ifndef raw_cpu_read_8
208 #define raw_cpu_read_8(pcp) raw_cpu_generic_read(pcp)
209 #endif
211 #ifndef raw_cpu_write_1
212 #define raw_cpu_write_1(pcp, val) raw_cpu_generic_to_op(pcp, val, =)
213 #endif
214 #ifndef raw_cpu_write_2
215 #define raw_cpu_write_2(pcp, val) raw_cpu_generic_to_op(pcp, val, =)
216 #endif
217 #ifndef raw_cpu_write_4
218 #define raw_cpu_write_4(pcp, val) raw_cpu_generic_to_op(pcp, val, =)
219 #endif
220 #ifndef raw_cpu_write_8
221 #define raw_cpu_write_8(pcp, val) raw_cpu_generic_to_op(pcp, val, =)
222 #endif
224 #ifndef raw_cpu_add_1
225 #define raw_cpu_add_1(pcp, val) raw_cpu_generic_to_op(pcp, val, +=)
226 #endif
227 #ifndef raw_cpu_add_2
228 #define raw_cpu_add_2(pcp, val) raw_cpu_generic_to_op(pcp, val, +=)
229 #endif
230 #ifndef raw_cpu_add_4
231 #define raw_cpu_add_4(pcp, val) raw_cpu_generic_to_op(pcp, val, +=)
232 #endif
233 #ifndef raw_cpu_add_8
234 #define raw_cpu_add_8(pcp, val) raw_cpu_generic_to_op(pcp, val, +=)
235 #endif
237 #ifndef raw_cpu_and_1
238 #define raw_cpu_and_1(pcp, val) raw_cpu_generic_to_op(pcp, val, &=)
239 #endif
240 #ifndef raw_cpu_and_2
241 #define raw_cpu_and_2(pcp, val) raw_cpu_generic_to_op(pcp, val, &=)
242 #endif
243 #ifndef raw_cpu_and_4
244 #define raw_cpu_and_4(pcp, val) raw_cpu_generic_to_op(pcp, val, &=)
245 #endif
246 #ifndef raw_cpu_and_8
247 #define raw_cpu_and_8(pcp, val) raw_cpu_generic_to_op(pcp, val, &=)
248 #endif
250 #ifndef raw_cpu_or_1
251 #define raw_cpu_or_1(pcp, val) raw_cpu_generic_to_op(pcp, val, |=)
252 #endif
253 #ifndef raw_cpu_or_2
254 #define raw_cpu_or_2(pcp, val) raw_cpu_generic_to_op(pcp, val, |=)
255 #endif
256 #ifndef raw_cpu_or_4
257 #define raw_cpu_or_4(pcp, val) raw_cpu_generic_to_op(pcp, val, |=)
258 #endif
259 #ifndef raw_cpu_or_8
260 #define raw_cpu_or_8(pcp, val) raw_cpu_generic_to_op(pcp, val, |=)
261 #endif
263 #ifndef raw_cpu_add_return_1
264 #define raw_cpu_add_return_1(pcp, val) raw_cpu_generic_add_return(pcp, val)
265 #endif
266 #ifndef raw_cpu_add_return_2
267 #define raw_cpu_add_return_2(pcp, val) raw_cpu_generic_add_return(pcp, val)
268 #endif
269 #ifndef raw_cpu_add_return_4
270 #define raw_cpu_add_return_4(pcp, val) raw_cpu_generic_add_return(pcp, val)
271 #endif
272 #ifndef raw_cpu_add_return_8
273 #define raw_cpu_add_return_8(pcp, val) raw_cpu_generic_add_return(pcp, val)
274 #endif
276 #ifndef raw_cpu_xchg_1
277 #define raw_cpu_xchg_1(pcp, nval) raw_cpu_generic_xchg(pcp, nval)
278 #endif
279 #ifndef raw_cpu_xchg_2
280 #define raw_cpu_xchg_2(pcp, nval) raw_cpu_generic_xchg(pcp, nval)
281 #endif
282 #ifndef raw_cpu_xchg_4
283 #define raw_cpu_xchg_4(pcp, nval) raw_cpu_generic_xchg(pcp, nval)
284 #endif
285 #ifndef raw_cpu_xchg_8
286 #define raw_cpu_xchg_8(pcp, nval) raw_cpu_generic_xchg(pcp, nval)
287 #endif
289 #ifndef raw_cpu_cmpxchg_1
290 #define raw_cpu_cmpxchg_1(pcp, oval, nval) \
291 raw_cpu_generic_cmpxchg(pcp, oval, nval)
292 #endif
293 #ifndef raw_cpu_cmpxchg_2
294 #define raw_cpu_cmpxchg_2(pcp, oval, nval) \
295 raw_cpu_generic_cmpxchg(pcp, oval, nval)
296 #endif
297 #ifndef raw_cpu_cmpxchg_4
298 #define raw_cpu_cmpxchg_4(pcp, oval, nval) \
299 raw_cpu_generic_cmpxchg(pcp, oval, nval)
300 #endif
301 #ifndef raw_cpu_cmpxchg_8
302 #define raw_cpu_cmpxchg_8(pcp, oval, nval) \
303 raw_cpu_generic_cmpxchg(pcp, oval, nval)
304 #endif
306 #ifndef raw_cpu_cmpxchg_double_1
307 #define raw_cpu_cmpxchg_double_1(pcp1, pcp2, oval1, oval2, nval1, nval2) \
308 raw_cpu_generic_cmpxchg_double(pcp1, pcp2, oval1, oval2, nval1, nval2)
309 #endif
310 #ifndef raw_cpu_cmpxchg_double_2
311 #define raw_cpu_cmpxchg_double_2(pcp1, pcp2, oval1, oval2, nval1, nval2) \
312 raw_cpu_generic_cmpxchg_double(pcp1, pcp2, oval1, oval2, nval1, nval2)
313 #endif
314 #ifndef raw_cpu_cmpxchg_double_4
315 #define raw_cpu_cmpxchg_double_4(pcp1, pcp2, oval1, oval2, nval1, nval2) \
316 raw_cpu_generic_cmpxchg_double(pcp1, pcp2, oval1, oval2, nval1, nval2)
317 #endif
318 #ifndef raw_cpu_cmpxchg_double_8
319 #define raw_cpu_cmpxchg_double_8(pcp1, pcp2, oval1, oval2, nval1, nval2) \
320 raw_cpu_generic_cmpxchg_double(pcp1, pcp2, oval1, oval2, nval1, nval2)
321 #endif
323 #ifndef this_cpu_read_1
324 #define this_cpu_read_1(pcp) this_cpu_generic_read(pcp)
325 #endif
326 #ifndef this_cpu_read_2
327 #define this_cpu_read_2(pcp) this_cpu_generic_read(pcp)
328 #endif
329 #ifndef this_cpu_read_4
330 #define this_cpu_read_4(pcp) this_cpu_generic_read(pcp)
331 #endif
332 #ifndef this_cpu_read_8
333 #define this_cpu_read_8(pcp) this_cpu_generic_read(pcp)
334 #endif
336 #ifndef this_cpu_write_1
337 #define this_cpu_write_1(pcp, val) this_cpu_generic_to_op(pcp, val, =)
338 #endif
339 #ifndef this_cpu_write_2
340 #define this_cpu_write_2(pcp, val) this_cpu_generic_to_op(pcp, val, =)
341 #endif
342 #ifndef this_cpu_write_4
343 #define this_cpu_write_4(pcp, val) this_cpu_generic_to_op(pcp, val, =)
344 #endif
345 #ifndef this_cpu_write_8
346 #define this_cpu_write_8(pcp, val) this_cpu_generic_to_op(pcp, val, =)
347 #endif
349 #ifndef this_cpu_add_1
350 #define this_cpu_add_1(pcp, val) this_cpu_generic_to_op(pcp, val, +=)
351 #endif
352 #ifndef this_cpu_add_2
353 #define this_cpu_add_2(pcp, val) this_cpu_generic_to_op(pcp, val, +=)
354 #endif
355 #ifndef this_cpu_add_4
356 #define this_cpu_add_4(pcp, val) this_cpu_generic_to_op(pcp, val, +=)
357 #endif
358 #ifndef this_cpu_add_8
359 #define this_cpu_add_8(pcp, val) this_cpu_generic_to_op(pcp, val, +=)
360 #endif
362 #ifndef this_cpu_and_1
363 #define this_cpu_and_1(pcp, val) this_cpu_generic_to_op(pcp, val, &=)
364 #endif
365 #ifndef this_cpu_and_2
366 #define this_cpu_and_2(pcp, val) this_cpu_generic_to_op(pcp, val, &=)
367 #endif
368 #ifndef this_cpu_and_4
369 #define this_cpu_and_4(pcp, val) this_cpu_generic_to_op(pcp, val, &=)
370 #endif
371 #ifndef this_cpu_and_8
372 #define this_cpu_and_8(pcp, val) this_cpu_generic_to_op(pcp, val, &=)
373 #endif
375 #ifndef this_cpu_or_1
376 #define this_cpu_or_1(pcp, val) this_cpu_generic_to_op(pcp, val, |=)
377 #endif
378 #ifndef this_cpu_or_2
379 #define this_cpu_or_2(pcp, val) this_cpu_generic_to_op(pcp, val, |=)
380 #endif
381 #ifndef this_cpu_or_4
382 #define this_cpu_or_4(pcp, val) this_cpu_generic_to_op(pcp, val, |=)
383 #endif
384 #ifndef this_cpu_or_8
385 #define this_cpu_or_8(pcp, val) this_cpu_generic_to_op(pcp, val, |=)
386 #endif
388 #ifndef this_cpu_add_return_1
389 #define this_cpu_add_return_1(pcp, val) this_cpu_generic_add_return(pcp, val)
390 #endif
391 #ifndef this_cpu_add_return_2
392 #define this_cpu_add_return_2(pcp, val) this_cpu_generic_add_return(pcp, val)
393 #endif
394 #ifndef this_cpu_add_return_4
395 #define this_cpu_add_return_4(pcp, val) this_cpu_generic_add_return(pcp, val)
396 #endif
397 #ifndef this_cpu_add_return_8
398 #define this_cpu_add_return_8(pcp, val) this_cpu_generic_add_return(pcp, val)
399 #endif
401 #ifndef this_cpu_xchg_1
402 #define this_cpu_xchg_1(pcp, nval) this_cpu_generic_xchg(pcp, nval)
403 #endif
404 #ifndef this_cpu_xchg_2
405 #define this_cpu_xchg_2(pcp, nval) this_cpu_generic_xchg(pcp, nval)
406 #endif
407 #ifndef this_cpu_xchg_4
408 #define this_cpu_xchg_4(pcp, nval) this_cpu_generic_xchg(pcp, nval)
409 #endif
410 #ifndef this_cpu_xchg_8
411 #define this_cpu_xchg_8(pcp, nval) this_cpu_generic_xchg(pcp, nval)
412 #endif
414 #ifndef this_cpu_cmpxchg_1
415 #define this_cpu_cmpxchg_1(pcp, oval, nval) \
416 this_cpu_generic_cmpxchg(pcp, oval, nval)
417 #endif
418 #ifndef this_cpu_cmpxchg_2
419 #define this_cpu_cmpxchg_2(pcp, oval, nval) \
420 this_cpu_generic_cmpxchg(pcp, oval, nval)
421 #endif
422 #ifndef this_cpu_cmpxchg_4
423 #define this_cpu_cmpxchg_4(pcp, oval, nval) \
424 this_cpu_generic_cmpxchg(pcp, oval, nval)
425 #endif
426 #ifndef this_cpu_cmpxchg_8
427 #define this_cpu_cmpxchg_8(pcp, oval, nval) \
428 this_cpu_generic_cmpxchg(pcp, oval, nval)
429 #endif
431 #ifndef this_cpu_cmpxchg_double_1
432 #define this_cpu_cmpxchg_double_1(pcp1, pcp2, oval1, oval2, nval1, nval2) \
433 this_cpu_generic_cmpxchg_double(pcp1, pcp2, oval1, oval2, nval1, nval2)
434 #endif
435 #ifndef this_cpu_cmpxchg_double_2
436 #define this_cpu_cmpxchg_double_2(pcp1, pcp2, oval1, oval2, nval1, nval2) \
437 this_cpu_generic_cmpxchg_double(pcp1, pcp2, oval1, oval2, nval1, nval2)
438 #endif
439 #ifndef this_cpu_cmpxchg_double_4
440 #define this_cpu_cmpxchg_double_4(pcp1, pcp2, oval1, oval2, nval1, nval2) \
441 this_cpu_generic_cmpxchg_double(pcp1, pcp2, oval1, oval2, nval1, nval2)
442 #endif
443 #ifndef this_cpu_cmpxchg_double_8
444 #define this_cpu_cmpxchg_double_8(pcp1, pcp2, oval1, oval2, nval1, nval2) \
445 this_cpu_generic_cmpxchg_double(pcp1, pcp2, oval1, oval2, nval1, nval2)
446 #endif
448 #endif /* _ASM_GENERIC_PERCPU_H_ */