提交 ba40a2aa 编写于 作者: A Alasdair G Kergon 提交者: Linus Torvalds

[PATCH] dm snapshot: tidy snapshot_map

This patch rearranges the snapshot_map code so that the functional changes in
subsequent patches are clearer.

The only functional change is to replace the existing read lock with a write
lock which the next patch needs.
Signed-off-by: NAlasdair G Kergon <agk@redhat.com>
Signed-off-by: NAndrew Morton <akpm@osdl.org>
Signed-off-by: NLinus Torvalds <torvalds@osdl.org>
上级 927ffe7c
...@@ -851,7 +851,6 @@ static int snapshot_map(struct dm_target *ti, struct bio *bio, ...@@ -851,7 +851,6 @@ static int snapshot_map(struct dm_target *ti, struct bio *bio,
{ {
struct exception *e; struct exception *e;
struct dm_snapshot *s = (struct dm_snapshot *) ti->private; struct dm_snapshot *s = (struct dm_snapshot *) ti->private;
int copy_needed = 0;
int r = 1; int r = 1;
chunk_t chunk; chunk_t chunk;
struct pending_exception *pe = NULL; struct pending_exception *pe = NULL;
...@@ -866,29 +865,28 @@ static int snapshot_map(struct dm_target *ti, struct bio *bio, ...@@ -866,29 +865,28 @@ static int snapshot_map(struct dm_target *ti, struct bio *bio,
if (unlikely(bio_barrier(bio))) if (unlikely(bio_barrier(bio)))
return -EOPNOTSUPP; return -EOPNOTSUPP;
/* FIXME: should only take write lock if we need
* to copy an exception */
down_write(&s->lock);
if (!s->valid) {
r = -EIO;
goto out_unlock;
}
/* If the block is already remapped - use that, else remap it */
e = lookup_exception(&s->complete, chunk);
if (e) {
remap_exception(s, e, bio);
goto out_unlock;
}
/* /*
* Write to snapshot - higher level takes care of RW/RO * Write to snapshot - higher level takes care of RW/RO
* flags so we should only get this if we are * flags so we should only get this if we are
* writeable. * writeable.
*/ */
if (bio_rw(bio) == WRITE) { if (bio_rw(bio) == WRITE) {
/* FIXME: should only take write lock if we need
* to copy an exception */
down_write(&s->lock);
if (!s->valid) {
r = -EIO;
goto out_unlock;
}
/* If the block is already remapped - use that, else remap it */
e = lookup_exception(&s->complete, chunk);
if (e) {
remap_exception(s, e, bio);
goto out_unlock;
}
pe = __find_pending_exception(s, bio); pe = __find_pending_exception(s, bio);
if (!pe) { if (!pe) {
__invalidate_snapshot(s, pe, -ENOMEM); __invalidate_snapshot(s, pe, -ENOMEM);
...@@ -899,45 +897,27 @@ static int snapshot_map(struct dm_target *ti, struct bio *bio, ...@@ -899,45 +897,27 @@ static int snapshot_map(struct dm_target *ti, struct bio *bio,
remap_exception(s, &pe->e, bio); remap_exception(s, &pe->e, bio);
bio_list_add(&pe->snapshot_bios, bio); bio_list_add(&pe->snapshot_bios, bio);
r = 0;
if (!pe->started) { if (!pe->started) {
/* this is protected by snap->lock */ /* this is protected by snap->lock */
pe->started = 1; pe->started = 1;
copy_needed = 1; up_write(&s->lock);
}
r = 0;
out_unlock:
up_write(&s->lock);
if (copy_needed)
start_copy(pe); start_copy(pe);
} else { goto out;
}
} else
/* /*
* FIXME: this read path scares me because we * FIXME: this read path scares me because we
* always use the origin when we have a pending * always use the origin when we have a pending
* exception. However I can't think of a * exception. However I can't think of a
* situation where this is wrong - ejt. * situation where this is wrong - ejt.
*/ */
bio->bi_bdev = s->origin->bdev;
/* Do reads */ out_unlock:
down_read(&s->lock); up_write(&s->lock);
out:
if (!s->valid) {
up_read(&s->lock);
return -EIO;
}
/* See if it it has been remapped */
e = lookup_exception(&s->complete, chunk);
if (e)
remap_exception(s, e, bio);
else
bio->bi_bdev = s->origin->bdev;
up_read(&s->lock);
}
return r; return r;
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册