diff options
Diffstat (limited to 'fs/f2fs/compress.c')
| -rw-r--r-- | fs/f2fs/compress.c | 84 | 
1 files changed, 33 insertions, 51 deletions
diff --git a/fs/f2fs/compress.c b/fs/f2fs/compress.c index 49121a21f749..d0c3aeba5945 100644 --- a/fs/f2fs/compress.c +++ b/fs/f2fs/compress.c @@ -154,6 +154,7 @@ void f2fs_destroy_compress_ctx(struct compress_ctx *cc, bool reuse)  	cc->rpages = NULL;  	cc->nr_rpages = 0;  	cc->nr_cpages = 0; +	cc->valid_nr_cpages = 0;  	if (!reuse)  		cc->cluster_idx = NULL_CLUSTER;  } @@ -620,7 +621,6 @@ static int f2fs_compress_pages(struct compress_ctx *cc)  	const struct f2fs_compress_ops *cops =  				f2fs_cops[fi->i_compress_algorithm];  	unsigned int max_len, new_nr_cpages; -	struct page **new_cpages;  	u32 chksum = 0;  	int i, ret; @@ -635,6 +635,7 @@ static int f2fs_compress_pages(struct compress_ctx *cc)  	max_len = COMPRESS_HEADER_SIZE + cc->clen;  	cc->nr_cpages = DIV_ROUND_UP(max_len, PAGE_SIZE); +	cc->valid_nr_cpages = cc->nr_cpages;  	cc->cpages = page_array_alloc(cc->inode, cc->nr_cpages);  	if (!cc->cpages) { @@ -685,13 +686,6 @@ static int f2fs_compress_pages(struct compress_ctx *cc)  	new_nr_cpages = DIV_ROUND_UP(cc->clen + COMPRESS_HEADER_SIZE, PAGE_SIZE); -	/* Now we're going to cut unnecessary tail pages */ -	new_cpages = page_array_alloc(cc->inode, new_nr_cpages); -	if (!new_cpages) { -		ret = -ENOMEM; -		goto out_vunmap_cbuf; -	} -  	/* zero out any unused part of the last page */  	memset(&cc->cbuf->cdata[cc->clen], 0,  			(new_nr_cpages * PAGE_SIZE) - @@ -701,10 +695,8 @@ static int f2fs_compress_pages(struct compress_ctx *cc)  	vm_unmap_ram(cc->rbuf, cc->cluster_size);  	for (i = 0; i < cc->nr_cpages; i++) { -		if (i < new_nr_cpages) { -			new_cpages[i] = cc->cpages[i]; +		if (i < new_nr_cpages)  			continue; -		}  		f2fs_compress_free_page(cc->cpages[i]);  		cc->cpages[i] = NULL;  	} @@ -712,9 +704,7 @@ static int f2fs_compress_pages(struct compress_ctx *cc)  	if (cops->destroy_compress_ctx)  		cops->destroy_compress_ctx(cc); -	page_array_free(cc->inode, cc->cpages, cc->nr_cpages); -	cc->cpages = new_cpages; -	cc->nr_cpages = new_nr_cpages; +	cc->valid_nr_cpages = new_nr_cpages;  	trace_f2fs_compress_pages_end(cc->inode, cc->cluster_idx,  							cc->clen, ret); @@ -1296,7 +1286,7 @@ static int f2fs_write_compressed_pages(struct compress_ctx *cc,  	psize = (loff_t)(cc->rpages[last_index]->index + 1) << PAGE_SHIFT; -	err = f2fs_get_node_info(fio.sbi, dn.nid, &ni); +	err = f2fs_get_node_info(fio.sbi, dn.nid, &ni, false);  	if (err)  		goto out_put_dnode; @@ -1308,14 +1298,14 @@ static int f2fs_write_compressed_pages(struct compress_ctx *cc,  	cic->magic = F2FS_COMPRESSED_PAGE_MAGIC;  	cic->inode = inode; -	atomic_set(&cic->pending_pages, cc->nr_cpages); +	atomic_set(&cic->pending_pages, cc->valid_nr_cpages);  	cic->rpages = page_array_alloc(cc->inode, cc->cluster_size);  	if (!cic->rpages)  		goto out_put_cic;  	cic->nr_rpages = cc->cluster_size; -	for (i = 0; i < cc->nr_cpages; i++) { +	for (i = 0; i < cc->valid_nr_cpages; i++) {  		f2fs_set_compressed_page(cc->cpages[i], inode,  					cc->rpages[i + 1]->index, cic);  		fio.compressed_page = cc->cpages[i]; @@ -1360,7 +1350,7 @@ static int f2fs_write_compressed_pages(struct compress_ctx *cc,  		if (fio.compr_blocks && __is_valid_data_blkaddr(blkaddr))  			fio.compr_blocks++; -		if (i > cc->nr_cpages) { +		if (i > cc->valid_nr_cpages) {  			if (__is_valid_data_blkaddr(blkaddr)) {  				f2fs_invalidate_blocks(sbi, blkaddr);  				f2fs_update_data_blkaddr(&dn, NEW_ADDR); @@ -1385,8 +1375,8 @@ unlock_continue:  	if (fio.compr_blocks)  		f2fs_i_compr_blocks_update(inode, fio.compr_blocks - 1, false); -	f2fs_i_compr_blocks_update(inode, cc->nr_cpages, true); -	add_compr_block_stat(inode, cc->nr_cpages); +	f2fs_i_compr_blocks_update(inode, cc->valid_nr_cpages, true); +	add_compr_block_stat(inode, cc->valid_nr_cpages);  	set_inode_flag(cc->inode, FI_APPEND_WRITE);  	if (cc->cluster_idx == 0) @@ -1424,9 +1414,7 @@ out_unlock_op:  	else  		f2fs_unlock_op(sbi);  out_free: -	for (i = 0; i < cc->nr_cpages; i++) { -		if (!cc->cpages[i]) -			continue; +	for (i = 0; i < cc->valid_nr_cpages; i++) {  		f2fs_compress_free_page(cc->cpages[i]);  		cc->cpages[i] = NULL;  	} @@ -1468,25 +1456,38 @@ static int f2fs_write_raw_pages(struct compress_ctx *cc,  					enum iostat_type io_type)  {  	struct address_space *mapping = cc->inode->i_mapping; -	int _submitted, compr_blocks, ret; -	int i = -1, err = 0; +	int _submitted, compr_blocks, ret, i;  	compr_blocks = f2fs_compressed_blocks(cc); -	if (compr_blocks < 0) { -		err = compr_blocks; -		goto out_err; + +	for (i = 0; i < cc->cluster_size; i++) { +		if (!cc->rpages[i]) +			continue; + +		redirty_page_for_writepage(wbc, cc->rpages[i]); +		unlock_page(cc->rpages[i]);  	} +	if (compr_blocks < 0) +		return compr_blocks; +  	for (i = 0; i < cc->cluster_size; i++) {  		if (!cc->rpages[i])  			continue;  retry_write: +		lock_page(cc->rpages[i]); +  		if (cc->rpages[i]->mapping != mapping) { +continue_unlock:  			unlock_page(cc->rpages[i]);  			continue;  		} -		BUG_ON(!PageLocked(cc->rpages[i])); +		if (!PageDirty(cc->rpages[i])) +			goto continue_unlock; + +		if (!clear_page_dirty_for_io(cc->rpages[i])) +			goto continue_unlock;  		ret = f2fs_write_single_data_page(cc->rpages[i], &_submitted,  						NULL, NULL, wbc, io_type, @@ -1501,26 +1502,15 @@ retry_write:  				 * avoid deadlock caused by cluster update race  				 * from foreground operation.  				 */ -				if (IS_NOQUOTA(cc->inode)) { -					err = 0; -					goto out_err; -				} +				if (IS_NOQUOTA(cc->inode)) +					return 0;  				ret = 0;  				cond_resched();  				congestion_wait(BLK_RW_ASYNC,  						DEFAULT_IO_TIMEOUT); -				lock_page(cc->rpages[i]); - -				if (!PageDirty(cc->rpages[i])) { -					unlock_page(cc->rpages[i]); -					continue; -				} - -				clear_page_dirty_for_io(cc->rpages[i]);  				goto retry_write;  			} -			err = ret; -			goto out_err; +			return ret;  		}  		*submitted += _submitted; @@ -1529,14 +1519,6 @@ retry_write:  	f2fs_balance_fs(F2FS_M_SB(mapping), true);  	return 0; -out_err: -	for (++i; i < cc->cluster_size; i++) { -		if (!cc->rpages[i]) -			continue; -		redirty_page_for_writepage(wbc, cc->rpages[i]); -		unlock_page(cc->rpages[i]); -	} -	return err;  }  int f2fs_write_multi_pages(struct compress_ctx *cc,  |