diff options
author | sephe <sephe@FreeBSD.org> | 2016-06-23 02:21:37 +0000 |
---|---|---|
committer | sephe <sephe@FreeBSD.org> | 2016-06-23 02:21:37 +0000 |
commit | ee9748f41f5818197d95b210279c755ad824dba7 (patch) | |
tree | 2a57196e39be8e665ce0e0b939e6b203603778ab /sys/i386 | |
parent | d5f44be4cd4d07380dce8e2eea32f6a97306d468 (diff) | |
download | FreeBSD-src-ee9748f41f5818197d95b210279c755ad824dba7.zip FreeBSD-src-ee9748f41f5818197d95b210279c755ad824dba7.tar.gz |
MFC 299912
atomic: Add testandclear on i386/amd64
Reviewed by: kib
Sponsored by: Microsoft OSTC
Differential Revision: https://reviews.freebsd.org/D6381
Diffstat (limited to 'sys/i386')
-rw-r--r-- | sys/i386/include/atomic.h | 26 |
1 files changed, 26 insertions, 0 deletions
diff --git a/sys/i386/include/atomic.h b/sys/i386/include/atomic.h index 0156b5b..6e71b05 100644 --- a/sys/i386/include/atomic.h +++ b/sys/i386/include/atomic.h @@ -86,6 +86,7 @@ void atomic_##NAME##_barr_##TYPE(volatile u_##TYPE *p, u_##TYPE v) int atomic_cmpset_int(volatile u_int *dst, u_int expect, u_int src); u_int atomic_fetchadd_int(volatile u_int *p, u_int v); int atomic_testandset_int(volatile u_int *p, u_int v); +int atomic_testandclear_int(volatile u_int *p, u_int v); #define ATOMIC_LOAD(TYPE, LOP) \ u_##TYPE atomic_load_acq_##TYPE(volatile u_##TYPE *p) @@ -224,6 +225,23 @@ atomic_testandset_int(volatile u_int *p, u_int v) return (res); } +static __inline int +atomic_testandclear_int(volatile u_int *p, u_int v) +{ + u_char res; + + __asm __volatile( + " " MPLOCKED " " + " btrl %2,%1 ; " + " setc %0 ; " + "# atomic_testandclear_int" + : "=q" (res), /* 0 */ + "+m" (*p) /* 1 */ + : "Ir" (v & 0x1f) /* 2 */ + : "cc"); + return (res); +} + /* * We assume that a = b will do atomic loads and stores. Due to the * IA32 memory model, a simple store guarantees release semantics. @@ -549,6 +567,13 @@ atomic_testandset_long(volatile u_long *p, u_int v) return (atomic_testandset_int((volatile u_int *)p, v)); } +static __inline int +atomic_testandclear_long(volatile u_long *p, u_int v) +{ + + return (atomic_testandclear_int((volatile u_int *)p, v)); +} + /* Read the current value and store a new value in the destination. */ #ifdef __GNUCLIKE_ASM @@ -675,6 +700,7 @@ u_long atomic_swap_long(volatile u_long *p, u_long v); #define atomic_readandclear_32 atomic_readandclear_int #define atomic_fetchadd_32 atomic_fetchadd_int #define atomic_testandset_32 atomic_testandset_int +#define atomic_testandclear_32 atomic_testandclear_int /* Operations on pointers. */ #define atomic_set_ptr(p, v) \ |