aboutsummaryrefslogtreecommitdiffstats
path: root/fs/f2fs
diff options
context:
space:
mode:
authorChao Yu <chao2.yu@samsung.com>2015-01-19 07:24:37 -0500
committerJaegeuk Kim <jaegeuk@kernel.org>2015-02-11 20:04:33 -0500
commit1601839e9e5bd5726d744c9c5919f87dc808bbcc (patch)
tree59ffa63672ff5c0faa4510ac4b48bcace7ccf63b /fs/f2fs
parent85dc2f2c6c84e99e9864ef660f79683aaad85f42 (diff)
f2fs: fix to release count of meta page in ->invalidatepage
We will encounter deadloop in below scenario: 1. increase page count for F2FS_DIRTY_META type in following path: ->recover_fsync_data ->recover_data ->do_recover_data ->recover_data_page ->change_curseg ->write_sum_page ->set_page_dirty 2. fail in recover_data() 3. invalidate meta pages in truncate_inode_pages_final without decreasing page count. 4. deadloop when sync_meta_pages as page count will always be non-zero. message: NMI watchdog: BUG: soft lockup - CPU#0 stuck for 22s! [<c1129a37>] pagevec_lookup_tag+0x27/0x30 [<f0e774c7>] sync_meta_pages+0x87/0x160 [f2fs] [<f0e86dd9>] recover_fsync_data+0xeb9/0xf10 [f2fs] [<f0e75398>] f2fs_fill_super+0x888/0x980 [f2fs] [<c11733ca>] mount_bdev+0x16a/0x1a0 [<f0e7180f>] f2fs_mount+0x1f/0x30 [f2fs] [<c1173da6>] mount_fs+0x36/0x170 [<c118b6f5>] vfs_kern_mount+0x55/0xe0 [<c118d63f>] do_mount+0x1df/0x9f0 [<c118e110>] SyS_mount+0x70/0xb0 [<c15a0c48>] sysenter_do_call+0x12/0x12 To avoid page count leak, let's add ->invalidatepage and ->releasepage in f2fs_meta_aops as f2fs_node_aops to release meta page count correctly. Signed-off-by: Chao Yu <chao2.yu@samsung.com> Signed-off-by: Jaegeuk Kim <jaegeuk@kernel.org>
Diffstat (limited to 'fs/f2fs')
-rw-r--r--fs/f2fs/checkpoint.c19
1 files changed, 19 insertions, 0 deletions
diff --git a/fs/f2fs/checkpoint.c b/fs/f2fs/checkpoint.c
index 231d8c9fea0a..79f82819086f 100644
--- a/fs/f2fs/checkpoint.c
+++ b/fs/f2fs/checkpoint.c
@@ -302,16 +302,35 @@ static int f2fs_set_meta_page_dirty(struct page *page)
302 if (!PageDirty(page)) { 302 if (!PageDirty(page)) {
303 __set_page_dirty_nobuffers(page); 303 __set_page_dirty_nobuffers(page);
304 inc_page_count(F2FS_P_SB(page), F2FS_DIRTY_META); 304 inc_page_count(F2FS_P_SB(page), F2FS_DIRTY_META);
305 SetPagePrivate(page);
305 f2fs_trace_pid(page); 306 f2fs_trace_pid(page);
306 return 1; 307 return 1;
307 } 308 }
308 return 0; 309 return 0;
309} 310}
310 311
312static void f2fs_invalidate_meta_page(struct page *page, unsigned int offset,
313 unsigned int length)
314{
315 struct inode *inode = page->mapping->host;
316
317 if (PageDirty(page))
318 dec_page_count(F2FS_I_SB(inode), F2FS_DIRTY_META);
319 ClearPagePrivate(page);
320}
321
322static int f2fs_release_meta_page(struct page *page, gfp_t wait)
323{
324 ClearPagePrivate(page);
325 return 1;
326}
327
311const struct address_space_operations f2fs_meta_aops = { 328const struct address_space_operations f2fs_meta_aops = {
312 .writepage = f2fs_write_meta_page, 329 .writepage = f2fs_write_meta_page,
313 .writepages = f2fs_write_meta_pages, 330 .writepages = f2fs_write_meta_pages,
314 .set_page_dirty = f2fs_set_meta_page_dirty, 331 .set_page_dirty = f2fs_set_meta_page_dirty,
332 .invalidatepage = f2fs_invalidate_meta_page,
333 .releasepage = f2fs_release_meta_page,
315}; 334};
316 335
317static void __add_ino_entry(struct f2fs_sb_info *sbi, nid_t ino, int type) 336static void __add_ino_entry(struct f2fs_sb_info *sbi, nid_t ino, int type)