diff options
Diffstat (limited to 'fs/f2fs/node.c')
| -rw-r--r-- | fs/f2fs/node.c | 89 | 
1 files changed, 57 insertions, 32 deletions
| diff --git a/fs/f2fs/node.c b/fs/f2fs/node.c index dd2e45a661aa..d338740d0fda 100644 --- a/fs/f2fs/node.c +++ b/fs/f2fs/node.c @@ -1,12 +1,9 @@ +// SPDX-License-Identifier: GPL-2.0  /*   * fs/f2fs/node.c   *   * Copyright (c) 2012 Samsung Electronics Co., Ltd.   *             http://www.samsung.com/ - * - * This program is free software; you can redistribute it and/or modify - * it under the terms of the GNU General Public License version 2 as - * published by the Free Software Foundation.   */  #include <linux/fs.h>  #include <linux/f2fs_fs.h> @@ -104,7 +101,7 @@ bool f2fs_available_free_memory(struct f2fs_sb_info *sbi, int type)  static void clear_node_page_dirty(struct page *page)  {  	if (PageDirty(page)) { -		f2fs_clear_radix_tree_dirty_tag(page); +		f2fs_clear_page_cache_dirty_tag(page);  		clear_page_dirty_for_io(page);  		dec_page_count(F2FS_P_SB(page), F2FS_DIRTY_NODES);  	} @@ -129,6 +126,8 @@ static struct page *get_next_nat_page(struct f2fs_sb_info *sbi, nid_t nid)  	/* get current nat block page with lock */  	src_page = get_current_nat_page(sbi, nid); +	if (IS_ERR(src_page)) +		return src_page;  	dst_page = f2fs_grab_meta_page(sbi, dst_off);  	f2fs_bug_on(sbi, PageDirty(src_page)); @@ -1307,9 +1306,7 @@ void f2fs_ra_node_page(struct f2fs_sb_info *sbi, nid_t nid)  	if (f2fs_check_nid_range(sbi, nid))  		return; -	rcu_read_lock(); -	apage = radix_tree_lookup(&NODE_MAPPING(sbi)->i_pages, nid); -	rcu_read_unlock(); +	apage = xa_load(&NODE_MAPPING(sbi)->i_pages, nid);  	if (apage)  		return; @@ -1542,8 +1539,10 @@ static int __write_node_page(struct page *page, bool atomic, bool *submitted,  	}  	if (__is_valid_data_blkaddr(ni.blk_addr) && -		!f2fs_is_valid_blkaddr(sbi, ni.blk_addr, DATA_GENERIC)) +		!f2fs_is_valid_blkaddr(sbi, ni.blk_addr, DATA_GENERIC)) { +		up_read(&sbi->node_write);  		goto redirty_out; +	}  	if (atomic && !test_opt(sbi, NOBARRIER))  		fio.op_flags |= REQ_PREFLUSH | REQ_FUA; @@ -1564,8 +1563,7 @@ static int __write_node_page(struct page *page, bool atomic, bool *submitted,  	up_read(&sbi->node_write);  	if (wbc->for_reclaim) { -		f2fs_submit_merged_write_cond(sbi, page->mapping->host, 0, -						page->index, NODE); +		f2fs_submit_merged_write_cond(sbi, NULL, page, 0, NODE);  		submitted = NULL;  	} @@ -1587,8 +1585,10 @@ redirty_out:  	return AOP_WRITEPAGE_ACTIVATE;  } -void f2fs_move_node_page(struct page *node_page, int gc_type) +int f2fs_move_node_page(struct page *node_page, int gc_type)  { +	int err = 0; +  	if (gc_type == FG_GC) {  		struct writeback_control wbc = {  			.sync_mode = WB_SYNC_ALL, @@ -1600,12 +1600,16 @@ void f2fs_move_node_page(struct page *node_page, int gc_type)  		f2fs_wait_on_page_writeback(node_page, NODE, true);  		f2fs_bug_on(F2FS_P_SB(node_page), PageWriteback(node_page)); -		if (!clear_page_dirty_for_io(node_page)) +		if (!clear_page_dirty_for_io(node_page)) { +			err = -EAGAIN;  			goto out_page; +		}  		if (__write_node_page(node_page, false, NULL, -					&wbc, false, FS_GC_NODE_IO, NULL)) +					&wbc, false, FS_GC_NODE_IO, NULL)) { +			err = -EAGAIN;  			unlock_page(node_page); +		}  		goto release_page;  	} else {  		/* set page dirty and write it */ @@ -1616,6 +1620,7 @@ out_page:  	unlock_page(node_page);  release_page:  	f2fs_put_page(node_page, 0); +	return err;  }  static int f2fs_write_node_page(struct page *page, @@ -1630,13 +1635,13 @@ int f2fs_fsync_node_pages(struct f2fs_sb_info *sbi, struct inode *inode,  			unsigned int *seq_id)  {  	pgoff_t index; -	pgoff_t last_idx = ULONG_MAX;  	struct pagevec pvec;  	int ret = 0;  	struct page *last_page = NULL;  	bool marked = false;  	nid_t ino = inode->i_ino;  	int nr_pages; +	int nwritten = 0;  	if (atomic) {  		last_page = last_fsync_dnode(sbi, ino); @@ -1714,7 +1719,7 @@ continue_unlock:  				f2fs_put_page(last_page, 0);  				break;  			} else if (submitted) { -				last_idx = page->index; +				nwritten++;  			}  			if (page == last_page) { @@ -1740,8 +1745,8 @@ continue_unlock:  		goto retry;  	}  out: -	if (last_idx != ULONG_MAX) -		f2fs_submit_merged_write_cond(sbi, NULL, ino, last_idx, NODE); +	if (nwritten) +		f2fs_submit_merged_write_cond(sbi, NULL, NULL, ino, NODE);  	return ret ? -EIO: 0;  } @@ -2268,15 +2273,19 @@ static int __f2fs_build_free_nids(struct f2fs_sb_info *sbi,  						nm_i->nat_block_bitmap)) {  			struct page *page = get_current_nat_page(sbi, nid); -			ret = scan_nat_page(sbi, page, nid); -			f2fs_put_page(page, 1); +			if (IS_ERR(page)) { +				ret = PTR_ERR(page); +			} else { +				ret = scan_nat_page(sbi, page, nid); +				f2fs_put_page(page, 1); +			}  			if (ret) {  				up_read(&nm_i->nat_tree_lock);  				f2fs_bug_on(sbi, !mount);  				f2fs_msg(sbi->sb, KERN_ERR,  					"NAT is corrupt, run fsck to fix it"); -				return -EINVAL; +				return ret;  			}  		} @@ -2353,8 +2362,9 @@ retry:  	spin_unlock(&nm_i->nid_list_lock);  	/* Let's scan nat pages and its caches to get free nids */ -	f2fs_build_free_nids(sbi, true, false); -	goto retry; +	if (!f2fs_build_free_nids(sbi, true, false)) +		goto retry; +	return false;  }  /* @@ -2537,7 +2547,7 @@ retry:  	if (!PageUptodate(ipage))  		SetPageUptodate(ipage);  	fill_node_footer(ipage, ino, ino, 0, true); -	set_cold_node(page, false); +	set_cold_node(ipage, false);  	src = F2FS_INODE(page);  	dst = F2FS_INODE(ipage); @@ -2560,6 +2570,13 @@ retry:  			F2FS_FITS_IN_INODE(src, le16_to_cpu(src->i_extra_isize),  								i_projid))  			dst->i_projid = src->i_projid; + +		if (f2fs_sb_has_inode_crtime(sbi->sb) && +			F2FS_FITS_IN_INODE(src, le16_to_cpu(src->i_extra_isize), +							i_crtime_nsec)) { +			dst->i_crtime = src->i_crtime; +			dst->i_crtime_nsec = src->i_crtime_nsec; +		}  	}  	new_ni = old_ni; @@ -2703,7 +2720,7 @@ static void __update_nat_bits(struct f2fs_sb_info *sbi, nid_t start_nid,  		__clear_bit_le(nat_index, nm_i->full_nat_bits);  } -static void __flush_nat_entry_set(struct f2fs_sb_info *sbi, +static int __flush_nat_entry_set(struct f2fs_sb_info *sbi,  		struct nat_entry_set *set, struct cp_control *cpc)  {  	struct curseg_info *curseg = CURSEG_I(sbi, CURSEG_HOT_DATA); @@ -2727,6 +2744,9 @@ static void __flush_nat_entry_set(struct f2fs_sb_info *sbi,  		down_write(&curseg->journal_rwsem);  	} else {  		page = get_next_nat_page(sbi, start_nid); +		if (IS_ERR(page)) +			return PTR_ERR(page); +  		nat_blk = page_address(page);  		f2fs_bug_on(sbi, !nat_blk);  	} @@ -2772,12 +2792,13 @@ static void __flush_nat_entry_set(struct f2fs_sb_info *sbi,  		radix_tree_delete(&NM_I(sbi)->nat_set_root, set->set);  		kmem_cache_free(nat_entry_set_slab, set);  	} +	return 0;  }  /*   * This function is called during the checkpointing process.   */ -void f2fs_flush_nat_entries(struct f2fs_sb_info *sbi, struct cp_control *cpc) +int f2fs_flush_nat_entries(struct f2fs_sb_info *sbi, struct cp_control *cpc)  {  	struct f2fs_nm_info *nm_i = NM_I(sbi);  	struct curseg_info *curseg = CURSEG_I(sbi, CURSEG_HOT_DATA); @@ -2787,6 +2808,7 @@ void f2fs_flush_nat_entries(struct f2fs_sb_info *sbi, struct cp_control *cpc)  	unsigned int found;  	nid_t set_idx = 0;  	LIST_HEAD(sets); +	int err = 0;  	/* during unmount, let's flush nat_bits before checking dirty_nat_cnt */  	if (enabled_nat_bits(sbi, cpc)) { @@ -2796,7 +2818,7 @@ void f2fs_flush_nat_entries(struct f2fs_sb_info *sbi, struct cp_control *cpc)  	}  	if (!nm_i->dirty_nat_cnt) -		return; +		return 0;  	down_write(&nm_i->nat_tree_lock); @@ -2819,11 +2841,16 @@ void f2fs_flush_nat_entries(struct f2fs_sb_info *sbi, struct cp_control *cpc)  	}  	/* flush dirty nats in nat entry set */ -	list_for_each_entry_safe(set, tmp, &sets, set_list) -		__flush_nat_entry_set(sbi, set, cpc); +	list_for_each_entry_safe(set, tmp, &sets, set_list) { +		err = __flush_nat_entry_set(sbi, set, cpc); +		if (err) +			break; +	}  	up_write(&nm_i->nat_tree_lock);  	/* Allow dirty nats by node block allocation in write_begin */ + +	return err;  }  static int __get_nat_bitmaps(struct f2fs_sb_info *sbi) @@ -2850,10 +2877,8 @@ static int __get_nat_bitmaps(struct f2fs_sb_info *sbi)  		struct page *page;  		page = f2fs_get_meta_page(sbi, nat_bits_addr++); -		if (IS_ERR(page)) { -			disable_nat_bits(sbi, true); +		if (IS_ERR(page))  			return PTR_ERR(page); -		}  		memcpy(nm_i->nat_bits + (i << F2FS_BLKSIZE_BITS),  					page_address(page), F2FS_BLKSIZE); | 
