Commit 8d6cdd78 authored by Lars Ellenberg's avatar Lars Ellenberg Committed by Philipp Reisner

drbd: conflicting writes: make wake_up of waiting peer_requests explicit

Signed-off-by: default avatarPhilipp Reisner <philipp.reisner@linbit.com>
Signed-off-by: default avatarLars Ellenberg <lars.ellenberg@linbit.com>
parent 0afd569a
...@@ -212,6 +212,16 @@ static void drbd_remove_request_interval(struct rb_root *root, ...@@ -212,6 +212,16 @@ static void drbd_remove_request_interval(struct rb_root *root,
wake_up(&mdev->misc_wait); wake_up(&mdev->misc_wait);
} }
static void maybe_wakeup_conflicting_requests(struct drbd_request *req)
{
const unsigned long s = req->rq_state;
if (s & RQ_LOCAL_PENDING && !(s & RQ_LOCAL_ABORTED))
return;
if (req->i.waiting)
/* Retry all conflicting peer requests. */
wake_up(&req->w.mdev->misc_wait);
}
/* Helper for __req_mod(). /* Helper for __req_mod().
* Set m->bio to the master bio, if it is fit to be completed, * Set m->bio to the master bio, if it is fit to be completed,
* or leave it alone (it is initialized to NULL in __req_mod), * or leave it alone (it is initialized to NULL in __req_mod),
...@@ -235,10 +245,6 @@ void _req_may_be_done(struct drbd_request *req, struct bio_and_error *m) ...@@ -235,10 +245,6 @@ void _req_may_be_done(struct drbd_request *req, struct bio_and_error *m)
*/ */
if (s & RQ_LOCAL_PENDING && !(s & RQ_LOCAL_ABORTED)) if (s & RQ_LOCAL_PENDING && !(s & RQ_LOCAL_ABORTED))
return; return;
if (req->i.waiting) {
/* Retry all conflicting peer requests. */
wake_up(&mdev->misc_wait);
}
if (s & RQ_NET_QUEUED) if (s & RQ_NET_QUEUED)
return; return;
if (s & RQ_NET_PENDING) if (s & RQ_NET_PENDING)
...@@ -388,6 +394,7 @@ int __req_mod(struct drbd_request *req, enum drbd_req_event what, ...@@ -388,6 +394,7 @@ int __req_mod(struct drbd_request *req, enum drbd_req_event what,
req->rq_state |= (RQ_LOCAL_COMPLETED|RQ_LOCAL_OK); req->rq_state |= (RQ_LOCAL_COMPLETED|RQ_LOCAL_OK);
req->rq_state &= ~RQ_LOCAL_PENDING; req->rq_state &= ~RQ_LOCAL_PENDING;
maybe_wakeup_conflicting_requests(req);
_req_may_be_done_not_susp(req, m); _req_may_be_done_not_susp(req, m);
put_ldev(mdev); put_ldev(mdev);
break; break;
...@@ -405,6 +412,7 @@ int __req_mod(struct drbd_request *req, enum drbd_req_event what, ...@@ -405,6 +412,7 @@ int __req_mod(struct drbd_request *req, enum drbd_req_event what,
req->rq_state &= ~RQ_LOCAL_PENDING; req->rq_state &= ~RQ_LOCAL_PENDING;
__drbd_chk_io_error(mdev, false); __drbd_chk_io_error(mdev, false);
maybe_wakeup_conflicting_requests(req);
_req_may_be_done_not_susp(req, m); _req_may_be_done_not_susp(req, m);
put_ldev(mdev); put_ldev(mdev);
break; break;
...@@ -615,6 +623,7 @@ int __req_mod(struct drbd_request *req, enum drbd_req_event what, ...@@ -615,6 +623,7 @@ int __req_mod(struct drbd_request *req, enum drbd_req_event what,
dec_ap_pending(mdev); dec_ap_pending(mdev);
atomic_sub(req->i.size >> 9, &mdev->ap_in_flight); atomic_sub(req->i.size >> 9, &mdev->ap_in_flight);
req->rq_state &= ~RQ_NET_PENDING; req->rq_state &= ~RQ_NET_PENDING;
maybe_wakeup_conflicting_requests(req);
_req_may_be_done_not_susp(req, m); _req_may_be_done_not_susp(req, m);
break; break;
...@@ -626,6 +635,7 @@ int __req_mod(struct drbd_request *req, enum drbd_req_event what, ...@@ -626,6 +635,7 @@ int __req_mod(struct drbd_request *req, enum drbd_req_event what,
*/ */
D_ASSERT(req->rq_state & RQ_NET_PENDING); D_ASSERT(req->rq_state & RQ_NET_PENDING);
req->rq_state |= RQ_POSTPONED; req->rq_state |= RQ_POSTPONED;
maybe_wakeup_conflicting_requests(req);
_req_may_be_done_not_susp(req, m); _req_may_be_done_not_susp(req, m);
break; break;
...@@ -643,6 +653,7 @@ int __req_mod(struct drbd_request *req, enum drbd_req_event what, ...@@ -643,6 +653,7 @@ int __req_mod(struct drbd_request *req, enum drbd_req_event what,
if (!(req->rq_state & RQ_WRITE)) if (!(req->rq_state & RQ_WRITE))
goto goto_read_retry_local; goto goto_read_retry_local;
maybe_wakeup_conflicting_requests(req);
_req_may_be_done_not_susp(req, m); _req_may_be_done_not_susp(req, m);
/* else: done by HANDED_OVER_TO_NETWORK */ /* else: done by HANDED_OVER_TO_NETWORK */
break; break;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment