diff options
Diffstat (limited to 'fs/jbd2/commit.c')
| -rw-r--r-- | fs/jbd2/commit.c | 25 | 
1 files changed, 21 insertions, 4 deletions
| diff --git a/fs/jbd2/commit.c b/fs/jbd2/commit.c index 6396fe70085b..27373f5792a4 100644 --- a/fs/jbd2/commit.c +++ b/fs/jbd2/commit.c @@ -985,12 +985,29 @@ restart_loop:  		 * pagesize and it is attached to the last partial page.  		 */  		if (buffer_freed(bh) && !jh->b_next_transaction) { +			struct address_space *mapping; +  			clear_buffer_freed(bh);  			clear_buffer_jbddirty(bh); -			clear_buffer_mapped(bh); -			clear_buffer_new(bh); -			clear_buffer_req(bh); -			bh->b_bdev = NULL; + +			/* +			 * Block device buffers need to stay mapped all the +			 * time, so it is enough to clear buffer_jbddirty and +			 * buffer_freed bits. For the file mapping buffers (i.e. +			 * journalled data) we need to unmap buffer and clear +			 * more bits. We also need to be careful about the check +			 * because the data page mapping can get cleared under +			 * out hands, which alse need not to clear more bits +			 * because the page and buffers will be freed and can +			 * never be reused once we are done with them. +			 */ +			mapping = READ_ONCE(bh->b_page->mapping); +			if (mapping && !sb_is_blkdev_sb(mapping->host->i_sb)) { +				clear_buffer_mapped(bh); +				clear_buffer_new(bh); +				clear_buffer_req(bh); +				bh->b_bdev = NULL; +			}  		}  		if (buffer_jbddirty(bh)) { | 
