]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
fs/ntfs3: fix wrong LCN in run_remove_range() when splitting a run
authorZhan Xusheng <zhanxusheng1024@gmail.com>
Fri, 8 May 2026 09:52:45 +0000 (17:52 +0800)
committerKonstantin Komarov <almaz.alexandrovich@paragon-software.com>
Tue, 2 Jun 2026 15:02:28 +0000 (17:02 +0200)
When run_remove_range() removes a middle portion of a non-sparse run,
it splits the run into head and tail parts.  The tail is inserted via
run_add_entry() but uses the original r->lcn as its starting LCN
instead of advancing it by the split offset.

For example, removing VCN range [10, 20) from a run
{vcn=0, lcn=100, len=30} should produce:
  {vcn=0,  lcn=100, len=10}   (head)
  {vcn=20, lcn=120, len=10}   (tail, lcn advanced by 20)

But the current code produces:
  {vcn=0,  lcn=100, len=10}
  {vcn=20, lcn=100, len=10}   (wrong: points to same physical clusters)

This creates overlapping physical mappings in the in-memory run tree,
which can corrupt cluster allocation decisions and lead to data
corruption.

The correct pattern is already used in run_insert_range():
  CLST lcn2 = r->lcn == SPARSE_LCN ? SPARSE_LCN : (r->lcn + len1);

Apply the same logic in run_remove_range().

Fixes: 10d7c95af043 ("fs/ntfs3: add delayed-allocation (delalloc) support")
Signed-off-by: Zhan Xusheng <zhanxusheng@xiaomi.com>
Signed-off-by: Konstantin Komarov <almaz.alexandrovich@paragon-software.com>
fs/ntfs3/run.c

index 19aa044fd1fcc9de43135c17db05d25d8a6c010b..ad7db67514ef7c5cc280e672334fe8d7ed47966b 100644 (file)
@@ -1297,9 +1297,12 @@ bool run_remove_range(struct runs_tree *run, CLST vcn, CLST len, CLST *done)
 
                if (r_end > end) {
                        /* Remove a middle part, split. */
+                       CLST tail_lcn = r->lcn == SPARSE_LCN ?
+                                       SPARSE_LCN : (r->lcn + (end - r->vcn));
+
                        *done += len;
                        r->len = d;
-                       return run_add_entry(run, end, r->lcn, r_end - end,
+                       return run_add_entry(run, end, tail_lcn, r_end - end,
                                             false);
                }
                /* Remove tail of run .*/