aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/block/drbd/drbd_receiver.c
diff options
context:
space:
mode:
authorLars Ellenberg <lars.ellenberg@linbit.com>2010-09-04 19:13:24 -0400
committerPhilipp Reisner <philipp.reisner@linbit.com>2010-10-14 12:38:34 -0400
commit1d53f09e170e477de67babd7a10e277479260d51 (patch)
treec795de9e5aeaf8324eeb25ea20c7b94f573bca83 /drivers/block/drbd/drbd_receiver.c
parent435f07402b3165b90592073bc0f8c6f8fa160ff9 (diff)
drbd: fix potential kernel BUG (NULL deref)
BUG trace would look like: lc_find drbd_rs_complete_io got_OVResult drbd_asender Could be triggered by explicit, or IO-error policy based, detach during online-verify. We may only dereference mdev->resync, if we first get_ldev(), as the disk may break any time, causing mdev->resync to disappear once all ldev references have been returned. Already in flight online-verify requests or replies may still come in, which we then need to ignore. Signed-off-by: Philipp Reisner <philipp.reisner@linbit.com> Signed-off-by: Lars Ellenberg <lars.ellenberg@linbit.com>
Diffstat (limited to 'drivers/block/drbd/drbd_receiver.c')
-rw-r--r--drivers/block/drbd/drbd_receiver.c15
1 files changed, 11 insertions, 4 deletions
diff --git a/drivers/block/drbd/drbd_receiver.c b/drivers/block/drbd/drbd_receiver.c
index 2c3edf0ac5ca..e4e4eddf04f2 100644
--- a/drivers/block/drbd/drbd_receiver.c
+++ b/drivers/block/drbd/drbd_receiver.c
@@ -4241,10 +4241,13 @@ static int got_IsInSync(struct drbd_conf *mdev, struct p_header80 *h)
4241 4241
4242 update_peer_seq(mdev, be32_to_cpu(p->seq_num)); 4242 update_peer_seq(mdev, be32_to_cpu(p->seq_num));
4243 4243
4244 drbd_rs_complete_io(mdev, sector); 4244 if (get_ldev(mdev)) {
4245 drbd_set_in_sync(mdev, sector, blksize); 4245 drbd_rs_complete_io(mdev, sector);
4246 /* rs_same_csums is supposed to count in units of BM_BLOCK_SIZE */ 4246 drbd_set_in_sync(mdev, sector, blksize);
4247 mdev->rs_same_csum += (blksize >> BM_BLOCK_SHIFT); 4247 /* rs_same_csums is supposed to count in units of BM_BLOCK_SIZE */
4248 mdev->rs_same_csum += (blksize >> BM_BLOCK_SHIFT);
4249 put_ldev(mdev);
4250 }
4248 dec_rs_pending(mdev); 4251 dec_rs_pending(mdev);
4249 atomic_add(blksize >> 9, &mdev->rs_sect_in); 4252 atomic_add(blksize >> 9, &mdev->rs_sect_in);
4250 4253
@@ -4423,6 +4426,9 @@ static int got_OVResult(struct drbd_conf *mdev, struct p_header80 *h)
4423 else 4426 else
4424 ov_oos_print(mdev); 4427 ov_oos_print(mdev);
4425 4428
4429 if (!get_ldev(mdev))
4430 return TRUE;
4431
4426 drbd_rs_complete_io(mdev, sector); 4432 drbd_rs_complete_io(mdev, sector);
4427 dec_rs_pending(mdev); 4433 dec_rs_pending(mdev);
4428 4434
@@ -4437,6 +4443,7 @@ static int got_OVResult(struct drbd_conf *mdev, struct p_header80 *h)
4437 drbd_resync_finished(mdev); 4443 drbd_resync_finished(mdev);
4438 } 4444 }
4439 } 4445 }
4446 put_ldev(mdev);
4440 return TRUE; 4447 return TRUE;
4441} 4448}
4442 4449