Commit 76341cab authored by Jeff Layton's avatar Jeff Layton

jbd2: don't clear and reset errors after waiting on writeback

Resetting this flag is almost certainly racy, and will be problematic
with some coming changes.

Make filemap_fdatawait_keep_errors return int, but not clear the flag(s).
Have jbd2 call it instead of filemap_fdatawait and don't attempt to
re-set the error flag if it fails.
Reviewed-by: default avatarJan Kara <jack@suse.cz>
Reviewed-by: default avatarCarlos Maiolino <cmaiolino@redhat.com>
Signed-off-by: default avatarJeff Layton <jlayton@redhat.com>
parent 87354e5d
...@@ -263,18 +263,10 @@ static int journal_finish_inode_data_buffers(journal_t *journal, ...@@ -263,18 +263,10 @@ static int journal_finish_inode_data_buffers(journal_t *journal,
continue; continue;
jinode->i_flags |= JI_COMMIT_RUNNING; jinode->i_flags |= JI_COMMIT_RUNNING;
spin_unlock(&journal->j_list_lock); spin_unlock(&journal->j_list_lock);
err = filemap_fdatawait(jinode->i_vfs_inode->i_mapping); err = filemap_fdatawait_keep_errors(
if (err) { jinode->i_vfs_inode->i_mapping);
/* if (!ret)
* Because AS_EIO is cleared by ret = err;
* filemap_fdatawait_range(), set it again so
* that user process can get -EIO from fsync().
*/
mapping_set_error(jinode->i_vfs_inode->i_mapping, -EIO);
if (!ret)
ret = err;
}
spin_lock(&journal->j_list_lock); spin_lock(&journal->j_list_lock);
jinode->i_flags &= ~JI_COMMIT_RUNNING; jinode->i_flags &= ~JI_COMMIT_RUNNING;
smp_mb(); smp_mb();
......
...@@ -2514,7 +2514,7 @@ extern int write_inode_now(struct inode *, int); ...@@ -2514,7 +2514,7 @@ extern int write_inode_now(struct inode *, int);
extern int filemap_fdatawrite(struct address_space *); extern int filemap_fdatawrite(struct address_space *);
extern int filemap_flush(struct address_space *); extern int filemap_flush(struct address_space *);
extern int filemap_fdatawait(struct address_space *); extern int filemap_fdatawait(struct address_space *);
extern void filemap_fdatawait_keep_errors(struct address_space *); extern int filemap_fdatawait_keep_errors(struct address_space *mapping);
extern int filemap_fdatawait_range(struct address_space *, loff_t lstart, extern int filemap_fdatawait_range(struct address_space *, loff_t lstart,
loff_t lend); loff_t lend);
extern int filemap_write_and_wait(struct address_space *mapping); extern int filemap_write_and_wait(struct address_space *mapping);
......
...@@ -309,6 +309,16 @@ int filemap_check_errors(struct address_space *mapping) ...@@ -309,6 +309,16 @@ int filemap_check_errors(struct address_space *mapping)
} }
EXPORT_SYMBOL(filemap_check_errors); EXPORT_SYMBOL(filemap_check_errors);
static int filemap_check_and_keep_errors(struct address_space *mapping)
{
/* Check for outstanding write errors */
if (test_bit(AS_EIO, &mapping->flags))
return -EIO;
if (test_bit(AS_ENOSPC, &mapping->flags))
return -ENOSPC;
return 0;
}
/** /**
* __filemap_fdatawrite_range - start writeback on mapping dirty pages in range * __filemap_fdatawrite_range - start writeback on mapping dirty pages in range
* @mapping: address space structure to write * @mapping: address space structure to write
...@@ -453,15 +463,17 @@ EXPORT_SYMBOL(filemap_fdatawait_range); ...@@ -453,15 +463,17 @@ EXPORT_SYMBOL(filemap_fdatawait_range);
* call sites are system-wide / filesystem-wide data flushers: e.g. sync(2), * call sites are system-wide / filesystem-wide data flushers: e.g. sync(2),
* fsfreeze(8) * fsfreeze(8)
*/ */
void filemap_fdatawait_keep_errors(struct address_space *mapping) int filemap_fdatawait_keep_errors(struct address_space *mapping)
{ {
loff_t i_size = i_size_read(mapping->host); loff_t i_size = i_size_read(mapping->host);
if (i_size == 0) if (i_size == 0)
return; return 0;
__filemap_fdatawait_range(mapping, 0, i_size - 1); __filemap_fdatawait_range(mapping, 0, i_size - 1);
return filemap_check_and_keep_errors(mapping);
} }
EXPORT_SYMBOL(filemap_fdatawait_keep_errors);
/** /**
* filemap_fdatawait - wait for all under-writeback pages to complete * filemap_fdatawait - wait for all under-writeback pages to complete
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment