|
@@ -292,9 +292,12 @@ static void do_release_stripe(struct r5conf *conf, struct stripe_head *sh,
|
|
|
BUG_ON(atomic_read(&conf->active_stripes)==0);
|
|
|
if (test_bit(STRIPE_HANDLE, &sh->state)) {
|
|
|
if (test_bit(STRIPE_DELAYED, &sh->state) &&
|
|
|
- !test_bit(STRIPE_PREREAD_ACTIVE, &sh->state))
|
|
|
+ !test_bit(STRIPE_PREREAD_ACTIVE, &sh->state)) {
|
|
|
list_add_tail(&sh->lru, &conf->delayed_list);
|
|
|
- else if (test_bit(STRIPE_BIT_DELAY, &sh->state) &&
|
|
|
+ if (atomic_read(&conf->preread_active_stripes)
|
|
|
+ < IO_THRESHOLD)
|
|
|
+ md_wakeup_thread(conf->mddev->thread);
|
|
|
+ } else if (test_bit(STRIPE_BIT_DELAY, &sh->state) &&
|
|
|
sh->bm_seq - conf->seq_write > 0)
|
|
|
list_add_tail(&sh->lru, &conf->bitmap_list);
|
|
|
else {
|
|
@@ -2886,8 +2889,11 @@ static int fetch_block(struct stripe_head *sh, struct stripe_head_state *s,
|
|
|
(s->failed >= 1 && fdev[0]->toread) ||
|
|
|
(s->failed >= 2 && fdev[1]->toread) ||
|
|
|
(sh->raid_conf->level <= 5 && s->failed && fdev[0]->towrite &&
|
|
|
+ (!test_bit(R5_Insync, &dev->flags) || test_bit(STRIPE_PREREAD_ACTIVE, &sh->state)) &&
|
|
|
!test_bit(R5_OVERWRITE, &fdev[0]->flags)) ||
|
|
|
- (sh->raid_conf->level == 6 && s->failed && s->to_write))) {
|
|
|
+ (sh->raid_conf->level == 6 && s->failed && s->to_write &&
|
|
|
+ s->to_write < sh->raid_conf->raid_disks - 2 &&
|
|
|
+ (!test_bit(R5_Insync, &dev->flags) || test_bit(STRIPE_PREREAD_ACTIVE, &sh->state))))) {
|
|
|
/* we would like to get this block, possibly by computing it,
|
|
|
* otherwise read it if the backing disk is insync
|
|
|
*/
|
|
@@ -3086,7 +3092,8 @@ static void handle_stripe_dirtying(struct r5conf *conf,
|
|
|
!test_bit(R5_LOCKED, &dev->flags) &&
|
|
|
!(test_bit(R5_UPTODATE, &dev->flags) ||
|
|
|
test_bit(R5_Wantcompute, &dev->flags))) {
|
|
|
- if (test_bit(R5_Insync, &dev->flags)) rcw++;
|
|
|
+ if (test_bit(R5_Insync, &dev->flags))
|
|
|
+ rcw++;
|
|
|
else
|
|
|
rcw += 2*disks;
|
|
|
}
|
|
@@ -3107,10 +3114,10 @@ static void handle_stripe_dirtying(struct r5conf *conf,
|
|
|
!(test_bit(R5_UPTODATE, &dev->flags) ||
|
|
|
test_bit(R5_Wantcompute, &dev->flags)) &&
|
|
|
test_bit(R5_Insync, &dev->flags)) {
|
|
|
- if (
|
|
|
- test_bit(STRIPE_PREREAD_ACTIVE, &sh->state)) {
|
|
|
- pr_debug("Read_old block "
|
|
|
- "%d for r-m-w\n", i);
|
|
|
+ if (test_bit(STRIPE_PREREAD_ACTIVE,
|
|
|
+ &sh->state)) {
|
|
|
+ pr_debug("Read_old block %d for r-m-w\n",
|
|
|
+ i);
|
|
|
set_bit(R5_LOCKED, &dev->flags);
|
|
|
set_bit(R5_Wantread, &dev->flags);
|
|
|
s->locked++;
|
|
@@ -3133,10 +3140,9 @@ static void handle_stripe_dirtying(struct r5conf *conf,
|
|
|
!(test_bit(R5_UPTODATE, &dev->flags) ||
|
|
|
test_bit(R5_Wantcompute, &dev->flags))) {
|
|
|
rcw++;
|
|
|
- if (!test_bit(R5_Insync, &dev->flags))
|
|
|
- continue; /* it's a failed drive */
|
|
|
- if (
|
|
|
- test_bit(STRIPE_PREREAD_ACTIVE, &sh->state)) {
|
|
|
+ if (test_bit(R5_Insync, &dev->flags) &&
|
|
|
+ test_bit(STRIPE_PREREAD_ACTIVE,
|
|
|
+ &sh->state)) {
|
|
|
pr_debug("Read_old block "
|
|
|
"%d for Reconstruct\n", i);
|
|
|
set_bit(R5_LOCKED, &dev->flags);
|