X-Git-Url: http://git.onelab.eu/?a=blobdiff_plain;f=include%2Fasm-arm%2Fatomic.h;h=24208a2dabe82da79dbcc86cb4fc854b11657473;hb=c7b5ebbddf7bcd3651947760f423e3783bbe6573;hp=61227d1a7ac98e32d78210310bbb72e15f44fb99;hpb=5273a3df6485dc2ad6aa7ddd441b9a21970f003b;p=linux-2.6.git diff --git a/include/asm-arm/atomic.h b/include/asm-arm/atomic.h index 61227d1a7..24208a2da 100644 --- a/include/asm-arm/atomic.h +++ b/include/asm-arm/atomic.h @@ -44,65 +44,32 @@ static inline void atomic_set(atomic_t *v, int i) : "cc"); } -static inline void atomic_add(int i, atomic_t *v) -{ - unsigned long tmp, tmp2; - - __asm__ __volatile__("@ atomic_add\n" -"1: ldrex %0, [%2]\n" -" add %0, %0, %3\n" -" strex %1, %0, [%2]\n" -" teq %1, #0\n" -" bne 1b" - : "=&r" (tmp), "=&r" (tmp2) - : "r" (&v->counter), "Ir" (i) - : "cc"); -} - -static inline void atomic_sub(int i, atomic_t *v) -{ - unsigned long tmp, tmp2; - - __asm__ __volatile__("@ atomic_sub\n" -"1: ldrex %0, [%2]\n" -" sub %0, %0, %3\n" -" strex %1, %0, [%2]\n" -" teq %1, #0\n" -" bne 1b" - : "=&r" (tmp), "=&r" (tmp2) - : "r" (&v->counter), "Ir" (i) - : "cc"); -} - -#define atomic_inc(v) atomic_add(1, v) -#define atomic_dec(v) atomic_sub(1, v) - -static inline int atomic_dec_and_test(atomic_t *v) +static inline int atomic_add_return(int i, atomic_t *v) { unsigned long tmp; int result; - __asm__ __volatile__("@ atomic_dec_and_test\n" + __asm__ __volatile__("@ atomic_add_return\n" "1: ldrex %0, [%2]\n" -" sub %0, %0, #1\n" +" add %0, %0, %3\n" " strex %1, %0, [%2]\n" " teq %1, #0\n" " bne 1b" : "=&r" (result), "=&r" (tmp) - : "r" (&v->counter) + : "r" (&v->counter), "Ir" (i) : "cc"); - return result == 0; + return result; } -static inline int atomic_add_negative(int i, atomic_t *v) +static inline int atomic_sub_return(int i, atomic_t *v) { unsigned long tmp; int result; - __asm__ __volatile__("@ atomic_add_negative\n" + __asm__ __volatile__("@ atomic_sub_return\n" "1: ldrex %0, [%2]\n" -" add %0, %0, %3\n" +" sub %0, %0, %3\n" " strex %1, %0, [%2]\n" " teq %1, #0\n" " bne 1b" @@ -110,7 +77,7 @@ static inline int atomic_add_negative(int i, atomic_t *v) : "r" (&v->counter), "Ir" (i) : "cc"); - return result < 0; + return result; } static inline void atomic_clear_mask(unsigned long mask, unsigned long *addr) @@ -138,66 +105,30 @@ static inline void atomic_clear_mask(unsigned long mask, unsigned long *addr) #define atomic_set(v,i) (((v)->counter) = (i)) -static inline void atomic_add(int i, atomic_t *v) -{ - unsigned long flags; - - local_irq_save(flags); - v->counter += i; - local_irq_restore(flags); -} - -static inline void atomic_sub(int i, atomic_t *v) -{ - unsigned long flags; - - local_irq_save(flags); - v->counter -= i; - local_irq_restore(flags); -} - -static inline void atomic_inc(atomic_t *v) -{ - unsigned long flags; - - local_irq_save(flags); - v->counter += 1; - local_irq_restore(flags); -} - -static inline void atomic_dec(atomic_t *v) -{ - unsigned long flags; - - local_irq_save(flags); - v->counter -= 1; - local_irq_restore(flags); -} - -static inline int atomic_dec_and_test(atomic_t *v) +static inline int atomic_add_return(int i, atomic_t *v) { unsigned long flags; int val; local_irq_save(flags); val = v->counter; - v->counter = val -= 1; + v->counter = val += i; local_irq_restore(flags); - return val == 0; + return val; } -static inline int atomic_add_negative(int i, atomic_t *v) +static inline int atomic_sub_return(int i, atomic_t *v) { unsigned long flags; int val; local_irq_save(flags); val = v->counter; - v->counter = val += i; + v->counter = val -= i; local_irq_restore(flags); - return val < 0; + return val; } static inline void atomic_clear_mask(unsigned long mask, unsigned long *addr) @@ -211,6 +142,16 @@ static inline void atomic_clear_mask(unsigned long mask, unsigned long *addr) #endif /* __LINUX_ARM_ARCH__ */ +#define atomic_add(i, v) (void) atomic_add_return(i, v) +#define atomic_inc(v) (void) atomic_add_return(1, v) +#define atomic_sub(i, v) (void) atomic_sub_return(i, v) +#define atomic_dec(v) (void) atomic_sub_return(1, v) + +#define atomic_inc_and_test(v) (atomic_add_return(1, v) == 0) +#define atomic_dec_and_test(v) (atomic_sub_return(1, v) == 0) + +#define atomic_add_negative(i,v) (atomic_add_return(i, v) < 0) + /* Atomic operations are already serializing on ARM */ #define smp_mb__before_atomic_dec() barrier() #define smp_mb__after_atomic_dec() barrier()