From a56e5d615865121de519f3e5e5ba6d97d924873a Mon Sep 17 00:00:00 2001 From: loli10K Date: Sat, 22 Oct 2016 19:49:29 +0200 Subject: [PATCH 1/2] Allow for '-o feature@=disabled' on the command line. Sometimes it is desired to specifically disable a feature directly on the 'zpool create' command line. Signed-off-by: Turbo Fredriksson Signed-off-by: loli10K --- cmd/zpool/zpool_main.c | 33 ++++--- lib/libzfs/libzfs_pool.c | 5 +- man/man8/zpool.8 | 19 +++- tests/runfiles/linux.run | 3 +- .../cli_root/zpool_create/Makefile.am | 3 +- .../zpool_create_features_004_neg.ksh | 2 +- .../zpool_create_features_005_pos.ksh | 97 +++++++++++++++++++ 7 files changed, 140 insertions(+), 22 deletions(-) create mode 100755 tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_features_005_pos.ksh diff --git a/cmd/zpool/zpool_main.c b/cmd/zpool/zpool_main.c index 3431ec67a0a7..b6702f22c099 100644 --- a/cmd/zpool/zpool_main.c +++ b/cmd/zpool/zpool_main.c @@ -920,6 +920,7 @@ zpool_do_labelclear(int argc, char **argv) * -m Set default mountpoint for the root dataset. By default it's * '/' * -o Set property=value. + * -o Set feature@feature=enabled|disabled. * -d Don't automatically enable all supported pool features * (individual features can be enabled with -o). * -O Set fsproperty=value in the pool's root file system @@ -1188,22 +1189,26 @@ zpool_do_create(int argc, char **argv) /* * Hand off to libzfs. */ - if (enable_all_pool_feat) { - spa_feature_t i; - for (i = 0; i < SPA_FEATURES; i++) { - char propname[MAXPATHLEN]; - zfeature_info_t *feat = &spa_feature_table[i]; - - (void) snprintf(propname, sizeof (propname), - "feature@%s", feat->fi_uname); + spa_feature_t i; + for (i = 0; i < SPA_FEATURES; i++) { + char propname[MAXPATHLEN]; + char *propval; + zfeature_info_t *feat = &spa_feature_table[i]; - /* - * Skip feature if user specified it manually - * on the command line. - */ - if (nvlist_exists(props, propname)) - continue; + (void) snprintf(propname, sizeof (propname), + "feature@%s", feat->fi_uname); + /* + * Only features contained in props will be enabled: + * remove from the nvlist every ZFS_FEATURE_DISABLED + * value and add every missing ZFS_FEATURE_ENABLED if + * enable_all_pool_feat is set. + */ + if (!nvlist_lookup_string(props, propname, &propval)) { + if (strcmp(propval, ZFS_FEATURE_DISABLED) == 0) + (void) nvlist_remove_all(props, + propname); + } else if (enable_all_pool_feat) { ret = add_prop_list(propname, ZFS_FEATURE_ENABLED, &props, B_TRUE); if (ret != 0) diff --git a/lib/libzfs/libzfs_pool.c b/lib/libzfs/libzfs_pool.c index ebca76834268..04689a93e6bb 100644 --- a/lib/libzfs/libzfs_pool.c +++ b/lib/libzfs/libzfs_pool.c @@ -502,10 +502,11 @@ zpool_valid_proplist(libzfs_handle_t *hdl, const char *poolname, } (void) nvpair_value_string(elem, &strval); - if (strcmp(strval, ZFS_FEATURE_ENABLED) != 0) { + if (strcmp(strval, ZFS_FEATURE_ENABLED) != 0 && + strcmp(strval, ZFS_FEATURE_DISABLED) != 0) { zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, "property '%s' can only be set to " - "'enabled'"), propname); + "'enabled' or 'disabled'"), propname); (void) zfs_error(hdl, EZFS_BADPROP, errbuf); goto error; } diff --git a/man/man8/zpool.8 b/man/man8/zpool.8 index 3518175699f8..43dccc05a3ef 100644 --- a/man/man8/zpool.8 +++ b/man/man8/zpool.8 @@ -41,8 +41,9 @@ zpool \- configures ZFS storage pools .LP .nf -\fBzpool create\fR [\fB-fnd\fR] [\fB-o\fR \fIproperty=value\fR] ... [\fB-O\fR \fIfile-system-property=value\fR] - ... [\fB-m\fR \fImountpoint\fR] [\fB-R\fR \fIroot\fR] [\fB-t\fR \fItname\fR] \fIpool\fR \fIvdev\fR ... +\fBzpool create\fR [\fB-fnd\fR] [\fB-o\fR \fIproperty=value\fR] ... [\fB-o\fR feature@\fIfeature=value\fR] + ... [\fB-O\fR \fIfile-system-property=value\fR] ... [\fB-m\fR \fImountpoint\fR] [\fB-R\fR \fIroot\fR] + ... [\fB-t\fR \fItname\fR] \fIpool\fR \fIvdev\fR ... .fi .LP @@ -877,7 +878,7 @@ Clears device errors in a pool. If no arguments are specified, all device errors .sp .ne 2 .na -\fB\fBzpool create\fR [\fB-fnd\fR] [\fB-o\fR \fIproperty=value\fR] ... [\fB-O\fR \fIfile-system-property=value\fR] ... [\fB-m\fR \fImountpoint\fR] [\fB-R\fR \fIroot\fR] [\fB-t\fR \fItname\fR] \fIpool\fR \fIvdev\fR ...\fR +\fB\fBzpool create\fR [\fB-fnd\fR] [\fB-o\fR \fIproperty=value\fR] ... [\fB-o\fR feature@\fIfeature=value\fR] ... [\fB-O\fR \fIfile-system-property=value\fR] ... [\fB-m\fR \fImountpoint\fR] [\fB-R\fR \fIroot\fR] [\fB-t\fR \fItname\fR] \fIpool\fR \fIvdev\fR ...\fR .ad .sp .6 .RS 4n @@ -930,6 +931,18 @@ Do not enable any features on the new pool. Individual features can be enabled b Sets the given pool properties. See the "Properties" section for a list of valid properties that can be set. .RE +.sp +.ne 2 +.na +\fB\fB-o\fR feature@\fIfeature=value\fR [\fB-o\fR feature@\fIfeature=value\fR] ...\fR +.ad +.sp .6 +.RS 4n +Sets the given pool feature. See \fBzpool-features(5)\fR for a list of valid features that can be set. +.sp +Value can be either \fBdisabled\fR or \fBenabled\fR. +.RE + .sp .ne 2 .na diff --git a/tests/runfiles/linux.run b/tests/runfiles/linux.run index 086ea2b273ea..b7ae839bccd0 100644 --- a/tests/runfiles/linux.run +++ b/tests/runfiles/linux.run @@ -249,7 +249,8 @@ tests = [ 'zpool_create_018_pos', 'zpool_create_019_pos', 'zpool_create_021_pos', 'zpool_create_022_pos', 'zpool_create_023_neg', 'zpool_create_features_001_pos', 'zpool_create_features_002_pos', - 'zpool_create_features_003_pos', 'zpool_create_features_004_neg'] + 'zpool_create_features_003_pos', 'zpool_create_features_004_neg', + 'zpool_create_features_005_pos'] # DISABLED: # zpool_destroy_001_pos - failure should be investigated diff --git a/tests/zfs-tests/tests/functional/cli_root/zpool_create/Makefile.am b/tests/zfs-tests/tests/functional/cli_root/zpool_create/Makefile.am index 0530dec1a8be..987e1699db9e 100644 --- a/tests/zfs-tests/tests/functional/cli_root/zpool_create/Makefile.am +++ b/tests/zfs-tests/tests/functional/cli_root/zpool_create/Makefile.am @@ -29,4 +29,5 @@ dist_pkgdata_SCRIPTS = \ zpool_create_features_001_pos.ksh \ zpool_create_features_002_pos.ksh \ zpool_create_features_003_pos.ksh \ - zpool_create_features_004_neg.ksh + zpool_create_features_004_neg.ksh \ + zpool_create_features_005_pos.ksh diff --git a/tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_features_004_neg.ksh b/tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_features_004_neg.ksh index a52e86251df6..2b465d5a41e2 100755 --- a/tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_features_004_neg.ksh +++ b/tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_features_004_neg.ksh @@ -39,7 +39,7 @@ verify_runnable "global" properties="\ -feature@async_destroy=disabled \ +feature@async_destroy=disable \ feature@async_destroy=active \ feature@xxx_fake_xxx=enabled \ unsupported@some_feature=inactive \ diff --git a/tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_features_005_pos.ksh b/tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_features_005_pos.ksh new file mode 100755 index 000000000000..c7e1ab246b72 --- /dev/null +++ b/tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_features_005_pos.ksh @@ -0,0 +1,97 @@ +#!/bin/ksh -p +# +# CDDL HEADER START +# +# The contents of this file are subject to the terms of the +# Common Development and Distribution License (the "License"). +# You may not use this file except in compliance with the License. +# +# You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE +# or http://www.opensolaris.org/os/licensing. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# When distributing Covered Code, include this CDDL HEADER in each +# file and include the License file at usr/src/OPENSOLARIS.LICENSE. +# If applicable, add the following below this CDDL HEADER, with the +# fields enclosed by brackets "[]" replaced with your own identifying +# information: Portions Copyright [yyyy] [name of copyright owner] +# +# CDDL HEADER END +# + +# +# Copyright (c) 2012 by Delphix. All rights reserved. +# + +. $STF_SUITE/include/libtest.shlib +. $STF_SUITE/tests/functional/cli_root/zpool_create/zpool_create.shlib + +################################################################################ +# +# Specifically disabling a feature, all other features should be enabled. +# +# 1. Loop through all existing features: +# a. Create a new pool with '-o feature@XXX=disabled'. +# b. Verify that every other features is in the 'enabled' state. +# +################################################################################ + +verify_runnable "global" + +function cleanup +{ + datasetexists $TESTPOOL && log_must $ZPOOL destroy $TESTPOOL +} + +function check_features +{ + feature="${1}" + feature_set=false + other_feature_set=false + + ${ZPOOL} get all ${TESTPOOL} | \ + grep feature@ | \ + while read line; do + set -- $(echo "${line}") + + if [[ "${3}" == "enabled" || "${3}" == "active" ]]; then + if [[ "feature@${feature}" == "${2}" ]]; then + feature_set=true + else + other_feature_set=true + fi + fi + done + + if [[ "${feature_set}" == "true" ]]; then + # This is a success + if [[ "${other_feature_set}" == "true" ]]; then + # .. but if _any_ of the other features is enabled, + # it's a failure! + return 0 + else + # All good - feature is enabled, all other disabled. + return 1 + fi + else + # Feature is not set - failure. + return 1 + fi +} + +log_onexit cleanup + +for feature in async_destroy bookmarks embedded_data empty_bpobj enabled_txg \ + extensible_dataset filesystem_limits hole_birth large_blocks \ + lz4_compress spacemap_histogram large_dnode userobj_accounting \ + sha512 skein edonr +do + log_assert "'zpool create' creates pools with ${feature} disabled" + + log_must $ZPOOL create -f -o "feature@${feature}=disabled" $TESTPOOL $DISKS + check_features ${feature} + log_must $ZPOOL destroy -f $TESTPOOL + + log_pass "'zpool create' creates pools with ${feature} disabled" +done From b06b75ee0b300c24054a199ae101b9fa9a581715 Mon Sep 17 00:00:00 2001 From: Brian Behlendorf Date: Tue, 25 Oct 2016 12:39:58 -0700 Subject: [PATCH 2/2] Fix zpool_create_features_005_pos test case Updated the test case so it behaves in the manor originally indended. * Consistent sytle with other test cases * log_pass causes immediate exit, moved outside loop * Simplified check_features function * log_must added to check_features call * Several features removed to keep test time short Signed-off-by: Brian Behlendorf --- .../zpool_create_features_005_pos.ksh | 70 +++++++++---------- 1 file changed, 33 insertions(+), 37 deletions(-) diff --git a/tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_features_005_pos.ksh b/tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_features_005_pos.ksh index c7e1ab246b72..488f11c9b7c3 100755 --- a/tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_features_005_pos.ksh +++ b/tests/zfs-tests/tests/functional/cli_root/zpool_create/zpool_create_features_005_pos.ksh @@ -33,7 +33,7 @@ # # 1. Loop through all existing features: # a. Create a new pool with '-o feature@XXX=disabled'. -# b. Verify that every other features is in the 'enabled' state. +# b. Verify that every other feature is 'enabled' or 'active'. # ################################################################################ @@ -46,52 +46,48 @@ function cleanup function check_features { - feature="${1}" - feature_set=false - other_feature_set=false + typeset feature="${1}" - ${ZPOOL} get all ${TESTPOOL} | \ - grep feature@ | \ - while read line; do + ${ZPOOL} get all ${TESTPOOL} | $GREP feature@ | while read line; do set -- $(echo "${line}") - if [[ "${3}" == "enabled" || "${3}" == "active" ]]; then - if [[ "feature@${feature}" == "${2}" ]]; then - feature_set=true - else - other_feature_set=true + if [[ "feature@${feature}" == "${2}" ]]; then + # Failure passed feature must be disabled. + if [[ "${3}" != "disabled" ]]; then + return 1; fi - fi - done - - if [[ "${feature_set}" == "true" ]]; then - # This is a success - if [[ "${other_feature_set}" == "true" ]]; then - # .. but if _any_ of the other features is enabled, - # it's a failure! - return 0 else - # All good - feature is enabled, all other disabled. - return 1 + # Failure other features must be enabled or active. + if [[ "${3}" != "enabled" && "${3}" != "active" ]]; then + return 2; + fi fi - else - # Feature is not set - failure. - return 1 - fi + done + + # All features enabled or active except the expected one. + return 0 } log_onexit cleanup -for feature in async_destroy bookmarks embedded_data empty_bpobj enabled_txg \ - extensible_dataset filesystem_limits hole_birth large_blocks \ - lz4_compress spacemap_histogram large_dnode userobj_accounting \ - sha512 skein edonr -do - log_assert "'zpool create' creates pools with ${feature} disabled" +# Several representative features are tested to keep the test time short. +# The features 'extensible_dataset' and 'enabled_txg' are intentionally +# excluded because other features depend on them. +set -A features \ + "hole_birth" \ + "large_blocks" \ + "large_dnode" \ + "userobj_accounting" - log_must $ZPOOL create -f -o "feature@${feature}=disabled" $TESTPOOL $DISKS - check_features ${feature} - log_must $ZPOOL destroy -f $TESTPOOL +typeset -i i=0 +while (( $i < ${#features[*]} )); do + log_assert "'zpool create' creates pools with ${features[i]} disabled" - log_pass "'zpool create' creates pools with ${feature} disabled" + log_must $ZPOOL create -f -o "feature@${features[i]}=disabled" \ + $TESTPOOL $DISKS + log_must check_features "${features[i]}" + log_must $ZPOOL destroy -f $TESTPOOL + (( i = i+1 )) done + +log_pass "'zpool create -o feature@feature=disabled' disables features"