diff options
Diffstat (limited to 'fs/f2fs/segment.h')
| -rw-r--r-- | fs/f2fs/segment.h | 81 | 
1 files changed, 52 insertions, 29 deletions
| diff --git a/fs/f2fs/segment.h b/fs/f2fs/segment.h index 269f690b4e24..5731682d7516 100644 --- a/fs/f2fs/segment.h +++ b/fs/f2fs/segment.h @@ -20,13 +20,8 @@  #define GET_L2R_SEGNO(free_i, segno)	(segno - free_i->start_segno)  #define GET_R2L_SEGNO(free_i, segno)	(segno + free_i->start_segno) -#define IS_DATASEG(t)							\ -	((t == CURSEG_HOT_DATA) || (t == CURSEG_COLD_DATA) ||		\ -	(t == CURSEG_WARM_DATA)) - -#define IS_NODESEG(t)							\ -	((t == CURSEG_HOT_NODE) || (t == CURSEG_COLD_NODE) ||		\ -	(t == CURSEG_WARM_NODE)) +#define IS_DATASEG(t)	(t <= CURSEG_COLD_DATA) +#define IS_NODESEG(t)	(t >= CURSEG_HOT_NODE)  #define IS_CURSEG(sbi, seg)						\  	((seg == CURSEG_I(sbi, CURSEG_HOT_DATA)->segno) ||	\ @@ -83,25 +78,20 @@  	(segno / SIT_ENTRY_PER_BLOCK)  #define	START_SEGNO(sit_i, segno)		\  	(SIT_BLOCK_OFFSET(sit_i, segno) * SIT_ENTRY_PER_BLOCK) +#define SIT_BLK_CNT(sbi)			\ +	((TOTAL_SEGS(sbi) + SIT_ENTRY_PER_BLOCK - 1) / SIT_ENTRY_PER_BLOCK)  #define f2fs_bitmap_size(nr)			\  	(BITS_TO_LONGS(nr) * sizeof(unsigned long))  #define TOTAL_SEGS(sbi)	(SM_I(sbi)->main_segments)  #define TOTAL_SECS(sbi)	(sbi->total_sections)  #define SECTOR_FROM_BLOCK(sbi, blk_addr)				\ -	(blk_addr << ((sbi)->log_blocksize - F2FS_LOG_SECTOR_SIZE)) +	(((sector_t)blk_addr) << (sbi)->log_sectors_per_block)  #define SECTOR_TO_BLOCK(sbi, sectors)					\ -	(sectors >> ((sbi)->log_blocksize - F2FS_LOG_SECTOR_SIZE)) +	(sectors >> (sbi)->log_sectors_per_block)  #define MAX_BIO_BLOCKS(max_hw_blocks)					\  	(min((int)max_hw_blocks, BIO_MAX_PAGES)) -/* during checkpoint, bio_private is used to synchronize the last bio */ -struct bio_private { -	struct f2fs_sb_info *sbi; -	bool is_sync; -	void *wait; -}; -  /*   * indicate a block allocation direction: RIGHT and LEFT.   * RIGHT means allocating new sections towards the end of volume. @@ -458,8 +448,8 @@ static inline int reserved_sections(struct f2fs_sb_info *sbi)  static inline bool need_SSR(struct f2fs_sb_info *sbi)  { -	return ((prefree_segments(sbi) / sbi->segs_per_sec) -			+ free_sections(sbi) < overprovision_sections(sbi)); +	return (prefree_segments(sbi) / sbi->segs_per_sec) +			+ free_sections(sbi) < overprovision_sections(sbi);  }  static inline bool has_not_enough_free_secs(struct f2fs_sb_info *sbi, int freed) @@ -467,38 +457,71 @@ static inline bool has_not_enough_free_secs(struct f2fs_sb_info *sbi, int freed)  	int node_secs = get_blocktype_secs(sbi, F2FS_DIRTY_NODES);  	int dent_secs = get_blocktype_secs(sbi, F2FS_DIRTY_DENTS); -	if (sbi->por_doing) +	if (unlikely(sbi->por_doing))  		return false; -	return ((free_sections(sbi) + freed) <= (node_secs + 2 * dent_secs + -						reserved_sections(sbi))); +	return (free_sections(sbi) + freed) <= (node_secs + 2 * dent_secs + +						reserved_sections(sbi));  }  static inline bool excess_prefree_segs(struct f2fs_sb_info *sbi)  { -	return (prefree_segments(sbi) > SM_I(sbi)->rec_prefree_segments); +	return prefree_segments(sbi) > SM_I(sbi)->rec_prefree_segments;  }  static inline int utilization(struct f2fs_sb_info *sbi)  { -	return div_u64((u64)valid_user_blocks(sbi) * 100, sbi->user_block_count); +	return div_u64((u64)valid_user_blocks(sbi) * 100, +					sbi->user_block_count);  }  /*   * Sometimes f2fs may be better to drop out-of-place update policy. - * So, if fs utilization is over MIN_IPU_UTIL, then f2fs tries to write - * data in the original place likewise other traditional file systems. - * But, currently set 100 in percentage, which means it is disabled. - * See below need_inplace_update(). + * And, users can control the policy through sysfs entries. + * There are five policies with triggering conditions as follows. + * F2FS_IPU_FORCE - all the time, + * F2FS_IPU_SSR - if SSR mode is activated, + * F2FS_IPU_UTIL - if FS utilization is over threashold, + * F2FS_IPU_SSR_UTIL - if SSR mode is activated and FS utilization is over + *                     threashold, + * F2FS_IPUT_DISABLE - disable IPU. (=default option)   */ -#define MIN_IPU_UTIL		100 +#define DEF_MIN_IPU_UTIL	70 + +enum { +	F2FS_IPU_FORCE, +	F2FS_IPU_SSR, +	F2FS_IPU_UTIL, +	F2FS_IPU_SSR_UTIL, +	F2FS_IPU_DISABLE, +}; +  static inline bool need_inplace_update(struct inode *inode)  {  	struct f2fs_sb_info *sbi = F2FS_SB(inode->i_sb); + +	/* IPU can be done only for the user data */  	if (S_ISDIR(inode->i_mode))  		return false; -	if (need_SSR(sbi) && utilization(sbi) > MIN_IPU_UTIL) + +	switch (SM_I(sbi)->ipu_policy) { +	case F2FS_IPU_FORCE:  		return true; +	case F2FS_IPU_SSR: +		if (need_SSR(sbi)) +			return true; +		break; +	case F2FS_IPU_UTIL: +		if (utilization(sbi) > SM_I(sbi)->min_ipu_util) +			return true; +		break; +	case F2FS_IPU_SSR_UTIL: +		if (need_SSR(sbi) && utilization(sbi) > SM_I(sbi)->min_ipu_util) +			return true; +		break; +	case F2FS_IPU_DISABLE: +		break; +	}  	return false;  } | 
