]> sourceware.org Git - lvm2.git/commitdiff
lvchange: avoid multiple metadata updates/reloads/backups
authorHeinz Mauelshagen <heinzm@redhat.com>
Sat, 1 Apr 2017 00:38:48 +0000 (02:38 +0200)
committerHeinz Mauelshagen <heinzm@redhat.com>
Sat, 1 Apr 2017 00:38:48 +0000 (02:38 +0200)
_lvchange_properties_single() processes multiple command
line arguments in a loop causing metadata updates and/or
backups per argument.

Optimize to only perform one update and/or backup
(but necessary interim ones; e.g. for --resync)
per command run.

Related: rhbz1437611

tools/lvchange.c

index 7239f1816bd5fd9853b0a5978feb82f62eea65ad..27b5184168bb7cf8fcc70c90209d47880d5826a8 100644 (file)
@@ -1,6 +1,6 @@
 /*
  * Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
- * Copyright (C) 2004-2016 Red Hat, Inc. All rights reserved.
+ * Copyright (C) 2004-2017 Red Hat, Inc. All rights reserved.
  *
  * This file is part of LVM2.
  *
 
 #include "memlock.h"
 
+/*
+ * Passed back from callee to request caller to
+ * commit / update and reload / backup metadata.
+ *
+ * This allows for one metadata update per command run
+ * (unless mandatory interim ones in callee).
+ */
+enum metadata_request {
+       MR_COMMIT        = 0x1,
+       MR_UPDATE_RELOAD = 0x2,
+       MR_BACKUP        = 0x4
+};
+
+static int _vg_write_commit(const struct logical_volume *lv, const char *what)
+{
+       log_very_verbose("Updating %slogical volume %s on disk(s).",
+                        what ? : "", display_lvname(lv));
+       if (!vg_write(lv->vg) || !vg_commit(lv->vg)) {
+               log_error("Failed to update %smetadata of %s on disk.",
+                         what ? : "", display_lvname(lv));
+               return 0;
+       }
+
+       return 1;
+}
+
 static int _lvchange_permission(struct cmd_context *cmd,
-                               struct logical_volume *lv)
+                               struct logical_volume *lv,
+                               enum metadata_request *mr)
 {
        uint32_t lv_access;
        struct lvinfo info;
@@ -67,14 +94,15 @@ static int _lvchange_permission(struct cmd_context *cmd,
                            display_lvname(lv));
        }
 
-       if (!lv_update_and_reload(lv))
-               return_0;
+       /* Request caller to update and reload metadata */
+       *mr |= MR_UPDATE_RELOAD;
 
        return 1;
 }
 
 static int _lvchange_pool_update(struct cmd_context *cmd,
-                                struct logical_volume *lv)
+                                struct logical_volume *lv,
+                                enum metadata_request *mr)
 {
        int update = 0;
        unsigned val;
@@ -110,9 +138,7 @@ static int _lvchange_pool_update(struct cmd_context *cmd,
        if (!update)
                return 0;
 
-       if (!lv_update_and_reload(lv))
-               return_0;
-
+       *mr |= MR_UPDATE_RELOAD;
        return 1;
 }
 
@@ -345,12 +371,8 @@ static int _lvchange_resync(struct cmd_context *cmd, struct logical_volume *lv)
        if (!seg_is_raid(seg) && !seg->log_lv) {
                if (lv_is_not_synced(lv)) {
                        lv->status &= ~LV_NOTSYNCED;
-                       log_very_verbose("Updating logical volume %s on disk(s).",
-                                        display_lvname(lv));
-                       if (!vg_write(lv->vg) || !vg_commit(lv->vg)) {
-                               log_error("Failed to update metadata on disk.");
+                       if (!_vg_write_commit(lv, NULL))
                                return 0;
-                       }
                }
 
                if (!_reactivate_lv(lv, active, exclusive)) {
@@ -375,8 +397,7 @@ static int _lvchange_resync(struct cmd_context *cmd, struct logical_volume *lv)
                return 0;
        }
 
-       if (!vg_write(lv->vg) || !vg_commit(lv->vg)) {
-               log_error("Failed to update intermediate VG metadata on disk.");
+       if (!_vg_write_commit(lv, "intermediate")) {
                if (!_reactivate_lv(lv, active, exclusive))
                        stack;
                return 0;
@@ -426,11 +447,8 @@ static int _lvchange_resync(struct cmd_context *cmd, struct logical_volume *lv)
                return 0;
        }
 
-       if (!vg_write(lv->vg) || !vg_commit(lv->vg)) {
-               log_error("Failed to update metadata on disk for %s.",
-                         display_lvname(lv));
+       if (!_vg_write_commit(lv, NULL))
                return 0;
-       }
 
        if (!_reactivate_lv(lv, active, exclusive)) {
                backup(lv->vg);
@@ -444,7 +462,9 @@ static int _lvchange_resync(struct cmd_context *cmd, struct logical_volume *lv)
        return 1;
 }
 
-static int _lvchange_alloc(struct cmd_context *cmd, struct logical_volume *lv)
+static int _lvchange_alloc(struct cmd_context *cmd,
+                          struct logical_volume *lv,
+                          enum metadata_request *mr)
 {
        int want_contiguous = arg_int_value(cmd, contiguous_ARG, 0);
        alloc_policy_t alloc = (alloc_policy_t)
@@ -467,16 +487,16 @@ static int _lvchange_alloc(struct cmd_context *cmd, struct logical_volume *lv)
        log_very_verbose("Updating logical volume %s on disk(s).", display_lvname(lv));
 
        /* No need to suspend LV for this change */
-       if (!vg_write(lv->vg) || !vg_commit(lv->vg))
-               return_0;
 
-       backup(lv->vg);
+       /* Request caller to commit+backuo metadata */
+       *mr |= (MR_COMMIT | MR_BACKUP);
 
        return 1;
 }
 
 static int _lvchange_errorwhenfull(struct cmd_context *cmd,
-                                  struct logical_volume *lv)
+                                  struct logical_volume *lv,
+                                  enum metadata_request *mr)
 {
        unsigned ewf = arg_int_value(cmd, errorwhenfull_ARG, 0);
 
@@ -491,14 +511,15 @@ static int _lvchange_errorwhenfull(struct cmd_context *cmd,
        else
                lv->status &= ~LV_ERROR_WHEN_FULL;
 
-       if (!lv_update_and_reload(lv))
-               return_0;
+       /* Request caller to update and reload metadata */
+       *mr |= MR_UPDATE_RELOAD;
 
        return 1;
 }
 
 static int _lvchange_readahead(struct cmd_context *cmd,
-                              struct logical_volume *lv)
+                              struct logical_volume *lv,
+                              enum metadata_request *mr)
 {
        unsigned read_ahead = 0;
        unsigned pagesize = (unsigned) lvm_getpagesize() >> SECTOR_SHIFT;
@@ -537,14 +558,15 @@ static int _lvchange_readahead(struct cmd_context *cmd,
        log_verbose("Setting read ahead to %u for %s.",
                    read_ahead, display_lvname(lv));
 
-       if (!lv_update_and_reload(lv))
-               return_0;
+       /* Request caller to update and reload metadata */
+       *mr |= MR_UPDATE_RELOAD;
 
        return 1;
 }
 
 static int _lvchange_persistent(struct cmd_context *cmd,
-                               struct logical_volume *lv)
+                               struct logical_volume *lv,
+                               enum metadata_request *mr)
 {
        enum activation_change activate = CHANGE_AN;
 
@@ -600,11 +622,8 @@ static int _lvchange_persistent(struct cmd_context *cmd,
                            lv->major, lv->minor, display_lvname(lv));
        }
 
-       log_very_verbose("Updating logical volume %s on disk(s).",
-                        display_lvname(lv));
-
-       if (!vg_write(lv->vg) || !vg_commit(lv->vg))
-               return_0;
+       if (!_vg_write_commit(lv, NULL))
+               return 0;
 
        if (activate != CHANGE_AN) {
                log_verbose("Re-activating logical volume %s.", display_lvname(lv));
@@ -615,12 +634,15 @@ static int _lvchange_persistent(struct cmd_context *cmd,
                }
        }
 
-       backup(lv->vg);
+       /* Request caller to backup metadata */
+       *mr |= MR_BACKUP;
 
        return 1;
 }
 
-static int _lvchange_cache(struct cmd_context *cmd, struct logical_volume *lv)
+static int _lvchange_cache(struct cmd_context *cmd,
+                          struct logical_volume *lv,
+                          enum metadata_request *mr)
 {
        cache_metadata_format_t format;
        cache_mode_t mode;
@@ -655,8 +677,8 @@ static int _lvchange_cache(struct cmd_context *cmd, struct logical_volume *lv)
            !cache_set_policy(first_seg(lv), name, settings))
                goto_out;
 
-       if (!lv_update_and_reload(lv))
-               goto_out;
+       /* Request caller to update and reload metadata */
+       *mr |= MR_UPDATE_RELOAD;
 
        r = 1;
 out:
@@ -666,7 +688,8 @@ out:
        return r;
 }
 
-static int _lvchange_tag(struct cmd_context *cmd, struct logical_volume *lv, int arg)
+static int _lvchange_tag(struct cmd_context *cmd, struct logical_volume *lv,
+                        int arg, enum metadata_request *mr)
 {
        if (!change_tag(cmd, NULL, lv, NULL, arg))
                return_0;
@@ -674,10 +697,8 @@ static int _lvchange_tag(struct cmd_context *cmd, struct logical_volume *lv, int
        log_very_verbose("Updating logical volume %s on disk(s).", display_lvname(lv));
 
        /* No need to suspend LV for this change */
-       if (!vg_write(lv->vg) || !vg_commit(lv->vg))
-               return_0;
-
-       backup(lv->vg);
+       /* Request caller to commit+backup metadata */
+       *mr |= (MR_COMMIT | MR_BACKUP);
 
        return 1;
 }
@@ -730,7 +751,8 @@ static int _lvchange_rebuild(struct logical_volume *lv)
        return lv_raid_rebuild(lv, rebuild_pvh);
 }
 
-static int _lvchange_writemostly(struct logical_volume *lv)
+static int _lvchange_writemostly(struct logical_volume *lv,
+                                enum metadata_request *mr)
 {
        int pv_count, i = 0;
        uint32_t s, writemostly;
@@ -847,13 +869,14 @@ static int _lvchange_writemostly(struct logical_volume *lv)
                }
        }
 
-       if (!lv_update_and_reload(lv))
-               return_0;
+       /* Request caller to update and reload metadata */
+       *mr |= MR_UPDATE_RELOAD;
 
        return 1;
 }
 
-static int _lvchange_recovery_rate(struct logical_volume *lv)
+static int _lvchange_recovery_rate(struct logical_volume *lv,
+                                  enum metadata_request *mr)
 {
        struct cmd_context *cmd = lv->vg->cmd;
        struct lv_segment *raid_seg = first_seg(lv);
@@ -871,13 +894,14 @@ static int _lvchange_recovery_rate(struct logical_volume *lv)
                return 0;
        }
 
-       if (!lv_update_and_reload(lv))
-               return_0;
+       /* Request caller to update and reload metadata */
+       *mr |= MR_UPDATE_RELOAD;
 
        return 1;
 }
 
-static int _lvchange_profile(struct logical_volume *lv)
+static int _lvchange_profile(struct logical_volume *lv,
+                            enum metadata_request *mr)
 {
        const char *old_profile_name, *new_profile_name;
        struct profile *new_profile;
@@ -900,15 +924,13 @@ static int _lvchange_profile(struct logical_volume *lv)
        log_verbose("Changing configuration profile for LV %s: %s -> %s.",
                    display_lvname(lv), old_profile_name, new_profile_name);
 
-       if (!vg_write(lv->vg) || !vg_commit(lv->vg))
-               return_0;
-
-       backup(lv->vg);
+       /* Request caller to commit+backup metadata */
+       *mr |= (MR_COMMIT | MR_BACKUP);
 
        return 1;
 }
 
-static int _lvchange_activation_skip(struct logical_volume *lv)
+static int _lvchange_activation_skip(struct logical_volume *lv, enum metadata_request *mr)
 {
        int skip = arg_int_value(lv->vg->cmd, setactivationskip_ARG, 0);
 
@@ -917,10 +939,26 @@ static int _lvchange_activation_skip(struct logical_volume *lv)
        log_verbose("Changing activation skip flag to %s for LV %s.",
                    display_lvname(lv), skip ? "enabled" : "disabled");
 
-       if (!vg_write(lv->vg) || !vg_commit(lv->vg))
-               return_0;
+       /* Request caller to commit+backup metadata */
+       *mr |= (MR_COMMIT | MR_BACKUP);
 
-       backup(lv->vg);
+       return 1;
+}
+
+/* Update and reload or commit and/or backup metadata for @lv as requested by @mr */
+static int _commit_update_backup(struct logical_volume *lv,
+                                const enum metadata_request mr)
+{
+       if (mr & MR_UPDATE_RELOAD) {
+               if (!lv_update_and_reload(lv))
+                       return_0;
+
+       } else if ((mr & MR_COMMIT) &&
+                  !_vg_write_commit(lv, NULL))
+               return 0;
+
+       if (mr & MR_BACKUP)
+               backup(lv->vg);
 
        return 1;
 }
@@ -944,6 +982,7 @@ static int _lvchange_properties_single(struct cmd_context *cmd,
 {
        int doit = 0, docmds = 0;
        int i, opt_enum;
+       enum metadata_request mr = 0;
 
        /*
         * If a persistent lv lock already exists from activation
@@ -962,6 +1001,7 @@ static int _lvchange_properties_single(struct cmd_context *cmd,
                if (!arg_is_set(cmd, opt_enum))
                        continue;
 
+               /* Archive will only happen once per run */
                if (!archive(lv->vg))
                        return_ECMD_FAILED;
 
@@ -969,60 +1009,60 @@ static int _lvchange_properties_single(struct cmd_context *cmd,
 
                switch (opt_enum) {
                case permission_ARG:
-                       doit += _lvchange_permission(cmd, lv);
+                       doit += _lvchange_permission(cmd, lv, &mr);
                        break;
 
                case alloc_ARG:
                case contiguous_ARG:
-                       doit += _lvchange_alloc(cmd, lv);
+                       doit += _lvchange_alloc(cmd, lv, &mr);
                        break;
 
                case errorwhenfull_ARG:
-                       doit += _lvchange_errorwhenfull(cmd, lv);
+                       doit += _lvchange_errorwhenfull(cmd, lv, &mr);
                        break;
 
                case readahead_ARG:
-                       doit += _lvchange_readahead(cmd, lv);
+                       doit += _lvchange_readahead(cmd, lv, &mr);
                        break;
 
                case persistent_ARG:
-                       doit += _lvchange_persistent(cmd, lv);
+                       doit += _lvchange_persistent(cmd, lv, &mr);
                        break;
 
                case discards_ARG:
                case zero_ARG:
-                       doit += _lvchange_pool_update(cmd, lv);
+                       doit += _lvchange_pool_update(cmd, lv, &mr);
                        break;
 
                case addtag_ARG:
                case deltag_ARG:
-                       doit += _lvchange_tag(cmd, lv, opt_enum);
+                       doit += _lvchange_tag(cmd, lv, opt_enum, &mr);
                        break;
 
                case writemostly_ARG:
                case writebehind_ARG:
-                       doit += _lvchange_writemostly(lv);
+                       doit += _lvchange_writemostly(lv, &mr);
                        break;
 
                case minrecoveryrate_ARG:
                case maxrecoveryrate_ARG:
-                       doit += _lvchange_recovery_rate(lv);
+                       doit += _lvchange_recovery_rate(lv, &mr);
                        break;
 
                case profile_ARG:
                case metadataprofile_ARG:
                case detachprofile_ARG:
-                       doit += _lvchange_profile(lv);
+                       doit += _lvchange_profile(lv, &mr);
                        break;
 
                case setactivationskip_ARG:
-                       doit += _lvchange_activation_skip(lv);
+                       doit += _lvchange_activation_skip(lv, &mr);
                        break;
 
                case cachemode_ARG:
                case cachepolicy_ARG:
                case cachesettings_ARG:
-                       doit += _lvchange_cache(cmd, lv);
+                       doit += _lvchange_cache(cmd, lv, &mr);
                        break;
 
                default:
@@ -1037,6 +1077,9 @@ static int _lvchange_properties_single(struct cmd_context *cmd,
        if (doit != docmds)
                return_ECMD_FAILED;
 
+       if (!_commit_update_backup(lv, mr))
+               return_ECMD_FAILED;
+
        return ECMD_PROCESSED;
 }
 
@@ -1379,7 +1422,12 @@ static int _lvchange_persistent_single(struct cmd_context *cmd,
                                       struct logical_volume *lv,
                                       struct processing_handle *handle)
 {
-       if (!_lvchange_persistent(cmd, lv))
+       enum metadata_request mr = 0;
+
+       if (!_lvchange_persistent(cmd, lv, &mr))
+               return_ECMD_FAILED;
+
+       if (!_commit_update_backup(lv, mr))
                return_ECMD_FAILED;
 
        return ECMD_PROCESSED;
@@ -1412,4 +1460,3 @@ int lvchange(struct cmd_context *cmd, int argc, char **argv)
                  cmd->command->command_index, cmd->command->command_id);
        return ECMD_FAILED;
 }
-
This page took 0.053535 seconds and 5 git commands to generate.