Wed Jul 24 03:35:50 2013 UTC ()
Add several operations to <linux/atomic.h>.

atomic_add_unless
atomic_clear_mask
atomic_inc_not_zero
atomic_set_mask
atomic_sub


(riastradh)
diff -r1.1.2.8 -r1.1.2.9 src/sys/external/bsd/drm2/include/linux/atomic.h

cvs diff -r1.1.2.8 -r1.1.2.9 src/sys/external/bsd/drm2/include/linux/atomic.h (expand / switch to context diff)
--- src/sys/external/bsd/drm2/include/linux/atomic.h 2013/07/24 02:28:36 1.1.2.8
+++ src/sys/external/bsd/drm2/include/linux/atomic.h 2013/07/24 03:35:50 1.1.2.9
@@ -1,4 +1,4 @@
-/*	$NetBSD: atomic.h,v 1.1.2.8 2013/07/24 02:28:36 riastradh Exp $	*/
+/*	$NetBSD: atomic.h,v 1.1.2.9 2013/07/24 03:35:50 riastradh Exp $	*/
 
 /*-
  * Copyright (c) 2013 The NetBSD Foundation, Inc.
@@ -61,6 +61,12 @@
 	atomic_add_int(&atomic->a_u.au_uint, addend);
 }
 
+static inline void
+atomic_sub(int subtrahend, atomic_t *atomic)
+{
+	atomic_add_int(&atomic->a_u.au_uint, -subtrahend);
+}
+
 static inline int
 atomic_add_return(int addend, atomic_t *atomic)
 {
@@ -83,6 +89,39 @@
 atomic_dec_and_test(atomic_t *atomic)
 {
 	return (-1 == (int)atomic_dec_uint_nv(&atomic->a_u.au_uint));
+}
+
+static inline void
+atomic_set_mask(unsigned long mask, atomic_t *atomic)
+{
+	atomic_or_uint(&atomic->a_u.au_uint, mask);
+}
+
+static inline void
+atomic_clear_mask(unsigned long mask, atomic_t *atomic)
+{
+	atomic_and_uint(&atomic->a_u.au_uint, ~mask);
+}
+
+static inline int
+atomic_add_unless(atomic_t *atomic, int addend, int zero)
+{
+	int value;
+
+	do {
+		value = atomic->a_u.au_int;
+		if (value == zero)
+			return 0;
+	} while (atomic_cas_uint(&atomic->a_u.au_uint, value, (value + addend))
+	    != value);
+
+	return 1;
+}
+
+static inline int
+atomic_inc_not_zero(atomic_t *atomic)
+{
+	return atomic_add_unless(atomic, 1, 0);
 }
 
 static inline void