summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorZdenek Kabelac <zkabelac@redhat.com>2017-01-03 14:47:46 +0100
committerZdenek Kabelac <zkabelac@redhat.com>2017-01-03 14:55:16 +0100
commit95d5877f7af781c53c7865868745dc683d2b0413 (patch)
treec928c991005001a980a21da80cd64ad643e22516
parent4fd41cf67f0b9ea44a103b059dbbb2d74d8289eb (diff)
downloadlvm2-95d5877f7af781c53c7865868745dc683d2b0413.tar.gz
cache: add missing udev wait
When we need to clear dirty cache content of cached LV, there is table reload which usually is shortly followed by next metadata change. However udev can't (as of now) process udev event while device is 'suspended'. So whenever sequence of 'suspend/resume/suspend' is needed, we need to wait first for finishing of 'resume' processing before starting next 'suspend'. Otherwise there is 'race' danger of triggering unwantend umount by systemd as such event will trigger SYSTEMD_READY=0 state for a moment for such changed device. Such race is pretty ugly to trace so we may need to review more sequencies for missing 'sync'. (Other option is to enhnace 'udev' rules processing to avoid such dramatic actions to be happening for suspended devices).
-rw-r--r--WHATS_NEW1
-rw-r--r--lib/metadata/cache_manip.c14
2 files changed, 14 insertions, 1 deletions
diff --git a/WHATS_NEW b/WHATS_NEW
index 9e16b278e..5fb590468 100644
--- a/WHATS_NEW
+++ b/WHATS_NEW
@@ -1,5 +1,6 @@
Version 2.02.169 -
=====================================
+ Add missing udev sync when flushing dirty cache content.
vgchange -p accepts only uint32 numbers.
Report thin LV date for merged LV when the merge is in progress.
Detect if snapshot merge really started before polling for progress.
diff --git a/lib/metadata/cache_manip.c b/lib/metadata/cache_manip.c
index 404b1af79..cddaf8252 100644
--- a/lib/metadata/cache_manip.c
+++ b/lib/metadata/cache_manip.c
@@ -424,9 +424,15 @@ int lv_cache_wait_for_clean(struct logical_volume *cache_lv, int *is_clean)
/* Switch to cleaner policy to flush the cache */
cache_seg->cleaner_policy = 1;
- /* Reaload kernel with "cleaner" policy */
+ /* Reload cache volume with "cleaner" policy */
if (!lv_update_and_reload_origin(cache_lv))
return_0;
+
+ if (!sync_local_dev_names(cache_lv->vg->cmd)) {
+ log_error("Failed to sync local devices when clearing cache volume %s.",
+ display_lvname(cache_lv));
+ return 0;
+ }
}
/*
@@ -436,6 +442,12 @@ int lv_cache_wait_for_clean(struct logical_volume *cache_lv, int *is_clean)
if (1) {
if (!lv_refresh_suspend_resume(lock_lv))
return_0;
+
+ if (!sync_local_dev_names(cache_lv->vg->cmd)) {
+ log_error("Failed to sync local devices after final clearing of cache %s.",
+ display_lvname(cache_lv));
+ return 0;
+ }
}
cache_seg->cleaner_policy = 0;