/* * Copyright (C) 2014 Stefan Kristiansson * * This file is licensed under the terms of the GNU General Public License * version 2. This program is licensed "as is" without any warranty of any * kind, whether express or implied. */ #ifndef __ASM_OPENRISC_CMPXCHG_H #define __ASM_OPENRISC_CMPXCHG_H #include /* * This function doesn't exist, so you'll get a linker error * if something tries to do an invalid cmpxchg(). */ extern void __cmpxchg_called_with_bad_pointer(void); #define __HAVE_ARCH_CMPXCHG 1 static inline unsigned long __cmpxchg(volatile void *ptr, unsigned long old, unsigned long new, int size) { if (size != 4) { __cmpxchg_called_with_bad_pointer(); return old; } __asm__ __volatile__( "1: l.lwa %0, 0(%1) \n" " l.sfeq %0, %2 \n" " l.bnf 2f \n" " l.nop \n" " l.swa 0(%1), %3 \n" " l.bnf 1b \n" " l.nop \n" "2: \n" : "=&r"(old) : "r"(ptr), "r"(old), "r"(new) : "cc", "memory"); return old; } #define cmpxchg(ptr, o, n) \ ({ \ (__typeof__(*(ptr))) __cmpxchg((ptr), \ (unsigned long)(o), \ (unsigned long)(n), \ sizeof(*(ptr))); \ }) /* * This function doesn't exist, so you'll get a linker error if * something tries to do an invalidly-sized xchg(). */ extern void __xchg_called_with_bad_pointer(void); static inline unsigned long __xchg(unsigned long val, volatile void *ptr, int size) { if (size != 4) { __xchg_called_with_bad_pointer(); return val; } __asm__ __volatile__( "1: l.lwa %0, 0(%1) \n" " l.swa 0(%1), %2 \n" " l.bnf 1b \n" " l.nop \n" : "=&r"(val) : "r"(ptr), "r"(val) : "cc", "memory"); return val; } #define xchg(ptr, with) \ ({ \ (__typeof__(*(ptr))) __xchg((unsigned long)(with), \ (ptr), \ sizeof(*(ptr))); \ }) #endif /* __ASM_OPENRISC_CMPXCHG_H */