| // 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); | 
 | } |