aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorChris Mason <chris.mason@oracle.com>2011-12-09 11:07:37 -0500
committerChris Mason <chris.mason@oracle.com>2011-12-09 11:07:37 -0500
commit5dbc8fca8ef5d719014f22345d990e957dcfc692 (patch)
treeadf84c6e10a33cf5cab21487b2443f80a8129509
parent1cf4ffdb3289624a6462c94f2ce05545b32ef736 (diff)
Btrfs: fix btrfs_end_bio to deal with write errors to a single mirror
btrfs_end_bio checks the number of errors on a bio against the max number of errors allowed before sending any EIOs up to the higher levels. If we got enough copies of the bio done for a given raid level, it is supposed to clear the bio error flag and return success. We have pointers to the original bio sent down by the higher layers and pointers to any cloned bios we made for raid purposes. If the original bio happens to be the one that got an io error, but not the last one to finish, it might not have the BIO_UPTODATE bit set. Then, when the last bio does finish, we'll call bio_end_io on the original bio. It won't have the uptodate bit set and we'll end up sending EIO to the higher layers. We already had a check for this, it just was conditional on getting the IO error on the very last bio. Make the check unconditional so we eat the EIOs properly. Signed-off-by: Chris Mason <chris.mason@oracle.com>
-rw-r--r--fs/btrfs/volumes.c2
1 files changed, 1 insertions, 1 deletions
diff --git a/fs/btrfs/volumes.c b/fs/btrfs/volumes.c
index 0a8c8f8304b1..91ea57a1474a 100644
--- a/fs/btrfs/volumes.c
+++ b/fs/btrfs/volumes.c
@@ -3258,7 +3258,7 @@ static void btrfs_end_bio(struct bio *bio, int err)
3258 */ 3258 */
3259 if (atomic_read(&bbio->error) > bbio->max_errors) { 3259 if (atomic_read(&bbio->error) > bbio->max_errors) {
3260 err = -EIO; 3260 err = -EIO;
3261 } else if (err) { 3261 } else {
3262 /* 3262 /*
3263 * this bio is actually up to date, we didn't 3263 * this bio is actually up to date, we didn't
3264 * go over the max number of errors 3264 * go over the max number of errors