Commit 0f73ea05 authored by Alasdair G Kergon's avatar Alasdair G Kergon Committed by Greg Kroah-Hartman

[PATCH] dm: bio split bvec fix

The code that handles bios that span table target boundaries by breaking
them up into smaller bios will not split an individual struct bio_vec into
more than two pieces.  Sometimes more than that are required.

This patch adds a loop to break the second piece up into as many pieces as
are necessary.

Cc: "Abhishek Gupta" <abhishekgupt@gmail.com>
Cc: Dan Smith <danms@us.ibm.com>
Signed-off-by: default avatarAlasdair G Kergon <agk@redhat.com>
Signed-off-by: default avatarAndrew Morton <akpm@osdl.org>
Signed-off-by: default avatarChris Wright <chrisw@sous-sol.org>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@suse.de>
parent e699844c
...@@ -533,30 +533,35 @@ static void __clone_and_map(struct clone_info *ci) ...@@ -533,30 +533,35 @@ static void __clone_and_map(struct clone_info *ci)
} else { } else {
/* /*
* Create two copy bios to deal with io that has * Handle a bvec that must be split between two or more targets.
* been split across a target.
*/ */
struct bio_vec *bv = bio->bi_io_vec + ci->idx; struct bio_vec *bv = bio->bi_io_vec + ci->idx;
sector_t remaining = to_sector(bv->bv_len);
unsigned int offset = 0;
clone = split_bvec(bio, ci->sector, ci->idx, do {
bv->bv_offset, max); if (offset) {
__map_bio(ti, clone, tio); ti = dm_table_find_target(ci->map, ci->sector);
max = max_io_len(ci->md, ci->sector, ti);
ci->sector += max; tio = alloc_tio(ci->md);
ci->sector_count -= max; tio->io = ci->io;
ti = dm_table_find_target(ci->map, ci->sector); tio->ti = ti;
memset(&tio->info, 0, sizeof(tio->info));
len = to_sector(bv->bv_len) - max; }
clone = split_bvec(bio, ci->sector, ci->idx,
bv->bv_offset + to_bytes(max), len); len = min(remaining, max);
tio = alloc_tio(ci->md);
tio->io = ci->io; clone = split_bvec(bio, ci->sector, ci->idx,
tio->ti = ti; bv->bv_offset + offset, len);
memset(&tio->info, 0, sizeof(tio->info));
__map_bio(ti, clone, tio); __map_bio(ti, clone, tio);
ci->sector += len;
ci->sector_count -= len;
offset += to_bytes(len);
} while (remaining -= len);
ci->sector += len;
ci->sector_count -= len;
ci->idx++; ci->idx++;
} }
} }
......
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