162306a36Sopenharmony_ci/* SPDX-License-Identifier: GPL-2.0 */
262306a36Sopenharmony_ci#ifndef __ARCH_M68K_CMPXCHG__
362306a36Sopenharmony_ci#define __ARCH_M68K_CMPXCHG__
462306a36Sopenharmony_ci
562306a36Sopenharmony_ci#include <linux/irqflags.h>
662306a36Sopenharmony_ci
762306a36Sopenharmony_ci#define __xg(type, x) ((volatile type *)(x))
862306a36Sopenharmony_ci
962306a36Sopenharmony_ciextern unsigned long __invalid_xchg_size(unsigned long, volatile void *, int);
1062306a36Sopenharmony_ci
1162306a36Sopenharmony_ci#ifndef CONFIG_RMW_INSNS
1262306a36Sopenharmony_cistatic inline unsigned long __arch_xchg(unsigned long x, volatile void * ptr, int size)
1362306a36Sopenharmony_ci{
1462306a36Sopenharmony_ci	unsigned long flags, tmp;
1562306a36Sopenharmony_ci
1662306a36Sopenharmony_ci	local_irq_save(flags);
1762306a36Sopenharmony_ci
1862306a36Sopenharmony_ci	switch (size) {
1962306a36Sopenharmony_ci	case 1:
2062306a36Sopenharmony_ci		tmp = *(u8 *)ptr;
2162306a36Sopenharmony_ci		*(u8 *)ptr = x;
2262306a36Sopenharmony_ci		x = tmp;
2362306a36Sopenharmony_ci		break;
2462306a36Sopenharmony_ci	case 2:
2562306a36Sopenharmony_ci		tmp = *(u16 *)ptr;
2662306a36Sopenharmony_ci		*(u16 *)ptr = x;
2762306a36Sopenharmony_ci		x = tmp;
2862306a36Sopenharmony_ci		break;
2962306a36Sopenharmony_ci	case 4:
3062306a36Sopenharmony_ci		tmp = *(u32 *)ptr;
3162306a36Sopenharmony_ci		*(u32 *)ptr = x;
3262306a36Sopenharmony_ci		x = tmp;
3362306a36Sopenharmony_ci		break;
3462306a36Sopenharmony_ci	default:
3562306a36Sopenharmony_ci		tmp = __invalid_xchg_size(x, ptr, size);
3662306a36Sopenharmony_ci		break;
3762306a36Sopenharmony_ci	}
3862306a36Sopenharmony_ci
3962306a36Sopenharmony_ci	local_irq_restore(flags);
4062306a36Sopenharmony_ci	return x;
4162306a36Sopenharmony_ci}
4262306a36Sopenharmony_ci#else
4362306a36Sopenharmony_cistatic inline unsigned long __arch_xchg(unsigned long x, volatile void * ptr, int size)
4462306a36Sopenharmony_ci{
4562306a36Sopenharmony_ci	switch (size) {
4662306a36Sopenharmony_ci	case 1:
4762306a36Sopenharmony_ci		__asm__ __volatile__
4862306a36Sopenharmony_ci			("moveb %2,%0\n\t"
4962306a36Sopenharmony_ci			 "1:\n\t"
5062306a36Sopenharmony_ci			 "casb %0,%1,%2\n\t"
5162306a36Sopenharmony_ci			 "jne 1b"
5262306a36Sopenharmony_ci			 : "=&d" (x) : "d" (x), "m" (*__xg(u8, ptr)) : "memory");
5362306a36Sopenharmony_ci		break;
5462306a36Sopenharmony_ci	case 2:
5562306a36Sopenharmony_ci		__asm__ __volatile__
5662306a36Sopenharmony_ci			("movew %2,%0\n\t"
5762306a36Sopenharmony_ci			 "1:\n\t"
5862306a36Sopenharmony_ci			 "casw %0,%1,%2\n\t"
5962306a36Sopenharmony_ci			 "jne 1b"
6062306a36Sopenharmony_ci			 : "=&d" (x) : "d" (x), "m" (*__xg(u16, ptr)) : "memory");
6162306a36Sopenharmony_ci		break;
6262306a36Sopenharmony_ci	case 4:
6362306a36Sopenharmony_ci		__asm__ __volatile__
6462306a36Sopenharmony_ci			("movel %2,%0\n\t"
6562306a36Sopenharmony_ci			 "1:\n\t"
6662306a36Sopenharmony_ci			 "casl %0,%1,%2\n\t"
6762306a36Sopenharmony_ci			 "jne 1b"
6862306a36Sopenharmony_ci			 : "=&d" (x) : "d" (x), "m" (*__xg(u32, ptr)) : "memory");
6962306a36Sopenharmony_ci		break;
7062306a36Sopenharmony_ci	default:
7162306a36Sopenharmony_ci		x = __invalid_xchg_size(x, ptr, size);
7262306a36Sopenharmony_ci		break;
7362306a36Sopenharmony_ci	}
7462306a36Sopenharmony_ci	return x;
7562306a36Sopenharmony_ci}
7662306a36Sopenharmony_ci#endif
7762306a36Sopenharmony_ci
7862306a36Sopenharmony_ci#define arch_xchg(ptr,x) ({(__typeof__(*(ptr)))__arch_xchg((unsigned long)(x),(ptr),sizeof(*(ptr)));})
7962306a36Sopenharmony_ci
8062306a36Sopenharmony_ci#include <asm-generic/cmpxchg-local.h>
8162306a36Sopenharmony_ci
8262306a36Sopenharmony_ci#define arch_cmpxchg64_local(ptr, o, n) __generic_cmpxchg64_local((ptr), (o), (n))
8362306a36Sopenharmony_ci
8462306a36Sopenharmony_ciextern unsigned long __invalid_cmpxchg_size(volatile void *,
8562306a36Sopenharmony_ci					    unsigned long, unsigned long, int);
8662306a36Sopenharmony_ci
8762306a36Sopenharmony_ci/*
8862306a36Sopenharmony_ci * Atomic compare and exchange.  Compare OLD with MEM, if identical,
8962306a36Sopenharmony_ci * store NEW in MEM.  Return the initial value in MEM.  Success is
9062306a36Sopenharmony_ci * indicated by comparing RETURN with OLD.
9162306a36Sopenharmony_ci */
9262306a36Sopenharmony_ci#ifdef CONFIG_RMW_INSNS
9362306a36Sopenharmony_ci
9462306a36Sopenharmony_cistatic inline unsigned long __cmpxchg(volatile void *p, unsigned long old,
9562306a36Sopenharmony_ci				      unsigned long new, int size)
9662306a36Sopenharmony_ci{
9762306a36Sopenharmony_ci	switch (size) {
9862306a36Sopenharmony_ci	case 1:
9962306a36Sopenharmony_ci		__asm__ __volatile__ ("casb %0,%2,%1"
10062306a36Sopenharmony_ci				      : "=d" (old), "=m" (*(char *)p)
10162306a36Sopenharmony_ci				      : "d" (new), "0" (old), "m" (*(char *)p));
10262306a36Sopenharmony_ci		break;
10362306a36Sopenharmony_ci	case 2:
10462306a36Sopenharmony_ci		__asm__ __volatile__ ("casw %0,%2,%1"
10562306a36Sopenharmony_ci				      : "=d" (old), "=m" (*(short *)p)
10662306a36Sopenharmony_ci				      : "d" (new), "0" (old), "m" (*(short *)p));
10762306a36Sopenharmony_ci		break;
10862306a36Sopenharmony_ci	case 4:
10962306a36Sopenharmony_ci		__asm__ __volatile__ ("casl %0,%2,%1"
11062306a36Sopenharmony_ci				      : "=d" (old), "=m" (*(int *)p)
11162306a36Sopenharmony_ci				      : "d" (new), "0" (old), "m" (*(int *)p));
11262306a36Sopenharmony_ci		break;
11362306a36Sopenharmony_ci	default:
11462306a36Sopenharmony_ci		old = __invalid_cmpxchg_size(p, old, new, size);
11562306a36Sopenharmony_ci		break;
11662306a36Sopenharmony_ci	}
11762306a36Sopenharmony_ci	return old;
11862306a36Sopenharmony_ci}
11962306a36Sopenharmony_ci
12062306a36Sopenharmony_ci#define arch_cmpxchg(ptr, o, n)						    \
12162306a36Sopenharmony_ci	({(__typeof__(*(ptr)))__cmpxchg((ptr), (unsigned long)(o),	    \
12262306a36Sopenharmony_ci			(unsigned long)(n), sizeof(*(ptr)));})
12362306a36Sopenharmony_ci#define arch_cmpxchg_local(ptr, o, n)					    \
12462306a36Sopenharmony_ci	({(__typeof__(*(ptr)))__cmpxchg((ptr), (unsigned long)(o),	    \
12562306a36Sopenharmony_ci			(unsigned long)(n), sizeof(*(ptr)));})
12662306a36Sopenharmony_ci
12762306a36Sopenharmony_ci#define arch_cmpxchg64(ptr, o, n)	arch_cmpxchg64_local((ptr), (o), (n))
12862306a36Sopenharmony_ci
12962306a36Sopenharmony_ci#else
13062306a36Sopenharmony_ci
13162306a36Sopenharmony_ci#include <asm-generic/cmpxchg.h>
13262306a36Sopenharmony_ci
13362306a36Sopenharmony_ci#endif
13462306a36Sopenharmony_ci
13562306a36Sopenharmony_ci#endif /* __ARCH_M68K_CMPXCHG__ */
136