xref: /openbmc/linux/arch/csky/include/asm/cmpxchg.h (revision a1dff44b354c0e2721aeae075a287d07daf1c76b)
1 /* SPDX-License-Identifier: GPL-2.0 */
2 
3 #ifndef __ASM_CSKY_CMPXCHG_H
4 #define __ASM_CSKY_CMPXCHG_H
5 
6 #ifdef CONFIG_SMP
7 #include <asm/barrier.h>
8 
9 extern void __bad_xchg(void);
10 
11 #define __xchg_relaxed(new, ptr, size)				\
12 ({								\
13 	__typeof__(ptr) __ptr = (ptr);				\
14 	__typeof__(new) __new = (new);				\
15 	__typeof__(*(ptr)) __ret;				\
16 	unsigned long tmp;					\
17 	switch (size) {						\
18 	case 4:							\
19 		asm volatile (					\
20 		"1:	ldex.w		%0, (%3) \n"		\
21 		"	mov		%1, %2   \n"		\
22 		"	stex.w		%1, (%3) \n"		\
23 		"	bez		%1, 1b   \n"		\
24 			: "=&r" (__ret), "=&r" (tmp)		\
25 			: "r" (__new), "r"(__ptr)		\
26 			:);					\
27 		break;						\
28 	default:						\
29 		__bad_xchg();					\
30 	}							\
31 	__ret;							\
32 })
33 
34 #define xchg_relaxed(ptr, x) \
35 		(__xchg_relaxed((x), (ptr), sizeof(*(ptr))))
36 
37 #define __cmpxchg_relaxed(ptr, old, new, size)			\
38 ({								\
39 	__typeof__(ptr) __ptr = (ptr);				\
40 	__typeof__(new) __new = (new);				\
41 	__typeof__(new) __tmp;					\
42 	__typeof__(old) __old = (old);				\
43 	__typeof__(*(ptr)) __ret;				\
44 	switch (size) {						\
45 	case 4:							\
46 		asm volatile (					\
47 		"1:	ldex.w		%0, (%3) \n"		\
48 		"	cmpne		%0, %4   \n"		\
49 		"	bt		2f       \n"		\
50 		"	mov		%1, %2   \n"		\
51 		"	stex.w		%1, (%3) \n"		\
52 		"	bez		%1, 1b   \n"		\
53 		"2:				 \n"		\
54 			: "=&r" (__ret), "=&r" (__tmp)		\
55 			: "r" (__new), "r"(__ptr), "r"(__old)	\
56 			:);					\
57 		break;						\
58 	default:						\
59 		__bad_xchg();					\
60 	}							\
61 	__ret;							\
62 })
63 
64 #define cmpxchg_relaxed(ptr, o, n) \
65 	(__cmpxchg_relaxed((ptr), (o), (n), sizeof(*(ptr))))
66 
67 #define cmpxchg(ptr, o, n) 					\
68 ({								\
69 	__typeof__(*(ptr)) __ret;				\
70 	__smp_release_fence();					\
71 	__ret = cmpxchg_relaxed(ptr, o, n);			\
72 	__smp_acquire_fence();					\
73 	__ret;							\
74 })
75 
76 #else
77 #include <asm-generic/cmpxchg.h>
78 #endif
79 
80 #endif /* __ASM_CSKY_CMPXCHG_H */
81