aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorEric Work <work.eric@gmail.com>2015-05-19 02:26:23 -0400
committerNeilBrown <neilb@suse.de>2015-05-20 19:14:25 -0400
commita81157768a00e8cf8a7b43b5ea5cac931262374f (patch)
tree824d88d9b79c668fb18594d4fe7bcbc414ee43bc
parent487696957e3bd64ccffe62c0ac4ff7bf662785ab (diff)
md/raid0: fix restore to sector variable in raid0_make_request
The variable "sector" in "raid0_make_request()" was improperly updated by a call to "sector_div()" which modifies its first argument in place. Commit 47d68979cc968535cb87f3e5f2e6a3533ea48fbd restored this variable after the call for later re-use. Unfortunetly the restore was done after the referenced variable "bio" was advanced. This lead to the original value and the restored value being different. Here we move this line to the proper place. One observed side effect of this bug was discarding a file though unlinking would cause an unrelated file's contents to be discarded. Signed-off-by: NeilBrown <neilb@suse.de> Fixes: 47d68979cc96 ("md/raid0: fix bug with chunksize not a power of 2.") Cc: stable@vger.kernel.org (any that received above backport) URL: https://bugzilla.kernel.org/show_bug.cgi?id=98501
-rw-r--r--drivers/md/raid0.c4
1 files changed, 3 insertions, 1 deletions
diff --git a/drivers/md/raid0.c b/drivers/md/raid0.c
index 6a68ef5246d4..efb654eb5399 100644
--- a/drivers/md/raid0.c
+++ b/drivers/md/raid0.c
@@ -524,6 +524,9 @@ static void raid0_make_request(struct mddev *mddev, struct bio *bio)
524 ? (sector & (chunk_sects-1)) 524 ? (sector & (chunk_sects-1))
525 : sector_div(sector, chunk_sects)); 525 : sector_div(sector, chunk_sects));
526 526
527 /* Restore due to sector_div */
528 sector = bio->bi_iter.bi_sector;
529
527 if (sectors < bio_sectors(bio)) { 530 if (sectors < bio_sectors(bio)) {
528 split = bio_split(bio, sectors, GFP_NOIO, fs_bio_set); 531 split = bio_split(bio, sectors, GFP_NOIO, fs_bio_set);
529 bio_chain(split, bio); 532 bio_chain(split, bio);
@@ -531,7 +534,6 @@ static void raid0_make_request(struct mddev *mddev, struct bio *bio)
531 split = bio; 534 split = bio;
532 } 535 }
533 536
534 sector = bio->bi_iter.bi_sector;
535 zone = find_zone(mddev->private, &sector); 537 zone = find_zone(mddev->private, &sector);
536 tmp_dev = map_sector(mddev, zone, sector, &sector); 538 tmp_dev = map_sector(mddev, zone, sector, &sector);
537 split->bi_bdev = tmp_dev->bdev; 539 split->bi_bdev = tmp_dev->bdev;