* [PATCH v2 1/2] maple_tree: Fix mas_spanning_rebalance() on insufficient data
2022-12-19 16:20 [PATCH v2 0/2] maple_tree: Fix mas_spanning_rebalance() on Liam Howlett
@ 2022-12-19 16:20 ` Liam Howlett
2022-12-19 16:20 ` [PATCH v2 2/2] test_maple_tree: Add test for " Liam Howlett
1 sibling, 0 replies; 3+ messages in thread
From: Liam Howlett @ 2022-12-19 16:20 UTC (permalink / raw)
To: linux-mm, linux-kernel, Andrew Morton, maple-tree
Cc: stable, Liam Howlett, Andrei Vagin, usama.anjum, Mike Rapoport
Mike Rapoport contacted me off-list with a regression in running criu.
Periodic tests fail with an RCU stall during execution. Although rare,
it is possible to hit this with other uses so this patch should be
backported to fix the regression.
An insufficient node was causing an out-of-bounds access on the node in
mas_leaf_max_gap(). The cause was the faulty detection of the new node
being a root node when overwriting many entries at the end of the tree.
Fix the detection of a new root and ensure there is sufficient data
prior to entering the spanning rebalance loop.
Cc: Andrei Vagin <avagin@gmail.com>
Cc: usama.anjum@collabora.com
CC: stable@vger.kernel.org
Reported-and-tested-by: Mike Rapoport <rppt@kernel.org>
Fixes: 54a611b60590 ("Maple Tree: add new data structure")
Signed-off-by: Liam R. Howlett <Liam.Howlett@oracle.com>
---
lib/maple_tree.c | 4 +++-
1 file changed, 3 insertions(+), 1 deletion(-)
diff --git a/lib/maple_tree.c b/lib/maple_tree.c
index fe3947b80069..26e2045d3cda 100644
--- a/lib/maple_tree.c
+++ b/lib/maple_tree.c
@@ -2994,7 +2994,9 @@ static int mas_spanning_rebalance(struct ma_state *mas,
mast->free = &free;
mast->destroy = &destroy;
l_mas.node = r_mas.node = m_mas.node = MAS_NONE;
- if (!(mast->orig_l->min && mast->orig_r->max == ULONG_MAX) &&
+
+ /* Check if this is not root and has sufficient data. */
+ if (((mast->orig_l->min != 0) || (mast->orig_r->max != ULONG_MAX)) &&
unlikely(mast->bn->b_end <= mt_min_slots[mast->bn->type]))
mast_spanning_rebalance(mast);
--
2.35.1
^ permalink raw reply [flat|nested] 3+ messages in thread* [PATCH v2 2/2] test_maple_tree: Add test for mas_spanning_rebalance() on insufficient data
2022-12-19 16:20 [PATCH v2 0/2] maple_tree: Fix mas_spanning_rebalance() on Liam Howlett
2022-12-19 16:20 ` [PATCH v2 1/2] maple_tree: Fix mas_spanning_rebalance() on insufficient data Liam Howlett
@ 2022-12-19 16:20 ` Liam Howlett
1 sibling, 0 replies; 3+ messages in thread
From: Liam Howlett @ 2022-12-19 16:20 UTC (permalink / raw)
To: linux-mm, linux-kernel, Andrew Morton, maple-tree
Cc: stable, Liam Howlett, Andrei Vagin, usama.anjum, Mike Rapoport
Add a test to the maple tree test suite for the spanning rebalance
insufficient node issue does not go undetected again.
Cc: Andrei Vagin <avagin@gmail.com>
Cc: usama.anjum@collabora.com
CC: stable@vger.kernel.org
Cc: Mike Rapoport <rppt@kernel.org>
Signed-off-by: Liam R. Howlett <Liam.Howlett@oracle.com>
---
lib/test_maple_tree.c | 23 +++++++++++++++++++++++
1 file changed, 23 insertions(+)
diff --git a/lib/test_maple_tree.c b/lib/test_maple_tree.c
index f425f169ef08..497fc93ccf9e 100644
--- a/lib/test_maple_tree.c
+++ b/lib/test_maple_tree.c
@@ -2498,6 +2498,25 @@ static noinline void check_dup(struct maple_tree *mt)
}
}
+static noinline void check_bnode_min_spanning(struct maple_tree *mt)
+{
+ int i = 50;
+ MA_STATE(mas, mt, 0, 0);
+
+ mt_set_non_kernel(9999);
+ mas_lock(&mas);
+ do {
+ mas_set_range(&mas, i*10, i*10+9);
+ mas_store(&mas, check_bnode_min_spanning);
+ } while (i--);
+
+ mas_set_range(&mas, 240, 509);
+ mas_store(&mas, NULL);
+ mas_unlock(&mas);
+ mas_destroy(&mas);
+ mt_set_non_kernel(0);
+}
+
static DEFINE_MTREE(tree);
static int maple_tree_seed(void)
{
@@ -2742,6 +2761,10 @@ static int maple_tree_seed(void)
check_dup(&tree);
mtree_destroy(&tree);
+ mt_init_flags(&tree, MT_FLAGS_ALLOC_RANGE);
+ check_bnode_min_spanning(&tree);
+ mtree_destroy(&tree);
+
#if defined(BENCH)
skip:
#endif
--
2.35.1
^ permalink raw reply [flat|nested] 3+ messages in thread