Merge branch 'locking-core-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip
Pull locking updates from Ingo Molnar: "So we have a laundry list of locking subsystem changes: - continuing barrier API and code improvements - futex enhancements - atomics API improvements - pvqspinlock enhancements: in particular lock stealing and adaptive spinning - qspinlock micro-enhancements" * 'locking-core-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip: futex: Allow FUTEX_CLOCK_REALTIME with FUTEX_WAIT op futex: Cleanup the goto confusion in requeue_pi() futex: Remove pointless put_pi_state calls in requeue() futex: Document pi_state refcounting in requeue code futex: Rename free_pi_state() to put_pi_state() futex: Drop refcount if requeue_pi() acquired the rtmutex locking/barriers, arch: Remove ambiguous statement in the smp_store_mb() documentation lcoking/barriers, arch: Use smp barriers in smp_store_release() locking/cmpxchg, arch: Remove tas() definitions locking/pvqspinlock: Queue node adaptive spinning locking/pvqspinlock: Allow limited lock stealing locking/pvqspinlock: Collect slowpath lock statistics sched/core, locking: Document Program-Order guarantees locking, sched: Introduce smp_cond_acquire() and use it locking/pvqspinlock, x86: Optimize the PV unlock code path locking/qspinlock: Avoid redundant read of next pointer locking/qspinlock: Prefetch the next node cacheline locking/qspinlock: Use _acquire/_release() versions of cmpxchg() & xchg() atomics: Add test for atomic operations with _relaxed variants
This commit is contained in:
@@ -27,6 +27,65 @@ do { \
|
||||
(unsigned long long)r); \
|
||||
} while (0)
|
||||
|
||||
/*
|
||||
* Test for a atomic operation family,
|
||||
* @test should be a macro accepting parameters (bit, op, ...)
|
||||
*/
|
||||
|
||||
#define FAMILY_TEST(test, bit, op, args...) \
|
||||
do { \
|
||||
test(bit, op, ##args); \
|
||||
test(bit, op##_acquire, ##args); \
|
||||
test(bit, op##_release, ##args); \
|
||||
test(bit, op##_relaxed, ##args); \
|
||||
} while (0)
|
||||
|
||||
#define TEST_RETURN(bit, op, c_op, val) \
|
||||
do { \
|
||||
atomic##bit##_set(&v, v0); \
|
||||
r = v0; \
|
||||
r c_op val; \
|
||||
BUG_ON(atomic##bit##_##op(val, &v) != r); \
|
||||
BUG_ON(atomic##bit##_read(&v) != r); \
|
||||
} while (0)
|
||||
|
||||
#define RETURN_FAMILY_TEST(bit, op, c_op, val) \
|
||||
do { \
|
||||
FAMILY_TEST(TEST_RETURN, bit, op, c_op, val); \
|
||||
} while (0)
|
||||
|
||||
#define TEST_ARGS(bit, op, init, ret, expect, args...) \
|
||||
do { \
|
||||
atomic##bit##_set(&v, init); \
|
||||
BUG_ON(atomic##bit##_##op(&v, ##args) != ret); \
|
||||
BUG_ON(atomic##bit##_read(&v) != expect); \
|
||||
} while (0)
|
||||
|
||||
#define XCHG_FAMILY_TEST(bit, init, new) \
|
||||
do { \
|
||||
FAMILY_TEST(TEST_ARGS, bit, xchg, init, init, new, new); \
|
||||
} while (0)
|
||||
|
||||
#define CMPXCHG_FAMILY_TEST(bit, init, new, wrong) \
|
||||
do { \
|
||||
FAMILY_TEST(TEST_ARGS, bit, cmpxchg, \
|
||||
init, init, new, init, new); \
|
||||
FAMILY_TEST(TEST_ARGS, bit, cmpxchg, \
|
||||
init, init, init, wrong, new); \
|
||||
} while (0)
|
||||
|
||||
#define INC_RETURN_FAMILY_TEST(bit, i) \
|
||||
do { \
|
||||
FAMILY_TEST(TEST_ARGS, bit, inc_return, \
|
||||
i, (i) + one, (i) + one); \
|
||||
} while (0)
|
||||
|
||||
#define DEC_RETURN_FAMILY_TEST(bit, i) \
|
||||
do { \
|
||||
FAMILY_TEST(TEST_ARGS, bit, dec_return, \
|
||||
i, (i) - one, (i) - one); \
|
||||
} while (0)
|
||||
|
||||
static __init void test_atomic(void)
|
||||
{
|
||||
int v0 = 0xaaa31337;
|
||||
@@ -45,6 +104,18 @@ static __init void test_atomic(void)
|
||||
TEST(, and, &=, v1);
|
||||
TEST(, xor, ^=, v1);
|
||||
TEST(, andnot, &= ~, v1);
|
||||
|
||||
RETURN_FAMILY_TEST(, add_return, +=, onestwos);
|
||||
RETURN_FAMILY_TEST(, add_return, +=, -one);
|
||||
RETURN_FAMILY_TEST(, sub_return, -=, onestwos);
|
||||
RETURN_FAMILY_TEST(, sub_return, -=, -one);
|
||||
|
||||
INC_RETURN_FAMILY_TEST(, v0);
|
||||
DEC_RETURN_FAMILY_TEST(, v0);
|
||||
|
||||
XCHG_FAMILY_TEST(, v0, v1);
|
||||
CMPXCHG_FAMILY_TEST(, v0, v1, onestwos);
|
||||
|
||||
}
|
||||
|
||||
#define INIT(c) do { atomic64_set(&v, c); r = c; } while (0)
|
||||
@@ -74,59 +145,26 @@ static __init void test_atomic64(void)
|
||||
TEST(64, xor, ^=, v1);
|
||||
TEST(64, andnot, &= ~, v1);
|
||||
|
||||
INIT(v0);
|
||||
r += onestwos;
|
||||
BUG_ON(atomic64_add_return(onestwos, &v) != r);
|
||||
BUG_ON(v.counter != r);
|
||||
|
||||
INIT(v0);
|
||||
r += -one;
|
||||
BUG_ON(atomic64_add_return(-one, &v) != r);
|
||||
BUG_ON(v.counter != r);
|
||||
|
||||
INIT(v0);
|
||||
r -= onestwos;
|
||||
BUG_ON(atomic64_sub_return(onestwos, &v) != r);
|
||||
BUG_ON(v.counter != r);
|
||||
|
||||
INIT(v0);
|
||||
r -= -one;
|
||||
BUG_ON(atomic64_sub_return(-one, &v) != r);
|
||||
BUG_ON(v.counter != r);
|
||||
RETURN_FAMILY_TEST(64, add_return, +=, onestwos);
|
||||
RETURN_FAMILY_TEST(64, add_return, +=, -one);
|
||||
RETURN_FAMILY_TEST(64, sub_return, -=, onestwos);
|
||||
RETURN_FAMILY_TEST(64, sub_return, -=, -one);
|
||||
|
||||
INIT(v0);
|
||||
atomic64_inc(&v);
|
||||
r += one;
|
||||
BUG_ON(v.counter != r);
|
||||
|
||||
INIT(v0);
|
||||
r += one;
|
||||
BUG_ON(atomic64_inc_return(&v) != r);
|
||||
BUG_ON(v.counter != r);
|
||||
|
||||
INIT(v0);
|
||||
atomic64_dec(&v);
|
||||
r -= one;
|
||||
BUG_ON(v.counter != r);
|
||||
|
||||
INIT(v0);
|
||||
r -= one;
|
||||
BUG_ON(atomic64_dec_return(&v) != r);
|
||||
BUG_ON(v.counter != r);
|
||||
INC_RETURN_FAMILY_TEST(64, v0);
|
||||
DEC_RETURN_FAMILY_TEST(64, v0);
|
||||
|
||||
INIT(v0);
|
||||
BUG_ON(atomic64_xchg(&v, v1) != v0);
|
||||
r = v1;
|
||||
BUG_ON(v.counter != r);
|
||||
|
||||
INIT(v0);
|
||||
BUG_ON(atomic64_cmpxchg(&v, v0, v1) != v0);
|
||||
r = v1;
|
||||
BUG_ON(v.counter != r);
|
||||
|
||||
INIT(v0);
|
||||
BUG_ON(atomic64_cmpxchg(&v, v2, v1) != v0);
|
||||
BUG_ON(v.counter != r);
|
||||
XCHG_FAMILY_TEST(64, v0, v1);
|
||||
CMPXCHG_FAMILY_TEST(64, v0, v1, v2);
|
||||
|
||||
INIT(v0);
|
||||
BUG_ON(atomic64_add_unless(&v, one, v0));
|
||||
|
Reference in New Issue
Block a user