mirror of
				https://git.proxmox.com/git/mirror_zfs.git
				synced 2025-10-26 01:45:00 +03:00 
			
		
		
		
	Vdev Properties Feature
Add properties, similar to pool properties, to each vdev. This makes use of the existing per-vdev ZAP that was added as part of device evacuation/removal. A large number of read-only properties are exposed, many of the members of struct vdev_t, that provide useful statistics. Adds support for read-only "removing" vdev property. Adds the "allocating" property that defaults to "on" and can be set to "off" to prevent future allocations from that top-level vdev. Supports user-defined vdev properties. Includes support for properties.vdev in SYSFS. Co-authored-by: Allan Jude <allan@klarasystems.com> Co-authored-by: Mark Maybee <mark.maybee@delphix.com> Reviewed-by: Matthew Ahrens <mahrens@delphix.com> Reviewed-by: Mark Maybee <mark.maybee@delphix.com> Signed-off-by: Allan Jude <allan@klarasystems.com> Closes #11711
This commit is contained in:
		
							parent
							
								
									5f64bf7fde
								
							
						
					
					
						commit
						2a673e76a9
					
				| @ -60,6 +60,7 @@ struct zpool_list { | |||||||
| 	uu_avl_t	*zl_avl; | 	uu_avl_t	*zl_avl; | ||||||
| 	uu_avl_pool_t	*zl_pool; | 	uu_avl_pool_t	*zl_pool; | ||||||
| 	zprop_list_t	**zl_proplist; | 	zprop_list_t	**zl_proplist; | ||||||
|  | 	zfs_type_t	zl_type; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| /* ARGSUSED */ | /* ARGSUSED */ | ||||||
| @ -90,8 +91,7 @@ add_pool(zpool_handle_t *zhp, void *data) | |||||||
| 	if (uu_avl_find(zlp->zl_avl, node, NULL, &idx) == NULL) { | 	if (uu_avl_find(zlp->zl_avl, node, NULL, &idx) == NULL) { | ||||||
| 		if (zlp->zl_proplist && | 		if (zlp->zl_proplist && | ||||||
| 		    zpool_expand_proplist(zhp, zlp->zl_proplist, | 		    zpool_expand_proplist(zhp, zlp->zl_proplist, | ||||||
| 		    zlp->zl_literal) | 		    zlp->zl_type, zlp->zl_literal) != 0) { | ||||||
| 		    != 0) { |  | ||||||
| 			zpool_close(zhp); | 			zpool_close(zhp); | ||||||
| 			free(node); | 			free(node); | ||||||
| 			return (-1); | 			return (-1); | ||||||
| @ -113,7 +113,7 @@ add_pool(zpool_handle_t *zhp, void *data) | |||||||
|  * line. |  * line. | ||||||
|  */ |  */ | ||||||
| zpool_list_t * | zpool_list_t * | ||||||
| pool_list_get(int argc, char **argv, zprop_list_t **proplist, | pool_list_get(int argc, char **argv, zprop_list_t **proplist, zfs_type_t type, | ||||||
|     boolean_t literal, int *err) |     boolean_t literal, int *err) | ||||||
| { | { | ||||||
| 	zpool_list_t *zlp; | 	zpool_list_t *zlp; | ||||||
| @ -131,6 +131,7 @@ pool_list_get(int argc, char **argv, zprop_list_t **proplist, | |||||||
| 		zpool_no_memory(); | 		zpool_no_memory(); | ||||||
| 
 | 
 | ||||||
| 	zlp->zl_proplist = proplist; | 	zlp->zl_proplist = proplist; | ||||||
|  | 	zlp->zl_type = type; | ||||||
| 
 | 
 | ||||||
| 	zlp->zl_literal = literal; | 	zlp->zl_literal = literal; | ||||||
| 
 | 
 | ||||||
| @ -248,12 +249,14 @@ pool_list_count(zpool_list_t *zlp) | |||||||
|  */ |  */ | ||||||
| int | int | ||||||
| for_each_pool(int argc, char **argv, boolean_t unavail, | for_each_pool(int argc, char **argv, boolean_t unavail, | ||||||
|     zprop_list_t **proplist, boolean_t literal, zpool_iter_f func, void *data) |     zprop_list_t **proplist, zfs_type_t type, boolean_t literal, | ||||||
|  |     zpool_iter_f func, void *data) | ||||||
| { | { | ||||||
| 	zpool_list_t *list; | 	zpool_list_t *list; | ||||||
| 	int ret = 0; | 	int ret = 0; | ||||||
| 
 | 
 | ||||||
| 	if ((list = pool_list_get(argc, argv, proplist, literal, &ret)) == NULL) | 	if ((list = pool_list_get(argc, argv, proplist, type, literal, | ||||||
|  | 	    &ret)) == NULL) | ||||||
| 		return (1); | 		return (1); | ||||||
| 
 | 
 | ||||||
| 	if (pool_list_iter(list, unavail, func, data) != 0) | 	if (pool_list_iter(list, unavail, func, data) != 0) | ||||||
| @ -678,8 +681,8 @@ all_pools_for_each_vdev_run(int argc, char **argv, char *cmd, | |||||||
| 	vcdl->g_zfs = g_zfs; | 	vcdl->g_zfs = g_zfs; | ||||||
| 
 | 
 | ||||||
| 	/* Gather our list of all vdevs in all pools */ | 	/* Gather our list of all vdevs in all pools */ | ||||||
| 	for_each_pool(argc, argv, B_TRUE, NULL, B_FALSE, | 	for_each_pool(argc, argv, B_TRUE, NULL, ZFS_TYPE_POOL, | ||||||
| 	    all_pools_for_each_vdev_gather_cb, vcdl); | 	    B_FALSE, all_pools_for_each_vdev_gather_cb, vcdl); | ||||||
| 
 | 
 | ||||||
| 	/* Run command on all vdevs in all pools */ | 	/* Run command on all vdevs in all pools */ | ||||||
| 	all_pools_for_each_vdev_run_vcdl(vcdl); | 	all_pools_for_each_vdev_run_vcdl(vcdl); | ||||||
|  | |||||||
| @ -32,6 +32,7 @@ | |||||||
|  * Copyright (c) 2017, Intel Corporation. |  * Copyright (c) 2017, Intel Corporation. | ||||||
|  * Copyright (c) 2019, loli10K <ezomori.nozomu@gmail.com> |  * Copyright (c) 2019, loli10K <ezomori.nozomu@gmail.com> | ||||||
|  * Copyright (c) 2021, Colm Buckley <colm@tuatha.org> |  * Copyright (c) 2021, Colm Buckley <colm@tuatha.org> | ||||||
|  |  * Copyright (c) 2021, Klara Inc. | ||||||
|  * Copyright [2021] Hewlett Packard Enterprise Development LP |  * Copyright [2021] Hewlett Packard Enterprise Development LP | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| @ -335,6 +336,7 @@ static zpool_command_t command_table[] = { | |||||||
| #define	VDEV_ALLOC_CLASS_LOGS	"logs" | #define	VDEV_ALLOC_CLASS_LOGS	"logs" | ||||||
| 
 | 
 | ||||||
| static zpool_command_t *current_command; | static zpool_command_t *current_command; | ||||||
|  | static zfs_type_t current_prop_type = (ZFS_TYPE_POOL | ZFS_TYPE_VDEV); | ||||||
| static char history_str[HIS_MAX_RECORD_LEN]; | static char history_str[HIS_MAX_RECORD_LEN]; | ||||||
| static boolean_t log_history = B_TRUE; | static boolean_t log_history = B_TRUE; | ||||||
| static uint_t timestamp_fmt = NODATE; | static uint_t timestamp_fmt = NODATE; | ||||||
| @ -470,7 +472,7 @@ zpool_collect_leaves(zpool_handle_t *zhp, nvlist_t *nvroot, nvlist_t *res) | |||||||
|  * Callback routine that will print out a pool property value. |  * Callback routine that will print out a pool property value. | ||||||
|  */ |  */ | ||||||
| static int | static int | ||||||
| print_prop_cb(int prop, void *cb) | print_pool_prop_cb(int prop, void *cb) | ||||||
| { | { | ||||||
| 	FILE *fp = cb; | 	FILE *fp = cb; | ||||||
| 
 | 
 | ||||||
| @ -489,6 +491,29 @@ print_prop_cb(int prop, void *cb) | |||||||
| 	return (ZPROP_CONT); | 	return (ZPROP_CONT); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * Callback routine that will print out a vdev property value. | ||||||
|  |  */ | ||||||
|  | static int | ||||||
|  | print_vdev_prop_cb(int prop, void *cb) | ||||||
|  | { | ||||||
|  | 	FILE *fp = cb; | ||||||
|  | 
 | ||||||
|  | 	(void) fprintf(fp, "\t%-19s  ", vdev_prop_to_name(prop)); | ||||||
|  | 
 | ||||||
|  | 	if (vdev_prop_readonly(prop)) | ||||||
|  | 		(void) fprintf(fp, "  NO   "); | ||||||
|  | 	else | ||||||
|  | 		(void) fprintf(fp, " YES   "); | ||||||
|  | 
 | ||||||
|  | 	if (vdev_prop_values(prop) == NULL) | ||||||
|  | 		(void) fprintf(fp, "-\n"); | ||||||
|  | 	else | ||||||
|  | 		(void) fprintf(fp, "%s\n", vdev_prop_values(prop)); | ||||||
|  | 
 | ||||||
|  | 	return (ZPROP_CONT); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Display usage message.  If we're inside a command, display only the usage for |  * Display usage message.  If we're inside a command, display only the usage for | ||||||
|  * that command.  Otherwise, iterate over the entire command table and display |  * that command.  Otherwise, iterate over the entire command table and display | ||||||
| @ -519,6 +544,7 @@ usage(boolean_t requested) | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (current_command != NULL && | 	if (current_command != NULL && | ||||||
|  | 	    current_prop_type != (ZFS_TYPE_POOL | ZFS_TYPE_VDEV) && | ||||||
| 	    ((strcmp(current_command->name, "set") == 0) || | 	    ((strcmp(current_command->name, "set") == 0) || | ||||||
| 	    (strcmp(current_command->name, "get") == 0) || | 	    (strcmp(current_command->name, "get") == 0) || | ||||||
| 	    (strcmp(current_command->name, "list") == 0))) { | 	    (strcmp(current_command->name, "list") == 0))) { | ||||||
| @ -530,14 +556,21 @@ usage(boolean_t requested) | |||||||
| 		    "PROPERTY", "EDIT", "VALUES"); | 		    "PROPERTY", "EDIT", "VALUES"); | ||||||
| 
 | 
 | ||||||
| 		/* Iterate over all properties */ | 		/* Iterate over all properties */ | ||||||
| 		(void) zprop_iter(print_prop_cb, fp, B_FALSE, B_TRUE, | 		if (current_prop_type == ZFS_TYPE_POOL) { | ||||||
| 		    ZFS_TYPE_POOL); | 			(void) zprop_iter(print_pool_prop_cb, fp, B_FALSE, | ||||||
|  | 			    B_TRUE, current_prop_type); | ||||||
| 
 | 
 | ||||||
| 			(void) fprintf(fp, "\t%-19s   ", "feature@..."); | 			(void) fprintf(fp, "\t%-19s   ", "feature@..."); | ||||||
| 		(void) fprintf(fp, "YES   disabled | enabled | active\n"); | 			(void) fprintf(fp, "YES   " | ||||||
|  | 			    "disabled | enabled | active\n"); | ||||||
| 
 | 
 | ||||||
| 		(void) fprintf(fp, gettext("\nThe feature@ properties must be " | 			(void) fprintf(fp, gettext("\nThe feature@ properties " | ||||||
| 		    "appended with a feature name.\nSee zpool-features(7).\n")); | 			    "must be appended with a feature name.\n" | ||||||
|  | 			    "See zpool-features(7).\n")); | ||||||
|  | 		} else if (current_prop_type == ZFS_TYPE_VDEV) { | ||||||
|  | 			(void) zprop_iter(print_vdev_prop_cb, fp, B_FALSE, | ||||||
|  | 			    B_TRUE, current_prop_type); | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| @ -795,9 +828,10 @@ add_prop_list(const char *propname, char *propval, nvlist_t **props, | |||||||
| 		    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 && | ||||||
| 		    !zpool_prop_feature(propname)) { | 		    (!zpool_prop_feature(propname) && | ||||||
|  | 		    !zpool_prop_vdev(propname))) { | ||||||
| 			(void) fprintf(stderr, gettext("property '%s' is " | 			(void) fprintf(stderr, gettext("property '%s' is " | ||||||
| 			    "not a valid pool property\n"), propname); | 			    "not a valid pool or vdev property\n"), propname); | ||||||
| 			return (2); | 			return (2); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| @ -832,7 +866,7 @@ add_prop_list(const char *propname, char *propval, nvlist_t **props, | |||||||
| 			return (2); | 			return (2); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if (zpool_prop_feature(propname)) | 		if (zpool_prop_feature(propname) || zpool_prop_vdev(propname)) | ||||||
| 			normnm = propname; | 			normnm = propname; | ||||||
| 		else | 		else | ||||||
| 			normnm = zpool_prop_to_name(prop); | 			normnm = zpool_prop_to_name(prop); | ||||||
| @ -1930,7 +1964,7 @@ zpool_do_export(int argc, char **argv) | |||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		return (for_each_pool(argc, argv, B_TRUE, NULL, | 		return (for_each_pool(argc, argv, B_TRUE, NULL, | ||||||
| 		    B_FALSE, zpool_export_one, &cb)); | 		    ZFS_TYPE_POOL, B_FALSE, zpool_export_one, &cb)); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/* check arguments */ | 	/* check arguments */ | ||||||
| @ -1939,8 +1973,8 @@ zpool_do_export(int argc, char **argv) | |||||||
| 		usage(B_FALSE); | 		usage(B_FALSE); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ret = for_each_pool(argc, argv, B_TRUE, NULL, B_FALSE, zpool_export_one, | 	ret = for_each_pool(argc, argv, B_TRUE, NULL, ZFS_TYPE_POOL, | ||||||
| 	    &cb); | 	    B_FALSE, zpool_export_one, &cb); | ||||||
| 
 | 
 | ||||||
| 	return (ret); | 	return (ret); | ||||||
| } | } | ||||||
| @ -2436,6 +2470,12 @@ print_status_config(zpool_handle_t *zhp, status_cbdata_t *cb, const char *name, | |||||||
| 		    1 << vs->vs_configured_ashift, 1 << vs->vs_physical_ashift); | 		    1 << vs->vs_configured_ashift, 1 << vs->vs_physical_ashift); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	if (vs->vs_scan_removing != 0) { | ||||||
|  | 		(void) printf(gettext("  (removing)")); | ||||||
|  | 	} else if (vs->vs_noalloc != 0) { | ||||||
|  | 		(void) printf(gettext("  (non-allocating)")); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	/* The root vdev has the scrub/resilver stats */ | 	/* The root vdev has the scrub/resilver stats */ | ||||||
| 	root = fnvlist_lookup_nvlist(zpool_get_config(zhp, NULL), | 	root = fnvlist_lookup_nvlist(zpool_get_config(zhp, NULL), | ||||||
| 	    ZPOOL_CONFIG_VDEV_TREE); | 	    ZPOOL_CONFIG_VDEV_TREE); | ||||||
| @ -3857,24 +3897,22 @@ zpool_do_sync(int argc, char **argv) | |||||||
| 	argv += optind; | 	argv += optind; | ||||||
| 
 | 
 | ||||||
| 	/* if argc == 0 we will execute zpool_sync_one on all pools */ | 	/* if argc == 0 we will execute zpool_sync_one on all pools */ | ||||||
| 	ret = for_each_pool(argc, argv, B_FALSE, NULL, B_FALSE, zpool_sync_one, | 	ret = for_each_pool(argc, argv, B_FALSE, NULL, ZFS_TYPE_POOL, | ||||||
| 	    &force); | 	    B_FALSE, zpool_sync_one, &force); | ||||||
| 
 | 
 | ||||||
| 	return (ret); | 	return (ret); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| typedef struct iostat_cbdata { | typedef struct iostat_cbdata { | ||||||
| 	uint64_t cb_flags; | 	uint64_t cb_flags; | ||||||
| 	int cb_name_flags; |  | ||||||
| 	int cb_namewidth; | 	int cb_namewidth; | ||||||
| 	int cb_iteration; | 	int cb_iteration; | ||||||
| 	char **cb_vdev_names; /* Only show these vdevs */ |  | ||||||
| 	unsigned int cb_vdev_names_count; |  | ||||||
| 	boolean_t cb_verbose; | 	boolean_t cb_verbose; | ||||||
| 	boolean_t cb_literal; | 	boolean_t cb_literal; | ||||||
| 	boolean_t cb_scripted; | 	boolean_t cb_scripted; | ||||||
| 	zpool_list_t *cb_list; | 	zpool_list_t *cb_list; | ||||||
| 	vdev_cmd_data_list_t *vcdl; | 	vdev_cmd_data_list_t *vcdl; | ||||||
|  | 	vdev_cbdata_t cb_vdevs; | ||||||
| } iostat_cbdata_t; | } iostat_cbdata_t; | ||||||
| 
 | 
 | ||||||
| /*  iostat labels */ | /*  iostat labels */ | ||||||
| @ -4128,7 +4166,7 @@ print_iostat_dashes(iostat_cbdata_t *cb, unsigned int force_column_width, | |||||||
| 
 | 
 | ||||||
| 	if (cb->cb_flags & IOS_ANYHISTO_M) { | 	if (cb->cb_flags & IOS_ANYHISTO_M) { | ||||||
| 		title = histo_to_title[IOS_HISTO_IDX(cb->cb_flags)]; | 		title = histo_to_title[IOS_HISTO_IDX(cb->cb_flags)]; | ||||||
| 	} else if (cb->cb_vdev_names_count) { | 	} else if (cb->cb_vdevs.cb_names_count) { | ||||||
| 		title = "vdev"; | 		title = "vdev"; | ||||||
| 	} else  { | 	} else  { | ||||||
| 		title = "pool"; | 		title = "pool"; | ||||||
| @ -4188,7 +4226,7 @@ print_iostat_header_impl(iostat_cbdata_t *cb, unsigned int force_column_width, | |||||||
| 
 | 
 | ||||||
| 	if (cb->cb_flags & IOS_ANYHISTO_M) { | 	if (cb->cb_flags & IOS_ANYHISTO_M) { | ||||||
| 		title = histo_to_title[IOS_HISTO_IDX(cb->cb_flags)]; | 		title = histo_to_title[IOS_HISTO_IDX(cb->cb_flags)]; | ||||||
| 	} else if (cb->cb_vdev_names_count) { | 	} else if (cb->cb_vdevs.cb_names_count) { | ||||||
| 		title = "vdev"; | 		title = "vdev"; | ||||||
| 	} else  { | 	} else  { | ||||||
| 		title = "pool"; | 		title = "pool"; | ||||||
| @ -4696,9 +4734,9 @@ print_vdev_stats(zpool_handle_t *zhp, const char *name, nvlist_t *oldnv, | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/* Do we only want to see a specific vdev? */ | 	/* Do we only want to see a specific vdev? */ | ||||||
| 	for (i = 0; i < cb->cb_vdev_names_count; i++) { | 	for (i = 0; i < cb->cb_vdevs.cb_names_count; i++) { | ||||||
| 		/* Yes we do.  Is this the vdev? */ | 		/* Yes we do.  Is this the vdev? */ | ||||||
| 		if (strcmp(name, cb->cb_vdev_names[i]) == 0) { | 		if (strcmp(name, cb->cb_vdevs.cb_names[i]) == 0) { | ||||||
| 			/*
 | 			/*
 | ||||||
| 			 * This is our vdev.  Since it is the only vdev we | 			 * This is our vdev.  Since it is the only vdev we | ||||||
| 			 * will be displaying, make depth = 0 so that it | 			 * will be displaying, make depth = 0 so that it | ||||||
| @ -4709,7 +4747,7 @@ print_vdev_stats(zpool_handle_t *zhp, const char *name, nvlist_t *oldnv, | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (cb->cb_vdev_names_count && (i == cb->cb_vdev_names_count)) { | 	if (cb->cb_vdevs.cb_names_count && (i == cb->cb_vdevs.cb_names_count)) { | ||||||
| 		/* Couldn't match the name */ | 		/* Couldn't match the name */ | ||||||
| 		goto children; | 		goto children; | ||||||
| 	} | 	} | ||||||
| @ -4816,7 +4854,7 @@ children: | |||||||
| 			continue; | 			continue; | ||||||
| 
 | 
 | ||||||
| 		vname = zpool_vdev_name(g_zfs, zhp, newchild[c], | 		vname = zpool_vdev_name(g_zfs, zhp, newchild[c], | ||||||
| 		    cb->cb_name_flags); | 		    cb->cb_vdevs.cb_name_flags); | ||||||
| 		ret += print_vdev_stats(zhp, vname, oldnv ? oldchild[c] : NULL, | 		ret += print_vdev_stats(zhp, vname, oldnv ? oldchild[c] : NULL, | ||||||
| 		    newchild[c], cb, depth + 2); | 		    newchild[c], cb, depth + 2); | ||||||
| 		free(vname); | 		free(vname); | ||||||
| @ -4850,7 +4888,8 @@ children: | |||||||
| 
 | 
 | ||||||
| 			if (!printed) { | 			if (!printed) { | ||||||
| 				if ((!(cb->cb_flags & IOS_ANYHISTO_M)) && | 				if ((!(cb->cb_flags & IOS_ANYHISTO_M)) && | ||||||
| 				    !cb->cb_scripted && !cb->cb_vdev_names) { | 				    !cb->cb_scripted && | ||||||
|  | 				    !cb->cb_vdevs.cb_names) { | ||||||
| 					print_iostat_dashes(cb, 0, | 					print_iostat_dashes(cb, 0, | ||||||
| 					    class_name[n]); | 					    class_name[n]); | ||||||
| 				} | 				} | ||||||
| @ -4859,7 +4898,7 @@ children: | |||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			vname = zpool_vdev_name(g_zfs, zhp, newchild[c], | 			vname = zpool_vdev_name(g_zfs, zhp, newchild[c], | ||||||
| 			    cb->cb_name_flags); | 			    cb->cb_vdevs.cb_name_flags); | ||||||
| 			ret += print_vdev_stats(zhp, vname, oldnv ? | 			ret += print_vdev_stats(zhp, vname, oldnv ? | ||||||
| 			    oldchild[c] : NULL, newchild[c], cb, depth + 2); | 			    oldchild[c] : NULL, newchild[c], cb, depth + 2); | ||||||
| 			free(vname); | 			free(vname); | ||||||
| @ -4883,14 +4922,14 @@ children: | |||||||
| 
 | 
 | ||||||
| 	if (children > 0) { | 	if (children > 0) { | ||||||
| 		if ((!(cb->cb_flags & IOS_ANYHISTO_M)) && !cb->cb_scripted && | 		if ((!(cb->cb_flags & IOS_ANYHISTO_M)) && !cb->cb_scripted && | ||||||
| 		    !cb->cb_vdev_names) { | 		    !cb->cb_vdevs.cb_names) { | ||||||
| 			print_iostat_dashes(cb, 0, "cache"); | 			print_iostat_dashes(cb, 0, "cache"); | ||||||
| 		} | 		} | ||||||
| 		printf("\n"); | 		printf("\n"); | ||||||
| 
 | 
 | ||||||
| 		for (c = 0; c < children; c++) { | 		for (c = 0; c < children; c++) { | ||||||
| 			vname = zpool_vdev_name(g_zfs, zhp, newchild[c], | 			vname = zpool_vdev_name(g_zfs, zhp, newchild[c], | ||||||
| 			    cb->cb_name_flags); | 			    cb->cb_vdevs.cb_name_flags); | ||||||
| 			ret += print_vdev_stats(zhp, vname, oldnv ? oldchild[c] | 			ret += print_vdev_stats(zhp, vname, oldnv ? oldchild[c] | ||||||
| 			    : NULL, newchild[c], cb, depth + 2); | 			    : NULL, newchild[c], cb, depth + 2); | ||||||
| 			free(vname); | 			free(vname); | ||||||
| @ -4946,7 +4985,8 @@ print_iostat(zpool_handle_t *zhp, void *data) | |||||||
| 	ret = print_vdev_stats(zhp, zpool_get_name(zhp), oldnvroot, newnvroot, | 	ret = print_vdev_stats(zhp, zpool_get_name(zhp), oldnvroot, newnvroot, | ||||||
| 	    cb, 0); | 	    cb, 0); | ||||||
| 	if ((ret != 0) && !(cb->cb_flags & IOS_ANYHISTO_M) && | 	if ((ret != 0) && !(cb->cb_flags & IOS_ANYHISTO_M) && | ||||||
| 	    !cb->cb_scripted && cb->cb_verbose && !cb->cb_vdev_names_count) { | 	    !cb->cb_scripted && cb->cb_verbose && | ||||||
|  | 	    !cb->cb_vdevs.cb_names_count) { | ||||||
| 		print_iostat_separator(cb); | 		print_iostat_separator(cb); | ||||||
| 		if (cb->vcdl != NULL) { | 		if (cb->vcdl != NULL) { | ||||||
| 			print_cmd_columns(cb->vcdl, 1); | 			print_cmd_columns(cb->vcdl, 1); | ||||||
| @ -5153,27 +5193,30 @@ get_stat_flags(zpool_list_t *list) | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Return 1 if cb_data->cb_vdev_names[0] is this vdev's name, 0 otherwise. |  * Return 1 if cb_data->cb_names[0] is this vdev's name, 0 otherwise. | ||||||
|  */ |  */ | ||||||
| static int | static int | ||||||
| is_vdev_cb(void *zhp_data, nvlist_t *nv, void *cb_data) | is_vdev_cb(void *zhp_data, nvlist_t *nv, void *cb_data) | ||||||
| { | { | ||||||
| 	iostat_cbdata_t *cb = cb_data; | 	vdev_cbdata_t *cb = cb_data; | ||||||
| 	char *name = NULL; | 	char *name = NULL; | ||||||
| 	int ret = 0; | 	int ret = 1; /* assume match */ | ||||||
| 	zpool_handle_t *zhp = zhp_data; | 	zpool_handle_t *zhp = zhp_data; | ||||||
| 
 | 
 | ||||||
| 	name = zpool_vdev_name(g_zfs, zhp, nv, cb->cb_name_flags); | 	name = zpool_vdev_name(g_zfs, zhp, nv, cb->cb_name_flags); | ||||||
| 
 | 
 | ||||||
| 	if (strcmp(name, cb->cb_vdev_names[0]) == 0) | 	if (strcmp(name, cb->cb_names[0])) { | ||||||
| 		ret = 1; /* match */ | 		free(name); | ||||||
|  | 		name = zpool_vdev_name(g_zfs, zhp, nv, VDEV_NAME_GUID); | ||||||
|  | 		ret = (strcmp(name, cb->cb_names[0]) == 0); | ||||||
|  | 	} | ||||||
| 	free(name); | 	free(name); | ||||||
| 
 | 
 | ||||||
| 	return (ret); | 	return (ret); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Returns 1 if cb_data->cb_vdev_names[0] is a vdev name, 0 otherwise. |  * Returns 1 if cb_data->cb_names[0] is a vdev name, 0 otherwise. | ||||||
|  */ |  */ | ||||||
| static int | static int | ||||||
| is_vdev(zpool_handle_t *zhp, void *cb_data) | is_vdev(zpool_handle_t *zhp, void *cb_data) | ||||||
| @ -5189,7 +5232,7 @@ is_vdev(zpool_handle_t *zhp, void *cb_data) | |||||||
|  */ |  */ | ||||||
| static int | static int | ||||||
| are_vdevs_in_pool(int argc, char **argv, char *pool_name, | are_vdevs_in_pool(int argc, char **argv, char *pool_name, | ||||||
|     iostat_cbdata_t *cb) |     vdev_cbdata_t *cb) | ||||||
| { | { | ||||||
| 	char **tmp_name; | 	char **tmp_name; | ||||||
| 	int ret = 0; | 	int ret = 0; | ||||||
| @ -5202,23 +5245,23 @@ are_vdevs_in_pool(int argc, char **argv, char *pool_name, | |||||||
| 	if (pool_name) | 	if (pool_name) | ||||||
| 		pool_count = 1; | 		pool_count = 1; | ||||||
| 
 | 
 | ||||||
| 	/* Temporarily hijack cb_vdev_names for a second... */ | 	/* Temporarily hijack cb_names for a second... */ | ||||||
| 	tmp_name = cb->cb_vdev_names; | 	tmp_name = cb->cb_names; | ||||||
| 
 | 
 | ||||||
| 	/* Go though our list of prospective vdev names */ | 	/* Go though our list of prospective vdev names */ | ||||||
| 	for (i = 0; i < argc; i++) { | 	for (i = 0; i < argc; i++) { | ||||||
| 		cb->cb_vdev_names = argv + i; | 		cb->cb_names = argv + i; | ||||||
| 
 | 
 | ||||||
| 		/* Is this name a vdev in our pools? */ | 		/* Is this name a vdev in our pools? */ | ||||||
| 		ret = for_each_pool(pool_count, &pool_name, B_TRUE, NULL, | 		ret = for_each_pool(pool_count, &pool_name, B_TRUE, NULL, | ||||||
| 		    B_FALSE, is_vdev, cb); | 		    ZFS_TYPE_POOL, B_FALSE, is_vdev, cb); | ||||||
| 		if (!ret) { | 		if (!ret) { | ||||||
| 			/* No match */ | 			/* No match */ | ||||||
| 			break; | 			break; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	cb->cb_vdev_names = tmp_name; | 	cb->cb_names = tmp_name; | ||||||
| 
 | 
 | ||||||
| 	return (ret); | 	return (ret); | ||||||
| } | } | ||||||
| @ -5239,8 +5282,8 @@ is_pool_cb(zpool_handle_t *zhp, void *data) | |||||||
| static int | static int | ||||||
| is_pool(char *name) | is_pool(char *name) | ||||||
| { | { | ||||||
| 	return (for_each_pool(0, NULL, B_TRUE, NULL, B_FALSE, is_pool_cb, | 	return (for_each_pool(0, NULL, B_TRUE, NULL, ZFS_TYPE_POOL, B_FALSE, | ||||||
| 	    name)); | 	    is_pool_cb, name)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /* Are all our argv[] strings pool names?  If so return 1, 0 otherwise. */ | /* Are all our argv[] strings pool names?  If so return 1, 0 otherwise. */ | ||||||
| @ -5263,7 +5306,7 @@ are_all_pools(int argc, char **argv) | |||||||
|  */ |  */ | ||||||
| static void | static void | ||||||
| error_list_unresolved_vdevs(int argc, char **argv, char *pool_name, | error_list_unresolved_vdevs(int argc, char **argv, char *pool_name, | ||||||
|     iostat_cbdata_t *cb) |     vdev_cbdata_t *cb) | ||||||
| { | { | ||||||
| 	int i; | 	int i; | ||||||
| 	char *name; | 	char *name; | ||||||
| @ -5287,7 +5330,7 @@ error_list_unresolved_vdevs(int argc, char **argv, char *pool_name, | |||||||
| /*
 | /*
 | ||||||
|  * Same as get_interval_count(), but with additional checks to not misinterpret |  * Same as get_interval_count(), but with additional checks to not misinterpret | ||||||
|  * guids as interval/count values.  Assumes VDEV_NAME_GUID is set in |  * guids as interval/count values.  Assumes VDEV_NAME_GUID is set in | ||||||
|  * cb.cb_name_flags. |  * cb.cb_vdevs.cb_name_flags. | ||||||
|  */ |  */ | ||||||
| static void | static void | ||||||
| get_interval_count_filter_guids(int *argc, char **argv, float *interval, | get_interval_count_filter_guids(int *argc, char **argv, float *interval, | ||||||
| @ -5297,7 +5340,8 @@ get_interval_count_filter_guids(int *argc, char **argv, float *interval, | |||||||
| 	int argc_for_interval = 0; | 	int argc_for_interval = 0; | ||||||
| 
 | 
 | ||||||
| 	/* Is the last arg an interval value?  Or a guid? */ | 	/* Is the last arg an interval value?  Or a guid? */ | ||||||
| 	if (*argc >= 1 && !are_vdevs_in_pool(1, &argv[*argc - 1], NULL, cb)) { | 	if (*argc >= 1 && !are_vdevs_in_pool(1, &argv[*argc - 1], NULL, | ||||||
|  | 	    &cb->cb_vdevs)) { | ||||||
| 		/*
 | 		/*
 | ||||||
| 		 * The last arg is not a guid, so it's probably an | 		 * The last arg is not a guid, so it's probably an | ||||||
| 		 * interval value. | 		 * interval value. | ||||||
| @ -5305,7 +5349,8 @@ get_interval_count_filter_guids(int *argc, char **argv, float *interval, | |||||||
| 		argc_for_interval++; | 		argc_for_interval++; | ||||||
| 
 | 
 | ||||||
| 		if (*argc >= 2 && | 		if (*argc >= 2 && | ||||||
| 		    !are_vdevs_in_pool(1, &argv[*argc - 2], NULL, cb)) { | 		    !are_vdevs_in_pool(1, &argv[*argc - 2], NULL, | ||||||
|  | 		    &cb->cb_vdevs)) { | ||||||
| 			/*
 | 			/*
 | ||||||
| 			 * The 2nd to last arg is not a guid, so it's probably | 			 * The 2nd to last arg is not a guid, so it's probably | ||||||
| 			 * an interval value. | 			 * an interval value. | ||||||
| @ -5448,7 +5493,7 @@ get_namewidth_iostat(zpool_handle_t *zhp, void *data) | |||||||
| 	 * get_namewidth() returns the maximum width of any name in that column | 	 * get_namewidth() returns the maximum width of any name in that column | ||||||
| 	 * for any pool/vdev/device line that will be output. | 	 * for any pool/vdev/device line that will be output. | ||||||
| 	 */ | 	 */ | ||||||
| 	width = get_namewidth(zhp, cb->cb_namewidth, cb->cb_name_flags, | 	width = get_namewidth(zhp, cb->cb_namewidth, cb->cb_vdevs.cb_name_flags, | ||||||
| 	    cb->cb_verbose); | 	    cb->cb_verbose); | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| @ -5626,11 +5671,11 @@ zpool_do_iostat(int argc, char **argv) | |||||||
| 	cb.cb_scripted = scripted; | 	cb.cb_scripted = scripted; | ||||||
| 
 | 
 | ||||||
| 	if (guid) | 	if (guid) | ||||||
| 		cb.cb_name_flags |= VDEV_NAME_GUID; | 		cb.cb_vdevs.cb_name_flags |= VDEV_NAME_GUID; | ||||||
| 	if (follow_links) | 	if (follow_links) | ||||||
| 		cb.cb_name_flags |= VDEV_NAME_FOLLOW_LINKS; | 		cb.cb_vdevs.cb_name_flags |= VDEV_NAME_FOLLOW_LINKS; | ||||||
| 	if (full_name) | 	if (full_name) | ||||||
| 		cb.cb_name_flags |= VDEV_NAME_PATH; | 		cb.cb_vdevs.cb_name_flags |= VDEV_NAME_PATH; | ||||||
| 	cb.cb_iteration = 0; | 	cb.cb_iteration = 0; | ||||||
| 	cb.cb_namewidth = 0; | 	cb.cb_namewidth = 0; | ||||||
| 	cb.cb_verbose = verbose; | 	cb.cb_verbose = verbose; | ||||||
| @ -5647,17 +5692,18 @@ zpool_do_iostat(int argc, char **argv) | |||||||
| 		/* No args, so just print the defaults. */ | 		/* No args, so just print the defaults. */ | ||||||
| 	} else if (are_all_pools(argc, argv)) { | 	} else if (are_all_pools(argc, argv)) { | ||||||
| 		/* All the args are pool names */ | 		/* All the args are pool names */ | ||||||
| 	} else if (are_vdevs_in_pool(argc, argv, NULL, &cb)) { | 	} else if (are_vdevs_in_pool(argc, argv, NULL, &cb.cb_vdevs)) { | ||||||
| 		/* All the args are vdevs */ | 		/* All the args are vdevs */ | ||||||
| 		cb.cb_vdev_names = argv; | 		cb.cb_vdevs.cb_names = argv; | ||||||
| 		cb.cb_vdev_names_count = argc; | 		cb.cb_vdevs.cb_names_count = argc; | ||||||
| 		argc = 0; /* No pools to process */ | 		argc = 0; /* No pools to process */ | ||||||
| 	} else if (are_all_pools(1, argv)) { | 	} else if (are_all_pools(1, argv)) { | ||||||
| 		/* The first arg is a pool name */ | 		/* The first arg is a pool name */ | ||||||
| 		if (are_vdevs_in_pool(argc - 1, argv + 1, argv[0], &cb)) { | 		if (are_vdevs_in_pool(argc - 1, argv + 1, argv[0], | ||||||
|  | 		    &cb.cb_vdevs)) { | ||||||
| 			/* ...and the rest are vdev names */ | 			/* ...and the rest are vdev names */ | ||||||
| 			cb.cb_vdev_names = argv + 1; | 			cb.cb_vdevs.cb_names = argv + 1; | ||||||
| 			cb.cb_vdev_names_count = argc - 1; | 			cb.cb_vdevs.cb_names_count = argc - 1; | ||||||
| 			argc = 1; /* One pool to process */ | 			argc = 1; /* One pool to process */ | ||||||
| 		} else { | 		} else { | ||||||
| 			fprintf(stderr, gettext("Expected either a list of ")); | 			fprintf(stderr, gettext("Expected either a list of ")); | ||||||
| @ -5665,7 +5711,7 @@ zpool_do_iostat(int argc, char **argv) | |||||||
| 			fprintf(stderr, " \"%s\", ", argv[0]); | 			fprintf(stderr, " \"%s\", ", argv[0]); | ||||||
| 			fprintf(stderr, gettext("but got:\n")); | 			fprintf(stderr, gettext("but got:\n")); | ||||||
| 			error_list_unresolved_vdevs(argc - 1, argv + 1, | 			error_list_unresolved_vdevs(argc - 1, argv + 1, | ||||||
| 			    argv[0], &cb); | 			    argv[0], &cb.cb_vdevs); | ||||||
| 			fprintf(stderr, "\n"); | 			fprintf(stderr, "\n"); | ||||||
| 			usage(B_FALSE); | 			usage(B_FALSE); | ||||||
| 			return (1); | 			return (1); | ||||||
| @ -5680,7 +5726,7 @@ zpool_do_iostat(int argc, char **argv) | |||||||
| 		return (1); | 		return (1); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (cb.cb_vdev_names_count != 0) { | 	if (cb.cb_vdevs.cb_names_count != 0) { | ||||||
| 		/*
 | 		/*
 | ||||||
| 		 * If user specified vdevs, it implies verbose. | 		 * If user specified vdevs, it implies verbose. | ||||||
| 		 */ | 		 */ | ||||||
| @ -5691,7 +5737,8 @@ zpool_do_iostat(int argc, char **argv) | |||||||
| 	 * Construct the list of all interesting pools. | 	 * Construct the list of all interesting pools. | ||||||
| 	 */ | 	 */ | ||||||
| 	ret = 0; | 	ret = 0; | ||||||
| 	if ((list = pool_list_get(argc, argv, NULL, parsable, &ret)) == NULL) | 	if ((list = pool_list_get(argc, argv, NULL, ZFS_TYPE_POOL, parsable, | ||||||
|  | 	    &ret)) == NULL) | ||||||
| 		return (1); | 		return (1); | ||||||
| 
 | 
 | ||||||
| 	if (pool_list_count(list) == 0 && argc != 0) { | 	if (pool_list_count(list) == 0 && argc != 0) { | ||||||
| @ -5799,8 +5846,9 @@ zpool_do_iostat(int argc, char **argv) | |||||||
| 			if (cmd != NULL && cb.cb_verbose && | 			if (cmd != NULL && cb.cb_verbose && | ||||||
| 			    !(cb.cb_flags & IOS_ANYHISTO_M)) { | 			    !(cb.cb_flags & IOS_ANYHISTO_M)) { | ||||||
| 				cb.vcdl = all_pools_for_each_vdev_run(argc, | 				cb.vcdl = all_pools_for_each_vdev_run(argc, | ||||||
| 				    argv, cmd, g_zfs, cb.cb_vdev_names, | 				    argv, cmd, g_zfs, cb.cb_vdevs.cb_names, | ||||||
| 				    cb.cb_vdev_names_count, cb.cb_name_flags); | 				    cb.cb_vdevs.cb_names_count, | ||||||
|  | 				    cb.cb_vdevs.cb_name_flags); | ||||||
| 			} else { | 			} else { | ||||||
| 				cb.vcdl = NULL; | 				cb.vcdl = NULL; | ||||||
| 			} | 			} | ||||||
| @ -5852,7 +5900,7 @@ zpool_do_iostat(int argc, char **argv) | |||||||
| 			if (((npools > 1 && !verbose && | 			if (((npools > 1 && !verbose && | ||||||
| 			    !(cb.cb_flags & IOS_ANYHISTO_M)) || | 			    !(cb.cb_flags & IOS_ANYHISTO_M)) || | ||||||
| 			    (!(cb.cb_flags & IOS_ANYHISTO_M) && | 			    (!(cb.cb_flags & IOS_ANYHISTO_M) && | ||||||
| 			    cb.cb_vdev_names_count)) && | 			    cb.cb_vdevs.cb_names_count)) && | ||||||
| 			    !cb.cb_scripted) { | 			    !cb.cb_scripted) { | ||||||
| 				print_iostat_separator(&cb); | 				print_iostat_separator(&cb); | ||||||
| 				if (cb.vcdl != NULL) | 				if (cb.vcdl != NULL) | ||||||
| @ -6314,6 +6362,7 @@ zpool_do_list(int argc, char **argv) | |||||||
| 	unsigned long count = 0; | 	unsigned long count = 0; | ||||||
| 	zpool_list_t *list; | 	zpool_list_t *list; | ||||||
| 	boolean_t first = B_TRUE; | 	boolean_t first = B_TRUE; | ||||||
|  | 	current_prop_type = ZFS_TYPE_POOL; | ||||||
| 
 | 
 | ||||||
| 	/* check options */ | 	/* check options */ | ||||||
| 	while ((c = getopt(argc, argv, ":gHLo:pPT:v")) != -1) { | 	while ((c = getopt(argc, argv, ":gHLo:pPT:v")) != -1) { | ||||||
| @ -6365,7 +6414,7 @@ zpool_do_list(int argc, char **argv) | |||||||
| 
 | 
 | ||||||
| 	for (;;) { | 	for (;;) { | ||||||
| 		if ((list = pool_list_get(argc, argv, &cb.cb_proplist, | 		if ((list = pool_list_get(argc, argv, &cb.cb_proplist, | ||||||
| 		    cb.cb_literal, &ret)) == NULL) | 		    ZFS_TYPE_POOL, cb.cb_literal, &ret)) == NULL) | ||||||
| 			return (1); | 			return (1); | ||||||
| 
 | 
 | ||||||
| 		if (pool_list_count(list) == 0) | 		if (pool_list_count(list) == 0) | ||||||
| @ -7121,8 +7170,8 @@ zpool_do_reopen(int argc, char **argv) | |||||||
| 	argv += optind; | 	argv += optind; | ||||||
| 
 | 
 | ||||||
| 	/* if argc == 0 we will execute zpool_reopen_one on all pools */ | 	/* if argc == 0 we will execute zpool_reopen_one on all pools */ | ||||||
| 	ret = for_each_pool(argc, argv, B_TRUE, NULL, B_FALSE, zpool_reopen_one, | 	ret = for_each_pool(argc, argv, B_TRUE, NULL, ZFS_TYPE_POOL, | ||||||
| 	    &scrub_restart); | 	    B_FALSE, zpool_reopen_one, &scrub_restart); | ||||||
| 
 | 
 | ||||||
| 	return (ret); | 	return (ret); | ||||||
| } | } | ||||||
| @ -7251,13 +7300,13 @@ zpool_do_scrub(int argc, char **argv) | |||||||
| 		usage(B_FALSE); | 		usage(B_FALSE); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	error = for_each_pool(argc, argv, B_TRUE, NULL, B_FALSE, | 	error = for_each_pool(argc, argv, B_TRUE, NULL, ZFS_TYPE_POOL, | ||||||
| 	    scrub_callback, &cb); | 	    B_FALSE, scrub_callback, &cb); | ||||||
| 
 | 
 | ||||||
| 	if (wait && !error) { | 	if (wait && !error) { | ||||||
| 		zpool_wait_activity_t act = ZPOOL_WAIT_SCRUB; | 		zpool_wait_activity_t act = ZPOOL_WAIT_SCRUB; | ||||||
| 		error = for_each_pool(argc, argv, B_TRUE, NULL, B_FALSE, | 		error = for_each_pool(argc, argv, B_TRUE, NULL, ZFS_TYPE_POOL, | ||||||
| 		    wait_callback, &act); | 		    B_FALSE, wait_callback, &act); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return (error); | 	return (error); | ||||||
| @ -7295,8 +7344,8 @@ zpool_do_resilver(int argc, char **argv) | |||||||
| 		usage(B_FALSE); | 		usage(B_FALSE); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return (for_each_pool(argc, argv, B_TRUE, NULL, B_FALSE, | 	return (for_each_pool(argc, argv, B_TRUE, NULL, ZFS_TYPE_POOL, | ||||||
| 	    scrub_callback, &cb)); | 	    B_FALSE, scrub_callback, &cb)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
| @ -8719,8 +8768,8 @@ zpool_do_status(int argc, char **argv) | |||||||
| 			cb.vcdl = all_pools_for_each_vdev_run(argc, argv, cmd, | 			cb.vcdl = all_pools_for_each_vdev_run(argc, argv, cmd, | ||||||
| 			    NULL, NULL, 0, 0); | 			    NULL, NULL, 0, 0); | ||||||
| 
 | 
 | ||||||
| 		ret = for_each_pool(argc, argv, B_TRUE, NULL, cb.cb_literal, | 		ret = for_each_pool(argc, argv, B_TRUE, NULL, ZFS_TYPE_POOL, | ||||||
| 		    status_callback, &cb); | 		    cb.cb_literal, status_callback, &cb); | ||||||
| 
 | 
 | ||||||
| 		if (cb.vcdl != NULL) | 		if (cb.vcdl != NULL) | ||||||
| 			free_vdev_cmd_data_list(cb.vcdl); | 			free_vdev_cmd_data_list(cb.vcdl); | ||||||
| @ -9279,8 +9328,8 @@ zpool_do_upgrade(int argc, char **argv) | |||||||
| 			(void) printf(gettext("\n")); | 			(void) printf(gettext("\n")); | ||||||
| 		} | 		} | ||||||
| 	} else { | 	} else { | ||||||
| 		ret = for_each_pool(argc, argv, B_FALSE, NULL, B_FALSE, | 		ret = for_each_pool(argc, argv, B_FALSE, NULL, ZFS_TYPE_POOL, | ||||||
| 		    upgrade_one, &cb); | 		    B_FALSE, upgrade_one, &cb); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return (ret); | 	return (ret); | ||||||
| @ -9476,8 +9525,8 @@ zpool_do_history(int argc, char **argv) | |||||||
| 	argc -= optind; | 	argc -= optind; | ||||||
| 	argv += optind; | 	argv += optind; | ||||||
| 
 | 
 | ||||||
| 	ret = for_each_pool(argc, argv, B_FALSE, NULL, B_FALSE, get_history_one, | 	ret = for_each_pool(argc, argv, B_FALSE, NULL, ZFS_TYPE_POOL, | ||||||
| 	    &cbdata); | 	    B_FALSE, get_history_one, &cbdata); | ||||||
| 
 | 
 | ||||||
| 	if (argc == 0 && cbdata.first == B_TRUE) { | 	if (argc == 0 && cbdata.first == B_TRUE) { | ||||||
| 		(void) fprintf(stderr, gettext("no pools available\n")); | 		(void) fprintf(stderr, gettext("no pools available\n")); | ||||||
| @ -9874,6 +9923,71 @@ zpool_do_events(int argc, char **argv) | |||||||
| 	return (ret); | 	return (ret); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static int | ||||||
|  | get_callback_vdev(zpool_handle_t *zhp, char *vdevname, void *data) | ||||||
|  | { | ||||||
|  | 	zprop_get_cbdata_t *cbp = (zprop_get_cbdata_t *)data; | ||||||
|  | 	char value[ZFS_MAXPROPLEN]; | ||||||
|  | 	zprop_source_t srctype; | ||||||
|  | 
 | ||||||
|  | 	for (zprop_list_t *pl = cbp->cb_proplist; pl != NULL; | ||||||
|  | 	    pl = pl->pl_next) { | ||||||
|  | 		char *prop_name; | ||||||
|  | 		/*
 | ||||||
|  | 		 * If the first property is pool name, it is a special | ||||||
|  | 		 * placeholder that we can skip. This will also skip | ||||||
|  | 		 * over the name property when 'all' is specified. | ||||||
|  | 		 */ | ||||||
|  | 		if (pl->pl_prop == ZPOOL_PROP_NAME && | ||||||
|  | 		    pl == cbp->cb_proplist) | ||||||
|  | 			continue; | ||||||
|  | 
 | ||||||
|  | 		if (pl->pl_prop == ZPROP_INVAL) { | ||||||
|  | 			prop_name = pl->pl_user_prop; | ||||||
|  | 		} else { | ||||||
|  | 			prop_name = (char *)vdev_prop_to_name(pl->pl_prop); | ||||||
|  | 		} | ||||||
|  | 		if (zpool_get_vdev_prop(zhp, vdevname, pl->pl_prop, | ||||||
|  | 		    prop_name, value, sizeof (value), &srctype, | ||||||
|  | 		    cbp->cb_literal) == 0) { | ||||||
|  | 			zprop_print_one_property(vdevname, cbp, prop_name, | ||||||
|  | 			    value, srctype, NULL, NULL); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return (0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | get_callback_vdev_width_cb(void *zhp_data, nvlist_t *nv, void *data) | ||||||
|  | { | ||||||
|  | 	zpool_handle_t *zhp = zhp_data; | ||||||
|  | 	zprop_get_cbdata_t *cbp = (zprop_get_cbdata_t *)data; | ||||||
|  | 	char *vdevname = zpool_vdev_name(g_zfs, zhp, nv, | ||||||
|  | 	    cbp->cb_vdevs.cb_name_flags); | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	/* Adjust the column widths for the vdev properties */ | ||||||
|  | 	ret = vdev_expand_proplist(zhp, vdevname, &cbp->cb_proplist); | ||||||
|  | 
 | ||||||
|  | 	return (ret); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | get_callback_vdev_cb(void *zhp_data, nvlist_t *nv, void *data) | ||||||
|  | { | ||||||
|  | 	zpool_handle_t *zhp = zhp_data; | ||||||
|  | 	zprop_get_cbdata_t *cbp = (zprop_get_cbdata_t *)data; | ||||||
|  | 	char *vdevname = zpool_vdev_name(g_zfs, zhp, nv, | ||||||
|  | 	    cbp->cb_vdevs.cb_name_flags); | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	/* Display the properties */ | ||||||
|  | 	ret = get_callback_vdev(zhp, vdevname, data); | ||||||
|  | 
 | ||||||
|  | 	return (ret); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static int | static int | ||||||
| get_callback(zpool_handle_t *zhp, void *data) | get_callback(zpool_handle_t *zhp, void *data) | ||||||
| { | { | ||||||
| @ -9881,12 +9995,33 @@ get_callback(zpool_handle_t *zhp, void *data) | |||||||
| 	char value[MAXNAMELEN]; | 	char value[MAXNAMELEN]; | ||||||
| 	zprop_source_t srctype; | 	zprop_source_t srctype; | ||||||
| 	zprop_list_t *pl; | 	zprop_list_t *pl; | ||||||
|  | 	int vid; | ||||||
| 
 | 
 | ||||||
|  | 	if (cbp->cb_type == ZFS_TYPE_VDEV) { | ||||||
|  | 		if (strcmp(cbp->cb_vdevs.cb_names[0], "all-vdevs") == 0) { | ||||||
|  | 			for_each_vdev(zhp, get_callback_vdev_width_cb, data); | ||||||
|  | 			for_each_vdev(zhp, get_callback_vdev_cb, data); | ||||||
|  | 		} else { | ||||||
|  | 			/* Adjust column widths for vdev properties */ | ||||||
|  | 			for (vid = 0; vid < cbp->cb_vdevs.cb_names_count; | ||||||
|  | 			    vid++) { | ||||||
|  | 				vdev_expand_proplist(zhp, | ||||||
|  | 				    cbp->cb_vdevs.cb_names[vid], | ||||||
|  | 				    &cbp->cb_proplist); | ||||||
|  | 			} | ||||||
|  | 			/* Display the properties */ | ||||||
|  | 			for (vid = 0; vid < cbp->cb_vdevs.cb_names_count; | ||||||
|  | 			    vid++) { | ||||||
|  | 				get_callback_vdev(zhp, | ||||||
|  | 				    cbp->cb_vdevs.cb_names[vid], data); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		assert(cbp->cb_type == ZFS_TYPE_POOL); | ||||||
| 		for (pl = cbp->cb_proplist; pl != NULL; pl = pl->pl_next) { | 		for (pl = cbp->cb_proplist; pl != NULL; pl = pl->pl_next) { | ||||||
| 
 |  | ||||||
| 			/*
 | 			/*
 | ||||||
| 		 * Skip the special fake placeholder. This will also skip | 			 * Skip the special fake placeholder. This will also | ||||||
| 		 * over the name property when 'all' is specified. | 			 * skip over the name property when 'all' is specified. | ||||||
| 			 */ | 			 */ | ||||||
| 			if (pl->pl_prop == ZPOOL_PROP_NAME && | 			if (pl->pl_prop == ZPOOL_PROP_NAME && | ||||||
| 			    pl == cbp->cb_proplist) | 			    pl == cbp->cb_proplist) | ||||||
| @ -9897,22 +10032,27 @@ get_callback(zpool_handle_t *zhp, void *data) | |||||||
| 			    zpool_prop_unsupported(pl->pl_user_prop))) { | 			    zpool_prop_unsupported(pl->pl_user_prop))) { | ||||||
| 				srctype = ZPROP_SRC_LOCAL; | 				srctype = ZPROP_SRC_LOCAL; | ||||||
| 
 | 
 | ||||||
| 			if (zpool_prop_get_feature(zhp, pl->pl_user_prop, | 				if (zpool_prop_get_feature(zhp, | ||||||
| 			    value, sizeof (value)) == 0) { | 				    pl->pl_user_prop, value, | ||||||
| 				zprop_print_one_property(zpool_get_name(zhp), | 				    sizeof (value)) == 0) { | ||||||
| 				    cbp, pl->pl_user_prop, value, srctype, | 					zprop_print_one_property( | ||||||
|  | 					    zpool_get_name(zhp), cbp, | ||||||
|  | 					    pl->pl_user_prop, value, srctype, | ||||||
| 					    NULL, NULL); | 					    NULL, NULL); | ||||||
| 				} | 				} | ||||||
| 			} else { | 			} else { | ||||||
| 				if (zpool_get_prop(zhp, pl->pl_prop, value, | 				if (zpool_get_prop(zhp, pl->pl_prop, value, | ||||||
| 			    sizeof (value), &srctype, cbp->cb_literal) != 0) | 				    sizeof (value), &srctype, | ||||||
|  | 				    cbp->cb_literal) != 0) | ||||||
| 					continue; | 					continue; | ||||||
| 
 | 
 | ||||||
| 			zprop_print_one_property(zpool_get_name(zhp), cbp, | 				zprop_print_one_property(zpool_get_name(zhp), | ||||||
| 			    zpool_prop_to_name(pl->pl_prop), value, srctype, | 				    cbp, zpool_prop_to_name(pl->pl_prop), | ||||||
| 			    NULL, NULL); | 				    value, srctype, NULL, NULL); | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	return (0); | 	return (0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -9936,6 +10076,7 @@ zpool_do_get(int argc, char **argv) | |||||||
| 	int ret; | 	int ret; | ||||||
| 	int c, i; | 	int c, i; | ||||||
| 	char *value; | 	char *value; | ||||||
|  | 	char *propstr = NULL; | ||||||
| 
 | 
 | ||||||
| 	cb.cb_first = B_TRUE; | 	cb.cb_first = B_TRUE; | ||||||
| 
 | 
 | ||||||
| @ -9948,6 +10089,8 @@ zpool_do_get(int argc, char **argv) | |||||||
| 	cb.cb_columns[2] = GET_COL_VALUE; | 	cb.cb_columns[2] = GET_COL_VALUE; | ||||||
| 	cb.cb_columns[3] = GET_COL_SOURCE; | 	cb.cb_columns[3] = GET_COL_SOURCE; | ||||||
| 	cb.cb_type = ZFS_TYPE_POOL; | 	cb.cb_type = ZFS_TYPE_POOL; | ||||||
|  | 	cb.cb_vdevs.cb_name_flags |= VDEV_NAME_TYPE_ID; | ||||||
|  | 	current_prop_type = cb.cb_type; | ||||||
| 
 | 
 | ||||||
| 	/* check options */ | 	/* check options */ | ||||||
| 	while ((c = getopt(argc, argv, ":Hpo:")) != -1) { | 	while ((c = getopt(argc, argv, ":Hpo:")) != -1) { | ||||||
| @ -10025,13 +10168,52 @@ zpool_do_get(int argc, char **argv) | |||||||
| 		usage(B_FALSE); | 		usage(B_FALSE); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (zprop_get_list(g_zfs, argv[0], &cb.cb_proplist, | 	/* Properties list is needed later by zprop_get_list() */ | ||||||
| 	    ZFS_TYPE_POOL) != 0) | 	propstr = argv[0]; | ||||||
| 		usage(B_FALSE); |  | ||||||
| 
 | 
 | ||||||
| 	argc--; | 	argc--; | ||||||
| 	argv++; | 	argv++; | ||||||
| 
 | 
 | ||||||
|  | 	if (argc == 0) { | ||||||
|  | 		/* No args, so just print the defaults. */ | ||||||
|  | 	} else if (are_all_pools(argc, argv)) { | ||||||
|  | 		/* All the args are pool names */ | ||||||
|  | 	} else if (are_all_pools(1, argv)) { | ||||||
|  | 		/* The first arg is a pool name */ | ||||||
|  | 		if ((argc == 2 && strcmp(argv[1], "all-vdevs") == 0) || | ||||||
|  | 		    are_vdevs_in_pool(argc - 1, argv + 1, argv[0], | ||||||
|  | 		    &cb.cb_vdevs)) { | ||||||
|  | 			/* ... and the rest are vdev names */ | ||||||
|  | 			cb.cb_vdevs.cb_names = argv + 1; | ||||||
|  | 			cb.cb_vdevs.cb_names_count = argc - 1; | ||||||
|  | 			cb.cb_type = ZFS_TYPE_VDEV; | ||||||
|  | 			argc = 1; /* One pool to process */ | ||||||
|  | 		} else { | ||||||
|  | 			fprintf(stderr, gettext("Expected a list of vdevs in" | ||||||
|  | 			    " \"%s\", but got:\n"), argv[0]); | ||||||
|  | 			error_list_unresolved_vdevs(argc - 1, argv + 1, | ||||||
|  | 			    argv[0], &cb.cb_vdevs); | ||||||
|  | 			fprintf(stderr, "\n"); | ||||||
|  | 			usage(B_FALSE); | ||||||
|  | 			return (1); | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		/*
 | ||||||
|  | 		 * The first arg isn't a pool name, | ||||||
|  | 		 */ | ||||||
|  | 		fprintf(stderr, gettext("missing pool name.\n")); | ||||||
|  | 		fprintf(stderr, "\n"); | ||||||
|  | 		usage(B_FALSE); | ||||||
|  | 		return (1); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (zprop_get_list(g_zfs, propstr, &cb.cb_proplist, | ||||||
|  | 	    cb.cb_type) != 0) { | ||||||
|  | 		/* Use correct list of valid properties (pool or vdev) */ | ||||||
|  | 		current_prop_type = cb.cb_type; | ||||||
|  | 		usage(B_FALSE); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	if (cb.cb_proplist != NULL) { | 	if (cb.cb_proplist != NULL) { | ||||||
| 		fake_name.pl_prop = ZPOOL_PROP_NAME; | 		fake_name.pl_prop = ZPOOL_PROP_NAME; | ||||||
| 		fake_name.pl_width = strlen(gettext("NAME")); | 		fake_name.pl_width = strlen(gettext("NAME")); | ||||||
| @ -10039,8 +10221,8 @@ zpool_do_get(int argc, char **argv) | |||||||
| 		cb.cb_proplist = &fake_name; | 		cb.cb_proplist = &fake_name; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ret = for_each_pool(argc, argv, B_TRUE, &cb.cb_proplist, cb.cb_literal, | 	ret = for_each_pool(argc, argv, B_TRUE, &cb.cb_proplist, cb.cb_type, | ||||||
| 	    get_callback, &cb); | 	    cb.cb_literal, get_callback, &cb); | ||||||
| 
 | 
 | ||||||
| 	if (cb.cb_proplist == &fake_name) | 	if (cb.cb_proplist == &fake_name) | ||||||
| 		zprop_free_list(fake_name.pl_next); | 		zprop_free_list(fake_name.pl_next); | ||||||
| @ -10053,14 +10235,15 @@ zpool_do_get(int argc, char **argv) | |||||||
| typedef struct set_cbdata { | typedef struct set_cbdata { | ||||||
| 	char *cb_propname; | 	char *cb_propname; | ||||||
| 	char *cb_value; | 	char *cb_value; | ||||||
|  | 	zfs_type_t cb_type; | ||||||
|  | 	vdev_cbdata_t cb_vdevs; | ||||||
| 	boolean_t cb_any_successful; | 	boolean_t cb_any_successful; | ||||||
| } set_cbdata_t; | } set_cbdata_t; | ||||||
| 
 | 
 | ||||||
| static int | static int | ||||||
| set_callback(zpool_handle_t *zhp, void *data) | set_pool_callback(zpool_handle_t *zhp, set_cbdata_t *cb) | ||||||
| { | { | ||||||
| 	int error; | 	int error; | ||||||
| 	set_cbdata_t *cb = (set_cbdata_t *)data; |  | ||||||
| 
 | 
 | ||||||
| 	/* Check if we have out-of-bounds features */ | 	/* Check if we have out-of-bounds features */ | ||||||
| 	if (strcmp(cb->cb_propname, ZPOOL_CONFIG_COMPATIBILITY) == 0) { | 	if (strcmp(cb->cb_propname, ZPOOL_CONFIG_COMPATIBILITY) == 0) { | ||||||
| @ -10121,9 +10304,24 @@ set_callback(zpool_handle_t *zhp, void *data) | |||||||
| 
 | 
 | ||||||
| 	error = zpool_set_prop(zhp, cb->cb_propname, cb->cb_value); | 	error = zpool_set_prop(zhp, cb->cb_propname, cb->cb_value); | ||||||
| 
 | 
 | ||||||
| 	if (!error) | 	return (error); | ||||||
| 		cb->cb_any_successful = B_TRUE; | } | ||||||
| 
 | 
 | ||||||
|  | static int | ||||||
|  | set_callback(zpool_handle_t *zhp, void *data) | ||||||
|  | { | ||||||
|  | 	int error; | ||||||
|  | 	set_cbdata_t *cb = (set_cbdata_t *)data; | ||||||
|  | 
 | ||||||
|  | 	if (cb->cb_type == ZFS_TYPE_VDEV) { | ||||||
|  | 		error = zpool_set_vdev_prop(zhp, *cb->cb_vdevs.cb_names, | ||||||
|  | 		    cb->cb_propname, cb->cb_value); | ||||||
|  | 	} else { | ||||||
|  | 		assert(cb->cb_type == ZFS_TYPE_POOL); | ||||||
|  | 		error = set_pool_callback(zhp, cb); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	cb->cb_any_successful = !error; | ||||||
| 	return (error); | 	return (error); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -10133,6 +10331,7 @@ zpool_do_set(int argc, char **argv) | |||||||
| 	set_cbdata_t cb = { 0 }; | 	set_cbdata_t cb = { 0 }; | ||||||
| 	int error; | 	int error; | ||||||
| 
 | 
 | ||||||
|  | 	current_prop_type = ZFS_TYPE_POOL; | ||||||
| 	if (argc > 1 && argv[1][0] == '-') { | 	if (argc > 1 && argv[1][0] == '-') { | ||||||
| 		(void) fprintf(stderr, gettext("invalid option '%c'\n"), | 		(void) fprintf(stderr, gettext("invalid option '%c'\n"), | ||||||
| 		    argv[1][1]); | 		    argv[1][1]); | ||||||
| @ -10150,12 +10349,14 @@ zpool_do_set(int argc, char **argv) | |||||||
| 		usage(B_FALSE); | 		usage(B_FALSE); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (argc > 3) { | 	if (argc > 4) { | ||||||
| 		(void) fprintf(stderr, gettext("too many pool names\n")); | 		(void) fprintf(stderr, gettext("too many pool names\n")); | ||||||
| 		usage(B_FALSE); | 		usage(B_FALSE); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	cb.cb_propname = argv[1]; | 	cb.cb_propname = argv[1]; | ||||||
|  | 	cb.cb_type = ZFS_TYPE_POOL; | ||||||
|  | 	cb.cb_vdevs.cb_name_flags |= VDEV_NAME_TYPE_ID; | ||||||
| 	cb.cb_value = strchr(cb.cb_propname, '='); | 	cb.cb_value = strchr(cb.cb_propname, '='); | ||||||
| 	if (cb.cb_value == NULL) { | 	if (cb.cb_value == NULL) { | ||||||
| 		(void) fprintf(stderr, gettext("missing value in " | 		(void) fprintf(stderr, gettext("missing value in " | ||||||
| @ -10165,9 +10366,33 @@ zpool_do_set(int argc, char **argv) | |||||||
| 
 | 
 | ||||||
| 	*(cb.cb_value) = '\0'; | 	*(cb.cb_value) = '\0'; | ||||||
| 	cb.cb_value++; | 	cb.cb_value++; | ||||||
|  | 	argc -= 2; | ||||||
|  | 	argv += 2; | ||||||
| 
 | 
 | ||||||
| 	error = for_each_pool(argc - 2, argv + 2, B_TRUE, NULL, B_FALSE, | 	if (are_vdevs_in_pool(argc, argv, NULL, &cb.cb_vdevs)) { | ||||||
| 	    set_callback, &cb); | 		/* Argument is a vdev */ | ||||||
|  | 		cb.cb_vdevs.cb_names = argv; | ||||||
|  | 		cb.cb_vdevs.cb_names_count = 1; | ||||||
|  | 		cb.cb_type = ZFS_TYPE_VDEV; | ||||||
|  | 		argc = 0; /* No pools to process */ | ||||||
|  | 	} else if (are_all_pools(1, argv)) { | ||||||
|  | 		/* The first arg is a pool name */ | ||||||
|  | 		if (are_vdevs_in_pool(argc - 1, argv + 1, argv[0], | ||||||
|  | 		    &cb.cb_vdevs)) { | ||||||
|  | 			/* 2nd argument is a vdev */ | ||||||
|  | 			cb.cb_vdevs.cb_names = argv + 1; | ||||||
|  | 			cb.cb_vdevs.cb_names_count = 1; | ||||||
|  | 			cb.cb_type = ZFS_TYPE_VDEV; | ||||||
|  | 			argc = 1; /* One pool to process */ | ||||||
|  | 		} else if (argc > 1) { | ||||||
|  | 			(void) fprintf(stderr, | ||||||
|  | 			    gettext("too many pool names\n")); | ||||||
|  | 			usage(B_FALSE); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	error = for_each_pool(argc, argv, B_TRUE, NULL, ZFS_TYPE_POOL, | ||||||
|  | 	    B_FALSE, set_callback, &cb); | ||||||
| 
 | 
 | ||||||
| 	return (error); | 	return (error); | ||||||
| } | } | ||||||
|  | |||||||
| @ -65,7 +65,7 @@ nvlist_t *split_mirror_vdev(zpool_handle_t *zhp, char *newname, | |||||||
| /*
 | /*
 | ||||||
|  * Pool list functions |  * Pool list functions | ||||||
|  */ |  */ | ||||||
| int for_each_pool(int, char **, boolean_t unavail, zprop_list_t **, | int for_each_pool(int, char **, boolean_t unavail, zprop_list_t **, zfs_type_t, | ||||||
|     boolean_t, zpool_iter_f, void *); |     boolean_t, zpool_iter_f, void *); | ||||||
| 
 | 
 | ||||||
| /* Vdev list functions */ | /* Vdev list functions */ | ||||||
| @ -73,7 +73,8 @@ int for_each_vdev(zpool_handle_t *zhp, pool_vdev_iter_f func, void *data); | |||||||
| 
 | 
 | ||||||
| typedef struct zpool_list zpool_list_t; | typedef struct zpool_list zpool_list_t; | ||||||
| 
 | 
 | ||||||
| zpool_list_t *pool_list_get(int, char **, zprop_list_t **, boolean_t, int *); | zpool_list_t *pool_list_get(int, char **, zprop_list_t **, zfs_type_t, | ||||||
|  |     boolean_t, int *); | ||||||
| void pool_list_update(zpool_list_t *); | void pool_list_update(zpool_list_t *); | ||||||
| int pool_list_iter(zpool_list_t *, int unavail, zpool_iter_f, void *); | int pool_list_iter(zpool_list_t *, int unavail, zpool_iter_f, void *); | ||||||
| void pool_list_free(zpool_list_t *); | void pool_list_free(zpool_list_t *); | ||||||
|  | |||||||
| @ -99,6 +99,7 @@ zfs_errno = enum_with_offset(1024, [ | |||||||
|         'ZFS_ERR_RESILVER_IN_PROGRESS', |         'ZFS_ERR_RESILVER_IN_PROGRESS', | ||||||
|         'ZFS_ERR_REBUILD_IN_PROGRESS', |         'ZFS_ERR_REBUILD_IN_PROGRESS', | ||||||
|         'ZFS_ERR_BADPROP', |         'ZFS_ERR_BADPROP', | ||||||
|  |         'ZFS_ERR_VDEV_NOTSUP', | ||||||
|     ], |     ], | ||||||
|     {} |     {} | ||||||
| ) | ) | ||||||
| @ -110,5 +111,6 @@ ZFS_ERR_NO_CHECKPOINT = zfs_errno.ZFS_ERR_NO_CHECKPOINT | |||||||
| ZFS_ERR_DEVRM_IN_PROGRESS = zfs_errno.ZFS_ERR_DEVRM_IN_PROGRESS | ZFS_ERR_DEVRM_IN_PROGRESS = zfs_errno.ZFS_ERR_DEVRM_IN_PROGRESS | ||||||
| ZFS_ERR_VDEV_TOO_BIG = zfs_errno.ZFS_ERR_VDEV_TOO_BIG | ZFS_ERR_VDEV_TOO_BIG = zfs_errno.ZFS_ERR_VDEV_TOO_BIG | ||||||
| ZFS_ERR_WRONG_PARENT = zfs_errno.ZFS_ERR_WRONG_PARENT | ZFS_ERR_WRONG_PARENT = zfs_errno.ZFS_ERR_WRONG_PARENT | ||||||
|  | ZFS_ERR_VDEV_NOTSUP = zfs_errno.ZFS_ERR_VDEV_NOTSUP | ||||||
| 
 | 
 | ||||||
| # vim: softtabstop=4 tabstop=4 expandtab shiftwidth=4 | # vim: softtabstop=4 tabstop=4 expandtab shiftwidth=4 | ||||||
|  | |||||||
| @ -150,6 +150,7 @@ typedef enum zfs_error { | |||||||
| 	EZFS_NO_RESILVER_DEFER,	/* pool doesn't support resilver_defer */ | 	EZFS_NO_RESILVER_DEFER,	/* pool doesn't support resilver_defer */ | ||||||
| 	EZFS_EXPORT_IN_PROGRESS,	/* currently exporting the pool */ | 	EZFS_EXPORT_IN_PROGRESS,	/* currently exporting the pool */ | ||||||
| 	EZFS_REBUILDING,	/* resilvering (sequential reconstrution) */ | 	EZFS_REBUILDING,	/* resilvering (sequential reconstrution) */ | ||||||
|  | 	EZFS_VDEV_NOTSUP,	/* ops not supported for this type of vdev */ | ||||||
| 	EZFS_UNKNOWN | 	EZFS_UNKNOWN | ||||||
| } zfs_error_t; | } zfs_error_t; | ||||||
| 
 | 
 | ||||||
| @ -336,6 +337,24 @@ _LIBZFS_H int zpool_props_refresh(zpool_handle_t *); | |||||||
| _LIBZFS_H const char *zpool_prop_to_name(zpool_prop_t); | _LIBZFS_H const char *zpool_prop_to_name(zpool_prop_t); | ||||||
| _LIBZFS_H const char *zpool_prop_values(zpool_prop_t); | _LIBZFS_H const char *zpool_prop_values(zpool_prop_t); | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * Functions to manage vdev properties | ||||||
|  |  */ | ||||||
|  | _LIBZFS_H int zpool_get_vdev_prop_value(nvlist_t *, vdev_prop_t, char *, char *, | ||||||
|  |     size_t, zprop_source_t *, boolean_t); | ||||||
|  | _LIBZFS_H int zpool_get_vdev_prop(zpool_handle_t *, const char *, vdev_prop_t, | ||||||
|  |     char *, char *, size_t, zprop_source_t *, boolean_t); | ||||||
|  | _LIBZFS_H int zpool_get_all_vdev_props(zpool_handle_t *, const char *, | ||||||
|  |     nvlist_t **); | ||||||
|  | _LIBZFS_H int zpool_set_vdev_prop(zpool_handle_t *, const char *, const char *, | ||||||
|  |     const char *); | ||||||
|  | 
 | ||||||
|  | _LIBZFS_H const char *vdev_prop_to_name(vdev_prop_t); | ||||||
|  | _LIBZFS_H const char *vdev_prop_values(vdev_prop_t); | ||||||
|  | _LIBZFS_H boolean_t vdev_prop_user(const char *name); | ||||||
|  | _LIBZFS_H const char *vdev_prop_column_name(vdev_prop_t); | ||||||
|  | _LIBZFS_H boolean_t vdev_prop_align_right(vdev_prop_t); | ||||||
|  | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Pool health statistics. |  * Pool health statistics. | ||||||
|  */ |  */ | ||||||
| @ -552,6 +571,8 @@ typedef struct zprop_list { | |||||||
| _LIBZFS_H int zfs_expand_proplist(zfs_handle_t *, zprop_list_t **, boolean_t, | _LIBZFS_H int zfs_expand_proplist(zfs_handle_t *, zprop_list_t **, boolean_t, | ||||||
|     boolean_t); |     boolean_t); | ||||||
| _LIBZFS_H void zfs_prune_proplist(zfs_handle_t *, uint8_t *); | _LIBZFS_H void zfs_prune_proplist(zfs_handle_t *, uint8_t *); | ||||||
|  | _LIBZFS_H int vdev_expand_proplist(zpool_handle_t *, const char *, | ||||||
|  |     zprop_list_t **); | ||||||
| 
 | 
 | ||||||
| #define	ZFS_MOUNTPOINT_NONE	"none" | #define	ZFS_MOUNTPOINT_NONE	"none" | ||||||
| #define	ZFS_MOUNTPOINT_LEGACY	"legacy" | #define	ZFS_MOUNTPOINT_LEGACY	"legacy" | ||||||
| @ -567,7 +588,7 @@ _LIBZFS_H void zfs_prune_proplist(zfs_handle_t *, uint8_t *); | |||||||
|  * zpool property management |  * zpool property management | ||||||
|  */ |  */ | ||||||
| _LIBZFS_H int zpool_expand_proplist(zpool_handle_t *, zprop_list_t **, | _LIBZFS_H int zpool_expand_proplist(zpool_handle_t *, zprop_list_t **, | ||||||
|     boolean_t); |     zfs_type_t, boolean_t); | ||||||
| _LIBZFS_H int zpool_prop_get_feature(zpool_handle_t *, const char *, char *, | _LIBZFS_H int zpool_prop_get_feature(zpool_handle_t *, const char *, char *, | ||||||
|     size_t); |     size_t); | ||||||
| _LIBZFS_H const char *zpool_prop_default_string(zpool_prop_t); | _LIBZFS_H const char *zpool_prop_default_string(zpool_prop_t); | ||||||
| @ -598,6 +619,12 @@ typedef enum { | |||||||
| /*
 | /*
 | ||||||
|  * Functions for printing zfs or zpool properties |  * Functions for printing zfs or zpool properties | ||||||
|  */ |  */ | ||||||
|  | typedef struct vdev_cbdata { | ||||||
|  | 	int cb_name_flags; | ||||||
|  | 	char **cb_names; | ||||||
|  | 	unsigned int cb_names_count; | ||||||
|  | } vdev_cbdata_t; | ||||||
|  | 
 | ||||||
| typedef struct zprop_get_cbdata { | typedef struct zprop_get_cbdata { | ||||||
| 	int cb_sources; | 	int cb_sources; | ||||||
| 	zfs_get_column_t cb_columns[ZFS_GET_NCOLS]; | 	zfs_get_column_t cb_columns[ZFS_GET_NCOLS]; | ||||||
| @ -607,6 +634,7 @@ typedef struct zprop_get_cbdata { | |||||||
| 	boolean_t cb_first; | 	boolean_t cb_first; | ||||||
| 	zprop_list_t *cb_proplist; | 	zprop_list_t *cb_proplist; | ||||||
| 	zfs_type_t cb_type; | 	zfs_type_t cb_type; | ||||||
|  | 	vdev_cbdata_t cb_vdevs; | ||||||
| } zprop_get_cbdata_t; | } zprop_get_cbdata_t; | ||||||
| 
 | 
 | ||||||
| _LIBZFS_H void zprop_print_one_property(const char *, zprop_get_cbdata_t *, | _LIBZFS_H void zprop_print_one_property(const char *, zprop_get_cbdata_t *, | ||||||
| @ -879,7 +907,7 @@ _LIBZFS_H void zfs_commit_shares(const char *); | |||||||
| _LIBZFS_H int zfs_nicestrtonum(libzfs_handle_t *, const char *, uint64_t *); | _LIBZFS_H int zfs_nicestrtonum(libzfs_handle_t *, const char *, uint64_t *); | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Utility functions to run an external process. |  * Utility functions to run an _LIBZFS_Hal process. | ||||||
|  */ |  */ | ||||||
| #define	STDOUT_VERBOSE	0x01 | #define	STDOUT_VERBOSE	0x01 | ||||||
| #define	STDERR_VERBOSE	0x02 | #define	STDERR_VERBOSE	0x02 | ||||||
|  | |||||||
| @ -146,6 +146,10 @@ _LIBZFS_CORE_H int lzc_wait_fs(const char *, zfs_wait_activity_t, boolean_t *); | |||||||
| 
 | 
 | ||||||
| _LIBZFS_CORE_H int lzc_set_bootenv(const char *, const nvlist_t *); | _LIBZFS_CORE_H int lzc_set_bootenv(const char *, const nvlist_t *); | ||||||
| _LIBZFS_CORE_H int lzc_get_bootenv(const char *, nvlist_t **); | _LIBZFS_CORE_H int lzc_get_bootenv(const char *, nvlist_t **); | ||||||
|  | 
 | ||||||
|  | _LIBZFS_CORE_H int lzc_get_vdev_prop(const char *, nvlist_t *, nvlist_t **); | ||||||
|  | _LIBZFS_CORE_H int lzc_set_vdev_prop(const char *, nvlist_t *, nvlist_t **); | ||||||
|  | 
 | ||||||
| #ifdef	__cplusplus | #ifdef	__cplusplus | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
|  | |||||||
| @ -54,7 +54,8 @@ typedef enum { | |||||||
| 	ZFS_TYPE_SNAPSHOT	= (1 << 1), | 	ZFS_TYPE_SNAPSHOT	= (1 << 1), | ||||||
| 	ZFS_TYPE_VOLUME		= (1 << 2), | 	ZFS_TYPE_VOLUME		= (1 << 2), | ||||||
| 	ZFS_TYPE_POOL		= (1 << 3), | 	ZFS_TYPE_POOL		= (1 << 3), | ||||||
| 	ZFS_TYPE_BOOKMARK	= (1 << 4) | 	ZFS_TYPE_BOOKMARK	= (1 << 4), | ||||||
|  | 	ZFS_TYPE_VDEV		= (1 << 5), | ||||||
| } zfs_type_t; | } zfs_type_t; | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
| @ -252,6 +253,7 @@ typedef enum { | |||||||
| 
 | 
 | ||||||
| /* Small enough to not hog a whole line of printout in zpool(8). */ | /* Small enough to not hog a whole line of printout in zpool(8). */ | ||||||
| #define	ZPROP_MAX_COMMENT	32 | #define	ZPROP_MAX_COMMENT	32 | ||||||
|  | #define	ZPROP_BOOLEAN_NA	2 | ||||||
| 
 | 
 | ||||||
| #define	ZPROP_VALUE		"value" | #define	ZPROP_VALUE		"value" | ||||||
| #define	ZPROP_SOURCE		"source" | #define	ZPROP_SOURCE		"source" | ||||||
| @ -298,6 +300,59 @@ typedef int (*zprop_func)(int, void *); | |||||||
|  */ |  */ | ||||||
| #define	ZFS_WRITTEN_PROP_PREFIX_LEN	8 | #define	ZFS_WRITTEN_PROP_PREFIX_LEN	8 | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * VDEV properties are identified by these constants and must be added to the | ||||||
|  |  * end of this list to ensure that external consumers are not affected | ||||||
|  |  * by the change. If you make any changes to this list, be sure to update | ||||||
|  |  * the property table in usr/src/common/zfs/zpool_prop.c. | ||||||
|  |  */ | ||||||
|  | typedef enum { | ||||||
|  | 	VDEV_PROP_INVAL = -1, | ||||||
|  | #define	VDEV_PROP_USER	VDEV_PROP_INVAL | ||||||
|  | 	VDEV_PROP_NAME, | ||||||
|  | 	VDEV_PROP_CAPACITY, | ||||||
|  | 	VDEV_PROP_STATE, | ||||||
|  | 	VDEV_PROP_GUID, | ||||||
|  | 	VDEV_PROP_ASIZE, | ||||||
|  | 	VDEV_PROP_PSIZE, | ||||||
|  | 	VDEV_PROP_ASHIFT, | ||||||
|  | 	VDEV_PROP_SIZE, | ||||||
|  | 	VDEV_PROP_FREE, | ||||||
|  | 	VDEV_PROP_ALLOCATED, | ||||||
|  | 	VDEV_PROP_COMMENT, | ||||||
|  | 	VDEV_PROP_EXPANDSZ, | ||||||
|  | 	VDEV_PROP_FRAGMENTATION, | ||||||
|  | 	VDEV_PROP_BOOTSIZE, | ||||||
|  | 	VDEV_PROP_PARITY, | ||||||
|  | 	VDEV_PROP_PATH, | ||||||
|  | 	VDEV_PROP_DEVID, | ||||||
|  | 	VDEV_PROP_PHYS_PATH, | ||||||
|  | 	VDEV_PROP_ENC_PATH, | ||||||
|  | 	VDEV_PROP_FRU, | ||||||
|  | 	VDEV_PROP_PARENT, | ||||||
|  | 	VDEV_PROP_CHILDREN, | ||||||
|  | 	VDEV_PROP_NUMCHILDREN, | ||||||
|  | 	VDEV_PROP_READ_ERRORS, | ||||||
|  | 	VDEV_PROP_WRITE_ERRORS, | ||||||
|  | 	VDEV_PROP_CHECKSUM_ERRORS, | ||||||
|  | 	VDEV_PROP_INITIALIZE_ERRORS, | ||||||
|  | 	VDEV_PROP_OPS_NULL, | ||||||
|  | 	VDEV_PROP_OPS_READ, | ||||||
|  | 	VDEV_PROP_OPS_WRITE, | ||||||
|  | 	VDEV_PROP_OPS_FREE, | ||||||
|  | 	VDEV_PROP_OPS_CLAIM, | ||||||
|  | 	VDEV_PROP_OPS_TRIM, | ||||||
|  | 	VDEV_PROP_BYTES_NULL, | ||||||
|  | 	VDEV_PROP_BYTES_READ, | ||||||
|  | 	VDEV_PROP_BYTES_WRITE, | ||||||
|  | 	VDEV_PROP_BYTES_FREE, | ||||||
|  | 	VDEV_PROP_BYTES_CLAIM, | ||||||
|  | 	VDEV_PROP_BYTES_TRIM, | ||||||
|  | 	VDEV_PROP_REMOVING, | ||||||
|  | 	VDEV_PROP_ALLOCATING, | ||||||
|  | 	VDEV_NUM_PROPS | ||||||
|  | } vdev_prop_t; | ||||||
|  | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Dataset property functions shared between libzfs and kernel. |  * Dataset property functions shared between libzfs and kernel. | ||||||
|  */ |  */ | ||||||
| @ -337,6 +392,22 @@ _SYS_FS_ZFS_H int zpool_prop_string_to_index(zpool_prop_t, const char *, | |||||||
|     uint64_t *); |     uint64_t *); | ||||||
| _SYS_FS_ZFS_H uint64_t zpool_prop_random_value(zpool_prop_t, uint64_t seed); | _SYS_FS_ZFS_H uint64_t zpool_prop_random_value(zpool_prop_t, uint64_t seed); | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * VDEV property functions shared between libzfs and kernel. | ||||||
|  |  */ | ||||||
|  | _SYS_FS_ZFS_H vdev_prop_t vdev_name_to_prop(const char *); | ||||||
|  | _SYS_FS_ZFS_H boolean_t vdev_prop_user(const char *name); | ||||||
|  | _SYS_FS_ZFS_H const char *vdev_prop_to_name(vdev_prop_t); | ||||||
|  | _SYS_FS_ZFS_H const char *vdev_prop_default_string(vdev_prop_t); | ||||||
|  | _SYS_FS_ZFS_H uint64_t vdev_prop_default_numeric(vdev_prop_t); | ||||||
|  | _SYS_FS_ZFS_H boolean_t vdev_prop_readonly(vdev_prop_t prop); | ||||||
|  | _SYS_FS_ZFS_H int vdev_prop_index_to_string(vdev_prop_t, uint64_t, | ||||||
|  |     const char **); | ||||||
|  | _SYS_FS_ZFS_H int vdev_prop_string_to_index(vdev_prop_t, const char *, | ||||||
|  |     uint64_t *); | ||||||
|  | _SYS_FS_ZFS_H boolean_t zpool_prop_vdev(const char *name); | ||||||
|  | _SYS_FS_ZFS_H uint64_t vdev_prop_random_value(vdev_prop_t prop, uint64_t seed); | ||||||
|  | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Definitions for the Delegation. |  * Definitions for the Delegation. | ||||||
|  */ |  */ | ||||||
| @ -712,6 +783,7 @@ typedef struct zpool_load_policy { | |||||||
| #define	ZPOOL_CONFIG_ORIG_GUID		"orig_guid" | #define	ZPOOL_CONFIG_ORIG_GUID		"orig_guid" | ||||||
| #define	ZPOOL_CONFIG_SPLIT_GUID		"split_guid" | #define	ZPOOL_CONFIG_SPLIT_GUID		"split_guid" | ||||||
| #define	ZPOOL_CONFIG_SPLIT_LIST		"guid_list" | #define	ZPOOL_CONFIG_SPLIT_LIST		"guid_list" | ||||||
|  | #define	ZPOOL_CONFIG_NONALLOCATING	"non_allocating" | ||||||
| #define	ZPOOL_CONFIG_REMOVING		"removing" | #define	ZPOOL_CONFIG_REMOVING		"removing" | ||||||
| #define	ZPOOL_CONFIG_RESILVER_TXG	"resilver_txg" | #define	ZPOOL_CONFIG_RESILVER_TXG	"resilver_txg" | ||||||
| #define	ZPOOL_CONFIG_REBUILD_TXG	"rebuild_txg" | #define	ZPOOL_CONFIG_REBUILD_TXG	"rebuild_txg" | ||||||
| @ -1109,6 +1181,7 @@ typedef struct vdev_stat { | |||||||
| 	uint64_t	vs_configured_ashift;   /* TLV vdev_ashift */ | 	uint64_t	vs_configured_ashift;   /* TLV vdev_ashift */ | ||||||
| 	uint64_t	vs_logical_ashift;	/* vdev_logical_ashift  */ | 	uint64_t	vs_logical_ashift;	/* vdev_logical_ashift  */ | ||||||
| 	uint64_t	vs_physical_ashift;	/* vdev_physical_ashift */ | 	uint64_t	vs_physical_ashift;	/* vdev_physical_ashift */ | ||||||
|  | 	uint64_t	vs_noalloc;		/* allocations halted?	*/ | ||||||
| } vdev_stat_t; | } vdev_stat_t; | ||||||
| 
 | 
 | ||||||
| /* BEGIN CSTYLED */ | /* BEGIN CSTYLED */ | ||||||
| @ -1362,6 +1435,8 @@ typedef enum zfs_ioc { | |||||||
| 	ZFS_IOC_GET_BOOKMARK_PROPS,		/* 0x5a52 */ | 	ZFS_IOC_GET_BOOKMARK_PROPS,		/* 0x5a52 */ | ||||||
| 	ZFS_IOC_WAIT,				/* 0x5a53 */ | 	ZFS_IOC_WAIT,				/* 0x5a53 */ | ||||||
| 	ZFS_IOC_WAIT_FS,			/* 0x5a54 */ | 	ZFS_IOC_WAIT_FS,			/* 0x5a54 */ | ||||||
|  | 	ZFS_IOC_VDEV_GET_PROPS,			/* 0x5a55 */ | ||||||
|  | 	ZFS_IOC_VDEV_SET_PROPS,			/* 0x5a56 */ | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Per-platform (Optional) - 8/128 numbers reserved. | 	 * Per-platform (Optional) - 8/128 numbers reserved. | ||||||
| @ -1417,6 +1492,7 @@ typedef enum { | |||||||
| 	ZFS_ERR_RESILVER_IN_PROGRESS, | 	ZFS_ERR_RESILVER_IN_PROGRESS, | ||||||
| 	ZFS_ERR_REBUILD_IN_PROGRESS, | 	ZFS_ERR_REBUILD_IN_PROGRESS, | ||||||
| 	ZFS_ERR_BADPROP, | 	ZFS_ERR_BADPROP, | ||||||
|  | 	ZFS_ERR_VDEV_NOTSUP, | ||||||
| } zfs_errno_t; | } zfs_errno_t; | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
| @ -1508,6 +1584,18 @@ typedef enum { | |||||||
| #define	ZPOOL_WAIT_TAG			"wait_tag" | #define	ZPOOL_WAIT_TAG			"wait_tag" | ||||||
| #define	ZPOOL_WAIT_WAITED		"wait_waited" | #define	ZPOOL_WAIT_WAITED		"wait_waited" | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * The following are names used when invoking ZFS_IOC_VDEV_GET_PROP. | ||||||
|  |  */ | ||||||
|  | #define	ZPOOL_VDEV_PROPS_GET_VDEV	"vdevprops_get_vdev" | ||||||
|  | #define	ZPOOL_VDEV_PROPS_GET_PROPS	"vdevprops_get_props" | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * The following are names used when invoking ZFS_IOC_VDEV_SET_PROP. | ||||||
|  |  */ | ||||||
|  | #define	ZPOOL_VDEV_PROPS_SET_VDEV	"vdevprops_set_vdev" | ||||||
|  | #define	ZPOOL_VDEV_PROPS_SET_PROPS	"vdevprops_set_props" | ||||||
|  | 
 | ||||||
| /*
 | /*
 | ||||||
|  * The following are names used when invoking ZFS_IOC_WAIT_FS. |  * The following are names used when invoking ZFS_IOC_WAIT_FS. | ||||||
|  */ |  */ | ||||||
|  | |||||||
| @ -792,7 +792,8 @@ extern int spa_vdev_attach(spa_t *spa, uint64_t guid, nvlist_t *nvroot, | |||||||
|     int replacing, int rebuild); |     int replacing, int rebuild); | ||||||
| extern int spa_vdev_detach(spa_t *spa, uint64_t guid, uint64_t pguid, | extern int spa_vdev_detach(spa_t *spa, uint64_t guid, uint64_t pguid, | ||||||
|     int replace_done); |     int replace_done); | ||||||
| extern int spa_vdev_remove(spa_t *spa, uint64_t guid, boolean_t unspare); | extern int spa_vdev_alloc(spa_t *spa, uint64_t guid); | ||||||
|  | extern int spa_vdev_noalloc(spa_t *spa, uint64_t guid); | ||||||
| extern boolean_t spa_vdev_remove_active(spa_t *spa); | extern boolean_t spa_vdev_remove_active(spa_t *spa); | ||||||
| extern int spa_vdev_initialize(spa_t *spa, nvlist_t *nv, uint64_t cmd_type, | extern int spa_vdev_initialize(spa_t *spa, nvlist_t *nv, uint64_t cmd_type, | ||||||
|     nvlist_t *vdev_errlist); |     nvlist_t *vdev_errlist); | ||||||
|  | |||||||
| @ -308,6 +308,7 @@ struct spa { | |||||||
| 	uint64_t	spa_missing_tvds;	/* unopenable tvds on load */ | 	uint64_t	spa_missing_tvds;	/* unopenable tvds on load */ | ||||||
| 	uint64_t	spa_missing_tvds_allowed; /* allow loading spa? */ | 	uint64_t	spa_missing_tvds_allowed; /* allow loading spa? */ | ||||||
| 
 | 
 | ||||||
|  | 	uint64_t	spa_nonallocating_dspace; | ||||||
| 	spa_removing_phys_t spa_removing_phys; | 	spa_removing_phys_t spa_removing_phys; | ||||||
| 	spa_vdev_removal_t *spa_vdev_removal; | 	spa_vdev_removal_t *spa_vdev_removal; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -219,6 +219,9 @@ typedef enum { | |||||||
| 
 | 
 | ||||||
| extern int vdev_label_init(vdev_t *vd, uint64_t txg, vdev_labeltype_t reason); | extern int vdev_label_init(vdev_t *vd, uint64_t txg, vdev_labeltype_t reason); | ||||||
| 
 | 
 | ||||||
|  | extern int vdev_prop_set(vdev_t *vd, nvlist_t *innvl, nvlist_t *outnvl); | ||||||
|  | extern int vdev_prop_get(vdev_t *vd, nvlist_t *nvprops, nvlist_t *outnvl); | ||||||
|  | 
 | ||||||
| #ifdef	__cplusplus | #ifdef	__cplusplus | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
|  | |||||||
| @ -295,6 +295,7 @@ struct vdev { | |||||||
| 	list_node_t	vdev_state_dirty_node; /* state dirty list	*/ | 	list_node_t	vdev_state_dirty_node; /* state dirty list	*/ | ||||||
| 	uint64_t	vdev_deflate_ratio; /* deflation ratio (x512)	*/ | 	uint64_t	vdev_deflate_ratio; /* deflation ratio (x512)	*/ | ||||||
| 	uint64_t	vdev_islog;	/* is an intent log device	*/ | 	uint64_t	vdev_islog;	/* is an intent log device	*/ | ||||||
|  | 	uint64_t	vdev_noalloc;	/* device is passivated?	*/ | ||||||
| 	uint64_t	vdev_removing;	/* device is being removed?	*/ | 	uint64_t	vdev_removing;	/* device is being removed?	*/ | ||||||
| 	boolean_t	vdev_ishole;	/* is a hole in the namespace	*/ | 	boolean_t	vdev_ishole;	/* is a hole in the namespace	*/ | ||||||
| 	uint64_t	vdev_top_zap; | 	uint64_t	vdev_top_zap; | ||||||
|  | |||||||
| @ -39,6 +39,7 @@ _SYS_ZFS_SYSFS_H boolean_t zfs_mod_supported(const char *, const char *); | |||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| #define	ZFS_SYSFS_POOL_PROPERTIES	"properties.pool" | #define	ZFS_SYSFS_POOL_PROPERTIES	"properties.pool" | ||||||
|  | #define	ZFS_SYSFS_VDEV_PROPERTIES	"properties.vdev" | ||||||
| #define	ZFS_SYSFS_DATASET_PROPERTIES	"properties.dataset" | #define	ZFS_SYSFS_DATASET_PROPERTIES	"properties.dataset" | ||||||
| #define	ZFS_SYSFS_KERNEL_FEATURES	"features.kernel" | #define	ZFS_SYSFS_KERNEL_FEATURES	"features.kernel" | ||||||
| #define	ZFS_SYSFS_POOL_FEATURES		"features.pool" | #define	ZFS_SYSFS_POOL_FEATURES		"features.pool" | ||||||
|  | |||||||
| @ -99,6 +99,13 @@ _ZFS_PROP_H void zpool_prop_init(void); | |||||||
| _ZFS_PROP_H zprop_type_t zpool_prop_get_type(zpool_prop_t); | _ZFS_PROP_H zprop_type_t zpool_prop_get_type(zpool_prop_t); | ||||||
| _ZFS_PROP_H zprop_desc_t *zpool_prop_get_table(void); | _ZFS_PROP_H zprop_desc_t *zpool_prop_get_table(void); | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * vdev property functions | ||||||
|  |  */ | ||||||
|  | _ZFS_PROP_H void vdev_prop_init(void); | ||||||
|  | _ZFS_PROP_H zprop_type_t vdev_prop_get_type(vdev_prop_t prop); | ||||||
|  | _ZFS_PROP_H zprop_desc_t *vdev_prop_get_table(void); | ||||||
|  | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Common routines to initialize property tables |  * Common routines to initialize property tables | ||||||
|  */ |  */ | ||||||
| @ -122,11 +129,13 @@ _ZFS_PROP_H int zprop_iter_common(zprop_func, void *, boolean_t, boolean_t, | |||||||
| _ZFS_PROP_H int zprop_name_to_prop(const char *, zfs_type_t); | _ZFS_PROP_H int zprop_name_to_prop(const char *, zfs_type_t); | ||||||
| _ZFS_PROP_H int zprop_string_to_index(int, const char *, uint64_t *, | _ZFS_PROP_H int zprop_string_to_index(int, const char *, uint64_t *, | ||||||
|     zfs_type_t); |     zfs_type_t); | ||||||
| _ZFS_PROP_H int zprop_index_to_string(int, uint64_t, const char **, zfs_type_t); | _ZFS_PROP_H int zprop_index_to_string(int, uint64_t, const char **, | ||||||
|  |     zfs_type_t); | ||||||
| _ZFS_PROP_H uint64_t zprop_random_value(int, uint64_t, zfs_type_t); | _ZFS_PROP_H uint64_t zprop_random_value(int, uint64_t, zfs_type_t); | ||||||
| _ZFS_PROP_H const char *zprop_values(int, zfs_type_t); | _ZFS_PROP_H const char *zprop_values(int, zfs_type_t); | ||||||
| _ZFS_PROP_H size_t zprop_width(int, boolean_t *, zfs_type_t); | _ZFS_PROP_H size_t zprop_width(int, boolean_t *, zfs_type_t); | ||||||
| _ZFS_PROP_H boolean_t zprop_valid_for_type(int, zfs_type_t, boolean_t); | _ZFS_PROP_H boolean_t zprop_valid_for_type(int, zfs_type_t, boolean_t); | ||||||
|  | _ZFS_PROP_H int zprop_valid_char(char c); | ||||||
| 
 | 
 | ||||||
| #ifdef	__cplusplus | #ifdef	__cplusplus | ||||||
| } | } | ||||||
|  | |||||||
| @ -259,6 +259,22 @@ | |||||||
|     <elf-symbol name='tpool_suspended' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='tpool_suspended' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='tpool_wait' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='tpool_wait' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='update_vdev_config_dev_strs' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='update_vdev_config_dev_strs' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_expand_proplist' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_name_to_prop' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_align_right' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_column_name' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_default_numeric' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_default_string' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_get_table' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_get_type' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_index_to_string' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_init' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_random_value' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_readonly' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_string_to_index' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_to_name' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_user' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='vdev_prop_values' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zfeature_depends_on' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zfeature_depends_on' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zfeature_is_supported' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zfeature_is_supported' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zfeature_is_valid_guid' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zfeature_is_valid_guid' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
| @ -463,6 +479,7 @@ | |||||||
|     <elf-symbol name='zpool_find_vdev' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_find_vdev' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_find_vdev_by_physpath' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_find_vdev_by_physpath' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_free_handles' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_free_handles' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='zpool_get_all_vdev_props' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_get_bootenv' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_get_bootenv' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_get_config' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_get_config' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_get_errlog' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_get_errlog' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
| @ -477,6 +494,8 @@ | |||||||
|     <elf-symbol name='zpool_get_state' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_get_state' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_get_state_str' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_get_state_str' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_get_status' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_get_status' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='zpool_get_vdev_prop' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='zpool_get_vdev_prop_value' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_history_unpack' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_history_unpack' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_import' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_import' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_import_props' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_import_props' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
| @ -514,6 +533,7 @@ | |||||||
|     <elf-symbol name='zpool_prop_to_name' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_prop_to_name' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_prop_unsupported' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_prop_unsupported' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_prop_values' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_prop_values' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='zpool_prop_vdev' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_props_refresh' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_props_refresh' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_read_label' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_read_label' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_refresh_stats' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_refresh_stats' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
| @ -523,6 +543,7 @@ | |||||||
|     <elf-symbol name='zpool_search_import' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_search_import' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_set_bootenv' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_set_bootenv' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_set_prop' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_set_prop' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='zpool_set_vdev_prop' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_skip_pool' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_skip_pool' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_state_to_name' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_state_to_name' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zpool_sync_one' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zpool_sync_one' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
| @ -557,6 +578,7 @@ | |||||||
|     <elf-symbol name='zprop_register_number' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zprop_register_number' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zprop_register_string' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zprop_register_string' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zprop_string_to_index' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zprop_string_to_index' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='zprop_valid_char' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zprop_valid_for_type' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zprop_valid_for_type' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zprop_values' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zprop_values' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='zprop_width' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='zprop_width' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
| @ -2683,6 +2705,73 @@ | |||||||
|       <parameter type-id='5d0c23fb' name='prop'/> |       <parameter type-id='5d0c23fb' name='prop'/> | ||||||
|       <return type-id='c19b74c3'/> |       <return type-id='c19b74c3'/> | ||||||
|     </function-decl> |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_get_table' mangled-name='vdev_prop_get_table' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_get_table'> | ||||||
|  |       <return type-id='76c8174b'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_init' mangled-name='vdev_prop_init' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_init'> | ||||||
|  |       <return type-id='48b5725f'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_name_to_prop' mangled-name='vdev_name_to_prop' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_name_to_prop'> | ||||||
|  |       <parameter type-id='80f4b756' name='propname'/> | ||||||
|  |       <return type-id='5aa5c90c'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_user' mangled-name='vdev_prop_user' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_user'> | ||||||
|  |       <parameter type-id='80f4b756' name='name'/> | ||||||
|  |       <return type-id='c19b74c3'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_to_name' mangled-name='vdev_prop_to_name' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_to_name'> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <return type-id='80f4b756'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_get_type' mangled-name='vdev_prop_get_type' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_get_type'> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <return type-id='31429eff'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_readonly' mangled-name='vdev_prop_readonly' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_readonly'> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <return type-id='c19b74c3'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_default_string' mangled-name='vdev_prop_default_string' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_default_string'> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <return type-id='80f4b756'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_default_numeric' mangled-name='vdev_prop_default_numeric' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_default_numeric'> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <return type-id='9c313c2d'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_string_to_index' mangled-name='vdev_prop_string_to_index' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_string_to_index'> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <parameter type-id='80f4b756' name='string'/> | ||||||
|  |       <parameter type-id='5d6479ae' name='index'/> | ||||||
|  |       <return type-id='95e97e5e'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_index_to_string' mangled-name='vdev_prop_index_to_string' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_index_to_string'> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <parameter type-id='9c313c2d' name='index'/> | ||||||
|  |       <parameter type-id='7d3cd834' name='string'/> | ||||||
|  |       <return type-id='95e97e5e'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='zpool_prop_vdev' mangled-name='zpool_prop_vdev' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zpool_prop_vdev'> | ||||||
|  |       <parameter type-id='80f4b756' name='name'/> | ||||||
|  |       <return type-id='c19b74c3'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_random_value' mangled-name='vdev_prop_random_value' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_random_value'> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <parameter type-id='9c313c2d' name='seed'/> | ||||||
|  |       <return type-id='9c313c2d'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_values' mangled-name='vdev_prop_values' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_values'> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <return type-id='80f4b756'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_column_name' mangled-name='vdev_prop_column_name' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_column_name'> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <return type-id='80f4b756'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='vdev_prop_align_right' mangled-name='vdev_prop_align_right' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_prop_align_right'> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <return type-id='c19b74c3'/> | ||||||
|  |     </function-decl> | ||||||
|   </abi-instr> |   </abi-instr> | ||||||
|   <abi-instr address-size='64' path='../../module/zcommon/zprop_common.c' language='LANG_C99'> |   <abi-instr address-size='64' path='../../module/zcommon/zprop_common.c' language='LANG_C99'> | ||||||
|     <function-decl name='zprop_register_impl' mangled-name='zprop_register_impl' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zprop_register_impl'> |     <function-decl name='zprop_register_impl' mangled-name='zprop_register_impl' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zprop_register_impl'> | ||||||
| @ -2784,6 +2873,10 @@ | |||||||
|       <parameter type-id='c19b74c3' name='headcheck'/> |       <parameter type-id='c19b74c3' name='headcheck'/> | ||||||
|       <return type-id='c19b74c3'/> |       <return type-id='c19b74c3'/> | ||||||
|     </function-decl> |     </function-decl> | ||||||
|  |     <function-decl name='zprop_valid_char' mangled-name='zprop_valid_char' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zprop_valid_char'> | ||||||
|  |       <parameter type-id='a84c031d' name='c'/> | ||||||
|  |       <return type-id='95e97e5e'/> | ||||||
|  |     </function-decl> | ||||||
|     <function-decl name='zprop_width' mangled-name='zprop_width' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zprop_width'> |     <function-decl name='zprop_width' mangled-name='zprop_width' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zprop_width'> | ||||||
|       <parameter type-id='95e97e5e' name='prop'/> |       <parameter type-id='95e97e5e' name='prop'/> | ||||||
|       <parameter type-id='37e3bd22' name='fixed'/> |       <parameter type-id='37e3bd22' name='fixed'/> | ||||||
| @ -2886,6 +2979,7 @@ | |||||||
|       <enumerator name='ZFS_TYPE_VOLUME' value='4'/> |       <enumerator name='ZFS_TYPE_VOLUME' value='4'/> | ||||||
|       <enumerator name='ZFS_TYPE_POOL' value='8'/> |       <enumerator name='ZFS_TYPE_POOL' value='8'/> | ||||||
|       <enumerator name='ZFS_TYPE_BOOKMARK' value='16'/> |       <enumerator name='ZFS_TYPE_BOOKMARK' value='16'/> | ||||||
|  |       <enumerator name='ZFS_TYPE_VDEV' value='32'/> | ||||||
|     </enum-decl> |     </enum-decl> | ||||||
|     <typedef-decl name='zfs_type_t' type-id='5d8f7321' id='2e45de5d'/> |     <typedef-decl name='zfs_type_t' type-id='5d8f7321' id='2e45de5d'/> | ||||||
|     <enum-decl name='dmu_objset_type' id='6b1b19f9'> |     <enum-decl name='dmu_objset_type' id='6b1b19f9'> | ||||||
| @ -4216,6 +4310,53 @@ | |||||||
|       <enumerator name='ZPOOL_NUM_PROPS' value='33'/> |       <enumerator name='ZPOOL_NUM_PROPS' value='33'/> | ||||||
|     </enum-decl> |     </enum-decl> | ||||||
|     <typedef-decl name='zpool_prop_t' type-id='af1ba157' id='5d0c23fb'/> |     <typedef-decl name='zpool_prop_t' type-id='af1ba157' id='5d0c23fb'/> | ||||||
|  |     <enum-decl name='vdev_prop_t' naming-typedef-id='5aa5c90c' id='1573bec8'> | ||||||
|  |       <underlying-type type-id='9cac1fee'/> | ||||||
|  |       <enumerator name='VDEV_PROP_INVAL' value='-1'/> | ||||||
|  |       <enumerator name='VDEV_PROP_NAME' value='0'/> | ||||||
|  |       <enumerator name='VDEV_PROP_CAPACITY' value='1'/> | ||||||
|  |       <enumerator name='VDEV_PROP_STATE' value='2'/> | ||||||
|  |       <enumerator name='VDEV_PROP_GUID' value='3'/> | ||||||
|  |       <enumerator name='VDEV_PROP_ASIZE' value='4'/> | ||||||
|  |       <enumerator name='VDEV_PROP_PSIZE' value='5'/> | ||||||
|  |       <enumerator name='VDEV_PROP_ASHIFT' value='6'/> | ||||||
|  |       <enumerator name='VDEV_PROP_SIZE' value='7'/> | ||||||
|  |       <enumerator name='VDEV_PROP_FREE' value='8'/> | ||||||
|  |       <enumerator name='VDEV_PROP_ALLOCATED' value='9'/> | ||||||
|  |       <enumerator name='VDEV_PROP_COMMENT' value='10'/> | ||||||
|  |       <enumerator name='VDEV_PROP_EXPANDSZ' value='11'/> | ||||||
|  |       <enumerator name='VDEV_PROP_FRAGMENTATION' value='12'/> | ||||||
|  |       <enumerator name='VDEV_PROP_BOOTSIZE' value='13'/> | ||||||
|  |       <enumerator name='VDEV_PROP_PARITY' value='14'/> | ||||||
|  |       <enumerator name='VDEV_PROP_PATH' value='15'/> | ||||||
|  |       <enumerator name='VDEV_PROP_DEVID' value='16'/> | ||||||
|  |       <enumerator name='VDEV_PROP_PHYS_PATH' value='17'/> | ||||||
|  |       <enumerator name='VDEV_PROP_ENC_PATH' value='18'/> | ||||||
|  |       <enumerator name='VDEV_PROP_FRU' value='19'/> | ||||||
|  |       <enumerator name='VDEV_PROP_PARENT' value='20'/> | ||||||
|  |       <enumerator name='VDEV_PROP_CHILDREN' value='21'/> | ||||||
|  |       <enumerator name='VDEV_PROP_NUMCHILDREN' value='22'/> | ||||||
|  |       <enumerator name='VDEV_PROP_READ_ERRORS' value='23'/> | ||||||
|  |       <enumerator name='VDEV_PROP_WRITE_ERRORS' value='24'/> | ||||||
|  |       <enumerator name='VDEV_PROP_CHECKSUM_ERRORS' value='25'/> | ||||||
|  |       <enumerator name='VDEV_PROP_INITIALIZE_ERRORS' value='26'/> | ||||||
|  |       <enumerator name='VDEV_PROP_OPS_NULL' value='27'/> | ||||||
|  |       <enumerator name='VDEV_PROP_OPS_READ' value='28'/> | ||||||
|  |       <enumerator name='VDEV_PROP_OPS_WRITE' value='29'/> | ||||||
|  |       <enumerator name='VDEV_PROP_OPS_FREE' value='30'/> | ||||||
|  |       <enumerator name='VDEV_PROP_OPS_CLAIM' value='31'/> | ||||||
|  |       <enumerator name='VDEV_PROP_OPS_TRIM' value='32'/> | ||||||
|  |       <enumerator name='VDEV_PROP_BYTES_NULL' value='33'/> | ||||||
|  |       <enumerator name='VDEV_PROP_BYTES_READ' value='34'/> | ||||||
|  |       <enumerator name='VDEV_PROP_BYTES_WRITE' value='35'/> | ||||||
|  |       <enumerator name='VDEV_PROP_BYTES_FREE' value='36'/> | ||||||
|  |       <enumerator name='VDEV_PROP_BYTES_CLAIM' value='37'/> | ||||||
|  |       <enumerator name='VDEV_PROP_BYTES_TRIM' value='38'/> | ||||||
|  |       <enumerator name='VDEV_PROP_REMOVING' value='39'/> | ||||||
|  |       <enumerator name='VDEV_PROP_ALLOCATING' value='40'/> | ||||||
|  |       <enumerator name='VDEV_NUM_PROPS' value='41'/> | ||||||
|  |     </enum-decl> | ||||||
|  |     <typedef-decl name='vdev_prop_t' type-id='1573bec8' id='5aa5c90c'/> | ||||||
|     <enum-decl name='vdev_state' id='21566197'> |     <enum-decl name='vdev_state' id='21566197'> | ||||||
|       <underlying-type type-id='9cac1fee'/> |       <underlying-type type-id='9cac1fee'/> | ||||||
|       <enumerator name='VDEV_STATE_UNKNOWN' value='0'/> |       <enumerator name='VDEV_STATE_UNKNOWN' value='0'/> | ||||||
| @ -4342,9 +4483,16 @@ | |||||||
|     <function-decl name='zpool_expand_proplist' mangled-name='zpool_expand_proplist' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zpool_expand_proplist'> |     <function-decl name='zpool_expand_proplist' mangled-name='zpool_expand_proplist' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zpool_expand_proplist'> | ||||||
|       <parameter type-id='4c81de99' name='zhp'/> |       <parameter type-id='4c81de99' name='zhp'/> | ||||||
|       <parameter type-id='e4378506' name='plp'/> |       <parameter type-id='e4378506' name='plp'/> | ||||||
|  |       <parameter type-id='2e45de5d' name='type'/> | ||||||
|       <parameter type-id='c19b74c3' name='literal'/> |       <parameter type-id='c19b74c3' name='literal'/> | ||||||
|       <return type-id='95e97e5e'/> |       <return type-id='95e97e5e'/> | ||||||
|     </function-decl> |     </function-decl> | ||||||
|  |     <function-decl name='vdev_expand_proplist' mangled-name='vdev_expand_proplist' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='vdev_expand_proplist'> | ||||||
|  |       <parameter type-id='4c81de99' name='zhp'/> | ||||||
|  |       <parameter type-id='80f4b756' name='vdevname'/> | ||||||
|  |       <parameter type-id='e4378506' name='plp'/> | ||||||
|  |       <return type-id='95e97e5e'/> | ||||||
|  |     </function-decl> | ||||||
|     <function-decl name='zpool_prop_get_feature' mangled-name='zpool_prop_get_feature' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zpool_prop_get_feature'> |     <function-decl name='zpool_prop_get_feature' mangled-name='zpool_prop_get_feature' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zpool_prop_get_feature'> | ||||||
|       <parameter type-id='4c81de99' name='zhp'/> |       <parameter type-id='4c81de99' name='zhp'/> | ||||||
|       <parameter type-id='80f4b756' name='propname'/> |       <parameter type-id='80f4b756' name='propname'/> | ||||||
| @ -4680,6 +4828,40 @@ | |||||||
|       <parameter type-id='b59d7dce' name='rlen'/> |       <parameter type-id='b59d7dce' name='rlen'/> | ||||||
|       <return type-id='901b78d1'/> |       <return type-id='901b78d1'/> | ||||||
|     </function-decl> |     </function-decl> | ||||||
|  |     <function-decl name='zpool_get_vdev_prop_value' mangled-name='zpool_get_vdev_prop_value' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zpool_get_vdev_prop_value'> | ||||||
|  |       <parameter type-id='5ce45b60' name='nvprop'/> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <parameter type-id='26a90f95' name='prop_name'/> | ||||||
|  |       <parameter type-id='26a90f95' name='buf'/> | ||||||
|  |       <parameter type-id='b59d7dce' name='len'/> | ||||||
|  |       <parameter type-id='debc6aa3' name='srctype'/> | ||||||
|  |       <parameter type-id='c19b74c3' name='literal'/> | ||||||
|  |       <return type-id='95e97e5e'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='zpool_get_vdev_prop' mangled-name='zpool_get_vdev_prop' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zpool_get_vdev_prop'> | ||||||
|  |       <parameter type-id='4c81de99' name='zhp'/> | ||||||
|  |       <parameter type-id='80f4b756' name='vdevname'/> | ||||||
|  |       <parameter type-id='5aa5c90c' name='prop'/> | ||||||
|  |       <parameter type-id='26a90f95' name='prop_name'/> | ||||||
|  |       <parameter type-id='26a90f95' name='buf'/> | ||||||
|  |       <parameter type-id='b59d7dce' name='len'/> | ||||||
|  |       <parameter type-id='debc6aa3' name='srctype'/> | ||||||
|  |       <parameter type-id='c19b74c3' name='literal'/> | ||||||
|  |       <return type-id='95e97e5e'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='zpool_get_all_vdev_props' mangled-name='zpool_get_all_vdev_props' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zpool_get_all_vdev_props'> | ||||||
|  |       <parameter type-id='4c81de99' name='zhp'/> | ||||||
|  |       <parameter type-id='80f4b756' name='vdevname'/> | ||||||
|  |       <parameter type-id='857bb57e' name='outnvl'/> | ||||||
|  |       <return type-id='95e97e5e'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='zpool_set_vdev_prop' mangled-name='zpool_set_vdev_prop' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='zpool_set_vdev_prop'> | ||||||
|  |       <parameter type-id='4c81de99' name='zhp'/> | ||||||
|  |       <parameter type-id='80f4b756' name='vdevname'/> | ||||||
|  |       <parameter type-id='80f4b756' name='propname'/> | ||||||
|  |       <parameter type-id='80f4b756' name='propval'/> | ||||||
|  |       <return type-id='95e97e5e'/> | ||||||
|  |     </function-decl> | ||||||
|   </abi-instr> |   </abi-instr> | ||||||
|   <abi-instr address-size='64' path='libzfs_sendrecv.c' language='LANG_C99'> |   <abi-instr address-size='64' path='libzfs_sendrecv.c' language='LANG_C99'> | ||||||
|     <class-decl name='sendflags' size-in-bits='544' is-struct='yes' visibility='default' id='f6aa15be'> |     <class-decl name='sendflags' size-in-bits='544' is-struct='yes' visibility='default' id='f6aa15be'> | ||||||
| @ -4922,7 +5104,19 @@ | |||||||
|       <enumerator name='GET_COL_SOURCE' value='5'/> |       <enumerator name='GET_COL_SOURCE' value='5'/> | ||||||
|     </enum-decl> |     </enum-decl> | ||||||
|     <typedef-decl name='zfs_get_column_t' type-id='223bdcaa' id='19cefcee'/> |     <typedef-decl name='zfs_get_column_t' type-id='223bdcaa' id='19cefcee'/> | ||||||
|     <class-decl name='zprop_get_cbdata' size-in-bits='640' is-struct='yes' visibility='default' id='f3d3c319'> |     <class-decl name='vdev_cbdata' size-in-bits='192' is-struct='yes' visibility='default' id='b8006be8'> | ||||||
|  |       <data-member access='public' layout-offset-in-bits='0'> | ||||||
|  |         <var-decl name='cb_name_flags' type-id='95e97e5e' visibility='default'/> | ||||||
|  |       </data-member> | ||||||
|  |       <data-member access='public' layout-offset-in-bits='64'> | ||||||
|  |         <var-decl name='cb_names' type-id='9b23c9ad' visibility='default'/> | ||||||
|  |       </data-member> | ||||||
|  |       <data-member access='public' layout-offset-in-bits='128'> | ||||||
|  |         <var-decl name='cb_names_count' type-id='f0981eeb' visibility='default'/> | ||||||
|  |       </data-member> | ||||||
|  |     </class-decl> | ||||||
|  |     <typedef-decl name='vdev_cbdata_t' type-id='b8006be8' id='a9679c94'/> | ||||||
|  |     <class-decl name='zprop_get_cbdata' size-in-bits='832' is-struct='yes' visibility='default' id='f3d3c319'> | ||||||
|       <data-member access='public' layout-offset-in-bits='0'> |       <data-member access='public' layout-offset-in-bits='0'> | ||||||
|         <var-decl name='cb_sources' type-id='95e97e5e' visibility='default'/> |         <var-decl name='cb_sources' type-id='95e97e5e' visibility='default'/> | ||||||
|       </data-member> |       </data-member> | ||||||
| @ -4947,6 +5141,9 @@ | |||||||
|       <data-member access='public' layout-offset-in-bits='576'> |       <data-member access='public' layout-offset-in-bits='576'> | ||||||
|         <var-decl name='cb_type' type-id='2e45de5d' visibility='default'/> |         <var-decl name='cb_type' type-id='2e45de5d' visibility='default'/> | ||||||
|       </data-member> |       </data-member> | ||||||
|  |       <data-member access='public' layout-offset-in-bits='640'> | ||||||
|  |         <var-decl name='cb_vdevs' type-id='a9679c94' visibility='default'/> | ||||||
|  |       </data-member> | ||||||
|     </class-decl> |     </class-decl> | ||||||
|     <typedef-decl name='zprop_get_cbdata_t' type-id='f3d3c319' id='f3d87113'/> |     <typedef-decl name='zprop_get_cbdata_t' type-id='f3d3c319' id='f3d87113'/> | ||||||
|     <typedef-decl name='zprop_func' type-id='2e711a2a' id='1ec3747a'/> |     <typedef-decl name='zprop_func' type-id='2e711a2a' id='1ec3747a'/> | ||||||
|  | |||||||
| @ -29,6 +29,7 @@ | |||||||
|  * Copyright (c) 2017, Intel Corporation. |  * Copyright (c) 2017, Intel Corporation. | ||||||
|  * Copyright (c) 2018, loli10K <ezomori.nozomu@gmail.com> |  * Copyright (c) 2018, loli10K <ezomori.nozomu@gmail.com> | ||||||
|  * Copyright (c) 2021, Colm Buckley <colm@tuatha.org> |  * Copyright (c) 2021, Colm Buckley <colm@tuatha.org> | ||||||
|  |  * Copyright (c) 2021, Klara Inc. | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| #include <errno.h> | #include <errno.h> | ||||||
| @ -61,6 +62,7 @@ static boolean_t zpool_vdev_is_interior(const char *name); | |||||||
| typedef struct prop_flags { | typedef struct prop_flags { | ||||||
| 	int create:1;	/* Validate property on creation */ | 	int create:1;	/* Validate property on creation */ | ||||||
| 	int import:1;	/* Validate property on import */ | 	int import:1;	/* Validate property on import */ | ||||||
|  | 	int vdevprop:1;	/* Validate property as a VDEV property */ | ||||||
| } prop_flags_t; | } prop_flags_t; | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
| @ -478,6 +480,35 @@ zpool_valid_proplist(libzfs_handle_t *hdl, const char *poolname, | |||||||
| 	while ((elem = nvlist_next_nvpair(props, elem)) != NULL) { | 	while ((elem = nvlist_next_nvpair(props, elem)) != NULL) { | ||||||
| 		const char *propname = nvpair_name(elem); | 		const char *propname = nvpair_name(elem); | ||||||
| 
 | 
 | ||||||
|  | 		if (flags.vdevprop && zpool_prop_vdev(propname)) { | ||||||
|  | 			vdev_prop_t vprop = vdev_name_to_prop(propname); | ||||||
|  | 
 | ||||||
|  | 			if (vdev_prop_readonly(vprop)) { | ||||||
|  | 				zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, "'%s' " | ||||||
|  | 				    "is readonly"), propname); | ||||||
|  | 				(void) zfs_error(hdl, EZFS_PROPREADONLY, | ||||||
|  | 				    errbuf); | ||||||
|  | 				goto error; | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			if (zprop_parse_value(hdl, elem, vprop, ZFS_TYPE_VDEV, | ||||||
|  | 			    retprops, &strval, &intval, errbuf) != 0) | ||||||
|  | 				goto error; | ||||||
|  | 
 | ||||||
|  | 			continue; | ||||||
|  | 		} else if (flags.vdevprop && vdev_prop_user(propname)) { | ||||||
|  | 			if (nvlist_add_nvpair(retprops, elem) != 0) { | ||||||
|  | 				(void) no_memory(hdl); | ||||||
|  | 				goto error; | ||||||
|  | 			} | ||||||
|  | 			continue; | ||||||
|  | 		} else if (flags.vdevprop) { | ||||||
|  | 			zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, | ||||||
|  | 			    "invalid property: '%s'"), propname); | ||||||
|  | 			(void) zfs_error(hdl, EZFS_BADPROP, errbuf); | ||||||
|  | 			goto error; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		prop = zpool_name_to_prop(propname); | 		prop = zpool_name_to_prop(propname); | ||||||
| 		if (prop == ZPOOL_PROP_INVAL && zpool_prop_feature(propname)) { | 		if (prop == ZPOOL_PROP_INVAL && zpool_prop_feature(propname)) { | ||||||
| 			int err; | 			int err; | ||||||
| @ -806,7 +837,7 @@ zpool_set_prop(zpool_handle_t *zhp, const char *propname, const char *propval) | |||||||
| 
 | 
 | ||||||
| int | int | ||||||
| zpool_expand_proplist(zpool_handle_t *zhp, zprop_list_t **plp, | zpool_expand_proplist(zpool_handle_t *zhp, zprop_list_t **plp, | ||||||
|     boolean_t literal) |     zfs_type_t type, boolean_t literal) | ||||||
| { | { | ||||||
| 	libzfs_handle_t *hdl = zhp->zpool_hdl; | 	libzfs_handle_t *hdl = zhp->zpool_hdl; | ||||||
| 	zprop_list_t *entry; | 	zprop_list_t *entry; | ||||||
| @ -817,9 +848,12 @@ zpool_expand_proplist(zpool_handle_t *zhp, zprop_list_t **plp, | |||||||
| 	boolean_t firstexpand = (NULL == *plp); | 	boolean_t firstexpand = (NULL == *plp); | ||||||
| 	int i; | 	int i; | ||||||
| 
 | 
 | ||||||
| 	if (zprop_expand_list(hdl, plp, ZFS_TYPE_POOL) != 0) | 	if (zprop_expand_list(hdl, plp, type) != 0) | ||||||
| 		return (-1); | 		return (-1); | ||||||
| 
 | 
 | ||||||
|  | 	if (type == ZFS_TYPE_VDEV) | ||||||
|  | 		return (0); | ||||||
|  | 
 | ||||||
| 	last = plp; | 	last = plp; | ||||||
| 	while (*last != NULL) | 	while (*last != NULL) | ||||||
| 		last = &(*last)->pl_next; | 		last = &(*last)->pl_next; | ||||||
| @ -899,6 +933,77 @@ zpool_expand_proplist(zpool_handle_t *zhp, zprop_list_t **plp, | |||||||
| 	return (0); | 	return (0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | int | ||||||
|  | vdev_expand_proplist(zpool_handle_t *zhp, const char *vdevname, | ||||||
|  |     zprop_list_t **plp) | ||||||
|  | { | ||||||
|  | 	zprop_list_t *entry; | ||||||
|  | 	char buf[ZFS_MAXPROPLEN]; | ||||||
|  | 	char *strval = NULL; | ||||||
|  | 	int err = 0; | ||||||
|  | 	nvpair_t *elem = NULL; | ||||||
|  | 	nvlist_t *vprops = NULL; | ||||||
|  | 	nvlist_t *propval = NULL; | ||||||
|  | 	const char *propname; | ||||||
|  | 	vdev_prop_t prop; | ||||||
|  | 	zprop_list_t **last; | ||||||
|  | 
 | ||||||
|  | 	for (entry = *plp; entry != NULL; entry = entry->pl_next) { | ||||||
|  | 		if (entry->pl_fixed) | ||||||
|  | 			continue; | ||||||
|  | 
 | ||||||
|  | 		if (zpool_get_vdev_prop(zhp, vdevname, entry->pl_prop, | ||||||
|  | 		    entry->pl_user_prop, buf, sizeof (buf), NULL, | ||||||
|  | 		    B_FALSE) == 0) { | ||||||
|  | 			if (strlen(buf) > entry->pl_width) | ||||||
|  | 				entry->pl_width = strlen(buf); | ||||||
|  | 		} | ||||||
|  | 		if (entry->pl_prop == VDEV_PROP_NAME && | ||||||
|  | 		    strlen(vdevname) > entry->pl_width) | ||||||
|  | 			entry->pl_width = strlen(vdevname); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Handle the all properties case */ | ||||||
|  | 	last = plp; | ||||||
|  | 	if (*last != NULL && (*last)->pl_all == B_TRUE) { | ||||||
|  | 		while (*last != NULL) | ||||||
|  | 			last = &(*last)->pl_next; | ||||||
|  | 
 | ||||||
|  | 		err = zpool_get_all_vdev_props(zhp, vdevname, &vprops); | ||||||
|  | 		if (err != 0) | ||||||
|  | 			return (err); | ||||||
|  | 
 | ||||||
|  | 		while ((elem = nvlist_next_nvpair(vprops, elem)) != NULL) { | ||||||
|  | 			propname = nvpair_name(elem); | ||||||
|  | 
 | ||||||
|  | 			/* Skip properties that are not user defined */ | ||||||
|  | 			if ((prop = vdev_name_to_prop(propname)) != | ||||||
|  | 			    VDEV_PROP_USER) | ||||||
|  | 				continue; | ||||||
|  | 
 | ||||||
|  | 			if (nvpair_value_nvlist(elem, &propval) != 0) | ||||||
|  | 				continue; | ||||||
|  | 
 | ||||||
|  | 			verify(nvlist_lookup_string(propval, ZPROP_VALUE, | ||||||
|  | 			    &strval) == 0); | ||||||
|  | 
 | ||||||
|  | 			if ((entry = zfs_alloc(zhp->zpool_hdl, | ||||||
|  | 			    sizeof (zprop_list_t))) == NULL) | ||||||
|  | 				return (ENOMEM); | ||||||
|  | 
 | ||||||
|  | 			entry->pl_prop = prop; | ||||||
|  | 			entry->pl_user_prop = zfs_strdup(zhp->zpool_hdl, | ||||||
|  | 			    propname); | ||||||
|  | 			entry->pl_width = strlen(strval); | ||||||
|  | 			entry->pl_all = B_TRUE; | ||||||
|  | 			*last = entry; | ||||||
|  | 			last = &entry->pl_next; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return (0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Get the state for the given feature on the given ZFS pool. |  * Get the state for the given feature on the given ZFS pool. | ||||||
|  */ |  */ | ||||||
| @ -4959,3 +5064,353 @@ zpool_load_compat(const char *compat, boolean_t *features, char *report, | |||||||
| 		strlcpy(report, gettext("compatibility set ok"), rlen); | 		strlcpy(report, gettext("compatibility set ok"), rlen); | ||||||
| 	return (ZPOOL_COMPATIBILITY_OK); | 	return (ZPOOL_COMPATIBILITY_OK); | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | zpool_vdev_guid(zpool_handle_t *zhp, const char *vdevname, uint64_t *vdev_guid) | ||||||
|  | { | ||||||
|  | 	nvlist_t *tgt; | ||||||
|  | 	boolean_t avail_spare, l2cache; | ||||||
|  | 
 | ||||||
|  | 	verify(zhp != NULL); | ||||||
|  | 	if (zpool_get_state(zhp) == POOL_STATE_UNAVAIL) { | ||||||
|  | 		char errbuf[1024]; | ||||||
|  | 		(void) snprintf(errbuf, sizeof (errbuf), | ||||||
|  | 		    dgettext(TEXT_DOMAIN, "pool is in an unavailable state")); | ||||||
|  | 		return (zfs_error(zhp->zpool_hdl, EZFS_POOLUNAVAIL, errbuf)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if ((tgt = zpool_find_vdev(zhp, vdevname, &avail_spare, &l2cache, | ||||||
|  | 	    NULL)) == NULL) { | ||||||
|  | 		char errbuf[1024]; | ||||||
|  | 		(void) snprintf(errbuf, sizeof (errbuf), | ||||||
|  | 		    dgettext(TEXT_DOMAIN, "can not find %s in %s"), | ||||||
|  | 		    vdevname, zhp->zpool_name); | ||||||
|  | 		return (zfs_error(zhp->zpool_hdl, EZFS_NODEVICE, errbuf)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	verify(nvlist_lookup_uint64(tgt, ZPOOL_CONFIG_GUID, vdev_guid) == 0); | ||||||
|  | 	return (0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Get a vdev property value for 'prop' and return the value in | ||||||
|  |  * a pre-allocated buffer. | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | zpool_get_vdev_prop_value(nvlist_t *nvprop, vdev_prop_t prop, char *prop_name, | ||||||
|  |     char *buf, size_t len, zprop_source_t *srctype, boolean_t literal) | ||||||
|  | { | ||||||
|  | 	nvlist_t *nv; | ||||||
|  | 	uint64_t intval; | ||||||
|  | 	char *strval; | ||||||
|  | 	zprop_source_t src = ZPROP_SRC_NONE; | ||||||
|  | 
 | ||||||
|  | 	if (prop == VDEV_PROP_USER) { | ||||||
|  | 		/* user property, prop_name must contain the property name */ | ||||||
|  | 		assert(prop_name != NULL); | ||||||
|  | 		if (nvlist_lookup_nvlist(nvprop, prop_name, &nv) == 0) { | ||||||
|  | 			verify(nvlist_lookup_uint64(nv, ZPROP_SOURCE, | ||||||
|  | 			    &intval) == 0); | ||||||
|  | 			src = intval; | ||||||
|  | 			verify(nvlist_lookup_string(nv, ZPROP_VALUE, | ||||||
|  | 			    &strval) == 0); | ||||||
|  | 		} else { | ||||||
|  | 			/* user prop not found */ | ||||||
|  | 			return (-1); | ||||||
|  | 		} | ||||||
|  | 		(void) strlcpy(buf, strval, len); | ||||||
|  | 		if (srctype) | ||||||
|  | 			*srctype = src; | ||||||
|  | 		return (0); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (prop_name == NULL) | ||||||
|  | 		prop_name = (char *)vdev_prop_to_name(prop); | ||||||
|  | 
 | ||||||
|  | 	switch (vdev_prop_get_type(prop)) { | ||||||
|  | 	case PROP_TYPE_STRING: | ||||||
|  | 		if (nvlist_lookup_nvlist(nvprop, prop_name, &nv) == 0) { | ||||||
|  | 			verify(nvlist_lookup_uint64(nv, ZPROP_SOURCE, | ||||||
|  | 			    &intval) == 0); | ||||||
|  | 			src = intval; | ||||||
|  | 			verify(nvlist_lookup_string(nv, ZPROP_VALUE, | ||||||
|  | 			    &strval) == 0); | ||||||
|  | 		} else { | ||||||
|  | 			src = ZPROP_SRC_DEFAULT; | ||||||
|  | 			if ((strval = (char *)vdev_prop_default_string(prop)) | ||||||
|  | 			    == NULL) | ||||||
|  | 				strval = "-"; | ||||||
|  | 		} | ||||||
|  | 		(void) strlcpy(buf, strval, len); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case PROP_TYPE_NUMBER: | ||||||
|  | 		if (nvlist_lookup_nvlist(nvprop, prop_name, &nv) == 0) { | ||||||
|  | 			verify(nvlist_lookup_uint64(nv, ZPROP_SOURCE, | ||||||
|  | 			    &intval) == 0); | ||||||
|  | 			src = intval; | ||||||
|  | 			verify(nvlist_lookup_uint64(nv, ZPROP_VALUE, | ||||||
|  | 			    &intval) == 0); | ||||||
|  | 		} else { | ||||||
|  | 			src = ZPROP_SRC_DEFAULT; | ||||||
|  | 			intval = vdev_prop_default_numeric(prop); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		switch (prop) { | ||||||
|  | 		case VDEV_PROP_ASIZE: | ||||||
|  | 		case VDEV_PROP_PSIZE: | ||||||
|  | 		case VDEV_PROP_SIZE: | ||||||
|  | 		case VDEV_PROP_ALLOCATED: | ||||||
|  | 		case VDEV_PROP_FREE: | ||||||
|  | 		case VDEV_PROP_READ_ERRORS: | ||||||
|  | 		case VDEV_PROP_WRITE_ERRORS: | ||||||
|  | 		case VDEV_PROP_CHECKSUM_ERRORS: | ||||||
|  | 		case VDEV_PROP_INITIALIZE_ERRORS: | ||||||
|  | 		case VDEV_PROP_OPS_NULL: | ||||||
|  | 		case VDEV_PROP_OPS_READ: | ||||||
|  | 		case VDEV_PROP_OPS_WRITE: | ||||||
|  | 		case VDEV_PROP_OPS_FREE: | ||||||
|  | 		case VDEV_PROP_OPS_CLAIM: | ||||||
|  | 		case VDEV_PROP_OPS_TRIM: | ||||||
|  | 		case VDEV_PROP_BYTES_NULL: | ||||||
|  | 		case VDEV_PROP_BYTES_READ: | ||||||
|  | 		case VDEV_PROP_BYTES_WRITE: | ||||||
|  | 		case VDEV_PROP_BYTES_FREE: | ||||||
|  | 		case VDEV_PROP_BYTES_CLAIM: | ||||||
|  | 		case VDEV_PROP_BYTES_TRIM: | ||||||
|  | 			if (literal) { | ||||||
|  | 				(void) snprintf(buf, len, "%llu", | ||||||
|  | 				    (u_longlong_t)intval); | ||||||
|  | 			} else { | ||||||
|  | 				(void) zfs_nicenum(intval, buf, len); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		case VDEV_PROP_EXPANDSZ: | ||||||
|  | 			if (intval == 0) { | ||||||
|  | 				(void) strlcpy(buf, "-", len); | ||||||
|  | 			} else if (literal) { | ||||||
|  | 				(void) snprintf(buf, len, "%llu", | ||||||
|  | 				    (u_longlong_t)intval); | ||||||
|  | 			} else { | ||||||
|  | 				(void) zfs_nicenum(intval, buf, len); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		case VDEV_PROP_CAPACITY: | ||||||
|  | 			if (literal) { | ||||||
|  | 				(void) snprintf(buf, len, "%llu", | ||||||
|  | 				    (u_longlong_t)intval); | ||||||
|  | 			} else { | ||||||
|  | 				(void) snprintf(buf, len, "%llu%%", | ||||||
|  | 				    (u_longlong_t)intval); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		case VDEV_PROP_FRAGMENTATION: | ||||||
|  | 			if (intval == UINT64_MAX) { | ||||||
|  | 				(void) strlcpy(buf, "-", len); | ||||||
|  | 			} else { | ||||||
|  | 				(void) snprintf(buf, len, "%llu%%", | ||||||
|  | 				    (u_longlong_t)intval); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		case VDEV_PROP_STATE: | ||||||
|  | 			if (literal) { | ||||||
|  | 				(void) snprintf(buf, len, "%llu", | ||||||
|  | 				    (u_longlong_t)intval); | ||||||
|  | 			} else { | ||||||
|  | 				(void) strlcpy(buf, zpool_state_to_name(intval, | ||||||
|  | 				    VDEV_AUX_NONE), len); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		default: | ||||||
|  | 			(void) snprintf(buf, len, "%llu", | ||||||
|  | 			    (u_longlong_t)intval); | ||||||
|  | 		} | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	case PROP_TYPE_INDEX: | ||||||
|  | 		if (nvlist_lookup_nvlist(nvprop, prop_name, &nv) == 0) { | ||||||
|  | 			verify(nvlist_lookup_uint64(nv, ZPROP_SOURCE, | ||||||
|  | 			    &intval) == 0); | ||||||
|  | 			src = intval; | ||||||
|  | 			verify(nvlist_lookup_uint64(nv, ZPROP_VALUE, | ||||||
|  | 			    &intval) == 0); | ||||||
|  | 		} else { | ||||||
|  | 			src = ZPROP_SRC_DEFAULT; | ||||||
|  | 			intval = vdev_prop_default_numeric(prop); | ||||||
|  | 		} | ||||||
|  | 		if (vdev_prop_index_to_string(prop, intval, | ||||||
|  | 		    (const char **)&strval) != 0) | ||||||
|  | 			return (-1); | ||||||
|  | 		(void) strlcpy(buf, strval, len); | ||||||
|  | 		break; | ||||||
|  | 
 | ||||||
|  | 	default: | ||||||
|  | 		abort(); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (srctype) | ||||||
|  | 		*srctype = src; | ||||||
|  | 
 | ||||||
|  | 	return (0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Get a vdev property value for 'prop_name' and return the value in | ||||||
|  |  * a pre-allocated buffer. | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | zpool_get_vdev_prop(zpool_handle_t *zhp, const char *vdevname, vdev_prop_t prop, | ||||||
|  |     char *prop_name, char *buf, size_t len, zprop_source_t *srctype, | ||||||
|  |     boolean_t literal) | ||||||
|  | { | ||||||
|  | 	nvlist_t *reqnvl, *reqprops; | ||||||
|  | 	nvlist_t *retprops = NULL; | ||||||
|  | 	uint64_t vdev_guid; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	if ((ret = zpool_vdev_guid(zhp, vdevname, &vdev_guid)) != 0) | ||||||
|  | 		return (ret); | ||||||
|  | 
 | ||||||
|  | 	if (nvlist_alloc(&reqnvl, NV_UNIQUE_NAME, 0) != 0) | ||||||
|  | 		return (no_memory(zhp->zpool_hdl)); | ||||||
|  | 	if (nvlist_alloc(&reqprops, NV_UNIQUE_NAME, 0) != 0) | ||||||
|  | 		return (no_memory(zhp->zpool_hdl)); | ||||||
|  | 
 | ||||||
|  | 	fnvlist_add_uint64(reqnvl, ZPOOL_VDEV_PROPS_GET_VDEV, vdev_guid); | ||||||
|  | 
 | ||||||
|  | 	if (prop != VDEV_PROP_USER) { | ||||||
|  | 		/* prop_name overrides prop value */ | ||||||
|  | 		if (prop_name != NULL) | ||||||
|  | 			prop = vdev_name_to_prop(prop_name); | ||||||
|  | 		else | ||||||
|  | 			prop_name = (char *)vdev_prop_to_name(prop); | ||||||
|  | 		assert(prop < VDEV_NUM_PROPS); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	assert(prop_name != NULL); | ||||||
|  | 	if (nvlist_add_uint64(reqprops, prop_name, prop) != 0) { | ||||||
|  | 		nvlist_free(reqnvl); | ||||||
|  | 		nvlist_free(reqprops); | ||||||
|  | 		return (no_memory(zhp->zpool_hdl)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fnvlist_add_nvlist(reqnvl, ZPOOL_VDEV_PROPS_GET_PROPS, reqprops); | ||||||
|  | 
 | ||||||
|  | 	ret = lzc_get_vdev_prop(zhp->zpool_name, reqnvl, &retprops); | ||||||
|  | 
 | ||||||
|  | 	if (ret == 0) { | ||||||
|  | 		ret = zpool_get_vdev_prop_value(retprops, prop, prop_name, buf, | ||||||
|  | 		    len, srctype, literal); | ||||||
|  | 	} else { | ||||||
|  | 		char errbuf[1024]; | ||||||
|  | 		(void) snprintf(errbuf, sizeof (errbuf), | ||||||
|  | 		    dgettext(TEXT_DOMAIN, "cannot get vdev property %s from" | ||||||
|  | 		    " %s in %s"), prop_name, vdevname, zhp->zpool_name); | ||||||
|  | 		(void) zpool_standard_error(zhp->zpool_hdl, ret, errbuf); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	nvlist_free(reqnvl); | ||||||
|  | 	nvlist_free(reqprops); | ||||||
|  | 	nvlist_free(retprops); | ||||||
|  | 
 | ||||||
|  | 	return (ret); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Get all vdev properties | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | zpool_get_all_vdev_props(zpool_handle_t *zhp, const char *vdevname, | ||||||
|  |     nvlist_t **outnvl) | ||||||
|  | { | ||||||
|  | 	nvlist_t *nvl = NULL; | ||||||
|  | 	uint64_t vdev_guid; | ||||||
|  | 	int ret; | ||||||
|  | 
 | ||||||
|  | 	if ((ret = zpool_vdev_guid(zhp, vdevname, &vdev_guid)) != 0) | ||||||
|  | 		return (ret); | ||||||
|  | 
 | ||||||
|  | 	if (nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0) != 0) | ||||||
|  | 		return (no_memory(zhp->zpool_hdl)); | ||||||
|  | 
 | ||||||
|  | 	fnvlist_add_uint64(nvl, ZPOOL_VDEV_PROPS_GET_VDEV, vdev_guid); | ||||||
|  | 
 | ||||||
|  | 	ret = lzc_get_vdev_prop(zhp->zpool_name, nvl, outnvl); | ||||||
|  | 
 | ||||||
|  | 	nvlist_free(nvl); | ||||||
|  | 
 | ||||||
|  | 	if (ret) { | ||||||
|  | 		char errbuf[1024]; | ||||||
|  | 		(void) snprintf(errbuf, sizeof (errbuf), | ||||||
|  | 		    dgettext(TEXT_DOMAIN, "cannot get vdev properties for" | ||||||
|  | 		    " %s in %s"), vdevname, zhp->zpool_name); | ||||||
|  | 		(void) zpool_standard_error(zhp->zpool_hdl, errno, errbuf); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return (ret); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Set vdev property | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | zpool_set_vdev_prop(zpool_handle_t *zhp, const char *vdevname, | ||||||
|  |     const char *propname, const char *propval) | ||||||
|  | { | ||||||
|  | 	int ret; | ||||||
|  | 	vdev_prop_t vprop; | ||||||
|  | 	nvlist_t *nvl = NULL; | ||||||
|  | 	nvlist_t *outnvl = NULL; | ||||||
|  | 	nvlist_t *props; | ||||||
|  | 	nvlist_t *realprops; | ||||||
|  | 	prop_flags_t flags = { 0 }; | ||||||
|  | 	uint64_t version; | ||||||
|  | 	uint64_t vdev_guid; | ||||||
|  | 
 | ||||||
|  | 	if ((ret = zpool_vdev_guid(zhp, vdevname, &vdev_guid)) != 0) | ||||||
|  | 		return (ret); | ||||||
|  | 
 | ||||||
|  | 	vprop = vdev_name_to_prop(propname); | ||||||
|  | 
 | ||||||
|  | 	if (nvlist_alloc(&nvl, NV_UNIQUE_NAME, 0) != 0) | ||||||
|  | 		return (no_memory(zhp->zpool_hdl)); | ||||||
|  | 	if (nvlist_alloc(&props, NV_UNIQUE_NAME, 0) != 0) | ||||||
|  | 		return (no_memory(zhp->zpool_hdl)); | ||||||
|  | 
 | ||||||
|  | 	fnvlist_add_uint64(nvl, ZPOOL_VDEV_PROPS_SET_VDEV, vdev_guid); | ||||||
|  | 
 | ||||||
|  | 	if (nvlist_add_string(props, propname, propval) != 0) { | ||||||
|  | 		nvlist_free(props); | ||||||
|  | 		return (no_memory(zhp->zpool_hdl)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	char errbuf[1024]; | ||||||
|  | 	(void) snprintf(errbuf, sizeof (errbuf), | ||||||
|  | 	    dgettext(TEXT_DOMAIN, "cannot set property %s for %s on %s"), | ||||||
|  | 	    propname, vdevname, zhp->zpool_name); | ||||||
|  | 
 | ||||||
|  | 	flags.vdevprop = 1; | ||||||
|  | 	version = zpool_get_prop_int(zhp, ZPOOL_PROP_VERSION, NULL); | ||||||
|  | 	if ((realprops = zpool_valid_proplist(zhp->zpool_hdl, | ||||||
|  | 	    zhp->zpool_name, props, version, flags, errbuf)) == NULL) { | ||||||
|  | 		nvlist_free(props); | ||||||
|  | 		nvlist_free(nvl); | ||||||
|  | 		return (-1); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	nvlist_free(props); | ||||||
|  | 	props = realprops; | ||||||
|  | 
 | ||||||
|  | 	fnvlist_add_nvlist(nvl, ZPOOL_VDEV_PROPS_SET_PROPS, props); | ||||||
|  | 
 | ||||||
|  | 	ret = lzc_set_vdev_prop(zhp->zpool_name, nvl, &outnvl); | ||||||
|  | 
 | ||||||
|  | 	nvlist_free(props); | ||||||
|  | 	nvlist_free(nvl); | ||||||
|  | 	nvlist_free(outnvl); | ||||||
|  | 
 | ||||||
|  | 	if (ret) | ||||||
|  | 		(void) zpool_standard_error(zhp->zpool_hdl, errno, errbuf); | ||||||
|  | 
 | ||||||
|  | 	return (ret); | ||||||
|  | } | ||||||
|  | |||||||
| @ -296,6 +296,9 @@ libzfs_error_description(libzfs_handle_t *hdl) | |||||||
| 	case EZFS_REBUILDING: | 	case EZFS_REBUILDING: | ||||||
| 		return (dgettext(TEXT_DOMAIN, "currently sequentially " | 		return (dgettext(TEXT_DOMAIN, "currently sequentially " | ||||||
| 		    "resilvering")); | 		    "resilvering")); | ||||||
|  | 	case EZFS_VDEV_NOTSUP: | ||||||
|  | 		return (dgettext(TEXT_DOMAIN, "operation not supported " | ||||||
|  | 		    "on this type of vdev")); | ||||||
| 	case EZFS_UNKNOWN: | 	case EZFS_UNKNOWN: | ||||||
| 		return (dgettext(TEXT_DOMAIN, "unknown error")); | 		return (dgettext(TEXT_DOMAIN, "unknown error")); | ||||||
| 	default: | 	default: | ||||||
| @ -716,6 +719,9 @@ zpool_standard_error_fmt(libzfs_handle_t *hdl, int error, const char *fmt, ...) | |||||||
| 	case ZFS_ERR_BADPROP: | 	case ZFS_ERR_BADPROP: | ||||||
| 		zfs_verror(hdl, EZFS_BADPROP, fmt, ap); | 		zfs_verror(hdl, EZFS_BADPROP, fmt, ap); | ||||||
| 		break; | 		break; | ||||||
|  | 	case ZFS_ERR_VDEV_NOTSUP: | ||||||
|  | 		zfs_verror(hdl, EZFS_VDEV_NOTSUP, fmt, ap); | ||||||
|  | 		break; | ||||||
| 	case ZFS_ERR_IOC_CMD_UNAVAIL: | 	case ZFS_ERR_IOC_CMD_UNAVAIL: | ||||||
| 		zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, "the loaded zfs " | 		zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, "the loaded zfs " | ||||||
| 		    "module does not support this operation. A reboot may " | 		    "module does not support this operation. A reboot may " | ||||||
| @ -1034,6 +1040,7 @@ libzfs_init(void) | |||||||
| 	zfs_prop_init(); | 	zfs_prop_init(); | ||||||
| 	zpool_prop_init(); | 	zpool_prop_init(); | ||||||
| 	zpool_feature_init(); | 	zpool_feature_init(); | ||||||
|  | 	vdev_prop_init(); | ||||||
| 	libzfs_mnttab_init(hdl); | 	libzfs_mnttab_init(hdl); | ||||||
| 	fletcher_4_init(); | 	fletcher_4_init(); | ||||||
| 
 | 
 | ||||||
| @ -1267,7 +1274,8 @@ zprop_print_headers(zprop_get_cbdata_t *cbp, zfs_type_t type) | |||||||
| 
 | 
 | ||||||
| 	/* first property is always NAME */ | 	/* first property is always NAME */ | ||||||
| 	assert(cbp->cb_proplist->pl_prop == | 	assert(cbp->cb_proplist->pl_prop == | ||||||
| 	    ((type == ZFS_TYPE_POOL) ?  ZPOOL_PROP_NAME : ZFS_PROP_NAME)); | 	    ((type == ZFS_TYPE_POOL) ? ZPOOL_PROP_NAME : | ||||||
|  | 	    ((type == ZFS_TYPE_VDEV) ? VDEV_PROP_NAME : ZFS_PROP_NAME))); | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Go through and calculate the widths for each column.  For the | 	 * Go through and calculate the widths for each column.  For the | ||||||
| @ -1284,12 +1292,16 @@ zprop_print_headers(zprop_get_cbdata_t *cbp, zfs_type_t type) | |||||||
| 		if (pl->pl_prop != ZPROP_INVAL) { | 		if (pl->pl_prop != ZPROP_INVAL) { | ||||||
| 			const char *propname = (type == ZFS_TYPE_POOL) ? | 			const char *propname = (type == ZFS_TYPE_POOL) ? | ||||||
| 			    zpool_prop_to_name(pl->pl_prop) : | 			    zpool_prop_to_name(pl->pl_prop) : | ||||||
| 			    zfs_prop_to_name(pl->pl_prop); | 			    ((type == ZFS_TYPE_VDEV) ? | ||||||
|  | 			    vdev_prop_to_name(pl->pl_prop) : | ||||||
|  | 			    zfs_prop_to_name(pl->pl_prop)); | ||||||
| 
 | 
 | ||||||
|  | 			assert(propname != NULL); | ||||||
| 			len = strlen(propname); | 			len = strlen(propname); | ||||||
| 			if (len > cbp->cb_colwidths[GET_COL_PROPERTY]) | 			if (len > cbp->cb_colwidths[GET_COL_PROPERTY]) | ||||||
| 				cbp->cb_colwidths[GET_COL_PROPERTY] = len; | 				cbp->cb_colwidths[GET_COL_PROPERTY] = len; | ||||||
| 		} else { | 		} else { | ||||||
|  | 			assert(pl->pl_user_prop != NULL); | ||||||
| 			len = strlen(pl->pl_user_prop); | 			len = strlen(pl->pl_user_prop); | ||||||
| 			if (len > cbp->cb_colwidths[GET_COL_PROPERTY]) | 			if (len > cbp->cb_colwidths[GET_COL_PROPERTY]) | ||||||
| 				cbp->cb_colwidths[GET_COL_PROPERTY] = len; | 				cbp->cb_colwidths[GET_COL_PROPERTY] = len; | ||||||
| @ -1314,9 +1326,10 @@ zprop_print_headers(zprop_get_cbdata_t *cbp, zfs_type_t type) | |||||||
| 		/*
 | 		/*
 | ||||||
| 		 * 'NAME' and 'SOURCE' columns | 		 * 'NAME' and 'SOURCE' columns | ||||||
| 		 */ | 		 */ | ||||||
| 		if (pl->pl_prop == (type == ZFS_TYPE_POOL ? ZPOOL_PROP_NAME : | 		if (pl->pl_prop == ((type == ZFS_TYPE_POOL) ? ZPOOL_PROP_NAME : | ||||||
| 		    ZFS_PROP_NAME) && | 		    ((type == ZFS_TYPE_VDEV) ? VDEV_PROP_NAME : | ||||||
| 		    pl->pl_width > cbp->cb_colwidths[GET_COL_NAME]) { | 		    ZFS_PROP_NAME)) && pl->pl_width > | ||||||
|  | 		    cbp->cb_colwidths[GET_COL_NAME]) { | ||||||
| 			cbp->cb_colwidths[GET_COL_NAME] = pl->pl_width; | 			cbp->cb_colwidths[GET_COL_NAME] = pl->pl_width; | ||||||
| 			cbp->cb_colwidths[GET_COL_SOURCE] = pl->pl_width + | 			cbp->cb_colwidths[GET_COL_SOURCE] = pl->pl_width + | ||||||
| 			    strlen(dgettext(TEXT_DOMAIN, "inherited from")); | 			    strlen(dgettext(TEXT_DOMAIN, "inherited from")); | ||||||
| @ -1597,6 +1610,9 @@ zprop_parse_value(libzfs_handle_t *hdl, nvpair_t *elem, int prop, | |||||||
| 	if (type == ZFS_TYPE_POOL) { | 	if (type == ZFS_TYPE_POOL) { | ||||||
| 		proptype = zpool_prop_get_type(prop); | 		proptype = zpool_prop_get_type(prop); | ||||||
| 		propname = zpool_prop_to_name(prop); | 		propname = zpool_prop_to_name(prop); | ||||||
|  | 	} else if (type == ZFS_TYPE_VDEV) { | ||||||
|  | 		proptype = vdev_prop_get_type(prop); | ||||||
|  | 		propname = vdev_prop_to_name(prop); | ||||||
| 	} else { | 	} else { | ||||||
| 		proptype = zfs_prop_get_type(prop); | 		proptype = zfs_prop_get_type(prop); | ||||||
| 		propname = zfs_prop_to_name(prop); | 		propname = zfs_prop_to_name(prop); | ||||||
| @ -1747,15 +1763,15 @@ addlist(libzfs_handle_t *hdl, char *propname, zprop_list_t **listp, | |||||||
| 		prop = ZPROP_INVAL; | 		prop = ZPROP_INVAL; | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * When no property table entry can be found, return failure if | 	 * Return failure if no property table entry was found and this isn't | ||||||
| 	 * this is a pool property or if this isn't a user-defined | 	 * a user-defined property. | ||||||
| 	 * dataset property, |  | ||||||
| 	 */ | 	 */ | ||||||
| 	if (prop == ZPROP_INVAL && ((type == ZFS_TYPE_POOL && | 	if (prop == ZPROP_INVAL && ((type == ZFS_TYPE_POOL && | ||||||
| 	    !zpool_prop_feature(propname) && | 	    !zpool_prop_feature(propname) && | ||||||
| 	    !zpool_prop_unsupported(propname)) || | 	    !zpool_prop_unsupported(propname)) || | ||||||
| 	    (type == ZFS_TYPE_DATASET && !zfs_prop_user(propname) && | 	    ((type == ZFS_TYPE_DATASET) && !zfs_prop_user(propname) && | ||||||
| 	    !zfs_prop_userquota(propname) && !zfs_prop_written(propname)))) { | 	    !zfs_prop_userquota(propname) && !zfs_prop_written(propname)) || | ||||||
|  | 	    ((type == ZFS_TYPE_VDEV) && !vdev_prop_user(propname)))) { | ||||||
| 		zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, | 		zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, | ||||||
| 		    "invalid property '%s'"), propname); | 		    "invalid property '%s'"), propname); | ||||||
| 		return (zfs_error(hdl, EZFS_BADPROP, | 		return (zfs_error(hdl, EZFS_BADPROP, | ||||||
| @ -1938,8 +1954,8 @@ zprop_expand_list(libzfs_handle_t *hdl, zprop_list_t **plp, zfs_type_t type) | |||||||
| 		if ((entry = zfs_alloc(hdl, sizeof (zprop_list_t))) == NULL) | 		if ((entry = zfs_alloc(hdl, sizeof (zprop_list_t))) == NULL) | ||||||
| 			return (-1); | 			return (-1); | ||||||
| 
 | 
 | ||||||
| 		entry->pl_prop = (type == ZFS_TYPE_POOL) ?  ZPOOL_PROP_NAME : | 		entry->pl_prop = ((type == ZFS_TYPE_POOL) ?  ZPOOL_PROP_NAME : | ||||||
| 		    ZFS_PROP_NAME; | 		    ((type == ZFS_TYPE_VDEV) ? VDEV_PROP_NAME : ZFS_PROP_NAME)); | ||||||
| 		entry->pl_width = zprop_width(entry->pl_prop, | 		entry->pl_width = zprop_width(entry->pl_prop, | ||||||
| 		    &entry->pl_fixed, type); | 		    &entry->pl_fixed, type); | ||||||
| 		entry->pl_all = B_TRUE; | 		entry->pl_all = B_TRUE; | ||||||
|  | |||||||
| @ -305,6 +305,10 @@ zfs_jail(zfs_handle_t *zhp, int jailid, int attach) | |||||||
| 		zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, | 		zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, | ||||||
| 		    "bookmarks can not be jailed")); | 		    "bookmarks can not be jailed")); | ||||||
| 		return (zfs_error(hdl, EZFS_BADTYPE, errbuf)); | 		return (zfs_error(hdl, EZFS_BADTYPE, errbuf)); | ||||||
|  | 	case ZFS_TYPE_VDEV: | ||||||
|  | 		zfs_error_aux(hdl, dgettext(TEXT_DOMAIN, | ||||||
|  | 		    "vdevs can not be jailed")); | ||||||
|  | 		return (zfs_error(hdl, EZFS_BADTYPE, errbuf)); | ||||||
| 	case ZFS_TYPE_POOL: | 	case ZFS_TYPE_POOL: | ||||||
| 	case ZFS_TYPE_FILESYSTEM: | 	case ZFS_TYPE_FILESYSTEM: | ||||||
| 		/* OK */ | 		/* OK */ | ||||||
|  | |||||||
| @ -167,6 +167,7 @@ | |||||||
|     <elf-symbol name='lzc_get_bookmarks' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='lzc_get_bookmarks' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='lzc_get_bootenv' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='lzc_get_bootenv' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='lzc_get_holds' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='lzc_get_holds' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='lzc_get_vdev_prop' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='lzc_hold' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='lzc_hold' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='lzc_initialize' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='lzc_initialize' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='lzc_ioctl_fd' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='lzc_ioctl_fd' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
| @ -192,6 +193,7 @@ | |||||||
|     <elf-symbol name='lzc_send_space' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='lzc_send_space' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='lzc_send_space_resume_redacted' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='lzc_send_space_resume_redacted' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='lzc_set_bootenv' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='lzc_set_bootenv' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|  |     <elf-symbol name='lzc_set_vdev_prop' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='lzc_snaprange_space' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='lzc_snaprange_space' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='lzc_snapshot' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='lzc_snapshot' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
|     <elf-symbol name='lzc_sync' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> |     <elf-symbol name='lzc_sync' type='func-type' binding='global-binding' visibility='default-visibility' is-defined='yes'/> | ||||||
| @ -1823,6 +1825,18 @@ | |||||||
|       <parameter type-id='857bb57e' name='outnvl'/> |       <parameter type-id='857bb57e' name='outnvl'/> | ||||||
|       <return type-id='95e97e5e'/> |       <return type-id='95e97e5e'/> | ||||||
|     </function-decl> |     </function-decl> | ||||||
|  |     <function-decl name='lzc_get_vdev_prop' mangled-name='lzc_get_vdev_prop' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='lzc_get_vdev_prop'> | ||||||
|  |       <parameter type-id='80f4b756' name='poolname'/> | ||||||
|  |       <parameter type-id='5ce45b60' name='innvl'/> | ||||||
|  |       <parameter type-id='857bb57e' name='outnvl'/> | ||||||
|  |       <return type-id='95e97e5e'/> | ||||||
|  |     </function-decl> | ||||||
|  |     <function-decl name='lzc_set_vdev_prop' mangled-name='lzc_set_vdev_prop' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='lzc_set_vdev_prop'> | ||||||
|  |       <parameter type-id='80f4b756' name='poolname'/> | ||||||
|  |       <parameter type-id='5ce45b60' name='innvl'/> | ||||||
|  |       <parameter type-id='857bb57e' name='outnvl'/> | ||||||
|  |       <return type-id='95e97e5e'/> | ||||||
|  |     </function-decl> | ||||||
|     <function-decl name='lzc_load_key' mangled-name='lzc_load_key' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='lzc_load_key'> |     <function-decl name='lzc_load_key' mangled-name='lzc_load_key' visibility='default' binding='global' size-in-bits='64' elf-symbol-id='lzc_load_key'> | ||||||
|       <parameter type-id='80f4b756' name='fsname'/> |       <parameter type-id='80f4b756' name='fsname'/> | ||||||
|       <parameter type-id='c19b74c3' name='noop'/> |       <parameter type-id='c19b74c3' name='noop'/> | ||||||
|  | |||||||
| @ -1394,6 +1394,18 @@ lzc_channel_program_nosync(const char *pool, const char *program, | |||||||
| 	    memlimit, argnvl, outnvl)); | 	    memlimit, argnvl, outnvl)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | int | ||||||
|  | lzc_get_vdev_prop(const char *poolname, nvlist_t *innvl, nvlist_t **outnvl) | ||||||
|  | { | ||||||
|  | 	return (lzc_ioctl(ZFS_IOC_VDEV_GET_PROPS, poolname, innvl, outnvl)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | lzc_set_vdev_prop(const char *poolname, nvlist_t *innvl, nvlist_t **outnvl) | ||||||
|  | { | ||||||
|  | 	return (lzc_ioctl(ZFS_IOC_VDEV_SET_PROPS, poolname, innvl, outnvl)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Performs key management functions |  * Performs key management functions | ||||||
|  * |  * | ||||||
|  | |||||||
| @ -1888,6 +1888,15 @@ for_each_vdev_cb(void *zhp, nvlist_t *nv, pool_vdev_iter_f func, | |||||||
| 	    ZPOOL_CONFIG_CHILDREN | 	    ZPOOL_CONFIG_CHILDREN | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
|  | 	if (nvlist_lookup_string(nv, ZPOOL_CONFIG_TYPE, &type) != 0) | ||||||
|  | 		return (ret); | ||||||
|  | 
 | ||||||
|  | 	/* Don't run our function on root or indirect vdevs */ | ||||||
|  | 	if ((strcmp(type, VDEV_TYPE_ROOT) != 0) && | ||||||
|  | 	    (strcmp(type, VDEV_TYPE_INDIRECT) != 0)) { | ||||||
|  | 		ret |= func(zhp, nv, data); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	for (i = 0; i < ARRAY_SIZE(list); i++) { | 	for (i = 0; i < ARRAY_SIZE(list); i++) { | ||||||
| 		if (nvlist_lookup_nvlist_array(nv, list[i], &child, | 		if (nvlist_lookup_nvlist_array(nv, list[i], &child, | ||||||
| 		    &children) == 0) { | 		    &children) == 0) { | ||||||
| @ -1906,14 +1915,6 @@ for_each_vdev_cb(void *zhp, nvlist_t *nv, pool_vdev_iter_f func, | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (nvlist_lookup_string(nv, ZPOOL_CONFIG_TYPE, &type) != 0) |  | ||||||
| 		return (ret); |  | ||||||
| 
 |  | ||||||
| 	/* Don't run our function on root vdevs */ |  | ||||||
| 	if (strcmp(type, VDEV_TYPE_ROOT) != 0) { |  | ||||||
| 		ret |= func(zhp, nv, data); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return (ret); | 	return (ret); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										172
									
								
								man/man7/vdevprops.7
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										172
									
								
								man/man7/vdevprops.7
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,172 @@ | |||||||
|  | .\" | ||||||
|  | .\" 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) 2021 Klara, Inc. | ||||||
|  | .\" | ||||||
|  | .Dd November 27, 2021 | ||||||
|  | .Dt VDEVPROPS 7 | ||||||
|  | .Os | ||||||
|  | . | ||||||
|  | .Sh NAME | ||||||
|  | .Nm vdevprops | ||||||
|  | .Nd native and user-defined properties of ZFS vdevs | ||||||
|  | . | ||||||
|  | .Sh DESCRIPTION | ||||||
|  | Properties are divided into two types, native properties and user-defined | ||||||
|  | .Pq or Qq user | ||||||
|  | properties. | ||||||
|  | Native properties either export internal statistics or control ZFS behavior. | ||||||
|  | In addition, native properties are either editable or read-only. | ||||||
|  | User properties have no effect on ZFS behavior, but you can use them to annotate | ||||||
|  | vdevs in a way that is meaningful in your environment. | ||||||
|  | For more information about user properties, see the | ||||||
|  | .Sx User Properties | ||||||
|  | section, below. | ||||||
|  | . | ||||||
|  | .Ss Native Properties | ||||||
|  | Every vdev has a set of properties that export statistics about the vdev | ||||||
|  | as well as control various behaviors. | ||||||
|  | Properties are NOT inherited from top-level vdevs. | ||||||
|  | .Pp | ||||||
|  | The values of numeric properties can be specified using human-readable suffixes | ||||||
|  | .Po for example, | ||||||
|  | .Sy k , KB , M , Gb , | ||||||
|  | and so forth, up to | ||||||
|  | .Sy Z | ||||||
|  | for zettabyte | ||||||
|  | .Pc . | ||||||
|  | The following are all valid | ||||||
|  | .Pq and equal | ||||||
|  | specifications: | ||||||
|  | .Li 1536M , 1.5g , 1.50GB . | ||||||
|  | .Pp | ||||||
|  | The values of non-numeric properties are case sensitive and must be lowercase. | ||||||
|  | .Pp | ||||||
|  | The following native properties consist of read-only statistics about the | ||||||
|  | vdev. | ||||||
|  | These properties can not be changed. | ||||||
|  | .Bl -tag -width "fragmentation" | ||||||
|  | .It Sy capacity | ||||||
|  | Percentage of vdev space used | ||||||
|  | .It Sy state | ||||||
|  | state of this vdev such as online, faulted, or offline | ||||||
|  | .It Sy guid | ||||||
|  | globaly unique id of this vdev | ||||||
|  | .It Sy asize | ||||||
|  | The allocable size of this vdev | ||||||
|  | .It Sy psize | ||||||
|  | The physical size of this vdev | ||||||
|  | .It Sy ashift | ||||||
|  | The physical sector size of this vdev expressed as the power of two | ||||||
|  | .It Sy size | ||||||
|  | The total size of this vdev | ||||||
|  | .It Sy free | ||||||
|  | The amount of remaining free space on this vdev | ||||||
|  | .It Sy allocated | ||||||
|  | The amount of allocated space on this vdev | ||||||
|  | .It Sy expandsize | ||||||
|  | How much this vdev can expand by | ||||||
|  | .It Sy fragmentation | ||||||
|  | Percent of fragmentation in this vdev | ||||||
|  | .It Sy parity | ||||||
|  | The level of parity for this vdev | ||||||
|  | .It Sy devid | ||||||
|  | The device id for this vdev | ||||||
|  | .It Sy physpath | ||||||
|  | The physical path to the device | ||||||
|  | .It Sy encpath | ||||||
|  | The enclosure path to the device | ||||||
|  | .It Sy fru | ||||||
|  | Field Replacable Unit, usually a model number | ||||||
|  | .It Sy parent | ||||||
|  | Parent of this vdev | ||||||
|  | .It Sy children | ||||||
|  | Comma separated list of children of this vdev | ||||||
|  | .It Sy numchildren | ||||||
|  | The number of children belonging to this vdev | ||||||
|  | .It Sy read_errors , write_errors , checksum_errors , initialize_errors | ||||||
|  | The number of errors of each type encountered by this vdev | ||||||
|  | .It Sy null_ops , read_ops , write_ops , free_ops , claim_ops , trim_ops | ||||||
|  | The number of I/O operations of each type performed by this vdev | ||||||
|  | .It Xo | ||||||
|  | .Sy null_bytes , read_bytes , write_bytes , free_bytes , claim_bytes , | ||||||
|  | .Sy trim_bytes | ||||||
|  | .Xc | ||||||
|  | The cumulative size of all operations of each type performed by this vdev | ||||||
|  | .It Sy removing | ||||||
|  | If this device is currently being removed from the pool | ||||||
|  | .El | ||||||
|  | .Pp | ||||||
|  | The following native properties can be used to change the behavior of a ZFS | ||||||
|  | dataset. | ||||||
|  | .Bl -tag -width "allocating" | ||||||
|  | .It Sy comment | ||||||
|  | A text comment up to 8192 characters long | ||||||
|  | .It Sy bootsize | ||||||
|  | The amount of space to reserve for the EFI system partition | ||||||
|  | .It Sy path | ||||||
|  | The path to the device for this vdev | ||||||
|  | .It Sy allocating | ||||||
|  | If this device should perform new allocations, used to disable a device | ||||||
|  | when it is scheduled for later removal. | ||||||
|  | See | ||||||
|  | .Xr zpool-remove 8 . | ||||||
|  | .El | ||||||
|  | .Ss User Properties | ||||||
|  | In addition to the standard native properties, ZFS supports arbitrary user | ||||||
|  | properties. | ||||||
|  | User properties have no effect on ZFS behavior, but applications or | ||||||
|  | administrators can use them to annotate vdevs. | ||||||
|  | .Pp | ||||||
|  | User property names must contain a colon | ||||||
|  | .Pq Qq Sy \&: | ||||||
|  | character to distinguish them from native properties. | ||||||
|  | They may contain lowercase letters, numbers, and the following punctuation | ||||||
|  | characters: colon | ||||||
|  | .Pq Qq Sy \&: , | ||||||
|  | dash | ||||||
|  | .Pq Qq Sy - , | ||||||
|  | period | ||||||
|  | .Pq Qq Sy \&. , | ||||||
|  | and underscore | ||||||
|  | .Pq Qq Sy _ . | ||||||
|  | The expected convention is that the property name is divided into two portions | ||||||
|  | such as | ||||||
|  | .Ar module : Ns Ar property , | ||||||
|  | but this namespace is not enforced by ZFS. | ||||||
|  | User property names can be at most 256 characters, and cannot begin with a dash | ||||||
|  | .Pq Qq Sy - . | ||||||
|  | .Pp | ||||||
|  | When making programmatic use of user properties, it is strongly suggested to use | ||||||
|  | a reversed DNS domain name for the | ||||||
|  | .Ar module | ||||||
|  | component of property names to reduce the chance that two | ||||||
|  | independently-developed packages use the same property name for different | ||||||
|  | purposes. | ||||||
|  | .Pp | ||||||
|  | The values of user properties are arbitrary strings and | ||||||
|  | are never validated. | ||||||
|  | Use the | ||||||
|  | .Nm zpool Cm set | ||||||
|  | command with a blank value to clear a user property. | ||||||
|  | Property values are limited to 8192 bytes. | ||||||
|  | .Sh SEE ALSO | ||||||
|  | .Xr zpoolprops 7 , | ||||||
|  | .Xr zpool-set 8 | ||||||
| @ -40,11 +40,27 @@ | |||||||
| .Op Fl o Ar field Ns Oo , Ns Ar field Oc Ns … | .Op Fl o Ar field Ns Oo , Ns Ar field Oc Ns … | ||||||
| .Sy all Ns | Ns Ar property Ns Oo , Ns Ar property Oc Ns … | .Sy all Ns | Ns Ar property Ns Oo , Ns Ar property Oc Ns … | ||||||
| .Oo Ar pool Oc Ns … | .Oo Ar pool Oc Ns … | ||||||
|  | . | ||||||
|  | .Nm zpool | ||||||
|  | .Cm get | ||||||
|  | .Op Fl Hp | ||||||
|  | .Op Fl o Ar field Ns Oo , Ns Ar field Oc Ns … | ||||||
|  | .Sy all Ns | Ns Ar property Ns Oo , Ns Ar property Oc Ns … | ||||||
|  | .Ar pool | ||||||
|  | .Oo Sy all-vdevs Ns | Ns | ||||||
|  | .Ar vdev Oc Ns … | ||||||
|  | . | ||||||
| .Nm zpool | .Nm zpool | ||||||
| .Cm set | .Cm set | ||||||
| .Ar property Ns = Ns Ar value | .Ar property Ns = Ns Ar value | ||||||
| .Ar pool | .Ar pool | ||||||
| . | . | ||||||
|  | .Nm zpool | ||||||
|  | .Cm set | ||||||
|  | .Ar property Ns = Ns Ar value | ||||||
|  | .Ar pool | ||||||
|  | .Ar vdev | ||||||
|  | . | ||||||
| .Sh DESCRIPTION | .Sh DESCRIPTION | ||||||
| .Bl -tag -width Ds | .Bl -tag -width Ds | ||||||
| .It Xo | .It Xo | ||||||
| @ -91,6 +107,56 @@ Display numbers in parsable (exact) values. | |||||||
| .El | .El | ||||||
| .It Xo | .It Xo | ||||||
| .Nm zpool | .Nm zpool | ||||||
|  | .Cm get | ||||||
|  | .Op Fl Hp | ||||||
|  | .Op Fl o Ar field Ns Oo , Ns Ar field Oc Ns … | ||||||
|  | .Sy all Ns | Ns Ar property Ns Oo , Ns Ar property Oc Ns … | ||||||
|  | .Ar pool | ||||||
|  | .Oo Sy all-vdevs Ns | Ns | ||||||
|  | .Ar vdev Oc Ns … | ||||||
|  | .Xc | ||||||
|  | Retrieves the given list of properties | ||||||
|  | .Po | ||||||
|  | or all properties if | ||||||
|  | .Sy all | ||||||
|  | is used | ||||||
|  | .Pc | ||||||
|  | for the specified vdevs | ||||||
|  | .Po | ||||||
|  | or all vdevs if | ||||||
|  | .Sy all-vdevs | ||||||
|  | is used | ||||||
|  | .Pc | ||||||
|  | in the specified pool. | ||||||
|  | These properties are displayed with the following fields: | ||||||
|  | .Bl -tag -compact -offset Ds -width "property" | ||||||
|  | .It Sy name | ||||||
|  | Name of vdev. | ||||||
|  | .It Sy property | ||||||
|  | Property name. | ||||||
|  | .It Sy value | ||||||
|  | Property value. | ||||||
|  | .It Sy source | ||||||
|  | Property source, either | ||||||
|  | .Sy default No or Sy local . | ||||||
|  | .El | ||||||
|  | .Pp | ||||||
|  | See the | ||||||
|  | .Xr vdevprops 7 | ||||||
|  | manual page for more information on the available pool properties. | ||||||
|  | .Bl -tag -compact -offset Ds -width "-o field" | ||||||
|  | .It Fl H | ||||||
|  | Scripted mode. | ||||||
|  | Do not display headers, and separate fields by a single tab instead of arbitrary | ||||||
|  | space. | ||||||
|  | .It Fl o Ar field | ||||||
|  | A comma-separated list of columns to display, defaults to | ||||||
|  | .Sy name , Ns Sy property , Ns Sy value , Ns Sy source . | ||||||
|  | .It Fl p | ||||||
|  | Display numbers in parsable (exact) values. | ||||||
|  | .El | ||||||
|  | .It Xo | ||||||
|  | .Nm zpool | ||||||
| .Cm set | .Cm set | ||||||
| .Ar property Ns = Ns Ar value | .Ar property Ns = Ns Ar value | ||||||
| .Ar pool | .Ar pool | ||||||
| @ -100,9 +166,22 @@ See the | |||||||
| .Xr zpoolprops 7 | .Xr zpoolprops 7 | ||||||
| manual page for more information on what properties can be set and acceptable | manual page for more information on what properties can be set and acceptable | ||||||
| values. | values. | ||||||
|  | .It Xo | ||||||
|  | .Nm zpool | ||||||
|  | .Cm set | ||||||
|  | .Ar property Ns = Ns Ar value | ||||||
|  | .Ar pool | ||||||
|  | .Ar vdev | ||||||
|  | .Xc | ||||||
|  | Sets the given property on the specified vdev in the specified pool. | ||||||
|  | See the | ||||||
|  | .Xr vdevprops 7 | ||||||
|  | manual page for more information on what properties can be set and acceptable | ||||||
|  | values. | ||||||
| .El | .El | ||||||
| . | . | ||||||
| .Sh SEE ALSO | .Sh SEE ALSO | ||||||
|  | .Xr vdevprops 7 , | ||||||
| .Xr zpool-features 7 , | .Xr zpool-features 7 , | ||||||
| .Xr zpoolprops 7 , | .Xr zpoolprops 7 , | ||||||
| .Xr zpool-list 8 | .Xr zpool-list 8 | ||||||
|  | |||||||
| @ -90,6 +90,7 @@ struct zfs_mod_kobj { | |||||||
| static zfs_mod_kobj_t kernel_features_kobj; | static zfs_mod_kobj_t kernel_features_kobj; | ||||||
| static zfs_mod_kobj_t pool_features_kobj; | static zfs_mod_kobj_t pool_features_kobj; | ||||||
| static zfs_mod_kobj_t dataset_props_kobj; | static zfs_mod_kobj_t dataset_props_kobj; | ||||||
|  | static zfs_mod_kobj_t vdev_props_kobj; | ||||||
| static zfs_mod_kobj_t pool_props_kobj; | static zfs_mod_kobj_t pool_props_kobj; | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
| @ -333,6 +334,20 @@ dataset_property_show(struct kobject *kobj, struct attribute *attr, char *buf) | |||||||
| 	return (len); | 	return (len); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static ssize_t | ||||||
|  | vdev_property_show(struct kobject *kobj, struct attribute *attr, char *buf) | ||||||
|  | { | ||||||
|  | 	vdev_prop_t prop = vdev_name_to_prop(kobject_name(kobj)); | ||||||
|  | 	zprop_desc_t *prop_tbl = vdev_prop_get_table(); | ||||||
|  | 	ssize_t len; | ||||||
|  | 
 | ||||||
|  | 	ASSERT3U(prop, <, VDEV_NUM_PROPS); | ||||||
|  | 
 | ||||||
|  | 	len = zprop_sysfs_show(attr->name, &prop_tbl[prop], buf, PAGE_SIZE); | ||||||
|  | 
 | ||||||
|  | 	return (len); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static ssize_t | static ssize_t | ||||||
| pool_property_show(struct kobject *kobj, struct attribute *attr, char *buf) | pool_property_show(struct kobject *kobj, struct attribute *attr, char *buf) | ||||||
| { | { | ||||||
| @ -577,6 +592,14 @@ zfs_sysfs_properties_init(zfs_mod_kobj_t *zfs_kobj, struct kobject *parent, | |||||||
| 		context.p2k_show_func = pool_property_show; | 		context.p2k_show_func = pool_property_show; | ||||||
| 		err = zfs_kobj_init(zfs_kobj, 0, ZPOOL_NUM_PROPS, | 		err = zfs_kobj_init(zfs_kobj, 0, ZPOOL_NUM_PROPS, | ||||||
| 		    pool_property_show); | 		    pool_property_show); | ||||||
|  | 	} else if (type == ZFS_TYPE_VDEV) { | ||||||
|  | 		name = ZFS_SYSFS_VDEV_PROPERTIES; | ||||||
|  | 		context.p2k_table = vdev_prop_get_table(); | ||||||
|  | 		context.p2k_attr_count = ZPOOL_PROP_ATTR_COUNT; | ||||||
|  | 		context.p2k_parent = zfs_kobj; | ||||||
|  | 		context.p2k_show_func = vdev_property_show; | ||||||
|  | 		err = zfs_kobj_init(zfs_kobj, 0, VDEV_NUM_PROPS, | ||||||
|  | 		    vdev_property_show); | ||||||
| 	} else { | 	} else { | ||||||
| 		name = ZFS_SYSFS_DATASET_PROPERTIES; | 		name = ZFS_SYSFS_DATASET_PROPERTIES; | ||||||
| 		context.p2k_table = zfs_prop_get_table(); | 		context.p2k_table = zfs_prop_get_table(); | ||||||
| @ -639,12 +662,22 @@ zfs_sysfs_init(void) | |||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	err = zfs_sysfs_properties_init(&vdev_props_kobj, parent, | ||||||
|  | 	    ZFS_TYPE_VDEV); | ||||||
|  | 	if (err) { | ||||||
|  | 		zfs_kobj_fini(&kernel_features_kobj); | ||||||
|  | 		zfs_kobj_fini(&pool_features_kobj); | ||||||
|  | 		zfs_kobj_fini(&pool_props_kobj); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	err = zfs_sysfs_properties_init(&dataset_props_kobj, parent, | 	err = zfs_sysfs_properties_init(&dataset_props_kobj, parent, | ||||||
| 	    ZFS_TYPE_FILESYSTEM); | 	    ZFS_TYPE_FILESYSTEM); | ||||||
| 	if (err) { | 	if (err) { | ||||||
| 		zfs_kobj_fini(&kernel_features_kobj); | 		zfs_kobj_fini(&kernel_features_kobj); | ||||||
| 		zfs_kobj_fini(&pool_features_kobj); | 		zfs_kobj_fini(&pool_features_kobj); | ||||||
| 		zfs_kobj_fini(&pool_props_kobj); | 		zfs_kobj_fini(&pool_props_kobj); | ||||||
|  | 		zfs_kobj_fini(&vdev_props_kobj); | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -657,6 +690,7 @@ zfs_sysfs_fini(void) | |||||||
| 	 */ | 	 */ | ||||||
| 	zfs_kobj_fini(&kernel_features_kobj); | 	zfs_kobj_fini(&kernel_features_kobj); | ||||||
| 	zfs_kobj_fini(&pool_features_kobj); | 	zfs_kobj_fini(&pool_features_kobj); | ||||||
| 	zfs_kobj_fini(&dataset_props_kobj); |  | ||||||
| 	zfs_kobj_fini(&pool_props_kobj); | 	zfs_kobj_fini(&pool_props_kobj); | ||||||
|  | 	zfs_kobj_fini(&vdev_props_kobj); | ||||||
|  | 	zfs_kobj_fini(&dataset_props_kobj); | ||||||
| } | } | ||||||
|  | |||||||
| @ -723,18 +723,6 @@ zfs_name_to_prop(const char *propname) | |||||||
| 	return (zprop_name_to_prop(propname, ZFS_TYPE_DATASET)); | 	return (zprop_name_to_prop(propname, ZFS_TYPE_DATASET)); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /*
 |  | ||||||
|  * For user property names, we allow all lowercase alphanumeric characters, plus |  | ||||||
|  * a few useful punctuation characters. |  | ||||||
|  */ |  | ||||||
| static int |  | ||||||
| valid_char(char c) |  | ||||||
| { |  | ||||||
| 	return ((c >= 'a' && c <= 'z') || |  | ||||||
| 	    (c >= '0' && c <= '9') || |  | ||||||
| 	    c == '-' || c == '_' || c == '.' || c == ':'); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /*
 | /*
 | ||||||
|  * Returns true if this is a valid user-defined property (one with a ':'). |  * Returns true if this is a valid user-defined property (one with a ':'). | ||||||
|  */ |  */ | ||||||
| @ -747,7 +735,7 @@ zfs_prop_user(const char *name) | |||||||
| 
 | 
 | ||||||
| 	for (i = 0; i < strlen(name); i++) { | 	for (i = 0; i < strlen(name); i++) { | ||||||
| 		c = name[i]; | 		c = name[i]; | ||||||
| 		if (!valid_char(c)) | 		if (!zprop_valid_char(c)) | ||||||
| 			return (B_FALSE); | 			return (B_FALSE); | ||||||
| 		if (c == ':') | 		if (c == ':') | ||||||
| 			foundsep = B_TRUE; | 			foundsep = B_TRUE; | ||||||
|  | |||||||
| @ -23,6 +23,7 @@ | |||||||
|  * Copyright 2011 Nexenta Systems, Inc. All rights reserved. |  * Copyright 2011 Nexenta Systems, Inc. All rights reserved. | ||||||
|  * Copyright (c) 2012, 2018 by Delphix. All rights reserved. |  * Copyright (c) 2012, 2018 by Delphix. All rights reserved. | ||||||
|  * Copyright (c) 2021, Colm Buckley <colm@tuatha.org> |  * Copyright (c) 2021, Colm Buckley <colm@tuatha.org> | ||||||
|  |  * Copyright (c) 2021, Klara Inc. | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| #include <sys/zio.h> | #include <sys/zio.h> | ||||||
| @ -40,6 +41,7 @@ | |||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| static zprop_desc_t zpool_prop_table[ZPOOL_NUM_PROPS]; | static zprop_desc_t zpool_prop_table[ZPOOL_NUM_PROPS]; | ||||||
|  | static zprop_desc_t vdev_prop_table[VDEV_NUM_PROPS]; | ||||||
| 
 | 
 | ||||||
| zprop_desc_t * | zprop_desc_t * | ||||||
| zpool_prop_get_table(void) | zpool_prop_get_table(void) | ||||||
| @ -260,12 +262,249 @@ zpool_prop_align_right(zpool_prop_t prop) | |||||||
| } | } | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
|  | zprop_desc_t * | ||||||
|  | vdev_prop_get_table(void) | ||||||
|  | { | ||||||
|  | 	return (vdev_prop_table); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void | ||||||
|  | vdev_prop_init(void) | ||||||
|  | { | ||||||
|  | 	static zprop_index_t boolean_table[] = { | ||||||
|  | 		{ "off",	0}, | ||||||
|  | 		{ "on",		1}, | ||||||
|  | 		{ NULL } | ||||||
|  | 	}; | ||||||
|  | 	static zprop_index_t boolean_na_table[] = { | ||||||
|  | 		{ "off",	0}, | ||||||
|  | 		{ "on",		1}, | ||||||
|  | 		{ "-",		2},	/* ZPROP_BOOLEAN_NA */ | ||||||
|  | 		{ NULL } | ||||||
|  | 	}; | ||||||
|  | 
 | ||||||
|  | 	/* string properties */ | ||||||
|  | 	zprop_register_string(VDEV_PROP_COMMENT, "comment", NULL, | ||||||
|  | 	    PROP_DEFAULT, ZFS_TYPE_VDEV, "<comment-string>", "COMMENT"); | ||||||
|  | 	zprop_register_string(VDEV_PROP_PATH, "path", NULL, | ||||||
|  | 	    PROP_DEFAULT, ZFS_TYPE_VDEV, "<device-path>", "PATH"); | ||||||
|  | 	zprop_register_string(VDEV_PROP_DEVID, "devid", NULL, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<devid>", "DEVID"); | ||||||
|  | 	zprop_register_string(VDEV_PROP_PHYS_PATH, "physpath", NULL, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<physpath>", "PHYSPATH"); | ||||||
|  | 	zprop_register_string(VDEV_PROP_ENC_PATH, "encpath", NULL, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<encpath>", "ENCPATH"); | ||||||
|  | 	zprop_register_string(VDEV_PROP_FRU, "fru", NULL, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<fru>", "FRU"); | ||||||
|  | 	zprop_register_string(VDEV_PROP_PARENT, "parent", NULL, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<parent>", "PARENT"); | ||||||
|  | 	zprop_register_string(VDEV_PROP_CHILDREN, "children", NULL, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<child[,...]>", "CHILDREN"); | ||||||
|  | 
 | ||||||
|  | 	/* readonly number properties */ | ||||||
|  | 	zprop_register_number(VDEV_PROP_SIZE, "size", 0, PROP_READONLY, | ||||||
|  | 	    ZFS_TYPE_VDEV, "<size>", "SIZE"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_FREE, "free", 0, PROP_READONLY, | ||||||
|  | 	    ZFS_TYPE_VDEV, "<size>", "FREE"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_ALLOCATED, "allocated", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<size>", "ALLOC"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_EXPANDSZ, "expandsize", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<size>", "EXPANDSZ"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_FRAGMENTATION, "fragmentation", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<percent>", "FRAG"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_CAPACITY, "capacity", 0, PROP_READONLY, | ||||||
|  | 	    ZFS_TYPE_VDEV, "<size>", "CAP"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_GUID, "guid", 0, PROP_READONLY, | ||||||
|  | 	    ZFS_TYPE_VDEV, "<guid>", "GUID"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_STATE, "state", 0, PROP_READONLY, | ||||||
|  | 	    ZFS_TYPE_VDEV, "<state>", "STATE"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_BOOTSIZE, "bootsize", 0, PROP_READONLY, | ||||||
|  | 	    ZFS_TYPE_VDEV, "<size>", "BOOTSIZE"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_ASIZE, "asize", 0, PROP_READONLY, | ||||||
|  | 	    ZFS_TYPE_VDEV, "<asize>", "ASIZE"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_PSIZE, "psize", 0, PROP_READONLY, | ||||||
|  | 	    ZFS_TYPE_VDEV, "<psize>", "PSIZE"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_ASHIFT, "ashift", 0, PROP_READONLY, | ||||||
|  | 	    ZFS_TYPE_VDEV, "<ashift>", "ASHIFT"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_PARITY, "parity", 0, PROP_READONLY, | ||||||
|  | 	    ZFS_TYPE_VDEV, "<parity>", "PARITY"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_NUMCHILDREN, "numchildren", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<number-of-children>", "NUMCHILD"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_READ_ERRORS, "read_errors", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<errors>", "RDERR"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_WRITE_ERRORS, "write_errors", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<errors>", "WRERR"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_CHECKSUM_ERRORS, "checksum_errors", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<errors>", "CKERR"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_INITIALIZE_ERRORS, | ||||||
|  | 	    "initialize_errors", 0, PROP_READONLY, ZFS_TYPE_VDEV, "<errors>", | ||||||
|  | 	    "INITERR"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_OPS_NULL, "null_ops", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<operations>", "NULLOP"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_OPS_READ, "read_ops", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<operations>", "READOP"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_OPS_WRITE, "write_ops", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<operations>", "WRITEOP"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_OPS_FREE, "free_ops", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<operations>", "FREEOP"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_OPS_CLAIM, "claim_ops", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<operations>", "CLAIMOP"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_OPS_TRIM, "trim_ops", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<operations>", "TRIMOP"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_BYTES_NULL, "null_bytes", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<bytes>", "NULLBYTE"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_BYTES_READ, "read_bytes", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<bytes>", "READBYTE"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_BYTES_WRITE, "write_bytes", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<bytes>", "WRITEBYTE"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_BYTES_FREE, "free_bytes", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<bytes>", "FREEBYTE"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_BYTES_CLAIM, "claim_bytes", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<bytes>", "CLAIMBYTE"); | ||||||
|  | 	zprop_register_number(VDEV_PROP_BYTES_TRIM, "trim_bytes", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "<bytes>", "TRIMBYTE"); | ||||||
|  | 
 | ||||||
|  | 	/* default numeric properties */ | ||||||
|  | 
 | ||||||
|  | 	/* default index (boolean) properties */ | ||||||
|  | 	zprop_register_index(VDEV_PROP_REMOVING, "removing", 0, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "on | off", "REMOVING", | ||||||
|  | 	    boolean_table); | ||||||
|  | 	zprop_register_index(VDEV_PROP_ALLOCATING, "allocating", 1, | ||||||
|  | 	    PROP_DEFAULT, ZFS_TYPE_VDEV, "on | off", "ALLOCATING", | ||||||
|  | 	    boolean_na_table); | ||||||
|  | 
 | ||||||
|  | 	/* default index properties */ | ||||||
|  | 
 | ||||||
|  | 	/* hidden properties */ | ||||||
|  | 	zprop_register_hidden(VDEV_PROP_NAME, "name", PROP_TYPE_STRING, | ||||||
|  | 	    PROP_READONLY, ZFS_TYPE_VDEV, "NAME"); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Given a property name and its type, returns the corresponding property ID. | ||||||
|  |  */ | ||||||
|  | vdev_prop_t | ||||||
|  | vdev_name_to_prop(const char *propname) | ||||||
|  | { | ||||||
|  | 	return (zprop_name_to_prop(propname, ZFS_TYPE_VDEV)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Returns true if this is a valid user-defined property (one with a ':'). | ||||||
|  |  */ | ||||||
|  | boolean_t | ||||||
|  | vdev_prop_user(const char *name) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	char c; | ||||||
|  | 	boolean_t foundsep = B_FALSE; | ||||||
|  | 
 | ||||||
|  | 	for (i = 0; i < strlen(name); i++) { | ||||||
|  | 		c = name[i]; | ||||||
|  | 		if (!zprop_valid_char(c)) | ||||||
|  | 			return (B_FALSE); | ||||||
|  | 		if (c == ':') | ||||||
|  | 			foundsep = B_TRUE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return (foundsep); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Given a pool property ID, returns the corresponding name. | ||||||
|  |  * Assuming the pool property ID is valid. | ||||||
|  |  */ | ||||||
|  | const char * | ||||||
|  | vdev_prop_to_name(vdev_prop_t prop) | ||||||
|  | { | ||||||
|  | 	return (vdev_prop_table[prop].pd_name); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | zprop_type_t | ||||||
|  | vdev_prop_get_type(vdev_prop_t prop) | ||||||
|  | { | ||||||
|  | 	return (vdev_prop_table[prop].pd_proptype); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | boolean_t | ||||||
|  | vdev_prop_readonly(vdev_prop_t prop) | ||||||
|  | { | ||||||
|  | 	return (vdev_prop_table[prop].pd_attr == PROP_READONLY); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const char * | ||||||
|  | vdev_prop_default_string(vdev_prop_t prop) | ||||||
|  | { | ||||||
|  | 	return (vdev_prop_table[prop].pd_strdefault); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | uint64_t | ||||||
|  | vdev_prop_default_numeric(vdev_prop_t prop) | ||||||
|  | { | ||||||
|  | 	return (vdev_prop_table[prop].pd_numdefault); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | vdev_prop_string_to_index(vdev_prop_t prop, const char *string, | ||||||
|  |     uint64_t *index) | ||||||
|  | { | ||||||
|  | 	return (zprop_string_to_index(prop, string, index, ZFS_TYPE_VDEV)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | vdev_prop_index_to_string(vdev_prop_t prop, uint64_t index, | ||||||
|  |     const char **string) | ||||||
|  | { | ||||||
|  | 	return (zprop_index_to_string(prop, index, string, ZFS_TYPE_VDEV)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Returns true if this is a valid vdev property. | ||||||
|  |  */ | ||||||
|  | boolean_t | ||||||
|  | zpool_prop_vdev(const char *name) | ||||||
|  | { | ||||||
|  | 	return (vdev_name_to_prop(name) != VDEV_PROP_INVAL); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | uint64_t | ||||||
|  | vdev_prop_random_value(vdev_prop_t prop, uint64_t seed) | ||||||
|  | { | ||||||
|  | 	return (zprop_random_value(prop, seed, ZFS_TYPE_VDEV)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #ifndef _KERNEL | ||||||
|  | const char * | ||||||
|  | vdev_prop_values(vdev_prop_t prop) | ||||||
|  | { | ||||||
|  | 	return (vdev_prop_table[prop].pd_values); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const char * | ||||||
|  | vdev_prop_column_name(vdev_prop_t prop) | ||||||
|  | { | ||||||
|  | 	return (vdev_prop_table[prop].pd_colname); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | boolean_t | ||||||
|  | vdev_prop_align_right(vdev_prop_t prop) | ||||||
|  | { | ||||||
|  | 	return (vdev_prop_table[prop].pd_rightalign); | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
| #if defined(_KERNEL) | #if defined(_KERNEL) | ||||||
| /* zpool property functions */ | /* zpool property functions */ | ||||||
| EXPORT_SYMBOL(zpool_prop_init); | EXPORT_SYMBOL(zpool_prop_init); | ||||||
| EXPORT_SYMBOL(zpool_prop_get_type); | EXPORT_SYMBOL(zpool_prop_get_type); | ||||||
| EXPORT_SYMBOL(zpool_prop_get_table); | EXPORT_SYMBOL(zpool_prop_get_table); | ||||||
| 
 | 
 | ||||||
|  | /* vdev property functions */ | ||||||
|  | EXPORT_SYMBOL(vdev_prop_init); | ||||||
|  | EXPORT_SYMBOL(vdev_prop_get_type); | ||||||
|  | EXPORT_SYMBOL(vdev_prop_get_table); | ||||||
|  | 
 | ||||||
| /* Pool property functions shared between libzfs and kernel. */ | /* Pool property functions shared between libzfs and kernel. */ | ||||||
| EXPORT_SYMBOL(zpool_name_to_prop); | EXPORT_SYMBOL(zpool_name_to_prop); | ||||||
| EXPORT_SYMBOL(zpool_prop_to_name); | EXPORT_SYMBOL(zpool_prop_to_name); | ||||||
| @ -276,4 +515,15 @@ EXPORT_SYMBOL(zpool_prop_feature); | |||||||
| EXPORT_SYMBOL(zpool_prop_unsupported); | EXPORT_SYMBOL(zpool_prop_unsupported); | ||||||
| EXPORT_SYMBOL(zpool_prop_index_to_string); | EXPORT_SYMBOL(zpool_prop_index_to_string); | ||||||
| EXPORT_SYMBOL(zpool_prop_string_to_index); | EXPORT_SYMBOL(zpool_prop_string_to_index); | ||||||
|  | EXPORT_SYMBOL(zpool_prop_vdev); | ||||||
|  | 
 | ||||||
|  | /* vdev property functions shared between libzfs and kernel. */ | ||||||
|  | EXPORT_SYMBOL(vdev_name_to_prop); | ||||||
|  | EXPORT_SYMBOL(vdev_prop_user); | ||||||
|  | EXPORT_SYMBOL(vdev_prop_to_name); | ||||||
|  | EXPORT_SYMBOL(vdev_prop_default_string); | ||||||
|  | EXPORT_SYMBOL(vdev_prop_default_numeric); | ||||||
|  | EXPORT_SYMBOL(vdev_prop_readonly); | ||||||
|  | EXPORT_SYMBOL(vdev_prop_index_to_string); | ||||||
|  | EXPORT_SYMBOL(vdev_prop_string_to_index); | ||||||
| #endif | #endif | ||||||
|  | |||||||
| @ -53,6 +53,8 @@ zprop_get_proptable(zfs_type_t type) | |||||||
| { | { | ||||||
| 	if (type == ZFS_TYPE_POOL) | 	if (type == ZFS_TYPE_POOL) | ||||||
| 		return (zpool_prop_get_table()); | 		return (zpool_prop_get_table()); | ||||||
|  | 	else if (type == ZFS_TYPE_VDEV) | ||||||
|  | 		return (vdev_prop_get_table()); | ||||||
| 	else | 	else | ||||||
| 		return (zfs_prop_get_table()); | 		return (zfs_prop_get_table()); | ||||||
| } | } | ||||||
| @ -62,6 +64,8 @@ zprop_get_numprops(zfs_type_t type) | |||||||
| { | { | ||||||
| 	if (type == ZFS_TYPE_POOL) | 	if (type == ZFS_TYPE_POOL) | ||||||
| 		return (ZPOOL_NUM_PROPS); | 		return (ZPOOL_NUM_PROPS); | ||||||
|  | 	else if (type == ZFS_TYPE_VDEV) | ||||||
|  | 		return (VDEV_NUM_PROPS); | ||||||
| 	else | 	else | ||||||
| 		return (ZFS_NUM_PROPS); | 		return (ZFS_NUM_PROPS); | ||||||
| } | } | ||||||
| @ -81,7 +85,8 @@ zfs_mod_supported_prop(const char *name, zfs_type_t type) | |||||||
| 	return (B_TRUE); | 	return (B_TRUE); | ||||||
| #else | #else | ||||||
| 	return (zfs_mod_supported(type == ZFS_TYPE_POOL ? | 	return (zfs_mod_supported(type == ZFS_TYPE_POOL ? | ||||||
| 	    ZFS_SYSFS_POOL_PROPERTIES : ZFS_SYSFS_DATASET_PROPERTIES, name)); | 	    ZFS_SYSFS_POOL_PROPERTIES : (type == ZFS_TYPE_VDEV ? | ||||||
|  | 	    ZFS_SYSFS_VDEV_PROPERTIES : ZFS_SYSFS_DATASET_PROPERTIES), name)); | ||||||
| #endif | #endif | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -235,6 +240,8 @@ propname_match(const char *p, size_t len, zprop_desc_t *prop_entry) | |||||||
| 	int c; | 	int c; | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
|  | 	ASSERT(propname != NULL); | ||||||
|  | 
 | ||||||
| 	if (len == strlen(propname) && | 	if (len == strlen(propname) && | ||||||
| 	    strncmp(p, propname, len) == 0) | 	    strncmp(p, propname, len) == 0) | ||||||
| 		return (B_TRUE); | 		return (B_TRUE); | ||||||
| @ -391,6 +398,18 @@ zprop_valid_for_type(int prop, zfs_type_t type, boolean_t headcheck) | |||||||
| 	return ((prop_tbl[prop].pd_types & type) != 0); | 	return ((prop_tbl[prop].pd_types & type) != 0); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * For user property names, we allow all lowercase alphanumeric characters, plus | ||||||
|  |  * a few useful punctuation characters. | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | zprop_valid_char(char c) | ||||||
|  | { | ||||||
|  | 	return ((c >= 'a' && c <= 'z') || | ||||||
|  | 	    (c >= '0' && c <= '9') || | ||||||
|  | 	    c == '-' || c == '_' || c == '.' || c == ':'); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| #ifndef _KERNEL | #ifndef _KERNEL | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
| @ -477,4 +496,5 @@ EXPORT_SYMBOL(zprop_index_to_string); | |||||||
| EXPORT_SYMBOL(zprop_random_value); | EXPORT_SYMBOL(zprop_random_value); | ||||||
| EXPORT_SYMBOL(zprop_values); | EXPORT_SYMBOL(zprop_values); | ||||||
| EXPORT_SYMBOL(zprop_valid_for_type); | EXPORT_SYMBOL(zprop_valid_for_type); | ||||||
|  | EXPORT_SYMBOL(zprop_valid_char); | ||||||
| #endif | #endif | ||||||
|  | |||||||
| @ -786,7 +786,7 @@ spa_prop_set(spa_t *spa, nvlist_t *nvp) | |||||||
| 			continue; | 			continue; | ||||||
| 
 | 
 | ||||||
| 		if (prop == ZPOOL_PROP_VERSION || prop == ZPOOL_PROP_INVAL) { | 		if (prop == ZPOOL_PROP_VERSION || prop == ZPOOL_PROP_INVAL) { | ||||||
| 			uint64_t ver; | 			uint64_t ver = 0; | ||||||
| 
 | 
 | ||||||
| 			if (prop == ZPOOL_PROP_VERSION) { | 			if (prop == ZPOOL_PROP_VERSION) { | ||||||
| 				VERIFY(nvpair_value_uint64(elem, &ver) == 0); | 				VERIFY(nvpair_value_uint64(elem, &ver) == 0); | ||||||
|  | |||||||
| @ -1833,36 +1833,27 @@ spa_update_dspace(spa_t *spa) | |||||||
| { | { | ||||||
| 	spa->spa_dspace = metaslab_class_get_dspace(spa_normal_class(spa)) + | 	spa->spa_dspace = metaslab_class_get_dspace(spa_normal_class(spa)) + | ||||||
| 	    ddt_get_dedup_dspace(spa); | 	    ddt_get_dedup_dspace(spa); | ||||||
| 	if (spa->spa_vdev_removal != NULL) { | 	if (spa->spa_nonallocating_dspace > 0) { | ||||||
| 		/*
 | 		/*
 | ||||||
| 		 * We can't allocate from the removing device, so subtract | 		 * Subtract the space provided by all non-allocating vdevs that | ||||||
| 		 * its size if it was included in dspace (i.e. if this is a | 		 * contribute to dspace.  If a file is overwritten, its old | ||||||
| 		 * normal-class vdev, not special/dedup).  This prevents the | 		 * blocks are freed and new blocks are allocated.  If there are | ||||||
| 		 * DMU/DSL from filling up the (now smaller) pool while we | 		 * no snapshots of the file, the available space should remain | ||||||
| 		 * are in the middle of removing the device. | 		 * the same.  The old blocks could be freed from the | ||||||
|  | 		 * non-allocating vdev, but the new blocks must be allocated on | ||||||
|  | 		 * other (allocating) vdevs.  By reserving the entire size of | ||||||
|  | 		 * the non-allocating vdevs (including allocated space), we | ||||||
|  | 		 * ensure that there will be enough space on the allocating | ||||||
|  | 		 * vdevs for this file overwrite to succeed. | ||||||
| 		 * | 		 * | ||||||
| 		 * Note that the DMU/DSL doesn't actually know or care | 		 * Note that the DMU/DSL doesn't actually know or care | ||||||
| 		 * how much space is allocated (it does its own tracking | 		 * how much space is allocated (it does its own tracking | ||||||
| 		 * of how much space has been logically used).  So it | 		 * of how much space has been logically used).  So it | ||||||
| 		 * doesn't matter that the data we are moving may be | 		 * doesn't matter that the data we are moving may be | ||||||
| 		 * allocated twice (on the old device and the new | 		 * allocated twice (on the old device and the new device). | ||||||
| 		 * device). |  | ||||||
| 		 */ | 		 */ | ||||||
| 		spa_config_enter(spa, SCL_VDEV, FTAG, RW_READER); | 		ASSERT3U(spa->spa_dspace, >=, spa->spa_nonallocating_dspace); | ||||||
| 		vdev_t *vd = | 		spa->spa_dspace -= spa->spa_nonallocating_dspace; | ||||||
| 		    vdev_lookup_top(spa, spa->spa_vdev_removal->svr_vdev_id); |  | ||||||
| 		/*
 |  | ||||||
| 		 * If the stars align, we can wind up here after |  | ||||||
| 		 * vdev_remove_complete() has cleared vd->vdev_mg but before |  | ||||||
| 		 * spa->spa_vdev_removal gets cleared, so we must check before |  | ||||||
| 		 * we dereference. |  | ||||||
| 		 */ |  | ||||||
| 		if (vd->vdev_mg && |  | ||||||
| 		    vd->vdev_mg->mg_class == spa_normal_class(spa)) { |  | ||||||
| 			spa->spa_dspace -= spa_deflate(spa) ? |  | ||||||
| 			    vd->vdev_stat.vs_dspace : vd->vdev_stat.vs_space; |  | ||||||
| 		} |  | ||||||
| 		spa_config_exit(spa, SCL_VDEV, FTAG); |  | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -2429,6 +2420,7 @@ spa_init(spa_mode_t mode) | |||||||
| 	zpool_prop_init(); | 	zpool_prop_init(); | ||||||
| 	zpool_feature_init(); | 	zpool_feature_init(); | ||||||
| 	spa_config_load(); | 	spa_config_load(); | ||||||
|  | 	vdev_prop_init(); | ||||||
| 	l2arc_start(); | 	l2arc_start(); | ||||||
| 	scan_init(); | 	scan_init(); | ||||||
| 	qat_init(); | 	qat_init(); | ||||||
|  | |||||||
| @ -28,6 +28,7 @@ | |||||||
|  * Copyright 2017 Joyent, Inc. |  * Copyright 2017 Joyent, Inc. | ||||||
|  * Copyright (c) 2017, Intel Corporation. |  * Copyright (c) 2017, Intel Corporation. | ||||||
|  * Copyright (c) 2019, Datto Inc. All rights reserved. |  * Copyright (c) 2019, Datto Inc. All rights reserved. | ||||||
|  |  * Copyright (c) 2021, Klara Inc. | ||||||
|  * Copyright [2021] Hewlett Packard Enterprise Development LP |  * Copyright [2021] Hewlett Packard Enterprise Development LP | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| @ -59,6 +60,7 @@ | |||||||
| #include <sys/vdev_trim.h> | #include <sys/vdev_trim.h> | ||||||
| #include <sys/zvol.h> | #include <sys/zvol.h> | ||||||
| #include <sys/zfs_ratelimit.h> | #include <sys/zfs_ratelimit.h> | ||||||
|  | #include "zfs_prop.h" | ||||||
| 
 | 
 | ||||||
| /*
 | /*
 | ||||||
|  * One metaslab from each (normal-class) vdev is used by the ZIL.  These are |  * One metaslab from each (normal-class) vdev is used by the ZIL.  These are | ||||||
| @ -865,6 +867,8 @@ vdev_alloc(spa_t *spa, vdev_t **vdp, nvlist_t *nv, vdev_t *parent, uint_t id, | |||||||
| 		    &vd->vdev_ms_shift); | 		    &vd->vdev_ms_shift); | ||||||
| 		(void) nvlist_lookup_uint64(nv, ZPOOL_CONFIG_ASIZE, | 		(void) nvlist_lookup_uint64(nv, ZPOOL_CONFIG_ASIZE, | ||||||
| 		    &vd->vdev_asize); | 		    &vd->vdev_asize); | ||||||
|  | 		(void) nvlist_lookup_uint64(nv, ZPOOL_CONFIG_NONALLOCATING, | ||||||
|  | 		    &vd->vdev_noalloc); | ||||||
| 		(void) nvlist_lookup_uint64(nv, ZPOOL_CONFIG_REMOVING, | 		(void) nvlist_lookup_uint64(nv, ZPOOL_CONFIG_REMOVING, | ||||||
| 		    &vd->vdev_removing); | 		    &vd->vdev_removing); | ||||||
| 		(void) nvlist_lookup_uint64(nv, ZPOOL_CONFIG_VDEV_TOP_ZAP, | 		(void) nvlist_lookup_uint64(nv, ZPOOL_CONFIG_VDEV_TOP_ZAP, | ||||||
| @ -1183,8 +1187,10 @@ vdev_top_transfer(vdev_t *svd, vdev_t *tvd) | |||||||
| 	ASSERT3P(tvd->vdev_indirect_mapping, ==, NULL); | 	ASSERT3P(tvd->vdev_indirect_mapping, ==, NULL); | ||||||
| 	ASSERT3P(tvd->vdev_indirect_births, ==, NULL); | 	ASSERT3P(tvd->vdev_indirect_births, ==, NULL); | ||||||
| 	ASSERT3P(tvd->vdev_obsolete_sm, ==, NULL); | 	ASSERT3P(tvd->vdev_obsolete_sm, ==, NULL); | ||||||
|  | 	ASSERT0(tvd->vdev_noalloc); | ||||||
| 	ASSERT0(tvd->vdev_removing); | 	ASSERT0(tvd->vdev_removing); | ||||||
| 	ASSERT0(tvd->vdev_rebuilding); | 	ASSERT0(tvd->vdev_rebuilding); | ||||||
|  | 	tvd->vdev_noalloc = svd->vdev_noalloc; | ||||||
| 	tvd->vdev_removing = svd->vdev_removing; | 	tvd->vdev_removing = svd->vdev_removing; | ||||||
| 	tvd->vdev_rebuilding = svd->vdev_rebuilding; | 	tvd->vdev_rebuilding = svd->vdev_rebuilding; | ||||||
| 	tvd->vdev_rebuild_config = svd->vdev_rebuild_config; | 	tvd->vdev_rebuild_config = svd->vdev_rebuild_config; | ||||||
| @ -1200,6 +1206,7 @@ vdev_top_transfer(vdev_t *svd, vdev_t *tvd) | |||||||
| 	svd->vdev_indirect_mapping = NULL; | 	svd->vdev_indirect_mapping = NULL; | ||||||
| 	svd->vdev_indirect_births = NULL; | 	svd->vdev_indirect_births = NULL; | ||||||
| 	svd->vdev_obsolete_sm = NULL; | 	svd->vdev_obsolete_sm = NULL; | ||||||
|  | 	svd->vdev_noalloc = 0; | ||||||
| 	svd->vdev_removing = 0; | 	svd->vdev_removing = 0; | ||||||
| 	svd->vdev_rebuilding = 0; | 	svd->vdev_rebuilding = 0; | ||||||
| 
 | 
 | ||||||
| @ -1498,11 +1505,15 @@ vdev_metaslab_init(vdev_t *vd, uint64_t txg) | |||||||
| 		spa_config_enter(spa, SCL_ALLOC, FTAG, RW_WRITER); | 		spa_config_enter(spa, SCL_ALLOC, FTAG, RW_WRITER); | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * If the vdev is being removed we don't activate | 	 * If the vdev is marked as non-allocating then don't | ||||||
| 	 * the metaslabs since we want to ensure that no new | 	 * activate the metaslabs since we want to ensure that | ||||||
| 	 * allocations are performed on this device. | 	 * no allocations are performed on this device. | ||||||
| 	 */ | 	 */ | ||||||
| 	if (!expanding && !vd->vdev_removing) { | 	if (vd->vdev_noalloc) { | ||||||
|  | 		/* track non-allocating vdev space */ | ||||||
|  | 		spa->spa_nonallocating_dspace += spa_deflate(spa) ? | ||||||
|  | 		    vd->vdev_stat.vs_dspace : vd->vdev_stat.vs_space; | ||||||
|  | 	} else if (!expanding) { | ||||||
| 		metaslab_group_activate(vd->vdev_mg); | 		metaslab_group_activate(vd->vdev_mg); | ||||||
| 		if (vd->vdev_log_mg != NULL) | 		if (vd->vdev_log_mg != NULL) | ||||||
| 			metaslab_group_activate(vd->vdev_log_mg); | 			metaslab_group_activate(vd->vdev_log_mg); | ||||||
| @ -4469,6 +4480,8 @@ vdev_get_stats_ex(vdev_t *vd, vdev_stat_t *vs, vdev_stat_ex_t *vsx) | |||||||
| 			vs->vs_fragmentation = (vd->vdev_mg != NULL) ? | 			vs->vs_fragmentation = (vd->vdev_mg != NULL) ? | ||||||
| 			    vd->vdev_mg->mg_fragmentation : 0; | 			    vd->vdev_mg->mg_fragmentation : 0; | ||||||
| 		} | 		} | ||||||
|  | 		vs->vs_noalloc = MAX(vd->vdev_noalloc, | ||||||
|  | 		    tvd ? tvd->vdev_noalloc : 0); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	vdev_get_stats_ex_impl(vd, vs, vsx); | 	vdev_get_stats_ex_impl(vd, vs, vsx); | ||||||
| @ -5375,6 +5388,23 @@ vdev_xlate_walk(vdev_t *vd, const range_seg64_t *logical_rs, | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static char * | ||||||
|  | vdev_name(vdev_t *vd, char *buf, int buflen) | ||||||
|  | { | ||||||
|  | 	if (vd->vdev_path == NULL) { | ||||||
|  | 		if (strcmp(vd->vdev_ops->vdev_op_type, "root") == 0) { | ||||||
|  | 			strlcpy(buf, vd->vdev_spa->spa_name, buflen); | ||||||
|  | 		} else if (!vd->vdev_ops->vdev_op_leaf) { | ||||||
|  | 			snprintf(buf, buflen, "%s-%llu", | ||||||
|  | 			    vd->vdev_ops->vdev_op_type, | ||||||
|  | 			    (u_longlong_t)vd->vdev_id); | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		strlcpy(buf, vd->vdev_path, buflen); | ||||||
|  | 	} | ||||||
|  | 	return (buf); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /*
 | /*
 | ||||||
|  * Look at the vdev tree and determine whether any devices are currently being |  * Look at the vdev tree and determine whether any devices are currently being | ||||||
|  * replaced. |  * replaced. | ||||||
| @ -5404,6 +5434,594 @@ vdev_replace_in_progress(vdev_t *vdev) | |||||||
| 	return (B_FALSE); | 	return (B_FALSE); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * Add a (source=src, propname=propval) list to an nvlist. | ||||||
|  |  */ | ||||||
|  | static void | ||||||
|  | vdev_prop_add_list(nvlist_t *nvl, const char *propname, char *strval, | ||||||
|  |     uint64_t intval, zprop_source_t src) | ||||||
|  | { | ||||||
|  | 	nvlist_t *propval; | ||||||
|  | 
 | ||||||
|  | 	propval = fnvlist_alloc(); | ||||||
|  | 	fnvlist_add_uint64(propval, ZPROP_SOURCE, src); | ||||||
|  | 
 | ||||||
|  | 	if (strval != NULL) | ||||||
|  | 		fnvlist_add_string(propval, ZPROP_VALUE, strval); | ||||||
|  | 	else | ||||||
|  | 		fnvlist_add_uint64(propval, ZPROP_VALUE, intval); | ||||||
|  | 
 | ||||||
|  | 	fnvlist_add_nvlist(nvl, propname, propval); | ||||||
|  | 	nvlist_free(propval); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void | ||||||
|  | vdev_props_set_sync(void *arg, dmu_tx_t *tx) | ||||||
|  | { | ||||||
|  | 	vdev_t *vd; | ||||||
|  | 	nvlist_t *nvp = arg; | ||||||
|  | 	spa_t *spa = dmu_tx_pool(tx)->dp_spa; | ||||||
|  | 	objset_t *mos = spa->spa_meta_objset; | ||||||
|  | 	nvpair_t *elem = NULL; | ||||||
|  | 	uint64_t vdev_guid; | ||||||
|  | 	nvlist_t *nvprops; | ||||||
|  | 
 | ||||||
|  | 	vdev_guid = fnvlist_lookup_uint64(nvp, ZPOOL_VDEV_PROPS_SET_VDEV); | ||||||
|  | 	nvprops = fnvlist_lookup_nvlist(nvp, ZPOOL_VDEV_PROPS_SET_PROPS); | ||||||
|  | 	vd = spa_lookup_by_guid(spa, vdev_guid, B_TRUE); | ||||||
|  | 	VERIFY(vd != NULL); | ||||||
|  | 
 | ||||||
|  | 	mutex_enter(&spa->spa_props_lock); | ||||||
|  | 
 | ||||||
|  | 	while ((elem = nvlist_next_nvpair(nvprops, elem)) != NULL) { | ||||||
|  | 		uint64_t intval, objid = 0; | ||||||
|  | 		char *strval; | ||||||
|  | 		vdev_prop_t prop; | ||||||
|  | 		const char *propname = nvpair_name(elem); | ||||||
|  | 		zprop_type_t proptype; | ||||||
|  | 
 | ||||||
|  | 		/*
 | ||||||
|  | 		 * Set vdev property values in the vdev props mos object. | ||||||
|  | 		 */ | ||||||
|  | 		if (vd->vdev_top_zap != 0) { | ||||||
|  | 			objid = vd->vdev_top_zap; | ||||||
|  | 		} else if (vd->vdev_leaf_zap != 0) { | ||||||
|  | 			objid = vd->vdev_leaf_zap; | ||||||
|  | 		} else { | ||||||
|  | 			panic("vdev not top or leaf"); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		switch (prop = vdev_name_to_prop(propname)) { | ||||||
|  | 		case VDEV_PROP_USER: | ||||||
|  | 			if (vdev_prop_user(propname)) { | ||||||
|  | 				strval = fnvpair_value_string(elem); | ||||||
|  | 				if (strlen(strval) == 0) { | ||||||
|  | 					/* remove the property if value == "" */ | ||||||
|  | 					(void) zap_remove(mos, objid, propname, | ||||||
|  | 					    tx); | ||||||
|  | 				} else { | ||||||
|  | 					VERIFY0(zap_update(mos, objid, propname, | ||||||
|  | 					    1, strlen(strval) + 1, strval, tx)); | ||||||
|  | 				} | ||||||
|  | 				spa_history_log_internal(spa, "vdev set", tx, | ||||||
|  | 				    "vdev_guid=%llu: %s=%s", | ||||||
|  | 				    (u_longlong_t)vdev_guid, nvpair_name(elem), | ||||||
|  | 				    strval); | ||||||
|  | 			} | ||||||
|  | 			break; | ||||||
|  | 		default: | ||||||
|  | 			/* normalize the property name */ | ||||||
|  | 			propname = vdev_prop_to_name(prop); | ||||||
|  | 			proptype = vdev_prop_get_type(prop); | ||||||
|  | 
 | ||||||
|  | 			if (nvpair_type(elem) == DATA_TYPE_STRING) { | ||||||
|  | 				ASSERT(proptype == PROP_TYPE_STRING); | ||||||
|  | 				strval = fnvpair_value_string(elem); | ||||||
|  | 				VERIFY0(zap_update(mos, objid, propname, | ||||||
|  | 				    1, strlen(strval) + 1, strval, tx)); | ||||||
|  | 				spa_history_log_internal(spa, "vdev set", tx, | ||||||
|  | 				    "vdev_guid=%llu: %s=%s", | ||||||
|  | 				    (u_longlong_t)vdev_guid, nvpair_name(elem), | ||||||
|  | 				    strval); | ||||||
|  | 			} else if (nvpair_type(elem) == DATA_TYPE_UINT64) { | ||||||
|  | 				intval = fnvpair_value_uint64(elem); | ||||||
|  | 
 | ||||||
|  | 				if (proptype == PROP_TYPE_INDEX) { | ||||||
|  | 					const char *unused; | ||||||
|  | 					VERIFY0(vdev_prop_index_to_string( | ||||||
|  | 					    prop, intval, &unused)); | ||||||
|  | 				} | ||||||
|  | 				VERIFY0(zap_update(mos, objid, propname, | ||||||
|  | 				    sizeof (uint64_t), 1, &intval, tx)); | ||||||
|  | 				spa_history_log_internal(spa, "vdev set", tx, | ||||||
|  | 				    "vdev_guid=%llu: %s=%lld", | ||||||
|  | 				    (u_longlong_t)vdev_guid, | ||||||
|  | 				    nvpair_name(elem), (longlong_t)intval); | ||||||
|  | 			} else { | ||||||
|  | 				panic("invalid vdev property type %u", | ||||||
|  | 				    nvpair_type(elem)); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	mutex_exit(&spa->spa_props_lock); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | vdev_prop_set(vdev_t *vd, nvlist_t *innvl, nvlist_t *outnvl) | ||||||
|  | { | ||||||
|  | 	spa_t *spa = vd->vdev_spa; | ||||||
|  | 	nvpair_t *elem = NULL; | ||||||
|  | 	uint64_t vdev_guid; | ||||||
|  | 	nvlist_t *nvprops; | ||||||
|  | 	int error; | ||||||
|  | 
 | ||||||
|  | 	ASSERT(vd != NULL); | ||||||
|  | 
 | ||||||
|  | 	if (nvlist_lookup_uint64(innvl, ZPOOL_VDEV_PROPS_SET_VDEV, | ||||||
|  | 	    &vdev_guid) != 0) | ||||||
|  | 		return (SET_ERROR(EINVAL)); | ||||||
|  | 
 | ||||||
|  | 	if (nvlist_lookup_nvlist(innvl, ZPOOL_VDEV_PROPS_SET_PROPS, | ||||||
|  | 	    &nvprops) != 0) | ||||||
|  | 		return (SET_ERROR(EINVAL)); | ||||||
|  | 
 | ||||||
|  | 	if ((vd = spa_lookup_by_guid(spa, vdev_guid, B_TRUE)) == NULL) | ||||||
|  | 		return (SET_ERROR(EINVAL)); | ||||||
|  | 
 | ||||||
|  | 	while ((elem = nvlist_next_nvpair(nvprops, elem)) != NULL) { | ||||||
|  | 		char *propname = nvpair_name(elem); | ||||||
|  | 		vdev_prop_t prop = vdev_name_to_prop(propname); | ||||||
|  | 		uint64_t intval = 0; | ||||||
|  | 		char *strval = NULL; | ||||||
|  | 
 | ||||||
|  | 		if (prop == VDEV_PROP_USER && !vdev_prop_user(propname)) { | ||||||
|  | 			error = EINVAL; | ||||||
|  | 			goto end; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (vdev_prop_readonly(prop)) { | ||||||
|  | 			error = EROFS; | ||||||
|  | 			goto end; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		/* Special Processing */ | ||||||
|  | 		switch (prop) { | ||||||
|  | 		case VDEV_PROP_PATH: | ||||||
|  | 			if (vd->vdev_path == NULL) { | ||||||
|  | 				error = EROFS; | ||||||
|  | 				break; | ||||||
|  | 			} | ||||||
|  | 			if (nvpair_value_string(elem, &strval) != 0) { | ||||||
|  | 				error = EINVAL; | ||||||
|  | 				break; | ||||||
|  | 			} | ||||||
|  | 			/* New path must start with /dev/ */ | ||||||
|  | 			if (strncmp(strval, "/dev/", 5)) { | ||||||
|  | 				error = EINVAL; | ||||||
|  | 				break; | ||||||
|  | 			} | ||||||
|  | 			error = spa_vdev_setpath(spa, vdev_guid, strval); | ||||||
|  | 			break; | ||||||
|  | 		case VDEV_PROP_ALLOCATING: | ||||||
|  | 			if (nvpair_value_uint64(elem, &intval) != 0) { | ||||||
|  | 				error = EINVAL; | ||||||
|  | 				break; | ||||||
|  | 			} | ||||||
|  | 			if (intval != vd->vdev_noalloc) | ||||||
|  | 				break; | ||||||
|  | 			if (intval == 0) | ||||||
|  | 				error = spa_vdev_noalloc(spa, vdev_guid); | ||||||
|  | 			else | ||||||
|  | 				error = spa_vdev_alloc(spa, vdev_guid); | ||||||
|  | 			break; | ||||||
|  | 		default: | ||||||
|  | 			/* Most processing is done in vdev_props_set_sync */ | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | end: | ||||||
|  | 		if (error != 0) { | ||||||
|  | 			intval = error; | ||||||
|  | 			vdev_prop_add_list(outnvl, propname, strval, intval, 0); | ||||||
|  | 			return (error); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return (dsl_sync_task(spa->spa_name, NULL, vdev_props_set_sync, | ||||||
|  | 	    innvl, 6, ZFS_SPACE_CHECK_EXTRA_RESERVED)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | vdev_prop_get(vdev_t *vd, nvlist_t *innvl, nvlist_t *outnvl) | ||||||
|  | { | ||||||
|  | 	spa_t *spa = vd->vdev_spa; | ||||||
|  | 	objset_t *mos = spa->spa_meta_objset; | ||||||
|  | 	int err = 0; | ||||||
|  | 	uint64_t objid; | ||||||
|  | 	uint64_t vdev_guid; | ||||||
|  | 	nvpair_t *elem = NULL; | ||||||
|  | 	nvlist_t *nvprops = NULL; | ||||||
|  | 	uint64_t intval = 0; | ||||||
|  | 	char *strval = NULL; | ||||||
|  | 	const char *propname = NULL; | ||||||
|  | 	vdev_prop_t prop; | ||||||
|  | 
 | ||||||
|  | 	ASSERT(vd != NULL); | ||||||
|  | 	ASSERT(mos != NULL); | ||||||
|  | 
 | ||||||
|  | 	if (nvlist_lookup_uint64(innvl, ZPOOL_VDEV_PROPS_GET_VDEV, | ||||||
|  | 	    &vdev_guid) != 0) | ||||||
|  | 		return (SET_ERROR(EINVAL)); | ||||||
|  | 
 | ||||||
|  | 	nvlist_lookup_nvlist(innvl, ZPOOL_VDEV_PROPS_GET_PROPS, &nvprops); | ||||||
|  | 
 | ||||||
|  | 	if (vd->vdev_top_zap != 0) { | ||||||
|  | 		objid = vd->vdev_top_zap; | ||||||
|  | 	} else if (vd->vdev_leaf_zap != 0) { | ||||||
|  | 		objid = vd->vdev_leaf_zap; | ||||||
|  | 	} else { | ||||||
|  | 		return (SET_ERROR(EINVAL)); | ||||||
|  | 	} | ||||||
|  | 	ASSERT(objid != 0); | ||||||
|  | 
 | ||||||
|  | 	mutex_enter(&spa->spa_props_lock); | ||||||
|  | 
 | ||||||
|  | 	if (nvprops != NULL) { | ||||||
|  | 		char namebuf[64] = { 0 }; | ||||||
|  | 
 | ||||||
|  | 		while ((elem = nvlist_next_nvpair(nvprops, elem)) != NULL) { | ||||||
|  | 			intval = 0; | ||||||
|  | 			strval = NULL; | ||||||
|  | 			propname = nvpair_name(elem); | ||||||
|  | 			prop = vdev_name_to_prop(propname); | ||||||
|  | 			zprop_source_t src = ZPROP_SRC_DEFAULT; | ||||||
|  | 			uint64_t integer_size, num_integers; | ||||||
|  | 
 | ||||||
|  | 			switch (prop) { | ||||||
|  | 			/* Special Read-only Properties */ | ||||||
|  | 			case VDEV_PROP_NAME: | ||||||
|  | 				strval = vdev_name(vd, namebuf, | ||||||
|  | 				    sizeof (namebuf)); | ||||||
|  | 				if (strval == NULL) | ||||||
|  | 					continue; | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, strval, 0, | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_CAPACITY: | ||||||
|  | 				/* percent used */ | ||||||
|  | 				intval = (vd->vdev_stat.vs_dspace == 0) ? 0 : | ||||||
|  | 				    (vd->vdev_stat.vs_alloc * 100 / | ||||||
|  | 				    vd->vdev_stat.vs_dspace); | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    intval, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_STATE: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_state, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_GUID: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_guid, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_ASIZE: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_asize, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_PSIZE: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_psize, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_ASHIFT: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_ashift, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_SIZE: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_dspace, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_FREE: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_dspace - | ||||||
|  | 				    vd->vdev_stat.vs_alloc, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_ALLOCATED: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_alloc, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_EXPANDSZ: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_esize, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_FRAGMENTATION: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_fragmentation, | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_PARITY: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vdev_get_nparity(vd), ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_PATH: | ||||||
|  | 				if (vd->vdev_path == NULL) | ||||||
|  | 					continue; | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, | ||||||
|  | 				    vd->vdev_path, 0, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_DEVID: | ||||||
|  | 				if (vd->vdev_devid == NULL) | ||||||
|  | 					continue; | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, | ||||||
|  | 				    vd->vdev_devid, 0, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_PHYS_PATH: | ||||||
|  | 				if (vd->vdev_physpath == NULL) | ||||||
|  | 					continue; | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, | ||||||
|  | 				    vd->vdev_physpath, 0, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_ENC_PATH: | ||||||
|  | 				if (vd->vdev_enc_sysfs_path == NULL) | ||||||
|  | 					continue; | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, | ||||||
|  | 				    vd->vdev_enc_sysfs_path, 0, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_FRU: | ||||||
|  | 				if (vd->vdev_fru == NULL) | ||||||
|  | 					continue; | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, | ||||||
|  | 				    vd->vdev_fru, 0, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_PARENT: | ||||||
|  | 				if (vd->vdev_parent != NULL) { | ||||||
|  | 					strval = vdev_name(vd->vdev_parent, | ||||||
|  | 					    namebuf, sizeof (namebuf)); | ||||||
|  | 					vdev_prop_add_list(outnvl, propname, | ||||||
|  | 					    strval, 0, ZPROP_SRC_NONE); | ||||||
|  | 				} | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_CHILDREN: | ||||||
|  | 				if (vd->vdev_children > 0) | ||||||
|  | 					strval = kmem_zalloc(ZAP_MAXVALUELEN, | ||||||
|  | 					    KM_SLEEP); | ||||||
|  | 				for (uint64_t i = 0; i < vd->vdev_children; | ||||||
|  | 				    i++) { | ||||||
|  | 					char *vname; | ||||||
|  | 
 | ||||||
|  | 					vname = vdev_name(vd->vdev_child[i], | ||||||
|  | 					    namebuf, sizeof (namebuf)); | ||||||
|  | 					if (vname == NULL) | ||||||
|  | 						vname = "(unknown)"; | ||||||
|  | 					if (strlen(strval) > 0) | ||||||
|  | 						strlcat(strval, ",", | ||||||
|  | 						    ZAP_MAXVALUELEN); | ||||||
|  | 					strlcat(strval, vname, ZAP_MAXVALUELEN); | ||||||
|  | 				} | ||||||
|  | 				if (strval != NULL) { | ||||||
|  | 					vdev_prop_add_list(outnvl, propname, | ||||||
|  | 					    strval, 0, ZPROP_SRC_NONE); | ||||||
|  | 					kmem_free(strval, ZAP_MAXVALUELEN); | ||||||
|  | 				} | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_NUMCHILDREN: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_children, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_READ_ERRORS: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_read_errors, | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_WRITE_ERRORS: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_write_errors, | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_CHECKSUM_ERRORS: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_checksum_errors, | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_INITIALIZE_ERRORS: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_initialize_errors, | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_OPS_NULL: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_ops[ZIO_TYPE_NULL], | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_OPS_READ: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_ops[ZIO_TYPE_READ], | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_OPS_WRITE: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_ops[ZIO_TYPE_WRITE], | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_OPS_FREE: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_ops[ZIO_TYPE_FREE], | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_OPS_CLAIM: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_ops[ZIO_TYPE_CLAIM], | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_OPS_TRIM: | ||||||
|  | 				/*
 | ||||||
|  | 				 * TRIM ops and bytes are reported to user | ||||||
|  | 				 * space as ZIO_TYPE_IOCTL.  This is done to | ||||||
|  | 				 * preserve the vdev_stat_t structure layout | ||||||
|  | 				 * for user space. | ||||||
|  | 				 */ | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_ops[ZIO_TYPE_IOCTL], | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_BYTES_NULL: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_bytes[ZIO_TYPE_NULL], | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_BYTES_READ: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_bytes[ZIO_TYPE_READ], | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_BYTES_WRITE: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_bytes[ZIO_TYPE_WRITE], | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_BYTES_FREE: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_bytes[ZIO_TYPE_FREE], | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_BYTES_CLAIM: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_bytes[ZIO_TYPE_CLAIM], | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_BYTES_TRIM: | ||||||
|  | 				/*
 | ||||||
|  | 				 * TRIM ops and bytes are reported to user | ||||||
|  | 				 * space as ZIO_TYPE_IOCTL.  This is done to | ||||||
|  | 				 * preserve the vdev_stat_t structure layout | ||||||
|  | 				 * for user space. | ||||||
|  | 				 */ | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_stat.vs_bytes[ZIO_TYPE_IOCTL], | ||||||
|  | 				    ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			case VDEV_PROP_REMOVING: | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, NULL, | ||||||
|  | 				    vd->vdev_removing, ZPROP_SRC_NONE); | ||||||
|  | 				continue; | ||||||
|  | 			/* Numeric Properites */ | ||||||
|  | 			case VDEV_PROP_ALLOCATING: | ||||||
|  | 				src = ZPROP_SRC_LOCAL; | ||||||
|  | 				strval = NULL; | ||||||
|  | 
 | ||||||
|  | 				err = zap_lookup(mos, objid, nvpair_name(elem), | ||||||
|  | 				    sizeof (uint64_t), 1, &intval); | ||||||
|  | 				if (err == ENOENT) { | ||||||
|  | 					intval = | ||||||
|  | 					    vdev_prop_default_numeric(prop); | ||||||
|  | 					err = 0; | ||||||
|  | 				} else if (err) | ||||||
|  | 					break; | ||||||
|  | 				if (intval == vdev_prop_default_numeric(prop)) | ||||||
|  | 					src = ZPROP_SRC_DEFAULT; | ||||||
|  | 
 | ||||||
|  | 				/* Leaf vdevs cannot have this property */ | ||||||
|  | 				if (vd->vdev_mg == NULL && | ||||||
|  | 				    vd->vdev_top != NULL) { | ||||||
|  | 					src = ZPROP_SRC_NONE; | ||||||
|  | 					intval = ZPROP_BOOLEAN_NA; | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, strval, | ||||||
|  | 				    intval, src); | ||||||
|  | 				break; | ||||||
|  | 			/* Text Properties */ | ||||||
|  | 			case VDEV_PROP_COMMENT: | ||||||
|  | 				/* Exists in the ZAP below */ | ||||||
|  | 				/* FALLTHRU */ | ||||||
|  | 			case VDEV_PROP_USER: | ||||||
|  | 				/* User Properites */ | ||||||
|  | 				src = ZPROP_SRC_LOCAL; | ||||||
|  | 
 | ||||||
|  | 				err = zap_length(mos, objid, nvpair_name(elem), | ||||||
|  | 				    &integer_size, &num_integers); | ||||||
|  | 				if (err) | ||||||
|  | 					break; | ||||||
|  | 
 | ||||||
|  | 				switch (integer_size) { | ||||||
|  | 				case 8: | ||||||
|  | 					/* User properties cannot be integers */ | ||||||
|  | 					err = EINVAL; | ||||||
|  | 					break; | ||||||
|  | 				case 1: | ||||||
|  | 					/* string property */ | ||||||
|  | 					strval = kmem_alloc(num_integers, | ||||||
|  | 					    KM_SLEEP); | ||||||
|  | 					err = zap_lookup(mos, objid, | ||||||
|  | 					    nvpair_name(elem), 1, | ||||||
|  | 					    num_integers, strval); | ||||||
|  | 					if (err) { | ||||||
|  | 						kmem_free(strval, | ||||||
|  | 						    num_integers); | ||||||
|  | 						break; | ||||||
|  | 					} | ||||||
|  | 					vdev_prop_add_list(outnvl, propname, | ||||||
|  | 					    strval, 0, src); | ||||||
|  | 					kmem_free(strval, num_integers); | ||||||
|  | 					break; | ||||||
|  | 				} | ||||||
|  | 				break; | ||||||
|  | 			default: | ||||||
|  | 				err = ENOENT; | ||||||
|  | 				break; | ||||||
|  | 			} | ||||||
|  | 			if (err) | ||||||
|  | 				break; | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		/*
 | ||||||
|  | 		 * Get all properties from the MOS vdev property object. | ||||||
|  | 		 */ | ||||||
|  | 		zap_cursor_t zc; | ||||||
|  | 		zap_attribute_t za; | ||||||
|  | 		for (zap_cursor_init(&zc, mos, objid); | ||||||
|  | 		    (err = zap_cursor_retrieve(&zc, &za)) == 0; | ||||||
|  | 		    zap_cursor_advance(&zc)) { | ||||||
|  | 			intval = 0; | ||||||
|  | 			strval = NULL; | ||||||
|  | 			zprop_source_t src = ZPROP_SRC_DEFAULT; | ||||||
|  | 			propname = za.za_name; | ||||||
|  | 			prop = vdev_name_to_prop(propname); | ||||||
|  | 
 | ||||||
|  | 			switch (za.za_integer_length) { | ||||||
|  | 			case 8: | ||||||
|  | 				/* We do not allow integer user properties */ | ||||||
|  | 				/* This is likely an internal value */ | ||||||
|  | 				break; | ||||||
|  | 			case 1: | ||||||
|  | 				/* string property */ | ||||||
|  | 				strval = kmem_alloc(za.za_num_integers, | ||||||
|  | 				    KM_SLEEP); | ||||||
|  | 				err = zap_lookup(mos, objid, za.za_name, 1, | ||||||
|  | 				    za.za_num_integers, strval); | ||||||
|  | 				if (err) { | ||||||
|  | 					kmem_free(strval, za.za_num_integers); | ||||||
|  | 					break; | ||||||
|  | 				} | ||||||
|  | 				vdev_prop_add_list(outnvl, propname, strval, 0, | ||||||
|  | 				    src); | ||||||
|  | 				kmem_free(strval, za.za_num_integers); | ||||||
|  | 				break; | ||||||
|  | 
 | ||||||
|  | 			default: | ||||||
|  | 				break; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		zap_cursor_fini(&zc); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	mutex_exit(&spa->spa_props_lock); | ||||||
|  | 	if (err && err != ENOENT) { | ||||||
|  | 		return (err); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return (0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| EXPORT_SYMBOL(vdev_fault); | EXPORT_SYMBOL(vdev_fault); | ||||||
| EXPORT_SYMBOL(vdev_degrade); | EXPORT_SYMBOL(vdev_degrade); | ||||||
| EXPORT_SYMBOL(vdev_online); | EXPORT_SYMBOL(vdev_online); | ||||||
|  | |||||||
| @ -496,6 +496,10 @@ vdev_config_generate(spa_t *spa, vdev_t *vd, boolean_t getstats, | |||||||
| 		fnvlist_add_uint64(nv, ZPOOL_CONFIG_ASIZE, | 		fnvlist_add_uint64(nv, ZPOOL_CONFIG_ASIZE, | ||||||
| 		    vd->vdev_asize); | 		    vd->vdev_asize); | ||||||
| 		fnvlist_add_uint64(nv, ZPOOL_CONFIG_IS_LOG, vd->vdev_islog); | 		fnvlist_add_uint64(nv, ZPOOL_CONFIG_IS_LOG, vd->vdev_islog); | ||||||
|  | 		if (vd->vdev_noalloc) { | ||||||
|  | 			fnvlist_add_uint64(nv, ZPOOL_CONFIG_NONALLOCATING, | ||||||
|  | 			    vd->vdev_noalloc); | ||||||
|  | 		} | ||||||
| 		if (vd->vdev_removing) { | 		if (vd->vdev_removing) { | ||||||
| 			fnvlist_add_uint64(nv, ZPOOL_CONFIG_REMOVING, | 			fnvlist_add_uint64(nv, ZPOOL_CONFIG_REMOVING, | ||||||
| 			    vd->vdev_removing); | 			    vd->vdev_removing); | ||||||
|  | |||||||
| @ -167,6 +167,176 @@ spa_nvlist_lookup_by_guid(nvlist_t **nvpp, int count, uint64_t target_guid) | |||||||
| 	return (NULL); | 	return (NULL); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static void | ||||||
|  | vdev_activate(vdev_t *vd) | ||||||
|  | { | ||||||
|  | 	metaslab_group_t *mg = vd->vdev_mg; | ||||||
|  | 	spa_t *spa = vd->vdev_spa; | ||||||
|  | 	uint64_t vdev_space = spa_deflate(spa) ? | ||||||
|  | 	    vd->vdev_stat.vs_dspace : vd->vdev_stat.vs_space; | ||||||
|  | 
 | ||||||
|  | 	ASSERT(!vd->vdev_islog); | ||||||
|  | 	ASSERT(vd->vdev_noalloc); | ||||||
|  | 
 | ||||||
|  | 	metaslab_group_activate(mg); | ||||||
|  | 	metaslab_group_activate(vd->vdev_log_mg); | ||||||
|  | 
 | ||||||
|  | 	ASSERT3U(spa->spa_nonallocating_dspace, >=, vdev_space); | ||||||
|  | 
 | ||||||
|  | 	spa->spa_nonallocating_dspace -= vdev_space; | ||||||
|  | 
 | ||||||
|  | 	vd->vdev_noalloc = B_FALSE; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | vdev_passivate(vdev_t *vd, uint64_t *txg) | ||||||
|  | { | ||||||
|  | 	spa_t *spa = vd->vdev_spa; | ||||||
|  | 	int error; | ||||||
|  | 
 | ||||||
|  | 	ASSERT(!vd->vdev_noalloc); | ||||||
|  | 
 | ||||||
|  | 	vdev_t *rvd = spa->spa_root_vdev; | ||||||
|  | 	metaslab_group_t *mg = vd->vdev_mg; | ||||||
|  | 	metaslab_class_t *normal = spa_normal_class(spa); | ||||||
|  | 	if (mg->mg_class == normal) { | ||||||
|  | 		/*
 | ||||||
|  | 		 * We must check that this is not the only allocating device in | ||||||
|  | 		 * the pool before passivating, otherwise we will not be able | ||||||
|  | 		 * to make progress because we can't allocate from any vdevs. | ||||||
|  | 		 */ | ||||||
|  | 		boolean_t last = B_TRUE; | ||||||
|  | 		for (uint64_t id = 0; id < rvd->vdev_children; id++) { | ||||||
|  | 			vdev_t *cvd = rvd->vdev_child[id]; | ||||||
|  | 
 | ||||||
|  | 			if (cvd == vd || | ||||||
|  | 			    cvd->vdev_ops == &vdev_indirect_ops) | ||||||
|  | 				continue; | ||||||
|  | 
 | ||||||
|  | 			metaslab_class_t *mc = cvd->vdev_mg->mg_class; | ||||||
|  | 			if (mc != normal) | ||||||
|  | 				continue; | ||||||
|  | 
 | ||||||
|  | 			if (!cvd->vdev_noalloc) { | ||||||
|  | 				last = B_FALSE; | ||||||
|  | 				break; | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		if (last) | ||||||
|  | 			return (SET_ERROR(EINVAL)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	metaslab_group_passivate(mg); | ||||||
|  | 	ASSERT(!vd->vdev_islog); | ||||||
|  | 	metaslab_group_passivate(vd->vdev_log_mg); | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * Wait for the youngest allocations and frees to sync, | ||||||
|  | 	 * and then wait for the deferral of those frees to finish. | ||||||
|  | 	 */ | ||||||
|  | 	spa_vdev_config_exit(spa, NULL, | ||||||
|  | 	    *txg + TXG_CONCURRENT_STATES + TXG_DEFER_SIZE, 0, FTAG); | ||||||
|  | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * We must ensure that no "stubby" log blocks are allocated | ||||||
|  | 	 * on the device to be removed.  These blocks could be | ||||||
|  | 	 * written at any time, including while we are in the middle | ||||||
|  | 	 * of copying them. | ||||||
|  | 	 */ | ||||||
|  | 	error = spa_reset_logs(spa); | ||||||
|  | 
 | ||||||
|  | 	*txg = spa_vdev_config_enter(spa); | ||||||
|  | 
 | ||||||
|  | 	if (error != 0) { | ||||||
|  | 		metaslab_group_activate(mg); | ||||||
|  | 		ASSERT(!vd->vdev_islog); | ||||||
|  | 		if (vd->vdev_log_mg != NULL) | ||||||
|  | 			metaslab_group_activate(vd->vdev_log_mg); | ||||||
|  | 		return (error); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	spa->spa_nonallocating_dspace += spa_deflate(spa) ? | ||||||
|  | 	    vd->vdev_stat.vs_dspace : vd->vdev_stat.vs_space; | ||||||
|  | 	vd->vdev_noalloc = B_TRUE; | ||||||
|  | 
 | ||||||
|  | 	return (0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * Turn off allocations for a top-level device from the pool. | ||||||
|  |  * | ||||||
|  |  * Turning off allocations for a top-level device can take a significant | ||||||
|  |  * amount of time. As a result we use the spa_vdev_config_[enter/exit] | ||||||
|  |  * functions which allow us to grab and release the spa_config_lock while | ||||||
|  |  * still holding the namespace lock. During each step the configuration | ||||||
|  |  * is synced out. | ||||||
|  |  */ | ||||||
|  | int | ||||||
|  | spa_vdev_noalloc(spa_t *spa, uint64_t guid) | ||||||
|  | { | ||||||
|  | 	vdev_t *vd; | ||||||
|  | 	uint64_t txg; | ||||||
|  | 	int error = 0; | ||||||
|  | 
 | ||||||
|  | 	ASSERT(!MUTEX_HELD(&spa_namespace_lock)); | ||||||
|  | 	ASSERT(spa_writeable(spa)); | ||||||
|  | 
 | ||||||
|  | 	txg = spa_vdev_enter(spa); | ||||||
|  | 
 | ||||||
|  | 	ASSERT(MUTEX_HELD(&spa_namespace_lock)); | ||||||
|  | 
 | ||||||
|  | 	vd = spa_lookup_by_guid(spa, guid, B_FALSE); | ||||||
|  | 
 | ||||||
|  | 	if (vd == NULL) | ||||||
|  | 		error = SET_ERROR(ENOENT); | ||||||
|  | 	else if (vd->vdev_mg == NULL) | ||||||
|  | 		error = SET_ERROR(ZFS_ERR_VDEV_NOTSUP); | ||||||
|  | 	else if (!vd->vdev_noalloc) | ||||||
|  | 		error = vdev_passivate(vd, &txg); | ||||||
|  | 
 | ||||||
|  | 	if (error == 0) { | ||||||
|  | 		vdev_dirty_leaves(vd, VDD_DTL, txg); | ||||||
|  | 		vdev_config_dirty(vd); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	error = spa_vdev_exit(spa, NULL, txg, error); | ||||||
|  | 
 | ||||||
|  | 	return (error); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int | ||||||
|  | spa_vdev_alloc(spa_t *spa, uint64_t guid) | ||||||
|  | { | ||||||
|  | 	vdev_t *vd; | ||||||
|  | 	uint64_t txg; | ||||||
|  | 	int error = 0; | ||||||
|  | 
 | ||||||
|  | 	ASSERT(!MUTEX_HELD(&spa_namespace_lock)); | ||||||
|  | 	ASSERT(spa_writeable(spa)); | ||||||
|  | 
 | ||||||
|  | 	txg = spa_vdev_enter(spa); | ||||||
|  | 
 | ||||||
|  | 	ASSERT(MUTEX_HELD(&spa_namespace_lock)); | ||||||
|  | 
 | ||||||
|  | 	vd = spa_lookup_by_guid(spa, guid, B_FALSE); | ||||||
|  | 
 | ||||||
|  | 	if (vd == NULL) | ||||||
|  | 		error = SET_ERROR(ENOENT); | ||||||
|  | 	else if (vd->vdev_mg == NULL) | ||||||
|  | 		error = SET_ERROR(ZFS_ERR_VDEV_NOTSUP); | ||||||
|  | 	else if (!vd->vdev_removing) | ||||||
|  | 		vdev_activate(vd); | ||||||
|  | 
 | ||||||
|  | 	if (error == 0) { | ||||||
|  | 		vdev_dirty_leaves(vd, VDD_DTL, txg); | ||||||
|  | 		vdev_config_dirty(vd); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	(void) spa_vdev_exit(spa, NULL, txg, error); | ||||||
|  | 
 | ||||||
|  | 	return (error); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static void | static void | ||||||
| spa_vdev_remove_aux(nvlist_t *config, char *name, nvlist_t **dev, int count, | spa_vdev_remove_aux(nvlist_t *config, char *name, nvlist_t **dev, int count, | ||||||
|     nvlist_t *dev_to_remove) |     nvlist_t *dev_to_remove) | ||||||
| @ -1193,6 +1363,8 @@ vdev_remove_complete(spa_t *spa) | |||||||
| 	ASSERT3P(vd->vdev_initialize_thread, ==, NULL); | 	ASSERT3P(vd->vdev_initialize_thread, ==, NULL); | ||||||
| 	ASSERT3P(vd->vdev_trim_thread, ==, NULL); | 	ASSERT3P(vd->vdev_trim_thread, ==, NULL); | ||||||
| 	ASSERT3P(vd->vdev_autotrim_thread, ==, NULL); | 	ASSERT3P(vd->vdev_autotrim_thread, ==, NULL); | ||||||
|  | 	uint64_t vdev_space = spa_deflate(spa) ? | ||||||
|  | 	    vd->vdev_stat.vs_dspace : vd->vdev_stat.vs_space; | ||||||
| 
 | 
 | ||||||
| 	sysevent_t *ev = spa_event_create(spa, vd, NULL, | 	sysevent_t *ev = spa_event_create(spa, vd, NULL, | ||||||
| 	    ESC_ZFS_VDEV_REMOVE_DEV); | 	    ESC_ZFS_VDEV_REMOVE_DEV); | ||||||
| @ -1200,6 +1372,12 @@ vdev_remove_complete(spa_t *spa) | |||||||
| 	zfs_dbgmsg("finishing device removal for vdev %llu in txg %llu", | 	zfs_dbgmsg("finishing device removal for vdev %llu in txg %llu", | ||||||
| 	    (u_longlong_t)vd->vdev_id, (u_longlong_t)txg); | 	    (u_longlong_t)vd->vdev_id, (u_longlong_t)txg); | ||||||
| 
 | 
 | ||||||
|  | 	ASSERT3U(0, !=, vdev_space); | ||||||
|  | 	ASSERT3U(spa->spa_nonallocating_dspace, >=, vdev_space); | ||||||
|  | 
 | ||||||
|  | 	/* the vdev is no longer part of the dspace */ | ||||||
|  | 	spa->spa_nonallocating_dspace -= vdev_space; | ||||||
|  | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Discard allocation state. | 	 * Discard allocation state. | ||||||
| 	 */ | 	 */ | ||||||
| @ -1619,6 +1797,28 @@ spa_vdev_remove_suspend(spa_t *spa) | |||||||
| 	mutex_exit(&svr->svr_lock); | 	mutex_exit(&svr->svr_lock); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * Return true if the "allocating" property has been set to "off" | ||||||
|  |  */ | ||||||
|  | static boolean_t | ||||||
|  | vdev_prop_allocating_off(vdev_t *vd) | ||||||
|  | { | ||||||
|  | 	uint64_t objid = vd->vdev_top_zap; | ||||||
|  | 	uint64_t allocating = 1; | ||||||
|  | 
 | ||||||
|  | 	/* no vdev property object => no props */ | ||||||
|  | 	if (objid != 0) { | ||||||
|  | 		spa_t *spa = vd->vdev_spa; | ||||||
|  | 		objset_t *mos = spa->spa_meta_objset; | ||||||
|  | 
 | ||||||
|  | 		mutex_enter(&spa->spa_props_lock); | ||||||
|  | 		(void) zap_lookup(mos, objid, "allocating", sizeof (uint64_t), | ||||||
|  | 		    1, &allocating); | ||||||
|  | 		mutex_exit(&spa->spa_props_lock); | ||||||
|  | 	} | ||||||
|  | 	return (allocating == 0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /* ARGSUSED */ | /* ARGSUSED */ | ||||||
| static int | static int | ||||||
| spa_vdev_remove_cancel_check(void *arg, dmu_tx_t *tx) | spa_vdev_remove_cancel_check(void *arg, dmu_tx_t *tx) | ||||||
| @ -1761,6 +1961,13 @@ spa_vdev_remove_cancel_sync(void *arg, dmu_tx_t *tx) | |||||||
| 	spa_finish_removal(spa, DSS_CANCELED, tx); | 	spa_finish_removal(spa, DSS_CANCELED, tx); | ||||||
| 
 | 
 | ||||||
| 	vd->vdev_removing = B_FALSE; | 	vd->vdev_removing = B_FALSE; | ||||||
|  | 
 | ||||||
|  | 	if (!vdev_prop_allocating_off(vd)) { | ||||||
|  | 		spa_config_enter(spa, SCL_ALLOC | SCL_VDEV, FTAG, RW_WRITER); | ||||||
|  | 		vdev_activate(vd); | ||||||
|  | 		spa_config_exit(spa, SCL_ALLOC | SCL_VDEV, FTAG); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	vdev_config_dirty(vd); | 	vdev_config_dirty(vd); | ||||||
| 
 | 
 | ||||||
| 	zfs_dbgmsg("canceled device removal for vdev %llu in %llu", | 	zfs_dbgmsg("canceled device removal for vdev %llu in %llu", | ||||||
| @ -1774,21 +1981,9 @@ spa_vdev_remove_cancel_sync(void *arg, dmu_tx_t *tx) | |||||||
| static int | static int | ||||||
| spa_vdev_remove_cancel_impl(spa_t *spa) | spa_vdev_remove_cancel_impl(spa_t *spa) | ||||||
| { | { | ||||||
| 	uint64_t vdid = spa->spa_vdev_removal->svr_vdev_id; |  | ||||||
| 
 |  | ||||||
| 	int error = dsl_sync_task(spa->spa_name, spa_vdev_remove_cancel_check, | 	int error = dsl_sync_task(spa->spa_name, spa_vdev_remove_cancel_check, | ||||||
| 	    spa_vdev_remove_cancel_sync, NULL, 0, | 	    spa_vdev_remove_cancel_sync, NULL, 0, | ||||||
| 	    ZFS_SPACE_CHECK_EXTRA_RESERVED); | 	    ZFS_SPACE_CHECK_EXTRA_RESERVED); | ||||||
| 
 |  | ||||||
| 	if (error == 0) { |  | ||||||
| 		spa_config_enter(spa, SCL_ALLOC | SCL_VDEV, FTAG, RW_WRITER); |  | ||||||
| 		vdev_t *vd = vdev_lookup_top(spa, vdid); |  | ||||||
| 		metaslab_group_activate(vd->vdev_mg); |  | ||||||
| 		ASSERT(!vd->vdev_islog); |  | ||||||
| 		metaslab_group_activate(vd->vdev_log_mg); |  | ||||||
| 		spa_config_exit(spa, SCL_ALLOC | SCL_VDEV, FTAG); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return (error); | 	return (error); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -1984,6 +2179,11 @@ spa_vdev_remove_top_check(vdev_t *vd) | |||||||
| 	if (!spa_feature_is_enabled(spa, SPA_FEATURE_DEVICE_REMOVAL)) | 	if (!spa_feature_is_enabled(spa, SPA_FEATURE_DEVICE_REMOVAL)) | ||||||
| 		return (SET_ERROR(ENOTSUP)); | 		return (SET_ERROR(ENOTSUP)); | ||||||
| 
 | 
 | ||||||
|  | 	/*
 | ||||||
|  | 	 * This device is already being removed | ||||||
|  | 	 */ | ||||||
|  | 	if (vd->vdev_removing) | ||||||
|  | 		return (SET_ERROR(EALREADY)); | ||||||
| 
 | 
 | ||||||
| 	metaslab_class_t *mc = vd->vdev_mg->mg_class; | 	metaslab_class_t *mc = vd->vdev_mg->mg_class; | ||||||
| 	metaslab_class_t *normal = spa_normal_class(spa); | 	metaslab_class_t *normal = spa_normal_class(spa); | ||||||
| @ -2002,21 +2202,13 @@ spa_vdev_remove_top_check(vdev_t *vd) | |||||||
| 		ASSERT3U(available, >=, vd->vdev_stat.vs_alloc); | 		ASSERT3U(available, >=, vd->vdev_stat.vs_alloc); | ||||||
| 		if (available < vd->vdev_stat.vs_alloc) | 		if (available < vd->vdev_stat.vs_alloc) | ||||||
| 			return (SET_ERROR(ENOSPC)); | 			return (SET_ERROR(ENOSPC)); | ||||||
| 	} else { | 	} else if (!vd->vdev_noalloc) { | ||||||
| 		/* available space in the pool's normal class */ | 		/* available space in the pool's normal class */ | ||||||
| 		uint64_t available = dsl_dir_space_available( | 		uint64_t available = dsl_dir_space_available( | ||||||
| 		    spa->spa_dsl_pool->dp_root_dir, NULL, 0, B_TRUE); | 		    spa->spa_dsl_pool->dp_root_dir, NULL, 0, B_TRUE); | ||||||
| 		if (available < | 		if (available < vd->vdev_stat.vs_dspace) | ||||||
| 		    vd->vdev_stat.vs_dspace + spa_get_slop_space(spa)) { |  | ||||||
| 			/*
 |  | ||||||
| 			 * This is a normal device. There has to be enough free |  | ||||||
| 			 * space to remove the device and leave double the |  | ||||||
| 			 * "slop" space (i.e. we must leave at least 3% of the |  | ||||||
| 			 * pool free, in addition to the normal slop space). |  | ||||||
| 			 */ |  | ||||||
| 			return (SET_ERROR(ENOSPC)); | 			return (SET_ERROR(ENOSPC)); | ||||||
| 	} | 	} | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * There can not be a removal in progress. | 	 * There can not be a removal in progress. | ||||||
| @ -2108,6 +2300,7 @@ static int | |||||||
| spa_vdev_remove_top(vdev_t *vd, uint64_t *txg) | spa_vdev_remove_top(vdev_t *vd, uint64_t *txg) | ||||||
| { | { | ||||||
| 	spa_t *spa = vd->vdev_spa; | 	spa_t *spa = vd->vdev_spa; | ||||||
|  | 	boolean_t set_noalloc = B_FALSE; | ||||||
| 	int error; | 	int error; | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| @ -2116,8 +2309,6 @@ spa_vdev_remove_top(vdev_t *vd, uint64_t *txg) | |||||||
| 	 * are errors. | 	 * are errors. | ||||||
| 	 */ | 	 */ | ||||||
| 	error = spa_vdev_remove_top_check(vd); | 	error = spa_vdev_remove_top_check(vd); | ||||||
| 	if (error != 0) |  | ||||||
| 		return (error); |  | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * Stop allocating from this vdev.  Note that we must check | 	 * Stop allocating from this vdev.  Note that we must check | ||||||
| @ -2127,31 +2318,22 @@ spa_vdev_remove_top(vdev_t *vd, uint64_t *txg) | |||||||
| 	 * The above check for sufficient free space serves this | 	 * The above check for sufficient free space serves this | ||||||
| 	 * purpose. | 	 * purpose. | ||||||
| 	 */ | 	 */ | ||||||
| 	metaslab_group_t *mg = vd->vdev_mg; | 	if (error == 0 && !vd->vdev_noalloc) { | ||||||
| 	metaslab_group_passivate(mg); | 		set_noalloc = B_TRUE; | ||||||
| 	ASSERT(!vd->vdev_islog); | 		error = vdev_passivate(vd, txg); | ||||||
| 	metaslab_group_passivate(vd->vdev_log_mg); | 	} | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	if (error != 0) | ||||||
| 	 * Wait for the youngest allocations and frees to sync, | 		return (error); | ||||||
| 	 * and then wait for the deferral of those frees to finish. |  | ||||||
| 	 */ |  | ||||||
| 	spa_vdev_config_exit(spa, NULL, |  | ||||||
| 	    *txg + TXG_CONCURRENT_STATES + TXG_DEFER_SIZE, 0, FTAG); |  | ||||||
| 
 |  | ||||||
| 	/*
 |  | ||||||
| 	 * We must ensure that no "stubby" log blocks are allocated |  | ||||||
| 	 * on the device to be removed.  These blocks could be |  | ||||||
| 	 * written at any time, including while we are in the middle |  | ||||||
| 	 * of copying them. |  | ||||||
| 	 */ |  | ||||||
| 	error = spa_reset_logs(spa); |  | ||||||
| 
 | 
 | ||||||
| 	/*
 | 	/*
 | ||||||
| 	 * We stop any initializing and TRIM that is currently in progress | 	 * We stop any initializing and TRIM that is currently in progress | ||||||
| 	 * but leave the state as "active". This will allow the process to | 	 * but leave the state as "active". This will allow the process to | ||||||
| 	 * resume if the removal is canceled sometime later. | 	 * resume if the removal is canceled sometime later. | ||||||
| 	 */ | 	 */ | ||||||
|  | 
 | ||||||
|  | 	spa_vdev_config_exit(spa, NULL, *txg, 0, FTAG); | ||||||
|  | 
 | ||||||
| 	vdev_initialize_stop_all(vd, VDEV_INITIALIZE_ACTIVE); | 	vdev_initialize_stop_all(vd, VDEV_INITIALIZE_ACTIVE); | ||||||
| 	vdev_trim_stop_all(vd, VDEV_TRIM_ACTIVE); | 	vdev_trim_stop_all(vd, VDEV_TRIM_ACTIVE); | ||||||
| 	vdev_autotrim_stop_wait(vd); | 	vdev_autotrim_stop_wait(vd); | ||||||
| @ -2162,13 +2344,11 @@ spa_vdev_remove_top(vdev_t *vd, uint64_t *txg) | |||||||
| 	 * Things might have changed while the config lock was dropped | 	 * Things might have changed while the config lock was dropped | ||||||
| 	 * (e.g. space usage).  Check for errors again. | 	 * (e.g. space usage).  Check for errors again. | ||||||
| 	 */ | 	 */ | ||||||
| 	if (error == 0) |  | ||||||
| 	error = spa_vdev_remove_top_check(vd); | 	error = spa_vdev_remove_top_check(vd); | ||||||
| 
 | 
 | ||||||
| 	if (error != 0) { | 	if (error != 0) { | ||||||
| 		metaslab_group_activate(mg); | 		if (set_noalloc) | ||||||
| 		ASSERT(!vd->vdev_islog); | 			vdev_activate(vd); | ||||||
| 		metaslab_group_activate(vd->vdev_log_mg); |  | ||||||
| 		spa_async_request(spa, SPA_ASYNC_INITIALIZE_RESTART); | 		spa_async_request(spa, SPA_ASYNC_INITIALIZE_RESTART); | ||||||
| 		spa_async_request(spa, SPA_ASYNC_TRIM_RESTART); | 		spa_async_request(spa, SPA_ASYNC_TRIM_RESTART); | ||||||
| 		spa_async_request(spa, SPA_ASYNC_AUTOTRIM_RESTART); | 		spa_async_request(spa, SPA_ASYNC_AUTOTRIM_RESTART); | ||||||
|  | |||||||
| @ -38,7 +38,7 @@ | |||||||
|  * Copyright (c) 2017 Open-E, Inc. All Rights Reserved. |  * Copyright (c) 2017 Open-E, Inc. All Rights Reserved. | ||||||
|  * Copyright (c) 2019 Datto Inc. |  * Copyright (c) 2019 Datto Inc. | ||||||
|  * Copyright (c) 2019, 2020 by Christian Schwarz. All rights reserved. |  * Copyright (c) 2019, 2020 by Christian Schwarz. All rights reserved. | ||||||
|  * Copyright (c) 2019, Klara Inc. |  * Copyright (c) 2019, 2021, Klara Inc. | ||||||
|  * Copyright (c) 2019, Allan Jude |  * Copyright (c) 2019, Allan Jude | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| @ -2981,6 +2981,96 @@ zfs_ioc_pool_get_props(zfs_cmd_t *zc) | |||||||
| 	return (error); | 	return (error); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | /*
 | ||||||
|  |  * innvl: { | ||||||
|  |  *     "vdevprops_set_vdev" -> guid | ||||||
|  |  *     "vdevprops_set_props" -> { prop -> value } | ||||||
|  |  * } | ||||||
|  |  * | ||||||
|  |  * outnvl: propname -> error code (int32) | ||||||
|  |  */ | ||||||
|  | static const zfs_ioc_key_t zfs_keys_vdev_set_props[] = { | ||||||
|  | 	{ZPOOL_VDEV_PROPS_SET_VDEV,	DATA_TYPE_UINT64,	0}, | ||||||
|  | 	{ZPOOL_VDEV_PROPS_SET_PROPS,	DATA_TYPE_NVLIST,	0} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | zfs_ioc_vdev_set_props(const char *poolname, nvlist_t *innvl, nvlist_t *outnvl) | ||||||
|  | { | ||||||
|  | 	spa_t *spa; | ||||||
|  | 	int error; | ||||||
|  | 	vdev_t *vd; | ||||||
|  | 	uint64_t vdev_guid; | ||||||
|  | 
 | ||||||
|  | 	/* Early validation */ | ||||||
|  | 	if (nvlist_lookup_uint64(innvl, ZPOOL_VDEV_PROPS_SET_VDEV, | ||||||
|  | 	    &vdev_guid) != 0) | ||||||
|  | 		return (SET_ERROR(EINVAL)); | ||||||
|  | 
 | ||||||
|  | 	if (outnvl == NULL) | ||||||
|  | 		return (SET_ERROR(EINVAL)); | ||||||
|  | 
 | ||||||
|  | 	if ((error = spa_open(poolname, &spa, FTAG)) != 0) | ||||||
|  | 		return (error); | ||||||
|  | 
 | ||||||
|  | 	ASSERT(spa_writeable(spa)); | ||||||
|  | 
 | ||||||
|  | 	if ((vd = spa_lookup_by_guid(spa, vdev_guid, B_TRUE)) == NULL) { | ||||||
|  | 		spa_close(spa, FTAG); | ||||||
|  | 		return (SET_ERROR(ENOENT)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	error = vdev_prop_set(vd, innvl, outnvl); | ||||||
|  | 
 | ||||||
|  | 	spa_close(spa, FTAG); | ||||||
|  | 
 | ||||||
|  | 	return (error); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /*
 | ||||||
|  |  * innvl: { | ||||||
|  |  *     "vdevprops_get_vdev" -> guid | ||||||
|  |  *     (optional) "vdevprops_get_props" -> { propname -> propid } | ||||||
|  |  * } | ||||||
|  |  * | ||||||
|  |  * outnvl: propname -> value | ||||||
|  |  */ | ||||||
|  | static const zfs_ioc_key_t zfs_keys_vdev_get_props[] = { | ||||||
|  | 	{ZPOOL_VDEV_PROPS_GET_VDEV,	DATA_TYPE_UINT64,	0}, | ||||||
|  | 	{ZPOOL_VDEV_PROPS_GET_PROPS,	DATA_TYPE_NVLIST,	ZK_OPTIONAL} | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | zfs_ioc_vdev_get_props(const char *poolname, nvlist_t *innvl, nvlist_t *outnvl) | ||||||
|  | { | ||||||
|  | 	spa_t *spa; | ||||||
|  | 	int error; | ||||||
|  | 	vdev_t *vd; | ||||||
|  | 	uint64_t vdev_guid; | ||||||
|  | 
 | ||||||
|  | 	/* Early validation */ | ||||||
|  | 	if (nvlist_lookup_uint64(innvl, ZPOOL_VDEV_PROPS_GET_VDEV, | ||||||
|  | 	    &vdev_guid) != 0) | ||||||
|  | 		return (SET_ERROR(EINVAL)); | ||||||
|  | 
 | ||||||
|  | 	if (outnvl == NULL) | ||||||
|  | 		return (SET_ERROR(EINVAL)); | ||||||
|  | 
 | ||||||
|  | 	if ((error = spa_open(poolname, &spa, FTAG)) != 0) | ||||||
|  | 		return (error); | ||||||
|  | 
 | ||||||
|  | 	if ((vd = spa_lookup_by_guid(spa, vdev_guid, B_TRUE)) == NULL) { | ||||||
|  | 		spa_close(spa, FTAG); | ||||||
|  | 		return (SET_ERROR(ENOENT)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	error = vdev_prop_get(vd, innvl, outnvl); | ||||||
|  | 
 | ||||||
|  | 	spa_close(spa, FTAG); | ||||||
|  | 
 | ||||||
|  | 	return (error); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /*
 | /*
 | ||||||
|  * inputs: |  * inputs: | ||||||
|  * zc_name		name of filesystem |  * zc_name		name of filesystem | ||||||
| @ -7107,6 +7197,16 @@ zfs_ioctl_init(void) | |||||||
| 	    POOL_CHECK_SUSPENDED, B_FALSE, B_TRUE, | 	    POOL_CHECK_SUSPENDED, B_FALSE, B_TRUE, | ||||||
| 	    zfs_keys_get_bootenv, ARRAY_SIZE(zfs_keys_get_bootenv)); | 	    zfs_keys_get_bootenv, ARRAY_SIZE(zfs_keys_get_bootenv)); | ||||||
| 
 | 
 | ||||||
|  | 	zfs_ioctl_register("zpool_vdev_get_props", ZFS_IOC_VDEV_GET_PROPS, | ||||||
|  | 	    zfs_ioc_vdev_get_props, zfs_secpolicy_read, POOL_NAME, | ||||||
|  | 	    POOL_CHECK_NONE, B_FALSE, B_FALSE, zfs_keys_vdev_get_props, | ||||||
|  | 	    ARRAY_SIZE(zfs_keys_vdev_get_props)); | ||||||
|  | 
 | ||||||
|  | 	zfs_ioctl_register("zpool_vdev_set_props", ZFS_IOC_VDEV_SET_PROPS, | ||||||
|  | 	    zfs_ioc_vdev_set_props, zfs_secpolicy_config, POOL_NAME, | ||||||
|  | 	    POOL_CHECK_SUSPENDED | POOL_CHECK_READONLY, B_FALSE, B_FALSE, | ||||||
|  | 	    zfs_keys_vdev_set_props, ARRAY_SIZE(zfs_keys_vdev_set_props)); | ||||||
|  | 
 | ||||||
| 	/* IOCTLS that use the legacy function signature */ | 	/* IOCTLS that use the legacy function signature */ | ||||||
| 
 | 
 | ||||||
| 	zfs_ioctl_register_legacy(ZFS_IOC_POOL_FREEZE, zfs_ioc_pool_freeze, | 	zfs_ioctl_register_legacy(ZFS_IOC_POOL_FREEZE, zfs_ioc_pool_freeze, | ||||||
|  | |||||||
| @ -3755,7 +3755,7 @@ zio_vdev_io_start(zio_t *zio) | |||||||
| 		 * Note: the code can handle other kinds of writes, | 		 * Note: the code can handle other kinds of writes, | ||||||
| 		 * but we don't expect them. | 		 * but we don't expect them. | ||||||
| 		 */ | 		 */ | ||||||
| 		if (zio->io_vd->vdev_removing) { | 		if (zio->io_vd->vdev_noalloc) { | ||||||
| 			ASSERT(zio->io_flags & | 			ASSERT(zio->io_flags & | ||||||
| 			    (ZIO_FLAG_PHYSICAL | ZIO_FLAG_SELF_HEAL | | 			    (ZIO_FLAG_PHYSICAL | ZIO_FLAG_SELF_HEAL | | ||||||
| 			    ZIO_FLAG_RESILVER | ZIO_FLAG_INDUCE_DAMAGE)); | 			    ZIO_FLAG_RESILVER | ZIO_FLAG_INDUCE_DAMAGE)); | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Allan Jude
						Allan Jude