Commit 298eaa89 authored by Mikulas Patocka's avatar Mikulas Patocka Committed by Mike Snitzer

dm snapshot: do not split read bios sent to snapshot-origin target

Change the snapshot-origin target so that only write bios are split on
chunk boundary.  Read bios are passed unchanged to the underlying
device, so they don't have to be split.

Later, we could change the target so that it accepts a larger write bio
if it spans an area that is completely covered by snapshot exceptions.
Signed-off-by: default avatarMikulas Patocka <mpatocka@redhat.com>
Signed-off-by: default avatarMike Snitzer <snitzer@redhat.com>
parent 599cdf3b
...@@ -2143,6 +2143,7 @@ static int origin_write_extent(struct dm_snapshot *merging_snap, ...@@ -2143,6 +2143,7 @@ static int origin_write_extent(struct dm_snapshot *merging_snap,
struct dm_origin { struct dm_origin {
struct dm_dev *dev; struct dm_dev *dev;
unsigned split_boundary;
}; };
/* /*
...@@ -2194,13 +2195,24 @@ static void origin_dtr(struct dm_target *ti) ...@@ -2194,13 +2195,24 @@ static void origin_dtr(struct dm_target *ti)
static int origin_map(struct dm_target *ti, struct bio *bio) static int origin_map(struct dm_target *ti, struct bio *bio)
{ {
struct dm_origin *o = ti->private; struct dm_origin *o = ti->private;
unsigned available_sectors;
bio->bi_bdev = o->dev->bdev; bio->bi_bdev = o->dev->bdev;
if (bio->bi_rw & REQ_FLUSH) if (unlikely(bio->bi_rw & REQ_FLUSH))
return DM_MAPIO_REMAPPED;
if (bio_rw(bio) != WRITE)
return DM_MAPIO_REMAPPED; return DM_MAPIO_REMAPPED;
available_sectors = o->split_boundary -
((unsigned)bio->bi_iter.bi_sector & (o->split_boundary - 1));
if (bio_sectors(bio) > available_sectors)
dm_accept_partial_bio(bio, available_sectors);
/* Only tell snapshots if this is a write */ /* Only tell snapshots if this is a write */
return (bio_rw(bio) == WRITE) ? do_origin(o->dev, bio) : DM_MAPIO_REMAPPED; return do_origin(o->dev, bio);
} }
/* /*
...@@ -2211,7 +2223,7 @@ static void origin_resume(struct dm_target *ti) ...@@ -2211,7 +2223,7 @@ static void origin_resume(struct dm_target *ti)
{ {
struct dm_origin *o = ti->private; struct dm_origin *o = ti->private;
ti->max_io_len = get_origin_minimum_chunksize(o->dev->bdev); o->split_boundary = get_origin_minimum_chunksize(o->dev->bdev);
} }
static void origin_status(struct dm_target *ti, status_type_t type, static void origin_status(struct dm_target *ti, status_type_t type,
......
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