Skip to content

Commit 69b8f2f

Browse files
committed
nilfs2: prevent general protection fault in nilfs_clear_dirty_page()
In a syzbot stress test that deliberately causes file system errors on nilfs2 with a corrupted disk image, it has been reported that nilfs_clear_dirty_page() called from nilfs_clear_dirty_pages() can cause a general protection fault. In nilfs_clear_dirty_pages(), when looking up dirty pages from the page cache and calling nilfs_clear_dirty_page() for each dirty page/folio retrieved, the back reference from the argument page to "mapping" may have been changed to NULL (and possibly others). It is necessary to check this after locking the page/folio. So, fix this issue by not calling nilfs_clear_dirty_page() on a page/folio after locking it in nilfs_clear_dirty_pages() if the back reference "mapping" from the page/folio is different from the "mapping" that held the page/folio just before. Link: https://lkml.kernel.org/r/[email protected] Signed-off-by: Ryusuke Konishi <[email protected]> Reported-by: [email protected] Closes: https://lkml.kernel.org/r/[email protected] Tested-by: Ryusuke Konishi <[email protected]> Cc: <[email protected]> Signed-off-by: Andrew Morton <[email protected]> [Converted folio back to page-based] Signed-off-by: Ryusuke Konishi <[email protected]>
1 parent d661f97 commit 69b8f2f

File tree

1 file changed

+9
-1
lines changed

1 file changed

+9
-1
lines changed

fs/nilfs2/page.c

+9-1
Original file line numberDiff line numberDiff line change
@@ -375,7 +375,15 @@ void nilfs_clear_dirty_pages(struct address_space *mapping, bool silent)
375375
struct page *page = pvec.pages[i];
376376

377377
lock_page(page);
378-
nilfs_clear_dirty_page(page, silent);
378+
379+
/*
380+
* This page may have been removed from the address
381+
* space by truncation or invalidation when the lock
382+
* was acquired. Skip processing in that case.
383+
*/
384+
if (likely(page->mapping == mapping))
385+
nilfs_clear_dirty_page(page, silent);
386+
379387
unlock_page(page);
380388
}
381389
pagevec_release(&pvec);

0 commit comments

Comments
 (0)