diff options
Diffstat (limited to 'drivers/block')
-rw-r--r-- | drivers/block/drbd/drbd_actlog.c | 24 |
1 files changed, 19 insertions, 5 deletions
diff --git a/drivers/block/drbd/drbd_actlog.c b/drivers/block/drbd/drbd_actlog.c index 28f85d950781..5570d9bdc863 100644 --- a/drivers/block/drbd/drbd_actlog.c +++ b/drivers/block/drbd/drbd_actlog.c | |||
@@ -176,14 +176,17 @@ static struct lc_element *_al_get(struct drbd_conf *mdev, unsigned int enr) | |||
176 | struct lc_element *al_ext; | 176 | struct lc_element *al_ext; |
177 | struct lc_element *tmp; | 177 | struct lc_element *tmp; |
178 | unsigned long al_flags = 0; | 178 | unsigned long al_flags = 0; |
179 | int wake; | ||
179 | 180 | ||
180 | spin_lock_irq(&mdev->al_lock); | 181 | spin_lock_irq(&mdev->al_lock); |
181 | tmp = lc_find(mdev->resync, enr/AL_EXT_PER_BM_SECT); | 182 | tmp = lc_find(mdev->resync, enr/AL_EXT_PER_BM_SECT); |
182 | if (unlikely(tmp != NULL)) { | 183 | if (unlikely(tmp != NULL)) { |
183 | struct bm_extent *bm_ext = lc_entry(tmp, struct bm_extent, lce); | 184 | struct bm_extent *bm_ext = lc_entry(tmp, struct bm_extent, lce); |
184 | if (test_bit(BME_NO_WRITES, &bm_ext->flags)) { | 185 | if (test_bit(BME_NO_WRITES, &bm_ext->flags)) { |
185 | set_bit(BME_PRIORITY, &bm_ext->flags); | 186 | wake = !test_and_set_bit(BME_PRIORITY, &bm_ext->flags); |
186 | spin_unlock_irq(&mdev->al_lock); | 187 | spin_unlock_irq(&mdev->al_lock); |
188 | if (wake) | ||
189 | wake_up(&mdev->al_wait); | ||
187 | return NULL; | 190 | return NULL; |
188 | } | 191 | } |
189 | } | 192 | } |
@@ -1135,7 +1138,10 @@ int drbd_rs_begin_io(struct drbd_conf *mdev, sector_t sector) | |||
1135 | unsigned int enr = BM_SECT_TO_EXT(sector); | 1138 | unsigned int enr = BM_SECT_TO_EXT(sector); |
1136 | struct bm_extent *bm_ext; | 1139 | struct bm_extent *bm_ext; |
1137 | int i, sig; | 1140 | int i, sig; |
1141 | int sa = 200; /* Step aside 200 times, then grab the extent and let app-IO wait. | ||
1142 | 200 times -> 20 seconds. */ | ||
1138 | 1143 | ||
1144 | retry: | ||
1139 | sig = wait_event_interruptible(mdev->al_wait, | 1145 | sig = wait_event_interruptible(mdev->al_wait, |
1140 | (bm_ext = _bme_get(mdev, enr))); | 1146 | (bm_ext = _bme_get(mdev, enr))); |
1141 | if (sig) | 1147 | if (sig) |
@@ -1146,16 +1152,24 @@ int drbd_rs_begin_io(struct drbd_conf *mdev, sector_t sector) | |||
1146 | 1152 | ||
1147 | for (i = 0; i < AL_EXT_PER_BM_SECT; i++) { | 1153 | for (i = 0; i < AL_EXT_PER_BM_SECT; i++) { |
1148 | sig = wait_event_interruptible(mdev->al_wait, | 1154 | sig = wait_event_interruptible(mdev->al_wait, |
1149 | !_is_in_al(mdev, enr * AL_EXT_PER_BM_SECT + i)); | 1155 | !_is_in_al(mdev, enr * AL_EXT_PER_BM_SECT + i) || |
1150 | if (sig) { | 1156 | (test_bit(BME_PRIORITY, &bm_ext->flags) && sa)); |
1157 | |||
1158 | if (sig || (test_bit(BME_PRIORITY, &bm_ext->flags) && sa)) { | ||
1151 | spin_lock_irq(&mdev->al_lock); | 1159 | spin_lock_irq(&mdev->al_lock); |
1152 | if (lc_put(mdev->resync, &bm_ext->lce) == 0) { | 1160 | if (lc_put(mdev->resync, &bm_ext->lce) == 0) { |
1153 | clear_bit(BME_NO_WRITES, &bm_ext->flags); | 1161 | bm_ext->flags = 0; /* clears BME_NO_WRITES and eventually BME_PRIORITY */ |
1154 | mdev->resync_locked--; | 1162 | mdev->resync_locked--; |
1155 | wake_up(&mdev->al_wait); | 1163 | wake_up(&mdev->al_wait); |
1156 | } | 1164 | } |
1157 | spin_unlock_irq(&mdev->al_lock); | 1165 | spin_unlock_irq(&mdev->al_lock); |
1158 | return -EINTR; | 1166 | if (sig) |
1167 | return -EINTR; | ||
1168 | if (schedule_timeout_interruptible(HZ/10)) | ||
1169 | return -EINTR; | ||
1170 | if (--sa == 0) | ||
1171 | dev_warn(DEV,"drbd_rs_begin_io() no longer stepping aside.\n"); | ||
1172 | goto retry; | ||
1159 | } | 1173 | } |
1160 | } | 1174 | } |
1161 | set_bit(BME_LOCKED, &bm_ext->flags); | 1175 | set_bit(BME_LOCKED, &bm_ext->flags); |