aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorHeinz Mauelshagen <heinzm@redhat.com>2014-03-12 11:13:39 -0400
committerMike Snitzer <snitzer@redhat.com>2014-03-12 13:52:00 -0400
commite893fba90c09f9b57fb97daae204ea9cc2c52fa5 (patch)
tree6743c251b72ea02946b5ed6db89f876a2024335c
parent8b9d96666529a979acf4825391efcc7c8a3e9f12 (diff)
dm cache: fix access beyond end of origin device
In order to avoid wasting cache space a partial block at the end of the origin device is not cached. Unfortunately, the check for such a partial block at the end of the origin device was flawed. Fix accesses beyond the end of the origin device that occured due to attempted promotion of an undetected partial block by: - initializing the per bio data struct to allow cache_end_io to work properly - recognizing access to the partial block at the end of the origin device - avoiding out of bounds access to the discard bitset Otherwise, users can experience errors like the following: attempt to access beyond end of device dm-5: rw=0, want=20971520, limit=20971456 ... device-mapper: cache: promotion failed; couldn't copy block Signed-off-by: Heinz Mauelshagen <heinzm@redhat.com> Acked-by: Joe Thornber <ejt@redhat.com> Signed-off-by: Mike Snitzer <snitzer@redhat.com> Cc: stable@vger.kernel.org
-rw-r--r--drivers/md/dm-cache-target.c8
1 files changed, 3 insertions, 5 deletions
diff --git a/drivers/md/dm-cache-target.c b/drivers/md/dm-cache-target.c
index 354bbc1b9a3c..074b9c8e4cf0 100644
--- a/drivers/md/dm-cache-target.c
+++ b/drivers/md/dm-cache-target.c
@@ -2465,20 +2465,18 @@ static int cache_map(struct dm_target *ti, struct bio *bio)
2465 bool discarded_block; 2465 bool discarded_block;
2466 struct dm_bio_prison_cell *cell; 2466 struct dm_bio_prison_cell *cell;
2467 struct policy_result lookup_result; 2467 struct policy_result lookup_result;
2468 struct per_bio_data *pb; 2468 struct per_bio_data *pb = init_per_bio_data(bio, pb_data_size);
2469 2469
2470 if (from_oblock(block) > from_oblock(cache->origin_blocks)) { 2470 if (unlikely(from_oblock(block) >= from_oblock(cache->origin_blocks))) {
2471 /* 2471 /*
2472 * This can only occur if the io goes to a partial block at 2472 * This can only occur if the io goes to a partial block at
2473 * the end of the origin device. We don't cache these. 2473 * the end of the origin device. We don't cache these.
2474 * Just remap to the origin and carry on. 2474 * Just remap to the origin and carry on.
2475 */ 2475 */
2476 remap_to_origin_clear_discard(cache, bio, block); 2476 remap_to_origin(cache, bio);
2477 return DM_MAPIO_REMAPPED; 2477 return DM_MAPIO_REMAPPED;
2478 } 2478 }
2479 2479
2480 pb = init_per_bio_data(bio, pb_data_size);
2481
2482 if (bio->bi_rw & (REQ_FLUSH | REQ_FUA | REQ_DISCARD)) { 2480 if (bio->bi_rw & (REQ_FLUSH | REQ_FUA | REQ_DISCARD)) {
2483 defer_bio(cache, bio); 2481 defer_bio(cache, bio);
2484 return DM_MAPIO_SUBMITTED; 2482 return DM_MAPIO_SUBMITTED;