提交 a9701a30 编写于 作者: N NeilBrown 提交者: Linus Torvalds

[PATCH] md: support BIO_RW_BARRIER for md/raid1

We can only accept BARRIER requests if all slaves handle
barriers, and that can, of course, change with time....

So we keep track of whether the whole array seems safe for barriers,
and also whether each individual rdev handles barriers.

We initially assumes barriers are OK.

When writing the superblock we try a barrier, and if that fails, we flag
things for no-barriers.  This will usually clear the flags fairly quickly.

If writing the superblock finds that BIO_RW_BARRIER is -ENOTSUPP, we need to
resubmit, so introduce function "md_super_wait" which waits for requests to
finish, and retries ENOTSUPP requests without the barrier flag.

When writing the real raid1, write requests which were BIO_RW_BARRIER but
which aresn't supported need to be retried.  So raid1d is enhanced to do this,
and when any bio write completes (i.e.  no retry needed) we remove it from the
r1bio, so that devices needing retry are easy to find.

We should hardly ever get -ENOTSUPP errors when writing data to the raid.
It should only happen if:
  1/ the device used to support BARRIER, but now doesn't.  Few devices
     change like this, though raid1 can!
or
  2/ the array has no persistent superblock, so there was no opportunity to
     pre-test for barriers when writing the superblock.
Signed-off-by: NNeil Brown <neilb@cse.unsw.edu.au>
Signed-off-by: NNeil Brown <neilb@suse.de>
Signed-off-by: NAndrew Morton <akpm@osdl.org>
Signed-off-by: NLinus Torvalds <torvalds@osdl.org>
上级 bd926c63
...@@ -301,7 +301,7 @@ static int write_sb_page(mddev_t *mddev, long offset, struct page *page, int wai ...@@ -301,7 +301,7 @@ static int write_sb_page(mddev_t *mddev, long offset, struct page *page, int wai
page); page);
if (wait) if (wait)
wait_event(mddev->sb_wait, atomic_read(&mddev->pending_writes)==0); md_super_wait(mddev);
return 0; return 0;
} }
...@@ -828,8 +828,7 @@ int bitmap_unplug(struct bitmap *bitmap) ...@@ -828,8 +828,7 @@ int bitmap_unplug(struct bitmap *bitmap)
wake_up_process(bitmap->writeback_daemon->tsk)); wake_up_process(bitmap->writeback_daemon->tsk));
spin_unlock_irq(&bitmap->write_lock); spin_unlock_irq(&bitmap->write_lock);
} else } else
wait_event(bitmap->mddev->sb_wait, md_super_wait(bitmap->mddev);
atomic_read(&bitmap->mddev->pending_writes)==0);
} }
return 0; return 0;
} }
......
...@@ -330,16 +330,44 @@ static void free_disk_sb(mdk_rdev_t * rdev) ...@@ -330,16 +330,44 @@ static void free_disk_sb(mdk_rdev_t * rdev)
static int super_written(struct bio *bio, unsigned int bytes_done, int error) static int super_written(struct bio *bio, unsigned int bytes_done, int error)
{ {
mdk_rdev_t *rdev = bio->bi_private; mdk_rdev_t *rdev = bio->bi_private;
mddev_t *mddev = rdev->mddev;
if (bio->bi_size) if (bio->bi_size)
return 1; return 1;
if (error || !test_bit(BIO_UPTODATE, &bio->bi_flags)) if (error || !test_bit(BIO_UPTODATE, &bio->bi_flags))
md_error(rdev->mddev, rdev); md_error(mddev, rdev);
if (atomic_dec_and_test(&mddev->pending_writes))
wake_up(&mddev->sb_wait);
bio_put(bio);
return 0;
}
static int super_written_barrier(struct bio *bio, unsigned int bytes_done, int error)
{
struct bio *bio2 = bio->bi_private;
mdk_rdev_t *rdev = bio2->bi_private;
mddev_t *mddev = rdev->mddev;
if (bio->bi_size)
return 1;
if (atomic_dec_and_test(&rdev->mddev->pending_writes)) if (!test_bit(BIO_UPTODATE, &bio->bi_flags) &&
wake_up(&rdev->mddev->sb_wait); error == -EOPNOTSUPP) {
unsigned long flags;
/* barriers don't appear to be supported :-( */
set_bit(BarriersNotsupp, &rdev->flags);
mddev->barriers_work = 0;
spin_lock_irqsave(&mddev->write_lock, flags);
bio2->bi_next = mddev->biolist;
mddev->biolist = bio2;
spin_unlock_irqrestore(&mddev->write_lock, flags);
wake_up(&mddev->sb_wait);
bio_put(bio); bio_put(bio);
return 0; return 0;
}
bio_put(bio2);
bio->bi_private = rdev;
return super_written(bio, bytes_done, error);
} }
void md_super_write(mddev_t *mddev, mdk_rdev_t *rdev, void md_super_write(mddev_t *mddev, mdk_rdev_t *rdev,
...@@ -350,16 +378,54 @@ void md_super_write(mddev_t *mddev, mdk_rdev_t *rdev, ...@@ -350,16 +378,54 @@ void md_super_write(mddev_t *mddev, mdk_rdev_t *rdev,
* and decrement it on completion, waking up sb_wait * and decrement it on completion, waking up sb_wait
* if zero is reached. * if zero is reached.
* If an error occurred, call md_error * If an error occurred, call md_error
*
* As we might need to resubmit the request if BIO_RW_BARRIER
* causes ENOTSUPP, we allocate a spare bio...
*/ */
struct bio *bio = bio_alloc(GFP_NOIO, 1); struct bio *bio = bio_alloc(GFP_NOIO, 1);
int rw = (1<<BIO_RW) | (1<<BIO_RW_SYNC);
bio->bi_bdev = rdev->bdev; bio->bi_bdev = rdev->bdev;
bio->bi_sector = sector; bio->bi_sector = sector;
bio_add_page(bio, page, size, 0); bio_add_page(bio, page, size, 0);
bio->bi_private = rdev; bio->bi_private = rdev;
bio->bi_end_io = super_written; bio->bi_end_io = super_written;
bio->bi_rw = rw;
atomic_inc(&mddev->pending_writes); atomic_inc(&mddev->pending_writes);
submit_bio((1<<BIO_RW)|(1<<BIO_RW_SYNC), bio); if (!test_bit(BarriersNotsupp, &rdev->flags)) {
struct bio *rbio;
rw |= (1<<BIO_RW_BARRIER);
rbio = bio_clone(bio, GFP_NOIO);
rbio->bi_private = bio;
rbio->bi_end_io = super_written_barrier;
submit_bio(rw, rbio);
} else
submit_bio(rw, bio);
}
void md_super_wait(mddev_t *mddev)
{
/* wait for all superblock writes that were scheduled to complete.
* if any had to be retried (due to BARRIER problems), retry them
*/
DEFINE_WAIT(wq);
for(;;) {
prepare_to_wait(&mddev->sb_wait, &wq, TASK_UNINTERRUPTIBLE);
if (atomic_read(&mddev->pending_writes)==0)
break;
while (mddev->biolist) {
struct bio *bio;
spin_lock_irq(&mddev->write_lock);
bio = mddev->biolist;
mddev->biolist = bio->bi_next ;
bio->bi_next = NULL;
spin_unlock_irq(&mddev->write_lock);
submit_bio(bio->bi_rw, bio);
}
schedule();
}
finish_wait(&mddev->sb_wait, &wq);
} }
static int bi_complete(struct bio *bio, unsigned int bytes_done, int error) static int bi_complete(struct bio *bio, unsigned int bytes_done, int error)
...@@ -1382,7 +1448,7 @@ static void md_update_sb(mddev_t * mddev) ...@@ -1382,7 +1448,7 @@ static void md_update_sb(mddev_t * mddev)
int sync_req; int sync_req;
repeat: repeat:
spin_lock(&mddev->write_lock); spin_lock_irq(&mddev->write_lock);
sync_req = mddev->in_sync; sync_req = mddev->in_sync;
mddev->utime = get_seconds(); mddev->utime = get_seconds();
mddev->events ++; mddev->events ++;
...@@ -1405,11 +1471,11 @@ static void md_update_sb(mddev_t * mddev) ...@@ -1405,11 +1471,11 @@ static void md_update_sb(mddev_t * mddev)
*/ */
if (!mddev->persistent) { if (!mddev->persistent) {
mddev->sb_dirty = 0; mddev->sb_dirty = 0;
spin_unlock(&mddev->write_lock); spin_unlock_irq(&mddev->write_lock);
wake_up(&mddev->sb_wait); wake_up(&mddev->sb_wait);
return; return;
} }
spin_unlock(&mddev->write_lock); spin_unlock_irq(&mddev->write_lock);
dprintk(KERN_INFO dprintk(KERN_INFO
"md: updating %s RAID superblock on device (in sync %d)\n", "md: updating %s RAID superblock on device (in sync %d)\n",
...@@ -1437,17 +1503,17 @@ static void md_update_sb(mddev_t * mddev) ...@@ -1437,17 +1503,17 @@ static void md_update_sb(mddev_t * mddev)
/* only need to write one superblock... */ /* only need to write one superblock... */
break; break;
} }
wait_event(mddev->sb_wait, atomic_read(&mddev->pending_writes)==0); md_super_wait(mddev);
/* if there was a failure, sb_dirty was set to 1, and we re-write super */ /* if there was a failure, sb_dirty was set to 1, and we re-write super */
spin_lock(&mddev->write_lock); spin_lock_irq(&mddev->write_lock);
if (mddev->in_sync != sync_req|| mddev->sb_dirty == 1) { if (mddev->in_sync != sync_req|| mddev->sb_dirty == 1) {
/* have to write it out again */ /* have to write it out again */
spin_unlock(&mddev->write_lock); spin_unlock_irq(&mddev->write_lock);
goto repeat; goto repeat;
} }
mddev->sb_dirty = 0; mddev->sb_dirty = 0;
spin_unlock(&mddev->write_lock); spin_unlock_irq(&mddev->write_lock);
wake_up(&mddev->sb_wait); wake_up(&mddev->sb_wait);
} }
...@@ -1989,6 +2055,7 @@ static int do_md_run(mddev_t * mddev) ...@@ -1989,6 +2055,7 @@ static int do_md_run(mddev_t * mddev)
mddev->recovery = 0; mddev->recovery = 0;
mddev->resync_max_sectors = mddev->size << 1; /* may be over-ridden by personality */ mddev->resync_max_sectors = mddev->size << 1; /* may be over-ridden by personality */
mddev->barriers_work = 1;
/* before we start the array running, initialise the bitmap */ /* before we start the array running, initialise the bitmap */
err = bitmap_create(mddev); err = bitmap_create(mddev);
...@@ -2107,7 +2174,7 @@ static int do_md_stop(mddev_t * mddev, int ro) ...@@ -2107,7 +2174,7 @@ static int do_md_stop(mddev_t * mddev, int ro)
mddev->ro = 1; mddev->ro = 1;
} else { } else {
bitmap_flush(mddev); bitmap_flush(mddev);
wait_event(mddev->sb_wait, atomic_read(&mddev->pending_writes)==0); md_super_wait(mddev);
if (mddev->ro) if (mddev->ro)
set_disk_ro(disk, 0); set_disk_ro(disk, 0);
blk_queue_make_request(mddev->queue, md_fail_request); blk_queue_make_request(mddev->queue, md_fail_request);
...@@ -3796,13 +3863,13 @@ void md_write_start(mddev_t *mddev, struct bio *bi) ...@@ -3796,13 +3863,13 @@ void md_write_start(mddev_t *mddev, struct bio *bi)
atomic_inc(&mddev->writes_pending); atomic_inc(&mddev->writes_pending);
if (mddev->in_sync) { if (mddev->in_sync) {
spin_lock(&mddev->write_lock); spin_lock_irq(&mddev->write_lock);
if (mddev->in_sync) { if (mddev->in_sync) {
mddev->in_sync = 0; mddev->in_sync = 0;
mddev->sb_dirty = 1; mddev->sb_dirty = 1;
md_wakeup_thread(mddev->thread); md_wakeup_thread(mddev->thread);
} }
spin_unlock(&mddev->write_lock); spin_unlock_irq(&mddev->write_lock);
} }
wait_event(mddev->sb_wait, mddev->sb_dirty==0); wait_event(mddev->sb_wait, mddev->sb_dirty==0);
} }
...@@ -4112,7 +4179,7 @@ void md_check_recovery(mddev_t *mddev) ...@@ -4112,7 +4179,7 @@ void md_check_recovery(mddev_t *mddev)
if (mddev_trylock(mddev)==0) { if (mddev_trylock(mddev)==0) {
int spares =0; int spares =0;
spin_lock(&mddev->write_lock); spin_lock_irq(&mddev->write_lock);
if (mddev->safemode && !atomic_read(&mddev->writes_pending) && if (mddev->safemode && !atomic_read(&mddev->writes_pending) &&
!mddev->in_sync && mddev->recovery_cp == MaxSector) { !mddev->in_sync && mddev->recovery_cp == MaxSector) {
mddev->in_sync = 1; mddev->in_sync = 1;
...@@ -4120,7 +4187,7 @@ void md_check_recovery(mddev_t *mddev) ...@@ -4120,7 +4187,7 @@ void md_check_recovery(mddev_t *mddev)
} }
if (mddev->safemode == 1) if (mddev->safemode == 1)
mddev->safemode = 0; mddev->safemode = 0;
spin_unlock(&mddev->write_lock); spin_unlock_irq(&mddev->write_lock);
if (mddev->sb_dirty) if (mddev->sb_dirty)
md_update_sb(mddev); md_update_sb(mddev);
......
...@@ -301,7 +301,7 @@ static int raid1_end_write_request(struct bio *bio, unsigned int bytes_done, int ...@@ -301,7 +301,7 @@ static int raid1_end_write_request(struct bio *bio, unsigned int bytes_done, int
{ {
int uptodate = test_bit(BIO_UPTODATE, &bio->bi_flags); int uptodate = test_bit(BIO_UPTODATE, &bio->bi_flags);
r1bio_t * r1_bio = (r1bio_t *)(bio->bi_private); r1bio_t * r1_bio = (r1bio_t *)(bio->bi_private);
int mirror, behind; int mirror, behind = test_bit(R1BIO_BehindIO, &r1_bio->state);
conf_t *conf = mddev_to_conf(r1_bio->mddev); conf_t *conf = mddev_to_conf(r1_bio->mddev);
if (bio->bi_size) if (bio->bi_size)
...@@ -311,9 +311,16 @@ static int raid1_end_write_request(struct bio *bio, unsigned int bytes_done, int ...@@ -311,9 +311,16 @@ static int raid1_end_write_request(struct bio *bio, unsigned int bytes_done, int
if (r1_bio->bios[mirror] == bio) if (r1_bio->bios[mirror] == bio)
break; break;
if (error == -ENOTSUPP && test_bit(R1BIO_Barrier, &r1_bio->state)) {
set_bit(BarriersNotsupp, &conf->mirrors[mirror].rdev->flags);
set_bit(R1BIO_BarrierRetry, &r1_bio->state);
r1_bio->mddev->barriers_work = 0;
} else {
/* /*
* this branch is our 'one mirror IO has finished' event handler: * this branch is our 'one mirror IO has finished' event handler:
*/ */
r1_bio->bios[mirror] = NULL;
bio_put(bio);
if (!uptodate) { if (!uptodate) {
md_error(r1_bio->mddev, conf->mirrors[mirror].rdev); md_error(r1_bio->mddev, conf->mirrors[mirror].rdev);
/* an I/O failed, we can't clear the bitmap */ /* an I/O failed, we can't clear the bitmap */
...@@ -332,7 +339,6 @@ static int raid1_end_write_request(struct bio *bio, unsigned int bytes_done, int ...@@ -332,7 +339,6 @@ static int raid1_end_write_request(struct bio *bio, unsigned int bytes_done, int
update_head_pos(mirror, r1_bio); update_head_pos(mirror, r1_bio);
behind = test_bit(R1BIO_BehindIO, &r1_bio->state);
if (behind) { if (behind) {
if (test_bit(WriteMostly, &conf->mirrors[mirror].rdev->flags)) if (test_bit(WriteMostly, &conf->mirrors[mirror].rdev->flags))
atomic_dec(&r1_bio->behind_remaining); atomic_dec(&r1_bio->behind_remaining);
...@@ -355,14 +361,23 @@ static int raid1_end_write_request(struct bio *bio, unsigned int bytes_done, int ...@@ -355,14 +361,23 @@ static int raid1_end_write_request(struct bio *bio, unsigned int bytes_done, int
} }
} }
} }
}
/* /*
* *
* Let's see if all mirrored write operations have finished * Let's see if all mirrored write operations have finished
* already. * already.
*/ */
if (atomic_dec_and_test(&r1_bio->remaining)) { if (atomic_dec_and_test(&r1_bio->remaining)) {
if (test_bit(R1BIO_BarrierRetry, &r1_bio->state)) {
reschedule_retry(r1_bio);
/* Don't dec_pending yet, we want to hold
* the reference over the retry
*/
return 0;
}
if (test_bit(R1BIO_BehindIO, &r1_bio->state)) { if (test_bit(R1BIO_BehindIO, &r1_bio->state)) {
/* free extra copy of the data pages */ /* free extra copy of the data pages */
/* FIXME bio has been freed!!! */
int i = bio->bi_vcnt; int i = bio->bi_vcnt;
while (i--) while (i--)
__free_page(bio->bi_io_vec[i].bv_page); __free_page(bio->bi_io_vec[i].bv_page);
...@@ -648,8 +663,9 @@ static int make_request(request_queue_t *q, struct bio * bio) ...@@ -648,8 +663,9 @@ static int make_request(request_queue_t *q, struct bio * bio)
struct bio_list bl; struct bio_list bl;
struct page **behind_pages = NULL; struct page **behind_pages = NULL;
const int rw = bio_data_dir(bio); const int rw = bio_data_dir(bio);
int do_barriers;
if (unlikely(bio_barrier(bio))) { if (unlikely(!mddev->barriers_work && bio_barrier(bio))) {
bio_endio(bio, bio->bi_size, -EOPNOTSUPP); bio_endio(bio, bio->bi_size, -EOPNOTSUPP);
return 0; return 0;
} }
...@@ -759,6 +775,10 @@ static int make_request(request_queue_t *q, struct bio * bio) ...@@ -759,6 +775,10 @@ static int make_request(request_queue_t *q, struct bio * bio)
atomic_set(&r1_bio->remaining, 0); atomic_set(&r1_bio->remaining, 0);
atomic_set(&r1_bio->behind_remaining, 0); atomic_set(&r1_bio->behind_remaining, 0);
do_barriers = bio->bi_rw & BIO_RW_BARRIER;
if (do_barriers)
set_bit(R1BIO_Barrier, &r1_bio->state);
bio_list_init(&bl); bio_list_init(&bl);
for (i = 0; i < disks; i++) { for (i = 0; i < disks; i++) {
struct bio *mbio; struct bio *mbio;
...@@ -771,7 +791,7 @@ static int make_request(request_queue_t *q, struct bio * bio) ...@@ -771,7 +791,7 @@ static int make_request(request_queue_t *q, struct bio * bio)
mbio->bi_sector = r1_bio->sector + conf->mirrors[i].rdev->data_offset; mbio->bi_sector = r1_bio->sector + conf->mirrors[i].rdev->data_offset;
mbio->bi_bdev = conf->mirrors[i].rdev->bdev; mbio->bi_bdev = conf->mirrors[i].rdev->bdev;
mbio->bi_end_io = raid1_end_write_request; mbio->bi_end_io = raid1_end_write_request;
mbio->bi_rw = WRITE; mbio->bi_rw = WRITE | do_barriers;
mbio->bi_private = r1_bio; mbio->bi_private = r1_bio;
if (behind_pages) { if (behind_pages) {
...@@ -1153,6 +1173,36 @@ static void raid1d(mddev_t *mddev) ...@@ -1153,6 +1173,36 @@ static void raid1d(mddev_t *mddev)
if (test_bit(R1BIO_IsSync, &r1_bio->state)) { if (test_bit(R1BIO_IsSync, &r1_bio->state)) {
sync_request_write(mddev, r1_bio); sync_request_write(mddev, r1_bio);
unplug = 1; unplug = 1;
} else if (test_bit(R1BIO_BarrierRetry, &r1_bio->state)) {
/* some requests in the r1bio were BIO_RW_BARRIER
* requests which failed with -ENOTSUPP. Hohumm..
* Better resubmit without the barrier.
* We know which devices to resubmit for, because
* all others have had their bios[] entry cleared.
*/
int i;
clear_bit(R1BIO_BarrierRetry, &r1_bio->state);
clear_bit(R1BIO_Barrier, &r1_bio->state);
for (i=0; i < conf->raid_disks; i++)
if (r1_bio->bios[i]) {
struct bio_vec *bvec;
int j;
bio = bio_clone(r1_bio->master_bio, GFP_NOIO);
/* copy pages from the failed bio, as
* this might be a write-behind device */
__bio_for_each_segment(bvec, bio, j, 0)
bvec->bv_page = bio_iovec_idx(r1_bio->bios[i], j)->bv_page;
bio_put(r1_bio->bios[i]);
bio->bi_sector = r1_bio->sector +
conf->mirrors[i].rdev->data_offset;
bio->bi_bdev = conf->mirrors[i].rdev->bdev;
bio->bi_end_io = raid1_end_write_request;
bio->bi_rw = WRITE;
bio->bi_private = r1_bio;
r1_bio->bios[i] = bio;
generic_make_request(bio);
}
} else { } else {
int disk; int disk;
bio = r1_bio->bios[r1_bio->read_disk]; bio = r1_bio->bios[r1_bio->read_disk];
......
...@@ -89,6 +89,7 @@ extern void md_print_devices (void); ...@@ -89,6 +89,7 @@ extern void md_print_devices (void);
extern void md_super_write(mddev_t *mddev, mdk_rdev_t *rdev, extern void md_super_write(mddev_t *mddev, mdk_rdev_t *rdev,
sector_t sector, int size, struct page *page); sector_t sector, int size, struct page *page);
extern void md_super_wait(mddev_t *mddev);
extern int sync_page_io(struct block_device *bdev, sector_t sector, int size, extern int sync_page_io(struct block_device *bdev, sector_t sector, int size,
struct page *page, int rw); struct page *page, int rw);
......
...@@ -122,6 +122,7 @@ struct mdk_rdev_s ...@@ -122,6 +122,7 @@ struct mdk_rdev_s
#define Faulty 1 /* device is known to have a fault */ #define Faulty 1 /* device is known to have a fault */
#define In_sync 2 /* device is in_sync with rest of array */ #define In_sync 2 /* device is in_sync with rest of array */
#define WriteMostly 4 /* Avoid reading if at all possible */ #define WriteMostly 4 /* Avoid reading if at all possible */
#define BarriersNotsupp 5 /* BIO_RW_BARRIER is not supported */
int desc_nr; /* descriptor index in the superblock */ int desc_nr; /* descriptor index in the superblock */
int raid_disk; /* role of device in array */ int raid_disk; /* role of device in array */
...@@ -210,6 +211,13 @@ struct mddev_s ...@@ -210,6 +211,13 @@ struct mddev_s
int degraded; /* whether md should consider int degraded; /* whether md should consider
* adding a spare * adding a spare
*/ */
int barriers_work; /* initialised to true, cleared as soon
* as a barrier request to slave
* fails. Only supported
*/
struct bio *biolist; /* bios that need to be retried
* because BIO_RW_BARRIER is not supported
*/
atomic_t recovery_active; /* blocks scheduled, but not written */ atomic_t recovery_active; /* blocks scheduled, but not written */
wait_queue_head_t recovery_wait; wait_queue_head_t recovery_wait;
......
...@@ -111,6 +111,8 @@ struct r1bio_s { ...@@ -111,6 +111,8 @@ struct r1bio_s {
#define R1BIO_IsSync 1 #define R1BIO_IsSync 1
#define R1BIO_Degraded 2 #define R1BIO_Degraded 2
#define R1BIO_BehindIO 3 #define R1BIO_BehindIO 3
#define R1BIO_Barrier 4
#define R1BIO_BarrierRetry 5
/* For write-behind requests, we call bi_end_io when /* For write-behind requests, we call bi_end_io when
* the last non-write-behind device completes, providing * the last non-write-behind device completes, providing
* any write was successful. Otherwise we call when * any write was successful. Otherwise we call when
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册