diff options
Diffstat (limited to 'fs/ext4/extents.c')
| -rw-r--r-- | fs/ext4/extents.c | 45 | 
1 files changed, 30 insertions, 15 deletions
| diff --git a/fs/ext4/extents.c b/fs/ext4/extents.c index 35f65cf4f318..3384dc4bed40 100644 --- a/fs/ext4/extents.c +++ b/fs/ext4/extents.c @@ -360,8 +360,10 @@ static int ext4_valid_extent(struct inode *inode, struct ext4_extent *ext)  {  	ext4_fsblk_t block = ext4_ext_pblock(ext);  	int len = ext4_ext_get_actual_len(ext); +	ext4_lblk_t lblock = le32_to_cpu(ext->ee_block); +	ext4_lblk_t last = lblock + len - 1; -	if (len == 0) +	if (lblock > last)  		return 0;  	return ext4_data_block_valid(EXT4_SB(inode->i_sb), block, len);  } @@ -387,11 +389,26 @@ static int ext4_valid_extent_entries(struct inode *inode,  	if (depth == 0) {  		/* leaf entries */  		struct ext4_extent *ext = EXT_FIRST_EXTENT(eh); +		struct ext4_super_block *es = EXT4_SB(inode->i_sb)->s_es; +		ext4_fsblk_t pblock = 0; +		ext4_lblk_t lblock = 0; +		ext4_lblk_t prev = 0; +		int len = 0;  		while (entries) {  			if (!ext4_valid_extent(inode, ext))  				return 0; + +			/* Check for overlapping extents */ +			lblock = le32_to_cpu(ext->ee_block); +			len = ext4_ext_get_actual_len(ext); +			if ((lblock <= prev) && prev) { +				pblock = ext4_ext_pblock(ext); +				es->s_last_error_block = cpu_to_le64(pblock); +				return 0; +			}  			ext++;  			entries--; +			prev = lblock + len - 1;  		}  	} else {  		struct ext4_extent_idx *ext_idx = EXT_FIRST_INDEX(eh); @@ -1834,8 +1851,7 @@ static unsigned int ext4_ext_check_overlap(struct ext4_sb_info *sbi,  	depth = ext_depth(inode);  	if (!path[depth].p_ext)  		goto out; -	b2 = le32_to_cpu(path[depth].p_ext->ee_block); -	b2 &= ~(sbi->s_cluster_ratio - 1); +	b2 = EXT4_LBLK_CMASK(sbi, le32_to_cpu(path[depth].p_ext->ee_block));  	/*  	 * get the next allocated block if the extent in the path @@ -1845,7 +1861,7 @@ static unsigned int ext4_ext_check_overlap(struct ext4_sb_info *sbi,  		b2 = ext4_ext_next_allocated_block(path);  		if (b2 == EXT_MAX_BLOCKS)  			goto out; -		b2 &= ~(sbi->s_cluster_ratio - 1); +		b2 = EXT4_LBLK_CMASK(sbi, b2);  	}  	/* check for wrap through zero on extent logical start block*/ @@ -2504,7 +2520,7 @@ static int ext4_remove_blocks(handle_t *handle, struct inode *inode,  		 * extent, we have to mark the cluster as used (store negative  		 * cluster number in partial_cluster).  		 */ -		unaligned = pblk & (sbi->s_cluster_ratio - 1); +		unaligned = EXT4_PBLK_COFF(sbi, pblk);  		if (unaligned && (ee_len == num) &&  		    (*partial_cluster != -((long long)EXT4_B2C(sbi, pblk))))  			*partial_cluster = EXT4_B2C(sbi, pblk); @@ -2598,7 +2614,7 @@ ext4_ext_rm_leaf(handle_t *handle, struct inode *inode,  			 * accidentally freeing it later on  			 */  			pblk = ext4_ext_pblock(ex); -			if (pblk & (sbi->s_cluster_ratio - 1)) +			if (EXT4_PBLK_COFF(sbi, pblk))  				*partial_cluster =  					-((long long)EXT4_B2C(sbi, pblk));  			ex--; @@ -3753,7 +3769,7 @@ int ext4_find_delalloc_cluster(struct inode *inode, ext4_lblk_t lblk)  {  	struct ext4_sb_info *sbi = EXT4_SB(inode->i_sb);  	ext4_lblk_t lblk_start, lblk_end; -	lblk_start = lblk & (~(sbi->s_cluster_ratio - 1)); +	lblk_start = EXT4_LBLK_CMASK(sbi, lblk);  	lblk_end = lblk_start + sbi->s_cluster_ratio - 1;  	return ext4_find_delalloc_range(inode, lblk_start, lblk_end); @@ -3812,9 +3828,9 @@ get_reserved_cluster_alloc(struct inode *inode, ext4_lblk_t lblk_start,  	trace_ext4_get_reserved_cluster_alloc(inode, lblk_start, num_blks);  	/* Check towards left side */ -	c_offset = lblk_start & (sbi->s_cluster_ratio - 1); +	c_offset = EXT4_LBLK_COFF(sbi, lblk_start);  	if (c_offset) { -		lblk_from = lblk_start & (~(sbi->s_cluster_ratio - 1)); +		lblk_from = EXT4_LBLK_CMASK(sbi, lblk_start);  		lblk_to = lblk_from + c_offset - 1;  		if (ext4_find_delalloc_range(inode, lblk_from, lblk_to)) @@ -3822,7 +3838,7 @@ get_reserved_cluster_alloc(struct inode *inode, ext4_lblk_t lblk_start,  	}  	/* Now check towards right. */ -	c_offset = (lblk_start + num_blks) & (sbi->s_cluster_ratio - 1); +	c_offset = EXT4_LBLK_COFF(sbi, lblk_start + num_blks);  	if (allocated_clusters && c_offset) {  		lblk_from = lblk_start + num_blks;  		lblk_to = lblk_from + (sbi->s_cluster_ratio - c_offset) - 1; @@ -4030,7 +4046,7 @@ static int get_implied_cluster_alloc(struct super_block *sb,  				     struct ext4_ext_path *path)  {  	struct ext4_sb_info *sbi = EXT4_SB(sb); -	ext4_lblk_t c_offset = map->m_lblk & (sbi->s_cluster_ratio-1); +	ext4_lblk_t c_offset = EXT4_LBLK_COFF(sbi, map->m_lblk);  	ext4_lblk_t ex_cluster_start, ex_cluster_end;  	ext4_lblk_t rr_cluster_start;  	ext4_lblk_t ee_block = le32_to_cpu(ex->ee_block); @@ -4048,8 +4064,7 @@ static int get_implied_cluster_alloc(struct super_block *sb,  	    (rr_cluster_start == ex_cluster_start)) {  		if (rr_cluster_start == ex_cluster_end)  			ee_start += ee_len - 1; -		map->m_pblk = (ee_start & ~(sbi->s_cluster_ratio - 1)) + -			c_offset; +		map->m_pblk = EXT4_PBLK_CMASK(sbi, ee_start) + c_offset;  		map->m_len = min(map->m_len,  				 (unsigned) sbi->s_cluster_ratio - c_offset);  		/* @@ -4203,7 +4218,7 @@ int ext4_ext_map_blocks(handle_t *handle, struct inode *inode,  	 */  	map->m_flags &= ~EXT4_MAP_FROM_CLUSTER;  	newex.ee_block = cpu_to_le32(map->m_lblk); -	cluster_offset = map->m_lblk & (sbi->s_cluster_ratio-1); +	cluster_offset = EXT4_LBLK_COFF(sbi, map->m_lblk);  	/*  	 * If we are doing bigalloc, check to see if the extent returned @@ -4271,7 +4286,7 @@ int ext4_ext_map_blocks(handle_t *handle, struct inode *inode,  	 * needed so that future calls to get_implied_cluster_alloc()  	 * work correctly.  	 */ -	offset = map->m_lblk & (sbi->s_cluster_ratio - 1); +	offset = EXT4_LBLK_COFF(sbi, map->m_lblk);  	ar.len = EXT4_NUM_B2C(sbi, offset+allocated);  	ar.goal -= offset;  	ar.logical -= offset; | 
