diff options
Diffstat (limited to 'fs/ext4/inode.c')
| -rw-r--r-- | fs/ext4/inode.c | 27 | 
1 files changed, 23 insertions, 4 deletions
| diff --git a/fs/ext4/inode.c b/fs/ext4/inode.c index d47264cafee0..986e2388f031 100644 --- a/fs/ext4/inode.c +++ b/fs/ext4/inode.c @@ -120,6 +120,9 @@ void ext4_evict_inode(struct inode *inode)  	int err;  	trace_ext4_evict_inode(inode); + +	ext4_ioend_wait(inode); +  	if (inode->i_nlink) {  		/*  		 * When journalling data dirty buffers are tracked only in the @@ -644,7 +647,7 @@ struct buffer_head *ext4_bread(handle_t *handle, struct inode *inode,  		return bh;  	if (buffer_uptodate(bh))  		return bh; -	ll_rw_block(READ_META, 1, &bh); +	ll_rw_block(READ | REQ_META | REQ_PRIO, 1, &bh);  	wait_on_buffer(bh);  	if (buffer_uptodate(bh))  		return bh; @@ -983,6 +986,8 @@ static int ext4_journalled_write_end(struct file *file,  	from = pos & (PAGE_CACHE_SIZE - 1);  	to = from + len; +	BUG_ON(!ext4_handle_valid(handle)); +  	if (copied < len) {  		if (!PageUptodate(page))  			copied = 0; @@ -1283,7 +1288,12 @@ static int mpage_da_submit_io(struct mpage_da_data *mpd,  			else if (test_opt(inode->i_sb, MBLK_IO_SUBMIT))  				err = ext4_bio_write_page(&io_submit, page,  							  len, mpd->wbc); -			else +			else if (buffer_uninit(page_bufs)) { +				ext4_set_bh_endio(page_bufs, inode); +				err = block_write_full_page_endio(page, +					noalloc_get_block_write, +					mpd->wbc, ext4_end_io_buffer_write); +			} else  				err = block_write_full_page(page,  					noalloc_get_block_write, mpd->wbc); @@ -1699,6 +1709,8 @@ static int __ext4_journalled_writepage(struct page *page,  		goto out;  	} +	BUG_ON(!ext4_handle_valid(handle)); +  	ret = walk_page_buffers(handle, page_bufs, 0, len, NULL,  				do_journal_get_write_access); @@ -2668,8 +2680,15 @@ static void ext4_end_io_buffer_write(struct buffer_head *bh, int uptodate)  		goto out;  	} -	io_end->flag = EXT4_IO_END_UNWRITTEN; +	/* +	 * It may be over-defensive here to check EXT4_IO_END_UNWRITTEN now, +	 * but being more careful is always safe for the future change. +	 */  	inode = io_end->inode; +	if (!(io_end->flag & EXT4_IO_END_UNWRITTEN)) { +		io_end->flag |= EXT4_IO_END_UNWRITTEN; +		atomic_inc(&EXT4_I(inode)->i_aiodio_unwritten); +	}  	/* Add the io_end to per-inode completed io list*/  	spin_lock_irqsave(&EXT4_I(inode)->i_completed_io_lock, flags); @@ -3279,7 +3298,7 @@ make_io:  		trace_ext4_load_inode(inode);  		get_bh(bh);  		bh->b_end_io = end_buffer_read_sync; -		submit_bh(READ_META, bh); +		submit_bh(READ | REQ_META | REQ_PRIO, bh);  		wait_on_buffer(bh);  		if (!buffer_uptodate(bh)) {  			EXT4_ERROR_INODE_BLOCK(inode, block, | 
