mirror of
				https://git.proxmox.com/git/mirror_zfs.git
				synced 2025-10-25 09:25:00 +03:00 
			
		
		
		
	Improvements to the 'compatibility' property
Several improvements to the operation of the 'compatibility' property: 1) Improved handling of unrecognized features: Change the way unrecognized features in compatibility files are handled. * invalid features in files under /usr/share/zfs/compatibility.d only get a warning (as these may refer to future features not yet in the library), * invalid features in files under /etc/zfs/compatibility.d get an error (as these are presumed to refer to the current system). 2) Improved error reporting from zpool_load_compat. Note: slight ABI change to zpool_load_compat for better error reporting. 3) compatibility=legacy inhibits all 'zpool upgrade' operations. 4) Detect when features are enabled outside current compatibility set * zpool set compatibility=foo <-- print a warning * zpool set feature@xxx=enabled <-- error * zpool status <-- indicate this state Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov> Signed-off-by: Colm Buckley <colm@tuatha.org> Closes #11861
This commit is contained in:
		
							parent
							
								
									7de1797cee
								
							
						
					
					
						commit
						1f3de97374
					
				| @ -786,7 +786,7 @@ add_prop_list(const char *propname, char *propval, nvlist_t **props, | |||||||
| 
 | 
 | ||||||
| 	if (poolprop) { | 	if (poolprop) { | ||||||
| 		const char *vname = zpool_prop_to_name(ZPOOL_PROP_VERSION); | 		const char *vname = zpool_prop_to_name(ZPOOL_PROP_VERSION); | ||||||
| 		const char *fname = | 		const char *cname = | ||||||
| 		    zpool_prop_to_name(ZPOOL_PROP_COMPATIBILITY); | 		    zpool_prop_to_name(ZPOOL_PROP_COMPATIBILITY); | ||||||
| 
 | 
 | ||||||
| 		if ((prop = zpool_name_to_prop(propname)) == ZPOOL_PROP_INVAL && | 		if ((prop = zpool_name_to_prop(propname)) == ZPOOL_PROP_INVAL && | ||||||
| @ -811,16 +811,19 @@ add_prop_list(const char *propname, char *propval, nvlist_t **props, | |||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/*
 | 		/*
 | ||||||
| 		 * compatibility property and version should not be specified | 		 * if version is specified, only "legacy" compatibility | ||||||
| 		 * at the same time. | 		 * may be requested | ||||||
| 		 */ | 		 */ | ||||||
| 		if ((prop == ZPOOL_PROP_COMPATIBILITY && | 		if ((prop == ZPOOL_PROP_COMPATIBILITY && | ||||||
|  | 		    strcmp(propval, ZPOOL_COMPAT_LEGACY) != 0 && | ||||||
| 		    nvlist_exists(proplist, vname)) || | 		    nvlist_exists(proplist, vname)) || | ||||||
| 		    (prop == ZPOOL_PROP_VERSION && | 		    (prop == ZPOOL_PROP_VERSION && | ||||||
| 		    nvlist_exists(proplist, fname))) { | 		    nvlist_exists(proplist, cname) && | ||||||
| 			(void) fprintf(stderr, gettext("'compatibility' and " | 		    strcmp(fnvlist_lookup_string(proplist, cname), | ||||||
| 			    "'version' properties cannot be specified " | 		    ZPOOL_COMPAT_LEGACY) != 0)) { | ||||||
| 			    "together\n")); | 			(void) fprintf(stderr, gettext("when 'version' is " | ||||||
|  | 			    "specified, the 'compatibility' feature may only " | ||||||
|  | 			    "be set to '" ZPOOL_COMPAT_LEGACY "'\n")); | ||||||
| 			return (2); | 			return (2); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| @ -1674,6 +1677,7 @@ zpool_do_create(int argc, char **argv) | |||||||
| 		 *    - enable_pool_features (ie: no '-d' or '-o version') | 		 *    - enable_pool_features (ie: no '-d' or '-o version') | ||||||
| 		 *    - it's supported by the kernel module | 		 *    - it's supported by the kernel module | ||||||
| 		 *    - it's in the requested feature set | 		 *    - it's in the requested feature set | ||||||
|  | 		 *  - warn if it's enabled but not in compat | ||||||
| 		 */ | 		 */ | ||||||
| 		for (spa_feature_t i = 0; i < SPA_FEATURES; i++) { | 		for (spa_feature_t i = 0; i < SPA_FEATURES; i++) { | ||||||
| 			char propname[MAXPATHLEN]; | 			char propname[MAXPATHLEN]; | ||||||
| @ -1687,6 +1691,14 @@ zpool_do_create(int argc, char **argv) | |||||||
| 				if (strcmp(propval, ZFS_FEATURE_DISABLED) == 0) | 				if (strcmp(propval, ZFS_FEATURE_DISABLED) == 0) | ||||||
| 					(void) nvlist_remove_all(props, | 					(void) nvlist_remove_all(props, | ||||||
| 					    propname); | 					    propname); | ||||||
|  | 				if (strcmp(propval, | ||||||
|  | 				    ZFS_FEATURE_ENABLED) == 0 && | ||||||
|  | 				    !requested_features[i]) | ||||||
|  | 					(void) fprintf(stderr, gettext( | ||||||
|  | 					    "Warning: feature \"%s\" enabled " | ||||||
|  | 					    "but is not in specified " | ||||||
|  | 					    "'compatibility' feature set.\n"), | ||||||
|  | 					    feat->fi_uname); | ||||||
| 			} else if ( | 			} else if ( | ||||||
| 			    enable_pool_features && | 			    enable_pool_features && | ||||||
| 			    feat->fi_zfs_mod_supported && | 			    feat->fi_zfs_mod_supported && | ||||||
| @ -2717,8 +2729,10 @@ show_import(nvlist_t *config, boolean_t report_error) | |||||||
| 
 | 
 | ||||||
| 	case ZPOOL_STATUS_FEAT_DISABLED: | 	case ZPOOL_STATUS_FEAT_DISABLED: | ||||||
| 		printf_color(ANSI_BOLD, gettext("status: ")); | 		printf_color(ANSI_BOLD, gettext("status: ")); | ||||||
| 		printf_color(ANSI_YELLOW, gettext("Some supported and " | 		printf_color(ANSI_YELLOW, gettext("Some supported " | ||||||
| 		    "requested features are not enabled on the pool.\n")); | 		    "features are not enabled on the pool.\n\t" | ||||||
|  | 		    "(Note that they may be intentionally disabled " | ||||||
|  | 		    "if the\n\t'compatibility' property is set.)\n")); | ||||||
| 		break; | 		break; | ||||||
| 
 | 
 | ||||||
| 	case ZPOOL_STATUS_COMPATIBILITY_ERR: | 	case ZPOOL_STATUS_COMPATIBILITY_ERR: | ||||||
| @ -2728,6 +2742,13 @@ show_import(nvlist_t *config, boolean_t report_error) | |||||||
| 		    "property.\n")); | 		    "property.\n")); | ||||||
| 		break; | 		break; | ||||||
| 
 | 
 | ||||||
|  | 	case ZPOOL_STATUS_INCOMPATIBLE_FEAT: | ||||||
|  | 		printf_color(ANSI_BOLD, gettext("status: ")); | ||||||
|  | 		printf_color(ANSI_YELLOW, gettext("One or more features " | ||||||
|  | 		    "are enabled on the pool despite not being\n" | ||||||
|  | 		    "requested by the 'compatibility' property.\n")); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
| 	case ZPOOL_STATUS_UNSUP_FEAT_READ: | 	case ZPOOL_STATUS_UNSUP_FEAT_READ: | ||||||
| 		printf_color(ANSI_BOLD, gettext("status: ")); | 		printf_color(ANSI_BOLD, gettext("status: ")); | ||||||
| 		printf_color(ANSI_YELLOW, gettext("The pool uses the following " | 		printf_color(ANSI_YELLOW, gettext("The pool uses the following " | ||||||
| @ -8055,7 +8076,8 @@ status_callback(zpool_handle_t *zhp, void *data) | |||||||
| 	    (reason == ZPOOL_STATUS_OK || | 	    (reason == ZPOOL_STATUS_OK || | ||||||
| 	    reason == ZPOOL_STATUS_VERSION_OLDER || | 	    reason == ZPOOL_STATUS_VERSION_OLDER || | ||||||
| 	    reason == ZPOOL_STATUS_FEAT_DISABLED || | 	    reason == ZPOOL_STATUS_FEAT_DISABLED || | ||||||
| 	    reason == ZPOOL_STATUS_COMPATIBILITY_ERR)) { | 	    reason == ZPOOL_STATUS_COMPATIBILITY_ERR || | ||||||
|  | 	    reason == ZPOOL_STATUS_INCOMPATIBLE_FEAT)) { | ||||||
| 		if (!cbp->cb_allpools) { | 		if (!cbp->cb_allpools) { | ||||||
| 			(void) printf(gettext("pool '%s' is healthy\n"), | 			(void) printf(gettext("pool '%s' is healthy\n"), | ||||||
| 			    zpool_get_name(zhp)); | 			    zpool_get_name(zhp)); | ||||||
| @ -8254,6 +8276,18 @@ status_callback(zpool_handle_t *zhp, void *data) | |||||||
| 		    ZPOOL_DATA_COMPAT_D ".\n")); | 		    ZPOOL_DATA_COMPAT_D ".\n")); | ||||||
| 		break; | 		break; | ||||||
| 
 | 
 | ||||||
|  | 	case ZPOOL_STATUS_INCOMPATIBLE_FEAT: | ||||||
|  | 		printf_color(ANSI_BOLD, gettext("status: ")); | ||||||
|  | 		printf_color(ANSI_YELLOW, gettext("One or more features " | ||||||
|  | 		    "are enabled on the pool despite not being\n\t" | ||||||
|  | 		    "requested by the 'compatibility' property.\n")); | ||||||
|  | 		printf_color(ANSI_BOLD, gettext("action: ")); | ||||||
|  | 		printf_color(ANSI_YELLOW, gettext("Consider setting " | ||||||
|  | 		    "'compatibility' to an appropriate value, or\n\t" | ||||||
|  | 		    "adding needed features to the relevant file in\n\t" | ||||||
|  | 		    ZPOOL_SYSCONF_COMPAT_D " or " ZPOOL_DATA_COMPAT_D ".\n")); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
| 	case ZPOOL_STATUS_UNSUP_FEAT_READ: | 	case ZPOOL_STATUS_UNSUP_FEAT_READ: | ||||||
| 		printf_color(ANSI_BOLD, gettext("status: ")); | 		printf_color(ANSI_BOLD, gettext("status: ")); | ||||||
| 		printf_color(ANSI_YELLOW, gettext("The pool cannot be accessed " | 		printf_color(ANSI_YELLOW, gettext("The pool cannot be accessed " | ||||||
| @ -8713,6 +8747,11 @@ upgrade_version(zpool_handle_t *zhp, uint64_t version) | |||||||
| 	verify(nvlist_lookup_uint64(config, ZPOOL_CONFIG_VERSION, | 	verify(nvlist_lookup_uint64(config, ZPOOL_CONFIG_VERSION, | ||||||
| 	    &oldversion) == 0); | 	    &oldversion) == 0); | ||||||
| 
 | 
 | ||||||
|  | 	char compat[ZFS_MAXPROPLEN]; | ||||||
|  | 	if (zpool_get_prop(zhp, ZPOOL_PROP_COMPATIBILITY, compat, | ||||||
|  | 	    ZFS_MAXPROPLEN, NULL, B_FALSE) != 0) | ||||||
|  | 		compat[0] = '\0'; | ||||||
|  | 
 | ||||||
| 	assert(SPA_VERSION_IS_SUPPORTED(oldversion)); | 	assert(SPA_VERSION_IS_SUPPORTED(oldversion)); | ||||||
| 	assert(oldversion < version); | 	assert(oldversion < version); | ||||||
| 
 | 
 | ||||||
| @ -8727,6 +8766,13 @@ upgrade_version(zpool_handle_t *zhp, uint64_t version) | |||||||
| 		return (1); | 		return (1); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if (strcmp(compat, ZPOOL_COMPAT_LEGACY) == 0) { | ||||||
|  | 		(void) fprintf(stderr, gettext("Upgrade not performed because " | ||||||
|  | 		    "'compatibility' property set to '" | ||||||
|  | 		    ZPOOL_COMPAT_LEGACY "'.\n")); | ||||||
|  | 		return (1); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	ret = zpool_upgrade(zhp, version); | 	ret = zpool_upgrade(zhp, version); | ||||||
| 	if (ret != 0) | 	if (ret != 0) | ||||||
| 		return (ret); | 		return (ret); | ||||||
| @ -8868,7 +8914,10 @@ upgrade_list_older_cb(zpool_handle_t *zhp, void *arg) | |||||||
| 			    "be upgraded to use feature flags.  After " | 			    "be upgraded to use feature flags.  After " | ||||||
| 			    "being upgraded, these pools\nwill no " | 			    "being upgraded, these pools\nwill no " | ||||||
| 			    "longer be accessible by software that does not " | 			    "longer be accessible by software that does not " | ||||||
| 			    "support feature\nflags.\n\n")); | 			    "support feature\nflags.\n\n" | ||||||
|  | 			    "Note that setting a pool's 'compatibility' " | ||||||
|  | 			    "feature to '" ZPOOL_COMPAT_LEGACY "' will\n" | ||||||
|  | 			    "inhibit upgrades.\n\n")); | ||||||
| 			(void) printf(gettext("VER  POOL\n")); | 			(void) printf(gettext("VER  POOL\n")); | ||||||
| 			(void) printf(gettext("---  ------------\n")); | 			(void) printf(gettext("---  ------------\n")); | ||||||
| 			cbp->cb_first = B_FALSE; | 			cbp->cb_first = B_FALSE; | ||||||
| @ -8914,7 +8963,11 @@ upgrade_list_disabled_cb(zpool_handle_t *zhp, void *arg) | |||||||
| 					    "software\nthat does not support " | 					    "software\nthat does not support " | ||||||
| 					    "the feature. See " | 					    "the feature. See " | ||||||
| 					    "zpool-features(5) for " | 					    "zpool-features(5) for " | ||||||
| 					    "details.\n\n")); | 					    "details.\n\n" | ||||||
|  | 					    "Note that the pool " | ||||||
|  | 					    "'compatibility' feature can be " | ||||||
|  | 					    "used to inhibit\nfeature " | ||||||
|  | 					    "upgrades.\n\n")); | ||||||
| 					(void) printf(gettext("POOL  " | 					(void) printf(gettext("POOL  " | ||||||
| 					    "FEATURE\n")); | 					    "FEATURE\n")); | ||||||
| 					(void) printf(gettext("------" | 					(void) printf(gettext("------" | ||||||
| @ -9970,6 +10023,63 @@ set_callback(zpool_handle_t *zhp, void *data) | |||||||
| 	int error; | 	int error; | ||||||
| 	set_cbdata_t *cb = (set_cbdata_t *)data; | 	set_cbdata_t *cb = (set_cbdata_t *)data; | ||||||
| 
 | 
 | ||||||
|  | 	/* Check if we have out-of-bounds features */ | ||||||
|  | 	if (strcmp(cb->cb_propname, ZPOOL_CONFIG_COMPATIBILITY) == 0) { | ||||||
|  | 		boolean_t features[SPA_FEATURES]; | ||||||
|  | 		if (zpool_do_load_compat(cb->cb_value, features) != | ||||||
|  | 		    ZPOOL_COMPATIBILITY_OK) | ||||||
|  | 			return (-1); | ||||||
|  | 
 | ||||||
|  | 		nvlist_t *enabled = zpool_get_features(zhp); | ||||||
|  | 		spa_feature_t i; | ||||||
|  | 		for (i = 0; i < SPA_FEATURES; i++) { | ||||||
|  | 			const char *fguid = spa_feature_table[i].fi_guid; | ||||||
|  | 			if (nvlist_exists(enabled, fguid) && !features[i]) | ||||||
|  | 				break; | ||||||
|  | 		} | ||||||
|  | 		if (i < SPA_FEATURES) | ||||||
|  | 			(void) fprintf(stderr, gettext("Warning: one or " | ||||||
|  | 			    "more features already enabled on pool '%s'\n" | ||||||
|  | 			    "are not present in this compatibility set.\n"), | ||||||
|  | 			    zpool_get_name(zhp)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* if we're setting a feature, check it's in compatibility set */ | ||||||
|  | 	if (zpool_prop_feature(cb->cb_propname) && | ||||||
|  | 	    strcmp(cb->cb_value, ZFS_FEATURE_ENABLED) == 0) { | ||||||
|  | 		char *fname = strchr(cb->cb_propname, '@') + 1; | ||||||
|  | 		spa_feature_t f; | ||||||
|  | 
 | ||||||
|  | 		if (zfeature_lookup_name(fname, &f) == 0) { | ||||||
|  | 			char compat[ZFS_MAXPROPLEN]; | ||||||
|  | 			if (zpool_get_prop(zhp, ZPOOL_PROP_COMPATIBILITY, | ||||||
|  | 			    compat, ZFS_MAXPROPLEN, NULL, B_FALSE) != 0) | ||||||
|  | 				compat[0] = '\0'; | ||||||
|  | 
 | ||||||
|  | 			boolean_t features[SPA_FEATURES]; | ||||||
|  | 			if (zpool_do_load_compat(compat, features) != | ||||||
|  | 			    ZPOOL_COMPATIBILITY_OK) { | ||||||
|  | 				(void) fprintf(stderr, gettext("Error: " | ||||||
|  | 				    "cannot enable feature '%s' on pool '%s'\n" | ||||||
|  | 				    "because the pool's 'compatibility' " | ||||||
|  | 				    "property cannot be parsed.\n"), | ||||||
|  | 				    fname, zpool_get_name(zhp)); | ||||||
|  | 				return (-1); | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			if (!features[f]) { | ||||||
|  | 				(void) fprintf(stderr, gettext("Error: " | ||||||
|  | 				    "cannot enable feature '%s' on pool '%s'\n" | ||||||
|  | 				    "as it is not specified in this pool's " | ||||||
|  | 				    "current compatibility set.\n" | ||||||
|  | 				    "Consider setting 'compatibility' to a " | ||||||
|  | 				    "less restrictive set, or to 'off'.\n"), | ||||||
|  | 				    fname, zpool_get_name(zhp)); | ||||||
|  | 				return (-1); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	error = zpool_set_prop(zhp, cb->cb_propname, cb->cb_value); | 	error = zpool_set_prop(zhp, cb->cb_propname, cb->cb_value); | ||||||
| 
 | 
 | ||||||
| 	if (!error) | 	if (!error) | ||||||
| @ -10492,28 +10602,25 @@ zpool_do_version(int argc, char **argv) | |||||||
| static zpool_compat_status_t | static zpool_compat_status_t | ||||||
| zpool_do_load_compat(const char *compat, boolean_t *list) | zpool_do_load_compat(const char *compat, boolean_t *list) | ||||||
| { | { | ||||||
| 	char badword[ZFS_MAXPROPLEN]; | 	char report[1024]; | ||||||
| 	char badfile[MAXPATHLEN]; | 
 | ||||||
| 	zpool_compat_status_t ret; | 	zpool_compat_status_t ret; | ||||||
| 
 | 
 | ||||||
| 	switch (ret = zpool_load_compat(compat, list, badword, badfile)) { | 	ret = zpool_load_compat(compat, list, report, 1024); | ||||||
|  | 	switch (ret) { | ||||||
|  | 
 | ||||||
| 	case ZPOOL_COMPATIBILITY_OK: | 	case ZPOOL_COMPATIBILITY_OK: | ||||||
| 		break; | 		break; | ||||||
| 	case ZPOOL_COMPATIBILITY_READERR: | 
 | ||||||
| 		(void) fprintf(stderr, gettext("error reading compatibility " |  | ||||||
| 		    "file '%s'\n"), badfile); |  | ||||||
| 		break; |  | ||||||
| 	case ZPOOL_COMPATIBILITY_BADFILE: |  | ||||||
| 		(void) fprintf(stderr, gettext("compatibility file '%s' " |  | ||||||
| 		    "too large or not newline-terminated\n"), badfile); |  | ||||||
| 		break; |  | ||||||
| 	case ZPOOL_COMPATIBILITY_BADWORD: |  | ||||||
| 		(void) fprintf(stderr, gettext("unknown feature '%s' in " |  | ||||||
| 		    "compatibility file '%s'\n"), badword, badfile); |  | ||||||
| 		break; |  | ||||||
| 	case ZPOOL_COMPATIBILITY_NOFILES: | 	case ZPOOL_COMPATIBILITY_NOFILES: | ||||||
| 		(void) fprintf(stderr, gettext("no compatibility files " | 	case ZPOOL_COMPATIBILITY_BADFILE: | ||||||
| 		    "specified\n")); | 	case ZPOOL_COMPATIBILITY_BADTOKEN: | ||||||
|  | 		(void) fprintf(stderr, "Error: %s\n", report); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case ZPOOL_COMPATIBILITY_WARNTOKEN: | ||||||
|  | 		(void) fprintf(stderr, "Warning: %s\n", report); | ||||||
|  | 		ret = ZPOOL_COMPATIBILITY_OK; | ||||||
| 		break; | 		break; | ||||||
| 	} | 	} | ||||||
| 	return (ret); | 	return (ret); | ||||||
|  | |||||||
| @ -393,6 +393,7 @@ typedef enum { | |||||||
| 	ZPOOL_STATUS_REBUILD_SCRUB,	/* recommend scrubbing the pool */ | 	ZPOOL_STATUS_REBUILD_SCRUB,	/* recommend scrubbing the pool */ | ||||||
| 	ZPOOL_STATUS_NON_NATIVE_ASHIFT,	/* (e.g. 512e dev with ashift of 9) */ | 	ZPOOL_STATUS_NON_NATIVE_ASHIFT,	/* (e.g. 512e dev with ashift of 9) */ | ||||||
| 	ZPOOL_STATUS_COMPATIBILITY_ERR,	/* bad 'compatibility' property */ | 	ZPOOL_STATUS_COMPATIBILITY_ERR,	/* bad 'compatibility' property */ | ||||||
|  | 	ZPOOL_STATUS_INCOMPATIBLE_FEAT,	/* feature set outside compatibility */ | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Finally, the following indicates a healthy pool. | 	 * Finally, the following indicates a healthy pool. | ||||||
| @ -922,14 +923,14 @@ extern int zpool_disable_datasets(zpool_handle_t *, boolean_t); | |||||||
|  */ |  */ | ||||||
| typedef enum { | typedef enum { | ||||||
| 	ZPOOL_COMPATIBILITY_OK, | 	ZPOOL_COMPATIBILITY_OK, | ||||||
| 	ZPOOL_COMPATIBILITY_READERR, | 	ZPOOL_COMPATIBILITY_WARNTOKEN, | ||||||
|  | 	ZPOOL_COMPATIBILITY_BADTOKEN, | ||||||
| 	ZPOOL_COMPATIBILITY_BADFILE, | 	ZPOOL_COMPATIBILITY_BADFILE, | ||||||
| 	ZPOOL_COMPATIBILITY_BADWORD, |  | ||||||
| 	ZPOOL_COMPATIBILITY_NOFILES | 	ZPOOL_COMPATIBILITY_NOFILES | ||||||
| } zpool_compat_status_t; | } zpool_compat_status_t; | ||||||
| 
 | 
 | ||||||
| extern zpool_compat_status_t zpool_load_compat(const char *, | extern zpool_compat_status_t zpool_load_compat(const char *, | ||||||
|     boolean_t *, char *, char *); |     boolean_t *, char *, size_t); | ||||||
| 
 | 
 | ||||||
| #ifdef __FreeBSD__ | #ifdef __FreeBSD__ | ||||||
| 
 | 
 | ||||||
|  | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -467,8 +467,7 @@ zpool_valid_proplist(libzfs_handle_t *hdl, const char *poolname, | |||||||
| 	char *slash, *check; | 	char *slash, *check; | ||||||
| 	struct stat64 statbuf; | 	struct stat64 statbuf; | ||||||
| 	zpool_handle_t *zhp; | 	zpool_handle_t *zhp; | ||||||
| 	char badword[ZFS_MAXPROPLEN]; | 	char report[1024]; | ||||||
| 	char badfile[MAXPATHLEN]; |  | ||||||
| 
 | 
 | ||||||
| 	if (nvlist_alloc(&retprops, NV_UNIQUE_NAME, 0) != 0) { | 	if (nvlist_alloc(&retprops, NV_UNIQUE_NAME, 0) != 0) { | ||||||
| 		(void) no_memory(hdl); | 		(void) no_memory(hdl); | ||||||
| @ -679,33 +678,14 @@ zpool_valid_proplist(libzfs_handle_t *hdl, const char *poolname, | |||||||
| 			break; | 			break; | ||||||
| 
 | 
 | ||||||
| 		case ZPOOL_PROP_COMPATIBILITY: | 		case ZPOOL_PROP_COMPATIBILITY: | ||||||
| 			switch (zpool_load_compat(strval, NULL, | 			switch (zpool_load_compat(strval, NULL, report, 1024)) { | ||||||
| 			    badword, badfile)) { |  | ||||||
| 			case ZPOOL_COMPATIBILITY_OK: | 			case ZPOOL_COMPATIBILITY_OK: | ||||||
|  | 			case ZPOOL_COMPATIBILITY_WARNTOKEN: | ||||||
| 				break; | 				break; | ||||||
| 			case ZPOOL_COMPATIBILITY_READERR: |  | ||||||
| 				zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, |  | ||||||
| 				    "error reading feature file '%s'"), |  | ||||||
| 				    badfile); |  | ||||||
| 				(void) zfs_error(hdl, EZFS_BADPROP, errbuf); |  | ||||||
| 				goto error; |  | ||||||
| 			case ZPOOL_COMPATIBILITY_BADFILE: | 			case ZPOOL_COMPATIBILITY_BADFILE: | ||||||
| 				zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, | 			case ZPOOL_COMPATIBILITY_BADTOKEN: | ||||||
| 				    "feature file '%s' too large or not " |  | ||||||
| 				    "newline-terminated"), |  | ||||||
| 				    badfile); |  | ||||||
| 				(void) zfs_error(hdl, EZFS_BADPROP, errbuf); |  | ||||||
| 				goto error; |  | ||||||
| 			case ZPOOL_COMPATIBILITY_BADWORD: |  | ||||||
| 				zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, |  | ||||||
| 				    "unknown feature '%s' in feature " |  | ||||||
| 				    "file '%s'"), |  | ||||||
| 				    badword, badfile); |  | ||||||
| 				(void) zfs_error(hdl, EZFS_BADPROP, errbuf); |  | ||||||
| 				goto error; |  | ||||||
| 			case ZPOOL_COMPATIBILITY_NOFILES: | 			case ZPOOL_COMPATIBILITY_NOFILES: | ||||||
| 				zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, | 				zfs_error_aux(hdl, report); | ||||||
| 				    "no feature files specified")); |  | ||||||
| 				(void) zfs_error(hdl, EZFS_BADPROP, errbuf); | 				(void) zfs_error(hdl, EZFS_BADPROP, errbuf); | ||||||
| 				goto error; | 				goto error; | ||||||
| 			} | 			} | ||||||
| @ -4742,8 +4722,8 @@ zpool_get_bootenv(zpool_handle_t *zhp, nvlist_t **nvlp) | |||||||
|  * Arguments: |  * Arguments: | ||||||
|  *  compatibility : string containing feature filenames |  *  compatibility : string containing feature filenames | ||||||
|  *  features : either NULL or pointer to array of boolean |  *  features : either NULL or pointer to array of boolean | ||||||
|  *  badtoken : either NULL or pointer to char[ZFS_MAXPROPLEN] |  *  report : either NULL or pointer to string buffer | ||||||
|  *  badfile : either NULL or pointer to char[MAXPATHLEN] |  *  rlen : length of "report" buffer | ||||||
|  * |  * | ||||||
|  * compatibility is NULL (unset), "", "off", "legacy", or list of |  * compatibility is NULL (unset), "", "off", "legacy", or list of | ||||||
|  * comma-separated filenames. filenames should either be absolute, |  * comma-separated filenames. filenames should either be absolute, | ||||||
| @ -4752,48 +4732,56 @@ zpool_get_bootenv(zpool_handle_t *zhp, nvlist_t **nvlp) | |||||||
|  *   2) ZPOOL_DATA_COMPAT_D (eg: /usr/share/zfs/compatibility.d). |  *   2) ZPOOL_DATA_COMPAT_D (eg: /usr/share/zfs/compatibility.d). | ||||||
|  * (Unset), "" or "off" => enable all features |  * (Unset), "" or "off" => enable all features | ||||||
|  * "legacy" => disable all features |  * "legacy" => disable all features | ||||||
|  |  * | ||||||
|  * Any feature names read from files which match unames in spa_feature_table |  * Any feature names read from files which match unames in spa_feature_table | ||||||
|  * will have the corresponding boolean set in the features array (if non-NULL). |  * will have the corresponding boolean set in the features array (if non-NULL). | ||||||
|  * If more than one feature set specified, only features present in *all* of |  * If more than one feature set specified, only features present in *all* of | ||||||
|  * them will be set. |  * them will be set. | ||||||
|  * |  * | ||||||
|  * An unreadable filename will be strlcpy'd to badfile (if non-NULL). |  * "report" if not NULL will be populated with a suitable status message. | ||||||
|  * An unrecognized feature will be strlcpy'd to badtoken (if non-NULL). |  | ||||||
|  * |  * | ||||||
|  * Return values: |  * Return values: | ||||||
|  *   ZPOOL_COMPATIBILITY_OK : files read and parsed ok |  *   ZPOOL_COMPATIBILITY_OK : files read and parsed ok | ||||||
|  *   ZPOOL_COMPATIBILITY_READERR : file could not be opened / mmap'd |  | ||||||
|  *   ZPOOL_COMPATIBILITY_BADFILE : file too big or not a text file |  *   ZPOOL_COMPATIBILITY_BADFILE : file too big or not a text file | ||||||
|  *   ZPOOL_COMPATIBILITY_BADWORD : file contains invalid feature name |  *   ZPOOL_COMPATIBILITY_BADTOKEN : SYSCONF file contains invalid feature name | ||||||
|  *   ZPOOL_COMPATIBILITY_NOFILES  : no file names found |  *   ZPOOL_COMPATIBILITY_WARNTOKEN : DATA file contains invalid feature name | ||||||
|  |  *   ZPOOL_COMPATIBILITY_NOFILES : no feature files found | ||||||
|  */ |  */ | ||||||
| zpool_compat_status_t | zpool_compat_status_t | ||||||
| zpool_load_compat(const char *compatibility, | zpool_load_compat(const char *compat, boolean_t *features, char *report, | ||||||
|     boolean_t *features, char *badtoken, char *badfile) |     size_t rlen) | ||||||
| { | { | ||||||
| 	int sdirfd, ddirfd, featfd; | 	int sdirfd, ddirfd, featfd; | ||||||
| 	int i; |  | ||||||
| 	struct stat fs; | 	struct stat fs; | ||||||
| 	char *fc;			/* mmap of file */ | 	char *fc; | ||||||
| 	char *ps, *ls, *ws;		/* strtok state */ | 	char *ps, *ls, *ws; | ||||||
| 	char *file, *line, *word; | 	char *file, *line, *word; | ||||||
| 	char filenames[ZFS_MAXPROPLEN]; | 
 | ||||||
| 	int filecount = 0; | 	char l_compat[ZFS_MAXPROPLEN]; | ||||||
|  | 
 | ||||||
|  | 	boolean_t ret_nofiles = B_TRUE; | ||||||
|  | 	boolean_t ret_badfile = B_FALSE; | ||||||
|  | 	boolean_t ret_badtoken = B_FALSE; | ||||||
|  | 	boolean_t ret_warntoken = B_FALSE; | ||||||
| 
 | 
 | ||||||
| 	/* special cases (unset), "" and "off" => enable all features */ | 	/* special cases (unset), "" and "off" => enable all features */ | ||||||
| 	if (compatibility == NULL || compatibility[0] == '\0' || | 	if (compat == NULL || compat[0] == '\0' || | ||||||
| 	    strcmp(compatibility, ZPOOL_COMPAT_OFF) == 0) { | 	    strcmp(compat, ZPOOL_COMPAT_OFF) == 0) { | ||||||
| 		if (features != NULL) | 		if (features != NULL) | ||||||
| 			for (i = 0; i < SPA_FEATURES; i++) | 			for (uint_t i = 0; i < SPA_FEATURES; i++) | ||||||
| 				features[i] = B_TRUE; | 				features[i] = B_TRUE; | ||||||
|  | 		if (report != NULL) | ||||||
|  | 			strlcpy(report, gettext("all features enabled"), rlen); | ||||||
| 		return (ZPOOL_COMPATIBILITY_OK); | 		return (ZPOOL_COMPATIBILITY_OK); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/* Final special case "legacy" => disable all features */ | 	/* Final special case "legacy" => disable all features */ | ||||||
| 	if (strcmp(compatibility, ZPOOL_COMPAT_LEGACY) == 0) { | 	if (strcmp(compat, ZPOOL_COMPAT_LEGACY) == 0) { | ||||||
| 		if (features != NULL) | 		if (features != NULL) | ||||||
| 			for (i = 0; i < SPA_FEATURES; i++) | 			for (uint_t i = 0; i < SPA_FEATURES; i++) | ||||||
| 				features[i] = B_FALSE; | 				features[i] = B_FALSE; | ||||||
|  | 		if (report != NULL) | ||||||
|  | 			strlcpy(report, gettext("all features disabled"), rlen); | ||||||
| 		return (ZPOOL_COMPATIBILITY_OK); | 		return (ZPOOL_COMPATIBILITY_OK); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -4801,9 +4789,12 @@ zpool_load_compat(const char *compatibility, | |||||||
| 	 * Start with all true; will be ANDed with results from each file | 	 * Start with all true; will be ANDed with results from each file | ||||||
| 	 */ | 	 */ | ||||||
| 	if (features != NULL) | 	if (features != NULL) | ||||||
| 		for (i = 0; i < SPA_FEATURES; i++) | 		for (uint_t i = 0; i < SPA_FEATURES; i++) | ||||||
| 			features[i] = B_TRUE; | 			features[i] = B_TRUE; | ||||||
| 
 | 
 | ||||||
|  | 	char err_badfile[1024] = ""; | ||||||
|  | 	char err_badtoken[1024] = ""; | ||||||
|  | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * We ignore errors from the directory open() | 	 * We ignore errors from the directory open() | ||||||
| 	 * as they're only needed if the filename is relative | 	 * as they're only needed if the filename is relative | ||||||
| @ -4815,32 +4806,33 @@ zpool_load_compat(const char *compatibility, | |||||||
| 	sdirfd = open(ZPOOL_SYSCONF_COMPAT_D, O_DIRECTORY | O_PATH | O_CLOEXEC); | 	sdirfd = open(ZPOOL_SYSCONF_COMPAT_D, O_DIRECTORY | O_PATH | O_CLOEXEC); | ||||||
| 	ddirfd = open(ZPOOL_DATA_COMPAT_D, O_DIRECTORY | O_PATH | O_CLOEXEC); | 	ddirfd = open(ZPOOL_DATA_COMPAT_D, O_DIRECTORY | O_PATH | O_CLOEXEC); | ||||||
| 
 | 
 | ||||||
| 	(void) strlcpy(filenames, compatibility, ZFS_MAXPROPLEN); | 	(void) strlcpy(l_compat, compat, ZFS_MAXPROPLEN); | ||||||
| 	file = strtok_r(filenames, ",", &ps); | 
 | ||||||
| 	while (file != NULL) { | 	for (file = strtok_r(l_compat, ",", &ps); | ||||||
| 		boolean_t features_local[SPA_FEATURES]; | 	    file != NULL; | ||||||
|  | 	    file = strtok_r(NULL, ",", &ps)) { | ||||||
|  | 
 | ||||||
|  | 		boolean_t l_features[SPA_FEATURES]; | ||||||
|  | 
 | ||||||
|  | 		enum { Z_SYSCONF, Z_DATA } source; | ||||||
| 
 | 
 | ||||||
| 		/* try sysconfdir first, then datadir */ | 		/* try sysconfdir first, then datadir */ | ||||||
| 		if ((featfd = openat(sdirfd, file, 0, O_RDONLY)) < 0) | 		source = Z_SYSCONF; | ||||||
|  | 		if ((featfd = openat(sdirfd, file, 0, O_RDONLY)) < 0) { | ||||||
| 			featfd = openat(ddirfd, file, 0, O_RDONLY); | 			featfd = openat(ddirfd, file, 0, O_RDONLY); | ||||||
| 
 | 			source = Z_DATA; | ||||||
| 		if (featfd < 0 || fstat(featfd, &fs) < 0) { |  | ||||||
| 			(void) close(featfd); |  | ||||||
| 			(void) close(sdirfd); |  | ||||||
| 			(void) close(ddirfd); |  | ||||||
| 			if (badfile != NULL) |  | ||||||
| 				(void) strlcpy(badfile, file, MAXPATHLEN); |  | ||||||
| 			return (ZPOOL_COMPATIBILITY_READERR); |  | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/* Too big or too small */ | 		/* File readable and correct size? */ | ||||||
| 		if (fs.st_size < 1 || fs.st_size > ZPOOL_COMPAT_MAXSIZE) { | 		if (featfd < 0 || | ||||||
|  | 		    fstat(featfd, &fs) < 0 || | ||||||
|  | 		    fs.st_size < 1 || | ||||||
|  | 		    fs.st_size > ZPOOL_COMPAT_MAXSIZE) { | ||||||
| 			(void) close(featfd); | 			(void) close(featfd); | ||||||
| 			(void) close(sdirfd); | 			strlcat(err_badfile, file, ZFS_MAXPROPLEN); | ||||||
| 			(void) close(ddirfd); | 			strlcat(err_badfile, " ", ZFS_MAXPROPLEN); | ||||||
| 			if (badfile != NULL) | 			ret_badfile = B_TRUE; | ||||||
| 				(void) strlcpy(badfile, file, MAXPATHLEN); | 			continue; | ||||||
| 			return (ZPOOL_COMPATIBILITY_BADFILE); |  | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/* private mmap() so we can strtok safely */ | 		/* private mmap() so we can strtok safely */ | ||||||
| @ -4848,73 +4840,99 @@ zpool_load_compat(const char *compatibility, | |||||||
| 		    PROT_READ|PROT_WRITE, MAP_PRIVATE, featfd, 0); | 		    PROT_READ|PROT_WRITE, MAP_PRIVATE, featfd, 0); | ||||||
| 		(void) close(featfd); | 		(void) close(featfd); | ||||||
| 
 | 
 | ||||||
| 		if (fc < 0) { | 		/* map ok, and last character == newline? */ | ||||||
| 			(void) close(sdirfd); | 		if (fc < 0 || fc[fs.st_size - 1] != '\n') { | ||||||
| 			(void) close(ddirfd); |  | ||||||
| 			if (badfile != NULL) |  | ||||||
| 				(void) strlcpy(badfile, file, MAXPATHLEN); |  | ||||||
| 			return (ZPOOL_COMPATIBILITY_READERR); |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		/* Text file sanity check - last char should be newline */ |  | ||||||
| 		if (fc[fs.st_size - 1] != '\n') { |  | ||||||
| 			(void) munmap((void *) fc, fs.st_size); | 			(void) munmap((void *) fc, fs.st_size); | ||||||
| 			(void) close(sdirfd); | 			strlcat(err_badfile, file, ZFS_MAXPROPLEN); | ||||||
| 			(void) close(ddirfd); | 			strlcat(err_badfile, " ", ZFS_MAXPROPLEN); | ||||||
| 			if (badfile != NULL) | 			ret_badfile = B_TRUE; | ||||||
| 				(void) strlcpy(badfile, file, MAXPATHLEN); | 			continue; | ||||||
| 			return (ZPOOL_COMPATIBILITY_BADFILE); |  | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/* replace with NUL to ensure we have a delimiter */ | 		ret_nofiles = B_FALSE; | ||||||
|  | 
 | ||||||
|  | 		for (uint_t i = 0; i < SPA_FEATURES; i++) | ||||||
|  | 			l_features[i] = B_FALSE; | ||||||
|  | 
 | ||||||
|  | 		/* replace last char with NUL to ensure we have a delimiter */ | ||||||
| 		fc[fs.st_size - 1] = '\0'; | 		fc[fs.st_size - 1] = '\0'; | ||||||
| 
 | 
 | ||||||
| 		for (i = 0; i < SPA_FEATURES; i++) | 		for (line = strtok_r(fc, "\n", &ls); | ||||||
| 			features_local[i] = B_FALSE; | 		    line != NULL; | ||||||
| 
 | 		    line = strtok_r(NULL, "\n", &ls)) { | ||||||
| 		line = strtok_r(fc, "\n", &ls); |  | ||||||
| 		while (line != NULL) { |  | ||||||
| 			/* discard comments */ | 			/* discard comments */ | ||||||
| 			*(strchrnul(line, '#')) = '\0'; | 			*(strchrnul(line, '#')) = '\0'; | ||||||
| 
 | 
 | ||||||
| 			word = strtok_r(line, ", \t", &ws); | 			for (word = strtok_r(line, ", \t", &ws); | ||||||
| 			while (word != NULL) { | 			    word != NULL; | ||||||
|  | 			    word = strtok_r(NULL, ", \t", &ws)) { | ||||||
| 				/* Find matching feature name */ | 				/* Find matching feature name */ | ||||||
| 				for (i = 0; i < SPA_FEATURES; i++) { | 				uint_t f; | ||||||
|  | 				for (f = 0; f < SPA_FEATURES; f++) { | ||||||
| 					zfeature_info_t *fi = | 					zfeature_info_t *fi = | ||||||
| 					    &spa_feature_table[i]; | 					    &spa_feature_table[f]; | ||||||
| 					if (strcmp(word, fi->fi_uname) == 0) { | 					if (strcmp(word, fi->fi_uname) == 0) { | ||||||
| 						features_local[i] = B_TRUE; | 						l_features[f] = B_TRUE; | ||||||
| 						break; | 						break; | ||||||
| 					} | 					} | ||||||
| 				} | 				} | ||||||
| 				if (i == SPA_FEATURES) { | 				if (f < SPA_FEATURES) | ||||||
| 					if (badtoken != NULL) | 					continue; | ||||||
| 						(void) strlcpy(badtoken, word, | 
 | ||||||
| 						    ZFS_MAXPROPLEN); | 				/* found an unrecognized word */ | ||||||
| 					if (badfile != NULL) | 				/* lightly sanitize it */ | ||||||
| 						(void) strlcpy(badfile, file, | 				if (strlen(word) > 32) | ||||||
| 						    MAXPATHLEN); | 					word[32] = '\0'; | ||||||
| 					(void) munmap((void *) fc, fs.st_size); | 				for (char *c = word; *c != '\0'; c++) | ||||||
| 					(void) close(sdirfd); | 					if (!isprint(*c)) | ||||||
| 					(void) close(ddirfd); | 						*c = '?'; | ||||||
| 					return (ZPOOL_COMPATIBILITY_BADWORD); | 
 | ||||||
|  | 				strlcat(err_badtoken, word, ZFS_MAXPROPLEN); | ||||||
|  | 				strlcat(err_badtoken, " ", ZFS_MAXPROPLEN); | ||||||
|  | 				if (source == Z_SYSCONF) | ||||||
|  | 					ret_badtoken = B_TRUE; | ||||||
|  | 				else | ||||||
|  | 					ret_warntoken = B_TRUE; | ||||||
| 			} | 			} | ||||||
| 				word = strtok_r(NULL, ", \t", &ws); |  | ||||||
| 			} |  | ||||||
| 			line = strtok_r(NULL, "\n", &ls); |  | ||||||
| 		} | 		} | ||||||
| 		(void) munmap((void *) fc, fs.st_size); | 		(void) munmap((void *) fc, fs.st_size); | ||||||
| 		if (features != NULL) { | 
 | ||||||
| 			for (i = 0; i < SPA_FEATURES; i++) | 		if (features != NULL) | ||||||
| 				features[i] &= features_local[i]; | 			for (uint_t i = 0; i < SPA_FEATURES; i++) | ||||||
| 		} | 				features[i] &= l_features[i]; | ||||||
| 		filecount++; |  | ||||||
| 		file = strtok_r(NULL, ",", &ps); |  | ||||||
| 	} | 	} | ||||||
| 	(void) close(sdirfd); | 	(void) close(sdirfd); | ||||||
| 	(void) close(ddirfd); | 	(void) close(ddirfd); | ||||||
| 	if (filecount == 0) | 
 | ||||||
|  | 	/* Return the most serious error */ | ||||||
|  | 	if (ret_badfile) { | ||||||
|  | 		if (report != NULL) | ||||||
|  | 			snprintf(report, rlen, gettext("could not read/" | ||||||
|  | 			    "parse feature file(s): %s"), err_badfile); | ||||||
|  | 		return (ZPOOL_COMPATIBILITY_BADFILE); | ||||||
|  | 	} | ||||||
|  | 	if (ret_nofiles) { | ||||||
|  | 		if (report != NULL) | ||||||
|  | 			strlcpy(report, | ||||||
|  | 			    gettext("no valid compatibility files specified"), | ||||||
|  | 			    rlen); | ||||||
| 		return (ZPOOL_COMPATIBILITY_NOFILES); | 		return (ZPOOL_COMPATIBILITY_NOFILES); | ||||||
|  | 	} | ||||||
|  | 	if (ret_badtoken) { | ||||||
|  | 		if (report != NULL) | ||||||
|  | 			snprintf(report, rlen, gettext("invalid feature " | ||||||
|  | 			    "name(s) in local compatibility files: %s"), | ||||||
|  | 			    err_badtoken); | ||||||
|  | 		return (ZPOOL_COMPATIBILITY_BADTOKEN); | ||||||
|  | 	} | ||||||
|  | 	if (ret_warntoken) { | ||||||
|  | 		if (report != NULL) | ||||||
|  | 			snprintf(report, rlen, gettext("unrecognized feature " | ||||||
|  | 			    "name(s) in distribution compatibility files: %s"), | ||||||
|  | 			    err_badtoken); | ||||||
|  | 		return (ZPOOL_COMPATIBILITY_WARNTOKEN); | ||||||
|  | 	} | ||||||
|  | 	if (report != NULL) | ||||||
|  | 		strlcpy(report, gettext("compatibility set ok"), rlen); | ||||||
| 	return (ZPOOL_COMPATIBILITY_OK); | 	return (ZPOOL_COMPATIBILITY_OK); | ||||||
| } | } | ||||||
|  | |||||||
| @ -89,6 +89,7 @@ static char *zfs_msgid_table[] = { | |||||||
| 	 *	ZPOOL_STATUS_REBUILDING | 	 *	ZPOOL_STATUS_REBUILDING | ||||||
| 	 *	ZPOOL_STATUS_REBUILD_SCRUB | 	 *	ZPOOL_STATUS_REBUILD_SCRUB | ||||||
| 	 *	ZPOOL_STATUS_COMPATIBILITY_ERR | 	 *	ZPOOL_STATUS_COMPATIBILITY_ERR | ||||||
|  | 	 *	ZPOOL_STATUS_INCOMPATIBLE_FEAT | ||||||
| 	 *	ZPOOL_STATUS_OK | 	 *	ZPOOL_STATUS_OK | ||||||
| 	 */ | 	 */ | ||||||
| }; | }; | ||||||
| @ -453,11 +454,17 @@ check_status(nvlist_t *config, boolean_t isimport, | |||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Outdated, but usable, version | 	 * Outdated, but usable, version | ||||||
| 	 */ | 	 */ | ||||||
| 	if (SPA_VERSION_IS_SUPPORTED(version) && version != SPA_VERSION) | 	if (SPA_VERSION_IS_SUPPORTED(version) && version != SPA_VERSION) { | ||||||
|  | 		/* "legacy" compatibility disables old version reporting */ | ||||||
|  | 		if (compat != NULL && strcmp(compat, ZPOOL_COMPAT_LEGACY) == 0) | ||||||
|  | 			return (ZPOOL_STATUS_OK); | ||||||
|  | 		else | ||||||
| 			return (ZPOOL_STATUS_VERSION_OLDER); | 			return (ZPOOL_STATUS_VERSION_OLDER); | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Usable pool with disabled features | 	 * Usable pool with disabled or superfluous features | ||||||
|  | 	 * (superfluous = beyond what's requested by 'compatibility') | ||||||
| 	 */ | 	 */ | ||||||
| 	if (version >= SPA_VERSION_FEATURES) { | 	if (version >= SPA_VERSION_FEATURES) { | ||||||
| 		int i; | 		int i; | ||||||
| @ -475,18 +482,23 @@ check_status(nvlist_t *config, boolean_t isimport, | |||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/* check against all features, or limited set? */ | 		/* check against all features, or limited set? */ | ||||||
| 		boolean_t pool_features[SPA_FEATURES]; | 		boolean_t c_features[SPA_FEATURES]; | ||||||
| 
 | 
 | ||||||
| 		if (zpool_load_compat(compat, pool_features, NULL, NULL) != | 		switch (zpool_load_compat(compat, c_features, NULL, 0)) { | ||||||
| 		    ZPOOL_COMPATIBILITY_OK) | 		case ZPOOL_COMPATIBILITY_OK: | ||||||
|  | 		case ZPOOL_COMPATIBILITY_WARNTOKEN: | ||||||
|  | 			break; | ||||||
|  | 		default: | ||||||
| 			return (ZPOOL_STATUS_COMPATIBILITY_ERR); | 			return (ZPOOL_STATUS_COMPATIBILITY_ERR); | ||||||
|  | 		} | ||||||
| 		for (i = 0; i < SPA_FEATURES; i++) { | 		for (i = 0; i < SPA_FEATURES; i++) { | ||||||
| 			zfeature_info_t *fi = &spa_feature_table[i]; | 			zfeature_info_t *fi = &spa_feature_table[i]; | ||||||
| 			if (!fi->fi_zfs_mod_supported) | 			if (!fi->fi_zfs_mod_supported) | ||||||
| 				continue; | 				continue; | ||||||
| 			if (pool_features[i] && | 			if (c_features[i] && !nvlist_exists(feat, fi->fi_guid)) | ||||||
| 			    !nvlist_exists(feat, fi->fi_guid)) |  | ||||||
| 				return (ZPOOL_STATUS_FEAT_DISABLED); | 				return (ZPOOL_STATUS_FEAT_DISABLED); | ||||||
|  | 			if (!c_features[i] && nvlist_exists(feat, fi->fi_guid)) | ||||||
|  | 				return (ZPOOL_STATUS_INCOMPATIBLE_FEAT); | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -166,6 +166,12 @@ enabled when using \fBzpool upgrade\fR. \fBzpool status\fR | |||||||
| will not show a warning about disabled features which are not part | will not show a warning about disabled features which are not part | ||||||
| of the requested feature set. | of the requested feature set. | ||||||
| .LP | .LP | ||||||
|  | The special value \fBlegacy\fR prevents any features from being enabled, | ||||||
|  | either via \fBzpool upgrade\fR or via \fBzpool set feature@XX=enabled\fR. | ||||||
|  | This setting also prevents pools from being upgraded to newer on-disk | ||||||
|  | versions. This is a safety measure to prevent new features from being | ||||||
|  | accidentally enabled, breaking compatibility. | ||||||
|  | .LP | ||||||
| By convention, compatibility files in \fB/usr/share/zfs/compatibility.d\fR | By convention, compatibility files in \fB/usr/share/zfs/compatibility.d\fR | ||||||
| are provided by the distribution package, and include feature sets | are provided by the distribution package, and include feature sets | ||||||
| supported by important versions of popular distributions, and feature | supported by important versions of popular distributions, and feature | ||||||
| @ -173,6 +179,15 @@ sets commonly supported at the start of each year.  Compatibility files | |||||||
| in \fB/etc/zfs/compatibility.d\fR, if present, will take precedence over | in \fB/etc/zfs/compatibility.d\fR, if present, will take precedence over | ||||||
| files with the same name in \fB/usr/share/zfs/compatibility.d\fR. | files with the same name in \fB/usr/share/zfs/compatibility.d\fR. | ||||||
| .LP | .LP | ||||||
|  | If an unrecognized feature is found in these files, an error message will | ||||||
|  | be shown. If the unrecognized feature is in a file in | ||||||
|  | \fB/etc/zfs/compatibility.d\fR, this is treated as an error and processing | ||||||
|  | will stop. If the unrecognized feature is under | ||||||
|  | \fB/usr/share/zfs/compatibility.d\fR, this is treated as a warning and | ||||||
|  | processing will continue. This difference is to allow distributions to | ||||||
|  | include features which might not be recognized by the currently-installed | ||||||
|  | binaries. | ||||||
|  | .LP | ||||||
| Compatibility files may include comments; any text from \fB#\fR to the end | Compatibility files may include comments; any text from \fB#\fR to the end | ||||||
| of the line is ignored. | of the line is ignored. | ||||||
| .LP | .LP | ||||||
|  | |||||||
| @ -55,11 +55,9 @@ formatted using a legacy ZFS version number. | |||||||
| These pools can continue to be used, but some features may not be available. | These pools can continue to be used, but some features may not be available. | ||||||
| Use | Use | ||||||
| .Nm zpool Cm upgrade Fl a | .Nm zpool Cm upgrade Fl a | ||||||
| to enable all features on all pools. (If a pool has specified compatibility | to enable all features on all pools (subject to the | ||||||
| feature sets using the |  | ||||||
| .Fl o Ar compatibility | .Fl o Ar compatibility | ||||||
| property, only the features present in all requested compatibility sets will | property). | ||||||
| be enabled on that pool.) |  | ||||||
| .It Xo | .It Xo | ||||||
| .Nm zpool | .Nm zpool | ||||||
| .Cm upgrade | .Cm upgrade | ||||||
| @ -75,11 +73,15 @@ for a description of feature flags features supported by the current software. | |||||||
| .Op Fl V Ar version | .Op Fl V Ar version | ||||||
| .Fl a Ns | Ns Ar pool Ns ... | .Fl a Ns | Ns Ar pool Ns ... | ||||||
| .Xc | .Xc | ||||||
| Enables all supported features on the given pool. (If the pool has specified | Enables all supported features on the given pool. | ||||||
| compatibility feature sets using the | .Pp | ||||||
|  | If the pool has specified compatibility feature sets using the | ||||||
| .Fl o Ar compatibility | .Fl o Ar compatibility | ||||||
| property, only the features present in all requested compatibility sets will be | property, only the features present in all requested compatibility sets will be | ||||||
| enabled.) | enabled. If this property is set to  | ||||||
|  | .Ar legacy | ||||||
|  | then no upgrade will take place. | ||||||
|  | .Pp | ||||||
| Once this is done, the pool will no longer be accessible on systems that do not | Once this is done, the pool will no longer be accessible on systems that do not | ||||||
| support feature flags. | support feature flags. | ||||||
| See | See | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Colm
						Colm