diff options
Diffstat (limited to 'lib/test_xarray.c')
| -rw-r--r-- | lib/test_xarray.c | 52 | 
1 files changed, 52 insertions, 0 deletions
| diff --git a/lib/test_xarray.c b/lib/test_xarray.c index 0e865bab4a10..080a39d22e73 100644 --- a/lib/test_xarray.c +++ b/lib/test_xarray.c @@ -1858,6 +1858,54 @@ static void check_split_1(struct xarray *xa, unsigned long index,  	xa_destroy(xa);  } +static void check_split_2(struct xarray *xa, unsigned long index, +				unsigned int order, unsigned int new_order) +{ +	XA_STATE_ORDER(xas, xa, index, new_order); +	unsigned int i, found; +	void *entry; + +	xa_store_order(xa, index, order, xa, GFP_KERNEL); +	xa_set_mark(xa, index, XA_MARK_1); + +	/* allocate a node for xas_try_split() */ +	xas_set_err(&xas, -ENOMEM); +	XA_BUG_ON(xa, !xas_nomem(&xas, GFP_KERNEL)); + +	xas_lock(&xas); +	xas_try_split(&xas, xa, order); +	if (((new_order / XA_CHUNK_SHIFT) < (order / XA_CHUNK_SHIFT)) && +	    new_order < order - 1) { +		XA_BUG_ON(xa, !xas_error(&xas) || xas_error(&xas) != -EINVAL); +		xas_unlock(&xas); +		goto out; +	} +	for (i = 0; i < (1 << order); i += (1 << new_order)) +		__xa_store(xa, index + i, xa_mk_index(index + i), 0); +	xas_unlock(&xas); + +	for (i = 0; i < (1 << order); i++) { +		unsigned int val = index + (i & ~((1 << new_order) - 1)); +		XA_BUG_ON(xa, xa_load(xa, index + i) != xa_mk_index(val)); +	} + +	xa_set_mark(xa, index, XA_MARK_0); +	XA_BUG_ON(xa, !xa_get_mark(xa, index, XA_MARK_0)); + +	xas_set_order(&xas, index, 0); +	found = 0; +	rcu_read_lock(); +	xas_for_each_marked(&xas, entry, ULONG_MAX, XA_MARK_1) { +		found++; +		XA_BUG_ON(xa, xa_is_internal(entry)); +	} +	rcu_read_unlock(); +	XA_BUG_ON(xa, found != 1 << (order - new_order)); +out: +	xas_destroy(&xas); +	xa_destroy(xa); +} +  static noinline void check_split(struct xarray *xa)  {  	unsigned int order, new_order; @@ -1869,6 +1917,10 @@ static noinline void check_split(struct xarray *xa)  			check_split_1(xa, 0, order, new_order);  			check_split_1(xa, 1UL << order, order, new_order);  			check_split_1(xa, 3UL << order, order, new_order); + +			check_split_2(xa, 0, order, new_order); +			check_split_2(xa, 1UL << order, order, new_order); +			check_split_2(xa, 3UL << order, order, new_order);  		}  	}  } | 
