diff options
-rw-r--r-- | drivers/block/drbd/drbd_int.h | 3 | ||||
-rw-r--r-- | drivers/block/drbd/drbd_receiver.c | 12 | ||||
-rw-r--r-- | drivers/block/drbd/drbd_worker.c | 4 |
3 files changed, 16 insertions, 3 deletions
diff --git a/drivers/block/drbd/drbd_int.h b/drivers/block/drbd/drbd_int.h index ec06e744be42..a529285b0cd3 100644 --- a/drivers/block/drbd/drbd_int.h +++ b/drivers/block/drbd/drbd_int.h | |||
@@ -205,8 +205,9 @@ enum drbd_packets { | |||
205 | /* P_CKPT_DISABLE_REQ = 0x26, * currently reserved for protocol D */ | 205 | /* P_CKPT_DISABLE_REQ = 0x26, * currently reserved for protocol D */ |
206 | P_DELAY_PROBE = 0x27, /* is used on BOTH sockets */ | 206 | P_DELAY_PROBE = 0x27, /* is used on BOTH sockets */ |
207 | P_OUT_OF_SYNC = 0x28, /* Mark as out of sync (Outrunning), data socket */ | 207 | P_OUT_OF_SYNC = 0x28, /* Mark as out of sync (Outrunning), data socket */ |
208 | P_RS_CANCEL = 0x29, /* meta: Used to cancel RS_DATA_REQUEST packet by SyncSource */ | ||
208 | 209 | ||
209 | P_MAX_CMD = 0x28, | 210 | P_MAX_CMD = 0x2A, |
210 | P_MAY_IGNORE = 0x100, /* Flag to test if (cmd > P_MAY_IGNORE) ... */ | 211 | P_MAY_IGNORE = 0x100, /* Flag to test if (cmd > P_MAY_IGNORE) ... */ |
211 | P_MAX_OPT_CMD = 0x101, | 212 | P_MAX_OPT_CMD = 0x101, |
212 | 213 | ||
diff --git a/drivers/block/drbd/drbd_receiver.c b/drivers/block/drbd/drbd_receiver.c index f0a0f66fbe68..bf865bd83414 100644 --- a/drivers/block/drbd/drbd_receiver.c +++ b/drivers/block/drbd/drbd_receiver.c | |||
@@ -4361,7 +4361,16 @@ static int got_NegRSDReply(struct drbd_conf *mdev, struct p_header80 *h) | |||
4361 | 4361 | ||
4362 | if (get_ldev_if_state(mdev, D_FAILED)) { | 4362 | if (get_ldev_if_state(mdev, D_FAILED)) { |
4363 | drbd_rs_complete_io(mdev, sector); | 4363 | drbd_rs_complete_io(mdev, sector); |
4364 | drbd_rs_failed_io(mdev, sector, size); | 4364 | switch (be16_to_cpu(h->command)) { |
4365 | case P_NEG_RS_DREPLY: | ||
4366 | drbd_rs_failed_io(mdev, sector, size); | ||
4367 | case P_RS_CANCEL: | ||
4368 | break; | ||
4369 | default: | ||
4370 | D_ASSERT(0); | ||
4371 | put_ldev(mdev); | ||
4372 | return false; | ||
4373 | } | ||
4365 | put_ldev(mdev); | 4374 | put_ldev(mdev); |
4366 | } | 4375 | } |
4367 | 4376 | ||
@@ -4459,6 +4468,7 @@ static struct asender_cmd *get_asender_cmd(int cmd) | |||
4459 | [P_STATE_CHG_REPLY] = { sizeof(struct p_req_state_reply), got_RqSReply }, | 4468 | [P_STATE_CHG_REPLY] = { sizeof(struct p_req_state_reply), got_RqSReply }, |
4460 | [P_RS_IS_IN_SYNC] = { sizeof(struct p_block_ack), got_IsInSync }, | 4469 | [P_RS_IS_IN_SYNC] = { sizeof(struct p_block_ack), got_IsInSync }, |
4461 | [P_DELAY_PROBE] = { sizeof(struct p_delay_probe93), got_skip }, | 4470 | [P_DELAY_PROBE] = { sizeof(struct p_delay_probe93), got_skip }, |
4471 | [P_RS_CANCEL] = { sizeof(struct p_block_ack), got_NegRSDReply}, | ||
4462 | [P_MAX_CMD] = { 0, NULL }, | 4472 | [P_MAX_CMD] = { 0, NULL }, |
4463 | }; | 4473 | }; |
4464 | if (cmd > P_MAX_CMD || asender_tbl[cmd].process == NULL) | 4474 | if (cmd > P_MAX_CMD || asender_tbl[cmd].process == NULL) |
diff --git a/drivers/block/drbd/drbd_worker.c b/drivers/block/drbd/drbd_worker.c index 7bfeb79e7105..1d7510ebaa43 100644 --- a/drivers/block/drbd/drbd_worker.c +++ b/drivers/block/drbd/drbd_worker.c | |||
@@ -988,7 +988,9 @@ int w_e_end_rsdata_req(struct drbd_conf *mdev, struct drbd_work *w, int cancel) | |||
988 | put_ldev(mdev); | 988 | put_ldev(mdev); |
989 | } | 989 | } |
990 | 990 | ||
991 | if (likely((e->flags & EE_WAS_ERROR) == 0)) { | 991 | if (mdev->state.conn == C_AHEAD) { |
992 | ok = drbd_send_ack(mdev, P_RS_CANCEL, e); | ||
993 | } else if (likely((e->flags & EE_WAS_ERROR) == 0)) { | ||
992 | if (likely(mdev->state.pdsk >= D_INCONSISTENT)) { | 994 | if (likely(mdev->state.pdsk >= D_INCONSISTENT)) { |
993 | inc_rs_pending(mdev); | 995 | inc_rs_pending(mdev); |
994 | ok = drbd_send_block(mdev, P_RS_DATA_REPLY, e); | 996 | ok = drbd_send_block(mdev, P_RS_DATA_REPLY, e); |