CVE-2026-45892 PUBLISHED

ext4: drop extent cache after doing PARTIAL_VALID1 zeroout

Assigner: Linux
Reserved: 13.05.2026 Published: 27.05.2026 Updated: 27.05.2026

In the Linux kernel, the following vulnerability has been resolved:

ext4: drop extent cache after doing PARTIAL_VALID1 zeroout

When splitting an unwritten extent in the middle and converting it to initialized in ext4_split_extent() with the EXT4_EXT_MAY_ZEROOUT and EXT4_EXT_DATA_VALID2 flags set, it could leave a stale unwritten extent.

Assume we have an unwritten file and buffered write in the middle of it without dioread_nolock enabled, it will allocate blocks as written extent.

<pre> 0 A B N [UUUUUUUUUUUU] on-disk extent U: unwritten extent [UUUUUUUUUUUU] extent status tree [--DDDDDDDD--] D: valid data |<- ->| ----> this range needs to be initialized </pre>

ext4_split_extent() first try to split this extent at B with EXT4_EXT_DATA_PARTIAL_VALID1 and EXT4_EXT_MAY_ZEROOUT flag set, but ext4_split_extent_at() failed to split this extent due to temporary lack of space. It zeroout B to N and leave the entire extent as unwritten.

<pre> 0 A B N [UUUUUUUUUUUU] on-disk extent [UUUUUUUUUUUU] extent status tree [--DDDDDDDDZZ] Z: zeroed data </pre>

ext4_split_extent() then try to split this extent at A with EXT4_EXT_DATA_VALID2 flag set. This time, it split successfully and leave an written extent from A to N.

<pre> 0 A B N [UUWWWWWWWWWW] on-disk extent W: written extent [UUUUUUUUUUUU] extent status tree [--DDDDDDDDZZ] </pre>

Finally ext4_map_create_blocks() only insert extent A to B to the extent status tree, and leave an stale unwritten extent in the status tree.

<pre> 0 A B N [UUWWWWWWWWWW] on-disk extent W: written extent [UUWWWWWWWWUU] extent status tree [--DDDDDDDDZZ] </pre>

Fix this issue by always cached extent status entry after zeroing out the second part.

Product Status

Vendor Linux
Product Linux
Versions Default: unaffected
  • affected from 1da177e4c3f41524e886b7f1b8a0c1fc7321cac2 to f0931a5c17005a0c4fc35bd1a001245effc3354b (excl.)
  • affected from 1da177e4c3f41524e886b7f1b8a0c1fc7321cac2 to d8ee559fccdef713f058cfe5f2c03dc9b18be3b1 (excl.)
  • affected from 1da177e4c3f41524e886b7f1b8a0c1fc7321cac2 to c2ee51d684adca7645e4aa74adca13f6750390bc (excl.)
  • affected from 1da177e4c3f41524e886b7f1b8a0c1fc7321cac2 to a1b962a821e7a52d48212ae269b45808b4411267 (excl.)
  • affected from 1da177e4c3f41524e886b7f1b8a0c1fc7321cac2 to 6d882ea3b0931b43530d44149b79fcd4ffc13030 (excl.)
  • affected from 0 to 6.6.130 (excl.)
  • affected from 0 to 6.12.75 (excl.)
  • affected from 0 to 6.18.14 (excl.)
  • affected from 0 to 6.19.4 (excl.)
Vendor Linux
Product Linux
Versions Default: affected
  • unaffected from 6.6.130 to 6.6.* (incl.)
  • unaffected from 6.12.75 to 6.12.* (incl.)
  • unaffected from 6.18.14 to 6.18.* (incl.)
  • unaffected from 6.19.4 to 6.19.* (incl.)
  • unaffected from 7.0 to * (incl.)

References