diff --git a/WHATS_NEW b/WHATS_NEW index c3d56de92..b1f8410fc 100644 --- a/WHATS_NEW +++ b/WHATS_NEW @@ -1,5 +1,6 @@ Version 2.02.112 - ===================================== + Check for zeroing of volume after segment type is fully detected. Better support for persistent major and minor options with lvcreate. Refactor lvcreate towards more complete validation of all supported options. Support lvcreate --type linear. diff --git a/tools/lvcreate.c b/tools/lvcreate.c index 2744b09fa..767405be4 100644 --- a/tools/lvcreate.c +++ b/tools/lvcreate.c @@ -588,19 +588,6 @@ static int _read_activation_params(struct cmd_context *cmd, lp->activate = (activation_change_t) arg_uint_value(cmd, activate_ARG, CHANGE_AY); - if (!is_change_activating(lp->activate)) { - if (lp->zero && !seg_is_thin(lp)) { - log_error("--activate n requires --zero n"); - return 0; - } - } else if (lp->activate == CHANGE_AAY) { - if (arg_count(cmd, zero_ARG) || arg_count(cmd, wipesignatures_ARG)) { - log_error("-Z and -W is incompatible with --activate a"); - return 0; - } - lp->zero = 0; - } - /* Read ahead */ lp->read_ahead = arg_uint_value(cmd, readahead_ARG, cmd->default_settings.read_ahead); @@ -616,13 +603,6 @@ static int _read_activation_params(struct cmd_context *cmd, "of %uK page size.", lp->read_ahead, pagesize >> 1); } - /* Permissions */ - if (!(lp->permission & LVM_WRITE)) { - /* Must not zero/wipe read only volume */ - lp->zero = 0; - lp->wipe_signatures = 0; - } - /* Persistent minor (and major), default 'n' */ if (!get_and_validate_major_minor(cmd, vg->fid->fmt, &lp->major, &lp->minor)) return_0; @@ -636,13 +616,6 @@ static int _read_activation_params(struct cmd_context *cmd, if (arg_is_set(cmd, ignoreactivationskip_ARG)) lp->activation_skip |= ACTIVATION_SKIP_IGNORE; - if (lp->zero && (lp->activation_skip & ACTIVATION_SKIP_SET_ENABLED) - && !(lp->activation_skip & ACTIVATION_SKIP_IGNORE)) { - log_error("--setactivationskip y requires either --zero n " - "or --ignoreactivationskip"); - return 0; - } - return 1; } @@ -984,16 +957,12 @@ static int _lvcreate_params(struct cmd_context *cmd, } } - /* - * Should we zero/wipe signatures on the lv. - */ - lp->zero = (lp->segtype->flags & SEG_CANNOT_BE_ZEROED) - ? 0 : arg_int_value(cmd, zero_ARG, 1); + /* Should we zero/wipe signatures on the lv, default to 'y' */ + lp->zero = arg_int_value(cmd, zero_ARG, 1); - if (arg_count(cmd, wipesignatures_ARG)) { + if (arg_is_set(cmd, wipesignatures_ARG)) { /* If -W/--wipesignatures is given on command line directly, respect it. */ - lp->wipe_signatures = (lp->segtype->flags & SEG_CANNOT_BE_ZEROED) - ? 0 : arg_int_value(cmd, wipesignatures_ARG, 1); + lp->wipe_signatures = arg_int_value(cmd, wipesignatures_ARG, 1); } else { /* * If -W/--wipesignatures is not given on command line, @@ -1346,6 +1315,53 @@ static int _check_pool_parameters(struct cmd_context *cmd, return 1; } +/* + * Check zero_ARG with default value set to value of wipesignatures_ARG + * with its default set to 'n'. So if user specifies on command line either + * -Zy or -Wy it will check for incompatible options will report error then. + * + * Catching cases like we cannot fulfill: + * lvcreate [-an][-pr][-aay][-ky] [-Zy][-Wy] + */ +static int _check_zero_parameters(struct cmd_context *cmd, struct lvcreate_params *lp) +{ + char buf[NAME_LEN + 128]; + + /* -Z has different meaning for thins */ + if (seg_is_thin(lp)) + return 1; + + /* If there is some problem, buffer will not be empty */ + if (dm_snprintf(buf, sizeof(buf), "%s%s%s%s%s%s%s", + lp->origin_name ? "origin " : "", + lp->origin_name ? : "", + lp->origin_name ? " " : "", + !(lp->permission & LVM_WRITE) ? "read-only " : "", + !is_change_activating(lp->activate) ? "inactive " : "", + (lp->activate == CHANGE_AAY) ? "auto activated " : "", + ((lp->activation_skip & ACTIVATION_SKIP_SET_ENABLED) && + !(lp->activation_skip & ACTIVATION_SKIP_IGNORE)) + ? "skipped from activation " : "") < 0) { + log_error(INTERNAL_ERROR "Buffer is too small for dm_snprintf()."); + return 0; + } + + if (buf[0] || (lp->segtype->flags & SEG_CANNOT_BE_ZEROED)) { + /* Found condition that prevents zeroing */ + if (arg_int_value(cmd, zero_ARG, arg_int_value(cmd, wipesignatures_ARG, 0))) { + if (!(lp->segtype->flags & SEG_CANNOT_BE_ZEROED)) { + log_error("Cannot zero %slogical volume with option -Zy or -Wy.", buf); + return 0; + } + log_print_unless_silent("Ignoring option -Zy or -Wy for unzeroable %s volume.", + lp->segtype->name); + } + lp->zero = lp->wipe_signatures = 0; + } + + return 1; +} + /* * Ensure the set of thin parameters extracted from the command line is consistent. @@ -1429,6 +1445,10 @@ int lvcreate(struct cmd_context *cmd, int argc, char **argv) if (!_check_pool_parameters(cmd, vg, &lp, &lcp)) goto_out; + /* All types are checked */ + if (!_check_zero_parameters(cmd, &lp)) + return_0; + if (!_update_extents_params(vg, &lp, &lcp)) goto_out;