diff options
author | Mikulas Patocka <mpatocka@redhat.com> | 2015-02-26 11:40:35 -0500 |
---|---|---|
committer | Mike Snitzer <snitzer@redhat.com> | 2015-02-27 14:49:47 -0500 |
commit | b735fede8d957d9d255e9c5cf3964cfa59799637 (patch) | |
tree | dc09f6c9db8d21128e1c5f793460eaaf5450c406 /drivers/md/dm.c | |
parent | ab7c7bb6f4ab95dbca96fcfc4463cd69843e3e24 (diff) | |
download | blackbird-op-linux-b735fede8d957d9d255e9c5cf3964cfa59799637.tar.gz blackbird-op-linux-b735fede8d957d9d255e9c5cf3964cfa59799637.zip |
dm snapshot: suspend origin when doing exception handover
In the function snapshot_resume we perform exception store handover. If
there is another active snapshot target, the exception store is moved
from this target to the target that is being resumed.
The problem is that if there is some pending exception, it will point to
an incorrect exception store after that handover, causing a crash due to
dm-snap-persistent.c:get_exception()'s BUG_ON.
This bug can be triggered by repeatedly changing snapshot permissions
with "lvchange -p r" and "lvchange -p rw" while there are writes on the
associated origin device.
To fix this bug, we must suspend the origin device when doing the
exception store handover to make sure that there are no pending
exceptions:
- introduce _origin_hash that keeps track of dm_origin structures.
- introduce functions __lookup_dm_origin, __insert_dm_origin and
__remove_dm_origin that manipulate the origin hash.
- modify snapshot_resume so that it calls dm_internal_suspend_fast() and
dm_internal_resume_fast() on the origin device.
NOTE to stable@ people:
When backporting to kernels 3.12-3.18, use dm_internal_suspend and
dm_internal_resume instead of dm_internal_suspend_fast and
dm_internal_resume_fast.
When backporting to kernels older than 3.12, you need to pick functions
dm_internal_suspend and dm_internal_resume from the commit
fd2ed4d252701d3bbed4cd3e3d267ad469bb832a.
Signed-off-by: Mikulas Patocka <mpatocka@redhat.com>
Signed-off-by: Mike Snitzer <snitzer@redhat.com>
Cc: stable@vger.kernel.org
Diffstat (limited to 'drivers/md/dm.c')
-rw-r--r-- | drivers/md/dm.c | 2 |
1 files changed, 2 insertions, 0 deletions
diff --git a/drivers/md/dm.c b/drivers/md/dm.c index e79d5ccfda64..6e2b2e97abe9 100644 --- a/drivers/md/dm.c +++ b/drivers/md/dm.c @@ -3121,6 +3121,7 @@ void dm_internal_suspend_fast(struct mapped_device *md) flush_workqueue(md->wq); dm_wait_for_completion(md, TASK_UNINTERRUPTIBLE); } +EXPORT_SYMBOL_GPL(dm_internal_suspend_fast); void dm_internal_resume_fast(struct mapped_device *md) { @@ -3132,6 +3133,7 @@ void dm_internal_resume_fast(struct mapped_device *md) done: mutex_unlock(&md->suspend_lock); } +EXPORT_SYMBOL_GPL(dm_internal_resume_fast); /*----------------------------------------------------------------- * Event notification. |