| // SPDX-License-Identifier: GPL-2.0-or-later |
| /* |
| * iteration_check_2.c: Check that deleting a tagged entry doesn't cause |
| * an RCU walker to finish early. |
| * Copyright (c) 2020 Oracle |
| * Author: Matthew Wilcox <willy@infradead.org> |
| */ |
| #include <pthread.h> |
| #include "test.h" |
| |
| static volatile bool test_complete; |
| |
| static void *iterator(void *arg) |
| { |
| XA_STATE(xas, arg, 0); |
| void *entry; |
| |
| rcu_register_thread(); |
| |
| while (!test_complete) { |
| xas_set(&xas, 0); |
| rcu_read_lock(); |
| xas_for_each_marked(&xas, entry, ULONG_MAX, XA_MARK_0) |
| ; |
| rcu_read_unlock(); |
| assert(xas.xa_index >= 100); |
| } |
| |
| rcu_unregister_thread(); |
| return NULL; |
| } |
| |
| static void *throbber(void *arg) |
| { |
| struct xarray *xa = arg; |
| |
| rcu_register_thread(); |
| |
| while (!test_complete) { |
| int i; |
| |
| for (i = 0; i < 100; i++) { |
| xa_store(xa, i, xa_mk_value(i), GFP_KERNEL); |
| xa_set_mark(xa, i, XA_MARK_0); |
| } |
| for (i = 0; i < 100; i++) |
| xa_erase(xa, i); |
| } |
| |
| rcu_unregister_thread(); |
| return NULL; |
| } |
| |
| void iteration_test2(unsigned test_duration) |
| { |
| pthread_t threads[2]; |
| DEFINE_XARRAY(array); |
| int i; |
| |
| printv(1, "Running iteration test 2 for %d seconds\n", test_duration); |
| |
| test_complete = false; |
| |
| xa_store(&array, 100, xa_mk_value(100), GFP_KERNEL); |
| xa_set_mark(&array, 100, XA_MARK_0); |
| |
| if (pthread_create(&threads[0], NULL, iterator, &array)) { |
| perror("create iterator thread"); |
| exit(1); |
| } |
| if (pthread_create(&threads[1], NULL, throbber, &array)) { |
| perror("create throbber thread"); |
| exit(1); |
| } |
| |
| sleep(test_duration); |
| test_complete = true; |
| |
| for (i = 0; i < 2; i++) { |
| if (pthread_join(threads[i], NULL)) { |
| perror("pthread_join"); |
| exit(1); |
| } |
| } |
| |
| xa_destroy(&array); |
| } |