// SPDX-FileCopyrightText: 2009 Mathieu Desnoyers // // SPDX-License-Identifier: GPL-2.0-or-later #include #include #include "tap.h" #define NR_TESTS 17 #define BYTE_PER_LONG (sizeof(unsigned long) / sizeof(unsigned char)) #define SHORT_PER_LONG (sizeof(unsigned long) / sizeof(unsigned short)) #define INT_PER_LONG (sizeof(unsigned long) / sizeof(unsigned int)) struct testvals { #ifdef UATOMIC_HAS_ATOMIC_BYTE unsigned char c[BYTE_PER_LONG]; #endif #ifdef UATOMIC_HAS_ATOMIC_SHORT unsigned short s[SHORT_PER_LONG]; #endif unsigned int i[INT_PER_LONG]; unsigned long l; }; static struct testvals vals; #define do_test(ptr) \ do { \ __typeof__(*(ptr)) v; \ \ uatomic_add(ptr, 10); \ ok1(uatomic_read(ptr) == 10); \ \ uatomic_add(ptr, -11UL); \ ok1(uatomic_read(ptr) == (__typeof__(*(ptr)))-1UL); \ \ v = uatomic_cmpxchg(ptr, -1UL, 22); \ ok1(uatomic_read(ptr) == 22); \ ok1(v == (__typeof__(*(ptr)))-1UL); \ \ v = uatomic_cmpxchg(ptr, 33, 44); \ ok1(uatomic_read(ptr) == 22); \ ok1(v == 22); \ \ v = uatomic_xchg(ptr, 55); \ ok1(uatomic_read(ptr) == 55); \ ok1(v == 22); \ \ uatomic_set(ptr, 22); \ uatomic_inc(ptr); \ ok1(uatomic_read(ptr) == 23); \ \ uatomic_dec(ptr); \ ok1(uatomic_read(ptr) == 22); \ \ v = uatomic_add_return(ptr, 74); \ ok1(v == 96); \ ok1(uatomic_read(ptr) == 96); \ \ uatomic_or(ptr, 58); \ ok1(uatomic_read(ptr) == 122); \ \ v = uatomic_sub_return(ptr, 1); \ ok1(v == 121); \ \ uatomic_sub(ptr, (unsigned int) 2); \ ok1(uatomic_read(ptr) == 119); \ \ uatomic_inc(ptr); \ uatomic_inc(ptr); \ ok1(uatomic_read(ptr) == 121); \ \ uatomic_and(ptr, 129); \ ok1(uatomic_read(ptr) == 1); \ \ } while (0) int main(void) { int nr_run = INT_PER_LONG + 1; unsigned long i; #ifdef UATOMIC_HAS_ATOMIC_BYTE nr_run += BYTE_PER_LONG; #endif #ifdef UATOMIC_HAS_ATOMIC_SHORT nr_run += SHORT_PER_LONG; #endif plan_tests(nr_run * NR_TESTS); #ifdef UATOMIC_HAS_ATOMIC_BYTE for (i = 0; i < BYTE_PER_LONG; i++) { diag("Test atomic ops on byte with %lu byte offset from long alignment", i); do_test(&vals.c[i]); } #endif #ifdef UATOMIC_HAS_ATOMIC_SHORT for (i = 0; i < SHORT_PER_LONG; i++) { diag("Test atomic ops on short with %lu byte offset from long alignment", i * sizeof(unsigned short)); do_test(&vals.s[i]); } #endif for (i = 0; i < INT_PER_LONG; i++) { diag("Test atomic ops on int with %lu byte offset from long alignment", i * sizeof(unsigned int)); do_test(&vals.i[i]); } diag("Test atomic ops on long"); do_test(&vals.l); return exit_status(); }