xref: /titanic_54/usr/src/cmd/zonecfg/zonecfg.c (revision 6e1ae2a33c618c4c2b14aec7d2f21743ddea5837)
17c478bd9Sstevel@tonic-gate /*
27c478bd9Sstevel@tonic-gate  * CDDL HEADER START
37c478bd9Sstevel@tonic-gate  *
47c478bd9Sstevel@tonic-gate  * The contents of this file are subject to the terms of the
5fb03efaaSdp  * Common Development and Distribution License (the "License").
6fb03efaaSdp  * You may not use this file except in compliance with the License.
77c478bd9Sstevel@tonic-gate  *
87c478bd9Sstevel@tonic-gate  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
97c478bd9Sstevel@tonic-gate  * or http://www.opensolaris.org/os/licensing.
107c478bd9Sstevel@tonic-gate  * See the License for the specific language governing permissions
117c478bd9Sstevel@tonic-gate  * and limitations under the License.
127c478bd9Sstevel@tonic-gate  *
137c478bd9Sstevel@tonic-gate  * When distributing Covered Code, include this CDDL HEADER in each
147c478bd9Sstevel@tonic-gate  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
157c478bd9Sstevel@tonic-gate  * If applicable, add the following below this CDDL HEADER, with the
167c478bd9Sstevel@tonic-gate  * fields enclosed by brackets "[]" replaced with your own identifying
177c478bd9Sstevel@tonic-gate  * information: Portions Copyright [yyyy] [name of copyright owner]
187c478bd9Sstevel@tonic-gate  *
197c478bd9Sstevel@tonic-gate  * CDDL HEADER END
207c478bd9Sstevel@tonic-gate  */
217e362f58Scomay 
227c478bd9Sstevel@tonic-gate /*
236d4d1c0dSbatschul  * Copyright (c) 2003, 2010, Oracle and/or its affiliates. All rights reserved.
247c478bd9Sstevel@tonic-gate  */
257c478bd9Sstevel@tonic-gate 
267c478bd9Sstevel@tonic-gate /*
277c478bd9Sstevel@tonic-gate  * zonecfg is a lex/yacc based command interpreter used to manage zone
287c478bd9Sstevel@tonic-gate  * configurations.  The lexer (see zonecfg_lex.l) builds up tokens, which
297c478bd9Sstevel@tonic-gate  * the grammar (see zonecfg_grammar.y) builds up into commands, some of
307c478bd9Sstevel@tonic-gate  * which takes resources and/or properties as arguments.  See the block
317c478bd9Sstevel@tonic-gate  * comments near the end of zonecfg_grammar.y for how the data structures
327c478bd9Sstevel@tonic-gate  * which keep track of these resources and properties are built up.
337c478bd9Sstevel@tonic-gate  *
347c478bd9Sstevel@tonic-gate  * The resource/property data structures are inserted into a command
357c478bd9Sstevel@tonic-gate  * structure (see zonecfg.h), which also keeps track of command names,
367c478bd9Sstevel@tonic-gate  * miscellaneous arguments, and function handlers.  The grammar selects
377c478bd9Sstevel@tonic-gate  * the appropriate function handler, each of which takes a pointer to a
387c478bd9Sstevel@tonic-gate  * command structure as its sole argument, and invokes it.  The grammar
397c478bd9Sstevel@tonic-gate  * itself is "entered" (a la the Matrix) by yyparse(), which is called
407c478bd9Sstevel@tonic-gate  * from read_input(), our main driving function.  That in turn is called
417c478bd9Sstevel@tonic-gate  * by one of do_interactive(), cmd_file() or one_command_at_a_time(), each
427c478bd9Sstevel@tonic-gate  * of which is called from main() depending on how the program was invoked.
437c478bd9Sstevel@tonic-gate  *
447c478bd9Sstevel@tonic-gate  * The rest of this module consists of the various function handlers and
457c478bd9Sstevel@tonic-gate  * their helper functions.  Some of these functions, particularly the
467c478bd9Sstevel@tonic-gate  * X_to_str() functions, which maps command, resource and property numbers
477c478bd9Sstevel@tonic-gate  * to strings, are used quite liberally, as doing so results in a better
487c478bd9Sstevel@tonic-gate  * program w/rt I18N, reducing the need for translation notes.
497c478bd9Sstevel@tonic-gate  */
507c478bd9Sstevel@tonic-gate 
517c478bd9Sstevel@tonic-gate #include <sys/mntent.h>
527c478bd9Sstevel@tonic-gate #include <sys/varargs.h>
537c478bd9Sstevel@tonic-gate #include <sys/sysmacros.h>
547c478bd9Sstevel@tonic-gate 
557c478bd9Sstevel@tonic-gate #include <errno.h>
569acbbeafSnn35248 #include <fcntl.h>
577c478bd9Sstevel@tonic-gate #include <strings.h>
587c478bd9Sstevel@tonic-gate #include <unistd.h>
597c478bd9Sstevel@tonic-gate #include <ctype.h>
607c478bd9Sstevel@tonic-gate #include <stdlib.h>
617c478bd9Sstevel@tonic-gate #include <assert.h>
627c478bd9Sstevel@tonic-gate #include <sys/stat.h>
637c478bd9Sstevel@tonic-gate #include <zone.h>
647c478bd9Sstevel@tonic-gate #include <arpa/inet.h>
657c478bd9Sstevel@tonic-gate #include <netdb.h>
667c478bd9Sstevel@tonic-gate #include <locale.h>
677c478bd9Sstevel@tonic-gate #include <libintl.h>
687c478bd9Sstevel@tonic-gate #include <alloca.h>
697c478bd9Sstevel@tonic-gate #include <signal.h>
709acbbeafSnn35248 #include <wait.h>
717c478bd9Sstevel@tonic-gate #include <libtecla.h>
72fa9e4066Sahrens #include <libzfs.h>
739acbbeafSnn35248 #include <sys/brand.h>
749acbbeafSnn35248 #include <libbrand.h>
755679c89fSjv227347 #include <sys/systeminfo.h>
76c9f134eaSjv227347 #include <libdladm.h>
77c9f134eaSjv227347 #include <libinetutil.h>
78cb8a054bSGlenn Faden #include <pwd.h>
797c478bd9Sstevel@tonic-gate 
807c478bd9Sstevel@tonic-gate #include <libzonecfg.h>
817c478bd9Sstevel@tonic-gate #include "zonecfg.h"
827c478bd9Sstevel@tonic-gate 
837c478bd9Sstevel@tonic-gate #if !defined(TEXT_DOMAIN)		/* should be defined by cc -D */
847c478bd9Sstevel@tonic-gate #define	TEXT_DOMAIN	"SYS_TEST"	/* Use this only if it wasn't */
857c478bd9Sstevel@tonic-gate #endif
867c478bd9Sstevel@tonic-gate 
877c478bd9Sstevel@tonic-gate #define	PAGER	"/usr/bin/more"
889acbbeafSnn35248 #define	EXEC_PREFIX	"exec "
899acbbeafSnn35248 #define	EXEC_LEN	(strlen(EXEC_PREFIX))
907c478bd9Sstevel@tonic-gate 
917c478bd9Sstevel@tonic-gate struct help {
927c478bd9Sstevel@tonic-gate 	uint_t	cmd_num;
937c478bd9Sstevel@tonic-gate 	char	*cmd_name;
947c478bd9Sstevel@tonic-gate 	uint_t	flags;
957c478bd9Sstevel@tonic-gate 	char	*short_usage;
967c478bd9Sstevel@tonic-gate };
977c478bd9Sstevel@tonic-gate 
987c478bd9Sstevel@tonic-gate extern int yyparse(void);
997c478bd9Sstevel@tonic-gate extern int lex_lineno;
1007c478bd9Sstevel@tonic-gate 
1017c478bd9Sstevel@tonic-gate #define	MAX_LINE_LEN	1024
1027c478bd9Sstevel@tonic-gate #define	MAX_CMD_HIST	1024
1039acbbeafSnn35248 #define	MAX_CMD_LEN	1024
1047c478bd9Sstevel@tonic-gate 
1050209230bSgjelinek #define	ONE_MB		1048576
1060209230bSgjelinek 
1077c478bd9Sstevel@tonic-gate /*
1087c478bd9Sstevel@tonic-gate  * Each SHELP_ should be a simple string.
1097c478bd9Sstevel@tonic-gate  */
1107c478bd9Sstevel@tonic-gate 
1117c478bd9Sstevel@tonic-gate #define	SHELP_ADD	"add <resource-type>\n\t(global scope)\n" \
1127c478bd9Sstevel@tonic-gate 	"add <property-name> <property-value>\n\t(resource scope)"
1137c478bd9Sstevel@tonic-gate #define	SHELP_CANCEL	"cancel"
1140209230bSgjelinek #define	SHELP_CLEAR	"clear <property-name>"
1157c478bd9Sstevel@tonic-gate #define	SHELP_COMMIT	"commit"
116ee519a1fSgjelinek #define	SHELP_CREATE	"create [-F] [ -a <path> | -b | -t <template> ]"
1177c478bd9Sstevel@tonic-gate #define	SHELP_DELETE	"delete [-F]"
1187c478bd9Sstevel@tonic-gate #define	SHELP_END	"end"
1197c478bd9Sstevel@tonic-gate #define	SHELP_EXIT	"exit [-F]"
1207c478bd9Sstevel@tonic-gate #define	SHELP_EXPORT	"export [-f output-file]"
1217c478bd9Sstevel@tonic-gate #define	SHELP_HELP	"help [commands] [syntax] [usage] [<command-name>]"
1227c478bd9Sstevel@tonic-gate #define	SHELP_INFO	"info [<resource-type> [property-name=property-value]*]"
1230209230bSgjelinek #define	SHELP_REMOVE	"remove [-F] <resource-type> " \
1240209230bSgjelinek 	"[ <property-name>=<property-value> ]*\n" \
1250209230bSgjelinek 	"\t(global scope)\n" \
1260209230bSgjelinek 	"remove <property-name> <property-value>\n" \
1270209230bSgjelinek 	"\t(resource scope)"
1287c478bd9Sstevel@tonic-gate #define	SHELP_REVERT	"revert [-F]"
1297c478bd9Sstevel@tonic-gate #define	SHELP_SELECT	"select <resource-type> { <property-name>=" \
1307c478bd9Sstevel@tonic-gate 	"<property-value> }"
1317c478bd9Sstevel@tonic-gate #define	SHELP_SET	"set <property-name>=<property-value>"
1327c478bd9Sstevel@tonic-gate #define	SHELP_VERIFY	"verify"
1337c478bd9Sstevel@tonic-gate 
1347c478bd9Sstevel@tonic-gate static struct help helptab[] = {
1357c478bd9Sstevel@tonic-gate 	{ CMD_ADD,	"add",		HELP_RES_PROPS,	SHELP_ADD, },
1367c478bd9Sstevel@tonic-gate 	{ CMD_CANCEL,	"cancel",	0,		SHELP_CANCEL, },
1370209230bSgjelinek 	{ CMD_CLEAR,	"clear",	HELP_PROPS,	SHELP_CLEAR, },
1387c478bd9Sstevel@tonic-gate 	{ CMD_COMMIT,	"commit",	0,		SHELP_COMMIT, },
1397c478bd9Sstevel@tonic-gate 	{ CMD_CREATE,	"create",	0,		SHELP_CREATE, },
1407c478bd9Sstevel@tonic-gate 	{ CMD_DELETE,	"delete",	0,		SHELP_DELETE, },
1417c478bd9Sstevel@tonic-gate 	{ CMD_END,	"end",		0,		SHELP_END, },
1427c478bd9Sstevel@tonic-gate 	{ CMD_EXIT,	"exit",		0,		SHELP_EXIT, },
1437c478bd9Sstevel@tonic-gate 	{ CMD_EXPORT,	"export",	0,		SHELP_EXPORT, },
1447c478bd9Sstevel@tonic-gate 	{ CMD_HELP,	"help",		0,		SHELP_HELP },
1457c478bd9Sstevel@tonic-gate 	{ CMD_INFO,	"info",		HELP_RES_PROPS,	SHELP_INFO, },
1467c478bd9Sstevel@tonic-gate 	{ CMD_REMOVE,	"remove",	HELP_RES_PROPS,	SHELP_REMOVE, },
1477c478bd9Sstevel@tonic-gate 	{ CMD_REVERT,	"revert",	0,		SHELP_REVERT, },
1487c478bd9Sstevel@tonic-gate 	{ CMD_SELECT,	"select",	HELP_RES_PROPS,	SHELP_SELECT, },
1497c478bd9Sstevel@tonic-gate 	{ CMD_SET,	"set",		HELP_PROPS,	SHELP_SET, },
1507c478bd9Sstevel@tonic-gate 	{ CMD_VERIFY,	"verify",	0,		SHELP_VERIFY, },
1517c478bd9Sstevel@tonic-gate 	{ 0 },
1527c478bd9Sstevel@tonic-gate };
1537c478bd9Sstevel@tonic-gate 
1547c478bd9Sstevel@tonic-gate #define	MAX_RT_STRLEN	16
1557c478bd9Sstevel@tonic-gate 
1567c478bd9Sstevel@tonic-gate /* These *must* match the order of the RT_ define's from zonecfg.h */
157c94c1ef0Sjv227347 char *res_types[] = {
1587c478bd9Sstevel@tonic-gate 	"unknown",
159087719fdSdp 	"zonename",
1607c478bd9Sstevel@tonic-gate 	"zonepath",
1617c478bd9Sstevel@tonic-gate 	"autoboot",
1627c478bd9Sstevel@tonic-gate 	"pool",
1637c478bd9Sstevel@tonic-gate 	"fs",
1647c478bd9Sstevel@tonic-gate 	"net",
1657c478bd9Sstevel@tonic-gate 	"device",
1667c478bd9Sstevel@tonic-gate 	"rctl",
1677c478bd9Sstevel@tonic-gate 	"attr",
168fa9e4066Sahrens 	"dataset",
169ffbafc53Scomay 	"limitpriv",
1703f2f09c1Sdp 	"bootargs",
1719acbbeafSnn35248 	"brand",
1720209230bSgjelinek 	"dedicated-cpu",
1730209230bSgjelinek 	"capped-memory",
1740209230bSgjelinek 	ALIAS_MAXLWPS,
1750209230bSgjelinek 	ALIAS_MAXSHMMEM,
1760209230bSgjelinek 	ALIAS_MAXSHMIDS,
1770209230bSgjelinek 	ALIAS_MAXMSGIDS,
1780209230bSgjelinek 	ALIAS_MAXSEMIDS,
1790209230bSgjelinek 	ALIAS_SHARES,
1800209230bSgjelinek 	"scheduling-class",
181f4b3ec61Sdh155122 	"ip-type",
182c97ad5cdSakolb 	"capped-cpu",
1835679c89fSjv227347 	"hostid",
184cb8a054bSGlenn Faden 	"admin",
1850fbb751dSJohn Levon 	"fs-allowed",
186ff19e029SMenno Lageman 	ALIAS_MAXPROCS,
1877c478bd9Sstevel@tonic-gate 	NULL
1887c478bd9Sstevel@tonic-gate };
1897c478bd9Sstevel@tonic-gate 
1907c478bd9Sstevel@tonic-gate /* These *must* match the order of the PT_ define's from zonecfg.h */
191c94c1ef0Sjv227347 char *prop_types[] = {
1927c478bd9Sstevel@tonic-gate 	"unknown",
193087719fdSdp 	"zonename",
1947c478bd9Sstevel@tonic-gate 	"zonepath",
1957c478bd9Sstevel@tonic-gate 	"autoboot",
1967c478bd9Sstevel@tonic-gate 	"pool",
1977c478bd9Sstevel@tonic-gate 	"dir",
1987c478bd9Sstevel@tonic-gate 	"special",
1997c478bd9Sstevel@tonic-gate 	"type",
2007c478bd9Sstevel@tonic-gate 	"options",
2017c478bd9Sstevel@tonic-gate 	"address",
2027c478bd9Sstevel@tonic-gate 	"physical",
2037c478bd9Sstevel@tonic-gate 	"name",
2047c478bd9Sstevel@tonic-gate 	"value",
2057c478bd9Sstevel@tonic-gate 	"match",
2067c478bd9Sstevel@tonic-gate 	"priv",
2077c478bd9Sstevel@tonic-gate 	"limit",
2087c478bd9Sstevel@tonic-gate 	"action",
2097c478bd9Sstevel@tonic-gate 	"raw",
210ffbafc53Scomay 	"limitpriv",
2113f2f09c1Sdp 	"bootargs",
2129acbbeafSnn35248 	"brand",
2130209230bSgjelinek 	"ncpus",
2140209230bSgjelinek 	"importance",
2150209230bSgjelinek 	"swap",
2160209230bSgjelinek 	"locked",
2170209230bSgjelinek 	ALIAS_SHARES,
2180209230bSgjelinek 	ALIAS_MAXLWPS,
2190209230bSgjelinek 	ALIAS_MAXSHMMEM,
2200209230bSgjelinek 	ALIAS_MAXSHMIDS,
2210209230bSgjelinek 	ALIAS_MAXMSGIDS,
2220209230bSgjelinek 	ALIAS_MAXSEMIDS,
2230209230bSgjelinek 	ALIAS_MAXLOCKEDMEM,
2240209230bSgjelinek 	ALIAS_MAXSWAP,
2250209230bSgjelinek 	"scheduling-class",
226f4b3ec61Sdh155122 	"ip-type",
227de860bd9Sgfaden 	"defrouter",
2285679c89fSjv227347 	"hostid",
229cb8a054bSGlenn Faden 	"user",
230cb8a054bSGlenn Faden 	"auths",
2310fbb751dSJohn Levon 	"fs-allowed",
232ff19e029SMenno Lageman 	ALIAS_MAXPROCS,
2337c478bd9Sstevel@tonic-gate 	NULL
2347c478bd9Sstevel@tonic-gate };
2357c478bd9Sstevel@tonic-gate 
236ffbafc53Scomay /* These *must* match the order of the PROP_VAL_ define's from zonecfg.h */
2377c478bd9Sstevel@tonic-gate static char *prop_val_types[] = {
2387c478bd9Sstevel@tonic-gate 	"simple",
2397c478bd9Sstevel@tonic-gate 	"complex",
2407c478bd9Sstevel@tonic-gate 	"list",
2417c478bd9Sstevel@tonic-gate };
2427c478bd9Sstevel@tonic-gate 
2437c478bd9Sstevel@tonic-gate /*
2447c478bd9Sstevel@tonic-gate  * The various _cmds[] lists below are for command tab-completion.
2457c478bd9Sstevel@tonic-gate  */
2467c478bd9Sstevel@tonic-gate 
2477c478bd9Sstevel@tonic-gate /*
2487c478bd9Sstevel@tonic-gate  * remove has a space afterwards because it has qualifiers; the other commands
2490209230bSgjelinek  * that have qualifiers (add, select, etc.) don't need a space here because
2507c478bd9Sstevel@tonic-gate  * they have their own _cmds[] lists below.
2517c478bd9Sstevel@tonic-gate  */
2527c478bd9Sstevel@tonic-gate static const char *global_scope_cmds[] = {
2537c478bd9Sstevel@tonic-gate 	"add",
2540209230bSgjelinek 	"clear",
2557c478bd9Sstevel@tonic-gate 	"commit",
2567c478bd9Sstevel@tonic-gate 	"create",
2577c478bd9Sstevel@tonic-gate 	"delete",
2587c478bd9Sstevel@tonic-gate 	"exit",
2597c478bd9Sstevel@tonic-gate 	"export",
2607c478bd9Sstevel@tonic-gate 	"help",
2617c478bd9Sstevel@tonic-gate 	"info",
2627c478bd9Sstevel@tonic-gate 	"remove ",
2637c478bd9Sstevel@tonic-gate 	"revert",
2647c478bd9Sstevel@tonic-gate 	"select",
2657c478bd9Sstevel@tonic-gate 	"set",
2667c478bd9Sstevel@tonic-gate 	"verify",
2677c478bd9Sstevel@tonic-gate 	NULL
2687c478bd9Sstevel@tonic-gate };
2697c478bd9Sstevel@tonic-gate 
2707c478bd9Sstevel@tonic-gate static const char *add_cmds[] = {
2717c478bd9Sstevel@tonic-gate 	"add fs",
2727c478bd9Sstevel@tonic-gate 	"add net",
2737c478bd9Sstevel@tonic-gate 	"add device",
2747c478bd9Sstevel@tonic-gate 	"add rctl",
2757c478bd9Sstevel@tonic-gate 	"add attr",
276fa9e4066Sahrens 	"add dataset",
2770209230bSgjelinek 	"add dedicated-cpu",
278c97ad5cdSakolb 	"add capped-cpu",
2790209230bSgjelinek 	"add capped-memory",
280cb8a054bSGlenn Faden 	"add admin",
2810209230bSgjelinek 	NULL
2820209230bSgjelinek };
2830209230bSgjelinek 
2840209230bSgjelinek static const char *clear_cmds[] = {
2850209230bSgjelinek 	"clear autoboot",
2860209230bSgjelinek 	"clear pool",
2870209230bSgjelinek 	"clear limitpriv",
2880209230bSgjelinek 	"clear bootargs",
2890209230bSgjelinek 	"clear scheduling-class",
290f4b3ec61Sdh155122 	"clear ip-type",
2910209230bSgjelinek 	"clear " ALIAS_MAXLWPS,
2920209230bSgjelinek 	"clear " ALIAS_MAXSHMMEM,
2930209230bSgjelinek 	"clear " ALIAS_MAXSHMIDS,
2940209230bSgjelinek 	"clear " ALIAS_MAXMSGIDS,
2950209230bSgjelinek 	"clear " ALIAS_MAXSEMIDS,
2960209230bSgjelinek 	"clear " ALIAS_SHARES,
297ff19e029SMenno Lageman 	"clear " ALIAS_MAXPROCS,
2987c478bd9Sstevel@tonic-gate 	NULL
2997c478bd9Sstevel@tonic-gate };
3007c478bd9Sstevel@tonic-gate 
3019e7542f4Sdp static const char *remove_cmds[] = {
3029e7542f4Sdp 	"remove fs ",
3039e7542f4Sdp 	"remove net ",
3049e7542f4Sdp 	"remove device ",
3059e7542f4Sdp 	"remove rctl ",
3069e7542f4Sdp 	"remove attr ",
3079e7542f4Sdp 	"remove dataset ",
3080209230bSgjelinek 	"remove dedicated-cpu ",
309c97ad5cdSakolb 	"remove capped-cpu ",
3100209230bSgjelinek 	"remove capped-memory ",
311cb8a054bSGlenn Faden 	"remove admin ",
3129e7542f4Sdp 	NULL
3139e7542f4Sdp };
3149e7542f4Sdp 
3157c478bd9Sstevel@tonic-gate static const char *select_cmds[] = {
3167c478bd9Sstevel@tonic-gate 	"select fs ",
3177c478bd9Sstevel@tonic-gate 	"select net ",
3187c478bd9Sstevel@tonic-gate 	"select device ",
3197c478bd9Sstevel@tonic-gate 	"select rctl ",
3207c478bd9Sstevel@tonic-gate 	"select attr ",
321fa9e4066Sahrens 	"select dataset ",
3220209230bSgjelinek 	"select dedicated-cpu",
323c97ad5cdSakolb 	"select capped-cpu",
3240209230bSgjelinek 	"select capped-memory",
325cb8a054bSGlenn Faden 	"select admin",
3267c478bd9Sstevel@tonic-gate 	NULL
3277c478bd9Sstevel@tonic-gate };
3287c478bd9Sstevel@tonic-gate 
3297c478bd9Sstevel@tonic-gate static const char *set_cmds[] = {
330087719fdSdp 	"set zonename=",
331087719fdSdp 	"set zonepath=",
3329acbbeafSnn35248 	"set brand=",
333087719fdSdp 	"set autoboot=",
334087719fdSdp 	"set pool=",
335ffbafc53Scomay 	"set limitpriv=",
3363f2f09c1Sdp 	"set bootargs=",
3370209230bSgjelinek 	"set scheduling-class=",
338f4b3ec61Sdh155122 	"set ip-type=",
3390209230bSgjelinek 	"set " ALIAS_MAXLWPS "=",
3400209230bSgjelinek 	"set " ALIAS_MAXSHMMEM "=",
3410209230bSgjelinek 	"set " ALIAS_MAXSHMIDS "=",
3420209230bSgjelinek 	"set " ALIAS_MAXMSGIDS "=",
3430209230bSgjelinek 	"set " ALIAS_MAXSEMIDS "=",
3440209230bSgjelinek 	"set " ALIAS_SHARES "=",
3455679c89fSjv227347 	"set hostid=",
3460fbb751dSJohn Levon 	"set fs-allowed=",
347ff19e029SMenno Lageman 	"set " ALIAS_MAXPROCS "=",
3487c478bd9Sstevel@tonic-gate 	NULL
3497c478bd9Sstevel@tonic-gate };
3507c478bd9Sstevel@tonic-gate 
3519e7542f4Sdp static const char *info_cmds[] = {
3529e7542f4Sdp 	"info fs ",
3539e7542f4Sdp 	"info net ",
3549e7542f4Sdp 	"info device ",
3559e7542f4Sdp 	"info rctl ",
3569e7542f4Sdp 	"info attr ",
3579e7542f4Sdp 	"info dataset ",
3580209230bSgjelinek 	"info capped-memory",
3590209230bSgjelinek 	"info dedicated-cpu",
360c97ad5cdSakolb 	"info capped-cpu",
3619e7542f4Sdp 	"info zonename",
3629e7542f4Sdp 	"info zonepath",
3639e7542f4Sdp 	"info autoboot",
3649e7542f4Sdp 	"info pool",
3659e7542f4Sdp 	"info limitpriv",
3669e7542f4Sdp 	"info bootargs",
3670209230bSgjelinek 	"info brand",
3680209230bSgjelinek 	"info scheduling-class",
369f4b3ec61Sdh155122 	"info ip-type",
3700209230bSgjelinek 	"info max-lwps",
3710209230bSgjelinek 	"info max-shm-memory",
3720209230bSgjelinek 	"info max-shm-ids",
3730209230bSgjelinek 	"info max-msg-ids",
3740209230bSgjelinek 	"info max-sem-ids",
3750209230bSgjelinek 	"info cpu-shares",
3765679c89fSjv227347 	"info hostid",
377cb8a054bSGlenn Faden 	"info admin",
3780fbb751dSJohn Levon 	"info fs-allowed",
379ff19e029SMenno Lageman 	"info max-processes",
3809e7542f4Sdp 	NULL
3819e7542f4Sdp };
3829e7542f4Sdp 
3837c478bd9Sstevel@tonic-gate static const char *fs_res_scope_cmds[] = {
3847c478bd9Sstevel@tonic-gate 	"add options ",
3857c478bd9Sstevel@tonic-gate 	"cancel",
3867c478bd9Sstevel@tonic-gate 	"end",
3877c478bd9Sstevel@tonic-gate 	"exit",
3887c478bd9Sstevel@tonic-gate 	"help",
3897c478bd9Sstevel@tonic-gate 	"info",
390ffbafc53Scomay 	"remove options ",
3917c478bd9Sstevel@tonic-gate 	"set dir=",
3927c478bd9Sstevel@tonic-gate 	"set raw=",
3937c478bd9Sstevel@tonic-gate 	"set special=",
3947c478bd9Sstevel@tonic-gate 	"set type=",
3950209230bSgjelinek 	"clear raw",
3967c478bd9Sstevel@tonic-gate 	NULL
3977c478bd9Sstevel@tonic-gate };
3987c478bd9Sstevel@tonic-gate 
3997c478bd9Sstevel@tonic-gate static const char *net_res_scope_cmds[] = {
4007c478bd9Sstevel@tonic-gate 	"cancel",
4017c478bd9Sstevel@tonic-gate 	"end",
4027c478bd9Sstevel@tonic-gate 	"exit",
4037c478bd9Sstevel@tonic-gate 	"help",
4047c478bd9Sstevel@tonic-gate 	"info",
4057c478bd9Sstevel@tonic-gate 	"set address=",
4067c478bd9Sstevel@tonic-gate 	"set physical=",
4071b3281c0SGerald Jelinek 	"set defrouter=",
4087c478bd9Sstevel@tonic-gate 	NULL
4097c478bd9Sstevel@tonic-gate };
4107c478bd9Sstevel@tonic-gate 
4117c478bd9Sstevel@tonic-gate static const char *device_res_scope_cmds[] = {
4127c478bd9Sstevel@tonic-gate 	"cancel",
4137c478bd9Sstevel@tonic-gate 	"end",
4147c478bd9Sstevel@tonic-gate 	"exit",
4157c478bd9Sstevel@tonic-gate 	"help",
4167c478bd9Sstevel@tonic-gate 	"info",
4177c478bd9Sstevel@tonic-gate 	"set match=",
4187c478bd9Sstevel@tonic-gate 	NULL
4197c478bd9Sstevel@tonic-gate };
4207c478bd9Sstevel@tonic-gate 
4217c478bd9Sstevel@tonic-gate static const char *attr_res_scope_cmds[] = {
4227c478bd9Sstevel@tonic-gate 	"cancel",
4237c478bd9Sstevel@tonic-gate 	"end",
4247c478bd9Sstevel@tonic-gate 	"exit",
4257c478bd9Sstevel@tonic-gate 	"help",
4267c478bd9Sstevel@tonic-gate 	"info",
4277c478bd9Sstevel@tonic-gate 	"set name=",
4287c478bd9Sstevel@tonic-gate 	"set type=",
4297c478bd9Sstevel@tonic-gate 	"set value=",
4307c478bd9Sstevel@tonic-gate 	NULL
4317c478bd9Sstevel@tonic-gate };
4327c478bd9Sstevel@tonic-gate 
4337c478bd9Sstevel@tonic-gate static const char *rctl_res_scope_cmds[] = {
4347c478bd9Sstevel@tonic-gate 	"add value ",
4357c478bd9Sstevel@tonic-gate 	"cancel",
4367c478bd9Sstevel@tonic-gate 	"end",
4377c478bd9Sstevel@tonic-gate 	"exit",
4387c478bd9Sstevel@tonic-gate 	"help",
4397c478bd9Sstevel@tonic-gate 	"info",
440ffbafc53Scomay 	"remove value ",
4417c478bd9Sstevel@tonic-gate 	"set name=",
4427c478bd9Sstevel@tonic-gate 	NULL
4437c478bd9Sstevel@tonic-gate };
4447c478bd9Sstevel@tonic-gate 
445fa9e4066Sahrens static const char *dataset_res_scope_cmds[] = {
446fa9e4066Sahrens 	"cancel",
447fa9e4066Sahrens 	"end",
448fa9e4066Sahrens 	"exit",
449fa9e4066Sahrens 	"help",
450fa9e4066Sahrens 	"info",
451fa9e4066Sahrens 	"set name=",
452fa9e4066Sahrens 	NULL
453fa9e4066Sahrens };
454fa9e4066Sahrens 
4550209230bSgjelinek static const char *pset_res_scope_cmds[] = {
4560209230bSgjelinek 	"cancel",
4570209230bSgjelinek 	"end",
4580209230bSgjelinek 	"exit",
4590209230bSgjelinek 	"help",
4600209230bSgjelinek 	"info",
4610209230bSgjelinek 	"set ncpus=",
4620209230bSgjelinek 	"set importance=",
4630209230bSgjelinek 	"clear importance",
4640209230bSgjelinek 	NULL
4650209230bSgjelinek };
4660209230bSgjelinek 
467c97ad5cdSakolb static const char *pcap_res_scope_cmds[] = {
468c97ad5cdSakolb 	"cancel",
469c97ad5cdSakolb 	"end",
470c97ad5cdSakolb 	"exit",
471c97ad5cdSakolb 	"help",
472c97ad5cdSakolb 	"info",
473c97ad5cdSakolb 	"set ncpus=",
474c97ad5cdSakolb 	NULL
475c97ad5cdSakolb };
476c97ad5cdSakolb 
4770209230bSgjelinek static const char *mcap_res_scope_cmds[] = {
4780209230bSgjelinek 	"cancel",
4790209230bSgjelinek 	"end",
4800209230bSgjelinek 	"exit",
4810209230bSgjelinek 	"help",
4820209230bSgjelinek 	"info",
4830209230bSgjelinek 	"set physical=",
4840209230bSgjelinek 	"set swap=",
4850209230bSgjelinek 	"set locked=",
4860209230bSgjelinek 	"clear physical",
4870209230bSgjelinek 	"clear swap",
4880209230bSgjelinek 	"clear locked",
4890209230bSgjelinek 	NULL
4900209230bSgjelinek };
4910209230bSgjelinek 
492cb8a054bSGlenn Faden static const char *admin_res_scope_cmds[] = {
493cb8a054bSGlenn Faden 	"cancel",
494cb8a054bSGlenn Faden 	"end",
495cb8a054bSGlenn Faden 	"exit",
496cb8a054bSGlenn Faden 	"help",
497cb8a054bSGlenn Faden 	"info",
498cb8a054bSGlenn Faden 	"set user=",
499cb8a054bSGlenn Faden 	"set auths=",
500cb8a054bSGlenn Faden 	NULL
501cb8a054bSGlenn Faden };
502cb8a054bSGlenn Faden 
5037c478bd9Sstevel@tonic-gate /* Global variables */
5047c478bd9Sstevel@tonic-gate 
5057c478bd9Sstevel@tonic-gate /* set early in main(), never modified thereafter, used all over the place */
5067c478bd9Sstevel@tonic-gate static char *execname;
5077c478bd9Sstevel@tonic-gate 
5087c478bd9Sstevel@tonic-gate /* set in main(), used all over the place */
5097c478bd9Sstevel@tonic-gate static zone_dochandle_t handle;
5107c478bd9Sstevel@tonic-gate 
5117c478bd9Sstevel@tonic-gate /* used all over the place */
512087719fdSdp static char zone[ZONENAME_MAX];
513087719fdSdp static char revert_zone[ZONENAME_MAX];
5147c478bd9Sstevel@tonic-gate 
5159acbbeafSnn35248 /* global brand operations */
516123807fbSedp static brand_handle_t brand;
5179acbbeafSnn35248 
5187c478bd9Sstevel@tonic-gate /* set in modifying functions, checked in read_input() */
519bbec428eSgjelinek static boolean_t need_to_commit = B_FALSE;
520bbec428eSgjelinek boolean_t saw_error;
5217c478bd9Sstevel@tonic-gate 
5227c478bd9Sstevel@tonic-gate /* set in yacc parser, checked in read_input() */
523bbec428eSgjelinek boolean_t newline_terminated;
5247c478bd9Sstevel@tonic-gate 
5257c478bd9Sstevel@tonic-gate /* set in main(), checked in lex error handler */
526bbec428eSgjelinek boolean_t cmd_file_mode;
5277c478bd9Sstevel@tonic-gate 
5287c478bd9Sstevel@tonic-gate /* set in exit_func(), checked in read_input() */
529bbec428eSgjelinek static boolean_t time_to_exit = B_FALSE, force_exit = B_FALSE;
5307c478bd9Sstevel@tonic-gate 
5317c478bd9Sstevel@tonic-gate /* used in short_usage() and zerr() */
5327c478bd9Sstevel@tonic-gate static char *cmd_file_name = NULL;
5337c478bd9Sstevel@tonic-gate 
5347c478bd9Sstevel@tonic-gate /* checked in read_input() and other places */
535bbec428eSgjelinek static boolean_t ok_to_prompt = B_FALSE;
5367c478bd9Sstevel@tonic-gate 
5377c478bd9Sstevel@tonic-gate /* set and checked in initialize() */
538bbec428eSgjelinek static boolean_t got_handle = B_FALSE;
5397c478bd9Sstevel@tonic-gate 
5407c478bd9Sstevel@tonic-gate /* initialized in do_interactive(), checked in initialize() */
541bbec428eSgjelinek static boolean_t interactive_mode;
5427c478bd9Sstevel@tonic-gate 
5430209230bSgjelinek /* set if configuring the global zone */
544bbec428eSgjelinek static boolean_t global_zone = B_FALSE;
5450209230bSgjelinek 
5467c478bd9Sstevel@tonic-gate /* set in main(), checked in multiple places */
547bbec428eSgjelinek static boolean_t read_only_mode;
5487c478bd9Sstevel@tonic-gate 
549bbec428eSgjelinek /* scope is outer/global or inner/resource */
550bbec428eSgjelinek static boolean_t global_scope = B_TRUE;
5517c478bd9Sstevel@tonic-gate static int resource_scope;	/* should be in the RT_ list from zonecfg.h */
5527c478bd9Sstevel@tonic-gate static int end_op = -1;		/* operation on end is either add or modify */
5537c478bd9Sstevel@tonic-gate 
5547c478bd9Sstevel@tonic-gate int num_prop_vals;		/* for grammar */
5557c478bd9Sstevel@tonic-gate 
5567c478bd9Sstevel@tonic-gate /*
5577c478bd9Sstevel@tonic-gate  * These are for keeping track of resources as they are specified as part of
5587c478bd9Sstevel@tonic-gate  * the multi-step process.  They should be initialized by add_resource() or
5597c478bd9Sstevel@tonic-gate  * select_func() and filled in by add_property() or set_func().
5607c478bd9Sstevel@tonic-gate  */
5617c478bd9Sstevel@tonic-gate static struct zone_fstab	old_fstab, in_progress_fstab;
5627c478bd9Sstevel@tonic-gate static struct zone_nwiftab	old_nwiftab, in_progress_nwiftab;
5637c478bd9Sstevel@tonic-gate static struct zone_devtab	old_devtab, in_progress_devtab;
5647c478bd9Sstevel@tonic-gate static struct zone_rctltab	old_rctltab, in_progress_rctltab;
5657c478bd9Sstevel@tonic-gate static struct zone_attrtab	old_attrtab, in_progress_attrtab;
566fa9e4066Sahrens static struct zone_dstab	old_dstab, in_progress_dstab;
5670209230bSgjelinek static struct zone_psettab	old_psettab, in_progress_psettab;
5680209230bSgjelinek static struct zone_mcaptab	old_mcaptab, in_progress_mcaptab;
569cb8a054bSGlenn Faden static struct zone_admintab	old_admintab, in_progress_admintab;
5707c478bd9Sstevel@tonic-gate 
5717c478bd9Sstevel@tonic-gate static GetLine *gl;	/* The gl_get_line() resource object */
5727c478bd9Sstevel@tonic-gate 
5730209230bSgjelinek static void bytes_to_units(char *str, char *buf, int bufsize);
5740209230bSgjelinek 
5757c478bd9Sstevel@tonic-gate /* Functions begin here */
5767c478bd9Sstevel@tonic-gate 
577bbec428eSgjelinek static boolean_t
5787c478bd9Sstevel@tonic-gate initial_match(const char *line1, const char *line2, int word_end)
5797c478bd9Sstevel@tonic-gate {
5807c478bd9Sstevel@tonic-gate 	if (word_end <= 0)
581bbec428eSgjelinek 		return (B_TRUE);
5827c478bd9Sstevel@tonic-gate 	return (strncmp(line1, line2, word_end) == 0);
5837c478bd9Sstevel@tonic-gate }
5847c478bd9Sstevel@tonic-gate 
5857c478bd9Sstevel@tonic-gate static int
5867c478bd9Sstevel@tonic-gate add_stuff(WordCompletion *cpl, const char *line1, const char **list,
5877c478bd9Sstevel@tonic-gate     int word_end)
5887c478bd9Sstevel@tonic-gate {
5897c478bd9Sstevel@tonic-gate 	int i, err;
5907c478bd9Sstevel@tonic-gate 
5917c478bd9Sstevel@tonic-gate 	for (i = 0; list[i] != NULL; i++) {
5927c478bd9Sstevel@tonic-gate 		if (initial_match(line1, list[i], word_end)) {
5937c478bd9Sstevel@tonic-gate 			err = cpl_add_completion(cpl, line1, 0, word_end,
5947c478bd9Sstevel@tonic-gate 			    list[i] + word_end, "", "");
5957c478bd9Sstevel@tonic-gate 			if (err != 0)
5967c478bd9Sstevel@tonic-gate 				return (err);
5977c478bd9Sstevel@tonic-gate 		}
5987c478bd9Sstevel@tonic-gate 	}
5997c478bd9Sstevel@tonic-gate 	return (0);
6007c478bd9Sstevel@tonic-gate }
6017c478bd9Sstevel@tonic-gate 
6027c478bd9Sstevel@tonic-gate static
6037c478bd9Sstevel@tonic-gate /* ARGSUSED */
6047c478bd9Sstevel@tonic-gate CPL_MATCH_FN(cmd_cpl_fn)
6057c478bd9Sstevel@tonic-gate {
6067c478bd9Sstevel@tonic-gate 	if (global_scope) {
6077c478bd9Sstevel@tonic-gate 		/*
6087c478bd9Sstevel@tonic-gate 		 * The MAX/MIN tests below are to make sure we have at least
6097c478bd9Sstevel@tonic-gate 		 * enough characters to distinguish from other prefixes (MAX)
6107c478bd9Sstevel@tonic-gate 		 * but only check MIN(what we have, what we're checking).
6117c478bd9Sstevel@tonic-gate 		 */
6127c478bd9Sstevel@tonic-gate 		if (strncmp(line, "add ", MAX(MIN(word_end, 4), 1)) == 0)
6137c478bd9Sstevel@tonic-gate 			return (add_stuff(cpl, line, add_cmds, word_end));
6140209230bSgjelinek 		if (strncmp(line, "clear ", MAX(MIN(word_end, 6), 2)) == 0)
6150209230bSgjelinek 			return (add_stuff(cpl, line, clear_cmds, word_end));
6167c478bd9Sstevel@tonic-gate 		if (strncmp(line, "select ", MAX(MIN(word_end, 7), 3)) == 0)
6177c478bd9Sstevel@tonic-gate 			return (add_stuff(cpl, line, select_cmds, word_end));
6187c478bd9Sstevel@tonic-gate 		if (strncmp(line, "set ", MAX(MIN(word_end, 4), 3)) == 0)
6197c478bd9Sstevel@tonic-gate 			return (add_stuff(cpl, line, set_cmds, word_end));
6209e7542f4Sdp 		if (strncmp(line, "remove ", MAX(MIN(word_end, 7), 1)) == 0)
6219e7542f4Sdp 			return (add_stuff(cpl, line, remove_cmds, word_end));
6229e7542f4Sdp 		if (strncmp(line, "info ", MAX(MIN(word_end, 5), 1)) == 0)
6239e7542f4Sdp 			return (add_stuff(cpl, line, info_cmds, word_end));
6247c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, global_scope_cmds, word_end));
6257c478bd9Sstevel@tonic-gate 	}
6267c478bd9Sstevel@tonic-gate 	switch (resource_scope) {
6277c478bd9Sstevel@tonic-gate 	case RT_FS:
6287c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, fs_res_scope_cmds, word_end));
6297c478bd9Sstevel@tonic-gate 	case RT_NET:
6307c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, net_res_scope_cmds, word_end));
6317c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
6327c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, device_res_scope_cmds, word_end));
6337c478bd9Sstevel@tonic-gate 	case RT_RCTL:
6347c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, rctl_res_scope_cmds, word_end));
6357c478bd9Sstevel@tonic-gate 	case RT_ATTR:
6367c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, attr_res_scope_cmds, word_end));
637fa9e4066Sahrens 	case RT_DATASET:
638fa9e4066Sahrens 		return (add_stuff(cpl, line, dataset_res_scope_cmds, word_end));
6390209230bSgjelinek 	case RT_DCPU:
6400209230bSgjelinek 		return (add_stuff(cpl, line, pset_res_scope_cmds, word_end));
641c97ad5cdSakolb 	case RT_PCAP:
642c97ad5cdSakolb 		return (add_stuff(cpl, line, pcap_res_scope_cmds, word_end));
6430209230bSgjelinek 	case RT_MCAP:
6440209230bSgjelinek 		return (add_stuff(cpl, line, mcap_res_scope_cmds, word_end));
645cb8a054bSGlenn Faden 	case RT_ADMIN:
646cb8a054bSGlenn Faden 		return (add_stuff(cpl, line, admin_res_scope_cmds, word_end));
6477c478bd9Sstevel@tonic-gate 	}
6487c478bd9Sstevel@tonic-gate 	return (0);
6497c478bd9Sstevel@tonic-gate }
6507c478bd9Sstevel@tonic-gate 
6517c478bd9Sstevel@tonic-gate /*
6527c478bd9Sstevel@tonic-gate  * For the main CMD_func() functions below, several of them call getopt()
6537c478bd9Sstevel@tonic-gate  * then check optind against argc to make sure an extra parameter was not
6547c478bd9Sstevel@tonic-gate  * passed in.  The reason this is not caught in the grammar is that the
6557c478bd9Sstevel@tonic-gate  * grammar just checks for a miscellaneous TOKEN, which is *expected* to
6567c478bd9Sstevel@tonic-gate  * be "-F" (for example), but could be anything.  So (for example) this
6577c478bd9Sstevel@tonic-gate  * check will prevent "create bogus".
6587c478bd9Sstevel@tonic-gate  */
6597c478bd9Sstevel@tonic-gate 
6607c478bd9Sstevel@tonic-gate cmd_t *
6617c478bd9Sstevel@tonic-gate alloc_cmd(void)
6627c478bd9Sstevel@tonic-gate {
6637c478bd9Sstevel@tonic-gate 	return (calloc(1, sizeof (cmd_t)));
6647c478bd9Sstevel@tonic-gate }
6657c478bd9Sstevel@tonic-gate 
6667c478bd9Sstevel@tonic-gate void
6677c478bd9Sstevel@tonic-gate free_cmd(cmd_t *cmd)
6687c478bd9Sstevel@tonic-gate {
6697c478bd9Sstevel@tonic-gate 	int i;
6707c478bd9Sstevel@tonic-gate 
6717c478bd9Sstevel@tonic-gate 	for (i = 0; i < MAX_EQ_PROP_PAIRS; i++)
6727c478bd9Sstevel@tonic-gate 		if (cmd->cmd_property_ptr[i] != NULL) {
6737c478bd9Sstevel@tonic-gate 			property_value_ptr_t pp = cmd->cmd_property_ptr[i];
6747c478bd9Sstevel@tonic-gate 
6757c478bd9Sstevel@tonic-gate 			switch (pp->pv_type) {
6767c478bd9Sstevel@tonic-gate 			case PROP_VAL_SIMPLE:
6777c478bd9Sstevel@tonic-gate 				free(pp->pv_simple);
6787c478bd9Sstevel@tonic-gate 				break;
6797c478bd9Sstevel@tonic-gate 			case PROP_VAL_COMPLEX:
6807c478bd9Sstevel@tonic-gate 				free_complex(pp->pv_complex);
6817c478bd9Sstevel@tonic-gate 				break;
6827c478bd9Sstevel@tonic-gate 			case PROP_VAL_LIST:
6837c478bd9Sstevel@tonic-gate 				free_list(pp->pv_list);
6847c478bd9Sstevel@tonic-gate 				break;
6857c478bd9Sstevel@tonic-gate 			}
6867c478bd9Sstevel@tonic-gate 		}
6877c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_argc; i++)
6887c478bd9Sstevel@tonic-gate 		free(cmd->cmd_argv[i]);
6897c478bd9Sstevel@tonic-gate 	free(cmd);
6907c478bd9Sstevel@tonic-gate }
6917c478bd9Sstevel@tonic-gate 
6927c478bd9Sstevel@tonic-gate complex_property_ptr_t
6937c478bd9Sstevel@tonic-gate alloc_complex(void)
6947c478bd9Sstevel@tonic-gate {
6957c478bd9Sstevel@tonic-gate 	return (calloc(1, sizeof (complex_property_t)));
6967c478bd9Sstevel@tonic-gate }
6977c478bd9Sstevel@tonic-gate 
6987c478bd9Sstevel@tonic-gate void
6997c478bd9Sstevel@tonic-gate free_complex(complex_property_ptr_t complex)
7007c478bd9Sstevel@tonic-gate {
7017c478bd9Sstevel@tonic-gate 	if (complex == NULL)
7027c478bd9Sstevel@tonic-gate 		return;
7037c478bd9Sstevel@tonic-gate 	free_complex(complex->cp_next);
7047c478bd9Sstevel@tonic-gate 	if (complex->cp_value != NULL)
7057c478bd9Sstevel@tonic-gate 		free(complex->cp_value);
7067c478bd9Sstevel@tonic-gate 	free(complex);
7077c478bd9Sstevel@tonic-gate }
7087c478bd9Sstevel@tonic-gate 
7097c478bd9Sstevel@tonic-gate list_property_ptr_t
7107c478bd9Sstevel@tonic-gate alloc_list(void)
7117c478bd9Sstevel@tonic-gate {
7127c478bd9Sstevel@tonic-gate 	return (calloc(1, sizeof (list_property_t)));
7137c478bd9Sstevel@tonic-gate }
7147c478bd9Sstevel@tonic-gate 
7157c478bd9Sstevel@tonic-gate void
7167c478bd9Sstevel@tonic-gate free_list(list_property_ptr_t list)
7177c478bd9Sstevel@tonic-gate {
7187c478bd9Sstevel@tonic-gate 	if (list == NULL)
7197c478bd9Sstevel@tonic-gate 		return;
7207c478bd9Sstevel@tonic-gate 	if (list->lp_simple != NULL)
7217c478bd9Sstevel@tonic-gate 		free(list->lp_simple);
7227c478bd9Sstevel@tonic-gate 	free_complex(list->lp_complex);
7237c478bd9Sstevel@tonic-gate 	free_list(list->lp_next);
7247c478bd9Sstevel@tonic-gate 	free(list);
7257c478bd9Sstevel@tonic-gate }
7267c478bd9Sstevel@tonic-gate 
7277c478bd9Sstevel@tonic-gate void
7287c478bd9Sstevel@tonic-gate free_outer_list(list_property_ptr_t list)
7297c478bd9Sstevel@tonic-gate {
7307c478bd9Sstevel@tonic-gate 	if (list == NULL)
7317c478bd9Sstevel@tonic-gate 		return;
7327c478bd9Sstevel@tonic-gate 	free_outer_list(list->lp_next);
7337c478bd9Sstevel@tonic-gate 	free(list);
7347c478bd9Sstevel@tonic-gate }
7357c478bd9Sstevel@tonic-gate 
7367c478bd9Sstevel@tonic-gate static struct zone_rctlvaltab *
7377c478bd9Sstevel@tonic-gate alloc_rctlvaltab(void)
7387c478bd9Sstevel@tonic-gate {
7397c478bd9Sstevel@tonic-gate 	return (calloc(1, sizeof (struct zone_rctlvaltab)));
7407c478bd9Sstevel@tonic-gate }
7417c478bd9Sstevel@tonic-gate 
7427c478bd9Sstevel@tonic-gate static char *
7437c478bd9Sstevel@tonic-gate rt_to_str(int res_type)
7447c478bd9Sstevel@tonic-gate {
7457c478bd9Sstevel@tonic-gate 	assert(res_type >= RT_MIN && res_type <= RT_MAX);
7467c478bd9Sstevel@tonic-gate 	return (res_types[res_type]);
7477c478bd9Sstevel@tonic-gate }
7487c478bd9Sstevel@tonic-gate 
7497c478bd9Sstevel@tonic-gate static char *
7507c478bd9Sstevel@tonic-gate pt_to_str(int prop_type)
7517c478bd9Sstevel@tonic-gate {
7527c478bd9Sstevel@tonic-gate 	assert(prop_type >= PT_MIN && prop_type <= PT_MAX);
7537c478bd9Sstevel@tonic-gate 	return (prop_types[prop_type]);
7547c478bd9Sstevel@tonic-gate }
7557c478bd9Sstevel@tonic-gate 
7567c478bd9Sstevel@tonic-gate static char *
7577c478bd9Sstevel@tonic-gate pvt_to_str(int pv_type)
7587c478bd9Sstevel@tonic-gate {
7597c478bd9Sstevel@tonic-gate 	assert(pv_type >= PROP_VAL_MIN && pv_type <= PROP_VAL_MAX);
7607c478bd9Sstevel@tonic-gate 	return (prop_val_types[pv_type]);
7617c478bd9Sstevel@tonic-gate }
7627c478bd9Sstevel@tonic-gate 
7637c478bd9Sstevel@tonic-gate static char *
7647c478bd9Sstevel@tonic-gate cmd_to_str(int cmd_num)
7657c478bd9Sstevel@tonic-gate {
7667c478bd9Sstevel@tonic-gate 	assert(cmd_num >= CMD_MIN && cmd_num <= CMD_MAX);
7677c478bd9Sstevel@tonic-gate 	return (helptab[cmd_num].cmd_name);
7687c478bd9Sstevel@tonic-gate }
7697c478bd9Sstevel@tonic-gate 
7703042b8b5Sbatschul /* PRINTFLIKE1 */
7713042b8b5Sbatschul static void
7723042b8b5Sbatschul zerr(const char *fmt, ...)
7733042b8b5Sbatschul {
7743042b8b5Sbatschul 	va_list alist;
7753042b8b5Sbatschul 	static int last_lineno;
7763042b8b5Sbatschul 
7773042b8b5Sbatschul 	/* lex_lineno has already been incremented in the lexer; compensate */
7783042b8b5Sbatschul 	if (cmd_file_mode && lex_lineno > last_lineno) {
7793042b8b5Sbatschul 		if (strcmp(cmd_file_name, "-") == 0)
7803042b8b5Sbatschul 			(void) fprintf(stderr, gettext("On line %d:\n"),
7813042b8b5Sbatschul 			    lex_lineno - 1);
7823042b8b5Sbatschul 		else
7833042b8b5Sbatschul 			(void) fprintf(stderr, gettext("On line %d of %s:\n"),
7843042b8b5Sbatschul 			    lex_lineno - 1, cmd_file_name);
7853042b8b5Sbatschul 		last_lineno = lex_lineno;
7863042b8b5Sbatschul 	}
7873042b8b5Sbatschul 	va_start(alist, fmt);
7883042b8b5Sbatschul 	(void) vfprintf(stderr, fmt, alist);
7893042b8b5Sbatschul 	(void) fprintf(stderr, "\n");
7903042b8b5Sbatschul 	va_end(alist);
7913042b8b5Sbatschul }
7923042b8b5Sbatschul 
7937c478bd9Sstevel@tonic-gate /*
7947c478bd9Sstevel@tonic-gate  * This is a separate function rather than a set of define's because of the
7957c478bd9Sstevel@tonic-gate  * gettext() wrapping.
7967c478bd9Sstevel@tonic-gate  */
7977c478bd9Sstevel@tonic-gate 
7987c478bd9Sstevel@tonic-gate /*
7997c478bd9Sstevel@tonic-gate  * TRANSLATION_NOTE
8007c478bd9Sstevel@tonic-gate  * Each string below should have \t follow \n whenever needed; the
8017c478bd9Sstevel@tonic-gate  * initial \t and the terminal \n will be provided by the calling function.
8027c478bd9Sstevel@tonic-gate  */
8037c478bd9Sstevel@tonic-gate 
8047c478bd9Sstevel@tonic-gate static char *
8057c478bd9Sstevel@tonic-gate long_help(int cmd_num)
8067c478bd9Sstevel@tonic-gate {
8077c478bd9Sstevel@tonic-gate 	static char line[1024];	/* arbitrary large amount */
8087c478bd9Sstevel@tonic-gate 
8097c478bd9Sstevel@tonic-gate 	assert(cmd_num >= CMD_MIN && cmd_num <= CMD_MAX);
8107c478bd9Sstevel@tonic-gate 	switch (cmd_num) {
8117c478bd9Sstevel@tonic-gate 		case CMD_HELP:
8127c478bd9Sstevel@tonic-gate 			return (gettext("Prints help message."));
8137c478bd9Sstevel@tonic-gate 		case CMD_CREATE:
8147c478bd9Sstevel@tonic-gate 			(void) snprintf(line, sizeof (line),
8157c478bd9Sstevel@tonic-gate 			    gettext("Creates a configuration for the "
8167c478bd9Sstevel@tonic-gate 			    "specified zone.  %s should be\n\tused to "
8177c478bd9Sstevel@tonic-gate 			    "begin configuring a new zone.  If overwriting an "
8187c478bd9Sstevel@tonic-gate 			    "existing\n\tconfiguration, the -F flag can be "
8197c478bd9Sstevel@tonic-gate 			    "used to force the action.  If\n\t-t template is "
8207c478bd9Sstevel@tonic-gate 			    "given, creates a configuration identical to the\n"
8217c478bd9Sstevel@tonic-gate 			    "\tspecified template, except that the zone name "
8229e518655Sgjelinek 			    "is changed from\n\ttemplate to zonename.  '%s -a' "
8239e518655Sgjelinek 			    "creates a configuration from a\n\tdetached "
8249e518655Sgjelinek 			    "zonepath.  '%s -b' results in a blank "
8259e518655Sgjelinek 			    "configuration.\n\t'%s' with no arguments applies "
8269e518655Sgjelinek 			    "the Sun default settings."),
8277c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_CREATE), cmd_to_str(CMD_CREATE),
8289e518655Sgjelinek 			    cmd_to_str(CMD_CREATE), cmd_to_str(CMD_CREATE));
8297c478bd9Sstevel@tonic-gate 			return (line);
8307c478bd9Sstevel@tonic-gate 		case CMD_EXIT:
8317c478bd9Sstevel@tonic-gate 			return (gettext("Exits the program.  The -F flag can "
8327c478bd9Sstevel@tonic-gate 			    "be used to force the action."));
8337c478bd9Sstevel@tonic-gate 		case CMD_EXPORT:
8347c478bd9Sstevel@tonic-gate 			return (gettext("Prints configuration to standard "
8357c478bd9Sstevel@tonic-gate 			    "output, or to output-file if\n\tspecified, in "
8367c478bd9Sstevel@tonic-gate 			    "a form suitable for use in a command-file."));
8377c478bd9Sstevel@tonic-gate 		case CMD_ADD:
8387c478bd9Sstevel@tonic-gate 			return (gettext("Add specified resource to "
8397c478bd9Sstevel@tonic-gate 			    "configuration."));
8407c478bd9Sstevel@tonic-gate 		case CMD_DELETE:
8417c478bd9Sstevel@tonic-gate 			return (gettext("Deletes the specified zone.  The -F "
8427c478bd9Sstevel@tonic-gate 			    "flag can be used to force the\n\taction."));
8437c478bd9Sstevel@tonic-gate 		case CMD_REMOVE:
8447c478bd9Sstevel@tonic-gate 			return (gettext("Remove specified resource from "
8450209230bSgjelinek 			    "configuration.  The -F flag can be used\n\tto "
8460209230bSgjelinek 			    "force the action."));
8477c478bd9Sstevel@tonic-gate 		case CMD_SELECT:
8487c478bd9Sstevel@tonic-gate 			(void) snprintf(line, sizeof (line),
8497c478bd9Sstevel@tonic-gate 			    gettext("Selects a resource to modify.  "
8507c478bd9Sstevel@tonic-gate 			    "Resource modification is completed\n\twith the "
8517c478bd9Sstevel@tonic-gate 			    "command \"%s\".  The property name/value pairs "
8527c478bd9Sstevel@tonic-gate 			    "must uniquely\n\tidentify a resource.  Note that "
8537c478bd9Sstevel@tonic-gate 			    "the curly braces ('{', '}') mean one\n\tor more "
8547c478bd9Sstevel@tonic-gate 			    "of whatever is between them."),
8557c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_END));
8567c478bd9Sstevel@tonic-gate 			return (line);
8577c478bd9Sstevel@tonic-gate 		case CMD_SET:
8587c478bd9Sstevel@tonic-gate 			return (gettext("Sets property values."));
8590209230bSgjelinek 		case CMD_CLEAR:
8600209230bSgjelinek 			return (gettext("Clears property values."));
8617c478bd9Sstevel@tonic-gate 		case CMD_INFO:
8627c478bd9Sstevel@tonic-gate 			return (gettext("Displays information about the "
8637c478bd9Sstevel@tonic-gate 			    "current configuration.  If resource\n\ttype is "
8647c478bd9Sstevel@tonic-gate 			    "specified, displays only information about "
8657c478bd9Sstevel@tonic-gate 			    "resources of\n\tthe relevant type.  If resource "
8667c478bd9Sstevel@tonic-gate 			    "id is specified, displays only\n\tinformation "
8677c478bd9Sstevel@tonic-gate 			    "about that resource."));
8687c478bd9Sstevel@tonic-gate 		case CMD_VERIFY:
8697c478bd9Sstevel@tonic-gate 			return (gettext("Verifies current configuration "
8707c478bd9Sstevel@tonic-gate 			    "for correctness (some resource types\n\thave "
8717c478bd9Sstevel@tonic-gate 			    "required properties)."));
8727c478bd9Sstevel@tonic-gate 		case CMD_COMMIT:
8737c478bd9Sstevel@tonic-gate 			(void) snprintf(line, sizeof (line),
8747c478bd9Sstevel@tonic-gate 			    gettext("Commits current configuration.  "
8757c478bd9Sstevel@tonic-gate 			    "Configuration must be committed to\n\tbe used by "
8767c478bd9Sstevel@tonic-gate 			    "%s.  Until the configuration is committed, "
8777c478bd9Sstevel@tonic-gate 			    "changes \n\tcan be removed with the %s "
8787c478bd9Sstevel@tonic-gate 			    "command.  This operation is\n\tattempted "
8797c478bd9Sstevel@tonic-gate 			    "automatically upon completion of a %s "
8807c478bd9Sstevel@tonic-gate 			    "session."), "zoneadm", cmd_to_str(CMD_REVERT),
8817c478bd9Sstevel@tonic-gate 			    "zonecfg");
8827c478bd9Sstevel@tonic-gate 			return (line);
8837c478bd9Sstevel@tonic-gate 		case CMD_REVERT:
8847c478bd9Sstevel@tonic-gate 			return (gettext("Reverts configuration back to the "
8857c478bd9Sstevel@tonic-gate 			    "last committed state.  The -F flag\n\tcan be "
8867c478bd9Sstevel@tonic-gate 			    "used to force the action."));
8877c478bd9Sstevel@tonic-gate 		case CMD_CANCEL:
8887c478bd9Sstevel@tonic-gate 			return (gettext("Cancels resource/property "
8897c478bd9Sstevel@tonic-gate 			    "specification."));
8907c478bd9Sstevel@tonic-gate 		case CMD_END:
8917c478bd9Sstevel@tonic-gate 			return (gettext("Ends resource/property "
8927c478bd9Sstevel@tonic-gate 			    "specification."));
8937c478bd9Sstevel@tonic-gate 	}
8947c478bd9Sstevel@tonic-gate 	/* NOTREACHED */
8957e362f58Scomay 	return (NULL);
8967c478bd9Sstevel@tonic-gate }
8977c478bd9Sstevel@tonic-gate 
8987c478bd9Sstevel@tonic-gate /*
8997c478bd9Sstevel@tonic-gate  * Called with verbose TRUE when help is explicitly requested, FALSE for
9007c478bd9Sstevel@tonic-gate  * unexpected errors.
9017c478bd9Sstevel@tonic-gate  */
9027c478bd9Sstevel@tonic-gate 
9037c478bd9Sstevel@tonic-gate void
904bbec428eSgjelinek usage(boolean_t verbose, uint_t flags)
9057c478bd9Sstevel@tonic-gate {
9063042b8b5Sbatschul 	FILE *fp = verbose ? stdout : stderr;
9073042b8b5Sbatschul 	FILE *newfp;
908bbec428eSgjelinek 	boolean_t need_to_close = B_FALSE;
9097c478bd9Sstevel@tonic-gate 	char *pager;
9107c478bd9Sstevel@tonic-gate 	int i;
9113042b8b5Sbatschul 	struct stat statbuf;
9127c478bd9Sstevel@tonic-gate 
9137c478bd9Sstevel@tonic-gate 	/* don't page error output */
9147c478bd9Sstevel@tonic-gate 	if (verbose && interactive_mode) {
9157c478bd9Sstevel@tonic-gate 		if ((pager = getenv("PAGER")) == NULL)
9167c478bd9Sstevel@tonic-gate 			pager = PAGER;
9173042b8b5Sbatschul 
9183042b8b5Sbatschul 		if (stat(pager, &statbuf) == 0) {
9197c478bd9Sstevel@tonic-gate 			if ((newfp = popen(pager, "w")) != NULL) {
920bbec428eSgjelinek 				need_to_close = B_TRUE;
9217c478bd9Sstevel@tonic-gate 				fp = newfp;
9227c478bd9Sstevel@tonic-gate 			}
9233042b8b5Sbatschul 		} else {
9243042b8b5Sbatschul 			zerr(gettext("PAGER %s does not exist (%s)."),
9253042b8b5Sbatschul 			    pager, strerror(errno));
9267c478bd9Sstevel@tonic-gate 		}
9273042b8b5Sbatschul 	}
9283042b8b5Sbatschul 
9297c478bd9Sstevel@tonic-gate 	if (flags & HELP_META) {
9307c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("More help is available for the "
9317c478bd9Sstevel@tonic-gate 		    "following:\n"));
9327c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\n\tcommands ('%s commands')\n",
9337c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_HELP));
9347c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\tsyntax ('%s syntax')\n",
9357c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_HELP));
9367c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\tusage ('%s usage')\n\n",
9377c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_HELP));
9387c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("You may also obtain help on any "
9397c478bd9Sstevel@tonic-gate 		    "command by typing '%s <command-name>.'\n"),
9407c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_HELP));
9417c478bd9Sstevel@tonic-gate 	}
9427c478bd9Sstevel@tonic-gate 	if (flags & HELP_RES_SCOPE) {
9437c478bd9Sstevel@tonic-gate 		switch (resource_scope) {
9447c478bd9Sstevel@tonic-gate 		case RT_FS:
9457c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
9467c478bd9Sstevel@tonic-gate 			    "used to configure a file-system.\n"),
9477c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
9487c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
9497c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9507c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_DIR), gettext("<path>"));
9517c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9527c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_SPECIAL), gettext("<path>"));
9537c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9547c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_RAW), gettext("<raw-device>"));
9557c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9567c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_TYPE), gettext("<file-system type>"));
9577c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s %s\n", cmd_to_str(CMD_ADD),
9587c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_OPTIONS),
9597c478bd9Sstevel@tonic-gate 			    gettext("<file-system options>"));
960ffbafc53Scomay 			(void) fprintf(fp, "\t%s %s %s\n",
961ffbafc53Scomay 			    cmd_to_str(CMD_REMOVE), pt_to_str(PT_OPTIONS),
962ffbafc53Scomay 			    gettext("<file-system options>"));
9637c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Consult the file-system "
9647c478bd9Sstevel@tonic-gate 			    "specific manual page, such as mount_ufs(1M), "
9657c478bd9Sstevel@tonic-gate 			    "for\ndetails about file-system options.  Note "
9667c478bd9Sstevel@tonic-gate 			    "that any file-system options with an\nembedded "
9677c478bd9Sstevel@tonic-gate 			    "'=' character must be enclosed in double quotes, "
9687c478bd9Sstevel@tonic-gate 			    /*CSTYLED*/
9697c478bd9Sstevel@tonic-gate 			    "such as \"%s=5\".\n"), MNTOPT_RETRY);
9707c478bd9Sstevel@tonic-gate 			break;
9717c478bd9Sstevel@tonic-gate 		case RT_NET:
9727c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
9737c478bd9Sstevel@tonic-gate 			    "used to configure a network interface.\n"),
9747c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
9757c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
9767c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9777c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_ADDRESS), gettext("<IP-address>"));
9787c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9797c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_PHYSICAL), gettext("<interface>"));
9807c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("See ifconfig(1M) for "
9817c478bd9Sstevel@tonic-gate 			    "details of the <interface> string.\n"));
982de860bd9Sgfaden 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
983de860bd9Sgfaden 			    pt_to_str(PT_DEFROUTER), gettext("<IP-address>"));
984de860bd9Sgfaden 			(void) fprintf(fp, gettext("%s %s and %s %s are valid "
985de860bd9Sgfaden 			    "if the %s property is set to %s, otherwise they "
986de860bd9Sgfaden 			    "must not be set.\n"),
987f4b3ec61Sdh155122 			    cmd_to_str(CMD_SET), pt_to_str(PT_ADDRESS),
988de860bd9Sgfaden 			    cmd_to_str(CMD_SET), pt_to_str(PT_DEFROUTER),
989f4b3ec61Sdh155122 			    pt_to_str(PT_IPTYPE), "shared");
9907c478bd9Sstevel@tonic-gate 			break;
9917c478bd9Sstevel@tonic-gate 		case RT_DEVICE:
9927c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
9937c478bd9Sstevel@tonic-gate 			    "used to configure a device node.\n"),
9947c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
9957c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
9967c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9977c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_MATCH), gettext("<device-path>"));
9987c478bd9Sstevel@tonic-gate 			break;
9997c478bd9Sstevel@tonic-gate 		case RT_RCTL:
10007c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
10017c478bd9Sstevel@tonic-gate 			    "used to configure a resource control.\n"),
10027c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
10037c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
10047c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10057c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_NAME), gettext("<string>"));
10067c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s (%s=%s,%s=%s,%s=%s)\n",
10077c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_ADD), pt_to_str(PT_VALUE),
10087c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_PRIV), gettext("<priv-value>"),
10097c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_LIMIT), gettext("<number>"),
10107c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_ACTION), gettext("<action-value>"));
1011ffbafc53Scomay 			(void) fprintf(fp, "\t%s %s (%s=%s,%s=%s,%s=%s)\n",
1012ffbafc53Scomay 			    cmd_to_str(CMD_REMOVE), pt_to_str(PT_VALUE),
1013ffbafc53Scomay 			    pt_to_str(PT_PRIV), gettext("<priv-value>"),
1014ffbafc53Scomay 			    pt_to_str(PT_LIMIT), gettext("<number>"),
1015ffbafc53Scomay 			    pt_to_str(PT_ACTION), gettext("<action-value>"));
10167c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "%s\n\t%s := privileged\n"
10177c478bd9Sstevel@tonic-gate 			    "\t%s := none | deny\n", gettext("Where"),
10187c478bd9Sstevel@tonic-gate 			    gettext("<priv-value>"), gettext("<action-value>"));
10197c478bd9Sstevel@tonic-gate 			break;
10207c478bd9Sstevel@tonic-gate 		case RT_ATTR:
10217c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
10227c478bd9Sstevel@tonic-gate 			    "used to configure a generic attribute.\n"),
10237c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
10247c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
10257c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10267c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_NAME), gettext("<name>"));
10277c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=boolean\n",
10287c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_SET), pt_to_str(PT_TYPE));
10297c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=true | false\n",
10307c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_SET), pt_to_str(PT_VALUE));
10317c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("or\n"));
10327c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=int\n", cmd_to_str(CMD_SET),
10337c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_TYPE));
10347c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10357c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_VALUE), gettext("<integer>"));
10367c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("or\n"));
10377c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=string\n",
10387c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_SET), pt_to_str(PT_TYPE));
10397c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10407c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_VALUE), gettext("<string>"));
10417c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("or\n"));
10427c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=uint\n",
10437c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_SET), pt_to_str(PT_TYPE));
10447c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10457c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_VALUE), gettext("<unsigned integer>"));
10467c478bd9Sstevel@tonic-gate 			break;
1047fa9e4066Sahrens 		case RT_DATASET:
1048fa9e4066Sahrens 			(void) fprintf(fp, gettext("The '%s' resource scope is "
1049fa9e4066Sahrens 			    "used to export ZFS datasets.\n"),
1050fa9e4066Sahrens 			    rt_to_str(resource_scope));
1051fa9e4066Sahrens 			(void) fprintf(fp, gettext("Valid commands:\n"));
1052fa9e4066Sahrens 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1053fa9e4066Sahrens 			    pt_to_str(PT_NAME), gettext("<name>"));
1054fa9e4066Sahrens 			break;
10550209230bSgjelinek 		case RT_DCPU:
10560209230bSgjelinek 			(void) fprintf(fp, gettext("The '%s' resource scope "
10570209230bSgjelinek 			    "configures the 'pools' facility to dedicate\na "
10580209230bSgjelinek 			    "subset of the system's processors to this zone "
10590209230bSgjelinek 			    "while it is running.\n"),
10600209230bSgjelinek 			    rt_to_str(resource_scope));
10610209230bSgjelinek 			(void) fprintf(fp, gettext("Valid commands:\n"));
10620209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10630209230bSgjelinek 			    pt_to_str(PT_NCPUS),
10640209230bSgjelinek 			    gettext("<unsigned integer | range>"));
10650209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10660209230bSgjelinek 			    pt_to_str(PT_IMPORTANCE),
10670209230bSgjelinek 			    gettext("<unsigned integer>"));
10680209230bSgjelinek 			break;
1069c97ad5cdSakolb 		case RT_PCAP:
1070c97ad5cdSakolb 			(void) fprintf(fp, gettext("The '%s' resource scope is "
1071c97ad5cdSakolb 			    "used to set an upper limit (a cap) on the\n"
1072c97ad5cdSakolb 			    "percentage of CPU that can be used by this zone.  "
1073c97ad5cdSakolb 			    "A '%s' value of 1\ncorresponds to one cpu.  The "
1074c97ad5cdSakolb 			    "value can be set higher than 1, up to the total\n"
1075c97ad5cdSakolb 			    "number of CPUs on the system.  The value can "
1076c97ad5cdSakolb 			    "also be less than 1,\nrepresenting a fraction of "
1077c97ad5cdSakolb 			    "a cpu.\n"),
1078c97ad5cdSakolb 			    rt_to_str(resource_scope), pt_to_str(PT_NCPUS));
1079c97ad5cdSakolb 			(void) fprintf(fp, gettext("Valid commands:\n"));
1080c97ad5cdSakolb 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1081c97ad5cdSakolb 			    pt_to_str(PT_NCPUS), gettext("<unsigned decimal>"));
1082c97ad5cdSakolb 			break;
10830209230bSgjelinek 		case RT_MCAP:
10840209230bSgjelinek 			(void) fprintf(fp, gettext("The '%s' resource scope is "
10850209230bSgjelinek 			    "used to set an upper limit (a cap) on the\n"
10860209230bSgjelinek 			    "amount of physical memory, swap space and locked "
10870209230bSgjelinek 			    "memory that can be used by\nthis zone.\n"),
10880209230bSgjelinek 			    rt_to_str(resource_scope));
10890209230bSgjelinek 			(void) fprintf(fp, gettext("Valid commands:\n"));
10900209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10910209230bSgjelinek 			    pt_to_str(PT_PHYSICAL),
10920209230bSgjelinek 			    gettext("<qualified unsigned decimal>"));
10930209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10940209230bSgjelinek 			    pt_to_str(PT_SWAP),
10950209230bSgjelinek 			    gettext("<qualified unsigned decimal>"));
10960209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10970209230bSgjelinek 			    pt_to_str(PT_LOCKED),
10980209230bSgjelinek 			    gettext("<qualified unsigned decimal>"));
10990209230bSgjelinek 			break;
1100cb8a054bSGlenn Faden 		case RT_ADMIN:
1101cb8a054bSGlenn Faden 			(void) fprintf(fp, gettext("The '%s' resource scope is "
1102cb8a054bSGlenn Faden 			    "used to delegate specific zone management\n"
1103cb8a054bSGlenn Faden 			    "rights to users and roles. These rights are "
1104cb8a054bSGlenn Faden 			    "only applicable to this zone.\n"),
1105cb8a054bSGlenn Faden 			    rt_to_str(resource_scope));
1106cb8a054bSGlenn Faden 			(void) fprintf(fp, gettext("Valid commands:\n"));
1107cb8a054bSGlenn Faden 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1108cb8a054bSGlenn Faden 			    pt_to_str(PT_USER),
1109cb8a054bSGlenn Faden 			    gettext("<single user or role name>"));
1110cb8a054bSGlenn Faden 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1111cb8a054bSGlenn Faden 			    pt_to_str(PT_AUTHS),
1112cb8a054bSGlenn Faden 			    gettext("<comma separated list>"));
1113cb8a054bSGlenn Faden 			break;
11147c478bd9Sstevel@tonic-gate 		}
11157c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("And from any resource scope, you "
11167c478bd9Sstevel@tonic-gate 		    "can:\n"));
11177c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", cmd_to_str(CMD_END),
11187c478bd9Sstevel@tonic-gate 		    gettext("(to conclude this operation)"));
11197c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", cmd_to_str(CMD_CANCEL),
11207c478bd9Sstevel@tonic-gate 		    gettext("(to cancel this operation)"));
11217c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", cmd_to_str(CMD_EXIT),
11227c478bd9Sstevel@tonic-gate 		    gettext("(to exit the zonecfg utility)"));
11237c478bd9Sstevel@tonic-gate 	}
11247c478bd9Sstevel@tonic-gate 	if (flags & HELP_USAGE) {
11257c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s:\t%s %s\n", gettext("usage"),
11267c478bd9Sstevel@tonic-gate 		    execname, cmd_to_str(CMD_HELP));
11277c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s -z <zone>\t\t\t(%s)\n",
11287c478bd9Sstevel@tonic-gate 		    execname, gettext("interactive"));
11297c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s -z <zone> <command>\n", execname);
11307c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s -z <zone> -f <command-file>\n",
11317c478bd9Sstevel@tonic-gate 		    execname);
11327c478bd9Sstevel@tonic-gate 	}
11337c478bd9Sstevel@tonic-gate 	if (flags & HELP_SUBCMDS) {
11347c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s:\n\n", gettext("Commands"));
11357c478bd9Sstevel@tonic-gate 		for (i = 0; i <= CMD_MAX; i++) {
11367c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "%s\n", helptab[i].short_usage);
11377c478bd9Sstevel@tonic-gate 			if (verbose)
11387c478bd9Sstevel@tonic-gate 				(void) fprintf(fp, "\t%s\n\n", long_help(i));
11397c478bd9Sstevel@tonic-gate 		}
11407c478bd9Sstevel@tonic-gate 	}
11417c478bd9Sstevel@tonic-gate 	if (flags & HELP_SYNTAX) {
11427c478bd9Sstevel@tonic-gate 		if (!verbose)
11437c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\n");
11447c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "<zone> := [A-Za-z0-9][A-Za-z0-9_.-]*\n");
11457c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("\t(except the reserved words "
11467c478bd9Sstevel@tonic-gate 		    "'%s' and anything starting with '%s')\n"), "global",
11477c478bd9Sstevel@tonic-gate 		    "SUNW");
11487c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
11497c478bd9Sstevel@tonic-gate 		    gettext("\tName must be less than %d characters.\n"),
11507c478bd9Sstevel@tonic-gate 		    ZONENAME_MAX);
11517c478bd9Sstevel@tonic-gate 		if (verbose)
11527c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\n");
11537c478bd9Sstevel@tonic-gate 	}
11547c478bd9Sstevel@tonic-gate 	if (flags & HELP_NETADDR) {
11557c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("\n<net-addr> :="));
11567c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
11577c478bd9Sstevel@tonic-gate 		    gettext("\t<IPv4-address>[/<IPv4-prefix-length>] |\n"));
11587c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
11597c478bd9Sstevel@tonic-gate 		    gettext("\t\t<IPv6-address>/<IPv6-prefix-length> |\n"));
11607c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
11617c478bd9Sstevel@tonic-gate 		    gettext("\t\t<hostname>[/<IPv4-prefix-length>]\n"));
11627c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("See inet(3SOCKET) for IPv4 and "
11637c478bd9Sstevel@tonic-gate 		    "IPv6 address syntax.\n"));
11647c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("<IPv4-prefix-length> := [0-32]\n"));
11657c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
11667c478bd9Sstevel@tonic-gate 		    gettext("<IPv6-prefix-length> := [0-128]\n"));
11677c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
11687c478bd9Sstevel@tonic-gate 		    gettext("<hostname> := [A-Za-z0-9][A-Za-z0-9-.]*\n"));
11697c478bd9Sstevel@tonic-gate 	}
11707c478bd9Sstevel@tonic-gate 	if (flags & HELP_RESOURCES) {
1171*6e1ae2a3SGary Pennington 		(void) fprintf(fp, "<%s> := %s | %s | %s | %s | %s |\n\t"
1172cb8a054bSGlenn Faden 		    "%s | %s | %s | %s | %s\n\n",
11737c478bd9Sstevel@tonic-gate 		    gettext("resource type"), rt_to_str(RT_FS),
1174*6e1ae2a3SGary Pennington 		    rt_to_str(RT_NET), rt_to_str(RT_DEVICE),
11759e7542f4Sdp 		    rt_to_str(RT_RCTL), rt_to_str(RT_ATTR),
11760209230bSgjelinek 		    rt_to_str(RT_DATASET), rt_to_str(RT_DCPU),
1177cb8a054bSGlenn Faden 		    rt_to_str(RT_PCAP), rt_to_str(RT_MCAP),
1178cb8a054bSGlenn Faden 		    rt_to_str(RT_ADMIN));
11797c478bd9Sstevel@tonic-gate 	}
11807c478bd9Sstevel@tonic-gate 	if (flags & HELP_PROPS) {
11817c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("For resource type ... there are "
11827c478bd9Sstevel@tonic-gate 		    "property types ...:\n"));
11837c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
1184087719fdSdp 		    pt_to_str(PT_ZONENAME));
1185087719fdSdp 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
11867c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_ZONEPATH));
11877c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
11889acbbeafSnn35248 		    pt_to_str(PT_BRAND));
11899acbbeafSnn35248 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
11907c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_AUTOBOOT));
11917c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
11923f2f09c1Sdp 		    pt_to_str(PT_BOOTARGS));
11933f2f09c1Sdp 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
11947c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_POOL));
1195ffbafc53Scomay 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
1196ffbafc53Scomay 		    pt_to_str(PT_LIMITPRIV));
11970209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
11980209230bSgjelinek 		    pt_to_str(PT_SCHED));
11990209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
1200f4b3ec61Sdh155122 		    pt_to_str(PT_IPTYPE));
1201f4b3ec61Sdh155122 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12025679c89fSjv227347 		    pt_to_str(PT_HOSTID));
12035679c89fSjv227347 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12040fbb751dSJohn Levon 		    pt_to_str(PT_FS_ALLOWED));
12050fbb751dSJohn Levon 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12060209230bSgjelinek 		    pt_to_str(PT_MAXLWPS));
12070209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
1208ff19e029SMenno Lageman 		    pt_to_str(PT_MAXPROCS));
1209ff19e029SMenno Lageman 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12100209230bSgjelinek 		    pt_to_str(PT_MAXSHMMEM));
12110209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12120209230bSgjelinek 		    pt_to_str(PT_MAXSHMIDS));
12130209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12140209230bSgjelinek 		    pt_to_str(PT_MAXMSGIDS));
12150209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12160209230bSgjelinek 		    pt_to_str(PT_MAXSEMIDS));
12170209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12180209230bSgjelinek 		    pt_to_str(PT_SHARES));
12196d4d1c0dSbatschul 		(void) fprintf(fp, "\t%s\t\t%s, %s, %s, %s, %s\n",
12206d4d1c0dSbatschul 		    rt_to_str(RT_FS), pt_to_str(PT_DIR),
12216d4d1c0dSbatschul 		    pt_to_str(PT_SPECIAL), pt_to_str(PT_RAW),
12226d4d1c0dSbatschul 		    pt_to_str(PT_TYPE), pt_to_str(PT_OPTIONS));
1223de860bd9Sgfaden 		(void) fprintf(fp, "\t%s\t\t%s, %s, %s\n", rt_to_str(RT_NET),
1224de860bd9Sgfaden 		    pt_to_str(PT_ADDRESS), pt_to_str(PT_PHYSICAL),
1225de860bd9Sgfaden 		    pt_to_str(PT_DEFROUTER));
12267c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t\t%s\n", rt_to_str(RT_DEVICE),
12277c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_MATCH));
12287c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t\t%s, %s\n", rt_to_str(RT_RCTL),
12297c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_NAME), pt_to_str(PT_VALUE));
12307c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t\t%s, %s, %s\n", rt_to_str(RT_ATTR),
12317c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_NAME), pt_to_str(PT_TYPE),
12327c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_VALUE));
1233fa9e4066Sahrens 		(void) fprintf(fp, "\t%s\t\t%s\n", rt_to_str(RT_DATASET),
1234fa9e4066Sahrens 		    pt_to_str(PT_NAME));
12350209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s, %s\n", rt_to_str(RT_DCPU),
12360209230bSgjelinek 		    pt_to_str(PT_NCPUS), pt_to_str(PT_IMPORTANCE));
1237c97ad5cdSakolb 		(void) fprintf(fp, "\t%s\t%s\n", rt_to_str(RT_PCAP),
1238c97ad5cdSakolb 		    pt_to_str(PT_NCPUS));
12390209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s, %s, %s\n", rt_to_str(RT_MCAP),
12400209230bSgjelinek 		    pt_to_str(PT_PHYSICAL), pt_to_str(PT_SWAP),
12410209230bSgjelinek 		    pt_to_str(PT_LOCKED));
1242cb8a054bSGlenn Faden 		(void) fprintf(fp, "\t%s\t\t%s, %s\n", rt_to_str(RT_ADMIN),
1243cb8a054bSGlenn Faden 		    pt_to_str(PT_USER), pt_to_str(PT_AUTHS));
12447c478bd9Sstevel@tonic-gate 	}
12457c478bd9Sstevel@tonic-gate 	if (need_to_close)
12467c478bd9Sstevel@tonic-gate 		(void) pclose(fp);
12477c478bd9Sstevel@tonic-gate }
12487c478bd9Sstevel@tonic-gate 
12497c478bd9Sstevel@tonic-gate static void
1250bbec428eSgjelinek zone_perror(char *prefix, int err, boolean_t set_saw)
12517c478bd9Sstevel@tonic-gate {
12527c478bd9Sstevel@tonic-gate 	zerr("%s: %s", prefix, zonecfg_strerror(err));
12537c478bd9Sstevel@tonic-gate 	if (set_saw)
1254bbec428eSgjelinek 		saw_error = B_TRUE;
12557c478bd9Sstevel@tonic-gate }
12567c478bd9Sstevel@tonic-gate 
12577c478bd9Sstevel@tonic-gate /*
12587c478bd9Sstevel@tonic-gate  * zone_perror() expects a single string, but for remove and select
12597c478bd9Sstevel@tonic-gate  * we have both the command and the resource type, so this wrapper
12607c478bd9Sstevel@tonic-gate  * function serves the same purpose in a slightly different way.
12617c478bd9Sstevel@tonic-gate  */
12627c478bd9Sstevel@tonic-gate 
12637c478bd9Sstevel@tonic-gate static void
1264bbec428eSgjelinek z_cmd_rt_perror(int cmd_num, int res_num, int err, boolean_t set_saw)
12657c478bd9Sstevel@tonic-gate {
12667c478bd9Sstevel@tonic-gate 	zerr("%s %s: %s", cmd_to_str(cmd_num), rt_to_str(res_num),
12677c478bd9Sstevel@tonic-gate 	    zonecfg_strerror(err));
12687c478bd9Sstevel@tonic-gate 	if (set_saw)
1269bbec428eSgjelinek 		saw_error = B_TRUE;
12707c478bd9Sstevel@tonic-gate }
12717c478bd9Sstevel@tonic-gate 
12727c478bd9Sstevel@tonic-gate /* returns Z_OK if successful, Z_foo from <libzonecfg.h> otherwise */
12737c478bd9Sstevel@tonic-gate static int
1274bbec428eSgjelinek initialize(boolean_t handle_expected)
12757c478bd9Sstevel@tonic-gate {
12767c478bd9Sstevel@tonic-gate 	int err;
12779acbbeafSnn35248 	char brandname[MAXNAMELEN];
12787c478bd9Sstevel@tonic-gate 
12797c478bd9Sstevel@tonic-gate 	if (zonecfg_check_handle(handle) != Z_OK) {
12807c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_get_handle(zone, handle)) == Z_OK) {
1281bbec428eSgjelinek 			got_handle = B_TRUE;
12829acbbeafSnn35248 			if (zonecfg_get_brand(handle, brandname,
12839acbbeafSnn35248 			    sizeof (brandname)) != Z_OK) {
12849acbbeafSnn35248 				zerr("Zone %s is inconsistent: missing "
12859acbbeafSnn35248 				    "brand attribute", zone);
12869acbbeafSnn35248 				exit(Z_ERR);
12879acbbeafSnn35248 			}
12889acbbeafSnn35248 			if ((brand = brand_open(brandname)) == NULL) {
12899acbbeafSnn35248 				zerr("Zone %s uses non-existent brand \"%s\"."
12909acbbeafSnn35248 				    "  Unable to continue", zone, brandname);
12919acbbeafSnn35248 				exit(Z_ERR);
12929acbbeafSnn35248 			}
1293cb8a054bSGlenn Faden 			/*
1294cb8a054bSGlenn Faden 			 * If the user_attr file is newer than
1295cb8a054bSGlenn Faden 			 * the zone config file, the admins
1296cb8a054bSGlenn Faden 			 * may need to be updated since the
1297cb8a054bSGlenn Faden 			 * RBAC files are authoritative for
1298cb8a054bSGlenn Faden 			 * authorization checks.
1299cb8a054bSGlenn Faden 			 */
1300cb8a054bSGlenn Faden 			err = zonecfg_update_userauths(handle, zone);
1301cb8a054bSGlenn Faden 			if (err == Z_OK) {
1302cb8a054bSGlenn Faden 				zerr(gettext("The administrative rights "
1303cb8a054bSGlenn Faden 				    "were updated to match "
1304cb8a054bSGlenn Faden 				    "the current RBAC configuration.\n"
1305cb8a054bSGlenn Faden 				    "Use \"info admin\" and \"revert\" to "
1306cb8a054bSGlenn Faden 				    "compare with the previous settings."));
1307cb8a054bSGlenn Faden 				need_to_commit = B_TRUE;
1308cb8a054bSGlenn Faden 			} else if (err != Z_NO_ENTRY) {
1309cb8a054bSGlenn Faden 				zerr(gettext("failed to update "
1310cb8a054bSGlenn Faden 				    "admin  rights."));
1311cb8a054bSGlenn Faden 				exit(Z_ERR);
1312cb8a054bSGlenn Faden 			} else if (need_to_commit) {
1313cb8a054bSGlenn Faden 				zerr(gettext("admin rights were updated "
1314cb8a054bSGlenn Faden 				    "to match RBAC configuration."));
1315cb8a054bSGlenn Faden 			}
1316cb8a054bSGlenn Faden 
13170209230bSgjelinek 		} else if (global_zone && err == Z_NO_ZONE && !got_handle &&
13180209230bSgjelinek 		    !read_only_mode) {
13190209230bSgjelinek 			/*
13200209230bSgjelinek 			 * We implicitly create the global zone config if it
13210209230bSgjelinek 			 * doesn't exist.
13220209230bSgjelinek 			 */
13230209230bSgjelinek 			zone_dochandle_t tmphandle;
13240209230bSgjelinek 
13250209230bSgjelinek 			if ((tmphandle = zonecfg_init_handle()) == NULL) {
1326bbec428eSgjelinek 				zone_perror(execname, Z_NOMEM, B_TRUE);
13270209230bSgjelinek 				exit(Z_ERR);
13280209230bSgjelinek 			}
13290209230bSgjelinek 
13300209230bSgjelinek 			err = zonecfg_get_template_handle("SUNWblank", zone,
13310209230bSgjelinek 			    tmphandle);
13320209230bSgjelinek 
13330209230bSgjelinek 			if (err != Z_OK) {
13340209230bSgjelinek 				zonecfg_fini_handle(tmphandle);
1335bbec428eSgjelinek 				zone_perror("SUNWblank", err, B_TRUE);
13360209230bSgjelinek 				return (err);
13370209230bSgjelinek 			}
13380209230bSgjelinek 
1339bbec428eSgjelinek 			need_to_commit = B_TRUE;
13400209230bSgjelinek 			zonecfg_fini_handle(handle);
13410209230bSgjelinek 			handle = tmphandle;
1342bbec428eSgjelinek 			got_handle = B_TRUE;
13430209230bSgjelinek 
13447c478bd9Sstevel@tonic-gate 		} else {
13457c478bd9Sstevel@tonic-gate 			zone_perror(zone, err, handle_expected || got_handle);
13467c478bd9Sstevel@tonic-gate 			if (err == Z_NO_ZONE && !got_handle &&
13477c478bd9Sstevel@tonic-gate 			    interactive_mode && !read_only_mode)
13487c478bd9Sstevel@tonic-gate 				(void) printf(gettext("Use '%s' to begin "
13497c478bd9Sstevel@tonic-gate 				    "configuring a new zone.\n"),
13507c478bd9Sstevel@tonic-gate 				    cmd_to_str(CMD_CREATE));
13517c478bd9Sstevel@tonic-gate 			return (err);
13527c478bd9Sstevel@tonic-gate 		}
13537c478bd9Sstevel@tonic-gate 	}
13547c478bd9Sstevel@tonic-gate 	return (Z_OK);
13557c478bd9Sstevel@tonic-gate }
13567c478bd9Sstevel@tonic-gate 
1357bbec428eSgjelinek static boolean_t
1358087719fdSdp state_atleast(zone_state_t state)
1359087719fdSdp {
1360087719fdSdp 	zone_state_t state_num;
1361087719fdSdp 	int err;
1362087719fdSdp 
1363087719fdSdp 	if ((err = zone_get_state(zone, &state_num)) != Z_OK) {
1364087719fdSdp 		/* all states are greater than "non-existent" */
1365087719fdSdp 		if (err == Z_NO_ZONE)
1366087719fdSdp 			return (B_FALSE);
1367087719fdSdp 		zerr(gettext("Unexpectedly failed to determine state "
1368087719fdSdp 		    "of zone %s: %s"), zone, zonecfg_strerror(err));
1369087719fdSdp 		exit(Z_ERR);
1370087719fdSdp 	}
1371087719fdSdp 	return (state_num >= state);
1372087719fdSdp }
1373087719fdSdp 
13747c478bd9Sstevel@tonic-gate /*
13757c478bd9Sstevel@tonic-gate  * short_usage() is for bad syntax: getopt() issues, too many arguments, etc.
13767c478bd9Sstevel@tonic-gate  */
13777c478bd9Sstevel@tonic-gate 
13787c478bd9Sstevel@tonic-gate void
13797c478bd9Sstevel@tonic-gate short_usage(int command)
13807c478bd9Sstevel@tonic-gate {
13817c478bd9Sstevel@tonic-gate 	/* lex_lineno has already been incremented in the lexer; compensate */
13827c478bd9Sstevel@tonic-gate 	if (cmd_file_mode) {
13837c478bd9Sstevel@tonic-gate 		if (strcmp(cmd_file_name, "-") == 0)
13847c478bd9Sstevel@tonic-gate 			(void) fprintf(stderr,
13857c478bd9Sstevel@tonic-gate 			    gettext("syntax error on line %d\n"),
13867c478bd9Sstevel@tonic-gate 			    lex_lineno - 1);
13877c478bd9Sstevel@tonic-gate 		else
13887c478bd9Sstevel@tonic-gate 			(void) fprintf(stderr,
13897c478bd9Sstevel@tonic-gate 			    gettext("syntax error on line %d of %s\n"),
13907c478bd9Sstevel@tonic-gate 			    lex_lineno - 1, cmd_file_name);
13917c478bd9Sstevel@tonic-gate 	}
13927c478bd9Sstevel@tonic-gate 	(void) fprintf(stderr, "%s:\n%s\n", gettext("usage"),
13937c478bd9Sstevel@tonic-gate 	    helptab[command].short_usage);
1394bbec428eSgjelinek 	saw_error = B_TRUE;
13957c478bd9Sstevel@tonic-gate }
13967c478bd9Sstevel@tonic-gate 
13977c478bd9Sstevel@tonic-gate /*
13987c478bd9Sstevel@tonic-gate  * long_usage() is for bad semantics: e.g., wrong property type for a given
13997c478bd9Sstevel@tonic-gate  * resource type.  It is also used by longer_usage() below.
14007c478bd9Sstevel@tonic-gate  */
14017c478bd9Sstevel@tonic-gate 
14027c478bd9Sstevel@tonic-gate void
1403bbec428eSgjelinek long_usage(uint_t cmd_num, boolean_t set_saw)
14047c478bd9Sstevel@tonic-gate {
14057c478bd9Sstevel@tonic-gate 	(void) fprintf(set_saw ? stderr : stdout, "%s:\n%s\n", gettext("usage"),
14067c478bd9Sstevel@tonic-gate 	    helptab[cmd_num].short_usage);
14077c478bd9Sstevel@tonic-gate 	(void) fprintf(set_saw ? stderr : stdout, "\t%s\n", long_help(cmd_num));
14087c478bd9Sstevel@tonic-gate 	if (set_saw)
1409bbec428eSgjelinek 		saw_error = B_TRUE;
14107c478bd9Sstevel@tonic-gate }
14117c478bd9Sstevel@tonic-gate 
14127c478bd9Sstevel@tonic-gate /*
14137c478bd9Sstevel@tonic-gate  * longer_usage() is for 'help foo' and 'foo -?': call long_usage() and also
14147c478bd9Sstevel@tonic-gate  * any extra usage() flags as appropriate for whatever command.
14157c478bd9Sstevel@tonic-gate  */
14167c478bd9Sstevel@tonic-gate 
14177c478bd9Sstevel@tonic-gate void
14187c478bd9Sstevel@tonic-gate longer_usage(uint_t cmd_num)
14197c478bd9Sstevel@tonic-gate {
1420bbec428eSgjelinek 	long_usage(cmd_num, B_FALSE);
14217c478bd9Sstevel@tonic-gate 	if (helptab[cmd_num].flags != 0) {
14227c478bd9Sstevel@tonic-gate 		(void) printf("\n");
1423bbec428eSgjelinek 		usage(B_TRUE, helptab[cmd_num].flags);
14247c478bd9Sstevel@tonic-gate 	}
14257c478bd9Sstevel@tonic-gate }
14267c478bd9Sstevel@tonic-gate 
14277c478bd9Sstevel@tonic-gate /*
14287c478bd9Sstevel@tonic-gate  * scope_usage() is simply used when a command is called from the wrong scope.
14297c478bd9Sstevel@tonic-gate  */
14307c478bd9Sstevel@tonic-gate 
14317c478bd9Sstevel@tonic-gate static void
14327c478bd9Sstevel@tonic-gate scope_usage(uint_t cmd_num)
14337c478bd9Sstevel@tonic-gate {
14347c478bd9Sstevel@tonic-gate 	zerr(gettext("The %s command only makes sense in the %s scope."),
14357c478bd9Sstevel@tonic-gate 	    cmd_to_str(cmd_num),
14367c478bd9Sstevel@tonic-gate 	    global_scope ?  gettext("resource") : gettext("global"));
1437bbec428eSgjelinek 	saw_error = B_TRUE;
14387c478bd9Sstevel@tonic-gate }
14397c478bd9Sstevel@tonic-gate 
14407c478bd9Sstevel@tonic-gate /*
1441bbec428eSgjelinek  * On input, B_TRUE => yes, B_FALSE => no.
1442bbec428eSgjelinek  * On return, B_TRUE => 1, B_FALSE => no, could not ask => -1.
14437c478bd9Sstevel@tonic-gate  */
14447c478bd9Sstevel@tonic-gate 
14457c478bd9Sstevel@tonic-gate static int
1446bbec428eSgjelinek ask_yesno(boolean_t default_answer, const char *question)
14477c478bd9Sstevel@tonic-gate {
14487c478bd9Sstevel@tonic-gate 	char line[64];	/* should be enough to answer yes or no */
14497c478bd9Sstevel@tonic-gate 
14507c478bd9Sstevel@tonic-gate 	if (!ok_to_prompt) {
1451bbec428eSgjelinek 		saw_error = B_TRUE;
14527c478bd9Sstevel@tonic-gate 		return (-1);
14537c478bd9Sstevel@tonic-gate 	}
14547c478bd9Sstevel@tonic-gate 	for (;;) {
1455087719fdSdp 		if (printf("%s (%s)? ", question,
1456087719fdSdp 		    default_answer ? "[y]/n" : "y/[n]") < 0)
1457087719fdSdp 			return (-1);
1458087719fdSdp 		if (fgets(line, sizeof (line), stdin) == NULL)
1459087719fdSdp 			return (-1);
1460087719fdSdp 
1461087719fdSdp 		if (line[0] == '\n')
14627c478bd9Sstevel@tonic-gate 			return (default_answer ? 1 : 0);
14637c478bd9Sstevel@tonic-gate 		if (tolower(line[0]) == 'y')
14647c478bd9Sstevel@tonic-gate 			return (1);
14657c478bd9Sstevel@tonic-gate 		if (tolower(line[0]) == 'n')
14667c478bd9Sstevel@tonic-gate 			return (0);
14677c478bd9Sstevel@tonic-gate 	}
14687c478bd9Sstevel@tonic-gate }
14697c478bd9Sstevel@tonic-gate 
14707c478bd9Sstevel@tonic-gate /*
14717c478bd9Sstevel@tonic-gate  * Prints warning if zone already exists.
14727c478bd9Sstevel@tonic-gate  * In interactive mode, prompts if we should continue anyway and returns Z_OK
14737c478bd9Sstevel@tonic-gate  * if so, Z_ERR if not.  In non-interactive mode, exits with Z_ERR.
14747c478bd9Sstevel@tonic-gate  *
14757c478bd9Sstevel@tonic-gate  * Note that if a zone exists and its state is >= INSTALLED, an error message
14767c478bd9Sstevel@tonic-gate  * will be printed and this function will return Z_ERR regardless of mode.
14777c478bd9Sstevel@tonic-gate  */
14787c478bd9Sstevel@tonic-gate 
14797c478bd9Sstevel@tonic-gate static int
1480bbec428eSgjelinek check_if_zone_already_exists(boolean_t force)
14817c478bd9Sstevel@tonic-gate {
14827c478bd9Sstevel@tonic-gate 	char line[ZONENAME_MAX + 128];	/* enough to ask a question */
14837c478bd9Sstevel@tonic-gate 	zone_dochandle_t tmphandle;
14847c478bd9Sstevel@tonic-gate 	int res, answer;
14857c478bd9Sstevel@tonic-gate 
14867c478bd9Sstevel@tonic-gate 	if ((tmphandle = zonecfg_init_handle()) == NULL) {
1487bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
14887c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
14897c478bd9Sstevel@tonic-gate 	}
14907c478bd9Sstevel@tonic-gate 	res = zonecfg_get_handle(zone, tmphandle);
14917c478bd9Sstevel@tonic-gate 	zonecfg_fini_handle(tmphandle);
1492087719fdSdp 	if (res != Z_OK)
14937c478bd9Sstevel@tonic-gate 		return (Z_OK);
1494087719fdSdp 
1495087719fdSdp 	if (state_atleast(ZONE_STATE_INSTALLED)) {
14967c478bd9Sstevel@tonic-gate 		zerr(gettext("Zone %s already installed; %s not allowed."),
14977c478bd9Sstevel@tonic-gate 		    zone, cmd_to_str(CMD_CREATE));
14987c478bd9Sstevel@tonic-gate 		return (Z_ERR);
14997c478bd9Sstevel@tonic-gate 	}
15007c478bd9Sstevel@tonic-gate 
15017c478bd9Sstevel@tonic-gate 	if (force) {
15027c478bd9Sstevel@tonic-gate 		(void) printf(gettext("Zone %s already exists; overwriting.\n"),
15037c478bd9Sstevel@tonic-gate 		    zone);
15047c478bd9Sstevel@tonic-gate 		return (Z_OK);
15057c478bd9Sstevel@tonic-gate 	}
15067c478bd9Sstevel@tonic-gate 	(void) snprintf(line, sizeof (line),
15077c478bd9Sstevel@tonic-gate 	    gettext("Zone %s already exists; %s anyway"), zone,
15087c478bd9Sstevel@tonic-gate 	    cmd_to_str(CMD_CREATE));
1509bbec428eSgjelinek 	if ((answer = ask_yesno(B_FALSE, line)) == -1) {
15107c478bd9Sstevel@tonic-gate 		zerr(gettext("Zone exists, input not from terminal and -F not "
15117c478bd9Sstevel@tonic-gate 		    "specified:\n%s command ignored, exiting."),
15127c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_CREATE));
15137c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
15147c478bd9Sstevel@tonic-gate 	}
15157c478bd9Sstevel@tonic-gate 	return (answer == 1 ? Z_OK : Z_ERR);
15167c478bd9Sstevel@tonic-gate }
15177c478bd9Sstevel@tonic-gate 
1518bbec428eSgjelinek static boolean_t
15197c478bd9Sstevel@tonic-gate zone_is_read_only(int cmd_num)
15207c478bd9Sstevel@tonic-gate {
15217c478bd9Sstevel@tonic-gate 	if (strncmp(zone, "SUNW", 4) == 0) {
15227c478bd9Sstevel@tonic-gate 		zerr(gettext("%s: zones beginning with SUNW are read-only."),
15237c478bd9Sstevel@tonic-gate 		    zone);
1524bbec428eSgjelinek 		saw_error = B_TRUE;
1525bbec428eSgjelinek 		return (B_TRUE);
15267c478bd9Sstevel@tonic-gate 	}
15277c478bd9Sstevel@tonic-gate 	if (read_only_mode) {
15287c478bd9Sstevel@tonic-gate 		zerr(gettext("%s: cannot %s in read-only mode."), zone,
15297c478bd9Sstevel@tonic-gate 		    cmd_to_str(cmd_num));
1530bbec428eSgjelinek 		saw_error = B_TRUE;
1531bbec428eSgjelinek 		return (B_TRUE);
15327c478bd9Sstevel@tonic-gate 	}
1533bbec428eSgjelinek 	return (B_FALSE);
15347c478bd9Sstevel@tonic-gate }
15357c478bd9Sstevel@tonic-gate 
15367c478bd9Sstevel@tonic-gate /*
15377c478bd9Sstevel@tonic-gate  * Create a new configuration.
15387c478bd9Sstevel@tonic-gate  */
15397c478bd9Sstevel@tonic-gate void
15407c478bd9Sstevel@tonic-gate create_func(cmd_t *cmd)
15417c478bd9Sstevel@tonic-gate {
15427c478bd9Sstevel@tonic-gate 	int err, arg;
15437c478bd9Sstevel@tonic-gate 	char zone_template[ZONENAME_MAX];
1544ee519a1fSgjelinek 	char attach_path[MAXPATHLEN];
15457c478bd9Sstevel@tonic-gate 	zone_dochandle_t tmphandle;
1546bbec428eSgjelinek 	boolean_t force = B_FALSE;
1547bbec428eSgjelinek 	boolean_t attach = B_FALSE;
1548bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
15497c478bd9Sstevel@tonic-gate 
15507c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
15517c478bd9Sstevel@tonic-gate 
15527c478bd9Sstevel@tonic-gate 	/* This is the default if no arguments are given. */
15537c478bd9Sstevel@tonic-gate 	(void) strlcpy(zone_template, "SUNWdefault", sizeof (zone_template));
15547c478bd9Sstevel@tonic-gate 
15557c478bd9Sstevel@tonic-gate 	optind = 0;
15569acbbeafSnn35248 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?a:bFt:"))
15579acbbeafSnn35248 	    != EOF) {
15587c478bd9Sstevel@tonic-gate 		switch (arg) {
15597c478bd9Sstevel@tonic-gate 		case '?':
15607c478bd9Sstevel@tonic-gate 			if (optopt == '?')
15617c478bd9Sstevel@tonic-gate 				longer_usage(CMD_CREATE);
15627c478bd9Sstevel@tonic-gate 			else
15637c478bd9Sstevel@tonic-gate 				short_usage(CMD_CREATE);
1564bbec428eSgjelinek 			arg_err = B_TRUE;
15657ec75eb8Sgjelinek 			break;
1566ee519a1fSgjelinek 		case 'a':
1567ee519a1fSgjelinek 			(void) strlcpy(attach_path, optarg,
1568ee519a1fSgjelinek 			    sizeof (attach_path));
1569bbec428eSgjelinek 			attach = B_TRUE;
1570ee519a1fSgjelinek 			break;
15717c478bd9Sstevel@tonic-gate 		case 'b':
15727c478bd9Sstevel@tonic-gate 			(void) strlcpy(zone_template, "SUNWblank",
15737c478bd9Sstevel@tonic-gate 			    sizeof (zone_template));
15747c478bd9Sstevel@tonic-gate 			break;
15757c478bd9Sstevel@tonic-gate 		case 'F':
1576bbec428eSgjelinek 			force = B_TRUE;
15777c478bd9Sstevel@tonic-gate 			break;
15787c478bd9Sstevel@tonic-gate 		case 't':
15797c478bd9Sstevel@tonic-gate 			(void) strlcpy(zone_template, optarg,
15807c478bd9Sstevel@tonic-gate 			    sizeof (zone_template));
15817c478bd9Sstevel@tonic-gate 			break;
15827c478bd9Sstevel@tonic-gate 		default:
15837c478bd9Sstevel@tonic-gate 			short_usage(CMD_CREATE);
1584bbec428eSgjelinek 			arg_err = B_TRUE;
15857ec75eb8Sgjelinek 			break;
15867ec75eb8Sgjelinek 		}
15877ec75eb8Sgjelinek 	}
15887ec75eb8Sgjelinek 	if (arg_err)
15897c478bd9Sstevel@tonic-gate 		return;
15907ec75eb8Sgjelinek 
15917c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
15927c478bd9Sstevel@tonic-gate 		short_usage(CMD_CREATE);
15937c478bd9Sstevel@tonic-gate 		return;
15947c478bd9Sstevel@tonic-gate 	}
15957c478bd9Sstevel@tonic-gate 
15967c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_CREATE))
15977c478bd9Sstevel@tonic-gate 		return;
15987c478bd9Sstevel@tonic-gate 
15997c478bd9Sstevel@tonic-gate 	if (check_if_zone_already_exists(force) != Z_OK)
16007c478bd9Sstevel@tonic-gate 		return;
16017c478bd9Sstevel@tonic-gate 
16027c478bd9Sstevel@tonic-gate 	/*
16037c478bd9Sstevel@tonic-gate 	 * Get a temporary handle first.  If that fails, the old handle
16047c478bd9Sstevel@tonic-gate 	 * will not be lost.  Then finish whichever one we don't need,
16057c478bd9Sstevel@tonic-gate 	 * to avoid leaks.  Then get the handle for zone_template, and
16067c478bd9Sstevel@tonic-gate 	 * set the name to zone: this "copy, rename" method is how
16077c478bd9Sstevel@tonic-gate 	 * create -[b|t] works.
16087c478bd9Sstevel@tonic-gate 	 */
16097c478bd9Sstevel@tonic-gate 	if ((tmphandle = zonecfg_init_handle()) == NULL) {
1610bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
16117c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
16127c478bd9Sstevel@tonic-gate 	}
1613ee519a1fSgjelinek 
1614ee519a1fSgjelinek 	if (attach)
161516ab8c7bSgjelinek 		err = zonecfg_get_attach_handle(attach_path, ZONE_DETACHED,
161616ab8c7bSgjelinek 		    zone, B_FALSE, tmphandle);
1617ee519a1fSgjelinek 	else
1618ee519a1fSgjelinek 		err = zonecfg_get_template_handle(zone_template, zone,
1619ee519a1fSgjelinek 		    tmphandle);
1620ee519a1fSgjelinek 
1621ee519a1fSgjelinek 	if (err != Z_OK) {
16227c478bd9Sstevel@tonic-gate 		zonecfg_fini_handle(tmphandle);
1623ee519a1fSgjelinek 		if (attach && err == Z_NO_ZONE)
1624ee519a1fSgjelinek 			(void) fprintf(stderr, gettext("invalid path to "
1625ee519a1fSgjelinek 			    "detached zone\n"));
1626ee519a1fSgjelinek 		else if (attach && err == Z_INVALID_DOCUMENT)
1627ee519a1fSgjelinek 			(void) fprintf(stderr, gettext("Cannot attach to an "
1628ee519a1fSgjelinek 			    "earlier release of the operating system\n"));
1629ee519a1fSgjelinek 		else
1630bbec428eSgjelinek 			zone_perror(zone_template, err, B_TRUE);
16317c478bd9Sstevel@tonic-gate 		return;
16327c478bd9Sstevel@tonic-gate 	}
1633087719fdSdp 
1634bbec428eSgjelinek 	need_to_commit = B_TRUE;
16357c478bd9Sstevel@tonic-gate 	zonecfg_fini_handle(handle);
16367c478bd9Sstevel@tonic-gate 	handle = tmphandle;
1637bbec428eSgjelinek 	got_handle = B_TRUE;
16387c478bd9Sstevel@tonic-gate }
16397c478bd9Sstevel@tonic-gate 
16407c478bd9Sstevel@tonic-gate /*
16417c478bd9Sstevel@tonic-gate  * This malloc()'s memory, which must be freed by the caller.
16427c478bd9Sstevel@tonic-gate  */
16437c478bd9Sstevel@tonic-gate static char *
16447c478bd9Sstevel@tonic-gate quoteit(char *instr)
16457c478bd9Sstevel@tonic-gate {
16467c478bd9Sstevel@tonic-gate 	char *outstr;
16477c478bd9Sstevel@tonic-gate 	size_t outstrsize = strlen(instr) + 3;	/* 2 quotes + '\0' */
16487c478bd9Sstevel@tonic-gate 
16497c478bd9Sstevel@tonic-gate 	if ((outstr = malloc(outstrsize)) == NULL) {
1650bbec428eSgjelinek 		zone_perror(zone, Z_NOMEM, B_FALSE);
16517c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
16527c478bd9Sstevel@tonic-gate 	}
16537c478bd9Sstevel@tonic-gate 	if (strchr(instr, ' ') == NULL) {
16547c478bd9Sstevel@tonic-gate 		(void) strlcpy(outstr, instr, outstrsize);
16557c478bd9Sstevel@tonic-gate 		return (outstr);
16567c478bd9Sstevel@tonic-gate 	}
16577c478bd9Sstevel@tonic-gate 	(void) snprintf(outstr, outstrsize, "\"%s\"", instr);
16587c478bd9Sstevel@tonic-gate 	return (outstr);
16597c478bd9Sstevel@tonic-gate }
16607c478bd9Sstevel@tonic-gate 
16617c478bd9Sstevel@tonic-gate static void
16627c478bd9Sstevel@tonic-gate export_prop(FILE *of, int prop_num, char *prop_id)
16637c478bd9Sstevel@tonic-gate {
16647c478bd9Sstevel@tonic-gate 	char *quote_str;
16657c478bd9Sstevel@tonic-gate 
16667c478bd9Sstevel@tonic-gate 	if (strlen(prop_id) == 0)
16677c478bd9Sstevel@tonic-gate 		return;
16687c478bd9Sstevel@tonic-gate 	quote_str = quoteit(prop_id);
16697c478bd9Sstevel@tonic-gate 	(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
16707c478bd9Sstevel@tonic-gate 	    pt_to_str(prop_num), quote_str);
16717c478bd9Sstevel@tonic-gate 	free(quote_str);
16727c478bd9Sstevel@tonic-gate }
16737c478bd9Sstevel@tonic-gate 
16747c478bd9Sstevel@tonic-gate void
16757c478bd9Sstevel@tonic-gate export_func(cmd_t *cmd)
16767c478bd9Sstevel@tonic-gate {
16777c478bd9Sstevel@tonic-gate 	struct zone_nwiftab nwiftab;
16787c478bd9Sstevel@tonic-gate 	struct zone_fstab fstab;
16797c478bd9Sstevel@tonic-gate 	struct zone_devtab devtab;
16807c478bd9Sstevel@tonic-gate 	struct zone_attrtab attrtab;
16817c478bd9Sstevel@tonic-gate 	struct zone_rctltab rctltab;
1682fa9e4066Sahrens 	struct zone_dstab dstab;
16830209230bSgjelinek 	struct zone_psettab psettab;
16840209230bSgjelinek 	struct zone_mcaptab mcaptab;
16857c478bd9Sstevel@tonic-gate 	struct zone_rctlvaltab *valptr;
1686cb8a054bSGlenn Faden 	struct zone_admintab admintab;
16877c478bd9Sstevel@tonic-gate 	int err, arg;
16887c478bd9Sstevel@tonic-gate 	char zonepath[MAXPATHLEN], outfile[MAXPATHLEN], pool[MAXNAMELEN];
16893f2f09c1Sdp 	char bootargs[BOOTARGS_MAX];
16900209230bSgjelinek 	char sched[MAXNAMELEN];
16919acbbeafSnn35248 	char brand[MAXNAMELEN];
16925679c89fSjv227347 	char hostidp[HW_HOSTID_LEN];
16930fbb751dSJohn Levon 	char fsallowedp[ZONE_FS_ALLOWED_MAX];
1694ffbafc53Scomay 	char *limitpriv;
16957c478bd9Sstevel@tonic-gate 	FILE *of;
16967c478bd9Sstevel@tonic-gate 	boolean_t autoboot;
1697f4b3ec61Sdh155122 	zone_iptype_t iptype;
1698bbec428eSgjelinek 	boolean_t need_to_close = B_FALSE;
1699bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
17007c478bd9Sstevel@tonic-gate 
17017c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
17027c478bd9Sstevel@tonic-gate 
17037c478bd9Sstevel@tonic-gate 	outfile[0] = '\0';
17047c478bd9Sstevel@tonic-gate 	optind = 0;
17057c478bd9Sstevel@tonic-gate 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?f:")) != EOF) {
17067c478bd9Sstevel@tonic-gate 		switch (arg) {
17077c478bd9Sstevel@tonic-gate 		case '?':
17087c478bd9Sstevel@tonic-gate 			if (optopt == '?')
17097c478bd9Sstevel@tonic-gate 				longer_usage(CMD_EXPORT);
17107c478bd9Sstevel@tonic-gate 			else
17117c478bd9Sstevel@tonic-gate 				short_usage(CMD_EXPORT);
1712bbec428eSgjelinek 			arg_err = B_TRUE;
17137ec75eb8Sgjelinek 			break;
17147c478bd9Sstevel@tonic-gate 		case 'f':
17157c478bd9Sstevel@tonic-gate 			(void) strlcpy(outfile, optarg, sizeof (outfile));
17167c478bd9Sstevel@tonic-gate 			break;
17177c478bd9Sstevel@tonic-gate 		default:
17187c478bd9Sstevel@tonic-gate 			short_usage(CMD_EXPORT);
1719bbec428eSgjelinek 			arg_err = B_TRUE;
17207ec75eb8Sgjelinek 			break;
17217ec75eb8Sgjelinek 		}
17227ec75eb8Sgjelinek 	}
17237ec75eb8Sgjelinek 	if (arg_err)
17247c478bd9Sstevel@tonic-gate 		return;
17257ec75eb8Sgjelinek 
17267c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
17277c478bd9Sstevel@tonic-gate 		short_usage(CMD_EXPORT);
17287c478bd9Sstevel@tonic-gate 		return;
17297c478bd9Sstevel@tonic-gate 	}
17307c478bd9Sstevel@tonic-gate 	if (strlen(outfile) == 0) {
17317c478bd9Sstevel@tonic-gate 		of = stdout;
17327c478bd9Sstevel@tonic-gate 	} else {
17337c478bd9Sstevel@tonic-gate 		if ((of = fopen(outfile, "w")) == NULL) {
17347c478bd9Sstevel@tonic-gate 			zerr(gettext("opening file %s: %s"),
17357c478bd9Sstevel@tonic-gate 			    outfile, strerror(errno));
17367c478bd9Sstevel@tonic-gate 			goto done;
17377c478bd9Sstevel@tonic-gate 		}
17387c478bd9Sstevel@tonic-gate 		setbuf(of, NULL);
1739bbec428eSgjelinek 		need_to_close = B_TRUE;
17407c478bd9Sstevel@tonic-gate 	}
17417c478bd9Sstevel@tonic-gate 
1742bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
17437c478bd9Sstevel@tonic-gate 		goto done;
17447c478bd9Sstevel@tonic-gate 
17457c478bd9Sstevel@tonic-gate 	(void) fprintf(of, "%s -b\n", cmd_to_str(CMD_CREATE));
17467c478bd9Sstevel@tonic-gate 
17477c478bd9Sstevel@tonic-gate 	if (zonecfg_get_zonepath(handle, zonepath, sizeof (zonepath)) == Z_OK &&
17487c478bd9Sstevel@tonic-gate 	    strlen(zonepath) > 0)
17497c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
17507c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_ZONEPATH), zonepath);
17517c478bd9Sstevel@tonic-gate 
17529acbbeafSnn35248 	if ((zone_get_brand(zone, brand, sizeof (brand)) == Z_OK) &&
17539acbbeafSnn35248 	    (strcmp(brand, NATIVE_BRAND_NAME) != 0))
17549acbbeafSnn35248 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
17559acbbeafSnn35248 		    pt_to_str(PT_BRAND), brand);
17569acbbeafSnn35248 
17577c478bd9Sstevel@tonic-gate 	if (zonecfg_get_autoboot(handle, &autoboot) == Z_OK)
17587c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
17597c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_AUTOBOOT), autoboot ? "true" : "false");
17607c478bd9Sstevel@tonic-gate 
17613f2f09c1Sdp 	if (zonecfg_get_bootargs(handle, bootargs, sizeof (bootargs)) == Z_OK &&
17623f2f09c1Sdp 	    strlen(bootargs) > 0) {
17633f2f09c1Sdp 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
17643f2f09c1Sdp 		    pt_to_str(PT_BOOTARGS), bootargs);
17653f2f09c1Sdp 	}
17663f2f09c1Sdp 
17677c478bd9Sstevel@tonic-gate 	if (zonecfg_get_pool(handle, pool, sizeof (pool)) == Z_OK &&
17687c478bd9Sstevel@tonic-gate 	    strlen(pool) > 0)
17697c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
17707c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_POOL), pool);
17717c478bd9Sstevel@tonic-gate 
1772ffbafc53Scomay 	if (zonecfg_get_limitpriv(handle, &limitpriv) == Z_OK &&
1773ffbafc53Scomay 	    strlen(limitpriv) > 0) {
1774ffbafc53Scomay 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
1775ffbafc53Scomay 		    pt_to_str(PT_LIMITPRIV), limitpriv);
1776ffbafc53Scomay 		free(limitpriv);
1777ffbafc53Scomay 	}
1778ffbafc53Scomay 
17790209230bSgjelinek 	if (zonecfg_get_sched_class(handle, sched, sizeof (sched)) == Z_OK &&
17800209230bSgjelinek 	    strlen(sched) > 0)
17810209230bSgjelinek 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
17820209230bSgjelinek 		    pt_to_str(PT_SCHED), sched);
17833f2f09c1Sdp 
1784f4b3ec61Sdh155122 	if (zonecfg_get_iptype(handle, &iptype) == Z_OK) {
1785f4b3ec61Sdh155122 		switch (iptype) {
1786f4b3ec61Sdh155122 		case ZS_SHARED:
1787f4b3ec61Sdh155122 			(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
1788f4b3ec61Sdh155122 			    pt_to_str(PT_IPTYPE), "shared");
1789f4b3ec61Sdh155122 			break;
1790f4b3ec61Sdh155122 		case ZS_EXCLUSIVE:
1791f4b3ec61Sdh155122 			(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
1792f4b3ec61Sdh155122 			    pt_to_str(PT_IPTYPE), "exclusive");
1793f4b3ec61Sdh155122 			break;
1794f4b3ec61Sdh155122 		}
1795f4b3ec61Sdh155122 	}
1796f4b3ec61Sdh155122 
17975679c89fSjv227347 	if (zonecfg_get_hostid(handle, hostidp, sizeof (hostidp)) == Z_OK) {
17985679c89fSjv227347 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
17995679c89fSjv227347 		    pt_to_str(PT_HOSTID), hostidp);
18005679c89fSjv227347 	}
18015679c89fSjv227347 
18020fbb751dSJohn Levon 	if (zonecfg_get_fs_allowed(handle, fsallowedp,
18030fbb751dSJohn Levon 	    sizeof (fsallowedp)) == Z_OK) {
18040fbb751dSJohn Levon 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
18050fbb751dSJohn Levon 		    pt_to_str(PT_FS_ALLOWED), fsallowedp);
18060fbb751dSJohn Levon 	}
18070fbb751dSJohn Levon 
18087c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setfsent(handle)) != Z_OK) {
1809bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
18107c478bd9Sstevel@tonic-gate 		goto done;
18117c478bd9Sstevel@tonic-gate 	}
18127c478bd9Sstevel@tonic-gate 	while (zonecfg_getfsent(handle, &fstab) == Z_OK) {
18137c478bd9Sstevel@tonic-gate 		zone_fsopt_t *optptr;
18147c478bd9Sstevel@tonic-gate 
18157c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
18167c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_FS));
18177c478bd9Sstevel@tonic-gate 		export_prop(of, PT_DIR, fstab.zone_fs_dir);
18187c478bd9Sstevel@tonic-gate 		export_prop(of, PT_SPECIAL, fstab.zone_fs_special);
18197c478bd9Sstevel@tonic-gate 		export_prop(of, PT_RAW, fstab.zone_fs_raw);
18207c478bd9Sstevel@tonic-gate 		export_prop(of, PT_TYPE, fstab.zone_fs_type);
18217c478bd9Sstevel@tonic-gate 		for (optptr = fstab.zone_fs_options; optptr != NULL;
18227c478bd9Sstevel@tonic-gate 		    optptr = optptr->zone_fsopt_next) {
18237c478bd9Sstevel@tonic-gate 			/*
18247c478bd9Sstevel@tonic-gate 			 * Simple property values with embedded equal signs
18257c478bd9Sstevel@tonic-gate 			 * need to be quoted to prevent the lexer from
18267c478bd9Sstevel@tonic-gate 			 * mis-parsing them as complex name=value pairs.
18277c478bd9Sstevel@tonic-gate 			 */
18287c478bd9Sstevel@tonic-gate 			if (strchr(optptr->zone_fsopt_opt, '='))
18297c478bd9Sstevel@tonic-gate 				(void) fprintf(of, "%s %s \"%s\"\n",
18307c478bd9Sstevel@tonic-gate 				    cmd_to_str(CMD_ADD),
18317c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_OPTIONS),
18327c478bd9Sstevel@tonic-gate 				    optptr->zone_fsopt_opt);
18337c478bd9Sstevel@tonic-gate 			else
18347c478bd9Sstevel@tonic-gate 				(void) fprintf(of, "%s %s %s\n",
18357c478bd9Sstevel@tonic-gate 				    cmd_to_str(CMD_ADD),
18367c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_OPTIONS),
18377c478bd9Sstevel@tonic-gate 				    optptr->zone_fsopt_opt);
18387c478bd9Sstevel@tonic-gate 		}
18397c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
18407c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(fstab.zone_fs_options);
18417c478bd9Sstevel@tonic-gate 	}
18427c478bd9Sstevel@tonic-gate 	(void) zonecfg_endfsent(handle);
18437c478bd9Sstevel@tonic-gate 
18447c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setnwifent(handle)) != Z_OK) {
1845bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
18467c478bd9Sstevel@tonic-gate 		goto done;
18477c478bd9Sstevel@tonic-gate 	}
18487c478bd9Sstevel@tonic-gate 	while (zonecfg_getnwifent(handle, &nwiftab) == Z_OK) {
18497c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
18507c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_NET));
18517c478bd9Sstevel@tonic-gate 		export_prop(of, PT_ADDRESS, nwiftab.zone_nwif_address);
18527c478bd9Sstevel@tonic-gate 		export_prop(of, PT_PHYSICAL, nwiftab.zone_nwif_physical);
1853de860bd9Sgfaden 		export_prop(of, PT_DEFROUTER, nwiftab.zone_nwif_defrouter);
18547c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
18557c478bd9Sstevel@tonic-gate 	}
18567c478bd9Sstevel@tonic-gate 	(void) zonecfg_endnwifent(handle);
18577c478bd9Sstevel@tonic-gate 
18587c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setdevent(handle)) != Z_OK) {
1859bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
18607c478bd9Sstevel@tonic-gate 		goto done;
18617c478bd9Sstevel@tonic-gate 	}
18627c478bd9Sstevel@tonic-gate 	while (zonecfg_getdevent(handle, &devtab) == Z_OK) {
18637c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
18647c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_DEVICE));
18657c478bd9Sstevel@tonic-gate 		export_prop(of, PT_MATCH, devtab.zone_dev_match);
18667c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
18677c478bd9Sstevel@tonic-gate 	}
18687c478bd9Sstevel@tonic-gate 	(void) zonecfg_enddevent(handle);
18697c478bd9Sstevel@tonic-gate 
18701b3281c0SGerald Jelinek 	if (zonecfg_getmcapent(handle, &mcaptab) == Z_OK) {
18711b3281c0SGerald Jelinek 		char buf[128];
18721b3281c0SGerald Jelinek 
18731b3281c0SGerald Jelinek 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
18741b3281c0SGerald Jelinek 		    rt_to_str(RT_MCAP));
18751b3281c0SGerald Jelinek 		bytes_to_units(mcaptab.zone_physmem_cap, buf, sizeof (buf));
18761b3281c0SGerald Jelinek 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
18771b3281c0SGerald Jelinek 		    pt_to_str(PT_PHYSICAL), buf);
18781b3281c0SGerald Jelinek 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
18791b3281c0SGerald Jelinek 	}
18801b3281c0SGerald Jelinek 
18817c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setrctlent(handle)) != Z_OK) {
1882bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
18837c478bd9Sstevel@tonic-gate 		goto done;
18847c478bd9Sstevel@tonic-gate 	}
18857c478bd9Sstevel@tonic-gate 	while (zonecfg_getrctlent(handle, &rctltab) == Z_OK) {
18867c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s rctl\n", cmd_to_str(CMD_ADD));
18877c478bd9Sstevel@tonic-gate 		export_prop(of, PT_NAME, rctltab.zone_rctl_name);
18887c478bd9Sstevel@tonic-gate 		for (valptr = rctltab.zone_rctl_valptr; valptr != NULL;
18897c478bd9Sstevel@tonic-gate 		    valptr = valptr->zone_rctlval_next) {
18907c478bd9Sstevel@tonic-gate 			fprintf(of, "%s %s (%s=%s,%s=%s,%s=%s)\n",
18917c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_ADD), pt_to_str(PT_VALUE),
18927c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_PRIV), valptr->zone_rctlval_priv,
18937c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_LIMIT), valptr->zone_rctlval_limit,
18947c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_ACTION), valptr->zone_rctlval_action);
18957c478bd9Sstevel@tonic-gate 		}
18967c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
18977c478bd9Sstevel@tonic-gate 		zonecfg_free_rctl_value_list(rctltab.zone_rctl_valptr);
18987c478bd9Sstevel@tonic-gate 	}
18997c478bd9Sstevel@tonic-gate 	(void) zonecfg_endrctlent(handle);
19007c478bd9Sstevel@tonic-gate 
19017c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setattrent(handle)) != Z_OK) {
1902bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
19037c478bd9Sstevel@tonic-gate 		goto done;
19047c478bd9Sstevel@tonic-gate 	}
19057c478bd9Sstevel@tonic-gate 	while (zonecfg_getattrent(handle, &attrtab) == Z_OK) {
19067c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
19077c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR));
19087c478bd9Sstevel@tonic-gate 		export_prop(of, PT_NAME, attrtab.zone_attr_name);
19097c478bd9Sstevel@tonic-gate 		export_prop(of, PT_TYPE, attrtab.zone_attr_type);
19107c478bd9Sstevel@tonic-gate 		export_prop(of, PT_VALUE, attrtab.zone_attr_value);
19117c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
19127c478bd9Sstevel@tonic-gate 	}
19137c478bd9Sstevel@tonic-gate 	(void) zonecfg_endattrent(handle);
19147c478bd9Sstevel@tonic-gate 
1915fa9e4066Sahrens 	if ((err = zonecfg_setdsent(handle)) != Z_OK) {
1916bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
1917fa9e4066Sahrens 		goto done;
1918fa9e4066Sahrens 	}
1919fa9e4066Sahrens 	while (zonecfg_getdsent(handle, &dstab) == Z_OK) {
1920fa9e4066Sahrens 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
1921fa9e4066Sahrens 		    rt_to_str(RT_DATASET));
1922fa9e4066Sahrens 		export_prop(of, PT_NAME, dstab.zone_dataset_name);
1923fa9e4066Sahrens 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
1924fa9e4066Sahrens 	}
1925fa9e4066Sahrens 	(void) zonecfg_enddsent(handle);
1926fa9e4066Sahrens 
19270209230bSgjelinek 	if (zonecfg_getpsetent(handle, &psettab) == Z_OK) {
19280209230bSgjelinek 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
19290209230bSgjelinek 		    rt_to_str(RT_DCPU));
19300209230bSgjelinek 		if (strcmp(psettab.zone_ncpu_min, psettab.zone_ncpu_max) == 0)
19310209230bSgjelinek 			(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19320209230bSgjelinek 			    pt_to_str(PT_NCPUS), psettab.zone_ncpu_max);
19330209230bSgjelinek 		else
19340209230bSgjelinek 			(void) fprintf(of, "%s %s=%s-%s\n", cmd_to_str(CMD_SET),
19350209230bSgjelinek 			    pt_to_str(PT_NCPUS), psettab.zone_ncpu_min,
19360209230bSgjelinek 			    psettab.zone_ncpu_max);
19370209230bSgjelinek 		if (psettab.zone_importance[0] != '\0')
19380209230bSgjelinek 			(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19390209230bSgjelinek 			    pt_to_str(PT_IMPORTANCE), psettab.zone_importance);
19400209230bSgjelinek 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
19410209230bSgjelinek 	}
19420209230bSgjelinek 
1943cb8a054bSGlenn Faden 	if ((err = zonecfg_setadminent(handle)) != Z_OK) {
1944cb8a054bSGlenn Faden 		zone_perror(zone, err, B_FALSE);
1945cb8a054bSGlenn Faden 		goto done;
1946cb8a054bSGlenn Faden 	}
1947cb8a054bSGlenn Faden 	while (zonecfg_getadminent(handle, &admintab) == Z_OK) {
1948cb8a054bSGlenn Faden 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
1949cb8a054bSGlenn Faden 		    rt_to_str(RT_ADMIN));
1950cb8a054bSGlenn Faden 		export_prop(of, PT_USER, admintab.zone_admin_user);
1951cb8a054bSGlenn Faden 		export_prop(of, PT_AUTHS, admintab.zone_admin_auths);
1952cb8a054bSGlenn Faden 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
1953cb8a054bSGlenn Faden 	}
1954cb8a054bSGlenn Faden 	(void) zonecfg_endadminent(handle);
1955cb8a054bSGlenn Faden 
1956c97ad5cdSakolb 	/*
1957c97ad5cdSakolb 	 * There is nothing to export for pcap since this resource is just
1958c97ad5cdSakolb 	 * a container for an rctl alias.
1959c97ad5cdSakolb 	 */
1960c97ad5cdSakolb 
19617c478bd9Sstevel@tonic-gate done:
19627c478bd9Sstevel@tonic-gate 	if (need_to_close)
19637c478bd9Sstevel@tonic-gate 		(void) fclose(of);
19647c478bd9Sstevel@tonic-gate }
19657c478bd9Sstevel@tonic-gate 
19667c478bd9Sstevel@tonic-gate void
19677c478bd9Sstevel@tonic-gate exit_func(cmd_t *cmd)
19687c478bd9Sstevel@tonic-gate {
19697c478bd9Sstevel@tonic-gate 	int arg, answer;
1970bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
19717c478bd9Sstevel@tonic-gate 
19727c478bd9Sstevel@tonic-gate 	optind = 0;
19737c478bd9Sstevel@tonic-gate 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?F")) != EOF) {
19747c478bd9Sstevel@tonic-gate 		switch (arg) {
19757c478bd9Sstevel@tonic-gate 		case '?':
19767c478bd9Sstevel@tonic-gate 			longer_usage(CMD_EXIT);
1977bbec428eSgjelinek 			arg_err = B_TRUE;
19787ec75eb8Sgjelinek 			break;
19797c478bd9Sstevel@tonic-gate 		case 'F':
1980bbec428eSgjelinek 			force_exit = B_TRUE;
19817c478bd9Sstevel@tonic-gate 			break;
19827c478bd9Sstevel@tonic-gate 		default:
19837c478bd9Sstevel@tonic-gate 			short_usage(CMD_EXIT);
1984bbec428eSgjelinek 			arg_err = B_TRUE;
19857ec75eb8Sgjelinek 			break;
19867ec75eb8Sgjelinek 		}
19877ec75eb8Sgjelinek 	}
19887ec75eb8Sgjelinek 	if (arg_err)
19897c478bd9Sstevel@tonic-gate 		return;
19907ec75eb8Sgjelinek 
19917c478bd9Sstevel@tonic-gate 	if (optind < cmd->cmd_argc) {
19927c478bd9Sstevel@tonic-gate 		short_usage(CMD_EXIT);
19937c478bd9Sstevel@tonic-gate 		return;
19947c478bd9Sstevel@tonic-gate 	}
19957c478bd9Sstevel@tonic-gate 
19967c478bd9Sstevel@tonic-gate 	if (global_scope || force_exit) {
1997bbec428eSgjelinek 		time_to_exit = B_TRUE;
19987c478bd9Sstevel@tonic-gate 		return;
19997c478bd9Sstevel@tonic-gate 	}
20007c478bd9Sstevel@tonic-gate 
2001bbec428eSgjelinek 	answer = ask_yesno(B_FALSE, "Resource incomplete; really quit");
20027c478bd9Sstevel@tonic-gate 	if (answer == -1) {
20037c478bd9Sstevel@tonic-gate 		zerr(gettext("Resource incomplete, input "
20047c478bd9Sstevel@tonic-gate 		    "not from terminal and -F not specified:\n%s command "
20057c478bd9Sstevel@tonic-gate 		    "ignored, but exiting anyway."), cmd_to_str(CMD_EXIT));
20067c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
20077c478bd9Sstevel@tonic-gate 	} else if (answer == 1) {
2008bbec428eSgjelinek 		time_to_exit = B_TRUE;
20097c478bd9Sstevel@tonic-gate 	}
20107c478bd9Sstevel@tonic-gate 	/* (answer == 0) => just return */
20117c478bd9Sstevel@tonic-gate }
20127c478bd9Sstevel@tonic-gate 
20137c478bd9Sstevel@tonic-gate static int
20147c478bd9Sstevel@tonic-gate validate_zonepath_syntax(char *path)
20157c478bd9Sstevel@tonic-gate {
20167c478bd9Sstevel@tonic-gate 	if (path[0] != '/') {
20177c478bd9Sstevel@tonic-gate 		zerr(gettext("%s is not an absolute path."), path);
20187c478bd9Sstevel@tonic-gate 		return (Z_ERR);
20197c478bd9Sstevel@tonic-gate 	}
20206fb06a2bSSusan Kamm-Worrell 	/* If path is all slashes, then fail */
20216fb06a2bSSusan Kamm-Worrell 	if (strspn(path, "/") == strlen(path)) {
20227c478bd9Sstevel@tonic-gate 		zerr(gettext("/ is not allowed as a %s."),
20237c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_ZONEPATH));
20247c478bd9Sstevel@tonic-gate 		return (Z_ERR);
20257c478bd9Sstevel@tonic-gate 	}
20267c478bd9Sstevel@tonic-gate 	return (Z_OK);
20277c478bd9Sstevel@tonic-gate }
20287c478bd9Sstevel@tonic-gate 
20297c478bd9Sstevel@tonic-gate static void
20307c478bd9Sstevel@tonic-gate add_resource(cmd_t *cmd)
20317c478bd9Sstevel@tonic-gate {
20327c478bd9Sstevel@tonic-gate 	int type;
20330209230bSgjelinek 	struct zone_psettab tmp_psettab;
20340209230bSgjelinek 	struct zone_mcaptab tmp_mcaptab;
2035c97ad5cdSakolb 	uint64_t tmp;
20360209230bSgjelinek 	uint64_t tmp_mcap;
20370209230bSgjelinek 	char pool[MAXNAMELEN];
20387c478bd9Sstevel@tonic-gate 
20397c478bd9Sstevel@tonic-gate 	if ((type = cmd->cmd_res_type) == RT_UNKNOWN) {
2040bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
20417c478bd9Sstevel@tonic-gate 		goto bad;
20427c478bd9Sstevel@tonic-gate 	}
20437c478bd9Sstevel@tonic-gate 
20447c478bd9Sstevel@tonic-gate 	switch (type) {
20457c478bd9Sstevel@tonic-gate 	case RT_FS:
20467c478bd9Sstevel@tonic-gate 		bzero(&in_progress_fstab, sizeof (in_progress_fstab));
20477c478bd9Sstevel@tonic-gate 		return;
20487c478bd9Sstevel@tonic-gate 	case RT_NET:
20497c478bd9Sstevel@tonic-gate 		bzero(&in_progress_nwiftab, sizeof (in_progress_nwiftab));
20507c478bd9Sstevel@tonic-gate 		return;
20517c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
20527c478bd9Sstevel@tonic-gate 		bzero(&in_progress_devtab, sizeof (in_progress_devtab));
20537c478bd9Sstevel@tonic-gate 		return;
20547c478bd9Sstevel@tonic-gate 	case RT_RCTL:
20550209230bSgjelinek 		if (global_zone)
20560209230bSgjelinek 			zerr(gettext("WARNING: Setting a global zone resource "
20570209230bSgjelinek 			    "control too low could deny\nservice "
20580209230bSgjelinek 			    "to even the root user; "
20590209230bSgjelinek 			    "this could render the system impossible\n"
20600209230bSgjelinek 			    "to administer.  Please use caution."));
20617c478bd9Sstevel@tonic-gate 		bzero(&in_progress_rctltab, sizeof (in_progress_rctltab));
20627c478bd9Sstevel@tonic-gate 		return;
20637c478bd9Sstevel@tonic-gate 	case RT_ATTR:
20647c478bd9Sstevel@tonic-gate 		bzero(&in_progress_attrtab, sizeof (in_progress_attrtab));
20657c478bd9Sstevel@tonic-gate 		return;
2066fa9e4066Sahrens 	case RT_DATASET:
2067fa9e4066Sahrens 		bzero(&in_progress_dstab, sizeof (in_progress_dstab));
2068fa9e4066Sahrens 		return;
20690209230bSgjelinek 	case RT_DCPU:
2070c97ad5cdSakolb 		/* Make sure there isn't already a cpu-set or cpu-cap entry. */
20710209230bSgjelinek 		if (zonecfg_lookup_pset(handle, &tmp_psettab) == Z_OK) {
20720209230bSgjelinek 			zerr(gettext("The %s resource already exists."),
20730209230bSgjelinek 			    rt_to_str(RT_DCPU));
20740209230bSgjelinek 			goto bad;
20750209230bSgjelinek 		}
2076c97ad5cdSakolb 		if (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &tmp) !=
2077c97ad5cdSakolb 		    Z_NO_ENTRY) {
2078c97ad5cdSakolb 			zerr(gettext("The %s resource already exists."),
2079c97ad5cdSakolb 			    rt_to_str(RT_PCAP));
2080c97ad5cdSakolb 			goto bad;
2081c97ad5cdSakolb 		}
20820209230bSgjelinek 
20830209230bSgjelinek 		/* Make sure the pool property isn't set. */
20840209230bSgjelinek 		if (zonecfg_get_pool(handle, pool, sizeof (pool)) == Z_OK &&
20850209230bSgjelinek 		    strlen(pool) > 0) {
20860209230bSgjelinek 			zerr(gettext("The %s property is already set.  "
20870209230bSgjelinek 			    "A persistent pool is incompatible with\nthe %s "
20880209230bSgjelinek 			    "resource."),
20890209230bSgjelinek 			    pt_to_str(PT_POOL), rt_to_str(RT_DCPU));
20900209230bSgjelinek 			goto bad;
20910209230bSgjelinek 		}
20920209230bSgjelinek 
20930209230bSgjelinek 		bzero(&in_progress_psettab, sizeof (in_progress_psettab));
20940209230bSgjelinek 		return;
2095c97ad5cdSakolb 	case RT_PCAP:
2096c97ad5cdSakolb 		/*
2097c97ad5cdSakolb 		 * Make sure there isn't already a cpu-set or incompatible
2098c97ad5cdSakolb 		 * cpu-cap rctls.
2099c97ad5cdSakolb 		 */
2100c97ad5cdSakolb 		if (zonecfg_lookup_pset(handle, &tmp_psettab) == Z_OK) {
2101c97ad5cdSakolb 			zerr(gettext("The %s resource already exists."),
2102c97ad5cdSakolb 			    rt_to_str(RT_DCPU));
2103c97ad5cdSakolb 			goto bad;
2104c97ad5cdSakolb 		}
2105c97ad5cdSakolb 
2106c97ad5cdSakolb 		switch (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &tmp)) {
2107c97ad5cdSakolb 		case Z_ALIAS_DISALLOW:
2108c97ad5cdSakolb 			zone_perror(rt_to_str(RT_PCAP), Z_ALIAS_DISALLOW,
2109bbec428eSgjelinek 			    B_FALSE);
2110c97ad5cdSakolb 			goto bad;
2111c97ad5cdSakolb 
2112c97ad5cdSakolb 		case Z_OK:
2113c97ad5cdSakolb 			zerr(gettext("The %s resource already exists."),
2114c97ad5cdSakolb 			    rt_to_str(RT_PCAP));
2115c97ad5cdSakolb 			goto bad;
2116c97ad5cdSakolb 
2117c97ad5cdSakolb 		default:
2118c97ad5cdSakolb 			break;
2119c97ad5cdSakolb 		}
2120c97ad5cdSakolb 		return;
21210209230bSgjelinek 	case RT_MCAP:
21220209230bSgjelinek 		/*
21230209230bSgjelinek 		 * Make sure there isn't already a mem-cap entry or max-swap
21240209230bSgjelinek 		 * or max-locked rctl.
21250209230bSgjelinek 		 */
21260209230bSgjelinek 		if (zonecfg_lookup_mcap(handle, &tmp_mcaptab) == Z_OK ||
21270209230bSgjelinek 		    zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP, &tmp_mcap)
21280209230bSgjelinek 		    == Z_OK ||
21290209230bSgjelinek 		    zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM,
21300209230bSgjelinek 		    &tmp_mcap) == Z_OK) {
21310209230bSgjelinek 			zerr(gettext("The %s resource or a related resource "
21320209230bSgjelinek 			    "control already exists."), rt_to_str(RT_MCAP));
21330209230bSgjelinek 			goto bad;
21340209230bSgjelinek 		}
21350209230bSgjelinek 		if (global_zone)
21360209230bSgjelinek 			zerr(gettext("WARNING: Setting a global zone memory "
21370209230bSgjelinek 			    "cap too low could deny\nservice "
21380209230bSgjelinek 			    "to even the root user; "
21390209230bSgjelinek 			    "this could render the system impossible\n"
21400209230bSgjelinek 			    "to administer.  Please use caution."));
21410209230bSgjelinek 		bzero(&in_progress_mcaptab, sizeof (in_progress_mcaptab));
21420209230bSgjelinek 		return;
2143cb8a054bSGlenn Faden 	case RT_ADMIN:
2144cb8a054bSGlenn Faden 		bzero(&in_progress_admintab, sizeof (in_progress_admintab));
2145cb8a054bSGlenn Faden 		return;
21467c478bd9Sstevel@tonic-gate 	default:
2147bbec428eSgjelinek 		zone_perror(rt_to_str(type), Z_NO_RESOURCE_TYPE, B_TRUE);
2148bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
2149bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
21507c478bd9Sstevel@tonic-gate 	}
21517c478bd9Sstevel@tonic-gate bad:
2152bbec428eSgjelinek 	global_scope = B_TRUE;
21537c478bd9Sstevel@tonic-gate 	end_op = -1;
21547c478bd9Sstevel@tonic-gate }
21557c478bd9Sstevel@tonic-gate 
21567c478bd9Sstevel@tonic-gate static void
21577c478bd9Sstevel@tonic-gate do_complex_rctl_val(complex_property_ptr_t cp)
21587c478bd9Sstevel@tonic-gate {
21597c478bd9Sstevel@tonic-gate 	struct zone_rctlvaltab *rctlvaltab;
21607c478bd9Sstevel@tonic-gate 	complex_property_ptr_t cx;
2161bbec428eSgjelinek 	boolean_t seen_priv = B_FALSE, seen_limit = B_FALSE,
2162bbec428eSgjelinek 	    seen_action = B_FALSE;
21637c478bd9Sstevel@tonic-gate 	rctlblk_t *rctlblk;
21647c478bd9Sstevel@tonic-gate 	int err;
21657c478bd9Sstevel@tonic-gate 
21667c478bd9Sstevel@tonic-gate 	if ((rctlvaltab = alloc_rctlvaltab()) == NULL) {
2167bbec428eSgjelinek 		zone_perror(zone, Z_NOMEM, B_TRUE);
21687c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
21697c478bd9Sstevel@tonic-gate 	}
21707c478bd9Sstevel@tonic-gate 	for (cx = cp; cx != NULL; cx = cx->cp_next) {
21717c478bd9Sstevel@tonic-gate 		switch (cx->cp_type) {
21727c478bd9Sstevel@tonic-gate 		case PT_PRIV:
21737c478bd9Sstevel@tonic-gate 			if (seen_priv) {
21747c478bd9Sstevel@tonic-gate 				zerr(gettext("%s already specified"),
21757c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_PRIV));
21767c478bd9Sstevel@tonic-gate 				goto bad;
21777c478bd9Sstevel@tonic-gate 			}
21787c478bd9Sstevel@tonic-gate 			(void) strlcpy(rctlvaltab->zone_rctlval_priv,
21797c478bd9Sstevel@tonic-gate 			    cx->cp_value,
21807c478bd9Sstevel@tonic-gate 			    sizeof (rctlvaltab->zone_rctlval_priv));
2181bbec428eSgjelinek 			seen_priv = B_TRUE;
21827c478bd9Sstevel@tonic-gate 			break;
21837c478bd9Sstevel@tonic-gate 		case PT_LIMIT:
21847c478bd9Sstevel@tonic-gate 			if (seen_limit) {
21857c478bd9Sstevel@tonic-gate 				zerr(gettext("%s already specified"),
21867c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_LIMIT));
21877c478bd9Sstevel@tonic-gate 				goto bad;
21887c478bd9Sstevel@tonic-gate 			}
21897c478bd9Sstevel@tonic-gate 			(void) strlcpy(rctlvaltab->zone_rctlval_limit,
21907c478bd9Sstevel@tonic-gate 			    cx->cp_value,
21917c478bd9Sstevel@tonic-gate 			    sizeof (rctlvaltab->zone_rctlval_limit));
2192bbec428eSgjelinek 			seen_limit = B_TRUE;
21937c478bd9Sstevel@tonic-gate 			break;
21947c478bd9Sstevel@tonic-gate 		case PT_ACTION:
21957c478bd9Sstevel@tonic-gate 			if (seen_action) {
21967c478bd9Sstevel@tonic-gate 				zerr(gettext("%s already specified"),
21977c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_ACTION));
21987c478bd9Sstevel@tonic-gate 				goto bad;
21997c478bd9Sstevel@tonic-gate 			}
22007c478bd9Sstevel@tonic-gate 			(void) strlcpy(rctlvaltab->zone_rctlval_action,
22017c478bd9Sstevel@tonic-gate 			    cx->cp_value,
22027c478bd9Sstevel@tonic-gate 			    sizeof (rctlvaltab->zone_rctlval_action));
2203bbec428eSgjelinek 			seen_action = B_TRUE;
22047c478bd9Sstevel@tonic-gate 			break;
22057c478bd9Sstevel@tonic-gate 		default:
22067c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(PT_VALUE),
2207bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
2208bbec428eSgjelinek 			long_usage(CMD_ADD, B_TRUE);
2209bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
22107c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(rctlvaltab);
22117c478bd9Sstevel@tonic-gate 			return;
22127c478bd9Sstevel@tonic-gate 		}
22137c478bd9Sstevel@tonic-gate 	}
22147c478bd9Sstevel@tonic-gate 	if (!seen_priv)
22157c478bd9Sstevel@tonic-gate 		zerr(gettext("%s not specified"), pt_to_str(PT_PRIV));
22167c478bd9Sstevel@tonic-gate 	if (!seen_limit)
22177c478bd9Sstevel@tonic-gate 		zerr(gettext("%s not specified"), pt_to_str(PT_LIMIT));
22187c478bd9Sstevel@tonic-gate 	if (!seen_action)
22197c478bd9Sstevel@tonic-gate 		zerr(gettext("%s not specified"), pt_to_str(PT_ACTION));
22207c478bd9Sstevel@tonic-gate 	if (!seen_priv || !seen_limit || !seen_action)
22217c478bd9Sstevel@tonic-gate 		goto bad;
22227c478bd9Sstevel@tonic-gate 	rctlvaltab->zone_rctlval_next = NULL;
22237c478bd9Sstevel@tonic-gate 	rctlblk = alloca(rctlblk_size());
22247c478bd9Sstevel@tonic-gate 	/*
22257c478bd9Sstevel@tonic-gate 	 * Make sure the rctl value looks roughly correct; we won't know if
22267c478bd9Sstevel@tonic-gate 	 * it's truly OK until we verify the configuration on the target
22277c478bd9Sstevel@tonic-gate 	 * system.
22287c478bd9Sstevel@tonic-gate 	 */
22297c478bd9Sstevel@tonic-gate 	if (zonecfg_construct_rctlblk(rctlvaltab, rctlblk) != Z_OK ||
22307c478bd9Sstevel@tonic-gate 	    !zonecfg_valid_rctlblk(rctlblk)) {
22317c478bd9Sstevel@tonic-gate 		zerr(gettext("Invalid %s %s specification"), rt_to_str(RT_RCTL),
22327c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_VALUE));
22337c478bd9Sstevel@tonic-gate 		goto bad;
22347c478bd9Sstevel@tonic-gate 	}
22357c478bd9Sstevel@tonic-gate 	err = zonecfg_add_rctl_value(&in_progress_rctltab, rctlvaltab);
22367c478bd9Sstevel@tonic-gate 	if (err != Z_OK)
2237bbec428eSgjelinek 		zone_perror(pt_to_str(PT_VALUE), err, B_TRUE);
22387c478bd9Sstevel@tonic-gate 	return;
22397c478bd9Sstevel@tonic-gate 
22407c478bd9Sstevel@tonic-gate bad:
22417c478bd9Sstevel@tonic-gate 	zonecfg_free_rctl_value_list(rctlvaltab);
22427c478bd9Sstevel@tonic-gate }
22437c478bd9Sstevel@tonic-gate 
22447c478bd9Sstevel@tonic-gate static void
22457c478bd9Sstevel@tonic-gate add_property(cmd_t *cmd)
22467c478bd9Sstevel@tonic-gate {
22477c478bd9Sstevel@tonic-gate 	char *prop_id;
22487c478bd9Sstevel@tonic-gate 	int err, res_type, prop_type;
22497c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
22507c478bd9Sstevel@tonic-gate 	list_property_ptr_t l;
22517c478bd9Sstevel@tonic-gate 
22527c478bd9Sstevel@tonic-gate 	res_type = resource_scope;
22537c478bd9Sstevel@tonic-gate 	prop_type = cmd->cmd_prop_name[0];
22547c478bd9Sstevel@tonic-gate 	if (res_type == RT_UNKNOWN || prop_type == PT_UNKNOWN) {
2255bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
22567c478bd9Sstevel@tonic-gate 		return;
22577c478bd9Sstevel@tonic-gate 	}
22587c478bd9Sstevel@tonic-gate 
22597c478bd9Sstevel@tonic-gate 	if (cmd->cmd_prop_nv_pairs != 1) {
2260bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
22617c478bd9Sstevel@tonic-gate 		return;
22627c478bd9Sstevel@tonic-gate 	}
22637c478bd9Sstevel@tonic-gate 
2264bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
22657c478bd9Sstevel@tonic-gate 		return;
22667c478bd9Sstevel@tonic-gate 
22677c478bd9Sstevel@tonic-gate 	switch (res_type) {
22687c478bd9Sstevel@tonic-gate 	case RT_FS:
22697c478bd9Sstevel@tonic-gate 		if (prop_type != PT_OPTIONS) {
22707c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
2271bbec428eSgjelinek 			    B_TRUE);
2272bbec428eSgjelinek 			long_usage(CMD_ADD, B_TRUE);
2273bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
22747c478bd9Sstevel@tonic-gate 			return;
22757c478bd9Sstevel@tonic-gate 		}
22767c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[0];
22777c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE &&
22787c478bd9Sstevel@tonic-gate 		    pp->pv_type != PROP_VAL_LIST) {
22797c478bd9Sstevel@tonic-gate 			zerr(gettext("A %s or %s value was expected here."),
22807c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_SIMPLE),
22817c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_LIST));
2282bbec428eSgjelinek 			saw_error = B_TRUE;
22837c478bd9Sstevel@tonic-gate 			return;
22847c478bd9Sstevel@tonic-gate 		}
22857c478bd9Sstevel@tonic-gate 		if (pp->pv_type == PROP_VAL_SIMPLE) {
22867c478bd9Sstevel@tonic-gate 			if (pp->pv_simple == NULL) {
2287bbec428eSgjelinek 				long_usage(CMD_ADD, B_TRUE);
22887c478bd9Sstevel@tonic-gate 				return;
22897c478bd9Sstevel@tonic-gate 			}
22907c478bd9Sstevel@tonic-gate 			prop_id = pp->pv_simple;
22917c478bd9Sstevel@tonic-gate 			err = zonecfg_add_fs_option(&in_progress_fstab,
22927c478bd9Sstevel@tonic-gate 			    prop_id);
22937c478bd9Sstevel@tonic-gate 			if (err != Z_OK)
2294bbec428eSgjelinek 				zone_perror(pt_to_str(prop_type), err, B_TRUE);
22957c478bd9Sstevel@tonic-gate 		} else {
22967c478bd9Sstevel@tonic-gate 			list_property_ptr_t list;
22977c478bd9Sstevel@tonic-gate 
22987c478bd9Sstevel@tonic-gate 			for (list = pp->pv_list; list != NULL;
22997c478bd9Sstevel@tonic-gate 			    list = list->lp_next) {
23007c478bd9Sstevel@tonic-gate 				prop_id = list->lp_simple;
23017c478bd9Sstevel@tonic-gate 				if (prop_id == NULL)
23027c478bd9Sstevel@tonic-gate 					break;
23037c478bd9Sstevel@tonic-gate 				err = zonecfg_add_fs_option(
23047c478bd9Sstevel@tonic-gate 				    &in_progress_fstab, prop_id);
23057c478bd9Sstevel@tonic-gate 				if (err != Z_OK)
23067c478bd9Sstevel@tonic-gate 					zone_perror(pt_to_str(prop_type), err,
2307bbec428eSgjelinek 					    B_TRUE);
23087c478bd9Sstevel@tonic-gate 			}
23097c478bd9Sstevel@tonic-gate 		}
23107c478bd9Sstevel@tonic-gate 		return;
23117c478bd9Sstevel@tonic-gate 	case RT_RCTL:
23127c478bd9Sstevel@tonic-gate 		if (prop_type != PT_VALUE) {
23137c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
2314bbec428eSgjelinek 			    B_TRUE);
2315bbec428eSgjelinek 			long_usage(CMD_ADD, B_TRUE);
2316bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
23177c478bd9Sstevel@tonic-gate 			return;
23187c478bd9Sstevel@tonic-gate 		}
23197c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[0];
23207c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_COMPLEX &&
23217c478bd9Sstevel@tonic-gate 		    pp->pv_type != PROP_VAL_LIST) {
23227c478bd9Sstevel@tonic-gate 			zerr(gettext("A %s or %s value was expected here."),
23237c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_COMPLEX),
23247c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_LIST));
2325bbec428eSgjelinek 			saw_error = B_TRUE;
23267c478bd9Sstevel@tonic-gate 			return;
23277c478bd9Sstevel@tonic-gate 		}
23287c478bd9Sstevel@tonic-gate 		if (pp->pv_type == PROP_VAL_COMPLEX) {
23297c478bd9Sstevel@tonic-gate 			do_complex_rctl_val(pp->pv_complex);
23307c478bd9Sstevel@tonic-gate 			return;
23317c478bd9Sstevel@tonic-gate 		}
23327c478bd9Sstevel@tonic-gate 		for (l = pp->pv_list; l != NULL; l = l->lp_next)
23337c478bd9Sstevel@tonic-gate 			do_complex_rctl_val(l->lp_complex);
23347c478bd9Sstevel@tonic-gate 		return;
23357c478bd9Sstevel@tonic-gate 	default:
2336bbec428eSgjelinek 		zone_perror(rt_to_str(res_type), Z_NO_RESOURCE_TYPE, B_TRUE);
2337bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
2338bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
23397c478bd9Sstevel@tonic-gate 		return;
23407c478bd9Sstevel@tonic-gate 	}
23417c478bd9Sstevel@tonic-gate }
23427c478bd9Sstevel@tonic-gate 
23430209230bSgjelinek static boolean_t
23440209230bSgjelinek gz_invalid_resource(int type)
23450209230bSgjelinek {
2346*6e1ae2a3SGary Pennington 	return (global_zone && (type == RT_FS ||
23470209230bSgjelinek 	    type == RT_NET || type == RT_DEVICE || type == RT_ATTR ||
23480209230bSgjelinek 	    type == RT_DATASET));
23490209230bSgjelinek }
23500209230bSgjelinek 
23510209230bSgjelinek static boolean_t
23520209230bSgjelinek gz_invalid_rt_property(int type)
23530209230bSgjelinek {
23540209230bSgjelinek 	return (global_zone && (type == RT_ZONENAME || type == RT_ZONEPATH ||
23550209230bSgjelinek 	    type == RT_AUTOBOOT || type == RT_LIMITPRIV ||
2356f4b3ec61Sdh155122 	    type == RT_BOOTARGS || type == RT_BRAND || type == RT_SCHED ||
23570fbb751dSJohn Levon 	    type == RT_IPTYPE || type == RT_HOSTID || type == RT_FS_ALLOWED));
23580209230bSgjelinek }
23590209230bSgjelinek 
23600209230bSgjelinek static boolean_t
23610209230bSgjelinek gz_invalid_property(int type)
23620209230bSgjelinek {
23630209230bSgjelinek 	return (global_zone && (type == PT_ZONENAME || type == PT_ZONEPATH ||
23640209230bSgjelinek 	    type == PT_AUTOBOOT || type == PT_LIMITPRIV ||
2365f4b3ec61Sdh155122 	    type == PT_BOOTARGS || type == PT_BRAND || type == PT_SCHED ||
23660fbb751dSJohn Levon 	    type == PT_IPTYPE || type == PT_HOSTID || type == PT_FS_ALLOWED));
23670209230bSgjelinek }
23680209230bSgjelinek 
23697c478bd9Sstevel@tonic-gate void
23707c478bd9Sstevel@tonic-gate add_func(cmd_t *cmd)
23717c478bd9Sstevel@tonic-gate {
23727c478bd9Sstevel@tonic-gate 	int arg;
2373bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
23747c478bd9Sstevel@tonic-gate 
23757c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
23767c478bd9Sstevel@tonic-gate 
23777c478bd9Sstevel@tonic-gate 	optind = 0;
23787ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
23797c478bd9Sstevel@tonic-gate 		switch (arg) {
23807c478bd9Sstevel@tonic-gate 		case '?':
23817c478bd9Sstevel@tonic-gate 			longer_usage(CMD_ADD);
2382bbec428eSgjelinek 			arg_err = B_TRUE;
23837ec75eb8Sgjelinek 			break;
23847c478bd9Sstevel@tonic-gate 		default:
23857c478bd9Sstevel@tonic-gate 			short_usage(CMD_ADD);
2386bbec428eSgjelinek 			arg_err = B_TRUE;
23877ec75eb8Sgjelinek 			break;
23887ec75eb8Sgjelinek 		}
23897ec75eb8Sgjelinek 	}
23907ec75eb8Sgjelinek 	if (arg_err)
23917c478bd9Sstevel@tonic-gate 		return;
23927ec75eb8Sgjelinek 
23937c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
23947c478bd9Sstevel@tonic-gate 		short_usage(CMD_ADD);
23957c478bd9Sstevel@tonic-gate 		return;
23967c478bd9Sstevel@tonic-gate 	}
23977c478bd9Sstevel@tonic-gate 
23987c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_ADD))
23997c478bd9Sstevel@tonic-gate 		return;
24007c478bd9Sstevel@tonic-gate 
2401bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
24027c478bd9Sstevel@tonic-gate 		return;
24037c478bd9Sstevel@tonic-gate 	if (global_scope) {
24040209230bSgjelinek 		if (gz_invalid_resource(cmd->cmd_res_type)) {
24050209230bSgjelinek 			zerr(gettext("Cannot add a %s resource to the "
24060209230bSgjelinek 			    "global zone."), rt_to_str(cmd->cmd_res_type));
2407bbec428eSgjelinek 			saw_error = B_TRUE;
24080209230bSgjelinek 			return;
24090209230bSgjelinek 		}
24100209230bSgjelinek 
2411bbec428eSgjelinek 		global_scope = B_FALSE;
24127c478bd9Sstevel@tonic-gate 		resource_scope = cmd->cmd_res_type;
24137c478bd9Sstevel@tonic-gate 		end_op = CMD_ADD;
24147c478bd9Sstevel@tonic-gate 		add_resource(cmd);
24157c478bd9Sstevel@tonic-gate 	} else
24167c478bd9Sstevel@tonic-gate 		add_property(cmd);
24177c478bd9Sstevel@tonic-gate }
24187c478bd9Sstevel@tonic-gate 
2419087719fdSdp /*
2420087719fdSdp  * This routine has an unusual implementation, because it tries very
2421087719fdSdp  * hard to succeed in the face of a variety of failure modes.
2422087719fdSdp  * The most common and most vexing occurs when the index file and
2423087719fdSdp  * the /etc/zones/<zonename.xml> file are not both present.  In
2424087719fdSdp  * this case, delete must eradicate as much of the zone state as is left
2425087719fdSdp  * so that the user can later create a new zone with the same name.
2426087719fdSdp  */
24277c478bd9Sstevel@tonic-gate void
24287c478bd9Sstevel@tonic-gate delete_func(cmd_t *cmd)
24297c478bd9Sstevel@tonic-gate {
24307c478bd9Sstevel@tonic-gate 	int err, arg, answer;
24317c478bd9Sstevel@tonic-gate 	char line[ZONENAME_MAX + 128];	/* enough to ask a question */
2432bbec428eSgjelinek 	boolean_t force = B_FALSE;
2433bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
24347c478bd9Sstevel@tonic-gate 
24357c478bd9Sstevel@tonic-gate 	optind = 0;
24367c478bd9Sstevel@tonic-gate 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?F")) != EOF) {
24377c478bd9Sstevel@tonic-gate 		switch (arg) {
24387c478bd9Sstevel@tonic-gate 		case '?':
24397c478bd9Sstevel@tonic-gate 			longer_usage(CMD_DELETE);
2440bbec428eSgjelinek 			arg_err = B_TRUE;
24417ec75eb8Sgjelinek 			break;
24427c478bd9Sstevel@tonic-gate 		case 'F':
2443bbec428eSgjelinek 			force = B_TRUE;
24447c478bd9Sstevel@tonic-gate 			break;
24457c478bd9Sstevel@tonic-gate 		default:
24467c478bd9Sstevel@tonic-gate 			short_usage(CMD_DELETE);
2447bbec428eSgjelinek 			arg_err = B_TRUE;
24487ec75eb8Sgjelinek 			break;
24497ec75eb8Sgjelinek 		}
24507ec75eb8Sgjelinek 	}
24517ec75eb8Sgjelinek 	if (arg_err)
24527c478bd9Sstevel@tonic-gate 		return;
24537ec75eb8Sgjelinek 
24547c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
24557c478bd9Sstevel@tonic-gate 		short_usage(CMD_DELETE);
24567c478bd9Sstevel@tonic-gate 		return;
24577c478bd9Sstevel@tonic-gate 	}
24587c478bd9Sstevel@tonic-gate 
24597c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_DELETE))
24607c478bd9Sstevel@tonic-gate 		return;
24617c478bd9Sstevel@tonic-gate 
2462087719fdSdp 	if (!force) {
2463087719fdSdp 		/*
2464087719fdSdp 		 * Initialize sets up the global called "handle" and warns the
2465087719fdSdp 		 * user if the zone is not configured.  In force mode, we don't
2466087719fdSdp 		 * trust that evaluation, and hence skip it.  (We don't need the
2467087719fdSdp 		 * handle to be loaded anyway, since zonecfg_destroy is done by
2468087719fdSdp 		 * zonename). However, we also have to take care to emulate the
2469087719fdSdp 		 * messages spit out by initialize; see below.
2470087719fdSdp 		 */
2471bbec428eSgjelinek 		if (initialize(B_TRUE) != Z_OK)
24727c478bd9Sstevel@tonic-gate 			return;
24737c478bd9Sstevel@tonic-gate 
24747c478bd9Sstevel@tonic-gate 		(void) snprintf(line, sizeof (line),
24757c478bd9Sstevel@tonic-gate 		    gettext("Are you sure you want to delete zone %s"), zone);
2476bbec428eSgjelinek 		if ((answer = ask_yesno(B_FALSE, line)) == -1) {
2477087719fdSdp 			zerr(gettext("Input not from terminal and -F not "
2478087719fdSdp 			    "specified:\n%s command ignored, exiting."),
2479087719fdSdp 			    cmd_to_str(CMD_DELETE));
24807c478bd9Sstevel@tonic-gate 			exit(Z_ERR);
24817c478bd9Sstevel@tonic-gate 		}
24827c478bd9Sstevel@tonic-gate 		if (answer != 1)
24837c478bd9Sstevel@tonic-gate 			return;
24847c478bd9Sstevel@tonic-gate 	}
24857c478bd9Sstevel@tonic-gate 
2486cb8a054bSGlenn Faden 	/*
2487cb8a054bSGlenn Faden 	 * This function removes the authorizations from user_attr
2488cb8a054bSGlenn Faden 	 * that correspond to those specified in the configuration
2489cb8a054bSGlenn Faden 	 */
2490cb8a054bSGlenn Faden 	if (initialize(B_TRUE) == Z_OK) {
2491cb8a054bSGlenn Faden 		(void) zonecfg_deauthorize_users(handle, zone);
2492cb8a054bSGlenn Faden 	}
2493087719fdSdp 	if ((err = zonecfg_destroy(zone, force)) != Z_OK) {
2494087719fdSdp 		if ((err == Z_BAD_ZONE_STATE) && !force) {
2495087719fdSdp 			zerr(gettext("Zone %s not in %s state; %s not "
2496087719fdSdp 			    "allowed.  Use -F to force %s."),
2497087719fdSdp 			    zone, zone_state_str(ZONE_STATE_CONFIGURED),
2498087719fdSdp 			    cmd_to_str(CMD_DELETE), cmd_to_str(CMD_DELETE));
2499087719fdSdp 		} else {
2500bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
25017c478bd9Sstevel@tonic-gate 		}
2502087719fdSdp 	}
2503bbec428eSgjelinek 	need_to_commit = B_FALSE;
2504087719fdSdp 
2505087719fdSdp 	/*
2506087719fdSdp 	 * Emulate initialize's messaging; if there wasn't a valid handle to
2507087719fdSdp 	 * begin with, then user had typed delete (or delete -F) multiple
2508087719fdSdp 	 * times.  So we emit a message.
2509087719fdSdp 	 *
2510087719fdSdp 	 * We only do this in the 'force' case because normally, initialize()
2511087719fdSdp 	 * takes care of this for us.
2512087719fdSdp 	 */
2513087719fdSdp 	if (force && zonecfg_check_handle(handle) != Z_OK && interactive_mode)
2514087719fdSdp 		(void) printf(gettext("Use '%s' to begin "
2515087719fdSdp 		    "configuring a new zone.\n"), cmd_to_str(CMD_CREATE));
25167c478bd9Sstevel@tonic-gate 
25177c478bd9Sstevel@tonic-gate 	/*
25187c478bd9Sstevel@tonic-gate 	 * Time for a new handle: finish the old one off first
25197c478bd9Sstevel@tonic-gate 	 * then get a new one properly to avoid leaks.
25207c478bd9Sstevel@tonic-gate 	 */
2521087719fdSdp 	if (got_handle) {
25227c478bd9Sstevel@tonic-gate 		zonecfg_fini_handle(handle);
25237c478bd9Sstevel@tonic-gate 		if ((handle = zonecfg_init_handle()) == NULL) {
2524bbec428eSgjelinek 			zone_perror(execname, Z_NOMEM, B_TRUE);
25257c478bd9Sstevel@tonic-gate 			exit(Z_ERR);
25267c478bd9Sstevel@tonic-gate 		}
25277c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_get_handle(zone, handle)) != Z_OK) {
25287c478bd9Sstevel@tonic-gate 			/* If there was no zone before, that's OK */
25297c478bd9Sstevel@tonic-gate 			if (err != Z_NO_ZONE)
2530bbec428eSgjelinek 				zone_perror(zone, err, B_TRUE);
2531bbec428eSgjelinek 			got_handle = B_FALSE;
25327c478bd9Sstevel@tonic-gate 		}
25337c478bd9Sstevel@tonic-gate 	}
2534087719fdSdp }
25357c478bd9Sstevel@tonic-gate 
25367c478bd9Sstevel@tonic-gate static int
2537bbec428eSgjelinek fill_in_fstab(cmd_t *cmd, struct zone_fstab *fstab, boolean_t fill_in_only)
25387c478bd9Sstevel@tonic-gate {
25397c478bd9Sstevel@tonic-gate 	int err, i;
25407c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
25417c478bd9Sstevel@tonic-gate 
2542bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
25437c478bd9Sstevel@tonic-gate 		return (err);
25447c478bd9Sstevel@tonic-gate 
2545e193d1e6Svp157776 	bzero(fstab, sizeof (*fstab));
25467c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
25477c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
25487c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
25497c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2550bbec428eSgjelinek 			saw_error = B_TRUE;
25517c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
25527c478bd9Sstevel@tonic-gate 		}
25537c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
25547c478bd9Sstevel@tonic-gate 		case PT_DIR:
25557c478bd9Sstevel@tonic-gate 			(void) strlcpy(fstab->zone_fs_dir, pp->pv_simple,
25567c478bd9Sstevel@tonic-gate 			    sizeof (fstab->zone_fs_dir));
25577c478bd9Sstevel@tonic-gate 			break;
25587c478bd9Sstevel@tonic-gate 		case PT_SPECIAL:
25597c478bd9Sstevel@tonic-gate 			(void) strlcpy(fstab->zone_fs_special, pp->pv_simple,
25607c478bd9Sstevel@tonic-gate 			    sizeof (fstab->zone_fs_special));
25617c478bd9Sstevel@tonic-gate 			break;
25627c478bd9Sstevel@tonic-gate 		case PT_RAW:
25637c478bd9Sstevel@tonic-gate 			(void) strlcpy(fstab->zone_fs_raw, pp->pv_simple,
25647c478bd9Sstevel@tonic-gate 			    sizeof (fstab->zone_fs_raw));
25657c478bd9Sstevel@tonic-gate 			break;
25667c478bd9Sstevel@tonic-gate 		case PT_TYPE:
25677c478bd9Sstevel@tonic-gate 			(void) strlcpy(fstab->zone_fs_type, pp->pv_simple,
25687c478bd9Sstevel@tonic-gate 			    sizeof (fstab->zone_fs_type));
25697c478bd9Sstevel@tonic-gate 			break;
25707c478bd9Sstevel@tonic-gate 		default:
25717c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2572bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
25737c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
25747c478bd9Sstevel@tonic-gate 		}
25757c478bd9Sstevel@tonic-gate 	}
25767c478bd9Sstevel@tonic-gate 	if (fill_in_only)
25777c478bd9Sstevel@tonic-gate 		return (Z_OK);
25787c478bd9Sstevel@tonic-gate 	return (zonecfg_lookup_filesystem(handle, fstab));
25797c478bd9Sstevel@tonic-gate }
25807c478bd9Sstevel@tonic-gate 
25817c478bd9Sstevel@tonic-gate static int
2582bbec428eSgjelinek fill_in_nwiftab(cmd_t *cmd, struct zone_nwiftab *nwiftab,
2583bbec428eSgjelinek     boolean_t fill_in_only)
25847c478bd9Sstevel@tonic-gate {
25857c478bd9Sstevel@tonic-gate 	int err, i;
25867c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
25877c478bd9Sstevel@tonic-gate 
2588bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
25897c478bd9Sstevel@tonic-gate 		return (err);
25907c478bd9Sstevel@tonic-gate 
2591e193d1e6Svp157776 	bzero(nwiftab, sizeof (*nwiftab));
25927c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
25937c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
25947c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
25957c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2596bbec428eSgjelinek 			saw_error = B_TRUE;
25977c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
25987c478bd9Sstevel@tonic-gate 		}
25997c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
26007c478bd9Sstevel@tonic-gate 		case PT_ADDRESS:
26017c478bd9Sstevel@tonic-gate 			(void) strlcpy(nwiftab->zone_nwif_address,
26027c478bd9Sstevel@tonic-gate 			    pp->pv_simple, sizeof (nwiftab->zone_nwif_address));
26037c478bd9Sstevel@tonic-gate 			break;
26047c478bd9Sstevel@tonic-gate 		case PT_PHYSICAL:
26057c478bd9Sstevel@tonic-gate 			(void) strlcpy(nwiftab->zone_nwif_physical,
26067c478bd9Sstevel@tonic-gate 			    pp->pv_simple,
26077c478bd9Sstevel@tonic-gate 			    sizeof (nwiftab->zone_nwif_physical));
26087c478bd9Sstevel@tonic-gate 			break;
2609de860bd9Sgfaden 		case PT_DEFROUTER:
2610de860bd9Sgfaden 			(void) strlcpy(nwiftab->zone_nwif_defrouter,
2611de860bd9Sgfaden 			    pp->pv_simple,
2612de860bd9Sgfaden 			    sizeof (nwiftab->zone_nwif_defrouter));
2613de860bd9Sgfaden 			break;
26147c478bd9Sstevel@tonic-gate 		default:
26157c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2616bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
26177c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
26187c478bd9Sstevel@tonic-gate 		}
26197c478bd9Sstevel@tonic-gate 	}
26207c478bd9Sstevel@tonic-gate 	if (fill_in_only)
26217c478bd9Sstevel@tonic-gate 		return (Z_OK);
26227c478bd9Sstevel@tonic-gate 	err = zonecfg_lookup_nwif(handle, nwiftab);
26237c478bd9Sstevel@tonic-gate 	return (err);
26247c478bd9Sstevel@tonic-gate }
26257c478bd9Sstevel@tonic-gate 
26267c478bd9Sstevel@tonic-gate static int
2627bbec428eSgjelinek fill_in_devtab(cmd_t *cmd, struct zone_devtab *devtab, boolean_t fill_in_only)
26287c478bd9Sstevel@tonic-gate {
26297c478bd9Sstevel@tonic-gate 	int err, i;
26307c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
26317c478bd9Sstevel@tonic-gate 
2632bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
26337c478bd9Sstevel@tonic-gate 		return (err);
26347c478bd9Sstevel@tonic-gate 
2635e193d1e6Svp157776 	bzero(devtab, sizeof (*devtab));
26367c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
26377c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
26387c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
26397c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2640bbec428eSgjelinek 			saw_error = B_TRUE;
26417c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
26427c478bd9Sstevel@tonic-gate 		}
26437c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
26447c478bd9Sstevel@tonic-gate 		case PT_MATCH:
26457c478bd9Sstevel@tonic-gate 			(void) strlcpy(devtab->zone_dev_match, pp->pv_simple,
26467c478bd9Sstevel@tonic-gate 			    sizeof (devtab->zone_dev_match));
26477c478bd9Sstevel@tonic-gate 			break;
26487c478bd9Sstevel@tonic-gate 		default:
26497c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2650bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
26517c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
26527c478bd9Sstevel@tonic-gate 		}
26537c478bd9Sstevel@tonic-gate 	}
26547c478bd9Sstevel@tonic-gate 	if (fill_in_only)
26557c478bd9Sstevel@tonic-gate 		return (Z_OK);
26567c478bd9Sstevel@tonic-gate 	err = zonecfg_lookup_dev(handle, devtab);
26577c478bd9Sstevel@tonic-gate 	return (err);
26587c478bd9Sstevel@tonic-gate }
26597c478bd9Sstevel@tonic-gate 
26607c478bd9Sstevel@tonic-gate static int
2661bbec428eSgjelinek fill_in_rctltab(cmd_t *cmd, struct zone_rctltab *rctltab,
2662bbec428eSgjelinek     boolean_t fill_in_only)
26637c478bd9Sstevel@tonic-gate {
26647c478bd9Sstevel@tonic-gate 	int err, i;
26657c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
26667c478bd9Sstevel@tonic-gate 
2667bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
26687c478bd9Sstevel@tonic-gate 		return (err);
26697c478bd9Sstevel@tonic-gate 
2670e193d1e6Svp157776 	bzero(rctltab, sizeof (*rctltab));
26717c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
26727c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
26737c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
26747c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2675bbec428eSgjelinek 			saw_error = B_TRUE;
26767c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
26777c478bd9Sstevel@tonic-gate 		}
26787c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
26797c478bd9Sstevel@tonic-gate 		case PT_NAME:
26807c478bd9Sstevel@tonic-gate 			(void) strlcpy(rctltab->zone_rctl_name, pp->pv_simple,
26817c478bd9Sstevel@tonic-gate 			    sizeof (rctltab->zone_rctl_name));
26827c478bd9Sstevel@tonic-gate 			break;
26837c478bd9Sstevel@tonic-gate 		default:
26847c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2685bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
26867c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
26877c478bd9Sstevel@tonic-gate 		}
26887c478bd9Sstevel@tonic-gate 	}
26897c478bd9Sstevel@tonic-gate 	if (fill_in_only)
26907c478bd9Sstevel@tonic-gate 		return (Z_OK);
26917c478bd9Sstevel@tonic-gate 	err = zonecfg_lookup_rctl(handle, rctltab);
26927c478bd9Sstevel@tonic-gate 	return (err);
26937c478bd9Sstevel@tonic-gate }
26947c478bd9Sstevel@tonic-gate 
26957c478bd9Sstevel@tonic-gate static int
2696bbec428eSgjelinek fill_in_attrtab(cmd_t *cmd, struct zone_attrtab *attrtab,
2697bbec428eSgjelinek     boolean_t fill_in_only)
26987c478bd9Sstevel@tonic-gate {
26997c478bd9Sstevel@tonic-gate 	int err, i;
27007c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
27017c478bd9Sstevel@tonic-gate 
2702bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
27037c478bd9Sstevel@tonic-gate 		return (err);
27047c478bd9Sstevel@tonic-gate 
2705e193d1e6Svp157776 	bzero(attrtab, sizeof (*attrtab));
27067c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
27077c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
27087c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
27097c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2710bbec428eSgjelinek 			saw_error = B_TRUE;
27117c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
27127c478bd9Sstevel@tonic-gate 		}
27137c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
27147c478bd9Sstevel@tonic-gate 		case PT_NAME:
27157c478bd9Sstevel@tonic-gate 			(void) strlcpy(attrtab->zone_attr_name, pp->pv_simple,
27167c478bd9Sstevel@tonic-gate 			    sizeof (attrtab->zone_attr_name));
27177c478bd9Sstevel@tonic-gate 			break;
27187c478bd9Sstevel@tonic-gate 		case PT_TYPE:
27197c478bd9Sstevel@tonic-gate 			(void) strlcpy(attrtab->zone_attr_type, pp->pv_simple,
27207c478bd9Sstevel@tonic-gate 			    sizeof (attrtab->zone_attr_type));
27217c478bd9Sstevel@tonic-gate 			break;
27227c478bd9Sstevel@tonic-gate 		case PT_VALUE:
27237c478bd9Sstevel@tonic-gate 			(void) strlcpy(attrtab->zone_attr_value, pp->pv_simple,
27247c478bd9Sstevel@tonic-gate 			    sizeof (attrtab->zone_attr_value));
27257c478bd9Sstevel@tonic-gate 			break;
27267c478bd9Sstevel@tonic-gate 		default:
27277c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2728bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
27297c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
27307c478bd9Sstevel@tonic-gate 		}
27317c478bd9Sstevel@tonic-gate 	}
27327c478bd9Sstevel@tonic-gate 	if (fill_in_only)
27337c478bd9Sstevel@tonic-gate 		return (Z_OK);
27347c478bd9Sstevel@tonic-gate 	err = zonecfg_lookup_attr(handle, attrtab);
27357c478bd9Sstevel@tonic-gate 	return (err);
27367c478bd9Sstevel@tonic-gate }
27377c478bd9Sstevel@tonic-gate 
2738fa9e4066Sahrens static int
2739bbec428eSgjelinek fill_in_dstab(cmd_t *cmd, struct zone_dstab *dstab, boolean_t fill_in_only)
2740fa9e4066Sahrens {
2741fa9e4066Sahrens 	int err, i;
2742fa9e4066Sahrens 	property_value_ptr_t pp;
2743fa9e4066Sahrens 
2744bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
2745fa9e4066Sahrens 		return (err);
2746fa9e4066Sahrens 
2747fa9e4066Sahrens 	dstab->zone_dataset_name[0] = '\0';
2748fa9e4066Sahrens 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
2749fa9e4066Sahrens 		pp = cmd->cmd_property_ptr[i];
2750fa9e4066Sahrens 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
2751fa9e4066Sahrens 			zerr(gettext("A simple value was expected here."));
2752bbec428eSgjelinek 			saw_error = B_TRUE;
2753fa9e4066Sahrens 			return (Z_INSUFFICIENT_SPEC);
2754fa9e4066Sahrens 		}
2755fa9e4066Sahrens 		switch (cmd->cmd_prop_name[i]) {
2756fa9e4066Sahrens 		case PT_NAME:
2757fa9e4066Sahrens 			(void) strlcpy(dstab->zone_dataset_name, pp->pv_simple,
2758fa9e4066Sahrens 			    sizeof (dstab->zone_dataset_name));
2759fa9e4066Sahrens 			break;
2760fa9e4066Sahrens 		default:
2761fa9e4066Sahrens 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2762bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
2763fa9e4066Sahrens 			return (Z_INSUFFICIENT_SPEC);
2764fa9e4066Sahrens 		}
2765fa9e4066Sahrens 	}
2766fa9e4066Sahrens 	if (fill_in_only)
2767fa9e4066Sahrens 		return (Z_OK);
2768fa9e4066Sahrens 	return (zonecfg_lookup_ds(handle, dstab));
2769fa9e4066Sahrens }
2770fa9e4066Sahrens 
2771cb8a054bSGlenn Faden static int
2772cb8a054bSGlenn Faden fill_in_admintab(cmd_t *cmd, struct zone_admintab *admintab,
2773cb8a054bSGlenn Faden     boolean_t fill_in_only)
2774cb8a054bSGlenn Faden {
2775cb8a054bSGlenn Faden 	int err, i;
2776cb8a054bSGlenn Faden 	property_value_ptr_t pp;
2777cb8a054bSGlenn Faden 
2778cb8a054bSGlenn Faden 	if ((err = initialize(B_TRUE)) != Z_OK)
2779cb8a054bSGlenn Faden 		return (err);
2780cb8a054bSGlenn Faden 
2781cb8a054bSGlenn Faden 	bzero(admintab, sizeof (*admintab));
2782cb8a054bSGlenn Faden 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
2783cb8a054bSGlenn Faden 		pp = cmd->cmd_property_ptr[i];
2784cb8a054bSGlenn Faden 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
2785cb8a054bSGlenn Faden 			zerr(gettext("A simple value was expected here."));
2786cb8a054bSGlenn Faden 			saw_error = B_TRUE;
2787cb8a054bSGlenn Faden 			return (Z_INSUFFICIENT_SPEC);
2788cb8a054bSGlenn Faden 		}
2789cb8a054bSGlenn Faden 		switch (cmd->cmd_prop_name[i]) {
2790cb8a054bSGlenn Faden 		case PT_USER:
2791cb8a054bSGlenn Faden 			(void) strlcpy(admintab->zone_admin_user, pp->pv_simple,
2792cb8a054bSGlenn Faden 			    sizeof (admintab->zone_admin_user));
2793cb8a054bSGlenn Faden 			break;
2794cb8a054bSGlenn Faden 		case PT_AUTHS:
2795cb8a054bSGlenn Faden 			(void) strlcpy(admintab->zone_admin_auths,
2796cb8a054bSGlenn Faden 			    pp->pv_simple, sizeof (admintab->zone_admin_auths));
2797cb8a054bSGlenn Faden 			break;
2798cb8a054bSGlenn Faden 		default:
2799cb8a054bSGlenn Faden 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2800cb8a054bSGlenn Faden 			    Z_NO_PROPERTY_TYPE, B_TRUE);
2801cb8a054bSGlenn Faden 			return (Z_INSUFFICIENT_SPEC);
2802cb8a054bSGlenn Faden 		}
2803cb8a054bSGlenn Faden 	}
2804cb8a054bSGlenn Faden 	if (fill_in_only)
2805cb8a054bSGlenn Faden 		return (Z_OK);
2806cb8a054bSGlenn Faden 	err = zonecfg_lookup_admin(handle, admintab);
2807cb8a054bSGlenn Faden 	return (err);
2808cb8a054bSGlenn Faden }
2809cb8a054bSGlenn Faden 
28107c478bd9Sstevel@tonic-gate static void
28110209230bSgjelinek remove_aliased_rctl(int type, char *name)
28127c478bd9Sstevel@tonic-gate {
28130209230bSgjelinek 	int err;
28140209230bSgjelinek 	uint64_t tmp;
28157c478bd9Sstevel@tonic-gate 
28160209230bSgjelinek 	if ((err = zonecfg_get_aliased_rctl(handle, name, &tmp)) != Z_OK) {
28170209230bSgjelinek 		zerr("%s %s: %s", cmd_to_str(CMD_CLEAR), pt_to_str(type),
28180209230bSgjelinek 		    zonecfg_strerror(err));
2819bbec428eSgjelinek 		saw_error = B_TRUE;
28207c478bd9Sstevel@tonic-gate 		return;
28217c478bd9Sstevel@tonic-gate 	}
28220209230bSgjelinek 	if ((err = zonecfg_rm_aliased_rctl(handle, name)) != Z_OK) {
28230209230bSgjelinek 		zerr("%s %s: %s", cmd_to_str(CMD_CLEAR), pt_to_str(type),
28240209230bSgjelinek 		    zonecfg_strerror(err));
2825bbec428eSgjelinek 		saw_error = B_TRUE;
28260209230bSgjelinek 	} else {
2827bbec428eSgjelinek 		need_to_commit = B_TRUE;
28280209230bSgjelinek 	}
28290209230bSgjelinek }
28307c478bd9Sstevel@tonic-gate 
28310209230bSgjelinek static boolean_t
28320209230bSgjelinek prompt_remove_resource(cmd_t *cmd, char *rsrc)
28330209230bSgjelinek {
28340209230bSgjelinek 	int num;
28350209230bSgjelinek 	int answer;
28360209230bSgjelinek 	int arg;
28370209230bSgjelinek 	boolean_t force = B_FALSE;
28380209230bSgjelinek 	char prompt[128];
2839bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
28407c478bd9Sstevel@tonic-gate 
28410209230bSgjelinek 	optind = 0;
28420209230bSgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "F")) != EOF) {
28430209230bSgjelinek 		switch (arg) {
28440209230bSgjelinek 		case 'F':
28450209230bSgjelinek 			force = B_TRUE;
28460209230bSgjelinek 			break;
28470209230bSgjelinek 		default:
2848bbec428eSgjelinek 			arg_err = B_TRUE;
28497ec75eb8Sgjelinek 			break;
28507ec75eb8Sgjelinek 		}
28517ec75eb8Sgjelinek 	}
28527ec75eb8Sgjelinek 	if (arg_err)
28530209230bSgjelinek 		return (B_FALSE);
28547ec75eb8Sgjelinek 
28550209230bSgjelinek 
28560209230bSgjelinek 	num = zonecfg_num_resources(handle, rsrc);
28570209230bSgjelinek 
28580209230bSgjelinek 	if (num == 0) {
28590209230bSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, cmd->cmd_res_type, Z_NO_ENTRY,
2860bbec428eSgjelinek 		    B_TRUE);
28610209230bSgjelinek 		return (B_FALSE);
28620209230bSgjelinek 	}
28630209230bSgjelinek 	if (num > 1 && !force) {
28640209230bSgjelinek 		if (!interactive_mode) {
28650209230bSgjelinek 			zerr(gettext("There are multiple instances of this "
28660209230bSgjelinek 			    "resource.  Either qualify the resource to\n"
28670209230bSgjelinek 			    "remove a single instance or use the -F option to "
28680209230bSgjelinek 			    "remove all instances."));
2869bbec428eSgjelinek 			saw_error = B_TRUE;
28700209230bSgjelinek 			return (B_FALSE);
28710209230bSgjelinek 		}
28720209230bSgjelinek 		(void) snprintf(prompt, sizeof (prompt), gettext(
28730209230bSgjelinek 		    "Are you sure you want to remove ALL '%s' resources"),
28740209230bSgjelinek 		    rsrc);
2875bbec428eSgjelinek 		answer = ask_yesno(B_FALSE, prompt);
28760209230bSgjelinek 		if (answer == -1) {
28770209230bSgjelinek 			zerr(gettext("Resource incomplete."));
28780209230bSgjelinek 			return (B_FALSE);
28790209230bSgjelinek 		}
28800209230bSgjelinek 		if (answer != 1)
28810209230bSgjelinek 			return (B_FALSE);
28820209230bSgjelinek 	}
28830209230bSgjelinek 	return (B_TRUE);
28840209230bSgjelinek }
28850209230bSgjelinek 
28860209230bSgjelinek static void
28870209230bSgjelinek remove_fs(cmd_t *cmd)
28880209230bSgjelinek {
28890209230bSgjelinek 	int err;
28900209230bSgjelinek 
28910209230bSgjelinek 	/* traditional, qualified fs removal */
28920209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
28930209230bSgjelinek 		struct zone_fstab fstab;
28940209230bSgjelinek 
2895bbec428eSgjelinek 		if ((err = fill_in_fstab(cmd, &fstab, B_FALSE)) != Z_OK) {
2896bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_FS, err, B_TRUE);
28977c478bd9Sstevel@tonic-gate 			return;
28987c478bd9Sstevel@tonic-gate 		}
28997c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_delete_filesystem(handle, &fstab)) != Z_OK)
2900bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_FS, err, B_TRUE);
29017c478bd9Sstevel@tonic-gate 		else
2902bbec428eSgjelinek 			need_to_commit = B_TRUE;
29037c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(fstab.zone_fs_options);
29047c478bd9Sstevel@tonic-gate 		return;
29050209230bSgjelinek 	}
29060209230bSgjelinek 
29070209230bSgjelinek 	/*
29080209230bSgjelinek 	 * unqualified fs removal.  remove all fs's but prompt if more
29090209230bSgjelinek 	 * than one.
29100209230bSgjelinek 	 */
29110209230bSgjelinek 	if (!prompt_remove_resource(cmd, "fs"))
29120209230bSgjelinek 		return;
29130209230bSgjelinek 
29140209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "fs")) != Z_OK)
2915bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_FS, err, B_TRUE);
29160209230bSgjelinek 	else
2917bbec428eSgjelinek 		need_to_commit = B_TRUE;
29180209230bSgjelinek }
29190209230bSgjelinek 
29200209230bSgjelinek static void
29210209230bSgjelinek remove_net(cmd_t *cmd)
29220209230bSgjelinek {
29230209230bSgjelinek 	int err;
29240209230bSgjelinek 
29250209230bSgjelinek 	/* traditional, qualified net removal */
29260209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
29270209230bSgjelinek 		struct zone_nwiftab nwiftab;
29280209230bSgjelinek 
2929bbec428eSgjelinek 		if ((err = fill_in_nwiftab(cmd, &nwiftab, B_FALSE)) != Z_OK) {
2930bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_NET, err, B_TRUE);
29317c478bd9Sstevel@tonic-gate 			return;
29327c478bd9Sstevel@tonic-gate 		}
29337c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_delete_nwif(handle, &nwiftab)) != Z_OK)
2934bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_NET, err, B_TRUE);
29357c478bd9Sstevel@tonic-gate 		else
2936bbec428eSgjelinek 			need_to_commit = B_TRUE;
29377c478bd9Sstevel@tonic-gate 		return;
29380209230bSgjelinek 	}
29390209230bSgjelinek 
29400209230bSgjelinek 	/*
29410209230bSgjelinek 	 * unqualified net removal.  remove all nets but prompt if more
29420209230bSgjelinek 	 * than one.
29430209230bSgjelinek 	 */
29440209230bSgjelinek 	if (!prompt_remove_resource(cmd, "net"))
29450209230bSgjelinek 		return;
29460209230bSgjelinek 
29470209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "net")) != Z_OK)
2948bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_NET, err, B_TRUE);
29490209230bSgjelinek 	else
2950bbec428eSgjelinek 		need_to_commit = B_TRUE;
29510209230bSgjelinek }
29520209230bSgjelinek 
29530209230bSgjelinek static void
29540209230bSgjelinek remove_device(cmd_t *cmd)
29550209230bSgjelinek {
29560209230bSgjelinek 	int err;
29570209230bSgjelinek 
29580209230bSgjelinek 	/* traditional, qualified device removal */
29590209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
29600209230bSgjelinek 		struct zone_devtab devtab;
29610209230bSgjelinek 
2962bbec428eSgjelinek 		if ((err = fill_in_devtab(cmd, &devtab, B_FALSE)) != Z_OK) {
2963bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_DEVICE, err, B_TRUE);
29647c478bd9Sstevel@tonic-gate 			return;
29657c478bd9Sstevel@tonic-gate 		}
29667c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_delete_dev(handle, &devtab)) != Z_OK)
2967bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_DEVICE, err, B_TRUE);
29687c478bd9Sstevel@tonic-gate 		else
2969bbec428eSgjelinek 			need_to_commit = B_TRUE;
29707c478bd9Sstevel@tonic-gate 		return;
29710209230bSgjelinek 	}
29720209230bSgjelinek 
29730209230bSgjelinek 	/*
29740209230bSgjelinek 	 * unqualified device removal.  remove all devices but prompt if more
29750209230bSgjelinek 	 * than one.
29760209230bSgjelinek 	 */
29770209230bSgjelinek 	if (!prompt_remove_resource(cmd, "device"))
29780209230bSgjelinek 		return;
29790209230bSgjelinek 
29800209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "device")) != Z_OK)
2981bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_DEVICE, err, B_TRUE);
29820209230bSgjelinek 	else
2983bbec428eSgjelinek 		need_to_commit = B_TRUE;
29840209230bSgjelinek }
29850209230bSgjelinek 
29860209230bSgjelinek static void
29870209230bSgjelinek remove_attr(cmd_t *cmd)
29880209230bSgjelinek {
29890209230bSgjelinek 	int err;
29900209230bSgjelinek 
29910209230bSgjelinek 	/* traditional, qualified attr removal */
29920209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
29930209230bSgjelinek 		struct zone_attrtab attrtab;
29940209230bSgjelinek 
2995bbec428eSgjelinek 		if ((err = fill_in_attrtab(cmd, &attrtab, B_FALSE)) != Z_OK) {
2996bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_ATTR, err, B_TRUE);
29970209230bSgjelinek 			return;
29980209230bSgjelinek 		}
29990209230bSgjelinek 		if ((err = zonecfg_delete_attr(handle, &attrtab)) != Z_OK)
3000bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_ATTR, err, B_TRUE);
30010209230bSgjelinek 		else
3002bbec428eSgjelinek 			need_to_commit = B_TRUE;
30030209230bSgjelinek 		return;
30040209230bSgjelinek 	}
30050209230bSgjelinek 
30060209230bSgjelinek 	/*
30070209230bSgjelinek 	 * unqualified attr removal.  remove all attrs but prompt if more
30080209230bSgjelinek 	 * than one.
30090209230bSgjelinek 	 */
30100209230bSgjelinek 	if (!prompt_remove_resource(cmd, "attr"))
30110209230bSgjelinek 		return;
30120209230bSgjelinek 
30130209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "attr")) != Z_OK)
3014bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_ATTR, err, B_TRUE);
30150209230bSgjelinek 	else
3016bbec428eSgjelinek 		need_to_commit = B_TRUE;
30170209230bSgjelinek }
30180209230bSgjelinek 
30190209230bSgjelinek static void
30200209230bSgjelinek remove_dataset(cmd_t *cmd)
30210209230bSgjelinek {
30220209230bSgjelinek 	int err;
30230209230bSgjelinek 
30240209230bSgjelinek 	/* traditional, qualified dataset removal */
30250209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
30260209230bSgjelinek 		struct zone_dstab dstab;
30270209230bSgjelinek 
3028bbec428eSgjelinek 		if ((err = fill_in_dstab(cmd, &dstab, B_FALSE)) != Z_OK) {
3029bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_DATASET, err, B_TRUE);
30300209230bSgjelinek 			return;
30310209230bSgjelinek 		}
30320209230bSgjelinek 		if ((err = zonecfg_delete_ds(handle, &dstab)) != Z_OK)
3033bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_DATASET, err, B_TRUE);
30340209230bSgjelinek 		else
3035bbec428eSgjelinek 			need_to_commit = B_TRUE;
30360209230bSgjelinek 		return;
30370209230bSgjelinek 	}
30380209230bSgjelinek 
30390209230bSgjelinek 	/*
30400209230bSgjelinek 	 * unqualified dataset removal.  remove all datasets but prompt if more
30410209230bSgjelinek 	 * than one.
30420209230bSgjelinek 	 */
30430209230bSgjelinek 	if (!prompt_remove_resource(cmd, "dataset"))
30440209230bSgjelinek 		return;
30450209230bSgjelinek 
30460209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "dataset")) != Z_OK)
3047bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_DATASET, err, B_TRUE);
30480209230bSgjelinek 	else
3049bbec428eSgjelinek 		need_to_commit = B_TRUE;
30500209230bSgjelinek }
30510209230bSgjelinek 
30520209230bSgjelinek static void
30530209230bSgjelinek remove_rctl(cmd_t *cmd)
30540209230bSgjelinek {
30550209230bSgjelinek 	int err;
30560209230bSgjelinek 
30570209230bSgjelinek 	/* traditional, qualified rctl removal */
30580209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
30590209230bSgjelinek 		struct zone_rctltab rctltab;
30600209230bSgjelinek 
3061bbec428eSgjelinek 		if ((err = fill_in_rctltab(cmd, &rctltab, B_FALSE)) != Z_OK) {
3062bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_RCTL, err, B_TRUE);
30637c478bd9Sstevel@tonic-gate 			return;
30647c478bd9Sstevel@tonic-gate 		}
30657c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_delete_rctl(handle, &rctltab)) != Z_OK)
3066bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_RCTL, err, B_TRUE);
30677c478bd9Sstevel@tonic-gate 		else
3068bbec428eSgjelinek 			need_to_commit = B_TRUE;
30697c478bd9Sstevel@tonic-gate 		zonecfg_free_rctl_value_list(rctltab.zone_rctl_valptr);
30707c478bd9Sstevel@tonic-gate 		return;
30717c478bd9Sstevel@tonic-gate 	}
30720209230bSgjelinek 
30730209230bSgjelinek 	/*
30740209230bSgjelinek 	 * unqualified rctl removal.  remove all rctls but prompt if more
30750209230bSgjelinek 	 * than one.
30760209230bSgjelinek 	 */
30770209230bSgjelinek 	if (!prompt_remove_resource(cmd, "rctl"))
30780209230bSgjelinek 		return;
30790209230bSgjelinek 
30800209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "rctl")) != Z_OK)
3081bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_RCTL, err, B_TRUE);
30827c478bd9Sstevel@tonic-gate 	else
3083bbec428eSgjelinek 		need_to_commit = B_TRUE;
30840209230bSgjelinek }
30850209230bSgjelinek 
30860209230bSgjelinek static void
30870209230bSgjelinek remove_pset()
30880209230bSgjelinek {
30890209230bSgjelinek 	int err;
30900209230bSgjelinek 	struct zone_psettab psettab;
30910209230bSgjelinek 
30920209230bSgjelinek 	if ((err = zonecfg_lookup_pset(handle, &psettab)) != Z_OK) {
3093bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_DCPU, err, B_TRUE);
30940209230bSgjelinek 		return;
30950209230bSgjelinek 	}
30960209230bSgjelinek 	if ((err = zonecfg_delete_pset(handle)) != Z_OK)
3097bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_DCPU, err, B_TRUE);
30980209230bSgjelinek 	else
3099bbec428eSgjelinek 		need_to_commit = B_TRUE;
31000209230bSgjelinek }
31010209230bSgjelinek 
31020209230bSgjelinek static void
3103c97ad5cdSakolb remove_pcap()
3104c97ad5cdSakolb {
3105c97ad5cdSakolb 	int err;
3106c97ad5cdSakolb 	uint64_t tmp;
3107c97ad5cdSakolb 
3108c97ad5cdSakolb 	if (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &tmp) != Z_OK) {
3109c97ad5cdSakolb 		zerr("%s %s: %s", cmd_to_str(CMD_REMOVE), rt_to_str(RT_PCAP),
3110c97ad5cdSakolb 		    zonecfg_strerror(Z_NO_RESOURCE_TYPE));
3111bbec428eSgjelinek 		saw_error = B_TRUE;
3112c97ad5cdSakolb 		return;
3113c97ad5cdSakolb 	}
3114c97ad5cdSakolb 
3115c97ad5cdSakolb 	if ((err = zonecfg_rm_aliased_rctl(handle, ALIAS_CPUCAP)) != Z_OK)
3116bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_PCAP, err, B_TRUE);
3117c97ad5cdSakolb 	else
3118bbec428eSgjelinek 		need_to_commit = B_TRUE;
3119c97ad5cdSakolb }
3120c97ad5cdSakolb 
3121c97ad5cdSakolb static void
31220209230bSgjelinek remove_mcap()
31230209230bSgjelinek {
31240209230bSgjelinek 	int err, res1, res2, res3;
31250209230bSgjelinek 	uint64_t tmp;
31260209230bSgjelinek 	struct zone_mcaptab mcaptab;
31270209230bSgjelinek 	boolean_t revert = B_FALSE;
31280209230bSgjelinek 
31290209230bSgjelinek 	res1 = zonecfg_lookup_mcap(handle, &mcaptab);
31300209230bSgjelinek 	res2 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP, &tmp);
31310209230bSgjelinek 	res3 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM, &tmp);
31320209230bSgjelinek 
31330209230bSgjelinek 	/* if none of these exist, there is no resource to remove */
31340209230bSgjelinek 	if (res1 != Z_OK && res2 != Z_OK && res3 != Z_OK) {
31350209230bSgjelinek 		zerr("%s %s: %s", cmd_to_str(CMD_REMOVE), rt_to_str(RT_MCAP),
31360209230bSgjelinek 		    zonecfg_strerror(Z_NO_RESOURCE_TYPE));
3137bbec428eSgjelinek 		saw_error = B_TRUE;
31380209230bSgjelinek 		return;
31390209230bSgjelinek 	}
31400209230bSgjelinek 	if (res1 == Z_OK) {
31410209230bSgjelinek 		if ((err = zonecfg_delete_mcap(handle)) != Z_OK) {
3142bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_MCAP, err, B_TRUE);
31430209230bSgjelinek 			revert = B_TRUE;
31440209230bSgjelinek 		} else {
3145bbec428eSgjelinek 			need_to_commit = B_TRUE;
31460209230bSgjelinek 		}
31470209230bSgjelinek 	}
31480209230bSgjelinek 	if (res2 == Z_OK) {
31490209230bSgjelinek 		if ((err = zonecfg_rm_aliased_rctl(handle, ALIAS_MAXSWAP))
31500209230bSgjelinek 		    != Z_OK) {
3151bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_MCAP, err, B_TRUE);
31520209230bSgjelinek 			revert = B_TRUE;
31530209230bSgjelinek 		} else {
3154bbec428eSgjelinek 			need_to_commit = B_TRUE;
31550209230bSgjelinek 		}
31560209230bSgjelinek 	}
31570209230bSgjelinek 	if (res3 == Z_OK) {
31580209230bSgjelinek 		if ((err = zonecfg_rm_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM))
31590209230bSgjelinek 		    != Z_OK) {
3160bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_MCAP, err, B_TRUE);
31610209230bSgjelinek 			revert = B_TRUE;
31620209230bSgjelinek 		} else {
3163bbec428eSgjelinek 			need_to_commit = B_TRUE;
31640209230bSgjelinek 		}
31650209230bSgjelinek 	}
31660209230bSgjelinek 
31670209230bSgjelinek 	if (revert)
3168bbec428eSgjelinek 		need_to_commit = B_FALSE;
31690209230bSgjelinek }
31700209230bSgjelinek 
31710209230bSgjelinek static void
3172cb8a054bSGlenn Faden remove_admin(cmd_t *cmd)
3173cb8a054bSGlenn Faden {
3174cb8a054bSGlenn Faden 	int err;
3175cb8a054bSGlenn Faden 
3176cb8a054bSGlenn Faden 	/* traditional, qualified attr removal */
3177cb8a054bSGlenn Faden 	if (cmd->cmd_prop_nv_pairs > 0) {
3178cb8a054bSGlenn Faden 		struct zone_admintab admintab;
3179cb8a054bSGlenn Faden 
3180cb8a054bSGlenn Faden 		if ((err = fill_in_admintab(cmd, &admintab, B_FALSE)) != Z_OK) {
3181cb8a054bSGlenn Faden 			z_cmd_rt_perror(CMD_REMOVE, RT_ADMIN,
3182cb8a054bSGlenn Faden 			    err, B_TRUE);
3183cb8a054bSGlenn Faden 			return;
3184cb8a054bSGlenn Faden 		}
3185cb8a054bSGlenn Faden 		if ((err = zonecfg_delete_admin(handle, &admintab,
3186cb8a054bSGlenn Faden 		    zone))
3187cb8a054bSGlenn Faden 		    != Z_OK)
3188cb8a054bSGlenn Faden 			z_cmd_rt_perror(CMD_REMOVE, RT_ADMIN,
3189cb8a054bSGlenn Faden 			    err, B_TRUE);
3190cb8a054bSGlenn Faden 		else
3191cb8a054bSGlenn Faden 			need_to_commit = B_TRUE;
3192cb8a054bSGlenn Faden 		return;
3193cb8a054bSGlenn Faden 	} else {
3194cb8a054bSGlenn Faden 		/*
3195cb8a054bSGlenn Faden 		 * unqualified admin removal.
3196cb8a054bSGlenn Faden 		 * remove all admins but prompt if more
3197cb8a054bSGlenn Faden 		 * than one.
3198cb8a054bSGlenn Faden 		 */
3199cb8a054bSGlenn Faden 		if (!prompt_remove_resource(cmd, "admin"))
3200cb8a054bSGlenn Faden 			return;
3201cb8a054bSGlenn Faden 
3202cb8a054bSGlenn Faden 		if ((err = zonecfg_delete_admins(handle, zone))
3203cb8a054bSGlenn Faden 		    != Z_OK)
3204cb8a054bSGlenn Faden 			z_cmd_rt_perror(CMD_REMOVE, RT_ADMIN,
3205cb8a054bSGlenn Faden 			    err, B_TRUE);
3206cb8a054bSGlenn Faden 		else
3207cb8a054bSGlenn Faden 			need_to_commit = B_TRUE;
3208cb8a054bSGlenn Faden 	}
3209cb8a054bSGlenn Faden }
3210cb8a054bSGlenn Faden 
3211cb8a054bSGlenn Faden static void
32120209230bSgjelinek remove_resource(cmd_t *cmd)
32130209230bSgjelinek {
32140209230bSgjelinek 	int type;
32150209230bSgjelinek 	int arg;
3216bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
32170209230bSgjelinek 
32180209230bSgjelinek 	if ((type = cmd->cmd_res_type) == RT_UNKNOWN) {
3219bbec428eSgjelinek 		long_usage(CMD_REMOVE, B_TRUE);
32200209230bSgjelinek 		return;
32210209230bSgjelinek 	}
32220209230bSgjelinek 
32230209230bSgjelinek 	optind = 0;
32240209230bSgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?F")) != EOF) {
32250209230bSgjelinek 		switch (arg) {
32260209230bSgjelinek 		case '?':
32270209230bSgjelinek 			longer_usage(CMD_REMOVE);
3228bbec428eSgjelinek 			arg_err = B_TRUE;
32297ec75eb8Sgjelinek 			break;
32300209230bSgjelinek 		case 'F':
32310209230bSgjelinek 			break;
32320209230bSgjelinek 		default:
32330209230bSgjelinek 			short_usage(CMD_REMOVE);
3234bbec428eSgjelinek 			arg_err = B_TRUE;
32357ec75eb8Sgjelinek 			break;
32367ec75eb8Sgjelinek 		}
32377ec75eb8Sgjelinek 	}
32387ec75eb8Sgjelinek 	if (arg_err)
32390209230bSgjelinek 		return;
32400209230bSgjelinek 
3241bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
32420209230bSgjelinek 		return;
32430209230bSgjelinek 
32440209230bSgjelinek 	switch (type) {
32450209230bSgjelinek 	case RT_FS:
32460209230bSgjelinek 		remove_fs(cmd);
32470209230bSgjelinek 		return;
32480209230bSgjelinek 	case RT_NET:
32490209230bSgjelinek 		remove_net(cmd);
32500209230bSgjelinek 		return;
32510209230bSgjelinek 	case RT_DEVICE:
32520209230bSgjelinek 		remove_device(cmd);
32530209230bSgjelinek 		return;
32540209230bSgjelinek 	case RT_RCTL:
32550209230bSgjelinek 		remove_rctl(cmd);
32560209230bSgjelinek 		return;
32570209230bSgjelinek 	case RT_ATTR:
32580209230bSgjelinek 		remove_attr(cmd);
32597c478bd9Sstevel@tonic-gate 		return;
3260fa9e4066Sahrens 	case RT_DATASET:
32610209230bSgjelinek 		remove_dataset(cmd);
3262fa9e4066Sahrens 		return;
32630209230bSgjelinek 	case RT_DCPU:
32640209230bSgjelinek 		remove_pset();
32650209230bSgjelinek 		return;
3266c97ad5cdSakolb 	case RT_PCAP:
3267c97ad5cdSakolb 		remove_pcap();
3268c97ad5cdSakolb 		return;
32690209230bSgjelinek 	case RT_MCAP:
32700209230bSgjelinek 		remove_mcap();
3271fa9e4066Sahrens 		return;
3272cb8a054bSGlenn Faden 	case RT_ADMIN:
3273cb8a054bSGlenn Faden 		remove_admin(cmd);
3274cb8a054bSGlenn Faden 		return;
32757c478bd9Sstevel@tonic-gate 	default:
3276bbec428eSgjelinek 		zone_perror(rt_to_str(type), Z_NO_RESOURCE_TYPE, B_TRUE);
3277bbec428eSgjelinek 		long_usage(CMD_REMOVE, B_TRUE);
3278bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
32797c478bd9Sstevel@tonic-gate 		return;
32807c478bd9Sstevel@tonic-gate 	}
32817c478bd9Sstevel@tonic-gate }
32827c478bd9Sstevel@tonic-gate 
32837c478bd9Sstevel@tonic-gate static void
32847c478bd9Sstevel@tonic-gate remove_property(cmd_t *cmd)
32857c478bd9Sstevel@tonic-gate {
32867c478bd9Sstevel@tonic-gate 	char *prop_id;
32877c478bd9Sstevel@tonic-gate 	int err, res_type, prop_type;
32887c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
32897c478bd9Sstevel@tonic-gate 	struct zone_rctlvaltab *rctlvaltab;
32907c478bd9Sstevel@tonic-gate 	complex_property_ptr_t cx;
32917c478bd9Sstevel@tonic-gate 
32927c478bd9Sstevel@tonic-gate 	res_type = resource_scope;
32937c478bd9Sstevel@tonic-gate 	prop_type = cmd->cmd_prop_name[0];
32947c478bd9Sstevel@tonic-gate 	if (res_type == RT_UNKNOWN || prop_type == PT_UNKNOWN) {
3295bbec428eSgjelinek 		long_usage(CMD_REMOVE, B_TRUE);
32967c478bd9Sstevel@tonic-gate 		return;
32977c478bd9Sstevel@tonic-gate 	}
32987c478bd9Sstevel@tonic-gate 
32997c478bd9Sstevel@tonic-gate 	if (cmd->cmd_prop_nv_pairs != 1) {
3300bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
33017c478bd9Sstevel@tonic-gate 		return;
33027c478bd9Sstevel@tonic-gate 	}
33037c478bd9Sstevel@tonic-gate 
3304bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
33057c478bd9Sstevel@tonic-gate 		return;
33067c478bd9Sstevel@tonic-gate 
33077c478bd9Sstevel@tonic-gate 	switch (res_type) {
33087c478bd9Sstevel@tonic-gate 	case RT_FS:
33097c478bd9Sstevel@tonic-gate 		if (prop_type != PT_OPTIONS) {
33107c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
3311bbec428eSgjelinek 			    B_TRUE);
3312bbec428eSgjelinek 			long_usage(CMD_REMOVE, B_TRUE);
3313bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
33147c478bd9Sstevel@tonic-gate 			return;
33157c478bd9Sstevel@tonic-gate 		}
33167c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[0];
33177c478bd9Sstevel@tonic-gate 		if (pp->pv_type == PROP_VAL_COMPLEX) {
33187c478bd9Sstevel@tonic-gate 			zerr(gettext("A %s or %s value was expected here."),
33197c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_SIMPLE),
33207c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_LIST));
3321bbec428eSgjelinek 			saw_error = B_TRUE;
33227c478bd9Sstevel@tonic-gate 			return;
33237c478bd9Sstevel@tonic-gate 		}
33247c478bd9Sstevel@tonic-gate 		if (pp->pv_type == PROP_VAL_SIMPLE) {
33257c478bd9Sstevel@tonic-gate 			if (pp->pv_simple == NULL) {
3326bbec428eSgjelinek 				long_usage(CMD_ADD, B_TRUE);
33277c478bd9Sstevel@tonic-gate 				return;
33287c478bd9Sstevel@tonic-gate 			}
33297c478bd9Sstevel@tonic-gate 			prop_id = pp->pv_simple;
33307c478bd9Sstevel@tonic-gate 			err = zonecfg_remove_fs_option(&in_progress_fstab,
33317c478bd9Sstevel@tonic-gate 			    prop_id);
33327c478bd9Sstevel@tonic-gate 			if (err != Z_OK)
3333bbec428eSgjelinek 				zone_perror(pt_to_str(prop_type), err, B_TRUE);
33347c478bd9Sstevel@tonic-gate 		} else {
33357c478bd9Sstevel@tonic-gate 			list_property_ptr_t list;
33367c478bd9Sstevel@tonic-gate 
33377c478bd9Sstevel@tonic-gate 			for (list = pp->pv_list; list != NULL;
33387c478bd9Sstevel@tonic-gate 			    list = list->lp_next) {
33397c478bd9Sstevel@tonic-gate 				prop_id = list->lp_simple;
33407c478bd9Sstevel@tonic-gate 				if (prop_id == NULL)
33417c478bd9Sstevel@tonic-gate 					break;
33427c478bd9Sstevel@tonic-gate 				err = zonecfg_remove_fs_option(
33437c478bd9Sstevel@tonic-gate 				    &in_progress_fstab, prop_id);
33447c478bd9Sstevel@tonic-gate 				if (err != Z_OK)
33457c478bd9Sstevel@tonic-gate 					zone_perror(pt_to_str(prop_type), err,
3346bbec428eSgjelinek 					    B_TRUE);
33477c478bd9Sstevel@tonic-gate 			}
33487c478bd9Sstevel@tonic-gate 		}
33497c478bd9Sstevel@tonic-gate 		return;
33507c478bd9Sstevel@tonic-gate 	case RT_RCTL:
33517c478bd9Sstevel@tonic-gate 		if (prop_type != PT_VALUE) {
33527c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
3353bbec428eSgjelinek 			    B_TRUE);
3354bbec428eSgjelinek 			long_usage(CMD_REMOVE, B_TRUE);
3355bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
33567c478bd9Sstevel@tonic-gate 			return;
33577c478bd9Sstevel@tonic-gate 		}
33587c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[0];
33597c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_COMPLEX) {
33607c478bd9Sstevel@tonic-gate 			zerr(gettext("A %s value was expected here."),
33617c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_COMPLEX));
3362bbec428eSgjelinek 			saw_error = B_TRUE;
33637c478bd9Sstevel@tonic-gate 			return;
33647c478bd9Sstevel@tonic-gate 		}
33657c478bd9Sstevel@tonic-gate 		if ((rctlvaltab = alloc_rctlvaltab()) == NULL) {
3366bbec428eSgjelinek 			zone_perror(zone, Z_NOMEM, B_TRUE);
33677c478bd9Sstevel@tonic-gate 			exit(Z_ERR);
33687c478bd9Sstevel@tonic-gate 		}
33697c478bd9Sstevel@tonic-gate 		for (cx = pp->pv_complex; cx != NULL; cx = cx->cp_next) {
33707c478bd9Sstevel@tonic-gate 			switch (cx->cp_type) {
33717c478bd9Sstevel@tonic-gate 			case PT_PRIV:
33727c478bd9Sstevel@tonic-gate 				(void) strlcpy(rctlvaltab->zone_rctlval_priv,
33737c478bd9Sstevel@tonic-gate 				    cx->cp_value,
33747c478bd9Sstevel@tonic-gate 				    sizeof (rctlvaltab->zone_rctlval_priv));
33757c478bd9Sstevel@tonic-gate 				break;
33767c478bd9Sstevel@tonic-gate 			case PT_LIMIT:
33777c478bd9Sstevel@tonic-gate 				(void) strlcpy(rctlvaltab->zone_rctlval_limit,
33787c478bd9Sstevel@tonic-gate 				    cx->cp_value,
33797c478bd9Sstevel@tonic-gate 				    sizeof (rctlvaltab->zone_rctlval_limit));
33807c478bd9Sstevel@tonic-gate 				break;
33817c478bd9Sstevel@tonic-gate 			case PT_ACTION:
33827c478bd9Sstevel@tonic-gate 				(void) strlcpy(rctlvaltab->zone_rctlval_action,
33837c478bd9Sstevel@tonic-gate 				    cx->cp_value,
33847c478bd9Sstevel@tonic-gate 				    sizeof (rctlvaltab->zone_rctlval_action));
33857c478bd9Sstevel@tonic-gate 				break;
33867c478bd9Sstevel@tonic-gate 			default:
33877c478bd9Sstevel@tonic-gate 				zone_perror(pt_to_str(prop_type),
3388bbec428eSgjelinek 				    Z_NO_PROPERTY_TYPE, B_TRUE);
3389bbec428eSgjelinek 				long_usage(CMD_ADD, B_TRUE);
3390bbec428eSgjelinek 				usage(B_FALSE, HELP_PROPS);
33917c478bd9Sstevel@tonic-gate 				zonecfg_free_rctl_value_list(rctlvaltab);
33927c478bd9Sstevel@tonic-gate 				return;
33937c478bd9Sstevel@tonic-gate 			}
33947c478bd9Sstevel@tonic-gate 		}
33957c478bd9Sstevel@tonic-gate 		rctlvaltab->zone_rctlval_next = NULL;
33967c478bd9Sstevel@tonic-gate 		err = zonecfg_remove_rctl_value(&in_progress_rctltab,
33977c478bd9Sstevel@tonic-gate 		    rctlvaltab);
33987c478bd9Sstevel@tonic-gate 		if (err != Z_OK)
3399bbec428eSgjelinek 			zone_perror(pt_to_str(prop_type), err, B_TRUE);
34007c478bd9Sstevel@tonic-gate 		zonecfg_free_rctl_value_list(rctlvaltab);
34017c478bd9Sstevel@tonic-gate 		return;
3402de860bd9Sgfaden 	case RT_NET:
3403de860bd9Sgfaden 		if (prop_type != PT_DEFROUTER) {
3404de860bd9Sgfaden 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
3405bbec428eSgjelinek 			    B_TRUE);
3406bbec428eSgjelinek 			long_usage(CMD_REMOVE, B_TRUE);
3407bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
3408de860bd9Sgfaden 			return;
3409de860bd9Sgfaden 		} else {
3410de860bd9Sgfaden 			bzero(&in_progress_nwiftab.zone_nwif_defrouter,
3411de860bd9Sgfaden 			    sizeof (in_progress_nwiftab.zone_nwif_defrouter));
3412de860bd9Sgfaden 			return;
3413de860bd9Sgfaden 		}
34147c478bd9Sstevel@tonic-gate 	default:
3415bbec428eSgjelinek 		zone_perror(rt_to_str(res_type), Z_NO_RESOURCE_TYPE, B_TRUE);
3416bbec428eSgjelinek 		long_usage(CMD_REMOVE, B_TRUE);
3417bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
34187c478bd9Sstevel@tonic-gate 		return;
34197c478bd9Sstevel@tonic-gate 	}
34207c478bd9Sstevel@tonic-gate }
34217c478bd9Sstevel@tonic-gate 
34227c478bd9Sstevel@tonic-gate void
34237c478bd9Sstevel@tonic-gate remove_func(cmd_t *cmd)
34247c478bd9Sstevel@tonic-gate {
34257c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_REMOVE))
34267c478bd9Sstevel@tonic-gate 		return;
34277c478bd9Sstevel@tonic-gate 
34287c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
34297c478bd9Sstevel@tonic-gate 
34300209230bSgjelinek 	if (global_scope) {
34310209230bSgjelinek 		if (gz_invalid_resource(cmd->cmd_res_type)) {
34320209230bSgjelinek 			zerr(gettext("%s is not a valid resource for the "
34330209230bSgjelinek 			    "global zone."), rt_to_str(cmd->cmd_res_type));
3434bbec428eSgjelinek 			saw_error = B_TRUE;
34350209230bSgjelinek 			return;
34360209230bSgjelinek 		}
34377c478bd9Sstevel@tonic-gate 		remove_resource(cmd);
34380209230bSgjelinek 	} else {
34397c478bd9Sstevel@tonic-gate 		remove_property(cmd);
34407c478bd9Sstevel@tonic-gate 	}
34410209230bSgjelinek }
34420209230bSgjelinek 
34430209230bSgjelinek static void
34440209230bSgjelinek clear_property(cmd_t *cmd)
34450209230bSgjelinek {
34460209230bSgjelinek 	int res_type, prop_type;
34470209230bSgjelinek 
34480209230bSgjelinek 	res_type = resource_scope;
34490209230bSgjelinek 	prop_type = cmd->cmd_res_type;
34500209230bSgjelinek 	if (res_type == RT_UNKNOWN || prop_type == PT_UNKNOWN) {
3451bbec428eSgjelinek 		long_usage(CMD_CLEAR, B_TRUE);
34520209230bSgjelinek 		return;
34530209230bSgjelinek 	}
34540209230bSgjelinek 
3455bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
34560209230bSgjelinek 		return;
34570209230bSgjelinek 
34580209230bSgjelinek 	switch (res_type) {
34590209230bSgjelinek 	case RT_FS:
34600209230bSgjelinek 		if (prop_type == PT_RAW) {
34610209230bSgjelinek 			in_progress_fstab.zone_fs_raw[0] = '\0';
3462bbec428eSgjelinek 			need_to_commit = B_TRUE;
34630209230bSgjelinek 			return;
34640209230bSgjelinek 		}
34650209230bSgjelinek 		break;
34660209230bSgjelinek 	case RT_DCPU:
34670209230bSgjelinek 		if (prop_type == PT_IMPORTANCE) {
34680209230bSgjelinek 			in_progress_psettab.zone_importance[0] = '\0';
3469bbec428eSgjelinek 			need_to_commit = B_TRUE;
34700209230bSgjelinek 			return;
34710209230bSgjelinek 		}
34720209230bSgjelinek 		break;
34730209230bSgjelinek 	case RT_MCAP:
34740209230bSgjelinek 		switch (prop_type) {
34750209230bSgjelinek 		case PT_PHYSICAL:
34760209230bSgjelinek 			in_progress_mcaptab.zone_physmem_cap[0] = '\0';
3477bbec428eSgjelinek 			need_to_commit = B_TRUE;
34780209230bSgjelinek 			return;
34790209230bSgjelinek 		case PT_SWAP:
34800209230bSgjelinek 			remove_aliased_rctl(PT_SWAP, ALIAS_MAXSWAP);
34810209230bSgjelinek 			return;
34820209230bSgjelinek 		case PT_LOCKED:
34830209230bSgjelinek 			remove_aliased_rctl(PT_LOCKED, ALIAS_MAXLOCKEDMEM);
34840209230bSgjelinek 			return;
34850209230bSgjelinek 		}
34860209230bSgjelinek 		break;
34870209230bSgjelinek 	default:
34880209230bSgjelinek 		break;
34890209230bSgjelinek 	}
34900209230bSgjelinek 
3491bbec428eSgjelinek 	zone_perror(pt_to_str(prop_type), Z_CLEAR_DISALLOW, B_TRUE);
34920209230bSgjelinek }
34930209230bSgjelinek 
34940209230bSgjelinek static void
34950209230bSgjelinek clear_global(cmd_t *cmd)
34960209230bSgjelinek {
34970209230bSgjelinek 	int err, type;
34980209230bSgjelinek 
34990209230bSgjelinek 	if ((type = cmd->cmd_res_type) == RT_UNKNOWN) {
3500bbec428eSgjelinek 		long_usage(CMD_CLEAR, B_TRUE);
35010209230bSgjelinek 		return;
35020209230bSgjelinek 	}
35030209230bSgjelinek 
3504bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
35050209230bSgjelinek 		return;
35060209230bSgjelinek 
35070209230bSgjelinek 	switch (type) {
35080209230bSgjelinek 	case PT_ZONENAME:
35090209230bSgjelinek 		/* FALLTHRU */
35100209230bSgjelinek 	case PT_ZONEPATH:
35110209230bSgjelinek 		/* FALLTHRU */
35120209230bSgjelinek 	case PT_BRAND:
3513bbec428eSgjelinek 		zone_perror(pt_to_str(type), Z_CLEAR_DISALLOW, B_TRUE);
35140209230bSgjelinek 		return;
35150209230bSgjelinek 	case PT_AUTOBOOT:
35160209230bSgjelinek 		/* false is default; we'll treat as equivalent to clearing */
35170209230bSgjelinek 		if ((err = zonecfg_set_autoboot(handle, B_FALSE)) != Z_OK)
3518bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_AUTOBOOT, err, B_TRUE);
35190209230bSgjelinek 		else
3520bbec428eSgjelinek 			need_to_commit = B_TRUE;
35210209230bSgjelinek 		return;
35220209230bSgjelinek 	case PT_POOL:
35230209230bSgjelinek 		if ((err = zonecfg_set_pool(handle, NULL)) != Z_OK)
3524bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_POOL, err, B_TRUE);
35250209230bSgjelinek 		else
3526bbec428eSgjelinek 			need_to_commit = B_TRUE;
35270209230bSgjelinek 		return;
35280209230bSgjelinek 	case PT_LIMITPRIV:
35290209230bSgjelinek 		if ((err = zonecfg_set_limitpriv(handle, NULL)) != Z_OK)
3530bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_LIMITPRIV, err, B_TRUE);
35310209230bSgjelinek 		else
3532bbec428eSgjelinek 			need_to_commit = B_TRUE;
35330209230bSgjelinek 		return;
35340209230bSgjelinek 	case PT_BOOTARGS:
35350209230bSgjelinek 		if ((err = zonecfg_set_bootargs(handle, NULL)) != Z_OK)
3536bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_BOOTARGS, err, B_TRUE);
35370209230bSgjelinek 		else
3538bbec428eSgjelinek 			need_to_commit = B_TRUE;
35390209230bSgjelinek 		return;
35400209230bSgjelinek 	case PT_SCHED:
35410209230bSgjelinek 		if ((err = zonecfg_set_sched(handle, NULL)) != Z_OK)
3542bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_SCHED, err, B_TRUE);
35430209230bSgjelinek 		else
3544bbec428eSgjelinek 			need_to_commit = B_TRUE;
35450209230bSgjelinek 		return;
3546f4b3ec61Sdh155122 	case PT_IPTYPE:
3547f4b3ec61Sdh155122 		/* shared is default; we'll treat as equivalent to clearing */
3548f4b3ec61Sdh155122 		if ((err = zonecfg_set_iptype(handle, ZS_SHARED)) != Z_OK)
3549bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_IPTYPE, err, B_TRUE);
3550f4b3ec61Sdh155122 		else
3551bbec428eSgjelinek 			need_to_commit = B_TRUE;
3552f4b3ec61Sdh155122 		return;
35530209230bSgjelinek 	case PT_MAXLWPS:
35540209230bSgjelinek 		remove_aliased_rctl(PT_MAXLWPS, ALIAS_MAXLWPS);
35550209230bSgjelinek 		return;
3556ff19e029SMenno Lageman 	case PT_MAXPROCS:
3557ff19e029SMenno Lageman 		remove_aliased_rctl(PT_MAXPROCS, ALIAS_MAXPROCS);
3558ff19e029SMenno Lageman 		return;
35590209230bSgjelinek 	case PT_MAXSHMMEM:
35600209230bSgjelinek 		remove_aliased_rctl(PT_MAXSHMMEM, ALIAS_MAXSHMMEM);
35610209230bSgjelinek 		return;
35620209230bSgjelinek 	case PT_MAXSHMIDS:
35630209230bSgjelinek 		remove_aliased_rctl(PT_MAXSHMIDS, ALIAS_MAXSHMIDS);
35640209230bSgjelinek 		return;
35650209230bSgjelinek 	case PT_MAXMSGIDS:
35660209230bSgjelinek 		remove_aliased_rctl(PT_MAXMSGIDS, ALIAS_MAXMSGIDS);
35670209230bSgjelinek 		return;
35680209230bSgjelinek 	case PT_MAXSEMIDS:
35690209230bSgjelinek 		remove_aliased_rctl(PT_MAXSEMIDS, ALIAS_MAXSEMIDS);
35700209230bSgjelinek 		return;
35710209230bSgjelinek 	case PT_SHARES:
35720209230bSgjelinek 		remove_aliased_rctl(PT_SHARES, ALIAS_SHARES);
35730209230bSgjelinek 		return;
35745679c89fSjv227347 	case PT_HOSTID:
35755679c89fSjv227347 		if ((err = zonecfg_set_hostid(handle, NULL)) != Z_OK)
35765679c89fSjv227347 			z_cmd_rt_perror(CMD_CLEAR, RT_HOSTID, err, B_TRUE);
35775679c89fSjv227347 		else
35785679c89fSjv227347 			need_to_commit = B_TRUE;
35795679c89fSjv227347 		return;
35800fbb751dSJohn Levon 	case PT_FS_ALLOWED:
35810fbb751dSJohn Levon 		if ((err = zonecfg_set_fs_allowed(handle, NULL)) != Z_OK)
35820fbb751dSJohn Levon 			z_cmd_rt_perror(CMD_CLEAR, RT_FS_ALLOWED, err, B_TRUE);
35830fbb751dSJohn Levon 		else
35840fbb751dSJohn Levon 			need_to_commit = B_TRUE;
35850fbb751dSJohn Levon 		return;
35860209230bSgjelinek 	default:
3587bbec428eSgjelinek 		zone_perror(pt_to_str(type), Z_NO_PROPERTY_TYPE, B_TRUE);
3588bbec428eSgjelinek 		long_usage(CMD_CLEAR, B_TRUE);
3589bbec428eSgjelinek 		usage(B_FALSE, HELP_PROPS);
35900209230bSgjelinek 		return;
35910209230bSgjelinek 	}
35920209230bSgjelinek }
35930209230bSgjelinek 
35940209230bSgjelinek void
35950209230bSgjelinek clear_func(cmd_t *cmd)
35960209230bSgjelinek {
35970209230bSgjelinek 	if (zone_is_read_only(CMD_CLEAR))
35980209230bSgjelinek 		return;
35990209230bSgjelinek 
36000209230bSgjelinek 	assert(cmd != NULL);
36010209230bSgjelinek 
36020209230bSgjelinek 	if (global_scope) {
36030209230bSgjelinek 		if (gz_invalid_property(cmd->cmd_res_type)) {
36040209230bSgjelinek 			zerr(gettext("%s is not a valid property for the "
36050209230bSgjelinek 			    "global zone."), pt_to_str(cmd->cmd_res_type));
3606bbec428eSgjelinek 			saw_error = B_TRUE;
36070209230bSgjelinek 			return;
36080209230bSgjelinek 		}
36090209230bSgjelinek 
36100209230bSgjelinek 		clear_global(cmd);
36110209230bSgjelinek 	} else {
36120209230bSgjelinek 		clear_property(cmd);
36130209230bSgjelinek 	}
36140209230bSgjelinek }
36157c478bd9Sstevel@tonic-gate 
36167c478bd9Sstevel@tonic-gate void
36177c478bd9Sstevel@tonic-gate select_func(cmd_t *cmd)
36187c478bd9Sstevel@tonic-gate {
36190209230bSgjelinek 	int type, err, res;
36200209230bSgjelinek 	uint64_t limit;
3621c97ad5cdSakolb 	uint64_t tmp;
36227c478bd9Sstevel@tonic-gate 
36237c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_SELECT))
36247c478bd9Sstevel@tonic-gate 		return;
36257c478bd9Sstevel@tonic-gate 
36267c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
36277c478bd9Sstevel@tonic-gate 
36287c478bd9Sstevel@tonic-gate 	if (global_scope) {
3629bbec428eSgjelinek 		global_scope = B_FALSE;
36307c478bd9Sstevel@tonic-gate 		resource_scope = cmd->cmd_res_type;
36317c478bd9Sstevel@tonic-gate 		end_op = CMD_SELECT;
36327c478bd9Sstevel@tonic-gate 	} else {
36337c478bd9Sstevel@tonic-gate 		scope_usage(CMD_SELECT);
36347c478bd9Sstevel@tonic-gate 		return;
36357c478bd9Sstevel@tonic-gate 	}
36367c478bd9Sstevel@tonic-gate 
36377c478bd9Sstevel@tonic-gate 	if ((type = cmd->cmd_res_type) == RT_UNKNOWN) {
3638bbec428eSgjelinek 		long_usage(CMD_SELECT, B_TRUE);
36397c478bd9Sstevel@tonic-gate 		return;
36407c478bd9Sstevel@tonic-gate 	}
36417c478bd9Sstevel@tonic-gate 
3642bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
36437c478bd9Sstevel@tonic-gate 		return;
36447c478bd9Sstevel@tonic-gate 
36457c478bd9Sstevel@tonic-gate 	switch (type) {
36467c478bd9Sstevel@tonic-gate 	case RT_FS:
3647bbec428eSgjelinek 		if ((err = fill_in_fstab(cmd, &old_fstab, B_FALSE)) != Z_OK) {
3648bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_FS, err, B_TRUE);
3649bbec428eSgjelinek 			global_scope = B_TRUE;
36507c478bd9Sstevel@tonic-gate 		}
36517c478bd9Sstevel@tonic-gate 		bcopy(&old_fstab, &in_progress_fstab,
36527c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_fstab));
36537c478bd9Sstevel@tonic-gate 		return;
36547c478bd9Sstevel@tonic-gate 	case RT_NET:
3655bbec428eSgjelinek 		if ((err = fill_in_nwiftab(cmd, &old_nwiftab, B_FALSE))
3656bbec428eSgjelinek 		    != Z_OK) {
3657bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_NET, err, B_TRUE);
3658bbec428eSgjelinek 			global_scope = B_TRUE;
36597c478bd9Sstevel@tonic-gate 		}
36607c478bd9Sstevel@tonic-gate 		bcopy(&old_nwiftab, &in_progress_nwiftab,
36617c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_nwiftab));
36627c478bd9Sstevel@tonic-gate 		return;
36637c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
3664bbec428eSgjelinek 		if ((err = fill_in_devtab(cmd, &old_devtab, B_FALSE)) != Z_OK) {
3665bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_DEVICE, err, B_TRUE);
3666bbec428eSgjelinek 			global_scope = B_TRUE;
36677c478bd9Sstevel@tonic-gate 		}
36687c478bd9Sstevel@tonic-gate 		bcopy(&old_devtab, &in_progress_devtab,
36697c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_devtab));
36707c478bd9Sstevel@tonic-gate 		return;
36717c478bd9Sstevel@tonic-gate 	case RT_RCTL:
3672bbec428eSgjelinek 		if ((err = fill_in_rctltab(cmd, &old_rctltab, B_FALSE))
3673bbec428eSgjelinek 		    != Z_OK) {
3674bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_RCTL, err, B_TRUE);
3675bbec428eSgjelinek 			global_scope = B_TRUE;
36767c478bd9Sstevel@tonic-gate 		}
36777c478bd9Sstevel@tonic-gate 		bcopy(&old_rctltab, &in_progress_rctltab,
36787c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_rctltab));
36797c478bd9Sstevel@tonic-gate 		return;
36807c478bd9Sstevel@tonic-gate 	case RT_ATTR:
3681bbec428eSgjelinek 		if ((err = fill_in_attrtab(cmd, &old_attrtab, B_FALSE))
3682bbec428eSgjelinek 		    != Z_OK) {
3683bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_ATTR, err, B_TRUE);
3684bbec428eSgjelinek 			global_scope = B_TRUE;
36857c478bd9Sstevel@tonic-gate 		}
36867c478bd9Sstevel@tonic-gate 		bcopy(&old_attrtab, &in_progress_attrtab,
36877c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_attrtab));
36887c478bd9Sstevel@tonic-gate 		return;
3689fa9e4066Sahrens 	case RT_DATASET:
3690bbec428eSgjelinek 		if ((err = fill_in_dstab(cmd, &old_dstab, B_FALSE)) != Z_OK) {
3691bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_DATASET, err, B_TRUE);
3692bbec428eSgjelinek 			global_scope = B_TRUE;
3693fa9e4066Sahrens 		}
3694fa9e4066Sahrens 		bcopy(&old_dstab, &in_progress_dstab,
3695fa9e4066Sahrens 		    sizeof (struct zone_dstab));
3696fa9e4066Sahrens 		return;
36970209230bSgjelinek 	case RT_DCPU:
36980209230bSgjelinek 		if ((err = zonecfg_lookup_pset(handle, &old_psettab)) != Z_OK) {
3699bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_DCPU, err, B_TRUE);
3700bbec428eSgjelinek 			global_scope = B_TRUE;
37010209230bSgjelinek 		}
37020209230bSgjelinek 		bcopy(&old_psettab, &in_progress_psettab,
37030209230bSgjelinek 		    sizeof (struct zone_psettab));
37040209230bSgjelinek 		return;
3705c97ad5cdSakolb 	case RT_PCAP:
3706c97ad5cdSakolb 		if ((err = zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &tmp))
3707c97ad5cdSakolb 		    != Z_OK) {
3708bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_PCAP, err, B_TRUE);
3709bbec428eSgjelinek 			global_scope = B_TRUE;
3710c97ad5cdSakolb 		}
3711c97ad5cdSakolb 		return;
37120209230bSgjelinek 	case RT_MCAP:
37130209230bSgjelinek 		/* if none of these exist, there is no resource to select */
37140209230bSgjelinek 		if ((res = zonecfg_lookup_mcap(handle, &old_mcaptab)) != Z_OK &&
37150209230bSgjelinek 		    zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP, &limit)
37160209230bSgjelinek 		    != Z_OK &&
37170209230bSgjelinek 		    zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM, &limit)
37180209230bSgjelinek 		    != Z_OK) {
37190209230bSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_MCAP, Z_NO_RESOURCE_TYPE,
3720bbec428eSgjelinek 			    B_TRUE);
3721bbec428eSgjelinek 			global_scope = B_TRUE;
37220209230bSgjelinek 		}
37230209230bSgjelinek 		if (res == Z_OK)
37240209230bSgjelinek 			bcopy(&old_mcaptab, &in_progress_mcaptab,
37250209230bSgjelinek 			    sizeof (struct zone_mcaptab));
37260209230bSgjelinek 		else
37270209230bSgjelinek 			bzero(&in_progress_mcaptab,
37280209230bSgjelinek 			    sizeof (in_progress_mcaptab));
37290209230bSgjelinek 		return;
3730cb8a054bSGlenn Faden 	case RT_ADMIN:
3731cb8a054bSGlenn Faden 		if ((err = fill_in_admintab(cmd, &old_admintab, B_FALSE))
3732cb8a054bSGlenn Faden 		    != Z_OK) {
3733cb8a054bSGlenn Faden 			z_cmd_rt_perror(CMD_SELECT, RT_ADMIN, err,
3734cb8a054bSGlenn Faden 			    B_TRUE);
3735cb8a054bSGlenn Faden 			global_scope = B_TRUE;
3736cb8a054bSGlenn Faden 		}
3737cb8a054bSGlenn Faden 		bcopy(&old_admintab, &in_progress_admintab,
3738cb8a054bSGlenn Faden 		    sizeof (struct zone_admintab));
3739cb8a054bSGlenn Faden 		return;
37407c478bd9Sstevel@tonic-gate 	default:
3741bbec428eSgjelinek 		zone_perror(rt_to_str(type), Z_NO_RESOURCE_TYPE, B_TRUE);
3742bbec428eSgjelinek 		long_usage(CMD_SELECT, B_TRUE);
3743bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
37447c478bd9Sstevel@tonic-gate 		return;
37457c478bd9Sstevel@tonic-gate 	}
37467c478bd9Sstevel@tonic-gate }
37477c478bd9Sstevel@tonic-gate 
37487c478bd9Sstevel@tonic-gate /*
37497c478bd9Sstevel@tonic-gate  * Network "addresses" can be one of the following forms:
37507c478bd9Sstevel@tonic-gate  *	<IPv4 address>
37517c478bd9Sstevel@tonic-gate  *	<IPv4 address>/<prefix length>
37527c478bd9Sstevel@tonic-gate  *	<IPv6 address>/<prefix length>
37537c478bd9Sstevel@tonic-gate  *	<host name>
37547c478bd9Sstevel@tonic-gate  *	<host name>/<prefix length>
37557c478bd9Sstevel@tonic-gate  * In other words, the "/" followed by a prefix length is allowed but not
37567c478bd9Sstevel@tonic-gate  * required for IPv4 addresses and host names, and required for IPv6 addresses.
37577c478bd9Sstevel@tonic-gate  * If a prefix length is given, it must be in the allowable range: 0 to 32 for
37587c478bd9Sstevel@tonic-gate  * IPv4 addresses and host names, 0 to 128 for IPv6 addresses.
37597c478bd9Sstevel@tonic-gate  * Host names must start with an alpha-numeric character, and all subsequent
37607c478bd9Sstevel@tonic-gate  * characters must be either alpha-numeric or "-".
37617c478bd9Sstevel@tonic-gate  */
37627c478bd9Sstevel@tonic-gate 
37637c478bd9Sstevel@tonic-gate static int
37647c478bd9Sstevel@tonic-gate validate_net_address_syntax(char *address)
37657c478bd9Sstevel@tonic-gate {
37667c478bd9Sstevel@tonic-gate 	char *slashp, part1[MAXHOSTNAMELEN];
37677c478bd9Sstevel@tonic-gate 	struct in6_addr in6;
37687c478bd9Sstevel@tonic-gate 	struct in_addr in4;
37697c478bd9Sstevel@tonic-gate 	int prefixlen, i;
37707c478bd9Sstevel@tonic-gate 
37717c478bd9Sstevel@tonic-gate 	/*
37727c478bd9Sstevel@tonic-gate 	 * Copy the part before any '/' into part1 or copy the whole
37737c478bd9Sstevel@tonic-gate 	 * thing if there is no '/'.
37747c478bd9Sstevel@tonic-gate 	 */
37757c478bd9Sstevel@tonic-gate 	if ((slashp = strchr(address, '/')) != NULL) {
37767c478bd9Sstevel@tonic-gate 		*slashp = '\0';
37777c478bd9Sstevel@tonic-gate 		(void) strlcpy(part1, address, sizeof (part1));
37787c478bd9Sstevel@tonic-gate 		*slashp = '/';
37797c478bd9Sstevel@tonic-gate 		prefixlen = atoi(++slashp);
37807c478bd9Sstevel@tonic-gate 	} else {
37817c478bd9Sstevel@tonic-gate 		(void) strlcpy(part1, address, sizeof (part1));
37827c478bd9Sstevel@tonic-gate 	}
37837c478bd9Sstevel@tonic-gate 
37847c478bd9Sstevel@tonic-gate 	if (inet_pton(AF_INET6, part1, &in6) == 1) {
37857c478bd9Sstevel@tonic-gate 		if (slashp == NULL) {
37867c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: IPv6 addresses "
37877c478bd9Sstevel@tonic-gate 			    "require /prefix-length suffix."), address);
37887c478bd9Sstevel@tonic-gate 			return (Z_ERR);
37897c478bd9Sstevel@tonic-gate 		}
37907c478bd9Sstevel@tonic-gate 		if (prefixlen < 0 || prefixlen > 128) {
37917c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: IPv6 address "
37927c478bd9Sstevel@tonic-gate 			    "prefix lengths must be 0 - 128."), address);
37937c478bd9Sstevel@tonic-gate 			return (Z_ERR);
37947c478bd9Sstevel@tonic-gate 		}
37957c478bd9Sstevel@tonic-gate 		return (Z_OK);
37967c478bd9Sstevel@tonic-gate 	}
37977c478bd9Sstevel@tonic-gate 
37987c478bd9Sstevel@tonic-gate 	/* At this point, any /prefix must be for IPv4. */
37997c478bd9Sstevel@tonic-gate 	if (slashp != NULL) {
38007c478bd9Sstevel@tonic-gate 		if (prefixlen < 0 || prefixlen > 32) {
38017c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: IPv4 address "
38027c478bd9Sstevel@tonic-gate 			    "prefix lengths must be 0 - 32."), address);
38037c478bd9Sstevel@tonic-gate 			return (Z_ERR);
38047c478bd9Sstevel@tonic-gate 		}
38057c478bd9Sstevel@tonic-gate 	}
38067c478bd9Sstevel@tonic-gate 	if (inet_pton(AF_INET, part1, &in4) == 1)
38077c478bd9Sstevel@tonic-gate 		return (Z_OK);
38087c478bd9Sstevel@tonic-gate 
38097c478bd9Sstevel@tonic-gate 	/* address may also be a host name */
38107c478bd9Sstevel@tonic-gate 	if (!isalnum(part1[0])) {
38117c478bd9Sstevel@tonic-gate 		zerr(gettext("%s: bogus host name or network address syntax"),
38127c478bd9Sstevel@tonic-gate 		    part1);
3813bbec428eSgjelinek 		saw_error = B_TRUE;
3814bbec428eSgjelinek 		usage(B_FALSE, HELP_NETADDR);
38157c478bd9Sstevel@tonic-gate 		return (Z_ERR);
38167c478bd9Sstevel@tonic-gate 	}
38177c478bd9Sstevel@tonic-gate 	for (i = 1; part1[i]; i++)
38187c478bd9Sstevel@tonic-gate 		if (!isalnum(part1[i]) && part1[i] != '-' && part1[i] != '.') {
38197c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: bogus host name or "
38207c478bd9Sstevel@tonic-gate 			    "network address syntax"), part1);
3821bbec428eSgjelinek 			saw_error = B_TRUE;
3822bbec428eSgjelinek 			usage(B_FALSE, HELP_NETADDR);
38237c478bd9Sstevel@tonic-gate 			return (Z_ERR);
38247c478bd9Sstevel@tonic-gate 		}
38257c478bd9Sstevel@tonic-gate 	return (Z_OK);
38267c478bd9Sstevel@tonic-gate }
38277c478bd9Sstevel@tonic-gate 
38287c478bd9Sstevel@tonic-gate static int
3829c9f134eaSjv227347 validate_net_physical_syntax(const char *ifname)
38307c478bd9Sstevel@tonic-gate {
3831c9f134eaSjv227347 	ifspec_t ifnameprop;
3832c9f134eaSjv227347 	zone_iptype_t iptype;
3833c9f134eaSjv227347 
383437b210dcSjv227347 	if (zonecfg_get_iptype(handle, &iptype) != Z_OK) {
3835c9f134eaSjv227347 		zerr(gettext("zone configuration has an invalid or nonexistent "
3836c9f134eaSjv227347 		    "ip-type property"));
38377c478bd9Sstevel@tonic-gate 		return (Z_ERR);
38387c478bd9Sstevel@tonic-gate 	}
3839c9f134eaSjv227347 	switch (iptype) {
3840c9f134eaSjv227347 	case ZS_SHARED:
3841c9f134eaSjv227347 		if (ifparse_ifspec(ifname, &ifnameprop) == B_FALSE) {
3842c9f134eaSjv227347 			zerr(gettext("%s: invalid physical interface name"),
3843c9f134eaSjv227347 			    ifname);
3844c9f134eaSjv227347 			return (Z_ERR);
3845c9f134eaSjv227347 		}
3846c9f134eaSjv227347 		if (ifnameprop.ifsp_lunvalid) {
3847c9f134eaSjv227347 			zerr(gettext("%s: LUNs not allowed in physical "
3848c9f134eaSjv227347 			    "interface names"), ifname);
3849c9f134eaSjv227347 			return (Z_ERR);
3850c9f134eaSjv227347 		}
3851c9f134eaSjv227347 		break;
3852c9f134eaSjv227347 	case ZS_EXCLUSIVE:
3853c9f134eaSjv227347 		if (dladm_valid_linkname(ifname) == B_FALSE) {
3854c9f134eaSjv227347 			if (strchr(ifname, ':') != NULL)
3855c9f134eaSjv227347 				zerr(gettext("%s: physical interface name "
3856c9f134eaSjv227347 				    "required; logical interface name not "
3857c9f134eaSjv227347 				    "allowed"), ifname);
3858c9f134eaSjv227347 			else
3859c9f134eaSjv227347 				zerr(gettext("%s: invalid physical interface "
3860c9f134eaSjv227347 				    "name"), ifname);
3861c9f134eaSjv227347 			return (Z_ERR);
3862c9f134eaSjv227347 		}
3863c9f134eaSjv227347 		break;
3864c9f134eaSjv227347 	}
3865c9f134eaSjv227347 	return (Z_OK);
3866c9f134eaSjv227347 }
38677c478bd9Sstevel@tonic-gate 
38687c478bd9Sstevel@tonic-gate static boolean_t
38697c478bd9Sstevel@tonic-gate valid_fs_type(const char *type)
38707c478bd9Sstevel@tonic-gate {
38717c478bd9Sstevel@tonic-gate 	/*
38727c478bd9Sstevel@tonic-gate 	 * Is this a valid path component?
38737c478bd9Sstevel@tonic-gate 	 */
38747c478bd9Sstevel@tonic-gate 	if (strlen(type) + 1 > MAXNAMELEN)
38757c478bd9Sstevel@tonic-gate 		return (B_FALSE);
38767c478bd9Sstevel@tonic-gate 	/*
38777c478bd9Sstevel@tonic-gate 	 * Make sure a bad value for "type" doesn't make
38787c478bd9Sstevel@tonic-gate 	 * /usr/lib/fs/<type>/mount turn into something else.
38797c478bd9Sstevel@tonic-gate 	 */
38807c478bd9Sstevel@tonic-gate 	if (strchr(type, '/') != NULL || type[0] == '\0' ||
38817c478bd9Sstevel@tonic-gate 	    strcmp(type, ".") == 0 || strcmp(type, "..") == 0)
38827c478bd9Sstevel@tonic-gate 		return (B_FALSE);
38837c478bd9Sstevel@tonic-gate 	/*
38847c478bd9Sstevel@tonic-gate 	 * More detailed verification happens later by zoneadm(1m).
38857c478bd9Sstevel@tonic-gate 	 */
38867c478bd9Sstevel@tonic-gate 	return (B_TRUE);
38877c478bd9Sstevel@tonic-gate }
38887c478bd9Sstevel@tonic-gate 
3889f4b3ec61Sdh155122 static boolean_t
3890f4b3ec61Sdh155122 allow_exclusive()
3891f4b3ec61Sdh155122 {
3892f4b3ec61Sdh155122 	brand_handle_t	bh;
3893f4b3ec61Sdh155122 	char		brand[MAXNAMELEN];
3894f4b3ec61Sdh155122 	boolean_t	ret;
3895f4b3ec61Sdh155122 
3896f4b3ec61Sdh155122 	if (zonecfg_get_brand(handle, brand, sizeof (brand)) != Z_OK) {
3897f4b3ec61Sdh155122 		zerr("%s: %s\n", zone, gettext("could not get zone brand"));
3898f4b3ec61Sdh155122 		return (B_FALSE);
3899f4b3ec61Sdh155122 	}
3900f4b3ec61Sdh155122 	if ((bh = brand_open(brand)) == NULL) {
3901f4b3ec61Sdh155122 		zerr("%s: %s\n", zone, gettext("unknown brand."));
3902f4b3ec61Sdh155122 		return (B_FALSE);
3903f4b3ec61Sdh155122 	}
3904f4b3ec61Sdh155122 	ret = brand_allow_exclusive_ip(bh);
3905f4b3ec61Sdh155122 	brand_close(bh);
3906f4b3ec61Sdh155122 	if (!ret)
3907f4b3ec61Sdh155122 		zerr(gettext("%s cannot be '%s' when %s is '%s'."),
3908f4b3ec61Sdh155122 		    pt_to_str(PT_IPTYPE), "exclusive",
3909f4b3ec61Sdh155122 		    pt_to_str(PT_BRAND), brand);
3910f4b3ec61Sdh155122 	return (ret);
3911f4b3ec61Sdh155122 }
3912f4b3ec61Sdh155122 
39130209230bSgjelinek static void
39140209230bSgjelinek set_aliased_rctl(char *alias, int prop_type, char *s)
39150209230bSgjelinek {
39160209230bSgjelinek 	uint64_t limit;
39170209230bSgjelinek 	int err;
39180209230bSgjelinek 	char tmp[128];
39190209230bSgjelinek 
39200209230bSgjelinek 	if (global_zone && strcmp(alias, ALIAS_SHARES) != 0)
39210209230bSgjelinek 		zerr(gettext("WARNING: Setting a global zone resource "
39220209230bSgjelinek 		    "control too low could deny\nservice "
39230209230bSgjelinek 		    "to even the root user; "
39240209230bSgjelinek 		    "this could render the system impossible\n"
39250209230bSgjelinek 		    "to administer.  Please use caution."));
39260209230bSgjelinek 
39270209230bSgjelinek 	/* convert memory based properties */
39280209230bSgjelinek 	if (prop_type == PT_MAXSHMMEM) {
39290209230bSgjelinek 		if (!zonecfg_valid_memlimit(s, &limit)) {
39300209230bSgjelinek 			zerr(gettext("A non-negative number with a required "
39310209230bSgjelinek 			    "scale suffix (K, M, G or T) was expected\nhere."));
3932bbec428eSgjelinek 			saw_error = B_TRUE;
39330209230bSgjelinek 			return;
39340209230bSgjelinek 		}
39350209230bSgjelinek 
39360209230bSgjelinek 		(void) snprintf(tmp, sizeof (tmp), "%llu", limit);
39370209230bSgjelinek 		s = tmp;
39380209230bSgjelinek 	}
39390209230bSgjelinek 
39400209230bSgjelinek 	if (!zonecfg_aliased_rctl_ok(handle, alias)) {
3941bbec428eSgjelinek 		zone_perror(pt_to_str(prop_type), Z_ALIAS_DISALLOW, B_FALSE);
3942bbec428eSgjelinek 		saw_error = B_TRUE;
39430209230bSgjelinek 	} else if (!zonecfg_valid_alias_limit(alias, s, &limit)) {
39440209230bSgjelinek 		zerr(gettext("%s property is out of range."),
39450209230bSgjelinek 		    pt_to_str(prop_type));
3946bbec428eSgjelinek 		saw_error = B_TRUE;
39470209230bSgjelinek 	} else if ((err = zonecfg_set_aliased_rctl(handle, alias, limit))
39480209230bSgjelinek 	    != Z_OK) {
3949bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
3950bbec428eSgjelinek 		saw_error = B_TRUE;
39510209230bSgjelinek 	} else {
3952bbec428eSgjelinek 		need_to_commit = B_TRUE;
39530209230bSgjelinek 	}
39540209230bSgjelinek }
39550209230bSgjelinek 
39567c478bd9Sstevel@tonic-gate void
39577c478bd9Sstevel@tonic-gate set_func(cmd_t *cmd)
39587c478bd9Sstevel@tonic-gate {
39597c478bd9Sstevel@tonic-gate 	char *prop_id;
3960555afedfScarlsonj 	int arg, err, res_type, prop_type;
39617c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
39627c478bd9Sstevel@tonic-gate 	boolean_t autoboot;
3963f4b3ec61Sdh155122 	zone_iptype_t iptype;
3964bbec428eSgjelinek 	boolean_t force_set = B_FALSE;
39650209230bSgjelinek 	size_t physmem_size = sizeof (in_progress_mcaptab.zone_physmem_cap);
39660209230bSgjelinek 	uint64_t mem_cap, mem_limit;
3967c97ad5cdSakolb 	float cap;
3968c97ad5cdSakolb 	char *unitp;
39690209230bSgjelinek 	struct zone_psettab tmp_psettab;
3970bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
39717c478bd9Sstevel@tonic-gate 
39727c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_SET))
39737c478bd9Sstevel@tonic-gate 		return;
39747c478bd9Sstevel@tonic-gate 
39757c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
39767c478bd9Sstevel@tonic-gate 
3977555afedfScarlsonj 	optind = opterr = 0;
3978555afedfScarlsonj 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "F")) != EOF) {
3979555afedfScarlsonj 		switch (arg) {
3980555afedfScarlsonj 		case 'F':
3981bbec428eSgjelinek 			force_set = B_TRUE;
3982555afedfScarlsonj 			break;
3983555afedfScarlsonj 		default:
3984555afedfScarlsonj 			if (optopt == '?')
3985555afedfScarlsonj 				longer_usage(CMD_SET);
3986555afedfScarlsonj 			else
3987555afedfScarlsonj 				short_usage(CMD_SET);
3988bbec428eSgjelinek 			arg_err = B_TRUE;
39897ec75eb8Sgjelinek 			break;
39907ec75eb8Sgjelinek 		}
39917ec75eb8Sgjelinek 	}
39927ec75eb8Sgjelinek 	if (arg_err)
3993555afedfScarlsonj 		return;
3994555afedfScarlsonj 
39957c478bd9Sstevel@tonic-gate 	prop_type = cmd->cmd_prop_name[0];
39967c478bd9Sstevel@tonic-gate 	if (global_scope) {
39970209230bSgjelinek 		if (gz_invalid_property(prop_type)) {
39980209230bSgjelinek 			zerr(gettext("%s is not a valid property for the "
39990209230bSgjelinek 			    "global zone."), pt_to_str(prop_type));
4000bbec428eSgjelinek 			saw_error = B_TRUE;
40010209230bSgjelinek 			return;
40020209230bSgjelinek 		}
40030209230bSgjelinek 
4004087719fdSdp 		if (prop_type == PT_ZONENAME) {
4005087719fdSdp 			res_type = RT_ZONENAME;
4006087719fdSdp 		} else if (prop_type == PT_ZONEPATH) {
40077c478bd9Sstevel@tonic-gate 			res_type = RT_ZONEPATH;
40087c478bd9Sstevel@tonic-gate 		} else if (prop_type == PT_AUTOBOOT) {
40097c478bd9Sstevel@tonic-gate 			res_type = RT_AUTOBOOT;
40109acbbeafSnn35248 		} else if (prop_type == PT_BRAND) {
40119acbbeafSnn35248 			res_type = RT_BRAND;
40127c478bd9Sstevel@tonic-gate 		} else if (prop_type == PT_POOL) {
40137c478bd9Sstevel@tonic-gate 			res_type = RT_POOL;
4014ffbafc53Scomay 		} else if (prop_type == PT_LIMITPRIV) {
4015ffbafc53Scomay 			res_type = RT_LIMITPRIV;
40163f2f09c1Sdp 		} else if (prop_type == PT_BOOTARGS) {
40173f2f09c1Sdp 			res_type = RT_BOOTARGS;
40180209230bSgjelinek 		} else if (prop_type == PT_SCHED) {
40190209230bSgjelinek 			res_type = RT_SCHED;
4020f4b3ec61Sdh155122 		} else if (prop_type == PT_IPTYPE) {
4021f4b3ec61Sdh155122 			res_type = RT_IPTYPE;
40220209230bSgjelinek 		} else if (prop_type == PT_MAXLWPS) {
40230209230bSgjelinek 			res_type = RT_MAXLWPS;
4024ff19e029SMenno Lageman 		} else if (prop_type == PT_MAXPROCS) {
4025ff19e029SMenno Lageman 			res_type = RT_MAXPROCS;
40260209230bSgjelinek 		} else if (prop_type == PT_MAXSHMMEM) {
40270209230bSgjelinek 			res_type = RT_MAXSHMMEM;
40280209230bSgjelinek 		} else if (prop_type == PT_MAXSHMIDS) {
40290209230bSgjelinek 			res_type = RT_MAXSHMIDS;
40300209230bSgjelinek 		} else if (prop_type == PT_MAXMSGIDS) {
40310209230bSgjelinek 			res_type = RT_MAXMSGIDS;
40320209230bSgjelinek 		} else if (prop_type == PT_MAXSEMIDS) {
40330209230bSgjelinek 			res_type = RT_MAXSEMIDS;
40340209230bSgjelinek 		} else if (prop_type == PT_SHARES) {
40350209230bSgjelinek 			res_type = RT_SHARES;
40365679c89fSjv227347 		} else if (prop_type == PT_HOSTID) {
40375679c89fSjv227347 			res_type = RT_HOSTID;
40380fbb751dSJohn Levon 		} else if (prop_type == PT_FS_ALLOWED) {
40390fbb751dSJohn Levon 			res_type = RT_FS_ALLOWED;
40407c478bd9Sstevel@tonic-gate 		} else {
40417c478bd9Sstevel@tonic-gate 			zerr(gettext("Cannot set a resource-specific property "
40427c478bd9Sstevel@tonic-gate 			    "from the global scope."));
4043bbec428eSgjelinek 			saw_error = B_TRUE;
40447c478bd9Sstevel@tonic-gate 			return;
40457c478bd9Sstevel@tonic-gate 		}
40467c478bd9Sstevel@tonic-gate 	} else {
40477c478bd9Sstevel@tonic-gate 		res_type = resource_scope;
40487c478bd9Sstevel@tonic-gate 	}
40497c478bd9Sstevel@tonic-gate 
4050555afedfScarlsonj 	if (force_set) {
4051555afedfScarlsonj 		if (res_type != RT_ZONEPATH) {
4052555afedfScarlsonj 			zerr(gettext("Only zonepath setting can be forced."));
4053bbec428eSgjelinek 			saw_error = B_TRUE;
4054555afedfScarlsonj 			return;
4055555afedfScarlsonj 		}
4056555afedfScarlsonj 		if (!zonecfg_in_alt_root()) {
4057555afedfScarlsonj 			zerr(gettext("Zonepath is changeable only in an "
4058555afedfScarlsonj 			    "alternate root."));
4059bbec428eSgjelinek 			saw_error = B_TRUE;
4060555afedfScarlsonj 			return;
4061555afedfScarlsonj 		}
4062555afedfScarlsonj 	}
4063555afedfScarlsonj 
40647c478bd9Sstevel@tonic-gate 	pp = cmd->cmd_property_ptr[0];
40657c478bd9Sstevel@tonic-gate 	/*
40667c478bd9Sstevel@tonic-gate 	 * A nasty expression but not that complicated:
40677c478bd9Sstevel@tonic-gate 	 * 1. fs options are simple or list (tested below)
40687c478bd9Sstevel@tonic-gate 	 * 2. rctl value's are complex or list (tested below)
40697c478bd9Sstevel@tonic-gate 	 * Anything else should be simple.
40707c478bd9Sstevel@tonic-gate 	 */
40717c478bd9Sstevel@tonic-gate 	if (!(res_type == RT_FS && prop_type == PT_OPTIONS) &&
40727c478bd9Sstevel@tonic-gate 	    !(res_type == RT_RCTL && prop_type == PT_VALUE) &&
40737c478bd9Sstevel@tonic-gate 	    (pp->pv_type != PROP_VAL_SIMPLE ||
40747c478bd9Sstevel@tonic-gate 	    (prop_id = pp->pv_simple) == NULL)) {
40757c478bd9Sstevel@tonic-gate 		zerr(gettext("A %s value was expected here."),
40767c478bd9Sstevel@tonic-gate 		    pvt_to_str(PROP_VAL_SIMPLE));
4077bbec428eSgjelinek 		saw_error = B_TRUE;
40787c478bd9Sstevel@tonic-gate 		return;
40797c478bd9Sstevel@tonic-gate 	}
40807c478bd9Sstevel@tonic-gate 	if (prop_type == PT_UNKNOWN) {
4081bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
40827c478bd9Sstevel@tonic-gate 		return;
40837c478bd9Sstevel@tonic-gate 	}
40847c478bd9Sstevel@tonic-gate 
4085087719fdSdp 	/*
4086087719fdSdp 	 * Special case: the user can change the zone name prior to 'create';
4087087719fdSdp 	 * if the zone already exists, we fall through letting initialize()
4088087719fdSdp 	 * and the rest of the logic run.
4089087719fdSdp 	 */
4090bbec428eSgjelinek 	if (res_type == RT_ZONENAME && got_handle == B_FALSE &&
4091087719fdSdp 	    !state_atleast(ZONE_STATE_CONFIGURED)) {
4092fb03efaaSdp 		if ((err = zonecfg_validate_zonename(prop_id)) != Z_OK) {
4093bbec428eSgjelinek 			zone_perror(prop_id, err, B_TRUE);
4094bbec428eSgjelinek 			usage(B_FALSE, HELP_SYNTAX);
4095fb03efaaSdp 			return;
4096fb03efaaSdp 		}
4097087719fdSdp 		(void) strlcpy(zone, prop_id, sizeof (zone));
4098087719fdSdp 		return;
4099087719fdSdp 	}
4100087719fdSdp 
4101bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
41027c478bd9Sstevel@tonic-gate 		return;
41037c478bd9Sstevel@tonic-gate 
41047c478bd9Sstevel@tonic-gate 	switch (res_type) {
4105087719fdSdp 	case RT_ZONENAME:
4106087719fdSdp 		if ((err = zonecfg_set_name(handle, prop_id)) != Z_OK) {
4107087719fdSdp 			/*
4108087719fdSdp 			 * Use prop_id instead of 'zone' here, since we're
4109087719fdSdp 			 * reporting a problem about the *new* zonename.
4110087719fdSdp 			 */
4111bbec428eSgjelinek 			zone_perror(prop_id, err, B_TRUE);
4112bbec428eSgjelinek 			usage(B_FALSE, HELP_SYNTAX);
4113087719fdSdp 		} else {
4114bbec428eSgjelinek 			need_to_commit = B_TRUE;
4115087719fdSdp 			(void) strlcpy(zone, prop_id, sizeof (zone));
4116087719fdSdp 		}
4117087719fdSdp 		return;
41187c478bd9Sstevel@tonic-gate 	case RT_ZONEPATH:
4119555afedfScarlsonj 		if (!force_set && state_atleast(ZONE_STATE_INSTALLED)) {
41207c478bd9Sstevel@tonic-gate 			zerr(gettext("Zone %s already installed; %s %s not "
41217c478bd9Sstevel@tonic-gate 			    "allowed."), zone, cmd_to_str(CMD_SET),
41227c478bd9Sstevel@tonic-gate 			    rt_to_str(RT_ZONEPATH));
41237c478bd9Sstevel@tonic-gate 			return;
41247c478bd9Sstevel@tonic-gate 		}
41257c478bd9Sstevel@tonic-gate 		if (validate_zonepath_syntax(prop_id) != Z_OK) {
4126bbec428eSgjelinek 			saw_error = B_TRUE;
41277c478bd9Sstevel@tonic-gate 			return;
41287c478bd9Sstevel@tonic-gate 		}
41297c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_set_zonepath(handle, prop_id)) != Z_OK)
4130bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
41317c478bd9Sstevel@tonic-gate 		else
4132bbec428eSgjelinek 			need_to_commit = B_TRUE;
41337c478bd9Sstevel@tonic-gate 		return;
41349acbbeafSnn35248 	case RT_BRAND:
41359acbbeafSnn35248 		if (state_atleast(ZONE_STATE_INSTALLED)) {
41369acbbeafSnn35248 			zerr(gettext("Zone %s already installed; %s %s not "
41379acbbeafSnn35248 			    "allowed."), zone, cmd_to_str(CMD_SET),
41389acbbeafSnn35248 			    rt_to_str(RT_BRAND));
41399acbbeafSnn35248 			return;
41409acbbeafSnn35248 		}
41419acbbeafSnn35248 		if ((err = zonecfg_set_brand(handle, prop_id)) != Z_OK)
4142bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
41439acbbeafSnn35248 		else
4144bbec428eSgjelinek 			need_to_commit = B_TRUE;
41459acbbeafSnn35248 		return;
41467c478bd9Sstevel@tonic-gate 	case RT_AUTOBOOT:
41477c478bd9Sstevel@tonic-gate 		if (strcmp(prop_id, "true") == 0) {
41487c478bd9Sstevel@tonic-gate 			autoboot = B_TRUE;
41497c478bd9Sstevel@tonic-gate 		} else if (strcmp(prop_id, "false") == 0) {
41507c478bd9Sstevel@tonic-gate 			autoboot = B_FALSE;
41517c478bd9Sstevel@tonic-gate 		} else {
41527c478bd9Sstevel@tonic-gate 			zerr(gettext("%s value must be '%s' or '%s'."),
41537c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_AUTOBOOT), "true", "false");
4154bbec428eSgjelinek 			saw_error = B_TRUE;
41557c478bd9Sstevel@tonic-gate 			return;
41567c478bd9Sstevel@tonic-gate 		}
41577c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_set_autoboot(handle, autoboot)) != Z_OK)
4158bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
41597c478bd9Sstevel@tonic-gate 		else
4160bbec428eSgjelinek 			need_to_commit = B_TRUE;
41617c478bd9Sstevel@tonic-gate 		return;
41627c478bd9Sstevel@tonic-gate 	case RT_POOL:
41630209230bSgjelinek 		/* don't allow use of the reserved temporary pool names */
41640209230bSgjelinek 		if (strncmp("SUNW", prop_id, 4) == 0) {
41650209230bSgjelinek 			zerr(gettext("pool names starting with SUNW are "
41660209230bSgjelinek 			    "reserved."));
4167bbec428eSgjelinek 			saw_error = B_TRUE;
41680209230bSgjelinek 			return;
41690209230bSgjelinek 		}
41700209230bSgjelinek 
41710209230bSgjelinek 		/* can't set pool if dedicated-cpu exists */
41720209230bSgjelinek 		if (zonecfg_lookup_pset(handle, &tmp_psettab) == Z_OK) {
41730209230bSgjelinek 			zerr(gettext("The %s resource already exists.  "
41740209230bSgjelinek 			    "A persistent pool is incompatible\nwith the %s "
41750209230bSgjelinek 			    "resource."), rt_to_str(RT_DCPU),
41760209230bSgjelinek 			    rt_to_str(RT_DCPU));
4177bbec428eSgjelinek 			saw_error = B_TRUE;
41780209230bSgjelinek 			return;
41790209230bSgjelinek 		}
41800209230bSgjelinek 
41817c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_set_pool(handle, prop_id)) != Z_OK)
4182bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
41837c478bd9Sstevel@tonic-gate 		else
4184bbec428eSgjelinek 			need_to_commit = B_TRUE;
41857c478bd9Sstevel@tonic-gate 		return;
4186ffbafc53Scomay 	case RT_LIMITPRIV:
4187ffbafc53Scomay 		if ((err = zonecfg_set_limitpriv(handle, prop_id)) != Z_OK)
4188bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
4189ffbafc53Scomay 		else
4190bbec428eSgjelinek 			need_to_commit = B_TRUE;
4191ffbafc53Scomay 		return;
41923f2f09c1Sdp 	case RT_BOOTARGS:
41933f2f09c1Sdp 		if ((err = zonecfg_set_bootargs(handle, prop_id)) != Z_OK)
4194bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
41953f2f09c1Sdp 		else
4196bbec428eSgjelinek 			need_to_commit = B_TRUE;
41973f2f09c1Sdp 		return;
41980209230bSgjelinek 	case RT_SCHED:
41990209230bSgjelinek 		if ((err = zonecfg_set_sched(handle, prop_id)) != Z_OK)
4200bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
42010209230bSgjelinek 		else
4202bbec428eSgjelinek 			need_to_commit = B_TRUE;
42030209230bSgjelinek 		return;
4204f4b3ec61Sdh155122 	case RT_IPTYPE:
4205f4b3ec61Sdh155122 		if (strcmp(prop_id, "shared") == 0) {
4206f4b3ec61Sdh155122 			iptype = ZS_SHARED;
4207f4b3ec61Sdh155122 		} else if (strcmp(prop_id, "exclusive") == 0) {
4208f4b3ec61Sdh155122 			iptype = ZS_EXCLUSIVE;
4209f4b3ec61Sdh155122 		} else {
4210f4b3ec61Sdh155122 			zerr(gettext("%s value must be '%s' or '%s'."),
4211f4b3ec61Sdh155122 			    pt_to_str(PT_IPTYPE), "shared", "exclusive");
4212bbec428eSgjelinek 			saw_error = B_TRUE;
4213f4b3ec61Sdh155122 			return;
4214f4b3ec61Sdh155122 		}
4215f4b3ec61Sdh155122 		if (iptype == ZS_EXCLUSIVE && !allow_exclusive()) {
4216bbec428eSgjelinek 			saw_error = B_TRUE;
4217f4b3ec61Sdh155122 			return;
4218f4b3ec61Sdh155122 		}
4219f4b3ec61Sdh155122 		if ((err = zonecfg_set_iptype(handle, iptype)) != Z_OK)
4220bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
4221f4b3ec61Sdh155122 		else
4222bbec428eSgjelinek 			need_to_commit = B_TRUE;
4223f4b3ec61Sdh155122 		return;
42240209230bSgjelinek 	case RT_MAXLWPS:
42250209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXLWPS, prop_type, prop_id);
42260209230bSgjelinek 		return;
4227ff19e029SMenno Lageman 	case RT_MAXPROCS:
4228ff19e029SMenno Lageman 		set_aliased_rctl(ALIAS_MAXPROCS, prop_type, prop_id);
4229ff19e029SMenno Lageman 		return;
42300209230bSgjelinek 	case RT_MAXSHMMEM:
42310209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXSHMMEM, prop_type, prop_id);
42320209230bSgjelinek 		return;
42330209230bSgjelinek 	case RT_MAXSHMIDS:
42340209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXSHMIDS, prop_type, prop_id);
42350209230bSgjelinek 		return;
42360209230bSgjelinek 	case RT_MAXMSGIDS:
42370209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXMSGIDS, prop_type, prop_id);
42380209230bSgjelinek 		return;
42390209230bSgjelinek 	case RT_MAXSEMIDS:
42400209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXSEMIDS, prop_type, prop_id);
42410209230bSgjelinek 		return;
42420209230bSgjelinek 	case RT_SHARES:
42430209230bSgjelinek 		set_aliased_rctl(ALIAS_SHARES, prop_type, prop_id);
42440209230bSgjelinek 		return;
42455679c89fSjv227347 	case RT_HOSTID:
42465679c89fSjv227347 		if ((err = zonecfg_set_hostid(handle, prop_id)) != Z_OK) {
42475679c89fSjv227347 			if (err == Z_TOO_BIG) {
42485679c89fSjv227347 				zerr(gettext("hostid string is too large: %s"),
42495679c89fSjv227347 				    prop_id);
42505679c89fSjv227347 				saw_error = B_TRUE;
42515679c89fSjv227347 			} else {
42525679c89fSjv227347 				zone_perror(pt_to_str(prop_type), err, B_TRUE);
42535679c89fSjv227347 			}
42545679c89fSjv227347 			return;
42555679c89fSjv227347 		}
42565679c89fSjv227347 		need_to_commit = B_TRUE;
42575679c89fSjv227347 		return;
42580fbb751dSJohn Levon 	case RT_FS_ALLOWED:
42590fbb751dSJohn Levon 		if ((err = zonecfg_set_fs_allowed(handle, prop_id)) != Z_OK)
42600fbb751dSJohn Levon 			zone_perror(zone, err, B_TRUE);
42610fbb751dSJohn Levon 		else
42620fbb751dSJohn Levon 			need_to_commit = B_TRUE;
42630fbb751dSJohn Levon 		return;
42647c478bd9Sstevel@tonic-gate 	case RT_FS:
42657c478bd9Sstevel@tonic-gate 		switch (prop_type) {
42667c478bd9Sstevel@tonic-gate 		case PT_DIR:
42677c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_fstab.zone_fs_dir, prop_id,
42687c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_fstab.zone_fs_dir));
42697c478bd9Sstevel@tonic-gate 			return;
42707c478bd9Sstevel@tonic-gate 		case PT_SPECIAL:
42717c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_fstab.zone_fs_special,
42727c478bd9Sstevel@tonic-gate 			    prop_id,
42737c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_fstab.zone_fs_special));
42747c478bd9Sstevel@tonic-gate 			return;
42757c478bd9Sstevel@tonic-gate 		case PT_RAW:
42767c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_fstab.zone_fs_raw,
42777c478bd9Sstevel@tonic-gate 			    prop_id, sizeof (in_progress_fstab.zone_fs_raw));
42787c478bd9Sstevel@tonic-gate 			return;
42797c478bd9Sstevel@tonic-gate 		case PT_TYPE:
42807c478bd9Sstevel@tonic-gate 			if (!valid_fs_type(prop_id)) {
42817c478bd9Sstevel@tonic-gate 				zerr(gettext("\"%s\" is not a valid %s."),
42827c478bd9Sstevel@tonic-gate 				    prop_id, pt_to_str(PT_TYPE));
4283bbec428eSgjelinek 				saw_error = B_TRUE;
42847c478bd9Sstevel@tonic-gate 				return;
42857c478bd9Sstevel@tonic-gate 			}
42867c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_fstab.zone_fs_type, prop_id,
42877c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_fstab.zone_fs_type));
42887c478bd9Sstevel@tonic-gate 			return;
42897c478bd9Sstevel@tonic-gate 		case PT_OPTIONS:
42907c478bd9Sstevel@tonic-gate 			if (pp->pv_type != PROP_VAL_SIMPLE &&
42917c478bd9Sstevel@tonic-gate 			    pp->pv_type != PROP_VAL_LIST) {
42927c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s or %s value was expected "
42937c478bd9Sstevel@tonic-gate 				    "here."), pvt_to_str(PROP_VAL_SIMPLE),
42947c478bd9Sstevel@tonic-gate 				    pvt_to_str(PROP_VAL_LIST));
4295bbec428eSgjelinek 				saw_error = B_TRUE;
42967c478bd9Sstevel@tonic-gate 				return;
42977c478bd9Sstevel@tonic-gate 			}
42987c478bd9Sstevel@tonic-gate 			zonecfg_free_fs_option_list(
42997c478bd9Sstevel@tonic-gate 			    in_progress_fstab.zone_fs_options);
43007c478bd9Sstevel@tonic-gate 			in_progress_fstab.zone_fs_options = NULL;
43017c478bd9Sstevel@tonic-gate 			if (!(pp->pv_type == PROP_VAL_LIST &&
43027c478bd9Sstevel@tonic-gate 			    pp->pv_list == NULL))
43037c478bd9Sstevel@tonic-gate 				add_property(cmd);
43047c478bd9Sstevel@tonic-gate 			return;
43057c478bd9Sstevel@tonic-gate 		default:
43067c478bd9Sstevel@tonic-gate 			break;
43077c478bd9Sstevel@tonic-gate 		}
4308bbec428eSgjelinek 		zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, B_TRUE);
4309bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
4310bbec428eSgjelinek 		usage(B_FALSE, HELP_PROPS);
43117c478bd9Sstevel@tonic-gate 		return;
43127c478bd9Sstevel@tonic-gate 	case RT_NET:
43137c478bd9Sstevel@tonic-gate 		switch (prop_type) {
43147c478bd9Sstevel@tonic-gate 		case PT_ADDRESS:
43157c478bd9Sstevel@tonic-gate 			if (validate_net_address_syntax(prop_id) != Z_OK) {
4316bbec428eSgjelinek 				saw_error = B_TRUE;
43177c478bd9Sstevel@tonic-gate 				return;
43187c478bd9Sstevel@tonic-gate 			}
43197c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_nwiftab.zone_nwif_address,
43207c478bd9Sstevel@tonic-gate 			    prop_id,
43217c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_nwiftab.zone_nwif_address));
43227c478bd9Sstevel@tonic-gate 			break;
43237c478bd9Sstevel@tonic-gate 		case PT_PHYSICAL:
43247c478bd9Sstevel@tonic-gate 			if (validate_net_physical_syntax(prop_id) != Z_OK) {
4325bbec428eSgjelinek 				saw_error = B_TRUE;
43267c478bd9Sstevel@tonic-gate 				return;
43277c478bd9Sstevel@tonic-gate 			}
43287c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_nwiftab.zone_nwif_physical,
43297c478bd9Sstevel@tonic-gate 			    prop_id,
43307c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_nwiftab.zone_nwif_physical));
43317c478bd9Sstevel@tonic-gate 			break;
4332de860bd9Sgfaden 		case PT_DEFROUTER:
4333de860bd9Sgfaden 			if (validate_net_address_syntax(prop_id) != Z_OK) {
4334bbec428eSgjelinek 				saw_error = B_TRUE;
4335de860bd9Sgfaden 				return;
4336de860bd9Sgfaden 			}
4337de860bd9Sgfaden 			(void) strlcpy(in_progress_nwiftab.zone_nwif_defrouter,
4338de860bd9Sgfaden 			    prop_id,
4339de860bd9Sgfaden 			    sizeof (in_progress_nwiftab.zone_nwif_defrouter));
4340de860bd9Sgfaden 			break;
43417c478bd9Sstevel@tonic-gate 		default:
43427c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4343bbec428eSgjelinek 			    B_TRUE);
4344bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4345bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
43467c478bd9Sstevel@tonic-gate 			return;
43477c478bd9Sstevel@tonic-gate 		}
43487c478bd9Sstevel@tonic-gate 		return;
43497c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
43507c478bd9Sstevel@tonic-gate 		switch (prop_type) {
43517c478bd9Sstevel@tonic-gate 		case PT_MATCH:
43527c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_devtab.zone_dev_match,
43537c478bd9Sstevel@tonic-gate 			    prop_id,
43547c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_devtab.zone_dev_match));
43557c478bd9Sstevel@tonic-gate 			break;
43567c478bd9Sstevel@tonic-gate 		default:
43577c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4358bbec428eSgjelinek 			    B_TRUE);
4359bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4360bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
43617c478bd9Sstevel@tonic-gate 			return;
43627c478bd9Sstevel@tonic-gate 		}
43637c478bd9Sstevel@tonic-gate 		return;
43647c478bd9Sstevel@tonic-gate 	case RT_RCTL:
43657c478bd9Sstevel@tonic-gate 		switch (prop_type) {
43667c478bd9Sstevel@tonic-gate 		case PT_NAME:
43677c478bd9Sstevel@tonic-gate 			if (!zonecfg_valid_rctlname(prop_id)) {
43687c478bd9Sstevel@tonic-gate 				zerr(gettext("'%s' is not a valid zone %s "
43697c478bd9Sstevel@tonic-gate 				    "name."), prop_id, rt_to_str(RT_RCTL));
43707c478bd9Sstevel@tonic-gate 				return;
43717c478bd9Sstevel@tonic-gate 			}
43727c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_rctltab.zone_rctl_name,
43737c478bd9Sstevel@tonic-gate 			    prop_id,
43747c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_rctltab.zone_rctl_name));
43757c478bd9Sstevel@tonic-gate 			break;
43767c478bd9Sstevel@tonic-gate 		case PT_VALUE:
43777c478bd9Sstevel@tonic-gate 			if (pp->pv_type != PROP_VAL_COMPLEX &&
43787c478bd9Sstevel@tonic-gate 			    pp->pv_type != PROP_VAL_LIST) {
43797c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s or %s value was expected "
43807c478bd9Sstevel@tonic-gate 				    "here."), pvt_to_str(PROP_VAL_COMPLEX),
43817c478bd9Sstevel@tonic-gate 				    pvt_to_str(PROP_VAL_LIST));
4382bbec428eSgjelinek 				saw_error = B_TRUE;
43837c478bd9Sstevel@tonic-gate 				return;
43847c478bd9Sstevel@tonic-gate 			}
43857c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(
43867c478bd9Sstevel@tonic-gate 			    in_progress_rctltab.zone_rctl_valptr);
43877c478bd9Sstevel@tonic-gate 			in_progress_rctltab.zone_rctl_valptr = NULL;
43887c478bd9Sstevel@tonic-gate 			if (!(pp->pv_type == PROP_VAL_LIST &&
43897c478bd9Sstevel@tonic-gate 			    pp->pv_list == NULL))
43907c478bd9Sstevel@tonic-gate 				add_property(cmd);
43917c478bd9Sstevel@tonic-gate 			break;
43927c478bd9Sstevel@tonic-gate 		default:
43937c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4394bbec428eSgjelinek 			    B_TRUE);
4395bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4396bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
43977c478bd9Sstevel@tonic-gate 			return;
43987c478bd9Sstevel@tonic-gate 		}
43997c478bd9Sstevel@tonic-gate 		return;
44007c478bd9Sstevel@tonic-gate 	case RT_ATTR:
44017c478bd9Sstevel@tonic-gate 		switch (prop_type) {
44027c478bd9Sstevel@tonic-gate 		case PT_NAME:
44037c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_attrtab.zone_attr_name,
44047c478bd9Sstevel@tonic-gate 			    prop_id,
44057c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_attrtab.zone_attr_name));
44067c478bd9Sstevel@tonic-gate 			break;
44077c478bd9Sstevel@tonic-gate 		case PT_TYPE:
44087c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_attrtab.zone_attr_type,
44097c478bd9Sstevel@tonic-gate 			    prop_id,
44107c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_attrtab.zone_attr_type));
44117c478bd9Sstevel@tonic-gate 			break;
44127c478bd9Sstevel@tonic-gate 		case PT_VALUE:
44137c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_attrtab.zone_attr_value,
44147c478bd9Sstevel@tonic-gate 			    prop_id,
44157c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_attrtab.zone_attr_value));
44167c478bd9Sstevel@tonic-gate 			break;
44177c478bd9Sstevel@tonic-gate 		default:
44187c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4419bbec428eSgjelinek 			    B_TRUE);
4420bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4421bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
44227c478bd9Sstevel@tonic-gate 			return;
44237c478bd9Sstevel@tonic-gate 		}
44247c478bd9Sstevel@tonic-gate 		return;
4425fa9e4066Sahrens 	case RT_DATASET:
4426fa9e4066Sahrens 		switch (prop_type) {
4427fa9e4066Sahrens 		case PT_NAME:
4428fa9e4066Sahrens 			(void) strlcpy(in_progress_dstab.zone_dataset_name,
4429fa9e4066Sahrens 			    prop_id,
4430fa9e4066Sahrens 			    sizeof (in_progress_dstab.zone_dataset_name));
4431fa9e4066Sahrens 			return;
4432fa9e4066Sahrens 		default:
4433fa9e4066Sahrens 			break;
4434fa9e4066Sahrens 		}
4435bbec428eSgjelinek 		zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, B_TRUE);
4436bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
4437bbec428eSgjelinek 		usage(B_FALSE, HELP_PROPS);
4438fa9e4066Sahrens 		return;
44390209230bSgjelinek 	case RT_DCPU:
44400209230bSgjelinek 		switch (prop_type) {
44410209230bSgjelinek 		char *lowp, *highp;
44420209230bSgjelinek 
44430209230bSgjelinek 		case PT_NCPUS:
44440209230bSgjelinek 			lowp = prop_id;
44450209230bSgjelinek 			if ((highp = strchr(prop_id, '-')) != NULL)
44460209230bSgjelinek 				*highp++ = '\0';
44470209230bSgjelinek 			else
44480209230bSgjelinek 				highp = lowp;
44490209230bSgjelinek 
44500209230bSgjelinek 			/* Make sure the input makes sense. */
44510209230bSgjelinek 			if (!zonecfg_valid_ncpus(lowp, highp)) {
44520209230bSgjelinek 				zerr(gettext("%s property is out of range."),
44530209230bSgjelinek 				    pt_to_str(PT_NCPUS));
4454bbec428eSgjelinek 				saw_error = B_TRUE;
44550209230bSgjelinek 				return;
44560209230bSgjelinek 			}
44570209230bSgjelinek 
44580209230bSgjelinek 			(void) strlcpy(
44590209230bSgjelinek 			    in_progress_psettab.zone_ncpu_min, lowp,
44600209230bSgjelinek 			    sizeof (in_progress_psettab.zone_ncpu_min));
44610209230bSgjelinek 			(void) strlcpy(
44620209230bSgjelinek 			    in_progress_psettab.zone_ncpu_max, highp,
44630209230bSgjelinek 			    sizeof (in_progress_psettab.zone_ncpu_max));
44640209230bSgjelinek 			return;
44650209230bSgjelinek 		case PT_IMPORTANCE:
44660209230bSgjelinek 			/* Make sure the value makes sense. */
44670209230bSgjelinek 			if (!zonecfg_valid_importance(prop_id)) {
44680209230bSgjelinek 				zerr(gettext("%s property is out of range."),
44690209230bSgjelinek 				    pt_to_str(PT_IMPORTANCE));
4470bbec428eSgjelinek 				saw_error = B_TRUE;
44710209230bSgjelinek 				return;
44720209230bSgjelinek 			}
44730209230bSgjelinek 
44740209230bSgjelinek 			(void) strlcpy(in_progress_psettab.zone_importance,
44750209230bSgjelinek 			    prop_id,
44760209230bSgjelinek 			    sizeof (in_progress_psettab.zone_importance));
44770209230bSgjelinek 			return;
44780209230bSgjelinek 		default:
44790209230bSgjelinek 			break;
44800209230bSgjelinek 		}
4481bbec428eSgjelinek 		zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, B_TRUE);
4482bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
4483bbec428eSgjelinek 		usage(B_FALSE, HELP_PROPS);
44840209230bSgjelinek 		return;
4485c97ad5cdSakolb 	case RT_PCAP:
4486c97ad5cdSakolb 		if (prop_type != PT_NCPUS) {
4487c97ad5cdSakolb 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4488bbec428eSgjelinek 			    B_TRUE);
4489bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4490bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
4491c97ad5cdSakolb 			return;
4492c97ad5cdSakolb 		}
4493c97ad5cdSakolb 
4494c97ad5cdSakolb 		/*
4495c97ad5cdSakolb 		 * We already checked that an rctl alias is allowed in
4496c97ad5cdSakolb 		 * the add_resource() function.
4497c97ad5cdSakolb 		 */
4498c97ad5cdSakolb 
4499c97ad5cdSakolb 		if ((cap = strtof(prop_id, &unitp)) <= 0 || *unitp != '\0' ||
4500c97ad5cdSakolb 		    (int)(cap * 100) < 1) {
4501c97ad5cdSakolb 			zerr(gettext("%s property is out of range."),
4502c97ad5cdSakolb 			    pt_to_str(PT_NCPUS));
4503bbec428eSgjelinek 			saw_error = B_TRUE;
4504c97ad5cdSakolb 			return;
4505c97ad5cdSakolb 		}
4506c97ad5cdSakolb 
4507c97ad5cdSakolb 		if ((err = zonecfg_set_aliased_rctl(handle, ALIAS_CPUCAP,
4508c97ad5cdSakolb 		    (int)(cap * 100))) != Z_OK)
4509bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
4510c97ad5cdSakolb 		else
4511bbec428eSgjelinek 			need_to_commit = B_TRUE;
4512c97ad5cdSakolb 		return;
45130209230bSgjelinek 	case RT_MCAP:
45140209230bSgjelinek 		switch (prop_type) {
45150209230bSgjelinek 		case PT_PHYSICAL:
45160209230bSgjelinek 			if (!zonecfg_valid_memlimit(prop_id, &mem_cap)) {
45170209230bSgjelinek 				zerr(gettext("A positive number with a "
45180209230bSgjelinek 				    "required scale suffix (K, M, G or T) was "
45190209230bSgjelinek 				    "expected here."));
4520bbec428eSgjelinek 				saw_error = B_TRUE;
45210209230bSgjelinek 			} else if (mem_cap < ONE_MB) {
45220209230bSgjelinek 				zerr(gettext("%s value is too small.  It must "
45230209230bSgjelinek 				    "be at least 1M."), pt_to_str(PT_PHYSICAL));
4524bbec428eSgjelinek 				saw_error = B_TRUE;
45250209230bSgjelinek 			} else {
45260209230bSgjelinek 				snprintf(in_progress_mcaptab.zone_physmem_cap,
45270209230bSgjelinek 				    physmem_size, "%llu", mem_cap);
45280209230bSgjelinek 			}
45290209230bSgjelinek 			break;
45300209230bSgjelinek 		case PT_SWAP:
45310209230bSgjelinek 			/*
45320209230bSgjelinek 			 * We have to check if an rctl is allowed here since
45330209230bSgjelinek 			 * there might already be a rctl defined that blocks
45340209230bSgjelinek 			 * the alias.
45350209230bSgjelinek 			 */
45360209230bSgjelinek 			if (!zonecfg_aliased_rctl_ok(handle, ALIAS_MAXSWAP)) {
45370209230bSgjelinek 				zone_perror(pt_to_str(PT_MAXSWAP),
4538bbec428eSgjelinek 				    Z_ALIAS_DISALLOW, B_FALSE);
4539bbec428eSgjelinek 				saw_error = B_TRUE;
45400209230bSgjelinek 				return;
45410209230bSgjelinek 			}
45420209230bSgjelinek 
45430209230bSgjelinek 			if (global_zone)
45440209230bSgjelinek 				mem_limit = ONE_MB * 100;
45450209230bSgjelinek 			else
45460209230bSgjelinek 				mem_limit = ONE_MB * 50;
45470209230bSgjelinek 
45480209230bSgjelinek 			if (!zonecfg_valid_memlimit(prop_id, &mem_cap)) {
45490209230bSgjelinek 				zerr(gettext("A positive number with a "
45500209230bSgjelinek 				    "required scale suffix (K, M, G or T) was "
45510209230bSgjelinek 				    "expected here."));
4552bbec428eSgjelinek 				saw_error = B_TRUE;
45530209230bSgjelinek 			} else if (mem_cap < mem_limit) {
45540209230bSgjelinek 				char buf[128];
45550209230bSgjelinek 
45560209230bSgjelinek 				(void) snprintf(buf, sizeof (buf), "%llu",
45570209230bSgjelinek 				    mem_limit);
45580209230bSgjelinek 				bytes_to_units(buf, buf, sizeof (buf));
45590209230bSgjelinek 				zerr(gettext("%s value is too small.  It must "
45600209230bSgjelinek 				    "be at least %s."), pt_to_str(PT_SWAP),
45610209230bSgjelinek 				    buf);
4562bbec428eSgjelinek 				saw_error = B_TRUE;
45630209230bSgjelinek 			} else {
45640209230bSgjelinek 				if ((err = zonecfg_set_aliased_rctl(handle,
45650209230bSgjelinek 				    ALIAS_MAXSWAP, mem_cap)) != Z_OK)
4566bbec428eSgjelinek 					zone_perror(zone, err, B_TRUE);
45670209230bSgjelinek 				else
4568bbec428eSgjelinek 					need_to_commit = B_TRUE;
45690209230bSgjelinek 			}
45700209230bSgjelinek 			break;
45710209230bSgjelinek 		case PT_LOCKED:
45720209230bSgjelinek 			/*
45730209230bSgjelinek 			 * We have to check if an rctl is allowed here since
45740209230bSgjelinek 			 * there might already be a rctl defined that blocks
45750209230bSgjelinek 			 * the alias.
45760209230bSgjelinek 			 */
45770209230bSgjelinek 			if (!zonecfg_aliased_rctl_ok(handle,
45780209230bSgjelinek 			    ALIAS_MAXLOCKEDMEM)) {
45790209230bSgjelinek 				zone_perror(pt_to_str(PT_LOCKED),
4580bbec428eSgjelinek 				    Z_ALIAS_DISALLOW, B_FALSE);
4581bbec428eSgjelinek 				saw_error = B_TRUE;
45820209230bSgjelinek 				return;
45830209230bSgjelinek 			}
45840209230bSgjelinek 
45850209230bSgjelinek 			if (!zonecfg_valid_memlimit(prop_id, &mem_cap)) {
45860209230bSgjelinek 				zerr(gettext("A non-negative number with a "
45870209230bSgjelinek 				    "required scale suffix (K, M, G or T) was "
45880209230bSgjelinek 				    "expected\nhere."));
4589bbec428eSgjelinek 				saw_error = B_TRUE;
45900209230bSgjelinek 			} else {
45910209230bSgjelinek 				if ((err = zonecfg_set_aliased_rctl(handle,
45920209230bSgjelinek 				    ALIAS_MAXLOCKEDMEM, mem_cap)) != Z_OK)
4593bbec428eSgjelinek 					zone_perror(zone, err, B_TRUE);
45940209230bSgjelinek 				else
4595bbec428eSgjelinek 					need_to_commit = B_TRUE;
45960209230bSgjelinek 			}
45970209230bSgjelinek 			break;
45980209230bSgjelinek 		default:
45990209230bSgjelinek 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4600bbec428eSgjelinek 			    B_TRUE);
4601bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4602bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
46030209230bSgjelinek 			return;
46040209230bSgjelinek 		}
46050209230bSgjelinek 		return;
4606cb8a054bSGlenn Faden 	case RT_ADMIN:
4607cb8a054bSGlenn Faden 		switch (prop_type) {
4608cb8a054bSGlenn Faden 		case PT_USER:
4609cb8a054bSGlenn Faden 			(void) strlcpy(in_progress_admintab.zone_admin_user,
4610cb8a054bSGlenn Faden 			    prop_id,
4611cb8a054bSGlenn Faden 			    sizeof (in_progress_admintab.zone_admin_user));
4612cb8a054bSGlenn Faden 			return;
4613cb8a054bSGlenn Faden 		case PT_AUTHS:
4614cb8a054bSGlenn Faden 			(void) strlcpy(in_progress_admintab.zone_admin_auths,
4615cb8a054bSGlenn Faden 			    prop_id,
4616cb8a054bSGlenn Faden 			    sizeof (in_progress_admintab.zone_admin_auths));
4617cb8a054bSGlenn Faden 			return;
4618cb8a054bSGlenn Faden 		default:
4619cb8a054bSGlenn Faden 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4620cb8a054bSGlenn Faden 			    B_TRUE);
4621cb8a054bSGlenn Faden 			long_usage(CMD_SET, B_TRUE);
4622cb8a054bSGlenn Faden 			usage(B_FALSE, HELP_PROPS);
4623cb8a054bSGlenn Faden 			return;
4624cb8a054bSGlenn Faden 		}
46257c478bd9Sstevel@tonic-gate 	default:
4626bbec428eSgjelinek 		zone_perror(rt_to_str(res_type), Z_NO_RESOURCE_TYPE, B_TRUE);
4627bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
4628bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
46297c478bd9Sstevel@tonic-gate 		return;
46307c478bd9Sstevel@tonic-gate 	}
46317c478bd9Sstevel@tonic-gate }
46327c478bd9Sstevel@tonic-gate 
46337c478bd9Sstevel@tonic-gate static void
4634bbec428eSgjelinek output_prop(FILE *fp, int pnum, char *pval, boolean_t print_notspec)
46357c478bd9Sstevel@tonic-gate {
46367c478bd9Sstevel@tonic-gate 	char *qstr;
46377c478bd9Sstevel@tonic-gate 
46387c478bd9Sstevel@tonic-gate 	if (*pval != '\0') {
46397c478bd9Sstevel@tonic-gate 		qstr = quoteit(pval);
46400209230bSgjelinek 		if (pnum == PT_SWAP || pnum == PT_LOCKED)
46410209230bSgjelinek 			(void) fprintf(fp, "\t[%s: %s]\n", pt_to_str(pnum),
46420209230bSgjelinek 			    qstr);
46430209230bSgjelinek 		else
46447c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s: %s\n", pt_to_str(pnum), qstr);
46457c478bd9Sstevel@tonic-gate 		free(qstr);
46467c478bd9Sstevel@tonic-gate 	} else if (print_notspec)
4647087719fdSdp 		(void) fprintf(fp, gettext("\t%s not specified\n"),
4648087719fdSdp 		    pt_to_str(pnum));
4649087719fdSdp }
4650087719fdSdp 
4651087719fdSdp static void
4652087719fdSdp info_zonename(zone_dochandle_t handle, FILE *fp)
4653087719fdSdp {
4654087719fdSdp 	char zonename[ZONENAME_MAX];
4655087719fdSdp 
4656087719fdSdp 	if (zonecfg_get_name(handle, zonename, sizeof (zonename)) == Z_OK)
4657087719fdSdp 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_ZONENAME),
4658087719fdSdp 		    zonename);
4659087719fdSdp 	else
4660087719fdSdp 		(void) fprintf(fp, gettext("%s not specified\n"),
4661087719fdSdp 		    pt_to_str(PT_ZONENAME));
46627c478bd9Sstevel@tonic-gate }
46637c478bd9Sstevel@tonic-gate 
46647c478bd9Sstevel@tonic-gate static void
46657c478bd9Sstevel@tonic-gate info_zonepath(zone_dochandle_t handle, FILE *fp)
46667c478bd9Sstevel@tonic-gate {
46677c478bd9Sstevel@tonic-gate 	char zonepath[MAXPATHLEN];
46687c478bd9Sstevel@tonic-gate 
46697c478bd9Sstevel@tonic-gate 	if (zonecfg_get_zonepath(handle, zonepath, sizeof (zonepath)) == Z_OK)
46707c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_ZONEPATH),
46717c478bd9Sstevel@tonic-gate 		    zonepath);
4672087719fdSdp 	else {
4673087719fdSdp 		(void) fprintf(fp, gettext("%s not specified\n"),
4674087719fdSdp 		    pt_to_str(PT_ZONEPATH));
4675087719fdSdp 	}
46767c478bd9Sstevel@tonic-gate }
46777c478bd9Sstevel@tonic-gate 
46787c478bd9Sstevel@tonic-gate static void
46799acbbeafSnn35248 info_brand(zone_dochandle_t handle, FILE *fp)
46809acbbeafSnn35248 {
46819acbbeafSnn35248 	char brand[MAXNAMELEN];
46829acbbeafSnn35248 
46839acbbeafSnn35248 	if (zonecfg_get_brand(handle, brand, sizeof (brand)) == Z_OK)
46849acbbeafSnn35248 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_BRAND),
46859acbbeafSnn35248 		    brand);
46869acbbeafSnn35248 	else
46879acbbeafSnn35248 		(void) fprintf(fp, "%s %s\n", pt_to_str(PT_BRAND),
46889acbbeafSnn35248 		    gettext("not specified"));
46899acbbeafSnn35248 }
46909acbbeafSnn35248 
46919acbbeafSnn35248 static void
46927c478bd9Sstevel@tonic-gate info_autoboot(zone_dochandle_t handle, FILE *fp)
46937c478bd9Sstevel@tonic-gate {
46947c478bd9Sstevel@tonic-gate 	boolean_t autoboot;
46957c478bd9Sstevel@tonic-gate 	int err;
46967c478bd9Sstevel@tonic-gate 
46977c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_get_autoboot(handle, &autoboot)) == Z_OK)
46987c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_AUTOBOOT),
46997c478bd9Sstevel@tonic-gate 		    autoboot ? "true" : "false");
47007c478bd9Sstevel@tonic-gate 	else
4701bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
47027c478bd9Sstevel@tonic-gate }
47037c478bd9Sstevel@tonic-gate 
47047c478bd9Sstevel@tonic-gate static void
47057c478bd9Sstevel@tonic-gate info_pool(zone_dochandle_t handle, FILE *fp)
47067c478bd9Sstevel@tonic-gate {
47077c478bd9Sstevel@tonic-gate 	char pool[MAXNAMELEN];
47087c478bd9Sstevel@tonic-gate 	int err;
47097c478bd9Sstevel@tonic-gate 
47107c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_get_pool(handle, pool, sizeof (pool))) == Z_OK)
47117c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_POOL), pool);
47127c478bd9Sstevel@tonic-gate 	else
4713bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
47147c478bd9Sstevel@tonic-gate }
47157c478bd9Sstevel@tonic-gate 
47167c478bd9Sstevel@tonic-gate static void
4717ffbafc53Scomay info_limitpriv(zone_dochandle_t handle, FILE *fp)
4718ffbafc53Scomay {
4719ffbafc53Scomay 	char *limitpriv;
4720ffbafc53Scomay 	int err;
4721ffbafc53Scomay 
4722ffbafc53Scomay 	if ((err = zonecfg_get_limitpriv(handle, &limitpriv)) == Z_OK) {
4723ffbafc53Scomay 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_LIMITPRIV),
4724ffbafc53Scomay 		    limitpriv);
4725ffbafc53Scomay 		free(limitpriv);
4726ffbafc53Scomay 	} else {
4727bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
4728ffbafc53Scomay 	}
4729ffbafc53Scomay }
4730ffbafc53Scomay 
4731ffbafc53Scomay static void
47323f2f09c1Sdp info_bootargs(zone_dochandle_t handle, FILE *fp)
47333f2f09c1Sdp {
47343f2f09c1Sdp 	char bootargs[BOOTARGS_MAX];
47353f2f09c1Sdp 	int err;
47363f2f09c1Sdp 
47373f2f09c1Sdp 	if ((err = zonecfg_get_bootargs(handle, bootargs,
47383f2f09c1Sdp 	    sizeof (bootargs))) == Z_OK) {
47393f2f09c1Sdp 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_BOOTARGS),
47403f2f09c1Sdp 		    bootargs);
47413f2f09c1Sdp 	} else {
4742bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
47433f2f09c1Sdp 	}
47443f2f09c1Sdp }
47453f2f09c1Sdp 
47463f2f09c1Sdp static void
47470209230bSgjelinek info_sched(zone_dochandle_t handle, FILE *fp)
47480209230bSgjelinek {
47490209230bSgjelinek 	char sched[MAXNAMELEN];
47500209230bSgjelinek 	int err;
47510209230bSgjelinek 
47520209230bSgjelinek 	if ((err = zonecfg_get_sched_class(handle, sched, sizeof (sched)))
47530209230bSgjelinek 	    == Z_OK) {
47540209230bSgjelinek 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_SCHED), sched);
47550209230bSgjelinek 	} else {
4756bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
47570209230bSgjelinek 	}
47580209230bSgjelinek }
47590209230bSgjelinek 
47600209230bSgjelinek static void
4761f4b3ec61Sdh155122 info_iptype(zone_dochandle_t handle, FILE *fp)
4762f4b3ec61Sdh155122 {
4763f4b3ec61Sdh155122 	zone_iptype_t iptype;
4764f4b3ec61Sdh155122 	int err;
4765f4b3ec61Sdh155122 
4766f4b3ec61Sdh155122 	if ((err = zonecfg_get_iptype(handle, &iptype)) == Z_OK) {
4767f4b3ec61Sdh155122 		switch (iptype) {
4768f4b3ec61Sdh155122 		case ZS_SHARED:
4769f4b3ec61Sdh155122 			(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_IPTYPE),
4770f4b3ec61Sdh155122 			    "shared");
4771f4b3ec61Sdh155122 			break;
4772f4b3ec61Sdh155122 		case ZS_EXCLUSIVE:
4773f4b3ec61Sdh155122 			(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_IPTYPE),
4774f4b3ec61Sdh155122 			    "exclusive");
4775f4b3ec61Sdh155122 			break;
4776f4b3ec61Sdh155122 		}
4777f4b3ec61Sdh155122 	} else {
4778bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
4779f4b3ec61Sdh155122 	}
4780f4b3ec61Sdh155122 }
4781f4b3ec61Sdh155122 
4782f4b3ec61Sdh155122 static void
47835679c89fSjv227347 info_hostid(zone_dochandle_t handle, FILE *fp)
47845679c89fSjv227347 {
47855679c89fSjv227347 	char hostidp[HW_HOSTID_LEN];
47860fbb751dSJohn Levon 	int err;
47875679c89fSjv227347 
47880fbb751dSJohn Levon 	if ((err = zonecfg_get_hostid(handle, hostidp,
47890fbb751dSJohn Levon 	    sizeof (hostidp))) == Z_OK) {
47905679c89fSjv227347 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_HOSTID), hostidp);
47910fbb751dSJohn Levon 	} else if (err == Z_BAD_PROPERTY) {
47920fbb751dSJohn Levon 		(void) fprintf(fp, "%s: \n", pt_to_str(PT_HOSTID));
47930fbb751dSJohn Levon 	} else {
47940fbb751dSJohn Levon 		zone_perror(zone, err, B_TRUE);
47950fbb751dSJohn Levon 	}
47960fbb751dSJohn Levon }
47970fbb751dSJohn Levon 
47980fbb751dSJohn Levon static void
47990fbb751dSJohn Levon info_fs_allowed(zone_dochandle_t handle, FILE *fp)
48000fbb751dSJohn Levon {
48010fbb751dSJohn Levon 	char fsallowedp[ZONE_FS_ALLOWED_MAX];
48020fbb751dSJohn Levon 	int err;
48030fbb751dSJohn Levon 
48040fbb751dSJohn Levon 	if ((err = zonecfg_get_fs_allowed(handle, fsallowedp,
48050fbb751dSJohn Levon 	    sizeof (fsallowedp))) == Z_OK) {
48060fbb751dSJohn Levon 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_FS_ALLOWED),
48070fbb751dSJohn Levon 		    fsallowedp);
48080fbb751dSJohn Levon 	} else if (err == Z_BAD_PROPERTY) {
48090fbb751dSJohn Levon 		(void) fprintf(fp, "%s: \n", pt_to_str(PT_FS_ALLOWED));
48100fbb751dSJohn Levon 	} else {
48110fbb751dSJohn Levon 		zone_perror(zone, err, B_TRUE);
48120fbb751dSJohn Levon 	}
48135679c89fSjv227347 }
48145679c89fSjv227347 
48155679c89fSjv227347 static void
48167c478bd9Sstevel@tonic-gate output_fs(FILE *fp, struct zone_fstab *fstab)
48177c478bd9Sstevel@tonic-gate {
48187c478bd9Sstevel@tonic-gate 	zone_fsopt_t *this;
48197c478bd9Sstevel@tonic-gate 
48207c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_FS));
48217c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_DIR, fstab->zone_fs_dir, B_TRUE);
48227c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_SPECIAL, fstab->zone_fs_special, B_TRUE);
48237c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_RAW, fstab->zone_fs_raw, B_TRUE);
48247c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_TYPE, fstab->zone_fs_type, B_TRUE);
48257c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "\t%s: [", pt_to_str(PT_OPTIONS));
48267c478bd9Sstevel@tonic-gate 	for (this = fstab->zone_fs_options; this != NULL;
48277c478bd9Sstevel@tonic-gate 	    this = this->zone_fsopt_next) {
48287c478bd9Sstevel@tonic-gate 		if (strchr(this->zone_fsopt_opt, '='))
48297c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\"%s\"", this->zone_fsopt_opt);
48307c478bd9Sstevel@tonic-gate 		else
48317c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "%s", this->zone_fsopt_opt);
48327c478bd9Sstevel@tonic-gate 		if (this->zone_fsopt_next != NULL)
48337c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, ",");
48347c478bd9Sstevel@tonic-gate 	}
48357c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "]\n");
48367c478bd9Sstevel@tonic-gate }
48377c478bd9Sstevel@tonic-gate 
48387c478bd9Sstevel@tonic-gate static void
48397c478bd9Sstevel@tonic-gate info_fs(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
48407c478bd9Sstevel@tonic-gate {
48417c478bd9Sstevel@tonic-gate 	struct zone_fstab lookup, user;
4842bbec428eSgjelinek 	boolean_t output = B_FALSE;
48437c478bd9Sstevel@tonic-gate 
48447c478bd9Sstevel@tonic-gate 	if (zonecfg_setfsent(handle) != Z_OK)
48457c478bd9Sstevel@tonic-gate 		return;
48467c478bd9Sstevel@tonic-gate 	while (zonecfg_getfsent(handle, &lookup) == Z_OK) {
48477c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
48487c478bd9Sstevel@tonic-gate 			output_fs(fp, &lookup);
48497c478bd9Sstevel@tonic-gate 			goto loopend;
48507c478bd9Sstevel@tonic-gate 		}
4851bbec428eSgjelinek 		if (fill_in_fstab(cmd, &user, B_TRUE) != Z_OK)
48527c478bd9Sstevel@tonic-gate 			goto loopend;
48537c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_fs_dir) > 0 &&
48547c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_fs_dir, lookup.zone_fs_dir) != 0)
48557c478bd9Sstevel@tonic-gate 			goto loopend;	/* no match */
48567c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_fs_special) > 0 &&
48577c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_fs_special, lookup.zone_fs_special) != 0)
48587c478bd9Sstevel@tonic-gate 			goto loopend;	/* no match */
48597c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_fs_type) > 0 &&
48607c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_fs_type, lookup.zone_fs_type) != 0)
48617c478bd9Sstevel@tonic-gate 			goto loopend;	/* no match */
48627c478bd9Sstevel@tonic-gate 		output_fs(fp, &lookup);
4863bbec428eSgjelinek 		output = B_TRUE;
48647c478bd9Sstevel@tonic-gate loopend:
48657c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(lookup.zone_fs_options);
48667c478bd9Sstevel@tonic-gate 	}
48677c478bd9Sstevel@tonic-gate 	(void) zonecfg_endfsent(handle);
48687c478bd9Sstevel@tonic-gate 	/*
48697c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
48707c478bd9Sstevel@tonic-gate 	 * nothing to output.
48717c478bd9Sstevel@tonic-gate 	 */
48727c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
48737c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
48747c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_FS));
48757c478bd9Sstevel@tonic-gate }
48767c478bd9Sstevel@tonic-gate 
48777c478bd9Sstevel@tonic-gate static void
48787c478bd9Sstevel@tonic-gate output_net(FILE *fp, struct zone_nwiftab *nwiftab)
48797c478bd9Sstevel@tonic-gate {
48807c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_NET));
48817c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_ADDRESS, nwiftab->zone_nwif_address, B_TRUE);
48827c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_PHYSICAL, nwiftab->zone_nwif_physical, B_TRUE);
4883de860bd9Sgfaden 	output_prop(fp, PT_DEFROUTER, nwiftab->zone_nwif_defrouter, B_TRUE);
48847c478bd9Sstevel@tonic-gate }
48857c478bd9Sstevel@tonic-gate 
48867c478bd9Sstevel@tonic-gate static void
48877c478bd9Sstevel@tonic-gate info_net(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
48887c478bd9Sstevel@tonic-gate {
48897c478bd9Sstevel@tonic-gate 	struct zone_nwiftab lookup, user;
4890bbec428eSgjelinek 	boolean_t output = B_FALSE;
48917c478bd9Sstevel@tonic-gate 
48927c478bd9Sstevel@tonic-gate 	if (zonecfg_setnwifent(handle) != Z_OK)
48937c478bd9Sstevel@tonic-gate 		return;
48947c478bd9Sstevel@tonic-gate 	while (zonecfg_getnwifent(handle, &lookup) == Z_OK) {
48957c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
48967c478bd9Sstevel@tonic-gate 			output_net(fp, &lookup);
48977c478bd9Sstevel@tonic-gate 			continue;
48987c478bd9Sstevel@tonic-gate 		}
4899bbec428eSgjelinek 		if (fill_in_nwiftab(cmd, &user, B_TRUE) != Z_OK)
49007c478bd9Sstevel@tonic-gate 			continue;
49017c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_nwif_physical) > 0 &&
49027c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_nwif_physical,
49037c478bd9Sstevel@tonic-gate 		    lookup.zone_nwif_physical) != 0)
49047c478bd9Sstevel@tonic-gate 			continue;	/* no match */
4905f4b3ec61Sdh155122 		/* If present make sure it matches */
49067c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_nwif_address) > 0 &&
49077c478bd9Sstevel@tonic-gate 		    !zonecfg_same_net_address(user.zone_nwif_address,
49087c478bd9Sstevel@tonic-gate 		    lookup.zone_nwif_address))
49097c478bd9Sstevel@tonic-gate 			continue;	/* no match */
49107c478bd9Sstevel@tonic-gate 		output_net(fp, &lookup);
4911bbec428eSgjelinek 		output = B_TRUE;
49127c478bd9Sstevel@tonic-gate 	}
49137c478bd9Sstevel@tonic-gate 	(void) zonecfg_endnwifent(handle);
49147c478bd9Sstevel@tonic-gate 	/*
49157c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
49167c478bd9Sstevel@tonic-gate 	 * nothing to output.
49177c478bd9Sstevel@tonic-gate 	 */
49187c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
49197c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
49207c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_NET));
49217c478bd9Sstevel@tonic-gate }
49227c478bd9Sstevel@tonic-gate 
49237c478bd9Sstevel@tonic-gate static void
49247c478bd9Sstevel@tonic-gate output_dev(FILE *fp, struct zone_devtab *devtab)
49257c478bd9Sstevel@tonic-gate {
492627e6fb21Sdp 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_DEVICE));
49277c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_MATCH, devtab->zone_dev_match, B_TRUE);
49287c478bd9Sstevel@tonic-gate }
49297c478bd9Sstevel@tonic-gate 
49307c478bd9Sstevel@tonic-gate static void
49317c478bd9Sstevel@tonic-gate info_dev(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
49327c478bd9Sstevel@tonic-gate {
49337c478bd9Sstevel@tonic-gate 	struct zone_devtab lookup, user;
4934bbec428eSgjelinek 	boolean_t output = B_FALSE;
49357c478bd9Sstevel@tonic-gate 
49367c478bd9Sstevel@tonic-gate 	if (zonecfg_setdevent(handle) != Z_OK)
49377c478bd9Sstevel@tonic-gate 		return;
49387c478bd9Sstevel@tonic-gate 	while (zonecfg_getdevent(handle, &lookup) == Z_OK) {
49397c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
49407c478bd9Sstevel@tonic-gate 			output_dev(fp, &lookup);
49417c478bd9Sstevel@tonic-gate 			continue;
49427c478bd9Sstevel@tonic-gate 		}
4943bbec428eSgjelinek 		if (fill_in_devtab(cmd, &user, B_TRUE) != Z_OK)
49447c478bd9Sstevel@tonic-gate 			continue;
49457c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_dev_match) > 0 &&
49467c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_dev_match, lookup.zone_dev_match) != 0)
49477c478bd9Sstevel@tonic-gate 			continue;	/* no match */
49487c478bd9Sstevel@tonic-gate 		output_dev(fp, &lookup);
4949bbec428eSgjelinek 		output = B_TRUE;
49507c478bd9Sstevel@tonic-gate 	}
49517c478bd9Sstevel@tonic-gate 	(void) zonecfg_enddevent(handle);
49527c478bd9Sstevel@tonic-gate 	/*
49537c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
49547c478bd9Sstevel@tonic-gate 	 * nothing to output.
49557c478bd9Sstevel@tonic-gate 	 */
49567c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
49577c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
49587c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_DEVICE));
49597c478bd9Sstevel@tonic-gate }
49607c478bd9Sstevel@tonic-gate 
49617c478bd9Sstevel@tonic-gate static void
49627c478bd9Sstevel@tonic-gate output_rctl(FILE *fp, struct zone_rctltab *rctltab)
49637c478bd9Sstevel@tonic-gate {
49647c478bd9Sstevel@tonic-gate 	struct zone_rctlvaltab *valptr;
49657c478bd9Sstevel@tonic-gate 
49667c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_RCTL));
49677c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_NAME, rctltab->zone_rctl_name, B_TRUE);
49687c478bd9Sstevel@tonic-gate 	for (valptr = rctltab->zone_rctl_valptr; valptr != NULL;
49697c478bd9Sstevel@tonic-gate 	    valptr = valptr->zone_rctlval_next) {
49707c478bd9Sstevel@tonic-gate 		fprintf(fp, "\t%s: (%s=%s,%s=%s,%s=%s)\n",
49717c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_VALUE),
49727c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_PRIV), valptr->zone_rctlval_priv,
49737c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_LIMIT), valptr->zone_rctlval_limit,
49747c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_ACTION), valptr->zone_rctlval_action);
49757c478bd9Sstevel@tonic-gate 	}
49767c478bd9Sstevel@tonic-gate }
49777c478bd9Sstevel@tonic-gate 
49787c478bd9Sstevel@tonic-gate static void
49797c478bd9Sstevel@tonic-gate info_rctl(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
49807c478bd9Sstevel@tonic-gate {
49817c478bd9Sstevel@tonic-gate 	struct zone_rctltab lookup, user;
4982bbec428eSgjelinek 	boolean_t output = B_FALSE;
49837c478bd9Sstevel@tonic-gate 
49847c478bd9Sstevel@tonic-gate 	if (zonecfg_setrctlent(handle) != Z_OK)
49857c478bd9Sstevel@tonic-gate 		return;
49867c478bd9Sstevel@tonic-gate 	while (zonecfg_getrctlent(handle, &lookup) == Z_OK) {
49877c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
49887c478bd9Sstevel@tonic-gate 			output_rctl(fp, &lookup);
4989bbec428eSgjelinek 		} else if (fill_in_rctltab(cmd, &user, B_TRUE) == Z_OK &&
49907c478bd9Sstevel@tonic-gate 		    (strlen(user.zone_rctl_name) == 0 ||
49917c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_rctl_name, lookup.zone_rctl_name) == 0)) {
49927c478bd9Sstevel@tonic-gate 			output_rctl(fp, &lookup);
4993bbec428eSgjelinek 			output = B_TRUE;
49947c478bd9Sstevel@tonic-gate 		}
49957c478bd9Sstevel@tonic-gate 		zonecfg_free_rctl_value_list(lookup.zone_rctl_valptr);
49967c478bd9Sstevel@tonic-gate 	}
49977c478bd9Sstevel@tonic-gate 	(void) zonecfg_endrctlent(handle);
49987c478bd9Sstevel@tonic-gate 	/*
49997c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
50007c478bd9Sstevel@tonic-gate 	 * nothing to output.
50017c478bd9Sstevel@tonic-gate 	 */
50027c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
50037c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
50047c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_RCTL));
50057c478bd9Sstevel@tonic-gate }
50067c478bd9Sstevel@tonic-gate 
50077c478bd9Sstevel@tonic-gate static void
50087c478bd9Sstevel@tonic-gate output_attr(FILE *fp, struct zone_attrtab *attrtab)
50097c478bd9Sstevel@tonic-gate {
50107c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_ATTR));
50117c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_NAME, attrtab->zone_attr_name, B_TRUE);
50127c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_TYPE, attrtab->zone_attr_type, B_TRUE);
50137c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_VALUE, attrtab->zone_attr_value, B_TRUE);
50147c478bd9Sstevel@tonic-gate }
50157c478bd9Sstevel@tonic-gate 
50167c478bd9Sstevel@tonic-gate static void
50177c478bd9Sstevel@tonic-gate info_attr(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
50187c478bd9Sstevel@tonic-gate {
50197c478bd9Sstevel@tonic-gate 	struct zone_attrtab lookup, user;
5020bbec428eSgjelinek 	boolean_t output = B_FALSE;
50217c478bd9Sstevel@tonic-gate 
50227c478bd9Sstevel@tonic-gate 	if (zonecfg_setattrent(handle) != Z_OK)
50237c478bd9Sstevel@tonic-gate 		return;
50247c478bd9Sstevel@tonic-gate 	while (zonecfg_getattrent(handle, &lookup) == Z_OK) {
50257c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
50267c478bd9Sstevel@tonic-gate 			output_attr(fp, &lookup);
50277c478bd9Sstevel@tonic-gate 			continue;
50287c478bd9Sstevel@tonic-gate 		}
5029bbec428eSgjelinek 		if (fill_in_attrtab(cmd, &user, B_TRUE) != Z_OK)
50307c478bd9Sstevel@tonic-gate 			continue;
50317c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_attr_name) > 0 &&
50327c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_attr_name, lookup.zone_attr_name) != 0)
50337c478bd9Sstevel@tonic-gate 			continue;	/* no match */
50347c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_attr_type) > 0 &&
50357c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_attr_type, lookup.zone_attr_type) != 0)
50367c478bd9Sstevel@tonic-gate 			continue;	/* no match */
50377c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_attr_value) > 0 &&
50387c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_attr_value, lookup.zone_attr_value) != 0)
50397c478bd9Sstevel@tonic-gate 			continue;	/* no match */
50407c478bd9Sstevel@tonic-gate 		output_attr(fp, &lookup);
5041bbec428eSgjelinek 		output = B_TRUE;
50427c478bd9Sstevel@tonic-gate 	}
50437c478bd9Sstevel@tonic-gate 	(void) zonecfg_endattrent(handle);
50447c478bd9Sstevel@tonic-gate 	/*
50457c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
50467c478bd9Sstevel@tonic-gate 	 * nothing to output.
50477c478bd9Sstevel@tonic-gate 	 */
50487c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
50497c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
50507c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR));
50517c478bd9Sstevel@tonic-gate }
50527c478bd9Sstevel@tonic-gate 
5053fa9e4066Sahrens static void
5054fa9e4066Sahrens output_ds(FILE *fp, struct zone_dstab *dstab)
5055fa9e4066Sahrens {
5056fa9e4066Sahrens 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_DATASET));
5057fa9e4066Sahrens 	output_prop(fp, PT_NAME, dstab->zone_dataset_name, B_TRUE);
5058fa9e4066Sahrens }
5059fa9e4066Sahrens 
5060fa9e4066Sahrens static void
5061fa9e4066Sahrens info_ds(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
5062fa9e4066Sahrens {
5063fa9e4066Sahrens 	struct zone_dstab lookup, user;
5064bbec428eSgjelinek 	boolean_t output = B_FALSE;
5065fa9e4066Sahrens 
50660209230bSgjelinek 	if (zonecfg_setdsent(handle) != Z_OK)
5067fa9e4066Sahrens 		return;
5068fa9e4066Sahrens 	while (zonecfg_getdsent(handle, &lookup) == Z_OK) {
5069fa9e4066Sahrens 		if (cmd->cmd_prop_nv_pairs == 0) {
5070fa9e4066Sahrens 			output_ds(fp, &lookup);
5071fa9e4066Sahrens 			continue;
5072fa9e4066Sahrens 		}
5073bbec428eSgjelinek 		if (fill_in_dstab(cmd, &user, B_TRUE) != Z_OK)
5074fa9e4066Sahrens 			continue;
5075fa9e4066Sahrens 		if (strlen(user.zone_dataset_name) > 0 &&
5076fa9e4066Sahrens 		    strcmp(user.zone_dataset_name,
5077fa9e4066Sahrens 		    lookup.zone_dataset_name) != 0)
5078fa9e4066Sahrens 			continue;	/* no match */
5079fa9e4066Sahrens 		output_ds(fp, &lookup);
5080bbec428eSgjelinek 		output = B_TRUE;
5081fa9e4066Sahrens 	}
5082fa9e4066Sahrens 	(void) zonecfg_enddsent(handle);
5083fa9e4066Sahrens 	/*
5084fa9e4066Sahrens 	 * If a property n/v pair was specified, warn the user if there was
5085fa9e4066Sahrens 	 * nothing to output.
5086fa9e4066Sahrens 	 */
5087fa9e4066Sahrens 	if (!output && cmd->cmd_prop_nv_pairs > 0)
5088fa9e4066Sahrens 		(void) printf(gettext("No such %s resource.\n"),
5089fa9e4066Sahrens 		    rt_to_str(RT_DATASET));
5090fa9e4066Sahrens }
5091fa9e4066Sahrens 
50920209230bSgjelinek static void
50930209230bSgjelinek output_pset(FILE *fp, struct zone_psettab *psettab)
50940209230bSgjelinek {
50950209230bSgjelinek 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_DCPU));
50960209230bSgjelinek 	if (strcmp(psettab->zone_ncpu_min, psettab->zone_ncpu_max) == 0)
50970209230bSgjelinek 		(void) fprintf(fp, "\t%s: %s\n", pt_to_str(PT_NCPUS),
50980209230bSgjelinek 		    psettab->zone_ncpu_max);
50990209230bSgjelinek 	else
51000209230bSgjelinek 		(void) fprintf(fp, "\t%s: %s-%s\n", pt_to_str(PT_NCPUS),
51010209230bSgjelinek 		    psettab->zone_ncpu_min, psettab->zone_ncpu_max);
51020209230bSgjelinek 	if (psettab->zone_importance[0] != '\0')
51030209230bSgjelinek 		(void) fprintf(fp, "\t%s: %s\n", pt_to_str(PT_IMPORTANCE),
51040209230bSgjelinek 		    psettab->zone_importance);
51050209230bSgjelinek }
51060209230bSgjelinek 
51070209230bSgjelinek static void
51080209230bSgjelinek info_pset(zone_dochandle_t handle, FILE *fp)
51090209230bSgjelinek {
51100209230bSgjelinek 	struct zone_psettab lookup;
51110209230bSgjelinek 
51120209230bSgjelinek 	if (zonecfg_getpsetent(handle, &lookup) == Z_OK)
51130209230bSgjelinek 		output_pset(fp, &lookup);
51140209230bSgjelinek }
51150209230bSgjelinek 
51160209230bSgjelinek static void
5117c97ad5cdSakolb output_pcap(FILE *fp)
5118c97ad5cdSakolb {
5119c97ad5cdSakolb 	uint64_t cap;
5120c97ad5cdSakolb 
5121c97ad5cdSakolb 	if (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &cap) == Z_OK) {
5122c97ad5cdSakolb 		float scaled = (float)cap / 100;
5123c97ad5cdSakolb 		(void) fprintf(fp, "%s:\n", rt_to_str(RT_PCAP));
5124c97ad5cdSakolb 		(void) fprintf(fp, "\t[%s: %.2f]\n", pt_to_str(PT_NCPUS),
5125c97ad5cdSakolb 		    scaled);
5126c97ad5cdSakolb 	}
5127c97ad5cdSakolb }
5128c97ad5cdSakolb 
5129c97ad5cdSakolb static void
5130c97ad5cdSakolb info_pcap(FILE *fp)
5131c97ad5cdSakolb {
5132c97ad5cdSakolb 	output_pcap(fp);
5133c97ad5cdSakolb }
5134c97ad5cdSakolb 
5135c97ad5cdSakolb 
5136c97ad5cdSakolb static void
51370209230bSgjelinek info_aliased_rctl(zone_dochandle_t handle, FILE *fp, char *alias)
51380209230bSgjelinek {
51390209230bSgjelinek 	uint64_t limit;
51400209230bSgjelinek 
51410209230bSgjelinek 	if (zonecfg_get_aliased_rctl(handle, alias, &limit) == Z_OK) {
51420209230bSgjelinek 		/* convert memory based properties */
51430209230bSgjelinek 		if (strcmp(alias, ALIAS_MAXSHMMEM) == 0) {
51440209230bSgjelinek 			char buf[128];
51450209230bSgjelinek 
51460209230bSgjelinek 			(void) snprintf(buf, sizeof (buf), "%llu", limit);
51470209230bSgjelinek 			bytes_to_units(buf, buf, sizeof (buf));
51480209230bSgjelinek 			(void) fprintf(fp, "[%s: %s]\n", alias, buf);
51490209230bSgjelinek 			return;
51500209230bSgjelinek 		}
51510209230bSgjelinek 
51520209230bSgjelinek 		(void) fprintf(fp, "[%s: %llu]\n", alias, limit);
51530209230bSgjelinek 	}
51540209230bSgjelinek }
51550209230bSgjelinek 
51560209230bSgjelinek static void
51570209230bSgjelinek bytes_to_units(char *str, char *buf, int bufsize)
51580209230bSgjelinek {
51590209230bSgjelinek 	unsigned long long num;
51600209230bSgjelinek 	unsigned long long save = 0;
51610209230bSgjelinek 	char *units = "BKMGT";
51620209230bSgjelinek 	char *up = units;
51630209230bSgjelinek 
51640209230bSgjelinek 	num = strtoll(str, NULL, 10);
51650209230bSgjelinek 
51660209230bSgjelinek 	if (num < 1024) {
51670209230bSgjelinek 		(void) snprintf(buf, bufsize, "%llu", num);
51680209230bSgjelinek 		return;
51690209230bSgjelinek 	}
51700209230bSgjelinek 
51710209230bSgjelinek 	while ((num >= 1024) && (*up != 'T')) {
51720209230bSgjelinek 		up++; /* next unit of measurement */
51730209230bSgjelinek 		save = num;
51740209230bSgjelinek 		num = (num + 512) >> 10;
51750209230bSgjelinek 	}
51760209230bSgjelinek 
51770209230bSgjelinek 	/* check if we should output a fraction.  snprintf will round for us */
51780209230bSgjelinek 	if (save % 1024 != 0 && ((save >> 10) < 10))
51790209230bSgjelinek 		(void) snprintf(buf, bufsize, "%2.1f%c", ((float)save / 1024),
51800209230bSgjelinek 		    *up);
51810209230bSgjelinek 	else
51820209230bSgjelinek 		(void) snprintf(buf, bufsize, "%llu%c", num, *up);
51830209230bSgjelinek }
51840209230bSgjelinek 
51850209230bSgjelinek static void
51860209230bSgjelinek output_mcap(FILE *fp, struct zone_mcaptab *mcaptab, int showswap,
51870209230bSgjelinek     uint64_t maxswap, int showlocked, uint64_t maxlocked)
51880209230bSgjelinek {
51890209230bSgjelinek 	char buf[128];
51900209230bSgjelinek 
51910209230bSgjelinek 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_MCAP));
51920209230bSgjelinek 	if (mcaptab->zone_physmem_cap[0] != '\0') {
51930209230bSgjelinek 		bytes_to_units(mcaptab->zone_physmem_cap, buf, sizeof (buf));
51940209230bSgjelinek 		output_prop(fp, PT_PHYSICAL, buf, B_TRUE);
51950209230bSgjelinek 	}
51960209230bSgjelinek 
51970209230bSgjelinek 	if (showswap == Z_OK) {
51980209230bSgjelinek 		(void) snprintf(buf, sizeof (buf), "%llu", maxswap);
51990209230bSgjelinek 		bytes_to_units(buf, buf, sizeof (buf));
52000209230bSgjelinek 		output_prop(fp, PT_SWAP, buf, B_TRUE);
52010209230bSgjelinek 	}
52020209230bSgjelinek 
52030209230bSgjelinek 	if (showlocked == Z_OK) {
52040209230bSgjelinek 		(void) snprintf(buf, sizeof (buf), "%llu", maxlocked);
52050209230bSgjelinek 		bytes_to_units(buf, buf, sizeof (buf));
52060209230bSgjelinek 		output_prop(fp, PT_LOCKED, buf, B_TRUE);
52070209230bSgjelinek 	}
52080209230bSgjelinek }
52090209230bSgjelinek 
52100209230bSgjelinek static void
52110209230bSgjelinek info_mcap(zone_dochandle_t handle, FILE *fp)
52120209230bSgjelinek {
52130209230bSgjelinek 	int res1, res2, res3;
52140209230bSgjelinek 	uint64_t swap_limit;
52150209230bSgjelinek 	uint64_t locked_limit;
52160209230bSgjelinek 	struct zone_mcaptab lookup;
52170209230bSgjelinek 
52180209230bSgjelinek 	bzero(&lookup, sizeof (lookup));
52190209230bSgjelinek 	res1 = zonecfg_getmcapent(handle, &lookup);
52200209230bSgjelinek 	res2 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP, &swap_limit);
52210209230bSgjelinek 	res3 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM,
52220209230bSgjelinek 	    &locked_limit);
52230209230bSgjelinek 
52240209230bSgjelinek 	if (res1 == Z_OK || res2 == Z_OK || res3 == Z_OK)
52250209230bSgjelinek 		output_mcap(fp, &lookup, res2, swap_limit, res3, locked_limit);
52260209230bSgjelinek }
52270209230bSgjelinek 
5228cb8a054bSGlenn Faden static void
5229cb8a054bSGlenn Faden output_auth(FILE *fp, struct zone_admintab *admintab)
5230cb8a054bSGlenn Faden {
5231cb8a054bSGlenn Faden 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_ADMIN));
5232cb8a054bSGlenn Faden 	output_prop(fp, PT_USER, admintab->zone_admin_user, B_TRUE);
5233cb8a054bSGlenn Faden 	output_prop(fp, PT_AUTHS, admintab->zone_admin_auths, B_TRUE);
5234cb8a054bSGlenn Faden }
5235cb8a054bSGlenn Faden 
5236cb8a054bSGlenn Faden static void
5237cb8a054bSGlenn Faden info_auth(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
5238cb8a054bSGlenn Faden {
5239cb8a054bSGlenn Faden 	struct zone_admintab lookup, user;
5240cb8a054bSGlenn Faden 	boolean_t output = B_FALSE;
5241cb8a054bSGlenn Faden 	int err;
5242cb8a054bSGlenn Faden 
5243cb8a054bSGlenn Faden 	if ((err = zonecfg_setadminent(handle)) != Z_OK) {
5244cb8a054bSGlenn Faden 		zone_perror(zone, err, B_TRUE);
5245cb8a054bSGlenn Faden 		return;
5246cb8a054bSGlenn Faden 	}
5247cb8a054bSGlenn Faden 	while (zonecfg_getadminent(handle, &lookup) == Z_OK) {
5248cb8a054bSGlenn Faden 		if (cmd->cmd_prop_nv_pairs == 0) {
5249cb8a054bSGlenn Faden 			output_auth(fp, &lookup);
5250cb8a054bSGlenn Faden 			continue;
5251cb8a054bSGlenn Faden 		}
5252cb8a054bSGlenn Faden 		if (fill_in_admintab(cmd, &user, B_TRUE) != Z_OK)
5253cb8a054bSGlenn Faden 			continue;
5254cb8a054bSGlenn Faden 		if (strlen(user.zone_admin_user) > 0 &&
5255cb8a054bSGlenn Faden 		    strcmp(user.zone_admin_user, lookup.zone_admin_user) != 0)
5256cb8a054bSGlenn Faden 			continue;	/* no match */
5257cb8a054bSGlenn Faden 		output_auth(fp, &lookup);
5258cb8a054bSGlenn Faden 		output = B_TRUE;
5259cb8a054bSGlenn Faden 	}
5260cb8a054bSGlenn Faden 	(void) zonecfg_endadminent(handle);
5261cb8a054bSGlenn Faden 	/*
5262cb8a054bSGlenn Faden 	 * If a property n/v pair was specified, warn the user if there was
5263cb8a054bSGlenn Faden 	 * nothing to output.
5264cb8a054bSGlenn Faden 	 */
5265cb8a054bSGlenn Faden 	if (!output && cmd->cmd_prop_nv_pairs > 0)
5266cb8a054bSGlenn Faden 		(void) printf(gettext("No such %s resource.\n"),
5267cb8a054bSGlenn Faden 		    rt_to_str(RT_ADMIN));
5268cb8a054bSGlenn Faden }
5269cb8a054bSGlenn Faden 
52707c478bd9Sstevel@tonic-gate void
52717c478bd9Sstevel@tonic-gate info_func(cmd_t *cmd)
52727c478bd9Sstevel@tonic-gate {
52737c478bd9Sstevel@tonic-gate 	FILE *fp = stdout;
5274bbec428eSgjelinek 	boolean_t need_to_close = B_FALSE;
52757c478bd9Sstevel@tonic-gate 	char *pager;
52760209230bSgjelinek 	int type;
52770209230bSgjelinek 	int res1, res2;
52780209230bSgjelinek 	uint64_t swap_limit;
52790209230bSgjelinek 	uint64_t locked_limit;
52803042b8b5Sbatschul 	struct stat statbuf;
52817c478bd9Sstevel@tonic-gate 
52827c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
52837c478bd9Sstevel@tonic-gate 
5284bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
52857c478bd9Sstevel@tonic-gate 		return;
52867c478bd9Sstevel@tonic-gate 
52877c478bd9Sstevel@tonic-gate 	/* don't page error output */
52887c478bd9Sstevel@tonic-gate 	if (interactive_mode) {
52897c478bd9Sstevel@tonic-gate 		if ((pager = getenv("PAGER")) == NULL)
52907c478bd9Sstevel@tonic-gate 			pager = PAGER;
52913042b8b5Sbatschul 
52923042b8b5Sbatschul 		if (stat(pager, &statbuf) == 0) {
52937c478bd9Sstevel@tonic-gate 			if ((fp = popen(pager, "w")) != NULL)
5294bbec428eSgjelinek 				need_to_close = B_TRUE;
52953042b8b5Sbatschul 			else
52963042b8b5Sbatschul 				fp = stdout;
52973042b8b5Sbatschul 		} else {
52983042b8b5Sbatschul 			zerr(gettext("PAGER %s does not exist (%s)."),
52993042b8b5Sbatschul 			    pager, strerror(errno));
53003042b8b5Sbatschul 		}
53013042b8b5Sbatschul 
53027c478bd9Sstevel@tonic-gate 		setbuf(fp, NULL);
53037c478bd9Sstevel@tonic-gate 	}
53047c478bd9Sstevel@tonic-gate 
53057c478bd9Sstevel@tonic-gate 	if (!global_scope) {
53067c478bd9Sstevel@tonic-gate 		switch (resource_scope) {
53077c478bd9Sstevel@tonic-gate 		case RT_FS:
53087c478bd9Sstevel@tonic-gate 			output_fs(fp, &in_progress_fstab);
53097c478bd9Sstevel@tonic-gate 			break;
53107c478bd9Sstevel@tonic-gate 		case RT_NET:
53117c478bd9Sstevel@tonic-gate 			output_net(fp, &in_progress_nwiftab);
53127c478bd9Sstevel@tonic-gate 			break;
53137c478bd9Sstevel@tonic-gate 		case RT_DEVICE:
53147c478bd9Sstevel@tonic-gate 			output_dev(fp, &in_progress_devtab);
53157c478bd9Sstevel@tonic-gate 			break;
53167c478bd9Sstevel@tonic-gate 		case RT_RCTL:
53177c478bd9Sstevel@tonic-gate 			output_rctl(fp, &in_progress_rctltab);
53187c478bd9Sstevel@tonic-gate 			break;
53197c478bd9Sstevel@tonic-gate 		case RT_ATTR:
53207c478bd9Sstevel@tonic-gate 			output_attr(fp, &in_progress_attrtab);
53217c478bd9Sstevel@tonic-gate 			break;
5322fa9e4066Sahrens 		case RT_DATASET:
5323fa9e4066Sahrens 			output_ds(fp, &in_progress_dstab);
5324fa9e4066Sahrens 			break;
53250209230bSgjelinek 		case RT_DCPU:
53260209230bSgjelinek 			output_pset(fp, &in_progress_psettab);
53270209230bSgjelinek 			break;
5328c97ad5cdSakolb 		case RT_PCAP:
5329c97ad5cdSakolb 			output_pcap(fp);
5330c97ad5cdSakolb 			break;
53310209230bSgjelinek 		case RT_MCAP:
53320209230bSgjelinek 			res1 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP,
53330209230bSgjelinek 			    &swap_limit);
53340209230bSgjelinek 			res2 = zonecfg_get_aliased_rctl(handle,
53350209230bSgjelinek 			    ALIAS_MAXLOCKEDMEM, &locked_limit);
53360209230bSgjelinek 			output_mcap(fp, &in_progress_mcaptab, res1, swap_limit,
53370209230bSgjelinek 			    res2, locked_limit);
53380209230bSgjelinek 			break;
5339cb8a054bSGlenn Faden 		case RT_ADMIN:
5340cb8a054bSGlenn Faden 			output_auth(fp, &in_progress_admintab);
5341cb8a054bSGlenn Faden 			break;
53427c478bd9Sstevel@tonic-gate 		}
53437c478bd9Sstevel@tonic-gate 		goto cleanup;
53447c478bd9Sstevel@tonic-gate 	}
53457c478bd9Sstevel@tonic-gate 
53460209230bSgjelinek 	type = cmd->cmd_res_type;
53470209230bSgjelinek 
53480209230bSgjelinek 	if (gz_invalid_rt_property(type)) {
53490209230bSgjelinek 		zerr(gettext("%s is not a valid property for the global zone."),
53500209230bSgjelinek 		    rt_to_str(type));
53510209230bSgjelinek 		goto cleanup;
53520209230bSgjelinek 	}
53530209230bSgjelinek 
53540209230bSgjelinek 	if (gz_invalid_resource(type)) {
53550209230bSgjelinek 		zerr(gettext("%s is not a valid resource for the global zone."),
53560209230bSgjelinek 		    rt_to_str(type));
53570209230bSgjelinek 		goto cleanup;
53580209230bSgjelinek 	}
53590209230bSgjelinek 
53607c478bd9Sstevel@tonic-gate 	switch (cmd->cmd_res_type) {
53617c478bd9Sstevel@tonic-gate 	case RT_UNKNOWN:
5362087719fdSdp 		info_zonename(handle, fp);
53630209230bSgjelinek 		if (!global_zone) {
53647c478bd9Sstevel@tonic-gate 			info_zonepath(handle, fp);
53659acbbeafSnn35248 			info_brand(handle, fp);
53667c478bd9Sstevel@tonic-gate 			info_autoboot(handle, fp);
53673f2f09c1Sdp 			info_bootargs(handle, fp);
53680209230bSgjelinek 		}
53697c478bd9Sstevel@tonic-gate 		info_pool(handle, fp);
53700209230bSgjelinek 		if (!global_zone) {
5371ffbafc53Scomay 			info_limitpriv(handle, fp);
53720209230bSgjelinek 			info_sched(handle, fp);
5373f4b3ec61Sdh155122 			info_iptype(handle, fp);
53745679c89fSjv227347 			info_hostid(handle, fp);
53750fbb751dSJohn Levon 			info_fs_allowed(handle, fp);
53760209230bSgjelinek 		}
53770209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXLWPS);
5378ff19e029SMenno Lageman 		info_aliased_rctl(handle, fp, ALIAS_MAXPROCS);
53790209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSHMMEM);
53800209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSHMIDS);
53810209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXMSGIDS);
53820209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSEMIDS);
53830209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_SHARES);
53840209230bSgjelinek 		if (!global_zone) {
53857c478bd9Sstevel@tonic-gate 			info_fs(handle, fp, cmd);
53867c478bd9Sstevel@tonic-gate 			info_net(handle, fp, cmd);
53877c478bd9Sstevel@tonic-gate 			info_dev(handle, fp, cmd);
53880209230bSgjelinek 		}
53890209230bSgjelinek 		info_pset(handle, fp);
5390c97ad5cdSakolb 		info_pcap(fp);
53910209230bSgjelinek 		info_mcap(handle, fp);
53920209230bSgjelinek 		if (!global_zone) {
53937c478bd9Sstevel@tonic-gate 			info_attr(handle, fp, cmd);
5394fa9e4066Sahrens 			info_ds(handle, fp, cmd);
5395cb8a054bSGlenn Faden 			info_auth(handle, fp, cmd);
53960209230bSgjelinek 		}
53970209230bSgjelinek 		info_rctl(handle, fp, cmd);
53987c478bd9Sstevel@tonic-gate 		break;
5399087719fdSdp 	case RT_ZONENAME:
5400087719fdSdp 		info_zonename(handle, fp);
5401087719fdSdp 		break;
54027c478bd9Sstevel@tonic-gate 	case RT_ZONEPATH:
54037c478bd9Sstevel@tonic-gate 		info_zonepath(handle, fp);
54047c478bd9Sstevel@tonic-gate 		break;
54059acbbeafSnn35248 	case RT_BRAND:
54069acbbeafSnn35248 		info_brand(handle, fp);
54079acbbeafSnn35248 		break;
54087c478bd9Sstevel@tonic-gate 	case RT_AUTOBOOT:
54097c478bd9Sstevel@tonic-gate 		info_autoboot(handle, fp);
54107c478bd9Sstevel@tonic-gate 		break;
54117c478bd9Sstevel@tonic-gate 	case RT_POOL:
54127c478bd9Sstevel@tonic-gate 		info_pool(handle, fp);
54137c478bd9Sstevel@tonic-gate 		break;
5414ffbafc53Scomay 	case RT_LIMITPRIV:
5415ffbafc53Scomay 		info_limitpriv(handle, fp);
5416ffbafc53Scomay 		break;
54173f2f09c1Sdp 	case RT_BOOTARGS:
54183f2f09c1Sdp 		info_bootargs(handle, fp);
54193f2f09c1Sdp 		break;
54200209230bSgjelinek 	case RT_SCHED:
54210209230bSgjelinek 		info_sched(handle, fp);
54220209230bSgjelinek 		break;
5423f4b3ec61Sdh155122 	case RT_IPTYPE:
5424f4b3ec61Sdh155122 		info_iptype(handle, fp);
5425f4b3ec61Sdh155122 		break;
54260209230bSgjelinek 	case RT_MAXLWPS:
54270209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXLWPS);
54280209230bSgjelinek 		break;
5429ff19e029SMenno Lageman 	case RT_MAXPROCS:
5430ff19e029SMenno Lageman 		info_aliased_rctl(handle, fp, ALIAS_MAXPROCS);
5431ff19e029SMenno Lageman 		break;
54320209230bSgjelinek 	case RT_MAXSHMMEM:
54330209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSHMMEM);
54340209230bSgjelinek 		break;
54350209230bSgjelinek 	case RT_MAXSHMIDS:
54360209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSHMIDS);
54370209230bSgjelinek 		break;
54380209230bSgjelinek 	case RT_MAXMSGIDS:
54390209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXMSGIDS);
54400209230bSgjelinek 		break;
54410209230bSgjelinek 	case RT_MAXSEMIDS:
54420209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSEMIDS);
54430209230bSgjelinek 		break;
54440209230bSgjelinek 	case RT_SHARES:
54450209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_SHARES);
54460209230bSgjelinek 		break;
54477c478bd9Sstevel@tonic-gate 	case RT_FS:
54487c478bd9Sstevel@tonic-gate 		info_fs(handle, fp, cmd);
54497c478bd9Sstevel@tonic-gate 		break;
54507c478bd9Sstevel@tonic-gate 	case RT_NET:
54517c478bd9Sstevel@tonic-gate 		info_net(handle, fp, cmd);
54527c478bd9Sstevel@tonic-gate 		break;
54537c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
54547c478bd9Sstevel@tonic-gate 		info_dev(handle, fp, cmd);
54557c478bd9Sstevel@tonic-gate 		break;
54567c478bd9Sstevel@tonic-gate 	case RT_RCTL:
54577c478bd9Sstevel@tonic-gate 		info_rctl(handle, fp, cmd);
54587c478bd9Sstevel@tonic-gate 		break;
54597c478bd9Sstevel@tonic-gate 	case RT_ATTR:
54607c478bd9Sstevel@tonic-gate 		info_attr(handle, fp, cmd);
54617c478bd9Sstevel@tonic-gate 		break;
5462fa9e4066Sahrens 	case RT_DATASET:
5463fa9e4066Sahrens 		info_ds(handle, fp, cmd);
5464fa9e4066Sahrens 		break;
54650209230bSgjelinek 	case RT_DCPU:
54660209230bSgjelinek 		info_pset(handle, fp);
54670209230bSgjelinek 		break;
5468c97ad5cdSakolb 	case RT_PCAP:
5469c97ad5cdSakolb 		info_pcap(fp);
5470c97ad5cdSakolb 		break;
54710209230bSgjelinek 	case RT_MCAP:
54720209230bSgjelinek 		info_mcap(handle, fp);
54730209230bSgjelinek 		break;
54745679c89fSjv227347 	case RT_HOSTID:
54755679c89fSjv227347 		info_hostid(handle, fp);
54765679c89fSjv227347 		break;
5477cb8a054bSGlenn Faden 	case RT_ADMIN:
5478cb8a054bSGlenn Faden 		info_auth(handle, fp, cmd);
5479cb8a054bSGlenn Faden 		break;
54800fbb751dSJohn Levon 	case RT_FS_ALLOWED:
54810fbb751dSJohn Levon 		info_fs_allowed(handle, fp);
54820fbb751dSJohn Levon 		break;
54837c478bd9Sstevel@tonic-gate 	default:
54847c478bd9Sstevel@tonic-gate 		zone_perror(rt_to_str(cmd->cmd_res_type), Z_NO_RESOURCE_TYPE,
5485bbec428eSgjelinek 		    B_TRUE);
54867c478bd9Sstevel@tonic-gate 	}
54877c478bd9Sstevel@tonic-gate 
54887c478bd9Sstevel@tonic-gate cleanup:
54897c478bd9Sstevel@tonic-gate 	if (need_to_close)
54907c478bd9Sstevel@tonic-gate 		(void) pclose(fp);
54917c478bd9Sstevel@tonic-gate }
54927c478bd9Sstevel@tonic-gate 
5493087719fdSdp /*
5494087719fdSdp  * Helper function for verify-- checks that a required string property
5495087719fdSdp  * exists.
5496087719fdSdp  */
5497087719fdSdp static void
5498087719fdSdp check_reqd_prop(char *attr, int rt, int pt, int *ret_val)
54997c478bd9Sstevel@tonic-gate {
5500087719fdSdp 	if (strlen(attr) == 0) {
5501087719fdSdp 		zerr(gettext("%s: %s not specified"), rt_to_str(rt),
5502087719fdSdp 		    pt_to_str(pt));
5503bbec428eSgjelinek 		saw_error = B_TRUE;
5504087719fdSdp 		if (*ret_val == Z_OK)
5505087719fdSdp 			*ret_val = Z_REQD_PROPERTY_MISSING;
55067c478bd9Sstevel@tonic-gate 	}
55077c478bd9Sstevel@tonic-gate }
55087c478bd9Sstevel@tonic-gate 
55099acbbeafSnn35248 static int
55109acbbeafSnn35248 do_subproc(char *cmdbuf)
55119acbbeafSnn35248 {
55129acbbeafSnn35248 	char inbuf[MAX_CMD_LEN];
55139acbbeafSnn35248 	FILE *file;
55149acbbeafSnn35248 	int status;
55159acbbeafSnn35248 
55169acbbeafSnn35248 	file = popen(cmdbuf, "r");
55179acbbeafSnn35248 	if (file == NULL) {
55189acbbeafSnn35248 		zerr(gettext("Could not launch: %s"), cmdbuf);
55199acbbeafSnn35248 		return (-1);
55209acbbeafSnn35248 	}
55219acbbeafSnn35248 
55229acbbeafSnn35248 	while (fgets(inbuf, sizeof (inbuf), file) != NULL)
55239acbbeafSnn35248 		fprintf(stderr, "%s", inbuf);
55249acbbeafSnn35248 	status = pclose(file);
55259acbbeafSnn35248 
55269acbbeafSnn35248 	if (WIFSIGNALED(status)) {
55279acbbeafSnn35248 		zerr(gettext("%s unexpectedly terminated due to signal %d"),
55289acbbeafSnn35248 		    cmdbuf, WTERMSIG(status));
55299acbbeafSnn35248 		return (-1);
55309acbbeafSnn35248 	}
55319acbbeafSnn35248 	assert(WIFEXITED(status));
55329acbbeafSnn35248 	return (WEXITSTATUS(status));
55339acbbeafSnn35248 }
55349acbbeafSnn35248 
55359acbbeafSnn35248 static int
55369acbbeafSnn35248 brand_verify(zone_dochandle_t handle)
55379acbbeafSnn35248 {
55386e65f9afSnn35248 	char xml_file[32];
55399acbbeafSnn35248 	char cmdbuf[MAX_CMD_LEN];
5540123807fbSedp 	brand_handle_t bh;
55419acbbeafSnn35248 	char brand[MAXNAMELEN];
55429acbbeafSnn35248 	int err;
55439acbbeafSnn35248 
55449acbbeafSnn35248 	if (zonecfg_get_brand(handle, brand, sizeof (brand)) != Z_OK) {
55459acbbeafSnn35248 		zerr("%s: %s\n", zone, gettext("could not get zone brand"));
55469acbbeafSnn35248 		return (Z_INVALID_DOCUMENT);
55479acbbeafSnn35248 	}
5548123807fbSedp 	if ((bh = brand_open(brand)) == NULL) {
55499acbbeafSnn35248 		zerr("%s: %s\n", zone, gettext("unknown brand."));
55509acbbeafSnn35248 		return (Z_INVALID_DOCUMENT);
55519acbbeafSnn35248 	}
55529acbbeafSnn35248 
55539acbbeafSnn35248 	/*
55549acbbeafSnn35248 	 * Fetch the verify command, if any, from the brand configuration
55559acbbeafSnn35248 	 * and build the command line to execute it.
55569acbbeafSnn35248 	 */
55579acbbeafSnn35248 	strcpy(cmdbuf, EXEC_PREFIX);
5558123807fbSedp 	err = brand_get_verify_cfg(bh, cmdbuf + EXEC_LEN,
55599acbbeafSnn35248 	    sizeof (cmdbuf) - (EXEC_LEN + (strlen(xml_file) + 1)));
5560123807fbSedp 	brand_close(bh);
55619acbbeafSnn35248 	if (err != Z_OK) {
55629acbbeafSnn35248 		zerr("%s: %s\n", zone,
55639acbbeafSnn35248 		    gettext("could not get brand verification command"));
55649acbbeafSnn35248 		return (Z_INVALID_DOCUMENT);
55659acbbeafSnn35248 	}
55669acbbeafSnn35248 
55679acbbeafSnn35248 	/*
55689acbbeafSnn35248 	 * If the brand doesn't provide a verification routine, we just
55699acbbeafSnn35248 	 * return success.
55709acbbeafSnn35248 	 */
55719acbbeafSnn35248 	if (strlen(cmdbuf) == EXEC_LEN)
55729acbbeafSnn35248 		return (Z_OK);
55739acbbeafSnn35248 
55749acbbeafSnn35248 	/*
55759acbbeafSnn35248 	 * Dump the current config information for this zone to a file.
55769acbbeafSnn35248 	 */
55776e65f9afSnn35248 	strcpy(xml_file, "/tmp/zonecfg_verify.XXXXXX");
55789acbbeafSnn35248 	if (mkstemp(xml_file) == NULL)
55799acbbeafSnn35248 		return (Z_TEMP_FILE);
55809acbbeafSnn35248 	if ((err = zonecfg_verify_save(handle, xml_file)) != Z_OK) {
55819acbbeafSnn35248 		(void) unlink(xml_file);
55829acbbeafSnn35248 		return (err);
55839acbbeafSnn35248 	}
55849acbbeafSnn35248 
55859acbbeafSnn35248 	/*
55869acbbeafSnn35248 	 * Execute the verification command.
55879acbbeafSnn35248 	 */
55889acbbeafSnn35248 	if ((strlcat(cmdbuf, " ", MAX_CMD_LEN) >= MAX_CMD_LEN) ||
55899acbbeafSnn35248 	    (strlcat(cmdbuf, xml_file, MAX_CMD_LEN) >= MAX_CMD_LEN)) {
55909acbbeafSnn35248 		err = Z_BRAND_ERROR;
55919acbbeafSnn35248 	} else {
55929acbbeafSnn35248 		err = do_subproc(cmdbuf);
55939acbbeafSnn35248 	}
55949acbbeafSnn35248 
55959acbbeafSnn35248 	(void) unlink(xml_file);
55969acbbeafSnn35248 	return ((err == Z_OK) ? Z_OK : Z_BRAND_ERROR);
55979acbbeafSnn35248 }
55989acbbeafSnn35248 
55997c478bd9Sstevel@tonic-gate /*
56007c478bd9Sstevel@tonic-gate  * See the DTD for which attributes are required for which resources.
56017c478bd9Sstevel@tonic-gate  *
56027c478bd9Sstevel@tonic-gate  * This function can be called by commit_func(), which needs to save things,
56037c478bd9Sstevel@tonic-gate  * in addition to the general call from parse_and_run(), which doesn't need
56047c478bd9Sstevel@tonic-gate  * things saved.  Since the parameters are standardized, we distinguish by
56057c478bd9Sstevel@tonic-gate  * having commit_func() call here with cmd->cmd_arg set to "save" to indicate
56067c478bd9Sstevel@tonic-gate  * that a save is needed.
56077c478bd9Sstevel@tonic-gate  */
56087c478bd9Sstevel@tonic-gate void
56097c478bd9Sstevel@tonic-gate verify_func(cmd_t *cmd)
56107c478bd9Sstevel@tonic-gate {
56117c478bd9Sstevel@tonic-gate 	struct zone_nwiftab nwiftab;
56127c478bd9Sstevel@tonic-gate 	struct zone_fstab fstab;
56137c478bd9Sstevel@tonic-gate 	struct zone_attrtab attrtab;
56147c478bd9Sstevel@tonic-gate 	struct zone_rctltab rctltab;
5615fa9e4066Sahrens 	struct zone_dstab dstab;
56160209230bSgjelinek 	struct zone_psettab psettab;
5617cb8a054bSGlenn Faden 	struct zone_admintab admintab;
56187c478bd9Sstevel@tonic-gate 	char zonepath[MAXPATHLEN];
56190209230bSgjelinek 	char sched[MAXNAMELEN];
56209acbbeafSnn35248 	char brand[MAXNAMELEN];
56215679c89fSjv227347 	char hostidp[HW_HOSTID_LEN];
56220fbb751dSJohn Levon 	char fsallowedp[ZONE_FS_ALLOWED_MAX];
56237c478bd9Sstevel@tonic-gate 	int err, ret_val = Z_OK, arg;
5624c97ad5cdSakolb 	int pset_res;
5625bbec428eSgjelinek 	boolean_t save = B_FALSE;
5626bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
5627f4b3ec61Sdh155122 	zone_iptype_t iptype;
56280209230bSgjelinek 	boolean_t has_cpu_shares = B_FALSE;
5629c97ad5cdSakolb 	boolean_t has_cpu_cap = B_FALSE;
56307c478bd9Sstevel@tonic-gate 
56317c478bd9Sstevel@tonic-gate 	optind = 0;
56327ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
56337c478bd9Sstevel@tonic-gate 		switch (arg) {
56347c478bd9Sstevel@tonic-gate 		case '?':
56357c478bd9Sstevel@tonic-gate 			longer_usage(CMD_VERIFY);
5636bbec428eSgjelinek 			arg_err = B_TRUE;
56377ec75eb8Sgjelinek 			break;
56387c478bd9Sstevel@tonic-gate 		default:
56397c478bd9Sstevel@tonic-gate 			short_usage(CMD_VERIFY);
5640bbec428eSgjelinek 			arg_err = B_TRUE;
56417ec75eb8Sgjelinek 			break;
56427ec75eb8Sgjelinek 		}
56437ec75eb8Sgjelinek 	}
56447ec75eb8Sgjelinek 	if (arg_err)
56457c478bd9Sstevel@tonic-gate 		return;
56467ec75eb8Sgjelinek 
56477c478bd9Sstevel@tonic-gate 	if (optind > cmd->cmd_argc) {
56487c478bd9Sstevel@tonic-gate 		short_usage(CMD_VERIFY);
56497c478bd9Sstevel@tonic-gate 		return;
56507c478bd9Sstevel@tonic-gate 	}
56517c478bd9Sstevel@tonic-gate 
56527c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_VERIFY))
56537c478bd9Sstevel@tonic-gate 		return;
56547c478bd9Sstevel@tonic-gate 
56557c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
56567c478bd9Sstevel@tonic-gate 
56577c478bd9Sstevel@tonic-gate 	if (cmd->cmd_argc > 0 && (strcmp(cmd->cmd_argv[0], "save") == 0))
5658bbec428eSgjelinek 		save = B_TRUE;
5659bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
56607c478bd9Sstevel@tonic-gate 		return;
56617c478bd9Sstevel@tonic-gate 
56620209230bSgjelinek 	if (zonecfg_get_zonepath(handle, zonepath, sizeof (zonepath)) != Z_OK &&
56630209230bSgjelinek 	    !global_zone) {
5664087719fdSdp 		zerr(gettext("%s not specified"), pt_to_str(PT_ZONEPATH));
56657c478bd9Sstevel@tonic-gate 		ret_val = Z_REQD_RESOURCE_MISSING;
5666bbec428eSgjelinek 		saw_error = B_TRUE;
56677c478bd9Sstevel@tonic-gate 	}
56680209230bSgjelinek 	if (strlen(zonepath) == 0 && !global_zone) {
5669087719fdSdp 		zerr(gettext("%s cannot be empty."), pt_to_str(PT_ZONEPATH));
56707c478bd9Sstevel@tonic-gate 		ret_val = Z_REQD_RESOURCE_MISSING;
5671bbec428eSgjelinek 		saw_error = B_TRUE;
56727c478bd9Sstevel@tonic-gate 	}
56737c478bd9Sstevel@tonic-gate 
56749acbbeafSnn35248 	if ((err = zonecfg_get_brand(handle, brand, sizeof (brand))) != Z_OK) {
5675bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
56769acbbeafSnn35248 		return;
56779acbbeafSnn35248 	}
56789acbbeafSnn35248 	if ((err = brand_verify(handle)) != Z_OK) {
5679bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
56809acbbeafSnn35248 		return;
56819acbbeafSnn35248 	}
56829acbbeafSnn35248 
5683f4b3ec61Sdh155122 	if (zonecfg_get_iptype(handle, &iptype) != Z_OK) {
5684f4b3ec61Sdh155122 		zerr("%s %s", gettext("cannot get"), pt_to_str(PT_IPTYPE));
5685f4b3ec61Sdh155122 		ret_val = Z_REQD_RESOURCE_MISSING;
5686bbec428eSgjelinek 		saw_error = B_TRUE;
5687f4b3ec61Sdh155122 	}
56887c478bd9Sstevel@tonic-gate 
56890fbb751dSJohn Levon 	if (zonecfg_get_hostid(handle, hostidp,
56900fbb751dSJohn Levon 	    sizeof (hostidp)) == Z_INVALID_PROPERTY) {
56910fbb751dSJohn Levon 		zerr(gettext("%s: invalid hostid: %s"),
56920fbb751dSJohn Levon 		    zone, hostidp);
56930fbb751dSJohn Levon 		return;
56940fbb751dSJohn Levon 	}
56950fbb751dSJohn Levon 
56960fbb751dSJohn Levon 	if (zonecfg_get_fs_allowed(handle, fsallowedp,
56970fbb751dSJohn Levon 	    sizeof (fsallowedp)) == Z_INVALID_PROPERTY) {
56980fbb751dSJohn Levon 		zerr(gettext("%s: invalid fs-allowed: %s"),
56990fbb751dSJohn Levon 		    zone, fsallowedp);
57005679c89fSjv227347 		return;
57015679c89fSjv227347 	}
57025679c89fSjv227347 
57037c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setfsent(handle)) != Z_OK) {
5704bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
57057c478bd9Sstevel@tonic-gate 		return;
57067c478bd9Sstevel@tonic-gate 	}
57077c478bd9Sstevel@tonic-gate 	while (zonecfg_getfsent(handle, &fstab) == Z_OK) {
5708087719fdSdp 		check_reqd_prop(fstab.zone_fs_dir, RT_FS, PT_DIR, &ret_val);
5709087719fdSdp 		check_reqd_prop(fstab.zone_fs_special, RT_FS, PT_SPECIAL,
5710087719fdSdp 		    &ret_val);
5711087719fdSdp 		check_reqd_prop(fstab.zone_fs_type, RT_FS, PT_TYPE, &ret_val);
5712087719fdSdp 
57137c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(fstab.zone_fs_options);
57147c478bd9Sstevel@tonic-gate 	}
57157c478bd9Sstevel@tonic-gate 	(void) zonecfg_endfsent(handle);
57167c478bd9Sstevel@tonic-gate 
57177c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setnwifent(handle)) != Z_OK) {
5718bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
57197c478bd9Sstevel@tonic-gate 		return;
57207c478bd9Sstevel@tonic-gate 	}
57217c478bd9Sstevel@tonic-gate 	while (zonecfg_getnwifent(handle, &nwiftab) == Z_OK) {
5722f4b3ec61Sdh155122 		/*
5723f4b3ec61Sdh155122 		 * physical is required in all cases.
5724de860bd9Sgfaden 		 * A shared IP requires an address,
5725de860bd9Sgfaden 		 * and may include a default router, while
5726de860bd9Sgfaden 		 * an exclusive IP must have neither an address
5727de860bd9Sgfaden 		 * nor a default router.
572801b4bc23Sjv227347 		 * The physical interface name must be valid in all cases.
5729f4b3ec61Sdh155122 		 */
5730087719fdSdp 		check_reqd_prop(nwiftab.zone_nwif_physical, RT_NET,
5731087719fdSdp 		    PT_PHYSICAL, &ret_val);
573201b4bc23Sjv227347 		if (validate_net_physical_syntax(nwiftab.zone_nwif_physical) !=
573301b4bc23Sjv227347 		    Z_OK) {
573401b4bc23Sjv227347 			saw_error = B_TRUE;
573501b4bc23Sjv227347 			if (ret_val == Z_OK)
573601b4bc23Sjv227347 				ret_val = Z_INVAL;
573701b4bc23Sjv227347 		}
5738f4b3ec61Sdh155122 
5739f4b3ec61Sdh155122 		switch (iptype) {
5740f4b3ec61Sdh155122 		case ZS_SHARED:
5741f4b3ec61Sdh155122 			check_reqd_prop(nwiftab.zone_nwif_address, RT_NET,
5742f4b3ec61Sdh155122 			    PT_ADDRESS, &ret_val);
5743f4b3ec61Sdh155122 			break;
5744f4b3ec61Sdh155122 		case ZS_EXCLUSIVE:
5745f4b3ec61Sdh155122 			if (strlen(nwiftab.zone_nwif_address) > 0) {
5746f4b3ec61Sdh155122 				zerr(gettext("%s: %s cannot be specified "
5747f4b3ec61Sdh155122 				    "for an exclusive IP type"),
5748f4b3ec61Sdh155122 				    rt_to_str(RT_NET), pt_to_str(PT_ADDRESS));
5749bbec428eSgjelinek 				saw_error = B_TRUE;
5750f4b3ec61Sdh155122 				if (ret_val == Z_OK)
5751f4b3ec61Sdh155122 					ret_val = Z_INVAL;
5752f4b3ec61Sdh155122 			}
5753de860bd9Sgfaden 			if (strlen(nwiftab.zone_nwif_defrouter) > 0) {
5754de860bd9Sgfaden 				zerr(gettext("%s: %s cannot be specified "
5755de860bd9Sgfaden 				    "for an exclusive IP type"),
5756de860bd9Sgfaden 				    rt_to_str(RT_NET), pt_to_str(PT_DEFROUTER));
5757bbec428eSgjelinek 				saw_error = B_TRUE;
5758de860bd9Sgfaden 				if (ret_val == Z_OK)
5759de860bd9Sgfaden 					ret_val = Z_INVAL;
5760de860bd9Sgfaden 			}
5761f4b3ec61Sdh155122 			break;
5762f4b3ec61Sdh155122 		}
57637c478bd9Sstevel@tonic-gate 	}
57647c478bd9Sstevel@tonic-gate 	(void) zonecfg_endnwifent(handle);
57657c478bd9Sstevel@tonic-gate 
57667c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setrctlent(handle)) != Z_OK) {
5767bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
57687c478bd9Sstevel@tonic-gate 		return;
57697c478bd9Sstevel@tonic-gate 	}
57707c478bd9Sstevel@tonic-gate 	while (zonecfg_getrctlent(handle, &rctltab) == Z_OK) {
5771087719fdSdp 		check_reqd_prop(rctltab.zone_rctl_name, RT_RCTL, PT_NAME,
5772087719fdSdp 		    &ret_val);
5773087719fdSdp 
57740209230bSgjelinek 		if (strcmp(rctltab.zone_rctl_name, "zone.cpu-shares") == 0)
57750209230bSgjelinek 			has_cpu_shares = B_TRUE;
57760209230bSgjelinek 
5777c97ad5cdSakolb 		if (strcmp(rctltab.zone_rctl_name, "zone.cpu-cap") == 0)
5778c97ad5cdSakolb 			has_cpu_cap = B_TRUE;
5779c97ad5cdSakolb 
57807c478bd9Sstevel@tonic-gate 		if (rctltab.zone_rctl_valptr == NULL) {
57817c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: no %s specified"),
57827c478bd9Sstevel@tonic-gate 			    rt_to_str(RT_RCTL), pt_to_str(PT_VALUE));
5783bbec428eSgjelinek 			saw_error = B_TRUE;
57847c478bd9Sstevel@tonic-gate 			if (ret_val == Z_OK)
57857c478bd9Sstevel@tonic-gate 				ret_val = Z_REQD_PROPERTY_MISSING;
57867c478bd9Sstevel@tonic-gate 		} else {
57877c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(rctltab.zone_rctl_valptr);
57887c478bd9Sstevel@tonic-gate 		}
57897c478bd9Sstevel@tonic-gate 	}
57907c478bd9Sstevel@tonic-gate 	(void) zonecfg_endrctlent(handle);
57917c478bd9Sstevel@tonic-gate 
5792c97ad5cdSakolb 	if ((pset_res = zonecfg_lookup_pset(handle, &psettab)) == Z_OK &&
5793c97ad5cdSakolb 	    has_cpu_shares) {
57940209230bSgjelinek 		zerr(gettext("%s zone.cpu-shares and %s are incompatible."),
57950209230bSgjelinek 		    rt_to_str(RT_RCTL), rt_to_str(RT_DCPU));
5796bbec428eSgjelinek 		saw_error = B_TRUE;
57970209230bSgjelinek 		if (ret_val == Z_OK)
57980209230bSgjelinek 			ret_val = Z_INCOMPATIBLE;
57990209230bSgjelinek 	}
58000209230bSgjelinek 
58010209230bSgjelinek 	if (has_cpu_shares && zonecfg_get_sched_class(handle, sched,
58020209230bSgjelinek 	    sizeof (sched)) == Z_OK && strlen(sched) > 0 &&
58030209230bSgjelinek 	    strcmp(sched, "FSS") != 0) {
58040209230bSgjelinek 		zerr(gettext("WARNING: %s zone.cpu-shares and %s=%s are "
58050209230bSgjelinek 		    "incompatible"),
58060209230bSgjelinek 		    rt_to_str(RT_RCTL), rt_to_str(RT_SCHED), sched);
5807bbec428eSgjelinek 		saw_error = B_TRUE;
58080209230bSgjelinek 		if (ret_val == Z_OK)
58090209230bSgjelinek 			ret_val = Z_INCOMPATIBLE;
58100209230bSgjelinek 	}
58110209230bSgjelinek 
5812c97ad5cdSakolb 	if (pset_res == Z_OK && has_cpu_cap) {
5813c97ad5cdSakolb 		zerr(gettext("%s zone.cpu-cap and the %s are incompatible."),
5814c97ad5cdSakolb 		    rt_to_str(RT_RCTL), rt_to_str(RT_DCPU));
5815bbec428eSgjelinek 		saw_error = B_TRUE;
5816c97ad5cdSakolb 		if (ret_val == Z_OK)
5817c97ad5cdSakolb 			ret_val = Z_INCOMPATIBLE;
5818c97ad5cdSakolb 	}
5819c97ad5cdSakolb 
58207c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setattrent(handle)) != Z_OK) {
5821bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
58227c478bd9Sstevel@tonic-gate 		return;
58237c478bd9Sstevel@tonic-gate 	}
58247c478bd9Sstevel@tonic-gate 	while (zonecfg_getattrent(handle, &attrtab) == Z_OK) {
5825087719fdSdp 		check_reqd_prop(attrtab.zone_attr_name, RT_ATTR, PT_NAME,
5826087719fdSdp 		    &ret_val);
5827087719fdSdp 		check_reqd_prop(attrtab.zone_attr_type, RT_ATTR, PT_TYPE,
5828087719fdSdp 		    &ret_val);
5829087719fdSdp 		check_reqd_prop(attrtab.zone_attr_value, RT_ATTR, PT_VALUE,
5830087719fdSdp 		    &ret_val);
58317c478bd9Sstevel@tonic-gate 	}
58327c478bd9Sstevel@tonic-gate 	(void) zonecfg_endattrent(handle);
58337c478bd9Sstevel@tonic-gate 
5834fa9e4066Sahrens 	if ((err = zonecfg_setdsent(handle)) != Z_OK) {
5835bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
5836fa9e4066Sahrens 		return;
5837fa9e4066Sahrens 	}
5838fa9e4066Sahrens 	while (zonecfg_getdsent(handle, &dstab) == Z_OK) {
5839fa9e4066Sahrens 		if (strlen(dstab.zone_dataset_name) == 0) {
5840fa9e4066Sahrens 			zerr("%s: %s %s", rt_to_str(RT_DATASET),
5841fa9e4066Sahrens 			    pt_to_str(PT_NAME), gettext("not specified"));
5842bbec428eSgjelinek 			saw_error = B_TRUE;
5843fa9e4066Sahrens 			if (ret_val == Z_OK)
5844fa9e4066Sahrens 				ret_val = Z_REQD_PROPERTY_MISSING;
5845fa9e4066Sahrens 		} else if (!zfs_name_valid(dstab.zone_dataset_name,
5846fa9e4066Sahrens 		    ZFS_TYPE_FILESYSTEM)) {
5847fa9e4066Sahrens 			zerr("%s: %s %s", rt_to_str(RT_DATASET),
5848fa9e4066Sahrens 			    pt_to_str(PT_NAME), gettext("invalid"));
5849bbec428eSgjelinek 			saw_error = B_TRUE;
5850fa9e4066Sahrens 			if (ret_val == Z_OK)
5851fa9e4066Sahrens 				ret_val = Z_BAD_PROPERTY;
5852fa9e4066Sahrens 		}
5853fa9e4066Sahrens 
5854fa9e4066Sahrens 	}
5855fa9e4066Sahrens 	(void) zonecfg_enddsent(handle);
5856fa9e4066Sahrens 
5857cb8a054bSGlenn Faden 	if ((err = zonecfg_setadminent(handle)) != Z_OK) {
5858cb8a054bSGlenn Faden 		zone_perror(zone, err, B_TRUE);
5859cb8a054bSGlenn Faden 		return;
5860cb8a054bSGlenn Faden 	}
5861cb8a054bSGlenn Faden 	while (zonecfg_getadminent(handle, &admintab) == Z_OK) {
5862cb8a054bSGlenn Faden 		check_reqd_prop(admintab.zone_admin_user, RT_ADMIN,
5863cb8a054bSGlenn Faden 		    PT_USER, &ret_val);
5864cb8a054bSGlenn Faden 		check_reqd_prop(admintab.zone_admin_auths, RT_ADMIN,
5865cb8a054bSGlenn Faden 		    PT_AUTHS, &ret_val);
5866cb8a054bSGlenn Faden 		if ((ret_val == Z_OK) && (getpwnam(admintab.zone_admin_user)
5867cb8a054bSGlenn Faden 		    == NULL)) {
5868cb8a054bSGlenn Faden 			zerr(gettext("%s %s is not a valid username"),
5869cb8a054bSGlenn Faden 			    pt_to_str(PT_USER),
5870cb8a054bSGlenn Faden 			    admintab.zone_admin_user);
5871cb8a054bSGlenn Faden 			ret_val = Z_BAD_PROPERTY;
5872cb8a054bSGlenn Faden 		}
5873cb8a054bSGlenn Faden 		if ((ret_val == Z_OK) && (!zonecfg_valid_auths(
5874cb8a054bSGlenn Faden 		    admintab.zone_admin_auths, zone))) {
5875cb8a054bSGlenn Faden 			ret_val = Z_BAD_PROPERTY;
5876cb8a054bSGlenn Faden 		}
5877cb8a054bSGlenn Faden 	}
5878cb8a054bSGlenn Faden 	(void) zonecfg_endadminent(handle);
5879cb8a054bSGlenn Faden 
58807c478bd9Sstevel@tonic-gate 	if (!global_scope) {
58817c478bd9Sstevel@tonic-gate 		zerr(gettext("resource specification incomplete"));
5882bbec428eSgjelinek 		saw_error = B_TRUE;
58837c478bd9Sstevel@tonic-gate 		if (ret_val == Z_OK)
58847c478bd9Sstevel@tonic-gate 			ret_val = Z_INSUFFICIENT_SPEC;
58857c478bd9Sstevel@tonic-gate 	}
58867c478bd9Sstevel@tonic-gate 
58877c478bd9Sstevel@tonic-gate 	if (save) {
5888087719fdSdp 		if (ret_val == Z_OK) {
5889087719fdSdp 			if ((ret_val = zonecfg_save(handle)) == Z_OK) {
5890bbec428eSgjelinek 				need_to_commit = B_FALSE;
5891087719fdSdp 				(void) strlcpy(revert_zone, zone,
5892087719fdSdp 				    sizeof (revert_zone));
5893087719fdSdp 			}
5894087719fdSdp 		} else {
5895087719fdSdp 			zerr(gettext("Zone %s failed to verify"), zone);
5896087719fdSdp 		}
58977c478bd9Sstevel@tonic-gate 	}
58987c478bd9Sstevel@tonic-gate 	if (ret_val != Z_OK)
5899bbec428eSgjelinek 		zone_perror(zone, ret_val, B_TRUE);
59007c478bd9Sstevel@tonic-gate }
59017c478bd9Sstevel@tonic-gate 
59027c478bd9Sstevel@tonic-gate void
59037c478bd9Sstevel@tonic-gate cancel_func(cmd_t *cmd)
59047c478bd9Sstevel@tonic-gate {
59057c478bd9Sstevel@tonic-gate 	int arg;
5906bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
59077c478bd9Sstevel@tonic-gate 
59087c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
59097c478bd9Sstevel@tonic-gate 
59107c478bd9Sstevel@tonic-gate 	optind = 0;
59117ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
59127c478bd9Sstevel@tonic-gate 		switch (arg) {
59137c478bd9Sstevel@tonic-gate 		case '?':
59147c478bd9Sstevel@tonic-gate 			longer_usage(CMD_CANCEL);
5915bbec428eSgjelinek 			arg_err = B_TRUE;
59167ec75eb8Sgjelinek 			break;
59177c478bd9Sstevel@tonic-gate 		default:
59187c478bd9Sstevel@tonic-gate 			short_usage(CMD_CANCEL);
5919bbec428eSgjelinek 			arg_err = B_TRUE;
59207ec75eb8Sgjelinek 			break;
59217ec75eb8Sgjelinek 		}
59227ec75eb8Sgjelinek 	}
59237ec75eb8Sgjelinek 	if (arg_err)
59247c478bd9Sstevel@tonic-gate 		return;
59257ec75eb8Sgjelinek 
59267c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
59277c478bd9Sstevel@tonic-gate 		short_usage(CMD_CANCEL);
59287c478bd9Sstevel@tonic-gate 		return;
59297c478bd9Sstevel@tonic-gate 	}
59307c478bd9Sstevel@tonic-gate 
59317c478bd9Sstevel@tonic-gate 	if (global_scope)
59327c478bd9Sstevel@tonic-gate 		scope_usage(CMD_CANCEL);
5933bbec428eSgjelinek 	global_scope = B_TRUE;
59347c478bd9Sstevel@tonic-gate 	zonecfg_free_fs_option_list(in_progress_fstab.zone_fs_options);
59357c478bd9Sstevel@tonic-gate 	bzero(&in_progress_fstab, sizeof (in_progress_fstab));
59367c478bd9Sstevel@tonic-gate 	bzero(&in_progress_nwiftab, sizeof (in_progress_nwiftab));
59377c478bd9Sstevel@tonic-gate 	bzero(&in_progress_devtab, sizeof (in_progress_devtab));
59387c478bd9Sstevel@tonic-gate 	zonecfg_free_rctl_value_list(in_progress_rctltab.zone_rctl_valptr);
59397c478bd9Sstevel@tonic-gate 	bzero(&in_progress_rctltab, sizeof (in_progress_rctltab));
59407c478bd9Sstevel@tonic-gate 	bzero(&in_progress_attrtab, sizeof (in_progress_attrtab));
5941fa9e4066Sahrens 	bzero(&in_progress_dstab, sizeof (in_progress_dstab));
59427c478bd9Sstevel@tonic-gate }
59437c478bd9Sstevel@tonic-gate 
59447c478bd9Sstevel@tonic-gate static int
59457c478bd9Sstevel@tonic-gate validate_attr_name(char *name)
59467c478bd9Sstevel@tonic-gate {
59477c478bd9Sstevel@tonic-gate 	int i;
59487c478bd9Sstevel@tonic-gate 
59497c478bd9Sstevel@tonic-gate 	if (!isalnum(name[0])) {
59507c478bd9Sstevel@tonic-gate 		zerr(gettext("Invalid %s %s %s: must start with an alpha-"
59517c478bd9Sstevel@tonic-gate 		    "numeric character."), rt_to_str(RT_ATTR),
59527c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_NAME), name);
59537c478bd9Sstevel@tonic-gate 		return (Z_INVAL);
59547c478bd9Sstevel@tonic-gate 	}
59557c478bd9Sstevel@tonic-gate 	for (i = 1; name[i]; i++)
59567c478bd9Sstevel@tonic-gate 		if (!isalnum(name[i]) && name[i] != '-' && name[i] != '.') {
59577c478bd9Sstevel@tonic-gate 			zerr(gettext("Invalid %s %s %s: can only contain "
59587c478bd9Sstevel@tonic-gate 			    "alpha-numeric characters, plus '-' and '.'."),
59597c478bd9Sstevel@tonic-gate 			    rt_to_str(RT_ATTR), pt_to_str(PT_NAME), name);
59607c478bd9Sstevel@tonic-gate 			return (Z_INVAL);
59617c478bd9Sstevel@tonic-gate 		}
59627c478bd9Sstevel@tonic-gate 	return (Z_OK);
59637c478bd9Sstevel@tonic-gate }
59647c478bd9Sstevel@tonic-gate 
59657c478bd9Sstevel@tonic-gate static int
59667c478bd9Sstevel@tonic-gate validate_attr_type_val(struct zone_attrtab *attrtab)
59677c478bd9Sstevel@tonic-gate {
59687c478bd9Sstevel@tonic-gate 	boolean_t boolval;
59697c478bd9Sstevel@tonic-gate 	int64_t intval;
59707c478bd9Sstevel@tonic-gate 	char strval[MAXNAMELEN];
59717c478bd9Sstevel@tonic-gate 	uint64_t uintval;
59727c478bd9Sstevel@tonic-gate 
59737c478bd9Sstevel@tonic-gate 	if (strcmp(attrtab->zone_attr_type, "boolean") == 0) {
59747c478bd9Sstevel@tonic-gate 		if (zonecfg_get_attr_boolean(attrtab, &boolval) == Z_OK)
59757c478bd9Sstevel@tonic-gate 			return (Z_OK);
59767c478bd9Sstevel@tonic-gate 		zerr(gettext("invalid %s value for %s=%s"),
59777c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR), pt_to_str(PT_TYPE), "boolean");
59787c478bd9Sstevel@tonic-gate 		return (Z_ERR);
59797c478bd9Sstevel@tonic-gate 	}
59807c478bd9Sstevel@tonic-gate 
59817c478bd9Sstevel@tonic-gate 	if (strcmp(attrtab->zone_attr_type, "int") == 0) {
59827c478bd9Sstevel@tonic-gate 		if (zonecfg_get_attr_int(attrtab, &intval) == Z_OK)
59837c478bd9Sstevel@tonic-gate 			return (Z_OK);
59847c478bd9Sstevel@tonic-gate 		zerr(gettext("invalid %s value for %s=%s"),
59857c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR), pt_to_str(PT_TYPE), "int");
59867c478bd9Sstevel@tonic-gate 		return (Z_ERR);
59877c478bd9Sstevel@tonic-gate 	}
59887c478bd9Sstevel@tonic-gate 
59897c478bd9Sstevel@tonic-gate 	if (strcmp(attrtab->zone_attr_type, "string") == 0) {
59907c478bd9Sstevel@tonic-gate 		if (zonecfg_get_attr_string(attrtab, strval,
59917c478bd9Sstevel@tonic-gate 		    sizeof (strval)) == Z_OK)
59927c478bd9Sstevel@tonic-gate 			return (Z_OK);
59937c478bd9Sstevel@tonic-gate 		zerr(gettext("invalid %s value for %s=%s"),
59947c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR), pt_to_str(PT_TYPE), "string");
59957c478bd9Sstevel@tonic-gate 		return (Z_ERR);
59967c478bd9Sstevel@tonic-gate 	}
59977c478bd9Sstevel@tonic-gate 
59987c478bd9Sstevel@tonic-gate 	if (strcmp(attrtab->zone_attr_type, "uint") == 0) {
59997c478bd9Sstevel@tonic-gate 		if (zonecfg_get_attr_uint(attrtab, &uintval) == Z_OK)
60007c478bd9Sstevel@tonic-gate 			return (Z_OK);
60017c478bd9Sstevel@tonic-gate 		zerr(gettext("invalid %s value for %s=%s"),
60027c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR), pt_to_str(PT_TYPE), "uint");
60037c478bd9Sstevel@tonic-gate 		return (Z_ERR);
60047c478bd9Sstevel@tonic-gate 	}
60057c478bd9Sstevel@tonic-gate 
60067c478bd9Sstevel@tonic-gate 	zerr(gettext("invalid %s %s '%s'"), rt_to_str(RT_ATTR),
60077c478bd9Sstevel@tonic-gate 	    pt_to_str(PT_TYPE), attrtab->zone_attr_type);
60087c478bd9Sstevel@tonic-gate 	return (Z_ERR);
60097c478bd9Sstevel@tonic-gate }
60107c478bd9Sstevel@tonic-gate 
6011087719fdSdp /*
6012087719fdSdp  * Helper function for end_func-- checks the existence of a given property
6013087719fdSdp  * and emits a message if not specified.
6014087719fdSdp  */
6015087719fdSdp static int
6016bbec428eSgjelinek end_check_reqd(char *attr, int pt, boolean_t *validation_failed)
6017087719fdSdp {
6018087719fdSdp 	if (strlen(attr) == 0) {
6019bbec428eSgjelinek 		*validation_failed = B_TRUE;
6020087719fdSdp 		zerr(gettext("%s not specified"), pt_to_str(pt));
6021087719fdSdp 		return (Z_ERR);
6022087719fdSdp 	}
6023087719fdSdp 	return (Z_OK);
6024087719fdSdp }
6025087719fdSdp 
60267c478bd9Sstevel@tonic-gate void
60277c478bd9Sstevel@tonic-gate end_func(cmd_t *cmd)
60287c478bd9Sstevel@tonic-gate {
6029bbec428eSgjelinek 	boolean_t validation_failed = B_FALSE;
6030bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
60317c478bd9Sstevel@tonic-gate 	struct zone_fstab tmp_fstab;
60327c478bd9Sstevel@tonic-gate 	struct zone_nwiftab tmp_nwiftab;
60337c478bd9Sstevel@tonic-gate 	struct zone_devtab tmp_devtab;
60347c478bd9Sstevel@tonic-gate 	struct zone_rctltab tmp_rctltab;
60357c478bd9Sstevel@tonic-gate 	struct zone_attrtab tmp_attrtab;
6036fa9e4066Sahrens 	struct zone_dstab tmp_dstab;
6037cb8a054bSGlenn Faden 	struct zone_admintab tmp_admintab;
60380209230bSgjelinek 	int err, arg, res1, res2, res3;
60390209230bSgjelinek 	uint64_t swap_limit;
60400209230bSgjelinek 	uint64_t locked_limit;
6041c97ad5cdSakolb 	uint64_t proc_cap;
60427c478bd9Sstevel@tonic-gate 
60437c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
60447c478bd9Sstevel@tonic-gate 
60457c478bd9Sstevel@tonic-gate 	optind = 0;
60467ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
60477c478bd9Sstevel@tonic-gate 		switch (arg) {
60487c478bd9Sstevel@tonic-gate 		case '?':
60497c478bd9Sstevel@tonic-gate 			longer_usage(CMD_END);
6050bbec428eSgjelinek 			arg_err = B_TRUE;
60517ec75eb8Sgjelinek 			break;
60527c478bd9Sstevel@tonic-gate 		default:
60537c478bd9Sstevel@tonic-gate 			short_usage(CMD_END);
6054bbec428eSgjelinek 			arg_err = B_TRUE;
60557ec75eb8Sgjelinek 			break;
60567ec75eb8Sgjelinek 		}
60577ec75eb8Sgjelinek 	}
60587ec75eb8Sgjelinek 	if (arg_err)
60597c478bd9Sstevel@tonic-gate 		return;
60607ec75eb8Sgjelinek 
60617c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
60627c478bd9Sstevel@tonic-gate 		short_usage(CMD_END);
60637c478bd9Sstevel@tonic-gate 		return;
60647c478bd9Sstevel@tonic-gate 	}
60657c478bd9Sstevel@tonic-gate 
60667c478bd9Sstevel@tonic-gate 	if (global_scope) {
60677c478bd9Sstevel@tonic-gate 		scope_usage(CMD_END);
60687c478bd9Sstevel@tonic-gate 		return;
60697c478bd9Sstevel@tonic-gate 	}
60707c478bd9Sstevel@tonic-gate 
60717c478bd9Sstevel@tonic-gate 	assert(end_op == CMD_ADD || end_op == CMD_SELECT);
60727c478bd9Sstevel@tonic-gate 
60737c478bd9Sstevel@tonic-gate 	switch (resource_scope) {
60747c478bd9Sstevel@tonic-gate 	case RT_FS:
60757c478bd9Sstevel@tonic-gate 		/* First make sure everything was filled in. */
6076087719fdSdp 		if (end_check_reqd(in_progress_fstab.zone_fs_dir,
6077087719fdSdp 		    PT_DIR, &validation_failed) == Z_OK) {
6078087719fdSdp 			if (in_progress_fstab.zone_fs_dir[0] != '/') {
6079087719fdSdp 				zerr(gettext("%s %s is not an absolute path."),
6080087719fdSdp 				    pt_to_str(PT_DIR),
6081087719fdSdp 				    in_progress_fstab.zone_fs_dir);
6082bbec428eSgjelinek 				validation_failed = B_TRUE;
60837c478bd9Sstevel@tonic-gate 			}
60847c478bd9Sstevel@tonic-gate 		}
6085087719fdSdp 
6086087719fdSdp 		(void) end_check_reqd(in_progress_fstab.zone_fs_special,
6087087719fdSdp 		    PT_SPECIAL, &validation_failed);
6088087719fdSdp 
60897c478bd9Sstevel@tonic-gate 		if (in_progress_fstab.zone_fs_raw[0] != '\0' &&
60907c478bd9Sstevel@tonic-gate 		    in_progress_fstab.zone_fs_raw[0] != '/') {
6091087719fdSdp 			zerr(gettext("%s %s is not an absolute path."),
6092087719fdSdp 			    pt_to_str(PT_RAW),
6093087719fdSdp 			    in_progress_fstab.zone_fs_raw);
6094bbec428eSgjelinek 			validation_failed = B_TRUE;
60957c478bd9Sstevel@tonic-gate 		}
6096087719fdSdp 
6097087719fdSdp 		(void) end_check_reqd(in_progress_fstab.zone_fs_type, PT_TYPE,
6098087719fdSdp 		    &validation_failed);
6099087719fdSdp 
6100087719fdSdp 		if (validation_failed) {
6101bbec428eSgjelinek 			saw_error = B_TRUE;
61027c478bd9Sstevel@tonic-gate 			return;
6103087719fdSdp 		}
6104087719fdSdp 
61057c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
61067c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
61077c478bd9Sstevel@tonic-gate 			bzero(&tmp_fstab, sizeof (tmp_fstab));
61087c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_fstab.zone_fs_dir,
61097c478bd9Sstevel@tonic-gate 			    in_progress_fstab.zone_fs_dir,
61107c478bd9Sstevel@tonic-gate 			    sizeof (tmp_fstab.zone_fs_dir));
61117c478bd9Sstevel@tonic-gate 			err = zonecfg_lookup_filesystem(handle, &tmp_fstab);
61127c478bd9Sstevel@tonic-gate 			zonecfg_free_fs_option_list(tmp_fstab.zone_fs_options);
61137c478bd9Sstevel@tonic-gate 			if (err == Z_OK) {
61147c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s resource "
61157c478bd9Sstevel@tonic-gate 				    "with the %s '%s' already exists."),
61167c478bd9Sstevel@tonic-gate 				    rt_to_str(RT_FS), pt_to_str(PT_DIR),
61177c478bd9Sstevel@tonic-gate 				    in_progress_fstab.zone_fs_dir);
6118bbec428eSgjelinek 				saw_error = B_TRUE;
61197c478bd9Sstevel@tonic-gate 				return;
61207c478bd9Sstevel@tonic-gate 			}
61217c478bd9Sstevel@tonic-gate 			err = zonecfg_add_filesystem(handle,
61227c478bd9Sstevel@tonic-gate 			    &in_progress_fstab);
61237c478bd9Sstevel@tonic-gate 		} else {
61247c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_filesystem(handle, &old_fstab,
61257c478bd9Sstevel@tonic-gate 			    &in_progress_fstab);
61267c478bd9Sstevel@tonic-gate 		}
61277c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(in_progress_fstab.zone_fs_options);
61287c478bd9Sstevel@tonic-gate 		in_progress_fstab.zone_fs_options = NULL;
61297c478bd9Sstevel@tonic-gate 		break;
6130087719fdSdp 
61317c478bd9Sstevel@tonic-gate 	case RT_NET:
6132f4b3ec61Sdh155122 		/*
6133f4b3ec61Sdh155122 		 * First make sure everything was filled in.
6134f4b3ec61Sdh155122 		 * Since we don't know whether IP will be shared
6135f4b3ec61Sdh155122 		 * or exclusive here, some checks are deferred until
6136f4b3ec61Sdh155122 		 * the verify command.
6137f4b3ec61Sdh155122 		 */
6138087719fdSdp 		(void) end_check_reqd(in_progress_nwiftab.zone_nwif_physical,
6139087719fdSdp 		    PT_PHYSICAL, &validation_failed);
6140087719fdSdp 
6141087719fdSdp 		if (validation_failed) {
6142bbec428eSgjelinek 			saw_error = B_TRUE;
61437c478bd9Sstevel@tonic-gate 			return;
6144087719fdSdp 		}
61457c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
61467c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
61477c478bd9Sstevel@tonic-gate 			bzero(&tmp_nwiftab, sizeof (tmp_nwiftab));
6148f4b3ec61Sdh155122 			(void) strlcpy(tmp_nwiftab.zone_nwif_physical,
6149f4b3ec61Sdh155122 			    in_progress_nwiftab.zone_nwif_physical,
6150f4b3ec61Sdh155122 			    sizeof (tmp_nwiftab.zone_nwif_physical));
61517c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_nwiftab.zone_nwif_address,
61527c478bd9Sstevel@tonic-gate 			    in_progress_nwiftab.zone_nwif_address,
61537c478bd9Sstevel@tonic-gate 			    sizeof (tmp_nwiftab.zone_nwif_address));
61547c478bd9Sstevel@tonic-gate 			if (zonecfg_lookup_nwif(handle, &tmp_nwiftab) == Z_OK) {
6155f4b3ec61Sdh155122 				zerr(gettext("A %s resource with the %s '%s', "
6156f4b3ec61Sdh155122 				    "and %s '%s' already exists."),
6157f4b3ec61Sdh155122 				    rt_to_str(RT_NET),
6158f4b3ec61Sdh155122 				    pt_to_str(PT_PHYSICAL),
6159f4b3ec61Sdh155122 				    in_progress_nwiftab.zone_nwif_physical,
6160f4b3ec61Sdh155122 				    pt_to_str(PT_ADDRESS),
61617c478bd9Sstevel@tonic-gate 				    in_progress_nwiftab.zone_nwif_address);
6162bbec428eSgjelinek 				saw_error = B_TRUE;
61637c478bd9Sstevel@tonic-gate 				return;
61647c478bd9Sstevel@tonic-gate 			}
61657c478bd9Sstevel@tonic-gate 			err = zonecfg_add_nwif(handle, &in_progress_nwiftab);
61667c478bd9Sstevel@tonic-gate 		} else {
61677c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_nwif(handle, &old_nwiftab,
61687c478bd9Sstevel@tonic-gate 			    &in_progress_nwiftab);
61697c478bd9Sstevel@tonic-gate 		}
61707c478bd9Sstevel@tonic-gate 		break;
6171087719fdSdp 
61727c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
61737c478bd9Sstevel@tonic-gate 		/* First make sure everything was filled in. */
6174087719fdSdp 		(void) end_check_reqd(in_progress_devtab.zone_dev_match,
6175087719fdSdp 		    PT_MATCH, &validation_failed);
6176087719fdSdp 
6177087719fdSdp 		if (validation_failed) {
6178bbec428eSgjelinek 			saw_error = B_TRUE;
61797c478bd9Sstevel@tonic-gate 			return;
6180087719fdSdp 		}
6181087719fdSdp 
61827c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
61837c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
61847c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_devtab.zone_dev_match,
61857c478bd9Sstevel@tonic-gate 			    in_progress_devtab.zone_dev_match,
61867c478bd9Sstevel@tonic-gate 			    sizeof (tmp_devtab.zone_dev_match));
61877c478bd9Sstevel@tonic-gate 			if (zonecfg_lookup_dev(handle, &tmp_devtab) == Z_OK) {
61887c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s resource with the %s '%s' "
61897c478bd9Sstevel@tonic-gate 				    "already exists."), rt_to_str(RT_DEVICE),
61907c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_MATCH),
61917c478bd9Sstevel@tonic-gate 				    in_progress_devtab.zone_dev_match);
6192bbec428eSgjelinek 				saw_error = B_TRUE;
61937c478bd9Sstevel@tonic-gate 				return;
61947c478bd9Sstevel@tonic-gate 			}
61957c478bd9Sstevel@tonic-gate 			err = zonecfg_add_dev(handle, &in_progress_devtab);
61967c478bd9Sstevel@tonic-gate 		} else {
61977c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_dev(handle, &old_devtab,
61987c478bd9Sstevel@tonic-gate 			    &in_progress_devtab);
61997c478bd9Sstevel@tonic-gate 		}
62007c478bd9Sstevel@tonic-gate 		break;
6201087719fdSdp 
62027c478bd9Sstevel@tonic-gate 	case RT_RCTL:
62037c478bd9Sstevel@tonic-gate 		/* First make sure everything was filled in. */
6204087719fdSdp 		(void) end_check_reqd(in_progress_rctltab.zone_rctl_name,
6205087719fdSdp 		    PT_NAME, &validation_failed);
6206087719fdSdp 
62077c478bd9Sstevel@tonic-gate 		if (in_progress_rctltab.zone_rctl_valptr == NULL) {
62087c478bd9Sstevel@tonic-gate 			zerr(gettext("no %s specified"), pt_to_str(PT_VALUE));
6209bbec428eSgjelinek 			validation_failed = B_TRUE;
62107c478bd9Sstevel@tonic-gate 		}
6211087719fdSdp 
6212087719fdSdp 		if (validation_failed) {
6213bbec428eSgjelinek 			saw_error = B_TRUE;
62147c478bd9Sstevel@tonic-gate 			return;
6215087719fdSdp 		}
6216087719fdSdp 
62177c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
62187c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
62197c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_rctltab.zone_rctl_name,
62207c478bd9Sstevel@tonic-gate 			    in_progress_rctltab.zone_rctl_name,
62217c478bd9Sstevel@tonic-gate 			    sizeof (tmp_rctltab.zone_rctl_name));
62227c478bd9Sstevel@tonic-gate 			tmp_rctltab.zone_rctl_valptr = NULL;
62237c478bd9Sstevel@tonic-gate 			err = zonecfg_lookup_rctl(handle, &tmp_rctltab);
62247c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(
62257c478bd9Sstevel@tonic-gate 			    tmp_rctltab.zone_rctl_valptr);
62267c478bd9Sstevel@tonic-gate 			if (err == Z_OK) {
62277c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s resource "
62287c478bd9Sstevel@tonic-gate 				    "with the %s '%s' already exists."),
62297c478bd9Sstevel@tonic-gate 				    rt_to_str(RT_RCTL), pt_to_str(PT_NAME),
62307c478bd9Sstevel@tonic-gate 				    in_progress_rctltab.zone_rctl_name);
6231bbec428eSgjelinek 				saw_error = B_TRUE;
62327c478bd9Sstevel@tonic-gate 				return;
62337c478bd9Sstevel@tonic-gate 			}
62347c478bd9Sstevel@tonic-gate 			err = zonecfg_add_rctl(handle, &in_progress_rctltab);
62357c478bd9Sstevel@tonic-gate 		} else {
62367c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_rctl(handle, &old_rctltab,
62377c478bd9Sstevel@tonic-gate 			    &in_progress_rctltab);
62387c478bd9Sstevel@tonic-gate 		}
62397c478bd9Sstevel@tonic-gate 		if (err == Z_OK) {
62407c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(
62417c478bd9Sstevel@tonic-gate 			    in_progress_rctltab.zone_rctl_valptr);
62427c478bd9Sstevel@tonic-gate 			in_progress_rctltab.zone_rctl_valptr = NULL;
62437c478bd9Sstevel@tonic-gate 		}
62447c478bd9Sstevel@tonic-gate 		break;
6245087719fdSdp 
62467c478bd9Sstevel@tonic-gate 	case RT_ATTR:
62477c478bd9Sstevel@tonic-gate 		/* First make sure everything was filled in. */
6248087719fdSdp 		(void) end_check_reqd(in_progress_attrtab.zone_attr_name,
6249087719fdSdp 		    PT_NAME, &validation_failed);
6250087719fdSdp 		(void) end_check_reqd(in_progress_attrtab.zone_attr_type,
6251087719fdSdp 		    PT_TYPE, &validation_failed);
6252087719fdSdp 		(void) end_check_reqd(in_progress_attrtab.zone_attr_value,
6253087719fdSdp 		    PT_VALUE, &validation_failed);
6254087719fdSdp 
62557c478bd9Sstevel@tonic-gate 		if (validate_attr_name(in_progress_attrtab.zone_attr_name) !=
6256087719fdSdp 		    Z_OK)
6257bbec428eSgjelinek 			validation_failed = B_TRUE;
6258087719fdSdp 
6259087719fdSdp 		if (validate_attr_type_val(&in_progress_attrtab) != Z_OK)
6260bbec428eSgjelinek 			validation_failed = B_TRUE;
6261087719fdSdp 
6262087719fdSdp 		if (validation_failed) {
6263bbec428eSgjelinek 			saw_error = B_TRUE;
62647c478bd9Sstevel@tonic-gate 			return;
6265087719fdSdp 		}
62667c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
62677c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
62687c478bd9Sstevel@tonic-gate 			bzero(&tmp_attrtab, sizeof (tmp_attrtab));
62697c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_attrtab.zone_attr_name,
62707c478bd9Sstevel@tonic-gate 			    in_progress_attrtab.zone_attr_name,
62717c478bd9Sstevel@tonic-gate 			    sizeof (tmp_attrtab.zone_attr_name));
62727c478bd9Sstevel@tonic-gate 			if (zonecfg_lookup_attr(handle, &tmp_attrtab) == Z_OK) {
62737c478bd9Sstevel@tonic-gate 				zerr(gettext("An %s resource "
62747c478bd9Sstevel@tonic-gate 				    "with the %s '%s' already exists."),
62757c478bd9Sstevel@tonic-gate 				    rt_to_str(RT_ATTR), pt_to_str(PT_NAME),
62767c478bd9Sstevel@tonic-gate 				    in_progress_attrtab.zone_attr_name);
6277bbec428eSgjelinek 				saw_error = B_TRUE;
62787c478bd9Sstevel@tonic-gate 				return;
62797c478bd9Sstevel@tonic-gate 			}
62807c478bd9Sstevel@tonic-gate 			err = zonecfg_add_attr(handle, &in_progress_attrtab);
62817c478bd9Sstevel@tonic-gate 		} else {
62827c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_attr(handle, &old_attrtab,
62837c478bd9Sstevel@tonic-gate 			    &in_progress_attrtab);
62847c478bd9Sstevel@tonic-gate 		}
62857c478bd9Sstevel@tonic-gate 		break;
6286fa9e4066Sahrens 	case RT_DATASET:
6287fa9e4066Sahrens 		/* First make sure everything was filled in. */
6288fa9e4066Sahrens 		if (strlen(in_progress_dstab.zone_dataset_name) == 0) {
6289fa9e4066Sahrens 			zerr("%s %s", pt_to_str(PT_NAME),
6290fa9e4066Sahrens 			    gettext("not specified"));
6291bbec428eSgjelinek 			saw_error = B_TRUE;
6292bbec428eSgjelinek 			validation_failed = B_TRUE;
6293fa9e4066Sahrens 		}
6294fa9e4066Sahrens 		if (validation_failed)
6295fa9e4066Sahrens 			return;
6296fa9e4066Sahrens 		if (end_op == CMD_ADD) {
6297fa9e4066Sahrens 			/* Make sure there isn't already one like this. */
6298fa9e4066Sahrens 			bzero(&tmp_dstab, sizeof (tmp_dstab));
6299fa9e4066Sahrens 			(void) strlcpy(tmp_dstab.zone_dataset_name,
6300fa9e4066Sahrens 			    in_progress_dstab.zone_dataset_name,
6301fa9e4066Sahrens 			    sizeof (tmp_dstab.zone_dataset_name));
6302fa9e4066Sahrens 			err = zonecfg_lookup_ds(handle, &tmp_dstab);
6303fa9e4066Sahrens 			if (err == Z_OK) {
6304fa9e4066Sahrens 				zerr(gettext("A %s resource "
6305fa9e4066Sahrens 				    "with the %s '%s' already exists."),
6306fa9e4066Sahrens 				    rt_to_str(RT_DATASET), pt_to_str(PT_NAME),
6307fa9e4066Sahrens 				    in_progress_dstab.zone_dataset_name);
6308bbec428eSgjelinek 				saw_error = B_TRUE;
6309fa9e4066Sahrens 				return;
6310fa9e4066Sahrens 			}
6311fa9e4066Sahrens 			err = zonecfg_add_ds(handle, &in_progress_dstab);
6312fa9e4066Sahrens 		} else {
6313fa9e4066Sahrens 			err = zonecfg_modify_ds(handle, &old_dstab,
6314fa9e4066Sahrens 			    &in_progress_dstab);
6315fa9e4066Sahrens 		}
6316fa9e4066Sahrens 		break;
63170209230bSgjelinek 	case RT_DCPU:
63180209230bSgjelinek 		/* Make sure everything was filled in. */
63190209230bSgjelinek 		if (end_check_reqd(in_progress_psettab.zone_ncpu_min,
63200209230bSgjelinek 		    PT_NCPUS, &validation_failed) != Z_OK) {
6321bbec428eSgjelinek 			saw_error = B_TRUE;
63220209230bSgjelinek 			return;
63230209230bSgjelinek 		}
63240209230bSgjelinek 
63250209230bSgjelinek 		if (end_op == CMD_ADD) {
63260209230bSgjelinek 			err = zonecfg_add_pset(handle, &in_progress_psettab);
63270209230bSgjelinek 		} else {
63280209230bSgjelinek 			err = zonecfg_modify_pset(handle, &in_progress_psettab);
63290209230bSgjelinek 		}
63300209230bSgjelinek 		break;
6331c97ad5cdSakolb 	case RT_PCAP:
6332c97ad5cdSakolb 		/* Make sure everything was filled in. */
6333c97ad5cdSakolb 		if (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &proc_cap)
6334c97ad5cdSakolb 		    != Z_OK) {
6335c97ad5cdSakolb 			zerr(gettext("%s not specified"), pt_to_str(PT_NCPUS));
6336bbec428eSgjelinek 			saw_error = B_TRUE;
6337bbec428eSgjelinek 			validation_failed = B_TRUE;
6338c97ad5cdSakolb 			return;
6339c97ad5cdSakolb 		}
6340c97ad5cdSakolb 		err = Z_OK;
6341c97ad5cdSakolb 		break;
63420209230bSgjelinek 	case RT_MCAP:
63430209230bSgjelinek 		/* Make sure everything was filled in. */
63440209230bSgjelinek 		res1 = strlen(in_progress_mcaptab.zone_physmem_cap) == 0 ?
63450209230bSgjelinek 		    Z_ERR : Z_OK;
63460209230bSgjelinek 		res2 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP,
63470209230bSgjelinek 		    &swap_limit);
63480209230bSgjelinek 		res3 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM,
63490209230bSgjelinek 		    &locked_limit);
63500209230bSgjelinek 
63510209230bSgjelinek 		if (res1 != Z_OK && res2 != Z_OK && res3 != Z_OK) {
63520209230bSgjelinek 			zerr(gettext("No property was specified.  One of %s, "
63530209230bSgjelinek 			    "%s or %s is required."), pt_to_str(PT_PHYSICAL),
63540209230bSgjelinek 			    pt_to_str(PT_SWAP), pt_to_str(PT_LOCKED));
6355bbec428eSgjelinek 			saw_error = B_TRUE;
63560209230bSgjelinek 			return;
63570209230bSgjelinek 		}
63580209230bSgjelinek 
63590209230bSgjelinek 		/* if phys & locked are both set, verify locked <= phys */
63600209230bSgjelinek 		if (res1 == Z_OK && res3 == Z_OK) {
63610209230bSgjelinek 			uint64_t phys_limit;
63620209230bSgjelinek 			char *endp;
63630209230bSgjelinek 
63640209230bSgjelinek 			phys_limit = strtoull(
63650209230bSgjelinek 			    in_progress_mcaptab.zone_physmem_cap, &endp, 10);
63660209230bSgjelinek 			if (phys_limit < locked_limit) {
63670209230bSgjelinek 				zerr(gettext("The %s cap must be less than or "
63680209230bSgjelinek 				    "equal to the %s cap."),
63690209230bSgjelinek 				    pt_to_str(PT_LOCKED),
63700209230bSgjelinek 				    pt_to_str(PT_PHYSICAL));
6371bbec428eSgjelinek 				saw_error = B_TRUE;
63720209230bSgjelinek 				return;
63730209230bSgjelinek 			}
63740209230bSgjelinek 		}
63750209230bSgjelinek 
63760209230bSgjelinek 		err = Z_OK;
63770209230bSgjelinek 		if (res1 == Z_OK) {
63780209230bSgjelinek 			/*
63790209230bSgjelinek 			 * We could be ending from either an add operation
63800209230bSgjelinek 			 * or a select operation.  Since all of the properties
63810209230bSgjelinek 			 * within this resource are optional, we always use
63820209230bSgjelinek 			 * modify on the mcap entry.  zonecfg_modify_mcap()
63830209230bSgjelinek 			 * will handle both adding and modifying a memory cap.
63840209230bSgjelinek 			 */
63850209230bSgjelinek 			err = zonecfg_modify_mcap(handle, &in_progress_mcaptab);
63860209230bSgjelinek 		} else if (end_op == CMD_SELECT) {
63870209230bSgjelinek 			/*
63880209230bSgjelinek 			 * If we're ending from a select and the physical
63890209230bSgjelinek 			 * memory cap is empty then the user could have cleared
63900209230bSgjelinek 			 * the physical cap value, so try to delete the entry.
63910209230bSgjelinek 			 */
63920209230bSgjelinek 			(void) zonecfg_delete_mcap(handle);
63930209230bSgjelinek 		}
63940209230bSgjelinek 		break;
6395cb8a054bSGlenn Faden 	case RT_ADMIN:
6396cb8a054bSGlenn Faden 		/* First make sure everything was filled in. */
6397cb8a054bSGlenn Faden 		if (end_check_reqd(in_progress_admintab.zone_admin_user,
6398cb8a054bSGlenn Faden 		    PT_USER, &validation_failed) == Z_OK) {
6399cb8a054bSGlenn Faden 			if (getpwnam(in_progress_admintab.zone_admin_user)
6400cb8a054bSGlenn Faden 			    == NULL) {
6401cb8a054bSGlenn Faden 				zerr(gettext("%s %s is not a valid username"),
6402cb8a054bSGlenn Faden 				    pt_to_str(PT_USER),
6403cb8a054bSGlenn Faden 				    in_progress_admintab.zone_admin_user);
6404cb8a054bSGlenn Faden 				validation_failed = B_TRUE;
6405cb8a054bSGlenn Faden 			}
6406cb8a054bSGlenn Faden 		}
6407cb8a054bSGlenn Faden 
6408cb8a054bSGlenn Faden 		if (end_check_reqd(in_progress_admintab.zone_admin_auths,
6409cb8a054bSGlenn Faden 		    PT_AUTHS, &validation_failed) == Z_OK) {
6410cb8a054bSGlenn Faden 			if (!zonecfg_valid_auths(
6411cb8a054bSGlenn Faden 			    in_progress_admintab.zone_admin_auths,
6412cb8a054bSGlenn Faden 			    zone)) {
6413cb8a054bSGlenn Faden 				validation_failed = B_TRUE;
6414cb8a054bSGlenn Faden 			}
6415cb8a054bSGlenn Faden 		}
6416cb8a054bSGlenn Faden 
6417cb8a054bSGlenn Faden 		if (validation_failed) {
6418cb8a054bSGlenn Faden 			saw_error = B_TRUE;
6419cb8a054bSGlenn Faden 			return;
6420cb8a054bSGlenn Faden 		}
6421cb8a054bSGlenn Faden 
6422cb8a054bSGlenn Faden 		if (end_op == CMD_ADD) {
6423cb8a054bSGlenn Faden 			/* Make sure there isn't already one like this. */
6424cb8a054bSGlenn Faden 			bzero(&tmp_admintab, sizeof (tmp_admintab));
6425cb8a054bSGlenn Faden 			(void) strlcpy(tmp_admintab.zone_admin_user,
6426cb8a054bSGlenn Faden 			    in_progress_admintab.zone_admin_user,
6427cb8a054bSGlenn Faden 			    sizeof (tmp_admintab.zone_admin_user));
6428cb8a054bSGlenn Faden 			err = zonecfg_lookup_admin(
6429cb8a054bSGlenn Faden 			    handle, &tmp_admintab);
6430cb8a054bSGlenn Faden 			if (err == Z_OK) {
6431cb8a054bSGlenn Faden 				zerr(gettext("A %s resource "
6432cb8a054bSGlenn Faden 				    "with the %s '%s' already exists."),
6433cb8a054bSGlenn Faden 				    rt_to_str(RT_ADMIN),
6434cb8a054bSGlenn Faden 				    pt_to_str(PT_USER),
6435cb8a054bSGlenn Faden 				    in_progress_admintab.zone_admin_user);
6436cb8a054bSGlenn Faden 				saw_error = B_TRUE;
6437cb8a054bSGlenn Faden 				return;
6438cb8a054bSGlenn Faden 			}
6439cb8a054bSGlenn Faden 			err = zonecfg_add_admin(handle,
6440cb8a054bSGlenn Faden 			    &in_progress_admintab, zone);
6441cb8a054bSGlenn Faden 		} else {
6442cb8a054bSGlenn Faden 			err = zonecfg_modify_admin(handle,
6443cb8a054bSGlenn Faden 			    &old_admintab, &in_progress_admintab,
6444cb8a054bSGlenn Faden 			    zone);
6445cb8a054bSGlenn Faden 		}
6446cb8a054bSGlenn Faden 		break;
64477c478bd9Sstevel@tonic-gate 	default:
64487c478bd9Sstevel@tonic-gate 		zone_perror(rt_to_str(resource_scope), Z_NO_RESOURCE_TYPE,
6449bbec428eSgjelinek 		    B_TRUE);
6450bbec428eSgjelinek 		saw_error = B_TRUE;
64517c478bd9Sstevel@tonic-gate 		return;
64527c478bd9Sstevel@tonic-gate 	}
64537c478bd9Sstevel@tonic-gate 
64547c478bd9Sstevel@tonic-gate 	if (err != Z_OK) {
6455bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
64567c478bd9Sstevel@tonic-gate 	} else {
6457bbec428eSgjelinek 		need_to_commit = B_TRUE;
6458bbec428eSgjelinek 		global_scope = B_TRUE;
64597c478bd9Sstevel@tonic-gate 		end_op = -1;
64607c478bd9Sstevel@tonic-gate 	}
64617c478bd9Sstevel@tonic-gate }
64627c478bd9Sstevel@tonic-gate 
64637c478bd9Sstevel@tonic-gate void
64647c478bd9Sstevel@tonic-gate commit_func(cmd_t *cmd)
64657c478bd9Sstevel@tonic-gate {
64667c478bd9Sstevel@tonic-gate 	int arg;
6467bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
64687c478bd9Sstevel@tonic-gate 
64697c478bd9Sstevel@tonic-gate 	optind = 0;
64707ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
64717c478bd9Sstevel@tonic-gate 		switch (arg) {
64727c478bd9Sstevel@tonic-gate 		case '?':
64737c478bd9Sstevel@tonic-gate 			longer_usage(CMD_COMMIT);
6474bbec428eSgjelinek 			arg_err = B_TRUE;
64757ec75eb8Sgjelinek 			break;
64767c478bd9Sstevel@tonic-gate 		default:
64777c478bd9Sstevel@tonic-gate 			short_usage(CMD_COMMIT);
6478bbec428eSgjelinek 			arg_err = B_TRUE;
64797ec75eb8Sgjelinek 			break;
64807ec75eb8Sgjelinek 		}
64817ec75eb8Sgjelinek 	}
64827ec75eb8Sgjelinek 	if (arg_err)
64837c478bd9Sstevel@tonic-gate 		return;
64847ec75eb8Sgjelinek 
64857c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
64867c478bd9Sstevel@tonic-gate 		short_usage(CMD_COMMIT);
64877c478bd9Sstevel@tonic-gate 		return;
64887c478bd9Sstevel@tonic-gate 	}
64897c478bd9Sstevel@tonic-gate 
64907c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_COMMIT))
64917c478bd9Sstevel@tonic-gate 		return;
64927c478bd9Sstevel@tonic-gate 
64937c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
64947c478bd9Sstevel@tonic-gate 
64957c478bd9Sstevel@tonic-gate 	cmd->cmd_argc = 1;
64967c478bd9Sstevel@tonic-gate 	/*
64977c478bd9Sstevel@tonic-gate 	 * cmd_arg normally comes from a strdup() in the lexer, and the
64987c478bd9Sstevel@tonic-gate 	 * whole cmd structure and its (char *) attributes are freed at
64997c478bd9Sstevel@tonic-gate 	 * the completion of each command, so the strdup() below is needed
65007c478bd9Sstevel@tonic-gate 	 * to match this and prevent a core dump from trying to free()
65017c478bd9Sstevel@tonic-gate 	 * something that can't be.
65027c478bd9Sstevel@tonic-gate 	 */
65037c478bd9Sstevel@tonic-gate 	if ((cmd->cmd_argv[0] = strdup("save")) == NULL) {
6504bbec428eSgjelinek 		zone_perror(zone, Z_NOMEM, B_TRUE);
65057c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
65067c478bd9Sstevel@tonic-gate 	}
65077c478bd9Sstevel@tonic-gate 	cmd->cmd_argv[1] = NULL;
65087c478bd9Sstevel@tonic-gate 	verify_func(cmd);
65097c478bd9Sstevel@tonic-gate }
65107c478bd9Sstevel@tonic-gate 
65117c478bd9Sstevel@tonic-gate void
65127c478bd9Sstevel@tonic-gate revert_func(cmd_t *cmd)
65137c478bd9Sstevel@tonic-gate {
65147c478bd9Sstevel@tonic-gate 	char line[128];	/* enough to ask a question */
6515bbec428eSgjelinek 	boolean_t force = B_FALSE;
6516bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
65177c478bd9Sstevel@tonic-gate 	int err, arg, answer;
65187c478bd9Sstevel@tonic-gate 
65197c478bd9Sstevel@tonic-gate 	optind = 0;
65207c478bd9Sstevel@tonic-gate 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?F")) != EOF) {
65217c478bd9Sstevel@tonic-gate 		switch (arg) {
65227c478bd9Sstevel@tonic-gate 		case '?':
65237c478bd9Sstevel@tonic-gate 			longer_usage(CMD_REVERT);
6524bbec428eSgjelinek 			arg_err = B_TRUE;
65257ec75eb8Sgjelinek 			break;
65267c478bd9Sstevel@tonic-gate 		case 'F':
6527bbec428eSgjelinek 			force = B_TRUE;
65287c478bd9Sstevel@tonic-gate 			break;
65297c478bd9Sstevel@tonic-gate 		default:
65307c478bd9Sstevel@tonic-gate 			short_usage(CMD_REVERT);
6531bbec428eSgjelinek 			arg_err = B_TRUE;
65327ec75eb8Sgjelinek 			break;
65337ec75eb8Sgjelinek 		}
65347ec75eb8Sgjelinek 	}
65357ec75eb8Sgjelinek 	if (arg_err)
65367c478bd9Sstevel@tonic-gate 		return;
65377ec75eb8Sgjelinek 
65387c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
65397c478bd9Sstevel@tonic-gate 		short_usage(CMD_REVERT);
65407c478bd9Sstevel@tonic-gate 		return;
65417c478bd9Sstevel@tonic-gate 	}
65427c478bd9Sstevel@tonic-gate 
65437c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_REVERT))
65447c478bd9Sstevel@tonic-gate 		return;
65457c478bd9Sstevel@tonic-gate 
65465b418297Sjv227347 	if (!global_scope) {
65475b418297Sjv227347 		zerr(gettext("You can only use %s in the global scope.\nUse"
65485b418297Sjv227347 		    " '%s' to cancel changes to a resource specification."),
65495b418297Sjv227347 		    cmd_to_str(CMD_REVERT), cmd_to_str(CMD_CANCEL));
65505b418297Sjv227347 		saw_error = B_TRUE;
65515b418297Sjv227347 		return;
65525b418297Sjv227347 	}
65535b418297Sjv227347 
65547c478bd9Sstevel@tonic-gate 	if (zonecfg_check_handle(handle) != Z_OK) {
65557c478bd9Sstevel@tonic-gate 		zerr(gettext("No changes to revert."));
6556bbec428eSgjelinek 		saw_error = B_TRUE;
65577c478bd9Sstevel@tonic-gate 		return;
65587c478bd9Sstevel@tonic-gate 	}
65597c478bd9Sstevel@tonic-gate 
65607c478bd9Sstevel@tonic-gate 	if (!force) {
65617c478bd9Sstevel@tonic-gate 		(void) snprintf(line, sizeof (line),
65627c478bd9Sstevel@tonic-gate 		    gettext("Are you sure you want to revert"));
6563bbec428eSgjelinek 		if ((answer = ask_yesno(B_FALSE, line)) == -1) {
65647c478bd9Sstevel@tonic-gate 			zerr(gettext("Input not from terminal and -F not "
65657c478bd9Sstevel@tonic-gate 			    "specified:\n%s command ignored, exiting."),
65667c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_REVERT));
65677c478bd9Sstevel@tonic-gate 			exit(Z_ERR);
65687c478bd9Sstevel@tonic-gate 		}
65697c478bd9Sstevel@tonic-gate 		if (answer != 1)
65707c478bd9Sstevel@tonic-gate 			return;
65717c478bd9Sstevel@tonic-gate 	}
65727c478bd9Sstevel@tonic-gate 
65737c478bd9Sstevel@tonic-gate 	/*
6574cb8a054bSGlenn Faden 	 * Reset any pending admins that were
6575cb8a054bSGlenn Faden 	 * removed from the previous zone
6576cb8a054bSGlenn Faden 	 */
6577cb8a054bSGlenn Faden 	zonecfg_remove_userauths(handle, "", zone, B_FALSE);
6578cb8a054bSGlenn Faden 
6579cb8a054bSGlenn Faden 	/*
65807c478bd9Sstevel@tonic-gate 	 * Time for a new handle: finish the old one off first
65817c478bd9Sstevel@tonic-gate 	 * then get a new one properly to avoid leaks.
65827c478bd9Sstevel@tonic-gate 	 */
65837c478bd9Sstevel@tonic-gate 	zonecfg_fini_handle(handle);
65847c478bd9Sstevel@tonic-gate 	if ((handle = zonecfg_init_handle()) == NULL) {
6585bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
65867c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
65877c478bd9Sstevel@tonic-gate 	}
6588cb8a054bSGlenn Faden 
6589087719fdSdp 	if ((err = zonecfg_get_handle(revert_zone, handle)) != Z_OK) {
6590bbec428eSgjelinek 		saw_error = B_TRUE;
6591bbec428eSgjelinek 		got_handle = B_FALSE;
65927c478bd9Sstevel@tonic-gate 		if (err == Z_NO_ZONE)
65937c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: no such saved zone to revert to."),
6594087719fdSdp 			    revert_zone);
65957c478bd9Sstevel@tonic-gate 		else
6596bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
65977c478bd9Sstevel@tonic-gate 	}
6598087719fdSdp 	(void) strlcpy(zone, revert_zone, sizeof (zone));
65997c478bd9Sstevel@tonic-gate }
66007c478bd9Sstevel@tonic-gate 
66017c478bd9Sstevel@tonic-gate void
66027c478bd9Sstevel@tonic-gate help_func(cmd_t *cmd)
66037c478bd9Sstevel@tonic-gate {
66047c478bd9Sstevel@tonic-gate 	int i;
66057c478bd9Sstevel@tonic-gate 
66067c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
66077c478bd9Sstevel@tonic-gate 
66087c478bd9Sstevel@tonic-gate 	if (cmd->cmd_argc == 0) {
6609bbec428eSgjelinek 		usage(B_TRUE, global_scope ? HELP_SUBCMDS : HELP_RES_SCOPE);
66107c478bd9Sstevel@tonic-gate 		return;
66117c478bd9Sstevel@tonic-gate 	}
66127c478bd9Sstevel@tonic-gate 	if (strcmp(cmd->cmd_argv[0], "usage") == 0) {
6613bbec428eSgjelinek 		usage(B_TRUE, HELP_USAGE);
66147c478bd9Sstevel@tonic-gate 		return;
66157c478bd9Sstevel@tonic-gate 	}
66167c478bd9Sstevel@tonic-gate 	if (strcmp(cmd->cmd_argv[0], "commands") == 0) {
6617bbec428eSgjelinek 		usage(B_TRUE, HELP_SUBCMDS);
66187c478bd9Sstevel@tonic-gate 		return;
66197c478bd9Sstevel@tonic-gate 	}
66207c478bd9Sstevel@tonic-gate 	if (strcmp(cmd->cmd_argv[0], "syntax") == 0) {
6621bbec428eSgjelinek 		usage(B_TRUE, HELP_SYNTAX | HELP_RES_PROPS);
66227c478bd9Sstevel@tonic-gate 		return;
66237c478bd9Sstevel@tonic-gate 	}
66247c478bd9Sstevel@tonic-gate 	if (strcmp(cmd->cmd_argv[0], "-?") == 0) {
66257c478bd9Sstevel@tonic-gate 		longer_usage(CMD_HELP);
66267c478bd9Sstevel@tonic-gate 		return;
66277c478bd9Sstevel@tonic-gate 	}
66287c478bd9Sstevel@tonic-gate 
66297c478bd9Sstevel@tonic-gate 	for (i = 0; i <= CMD_MAX; i++) {
66307c478bd9Sstevel@tonic-gate 		if (strcmp(cmd->cmd_argv[0], cmd_to_str(i)) == 0) {
66317c478bd9Sstevel@tonic-gate 			longer_usage(i);
66327c478bd9Sstevel@tonic-gate 			return;
66337c478bd9Sstevel@tonic-gate 		}
66347c478bd9Sstevel@tonic-gate 	}
66357c478bd9Sstevel@tonic-gate 	/* We do not use zerr() here because we do not want its extra \n. */
66367c478bd9Sstevel@tonic-gate 	(void) fprintf(stderr, gettext("Unknown help subject %s.  "),
66377c478bd9Sstevel@tonic-gate 	    cmd->cmd_argv[0]);
6638bbec428eSgjelinek 	usage(B_FALSE, HELP_META);
66397c478bd9Sstevel@tonic-gate }
66407c478bd9Sstevel@tonic-gate 
66417c478bd9Sstevel@tonic-gate static int
66427c478bd9Sstevel@tonic-gate string_to_yyin(char *string)
66437c478bd9Sstevel@tonic-gate {
66447c478bd9Sstevel@tonic-gate 	if ((yyin = tmpfile()) == NULL) {
6645bbec428eSgjelinek 		zone_perror(execname, Z_TEMP_FILE, B_TRUE);
66467c478bd9Sstevel@tonic-gate 		return (Z_ERR);
66477c478bd9Sstevel@tonic-gate 	}
66487c478bd9Sstevel@tonic-gate 	if (fwrite(string, strlen(string), 1, yyin) != 1) {
6649bbec428eSgjelinek 		zone_perror(execname, Z_TEMP_FILE, B_TRUE);
66507c478bd9Sstevel@tonic-gate 		return (Z_ERR);
66517c478bd9Sstevel@tonic-gate 	}
66527c478bd9Sstevel@tonic-gate 	if (fseek(yyin, 0, SEEK_SET) != 0) {
6653bbec428eSgjelinek 		zone_perror(execname, Z_TEMP_FILE, B_TRUE);
66547c478bd9Sstevel@tonic-gate 		return (Z_ERR);
66557c478bd9Sstevel@tonic-gate 	}
66567c478bd9Sstevel@tonic-gate 	return (Z_OK);
66577c478bd9Sstevel@tonic-gate }
66587c478bd9Sstevel@tonic-gate 
66597c478bd9Sstevel@tonic-gate /* This is the back-end helper function for read_input() below. */
66607c478bd9Sstevel@tonic-gate 
66617c478bd9Sstevel@tonic-gate static int
66627c478bd9Sstevel@tonic-gate cleanup()
66637c478bd9Sstevel@tonic-gate {
66647c478bd9Sstevel@tonic-gate 	int answer;
66657c478bd9Sstevel@tonic-gate 	cmd_t *cmd;
66667c478bd9Sstevel@tonic-gate 
66677c478bd9Sstevel@tonic-gate 	if (!interactive_mode && !cmd_file_mode) {
66687c478bd9Sstevel@tonic-gate 		/*
66697c478bd9Sstevel@tonic-gate 		 * If we're not in interactive mode, and we're not in command
66707c478bd9Sstevel@tonic-gate 		 * file mode, then we must be in commands-from-the-command-line
66717c478bd9Sstevel@tonic-gate 		 * mode.  As such, we can't loop back and ask for more input.
66727c478bd9Sstevel@tonic-gate 		 * It was OK to prompt for such things as whether or not to
66737c478bd9Sstevel@tonic-gate 		 * really delete a zone in the command handler called from
66747c478bd9Sstevel@tonic-gate 		 * yyparse() above, but "really quit?" makes no sense in this
66757c478bd9Sstevel@tonic-gate 		 * context.  So disable prompting.
66767c478bd9Sstevel@tonic-gate 		 */
6677bbec428eSgjelinek 		ok_to_prompt = B_FALSE;
66787c478bd9Sstevel@tonic-gate 	}
66797c478bd9Sstevel@tonic-gate 	if (!global_scope) {
66807c478bd9Sstevel@tonic-gate 		if (!time_to_exit) {
66817c478bd9Sstevel@tonic-gate 			/*
66827c478bd9Sstevel@tonic-gate 			 * Just print a simple error message in the -1 case,
66837c478bd9Sstevel@tonic-gate 			 * since exit_func() already handles that case, and
66847c478bd9Sstevel@tonic-gate 			 * EOF means we are finished anyway.
66857c478bd9Sstevel@tonic-gate 			 */
6686bbec428eSgjelinek 			answer = ask_yesno(B_FALSE,
66877c478bd9Sstevel@tonic-gate 			    gettext("Resource incomplete; really quit"));
66887c478bd9Sstevel@tonic-gate 			if (answer == -1) {
66897c478bd9Sstevel@tonic-gate 				zerr(gettext("Resource incomplete."));
66907c478bd9Sstevel@tonic-gate 				return (Z_ERR);
66917c478bd9Sstevel@tonic-gate 			}
66927c478bd9Sstevel@tonic-gate 			if (answer != 1) {
66937c478bd9Sstevel@tonic-gate 				yyin = stdin;
66947c478bd9Sstevel@tonic-gate 				return (Z_REPEAT);
66957c478bd9Sstevel@tonic-gate 			}
66967c478bd9Sstevel@tonic-gate 		} else {
6697bbec428eSgjelinek 			saw_error = B_TRUE;
66987c478bd9Sstevel@tonic-gate 		}
66997c478bd9Sstevel@tonic-gate 	}
67007c478bd9Sstevel@tonic-gate 	/*
67017c478bd9Sstevel@tonic-gate 	 * Make sure we tried something and that the handle checks
67027c478bd9Sstevel@tonic-gate 	 * out, or we would get a false error trying to commit.
67037c478bd9Sstevel@tonic-gate 	 */
67047c478bd9Sstevel@tonic-gate 	if (need_to_commit && zonecfg_check_handle(handle) == Z_OK) {
67057c478bd9Sstevel@tonic-gate 		if ((cmd = alloc_cmd()) == NULL) {
6706bbec428eSgjelinek 			zone_perror(zone, Z_NOMEM, B_TRUE);
67077c478bd9Sstevel@tonic-gate 			return (Z_ERR);
67087c478bd9Sstevel@tonic-gate 		}
67097c478bd9Sstevel@tonic-gate 		cmd->cmd_argc = 0;
67107c478bd9Sstevel@tonic-gate 		cmd->cmd_argv[0] = NULL;
67117c478bd9Sstevel@tonic-gate 		commit_func(cmd);
67127c478bd9Sstevel@tonic-gate 		free_cmd(cmd);
67137c478bd9Sstevel@tonic-gate 		/*
67147c478bd9Sstevel@tonic-gate 		 * need_to_commit will get set back to FALSE if the
67157c478bd9Sstevel@tonic-gate 		 * configuration is saved successfully.
67167c478bd9Sstevel@tonic-gate 		 */
67177c478bd9Sstevel@tonic-gate 		if (need_to_commit) {
67187c478bd9Sstevel@tonic-gate 			if (force_exit) {
67197c478bd9Sstevel@tonic-gate 				zerr(gettext("Configuration not saved."));
67207c478bd9Sstevel@tonic-gate 				return (Z_ERR);
67217c478bd9Sstevel@tonic-gate 			}
6722bbec428eSgjelinek 			answer = ask_yesno(B_FALSE,
67237c478bd9Sstevel@tonic-gate 			    gettext("Configuration not saved; really quit"));
67247c478bd9Sstevel@tonic-gate 			if (answer == -1) {
67257c478bd9Sstevel@tonic-gate 				zerr(gettext("Configuration not saved."));
67267c478bd9Sstevel@tonic-gate 				return (Z_ERR);
67277c478bd9Sstevel@tonic-gate 			}
67287c478bd9Sstevel@tonic-gate 			if (answer != 1) {
6729bbec428eSgjelinek 				time_to_exit = B_FALSE;
67307c478bd9Sstevel@tonic-gate 				yyin = stdin;
67317c478bd9Sstevel@tonic-gate 				return (Z_REPEAT);
67327c478bd9Sstevel@tonic-gate 			}
67337c478bd9Sstevel@tonic-gate 		}
67347c478bd9Sstevel@tonic-gate 	}
67357c478bd9Sstevel@tonic-gate 	return ((need_to_commit || saw_error) ? Z_ERR : Z_OK);
67367c478bd9Sstevel@tonic-gate }
67377c478bd9Sstevel@tonic-gate 
67387c478bd9Sstevel@tonic-gate /*
67397c478bd9Sstevel@tonic-gate  * read_input() is the driver of this program.  It is a wrapper around
67407c478bd9Sstevel@tonic-gate  * yyparse(), printing appropriate prompts when needed, checking for
67417c478bd9Sstevel@tonic-gate  * exit conditions and reacting appropriately [the latter in its cleanup()
67427c478bd9Sstevel@tonic-gate  * helper function].
67437c478bd9Sstevel@tonic-gate  *
67447c478bd9Sstevel@tonic-gate  * Like most zonecfg functions, it returns Z_OK or Z_ERR, *or* Z_REPEAT
67457c478bd9Sstevel@tonic-gate  * so do_interactive() knows that we are not really done (i.e, we asked
67467c478bd9Sstevel@tonic-gate  * the user if we should really quit and the user said no).
67477c478bd9Sstevel@tonic-gate  */
67487c478bd9Sstevel@tonic-gate static int
67497c478bd9Sstevel@tonic-gate read_input()
67507c478bd9Sstevel@tonic-gate {
6751bbec428eSgjelinek 	boolean_t yyin_is_a_tty = isatty(fileno(yyin));
67527c478bd9Sstevel@tonic-gate 	/*
67537c478bd9Sstevel@tonic-gate 	 * The prompt is "e:z> " or "e:z:r> " where e is execname, z is zone
67547c478bd9Sstevel@tonic-gate 	 * and r is resource_scope: 5 is for the two ":"s + "> " + terminator.
67557c478bd9Sstevel@tonic-gate 	 */
67567c478bd9Sstevel@tonic-gate 	char prompt[MAXPATHLEN + ZONENAME_MAX + MAX_RT_STRLEN + 5], *line;
67577c478bd9Sstevel@tonic-gate 
67587c478bd9Sstevel@tonic-gate 	/* yyin should have been set to the appropriate (FILE *) if not stdin */
6759bbec428eSgjelinek 	newline_terminated = B_TRUE;
67607c478bd9Sstevel@tonic-gate 	for (;;) {
67617c478bd9Sstevel@tonic-gate 		if (yyin_is_a_tty) {
67627c478bd9Sstevel@tonic-gate 			if (newline_terminated) {
67637c478bd9Sstevel@tonic-gate 				if (global_scope)
67647c478bd9Sstevel@tonic-gate 					(void) snprintf(prompt, sizeof (prompt),
67657c478bd9Sstevel@tonic-gate 					    "%s:%s> ", execname, zone);
67667c478bd9Sstevel@tonic-gate 				else
67677c478bd9Sstevel@tonic-gate 					(void) snprintf(prompt, sizeof (prompt),
67687c478bd9Sstevel@tonic-gate 					    "%s:%s:%s> ", execname, zone,
67697c478bd9Sstevel@tonic-gate 					    rt_to_str(resource_scope));
67707c478bd9Sstevel@tonic-gate 			}
67717c478bd9Sstevel@tonic-gate 			/*
67727c478bd9Sstevel@tonic-gate 			 * If the user hits ^C then we want to catch it and
67737c478bd9Sstevel@tonic-gate 			 * start over.  If the user hits EOF then we want to
67747c478bd9Sstevel@tonic-gate 			 * bail out.
67757c478bd9Sstevel@tonic-gate 			 */
67767c478bd9Sstevel@tonic-gate 			line = gl_get_line(gl, prompt, NULL, -1);
67777c478bd9Sstevel@tonic-gate 			if (gl_return_status(gl) == GLR_SIGNAL) {
67787c478bd9Sstevel@tonic-gate 				gl_abandon_line(gl);
67797c478bd9Sstevel@tonic-gate 				continue;
67807c478bd9Sstevel@tonic-gate 			}
67817c478bd9Sstevel@tonic-gate 			if (line == NULL)
67827c478bd9Sstevel@tonic-gate 				break;
67837c478bd9Sstevel@tonic-gate 			(void) string_to_yyin(line);
67847c478bd9Sstevel@tonic-gate 			while (!feof(yyin))
67857c478bd9Sstevel@tonic-gate 				yyparse();
67867c478bd9Sstevel@tonic-gate 		} else {
67877c478bd9Sstevel@tonic-gate 			yyparse();
67887c478bd9Sstevel@tonic-gate 		}
67897c478bd9Sstevel@tonic-gate 		/* Bail out on an error in command file mode. */
67907c478bd9Sstevel@tonic-gate 		if (saw_error && cmd_file_mode && !interactive_mode)
6791bbec428eSgjelinek 			time_to_exit = B_TRUE;
67927c478bd9Sstevel@tonic-gate 		if (time_to_exit || (!yyin_is_a_tty && feof(yyin)))
67937c478bd9Sstevel@tonic-gate 			break;
67947c478bd9Sstevel@tonic-gate 	}
67957c478bd9Sstevel@tonic-gate 	return (cleanup());
67967c478bd9Sstevel@tonic-gate }
67977c478bd9Sstevel@tonic-gate 
67987c478bd9Sstevel@tonic-gate /*
67997c478bd9Sstevel@tonic-gate  * This function is used in the zonecfg-interactive-mode scenario: it just
68007c478bd9Sstevel@tonic-gate  * calls read_input() until we are done.
68017c478bd9Sstevel@tonic-gate  */
68027c478bd9Sstevel@tonic-gate 
68037c478bd9Sstevel@tonic-gate static int
68047c478bd9Sstevel@tonic-gate do_interactive(void)
68057c478bd9Sstevel@tonic-gate {
68067c478bd9Sstevel@tonic-gate 	int err;
68077c478bd9Sstevel@tonic-gate 
6808bbec428eSgjelinek 	interactive_mode = B_TRUE;
68097c478bd9Sstevel@tonic-gate 	if (!read_only_mode) {
68107c478bd9Sstevel@tonic-gate 		/*
68117c478bd9Sstevel@tonic-gate 		 * Try to set things up proactively in interactive mode, so
68127c478bd9Sstevel@tonic-gate 		 * that if the zone in question does not exist yet, we can
68137c478bd9Sstevel@tonic-gate 		 * provide the user with a clue.
68147c478bd9Sstevel@tonic-gate 		 */
6815bbec428eSgjelinek 		(void) initialize(B_FALSE);
68167c478bd9Sstevel@tonic-gate 	}
6817087719fdSdp 	do {
68187c478bd9Sstevel@tonic-gate 		err = read_input();
6819087719fdSdp 	} while (err == Z_REPEAT);
68207c478bd9Sstevel@tonic-gate 	return (err);
68217c478bd9Sstevel@tonic-gate }
68227c478bd9Sstevel@tonic-gate 
68237c478bd9Sstevel@tonic-gate /*
68247c478bd9Sstevel@tonic-gate  * cmd_file is slightly more complicated, as it has to open the command file
68257c478bd9Sstevel@tonic-gate  * and set yyin appropriately.  Once that is done, though, it just calls
68267c478bd9Sstevel@tonic-gate  * read_input(), and only once, since prompting is not possible.
68277c478bd9Sstevel@tonic-gate  */
68287c478bd9Sstevel@tonic-gate 
68297c478bd9Sstevel@tonic-gate static int
68307c478bd9Sstevel@tonic-gate cmd_file(char *file)
68317c478bd9Sstevel@tonic-gate {
68327c478bd9Sstevel@tonic-gate 	FILE *infile;
68337c478bd9Sstevel@tonic-gate 	int err;
68347c478bd9Sstevel@tonic-gate 	struct stat statbuf;
6835bbec428eSgjelinek 	boolean_t using_real_file = (strcmp(file, "-") != 0);
68367c478bd9Sstevel@tonic-gate 
68377c478bd9Sstevel@tonic-gate 	if (using_real_file) {
68387c478bd9Sstevel@tonic-gate 		/*
68397c478bd9Sstevel@tonic-gate 		 * zerr() prints a line number in cmd_file_mode, which we do
68407c478bd9Sstevel@tonic-gate 		 * not want here, so temporarily unset it.
68417c478bd9Sstevel@tonic-gate 		 */
6842bbec428eSgjelinek 		cmd_file_mode = B_FALSE;
68437c478bd9Sstevel@tonic-gate 		if ((infile = fopen(file, "r")) == NULL) {
68447c478bd9Sstevel@tonic-gate 			zerr(gettext("could not open file %s: %s"),
68457c478bd9Sstevel@tonic-gate 			    file, strerror(errno));
68467c478bd9Sstevel@tonic-gate 			return (Z_ERR);
68477c478bd9Sstevel@tonic-gate 		}
68487c478bd9Sstevel@tonic-gate 		if ((err = fstat(fileno(infile), &statbuf)) != 0) {
68497c478bd9Sstevel@tonic-gate 			zerr(gettext("could not stat file %s: %s"),
68507c478bd9Sstevel@tonic-gate 			    file, strerror(errno));
68517c478bd9Sstevel@tonic-gate 			err = Z_ERR;
68527c478bd9Sstevel@tonic-gate 			goto done;
68537c478bd9Sstevel@tonic-gate 		}
68547c478bd9Sstevel@tonic-gate 		if (!S_ISREG(statbuf.st_mode)) {
68557c478bd9Sstevel@tonic-gate 			zerr(gettext("%s is not a regular file."), file);
68567c478bd9Sstevel@tonic-gate 			err = Z_ERR;
68577c478bd9Sstevel@tonic-gate 			goto done;
68587c478bd9Sstevel@tonic-gate 		}
68597c478bd9Sstevel@tonic-gate 		yyin = infile;
6860bbec428eSgjelinek 		cmd_file_mode = B_TRUE;
6861bbec428eSgjelinek 		ok_to_prompt = B_FALSE;
68627c478bd9Sstevel@tonic-gate 	} else {
68637c478bd9Sstevel@tonic-gate 		/*
68647c478bd9Sstevel@tonic-gate 		 * "-f -" is essentially the same as interactive mode,
68657c478bd9Sstevel@tonic-gate 		 * so treat it that way.
68667c478bd9Sstevel@tonic-gate 		 */
6867bbec428eSgjelinek 		interactive_mode = B_TRUE;
68687c478bd9Sstevel@tonic-gate 	}
68697c478bd9Sstevel@tonic-gate 	/* Z_REPEAT is for interactive mode; treat it like Z_ERR here. */
68707c478bd9Sstevel@tonic-gate 	if ((err = read_input()) == Z_REPEAT)
68717c478bd9Sstevel@tonic-gate 		err = Z_ERR;
68727c478bd9Sstevel@tonic-gate done:
68737c478bd9Sstevel@tonic-gate 	if (using_real_file)
68747c478bd9Sstevel@tonic-gate 		(void) fclose(infile);
68757c478bd9Sstevel@tonic-gate 	return (err);
68767c478bd9Sstevel@tonic-gate }
68777c478bd9Sstevel@tonic-gate 
68787c478bd9Sstevel@tonic-gate /*
68797c478bd9Sstevel@tonic-gate  * Since yacc is based on reading from a (FILE *) whereas what we get from
68807c478bd9Sstevel@tonic-gate  * the command line is in argv format, we need to convert when the user
68817c478bd9Sstevel@tonic-gate  * gives us commands directly from the command line.  That is done here by
68827c478bd9Sstevel@tonic-gate  * concatenating the argv list into a space-separated string, writing it
68837c478bd9Sstevel@tonic-gate  * to a temp file, and rewinding the file so yyin can be set to it.  Then
68847c478bd9Sstevel@tonic-gate  * we call read_input(), and only once, since prompting about whether to
68857c478bd9Sstevel@tonic-gate  * continue or quit would make no sense in this context.
68867c478bd9Sstevel@tonic-gate  */
68877c478bd9Sstevel@tonic-gate 
68887c478bd9Sstevel@tonic-gate static int
68897c478bd9Sstevel@tonic-gate one_command_at_a_time(int argc, char *argv[])
68907c478bd9Sstevel@tonic-gate {
68917c478bd9Sstevel@tonic-gate 	char *command;
68927c478bd9Sstevel@tonic-gate 	size_t len = 2; /* terminal \n\0 */
68937c478bd9Sstevel@tonic-gate 	int i, err;
68947c478bd9Sstevel@tonic-gate 
68957c478bd9Sstevel@tonic-gate 	for (i = 0; i < argc; i++)
68967c478bd9Sstevel@tonic-gate 		len += strlen(argv[i]) + 1;
68977c478bd9Sstevel@tonic-gate 	if ((command = malloc(len)) == NULL) {
6898bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
68997c478bd9Sstevel@tonic-gate 		return (Z_ERR);
69007c478bd9Sstevel@tonic-gate 	}
69017c478bd9Sstevel@tonic-gate 	(void) strlcpy(command, argv[0], len);
69027c478bd9Sstevel@tonic-gate 	for (i = 1; i < argc; i++) {
69037c478bd9Sstevel@tonic-gate 		(void) strlcat(command, " ", len);
69047c478bd9Sstevel@tonic-gate 		(void) strlcat(command, argv[i], len);
69057c478bd9Sstevel@tonic-gate 	}
69067c478bd9Sstevel@tonic-gate 	(void) strlcat(command, "\n", len);
69077c478bd9Sstevel@tonic-gate 	err = string_to_yyin(command);
69087c478bd9Sstevel@tonic-gate 	free(command);
69097c478bd9Sstevel@tonic-gate 	if (err != Z_OK)
69107c478bd9Sstevel@tonic-gate 		return (err);
69117c478bd9Sstevel@tonic-gate 	while (!feof(yyin))
69127c478bd9Sstevel@tonic-gate 		yyparse();
69137c478bd9Sstevel@tonic-gate 	return (cleanup());
69147c478bd9Sstevel@tonic-gate }
69157c478bd9Sstevel@tonic-gate 
69167c478bd9Sstevel@tonic-gate static char *
69177c478bd9Sstevel@tonic-gate get_execbasename(char *execfullname)
69187c478bd9Sstevel@tonic-gate {
69197c478bd9Sstevel@tonic-gate 	char *last_slash, *execbasename;
69207c478bd9Sstevel@tonic-gate 
69217c478bd9Sstevel@tonic-gate 	/* guard against '/' at end of command invocation */
69227c478bd9Sstevel@tonic-gate 	for (;;) {
69237c478bd9Sstevel@tonic-gate 		last_slash = strrchr(execfullname, '/');
69247c478bd9Sstevel@tonic-gate 		if (last_slash == NULL) {
69257c478bd9Sstevel@tonic-gate 			execbasename = execfullname;
69267c478bd9Sstevel@tonic-gate 			break;
69277c478bd9Sstevel@tonic-gate 		} else {
69287c478bd9Sstevel@tonic-gate 			execbasename = last_slash + 1;
69297c478bd9Sstevel@tonic-gate 			if (*execbasename == '\0') {
69307c478bd9Sstevel@tonic-gate 				*last_slash = '\0';
69317c478bd9Sstevel@tonic-gate 				continue;
69327c478bd9Sstevel@tonic-gate 			}
69337c478bd9Sstevel@tonic-gate 			break;
69347c478bd9Sstevel@tonic-gate 		}
69357c478bd9Sstevel@tonic-gate 	}
69367c478bd9Sstevel@tonic-gate 	return (execbasename);
69377c478bd9Sstevel@tonic-gate }
69387c478bd9Sstevel@tonic-gate 
69397c478bd9Sstevel@tonic-gate int
69407c478bd9Sstevel@tonic-gate main(int argc, char *argv[])
69417c478bd9Sstevel@tonic-gate {
69427c478bd9Sstevel@tonic-gate 	int err, arg;
6943555afedfScarlsonj 	struct stat st;
69447c478bd9Sstevel@tonic-gate 
69457c478bd9Sstevel@tonic-gate 	/* This must be before anything goes to stdout. */
69467c478bd9Sstevel@tonic-gate 	setbuf(stdout, NULL);
69477c478bd9Sstevel@tonic-gate 
6948bbec428eSgjelinek 	saw_error = B_FALSE;
6949bbec428eSgjelinek 	cmd_file_mode = B_FALSE;
69507c478bd9Sstevel@tonic-gate 	execname = get_execbasename(argv[0]);
69517c478bd9Sstevel@tonic-gate 
69527c478bd9Sstevel@tonic-gate 	(void) setlocale(LC_ALL, "");
69537c478bd9Sstevel@tonic-gate 	(void) textdomain(TEXT_DOMAIN);
69547c478bd9Sstevel@tonic-gate 
69557c478bd9Sstevel@tonic-gate 	if (getzoneid() != GLOBAL_ZONEID) {
69567c478bd9Sstevel@tonic-gate 		zerr(gettext("%s can only be run from the global zone."),
69577c478bd9Sstevel@tonic-gate 		    execname);
69587c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
69597c478bd9Sstevel@tonic-gate 	}
69607c478bd9Sstevel@tonic-gate 
69617c478bd9Sstevel@tonic-gate 	if (argc < 2) {
6962bbec428eSgjelinek 		usage(B_FALSE, HELP_USAGE | HELP_SUBCMDS);
69637c478bd9Sstevel@tonic-gate 		exit(Z_USAGE);
69647c478bd9Sstevel@tonic-gate 	}
69657c478bd9Sstevel@tonic-gate 	if (strcmp(argv[1], cmd_to_str(CMD_HELP)) == 0) {
69667c478bd9Sstevel@tonic-gate 		(void) one_command_at_a_time(argc - 1, &(argv[1]));
69677c478bd9Sstevel@tonic-gate 		exit(Z_OK);
69687c478bd9Sstevel@tonic-gate 	}
69697c478bd9Sstevel@tonic-gate 
6970555afedfScarlsonj 	while ((arg = getopt(argc, argv, "?f:R:z:")) != EOF) {
69717c478bd9Sstevel@tonic-gate 		switch (arg) {
69727c478bd9Sstevel@tonic-gate 		case '?':
69737c478bd9Sstevel@tonic-gate 			if (optopt == '?')
6974bbec428eSgjelinek 				usage(B_TRUE, HELP_USAGE | HELP_SUBCMDS);
69757c478bd9Sstevel@tonic-gate 			else
6976bbec428eSgjelinek 				usage(B_FALSE, HELP_USAGE);
69777c478bd9Sstevel@tonic-gate 			exit(Z_USAGE);
69787c478bd9Sstevel@tonic-gate 			/* NOTREACHED */
69797c478bd9Sstevel@tonic-gate 		case 'f':
69807c478bd9Sstevel@tonic-gate 			cmd_file_name = optarg;
6981bbec428eSgjelinek 			cmd_file_mode = B_TRUE;
69827c478bd9Sstevel@tonic-gate 			break;
6983555afedfScarlsonj 		case 'R':
6984555afedfScarlsonj 			if (*optarg != '/') {
6985555afedfScarlsonj 				zerr(gettext("root path must be absolute: %s"),
6986555afedfScarlsonj 				    optarg);
6987555afedfScarlsonj 				exit(Z_USAGE);
6988555afedfScarlsonj 			}
6989555afedfScarlsonj 			if (stat(optarg, &st) == -1 || !S_ISDIR(st.st_mode)) {
6990555afedfScarlsonj 				zerr(gettext(
6991555afedfScarlsonj 				    "root path must be a directory: %s"),
6992555afedfScarlsonj 				    optarg);
6993555afedfScarlsonj 				exit(Z_USAGE);
6994555afedfScarlsonj 			}
6995555afedfScarlsonj 			zonecfg_set_root(optarg);
6996555afedfScarlsonj 			break;
69977c478bd9Sstevel@tonic-gate 		case 'z':
69980209230bSgjelinek 			if (strcmp(optarg, GLOBAL_ZONENAME) == 0) {
6999bbec428eSgjelinek 				global_zone = B_TRUE;
70000209230bSgjelinek 			} else if (zonecfg_validate_zonename(optarg) != Z_OK) {
7001bbec428eSgjelinek 				zone_perror(optarg, Z_BOGUS_ZONE_NAME, B_TRUE);
7002bbec428eSgjelinek 				usage(B_FALSE, HELP_SYNTAX);
7003087719fdSdp 				exit(Z_USAGE);
7004087719fdSdp 			}
7005087719fdSdp 			(void) strlcpy(zone, optarg, sizeof (zone));
7006087719fdSdp 			(void) strlcpy(revert_zone, optarg, sizeof (zone));
70077c478bd9Sstevel@tonic-gate 			break;
70087c478bd9Sstevel@tonic-gate 		default:
7009bbec428eSgjelinek 			usage(B_FALSE, HELP_USAGE);
70107c478bd9Sstevel@tonic-gate 			exit(Z_USAGE);
70117c478bd9Sstevel@tonic-gate 		}
70127c478bd9Sstevel@tonic-gate 	}
70137c478bd9Sstevel@tonic-gate 
7014087719fdSdp 	if (optind > argc || strcmp(zone, "") == 0) {
7015bbec428eSgjelinek 		usage(B_FALSE, HELP_USAGE);
70167c478bd9Sstevel@tonic-gate 		exit(Z_USAGE);
70177c478bd9Sstevel@tonic-gate 	}
70187c478bd9Sstevel@tonic-gate 
7019087719fdSdp 	if ((err = zonecfg_access(zone, W_OK)) == Z_OK) {
7020bbec428eSgjelinek 		read_only_mode = B_FALSE;
7021087719fdSdp 	} else if (err == Z_ACCES) {
7022bbec428eSgjelinek 		read_only_mode = B_TRUE;
70237c478bd9Sstevel@tonic-gate 		/* skip this message in one-off from command line mode */
70247c478bd9Sstevel@tonic-gate 		if (optind == argc)
70257c478bd9Sstevel@tonic-gate 			(void) fprintf(stderr, gettext("WARNING: you do not "
70267c478bd9Sstevel@tonic-gate 			    "have write access to this zone's configuration "
70277c478bd9Sstevel@tonic-gate 			    "file;\ngoing into read-only mode.\n"));
7028087719fdSdp 	} else {
7029087719fdSdp 		fprintf(stderr, "%s: Could not access zone configuration "
7030087719fdSdp 		    "store: %s\n", execname, zonecfg_strerror(err));
7031087719fdSdp 		exit(Z_ERR);
70327c478bd9Sstevel@tonic-gate 	}
70337c478bd9Sstevel@tonic-gate 
70347c478bd9Sstevel@tonic-gate 	if ((handle = zonecfg_init_handle()) == NULL) {
7035bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
70367c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
70377c478bd9Sstevel@tonic-gate 	}
70387c478bd9Sstevel@tonic-gate 
70397c478bd9Sstevel@tonic-gate 	/*
70407c478bd9Sstevel@tonic-gate 	 * This may get set back to FALSE again in cmd_file() if cmd_file_name
70417c478bd9Sstevel@tonic-gate 	 * is a "real" file as opposed to "-" (i.e. meaning use stdin).
70427c478bd9Sstevel@tonic-gate 	 */
70437c478bd9Sstevel@tonic-gate 	if (isatty(STDIN_FILENO))
7044bbec428eSgjelinek 		ok_to_prompt = B_TRUE;
70457c478bd9Sstevel@tonic-gate 	if ((gl = new_GetLine(MAX_LINE_LEN, MAX_CMD_HIST)) == NULL)
70467c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
70477c478bd9Sstevel@tonic-gate 	if (gl_customize_completion(gl, NULL, cmd_cpl_fn) != 0)
70487c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
70497c478bd9Sstevel@tonic-gate 	(void) sigset(SIGINT, SIG_IGN);
70507c478bd9Sstevel@tonic-gate 	if (optind == argc) {
70517c478bd9Sstevel@tonic-gate 		if (!cmd_file_mode)
70527c478bd9Sstevel@tonic-gate 			err = do_interactive();
70537c478bd9Sstevel@tonic-gate 		else
70547c478bd9Sstevel@tonic-gate 			err = cmd_file(cmd_file_name);
70557c478bd9Sstevel@tonic-gate 	} else {
70567c478bd9Sstevel@tonic-gate 		err = one_command_at_a_time(argc - optind, &(argv[optind]));
70577c478bd9Sstevel@tonic-gate 	}
70587c478bd9Sstevel@tonic-gate 	zonecfg_fini_handle(handle);
70599acbbeafSnn35248 	if (brand != NULL)
70609acbbeafSnn35248 		brand_close(brand);
70617c478bd9Sstevel@tonic-gate 	(void) del_GetLine(gl);
70627c478bd9Sstevel@tonic-gate 	return (err);
70637c478bd9Sstevel@tonic-gate }
7064