xref: /titanic_54/usr/src/cmd/zonecfg/zonecfg.c (revision 12a8b243cfaa0a3c7097fde03e4b94e7c49f0492)
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>
79550b6e40SSowmini Varadhan #include <inet/ip.h>
807c478bd9Sstevel@tonic-gate 
817c478bd9Sstevel@tonic-gate #include <libzonecfg.h>
827c478bd9Sstevel@tonic-gate #include "zonecfg.h"
837c478bd9Sstevel@tonic-gate 
847c478bd9Sstevel@tonic-gate #if !defined(TEXT_DOMAIN)		/* should be defined by cc -D */
857c478bd9Sstevel@tonic-gate #define	TEXT_DOMAIN	"SYS_TEST"	/* Use this only if it wasn't */
867c478bd9Sstevel@tonic-gate #endif
877c478bd9Sstevel@tonic-gate 
887c478bd9Sstevel@tonic-gate #define	PAGER	"/usr/bin/more"
899acbbeafSnn35248 #define	EXEC_PREFIX	"exec "
909acbbeafSnn35248 #define	EXEC_LEN	(strlen(EXEC_PREFIX))
917c478bd9Sstevel@tonic-gate 
927c478bd9Sstevel@tonic-gate struct help {
937c478bd9Sstevel@tonic-gate 	uint_t	cmd_num;
947c478bd9Sstevel@tonic-gate 	char	*cmd_name;
957c478bd9Sstevel@tonic-gate 	uint_t	flags;
967c478bd9Sstevel@tonic-gate 	char	*short_usage;
977c478bd9Sstevel@tonic-gate };
987c478bd9Sstevel@tonic-gate 
997c478bd9Sstevel@tonic-gate extern int yyparse(void);
1007c478bd9Sstevel@tonic-gate extern int lex_lineno;
1017c478bd9Sstevel@tonic-gate 
1027c478bd9Sstevel@tonic-gate #define	MAX_LINE_LEN	1024
1037c478bd9Sstevel@tonic-gate #define	MAX_CMD_HIST	1024
1049acbbeafSnn35248 #define	MAX_CMD_LEN	1024
1057c478bd9Sstevel@tonic-gate 
1060209230bSgjelinek #define	ONE_MB		1048576
1070209230bSgjelinek 
1087c478bd9Sstevel@tonic-gate /*
1097c478bd9Sstevel@tonic-gate  * Each SHELP_ should be a simple string.
1107c478bd9Sstevel@tonic-gate  */
1117c478bd9Sstevel@tonic-gate 
1127c478bd9Sstevel@tonic-gate #define	SHELP_ADD	"add <resource-type>\n\t(global scope)\n" \
1137c478bd9Sstevel@tonic-gate 	"add <property-name> <property-value>\n\t(resource scope)"
1147c478bd9Sstevel@tonic-gate #define	SHELP_CANCEL	"cancel"
1150209230bSgjelinek #define	SHELP_CLEAR	"clear <property-name>"
1167c478bd9Sstevel@tonic-gate #define	SHELP_COMMIT	"commit"
117ee519a1fSgjelinek #define	SHELP_CREATE	"create [-F] [ -a <path> | -b | -t <template> ]"
1187c478bd9Sstevel@tonic-gate #define	SHELP_DELETE	"delete [-F]"
1197c478bd9Sstevel@tonic-gate #define	SHELP_END	"end"
1207c478bd9Sstevel@tonic-gate #define	SHELP_EXIT	"exit [-F]"
1217c478bd9Sstevel@tonic-gate #define	SHELP_EXPORT	"export [-f output-file]"
1227c478bd9Sstevel@tonic-gate #define	SHELP_HELP	"help [commands] [syntax] [usage] [<command-name>]"
1237c478bd9Sstevel@tonic-gate #define	SHELP_INFO	"info [<resource-type> [property-name=property-value]*]"
1240209230bSgjelinek #define	SHELP_REMOVE	"remove [-F] <resource-type> " \
1250209230bSgjelinek 	"[ <property-name>=<property-value> ]*\n" \
1260209230bSgjelinek 	"\t(global scope)\n" \
1270209230bSgjelinek 	"remove <property-name> <property-value>\n" \
1280209230bSgjelinek 	"\t(resource scope)"
1297c478bd9Sstevel@tonic-gate #define	SHELP_REVERT	"revert [-F]"
1307c478bd9Sstevel@tonic-gate #define	SHELP_SELECT	"select <resource-type> { <property-name>=" \
1317c478bd9Sstevel@tonic-gate 	"<property-value> }"
1327c478bd9Sstevel@tonic-gate #define	SHELP_SET	"set <property-name>=<property-value>"
1337c478bd9Sstevel@tonic-gate #define	SHELP_VERIFY	"verify"
1347c478bd9Sstevel@tonic-gate 
1357c478bd9Sstevel@tonic-gate static struct help helptab[] = {
1367c478bd9Sstevel@tonic-gate 	{ CMD_ADD,	"add",		HELP_RES_PROPS,	SHELP_ADD, },
1377c478bd9Sstevel@tonic-gate 	{ CMD_CANCEL,	"cancel",	0,		SHELP_CANCEL, },
1380209230bSgjelinek 	{ CMD_CLEAR,	"clear",	HELP_PROPS,	SHELP_CLEAR, },
1397c478bd9Sstevel@tonic-gate 	{ CMD_COMMIT,	"commit",	0,		SHELP_COMMIT, },
1407c478bd9Sstevel@tonic-gate 	{ CMD_CREATE,	"create",	0,		SHELP_CREATE, },
1417c478bd9Sstevel@tonic-gate 	{ CMD_DELETE,	"delete",	0,		SHELP_DELETE, },
1427c478bd9Sstevel@tonic-gate 	{ CMD_END,	"end",		0,		SHELP_END, },
1437c478bd9Sstevel@tonic-gate 	{ CMD_EXIT,	"exit",		0,		SHELP_EXIT, },
1447c478bd9Sstevel@tonic-gate 	{ CMD_EXPORT,	"export",	0,		SHELP_EXPORT, },
1457c478bd9Sstevel@tonic-gate 	{ CMD_HELP,	"help",		0,		SHELP_HELP },
1467c478bd9Sstevel@tonic-gate 	{ CMD_INFO,	"info",		HELP_RES_PROPS,	SHELP_INFO, },
1477c478bd9Sstevel@tonic-gate 	{ CMD_REMOVE,	"remove",	HELP_RES_PROPS,	SHELP_REMOVE, },
1487c478bd9Sstevel@tonic-gate 	{ CMD_REVERT,	"revert",	0,		SHELP_REVERT, },
1497c478bd9Sstevel@tonic-gate 	{ CMD_SELECT,	"select",	HELP_RES_PROPS,	SHELP_SELECT, },
1507c478bd9Sstevel@tonic-gate 	{ CMD_SET,	"set",		HELP_PROPS,	SHELP_SET, },
1517c478bd9Sstevel@tonic-gate 	{ CMD_VERIFY,	"verify",	0,		SHELP_VERIFY, },
1527c478bd9Sstevel@tonic-gate 	{ 0 },
1537c478bd9Sstevel@tonic-gate };
1547c478bd9Sstevel@tonic-gate 
1557c478bd9Sstevel@tonic-gate #define	MAX_RT_STRLEN	16
1567c478bd9Sstevel@tonic-gate 
1577c478bd9Sstevel@tonic-gate /* These *must* match the order of the RT_ define's from zonecfg.h */
158c94c1ef0Sjv227347 char *res_types[] = {
1597c478bd9Sstevel@tonic-gate 	"unknown",
160087719fdSdp 	"zonename",
1617c478bd9Sstevel@tonic-gate 	"zonepath",
1627c478bd9Sstevel@tonic-gate 	"autoboot",
1637c478bd9Sstevel@tonic-gate 	"pool",
1647c478bd9Sstevel@tonic-gate 	"fs",
1657c478bd9Sstevel@tonic-gate 	"net",
1667c478bd9Sstevel@tonic-gate 	"device",
1677c478bd9Sstevel@tonic-gate 	"rctl",
1687c478bd9Sstevel@tonic-gate 	"attr",
169fa9e4066Sahrens 	"dataset",
170ffbafc53Scomay 	"limitpriv",
1713f2f09c1Sdp 	"bootargs",
1729acbbeafSnn35248 	"brand",
1730209230bSgjelinek 	"dedicated-cpu",
1740209230bSgjelinek 	"capped-memory",
1750209230bSgjelinek 	ALIAS_MAXLWPS,
1760209230bSgjelinek 	ALIAS_MAXSHMMEM,
1770209230bSgjelinek 	ALIAS_MAXSHMIDS,
1780209230bSgjelinek 	ALIAS_MAXMSGIDS,
1790209230bSgjelinek 	ALIAS_MAXSEMIDS,
1800209230bSgjelinek 	ALIAS_SHARES,
1810209230bSgjelinek 	"scheduling-class",
182f4b3ec61Sdh155122 	"ip-type",
183c97ad5cdSakolb 	"capped-cpu",
1845679c89fSjv227347 	"hostid",
185cb8a054bSGlenn Faden 	"admin",
1860fbb751dSJohn Levon 	"fs-allowed",
187ff19e029SMenno Lageman 	ALIAS_MAXPROCS,
1887c478bd9Sstevel@tonic-gate 	NULL
1897c478bd9Sstevel@tonic-gate };
1907c478bd9Sstevel@tonic-gate 
1917c478bd9Sstevel@tonic-gate /* These *must* match the order of the PT_ define's from zonecfg.h */
192c94c1ef0Sjv227347 char *prop_types[] = {
1937c478bd9Sstevel@tonic-gate 	"unknown",
194087719fdSdp 	"zonename",
1957c478bd9Sstevel@tonic-gate 	"zonepath",
1967c478bd9Sstevel@tonic-gate 	"autoboot",
1977c478bd9Sstevel@tonic-gate 	"pool",
1987c478bd9Sstevel@tonic-gate 	"dir",
1997c478bd9Sstevel@tonic-gate 	"special",
2007c478bd9Sstevel@tonic-gate 	"type",
2017c478bd9Sstevel@tonic-gate 	"options",
2027c478bd9Sstevel@tonic-gate 	"address",
2037c478bd9Sstevel@tonic-gate 	"physical",
2047c478bd9Sstevel@tonic-gate 	"name",
2057c478bd9Sstevel@tonic-gate 	"value",
2067c478bd9Sstevel@tonic-gate 	"match",
2077c478bd9Sstevel@tonic-gate 	"priv",
2087c478bd9Sstevel@tonic-gate 	"limit",
2097c478bd9Sstevel@tonic-gate 	"action",
2107c478bd9Sstevel@tonic-gate 	"raw",
211ffbafc53Scomay 	"limitpriv",
2123f2f09c1Sdp 	"bootargs",
2139acbbeafSnn35248 	"brand",
2140209230bSgjelinek 	"ncpus",
2150209230bSgjelinek 	"importance",
2160209230bSgjelinek 	"swap",
2170209230bSgjelinek 	"locked",
2180209230bSgjelinek 	ALIAS_SHARES,
2190209230bSgjelinek 	ALIAS_MAXLWPS,
2200209230bSgjelinek 	ALIAS_MAXSHMMEM,
2210209230bSgjelinek 	ALIAS_MAXSHMIDS,
2220209230bSgjelinek 	ALIAS_MAXMSGIDS,
2230209230bSgjelinek 	ALIAS_MAXSEMIDS,
2240209230bSgjelinek 	ALIAS_MAXLOCKEDMEM,
2250209230bSgjelinek 	ALIAS_MAXSWAP,
2260209230bSgjelinek 	"scheduling-class",
227f4b3ec61Sdh155122 	"ip-type",
228de860bd9Sgfaden 	"defrouter",
2295679c89fSjv227347 	"hostid",
230cb8a054bSGlenn Faden 	"user",
231cb8a054bSGlenn Faden 	"auths",
2320fbb751dSJohn Levon 	"fs-allowed",
233ff19e029SMenno Lageman 	ALIAS_MAXPROCS,
234550b6e40SSowmini Varadhan 	"allowed-address",
2357c478bd9Sstevel@tonic-gate 	NULL
2367c478bd9Sstevel@tonic-gate };
2377c478bd9Sstevel@tonic-gate 
238ffbafc53Scomay /* These *must* match the order of the PROP_VAL_ define's from zonecfg.h */
2397c478bd9Sstevel@tonic-gate static char *prop_val_types[] = {
2407c478bd9Sstevel@tonic-gate 	"simple",
2417c478bd9Sstevel@tonic-gate 	"complex",
2427c478bd9Sstevel@tonic-gate 	"list",
2437c478bd9Sstevel@tonic-gate };
2447c478bd9Sstevel@tonic-gate 
2457c478bd9Sstevel@tonic-gate /*
2467c478bd9Sstevel@tonic-gate  * The various _cmds[] lists below are for command tab-completion.
2477c478bd9Sstevel@tonic-gate  */
2487c478bd9Sstevel@tonic-gate 
2497c478bd9Sstevel@tonic-gate /*
2507c478bd9Sstevel@tonic-gate  * remove has a space afterwards because it has qualifiers; the other commands
2510209230bSgjelinek  * that have qualifiers (add, select, etc.) don't need a space here because
2527c478bd9Sstevel@tonic-gate  * they have their own _cmds[] lists below.
2537c478bd9Sstevel@tonic-gate  */
2547c478bd9Sstevel@tonic-gate static const char *global_scope_cmds[] = {
2557c478bd9Sstevel@tonic-gate 	"add",
2560209230bSgjelinek 	"clear",
2577c478bd9Sstevel@tonic-gate 	"commit",
2587c478bd9Sstevel@tonic-gate 	"create",
2597c478bd9Sstevel@tonic-gate 	"delete",
2607c478bd9Sstevel@tonic-gate 	"exit",
2617c478bd9Sstevel@tonic-gate 	"export",
2627c478bd9Sstevel@tonic-gate 	"help",
2637c478bd9Sstevel@tonic-gate 	"info",
2647c478bd9Sstevel@tonic-gate 	"remove ",
2657c478bd9Sstevel@tonic-gate 	"revert",
2667c478bd9Sstevel@tonic-gate 	"select",
2677c478bd9Sstevel@tonic-gate 	"set",
2687c478bd9Sstevel@tonic-gate 	"verify",
2697c478bd9Sstevel@tonic-gate 	NULL
2707c478bd9Sstevel@tonic-gate };
2717c478bd9Sstevel@tonic-gate 
2727c478bd9Sstevel@tonic-gate static const char *add_cmds[] = {
2737c478bd9Sstevel@tonic-gate 	"add fs",
2747c478bd9Sstevel@tonic-gate 	"add net",
2757c478bd9Sstevel@tonic-gate 	"add device",
2767c478bd9Sstevel@tonic-gate 	"add rctl",
2777c478bd9Sstevel@tonic-gate 	"add attr",
278fa9e4066Sahrens 	"add dataset",
2790209230bSgjelinek 	"add dedicated-cpu",
280c97ad5cdSakolb 	"add capped-cpu",
2810209230bSgjelinek 	"add capped-memory",
282cb8a054bSGlenn Faden 	"add admin",
2830209230bSgjelinek 	NULL
2840209230bSgjelinek };
2850209230bSgjelinek 
2860209230bSgjelinek static const char *clear_cmds[] = {
2870209230bSgjelinek 	"clear autoboot",
2880209230bSgjelinek 	"clear pool",
2890209230bSgjelinek 	"clear limitpriv",
2900209230bSgjelinek 	"clear bootargs",
2910209230bSgjelinek 	"clear scheduling-class",
292f4b3ec61Sdh155122 	"clear ip-type",
2930209230bSgjelinek 	"clear " ALIAS_MAXLWPS,
2940209230bSgjelinek 	"clear " ALIAS_MAXSHMMEM,
2950209230bSgjelinek 	"clear " ALIAS_MAXSHMIDS,
2960209230bSgjelinek 	"clear " ALIAS_MAXMSGIDS,
2970209230bSgjelinek 	"clear " ALIAS_MAXSEMIDS,
2980209230bSgjelinek 	"clear " ALIAS_SHARES,
299ff19e029SMenno Lageman 	"clear " ALIAS_MAXPROCS,
3007c478bd9Sstevel@tonic-gate 	NULL
3017c478bd9Sstevel@tonic-gate };
3027c478bd9Sstevel@tonic-gate 
3039e7542f4Sdp static const char *remove_cmds[] = {
3049e7542f4Sdp 	"remove fs ",
3059e7542f4Sdp 	"remove net ",
3069e7542f4Sdp 	"remove device ",
3079e7542f4Sdp 	"remove rctl ",
3089e7542f4Sdp 	"remove attr ",
3099e7542f4Sdp 	"remove dataset ",
3100209230bSgjelinek 	"remove dedicated-cpu ",
311c97ad5cdSakolb 	"remove capped-cpu ",
3120209230bSgjelinek 	"remove capped-memory ",
313cb8a054bSGlenn Faden 	"remove admin ",
3149e7542f4Sdp 	NULL
3159e7542f4Sdp };
3169e7542f4Sdp 
3177c478bd9Sstevel@tonic-gate static const char *select_cmds[] = {
3187c478bd9Sstevel@tonic-gate 	"select fs ",
3197c478bd9Sstevel@tonic-gate 	"select net ",
3207c478bd9Sstevel@tonic-gate 	"select device ",
3217c478bd9Sstevel@tonic-gate 	"select rctl ",
3227c478bd9Sstevel@tonic-gate 	"select attr ",
323fa9e4066Sahrens 	"select dataset ",
3240209230bSgjelinek 	"select dedicated-cpu",
325c97ad5cdSakolb 	"select capped-cpu",
3260209230bSgjelinek 	"select capped-memory",
327cb8a054bSGlenn Faden 	"select admin",
3287c478bd9Sstevel@tonic-gate 	NULL
3297c478bd9Sstevel@tonic-gate };
3307c478bd9Sstevel@tonic-gate 
3317c478bd9Sstevel@tonic-gate static const char *set_cmds[] = {
332087719fdSdp 	"set zonename=",
333087719fdSdp 	"set zonepath=",
3349acbbeafSnn35248 	"set brand=",
335087719fdSdp 	"set autoboot=",
336087719fdSdp 	"set pool=",
337ffbafc53Scomay 	"set limitpriv=",
3383f2f09c1Sdp 	"set bootargs=",
3390209230bSgjelinek 	"set scheduling-class=",
340f4b3ec61Sdh155122 	"set ip-type=",
3410209230bSgjelinek 	"set " ALIAS_MAXLWPS "=",
3420209230bSgjelinek 	"set " ALIAS_MAXSHMMEM "=",
3430209230bSgjelinek 	"set " ALIAS_MAXSHMIDS "=",
3440209230bSgjelinek 	"set " ALIAS_MAXMSGIDS "=",
3450209230bSgjelinek 	"set " ALIAS_MAXSEMIDS "=",
3460209230bSgjelinek 	"set " ALIAS_SHARES "=",
3475679c89fSjv227347 	"set hostid=",
3480fbb751dSJohn Levon 	"set fs-allowed=",
349ff19e029SMenno Lageman 	"set " ALIAS_MAXPROCS "=",
3507c478bd9Sstevel@tonic-gate 	NULL
3517c478bd9Sstevel@tonic-gate };
3527c478bd9Sstevel@tonic-gate 
3539e7542f4Sdp static const char *info_cmds[] = {
3549e7542f4Sdp 	"info fs ",
3559e7542f4Sdp 	"info net ",
3569e7542f4Sdp 	"info device ",
3579e7542f4Sdp 	"info rctl ",
3589e7542f4Sdp 	"info attr ",
3599e7542f4Sdp 	"info dataset ",
3600209230bSgjelinek 	"info capped-memory",
3610209230bSgjelinek 	"info dedicated-cpu",
362c97ad5cdSakolb 	"info capped-cpu",
3639e7542f4Sdp 	"info zonename",
3649e7542f4Sdp 	"info zonepath",
3659e7542f4Sdp 	"info autoboot",
3669e7542f4Sdp 	"info pool",
3679e7542f4Sdp 	"info limitpriv",
3689e7542f4Sdp 	"info bootargs",
3690209230bSgjelinek 	"info brand",
3700209230bSgjelinek 	"info scheduling-class",
371f4b3ec61Sdh155122 	"info ip-type",
3720209230bSgjelinek 	"info max-lwps",
3730209230bSgjelinek 	"info max-shm-memory",
3740209230bSgjelinek 	"info max-shm-ids",
3750209230bSgjelinek 	"info max-msg-ids",
3760209230bSgjelinek 	"info max-sem-ids",
3770209230bSgjelinek 	"info cpu-shares",
3785679c89fSjv227347 	"info hostid",
379cb8a054bSGlenn Faden 	"info admin",
3800fbb751dSJohn Levon 	"info fs-allowed",
381ff19e029SMenno Lageman 	"info max-processes",
3829e7542f4Sdp 	NULL
3839e7542f4Sdp };
3849e7542f4Sdp 
3857c478bd9Sstevel@tonic-gate static const char *fs_res_scope_cmds[] = {
3867c478bd9Sstevel@tonic-gate 	"add options ",
3877c478bd9Sstevel@tonic-gate 	"cancel",
3887c478bd9Sstevel@tonic-gate 	"end",
3897c478bd9Sstevel@tonic-gate 	"exit",
3907c478bd9Sstevel@tonic-gate 	"help",
3917c478bd9Sstevel@tonic-gate 	"info",
392ffbafc53Scomay 	"remove options ",
3937c478bd9Sstevel@tonic-gate 	"set dir=",
3947c478bd9Sstevel@tonic-gate 	"set raw=",
3957c478bd9Sstevel@tonic-gate 	"set special=",
3967c478bd9Sstevel@tonic-gate 	"set type=",
3970209230bSgjelinek 	"clear raw",
3987c478bd9Sstevel@tonic-gate 	NULL
3997c478bd9Sstevel@tonic-gate };
4007c478bd9Sstevel@tonic-gate 
4017c478bd9Sstevel@tonic-gate static const char *net_res_scope_cmds[] = {
4027c478bd9Sstevel@tonic-gate 	"cancel",
4037c478bd9Sstevel@tonic-gate 	"end",
4047c478bd9Sstevel@tonic-gate 	"exit",
4057c478bd9Sstevel@tonic-gate 	"help",
4067c478bd9Sstevel@tonic-gate 	"info",
4077c478bd9Sstevel@tonic-gate 	"set address=",
4087c478bd9Sstevel@tonic-gate 	"set physical=",
4091b3281c0SGerald Jelinek 	"set defrouter=",
4107c478bd9Sstevel@tonic-gate 	NULL
4117c478bd9Sstevel@tonic-gate };
4127c478bd9Sstevel@tonic-gate 
4137c478bd9Sstevel@tonic-gate static const char *device_res_scope_cmds[] = {
4147c478bd9Sstevel@tonic-gate 	"cancel",
4157c478bd9Sstevel@tonic-gate 	"end",
4167c478bd9Sstevel@tonic-gate 	"exit",
4177c478bd9Sstevel@tonic-gate 	"help",
4187c478bd9Sstevel@tonic-gate 	"info",
4197c478bd9Sstevel@tonic-gate 	"set match=",
4207c478bd9Sstevel@tonic-gate 	NULL
4217c478bd9Sstevel@tonic-gate };
4227c478bd9Sstevel@tonic-gate 
4237c478bd9Sstevel@tonic-gate static const char *attr_res_scope_cmds[] = {
4247c478bd9Sstevel@tonic-gate 	"cancel",
4257c478bd9Sstevel@tonic-gate 	"end",
4267c478bd9Sstevel@tonic-gate 	"exit",
4277c478bd9Sstevel@tonic-gate 	"help",
4287c478bd9Sstevel@tonic-gate 	"info",
4297c478bd9Sstevel@tonic-gate 	"set name=",
4307c478bd9Sstevel@tonic-gate 	"set type=",
4317c478bd9Sstevel@tonic-gate 	"set value=",
4327c478bd9Sstevel@tonic-gate 	NULL
4337c478bd9Sstevel@tonic-gate };
4347c478bd9Sstevel@tonic-gate 
4357c478bd9Sstevel@tonic-gate static const char *rctl_res_scope_cmds[] = {
4367c478bd9Sstevel@tonic-gate 	"add value ",
4377c478bd9Sstevel@tonic-gate 	"cancel",
4387c478bd9Sstevel@tonic-gate 	"end",
4397c478bd9Sstevel@tonic-gate 	"exit",
4407c478bd9Sstevel@tonic-gate 	"help",
4417c478bd9Sstevel@tonic-gate 	"info",
442ffbafc53Scomay 	"remove value ",
4437c478bd9Sstevel@tonic-gate 	"set name=",
4447c478bd9Sstevel@tonic-gate 	NULL
4457c478bd9Sstevel@tonic-gate };
4467c478bd9Sstevel@tonic-gate 
447fa9e4066Sahrens static const char *dataset_res_scope_cmds[] = {
448fa9e4066Sahrens 	"cancel",
449fa9e4066Sahrens 	"end",
450fa9e4066Sahrens 	"exit",
451fa9e4066Sahrens 	"help",
452fa9e4066Sahrens 	"info",
453fa9e4066Sahrens 	"set name=",
454fa9e4066Sahrens 	NULL
455fa9e4066Sahrens };
456fa9e4066Sahrens 
4570209230bSgjelinek static const char *pset_res_scope_cmds[] = {
4580209230bSgjelinek 	"cancel",
4590209230bSgjelinek 	"end",
4600209230bSgjelinek 	"exit",
4610209230bSgjelinek 	"help",
4620209230bSgjelinek 	"info",
4630209230bSgjelinek 	"set ncpus=",
4640209230bSgjelinek 	"set importance=",
4650209230bSgjelinek 	"clear importance",
4660209230bSgjelinek 	NULL
4670209230bSgjelinek };
4680209230bSgjelinek 
469c97ad5cdSakolb static const char *pcap_res_scope_cmds[] = {
470c97ad5cdSakolb 	"cancel",
471c97ad5cdSakolb 	"end",
472c97ad5cdSakolb 	"exit",
473c97ad5cdSakolb 	"help",
474c97ad5cdSakolb 	"info",
475c97ad5cdSakolb 	"set ncpus=",
476c97ad5cdSakolb 	NULL
477c97ad5cdSakolb };
478c97ad5cdSakolb 
4790209230bSgjelinek static const char *mcap_res_scope_cmds[] = {
4800209230bSgjelinek 	"cancel",
4810209230bSgjelinek 	"end",
4820209230bSgjelinek 	"exit",
4830209230bSgjelinek 	"help",
4840209230bSgjelinek 	"info",
4850209230bSgjelinek 	"set physical=",
4860209230bSgjelinek 	"set swap=",
4870209230bSgjelinek 	"set locked=",
4880209230bSgjelinek 	"clear physical",
4890209230bSgjelinek 	"clear swap",
4900209230bSgjelinek 	"clear locked",
4910209230bSgjelinek 	NULL
4920209230bSgjelinek };
4930209230bSgjelinek 
494cb8a054bSGlenn Faden static const char *admin_res_scope_cmds[] = {
495cb8a054bSGlenn Faden 	"cancel",
496cb8a054bSGlenn Faden 	"end",
497cb8a054bSGlenn Faden 	"exit",
498cb8a054bSGlenn Faden 	"help",
499cb8a054bSGlenn Faden 	"info",
500cb8a054bSGlenn Faden 	"set user=",
501cb8a054bSGlenn Faden 	"set auths=",
502cb8a054bSGlenn Faden 	NULL
503cb8a054bSGlenn Faden };
504cb8a054bSGlenn Faden 
505550b6e40SSowmini Varadhan struct xif {
506550b6e40SSowmini Varadhan 	struct xif	*xif_next;
507550b6e40SSowmini Varadhan 	char		xif_name[LIFNAMSIZ];
508550b6e40SSowmini Varadhan 	boolean_t	xif_has_address;
509550b6e40SSowmini Varadhan 	boolean_t	xif_has_defrouter;
510550b6e40SSowmini Varadhan };
511550b6e40SSowmini Varadhan 
5127c478bd9Sstevel@tonic-gate /* Global variables */
5137c478bd9Sstevel@tonic-gate 
514550b6e40SSowmini Varadhan /* list of network interfaces specified for exclusive IP zone */
515550b6e40SSowmini Varadhan struct xif *xif;
516550b6e40SSowmini Varadhan 
5177c478bd9Sstevel@tonic-gate /* set early in main(), never modified thereafter, used all over the place */
5187c478bd9Sstevel@tonic-gate static char *execname;
5197c478bd9Sstevel@tonic-gate 
5207c478bd9Sstevel@tonic-gate /* set in main(), used all over the place */
5217c478bd9Sstevel@tonic-gate static zone_dochandle_t handle;
5227c478bd9Sstevel@tonic-gate 
5237c478bd9Sstevel@tonic-gate /* used all over the place */
524087719fdSdp static char zone[ZONENAME_MAX];
525087719fdSdp static char revert_zone[ZONENAME_MAX];
5267c478bd9Sstevel@tonic-gate 
5279acbbeafSnn35248 /* global brand operations */
528123807fbSedp static brand_handle_t brand;
5299acbbeafSnn35248 
5307c478bd9Sstevel@tonic-gate /* set in modifying functions, checked in read_input() */
531bbec428eSgjelinek static boolean_t need_to_commit = B_FALSE;
532bbec428eSgjelinek boolean_t saw_error;
5337c478bd9Sstevel@tonic-gate 
5347c478bd9Sstevel@tonic-gate /* set in yacc parser, checked in read_input() */
535bbec428eSgjelinek boolean_t newline_terminated;
5367c478bd9Sstevel@tonic-gate 
5377c478bd9Sstevel@tonic-gate /* set in main(), checked in lex error handler */
538bbec428eSgjelinek boolean_t cmd_file_mode;
5397c478bd9Sstevel@tonic-gate 
5407c478bd9Sstevel@tonic-gate /* set in exit_func(), checked in read_input() */
541bbec428eSgjelinek static boolean_t time_to_exit = B_FALSE, force_exit = B_FALSE;
5427c478bd9Sstevel@tonic-gate 
5437c478bd9Sstevel@tonic-gate /* used in short_usage() and zerr() */
5447c478bd9Sstevel@tonic-gate static char *cmd_file_name = NULL;
5457c478bd9Sstevel@tonic-gate 
5467c478bd9Sstevel@tonic-gate /* checked in read_input() and other places */
547bbec428eSgjelinek static boolean_t ok_to_prompt = B_FALSE;
5487c478bd9Sstevel@tonic-gate 
5497c478bd9Sstevel@tonic-gate /* set and checked in initialize() */
550bbec428eSgjelinek static boolean_t got_handle = B_FALSE;
5517c478bd9Sstevel@tonic-gate 
5527c478bd9Sstevel@tonic-gate /* initialized in do_interactive(), checked in initialize() */
553bbec428eSgjelinek static boolean_t interactive_mode;
5547c478bd9Sstevel@tonic-gate 
5550209230bSgjelinek /* set if configuring the global zone */
556bbec428eSgjelinek static boolean_t global_zone = B_FALSE;
5570209230bSgjelinek 
5587c478bd9Sstevel@tonic-gate /* set in main(), checked in multiple places */
559bbec428eSgjelinek static boolean_t read_only_mode;
5607c478bd9Sstevel@tonic-gate 
561bbec428eSgjelinek /* scope is outer/global or inner/resource */
562bbec428eSgjelinek static boolean_t global_scope = B_TRUE;
5637c478bd9Sstevel@tonic-gate static int resource_scope;	/* should be in the RT_ list from zonecfg.h */
5647c478bd9Sstevel@tonic-gate static int end_op = -1;		/* operation on end is either add or modify */
5657c478bd9Sstevel@tonic-gate 
5667c478bd9Sstevel@tonic-gate int num_prop_vals;		/* for grammar */
5677c478bd9Sstevel@tonic-gate 
5687c478bd9Sstevel@tonic-gate /*
5697c478bd9Sstevel@tonic-gate  * These are for keeping track of resources as they are specified as part of
5707c478bd9Sstevel@tonic-gate  * the multi-step process.  They should be initialized by add_resource() or
5717c478bd9Sstevel@tonic-gate  * select_func() and filled in by add_property() or set_func().
5727c478bd9Sstevel@tonic-gate  */
5737c478bd9Sstevel@tonic-gate static struct zone_fstab	old_fstab, in_progress_fstab;
5747c478bd9Sstevel@tonic-gate static struct zone_nwiftab	old_nwiftab, in_progress_nwiftab;
5757c478bd9Sstevel@tonic-gate static struct zone_devtab	old_devtab, in_progress_devtab;
5767c478bd9Sstevel@tonic-gate static struct zone_rctltab	old_rctltab, in_progress_rctltab;
5777c478bd9Sstevel@tonic-gate static struct zone_attrtab	old_attrtab, in_progress_attrtab;
578fa9e4066Sahrens static struct zone_dstab	old_dstab, in_progress_dstab;
5790209230bSgjelinek static struct zone_psettab	old_psettab, in_progress_psettab;
5800209230bSgjelinek static struct zone_mcaptab	old_mcaptab, in_progress_mcaptab;
581cb8a054bSGlenn Faden static struct zone_admintab	old_admintab, in_progress_admintab;
5827c478bd9Sstevel@tonic-gate 
5837c478bd9Sstevel@tonic-gate static GetLine *gl;	/* The gl_get_line() resource object */
5847c478bd9Sstevel@tonic-gate 
5850209230bSgjelinek static void bytes_to_units(char *str, char *buf, int bufsize);
5860209230bSgjelinek 
5877c478bd9Sstevel@tonic-gate /* Functions begin here */
5887c478bd9Sstevel@tonic-gate 
589bbec428eSgjelinek static boolean_t
5907c478bd9Sstevel@tonic-gate initial_match(const char *line1, const char *line2, int word_end)
5917c478bd9Sstevel@tonic-gate {
5927c478bd9Sstevel@tonic-gate 	if (word_end <= 0)
593bbec428eSgjelinek 		return (B_TRUE);
5947c478bd9Sstevel@tonic-gate 	return (strncmp(line1, line2, word_end) == 0);
5957c478bd9Sstevel@tonic-gate }
5967c478bd9Sstevel@tonic-gate 
5977c478bd9Sstevel@tonic-gate static int
5987c478bd9Sstevel@tonic-gate add_stuff(WordCompletion *cpl, const char *line1, const char **list,
5997c478bd9Sstevel@tonic-gate     int word_end)
6007c478bd9Sstevel@tonic-gate {
6017c478bd9Sstevel@tonic-gate 	int i, err;
6027c478bd9Sstevel@tonic-gate 
6037c478bd9Sstevel@tonic-gate 	for (i = 0; list[i] != NULL; i++) {
6047c478bd9Sstevel@tonic-gate 		if (initial_match(line1, list[i], word_end)) {
6057c478bd9Sstevel@tonic-gate 			err = cpl_add_completion(cpl, line1, 0, word_end,
6067c478bd9Sstevel@tonic-gate 			    list[i] + word_end, "", "");
6077c478bd9Sstevel@tonic-gate 			if (err != 0)
6087c478bd9Sstevel@tonic-gate 				return (err);
6097c478bd9Sstevel@tonic-gate 		}
6107c478bd9Sstevel@tonic-gate 	}
6117c478bd9Sstevel@tonic-gate 	return (0);
6127c478bd9Sstevel@tonic-gate }
6137c478bd9Sstevel@tonic-gate 
6147c478bd9Sstevel@tonic-gate static
6157c478bd9Sstevel@tonic-gate /* ARGSUSED */
6167c478bd9Sstevel@tonic-gate CPL_MATCH_FN(cmd_cpl_fn)
6177c478bd9Sstevel@tonic-gate {
6187c478bd9Sstevel@tonic-gate 	if (global_scope) {
6197c478bd9Sstevel@tonic-gate 		/*
6207c478bd9Sstevel@tonic-gate 		 * The MAX/MIN tests below are to make sure we have at least
6217c478bd9Sstevel@tonic-gate 		 * enough characters to distinguish from other prefixes (MAX)
6227c478bd9Sstevel@tonic-gate 		 * but only check MIN(what we have, what we're checking).
6237c478bd9Sstevel@tonic-gate 		 */
6247c478bd9Sstevel@tonic-gate 		if (strncmp(line, "add ", MAX(MIN(word_end, 4), 1)) == 0)
6257c478bd9Sstevel@tonic-gate 			return (add_stuff(cpl, line, add_cmds, word_end));
6260209230bSgjelinek 		if (strncmp(line, "clear ", MAX(MIN(word_end, 6), 2)) == 0)
6270209230bSgjelinek 			return (add_stuff(cpl, line, clear_cmds, word_end));
6287c478bd9Sstevel@tonic-gate 		if (strncmp(line, "select ", MAX(MIN(word_end, 7), 3)) == 0)
6297c478bd9Sstevel@tonic-gate 			return (add_stuff(cpl, line, select_cmds, word_end));
6307c478bd9Sstevel@tonic-gate 		if (strncmp(line, "set ", MAX(MIN(word_end, 4), 3)) == 0)
6317c478bd9Sstevel@tonic-gate 			return (add_stuff(cpl, line, set_cmds, word_end));
6329e7542f4Sdp 		if (strncmp(line, "remove ", MAX(MIN(word_end, 7), 1)) == 0)
6339e7542f4Sdp 			return (add_stuff(cpl, line, remove_cmds, word_end));
6349e7542f4Sdp 		if (strncmp(line, "info ", MAX(MIN(word_end, 5), 1)) == 0)
6359e7542f4Sdp 			return (add_stuff(cpl, line, info_cmds, word_end));
6367c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, global_scope_cmds, word_end));
6377c478bd9Sstevel@tonic-gate 	}
6387c478bd9Sstevel@tonic-gate 	switch (resource_scope) {
6397c478bd9Sstevel@tonic-gate 	case RT_FS:
6407c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, fs_res_scope_cmds, word_end));
6417c478bd9Sstevel@tonic-gate 	case RT_NET:
6427c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, net_res_scope_cmds, word_end));
6437c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
6447c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, device_res_scope_cmds, word_end));
6457c478bd9Sstevel@tonic-gate 	case RT_RCTL:
6467c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, rctl_res_scope_cmds, word_end));
6477c478bd9Sstevel@tonic-gate 	case RT_ATTR:
6487c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, attr_res_scope_cmds, word_end));
649fa9e4066Sahrens 	case RT_DATASET:
650fa9e4066Sahrens 		return (add_stuff(cpl, line, dataset_res_scope_cmds, word_end));
6510209230bSgjelinek 	case RT_DCPU:
6520209230bSgjelinek 		return (add_stuff(cpl, line, pset_res_scope_cmds, word_end));
653c97ad5cdSakolb 	case RT_PCAP:
654c97ad5cdSakolb 		return (add_stuff(cpl, line, pcap_res_scope_cmds, word_end));
6550209230bSgjelinek 	case RT_MCAP:
6560209230bSgjelinek 		return (add_stuff(cpl, line, mcap_res_scope_cmds, word_end));
657cb8a054bSGlenn Faden 	case RT_ADMIN:
658cb8a054bSGlenn Faden 		return (add_stuff(cpl, line, admin_res_scope_cmds, word_end));
6597c478bd9Sstevel@tonic-gate 	}
6607c478bd9Sstevel@tonic-gate 	return (0);
6617c478bd9Sstevel@tonic-gate }
6627c478bd9Sstevel@tonic-gate 
6637c478bd9Sstevel@tonic-gate /*
6647c478bd9Sstevel@tonic-gate  * For the main CMD_func() functions below, several of them call getopt()
6657c478bd9Sstevel@tonic-gate  * then check optind against argc to make sure an extra parameter was not
6667c478bd9Sstevel@tonic-gate  * passed in.  The reason this is not caught in the grammar is that the
6677c478bd9Sstevel@tonic-gate  * grammar just checks for a miscellaneous TOKEN, which is *expected* to
6687c478bd9Sstevel@tonic-gate  * be "-F" (for example), but could be anything.  So (for example) this
6697c478bd9Sstevel@tonic-gate  * check will prevent "create bogus".
6707c478bd9Sstevel@tonic-gate  */
6717c478bd9Sstevel@tonic-gate 
6727c478bd9Sstevel@tonic-gate cmd_t *
6737c478bd9Sstevel@tonic-gate alloc_cmd(void)
6747c478bd9Sstevel@tonic-gate {
6757c478bd9Sstevel@tonic-gate 	return (calloc(1, sizeof (cmd_t)));
6767c478bd9Sstevel@tonic-gate }
6777c478bd9Sstevel@tonic-gate 
6787c478bd9Sstevel@tonic-gate void
6797c478bd9Sstevel@tonic-gate free_cmd(cmd_t *cmd)
6807c478bd9Sstevel@tonic-gate {
6817c478bd9Sstevel@tonic-gate 	int i;
6827c478bd9Sstevel@tonic-gate 
6837c478bd9Sstevel@tonic-gate 	for (i = 0; i < MAX_EQ_PROP_PAIRS; i++)
6847c478bd9Sstevel@tonic-gate 		if (cmd->cmd_property_ptr[i] != NULL) {
6857c478bd9Sstevel@tonic-gate 			property_value_ptr_t pp = cmd->cmd_property_ptr[i];
6867c478bd9Sstevel@tonic-gate 
6877c478bd9Sstevel@tonic-gate 			switch (pp->pv_type) {
6887c478bd9Sstevel@tonic-gate 			case PROP_VAL_SIMPLE:
6897c478bd9Sstevel@tonic-gate 				free(pp->pv_simple);
6907c478bd9Sstevel@tonic-gate 				break;
6917c478bd9Sstevel@tonic-gate 			case PROP_VAL_COMPLEX:
6927c478bd9Sstevel@tonic-gate 				free_complex(pp->pv_complex);
6937c478bd9Sstevel@tonic-gate 				break;
6947c478bd9Sstevel@tonic-gate 			case PROP_VAL_LIST:
6957c478bd9Sstevel@tonic-gate 				free_list(pp->pv_list);
6967c478bd9Sstevel@tonic-gate 				break;
6977c478bd9Sstevel@tonic-gate 			}
6987c478bd9Sstevel@tonic-gate 		}
6997c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_argc; i++)
7007c478bd9Sstevel@tonic-gate 		free(cmd->cmd_argv[i]);
7017c478bd9Sstevel@tonic-gate 	free(cmd);
7027c478bd9Sstevel@tonic-gate }
7037c478bd9Sstevel@tonic-gate 
7047c478bd9Sstevel@tonic-gate complex_property_ptr_t
7057c478bd9Sstevel@tonic-gate alloc_complex(void)
7067c478bd9Sstevel@tonic-gate {
7077c478bd9Sstevel@tonic-gate 	return (calloc(1, sizeof (complex_property_t)));
7087c478bd9Sstevel@tonic-gate }
7097c478bd9Sstevel@tonic-gate 
7107c478bd9Sstevel@tonic-gate void
7117c478bd9Sstevel@tonic-gate free_complex(complex_property_ptr_t complex)
7127c478bd9Sstevel@tonic-gate {
7137c478bd9Sstevel@tonic-gate 	if (complex == NULL)
7147c478bd9Sstevel@tonic-gate 		return;
7157c478bd9Sstevel@tonic-gate 	free_complex(complex->cp_next);
7167c478bd9Sstevel@tonic-gate 	if (complex->cp_value != NULL)
7177c478bd9Sstevel@tonic-gate 		free(complex->cp_value);
7187c478bd9Sstevel@tonic-gate 	free(complex);
7197c478bd9Sstevel@tonic-gate }
7207c478bd9Sstevel@tonic-gate 
7217c478bd9Sstevel@tonic-gate list_property_ptr_t
7227c478bd9Sstevel@tonic-gate alloc_list(void)
7237c478bd9Sstevel@tonic-gate {
7247c478bd9Sstevel@tonic-gate 	return (calloc(1, sizeof (list_property_t)));
7257c478bd9Sstevel@tonic-gate }
7267c478bd9Sstevel@tonic-gate 
7277c478bd9Sstevel@tonic-gate void
7287c478bd9Sstevel@tonic-gate free_list(list_property_ptr_t list)
7297c478bd9Sstevel@tonic-gate {
7307c478bd9Sstevel@tonic-gate 	if (list == NULL)
7317c478bd9Sstevel@tonic-gate 		return;
7327c478bd9Sstevel@tonic-gate 	if (list->lp_simple != NULL)
7337c478bd9Sstevel@tonic-gate 		free(list->lp_simple);
7347c478bd9Sstevel@tonic-gate 	free_complex(list->lp_complex);
7357c478bd9Sstevel@tonic-gate 	free_list(list->lp_next);
7367c478bd9Sstevel@tonic-gate 	free(list);
7377c478bd9Sstevel@tonic-gate }
7387c478bd9Sstevel@tonic-gate 
7397c478bd9Sstevel@tonic-gate void
7407c478bd9Sstevel@tonic-gate free_outer_list(list_property_ptr_t list)
7417c478bd9Sstevel@tonic-gate {
7427c478bd9Sstevel@tonic-gate 	if (list == NULL)
7437c478bd9Sstevel@tonic-gate 		return;
7447c478bd9Sstevel@tonic-gate 	free_outer_list(list->lp_next);
7457c478bd9Sstevel@tonic-gate 	free(list);
7467c478bd9Sstevel@tonic-gate }
7477c478bd9Sstevel@tonic-gate 
7487c478bd9Sstevel@tonic-gate static struct zone_rctlvaltab *
7497c478bd9Sstevel@tonic-gate alloc_rctlvaltab(void)
7507c478bd9Sstevel@tonic-gate {
7517c478bd9Sstevel@tonic-gate 	return (calloc(1, sizeof (struct zone_rctlvaltab)));
7527c478bd9Sstevel@tonic-gate }
7537c478bd9Sstevel@tonic-gate 
7547c478bd9Sstevel@tonic-gate static char *
7557c478bd9Sstevel@tonic-gate rt_to_str(int res_type)
7567c478bd9Sstevel@tonic-gate {
7577c478bd9Sstevel@tonic-gate 	assert(res_type >= RT_MIN && res_type <= RT_MAX);
7587c478bd9Sstevel@tonic-gate 	return (res_types[res_type]);
7597c478bd9Sstevel@tonic-gate }
7607c478bd9Sstevel@tonic-gate 
7617c478bd9Sstevel@tonic-gate static char *
7627c478bd9Sstevel@tonic-gate pt_to_str(int prop_type)
7637c478bd9Sstevel@tonic-gate {
7647c478bd9Sstevel@tonic-gate 	assert(prop_type >= PT_MIN && prop_type <= PT_MAX);
7657c478bd9Sstevel@tonic-gate 	return (prop_types[prop_type]);
7667c478bd9Sstevel@tonic-gate }
7677c478bd9Sstevel@tonic-gate 
7687c478bd9Sstevel@tonic-gate static char *
7697c478bd9Sstevel@tonic-gate pvt_to_str(int pv_type)
7707c478bd9Sstevel@tonic-gate {
7717c478bd9Sstevel@tonic-gate 	assert(pv_type >= PROP_VAL_MIN && pv_type <= PROP_VAL_MAX);
7727c478bd9Sstevel@tonic-gate 	return (prop_val_types[pv_type]);
7737c478bd9Sstevel@tonic-gate }
7747c478bd9Sstevel@tonic-gate 
7757c478bd9Sstevel@tonic-gate static char *
7767c478bd9Sstevel@tonic-gate cmd_to_str(int cmd_num)
7777c478bd9Sstevel@tonic-gate {
7787c478bd9Sstevel@tonic-gate 	assert(cmd_num >= CMD_MIN && cmd_num <= CMD_MAX);
7797c478bd9Sstevel@tonic-gate 	return (helptab[cmd_num].cmd_name);
7807c478bd9Sstevel@tonic-gate }
7817c478bd9Sstevel@tonic-gate 
7823042b8b5Sbatschul /* PRINTFLIKE1 */
7833042b8b5Sbatschul static void
7843042b8b5Sbatschul zerr(const char *fmt, ...)
7853042b8b5Sbatschul {
7863042b8b5Sbatschul 	va_list alist;
7873042b8b5Sbatschul 	static int last_lineno;
7883042b8b5Sbatschul 
7893042b8b5Sbatschul 	/* lex_lineno has already been incremented in the lexer; compensate */
7903042b8b5Sbatschul 	if (cmd_file_mode && lex_lineno > last_lineno) {
7913042b8b5Sbatschul 		if (strcmp(cmd_file_name, "-") == 0)
7923042b8b5Sbatschul 			(void) fprintf(stderr, gettext("On line %d:\n"),
7933042b8b5Sbatschul 			    lex_lineno - 1);
7943042b8b5Sbatschul 		else
7953042b8b5Sbatschul 			(void) fprintf(stderr, gettext("On line %d of %s:\n"),
7963042b8b5Sbatschul 			    lex_lineno - 1, cmd_file_name);
7973042b8b5Sbatschul 		last_lineno = lex_lineno;
7983042b8b5Sbatschul 	}
7993042b8b5Sbatschul 	va_start(alist, fmt);
8003042b8b5Sbatschul 	(void) vfprintf(stderr, fmt, alist);
8013042b8b5Sbatschul 	(void) fprintf(stderr, "\n");
8023042b8b5Sbatschul 	va_end(alist);
8033042b8b5Sbatschul }
8043042b8b5Sbatschul 
8057c478bd9Sstevel@tonic-gate /*
8067c478bd9Sstevel@tonic-gate  * This is a separate function rather than a set of define's because of the
8077c478bd9Sstevel@tonic-gate  * gettext() wrapping.
8087c478bd9Sstevel@tonic-gate  */
8097c478bd9Sstevel@tonic-gate 
8107c478bd9Sstevel@tonic-gate /*
8117c478bd9Sstevel@tonic-gate  * TRANSLATION_NOTE
8127c478bd9Sstevel@tonic-gate  * Each string below should have \t follow \n whenever needed; the
8137c478bd9Sstevel@tonic-gate  * initial \t and the terminal \n will be provided by the calling function.
8147c478bd9Sstevel@tonic-gate  */
8157c478bd9Sstevel@tonic-gate 
8167c478bd9Sstevel@tonic-gate static char *
8177c478bd9Sstevel@tonic-gate long_help(int cmd_num)
8187c478bd9Sstevel@tonic-gate {
8197c478bd9Sstevel@tonic-gate 	static char line[1024];	/* arbitrary large amount */
8207c478bd9Sstevel@tonic-gate 
8217c478bd9Sstevel@tonic-gate 	assert(cmd_num >= CMD_MIN && cmd_num <= CMD_MAX);
8227c478bd9Sstevel@tonic-gate 	switch (cmd_num) {
8237c478bd9Sstevel@tonic-gate 		case CMD_HELP:
8247c478bd9Sstevel@tonic-gate 			return (gettext("Prints help message."));
8257c478bd9Sstevel@tonic-gate 		case CMD_CREATE:
8267c478bd9Sstevel@tonic-gate 			(void) snprintf(line, sizeof (line),
8277c478bd9Sstevel@tonic-gate 			    gettext("Creates a configuration for the "
8287c478bd9Sstevel@tonic-gate 			    "specified zone.  %s should be\n\tused to "
8297c478bd9Sstevel@tonic-gate 			    "begin configuring a new zone.  If overwriting an "
8307c478bd9Sstevel@tonic-gate 			    "existing\n\tconfiguration, the -F flag can be "
8317c478bd9Sstevel@tonic-gate 			    "used to force the action.  If\n\t-t template is "
8327c478bd9Sstevel@tonic-gate 			    "given, creates a configuration identical to the\n"
8337c478bd9Sstevel@tonic-gate 			    "\tspecified template, except that the zone name "
8349e518655Sgjelinek 			    "is changed from\n\ttemplate to zonename.  '%s -a' "
8359e518655Sgjelinek 			    "creates a configuration from a\n\tdetached "
8369e518655Sgjelinek 			    "zonepath.  '%s -b' results in a blank "
8379e518655Sgjelinek 			    "configuration.\n\t'%s' with no arguments applies "
8389e518655Sgjelinek 			    "the Sun default settings."),
8397c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_CREATE), cmd_to_str(CMD_CREATE),
8409e518655Sgjelinek 			    cmd_to_str(CMD_CREATE), cmd_to_str(CMD_CREATE));
8417c478bd9Sstevel@tonic-gate 			return (line);
8427c478bd9Sstevel@tonic-gate 		case CMD_EXIT:
8437c478bd9Sstevel@tonic-gate 			return (gettext("Exits the program.  The -F flag can "
8447c478bd9Sstevel@tonic-gate 			    "be used to force the action."));
8457c478bd9Sstevel@tonic-gate 		case CMD_EXPORT:
8467c478bd9Sstevel@tonic-gate 			return (gettext("Prints configuration to standard "
8477c478bd9Sstevel@tonic-gate 			    "output, or to output-file if\n\tspecified, in "
8487c478bd9Sstevel@tonic-gate 			    "a form suitable for use in a command-file."));
8497c478bd9Sstevel@tonic-gate 		case CMD_ADD:
8507c478bd9Sstevel@tonic-gate 			return (gettext("Add specified resource to "
8517c478bd9Sstevel@tonic-gate 			    "configuration."));
8527c478bd9Sstevel@tonic-gate 		case CMD_DELETE:
8537c478bd9Sstevel@tonic-gate 			return (gettext("Deletes the specified zone.  The -F "
8547c478bd9Sstevel@tonic-gate 			    "flag can be used to force the\n\taction."));
8557c478bd9Sstevel@tonic-gate 		case CMD_REMOVE:
8567c478bd9Sstevel@tonic-gate 			return (gettext("Remove specified resource from "
8570209230bSgjelinek 			    "configuration.  The -F flag can be used\n\tto "
8580209230bSgjelinek 			    "force the action."));
8597c478bd9Sstevel@tonic-gate 		case CMD_SELECT:
8607c478bd9Sstevel@tonic-gate 			(void) snprintf(line, sizeof (line),
8617c478bd9Sstevel@tonic-gate 			    gettext("Selects a resource to modify.  "
8627c478bd9Sstevel@tonic-gate 			    "Resource modification is completed\n\twith the "
8637c478bd9Sstevel@tonic-gate 			    "command \"%s\".  The property name/value pairs "
8647c478bd9Sstevel@tonic-gate 			    "must uniquely\n\tidentify a resource.  Note that "
8657c478bd9Sstevel@tonic-gate 			    "the curly braces ('{', '}') mean one\n\tor more "
8667c478bd9Sstevel@tonic-gate 			    "of whatever is between them."),
8677c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_END));
8687c478bd9Sstevel@tonic-gate 			return (line);
8697c478bd9Sstevel@tonic-gate 		case CMD_SET:
8707c478bd9Sstevel@tonic-gate 			return (gettext("Sets property values."));
8710209230bSgjelinek 		case CMD_CLEAR:
8720209230bSgjelinek 			return (gettext("Clears property values."));
8737c478bd9Sstevel@tonic-gate 		case CMD_INFO:
8747c478bd9Sstevel@tonic-gate 			return (gettext("Displays information about the "
8757c478bd9Sstevel@tonic-gate 			    "current configuration.  If resource\n\ttype is "
8767c478bd9Sstevel@tonic-gate 			    "specified, displays only information about "
8777c478bd9Sstevel@tonic-gate 			    "resources of\n\tthe relevant type.  If resource "
8787c478bd9Sstevel@tonic-gate 			    "id is specified, displays only\n\tinformation "
8797c478bd9Sstevel@tonic-gate 			    "about that resource."));
8807c478bd9Sstevel@tonic-gate 		case CMD_VERIFY:
8817c478bd9Sstevel@tonic-gate 			return (gettext("Verifies current configuration "
8827c478bd9Sstevel@tonic-gate 			    "for correctness (some resource types\n\thave "
8837c478bd9Sstevel@tonic-gate 			    "required properties)."));
8847c478bd9Sstevel@tonic-gate 		case CMD_COMMIT:
8857c478bd9Sstevel@tonic-gate 			(void) snprintf(line, sizeof (line),
8867c478bd9Sstevel@tonic-gate 			    gettext("Commits current configuration.  "
8877c478bd9Sstevel@tonic-gate 			    "Configuration must be committed to\n\tbe used by "
8887c478bd9Sstevel@tonic-gate 			    "%s.  Until the configuration is committed, "
8897c478bd9Sstevel@tonic-gate 			    "changes \n\tcan be removed with the %s "
8907c478bd9Sstevel@tonic-gate 			    "command.  This operation is\n\tattempted "
8917c478bd9Sstevel@tonic-gate 			    "automatically upon completion of a %s "
8927c478bd9Sstevel@tonic-gate 			    "session."), "zoneadm", cmd_to_str(CMD_REVERT),
8937c478bd9Sstevel@tonic-gate 			    "zonecfg");
8947c478bd9Sstevel@tonic-gate 			return (line);
8957c478bd9Sstevel@tonic-gate 		case CMD_REVERT:
8967c478bd9Sstevel@tonic-gate 			return (gettext("Reverts configuration back to the "
8977c478bd9Sstevel@tonic-gate 			    "last committed state.  The -F flag\n\tcan be "
8987c478bd9Sstevel@tonic-gate 			    "used to force the action."));
8997c478bd9Sstevel@tonic-gate 		case CMD_CANCEL:
9007c478bd9Sstevel@tonic-gate 			return (gettext("Cancels resource/property "
9017c478bd9Sstevel@tonic-gate 			    "specification."));
9027c478bd9Sstevel@tonic-gate 		case CMD_END:
9037c478bd9Sstevel@tonic-gate 			return (gettext("Ends resource/property "
9047c478bd9Sstevel@tonic-gate 			    "specification."));
9057c478bd9Sstevel@tonic-gate 	}
9067c478bd9Sstevel@tonic-gate 	/* NOTREACHED */
9077e362f58Scomay 	return (NULL);
9087c478bd9Sstevel@tonic-gate }
9097c478bd9Sstevel@tonic-gate 
9107c478bd9Sstevel@tonic-gate /*
9117c478bd9Sstevel@tonic-gate  * Called with verbose TRUE when help is explicitly requested, FALSE for
9127c478bd9Sstevel@tonic-gate  * unexpected errors.
9137c478bd9Sstevel@tonic-gate  */
9147c478bd9Sstevel@tonic-gate 
9157c478bd9Sstevel@tonic-gate void
916bbec428eSgjelinek usage(boolean_t verbose, uint_t flags)
9177c478bd9Sstevel@tonic-gate {
9183042b8b5Sbatschul 	FILE *fp = verbose ? stdout : stderr;
9193042b8b5Sbatschul 	FILE *newfp;
920bbec428eSgjelinek 	boolean_t need_to_close = B_FALSE;
921*12a8b243SSusan Kamm-Worrell 	char *pager, *space;
9227c478bd9Sstevel@tonic-gate 	int i;
9233042b8b5Sbatschul 	struct stat statbuf;
9247c478bd9Sstevel@tonic-gate 
9257c478bd9Sstevel@tonic-gate 	/* don't page error output */
9267c478bd9Sstevel@tonic-gate 	if (verbose && interactive_mode) {
9277c478bd9Sstevel@tonic-gate 		if ((pager = getenv("PAGER")) == NULL)
9287c478bd9Sstevel@tonic-gate 			pager = PAGER;
9293042b8b5Sbatschul 
930*12a8b243SSusan Kamm-Worrell 		space = strchr(pager, ' ');
931*12a8b243SSusan Kamm-Worrell 		if (space)
932*12a8b243SSusan Kamm-Worrell 			*space = '\0';
9333042b8b5Sbatschul 		if (stat(pager, &statbuf) == 0) {
934*12a8b243SSusan Kamm-Worrell 			if (space)
935*12a8b243SSusan Kamm-Worrell 				*space = ' ';
9367c478bd9Sstevel@tonic-gate 			if ((newfp = popen(pager, "w")) != NULL) {
937bbec428eSgjelinek 				need_to_close = B_TRUE;
9387c478bd9Sstevel@tonic-gate 				fp = newfp;
9397c478bd9Sstevel@tonic-gate 			}
9403042b8b5Sbatschul 		} else {
9413042b8b5Sbatschul 			zerr(gettext("PAGER %s does not exist (%s)."),
9423042b8b5Sbatschul 			    pager, strerror(errno));
9437c478bd9Sstevel@tonic-gate 		}
9443042b8b5Sbatschul 	}
9453042b8b5Sbatschul 
9467c478bd9Sstevel@tonic-gate 	if (flags & HELP_META) {
9477c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("More help is available for the "
9487c478bd9Sstevel@tonic-gate 		    "following:\n"));
9497c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\n\tcommands ('%s commands')\n",
9507c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_HELP));
9517c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\tsyntax ('%s syntax')\n",
9527c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_HELP));
9537c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\tusage ('%s usage')\n\n",
9547c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_HELP));
9557c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("You may also obtain help on any "
9567c478bd9Sstevel@tonic-gate 		    "command by typing '%s <command-name>.'\n"),
9577c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_HELP));
9587c478bd9Sstevel@tonic-gate 	}
9597c478bd9Sstevel@tonic-gate 	if (flags & HELP_RES_SCOPE) {
9607c478bd9Sstevel@tonic-gate 		switch (resource_scope) {
9617c478bd9Sstevel@tonic-gate 		case RT_FS:
9627c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
9637c478bd9Sstevel@tonic-gate 			    "used to configure a file-system.\n"),
9647c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
9657c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
9667c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9677c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_DIR), gettext("<path>"));
9687c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9697c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_SPECIAL), gettext("<path>"));
9707c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9717c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_RAW), gettext("<raw-device>"));
9727c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9737c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_TYPE), gettext("<file-system type>"));
9747c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s %s\n", cmd_to_str(CMD_ADD),
9757c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_OPTIONS),
9767c478bd9Sstevel@tonic-gate 			    gettext("<file-system options>"));
977ffbafc53Scomay 			(void) fprintf(fp, "\t%s %s %s\n",
978ffbafc53Scomay 			    cmd_to_str(CMD_REMOVE), pt_to_str(PT_OPTIONS),
979ffbafc53Scomay 			    gettext("<file-system options>"));
9807c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Consult the file-system "
9817c478bd9Sstevel@tonic-gate 			    "specific manual page, such as mount_ufs(1M), "
9827c478bd9Sstevel@tonic-gate 			    "for\ndetails about file-system options.  Note "
9837c478bd9Sstevel@tonic-gate 			    "that any file-system options with an\nembedded "
9847c478bd9Sstevel@tonic-gate 			    "'=' character must be enclosed in double quotes, "
9857c478bd9Sstevel@tonic-gate 			    /*CSTYLED*/
9867c478bd9Sstevel@tonic-gate 			    "such as \"%s=5\".\n"), MNTOPT_RETRY);
9877c478bd9Sstevel@tonic-gate 			break;
9887c478bd9Sstevel@tonic-gate 		case RT_NET:
9897c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
9907c478bd9Sstevel@tonic-gate 			    "used to configure a network interface.\n"),
9917c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
9927c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
9937c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9947c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_ADDRESS), gettext("<IP-address>"));
9957c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
996550b6e40SSowmini Varadhan 			    pt_to_str(PT_ALLOWED_ADDRESS),
997550b6e40SSowmini Varadhan 			    gettext("<IP-address>"));
998550b6e40SSowmini Varadhan 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
9997c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_PHYSICAL), gettext("<interface>"));
10007c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("See ifconfig(1M) for "
10017c478bd9Sstevel@tonic-gate 			    "details of the <interface> string.\n"));
1002550b6e40SSowmini Varadhan 			(void) fprintf(fp, gettext("%s %s is valid "
1003550b6e40SSowmini Varadhan 			    "if the %s property is set to %s, otherwise it "
1004de860bd9Sgfaden 			    "must not be set.\n"),
1005f4b3ec61Sdh155122 			    cmd_to_str(CMD_SET), pt_to_str(PT_ADDRESS),
1006550b6e40SSowmini Varadhan 			    pt_to_str(PT_IPTYPE), gettext("shared"));
1007550b6e40SSowmini Varadhan 			(void) fprintf(fp, gettext("%s %s is valid "
1008550b6e40SSowmini Varadhan 			    "if the %s property is set to %s, otherwise it "
1009550b6e40SSowmini Varadhan 			    "must not be set.\n"),
1010550b6e40SSowmini Varadhan 			    cmd_to_str(CMD_SET), pt_to_str(PT_ALLOWED_ADDRESS),
1011550b6e40SSowmini Varadhan 			    pt_to_str(PT_IPTYPE), gettext("exclusive"));
1012550b6e40SSowmini Varadhan 			(void) fprintf(fp, gettext("\t%s %s=%s\n%s %s "
1013550b6e40SSowmini Varadhan 			    "is valid if the %s or %s property is set, "
1014550b6e40SSowmini Varadhan 			    "otherwise it must not be set\n"),
1015550b6e40SSowmini Varadhan 			    cmd_to_str(CMD_SET),
1016550b6e40SSowmini Varadhan 			    pt_to_str(PT_DEFROUTER), gettext("<IP-address>"),
1017de860bd9Sgfaden 			    cmd_to_str(CMD_SET), pt_to_str(PT_DEFROUTER),
1018550b6e40SSowmini Varadhan 			    gettext(pt_to_str(PT_ADDRESS)),
1019550b6e40SSowmini Varadhan 			    gettext(pt_to_str(PT_ALLOWED_ADDRESS)));
10207c478bd9Sstevel@tonic-gate 			break;
10217c478bd9Sstevel@tonic-gate 		case RT_DEVICE:
10227c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
10237c478bd9Sstevel@tonic-gate 			    "used to configure a device node.\n"),
10247c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
10257c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
10267c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10277c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_MATCH), gettext("<device-path>"));
10287c478bd9Sstevel@tonic-gate 			break;
10297c478bd9Sstevel@tonic-gate 		case RT_RCTL:
10307c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
10317c478bd9Sstevel@tonic-gate 			    "used to configure a resource control.\n"),
10327c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
10337c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
10347c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10357c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_NAME), gettext("<string>"));
10367c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s (%s=%s,%s=%s,%s=%s)\n",
10377c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_ADD), pt_to_str(PT_VALUE),
10387c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_PRIV), gettext("<priv-value>"),
10397c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_LIMIT), gettext("<number>"),
10407c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_ACTION), gettext("<action-value>"));
1041ffbafc53Scomay 			(void) fprintf(fp, "\t%s %s (%s=%s,%s=%s,%s=%s)\n",
1042ffbafc53Scomay 			    cmd_to_str(CMD_REMOVE), pt_to_str(PT_VALUE),
1043ffbafc53Scomay 			    pt_to_str(PT_PRIV), gettext("<priv-value>"),
1044ffbafc53Scomay 			    pt_to_str(PT_LIMIT), gettext("<number>"),
1045ffbafc53Scomay 			    pt_to_str(PT_ACTION), gettext("<action-value>"));
10467c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "%s\n\t%s := privileged\n"
10477c478bd9Sstevel@tonic-gate 			    "\t%s := none | deny\n", gettext("Where"),
10487c478bd9Sstevel@tonic-gate 			    gettext("<priv-value>"), gettext("<action-value>"));
10497c478bd9Sstevel@tonic-gate 			break;
10507c478bd9Sstevel@tonic-gate 		case RT_ATTR:
10517c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
10527c478bd9Sstevel@tonic-gate 			    "used to configure a generic attribute.\n"),
10537c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
10547c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
10557c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10567c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_NAME), gettext("<name>"));
10577c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=boolean\n",
10587c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_SET), pt_to_str(PT_TYPE));
10597c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=true | false\n",
10607c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_SET), pt_to_str(PT_VALUE));
10617c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("or\n"));
10627c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=int\n", cmd_to_str(CMD_SET),
10637c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_TYPE));
10647c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10657c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_VALUE), gettext("<integer>"));
10667c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("or\n"));
10677c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=string\n",
10687c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_SET), pt_to_str(PT_TYPE));
10697c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10707c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_VALUE), gettext("<string>"));
10717c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("or\n"));
10727c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=uint\n",
10737c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_SET), pt_to_str(PT_TYPE));
10747c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10757c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_VALUE), gettext("<unsigned integer>"));
10767c478bd9Sstevel@tonic-gate 			break;
1077fa9e4066Sahrens 		case RT_DATASET:
1078fa9e4066Sahrens 			(void) fprintf(fp, gettext("The '%s' resource scope is "
1079fa9e4066Sahrens 			    "used to export ZFS datasets.\n"),
1080fa9e4066Sahrens 			    rt_to_str(resource_scope));
1081fa9e4066Sahrens 			(void) fprintf(fp, gettext("Valid commands:\n"));
1082fa9e4066Sahrens 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1083fa9e4066Sahrens 			    pt_to_str(PT_NAME), gettext("<name>"));
1084fa9e4066Sahrens 			break;
10850209230bSgjelinek 		case RT_DCPU:
10860209230bSgjelinek 			(void) fprintf(fp, gettext("The '%s' resource scope "
10870209230bSgjelinek 			    "configures the 'pools' facility to dedicate\na "
10880209230bSgjelinek 			    "subset of the system's processors to this zone "
10890209230bSgjelinek 			    "while it is running.\n"),
10900209230bSgjelinek 			    rt_to_str(resource_scope));
10910209230bSgjelinek 			(void) fprintf(fp, gettext("Valid commands:\n"));
10920209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10930209230bSgjelinek 			    pt_to_str(PT_NCPUS),
10940209230bSgjelinek 			    gettext("<unsigned integer | range>"));
10950209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10960209230bSgjelinek 			    pt_to_str(PT_IMPORTANCE),
10970209230bSgjelinek 			    gettext("<unsigned integer>"));
10980209230bSgjelinek 			break;
1099c97ad5cdSakolb 		case RT_PCAP:
1100c97ad5cdSakolb 			(void) fprintf(fp, gettext("The '%s' resource scope is "
1101c97ad5cdSakolb 			    "used to set an upper limit (a cap) on the\n"
1102c97ad5cdSakolb 			    "percentage of CPU that can be used by this zone.  "
1103c97ad5cdSakolb 			    "A '%s' value of 1\ncorresponds to one cpu.  The "
1104c97ad5cdSakolb 			    "value can be set higher than 1, up to the total\n"
1105c97ad5cdSakolb 			    "number of CPUs on the system.  The value can "
1106c97ad5cdSakolb 			    "also be less than 1,\nrepresenting a fraction of "
1107c97ad5cdSakolb 			    "a cpu.\n"),
1108c97ad5cdSakolb 			    rt_to_str(resource_scope), pt_to_str(PT_NCPUS));
1109c97ad5cdSakolb 			(void) fprintf(fp, gettext("Valid commands:\n"));
1110c97ad5cdSakolb 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1111c97ad5cdSakolb 			    pt_to_str(PT_NCPUS), gettext("<unsigned decimal>"));
1112c97ad5cdSakolb 			break;
11130209230bSgjelinek 		case RT_MCAP:
11140209230bSgjelinek 			(void) fprintf(fp, gettext("The '%s' resource scope is "
11150209230bSgjelinek 			    "used to set an upper limit (a cap) on the\n"
11160209230bSgjelinek 			    "amount of physical memory, swap space and locked "
11170209230bSgjelinek 			    "memory that can be used by\nthis zone.\n"),
11180209230bSgjelinek 			    rt_to_str(resource_scope));
11190209230bSgjelinek 			(void) fprintf(fp, gettext("Valid commands:\n"));
11200209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
11210209230bSgjelinek 			    pt_to_str(PT_PHYSICAL),
11220209230bSgjelinek 			    gettext("<qualified unsigned decimal>"));
11230209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
11240209230bSgjelinek 			    pt_to_str(PT_SWAP),
11250209230bSgjelinek 			    gettext("<qualified unsigned decimal>"));
11260209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
11270209230bSgjelinek 			    pt_to_str(PT_LOCKED),
11280209230bSgjelinek 			    gettext("<qualified unsigned decimal>"));
11290209230bSgjelinek 			break;
1130cb8a054bSGlenn Faden 		case RT_ADMIN:
1131cb8a054bSGlenn Faden 			(void) fprintf(fp, gettext("The '%s' resource scope is "
1132cb8a054bSGlenn Faden 			    "used to delegate specific zone management\n"
1133cb8a054bSGlenn Faden 			    "rights to users and roles. These rights are "
1134cb8a054bSGlenn Faden 			    "only applicable to this zone.\n"),
1135cb8a054bSGlenn Faden 			    rt_to_str(resource_scope));
1136cb8a054bSGlenn Faden 			(void) fprintf(fp, gettext("Valid commands:\n"));
1137cb8a054bSGlenn Faden 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1138cb8a054bSGlenn Faden 			    pt_to_str(PT_USER),
1139cb8a054bSGlenn Faden 			    gettext("<single user or role name>"));
1140cb8a054bSGlenn Faden 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1141cb8a054bSGlenn Faden 			    pt_to_str(PT_AUTHS),
1142cb8a054bSGlenn Faden 			    gettext("<comma separated list>"));
1143cb8a054bSGlenn Faden 			break;
11447c478bd9Sstevel@tonic-gate 		}
11457c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("And from any resource scope, you "
11467c478bd9Sstevel@tonic-gate 		    "can:\n"));
11477c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", cmd_to_str(CMD_END),
11487c478bd9Sstevel@tonic-gate 		    gettext("(to conclude this operation)"));
11497c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", cmd_to_str(CMD_CANCEL),
11507c478bd9Sstevel@tonic-gate 		    gettext("(to cancel this operation)"));
11517c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", cmd_to_str(CMD_EXIT),
11527c478bd9Sstevel@tonic-gate 		    gettext("(to exit the zonecfg utility)"));
11537c478bd9Sstevel@tonic-gate 	}
11547c478bd9Sstevel@tonic-gate 	if (flags & HELP_USAGE) {
11557c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s:\t%s %s\n", gettext("usage"),
11567c478bd9Sstevel@tonic-gate 		    execname, cmd_to_str(CMD_HELP));
11577c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s -z <zone>\t\t\t(%s)\n",
11587c478bd9Sstevel@tonic-gate 		    execname, gettext("interactive"));
11597c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s -z <zone> <command>\n", execname);
11607c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s -z <zone> -f <command-file>\n",
11617c478bd9Sstevel@tonic-gate 		    execname);
11627c478bd9Sstevel@tonic-gate 	}
11637c478bd9Sstevel@tonic-gate 	if (flags & HELP_SUBCMDS) {
11647c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s:\n\n", gettext("Commands"));
11657c478bd9Sstevel@tonic-gate 		for (i = 0; i <= CMD_MAX; i++) {
11667c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "%s\n", helptab[i].short_usage);
11677c478bd9Sstevel@tonic-gate 			if (verbose)
11687c478bd9Sstevel@tonic-gate 				(void) fprintf(fp, "\t%s\n\n", long_help(i));
11697c478bd9Sstevel@tonic-gate 		}
11707c478bd9Sstevel@tonic-gate 	}
11717c478bd9Sstevel@tonic-gate 	if (flags & HELP_SYNTAX) {
11727c478bd9Sstevel@tonic-gate 		if (!verbose)
11737c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\n");
11747c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "<zone> := [A-Za-z0-9][A-Za-z0-9_.-]*\n");
11757c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("\t(except the reserved words "
11767c478bd9Sstevel@tonic-gate 		    "'%s' and anything starting with '%s')\n"), "global",
11777c478bd9Sstevel@tonic-gate 		    "SUNW");
11787c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
11797c478bd9Sstevel@tonic-gate 		    gettext("\tName must be less than %d characters.\n"),
11807c478bd9Sstevel@tonic-gate 		    ZONENAME_MAX);
11817c478bd9Sstevel@tonic-gate 		if (verbose)
11827c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\n");
11837c478bd9Sstevel@tonic-gate 	}
11847c478bd9Sstevel@tonic-gate 	if (flags & HELP_NETADDR) {
11857c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("\n<net-addr> :="));
11867c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
11877c478bd9Sstevel@tonic-gate 		    gettext("\t<IPv4-address>[/<IPv4-prefix-length>] |\n"));
11887c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
11897c478bd9Sstevel@tonic-gate 		    gettext("\t\t<IPv6-address>/<IPv6-prefix-length> |\n"));
11907c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
11917c478bd9Sstevel@tonic-gate 		    gettext("\t\t<hostname>[/<IPv4-prefix-length>]\n"));
11927c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("See inet(3SOCKET) for IPv4 and "
11937c478bd9Sstevel@tonic-gate 		    "IPv6 address syntax.\n"));
11947c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("<IPv4-prefix-length> := [0-32]\n"));
11957c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
11967c478bd9Sstevel@tonic-gate 		    gettext("<IPv6-prefix-length> := [0-128]\n"));
11977c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
11987c478bd9Sstevel@tonic-gate 		    gettext("<hostname> := [A-Za-z0-9][A-Za-z0-9-.]*\n"));
11997c478bd9Sstevel@tonic-gate 	}
12007c478bd9Sstevel@tonic-gate 	if (flags & HELP_RESOURCES) {
12016e1ae2a3SGary Pennington 		(void) fprintf(fp, "<%s> := %s | %s | %s | %s | %s |\n\t"
1202cb8a054bSGlenn Faden 		    "%s | %s | %s | %s | %s\n\n",
12037c478bd9Sstevel@tonic-gate 		    gettext("resource type"), rt_to_str(RT_FS),
12046e1ae2a3SGary Pennington 		    rt_to_str(RT_NET), rt_to_str(RT_DEVICE),
12059e7542f4Sdp 		    rt_to_str(RT_RCTL), rt_to_str(RT_ATTR),
12060209230bSgjelinek 		    rt_to_str(RT_DATASET), rt_to_str(RT_DCPU),
1207cb8a054bSGlenn Faden 		    rt_to_str(RT_PCAP), rt_to_str(RT_MCAP),
1208cb8a054bSGlenn Faden 		    rt_to_str(RT_ADMIN));
12097c478bd9Sstevel@tonic-gate 	}
12107c478bd9Sstevel@tonic-gate 	if (flags & HELP_PROPS) {
12117c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("For resource type ... there are "
12127c478bd9Sstevel@tonic-gate 		    "property types ...:\n"));
12137c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
1214087719fdSdp 		    pt_to_str(PT_ZONENAME));
1215087719fdSdp 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12167c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_ZONEPATH));
12177c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12189acbbeafSnn35248 		    pt_to_str(PT_BRAND));
12199acbbeafSnn35248 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12207c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_AUTOBOOT));
12217c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12223f2f09c1Sdp 		    pt_to_str(PT_BOOTARGS));
12233f2f09c1Sdp 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12247c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_POOL));
1225ffbafc53Scomay 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
1226ffbafc53Scomay 		    pt_to_str(PT_LIMITPRIV));
12270209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12280209230bSgjelinek 		    pt_to_str(PT_SCHED));
12290209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
1230f4b3ec61Sdh155122 		    pt_to_str(PT_IPTYPE));
1231f4b3ec61Sdh155122 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12325679c89fSjv227347 		    pt_to_str(PT_HOSTID));
12335679c89fSjv227347 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12340fbb751dSJohn Levon 		    pt_to_str(PT_FS_ALLOWED));
12350fbb751dSJohn Levon 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12360209230bSgjelinek 		    pt_to_str(PT_MAXLWPS));
12370209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
1238ff19e029SMenno Lageman 		    pt_to_str(PT_MAXPROCS));
1239ff19e029SMenno Lageman 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12400209230bSgjelinek 		    pt_to_str(PT_MAXSHMMEM));
12410209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12420209230bSgjelinek 		    pt_to_str(PT_MAXSHMIDS));
12430209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12440209230bSgjelinek 		    pt_to_str(PT_MAXMSGIDS));
12450209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12460209230bSgjelinek 		    pt_to_str(PT_MAXSEMIDS));
12470209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
12480209230bSgjelinek 		    pt_to_str(PT_SHARES));
12496d4d1c0dSbatschul 		(void) fprintf(fp, "\t%s\t\t%s, %s, %s, %s, %s\n",
12506d4d1c0dSbatschul 		    rt_to_str(RT_FS), pt_to_str(PT_DIR),
12516d4d1c0dSbatschul 		    pt_to_str(PT_SPECIAL), pt_to_str(PT_RAW),
12526d4d1c0dSbatschul 		    pt_to_str(PT_TYPE), pt_to_str(PT_OPTIONS));
1253550b6e40SSowmini Varadhan 		(void) fprintf(fp, "\t%s\t\t%s, %s, %s|%s\n", rt_to_str(RT_NET),
1254550b6e40SSowmini Varadhan 		    pt_to_str(PT_ADDRESS), pt_to_str(PT_ALLOWED_ADDRESS),
1255550b6e40SSowmini Varadhan 		    pt_to_str(PT_PHYSICAL), pt_to_str(PT_DEFROUTER));
12567c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t\t%s\n", rt_to_str(RT_DEVICE),
12577c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_MATCH));
12587c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t\t%s, %s\n", rt_to_str(RT_RCTL),
12597c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_NAME), pt_to_str(PT_VALUE));
12607c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t\t%s, %s, %s\n", rt_to_str(RT_ATTR),
12617c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_NAME), pt_to_str(PT_TYPE),
12627c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_VALUE));
1263fa9e4066Sahrens 		(void) fprintf(fp, "\t%s\t\t%s\n", rt_to_str(RT_DATASET),
1264fa9e4066Sahrens 		    pt_to_str(PT_NAME));
12650209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s, %s\n", rt_to_str(RT_DCPU),
12660209230bSgjelinek 		    pt_to_str(PT_NCPUS), pt_to_str(PT_IMPORTANCE));
1267c97ad5cdSakolb 		(void) fprintf(fp, "\t%s\t%s\n", rt_to_str(RT_PCAP),
1268c97ad5cdSakolb 		    pt_to_str(PT_NCPUS));
12690209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s, %s, %s\n", rt_to_str(RT_MCAP),
12700209230bSgjelinek 		    pt_to_str(PT_PHYSICAL), pt_to_str(PT_SWAP),
12710209230bSgjelinek 		    pt_to_str(PT_LOCKED));
1272cb8a054bSGlenn Faden 		(void) fprintf(fp, "\t%s\t\t%s, %s\n", rt_to_str(RT_ADMIN),
1273cb8a054bSGlenn Faden 		    pt_to_str(PT_USER), pt_to_str(PT_AUTHS));
12747c478bd9Sstevel@tonic-gate 	}
12757c478bd9Sstevel@tonic-gate 	if (need_to_close)
12767c478bd9Sstevel@tonic-gate 		(void) pclose(fp);
12777c478bd9Sstevel@tonic-gate }
12787c478bd9Sstevel@tonic-gate 
12797c478bd9Sstevel@tonic-gate static void
1280bbec428eSgjelinek zone_perror(char *prefix, int err, boolean_t set_saw)
12817c478bd9Sstevel@tonic-gate {
12827c478bd9Sstevel@tonic-gate 	zerr("%s: %s", prefix, zonecfg_strerror(err));
12837c478bd9Sstevel@tonic-gate 	if (set_saw)
1284bbec428eSgjelinek 		saw_error = B_TRUE;
12857c478bd9Sstevel@tonic-gate }
12867c478bd9Sstevel@tonic-gate 
12877c478bd9Sstevel@tonic-gate /*
12887c478bd9Sstevel@tonic-gate  * zone_perror() expects a single string, but for remove and select
12897c478bd9Sstevel@tonic-gate  * we have both the command and the resource type, so this wrapper
12907c478bd9Sstevel@tonic-gate  * function serves the same purpose in a slightly different way.
12917c478bd9Sstevel@tonic-gate  */
12927c478bd9Sstevel@tonic-gate 
12937c478bd9Sstevel@tonic-gate static void
1294bbec428eSgjelinek z_cmd_rt_perror(int cmd_num, int res_num, int err, boolean_t set_saw)
12957c478bd9Sstevel@tonic-gate {
12967c478bd9Sstevel@tonic-gate 	zerr("%s %s: %s", cmd_to_str(cmd_num), rt_to_str(res_num),
12977c478bd9Sstevel@tonic-gate 	    zonecfg_strerror(err));
12987c478bd9Sstevel@tonic-gate 	if (set_saw)
1299bbec428eSgjelinek 		saw_error = B_TRUE;
13007c478bd9Sstevel@tonic-gate }
13017c478bd9Sstevel@tonic-gate 
13027c478bd9Sstevel@tonic-gate /* returns Z_OK if successful, Z_foo from <libzonecfg.h> otherwise */
13037c478bd9Sstevel@tonic-gate static int
1304bbec428eSgjelinek initialize(boolean_t handle_expected)
13057c478bd9Sstevel@tonic-gate {
13067c478bd9Sstevel@tonic-gate 	int err;
13079acbbeafSnn35248 	char brandname[MAXNAMELEN];
13087c478bd9Sstevel@tonic-gate 
13097c478bd9Sstevel@tonic-gate 	if (zonecfg_check_handle(handle) != Z_OK) {
13107c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_get_handle(zone, handle)) == Z_OK) {
1311bbec428eSgjelinek 			got_handle = B_TRUE;
13129acbbeafSnn35248 			if (zonecfg_get_brand(handle, brandname,
13139acbbeafSnn35248 			    sizeof (brandname)) != Z_OK) {
13149acbbeafSnn35248 				zerr("Zone %s is inconsistent: missing "
13159acbbeafSnn35248 				    "brand attribute", zone);
13169acbbeafSnn35248 				exit(Z_ERR);
13179acbbeafSnn35248 			}
13189acbbeafSnn35248 			if ((brand = brand_open(brandname)) == NULL) {
13199acbbeafSnn35248 				zerr("Zone %s uses non-existent brand \"%s\"."
13209acbbeafSnn35248 				    "  Unable to continue", zone, brandname);
13219acbbeafSnn35248 				exit(Z_ERR);
13229acbbeafSnn35248 			}
1323cb8a054bSGlenn Faden 			/*
1324cb8a054bSGlenn Faden 			 * If the user_attr file is newer than
1325cb8a054bSGlenn Faden 			 * the zone config file, the admins
1326cb8a054bSGlenn Faden 			 * may need to be updated since the
1327cb8a054bSGlenn Faden 			 * RBAC files are authoritative for
1328cb8a054bSGlenn Faden 			 * authorization checks.
1329cb8a054bSGlenn Faden 			 */
1330cb8a054bSGlenn Faden 			err = zonecfg_update_userauths(handle, zone);
1331cb8a054bSGlenn Faden 			if (err == Z_OK) {
1332cb8a054bSGlenn Faden 				zerr(gettext("The administrative rights "
1333cb8a054bSGlenn Faden 				    "were updated to match "
1334cb8a054bSGlenn Faden 				    "the current RBAC configuration.\n"
1335cb8a054bSGlenn Faden 				    "Use \"info admin\" and \"revert\" to "
1336cb8a054bSGlenn Faden 				    "compare with the previous settings."));
1337cb8a054bSGlenn Faden 				need_to_commit = B_TRUE;
1338cb8a054bSGlenn Faden 			} else if (err != Z_NO_ENTRY) {
1339cb8a054bSGlenn Faden 				zerr(gettext("failed to update "
1340cb8a054bSGlenn Faden 				    "admin  rights."));
1341cb8a054bSGlenn Faden 				exit(Z_ERR);
1342cb8a054bSGlenn Faden 			} else if (need_to_commit) {
1343cb8a054bSGlenn Faden 				zerr(gettext("admin rights were updated "
1344cb8a054bSGlenn Faden 				    "to match RBAC configuration."));
1345cb8a054bSGlenn Faden 			}
1346cb8a054bSGlenn Faden 
13470209230bSgjelinek 		} else if (global_zone && err == Z_NO_ZONE && !got_handle &&
13480209230bSgjelinek 		    !read_only_mode) {
13490209230bSgjelinek 			/*
13500209230bSgjelinek 			 * We implicitly create the global zone config if it
13510209230bSgjelinek 			 * doesn't exist.
13520209230bSgjelinek 			 */
13530209230bSgjelinek 			zone_dochandle_t tmphandle;
13540209230bSgjelinek 
13550209230bSgjelinek 			if ((tmphandle = zonecfg_init_handle()) == NULL) {
1356bbec428eSgjelinek 				zone_perror(execname, Z_NOMEM, B_TRUE);
13570209230bSgjelinek 				exit(Z_ERR);
13580209230bSgjelinek 			}
13590209230bSgjelinek 
13600209230bSgjelinek 			err = zonecfg_get_template_handle("SUNWblank", zone,
13610209230bSgjelinek 			    tmphandle);
13620209230bSgjelinek 
13630209230bSgjelinek 			if (err != Z_OK) {
13640209230bSgjelinek 				zonecfg_fini_handle(tmphandle);
1365bbec428eSgjelinek 				zone_perror("SUNWblank", err, B_TRUE);
13660209230bSgjelinek 				return (err);
13670209230bSgjelinek 			}
13680209230bSgjelinek 
1369bbec428eSgjelinek 			need_to_commit = B_TRUE;
13700209230bSgjelinek 			zonecfg_fini_handle(handle);
13710209230bSgjelinek 			handle = tmphandle;
1372bbec428eSgjelinek 			got_handle = B_TRUE;
13730209230bSgjelinek 
13747c478bd9Sstevel@tonic-gate 		} else {
13757c478bd9Sstevel@tonic-gate 			zone_perror(zone, err, handle_expected || got_handle);
13767c478bd9Sstevel@tonic-gate 			if (err == Z_NO_ZONE && !got_handle &&
13777c478bd9Sstevel@tonic-gate 			    interactive_mode && !read_only_mode)
13787c478bd9Sstevel@tonic-gate 				(void) printf(gettext("Use '%s' to begin "
13797c478bd9Sstevel@tonic-gate 				    "configuring a new zone.\n"),
13807c478bd9Sstevel@tonic-gate 				    cmd_to_str(CMD_CREATE));
13817c478bd9Sstevel@tonic-gate 			return (err);
13827c478bd9Sstevel@tonic-gate 		}
13837c478bd9Sstevel@tonic-gate 	}
13847c478bd9Sstevel@tonic-gate 	return (Z_OK);
13857c478bd9Sstevel@tonic-gate }
13867c478bd9Sstevel@tonic-gate 
1387bbec428eSgjelinek static boolean_t
1388087719fdSdp state_atleast(zone_state_t state)
1389087719fdSdp {
1390087719fdSdp 	zone_state_t state_num;
1391087719fdSdp 	int err;
1392087719fdSdp 
1393087719fdSdp 	if ((err = zone_get_state(zone, &state_num)) != Z_OK) {
1394087719fdSdp 		/* all states are greater than "non-existent" */
1395087719fdSdp 		if (err == Z_NO_ZONE)
1396087719fdSdp 			return (B_FALSE);
1397087719fdSdp 		zerr(gettext("Unexpectedly failed to determine state "
1398087719fdSdp 		    "of zone %s: %s"), zone, zonecfg_strerror(err));
1399087719fdSdp 		exit(Z_ERR);
1400087719fdSdp 	}
1401087719fdSdp 	return (state_num >= state);
1402087719fdSdp }
1403087719fdSdp 
14047c478bd9Sstevel@tonic-gate /*
14057c478bd9Sstevel@tonic-gate  * short_usage() is for bad syntax: getopt() issues, too many arguments, etc.
14067c478bd9Sstevel@tonic-gate  */
14077c478bd9Sstevel@tonic-gate 
14087c478bd9Sstevel@tonic-gate void
14097c478bd9Sstevel@tonic-gate short_usage(int command)
14107c478bd9Sstevel@tonic-gate {
14117c478bd9Sstevel@tonic-gate 	/* lex_lineno has already been incremented in the lexer; compensate */
14127c478bd9Sstevel@tonic-gate 	if (cmd_file_mode) {
14137c478bd9Sstevel@tonic-gate 		if (strcmp(cmd_file_name, "-") == 0)
14147c478bd9Sstevel@tonic-gate 			(void) fprintf(stderr,
14157c478bd9Sstevel@tonic-gate 			    gettext("syntax error on line %d\n"),
14167c478bd9Sstevel@tonic-gate 			    lex_lineno - 1);
14177c478bd9Sstevel@tonic-gate 		else
14187c478bd9Sstevel@tonic-gate 			(void) fprintf(stderr,
14197c478bd9Sstevel@tonic-gate 			    gettext("syntax error on line %d of %s\n"),
14207c478bd9Sstevel@tonic-gate 			    lex_lineno - 1, cmd_file_name);
14217c478bd9Sstevel@tonic-gate 	}
14227c478bd9Sstevel@tonic-gate 	(void) fprintf(stderr, "%s:\n%s\n", gettext("usage"),
14237c478bd9Sstevel@tonic-gate 	    helptab[command].short_usage);
1424bbec428eSgjelinek 	saw_error = B_TRUE;
14257c478bd9Sstevel@tonic-gate }
14267c478bd9Sstevel@tonic-gate 
14277c478bd9Sstevel@tonic-gate /*
14287c478bd9Sstevel@tonic-gate  * long_usage() is for bad semantics: e.g., wrong property type for a given
14297c478bd9Sstevel@tonic-gate  * resource type.  It is also used by longer_usage() below.
14307c478bd9Sstevel@tonic-gate  */
14317c478bd9Sstevel@tonic-gate 
14327c478bd9Sstevel@tonic-gate void
1433bbec428eSgjelinek long_usage(uint_t cmd_num, boolean_t set_saw)
14347c478bd9Sstevel@tonic-gate {
14357c478bd9Sstevel@tonic-gate 	(void) fprintf(set_saw ? stderr : stdout, "%s:\n%s\n", gettext("usage"),
14367c478bd9Sstevel@tonic-gate 	    helptab[cmd_num].short_usage);
14377c478bd9Sstevel@tonic-gate 	(void) fprintf(set_saw ? stderr : stdout, "\t%s\n", long_help(cmd_num));
14387c478bd9Sstevel@tonic-gate 	if (set_saw)
1439bbec428eSgjelinek 		saw_error = B_TRUE;
14407c478bd9Sstevel@tonic-gate }
14417c478bd9Sstevel@tonic-gate 
14427c478bd9Sstevel@tonic-gate /*
14437c478bd9Sstevel@tonic-gate  * longer_usage() is for 'help foo' and 'foo -?': call long_usage() and also
14447c478bd9Sstevel@tonic-gate  * any extra usage() flags as appropriate for whatever command.
14457c478bd9Sstevel@tonic-gate  */
14467c478bd9Sstevel@tonic-gate 
14477c478bd9Sstevel@tonic-gate void
14487c478bd9Sstevel@tonic-gate longer_usage(uint_t cmd_num)
14497c478bd9Sstevel@tonic-gate {
1450bbec428eSgjelinek 	long_usage(cmd_num, B_FALSE);
14517c478bd9Sstevel@tonic-gate 	if (helptab[cmd_num].flags != 0) {
14527c478bd9Sstevel@tonic-gate 		(void) printf("\n");
1453bbec428eSgjelinek 		usage(B_TRUE, helptab[cmd_num].flags);
14547c478bd9Sstevel@tonic-gate 	}
14557c478bd9Sstevel@tonic-gate }
14567c478bd9Sstevel@tonic-gate 
14577c478bd9Sstevel@tonic-gate /*
14587c478bd9Sstevel@tonic-gate  * scope_usage() is simply used when a command is called from the wrong scope.
14597c478bd9Sstevel@tonic-gate  */
14607c478bd9Sstevel@tonic-gate 
14617c478bd9Sstevel@tonic-gate static void
14627c478bd9Sstevel@tonic-gate scope_usage(uint_t cmd_num)
14637c478bd9Sstevel@tonic-gate {
14647c478bd9Sstevel@tonic-gate 	zerr(gettext("The %s command only makes sense in the %s scope."),
14657c478bd9Sstevel@tonic-gate 	    cmd_to_str(cmd_num),
14667c478bd9Sstevel@tonic-gate 	    global_scope ?  gettext("resource") : gettext("global"));
1467bbec428eSgjelinek 	saw_error = B_TRUE;
14687c478bd9Sstevel@tonic-gate }
14697c478bd9Sstevel@tonic-gate 
14707c478bd9Sstevel@tonic-gate /*
1471bbec428eSgjelinek  * On input, B_TRUE => yes, B_FALSE => no.
1472bbec428eSgjelinek  * On return, B_TRUE => 1, B_FALSE => no, could not ask => -1.
14737c478bd9Sstevel@tonic-gate  */
14747c478bd9Sstevel@tonic-gate 
14757c478bd9Sstevel@tonic-gate static int
1476bbec428eSgjelinek ask_yesno(boolean_t default_answer, const char *question)
14777c478bd9Sstevel@tonic-gate {
14787c478bd9Sstevel@tonic-gate 	char line[64];	/* should be enough to answer yes or no */
14797c478bd9Sstevel@tonic-gate 
14807c478bd9Sstevel@tonic-gate 	if (!ok_to_prompt) {
1481bbec428eSgjelinek 		saw_error = B_TRUE;
14827c478bd9Sstevel@tonic-gate 		return (-1);
14837c478bd9Sstevel@tonic-gate 	}
14847c478bd9Sstevel@tonic-gate 	for (;;) {
1485087719fdSdp 		if (printf("%s (%s)? ", question,
1486087719fdSdp 		    default_answer ? "[y]/n" : "y/[n]") < 0)
1487087719fdSdp 			return (-1);
1488087719fdSdp 		if (fgets(line, sizeof (line), stdin) == NULL)
1489087719fdSdp 			return (-1);
1490087719fdSdp 
1491087719fdSdp 		if (line[0] == '\n')
14927c478bd9Sstevel@tonic-gate 			return (default_answer ? 1 : 0);
14937c478bd9Sstevel@tonic-gate 		if (tolower(line[0]) == 'y')
14947c478bd9Sstevel@tonic-gate 			return (1);
14957c478bd9Sstevel@tonic-gate 		if (tolower(line[0]) == 'n')
14967c478bd9Sstevel@tonic-gate 			return (0);
14977c478bd9Sstevel@tonic-gate 	}
14987c478bd9Sstevel@tonic-gate }
14997c478bd9Sstevel@tonic-gate 
15007c478bd9Sstevel@tonic-gate /*
15017c478bd9Sstevel@tonic-gate  * Prints warning if zone already exists.
15027c478bd9Sstevel@tonic-gate  * In interactive mode, prompts if we should continue anyway and returns Z_OK
15037c478bd9Sstevel@tonic-gate  * if so, Z_ERR if not.  In non-interactive mode, exits with Z_ERR.
15047c478bd9Sstevel@tonic-gate  *
15057c478bd9Sstevel@tonic-gate  * Note that if a zone exists and its state is >= INSTALLED, an error message
15067c478bd9Sstevel@tonic-gate  * will be printed and this function will return Z_ERR regardless of mode.
15077c478bd9Sstevel@tonic-gate  */
15087c478bd9Sstevel@tonic-gate 
15097c478bd9Sstevel@tonic-gate static int
1510bbec428eSgjelinek check_if_zone_already_exists(boolean_t force)
15117c478bd9Sstevel@tonic-gate {
15127c478bd9Sstevel@tonic-gate 	char line[ZONENAME_MAX + 128];	/* enough to ask a question */
15137c478bd9Sstevel@tonic-gate 	zone_dochandle_t tmphandle;
15147c478bd9Sstevel@tonic-gate 	int res, answer;
15157c478bd9Sstevel@tonic-gate 
15167c478bd9Sstevel@tonic-gate 	if ((tmphandle = zonecfg_init_handle()) == NULL) {
1517bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
15187c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
15197c478bd9Sstevel@tonic-gate 	}
15207c478bd9Sstevel@tonic-gate 	res = zonecfg_get_handle(zone, tmphandle);
15217c478bd9Sstevel@tonic-gate 	zonecfg_fini_handle(tmphandle);
1522087719fdSdp 	if (res != Z_OK)
15237c478bd9Sstevel@tonic-gate 		return (Z_OK);
1524087719fdSdp 
1525087719fdSdp 	if (state_atleast(ZONE_STATE_INSTALLED)) {
15267c478bd9Sstevel@tonic-gate 		zerr(gettext("Zone %s already installed; %s not allowed."),
15277c478bd9Sstevel@tonic-gate 		    zone, cmd_to_str(CMD_CREATE));
15287c478bd9Sstevel@tonic-gate 		return (Z_ERR);
15297c478bd9Sstevel@tonic-gate 	}
15307c478bd9Sstevel@tonic-gate 
15317c478bd9Sstevel@tonic-gate 	if (force) {
15327c478bd9Sstevel@tonic-gate 		(void) printf(gettext("Zone %s already exists; overwriting.\n"),
15337c478bd9Sstevel@tonic-gate 		    zone);
15347c478bd9Sstevel@tonic-gate 		return (Z_OK);
15357c478bd9Sstevel@tonic-gate 	}
15367c478bd9Sstevel@tonic-gate 	(void) snprintf(line, sizeof (line),
15377c478bd9Sstevel@tonic-gate 	    gettext("Zone %s already exists; %s anyway"), zone,
15387c478bd9Sstevel@tonic-gate 	    cmd_to_str(CMD_CREATE));
1539bbec428eSgjelinek 	if ((answer = ask_yesno(B_FALSE, line)) == -1) {
15407c478bd9Sstevel@tonic-gate 		zerr(gettext("Zone exists, input not from terminal and -F not "
15417c478bd9Sstevel@tonic-gate 		    "specified:\n%s command ignored, exiting."),
15427c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_CREATE));
15437c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
15447c478bd9Sstevel@tonic-gate 	}
15457c478bd9Sstevel@tonic-gate 	return (answer == 1 ? Z_OK : Z_ERR);
15467c478bd9Sstevel@tonic-gate }
15477c478bd9Sstevel@tonic-gate 
1548bbec428eSgjelinek static boolean_t
15497c478bd9Sstevel@tonic-gate zone_is_read_only(int cmd_num)
15507c478bd9Sstevel@tonic-gate {
15517c478bd9Sstevel@tonic-gate 	if (strncmp(zone, "SUNW", 4) == 0) {
15527c478bd9Sstevel@tonic-gate 		zerr(gettext("%s: zones beginning with SUNW are read-only."),
15537c478bd9Sstevel@tonic-gate 		    zone);
1554bbec428eSgjelinek 		saw_error = B_TRUE;
1555bbec428eSgjelinek 		return (B_TRUE);
15567c478bd9Sstevel@tonic-gate 	}
15577c478bd9Sstevel@tonic-gate 	if (read_only_mode) {
15587c478bd9Sstevel@tonic-gate 		zerr(gettext("%s: cannot %s in read-only mode."), zone,
15597c478bd9Sstevel@tonic-gate 		    cmd_to_str(cmd_num));
1560bbec428eSgjelinek 		saw_error = B_TRUE;
1561bbec428eSgjelinek 		return (B_TRUE);
15627c478bd9Sstevel@tonic-gate 	}
1563bbec428eSgjelinek 	return (B_FALSE);
15647c478bd9Sstevel@tonic-gate }
15657c478bd9Sstevel@tonic-gate 
15667c478bd9Sstevel@tonic-gate /*
15677c478bd9Sstevel@tonic-gate  * Create a new configuration.
15687c478bd9Sstevel@tonic-gate  */
15697c478bd9Sstevel@tonic-gate void
15707c478bd9Sstevel@tonic-gate create_func(cmd_t *cmd)
15717c478bd9Sstevel@tonic-gate {
15727c478bd9Sstevel@tonic-gate 	int err, arg;
15737c478bd9Sstevel@tonic-gate 	char zone_template[ZONENAME_MAX];
1574ee519a1fSgjelinek 	char attach_path[MAXPATHLEN];
15757c478bd9Sstevel@tonic-gate 	zone_dochandle_t tmphandle;
1576bbec428eSgjelinek 	boolean_t force = B_FALSE;
1577bbec428eSgjelinek 	boolean_t attach = B_FALSE;
1578bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
15797c478bd9Sstevel@tonic-gate 
15807c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
15817c478bd9Sstevel@tonic-gate 
15827c478bd9Sstevel@tonic-gate 	/* This is the default if no arguments are given. */
15837c478bd9Sstevel@tonic-gate 	(void) strlcpy(zone_template, "SUNWdefault", sizeof (zone_template));
15847c478bd9Sstevel@tonic-gate 
15857c478bd9Sstevel@tonic-gate 	optind = 0;
15869acbbeafSnn35248 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?a:bFt:"))
15879acbbeafSnn35248 	    != EOF) {
15887c478bd9Sstevel@tonic-gate 		switch (arg) {
15897c478bd9Sstevel@tonic-gate 		case '?':
15907c478bd9Sstevel@tonic-gate 			if (optopt == '?')
15917c478bd9Sstevel@tonic-gate 				longer_usage(CMD_CREATE);
15927c478bd9Sstevel@tonic-gate 			else
15937c478bd9Sstevel@tonic-gate 				short_usage(CMD_CREATE);
1594bbec428eSgjelinek 			arg_err = B_TRUE;
15957ec75eb8Sgjelinek 			break;
1596ee519a1fSgjelinek 		case 'a':
1597ee519a1fSgjelinek 			(void) strlcpy(attach_path, optarg,
1598ee519a1fSgjelinek 			    sizeof (attach_path));
1599bbec428eSgjelinek 			attach = B_TRUE;
1600ee519a1fSgjelinek 			break;
16017c478bd9Sstevel@tonic-gate 		case 'b':
16027c478bd9Sstevel@tonic-gate 			(void) strlcpy(zone_template, "SUNWblank",
16037c478bd9Sstevel@tonic-gate 			    sizeof (zone_template));
16047c478bd9Sstevel@tonic-gate 			break;
16057c478bd9Sstevel@tonic-gate 		case 'F':
1606bbec428eSgjelinek 			force = B_TRUE;
16077c478bd9Sstevel@tonic-gate 			break;
16087c478bd9Sstevel@tonic-gate 		case 't':
16097c478bd9Sstevel@tonic-gate 			(void) strlcpy(zone_template, optarg,
16107c478bd9Sstevel@tonic-gate 			    sizeof (zone_template));
16117c478bd9Sstevel@tonic-gate 			break;
16127c478bd9Sstevel@tonic-gate 		default:
16137c478bd9Sstevel@tonic-gate 			short_usage(CMD_CREATE);
1614bbec428eSgjelinek 			arg_err = B_TRUE;
16157ec75eb8Sgjelinek 			break;
16167ec75eb8Sgjelinek 		}
16177ec75eb8Sgjelinek 	}
16187ec75eb8Sgjelinek 	if (arg_err)
16197c478bd9Sstevel@tonic-gate 		return;
16207ec75eb8Sgjelinek 
16217c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
16227c478bd9Sstevel@tonic-gate 		short_usage(CMD_CREATE);
16237c478bd9Sstevel@tonic-gate 		return;
16247c478bd9Sstevel@tonic-gate 	}
16257c478bd9Sstevel@tonic-gate 
16267c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_CREATE))
16277c478bd9Sstevel@tonic-gate 		return;
16287c478bd9Sstevel@tonic-gate 
16297c478bd9Sstevel@tonic-gate 	if (check_if_zone_already_exists(force) != Z_OK)
16307c478bd9Sstevel@tonic-gate 		return;
16317c478bd9Sstevel@tonic-gate 
16327c478bd9Sstevel@tonic-gate 	/*
16337c478bd9Sstevel@tonic-gate 	 * Get a temporary handle first.  If that fails, the old handle
16347c478bd9Sstevel@tonic-gate 	 * will not be lost.  Then finish whichever one we don't need,
16357c478bd9Sstevel@tonic-gate 	 * to avoid leaks.  Then get the handle for zone_template, and
16367c478bd9Sstevel@tonic-gate 	 * set the name to zone: this "copy, rename" method is how
16377c478bd9Sstevel@tonic-gate 	 * create -[b|t] works.
16387c478bd9Sstevel@tonic-gate 	 */
16397c478bd9Sstevel@tonic-gate 	if ((tmphandle = zonecfg_init_handle()) == NULL) {
1640bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
16417c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
16427c478bd9Sstevel@tonic-gate 	}
1643ee519a1fSgjelinek 
1644ee519a1fSgjelinek 	if (attach)
164516ab8c7bSgjelinek 		err = zonecfg_get_attach_handle(attach_path, ZONE_DETACHED,
164616ab8c7bSgjelinek 		    zone, B_FALSE, tmphandle);
1647ee519a1fSgjelinek 	else
1648ee519a1fSgjelinek 		err = zonecfg_get_template_handle(zone_template, zone,
1649ee519a1fSgjelinek 		    tmphandle);
1650ee519a1fSgjelinek 
1651ee519a1fSgjelinek 	if (err != Z_OK) {
16527c478bd9Sstevel@tonic-gate 		zonecfg_fini_handle(tmphandle);
1653ee519a1fSgjelinek 		if (attach && err == Z_NO_ZONE)
1654ee519a1fSgjelinek 			(void) fprintf(stderr, gettext("invalid path to "
1655ee519a1fSgjelinek 			    "detached zone\n"));
1656ee519a1fSgjelinek 		else if (attach && err == Z_INVALID_DOCUMENT)
1657ee519a1fSgjelinek 			(void) fprintf(stderr, gettext("Cannot attach to an "
1658ee519a1fSgjelinek 			    "earlier release of the operating system\n"));
1659ee519a1fSgjelinek 		else
1660bbec428eSgjelinek 			zone_perror(zone_template, err, B_TRUE);
16617c478bd9Sstevel@tonic-gate 		return;
16627c478bd9Sstevel@tonic-gate 	}
1663087719fdSdp 
1664bbec428eSgjelinek 	need_to_commit = B_TRUE;
16657c478bd9Sstevel@tonic-gate 	zonecfg_fini_handle(handle);
16667c478bd9Sstevel@tonic-gate 	handle = tmphandle;
1667bbec428eSgjelinek 	got_handle = B_TRUE;
16687c478bd9Sstevel@tonic-gate }
16697c478bd9Sstevel@tonic-gate 
16707c478bd9Sstevel@tonic-gate /*
16717c478bd9Sstevel@tonic-gate  * This malloc()'s memory, which must be freed by the caller.
16727c478bd9Sstevel@tonic-gate  */
16737c478bd9Sstevel@tonic-gate static char *
16747c478bd9Sstevel@tonic-gate quoteit(char *instr)
16757c478bd9Sstevel@tonic-gate {
16767c478bd9Sstevel@tonic-gate 	char *outstr;
16777c478bd9Sstevel@tonic-gate 	size_t outstrsize = strlen(instr) + 3;	/* 2 quotes + '\0' */
16787c478bd9Sstevel@tonic-gate 
16797c478bd9Sstevel@tonic-gate 	if ((outstr = malloc(outstrsize)) == NULL) {
1680bbec428eSgjelinek 		zone_perror(zone, Z_NOMEM, B_FALSE);
16817c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
16827c478bd9Sstevel@tonic-gate 	}
16837c478bd9Sstevel@tonic-gate 	if (strchr(instr, ' ') == NULL) {
16847c478bd9Sstevel@tonic-gate 		(void) strlcpy(outstr, instr, outstrsize);
16857c478bd9Sstevel@tonic-gate 		return (outstr);
16867c478bd9Sstevel@tonic-gate 	}
16877c478bd9Sstevel@tonic-gate 	(void) snprintf(outstr, outstrsize, "\"%s\"", instr);
16887c478bd9Sstevel@tonic-gate 	return (outstr);
16897c478bd9Sstevel@tonic-gate }
16907c478bd9Sstevel@tonic-gate 
16917c478bd9Sstevel@tonic-gate static void
16927c478bd9Sstevel@tonic-gate export_prop(FILE *of, int prop_num, char *prop_id)
16937c478bd9Sstevel@tonic-gate {
16947c478bd9Sstevel@tonic-gate 	char *quote_str;
16957c478bd9Sstevel@tonic-gate 
16967c478bd9Sstevel@tonic-gate 	if (strlen(prop_id) == 0)
16977c478bd9Sstevel@tonic-gate 		return;
16987c478bd9Sstevel@tonic-gate 	quote_str = quoteit(prop_id);
16997c478bd9Sstevel@tonic-gate 	(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
17007c478bd9Sstevel@tonic-gate 	    pt_to_str(prop_num), quote_str);
17017c478bd9Sstevel@tonic-gate 	free(quote_str);
17027c478bd9Sstevel@tonic-gate }
17037c478bd9Sstevel@tonic-gate 
17047c478bd9Sstevel@tonic-gate void
17057c478bd9Sstevel@tonic-gate export_func(cmd_t *cmd)
17067c478bd9Sstevel@tonic-gate {
17077c478bd9Sstevel@tonic-gate 	struct zone_nwiftab nwiftab;
17087c478bd9Sstevel@tonic-gate 	struct zone_fstab fstab;
17097c478bd9Sstevel@tonic-gate 	struct zone_devtab devtab;
17107c478bd9Sstevel@tonic-gate 	struct zone_attrtab attrtab;
17117c478bd9Sstevel@tonic-gate 	struct zone_rctltab rctltab;
1712fa9e4066Sahrens 	struct zone_dstab dstab;
17130209230bSgjelinek 	struct zone_psettab psettab;
17140209230bSgjelinek 	struct zone_mcaptab mcaptab;
17157c478bd9Sstevel@tonic-gate 	struct zone_rctlvaltab *valptr;
1716cb8a054bSGlenn Faden 	struct zone_admintab admintab;
17177c478bd9Sstevel@tonic-gate 	int err, arg;
17187c478bd9Sstevel@tonic-gate 	char zonepath[MAXPATHLEN], outfile[MAXPATHLEN], pool[MAXNAMELEN];
17193f2f09c1Sdp 	char bootargs[BOOTARGS_MAX];
17200209230bSgjelinek 	char sched[MAXNAMELEN];
17219acbbeafSnn35248 	char brand[MAXNAMELEN];
17225679c89fSjv227347 	char hostidp[HW_HOSTID_LEN];
17230fbb751dSJohn Levon 	char fsallowedp[ZONE_FS_ALLOWED_MAX];
1724ffbafc53Scomay 	char *limitpriv;
17257c478bd9Sstevel@tonic-gate 	FILE *of;
17267c478bd9Sstevel@tonic-gate 	boolean_t autoboot;
1727f4b3ec61Sdh155122 	zone_iptype_t iptype;
1728bbec428eSgjelinek 	boolean_t need_to_close = B_FALSE;
1729bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
17307c478bd9Sstevel@tonic-gate 
17317c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
17327c478bd9Sstevel@tonic-gate 
17337c478bd9Sstevel@tonic-gate 	outfile[0] = '\0';
17347c478bd9Sstevel@tonic-gate 	optind = 0;
17357c478bd9Sstevel@tonic-gate 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?f:")) != EOF) {
17367c478bd9Sstevel@tonic-gate 		switch (arg) {
17377c478bd9Sstevel@tonic-gate 		case '?':
17387c478bd9Sstevel@tonic-gate 			if (optopt == '?')
17397c478bd9Sstevel@tonic-gate 				longer_usage(CMD_EXPORT);
17407c478bd9Sstevel@tonic-gate 			else
17417c478bd9Sstevel@tonic-gate 				short_usage(CMD_EXPORT);
1742bbec428eSgjelinek 			arg_err = B_TRUE;
17437ec75eb8Sgjelinek 			break;
17447c478bd9Sstevel@tonic-gate 		case 'f':
17457c478bd9Sstevel@tonic-gate 			(void) strlcpy(outfile, optarg, sizeof (outfile));
17467c478bd9Sstevel@tonic-gate 			break;
17477c478bd9Sstevel@tonic-gate 		default:
17487c478bd9Sstevel@tonic-gate 			short_usage(CMD_EXPORT);
1749bbec428eSgjelinek 			arg_err = B_TRUE;
17507ec75eb8Sgjelinek 			break;
17517ec75eb8Sgjelinek 		}
17527ec75eb8Sgjelinek 	}
17537ec75eb8Sgjelinek 	if (arg_err)
17547c478bd9Sstevel@tonic-gate 		return;
17557ec75eb8Sgjelinek 
17567c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
17577c478bd9Sstevel@tonic-gate 		short_usage(CMD_EXPORT);
17587c478bd9Sstevel@tonic-gate 		return;
17597c478bd9Sstevel@tonic-gate 	}
17607c478bd9Sstevel@tonic-gate 	if (strlen(outfile) == 0) {
17617c478bd9Sstevel@tonic-gate 		of = stdout;
17627c478bd9Sstevel@tonic-gate 	} else {
17637c478bd9Sstevel@tonic-gate 		if ((of = fopen(outfile, "w")) == NULL) {
17647c478bd9Sstevel@tonic-gate 			zerr(gettext("opening file %s: %s"),
17657c478bd9Sstevel@tonic-gate 			    outfile, strerror(errno));
17667c478bd9Sstevel@tonic-gate 			goto done;
17677c478bd9Sstevel@tonic-gate 		}
17687c478bd9Sstevel@tonic-gate 		setbuf(of, NULL);
1769bbec428eSgjelinek 		need_to_close = B_TRUE;
17707c478bd9Sstevel@tonic-gate 	}
17717c478bd9Sstevel@tonic-gate 
1772bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
17737c478bd9Sstevel@tonic-gate 		goto done;
17747c478bd9Sstevel@tonic-gate 
17757c478bd9Sstevel@tonic-gate 	(void) fprintf(of, "%s -b\n", cmd_to_str(CMD_CREATE));
17767c478bd9Sstevel@tonic-gate 
17777c478bd9Sstevel@tonic-gate 	if (zonecfg_get_zonepath(handle, zonepath, sizeof (zonepath)) == Z_OK &&
17787c478bd9Sstevel@tonic-gate 	    strlen(zonepath) > 0)
17797c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
17807c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_ZONEPATH), zonepath);
17817c478bd9Sstevel@tonic-gate 
17829acbbeafSnn35248 	if ((zone_get_brand(zone, brand, sizeof (brand)) == Z_OK) &&
17839acbbeafSnn35248 	    (strcmp(brand, NATIVE_BRAND_NAME) != 0))
17849acbbeafSnn35248 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
17859acbbeafSnn35248 		    pt_to_str(PT_BRAND), brand);
17869acbbeafSnn35248 
17877c478bd9Sstevel@tonic-gate 	if (zonecfg_get_autoboot(handle, &autoboot) == Z_OK)
17887c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
17897c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_AUTOBOOT), autoboot ? "true" : "false");
17907c478bd9Sstevel@tonic-gate 
17913f2f09c1Sdp 	if (zonecfg_get_bootargs(handle, bootargs, sizeof (bootargs)) == Z_OK &&
17923f2f09c1Sdp 	    strlen(bootargs) > 0) {
17933f2f09c1Sdp 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
17943f2f09c1Sdp 		    pt_to_str(PT_BOOTARGS), bootargs);
17953f2f09c1Sdp 	}
17963f2f09c1Sdp 
17977c478bd9Sstevel@tonic-gate 	if (zonecfg_get_pool(handle, pool, sizeof (pool)) == Z_OK &&
17987c478bd9Sstevel@tonic-gate 	    strlen(pool) > 0)
17997c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
18007c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_POOL), pool);
18017c478bd9Sstevel@tonic-gate 
1802ffbafc53Scomay 	if (zonecfg_get_limitpriv(handle, &limitpriv) == Z_OK &&
1803ffbafc53Scomay 	    strlen(limitpriv) > 0) {
1804ffbafc53Scomay 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
1805ffbafc53Scomay 		    pt_to_str(PT_LIMITPRIV), limitpriv);
1806ffbafc53Scomay 		free(limitpriv);
1807ffbafc53Scomay 	}
1808ffbafc53Scomay 
18090209230bSgjelinek 	if (zonecfg_get_sched_class(handle, sched, sizeof (sched)) == Z_OK &&
18100209230bSgjelinek 	    strlen(sched) > 0)
18110209230bSgjelinek 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
18120209230bSgjelinek 		    pt_to_str(PT_SCHED), sched);
18133f2f09c1Sdp 
1814f4b3ec61Sdh155122 	if (zonecfg_get_iptype(handle, &iptype) == Z_OK) {
1815f4b3ec61Sdh155122 		switch (iptype) {
1816f4b3ec61Sdh155122 		case ZS_SHARED:
1817f4b3ec61Sdh155122 			(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
1818f4b3ec61Sdh155122 			    pt_to_str(PT_IPTYPE), "shared");
1819f4b3ec61Sdh155122 			break;
1820f4b3ec61Sdh155122 		case ZS_EXCLUSIVE:
1821f4b3ec61Sdh155122 			(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
1822f4b3ec61Sdh155122 			    pt_to_str(PT_IPTYPE), "exclusive");
1823f4b3ec61Sdh155122 			break;
1824f4b3ec61Sdh155122 		}
1825f4b3ec61Sdh155122 	}
1826f4b3ec61Sdh155122 
18275679c89fSjv227347 	if (zonecfg_get_hostid(handle, hostidp, sizeof (hostidp)) == Z_OK) {
18285679c89fSjv227347 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
18295679c89fSjv227347 		    pt_to_str(PT_HOSTID), hostidp);
18305679c89fSjv227347 	}
18315679c89fSjv227347 
18320fbb751dSJohn Levon 	if (zonecfg_get_fs_allowed(handle, fsallowedp,
18330fbb751dSJohn Levon 	    sizeof (fsallowedp)) == Z_OK) {
18340fbb751dSJohn Levon 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
18350fbb751dSJohn Levon 		    pt_to_str(PT_FS_ALLOWED), fsallowedp);
18360fbb751dSJohn Levon 	}
18370fbb751dSJohn Levon 
18387c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setfsent(handle)) != Z_OK) {
1839bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
18407c478bd9Sstevel@tonic-gate 		goto done;
18417c478bd9Sstevel@tonic-gate 	}
18427c478bd9Sstevel@tonic-gate 	while (zonecfg_getfsent(handle, &fstab) == Z_OK) {
18437c478bd9Sstevel@tonic-gate 		zone_fsopt_t *optptr;
18447c478bd9Sstevel@tonic-gate 
18457c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
18467c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_FS));
18477c478bd9Sstevel@tonic-gate 		export_prop(of, PT_DIR, fstab.zone_fs_dir);
18487c478bd9Sstevel@tonic-gate 		export_prop(of, PT_SPECIAL, fstab.zone_fs_special);
18497c478bd9Sstevel@tonic-gate 		export_prop(of, PT_RAW, fstab.zone_fs_raw);
18507c478bd9Sstevel@tonic-gate 		export_prop(of, PT_TYPE, fstab.zone_fs_type);
18517c478bd9Sstevel@tonic-gate 		for (optptr = fstab.zone_fs_options; optptr != NULL;
18527c478bd9Sstevel@tonic-gate 		    optptr = optptr->zone_fsopt_next) {
18537c478bd9Sstevel@tonic-gate 			/*
18547c478bd9Sstevel@tonic-gate 			 * Simple property values with embedded equal signs
18557c478bd9Sstevel@tonic-gate 			 * need to be quoted to prevent the lexer from
18567c478bd9Sstevel@tonic-gate 			 * mis-parsing them as complex name=value pairs.
18577c478bd9Sstevel@tonic-gate 			 */
18587c478bd9Sstevel@tonic-gate 			if (strchr(optptr->zone_fsopt_opt, '='))
18597c478bd9Sstevel@tonic-gate 				(void) fprintf(of, "%s %s \"%s\"\n",
18607c478bd9Sstevel@tonic-gate 				    cmd_to_str(CMD_ADD),
18617c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_OPTIONS),
18627c478bd9Sstevel@tonic-gate 				    optptr->zone_fsopt_opt);
18637c478bd9Sstevel@tonic-gate 			else
18647c478bd9Sstevel@tonic-gate 				(void) fprintf(of, "%s %s %s\n",
18657c478bd9Sstevel@tonic-gate 				    cmd_to_str(CMD_ADD),
18667c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_OPTIONS),
18677c478bd9Sstevel@tonic-gate 				    optptr->zone_fsopt_opt);
18687c478bd9Sstevel@tonic-gate 		}
18697c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
18707c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(fstab.zone_fs_options);
18717c478bd9Sstevel@tonic-gate 	}
18727c478bd9Sstevel@tonic-gate 	(void) zonecfg_endfsent(handle);
18737c478bd9Sstevel@tonic-gate 
18747c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setnwifent(handle)) != Z_OK) {
1875bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
18767c478bd9Sstevel@tonic-gate 		goto done;
18777c478bd9Sstevel@tonic-gate 	}
18787c478bd9Sstevel@tonic-gate 	while (zonecfg_getnwifent(handle, &nwiftab) == Z_OK) {
18797c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
18807c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_NET));
18817c478bd9Sstevel@tonic-gate 		export_prop(of, PT_ADDRESS, nwiftab.zone_nwif_address);
1882550b6e40SSowmini Varadhan 		export_prop(of, PT_ALLOWED_ADDRESS,
1883550b6e40SSowmini Varadhan 		    nwiftab.zone_nwif_allowed_address);
18847c478bd9Sstevel@tonic-gate 		export_prop(of, PT_PHYSICAL, nwiftab.zone_nwif_physical);
1885de860bd9Sgfaden 		export_prop(of, PT_DEFROUTER, nwiftab.zone_nwif_defrouter);
18867c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
18877c478bd9Sstevel@tonic-gate 	}
18887c478bd9Sstevel@tonic-gate 	(void) zonecfg_endnwifent(handle);
18897c478bd9Sstevel@tonic-gate 
18907c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setdevent(handle)) != Z_OK) {
1891bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
18927c478bd9Sstevel@tonic-gate 		goto done;
18937c478bd9Sstevel@tonic-gate 	}
18947c478bd9Sstevel@tonic-gate 	while (zonecfg_getdevent(handle, &devtab) == Z_OK) {
18957c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
18967c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_DEVICE));
18977c478bd9Sstevel@tonic-gate 		export_prop(of, PT_MATCH, devtab.zone_dev_match);
18987c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
18997c478bd9Sstevel@tonic-gate 	}
19007c478bd9Sstevel@tonic-gate 	(void) zonecfg_enddevent(handle);
19017c478bd9Sstevel@tonic-gate 
19021b3281c0SGerald Jelinek 	if (zonecfg_getmcapent(handle, &mcaptab) == Z_OK) {
19031b3281c0SGerald Jelinek 		char buf[128];
19041b3281c0SGerald Jelinek 
19051b3281c0SGerald Jelinek 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
19061b3281c0SGerald Jelinek 		    rt_to_str(RT_MCAP));
19071b3281c0SGerald Jelinek 		bytes_to_units(mcaptab.zone_physmem_cap, buf, sizeof (buf));
19081b3281c0SGerald Jelinek 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19091b3281c0SGerald Jelinek 		    pt_to_str(PT_PHYSICAL), buf);
19101b3281c0SGerald Jelinek 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
19111b3281c0SGerald Jelinek 	}
19121b3281c0SGerald Jelinek 
19137c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setrctlent(handle)) != Z_OK) {
1914bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
19157c478bd9Sstevel@tonic-gate 		goto done;
19167c478bd9Sstevel@tonic-gate 	}
19177c478bd9Sstevel@tonic-gate 	while (zonecfg_getrctlent(handle, &rctltab) == Z_OK) {
19187c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s rctl\n", cmd_to_str(CMD_ADD));
19197c478bd9Sstevel@tonic-gate 		export_prop(of, PT_NAME, rctltab.zone_rctl_name);
19207c478bd9Sstevel@tonic-gate 		for (valptr = rctltab.zone_rctl_valptr; valptr != NULL;
19217c478bd9Sstevel@tonic-gate 		    valptr = valptr->zone_rctlval_next) {
19227c478bd9Sstevel@tonic-gate 			fprintf(of, "%s %s (%s=%s,%s=%s,%s=%s)\n",
19237c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_ADD), pt_to_str(PT_VALUE),
19247c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_PRIV), valptr->zone_rctlval_priv,
19257c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_LIMIT), valptr->zone_rctlval_limit,
19267c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_ACTION), valptr->zone_rctlval_action);
19277c478bd9Sstevel@tonic-gate 		}
19287c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
19297c478bd9Sstevel@tonic-gate 		zonecfg_free_rctl_value_list(rctltab.zone_rctl_valptr);
19307c478bd9Sstevel@tonic-gate 	}
19317c478bd9Sstevel@tonic-gate 	(void) zonecfg_endrctlent(handle);
19327c478bd9Sstevel@tonic-gate 
19337c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setattrent(handle)) != Z_OK) {
1934bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
19357c478bd9Sstevel@tonic-gate 		goto done;
19367c478bd9Sstevel@tonic-gate 	}
19377c478bd9Sstevel@tonic-gate 	while (zonecfg_getattrent(handle, &attrtab) == Z_OK) {
19387c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
19397c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR));
19407c478bd9Sstevel@tonic-gate 		export_prop(of, PT_NAME, attrtab.zone_attr_name);
19417c478bd9Sstevel@tonic-gate 		export_prop(of, PT_TYPE, attrtab.zone_attr_type);
19427c478bd9Sstevel@tonic-gate 		export_prop(of, PT_VALUE, attrtab.zone_attr_value);
19437c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
19447c478bd9Sstevel@tonic-gate 	}
19457c478bd9Sstevel@tonic-gate 	(void) zonecfg_endattrent(handle);
19467c478bd9Sstevel@tonic-gate 
1947fa9e4066Sahrens 	if ((err = zonecfg_setdsent(handle)) != Z_OK) {
1948bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
1949fa9e4066Sahrens 		goto done;
1950fa9e4066Sahrens 	}
1951fa9e4066Sahrens 	while (zonecfg_getdsent(handle, &dstab) == Z_OK) {
1952fa9e4066Sahrens 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
1953fa9e4066Sahrens 		    rt_to_str(RT_DATASET));
1954fa9e4066Sahrens 		export_prop(of, PT_NAME, dstab.zone_dataset_name);
1955fa9e4066Sahrens 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
1956fa9e4066Sahrens 	}
1957fa9e4066Sahrens 	(void) zonecfg_enddsent(handle);
1958fa9e4066Sahrens 
19590209230bSgjelinek 	if (zonecfg_getpsetent(handle, &psettab) == Z_OK) {
19600209230bSgjelinek 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
19610209230bSgjelinek 		    rt_to_str(RT_DCPU));
19620209230bSgjelinek 		if (strcmp(psettab.zone_ncpu_min, psettab.zone_ncpu_max) == 0)
19630209230bSgjelinek 			(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19640209230bSgjelinek 			    pt_to_str(PT_NCPUS), psettab.zone_ncpu_max);
19650209230bSgjelinek 		else
19660209230bSgjelinek 			(void) fprintf(of, "%s %s=%s-%s\n", cmd_to_str(CMD_SET),
19670209230bSgjelinek 			    pt_to_str(PT_NCPUS), psettab.zone_ncpu_min,
19680209230bSgjelinek 			    psettab.zone_ncpu_max);
19690209230bSgjelinek 		if (psettab.zone_importance[0] != '\0')
19700209230bSgjelinek 			(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19710209230bSgjelinek 			    pt_to_str(PT_IMPORTANCE), psettab.zone_importance);
19720209230bSgjelinek 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
19730209230bSgjelinek 	}
19740209230bSgjelinek 
1975cb8a054bSGlenn Faden 	if ((err = zonecfg_setadminent(handle)) != Z_OK) {
1976cb8a054bSGlenn Faden 		zone_perror(zone, err, B_FALSE);
1977cb8a054bSGlenn Faden 		goto done;
1978cb8a054bSGlenn Faden 	}
1979cb8a054bSGlenn Faden 	while (zonecfg_getadminent(handle, &admintab) == Z_OK) {
1980cb8a054bSGlenn Faden 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
1981cb8a054bSGlenn Faden 		    rt_to_str(RT_ADMIN));
1982cb8a054bSGlenn Faden 		export_prop(of, PT_USER, admintab.zone_admin_user);
1983cb8a054bSGlenn Faden 		export_prop(of, PT_AUTHS, admintab.zone_admin_auths);
1984cb8a054bSGlenn Faden 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
1985cb8a054bSGlenn Faden 	}
1986cb8a054bSGlenn Faden 	(void) zonecfg_endadminent(handle);
1987cb8a054bSGlenn Faden 
1988c97ad5cdSakolb 	/*
1989c97ad5cdSakolb 	 * There is nothing to export for pcap since this resource is just
1990c97ad5cdSakolb 	 * a container for an rctl alias.
1991c97ad5cdSakolb 	 */
1992c97ad5cdSakolb 
19937c478bd9Sstevel@tonic-gate done:
19947c478bd9Sstevel@tonic-gate 	if (need_to_close)
19957c478bd9Sstevel@tonic-gate 		(void) fclose(of);
19967c478bd9Sstevel@tonic-gate }
19977c478bd9Sstevel@tonic-gate 
19987c478bd9Sstevel@tonic-gate void
19997c478bd9Sstevel@tonic-gate exit_func(cmd_t *cmd)
20007c478bd9Sstevel@tonic-gate {
20017c478bd9Sstevel@tonic-gate 	int arg, answer;
2002bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
20037c478bd9Sstevel@tonic-gate 
20047c478bd9Sstevel@tonic-gate 	optind = 0;
20057c478bd9Sstevel@tonic-gate 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?F")) != EOF) {
20067c478bd9Sstevel@tonic-gate 		switch (arg) {
20077c478bd9Sstevel@tonic-gate 		case '?':
20087c478bd9Sstevel@tonic-gate 			longer_usage(CMD_EXIT);
2009bbec428eSgjelinek 			arg_err = B_TRUE;
20107ec75eb8Sgjelinek 			break;
20117c478bd9Sstevel@tonic-gate 		case 'F':
2012bbec428eSgjelinek 			force_exit = B_TRUE;
20137c478bd9Sstevel@tonic-gate 			break;
20147c478bd9Sstevel@tonic-gate 		default:
20157c478bd9Sstevel@tonic-gate 			short_usage(CMD_EXIT);
2016bbec428eSgjelinek 			arg_err = B_TRUE;
20177ec75eb8Sgjelinek 			break;
20187ec75eb8Sgjelinek 		}
20197ec75eb8Sgjelinek 	}
20207ec75eb8Sgjelinek 	if (arg_err)
20217c478bd9Sstevel@tonic-gate 		return;
20227ec75eb8Sgjelinek 
20237c478bd9Sstevel@tonic-gate 	if (optind < cmd->cmd_argc) {
20247c478bd9Sstevel@tonic-gate 		short_usage(CMD_EXIT);
20257c478bd9Sstevel@tonic-gate 		return;
20267c478bd9Sstevel@tonic-gate 	}
20277c478bd9Sstevel@tonic-gate 
20287c478bd9Sstevel@tonic-gate 	if (global_scope || force_exit) {
2029bbec428eSgjelinek 		time_to_exit = B_TRUE;
20307c478bd9Sstevel@tonic-gate 		return;
20317c478bd9Sstevel@tonic-gate 	}
20327c478bd9Sstevel@tonic-gate 
2033bbec428eSgjelinek 	answer = ask_yesno(B_FALSE, "Resource incomplete; really quit");
20347c478bd9Sstevel@tonic-gate 	if (answer == -1) {
20357c478bd9Sstevel@tonic-gate 		zerr(gettext("Resource incomplete, input "
20367c478bd9Sstevel@tonic-gate 		    "not from terminal and -F not specified:\n%s command "
20377c478bd9Sstevel@tonic-gate 		    "ignored, but exiting anyway."), cmd_to_str(CMD_EXIT));
20387c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
20397c478bd9Sstevel@tonic-gate 	} else if (answer == 1) {
2040bbec428eSgjelinek 		time_to_exit = B_TRUE;
20417c478bd9Sstevel@tonic-gate 	}
20427c478bd9Sstevel@tonic-gate 	/* (answer == 0) => just return */
20437c478bd9Sstevel@tonic-gate }
20447c478bd9Sstevel@tonic-gate 
20457c478bd9Sstevel@tonic-gate static int
20467c478bd9Sstevel@tonic-gate validate_zonepath_syntax(char *path)
20477c478bd9Sstevel@tonic-gate {
20487c478bd9Sstevel@tonic-gate 	if (path[0] != '/') {
20497c478bd9Sstevel@tonic-gate 		zerr(gettext("%s is not an absolute path."), path);
20507c478bd9Sstevel@tonic-gate 		return (Z_ERR);
20517c478bd9Sstevel@tonic-gate 	}
20526fb06a2bSSusan Kamm-Worrell 	/* If path is all slashes, then fail */
20536fb06a2bSSusan Kamm-Worrell 	if (strspn(path, "/") == strlen(path)) {
20547c478bd9Sstevel@tonic-gate 		zerr(gettext("/ is not allowed as a %s."),
20557c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_ZONEPATH));
20567c478bd9Sstevel@tonic-gate 		return (Z_ERR);
20577c478bd9Sstevel@tonic-gate 	}
20587c478bd9Sstevel@tonic-gate 	return (Z_OK);
20597c478bd9Sstevel@tonic-gate }
20607c478bd9Sstevel@tonic-gate 
20617c478bd9Sstevel@tonic-gate static void
20627c478bd9Sstevel@tonic-gate add_resource(cmd_t *cmd)
20637c478bd9Sstevel@tonic-gate {
20647c478bd9Sstevel@tonic-gate 	int type;
20650209230bSgjelinek 	struct zone_psettab tmp_psettab;
20660209230bSgjelinek 	struct zone_mcaptab tmp_mcaptab;
2067c97ad5cdSakolb 	uint64_t tmp;
20680209230bSgjelinek 	uint64_t tmp_mcap;
20690209230bSgjelinek 	char pool[MAXNAMELEN];
20707c478bd9Sstevel@tonic-gate 
20717c478bd9Sstevel@tonic-gate 	if ((type = cmd->cmd_res_type) == RT_UNKNOWN) {
2072bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
20737c478bd9Sstevel@tonic-gate 		goto bad;
20747c478bd9Sstevel@tonic-gate 	}
20757c478bd9Sstevel@tonic-gate 
20767c478bd9Sstevel@tonic-gate 	switch (type) {
20777c478bd9Sstevel@tonic-gate 	case RT_FS:
20787c478bd9Sstevel@tonic-gate 		bzero(&in_progress_fstab, sizeof (in_progress_fstab));
20797c478bd9Sstevel@tonic-gate 		return;
20807c478bd9Sstevel@tonic-gate 	case RT_NET:
20817c478bd9Sstevel@tonic-gate 		bzero(&in_progress_nwiftab, sizeof (in_progress_nwiftab));
20827c478bd9Sstevel@tonic-gate 		return;
20837c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
20847c478bd9Sstevel@tonic-gate 		bzero(&in_progress_devtab, sizeof (in_progress_devtab));
20857c478bd9Sstevel@tonic-gate 		return;
20867c478bd9Sstevel@tonic-gate 	case RT_RCTL:
20870209230bSgjelinek 		if (global_zone)
20880209230bSgjelinek 			zerr(gettext("WARNING: Setting a global zone resource "
20890209230bSgjelinek 			    "control too low could deny\nservice "
20900209230bSgjelinek 			    "to even the root user; "
20910209230bSgjelinek 			    "this could render the system impossible\n"
20920209230bSgjelinek 			    "to administer.  Please use caution."));
20937c478bd9Sstevel@tonic-gate 		bzero(&in_progress_rctltab, sizeof (in_progress_rctltab));
20947c478bd9Sstevel@tonic-gate 		return;
20957c478bd9Sstevel@tonic-gate 	case RT_ATTR:
20967c478bd9Sstevel@tonic-gate 		bzero(&in_progress_attrtab, sizeof (in_progress_attrtab));
20977c478bd9Sstevel@tonic-gate 		return;
2098fa9e4066Sahrens 	case RT_DATASET:
2099fa9e4066Sahrens 		bzero(&in_progress_dstab, sizeof (in_progress_dstab));
2100fa9e4066Sahrens 		return;
21010209230bSgjelinek 	case RT_DCPU:
2102c97ad5cdSakolb 		/* Make sure there isn't already a cpu-set or cpu-cap entry. */
21030209230bSgjelinek 		if (zonecfg_lookup_pset(handle, &tmp_psettab) == Z_OK) {
21040209230bSgjelinek 			zerr(gettext("The %s resource already exists."),
21050209230bSgjelinek 			    rt_to_str(RT_DCPU));
21060209230bSgjelinek 			goto bad;
21070209230bSgjelinek 		}
2108c97ad5cdSakolb 		if (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &tmp) !=
2109c97ad5cdSakolb 		    Z_NO_ENTRY) {
2110c97ad5cdSakolb 			zerr(gettext("The %s resource already exists."),
2111c97ad5cdSakolb 			    rt_to_str(RT_PCAP));
2112c97ad5cdSakolb 			goto bad;
2113c97ad5cdSakolb 		}
21140209230bSgjelinek 
21150209230bSgjelinek 		/* Make sure the pool property isn't set. */
21160209230bSgjelinek 		if (zonecfg_get_pool(handle, pool, sizeof (pool)) == Z_OK &&
21170209230bSgjelinek 		    strlen(pool) > 0) {
21180209230bSgjelinek 			zerr(gettext("The %s property is already set.  "
21190209230bSgjelinek 			    "A persistent pool is incompatible with\nthe %s "
21200209230bSgjelinek 			    "resource."),
21210209230bSgjelinek 			    pt_to_str(PT_POOL), rt_to_str(RT_DCPU));
21220209230bSgjelinek 			goto bad;
21230209230bSgjelinek 		}
21240209230bSgjelinek 
21250209230bSgjelinek 		bzero(&in_progress_psettab, sizeof (in_progress_psettab));
21260209230bSgjelinek 		return;
2127c97ad5cdSakolb 	case RT_PCAP:
2128c97ad5cdSakolb 		/*
2129c97ad5cdSakolb 		 * Make sure there isn't already a cpu-set or incompatible
2130c97ad5cdSakolb 		 * cpu-cap rctls.
2131c97ad5cdSakolb 		 */
2132c97ad5cdSakolb 		if (zonecfg_lookup_pset(handle, &tmp_psettab) == Z_OK) {
2133c97ad5cdSakolb 			zerr(gettext("The %s resource already exists."),
2134c97ad5cdSakolb 			    rt_to_str(RT_DCPU));
2135c97ad5cdSakolb 			goto bad;
2136c97ad5cdSakolb 		}
2137c97ad5cdSakolb 
2138c97ad5cdSakolb 		switch (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &tmp)) {
2139c97ad5cdSakolb 		case Z_ALIAS_DISALLOW:
2140c97ad5cdSakolb 			zone_perror(rt_to_str(RT_PCAP), Z_ALIAS_DISALLOW,
2141bbec428eSgjelinek 			    B_FALSE);
2142c97ad5cdSakolb 			goto bad;
2143c97ad5cdSakolb 
2144c97ad5cdSakolb 		case Z_OK:
2145c97ad5cdSakolb 			zerr(gettext("The %s resource already exists."),
2146c97ad5cdSakolb 			    rt_to_str(RT_PCAP));
2147c97ad5cdSakolb 			goto bad;
2148c97ad5cdSakolb 
2149c97ad5cdSakolb 		default:
2150c97ad5cdSakolb 			break;
2151c97ad5cdSakolb 		}
2152c97ad5cdSakolb 		return;
21530209230bSgjelinek 	case RT_MCAP:
21540209230bSgjelinek 		/*
21550209230bSgjelinek 		 * Make sure there isn't already a mem-cap entry or max-swap
21560209230bSgjelinek 		 * or max-locked rctl.
21570209230bSgjelinek 		 */
21580209230bSgjelinek 		if (zonecfg_lookup_mcap(handle, &tmp_mcaptab) == Z_OK ||
21590209230bSgjelinek 		    zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP, &tmp_mcap)
21600209230bSgjelinek 		    == Z_OK ||
21610209230bSgjelinek 		    zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM,
21620209230bSgjelinek 		    &tmp_mcap) == Z_OK) {
21630209230bSgjelinek 			zerr(gettext("The %s resource or a related resource "
21640209230bSgjelinek 			    "control already exists."), rt_to_str(RT_MCAP));
21650209230bSgjelinek 			goto bad;
21660209230bSgjelinek 		}
21670209230bSgjelinek 		if (global_zone)
21680209230bSgjelinek 			zerr(gettext("WARNING: Setting a global zone memory "
21690209230bSgjelinek 			    "cap too low could deny\nservice "
21700209230bSgjelinek 			    "to even the root user; "
21710209230bSgjelinek 			    "this could render the system impossible\n"
21720209230bSgjelinek 			    "to administer.  Please use caution."));
21730209230bSgjelinek 		bzero(&in_progress_mcaptab, sizeof (in_progress_mcaptab));
21740209230bSgjelinek 		return;
2175cb8a054bSGlenn Faden 	case RT_ADMIN:
2176cb8a054bSGlenn Faden 		bzero(&in_progress_admintab, sizeof (in_progress_admintab));
2177cb8a054bSGlenn Faden 		return;
21787c478bd9Sstevel@tonic-gate 	default:
2179bbec428eSgjelinek 		zone_perror(rt_to_str(type), Z_NO_RESOURCE_TYPE, B_TRUE);
2180bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
2181bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
21827c478bd9Sstevel@tonic-gate 	}
21837c478bd9Sstevel@tonic-gate bad:
2184bbec428eSgjelinek 	global_scope = B_TRUE;
21857c478bd9Sstevel@tonic-gate 	end_op = -1;
21867c478bd9Sstevel@tonic-gate }
21877c478bd9Sstevel@tonic-gate 
21887c478bd9Sstevel@tonic-gate static void
21897c478bd9Sstevel@tonic-gate do_complex_rctl_val(complex_property_ptr_t cp)
21907c478bd9Sstevel@tonic-gate {
21917c478bd9Sstevel@tonic-gate 	struct zone_rctlvaltab *rctlvaltab;
21927c478bd9Sstevel@tonic-gate 	complex_property_ptr_t cx;
2193bbec428eSgjelinek 	boolean_t seen_priv = B_FALSE, seen_limit = B_FALSE,
2194bbec428eSgjelinek 	    seen_action = B_FALSE;
21957c478bd9Sstevel@tonic-gate 	rctlblk_t *rctlblk;
21967c478bd9Sstevel@tonic-gate 	int err;
21977c478bd9Sstevel@tonic-gate 
21987c478bd9Sstevel@tonic-gate 	if ((rctlvaltab = alloc_rctlvaltab()) == NULL) {
2199bbec428eSgjelinek 		zone_perror(zone, Z_NOMEM, B_TRUE);
22007c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
22017c478bd9Sstevel@tonic-gate 	}
22027c478bd9Sstevel@tonic-gate 	for (cx = cp; cx != NULL; cx = cx->cp_next) {
22037c478bd9Sstevel@tonic-gate 		switch (cx->cp_type) {
22047c478bd9Sstevel@tonic-gate 		case PT_PRIV:
22057c478bd9Sstevel@tonic-gate 			if (seen_priv) {
22067c478bd9Sstevel@tonic-gate 				zerr(gettext("%s already specified"),
22077c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_PRIV));
22087c478bd9Sstevel@tonic-gate 				goto bad;
22097c478bd9Sstevel@tonic-gate 			}
22107c478bd9Sstevel@tonic-gate 			(void) strlcpy(rctlvaltab->zone_rctlval_priv,
22117c478bd9Sstevel@tonic-gate 			    cx->cp_value,
22127c478bd9Sstevel@tonic-gate 			    sizeof (rctlvaltab->zone_rctlval_priv));
2213bbec428eSgjelinek 			seen_priv = B_TRUE;
22147c478bd9Sstevel@tonic-gate 			break;
22157c478bd9Sstevel@tonic-gate 		case PT_LIMIT:
22167c478bd9Sstevel@tonic-gate 			if (seen_limit) {
22177c478bd9Sstevel@tonic-gate 				zerr(gettext("%s already specified"),
22187c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_LIMIT));
22197c478bd9Sstevel@tonic-gate 				goto bad;
22207c478bd9Sstevel@tonic-gate 			}
22217c478bd9Sstevel@tonic-gate 			(void) strlcpy(rctlvaltab->zone_rctlval_limit,
22227c478bd9Sstevel@tonic-gate 			    cx->cp_value,
22237c478bd9Sstevel@tonic-gate 			    sizeof (rctlvaltab->zone_rctlval_limit));
2224bbec428eSgjelinek 			seen_limit = B_TRUE;
22257c478bd9Sstevel@tonic-gate 			break;
22267c478bd9Sstevel@tonic-gate 		case PT_ACTION:
22277c478bd9Sstevel@tonic-gate 			if (seen_action) {
22287c478bd9Sstevel@tonic-gate 				zerr(gettext("%s already specified"),
22297c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_ACTION));
22307c478bd9Sstevel@tonic-gate 				goto bad;
22317c478bd9Sstevel@tonic-gate 			}
22327c478bd9Sstevel@tonic-gate 			(void) strlcpy(rctlvaltab->zone_rctlval_action,
22337c478bd9Sstevel@tonic-gate 			    cx->cp_value,
22347c478bd9Sstevel@tonic-gate 			    sizeof (rctlvaltab->zone_rctlval_action));
2235bbec428eSgjelinek 			seen_action = B_TRUE;
22367c478bd9Sstevel@tonic-gate 			break;
22377c478bd9Sstevel@tonic-gate 		default:
22387c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(PT_VALUE),
2239bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
2240bbec428eSgjelinek 			long_usage(CMD_ADD, B_TRUE);
2241bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
22427c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(rctlvaltab);
22437c478bd9Sstevel@tonic-gate 			return;
22447c478bd9Sstevel@tonic-gate 		}
22457c478bd9Sstevel@tonic-gate 	}
22467c478bd9Sstevel@tonic-gate 	if (!seen_priv)
22477c478bd9Sstevel@tonic-gate 		zerr(gettext("%s not specified"), pt_to_str(PT_PRIV));
22487c478bd9Sstevel@tonic-gate 	if (!seen_limit)
22497c478bd9Sstevel@tonic-gate 		zerr(gettext("%s not specified"), pt_to_str(PT_LIMIT));
22507c478bd9Sstevel@tonic-gate 	if (!seen_action)
22517c478bd9Sstevel@tonic-gate 		zerr(gettext("%s not specified"), pt_to_str(PT_ACTION));
22527c478bd9Sstevel@tonic-gate 	if (!seen_priv || !seen_limit || !seen_action)
22537c478bd9Sstevel@tonic-gate 		goto bad;
22547c478bd9Sstevel@tonic-gate 	rctlvaltab->zone_rctlval_next = NULL;
22557c478bd9Sstevel@tonic-gate 	rctlblk = alloca(rctlblk_size());
22567c478bd9Sstevel@tonic-gate 	/*
22577c478bd9Sstevel@tonic-gate 	 * Make sure the rctl value looks roughly correct; we won't know if
22587c478bd9Sstevel@tonic-gate 	 * it's truly OK until we verify the configuration on the target
22597c478bd9Sstevel@tonic-gate 	 * system.
22607c478bd9Sstevel@tonic-gate 	 */
22617c478bd9Sstevel@tonic-gate 	if (zonecfg_construct_rctlblk(rctlvaltab, rctlblk) != Z_OK ||
22627c478bd9Sstevel@tonic-gate 	    !zonecfg_valid_rctlblk(rctlblk)) {
22637c478bd9Sstevel@tonic-gate 		zerr(gettext("Invalid %s %s specification"), rt_to_str(RT_RCTL),
22647c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_VALUE));
22657c478bd9Sstevel@tonic-gate 		goto bad;
22667c478bd9Sstevel@tonic-gate 	}
22677c478bd9Sstevel@tonic-gate 	err = zonecfg_add_rctl_value(&in_progress_rctltab, rctlvaltab);
22687c478bd9Sstevel@tonic-gate 	if (err != Z_OK)
2269bbec428eSgjelinek 		zone_perror(pt_to_str(PT_VALUE), err, B_TRUE);
22707c478bd9Sstevel@tonic-gate 	return;
22717c478bd9Sstevel@tonic-gate 
22727c478bd9Sstevel@tonic-gate bad:
22737c478bd9Sstevel@tonic-gate 	zonecfg_free_rctl_value_list(rctlvaltab);
22747c478bd9Sstevel@tonic-gate }
22757c478bd9Sstevel@tonic-gate 
22767c478bd9Sstevel@tonic-gate static void
22777c478bd9Sstevel@tonic-gate add_property(cmd_t *cmd)
22787c478bd9Sstevel@tonic-gate {
22797c478bd9Sstevel@tonic-gate 	char *prop_id;
22807c478bd9Sstevel@tonic-gate 	int err, res_type, prop_type;
22817c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
22827c478bd9Sstevel@tonic-gate 	list_property_ptr_t l;
22837c478bd9Sstevel@tonic-gate 
22847c478bd9Sstevel@tonic-gate 	res_type = resource_scope;
22857c478bd9Sstevel@tonic-gate 	prop_type = cmd->cmd_prop_name[0];
22867c478bd9Sstevel@tonic-gate 	if (res_type == RT_UNKNOWN || prop_type == PT_UNKNOWN) {
2287bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
22887c478bd9Sstevel@tonic-gate 		return;
22897c478bd9Sstevel@tonic-gate 	}
22907c478bd9Sstevel@tonic-gate 
22917c478bd9Sstevel@tonic-gate 	if (cmd->cmd_prop_nv_pairs != 1) {
2292bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
22937c478bd9Sstevel@tonic-gate 		return;
22947c478bd9Sstevel@tonic-gate 	}
22957c478bd9Sstevel@tonic-gate 
2296bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
22977c478bd9Sstevel@tonic-gate 		return;
22987c478bd9Sstevel@tonic-gate 
22997c478bd9Sstevel@tonic-gate 	switch (res_type) {
23007c478bd9Sstevel@tonic-gate 	case RT_FS:
23017c478bd9Sstevel@tonic-gate 		if (prop_type != PT_OPTIONS) {
23027c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
2303bbec428eSgjelinek 			    B_TRUE);
2304bbec428eSgjelinek 			long_usage(CMD_ADD, B_TRUE);
2305bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
23067c478bd9Sstevel@tonic-gate 			return;
23077c478bd9Sstevel@tonic-gate 		}
23087c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[0];
23097c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE &&
23107c478bd9Sstevel@tonic-gate 		    pp->pv_type != PROP_VAL_LIST) {
23117c478bd9Sstevel@tonic-gate 			zerr(gettext("A %s or %s value was expected here."),
23127c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_SIMPLE),
23137c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_LIST));
2314bbec428eSgjelinek 			saw_error = B_TRUE;
23157c478bd9Sstevel@tonic-gate 			return;
23167c478bd9Sstevel@tonic-gate 		}
23177c478bd9Sstevel@tonic-gate 		if (pp->pv_type == PROP_VAL_SIMPLE) {
23187c478bd9Sstevel@tonic-gate 			if (pp->pv_simple == NULL) {
2319bbec428eSgjelinek 				long_usage(CMD_ADD, B_TRUE);
23207c478bd9Sstevel@tonic-gate 				return;
23217c478bd9Sstevel@tonic-gate 			}
23227c478bd9Sstevel@tonic-gate 			prop_id = pp->pv_simple;
23237c478bd9Sstevel@tonic-gate 			err = zonecfg_add_fs_option(&in_progress_fstab,
23247c478bd9Sstevel@tonic-gate 			    prop_id);
23257c478bd9Sstevel@tonic-gate 			if (err != Z_OK)
2326bbec428eSgjelinek 				zone_perror(pt_to_str(prop_type), err, B_TRUE);
23277c478bd9Sstevel@tonic-gate 		} else {
23287c478bd9Sstevel@tonic-gate 			list_property_ptr_t list;
23297c478bd9Sstevel@tonic-gate 
23307c478bd9Sstevel@tonic-gate 			for (list = pp->pv_list; list != NULL;
23317c478bd9Sstevel@tonic-gate 			    list = list->lp_next) {
23327c478bd9Sstevel@tonic-gate 				prop_id = list->lp_simple;
23337c478bd9Sstevel@tonic-gate 				if (prop_id == NULL)
23347c478bd9Sstevel@tonic-gate 					break;
23357c478bd9Sstevel@tonic-gate 				err = zonecfg_add_fs_option(
23367c478bd9Sstevel@tonic-gate 				    &in_progress_fstab, prop_id);
23377c478bd9Sstevel@tonic-gate 				if (err != Z_OK)
23387c478bd9Sstevel@tonic-gate 					zone_perror(pt_to_str(prop_type), err,
2339bbec428eSgjelinek 					    B_TRUE);
23407c478bd9Sstevel@tonic-gate 			}
23417c478bd9Sstevel@tonic-gate 		}
23427c478bd9Sstevel@tonic-gate 		return;
23437c478bd9Sstevel@tonic-gate 	case RT_RCTL:
23447c478bd9Sstevel@tonic-gate 		if (prop_type != PT_VALUE) {
23457c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
2346bbec428eSgjelinek 			    B_TRUE);
2347bbec428eSgjelinek 			long_usage(CMD_ADD, B_TRUE);
2348bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
23497c478bd9Sstevel@tonic-gate 			return;
23507c478bd9Sstevel@tonic-gate 		}
23517c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[0];
23527c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_COMPLEX &&
23537c478bd9Sstevel@tonic-gate 		    pp->pv_type != PROP_VAL_LIST) {
23547c478bd9Sstevel@tonic-gate 			zerr(gettext("A %s or %s value was expected here."),
23557c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_COMPLEX),
23567c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_LIST));
2357bbec428eSgjelinek 			saw_error = B_TRUE;
23587c478bd9Sstevel@tonic-gate 			return;
23597c478bd9Sstevel@tonic-gate 		}
23607c478bd9Sstevel@tonic-gate 		if (pp->pv_type == PROP_VAL_COMPLEX) {
23617c478bd9Sstevel@tonic-gate 			do_complex_rctl_val(pp->pv_complex);
23627c478bd9Sstevel@tonic-gate 			return;
23637c478bd9Sstevel@tonic-gate 		}
23647c478bd9Sstevel@tonic-gate 		for (l = pp->pv_list; l != NULL; l = l->lp_next)
23657c478bd9Sstevel@tonic-gate 			do_complex_rctl_val(l->lp_complex);
23667c478bd9Sstevel@tonic-gate 		return;
23677c478bd9Sstevel@tonic-gate 	default:
2368bbec428eSgjelinek 		zone_perror(rt_to_str(res_type), Z_NO_RESOURCE_TYPE, B_TRUE);
2369bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
2370bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
23717c478bd9Sstevel@tonic-gate 		return;
23727c478bd9Sstevel@tonic-gate 	}
23737c478bd9Sstevel@tonic-gate }
23747c478bd9Sstevel@tonic-gate 
23750209230bSgjelinek static boolean_t
23760209230bSgjelinek gz_invalid_resource(int type)
23770209230bSgjelinek {
23786e1ae2a3SGary Pennington 	return (global_zone && (type == RT_FS ||
23790209230bSgjelinek 	    type == RT_NET || type == RT_DEVICE || type == RT_ATTR ||
23800209230bSgjelinek 	    type == RT_DATASET));
23810209230bSgjelinek }
23820209230bSgjelinek 
23830209230bSgjelinek static boolean_t
23840209230bSgjelinek gz_invalid_rt_property(int type)
23850209230bSgjelinek {
23860209230bSgjelinek 	return (global_zone && (type == RT_ZONENAME || type == RT_ZONEPATH ||
23870209230bSgjelinek 	    type == RT_AUTOBOOT || type == RT_LIMITPRIV ||
2388f4b3ec61Sdh155122 	    type == RT_BOOTARGS || type == RT_BRAND || type == RT_SCHED ||
23890fbb751dSJohn Levon 	    type == RT_IPTYPE || type == RT_HOSTID || type == RT_FS_ALLOWED));
23900209230bSgjelinek }
23910209230bSgjelinek 
23920209230bSgjelinek static boolean_t
23930209230bSgjelinek gz_invalid_property(int type)
23940209230bSgjelinek {
23950209230bSgjelinek 	return (global_zone && (type == PT_ZONENAME || type == PT_ZONEPATH ||
23960209230bSgjelinek 	    type == PT_AUTOBOOT || type == PT_LIMITPRIV ||
2397f4b3ec61Sdh155122 	    type == PT_BOOTARGS || type == PT_BRAND || type == PT_SCHED ||
23980fbb751dSJohn Levon 	    type == PT_IPTYPE || type == PT_HOSTID || type == PT_FS_ALLOWED));
23990209230bSgjelinek }
24000209230bSgjelinek 
24017c478bd9Sstevel@tonic-gate void
24027c478bd9Sstevel@tonic-gate add_func(cmd_t *cmd)
24037c478bd9Sstevel@tonic-gate {
24047c478bd9Sstevel@tonic-gate 	int arg;
2405bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
24067c478bd9Sstevel@tonic-gate 
24077c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
24087c478bd9Sstevel@tonic-gate 
24097c478bd9Sstevel@tonic-gate 	optind = 0;
24107ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
24117c478bd9Sstevel@tonic-gate 		switch (arg) {
24127c478bd9Sstevel@tonic-gate 		case '?':
24137c478bd9Sstevel@tonic-gate 			longer_usage(CMD_ADD);
2414bbec428eSgjelinek 			arg_err = B_TRUE;
24157ec75eb8Sgjelinek 			break;
24167c478bd9Sstevel@tonic-gate 		default:
24177c478bd9Sstevel@tonic-gate 			short_usage(CMD_ADD);
2418bbec428eSgjelinek 			arg_err = B_TRUE;
24197ec75eb8Sgjelinek 			break;
24207ec75eb8Sgjelinek 		}
24217ec75eb8Sgjelinek 	}
24227ec75eb8Sgjelinek 	if (arg_err)
24237c478bd9Sstevel@tonic-gate 		return;
24247ec75eb8Sgjelinek 
24257c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
24267c478bd9Sstevel@tonic-gate 		short_usage(CMD_ADD);
24277c478bd9Sstevel@tonic-gate 		return;
24287c478bd9Sstevel@tonic-gate 	}
24297c478bd9Sstevel@tonic-gate 
24307c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_ADD))
24317c478bd9Sstevel@tonic-gate 		return;
24327c478bd9Sstevel@tonic-gate 
2433bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
24347c478bd9Sstevel@tonic-gate 		return;
24357c478bd9Sstevel@tonic-gate 	if (global_scope) {
24360209230bSgjelinek 		if (gz_invalid_resource(cmd->cmd_res_type)) {
24370209230bSgjelinek 			zerr(gettext("Cannot add a %s resource to the "
24380209230bSgjelinek 			    "global zone."), rt_to_str(cmd->cmd_res_type));
2439bbec428eSgjelinek 			saw_error = B_TRUE;
24400209230bSgjelinek 			return;
24410209230bSgjelinek 		}
24420209230bSgjelinek 
2443bbec428eSgjelinek 		global_scope = B_FALSE;
24447c478bd9Sstevel@tonic-gate 		resource_scope = cmd->cmd_res_type;
24457c478bd9Sstevel@tonic-gate 		end_op = CMD_ADD;
24467c478bd9Sstevel@tonic-gate 		add_resource(cmd);
24477c478bd9Sstevel@tonic-gate 	} else
24487c478bd9Sstevel@tonic-gate 		add_property(cmd);
24497c478bd9Sstevel@tonic-gate }
24507c478bd9Sstevel@tonic-gate 
2451087719fdSdp /*
2452087719fdSdp  * This routine has an unusual implementation, because it tries very
2453087719fdSdp  * hard to succeed in the face of a variety of failure modes.
2454087719fdSdp  * The most common and most vexing occurs when the index file and
2455087719fdSdp  * the /etc/zones/<zonename.xml> file are not both present.  In
2456087719fdSdp  * this case, delete must eradicate as much of the zone state as is left
2457087719fdSdp  * so that the user can later create a new zone with the same name.
2458087719fdSdp  */
24597c478bd9Sstevel@tonic-gate void
24607c478bd9Sstevel@tonic-gate delete_func(cmd_t *cmd)
24617c478bd9Sstevel@tonic-gate {
24627c478bd9Sstevel@tonic-gate 	int err, arg, answer;
24637c478bd9Sstevel@tonic-gate 	char line[ZONENAME_MAX + 128];	/* enough to ask a question */
2464bbec428eSgjelinek 	boolean_t force = B_FALSE;
2465bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
24667c478bd9Sstevel@tonic-gate 
24677c478bd9Sstevel@tonic-gate 	optind = 0;
24687c478bd9Sstevel@tonic-gate 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?F")) != EOF) {
24697c478bd9Sstevel@tonic-gate 		switch (arg) {
24707c478bd9Sstevel@tonic-gate 		case '?':
24717c478bd9Sstevel@tonic-gate 			longer_usage(CMD_DELETE);
2472bbec428eSgjelinek 			arg_err = B_TRUE;
24737ec75eb8Sgjelinek 			break;
24747c478bd9Sstevel@tonic-gate 		case 'F':
2475bbec428eSgjelinek 			force = B_TRUE;
24767c478bd9Sstevel@tonic-gate 			break;
24777c478bd9Sstevel@tonic-gate 		default:
24787c478bd9Sstevel@tonic-gate 			short_usage(CMD_DELETE);
2479bbec428eSgjelinek 			arg_err = B_TRUE;
24807ec75eb8Sgjelinek 			break;
24817ec75eb8Sgjelinek 		}
24827ec75eb8Sgjelinek 	}
24837ec75eb8Sgjelinek 	if (arg_err)
24847c478bd9Sstevel@tonic-gate 		return;
24857ec75eb8Sgjelinek 
24867c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
24877c478bd9Sstevel@tonic-gate 		short_usage(CMD_DELETE);
24887c478bd9Sstevel@tonic-gate 		return;
24897c478bd9Sstevel@tonic-gate 	}
24907c478bd9Sstevel@tonic-gate 
24917c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_DELETE))
24927c478bd9Sstevel@tonic-gate 		return;
24937c478bd9Sstevel@tonic-gate 
2494087719fdSdp 	if (!force) {
2495087719fdSdp 		/*
2496087719fdSdp 		 * Initialize sets up the global called "handle" and warns the
2497087719fdSdp 		 * user if the zone is not configured.  In force mode, we don't
2498087719fdSdp 		 * trust that evaluation, and hence skip it.  (We don't need the
2499087719fdSdp 		 * handle to be loaded anyway, since zonecfg_destroy is done by
2500087719fdSdp 		 * zonename). However, we also have to take care to emulate the
2501087719fdSdp 		 * messages spit out by initialize; see below.
2502087719fdSdp 		 */
2503bbec428eSgjelinek 		if (initialize(B_TRUE) != Z_OK)
25047c478bd9Sstevel@tonic-gate 			return;
25057c478bd9Sstevel@tonic-gate 
25067c478bd9Sstevel@tonic-gate 		(void) snprintf(line, sizeof (line),
25077c478bd9Sstevel@tonic-gate 		    gettext("Are you sure you want to delete zone %s"), zone);
2508bbec428eSgjelinek 		if ((answer = ask_yesno(B_FALSE, line)) == -1) {
2509087719fdSdp 			zerr(gettext("Input not from terminal and -F not "
2510087719fdSdp 			    "specified:\n%s command ignored, exiting."),
2511087719fdSdp 			    cmd_to_str(CMD_DELETE));
25127c478bd9Sstevel@tonic-gate 			exit(Z_ERR);
25137c478bd9Sstevel@tonic-gate 		}
25147c478bd9Sstevel@tonic-gate 		if (answer != 1)
25157c478bd9Sstevel@tonic-gate 			return;
25167c478bd9Sstevel@tonic-gate 	}
25177c478bd9Sstevel@tonic-gate 
2518cb8a054bSGlenn Faden 	/*
2519cb8a054bSGlenn Faden 	 * This function removes the authorizations from user_attr
2520cb8a054bSGlenn Faden 	 * that correspond to those specified in the configuration
2521cb8a054bSGlenn Faden 	 */
2522cb8a054bSGlenn Faden 	if (initialize(B_TRUE) == Z_OK) {
2523cb8a054bSGlenn Faden 		(void) zonecfg_deauthorize_users(handle, zone);
2524cb8a054bSGlenn Faden 	}
2525087719fdSdp 	if ((err = zonecfg_destroy(zone, force)) != Z_OK) {
2526087719fdSdp 		if ((err == Z_BAD_ZONE_STATE) && !force) {
2527087719fdSdp 			zerr(gettext("Zone %s not in %s state; %s not "
2528087719fdSdp 			    "allowed.  Use -F to force %s."),
2529087719fdSdp 			    zone, zone_state_str(ZONE_STATE_CONFIGURED),
2530087719fdSdp 			    cmd_to_str(CMD_DELETE), cmd_to_str(CMD_DELETE));
2531087719fdSdp 		} else {
2532bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
25337c478bd9Sstevel@tonic-gate 		}
2534087719fdSdp 	}
2535bbec428eSgjelinek 	need_to_commit = B_FALSE;
2536087719fdSdp 
2537087719fdSdp 	/*
2538087719fdSdp 	 * Emulate initialize's messaging; if there wasn't a valid handle to
2539087719fdSdp 	 * begin with, then user had typed delete (or delete -F) multiple
2540087719fdSdp 	 * times.  So we emit a message.
2541087719fdSdp 	 *
2542087719fdSdp 	 * We only do this in the 'force' case because normally, initialize()
2543087719fdSdp 	 * takes care of this for us.
2544087719fdSdp 	 */
2545087719fdSdp 	if (force && zonecfg_check_handle(handle) != Z_OK && interactive_mode)
2546087719fdSdp 		(void) printf(gettext("Use '%s' to begin "
2547087719fdSdp 		    "configuring a new zone.\n"), cmd_to_str(CMD_CREATE));
25487c478bd9Sstevel@tonic-gate 
25497c478bd9Sstevel@tonic-gate 	/*
25507c478bd9Sstevel@tonic-gate 	 * Time for a new handle: finish the old one off first
25517c478bd9Sstevel@tonic-gate 	 * then get a new one properly to avoid leaks.
25527c478bd9Sstevel@tonic-gate 	 */
2553087719fdSdp 	if (got_handle) {
25547c478bd9Sstevel@tonic-gate 		zonecfg_fini_handle(handle);
25557c478bd9Sstevel@tonic-gate 		if ((handle = zonecfg_init_handle()) == NULL) {
2556bbec428eSgjelinek 			zone_perror(execname, Z_NOMEM, B_TRUE);
25577c478bd9Sstevel@tonic-gate 			exit(Z_ERR);
25587c478bd9Sstevel@tonic-gate 		}
25597c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_get_handle(zone, handle)) != Z_OK) {
25607c478bd9Sstevel@tonic-gate 			/* If there was no zone before, that's OK */
25617c478bd9Sstevel@tonic-gate 			if (err != Z_NO_ZONE)
2562bbec428eSgjelinek 				zone_perror(zone, err, B_TRUE);
2563bbec428eSgjelinek 			got_handle = B_FALSE;
25647c478bd9Sstevel@tonic-gate 		}
25657c478bd9Sstevel@tonic-gate 	}
2566087719fdSdp }
25677c478bd9Sstevel@tonic-gate 
25687c478bd9Sstevel@tonic-gate static int
2569bbec428eSgjelinek fill_in_fstab(cmd_t *cmd, struct zone_fstab *fstab, boolean_t fill_in_only)
25707c478bd9Sstevel@tonic-gate {
25717c478bd9Sstevel@tonic-gate 	int err, i;
25727c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
25737c478bd9Sstevel@tonic-gate 
2574bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
25757c478bd9Sstevel@tonic-gate 		return (err);
25767c478bd9Sstevel@tonic-gate 
2577e193d1e6Svp157776 	bzero(fstab, sizeof (*fstab));
25787c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
25797c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
25807c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
25817c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2582bbec428eSgjelinek 			saw_error = B_TRUE;
25837c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
25847c478bd9Sstevel@tonic-gate 		}
25857c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
25867c478bd9Sstevel@tonic-gate 		case PT_DIR:
25877c478bd9Sstevel@tonic-gate 			(void) strlcpy(fstab->zone_fs_dir, pp->pv_simple,
25887c478bd9Sstevel@tonic-gate 			    sizeof (fstab->zone_fs_dir));
25897c478bd9Sstevel@tonic-gate 			break;
25907c478bd9Sstevel@tonic-gate 		case PT_SPECIAL:
25917c478bd9Sstevel@tonic-gate 			(void) strlcpy(fstab->zone_fs_special, pp->pv_simple,
25927c478bd9Sstevel@tonic-gate 			    sizeof (fstab->zone_fs_special));
25937c478bd9Sstevel@tonic-gate 			break;
25947c478bd9Sstevel@tonic-gate 		case PT_RAW:
25957c478bd9Sstevel@tonic-gate 			(void) strlcpy(fstab->zone_fs_raw, pp->pv_simple,
25967c478bd9Sstevel@tonic-gate 			    sizeof (fstab->zone_fs_raw));
25977c478bd9Sstevel@tonic-gate 			break;
25987c478bd9Sstevel@tonic-gate 		case PT_TYPE:
25997c478bd9Sstevel@tonic-gate 			(void) strlcpy(fstab->zone_fs_type, pp->pv_simple,
26007c478bd9Sstevel@tonic-gate 			    sizeof (fstab->zone_fs_type));
26017c478bd9Sstevel@tonic-gate 			break;
26027c478bd9Sstevel@tonic-gate 		default:
26037c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2604bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
26057c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
26067c478bd9Sstevel@tonic-gate 		}
26077c478bd9Sstevel@tonic-gate 	}
26087c478bd9Sstevel@tonic-gate 	if (fill_in_only)
26097c478bd9Sstevel@tonic-gate 		return (Z_OK);
26107c478bd9Sstevel@tonic-gate 	return (zonecfg_lookup_filesystem(handle, fstab));
26117c478bd9Sstevel@tonic-gate }
26127c478bd9Sstevel@tonic-gate 
26137c478bd9Sstevel@tonic-gate static int
2614bbec428eSgjelinek fill_in_nwiftab(cmd_t *cmd, struct zone_nwiftab *nwiftab,
2615bbec428eSgjelinek     boolean_t fill_in_only)
26167c478bd9Sstevel@tonic-gate {
26177c478bd9Sstevel@tonic-gate 	int err, i;
26187c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
26197c478bd9Sstevel@tonic-gate 
2620bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
26217c478bd9Sstevel@tonic-gate 		return (err);
26227c478bd9Sstevel@tonic-gate 
2623e193d1e6Svp157776 	bzero(nwiftab, sizeof (*nwiftab));
26247c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
26257c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
26267c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
26277c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2628bbec428eSgjelinek 			saw_error = B_TRUE;
26297c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
26307c478bd9Sstevel@tonic-gate 		}
26317c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
26327c478bd9Sstevel@tonic-gate 		case PT_ADDRESS:
26337c478bd9Sstevel@tonic-gate 			(void) strlcpy(nwiftab->zone_nwif_address,
26347c478bd9Sstevel@tonic-gate 			    pp->pv_simple, sizeof (nwiftab->zone_nwif_address));
26357c478bd9Sstevel@tonic-gate 			break;
2636550b6e40SSowmini Varadhan 		case PT_ALLOWED_ADDRESS:
2637550b6e40SSowmini Varadhan 			(void) strlcpy(nwiftab->zone_nwif_allowed_address,
2638550b6e40SSowmini Varadhan 			    pp->pv_simple,
2639550b6e40SSowmini Varadhan 			    sizeof (nwiftab->zone_nwif_allowed_address));
2640550b6e40SSowmini Varadhan 			break;
26417c478bd9Sstevel@tonic-gate 		case PT_PHYSICAL:
26427c478bd9Sstevel@tonic-gate 			(void) strlcpy(nwiftab->zone_nwif_physical,
26437c478bd9Sstevel@tonic-gate 			    pp->pv_simple,
26447c478bd9Sstevel@tonic-gate 			    sizeof (nwiftab->zone_nwif_physical));
26457c478bd9Sstevel@tonic-gate 			break;
2646de860bd9Sgfaden 		case PT_DEFROUTER:
2647de860bd9Sgfaden 			(void) strlcpy(nwiftab->zone_nwif_defrouter,
2648de860bd9Sgfaden 			    pp->pv_simple,
2649de860bd9Sgfaden 			    sizeof (nwiftab->zone_nwif_defrouter));
2650de860bd9Sgfaden 			break;
26517c478bd9Sstevel@tonic-gate 		default:
26527c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2653bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
26547c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
26557c478bd9Sstevel@tonic-gate 		}
26567c478bd9Sstevel@tonic-gate 	}
26577c478bd9Sstevel@tonic-gate 	if (fill_in_only)
26587c478bd9Sstevel@tonic-gate 		return (Z_OK);
26597c478bd9Sstevel@tonic-gate 	err = zonecfg_lookup_nwif(handle, nwiftab);
26607c478bd9Sstevel@tonic-gate 	return (err);
26617c478bd9Sstevel@tonic-gate }
26627c478bd9Sstevel@tonic-gate 
26637c478bd9Sstevel@tonic-gate static int
2664bbec428eSgjelinek fill_in_devtab(cmd_t *cmd, struct zone_devtab *devtab, boolean_t fill_in_only)
26657c478bd9Sstevel@tonic-gate {
26667c478bd9Sstevel@tonic-gate 	int err, i;
26677c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
26687c478bd9Sstevel@tonic-gate 
2669bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
26707c478bd9Sstevel@tonic-gate 		return (err);
26717c478bd9Sstevel@tonic-gate 
2672e193d1e6Svp157776 	bzero(devtab, sizeof (*devtab));
26737c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
26747c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
26757c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
26767c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2677bbec428eSgjelinek 			saw_error = B_TRUE;
26787c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
26797c478bd9Sstevel@tonic-gate 		}
26807c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
26817c478bd9Sstevel@tonic-gate 		case PT_MATCH:
26827c478bd9Sstevel@tonic-gate 			(void) strlcpy(devtab->zone_dev_match, pp->pv_simple,
26837c478bd9Sstevel@tonic-gate 			    sizeof (devtab->zone_dev_match));
26847c478bd9Sstevel@tonic-gate 			break;
26857c478bd9Sstevel@tonic-gate 		default:
26867c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2687bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
26887c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
26897c478bd9Sstevel@tonic-gate 		}
26907c478bd9Sstevel@tonic-gate 	}
26917c478bd9Sstevel@tonic-gate 	if (fill_in_only)
26927c478bd9Sstevel@tonic-gate 		return (Z_OK);
26937c478bd9Sstevel@tonic-gate 	err = zonecfg_lookup_dev(handle, devtab);
26947c478bd9Sstevel@tonic-gate 	return (err);
26957c478bd9Sstevel@tonic-gate }
26967c478bd9Sstevel@tonic-gate 
26977c478bd9Sstevel@tonic-gate static int
2698bbec428eSgjelinek fill_in_rctltab(cmd_t *cmd, struct zone_rctltab *rctltab,
2699bbec428eSgjelinek     boolean_t fill_in_only)
27007c478bd9Sstevel@tonic-gate {
27017c478bd9Sstevel@tonic-gate 	int err, i;
27027c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
27037c478bd9Sstevel@tonic-gate 
2704bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
27057c478bd9Sstevel@tonic-gate 		return (err);
27067c478bd9Sstevel@tonic-gate 
2707e193d1e6Svp157776 	bzero(rctltab, sizeof (*rctltab));
27087c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
27097c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
27107c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
27117c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2712bbec428eSgjelinek 			saw_error = B_TRUE;
27137c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
27147c478bd9Sstevel@tonic-gate 		}
27157c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
27167c478bd9Sstevel@tonic-gate 		case PT_NAME:
27177c478bd9Sstevel@tonic-gate 			(void) strlcpy(rctltab->zone_rctl_name, pp->pv_simple,
27187c478bd9Sstevel@tonic-gate 			    sizeof (rctltab->zone_rctl_name));
27197c478bd9Sstevel@tonic-gate 			break;
27207c478bd9Sstevel@tonic-gate 		default:
27217c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2722bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
27237c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
27247c478bd9Sstevel@tonic-gate 		}
27257c478bd9Sstevel@tonic-gate 	}
27267c478bd9Sstevel@tonic-gate 	if (fill_in_only)
27277c478bd9Sstevel@tonic-gate 		return (Z_OK);
27287c478bd9Sstevel@tonic-gate 	err = zonecfg_lookup_rctl(handle, rctltab);
27297c478bd9Sstevel@tonic-gate 	return (err);
27307c478bd9Sstevel@tonic-gate }
27317c478bd9Sstevel@tonic-gate 
27327c478bd9Sstevel@tonic-gate static int
2733bbec428eSgjelinek fill_in_attrtab(cmd_t *cmd, struct zone_attrtab *attrtab,
2734bbec428eSgjelinek     boolean_t fill_in_only)
27357c478bd9Sstevel@tonic-gate {
27367c478bd9Sstevel@tonic-gate 	int err, i;
27377c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
27387c478bd9Sstevel@tonic-gate 
2739bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
27407c478bd9Sstevel@tonic-gate 		return (err);
27417c478bd9Sstevel@tonic-gate 
2742e193d1e6Svp157776 	bzero(attrtab, sizeof (*attrtab));
27437c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
27447c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
27457c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
27467c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2747bbec428eSgjelinek 			saw_error = B_TRUE;
27487c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
27497c478bd9Sstevel@tonic-gate 		}
27507c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
27517c478bd9Sstevel@tonic-gate 		case PT_NAME:
27527c478bd9Sstevel@tonic-gate 			(void) strlcpy(attrtab->zone_attr_name, pp->pv_simple,
27537c478bd9Sstevel@tonic-gate 			    sizeof (attrtab->zone_attr_name));
27547c478bd9Sstevel@tonic-gate 			break;
27557c478bd9Sstevel@tonic-gate 		case PT_TYPE:
27567c478bd9Sstevel@tonic-gate 			(void) strlcpy(attrtab->zone_attr_type, pp->pv_simple,
27577c478bd9Sstevel@tonic-gate 			    sizeof (attrtab->zone_attr_type));
27587c478bd9Sstevel@tonic-gate 			break;
27597c478bd9Sstevel@tonic-gate 		case PT_VALUE:
27607c478bd9Sstevel@tonic-gate 			(void) strlcpy(attrtab->zone_attr_value, pp->pv_simple,
27617c478bd9Sstevel@tonic-gate 			    sizeof (attrtab->zone_attr_value));
27627c478bd9Sstevel@tonic-gate 			break;
27637c478bd9Sstevel@tonic-gate 		default:
27647c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2765bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
27667c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
27677c478bd9Sstevel@tonic-gate 		}
27687c478bd9Sstevel@tonic-gate 	}
27697c478bd9Sstevel@tonic-gate 	if (fill_in_only)
27707c478bd9Sstevel@tonic-gate 		return (Z_OK);
27717c478bd9Sstevel@tonic-gate 	err = zonecfg_lookup_attr(handle, attrtab);
27727c478bd9Sstevel@tonic-gate 	return (err);
27737c478bd9Sstevel@tonic-gate }
27747c478bd9Sstevel@tonic-gate 
2775fa9e4066Sahrens static int
2776bbec428eSgjelinek fill_in_dstab(cmd_t *cmd, struct zone_dstab *dstab, boolean_t fill_in_only)
2777fa9e4066Sahrens {
2778fa9e4066Sahrens 	int err, i;
2779fa9e4066Sahrens 	property_value_ptr_t pp;
2780fa9e4066Sahrens 
2781bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
2782fa9e4066Sahrens 		return (err);
2783fa9e4066Sahrens 
2784fa9e4066Sahrens 	dstab->zone_dataset_name[0] = '\0';
2785fa9e4066Sahrens 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
2786fa9e4066Sahrens 		pp = cmd->cmd_property_ptr[i];
2787fa9e4066Sahrens 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
2788fa9e4066Sahrens 			zerr(gettext("A simple value was expected here."));
2789bbec428eSgjelinek 			saw_error = B_TRUE;
2790fa9e4066Sahrens 			return (Z_INSUFFICIENT_SPEC);
2791fa9e4066Sahrens 		}
2792fa9e4066Sahrens 		switch (cmd->cmd_prop_name[i]) {
2793fa9e4066Sahrens 		case PT_NAME:
2794fa9e4066Sahrens 			(void) strlcpy(dstab->zone_dataset_name, pp->pv_simple,
2795fa9e4066Sahrens 			    sizeof (dstab->zone_dataset_name));
2796fa9e4066Sahrens 			break;
2797fa9e4066Sahrens 		default:
2798fa9e4066Sahrens 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2799bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
2800fa9e4066Sahrens 			return (Z_INSUFFICIENT_SPEC);
2801fa9e4066Sahrens 		}
2802fa9e4066Sahrens 	}
2803fa9e4066Sahrens 	if (fill_in_only)
2804fa9e4066Sahrens 		return (Z_OK);
2805fa9e4066Sahrens 	return (zonecfg_lookup_ds(handle, dstab));
2806fa9e4066Sahrens }
2807fa9e4066Sahrens 
2808cb8a054bSGlenn Faden static int
2809cb8a054bSGlenn Faden fill_in_admintab(cmd_t *cmd, struct zone_admintab *admintab,
2810cb8a054bSGlenn Faden     boolean_t fill_in_only)
2811cb8a054bSGlenn Faden {
2812cb8a054bSGlenn Faden 	int err, i;
2813cb8a054bSGlenn Faden 	property_value_ptr_t pp;
2814cb8a054bSGlenn Faden 
2815cb8a054bSGlenn Faden 	if ((err = initialize(B_TRUE)) != Z_OK)
2816cb8a054bSGlenn Faden 		return (err);
2817cb8a054bSGlenn Faden 
2818cb8a054bSGlenn Faden 	bzero(admintab, sizeof (*admintab));
2819cb8a054bSGlenn Faden 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
2820cb8a054bSGlenn Faden 		pp = cmd->cmd_property_ptr[i];
2821cb8a054bSGlenn Faden 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
2822cb8a054bSGlenn Faden 			zerr(gettext("A simple value was expected here."));
2823cb8a054bSGlenn Faden 			saw_error = B_TRUE;
2824cb8a054bSGlenn Faden 			return (Z_INSUFFICIENT_SPEC);
2825cb8a054bSGlenn Faden 		}
2826cb8a054bSGlenn Faden 		switch (cmd->cmd_prop_name[i]) {
2827cb8a054bSGlenn Faden 		case PT_USER:
2828cb8a054bSGlenn Faden 			(void) strlcpy(admintab->zone_admin_user, pp->pv_simple,
2829cb8a054bSGlenn Faden 			    sizeof (admintab->zone_admin_user));
2830cb8a054bSGlenn Faden 			break;
2831cb8a054bSGlenn Faden 		case PT_AUTHS:
2832cb8a054bSGlenn Faden 			(void) strlcpy(admintab->zone_admin_auths,
2833cb8a054bSGlenn Faden 			    pp->pv_simple, sizeof (admintab->zone_admin_auths));
2834cb8a054bSGlenn Faden 			break;
2835cb8a054bSGlenn Faden 		default:
2836cb8a054bSGlenn Faden 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2837cb8a054bSGlenn Faden 			    Z_NO_PROPERTY_TYPE, B_TRUE);
2838cb8a054bSGlenn Faden 			return (Z_INSUFFICIENT_SPEC);
2839cb8a054bSGlenn Faden 		}
2840cb8a054bSGlenn Faden 	}
2841cb8a054bSGlenn Faden 	if (fill_in_only)
2842cb8a054bSGlenn Faden 		return (Z_OK);
2843cb8a054bSGlenn Faden 	err = zonecfg_lookup_admin(handle, admintab);
2844cb8a054bSGlenn Faden 	return (err);
2845cb8a054bSGlenn Faden }
2846cb8a054bSGlenn Faden 
28477c478bd9Sstevel@tonic-gate static void
28480209230bSgjelinek remove_aliased_rctl(int type, char *name)
28497c478bd9Sstevel@tonic-gate {
28500209230bSgjelinek 	int err;
28510209230bSgjelinek 	uint64_t tmp;
28527c478bd9Sstevel@tonic-gate 
28530209230bSgjelinek 	if ((err = zonecfg_get_aliased_rctl(handle, name, &tmp)) != Z_OK) {
28540209230bSgjelinek 		zerr("%s %s: %s", cmd_to_str(CMD_CLEAR), pt_to_str(type),
28550209230bSgjelinek 		    zonecfg_strerror(err));
2856bbec428eSgjelinek 		saw_error = B_TRUE;
28577c478bd9Sstevel@tonic-gate 		return;
28587c478bd9Sstevel@tonic-gate 	}
28590209230bSgjelinek 	if ((err = zonecfg_rm_aliased_rctl(handle, name)) != Z_OK) {
28600209230bSgjelinek 		zerr("%s %s: %s", cmd_to_str(CMD_CLEAR), pt_to_str(type),
28610209230bSgjelinek 		    zonecfg_strerror(err));
2862bbec428eSgjelinek 		saw_error = B_TRUE;
28630209230bSgjelinek 	} else {
2864bbec428eSgjelinek 		need_to_commit = B_TRUE;
28650209230bSgjelinek 	}
28660209230bSgjelinek }
28677c478bd9Sstevel@tonic-gate 
28680209230bSgjelinek static boolean_t
28690209230bSgjelinek prompt_remove_resource(cmd_t *cmd, char *rsrc)
28700209230bSgjelinek {
28710209230bSgjelinek 	int num;
28720209230bSgjelinek 	int answer;
28730209230bSgjelinek 	int arg;
28740209230bSgjelinek 	boolean_t force = B_FALSE;
28750209230bSgjelinek 	char prompt[128];
2876bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
28777c478bd9Sstevel@tonic-gate 
28780209230bSgjelinek 	optind = 0;
28790209230bSgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "F")) != EOF) {
28800209230bSgjelinek 		switch (arg) {
28810209230bSgjelinek 		case 'F':
28820209230bSgjelinek 			force = B_TRUE;
28830209230bSgjelinek 			break;
28840209230bSgjelinek 		default:
2885bbec428eSgjelinek 			arg_err = B_TRUE;
28867ec75eb8Sgjelinek 			break;
28877ec75eb8Sgjelinek 		}
28887ec75eb8Sgjelinek 	}
28897ec75eb8Sgjelinek 	if (arg_err)
28900209230bSgjelinek 		return (B_FALSE);
28917ec75eb8Sgjelinek 
28920209230bSgjelinek 
28930209230bSgjelinek 	num = zonecfg_num_resources(handle, rsrc);
28940209230bSgjelinek 
28950209230bSgjelinek 	if (num == 0) {
28960209230bSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, cmd->cmd_res_type, Z_NO_ENTRY,
2897bbec428eSgjelinek 		    B_TRUE);
28980209230bSgjelinek 		return (B_FALSE);
28990209230bSgjelinek 	}
29000209230bSgjelinek 	if (num > 1 && !force) {
29010209230bSgjelinek 		if (!interactive_mode) {
29020209230bSgjelinek 			zerr(gettext("There are multiple instances of this "
29030209230bSgjelinek 			    "resource.  Either qualify the resource to\n"
29040209230bSgjelinek 			    "remove a single instance or use the -F option to "
29050209230bSgjelinek 			    "remove all instances."));
2906bbec428eSgjelinek 			saw_error = B_TRUE;
29070209230bSgjelinek 			return (B_FALSE);
29080209230bSgjelinek 		}
29090209230bSgjelinek 		(void) snprintf(prompt, sizeof (prompt), gettext(
29100209230bSgjelinek 		    "Are you sure you want to remove ALL '%s' resources"),
29110209230bSgjelinek 		    rsrc);
2912bbec428eSgjelinek 		answer = ask_yesno(B_FALSE, prompt);
29130209230bSgjelinek 		if (answer == -1) {
29140209230bSgjelinek 			zerr(gettext("Resource incomplete."));
29150209230bSgjelinek 			return (B_FALSE);
29160209230bSgjelinek 		}
29170209230bSgjelinek 		if (answer != 1)
29180209230bSgjelinek 			return (B_FALSE);
29190209230bSgjelinek 	}
29200209230bSgjelinek 	return (B_TRUE);
29210209230bSgjelinek }
29220209230bSgjelinek 
29230209230bSgjelinek static void
29240209230bSgjelinek remove_fs(cmd_t *cmd)
29250209230bSgjelinek {
29260209230bSgjelinek 	int err;
29270209230bSgjelinek 
29280209230bSgjelinek 	/* traditional, qualified fs removal */
29290209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
29300209230bSgjelinek 		struct zone_fstab fstab;
29310209230bSgjelinek 
2932bbec428eSgjelinek 		if ((err = fill_in_fstab(cmd, &fstab, B_FALSE)) != Z_OK) {
2933bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_FS, err, B_TRUE);
29347c478bd9Sstevel@tonic-gate 			return;
29357c478bd9Sstevel@tonic-gate 		}
29367c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_delete_filesystem(handle, &fstab)) != Z_OK)
2937bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_FS, err, B_TRUE);
29387c478bd9Sstevel@tonic-gate 		else
2939bbec428eSgjelinek 			need_to_commit = B_TRUE;
29407c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(fstab.zone_fs_options);
29417c478bd9Sstevel@tonic-gate 		return;
29420209230bSgjelinek 	}
29430209230bSgjelinek 
29440209230bSgjelinek 	/*
29450209230bSgjelinek 	 * unqualified fs removal.  remove all fs's but prompt if more
29460209230bSgjelinek 	 * than one.
29470209230bSgjelinek 	 */
29480209230bSgjelinek 	if (!prompt_remove_resource(cmd, "fs"))
29490209230bSgjelinek 		return;
29500209230bSgjelinek 
29510209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "fs")) != Z_OK)
2952bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_FS, err, B_TRUE);
29530209230bSgjelinek 	else
2954bbec428eSgjelinek 		need_to_commit = B_TRUE;
29550209230bSgjelinek }
29560209230bSgjelinek 
29570209230bSgjelinek static void
29580209230bSgjelinek remove_net(cmd_t *cmd)
29590209230bSgjelinek {
29600209230bSgjelinek 	int err;
29610209230bSgjelinek 
29620209230bSgjelinek 	/* traditional, qualified net removal */
29630209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
29640209230bSgjelinek 		struct zone_nwiftab nwiftab;
29650209230bSgjelinek 
2966bbec428eSgjelinek 		if ((err = fill_in_nwiftab(cmd, &nwiftab, B_FALSE)) != Z_OK) {
2967bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_NET, err, B_TRUE);
29687c478bd9Sstevel@tonic-gate 			return;
29697c478bd9Sstevel@tonic-gate 		}
29707c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_delete_nwif(handle, &nwiftab)) != Z_OK)
2971bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_NET, err, B_TRUE);
29727c478bd9Sstevel@tonic-gate 		else
2973bbec428eSgjelinek 			need_to_commit = B_TRUE;
29747c478bd9Sstevel@tonic-gate 		return;
29750209230bSgjelinek 	}
29760209230bSgjelinek 
29770209230bSgjelinek 	/*
29780209230bSgjelinek 	 * unqualified net removal.  remove all nets but prompt if more
29790209230bSgjelinek 	 * than one.
29800209230bSgjelinek 	 */
29810209230bSgjelinek 	if (!prompt_remove_resource(cmd, "net"))
29820209230bSgjelinek 		return;
29830209230bSgjelinek 
29840209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "net")) != Z_OK)
2985bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_NET, err, B_TRUE);
29860209230bSgjelinek 	else
2987bbec428eSgjelinek 		need_to_commit = B_TRUE;
29880209230bSgjelinek }
29890209230bSgjelinek 
29900209230bSgjelinek static void
29910209230bSgjelinek remove_device(cmd_t *cmd)
29920209230bSgjelinek {
29930209230bSgjelinek 	int err;
29940209230bSgjelinek 
29950209230bSgjelinek 	/* traditional, qualified device removal */
29960209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
29970209230bSgjelinek 		struct zone_devtab devtab;
29980209230bSgjelinek 
2999bbec428eSgjelinek 		if ((err = fill_in_devtab(cmd, &devtab, B_FALSE)) != Z_OK) {
3000bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_DEVICE, err, B_TRUE);
30017c478bd9Sstevel@tonic-gate 			return;
30027c478bd9Sstevel@tonic-gate 		}
30037c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_delete_dev(handle, &devtab)) != Z_OK)
3004bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_DEVICE, err, B_TRUE);
30057c478bd9Sstevel@tonic-gate 		else
3006bbec428eSgjelinek 			need_to_commit = B_TRUE;
30077c478bd9Sstevel@tonic-gate 		return;
30080209230bSgjelinek 	}
30090209230bSgjelinek 
30100209230bSgjelinek 	/*
30110209230bSgjelinek 	 * unqualified device removal.  remove all devices but prompt if more
30120209230bSgjelinek 	 * than one.
30130209230bSgjelinek 	 */
30140209230bSgjelinek 	if (!prompt_remove_resource(cmd, "device"))
30150209230bSgjelinek 		return;
30160209230bSgjelinek 
30170209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "device")) != Z_OK)
3018bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_DEVICE, err, B_TRUE);
30190209230bSgjelinek 	else
3020bbec428eSgjelinek 		need_to_commit = B_TRUE;
30210209230bSgjelinek }
30220209230bSgjelinek 
30230209230bSgjelinek static void
30240209230bSgjelinek remove_attr(cmd_t *cmd)
30250209230bSgjelinek {
30260209230bSgjelinek 	int err;
30270209230bSgjelinek 
30280209230bSgjelinek 	/* traditional, qualified attr removal */
30290209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
30300209230bSgjelinek 		struct zone_attrtab attrtab;
30310209230bSgjelinek 
3032bbec428eSgjelinek 		if ((err = fill_in_attrtab(cmd, &attrtab, B_FALSE)) != Z_OK) {
3033bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_ATTR, err, B_TRUE);
30340209230bSgjelinek 			return;
30350209230bSgjelinek 		}
30360209230bSgjelinek 		if ((err = zonecfg_delete_attr(handle, &attrtab)) != Z_OK)
3037bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_ATTR, err, B_TRUE);
30380209230bSgjelinek 		else
3039bbec428eSgjelinek 			need_to_commit = B_TRUE;
30400209230bSgjelinek 		return;
30410209230bSgjelinek 	}
30420209230bSgjelinek 
30430209230bSgjelinek 	/*
30440209230bSgjelinek 	 * unqualified attr removal.  remove all attrs but prompt if more
30450209230bSgjelinek 	 * than one.
30460209230bSgjelinek 	 */
30470209230bSgjelinek 	if (!prompt_remove_resource(cmd, "attr"))
30480209230bSgjelinek 		return;
30490209230bSgjelinek 
30500209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "attr")) != Z_OK)
3051bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_ATTR, err, B_TRUE);
30520209230bSgjelinek 	else
3053bbec428eSgjelinek 		need_to_commit = B_TRUE;
30540209230bSgjelinek }
30550209230bSgjelinek 
30560209230bSgjelinek static void
30570209230bSgjelinek remove_dataset(cmd_t *cmd)
30580209230bSgjelinek {
30590209230bSgjelinek 	int err;
30600209230bSgjelinek 
30610209230bSgjelinek 	/* traditional, qualified dataset removal */
30620209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
30630209230bSgjelinek 		struct zone_dstab dstab;
30640209230bSgjelinek 
3065bbec428eSgjelinek 		if ((err = fill_in_dstab(cmd, &dstab, B_FALSE)) != Z_OK) {
3066bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_DATASET, err, B_TRUE);
30670209230bSgjelinek 			return;
30680209230bSgjelinek 		}
30690209230bSgjelinek 		if ((err = zonecfg_delete_ds(handle, &dstab)) != Z_OK)
3070bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_DATASET, err, B_TRUE);
30710209230bSgjelinek 		else
3072bbec428eSgjelinek 			need_to_commit = B_TRUE;
30730209230bSgjelinek 		return;
30740209230bSgjelinek 	}
30750209230bSgjelinek 
30760209230bSgjelinek 	/*
30770209230bSgjelinek 	 * unqualified dataset removal.  remove all datasets but prompt if more
30780209230bSgjelinek 	 * than one.
30790209230bSgjelinek 	 */
30800209230bSgjelinek 	if (!prompt_remove_resource(cmd, "dataset"))
30810209230bSgjelinek 		return;
30820209230bSgjelinek 
30830209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "dataset")) != Z_OK)
3084bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_DATASET, err, B_TRUE);
30850209230bSgjelinek 	else
3086bbec428eSgjelinek 		need_to_commit = B_TRUE;
30870209230bSgjelinek }
30880209230bSgjelinek 
30890209230bSgjelinek static void
30900209230bSgjelinek remove_rctl(cmd_t *cmd)
30910209230bSgjelinek {
30920209230bSgjelinek 	int err;
30930209230bSgjelinek 
30940209230bSgjelinek 	/* traditional, qualified rctl removal */
30950209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
30960209230bSgjelinek 		struct zone_rctltab rctltab;
30970209230bSgjelinek 
3098bbec428eSgjelinek 		if ((err = fill_in_rctltab(cmd, &rctltab, B_FALSE)) != Z_OK) {
3099bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_RCTL, err, B_TRUE);
31007c478bd9Sstevel@tonic-gate 			return;
31017c478bd9Sstevel@tonic-gate 		}
31027c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_delete_rctl(handle, &rctltab)) != Z_OK)
3103bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_RCTL, err, B_TRUE);
31047c478bd9Sstevel@tonic-gate 		else
3105bbec428eSgjelinek 			need_to_commit = B_TRUE;
31067c478bd9Sstevel@tonic-gate 		zonecfg_free_rctl_value_list(rctltab.zone_rctl_valptr);
31077c478bd9Sstevel@tonic-gate 		return;
31087c478bd9Sstevel@tonic-gate 	}
31090209230bSgjelinek 
31100209230bSgjelinek 	/*
31110209230bSgjelinek 	 * unqualified rctl removal.  remove all rctls but prompt if more
31120209230bSgjelinek 	 * than one.
31130209230bSgjelinek 	 */
31140209230bSgjelinek 	if (!prompt_remove_resource(cmd, "rctl"))
31150209230bSgjelinek 		return;
31160209230bSgjelinek 
31170209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "rctl")) != Z_OK)
3118bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_RCTL, err, B_TRUE);
31197c478bd9Sstevel@tonic-gate 	else
3120bbec428eSgjelinek 		need_to_commit = B_TRUE;
31210209230bSgjelinek }
31220209230bSgjelinek 
31230209230bSgjelinek static void
31240209230bSgjelinek remove_pset()
31250209230bSgjelinek {
31260209230bSgjelinek 	int err;
31270209230bSgjelinek 	struct zone_psettab psettab;
31280209230bSgjelinek 
31290209230bSgjelinek 	if ((err = zonecfg_lookup_pset(handle, &psettab)) != Z_OK) {
3130bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_DCPU, err, B_TRUE);
31310209230bSgjelinek 		return;
31320209230bSgjelinek 	}
31330209230bSgjelinek 	if ((err = zonecfg_delete_pset(handle)) != Z_OK)
3134bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_DCPU, err, B_TRUE);
31350209230bSgjelinek 	else
3136bbec428eSgjelinek 		need_to_commit = B_TRUE;
31370209230bSgjelinek }
31380209230bSgjelinek 
31390209230bSgjelinek static void
3140c97ad5cdSakolb remove_pcap()
3141c97ad5cdSakolb {
3142c97ad5cdSakolb 	int err;
3143c97ad5cdSakolb 	uint64_t tmp;
3144c97ad5cdSakolb 
3145c97ad5cdSakolb 	if (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &tmp) != Z_OK) {
3146c97ad5cdSakolb 		zerr("%s %s: %s", cmd_to_str(CMD_REMOVE), rt_to_str(RT_PCAP),
3147c97ad5cdSakolb 		    zonecfg_strerror(Z_NO_RESOURCE_TYPE));
3148bbec428eSgjelinek 		saw_error = B_TRUE;
3149c97ad5cdSakolb 		return;
3150c97ad5cdSakolb 	}
3151c97ad5cdSakolb 
3152c97ad5cdSakolb 	if ((err = zonecfg_rm_aliased_rctl(handle, ALIAS_CPUCAP)) != Z_OK)
3153bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_PCAP, err, B_TRUE);
3154c97ad5cdSakolb 	else
3155bbec428eSgjelinek 		need_to_commit = B_TRUE;
3156c97ad5cdSakolb }
3157c97ad5cdSakolb 
3158c97ad5cdSakolb static void
31590209230bSgjelinek remove_mcap()
31600209230bSgjelinek {
31610209230bSgjelinek 	int err, res1, res2, res3;
31620209230bSgjelinek 	uint64_t tmp;
31630209230bSgjelinek 	struct zone_mcaptab mcaptab;
31640209230bSgjelinek 	boolean_t revert = B_FALSE;
31650209230bSgjelinek 
31660209230bSgjelinek 	res1 = zonecfg_lookup_mcap(handle, &mcaptab);
31670209230bSgjelinek 	res2 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP, &tmp);
31680209230bSgjelinek 	res3 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM, &tmp);
31690209230bSgjelinek 
31700209230bSgjelinek 	/* if none of these exist, there is no resource to remove */
31710209230bSgjelinek 	if (res1 != Z_OK && res2 != Z_OK && res3 != Z_OK) {
31720209230bSgjelinek 		zerr("%s %s: %s", cmd_to_str(CMD_REMOVE), rt_to_str(RT_MCAP),
31730209230bSgjelinek 		    zonecfg_strerror(Z_NO_RESOURCE_TYPE));
3174bbec428eSgjelinek 		saw_error = B_TRUE;
31750209230bSgjelinek 		return;
31760209230bSgjelinek 	}
31770209230bSgjelinek 	if (res1 == Z_OK) {
31780209230bSgjelinek 		if ((err = zonecfg_delete_mcap(handle)) != Z_OK) {
3179bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_MCAP, err, B_TRUE);
31800209230bSgjelinek 			revert = B_TRUE;
31810209230bSgjelinek 		} else {
3182bbec428eSgjelinek 			need_to_commit = B_TRUE;
31830209230bSgjelinek 		}
31840209230bSgjelinek 	}
31850209230bSgjelinek 	if (res2 == Z_OK) {
31860209230bSgjelinek 		if ((err = zonecfg_rm_aliased_rctl(handle, ALIAS_MAXSWAP))
31870209230bSgjelinek 		    != Z_OK) {
3188bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_MCAP, err, B_TRUE);
31890209230bSgjelinek 			revert = B_TRUE;
31900209230bSgjelinek 		} else {
3191bbec428eSgjelinek 			need_to_commit = B_TRUE;
31920209230bSgjelinek 		}
31930209230bSgjelinek 	}
31940209230bSgjelinek 	if (res3 == Z_OK) {
31950209230bSgjelinek 		if ((err = zonecfg_rm_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM))
31960209230bSgjelinek 		    != Z_OK) {
3197bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_MCAP, err, B_TRUE);
31980209230bSgjelinek 			revert = B_TRUE;
31990209230bSgjelinek 		} else {
3200bbec428eSgjelinek 			need_to_commit = B_TRUE;
32010209230bSgjelinek 		}
32020209230bSgjelinek 	}
32030209230bSgjelinek 
32040209230bSgjelinek 	if (revert)
3205bbec428eSgjelinek 		need_to_commit = B_FALSE;
32060209230bSgjelinek }
32070209230bSgjelinek 
32080209230bSgjelinek static void
3209cb8a054bSGlenn Faden remove_admin(cmd_t *cmd)
3210cb8a054bSGlenn Faden {
3211cb8a054bSGlenn Faden 	int err;
3212cb8a054bSGlenn Faden 
3213cb8a054bSGlenn Faden 	/* traditional, qualified attr removal */
3214cb8a054bSGlenn Faden 	if (cmd->cmd_prop_nv_pairs > 0) {
3215cb8a054bSGlenn Faden 		struct zone_admintab admintab;
3216cb8a054bSGlenn Faden 
3217cb8a054bSGlenn Faden 		if ((err = fill_in_admintab(cmd, &admintab, B_FALSE)) != Z_OK) {
3218cb8a054bSGlenn Faden 			z_cmd_rt_perror(CMD_REMOVE, RT_ADMIN,
3219cb8a054bSGlenn Faden 			    err, B_TRUE);
3220cb8a054bSGlenn Faden 			return;
3221cb8a054bSGlenn Faden 		}
3222cb8a054bSGlenn Faden 		if ((err = zonecfg_delete_admin(handle, &admintab,
3223cb8a054bSGlenn Faden 		    zone))
3224cb8a054bSGlenn Faden 		    != Z_OK)
3225cb8a054bSGlenn Faden 			z_cmd_rt_perror(CMD_REMOVE, RT_ADMIN,
3226cb8a054bSGlenn Faden 			    err, B_TRUE);
3227cb8a054bSGlenn Faden 		else
3228cb8a054bSGlenn Faden 			need_to_commit = B_TRUE;
3229cb8a054bSGlenn Faden 		return;
3230cb8a054bSGlenn Faden 	} else {
3231cb8a054bSGlenn Faden 		/*
3232cb8a054bSGlenn Faden 		 * unqualified admin removal.
3233cb8a054bSGlenn Faden 		 * remove all admins but prompt if more
3234cb8a054bSGlenn Faden 		 * than one.
3235cb8a054bSGlenn Faden 		 */
3236cb8a054bSGlenn Faden 		if (!prompt_remove_resource(cmd, "admin"))
3237cb8a054bSGlenn Faden 			return;
3238cb8a054bSGlenn Faden 
3239cb8a054bSGlenn Faden 		if ((err = zonecfg_delete_admins(handle, zone))
3240cb8a054bSGlenn Faden 		    != Z_OK)
3241cb8a054bSGlenn Faden 			z_cmd_rt_perror(CMD_REMOVE, RT_ADMIN,
3242cb8a054bSGlenn Faden 			    err, B_TRUE);
3243cb8a054bSGlenn Faden 		else
3244cb8a054bSGlenn Faden 			need_to_commit = B_TRUE;
3245cb8a054bSGlenn Faden 	}
3246cb8a054bSGlenn Faden }
3247cb8a054bSGlenn Faden 
3248cb8a054bSGlenn Faden static void
32490209230bSgjelinek remove_resource(cmd_t *cmd)
32500209230bSgjelinek {
32510209230bSgjelinek 	int type;
32520209230bSgjelinek 	int arg;
3253bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
32540209230bSgjelinek 
32550209230bSgjelinek 	if ((type = cmd->cmd_res_type) == RT_UNKNOWN) {
3256bbec428eSgjelinek 		long_usage(CMD_REMOVE, B_TRUE);
32570209230bSgjelinek 		return;
32580209230bSgjelinek 	}
32590209230bSgjelinek 
32600209230bSgjelinek 	optind = 0;
32610209230bSgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?F")) != EOF) {
32620209230bSgjelinek 		switch (arg) {
32630209230bSgjelinek 		case '?':
32640209230bSgjelinek 			longer_usage(CMD_REMOVE);
3265bbec428eSgjelinek 			arg_err = B_TRUE;
32667ec75eb8Sgjelinek 			break;
32670209230bSgjelinek 		case 'F':
32680209230bSgjelinek 			break;
32690209230bSgjelinek 		default:
32700209230bSgjelinek 			short_usage(CMD_REMOVE);
3271bbec428eSgjelinek 			arg_err = B_TRUE;
32727ec75eb8Sgjelinek 			break;
32737ec75eb8Sgjelinek 		}
32747ec75eb8Sgjelinek 	}
32757ec75eb8Sgjelinek 	if (arg_err)
32760209230bSgjelinek 		return;
32770209230bSgjelinek 
3278bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
32790209230bSgjelinek 		return;
32800209230bSgjelinek 
32810209230bSgjelinek 	switch (type) {
32820209230bSgjelinek 	case RT_FS:
32830209230bSgjelinek 		remove_fs(cmd);
32840209230bSgjelinek 		return;
32850209230bSgjelinek 	case RT_NET:
32860209230bSgjelinek 		remove_net(cmd);
32870209230bSgjelinek 		return;
32880209230bSgjelinek 	case RT_DEVICE:
32890209230bSgjelinek 		remove_device(cmd);
32900209230bSgjelinek 		return;
32910209230bSgjelinek 	case RT_RCTL:
32920209230bSgjelinek 		remove_rctl(cmd);
32930209230bSgjelinek 		return;
32940209230bSgjelinek 	case RT_ATTR:
32950209230bSgjelinek 		remove_attr(cmd);
32967c478bd9Sstevel@tonic-gate 		return;
3297fa9e4066Sahrens 	case RT_DATASET:
32980209230bSgjelinek 		remove_dataset(cmd);
3299fa9e4066Sahrens 		return;
33000209230bSgjelinek 	case RT_DCPU:
33010209230bSgjelinek 		remove_pset();
33020209230bSgjelinek 		return;
3303c97ad5cdSakolb 	case RT_PCAP:
3304c97ad5cdSakolb 		remove_pcap();
3305c97ad5cdSakolb 		return;
33060209230bSgjelinek 	case RT_MCAP:
33070209230bSgjelinek 		remove_mcap();
3308fa9e4066Sahrens 		return;
3309cb8a054bSGlenn Faden 	case RT_ADMIN:
3310cb8a054bSGlenn Faden 		remove_admin(cmd);
3311cb8a054bSGlenn Faden 		return;
33127c478bd9Sstevel@tonic-gate 	default:
3313bbec428eSgjelinek 		zone_perror(rt_to_str(type), Z_NO_RESOURCE_TYPE, B_TRUE);
3314bbec428eSgjelinek 		long_usage(CMD_REMOVE, B_TRUE);
3315bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
33167c478bd9Sstevel@tonic-gate 		return;
33177c478bd9Sstevel@tonic-gate 	}
33187c478bd9Sstevel@tonic-gate }
33197c478bd9Sstevel@tonic-gate 
33207c478bd9Sstevel@tonic-gate static void
33217c478bd9Sstevel@tonic-gate remove_property(cmd_t *cmd)
33227c478bd9Sstevel@tonic-gate {
33237c478bd9Sstevel@tonic-gate 	char *prop_id;
33247c478bd9Sstevel@tonic-gate 	int err, res_type, prop_type;
33257c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
33267c478bd9Sstevel@tonic-gate 	struct zone_rctlvaltab *rctlvaltab;
33277c478bd9Sstevel@tonic-gate 	complex_property_ptr_t cx;
33287c478bd9Sstevel@tonic-gate 
33297c478bd9Sstevel@tonic-gate 	res_type = resource_scope;
33307c478bd9Sstevel@tonic-gate 	prop_type = cmd->cmd_prop_name[0];
33317c478bd9Sstevel@tonic-gate 	if (res_type == RT_UNKNOWN || prop_type == PT_UNKNOWN) {
3332bbec428eSgjelinek 		long_usage(CMD_REMOVE, B_TRUE);
33337c478bd9Sstevel@tonic-gate 		return;
33347c478bd9Sstevel@tonic-gate 	}
33357c478bd9Sstevel@tonic-gate 
33367c478bd9Sstevel@tonic-gate 	if (cmd->cmd_prop_nv_pairs != 1) {
3337bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
33387c478bd9Sstevel@tonic-gate 		return;
33397c478bd9Sstevel@tonic-gate 	}
33407c478bd9Sstevel@tonic-gate 
3341bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
33427c478bd9Sstevel@tonic-gate 		return;
33437c478bd9Sstevel@tonic-gate 
33447c478bd9Sstevel@tonic-gate 	switch (res_type) {
33457c478bd9Sstevel@tonic-gate 	case RT_FS:
33467c478bd9Sstevel@tonic-gate 		if (prop_type != PT_OPTIONS) {
33477c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
3348bbec428eSgjelinek 			    B_TRUE);
3349bbec428eSgjelinek 			long_usage(CMD_REMOVE, B_TRUE);
3350bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
33517c478bd9Sstevel@tonic-gate 			return;
33527c478bd9Sstevel@tonic-gate 		}
33537c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[0];
33547c478bd9Sstevel@tonic-gate 		if (pp->pv_type == PROP_VAL_COMPLEX) {
33557c478bd9Sstevel@tonic-gate 			zerr(gettext("A %s or %s value was expected here."),
33567c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_SIMPLE),
33577c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_LIST));
3358bbec428eSgjelinek 			saw_error = B_TRUE;
33597c478bd9Sstevel@tonic-gate 			return;
33607c478bd9Sstevel@tonic-gate 		}
33617c478bd9Sstevel@tonic-gate 		if (pp->pv_type == PROP_VAL_SIMPLE) {
33627c478bd9Sstevel@tonic-gate 			if (pp->pv_simple == NULL) {
3363bbec428eSgjelinek 				long_usage(CMD_ADD, B_TRUE);
33647c478bd9Sstevel@tonic-gate 				return;
33657c478bd9Sstevel@tonic-gate 			}
33667c478bd9Sstevel@tonic-gate 			prop_id = pp->pv_simple;
33677c478bd9Sstevel@tonic-gate 			err = zonecfg_remove_fs_option(&in_progress_fstab,
33687c478bd9Sstevel@tonic-gate 			    prop_id);
33697c478bd9Sstevel@tonic-gate 			if (err != Z_OK)
3370bbec428eSgjelinek 				zone_perror(pt_to_str(prop_type), err, B_TRUE);
33717c478bd9Sstevel@tonic-gate 		} else {
33727c478bd9Sstevel@tonic-gate 			list_property_ptr_t list;
33737c478bd9Sstevel@tonic-gate 
33747c478bd9Sstevel@tonic-gate 			for (list = pp->pv_list; list != NULL;
33757c478bd9Sstevel@tonic-gate 			    list = list->lp_next) {
33767c478bd9Sstevel@tonic-gate 				prop_id = list->lp_simple;
33777c478bd9Sstevel@tonic-gate 				if (prop_id == NULL)
33787c478bd9Sstevel@tonic-gate 					break;
33797c478bd9Sstevel@tonic-gate 				err = zonecfg_remove_fs_option(
33807c478bd9Sstevel@tonic-gate 				    &in_progress_fstab, prop_id);
33817c478bd9Sstevel@tonic-gate 				if (err != Z_OK)
33827c478bd9Sstevel@tonic-gate 					zone_perror(pt_to_str(prop_type), err,
3383bbec428eSgjelinek 					    B_TRUE);
33847c478bd9Sstevel@tonic-gate 			}
33857c478bd9Sstevel@tonic-gate 		}
33867c478bd9Sstevel@tonic-gate 		return;
33877c478bd9Sstevel@tonic-gate 	case RT_RCTL:
33887c478bd9Sstevel@tonic-gate 		if (prop_type != PT_VALUE) {
33897c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
3390bbec428eSgjelinek 			    B_TRUE);
3391bbec428eSgjelinek 			long_usage(CMD_REMOVE, B_TRUE);
3392bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
33937c478bd9Sstevel@tonic-gate 			return;
33947c478bd9Sstevel@tonic-gate 		}
33957c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[0];
33967c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_COMPLEX) {
33977c478bd9Sstevel@tonic-gate 			zerr(gettext("A %s value was expected here."),
33987c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_COMPLEX));
3399bbec428eSgjelinek 			saw_error = B_TRUE;
34007c478bd9Sstevel@tonic-gate 			return;
34017c478bd9Sstevel@tonic-gate 		}
34027c478bd9Sstevel@tonic-gate 		if ((rctlvaltab = alloc_rctlvaltab()) == NULL) {
3403bbec428eSgjelinek 			zone_perror(zone, Z_NOMEM, B_TRUE);
34047c478bd9Sstevel@tonic-gate 			exit(Z_ERR);
34057c478bd9Sstevel@tonic-gate 		}
34067c478bd9Sstevel@tonic-gate 		for (cx = pp->pv_complex; cx != NULL; cx = cx->cp_next) {
34077c478bd9Sstevel@tonic-gate 			switch (cx->cp_type) {
34087c478bd9Sstevel@tonic-gate 			case PT_PRIV:
34097c478bd9Sstevel@tonic-gate 				(void) strlcpy(rctlvaltab->zone_rctlval_priv,
34107c478bd9Sstevel@tonic-gate 				    cx->cp_value,
34117c478bd9Sstevel@tonic-gate 				    sizeof (rctlvaltab->zone_rctlval_priv));
34127c478bd9Sstevel@tonic-gate 				break;
34137c478bd9Sstevel@tonic-gate 			case PT_LIMIT:
34147c478bd9Sstevel@tonic-gate 				(void) strlcpy(rctlvaltab->zone_rctlval_limit,
34157c478bd9Sstevel@tonic-gate 				    cx->cp_value,
34167c478bd9Sstevel@tonic-gate 				    sizeof (rctlvaltab->zone_rctlval_limit));
34177c478bd9Sstevel@tonic-gate 				break;
34187c478bd9Sstevel@tonic-gate 			case PT_ACTION:
34197c478bd9Sstevel@tonic-gate 				(void) strlcpy(rctlvaltab->zone_rctlval_action,
34207c478bd9Sstevel@tonic-gate 				    cx->cp_value,
34217c478bd9Sstevel@tonic-gate 				    sizeof (rctlvaltab->zone_rctlval_action));
34227c478bd9Sstevel@tonic-gate 				break;
34237c478bd9Sstevel@tonic-gate 			default:
34247c478bd9Sstevel@tonic-gate 				zone_perror(pt_to_str(prop_type),
3425bbec428eSgjelinek 				    Z_NO_PROPERTY_TYPE, B_TRUE);
3426bbec428eSgjelinek 				long_usage(CMD_ADD, B_TRUE);
3427bbec428eSgjelinek 				usage(B_FALSE, HELP_PROPS);
34287c478bd9Sstevel@tonic-gate 				zonecfg_free_rctl_value_list(rctlvaltab);
34297c478bd9Sstevel@tonic-gate 				return;
34307c478bd9Sstevel@tonic-gate 			}
34317c478bd9Sstevel@tonic-gate 		}
34327c478bd9Sstevel@tonic-gate 		rctlvaltab->zone_rctlval_next = NULL;
34337c478bd9Sstevel@tonic-gate 		err = zonecfg_remove_rctl_value(&in_progress_rctltab,
34347c478bd9Sstevel@tonic-gate 		    rctlvaltab);
34357c478bd9Sstevel@tonic-gate 		if (err != Z_OK)
3436bbec428eSgjelinek 			zone_perror(pt_to_str(prop_type), err, B_TRUE);
34377c478bd9Sstevel@tonic-gate 		zonecfg_free_rctl_value_list(rctlvaltab);
34387c478bd9Sstevel@tonic-gate 		return;
3439de860bd9Sgfaden 	case RT_NET:
3440de860bd9Sgfaden 		if (prop_type != PT_DEFROUTER) {
3441de860bd9Sgfaden 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
3442bbec428eSgjelinek 			    B_TRUE);
3443bbec428eSgjelinek 			long_usage(CMD_REMOVE, B_TRUE);
3444bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
3445de860bd9Sgfaden 			return;
3446de860bd9Sgfaden 		} else {
3447de860bd9Sgfaden 			bzero(&in_progress_nwiftab.zone_nwif_defrouter,
3448de860bd9Sgfaden 			    sizeof (in_progress_nwiftab.zone_nwif_defrouter));
3449de860bd9Sgfaden 			return;
3450de860bd9Sgfaden 		}
34517c478bd9Sstevel@tonic-gate 	default:
3452bbec428eSgjelinek 		zone_perror(rt_to_str(res_type), Z_NO_RESOURCE_TYPE, B_TRUE);
3453bbec428eSgjelinek 		long_usage(CMD_REMOVE, B_TRUE);
3454bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
34557c478bd9Sstevel@tonic-gate 		return;
34567c478bd9Sstevel@tonic-gate 	}
34577c478bd9Sstevel@tonic-gate }
34587c478bd9Sstevel@tonic-gate 
34597c478bd9Sstevel@tonic-gate void
34607c478bd9Sstevel@tonic-gate remove_func(cmd_t *cmd)
34617c478bd9Sstevel@tonic-gate {
34627c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_REMOVE))
34637c478bd9Sstevel@tonic-gate 		return;
34647c478bd9Sstevel@tonic-gate 
34657c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
34667c478bd9Sstevel@tonic-gate 
34670209230bSgjelinek 	if (global_scope) {
34680209230bSgjelinek 		if (gz_invalid_resource(cmd->cmd_res_type)) {
34690209230bSgjelinek 			zerr(gettext("%s is not a valid resource for the "
34700209230bSgjelinek 			    "global zone."), rt_to_str(cmd->cmd_res_type));
3471bbec428eSgjelinek 			saw_error = B_TRUE;
34720209230bSgjelinek 			return;
34730209230bSgjelinek 		}
34747c478bd9Sstevel@tonic-gate 		remove_resource(cmd);
34750209230bSgjelinek 	} else {
34767c478bd9Sstevel@tonic-gate 		remove_property(cmd);
34777c478bd9Sstevel@tonic-gate 	}
34780209230bSgjelinek }
34790209230bSgjelinek 
34800209230bSgjelinek static void
34810209230bSgjelinek clear_property(cmd_t *cmd)
34820209230bSgjelinek {
34830209230bSgjelinek 	int res_type, prop_type;
34840209230bSgjelinek 
34850209230bSgjelinek 	res_type = resource_scope;
34860209230bSgjelinek 	prop_type = cmd->cmd_res_type;
34870209230bSgjelinek 	if (res_type == RT_UNKNOWN || prop_type == PT_UNKNOWN) {
3488bbec428eSgjelinek 		long_usage(CMD_CLEAR, B_TRUE);
34890209230bSgjelinek 		return;
34900209230bSgjelinek 	}
34910209230bSgjelinek 
3492bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
34930209230bSgjelinek 		return;
34940209230bSgjelinek 
34950209230bSgjelinek 	switch (res_type) {
34960209230bSgjelinek 	case RT_FS:
34970209230bSgjelinek 		if (prop_type == PT_RAW) {
34980209230bSgjelinek 			in_progress_fstab.zone_fs_raw[0] = '\0';
3499bbec428eSgjelinek 			need_to_commit = B_TRUE;
35000209230bSgjelinek 			return;
35010209230bSgjelinek 		}
35020209230bSgjelinek 		break;
35030209230bSgjelinek 	case RT_DCPU:
35040209230bSgjelinek 		if (prop_type == PT_IMPORTANCE) {
35050209230bSgjelinek 			in_progress_psettab.zone_importance[0] = '\0';
3506bbec428eSgjelinek 			need_to_commit = B_TRUE;
35070209230bSgjelinek 			return;
35080209230bSgjelinek 		}
35090209230bSgjelinek 		break;
35100209230bSgjelinek 	case RT_MCAP:
35110209230bSgjelinek 		switch (prop_type) {
35120209230bSgjelinek 		case PT_PHYSICAL:
35130209230bSgjelinek 			in_progress_mcaptab.zone_physmem_cap[0] = '\0';
3514bbec428eSgjelinek 			need_to_commit = B_TRUE;
35150209230bSgjelinek 			return;
35160209230bSgjelinek 		case PT_SWAP:
35170209230bSgjelinek 			remove_aliased_rctl(PT_SWAP, ALIAS_MAXSWAP);
35180209230bSgjelinek 			return;
35190209230bSgjelinek 		case PT_LOCKED:
35200209230bSgjelinek 			remove_aliased_rctl(PT_LOCKED, ALIAS_MAXLOCKEDMEM);
35210209230bSgjelinek 			return;
35220209230bSgjelinek 		}
35230209230bSgjelinek 		break;
35240209230bSgjelinek 	default:
35250209230bSgjelinek 		break;
35260209230bSgjelinek 	}
35270209230bSgjelinek 
3528bbec428eSgjelinek 	zone_perror(pt_to_str(prop_type), Z_CLEAR_DISALLOW, B_TRUE);
35290209230bSgjelinek }
35300209230bSgjelinek 
35310209230bSgjelinek static void
35320209230bSgjelinek clear_global(cmd_t *cmd)
35330209230bSgjelinek {
35340209230bSgjelinek 	int err, type;
35350209230bSgjelinek 
35360209230bSgjelinek 	if ((type = cmd->cmd_res_type) == RT_UNKNOWN) {
3537bbec428eSgjelinek 		long_usage(CMD_CLEAR, B_TRUE);
35380209230bSgjelinek 		return;
35390209230bSgjelinek 	}
35400209230bSgjelinek 
3541bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
35420209230bSgjelinek 		return;
35430209230bSgjelinek 
35440209230bSgjelinek 	switch (type) {
35450209230bSgjelinek 	case PT_ZONENAME:
35460209230bSgjelinek 		/* FALLTHRU */
35470209230bSgjelinek 	case PT_ZONEPATH:
35480209230bSgjelinek 		/* FALLTHRU */
35490209230bSgjelinek 	case PT_BRAND:
3550bbec428eSgjelinek 		zone_perror(pt_to_str(type), Z_CLEAR_DISALLOW, B_TRUE);
35510209230bSgjelinek 		return;
35520209230bSgjelinek 	case PT_AUTOBOOT:
35530209230bSgjelinek 		/* false is default; we'll treat as equivalent to clearing */
35540209230bSgjelinek 		if ((err = zonecfg_set_autoboot(handle, B_FALSE)) != Z_OK)
3555bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_AUTOBOOT, err, B_TRUE);
35560209230bSgjelinek 		else
3557bbec428eSgjelinek 			need_to_commit = B_TRUE;
35580209230bSgjelinek 		return;
35590209230bSgjelinek 	case PT_POOL:
35600209230bSgjelinek 		if ((err = zonecfg_set_pool(handle, NULL)) != Z_OK)
3561bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_POOL, err, B_TRUE);
35620209230bSgjelinek 		else
3563bbec428eSgjelinek 			need_to_commit = B_TRUE;
35640209230bSgjelinek 		return;
35650209230bSgjelinek 	case PT_LIMITPRIV:
35660209230bSgjelinek 		if ((err = zonecfg_set_limitpriv(handle, NULL)) != Z_OK)
3567bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_LIMITPRIV, err, B_TRUE);
35680209230bSgjelinek 		else
3569bbec428eSgjelinek 			need_to_commit = B_TRUE;
35700209230bSgjelinek 		return;
35710209230bSgjelinek 	case PT_BOOTARGS:
35720209230bSgjelinek 		if ((err = zonecfg_set_bootargs(handle, NULL)) != Z_OK)
3573bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_BOOTARGS, err, B_TRUE);
35740209230bSgjelinek 		else
3575bbec428eSgjelinek 			need_to_commit = B_TRUE;
35760209230bSgjelinek 		return;
35770209230bSgjelinek 	case PT_SCHED:
35780209230bSgjelinek 		if ((err = zonecfg_set_sched(handle, NULL)) != Z_OK)
3579bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_SCHED, err, B_TRUE);
35800209230bSgjelinek 		else
3581bbec428eSgjelinek 			need_to_commit = B_TRUE;
35820209230bSgjelinek 		return;
3583f4b3ec61Sdh155122 	case PT_IPTYPE:
3584f4b3ec61Sdh155122 		/* shared is default; we'll treat as equivalent to clearing */
3585f4b3ec61Sdh155122 		if ((err = zonecfg_set_iptype(handle, ZS_SHARED)) != Z_OK)
3586bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_IPTYPE, err, B_TRUE);
3587f4b3ec61Sdh155122 		else
3588bbec428eSgjelinek 			need_to_commit = B_TRUE;
3589f4b3ec61Sdh155122 		return;
35900209230bSgjelinek 	case PT_MAXLWPS:
35910209230bSgjelinek 		remove_aliased_rctl(PT_MAXLWPS, ALIAS_MAXLWPS);
35920209230bSgjelinek 		return;
3593ff19e029SMenno Lageman 	case PT_MAXPROCS:
3594ff19e029SMenno Lageman 		remove_aliased_rctl(PT_MAXPROCS, ALIAS_MAXPROCS);
3595ff19e029SMenno Lageman 		return;
35960209230bSgjelinek 	case PT_MAXSHMMEM:
35970209230bSgjelinek 		remove_aliased_rctl(PT_MAXSHMMEM, ALIAS_MAXSHMMEM);
35980209230bSgjelinek 		return;
35990209230bSgjelinek 	case PT_MAXSHMIDS:
36000209230bSgjelinek 		remove_aliased_rctl(PT_MAXSHMIDS, ALIAS_MAXSHMIDS);
36010209230bSgjelinek 		return;
36020209230bSgjelinek 	case PT_MAXMSGIDS:
36030209230bSgjelinek 		remove_aliased_rctl(PT_MAXMSGIDS, ALIAS_MAXMSGIDS);
36040209230bSgjelinek 		return;
36050209230bSgjelinek 	case PT_MAXSEMIDS:
36060209230bSgjelinek 		remove_aliased_rctl(PT_MAXSEMIDS, ALIAS_MAXSEMIDS);
36070209230bSgjelinek 		return;
36080209230bSgjelinek 	case PT_SHARES:
36090209230bSgjelinek 		remove_aliased_rctl(PT_SHARES, ALIAS_SHARES);
36100209230bSgjelinek 		return;
36115679c89fSjv227347 	case PT_HOSTID:
36125679c89fSjv227347 		if ((err = zonecfg_set_hostid(handle, NULL)) != Z_OK)
36135679c89fSjv227347 			z_cmd_rt_perror(CMD_CLEAR, RT_HOSTID, err, B_TRUE);
36145679c89fSjv227347 		else
36155679c89fSjv227347 			need_to_commit = B_TRUE;
36165679c89fSjv227347 		return;
36170fbb751dSJohn Levon 	case PT_FS_ALLOWED:
36180fbb751dSJohn Levon 		if ((err = zonecfg_set_fs_allowed(handle, NULL)) != Z_OK)
36190fbb751dSJohn Levon 			z_cmd_rt_perror(CMD_CLEAR, RT_FS_ALLOWED, err, B_TRUE);
36200fbb751dSJohn Levon 		else
36210fbb751dSJohn Levon 			need_to_commit = B_TRUE;
36220fbb751dSJohn Levon 		return;
36230209230bSgjelinek 	default:
3624bbec428eSgjelinek 		zone_perror(pt_to_str(type), Z_NO_PROPERTY_TYPE, B_TRUE);
3625bbec428eSgjelinek 		long_usage(CMD_CLEAR, B_TRUE);
3626bbec428eSgjelinek 		usage(B_FALSE, HELP_PROPS);
36270209230bSgjelinek 		return;
36280209230bSgjelinek 	}
36290209230bSgjelinek }
36300209230bSgjelinek 
36310209230bSgjelinek void
36320209230bSgjelinek clear_func(cmd_t *cmd)
36330209230bSgjelinek {
36340209230bSgjelinek 	if (zone_is_read_only(CMD_CLEAR))
36350209230bSgjelinek 		return;
36360209230bSgjelinek 
36370209230bSgjelinek 	assert(cmd != NULL);
36380209230bSgjelinek 
36390209230bSgjelinek 	if (global_scope) {
36400209230bSgjelinek 		if (gz_invalid_property(cmd->cmd_res_type)) {
36410209230bSgjelinek 			zerr(gettext("%s is not a valid property for the "
36420209230bSgjelinek 			    "global zone."), pt_to_str(cmd->cmd_res_type));
3643bbec428eSgjelinek 			saw_error = B_TRUE;
36440209230bSgjelinek 			return;
36450209230bSgjelinek 		}
36460209230bSgjelinek 
36470209230bSgjelinek 		clear_global(cmd);
36480209230bSgjelinek 	} else {
36490209230bSgjelinek 		clear_property(cmd);
36500209230bSgjelinek 	}
36510209230bSgjelinek }
36527c478bd9Sstevel@tonic-gate 
36537c478bd9Sstevel@tonic-gate void
36547c478bd9Sstevel@tonic-gate select_func(cmd_t *cmd)
36557c478bd9Sstevel@tonic-gate {
36560209230bSgjelinek 	int type, err, res;
36570209230bSgjelinek 	uint64_t limit;
3658c97ad5cdSakolb 	uint64_t tmp;
36597c478bd9Sstevel@tonic-gate 
36607c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_SELECT))
36617c478bd9Sstevel@tonic-gate 		return;
36627c478bd9Sstevel@tonic-gate 
36637c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
36647c478bd9Sstevel@tonic-gate 
36657c478bd9Sstevel@tonic-gate 	if (global_scope) {
3666bbec428eSgjelinek 		global_scope = B_FALSE;
36677c478bd9Sstevel@tonic-gate 		resource_scope = cmd->cmd_res_type;
36687c478bd9Sstevel@tonic-gate 		end_op = CMD_SELECT;
36697c478bd9Sstevel@tonic-gate 	} else {
36707c478bd9Sstevel@tonic-gate 		scope_usage(CMD_SELECT);
36717c478bd9Sstevel@tonic-gate 		return;
36727c478bd9Sstevel@tonic-gate 	}
36737c478bd9Sstevel@tonic-gate 
36747c478bd9Sstevel@tonic-gate 	if ((type = cmd->cmd_res_type) == RT_UNKNOWN) {
3675bbec428eSgjelinek 		long_usage(CMD_SELECT, B_TRUE);
36767c478bd9Sstevel@tonic-gate 		return;
36777c478bd9Sstevel@tonic-gate 	}
36787c478bd9Sstevel@tonic-gate 
3679bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
36807c478bd9Sstevel@tonic-gate 		return;
36817c478bd9Sstevel@tonic-gate 
36827c478bd9Sstevel@tonic-gate 	switch (type) {
36837c478bd9Sstevel@tonic-gate 	case RT_FS:
3684bbec428eSgjelinek 		if ((err = fill_in_fstab(cmd, &old_fstab, B_FALSE)) != Z_OK) {
3685bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_FS, err, B_TRUE);
3686bbec428eSgjelinek 			global_scope = B_TRUE;
36877c478bd9Sstevel@tonic-gate 		}
36887c478bd9Sstevel@tonic-gate 		bcopy(&old_fstab, &in_progress_fstab,
36897c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_fstab));
36907c478bd9Sstevel@tonic-gate 		return;
36917c478bd9Sstevel@tonic-gate 	case RT_NET:
3692bbec428eSgjelinek 		if ((err = fill_in_nwiftab(cmd, &old_nwiftab, B_FALSE))
3693bbec428eSgjelinek 		    != Z_OK) {
3694bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_NET, err, B_TRUE);
3695bbec428eSgjelinek 			global_scope = B_TRUE;
36967c478bd9Sstevel@tonic-gate 		}
36977c478bd9Sstevel@tonic-gate 		bcopy(&old_nwiftab, &in_progress_nwiftab,
36987c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_nwiftab));
36997c478bd9Sstevel@tonic-gate 		return;
37007c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
3701bbec428eSgjelinek 		if ((err = fill_in_devtab(cmd, &old_devtab, B_FALSE)) != Z_OK) {
3702bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_DEVICE, err, B_TRUE);
3703bbec428eSgjelinek 			global_scope = B_TRUE;
37047c478bd9Sstevel@tonic-gate 		}
37057c478bd9Sstevel@tonic-gate 		bcopy(&old_devtab, &in_progress_devtab,
37067c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_devtab));
37077c478bd9Sstevel@tonic-gate 		return;
37087c478bd9Sstevel@tonic-gate 	case RT_RCTL:
3709bbec428eSgjelinek 		if ((err = fill_in_rctltab(cmd, &old_rctltab, B_FALSE))
3710bbec428eSgjelinek 		    != Z_OK) {
3711bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_RCTL, err, B_TRUE);
3712bbec428eSgjelinek 			global_scope = B_TRUE;
37137c478bd9Sstevel@tonic-gate 		}
37147c478bd9Sstevel@tonic-gate 		bcopy(&old_rctltab, &in_progress_rctltab,
37157c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_rctltab));
37167c478bd9Sstevel@tonic-gate 		return;
37177c478bd9Sstevel@tonic-gate 	case RT_ATTR:
3718bbec428eSgjelinek 		if ((err = fill_in_attrtab(cmd, &old_attrtab, B_FALSE))
3719bbec428eSgjelinek 		    != Z_OK) {
3720bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_ATTR, err, B_TRUE);
3721bbec428eSgjelinek 			global_scope = B_TRUE;
37227c478bd9Sstevel@tonic-gate 		}
37237c478bd9Sstevel@tonic-gate 		bcopy(&old_attrtab, &in_progress_attrtab,
37247c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_attrtab));
37257c478bd9Sstevel@tonic-gate 		return;
3726fa9e4066Sahrens 	case RT_DATASET:
3727bbec428eSgjelinek 		if ((err = fill_in_dstab(cmd, &old_dstab, B_FALSE)) != Z_OK) {
3728bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_DATASET, err, B_TRUE);
3729bbec428eSgjelinek 			global_scope = B_TRUE;
3730fa9e4066Sahrens 		}
3731fa9e4066Sahrens 		bcopy(&old_dstab, &in_progress_dstab,
3732fa9e4066Sahrens 		    sizeof (struct zone_dstab));
3733fa9e4066Sahrens 		return;
37340209230bSgjelinek 	case RT_DCPU:
37350209230bSgjelinek 		if ((err = zonecfg_lookup_pset(handle, &old_psettab)) != Z_OK) {
3736bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_DCPU, err, B_TRUE);
3737bbec428eSgjelinek 			global_scope = B_TRUE;
37380209230bSgjelinek 		}
37390209230bSgjelinek 		bcopy(&old_psettab, &in_progress_psettab,
37400209230bSgjelinek 		    sizeof (struct zone_psettab));
37410209230bSgjelinek 		return;
3742c97ad5cdSakolb 	case RT_PCAP:
3743c97ad5cdSakolb 		if ((err = zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &tmp))
3744c97ad5cdSakolb 		    != Z_OK) {
3745bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_PCAP, err, B_TRUE);
3746bbec428eSgjelinek 			global_scope = B_TRUE;
3747c97ad5cdSakolb 		}
3748c97ad5cdSakolb 		return;
37490209230bSgjelinek 	case RT_MCAP:
37500209230bSgjelinek 		/* if none of these exist, there is no resource to select */
37510209230bSgjelinek 		if ((res = zonecfg_lookup_mcap(handle, &old_mcaptab)) != Z_OK &&
37520209230bSgjelinek 		    zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP, &limit)
37530209230bSgjelinek 		    != Z_OK &&
37540209230bSgjelinek 		    zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM, &limit)
37550209230bSgjelinek 		    != Z_OK) {
37560209230bSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_MCAP, Z_NO_RESOURCE_TYPE,
3757bbec428eSgjelinek 			    B_TRUE);
3758bbec428eSgjelinek 			global_scope = B_TRUE;
37590209230bSgjelinek 		}
37600209230bSgjelinek 		if (res == Z_OK)
37610209230bSgjelinek 			bcopy(&old_mcaptab, &in_progress_mcaptab,
37620209230bSgjelinek 			    sizeof (struct zone_mcaptab));
37630209230bSgjelinek 		else
37640209230bSgjelinek 			bzero(&in_progress_mcaptab,
37650209230bSgjelinek 			    sizeof (in_progress_mcaptab));
37660209230bSgjelinek 		return;
3767cb8a054bSGlenn Faden 	case RT_ADMIN:
3768cb8a054bSGlenn Faden 		if ((err = fill_in_admintab(cmd, &old_admintab, B_FALSE))
3769cb8a054bSGlenn Faden 		    != Z_OK) {
3770cb8a054bSGlenn Faden 			z_cmd_rt_perror(CMD_SELECT, RT_ADMIN, err,
3771cb8a054bSGlenn Faden 			    B_TRUE);
3772cb8a054bSGlenn Faden 			global_scope = B_TRUE;
3773cb8a054bSGlenn Faden 		}
3774cb8a054bSGlenn Faden 		bcopy(&old_admintab, &in_progress_admintab,
3775cb8a054bSGlenn Faden 		    sizeof (struct zone_admintab));
3776cb8a054bSGlenn Faden 		return;
37777c478bd9Sstevel@tonic-gate 	default:
3778bbec428eSgjelinek 		zone_perror(rt_to_str(type), Z_NO_RESOURCE_TYPE, B_TRUE);
3779bbec428eSgjelinek 		long_usage(CMD_SELECT, B_TRUE);
3780bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
37817c478bd9Sstevel@tonic-gate 		return;
37827c478bd9Sstevel@tonic-gate 	}
37837c478bd9Sstevel@tonic-gate }
37847c478bd9Sstevel@tonic-gate 
37857c478bd9Sstevel@tonic-gate /*
37867c478bd9Sstevel@tonic-gate  * Network "addresses" can be one of the following forms:
37877c478bd9Sstevel@tonic-gate  *	<IPv4 address>
37887c478bd9Sstevel@tonic-gate  *	<IPv4 address>/<prefix length>
37897c478bd9Sstevel@tonic-gate  *	<IPv6 address>/<prefix length>
37907c478bd9Sstevel@tonic-gate  *	<host name>
37917c478bd9Sstevel@tonic-gate  *	<host name>/<prefix length>
37927c478bd9Sstevel@tonic-gate  * In other words, the "/" followed by a prefix length is allowed but not
37937c478bd9Sstevel@tonic-gate  * required for IPv4 addresses and host names, and required for IPv6 addresses.
37947c478bd9Sstevel@tonic-gate  * If a prefix length is given, it must be in the allowable range: 0 to 32 for
37957c478bd9Sstevel@tonic-gate  * IPv4 addresses and host names, 0 to 128 for IPv6 addresses.
37967c478bd9Sstevel@tonic-gate  * Host names must start with an alpha-numeric character, and all subsequent
37977c478bd9Sstevel@tonic-gate  * characters must be either alpha-numeric or "-".
3798550b6e40SSowmini Varadhan  *
3799550b6e40SSowmini Varadhan  * In some cases, e.g., the nexthop for the defrouter, the context indicates
3800550b6e40SSowmini Varadhan  * that this is the IPV4_ABITS or IPV6_ABITS netmask, in which case we don't
3801550b6e40SSowmini Varadhan  * require the /<prefix length> (and should ignore it if provided).
38027c478bd9Sstevel@tonic-gate  */
38037c478bd9Sstevel@tonic-gate 
38047c478bd9Sstevel@tonic-gate static int
3805550b6e40SSowmini Varadhan validate_net_address_syntax(char *address, boolean_t ishost)
38067c478bd9Sstevel@tonic-gate {
38077c478bd9Sstevel@tonic-gate 	char *slashp, part1[MAXHOSTNAMELEN];
38087c478bd9Sstevel@tonic-gate 	struct in6_addr in6;
38097c478bd9Sstevel@tonic-gate 	struct in_addr in4;
38107c478bd9Sstevel@tonic-gate 	int prefixlen, i;
38117c478bd9Sstevel@tonic-gate 
38127c478bd9Sstevel@tonic-gate 	/*
38137c478bd9Sstevel@tonic-gate 	 * Copy the part before any '/' into part1 or copy the whole
38147c478bd9Sstevel@tonic-gate 	 * thing if there is no '/'.
38157c478bd9Sstevel@tonic-gate 	 */
38167c478bd9Sstevel@tonic-gate 	if ((slashp = strchr(address, '/')) != NULL) {
38177c478bd9Sstevel@tonic-gate 		*slashp = '\0';
38187c478bd9Sstevel@tonic-gate 		(void) strlcpy(part1, address, sizeof (part1));
38197c478bd9Sstevel@tonic-gate 		*slashp = '/';
38207c478bd9Sstevel@tonic-gate 		prefixlen = atoi(++slashp);
38217c478bd9Sstevel@tonic-gate 	} else {
38227c478bd9Sstevel@tonic-gate 		(void) strlcpy(part1, address, sizeof (part1));
38237c478bd9Sstevel@tonic-gate 	}
38247c478bd9Sstevel@tonic-gate 
3825550b6e40SSowmini Varadhan 	if (ishost && slashp != NULL) {
3826550b6e40SSowmini Varadhan 		zerr(gettext("Warning: prefix length in %s is not required and "
3827550b6e40SSowmini Varadhan 		    "will be ignored. The default host-prefix length "
3828550b6e40SSowmini Varadhan 		    "will be used"), address);
3829550b6e40SSowmini Varadhan 	}
3830550b6e40SSowmini Varadhan 
3831550b6e40SSowmini Varadhan 
38327c478bd9Sstevel@tonic-gate 	if (inet_pton(AF_INET6, part1, &in6) == 1) {
3833550b6e40SSowmini Varadhan 		if (ishost) {
3834550b6e40SSowmini Varadhan 			prefixlen = IPV6_ABITS;
3835550b6e40SSowmini Varadhan 		} else if (slashp == NULL) {
38367c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: IPv6 addresses "
38377c478bd9Sstevel@tonic-gate 			    "require /prefix-length suffix."), address);
38387c478bd9Sstevel@tonic-gate 			return (Z_ERR);
38397c478bd9Sstevel@tonic-gate 		}
38407c478bd9Sstevel@tonic-gate 		if (prefixlen < 0 || prefixlen > 128) {
38417c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: IPv6 address "
38427c478bd9Sstevel@tonic-gate 			    "prefix lengths must be 0 - 128."), address);
38437c478bd9Sstevel@tonic-gate 			return (Z_ERR);
38447c478bd9Sstevel@tonic-gate 		}
38457c478bd9Sstevel@tonic-gate 		return (Z_OK);
38467c478bd9Sstevel@tonic-gate 	}
38477c478bd9Sstevel@tonic-gate 
38487c478bd9Sstevel@tonic-gate 	/* At this point, any /prefix must be for IPv4. */
3849550b6e40SSowmini Varadhan 	if (ishost)
3850550b6e40SSowmini Varadhan 		prefixlen = IPV4_ABITS;
3851550b6e40SSowmini Varadhan 	else if (slashp != NULL) {
38527c478bd9Sstevel@tonic-gate 		if (prefixlen < 0 || prefixlen > 32) {
38537c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: IPv4 address "
38547c478bd9Sstevel@tonic-gate 			    "prefix lengths must be 0 - 32."), address);
38557c478bd9Sstevel@tonic-gate 			return (Z_ERR);
38567c478bd9Sstevel@tonic-gate 		}
38577c478bd9Sstevel@tonic-gate 	}
3858550b6e40SSowmini Varadhan 
38597c478bd9Sstevel@tonic-gate 	if (inet_pton(AF_INET, part1, &in4) == 1)
38607c478bd9Sstevel@tonic-gate 		return (Z_OK);
38617c478bd9Sstevel@tonic-gate 
38627c478bd9Sstevel@tonic-gate 	/* address may also be a host name */
38637c478bd9Sstevel@tonic-gate 	if (!isalnum(part1[0])) {
38647c478bd9Sstevel@tonic-gate 		zerr(gettext("%s: bogus host name or network address syntax"),
38657c478bd9Sstevel@tonic-gate 		    part1);
3866bbec428eSgjelinek 		saw_error = B_TRUE;
3867bbec428eSgjelinek 		usage(B_FALSE, HELP_NETADDR);
38687c478bd9Sstevel@tonic-gate 		return (Z_ERR);
38697c478bd9Sstevel@tonic-gate 	}
38707c478bd9Sstevel@tonic-gate 	for (i = 1; part1[i]; i++)
38717c478bd9Sstevel@tonic-gate 		if (!isalnum(part1[i]) && part1[i] != '-' && part1[i] != '.') {
38727c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: bogus host name or "
38737c478bd9Sstevel@tonic-gate 			    "network address syntax"), part1);
3874bbec428eSgjelinek 			saw_error = B_TRUE;
3875bbec428eSgjelinek 			usage(B_FALSE, HELP_NETADDR);
38767c478bd9Sstevel@tonic-gate 			return (Z_ERR);
38777c478bd9Sstevel@tonic-gate 		}
38787c478bd9Sstevel@tonic-gate 	return (Z_OK);
38797c478bd9Sstevel@tonic-gate }
38807c478bd9Sstevel@tonic-gate 
38817c478bd9Sstevel@tonic-gate static int
3882c9f134eaSjv227347 validate_net_physical_syntax(const char *ifname)
38837c478bd9Sstevel@tonic-gate {
3884c9f134eaSjv227347 	ifspec_t ifnameprop;
3885c9f134eaSjv227347 	zone_iptype_t iptype;
3886c9f134eaSjv227347 
388737b210dcSjv227347 	if (zonecfg_get_iptype(handle, &iptype) != Z_OK) {
3888c9f134eaSjv227347 		zerr(gettext("zone configuration has an invalid or nonexistent "
3889c9f134eaSjv227347 		    "ip-type property"));
38907c478bd9Sstevel@tonic-gate 		return (Z_ERR);
38917c478bd9Sstevel@tonic-gate 	}
3892c9f134eaSjv227347 	switch (iptype) {
3893c9f134eaSjv227347 	case ZS_SHARED:
3894c9f134eaSjv227347 		if (ifparse_ifspec(ifname, &ifnameprop) == B_FALSE) {
3895c9f134eaSjv227347 			zerr(gettext("%s: invalid physical interface name"),
3896c9f134eaSjv227347 			    ifname);
3897c9f134eaSjv227347 			return (Z_ERR);
3898c9f134eaSjv227347 		}
3899c9f134eaSjv227347 		if (ifnameprop.ifsp_lunvalid) {
3900c9f134eaSjv227347 			zerr(gettext("%s: LUNs not allowed in physical "
3901c9f134eaSjv227347 			    "interface names"), ifname);
3902c9f134eaSjv227347 			return (Z_ERR);
3903c9f134eaSjv227347 		}
3904c9f134eaSjv227347 		break;
3905c9f134eaSjv227347 	case ZS_EXCLUSIVE:
3906c9f134eaSjv227347 		if (dladm_valid_linkname(ifname) == B_FALSE) {
3907c9f134eaSjv227347 			if (strchr(ifname, ':') != NULL)
3908c9f134eaSjv227347 				zerr(gettext("%s: physical interface name "
3909c9f134eaSjv227347 				    "required; logical interface name not "
3910c9f134eaSjv227347 				    "allowed"), ifname);
3911c9f134eaSjv227347 			else
3912c9f134eaSjv227347 				zerr(gettext("%s: invalid physical interface "
3913c9f134eaSjv227347 				    "name"), ifname);
3914c9f134eaSjv227347 			return (Z_ERR);
3915c9f134eaSjv227347 		}
3916c9f134eaSjv227347 		break;
3917c9f134eaSjv227347 	}
3918c9f134eaSjv227347 	return (Z_OK);
3919c9f134eaSjv227347 }
39207c478bd9Sstevel@tonic-gate 
39217c478bd9Sstevel@tonic-gate static boolean_t
39227c478bd9Sstevel@tonic-gate valid_fs_type(const char *type)
39237c478bd9Sstevel@tonic-gate {
39247c478bd9Sstevel@tonic-gate 	/*
39257c478bd9Sstevel@tonic-gate 	 * Is this a valid path component?
39267c478bd9Sstevel@tonic-gate 	 */
39277c478bd9Sstevel@tonic-gate 	if (strlen(type) + 1 > MAXNAMELEN)
39287c478bd9Sstevel@tonic-gate 		return (B_FALSE);
39297c478bd9Sstevel@tonic-gate 	/*
39307c478bd9Sstevel@tonic-gate 	 * Make sure a bad value for "type" doesn't make
39317c478bd9Sstevel@tonic-gate 	 * /usr/lib/fs/<type>/mount turn into something else.
39327c478bd9Sstevel@tonic-gate 	 */
39337c478bd9Sstevel@tonic-gate 	if (strchr(type, '/') != NULL || type[0] == '\0' ||
39347c478bd9Sstevel@tonic-gate 	    strcmp(type, ".") == 0 || strcmp(type, "..") == 0)
39357c478bd9Sstevel@tonic-gate 		return (B_FALSE);
39367c478bd9Sstevel@tonic-gate 	/*
39377c478bd9Sstevel@tonic-gate 	 * More detailed verification happens later by zoneadm(1m).
39387c478bd9Sstevel@tonic-gate 	 */
39397c478bd9Sstevel@tonic-gate 	return (B_TRUE);
39407c478bd9Sstevel@tonic-gate }
39417c478bd9Sstevel@tonic-gate 
3942f4b3ec61Sdh155122 static boolean_t
3943f4b3ec61Sdh155122 allow_exclusive()
3944f4b3ec61Sdh155122 {
3945f4b3ec61Sdh155122 	brand_handle_t	bh;
3946f4b3ec61Sdh155122 	char		brand[MAXNAMELEN];
3947f4b3ec61Sdh155122 	boolean_t	ret;
3948f4b3ec61Sdh155122 
3949f4b3ec61Sdh155122 	if (zonecfg_get_brand(handle, brand, sizeof (brand)) != Z_OK) {
3950f4b3ec61Sdh155122 		zerr("%s: %s\n", zone, gettext("could not get zone brand"));
3951f4b3ec61Sdh155122 		return (B_FALSE);
3952f4b3ec61Sdh155122 	}
3953f4b3ec61Sdh155122 	if ((bh = brand_open(brand)) == NULL) {
3954f4b3ec61Sdh155122 		zerr("%s: %s\n", zone, gettext("unknown brand."));
3955f4b3ec61Sdh155122 		return (B_FALSE);
3956f4b3ec61Sdh155122 	}
3957f4b3ec61Sdh155122 	ret = brand_allow_exclusive_ip(bh);
3958f4b3ec61Sdh155122 	brand_close(bh);
3959f4b3ec61Sdh155122 	if (!ret)
3960f4b3ec61Sdh155122 		zerr(gettext("%s cannot be '%s' when %s is '%s'."),
3961f4b3ec61Sdh155122 		    pt_to_str(PT_IPTYPE), "exclusive",
3962f4b3ec61Sdh155122 		    pt_to_str(PT_BRAND), brand);
3963f4b3ec61Sdh155122 	return (ret);
3964f4b3ec61Sdh155122 }
3965f4b3ec61Sdh155122 
39660209230bSgjelinek static void
39670209230bSgjelinek set_aliased_rctl(char *alias, int prop_type, char *s)
39680209230bSgjelinek {
39690209230bSgjelinek 	uint64_t limit;
39700209230bSgjelinek 	int err;
39710209230bSgjelinek 	char tmp[128];
39720209230bSgjelinek 
39730209230bSgjelinek 	if (global_zone && strcmp(alias, ALIAS_SHARES) != 0)
39740209230bSgjelinek 		zerr(gettext("WARNING: Setting a global zone resource "
39750209230bSgjelinek 		    "control too low could deny\nservice "
39760209230bSgjelinek 		    "to even the root user; "
39770209230bSgjelinek 		    "this could render the system impossible\n"
39780209230bSgjelinek 		    "to administer.  Please use caution."));
39790209230bSgjelinek 
39800209230bSgjelinek 	/* convert memory based properties */
39810209230bSgjelinek 	if (prop_type == PT_MAXSHMMEM) {
39820209230bSgjelinek 		if (!zonecfg_valid_memlimit(s, &limit)) {
39830209230bSgjelinek 			zerr(gettext("A non-negative number with a required "
39840209230bSgjelinek 			    "scale suffix (K, M, G or T) was expected\nhere."));
3985bbec428eSgjelinek 			saw_error = B_TRUE;
39860209230bSgjelinek 			return;
39870209230bSgjelinek 		}
39880209230bSgjelinek 
39890209230bSgjelinek 		(void) snprintf(tmp, sizeof (tmp), "%llu", limit);
39900209230bSgjelinek 		s = tmp;
39910209230bSgjelinek 	}
39920209230bSgjelinek 
39930209230bSgjelinek 	if (!zonecfg_aliased_rctl_ok(handle, alias)) {
3994bbec428eSgjelinek 		zone_perror(pt_to_str(prop_type), Z_ALIAS_DISALLOW, B_FALSE);
3995bbec428eSgjelinek 		saw_error = B_TRUE;
39960209230bSgjelinek 	} else if (!zonecfg_valid_alias_limit(alias, s, &limit)) {
39970209230bSgjelinek 		zerr(gettext("%s property is out of range."),
39980209230bSgjelinek 		    pt_to_str(prop_type));
3999bbec428eSgjelinek 		saw_error = B_TRUE;
40000209230bSgjelinek 	} else if ((err = zonecfg_set_aliased_rctl(handle, alias, limit))
40010209230bSgjelinek 	    != Z_OK) {
4002bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
4003bbec428eSgjelinek 		saw_error = B_TRUE;
40040209230bSgjelinek 	} else {
4005bbec428eSgjelinek 		need_to_commit = B_TRUE;
40060209230bSgjelinek 	}
40070209230bSgjelinek }
40080209230bSgjelinek 
4009550b6e40SSowmini Varadhan static void
4010550b6e40SSowmini Varadhan set_in_progress_nwiftab_address(char *prop_id, int prop_type)
4011550b6e40SSowmini Varadhan {
4012550b6e40SSowmini Varadhan 	if (prop_type == PT_ADDRESS) {
4013550b6e40SSowmini Varadhan 		(void) strlcpy(in_progress_nwiftab.zone_nwif_address, prop_id,
4014550b6e40SSowmini Varadhan 		    sizeof (in_progress_nwiftab.zone_nwif_address));
4015550b6e40SSowmini Varadhan 	} else {
4016550b6e40SSowmini Varadhan 		assert(prop_type == PT_ALLOWED_ADDRESS);
4017550b6e40SSowmini Varadhan 		(void) strlcpy(in_progress_nwiftab.zone_nwif_allowed_address,
4018550b6e40SSowmini Varadhan 		    prop_id,
4019550b6e40SSowmini Varadhan 		    sizeof (in_progress_nwiftab.zone_nwif_allowed_address));
4020550b6e40SSowmini Varadhan 	}
4021550b6e40SSowmini Varadhan }
4022550b6e40SSowmini Varadhan 
40237c478bd9Sstevel@tonic-gate void
40247c478bd9Sstevel@tonic-gate set_func(cmd_t *cmd)
40257c478bd9Sstevel@tonic-gate {
40267c478bd9Sstevel@tonic-gate 	char *prop_id;
4027555afedfScarlsonj 	int arg, err, res_type, prop_type;
40287c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
40297c478bd9Sstevel@tonic-gate 	boolean_t autoboot;
4030f4b3ec61Sdh155122 	zone_iptype_t iptype;
4031bbec428eSgjelinek 	boolean_t force_set = B_FALSE;
40320209230bSgjelinek 	size_t physmem_size = sizeof (in_progress_mcaptab.zone_physmem_cap);
40330209230bSgjelinek 	uint64_t mem_cap, mem_limit;
4034c97ad5cdSakolb 	float cap;
4035c97ad5cdSakolb 	char *unitp;
40360209230bSgjelinek 	struct zone_psettab tmp_psettab;
4037bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
40387c478bd9Sstevel@tonic-gate 
40397c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_SET))
40407c478bd9Sstevel@tonic-gate 		return;
40417c478bd9Sstevel@tonic-gate 
40427c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
40437c478bd9Sstevel@tonic-gate 
4044555afedfScarlsonj 	optind = opterr = 0;
4045555afedfScarlsonj 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "F")) != EOF) {
4046555afedfScarlsonj 		switch (arg) {
4047555afedfScarlsonj 		case 'F':
4048bbec428eSgjelinek 			force_set = B_TRUE;
4049555afedfScarlsonj 			break;
4050555afedfScarlsonj 		default:
4051555afedfScarlsonj 			if (optopt == '?')
4052555afedfScarlsonj 				longer_usage(CMD_SET);
4053555afedfScarlsonj 			else
4054555afedfScarlsonj 				short_usage(CMD_SET);
4055bbec428eSgjelinek 			arg_err = B_TRUE;
40567ec75eb8Sgjelinek 			break;
40577ec75eb8Sgjelinek 		}
40587ec75eb8Sgjelinek 	}
40597ec75eb8Sgjelinek 	if (arg_err)
4060555afedfScarlsonj 		return;
4061555afedfScarlsonj 
40627c478bd9Sstevel@tonic-gate 	prop_type = cmd->cmd_prop_name[0];
40637c478bd9Sstevel@tonic-gate 	if (global_scope) {
40640209230bSgjelinek 		if (gz_invalid_property(prop_type)) {
40650209230bSgjelinek 			zerr(gettext("%s is not a valid property for the "
40660209230bSgjelinek 			    "global zone."), pt_to_str(prop_type));
4067bbec428eSgjelinek 			saw_error = B_TRUE;
40680209230bSgjelinek 			return;
40690209230bSgjelinek 		}
40700209230bSgjelinek 
4071087719fdSdp 		if (prop_type == PT_ZONENAME) {
4072087719fdSdp 			res_type = RT_ZONENAME;
4073087719fdSdp 		} else if (prop_type == PT_ZONEPATH) {
40747c478bd9Sstevel@tonic-gate 			res_type = RT_ZONEPATH;
40757c478bd9Sstevel@tonic-gate 		} else if (prop_type == PT_AUTOBOOT) {
40767c478bd9Sstevel@tonic-gate 			res_type = RT_AUTOBOOT;
40779acbbeafSnn35248 		} else if (prop_type == PT_BRAND) {
40789acbbeafSnn35248 			res_type = RT_BRAND;
40797c478bd9Sstevel@tonic-gate 		} else if (prop_type == PT_POOL) {
40807c478bd9Sstevel@tonic-gate 			res_type = RT_POOL;
4081ffbafc53Scomay 		} else if (prop_type == PT_LIMITPRIV) {
4082ffbafc53Scomay 			res_type = RT_LIMITPRIV;
40833f2f09c1Sdp 		} else if (prop_type == PT_BOOTARGS) {
40843f2f09c1Sdp 			res_type = RT_BOOTARGS;
40850209230bSgjelinek 		} else if (prop_type == PT_SCHED) {
40860209230bSgjelinek 			res_type = RT_SCHED;
4087f4b3ec61Sdh155122 		} else if (prop_type == PT_IPTYPE) {
4088f4b3ec61Sdh155122 			res_type = RT_IPTYPE;
40890209230bSgjelinek 		} else if (prop_type == PT_MAXLWPS) {
40900209230bSgjelinek 			res_type = RT_MAXLWPS;
4091ff19e029SMenno Lageman 		} else if (prop_type == PT_MAXPROCS) {
4092ff19e029SMenno Lageman 			res_type = RT_MAXPROCS;
40930209230bSgjelinek 		} else if (prop_type == PT_MAXSHMMEM) {
40940209230bSgjelinek 			res_type = RT_MAXSHMMEM;
40950209230bSgjelinek 		} else if (prop_type == PT_MAXSHMIDS) {
40960209230bSgjelinek 			res_type = RT_MAXSHMIDS;
40970209230bSgjelinek 		} else if (prop_type == PT_MAXMSGIDS) {
40980209230bSgjelinek 			res_type = RT_MAXMSGIDS;
40990209230bSgjelinek 		} else if (prop_type == PT_MAXSEMIDS) {
41000209230bSgjelinek 			res_type = RT_MAXSEMIDS;
41010209230bSgjelinek 		} else if (prop_type == PT_SHARES) {
41020209230bSgjelinek 			res_type = RT_SHARES;
41035679c89fSjv227347 		} else if (prop_type == PT_HOSTID) {
41045679c89fSjv227347 			res_type = RT_HOSTID;
41050fbb751dSJohn Levon 		} else if (prop_type == PT_FS_ALLOWED) {
41060fbb751dSJohn Levon 			res_type = RT_FS_ALLOWED;
41077c478bd9Sstevel@tonic-gate 		} else {
41087c478bd9Sstevel@tonic-gate 			zerr(gettext("Cannot set a resource-specific property "
41097c478bd9Sstevel@tonic-gate 			    "from the global scope."));
4110bbec428eSgjelinek 			saw_error = B_TRUE;
41117c478bd9Sstevel@tonic-gate 			return;
41127c478bd9Sstevel@tonic-gate 		}
41137c478bd9Sstevel@tonic-gate 	} else {
41147c478bd9Sstevel@tonic-gate 		res_type = resource_scope;
41157c478bd9Sstevel@tonic-gate 	}
41167c478bd9Sstevel@tonic-gate 
4117555afedfScarlsonj 	if (force_set) {
4118555afedfScarlsonj 		if (res_type != RT_ZONEPATH) {
4119555afedfScarlsonj 			zerr(gettext("Only zonepath setting can be forced."));
4120bbec428eSgjelinek 			saw_error = B_TRUE;
4121555afedfScarlsonj 			return;
4122555afedfScarlsonj 		}
4123555afedfScarlsonj 		if (!zonecfg_in_alt_root()) {
4124555afedfScarlsonj 			zerr(gettext("Zonepath is changeable only in an "
4125555afedfScarlsonj 			    "alternate root."));
4126bbec428eSgjelinek 			saw_error = B_TRUE;
4127555afedfScarlsonj 			return;
4128555afedfScarlsonj 		}
4129555afedfScarlsonj 	}
4130555afedfScarlsonj 
41317c478bd9Sstevel@tonic-gate 	pp = cmd->cmd_property_ptr[0];
41327c478bd9Sstevel@tonic-gate 	/*
41337c478bd9Sstevel@tonic-gate 	 * A nasty expression but not that complicated:
41347c478bd9Sstevel@tonic-gate 	 * 1. fs options are simple or list (tested below)
41357c478bd9Sstevel@tonic-gate 	 * 2. rctl value's are complex or list (tested below)
41367c478bd9Sstevel@tonic-gate 	 * Anything else should be simple.
41377c478bd9Sstevel@tonic-gate 	 */
41387c478bd9Sstevel@tonic-gate 	if (!(res_type == RT_FS && prop_type == PT_OPTIONS) &&
41397c478bd9Sstevel@tonic-gate 	    !(res_type == RT_RCTL && prop_type == PT_VALUE) &&
41407c478bd9Sstevel@tonic-gate 	    (pp->pv_type != PROP_VAL_SIMPLE ||
41417c478bd9Sstevel@tonic-gate 	    (prop_id = pp->pv_simple) == NULL)) {
41427c478bd9Sstevel@tonic-gate 		zerr(gettext("A %s value was expected here."),
41437c478bd9Sstevel@tonic-gate 		    pvt_to_str(PROP_VAL_SIMPLE));
4144bbec428eSgjelinek 		saw_error = B_TRUE;
41457c478bd9Sstevel@tonic-gate 		return;
41467c478bd9Sstevel@tonic-gate 	}
41477c478bd9Sstevel@tonic-gate 	if (prop_type == PT_UNKNOWN) {
4148bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
41497c478bd9Sstevel@tonic-gate 		return;
41507c478bd9Sstevel@tonic-gate 	}
41517c478bd9Sstevel@tonic-gate 
4152087719fdSdp 	/*
4153087719fdSdp 	 * Special case: the user can change the zone name prior to 'create';
4154087719fdSdp 	 * if the zone already exists, we fall through letting initialize()
4155087719fdSdp 	 * and the rest of the logic run.
4156087719fdSdp 	 */
4157bbec428eSgjelinek 	if (res_type == RT_ZONENAME && got_handle == B_FALSE &&
4158087719fdSdp 	    !state_atleast(ZONE_STATE_CONFIGURED)) {
4159fb03efaaSdp 		if ((err = zonecfg_validate_zonename(prop_id)) != Z_OK) {
4160bbec428eSgjelinek 			zone_perror(prop_id, err, B_TRUE);
4161bbec428eSgjelinek 			usage(B_FALSE, HELP_SYNTAX);
4162fb03efaaSdp 			return;
4163fb03efaaSdp 		}
4164087719fdSdp 		(void) strlcpy(zone, prop_id, sizeof (zone));
4165087719fdSdp 		return;
4166087719fdSdp 	}
4167087719fdSdp 
4168bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
41697c478bd9Sstevel@tonic-gate 		return;
41707c478bd9Sstevel@tonic-gate 
41717c478bd9Sstevel@tonic-gate 	switch (res_type) {
4172087719fdSdp 	case RT_ZONENAME:
4173087719fdSdp 		if ((err = zonecfg_set_name(handle, prop_id)) != Z_OK) {
4174087719fdSdp 			/*
4175087719fdSdp 			 * Use prop_id instead of 'zone' here, since we're
4176087719fdSdp 			 * reporting a problem about the *new* zonename.
4177087719fdSdp 			 */
4178bbec428eSgjelinek 			zone_perror(prop_id, err, B_TRUE);
4179bbec428eSgjelinek 			usage(B_FALSE, HELP_SYNTAX);
4180087719fdSdp 		} else {
4181bbec428eSgjelinek 			need_to_commit = B_TRUE;
4182087719fdSdp 			(void) strlcpy(zone, prop_id, sizeof (zone));
4183087719fdSdp 		}
4184087719fdSdp 		return;
41857c478bd9Sstevel@tonic-gate 	case RT_ZONEPATH:
4186555afedfScarlsonj 		if (!force_set && state_atleast(ZONE_STATE_INSTALLED)) {
41877c478bd9Sstevel@tonic-gate 			zerr(gettext("Zone %s already installed; %s %s not "
41887c478bd9Sstevel@tonic-gate 			    "allowed."), zone, cmd_to_str(CMD_SET),
41897c478bd9Sstevel@tonic-gate 			    rt_to_str(RT_ZONEPATH));
41907c478bd9Sstevel@tonic-gate 			return;
41917c478bd9Sstevel@tonic-gate 		}
41927c478bd9Sstevel@tonic-gate 		if (validate_zonepath_syntax(prop_id) != Z_OK) {
4193bbec428eSgjelinek 			saw_error = B_TRUE;
41947c478bd9Sstevel@tonic-gate 			return;
41957c478bd9Sstevel@tonic-gate 		}
41967c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_set_zonepath(handle, prop_id)) != Z_OK)
4197bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
41987c478bd9Sstevel@tonic-gate 		else
4199bbec428eSgjelinek 			need_to_commit = B_TRUE;
42007c478bd9Sstevel@tonic-gate 		return;
42019acbbeafSnn35248 	case RT_BRAND:
42029acbbeafSnn35248 		if (state_atleast(ZONE_STATE_INSTALLED)) {
42039acbbeafSnn35248 			zerr(gettext("Zone %s already installed; %s %s not "
42049acbbeafSnn35248 			    "allowed."), zone, cmd_to_str(CMD_SET),
42059acbbeafSnn35248 			    rt_to_str(RT_BRAND));
42069acbbeafSnn35248 			return;
42079acbbeafSnn35248 		}
42089acbbeafSnn35248 		if ((err = zonecfg_set_brand(handle, prop_id)) != Z_OK)
4209bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
42109acbbeafSnn35248 		else
4211bbec428eSgjelinek 			need_to_commit = B_TRUE;
42129acbbeafSnn35248 		return;
42137c478bd9Sstevel@tonic-gate 	case RT_AUTOBOOT:
42147c478bd9Sstevel@tonic-gate 		if (strcmp(prop_id, "true") == 0) {
42157c478bd9Sstevel@tonic-gate 			autoboot = B_TRUE;
42167c478bd9Sstevel@tonic-gate 		} else if (strcmp(prop_id, "false") == 0) {
42177c478bd9Sstevel@tonic-gate 			autoboot = B_FALSE;
42187c478bd9Sstevel@tonic-gate 		} else {
42197c478bd9Sstevel@tonic-gate 			zerr(gettext("%s value must be '%s' or '%s'."),
42207c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_AUTOBOOT), "true", "false");
4221bbec428eSgjelinek 			saw_error = B_TRUE;
42227c478bd9Sstevel@tonic-gate 			return;
42237c478bd9Sstevel@tonic-gate 		}
42247c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_set_autoboot(handle, autoboot)) != Z_OK)
4225bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
42267c478bd9Sstevel@tonic-gate 		else
4227bbec428eSgjelinek 			need_to_commit = B_TRUE;
42287c478bd9Sstevel@tonic-gate 		return;
42297c478bd9Sstevel@tonic-gate 	case RT_POOL:
42300209230bSgjelinek 		/* don't allow use of the reserved temporary pool names */
42310209230bSgjelinek 		if (strncmp("SUNW", prop_id, 4) == 0) {
42320209230bSgjelinek 			zerr(gettext("pool names starting with SUNW are "
42330209230bSgjelinek 			    "reserved."));
4234bbec428eSgjelinek 			saw_error = B_TRUE;
42350209230bSgjelinek 			return;
42360209230bSgjelinek 		}
42370209230bSgjelinek 
42380209230bSgjelinek 		/* can't set pool if dedicated-cpu exists */
42390209230bSgjelinek 		if (zonecfg_lookup_pset(handle, &tmp_psettab) == Z_OK) {
42400209230bSgjelinek 			zerr(gettext("The %s resource already exists.  "
42410209230bSgjelinek 			    "A persistent pool is incompatible\nwith the %s "
42420209230bSgjelinek 			    "resource."), rt_to_str(RT_DCPU),
42430209230bSgjelinek 			    rt_to_str(RT_DCPU));
4244bbec428eSgjelinek 			saw_error = B_TRUE;
42450209230bSgjelinek 			return;
42460209230bSgjelinek 		}
42470209230bSgjelinek 
42487c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_set_pool(handle, prop_id)) != Z_OK)
4249bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
42507c478bd9Sstevel@tonic-gate 		else
4251bbec428eSgjelinek 			need_to_commit = B_TRUE;
42527c478bd9Sstevel@tonic-gate 		return;
4253ffbafc53Scomay 	case RT_LIMITPRIV:
4254ffbafc53Scomay 		if ((err = zonecfg_set_limitpriv(handle, prop_id)) != Z_OK)
4255bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
4256ffbafc53Scomay 		else
4257bbec428eSgjelinek 			need_to_commit = B_TRUE;
4258ffbafc53Scomay 		return;
42593f2f09c1Sdp 	case RT_BOOTARGS:
42603f2f09c1Sdp 		if ((err = zonecfg_set_bootargs(handle, prop_id)) != Z_OK)
4261bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
42623f2f09c1Sdp 		else
4263bbec428eSgjelinek 			need_to_commit = B_TRUE;
42643f2f09c1Sdp 		return;
42650209230bSgjelinek 	case RT_SCHED:
42660209230bSgjelinek 		if ((err = zonecfg_set_sched(handle, prop_id)) != Z_OK)
4267bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
42680209230bSgjelinek 		else
4269bbec428eSgjelinek 			need_to_commit = B_TRUE;
42700209230bSgjelinek 		return;
4271f4b3ec61Sdh155122 	case RT_IPTYPE:
4272f4b3ec61Sdh155122 		if (strcmp(prop_id, "shared") == 0) {
4273f4b3ec61Sdh155122 			iptype = ZS_SHARED;
4274f4b3ec61Sdh155122 		} else if (strcmp(prop_id, "exclusive") == 0) {
4275f4b3ec61Sdh155122 			iptype = ZS_EXCLUSIVE;
4276f4b3ec61Sdh155122 		} else {
4277f4b3ec61Sdh155122 			zerr(gettext("%s value must be '%s' or '%s'."),
4278f4b3ec61Sdh155122 			    pt_to_str(PT_IPTYPE), "shared", "exclusive");
4279bbec428eSgjelinek 			saw_error = B_TRUE;
4280f4b3ec61Sdh155122 			return;
4281f4b3ec61Sdh155122 		}
4282f4b3ec61Sdh155122 		if (iptype == ZS_EXCLUSIVE && !allow_exclusive()) {
4283bbec428eSgjelinek 			saw_error = B_TRUE;
4284f4b3ec61Sdh155122 			return;
4285f4b3ec61Sdh155122 		}
4286f4b3ec61Sdh155122 		if ((err = zonecfg_set_iptype(handle, iptype)) != Z_OK)
4287bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
4288f4b3ec61Sdh155122 		else
4289bbec428eSgjelinek 			need_to_commit = B_TRUE;
4290f4b3ec61Sdh155122 		return;
42910209230bSgjelinek 	case RT_MAXLWPS:
42920209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXLWPS, prop_type, prop_id);
42930209230bSgjelinek 		return;
4294ff19e029SMenno Lageman 	case RT_MAXPROCS:
4295ff19e029SMenno Lageman 		set_aliased_rctl(ALIAS_MAXPROCS, prop_type, prop_id);
4296ff19e029SMenno Lageman 		return;
42970209230bSgjelinek 	case RT_MAXSHMMEM:
42980209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXSHMMEM, prop_type, prop_id);
42990209230bSgjelinek 		return;
43000209230bSgjelinek 	case RT_MAXSHMIDS:
43010209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXSHMIDS, prop_type, prop_id);
43020209230bSgjelinek 		return;
43030209230bSgjelinek 	case RT_MAXMSGIDS:
43040209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXMSGIDS, prop_type, prop_id);
43050209230bSgjelinek 		return;
43060209230bSgjelinek 	case RT_MAXSEMIDS:
43070209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXSEMIDS, prop_type, prop_id);
43080209230bSgjelinek 		return;
43090209230bSgjelinek 	case RT_SHARES:
43100209230bSgjelinek 		set_aliased_rctl(ALIAS_SHARES, prop_type, prop_id);
43110209230bSgjelinek 		return;
43125679c89fSjv227347 	case RT_HOSTID:
43135679c89fSjv227347 		if ((err = zonecfg_set_hostid(handle, prop_id)) != Z_OK) {
43145679c89fSjv227347 			if (err == Z_TOO_BIG) {
43155679c89fSjv227347 				zerr(gettext("hostid string is too large: %s"),
43165679c89fSjv227347 				    prop_id);
43175679c89fSjv227347 				saw_error = B_TRUE;
43185679c89fSjv227347 			} else {
43195679c89fSjv227347 				zone_perror(pt_to_str(prop_type), err, B_TRUE);
43205679c89fSjv227347 			}
43215679c89fSjv227347 			return;
43225679c89fSjv227347 		}
43235679c89fSjv227347 		need_to_commit = B_TRUE;
43245679c89fSjv227347 		return;
43250fbb751dSJohn Levon 	case RT_FS_ALLOWED:
43260fbb751dSJohn Levon 		if ((err = zonecfg_set_fs_allowed(handle, prop_id)) != Z_OK)
43270fbb751dSJohn Levon 			zone_perror(zone, err, B_TRUE);
43280fbb751dSJohn Levon 		else
43290fbb751dSJohn Levon 			need_to_commit = B_TRUE;
43300fbb751dSJohn Levon 		return;
43317c478bd9Sstevel@tonic-gate 	case RT_FS:
43327c478bd9Sstevel@tonic-gate 		switch (prop_type) {
43337c478bd9Sstevel@tonic-gate 		case PT_DIR:
43347c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_fstab.zone_fs_dir, prop_id,
43357c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_fstab.zone_fs_dir));
43367c478bd9Sstevel@tonic-gate 			return;
43377c478bd9Sstevel@tonic-gate 		case PT_SPECIAL:
43387c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_fstab.zone_fs_special,
43397c478bd9Sstevel@tonic-gate 			    prop_id,
43407c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_fstab.zone_fs_special));
43417c478bd9Sstevel@tonic-gate 			return;
43427c478bd9Sstevel@tonic-gate 		case PT_RAW:
43437c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_fstab.zone_fs_raw,
43447c478bd9Sstevel@tonic-gate 			    prop_id, sizeof (in_progress_fstab.zone_fs_raw));
43457c478bd9Sstevel@tonic-gate 			return;
43467c478bd9Sstevel@tonic-gate 		case PT_TYPE:
43477c478bd9Sstevel@tonic-gate 			if (!valid_fs_type(prop_id)) {
43487c478bd9Sstevel@tonic-gate 				zerr(gettext("\"%s\" is not a valid %s."),
43497c478bd9Sstevel@tonic-gate 				    prop_id, pt_to_str(PT_TYPE));
4350bbec428eSgjelinek 				saw_error = B_TRUE;
43517c478bd9Sstevel@tonic-gate 				return;
43527c478bd9Sstevel@tonic-gate 			}
43537c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_fstab.zone_fs_type, prop_id,
43547c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_fstab.zone_fs_type));
43557c478bd9Sstevel@tonic-gate 			return;
43567c478bd9Sstevel@tonic-gate 		case PT_OPTIONS:
43577c478bd9Sstevel@tonic-gate 			if (pp->pv_type != PROP_VAL_SIMPLE &&
43587c478bd9Sstevel@tonic-gate 			    pp->pv_type != PROP_VAL_LIST) {
43597c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s or %s value was expected "
43607c478bd9Sstevel@tonic-gate 				    "here."), pvt_to_str(PROP_VAL_SIMPLE),
43617c478bd9Sstevel@tonic-gate 				    pvt_to_str(PROP_VAL_LIST));
4362bbec428eSgjelinek 				saw_error = B_TRUE;
43637c478bd9Sstevel@tonic-gate 				return;
43647c478bd9Sstevel@tonic-gate 			}
43657c478bd9Sstevel@tonic-gate 			zonecfg_free_fs_option_list(
43667c478bd9Sstevel@tonic-gate 			    in_progress_fstab.zone_fs_options);
43677c478bd9Sstevel@tonic-gate 			in_progress_fstab.zone_fs_options = NULL;
43687c478bd9Sstevel@tonic-gate 			if (!(pp->pv_type == PROP_VAL_LIST &&
43697c478bd9Sstevel@tonic-gate 			    pp->pv_list == NULL))
43707c478bd9Sstevel@tonic-gate 				add_property(cmd);
43717c478bd9Sstevel@tonic-gate 			return;
43727c478bd9Sstevel@tonic-gate 		default:
43737c478bd9Sstevel@tonic-gate 			break;
43747c478bd9Sstevel@tonic-gate 		}
4375bbec428eSgjelinek 		zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, B_TRUE);
4376bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
4377bbec428eSgjelinek 		usage(B_FALSE, HELP_PROPS);
43787c478bd9Sstevel@tonic-gate 		return;
43797c478bd9Sstevel@tonic-gate 	case RT_NET:
43807c478bd9Sstevel@tonic-gate 		switch (prop_type) {
43817c478bd9Sstevel@tonic-gate 		case PT_ADDRESS:
4382550b6e40SSowmini Varadhan 		case PT_ALLOWED_ADDRESS:
4383550b6e40SSowmini Varadhan 			if (validate_net_address_syntax(prop_id, B_FALSE)
4384550b6e40SSowmini Varadhan 			    != Z_OK) {
4385bbec428eSgjelinek 				saw_error = B_TRUE;
43867c478bd9Sstevel@tonic-gate 				return;
43877c478bd9Sstevel@tonic-gate 			}
4388550b6e40SSowmini Varadhan 			set_in_progress_nwiftab_address(prop_id, prop_type);
43897c478bd9Sstevel@tonic-gate 			break;
43907c478bd9Sstevel@tonic-gate 		case PT_PHYSICAL:
43917c478bd9Sstevel@tonic-gate 			if (validate_net_physical_syntax(prop_id) != Z_OK) {
4392bbec428eSgjelinek 				saw_error = B_TRUE;
43937c478bd9Sstevel@tonic-gate 				return;
43947c478bd9Sstevel@tonic-gate 			}
43957c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_nwiftab.zone_nwif_physical,
43967c478bd9Sstevel@tonic-gate 			    prop_id,
43977c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_nwiftab.zone_nwif_physical));
43987c478bd9Sstevel@tonic-gate 			break;
4399de860bd9Sgfaden 		case PT_DEFROUTER:
4400550b6e40SSowmini Varadhan 			if (validate_net_address_syntax(prop_id, B_TRUE)
4401550b6e40SSowmini Varadhan 			    != Z_OK) {
4402bbec428eSgjelinek 				saw_error = B_TRUE;
4403de860bd9Sgfaden 				return;
4404de860bd9Sgfaden 			}
4405de860bd9Sgfaden 			(void) strlcpy(in_progress_nwiftab.zone_nwif_defrouter,
4406de860bd9Sgfaden 			    prop_id,
4407de860bd9Sgfaden 			    sizeof (in_progress_nwiftab.zone_nwif_defrouter));
4408de860bd9Sgfaden 			break;
44097c478bd9Sstevel@tonic-gate 		default:
44107c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4411bbec428eSgjelinek 			    B_TRUE);
4412bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4413bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
44147c478bd9Sstevel@tonic-gate 			return;
44157c478bd9Sstevel@tonic-gate 		}
44167c478bd9Sstevel@tonic-gate 		return;
44177c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
44187c478bd9Sstevel@tonic-gate 		switch (prop_type) {
44197c478bd9Sstevel@tonic-gate 		case PT_MATCH:
44207c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_devtab.zone_dev_match,
44217c478bd9Sstevel@tonic-gate 			    prop_id,
44227c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_devtab.zone_dev_match));
44237c478bd9Sstevel@tonic-gate 			break;
44247c478bd9Sstevel@tonic-gate 		default:
44257c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4426bbec428eSgjelinek 			    B_TRUE);
4427bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4428bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
44297c478bd9Sstevel@tonic-gate 			return;
44307c478bd9Sstevel@tonic-gate 		}
44317c478bd9Sstevel@tonic-gate 		return;
44327c478bd9Sstevel@tonic-gate 	case RT_RCTL:
44337c478bd9Sstevel@tonic-gate 		switch (prop_type) {
44347c478bd9Sstevel@tonic-gate 		case PT_NAME:
44357c478bd9Sstevel@tonic-gate 			if (!zonecfg_valid_rctlname(prop_id)) {
44367c478bd9Sstevel@tonic-gate 				zerr(gettext("'%s' is not a valid zone %s "
44377c478bd9Sstevel@tonic-gate 				    "name."), prop_id, rt_to_str(RT_RCTL));
44387c478bd9Sstevel@tonic-gate 				return;
44397c478bd9Sstevel@tonic-gate 			}
44407c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_rctltab.zone_rctl_name,
44417c478bd9Sstevel@tonic-gate 			    prop_id,
44427c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_rctltab.zone_rctl_name));
44437c478bd9Sstevel@tonic-gate 			break;
44447c478bd9Sstevel@tonic-gate 		case PT_VALUE:
44457c478bd9Sstevel@tonic-gate 			if (pp->pv_type != PROP_VAL_COMPLEX &&
44467c478bd9Sstevel@tonic-gate 			    pp->pv_type != PROP_VAL_LIST) {
44477c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s or %s value was expected "
44487c478bd9Sstevel@tonic-gate 				    "here."), pvt_to_str(PROP_VAL_COMPLEX),
44497c478bd9Sstevel@tonic-gate 				    pvt_to_str(PROP_VAL_LIST));
4450bbec428eSgjelinek 				saw_error = B_TRUE;
44517c478bd9Sstevel@tonic-gate 				return;
44527c478bd9Sstevel@tonic-gate 			}
44537c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(
44547c478bd9Sstevel@tonic-gate 			    in_progress_rctltab.zone_rctl_valptr);
44557c478bd9Sstevel@tonic-gate 			in_progress_rctltab.zone_rctl_valptr = NULL;
44567c478bd9Sstevel@tonic-gate 			if (!(pp->pv_type == PROP_VAL_LIST &&
44577c478bd9Sstevel@tonic-gate 			    pp->pv_list == NULL))
44587c478bd9Sstevel@tonic-gate 				add_property(cmd);
44597c478bd9Sstevel@tonic-gate 			break;
44607c478bd9Sstevel@tonic-gate 		default:
44617c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4462bbec428eSgjelinek 			    B_TRUE);
4463bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4464bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
44657c478bd9Sstevel@tonic-gate 			return;
44667c478bd9Sstevel@tonic-gate 		}
44677c478bd9Sstevel@tonic-gate 		return;
44687c478bd9Sstevel@tonic-gate 	case RT_ATTR:
44697c478bd9Sstevel@tonic-gate 		switch (prop_type) {
44707c478bd9Sstevel@tonic-gate 		case PT_NAME:
44717c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_attrtab.zone_attr_name,
44727c478bd9Sstevel@tonic-gate 			    prop_id,
44737c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_attrtab.zone_attr_name));
44747c478bd9Sstevel@tonic-gate 			break;
44757c478bd9Sstevel@tonic-gate 		case PT_TYPE:
44767c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_attrtab.zone_attr_type,
44777c478bd9Sstevel@tonic-gate 			    prop_id,
44787c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_attrtab.zone_attr_type));
44797c478bd9Sstevel@tonic-gate 			break;
44807c478bd9Sstevel@tonic-gate 		case PT_VALUE:
44817c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_attrtab.zone_attr_value,
44827c478bd9Sstevel@tonic-gate 			    prop_id,
44837c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_attrtab.zone_attr_value));
44847c478bd9Sstevel@tonic-gate 			break;
44857c478bd9Sstevel@tonic-gate 		default:
44867c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4487bbec428eSgjelinek 			    B_TRUE);
4488bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4489bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
44907c478bd9Sstevel@tonic-gate 			return;
44917c478bd9Sstevel@tonic-gate 		}
44927c478bd9Sstevel@tonic-gate 		return;
4493fa9e4066Sahrens 	case RT_DATASET:
4494fa9e4066Sahrens 		switch (prop_type) {
4495fa9e4066Sahrens 		case PT_NAME:
4496fa9e4066Sahrens 			(void) strlcpy(in_progress_dstab.zone_dataset_name,
4497fa9e4066Sahrens 			    prop_id,
4498fa9e4066Sahrens 			    sizeof (in_progress_dstab.zone_dataset_name));
4499fa9e4066Sahrens 			return;
4500fa9e4066Sahrens 		default:
4501fa9e4066Sahrens 			break;
4502fa9e4066Sahrens 		}
4503bbec428eSgjelinek 		zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, B_TRUE);
4504bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
4505bbec428eSgjelinek 		usage(B_FALSE, HELP_PROPS);
4506fa9e4066Sahrens 		return;
45070209230bSgjelinek 	case RT_DCPU:
45080209230bSgjelinek 		switch (prop_type) {
45090209230bSgjelinek 		char *lowp, *highp;
45100209230bSgjelinek 
45110209230bSgjelinek 		case PT_NCPUS:
45120209230bSgjelinek 			lowp = prop_id;
45130209230bSgjelinek 			if ((highp = strchr(prop_id, '-')) != NULL)
45140209230bSgjelinek 				*highp++ = '\0';
45150209230bSgjelinek 			else
45160209230bSgjelinek 				highp = lowp;
45170209230bSgjelinek 
45180209230bSgjelinek 			/* Make sure the input makes sense. */
45190209230bSgjelinek 			if (!zonecfg_valid_ncpus(lowp, highp)) {
45200209230bSgjelinek 				zerr(gettext("%s property is out of range."),
45210209230bSgjelinek 				    pt_to_str(PT_NCPUS));
4522bbec428eSgjelinek 				saw_error = B_TRUE;
45230209230bSgjelinek 				return;
45240209230bSgjelinek 			}
45250209230bSgjelinek 
45260209230bSgjelinek 			(void) strlcpy(
45270209230bSgjelinek 			    in_progress_psettab.zone_ncpu_min, lowp,
45280209230bSgjelinek 			    sizeof (in_progress_psettab.zone_ncpu_min));
45290209230bSgjelinek 			(void) strlcpy(
45300209230bSgjelinek 			    in_progress_psettab.zone_ncpu_max, highp,
45310209230bSgjelinek 			    sizeof (in_progress_psettab.zone_ncpu_max));
45320209230bSgjelinek 			return;
45330209230bSgjelinek 		case PT_IMPORTANCE:
45340209230bSgjelinek 			/* Make sure the value makes sense. */
45350209230bSgjelinek 			if (!zonecfg_valid_importance(prop_id)) {
45360209230bSgjelinek 				zerr(gettext("%s property is out of range."),
45370209230bSgjelinek 				    pt_to_str(PT_IMPORTANCE));
4538bbec428eSgjelinek 				saw_error = B_TRUE;
45390209230bSgjelinek 				return;
45400209230bSgjelinek 			}
45410209230bSgjelinek 
45420209230bSgjelinek 			(void) strlcpy(in_progress_psettab.zone_importance,
45430209230bSgjelinek 			    prop_id,
45440209230bSgjelinek 			    sizeof (in_progress_psettab.zone_importance));
45450209230bSgjelinek 			return;
45460209230bSgjelinek 		default:
45470209230bSgjelinek 			break;
45480209230bSgjelinek 		}
4549bbec428eSgjelinek 		zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, B_TRUE);
4550bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
4551bbec428eSgjelinek 		usage(B_FALSE, HELP_PROPS);
45520209230bSgjelinek 		return;
4553c97ad5cdSakolb 	case RT_PCAP:
4554c97ad5cdSakolb 		if (prop_type != PT_NCPUS) {
4555c97ad5cdSakolb 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4556bbec428eSgjelinek 			    B_TRUE);
4557bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4558bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
4559c97ad5cdSakolb 			return;
4560c97ad5cdSakolb 		}
4561c97ad5cdSakolb 
4562c97ad5cdSakolb 		/*
4563c97ad5cdSakolb 		 * We already checked that an rctl alias is allowed in
4564c97ad5cdSakolb 		 * the add_resource() function.
4565c97ad5cdSakolb 		 */
4566c97ad5cdSakolb 
4567c97ad5cdSakolb 		if ((cap = strtof(prop_id, &unitp)) <= 0 || *unitp != '\0' ||
4568c97ad5cdSakolb 		    (int)(cap * 100) < 1) {
4569c97ad5cdSakolb 			zerr(gettext("%s property is out of range."),
4570c97ad5cdSakolb 			    pt_to_str(PT_NCPUS));
4571bbec428eSgjelinek 			saw_error = B_TRUE;
4572c97ad5cdSakolb 			return;
4573c97ad5cdSakolb 		}
4574c97ad5cdSakolb 
4575c97ad5cdSakolb 		if ((err = zonecfg_set_aliased_rctl(handle, ALIAS_CPUCAP,
4576c97ad5cdSakolb 		    (int)(cap * 100))) != Z_OK)
4577bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
4578c97ad5cdSakolb 		else
4579bbec428eSgjelinek 			need_to_commit = B_TRUE;
4580c97ad5cdSakolb 		return;
45810209230bSgjelinek 	case RT_MCAP:
45820209230bSgjelinek 		switch (prop_type) {
45830209230bSgjelinek 		case PT_PHYSICAL:
45840209230bSgjelinek 			if (!zonecfg_valid_memlimit(prop_id, &mem_cap)) {
45850209230bSgjelinek 				zerr(gettext("A positive number with a "
45860209230bSgjelinek 				    "required scale suffix (K, M, G or T) was "
45870209230bSgjelinek 				    "expected here."));
4588bbec428eSgjelinek 				saw_error = B_TRUE;
45890209230bSgjelinek 			} else if (mem_cap < ONE_MB) {
45900209230bSgjelinek 				zerr(gettext("%s value is too small.  It must "
45910209230bSgjelinek 				    "be at least 1M."), pt_to_str(PT_PHYSICAL));
4592bbec428eSgjelinek 				saw_error = B_TRUE;
45930209230bSgjelinek 			} else {
45940209230bSgjelinek 				snprintf(in_progress_mcaptab.zone_physmem_cap,
45950209230bSgjelinek 				    physmem_size, "%llu", mem_cap);
45960209230bSgjelinek 			}
45970209230bSgjelinek 			break;
45980209230bSgjelinek 		case PT_SWAP:
45990209230bSgjelinek 			/*
46000209230bSgjelinek 			 * We have to check if an rctl is allowed here since
46010209230bSgjelinek 			 * there might already be a rctl defined that blocks
46020209230bSgjelinek 			 * the alias.
46030209230bSgjelinek 			 */
46040209230bSgjelinek 			if (!zonecfg_aliased_rctl_ok(handle, ALIAS_MAXSWAP)) {
46050209230bSgjelinek 				zone_perror(pt_to_str(PT_MAXSWAP),
4606bbec428eSgjelinek 				    Z_ALIAS_DISALLOW, B_FALSE);
4607bbec428eSgjelinek 				saw_error = B_TRUE;
46080209230bSgjelinek 				return;
46090209230bSgjelinek 			}
46100209230bSgjelinek 
46110209230bSgjelinek 			if (global_zone)
46120209230bSgjelinek 				mem_limit = ONE_MB * 100;
46130209230bSgjelinek 			else
46140209230bSgjelinek 				mem_limit = ONE_MB * 50;
46150209230bSgjelinek 
46160209230bSgjelinek 			if (!zonecfg_valid_memlimit(prop_id, &mem_cap)) {
46170209230bSgjelinek 				zerr(gettext("A positive number with a "
46180209230bSgjelinek 				    "required scale suffix (K, M, G or T) was "
46190209230bSgjelinek 				    "expected here."));
4620bbec428eSgjelinek 				saw_error = B_TRUE;
46210209230bSgjelinek 			} else if (mem_cap < mem_limit) {
46220209230bSgjelinek 				char buf[128];
46230209230bSgjelinek 
46240209230bSgjelinek 				(void) snprintf(buf, sizeof (buf), "%llu",
46250209230bSgjelinek 				    mem_limit);
46260209230bSgjelinek 				bytes_to_units(buf, buf, sizeof (buf));
46270209230bSgjelinek 				zerr(gettext("%s value is too small.  It must "
46280209230bSgjelinek 				    "be at least %s."), pt_to_str(PT_SWAP),
46290209230bSgjelinek 				    buf);
4630bbec428eSgjelinek 				saw_error = B_TRUE;
46310209230bSgjelinek 			} else {
46320209230bSgjelinek 				if ((err = zonecfg_set_aliased_rctl(handle,
46330209230bSgjelinek 				    ALIAS_MAXSWAP, mem_cap)) != Z_OK)
4634bbec428eSgjelinek 					zone_perror(zone, err, B_TRUE);
46350209230bSgjelinek 				else
4636bbec428eSgjelinek 					need_to_commit = B_TRUE;
46370209230bSgjelinek 			}
46380209230bSgjelinek 			break;
46390209230bSgjelinek 		case PT_LOCKED:
46400209230bSgjelinek 			/*
46410209230bSgjelinek 			 * We have to check if an rctl is allowed here since
46420209230bSgjelinek 			 * there might already be a rctl defined that blocks
46430209230bSgjelinek 			 * the alias.
46440209230bSgjelinek 			 */
46450209230bSgjelinek 			if (!zonecfg_aliased_rctl_ok(handle,
46460209230bSgjelinek 			    ALIAS_MAXLOCKEDMEM)) {
46470209230bSgjelinek 				zone_perror(pt_to_str(PT_LOCKED),
4648bbec428eSgjelinek 				    Z_ALIAS_DISALLOW, B_FALSE);
4649bbec428eSgjelinek 				saw_error = B_TRUE;
46500209230bSgjelinek 				return;
46510209230bSgjelinek 			}
46520209230bSgjelinek 
46530209230bSgjelinek 			if (!zonecfg_valid_memlimit(prop_id, &mem_cap)) {
46540209230bSgjelinek 				zerr(gettext("A non-negative number with a "
46550209230bSgjelinek 				    "required scale suffix (K, M, G or T) was "
46560209230bSgjelinek 				    "expected\nhere."));
4657bbec428eSgjelinek 				saw_error = B_TRUE;
46580209230bSgjelinek 			} else {
46590209230bSgjelinek 				if ((err = zonecfg_set_aliased_rctl(handle,
46600209230bSgjelinek 				    ALIAS_MAXLOCKEDMEM, mem_cap)) != Z_OK)
4661bbec428eSgjelinek 					zone_perror(zone, err, B_TRUE);
46620209230bSgjelinek 				else
4663bbec428eSgjelinek 					need_to_commit = B_TRUE;
46640209230bSgjelinek 			}
46650209230bSgjelinek 			break;
46660209230bSgjelinek 		default:
46670209230bSgjelinek 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4668bbec428eSgjelinek 			    B_TRUE);
4669bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4670bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
46710209230bSgjelinek 			return;
46720209230bSgjelinek 		}
46730209230bSgjelinek 		return;
4674cb8a054bSGlenn Faden 	case RT_ADMIN:
4675cb8a054bSGlenn Faden 		switch (prop_type) {
4676cb8a054bSGlenn Faden 		case PT_USER:
4677cb8a054bSGlenn Faden 			(void) strlcpy(in_progress_admintab.zone_admin_user,
4678cb8a054bSGlenn Faden 			    prop_id,
4679cb8a054bSGlenn Faden 			    sizeof (in_progress_admintab.zone_admin_user));
4680cb8a054bSGlenn Faden 			return;
4681cb8a054bSGlenn Faden 		case PT_AUTHS:
4682cb8a054bSGlenn Faden 			(void) strlcpy(in_progress_admintab.zone_admin_auths,
4683cb8a054bSGlenn Faden 			    prop_id,
4684cb8a054bSGlenn Faden 			    sizeof (in_progress_admintab.zone_admin_auths));
4685cb8a054bSGlenn Faden 			return;
4686cb8a054bSGlenn Faden 		default:
4687cb8a054bSGlenn Faden 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4688cb8a054bSGlenn Faden 			    B_TRUE);
4689cb8a054bSGlenn Faden 			long_usage(CMD_SET, B_TRUE);
4690cb8a054bSGlenn Faden 			usage(B_FALSE, HELP_PROPS);
4691cb8a054bSGlenn Faden 			return;
4692cb8a054bSGlenn Faden 		}
46937c478bd9Sstevel@tonic-gate 	default:
4694bbec428eSgjelinek 		zone_perror(rt_to_str(res_type), Z_NO_RESOURCE_TYPE, B_TRUE);
4695bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
4696bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
46977c478bd9Sstevel@tonic-gate 		return;
46987c478bd9Sstevel@tonic-gate 	}
46997c478bd9Sstevel@tonic-gate }
47007c478bd9Sstevel@tonic-gate 
47017c478bd9Sstevel@tonic-gate static void
4702bbec428eSgjelinek output_prop(FILE *fp, int pnum, char *pval, boolean_t print_notspec)
47037c478bd9Sstevel@tonic-gate {
47047c478bd9Sstevel@tonic-gate 	char *qstr;
47057c478bd9Sstevel@tonic-gate 
47067c478bd9Sstevel@tonic-gate 	if (*pval != '\0') {
47077c478bd9Sstevel@tonic-gate 		qstr = quoteit(pval);
47080209230bSgjelinek 		if (pnum == PT_SWAP || pnum == PT_LOCKED)
47090209230bSgjelinek 			(void) fprintf(fp, "\t[%s: %s]\n", pt_to_str(pnum),
47100209230bSgjelinek 			    qstr);
47110209230bSgjelinek 		else
47127c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s: %s\n", pt_to_str(pnum), qstr);
47137c478bd9Sstevel@tonic-gate 		free(qstr);
47147c478bd9Sstevel@tonic-gate 	} else if (print_notspec)
4715087719fdSdp 		(void) fprintf(fp, gettext("\t%s not specified\n"),
4716087719fdSdp 		    pt_to_str(pnum));
4717087719fdSdp }
4718087719fdSdp 
4719087719fdSdp static void
4720087719fdSdp info_zonename(zone_dochandle_t handle, FILE *fp)
4721087719fdSdp {
4722087719fdSdp 	char zonename[ZONENAME_MAX];
4723087719fdSdp 
4724087719fdSdp 	if (zonecfg_get_name(handle, zonename, sizeof (zonename)) == Z_OK)
4725087719fdSdp 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_ZONENAME),
4726087719fdSdp 		    zonename);
4727087719fdSdp 	else
4728087719fdSdp 		(void) fprintf(fp, gettext("%s not specified\n"),
4729087719fdSdp 		    pt_to_str(PT_ZONENAME));
47307c478bd9Sstevel@tonic-gate }
47317c478bd9Sstevel@tonic-gate 
47327c478bd9Sstevel@tonic-gate static void
47337c478bd9Sstevel@tonic-gate info_zonepath(zone_dochandle_t handle, FILE *fp)
47347c478bd9Sstevel@tonic-gate {
47357c478bd9Sstevel@tonic-gate 	char zonepath[MAXPATHLEN];
47367c478bd9Sstevel@tonic-gate 
47377c478bd9Sstevel@tonic-gate 	if (zonecfg_get_zonepath(handle, zonepath, sizeof (zonepath)) == Z_OK)
47387c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_ZONEPATH),
47397c478bd9Sstevel@tonic-gate 		    zonepath);
4740087719fdSdp 	else {
4741087719fdSdp 		(void) fprintf(fp, gettext("%s not specified\n"),
4742087719fdSdp 		    pt_to_str(PT_ZONEPATH));
4743087719fdSdp 	}
47447c478bd9Sstevel@tonic-gate }
47457c478bd9Sstevel@tonic-gate 
47467c478bd9Sstevel@tonic-gate static void
47479acbbeafSnn35248 info_brand(zone_dochandle_t handle, FILE *fp)
47489acbbeafSnn35248 {
47499acbbeafSnn35248 	char brand[MAXNAMELEN];
47509acbbeafSnn35248 
47519acbbeafSnn35248 	if (zonecfg_get_brand(handle, brand, sizeof (brand)) == Z_OK)
47529acbbeafSnn35248 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_BRAND),
47539acbbeafSnn35248 		    brand);
47549acbbeafSnn35248 	else
47559acbbeafSnn35248 		(void) fprintf(fp, "%s %s\n", pt_to_str(PT_BRAND),
47569acbbeafSnn35248 		    gettext("not specified"));
47579acbbeafSnn35248 }
47589acbbeafSnn35248 
47599acbbeafSnn35248 static void
47607c478bd9Sstevel@tonic-gate info_autoboot(zone_dochandle_t handle, FILE *fp)
47617c478bd9Sstevel@tonic-gate {
47627c478bd9Sstevel@tonic-gate 	boolean_t autoboot;
47637c478bd9Sstevel@tonic-gate 	int err;
47647c478bd9Sstevel@tonic-gate 
47657c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_get_autoboot(handle, &autoboot)) == Z_OK)
47667c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_AUTOBOOT),
47677c478bd9Sstevel@tonic-gate 		    autoboot ? "true" : "false");
47687c478bd9Sstevel@tonic-gate 	else
4769bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
47707c478bd9Sstevel@tonic-gate }
47717c478bd9Sstevel@tonic-gate 
47727c478bd9Sstevel@tonic-gate static void
47737c478bd9Sstevel@tonic-gate info_pool(zone_dochandle_t handle, FILE *fp)
47747c478bd9Sstevel@tonic-gate {
47757c478bd9Sstevel@tonic-gate 	char pool[MAXNAMELEN];
47767c478bd9Sstevel@tonic-gate 	int err;
47777c478bd9Sstevel@tonic-gate 
47787c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_get_pool(handle, pool, sizeof (pool))) == Z_OK)
47797c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_POOL), pool);
47807c478bd9Sstevel@tonic-gate 	else
4781bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
47827c478bd9Sstevel@tonic-gate }
47837c478bd9Sstevel@tonic-gate 
47847c478bd9Sstevel@tonic-gate static void
4785ffbafc53Scomay info_limitpriv(zone_dochandle_t handle, FILE *fp)
4786ffbafc53Scomay {
4787ffbafc53Scomay 	char *limitpriv;
4788ffbafc53Scomay 	int err;
4789ffbafc53Scomay 
4790ffbafc53Scomay 	if ((err = zonecfg_get_limitpriv(handle, &limitpriv)) == Z_OK) {
4791ffbafc53Scomay 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_LIMITPRIV),
4792ffbafc53Scomay 		    limitpriv);
4793ffbafc53Scomay 		free(limitpriv);
4794ffbafc53Scomay 	} else {
4795bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
4796ffbafc53Scomay 	}
4797ffbafc53Scomay }
4798ffbafc53Scomay 
4799ffbafc53Scomay static void
48003f2f09c1Sdp info_bootargs(zone_dochandle_t handle, FILE *fp)
48013f2f09c1Sdp {
48023f2f09c1Sdp 	char bootargs[BOOTARGS_MAX];
48033f2f09c1Sdp 	int err;
48043f2f09c1Sdp 
48053f2f09c1Sdp 	if ((err = zonecfg_get_bootargs(handle, bootargs,
48063f2f09c1Sdp 	    sizeof (bootargs))) == Z_OK) {
48073f2f09c1Sdp 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_BOOTARGS),
48083f2f09c1Sdp 		    bootargs);
48093f2f09c1Sdp 	} else {
4810bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
48113f2f09c1Sdp 	}
48123f2f09c1Sdp }
48133f2f09c1Sdp 
48143f2f09c1Sdp static void
48150209230bSgjelinek info_sched(zone_dochandle_t handle, FILE *fp)
48160209230bSgjelinek {
48170209230bSgjelinek 	char sched[MAXNAMELEN];
48180209230bSgjelinek 	int err;
48190209230bSgjelinek 
48200209230bSgjelinek 	if ((err = zonecfg_get_sched_class(handle, sched, sizeof (sched)))
48210209230bSgjelinek 	    == Z_OK) {
48220209230bSgjelinek 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_SCHED), sched);
48230209230bSgjelinek 	} else {
4824bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
48250209230bSgjelinek 	}
48260209230bSgjelinek }
48270209230bSgjelinek 
48280209230bSgjelinek static void
4829f4b3ec61Sdh155122 info_iptype(zone_dochandle_t handle, FILE *fp)
4830f4b3ec61Sdh155122 {
4831f4b3ec61Sdh155122 	zone_iptype_t iptype;
4832f4b3ec61Sdh155122 	int err;
4833f4b3ec61Sdh155122 
4834f4b3ec61Sdh155122 	if ((err = zonecfg_get_iptype(handle, &iptype)) == Z_OK) {
4835f4b3ec61Sdh155122 		switch (iptype) {
4836f4b3ec61Sdh155122 		case ZS_SHARED:
4837f4b3ec61Sdh155122 			(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_IPTYPE),
4838f4b3ec61Sdh155122 			    "shared");
4839f4b3ec61Sdh155122 			break;
4840f4b3ec61Sdh155122 		case ZS_EXCLUSIVE:
4841f4b3ec61Sdh155122 			(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_IPTYPE),
4842f4b3ec61Sdh155122 			    "exclusive");
4843f4b3ec61Sdh155122 			break;
4844f4b3ec61Sdh155122 		}
4845f4b3ec61Sdh155122 	} else {
4846bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
4847f4b3ec61Sdh155122 	}
4848f4b3ec61Sdh155122 }
4849f4b3ec61Sdh155122 
4850f4b3ec61Sdh155122 static void
48515679c89fSjv227347 info_hostid(zone_dochandle_t handle, FILE *fp)
48525679c89fSjv227347 {
48535679c89fSjv227347 	char hostidp[HW_HOSTID_LEN];
48540fbb751dSJohn Levon 	int err;
48555679c89fSjv227347 
48560fbb751dSJohn Levon 	if ((err = zonecfg_get_hostid(handle, hostidp,
48570fbb751dSJohn Levon 	    sizeof (hostidp))) == Z_OK) {
48585679c89fSjv227347 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_HOSTID), hostidp);
48590fbb751dSJohn Levon 	} else if (err == Z_BAD_PROPERTY) {
48600fbb751dSJohn Levon 		(void) fprintf(fp, "%s: \n", pt_to_str(PT_HOSTID));
48610fbb751dSJohn Levon 	} else {
48620fbb751dSJohn Levon 		zone_perror(zone, err, B_TRUE);
48630fbb751dSJohn Levon 	}
48640fbb751dSJohn Levon }
48650fbb751dSJohn Levon 
48660fbb751dSJohn Levon static void
48670fbb751dSJohn Levon info_fs_allowed(zone_dochandle_t handle, FILE *fp)
48680fbb751dSJohn Levon {
48690fbb751dSJohn Levon 	char fsallowedp[ZONE_FS_ALLOWED_MAX];
48700fbb751dSJohn Levon 	int err;
48710fbb751dSJohn Levon 
48720fbb751dSJohn Levon 	if ((err = zonecfg_get_fs_allowed(handle, fsallowedp,
48730fbb751dSJohn Levon 	    sizeof (fsallowedp))) == Z_OK) {
48740fbb751dSJohn Levon 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_FS_ALLOWED),
48750fbb751dSJohn Levon 		    fsallowedp);
48760fbb751dSJohn Levon 	} else if (err == Z_BAD_PROPERTY) {
48770fbb751dSJohn Levon 		(void) fprintf(fp, "%s: \n", pt_to_str(PT_FS_ALLOWED));
48780fbb751dSJohn Levon 	} else {
48790fbb751dSJohn Levon 		zone_perror(zone, err, B_TRUE);
48800fbb751dSJohn Levon 	}
48815679c89fSjv227347 }
48825679c89fSjv227347 
48835679c89fSjv227347 static void
48847c478bd9Sstevel@tonic-gate output_fs(FILE *fp, struct zone_fstab *fstab)
48857c478bd9Sstevel@tonic-gate {
48867c478bd9Sstevel@tonic-gate 	zone_fsopt_t *this;
48877c478bd9Sstevel@tonic-gate 
48887c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_FS));
48897c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_DIR, fstab->zone_fs_dir, B_TRUE);
48907c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_SPECIAL, fstab->zone_fs_special, B_TRUE);
48917c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_RAW, fstab->zone_fs_raw, B_TRUE);
48927c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_TYPE, fstab->zone_fs_type, B_TRUE);
48937c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "\t%s: [", pt_to_str(PT_OPTIONS));
48947c478bd9Sstevel@tonic-gate 	for (this = fstab->zone_fs_options; this != NULL;
48957c478bd9Sstevel@tonic-gate 	    this = this->zone_fsopt_next) {
48967c478bd9Sstevel@tonic-gate 		if (strchr(this->zone_fsopt_opt, '='))
48977c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\"%s\"", this->zone_fsopt_opt);
48987c478bd9Sstevel@tonic-gate 		else
48997c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "%s", this->zone_fsopt_opt);
49007c478bd9Sstevel@tonic-gate 		if (this->zone_fsopt_next != NULL)
49017c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, ",");
49027c478bd9Sstevel@tonic-gate 	}
49037c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "]\n");
49047c478bd9Sstevel@tonic-gate }
49057c478bd9Sstevel@tonic-gate 
49067c478bd9Sstevel@tonic-gate static void
49077c478bd9Sstevel@tonic-gate info_fs(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
49087c478bd9Sstevel@tonic-gate {
49097c478bd9Sstevel@tonic-gate 	struct zone_fstab lookup, user;
4910bbec428eSgjelinek 	boolean_t output = B_FALSE;
49117c478bd9Sstevel@tonic-gate 
49127c478bd9Sstevel@tonic-gate 	if (zonecfg_setfsent(handle) != Z_OK)
49137c478bd9Sstevel@tonic-gate 		return;
49147c478bd9Sstevel@tonic-gate 	while (zonecfg_getfsent(handle, &lookup) == Z_OK) {
49157c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
49167c478bd9Sstevel@tonic-gate 			output_fs(fp, &lookup);
49177c478bd9Sstevel@tonic-gate 			goto loopend;
49187c478bd9Sstevel@tonic-gate 		}
4919bbec428eSgjelinek 		if (fill_in_fstab(cmd, &user, B_TRUE) != Z_OK)
49207c478bd9Sstevel@tonic-gate 			goto loopend;
49217c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_fs_dir) > 0 &&
49227c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_fs_dir, lookup.zone_fs_dir) != 0)
49237c478bd9Sstevel@tonic-gate 			goto loopend;	/* no match */
49247c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_fs_special) > 0 &&
49257c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_fs_special, lookup.zone_fs_special) != 0)
49267c478bd9Sstevel@tonic-gate 			goto loopend;	/* no match */
49277c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_fs_type) > 0 &&
49287c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_fs_type, lookup.zone_fs_type) != 0)
49297c478bd9Sstevel@tonic-gate 			goto loopend;	/* no match */
49307c478bd9Sstevel@tonic-gate 		output_fs(fp, &lookup);
4931bbec428eSgjelinek 		output = B_TRUE;
49327c478bd9Sstevel@tonic-gate loopend:
49337c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(lookup.zone_fs_options);
49347c478bd9Sstevel@tonic-gate 	}
49357c478bd9Sstevel@tonic-gate 	(void) zonecfg_endfsent(handle);
49367c478bd9Sstevel@tonic-gate 	/*
49377c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
49387c478bd9Sstevel@tonic-gate 	 * nothing to output.
49397c478bd9Sstevel@tonic-gate 	 */
49407c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
49417c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
49427c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_FS));
49437c478bd9Sstevel@tonic-gate }
49447c478bd9Sstevel@tonic-gate 
49457c478bd9Sstevel@tonic-gate static void
49467c478bd9Sstevel@tonic-gate output_net(FILE *fp, struct zone_nwiftab *nwiftab)
49477c478bd9Sstevel@tonic-gate {
49487c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_NET));
49497c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_ADDRESS, nwiftab->zone_nwif_address, B_TRUE);
4950550b6e40SSowmini Varadhan 	output_prop(fp, PT_ALLOWED_ADDRESS,
4951550b6e40SSowmini Varadhan 	    nwiftab->zone_nwif_allowed_address, B_TRUE);
49527c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_PHYSICAL, nwiftab->zone_nwif_physical, B_TRUE);
4953de860bd9Sgfaden 	output_prop(fp, PT_DEFROUTER, nwiftab->zone_nwif_defrouter, B_TRUE);
49547c478bd9Sstevel@tonic-gate }
49557c478bd9Sstevel@tonic-gate 
49567c478bd9Sstevel@tonic-gate static void
49577c478bd9Sstevel@tonic-gate info_net(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
49587c478bd9Sstevel@tonic-gate {
49597c478bd9Sstevel@tonic-gate 	struct zone_nwiftab lookup, user;
4960bbec428eSgjelinek 	boolean_t output = B_FALSE;
49617c478bd9Sstevel@tonic-gate 
49627c478bd9Sstevel@tonic-gate 	if (zonecfg_setnwifent(handle) != Z_OK)
49637c478bd9Sstevel@tonic-gate 		return;
49647c478bd9Sstevel@tonic-gate 	while (zonecfg_getnwifent(handle, &lookup) == Z_OK) {
49657c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
49667c478bd9Sstevel@tonic-gate 			output_net(fp, &lookup);
49677c478bd9Sstevel@tonic-gate 			continue;
49687c478bd9Sstevel@tonic-gate 		}
4969bbec428eSgjelinek 		if (fill_in_nwiftab(cmd, &user, B_TRUE) != Z_OK)
49707c478bd9Sstevel@tonic-gate 			continue;
49717c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_nwif_physical) > 0 &&
49727c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_nwif_physical,
49737c478bd9Sstevel@tonic-gate 		    lookup.zone_nwif_physical) != 0)
49747c478bd9Sstevel@tonic-gate 			continue;	/* no match */
4975f4b3ec61Sdh155122 		/* If present make sure it matches */
49767c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_nwif_address) > 0 &&
49777c478bd9Sstevel@tonic-gate 		    !zonecfg_same_net_address(user.zone_nwif_address,
49787c478bd9Sstevel@tonic-gate 		    lookup.zone_nwif_address))
49797c478bd9Sstevel@tonic-gate 			continue;	/* no match */
49807c478bd9Sstevel@tonic-gate 		output_net(fp, &lookup);
4981bbec428eSgjelinek 		output = B_TRUE;
49827c478bd9Sstevel@tonic-gate 	}
49837c478bd9Sstevel@tonic-gate 	(void) zonecfg_endnwifent(handle);
49847c478bd9Sstevel@tonic-gate 	/*
49857c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
49867c478bd9Sstevel@tonic-gate 	 * nothing to output.
49877c478bd9Sstevel@tonic-gate 	 */
49887c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
49897c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
49907c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_NET));
49917c478bd9Sstevel@tonic-gate }
49927c478bd9Sstevel@tonic-gate 
49937c478bd9Sstevel@tonic-gate static void
49947c478bd9Sstevel@tonic-gate output_dev(FILE *fp, struct zone_devtab *devtab)
49957c478bd9Sstevel@tonic-gate {
499627e6fb21Sdp 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_DEVICE));
49977c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_MATCH, devtab->zone_dev_match, B_TRUE);
49987c478bd9Sstevel@tonic-gate }
49997c478bd9Sstevel@tonic-gate 
50007c478bd9Sstevel@tonic-gate static void
50017c478bd9Sstevel@tonic-gate info_dev(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
50027c478bd9Sstevel@tonic-gate {
50037c478bd9Sstevel@tonic-gate 	struct zone_devtab lookup, user;
5004bbec428eSgjelinek 	boolean_t output = B_FALSE;
50057c478bd9Sstevel@tonic-gate 
50067c478bd9Sstevel@tonic-gate 	if (zonecfg_setdevent(handle) != Z_OK)
50077c478bd9Sstevel@tonic-gate 		return;
50087c478bd9Sstevel@tonic-gate 	while (zonecfg_getdevent(handle, &lookup) == Z_OK) {
50097c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
50107c478bd9Sstevel@tonic-gate 			output_dev(fp, &lookup);
50117c478bd9Sstevel@tonic-gate 			continue;
50127c478bd9Sstevel@tonic-gate 		}
5013bbec428eSgjelinek 		if (fill_in_devtab(cmd, &user, B_TRUE) != Z_OK)
50147c478bd9Sstevel@tonic-gate 			continue;
50157c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_dev_match) > 0 &&
50167c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_dev_match, lookup.zone_dev_match) != 0)
50177c478bd9Sstevel@tonic-gate 			continue;	/* no match */
50187c478bd9Sstevel@tonic-gate 		output_dev(fp, &lookup);
5019bbec428eSgjelinek 		output = B_TRUE;
50207c478bd9Sstevel@tonic-gate 	}
50217c478bd9Sstevel@tonic-gate 	(void) zonecfg_enddevent(handle);
50227c478bd9Sstevel@tonic-gate 	/*
50237c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
50247c478bd9Sstevel@tonic-gate 	 * nothing to output.
50257c478bd9Sstevel@tonic-gate 	 */
50267c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
50277c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
50287c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_DEVICE));
50297c478bd9Sstevel@tonic-gate }
50307c478bd9Sstevel@tonic-gate 
50317c478bd9Sstevel@tonic-gate static void
50327c478bd9Sstevel@tonic-gate output_rctl(FILE *fp, struct zone_rctltab *rctltab)
50337c478bd9Sstevel@tonic-gate {
50347c478bd9Sstevel@tonic-gate 	struct zone_rctlvaltab *valptr;
50357c478bd9Sstevel@tonic-gate 
50367c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_RCTL));
50377c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_NAME, rctltab->zone_rctl_name, B_TRUE);
50387c478bd9Sstevel@tonic-gate 	for (valptr = rctltab->zone_rctl_valptr; valptr != NULL;
50397c478bd9Sstevel@tonic-gate 	    valptr = valptr->zone_rctlval_next) {
50407c478bd9Sstevel@tonic-gate 		fprintf(fp, "\t%s: (%s=%s,%s=%s,%s=%s)\n",
50417c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_VALUE),
50427c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_PRIV), valptr->zone_rctlval_priv,
50437c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_LIMIT), valptr->zone_rctlval_limit,
50447c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_ACTION), valptr->zone_rctlval_action);
50457c478bd9Sstevel@tonic-gate 	}
50467c478bd9Sstevel@tonic-gate }
50477c478bd9Sstevel@tonic-gate 
50487c478bd9Sstevel@tonic-gate static void
50497c478bd9Sstevel@tonic-gate info_rctl(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
50507c478bd9Sstevel@tonic-gate {
50517c478bd9Sstevel@tonic-gate 	struct zone_rctltab lookup, user;
5052bbec428eSgjelinek 	boolean_t output = B_FALSE;
50537c478bd9Sstevel@tonic-gate 
50547c478bd9Sstevel@tonic-gate 	if (zonecfg_setrctlent(handle) != Z_OK)
50557c478bd9Sstevel@tonic-gate 		return;
50567c478bd9Sstevel@tonic-gate 	while (zonecfg_getrctlent(handle, &lookup) == Z_OK) {
50577c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
50587c478bd9Sstevel@tonic-gate 			output_rctl(fp, &lookup);
5059bbec428eSgjelinek 		} else if (fill_in_rctltab(cmd, &user, B_TRUE) == Z_OK &&
50607c478bd9Sstevel@tonic-gate 		    (strlen(user.zone_rctl_name) == 0 ||
50617c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_rctl_name, lookup.zone_rctl_name) == 0)) {
50627c478bd9Sstevel@tonic-gate 			output_rctl(fp, &lookup);
5063bbec428eSgjelinek 			output = B_TRUE;
50647c478bd9Sstevel@tonic-gate 		}
50657c478bd9Sstevel@tonic-gate 		zonecfg_free_rctl_value_list(lookup.zone_rctl_valptr);
50667c478bd9Sstevel@tonic-gate 	}
50677c478bd9Sstevel@tonic-gate 	(void) zonecfg_endrctlent(handle);
50687c478bd9Sstevel@tonic-gate 	/*
50697c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
50707c478bd9Sstevel@tonic-gate 	 * nothing to output.
50717c478bd9Sstevel@tonic-gate 	 */
50727c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
50737c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
50747c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_RCTL));
50757c478bd9Sstevel@tonic-gate }
50767c478bd9Sstevel@tonic-gate 
50777c478bd9Sstevel@tonic-gate static void
50787c478bd9Sstevel@tonic-gate output_attr(FILE *fp, struct zone_attrtab *attrtab)
50797c478bd9Sstevel@tonic-gate {
50807c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_ATTR));
50817c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_NAME, attrtab->zone_attr_name, B_TRUE);
50827c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_TYPE, attrtab->zone_attr_type, B_TRUE);
50837c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_VALUE, attrtab->zone_attr_value, B_TRUE);
50847c478bd9Sstevel@tonic-gate }
50857c478bd9Sstevel@tonic-gate 
50867c478bd9Sstevel@tonic-gate static void
50877c478bd9Sstevel@tonic-gate info_attr(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
50887c478bd9Sstevel@tonic-gate {
50897c478bd9Sstevel@tonic-gate 	struct zone_attrtab lookup, user;
5090bbec428eSgjelinek 	boolean_t output = B_FALSE;
50917c478bd9Sstevel@tonic-gate 
50927c478bd9Sstevel@tonic-gate 	if (zonecfg_setattrent(handle) != Z_OK)
50937c478bd9Sstevel@tonic-gate 		return;
50947c478bd9Sstevel@tonic-gate 	while (zonecfg_getattrent(handle, &lookup) == Z_OK) {
50957c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
50967c478bd9Sstevel@tonic-gate 			output_attr(fp, &lookup);
50977c478bd9Sstevel@tonic-gate 			continue;
50987c478bd9Sstevel@tonic-gate 		}
5099bbec428eSgjelinek 		if (fill_in_attrtab(cmd, &user, B_TRUE) != Z_OK)
51007c478bd9Sstevel@tonic-gate 			continue;
51017c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_attr_name) > 0 &&
51027c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_attr_name, lookup.zone_attr_name) != 0)
51037c478bd9Sstevel@tonic-gate 			continue;	/* no match */
51047c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_attr_type) > 0 &&
51057c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_attr_type, lookup.zone_attr_type) != 0)
51067c478bd9Sstevel@tonic-gate 			continue;	/* no match */
51077c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_attr_value) > 0 &&
51087c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_attr_value, lookup.zone_attr_value) != 0)
51097c478bd9Sstevel@tonic-gate 			continue;	/* no match */
51107c478bd9Sstevel@tonic-gate 		output_attr(fp, &lookup);
5111bbec428eSgjelinek 		output = B_TRUE;
51127c478bd9Sstevel@tonic-gate 	}
51137c478bd9Sstevel@tonic-gate 	(void) zonecfg_endattrent(handle);
51147c478bd9Sstevel@tonic-gate 	/*
51157c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
51167c478bd9Sstevel@tonic-gate 	 * nothing to output.
51177c478bd9Sstevel@tonic-gate 	 */
51187c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
51197c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
51207c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR));
51217c478bd9Sstevel@tonic-gate }
51227c478bd9Sstevel@tonic-gate 
5123fa9e4066Sahrens static void
5124fa9e4066Sahrens output_ds(FILE *fp, struct zone_dstab *dstab)
5125fa9e4066Sahrens {
5126fa9e4066Sahrens 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_DATASET));
5127fa9e4066Sahrens 	output_prop(fp, PT_NAME, dstab->zone_dataset_name, B_TRUE);
5128fa9e4066Sahrens }
5129fa9e4066Sahrens 
5130fa9e4066Sahrens static void
5131fa9e4066Sahrens info_ds(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
5132fa9e4066Sahrens {
5133fa9e4066Sahrens 	struct zone_dstab lookup, user;
5134bbec428eSgjelinek 	boolean_t output = B_FALSE;
5135fa9e4066Sahrens 
51360209230bSgjelinek 	if (zonecfg_setdsent(handle) != Z_OK)
5137fa9e4066Sahrens 		return;
5138fa9e4066Sahrens 	while (zonecfg_getdsent(handle, &lookup) == Z_OK) {
5139fa9e4066Sahrens 		if (cmd->cmd_prop_nv_pairs == 0) {
5140fa9e4066Sahrens 			output_ds(fp, &lookup);
5141fa9e4066Sahrens 			continue;
5142fa9e4066Sahrens 		}
5143bbec428eSgjelinek 		if (fill_in_dstab(cmd, &user, B_TRUE) != Z_OK)
5144fa9e4066Sahrens 			continue;
5145fa9e4066Sahrens 		if (strlen(user.zone_dataset_name) > 0 &&
5146fa9e4066Sahrens 		    strcmp(user.zone_dataset_name,
5147fa9e4066Sahrens 		    lookup.zone_dataset_name) != 0)
5148fa9e4066Sahrens 			continue;	/* no match */
5149fa9e4066Sahrens 		output_ds(fp, &lookup);
5150bbec428eSgjelinek 		output = B_TRUE;
5151fa9e4066Sahrens 	}
5152fa9e4066Sahrens 	(void) zonecfg_enddsent(handle);
5153fa9e4066Sahrens 	/*
5154fa9e4066Sahrens 	 * If a property n/v pair was specified, warn the user if there was
5155fa9e4066Sahrens 	 * nothing to output.
5156fa9e4066Sahrens 	 */
5157fa9e4066Sahrens 	if (!output && cmd->cmd_prop_nv_pairs > 0)
5158fa9e4066Sahrens 		(void) printf(gettext("No such %s resource.\n"),
5159fa9e4066Sahrens 		    rt_to_str(RT_DATASET));
5160fa9e4066Sahrens }
5161fa9e4066Sahrens 
51620209230bSgjelinek static void
51630209230bSgjelinek output_pset(FILE *fp, struct zone_psettab *psettab)
51640209230bSgjelinek {
51650209230bSgjelinek 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_DCPU));
51660209230bSgjelinek 	if (strcmp(psettab->zone_ncpu_min, psettab->zone_ncpu_max) == 0)
51670209230bSgjelinek 		(void) fprintf(fp, "\t%s: %s\n", pt_to_str(PT_NCPUS),
51680209230bSgjelinek 		    psettab->zone_ncpu_max);
51690209230bSgjelinek 	else
51700209230bSgjelinek 		(void) fprintf(fp, "\t%s: %s-%s\n", pt_to_str(PT_NCPUS),
51710209230bSgjelinek 		    psettab->zone_ncpu_min, psettab->zone_ncpu_max);
51720209230bSgjelinek 	if (psettab->zone_importance[0] != '\0')
51730209230bSgjelinek 		(void) fprintf(fp, "\t%s: %s\n", pt_to_str(PT_IMPORTANCE),
51740209230bSgjelinek 		    psettab->zone_importance);
51750209230bSgjelinek }
51760209230bSgjelinek 
51770209230bSgjelinek static void
51780209230bSgjelinek info_pset(zone_dochandle_t handle, FILE *fp)
51790209230bSgjelinek {
51800209230bSgjelinek 	struct zone_psettab lookup;
51810209230bSgjelinek 
51820209230bSgjelinek 	if (zonecfg_getpsetent(handle, &lookup) == Z_OK)
51830209230bSgjelinek 		output_pset(fp, &lookup);
51840209230bSgjelinek }
51850209230bSgjelinek 
51860209230bSgjelinek static void
5187c97ad5cdSakolb output_pcap(FILE *fp)
5188c97ad5cdSakolb {
5189c97ad5cdSakolb 	uint64_t cap;
5190c97ad5cdSakolb 
5191c97ad5cdSakolb 	if (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &cap) == Z_OK) {
5192c97ad5cdSakolb 		float scaled = (float)cap / 100;
5193c97ad5cdSakolb 		(void) fprintf(fp, "%s:\n", rt_to_str(RT_PCAP));
5194c97ad5cdSakolb 		(void) fprintf(fp, "\t[%s: %.2f]\n", pt_to_str(PT_NCPUS),
5195c97ad5cdSakolb 		    scaled);
5196c97ad5cdSakolb 	}
5197c97ad5cdSakolb }
5198c97ad5cdSakolb 
5199c97ad5cdSakolb static void
5200c97ad5cdSakolb info_pcap(FILE *fp)
5201c97ad5cdSakolb {
5202c97ad5cdSakolb 	output_pcap(fp);
5203c97ad5cdSakolb }
5204c97ad5cdSakolb 
5205c97ad5cdSakolb 
5206c97ad5cdSakolb static void
52070209230bSgjelinek info_aliased_rctl(zone_dochandle_t handle, FILE *fp, char *alias)
52080209230bSgjelinek {
52090209230bSgjelinek 	uint64_t limit;
52100209230bSgjelinek 
52110209230bSgjelinek 	if (zonecfg_get_aliased_rctl(handle, alias, &limit) == Z_OK) {
52120209230bSgjelinek 		/* convert memory based properties */
52130209230bSgjelinek 		if (strcmp(alias, ALIAS_MAXSHMMEM) == 0) {
52140209230bSgjelinek 			char buf[128];
52150209230bSgjelinek 
52160209230bSgjelinek 			(void) snprintf(buf, sizeof (buf), "%llu", limit);
52170209230bSgjelinek 			bytes_to_units(buf, buf, sizeof (buf));
52180209230bSgjelinek 			(void) fprintf(fp, "[%s: %s]\n", alias, buf);
52190209230bSgjelinek 			return;
52200209230bSgjelinek 		}
52210209230bSgjelinek 
52220209230bSgjelinek 		(void) fprintf(fp, "[%s: %llu]\n", alias, limit);
52230209230bSgjelinek 	}
52240209230bSgjelinek }
52250209230bSgjelinek 
52260209230bSgjelinek static void
52270209230bSgjelinek bytes_to_units(char *str, char *buf, int bufsize)
52280209230bSgjelinek {
52290209230bSgjelinek 	unsigned long long num;
52300209230bSgjelinek 	unsigned long long save = 0;
52310209230bSgjelinek 	char *units = "BKMGT";
52320209230bSgjelinek 	char *up = units;
52330209230bSgjelinek 
52340209230bSgjelinek 	num = strtoll(str, NULL, 10);
52350209230bSgjelinek 
52360209230bSgjelinek 	if (num < 1024) {
52370209230bSgjelinek 		(void) snprintf(buf, bufsize, "%llu", num);
52380209230bSgjelinek 		return;
52390209230bSgjelinek 	}
52400209230bSgjelinek 
52410209230bSgjelinek 	while ((num >= 1024) && (*up != 'T')) {
52420209230bSgjelinek 		up++; /* next unit of measurement */
52430209230bSgjelinek 		save = num;
52440209230bSgjelinek 		num = (num + 512) >> 10;
52450209230bSgjelinek 	}
52460209230bSgjelinek 
52470209230bSgjelinek 	/* check if we should output a fraction.  snprintf will round for us */
52480209230bSgjelinek 	if (save % 1024 != 0 && ((save >> 10) < 10))
52490209230bSgjelinek 		(void) snprintf(buf, bufsize, "%2.1f%c", ((float)save / 1024),
52500209230bSgjelinek 		    *up);
52510209230bSgjelinek 	else
52520209230bSgjelinek 		(void) snprintf(buf, bufsize, "%llu%c", num, *up);
52530209230bSgjelinek }
52540209230bSgjelinek 
52550209230bSgjelinek static void
52560209230bSgjelinek output_mcap(FILE *fp, struct zone_mcaptab *mcaptab, int showswap,
52570209230bSgjelinek     uint64_t maxswap, int showlocked, uint64_t maxlocked)
52580209230bSgjelinek {
52590209230bSgjelinek 	char buf[128];
52600209230bSgjelinek 
52610209230bSgjelinek 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_MCAP));
52620209230bSgjelinek 	if (mcaptab->zone_physmem_cap[0] != '\0') {
52630209230bSgjelinek 		bytes_to_units(mcaptab->zone_physmem_cap, buf, sizeof (buf));
52640209230bSgjelinek 		output_prop(fp, PT_PHYSICAL, buf, B_TRUE);
52650209230bSgjelinek 	}
52660209230bSgjelinek 
52670209230bSgjelinek 	if (showswap == Z_OK) {
52680209230bSgjelinek 		(void) snprintf(buf, sizeof (buf), "%llu", maxswap);
52690209230bSgjelinek 		bytes_to_units(buf, buf, sizeof (buf));
52700209230bSgjelinek 		output_prop(fp, PT_SWAP, buf, B_TRUE);
52710209230bSgjelinek 	}
52720209230bSgjelinek 
52730209230bSgjelinek 	if (showlocked == Z_OK) {
52740209230bSgjelinek 		(void) snprintf(buf, sizeof (buf), "%llu", maxlocked);
52750209230bSgjelinek 		bytes_to_units(buf, buf, sizeof (buf));
52760209230bSgjelinek 		output_prop(fp, PT_LOCKED, buf, B_TRUE);
52770209230bSgjelinek 	}
52780209230bSgjelinek }
52790209230bSgjelinek 
52800209230bSgjelinek static void
52810209230bSgjelinek info_mcap(zone_dochandle_t handle, FILE *fp)
52820209230bSgjelinek {
52830209230bSgjelinek 	int res1, res2, res3;
52840209230bSgjelinek 	uint64_t swap_limit;
52850209230bSgjelinek 	uint64_t locked_limit;
52860209230bSgjelinek 	struct zone_mcaptab lookup;
52870209230bSgjelinek 
52880209230bSgjelinek 	bzero(&lookup, sizeof (lookup));
52890209230bSgjelinek 	res1 = zonecfg_getmcapent(handle, &lookup);
52900209230bSgjelinek 	res2 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP, &swap_limit);
52910209230bSgjelinek 	res3 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM,
52920209230bSgjelinek 	    &locked_limit);
52930209230bSgjelinek 
52940209230bSgjelinek 	if (res1 == Z_OK || res2 == Z_OK || res3 == Z_OK)
52950209230bSgjelinek 		output_mcap(fp, &lookup, res2, swap_limit, res3, locked_limit);
52960209230bSgjelinek }
52970209230bSgjelinek 
5298cb8a054bSGlenn Faden static void
5299cb8a054bSGlenn Faden output_auth(FILE *fp, struct zone_admintab *admintab)
5300cb8a054bSGlenn Faden {
5301cb8a054bSGlenn Faden 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_ADMIN));
5302cb8a054bSGlenn Faden 	output_prop(fp, PT_USER, admintab->zone_admin_user, B_TRUE);
5303cb8a054bSGlenn Faden 	output_prop(fp, PT_AUTHS, admintab->zone_admin_auths, B_TRUE);
5304cb8a054bSGlenn Faden }
5305cb8a054bSGlenn Faden 
5306cb8a054bSGlenn Faden static void
5307cb8a054bSGlenn Faden info_auth(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
5308cb8a054bSGlenn Faden {
5309cb8a054bSGlenn Faden 	struct zone_admintab lookup, user;
5310cb8a054bSGlenn Faden 	boolean_t output = B_FALSE;
5311cb8a054bSGlenn Faden 	int err;
5312cb8a054bSGlenn Faden 
5313cb8a054bSGlenn Faden 	if ((err = zonecfg_setadminent(handle)) != Z_OK) {
5314cb8a054bSGlenn Faden 		zone_perror(zone, err, B_TRUE);
5315cb8a054bSGlenn Faden 		return;
5316cb8a054bSGlenn Faden 	}
5317cb8a054bSGlenn Faden 	while (zonecfg_getadminent(handle, &lookup) == Z_OK) {
5318cb8a054bSGlenn Faden 		if (cmd->cmd_prop_nv_pairs == 0) {
5319cb8a054bSGlenn Faden 			output_auth(fp, &lookup);
5320cb8a054bSGlenn Faden 			continue;
5321cb8a054bSGlenn Faden 		}
5322cb8a054bSGlenn Faden 		if (fill_in_admintab(cmd, &user, B_TRUE) != Z_OK)
5323cb8a054bSGlenn Faden 			continue;
5324cb8a054bSGlenn Faden 		if (strlen(user.zone_admin_user) > 0 &&
5325cb8a054bSGlenn Faden 		    strcmp(user.zone_admin_user, lookup.zone_admin_user) != 0)
5326cb8a054bSGlenn Faden 			continue;	/* no match */
5327cb8a054bSGlenn Faden 		output_auth(fp, &lookup);
5328cb8a054bSGlenn Faden 		output = B_TRUE;
5329cb8a054bSGlenn Faden 	}
5330cb8a054bSGlenn Faden 	(void) zonecfg_endadminent(handle);
5331cb8a054bSGlenn Faden 	/*
5332cb8a054bSGlenn Faden 	 * If a property n/v pair was specified, warn the user if there was
5333cb8a054bSGlenn Faden 	 * nothing to output.
5334cb8a054bSGlenn Faden 	 */
5335cb8a054bSGlenn Faden 	if (!output && cmd->cmd_prop_nv_pairs > 0)
5336cb8a054bSGlenn Faden 		(void) printf(gettext("No such %s resource.\n"),
5337cb8a054bSGlenn Faden 		    rt_to_str(RT_ADMIN));
5338cb8a054bSGlenn Faden }
5339cb8a054bSGlenn Faden 
53407c478bd9Sstevel@tonic-gate void
53417c478bd9Sstevel@tonic-gate info_func(cmd_t *cmd)
53427c478bd9Sstevel@tonic-gate {
53437c478bd9Sstevel@tonic-gate 	FILE *fp = stdout;
5344bbec428eSgjelinek 	boolean_t need_to_close = B_FALSE;
5345*12a8b243SSusan Kamm-Worrell 	char *pager, *space;
53460209230bSgjelinek 	int type;
53470209230bSgjelinek 	int res1, res2;
53480209230bSgjelinek 	uint64_t swap_limit;
53490209230bSgjelinek 	uint64_t locked_limit;
53503042b8b5Sbatschul 	struct stat statbuf;
53517c478bd9Sstevel@tonic-gate 
53527c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
53537c478bd9Sstevel@tonic-gate 
5354bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
53557c478bd9Sstevel@tonic-gate 		return;
53567c478bd9Sstevel@tonic-gate 
53577c478bd9Sstevel@tonic-gate 	/* don't page error output */
53587c478bd9Sstevel@tonic-gate 	if (interactive_mode) {
53597c478bd9Sstevel@tonic-gate 		if ((pager = getenv("PAGER")) == NULL)
53607c478bd9Sstevel@tonic-gate 			pager = PAGER;
5361*12a8b243SSusan Kamm-Worrell 		space = strchr(pager, ' ');
5362*12a8b243SSusan Kamm-Worrell 		if (space)
5363*12a8b243SSusan Kamm-Worrell 			*space = '\0';
53643042b8b5Sbatschul 		if (stat(pager, &statbuf) == 0) {
5365*12a8b243SSusan Kamm-Worrell 			if (space)
5366*12a8b243SSusan Kamm-Worrell 				*space = ' ';
53677c478bd9Sstevel@tonic-gate 			if ((fp = popen(pager, "w")) != NULL)
5368bbec428eSgjelinek 				need_to_close = B_TRUE;
53693042b8b5Sbatschul 			else
53703042b8b5Sbatschul 				fp = stdout;
53713042b8b5Sbatschul 		} else {
53723042b8b5Sbatschul 			zerr(gettext("PAGER %s does not exist (%s)."),
53733042b8b5Sbatschul 			    pager, strerror(errno));
53743042b8b5Sbatschul 		}
53753042b8b5Sbatschul 
53767c478bd9Sstevel@tonic-gate 		setbuf(fp, NULL);
53777c478bd9Sstevel@tonic-gate 	}
53787c478bd9Sstevel@tonic-gate 
53797c478bd9Sstevel@tonic-gate 	if (!global_scope) {
53807c478bd9Sstevel@tonic-gate 		switch (resource_scope) {
53817c478bd9Sstevel@tonic-gate 		case RT_FS:
53827c478bd9Sstevel@tonic-gate 			output_fs(fp, &in_progress_fstab);
53837c478bd9Sstevel@tonic-gate 			break;
53847c478bd9Sstevel@tonic-gate 		case RT_NET:
53857c478bd9Sstevel@tonic-gate 			output_net(fp, &in_progress_nwiftab);
53867c478bd9Sstevel@tonic-gate 			break;
53877c478bd9Sstevel@tonic-gate 		case RT_DEVICE:
53887c478bd9Sstevel@tonic-gate 			output_dev(fp, &in_progress_devtab);
53897c478bd9Sstevel@tonic-gate 			break;
53907c478bd9Sstevel@tonic-gate 		case RT_RCTL:
53917c478bd9Sstevel@tonic-gate 			output_rctl(fp, &in_progress_rctltab);
53927c478bd9Sstevel@tonic-gate 			break;
53937c478bd9Sstevel@tonic-gate 		case RT_ATTR:
53947c478bd9Sstevel@tonic-gate 			output_attr(fp, &in_progress_attrtab);
53957c478bd9Sstevel@tonic-gate 			break;
5396fa9e4066Sahrens 		case RT_DATASET:
5397fa9e4066Sahrens 			output_ds(fp, &in_progress_dstab);
5398fa9e4066Sahrens 			break;
53990209230bSgjelinek 		case RT_DCPU:
54000209230bSgjelinek 			output_pset(fp, &in_progress_psettab);
54010209230bSgjelinek 			break;
5402c97ad5cdSakolb 		case RT_PCAP:
5403c97ad5cdSakolb 			output_pcap(fp);
5404c97ad5cdSakolb 			break;
54050209230bSgjelinek 		case RT_MCAP:
54060209230bSgjelinek 			res1 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP,
54070209230bSgjelinek 			    &swap_limit);
54080209230bSgjelinek 			res2 = zonecfg_get_aliased_rctl(handle,
54090209230bSgjelinek 			    ALIAS_MAXLOCKEDMEM, &locked_limit);
54100209230bSgjelinek 			output_mcap(fp, &in_progress_mcaptab, res1, swap_limit,
54110209230bSgjelinek 			    res2, locked_limit);
54120209230bSgjelinek 			break;
5413cb8a054bSGlenn Faden 		case RT_ADMIN:
5414cb8a054bSGlenn Faden 			output_auth(fp, &in_progress_admintab);
5415cb8a054bSGlenn Faden 			break;
54167c478bd9Sstevel@tonic-gate 		}
54177c478bd9Sstevel@tonic-gate 		goto cleanup;
54187c478bd9Sstevel@tonic-gate 	}
54197c478bd9Sstevel@tonic-gate 
54200209230bSgjelinek 	type = cmd->cmd_res_type;
54210209230bSgjelinek 
54220209230bSgjelinek 	if (gz_invalid_rt_property(type)) {
54230209230bSgjelinek 		zerr(gettext("%s is not a valid property for the global zone."),
54240209230bSgjelinek 		    rt_to_str(type));
54250209230bSgjelinek 		goto cleanup;
54260209230bSgjelinek 	}
54270209230bSgjelinek 
54280209230bSgjelinek 	if (gz_invalid_resource(type)) {
54290209230bSgjelinek 		zerr(gettext("%s is not a valid resource for the global zone."),
54300209230bSgjelinek 		    rt_to_str(type));
54310209230bSgjelinek 		goto cleanup;
54320209230bSgjelinek 	}
54330209230bSgjelinek 
54347c478bd9Sstevel@tonic-gate 	switch (cmd->cmd_res_type) {
54357c478bd9Sstevel@tonic-gate 	case RT_UNKNOWN:
5436087719fdSdp 		info_zonename(handle, fp);
54370209230bSgjelinek 		if (!global_zone) {
54387c478bd9Sstevel@tonic-gate 			info_zonepath(handle, fp);
54399acbbeafSnn35248 			info_brand(handle, fp);
54407c478bd9Sstevel@tonic-gate 			info_autoboot(handle, fp);
54413f2f09c1Sdp 			info_bootargs(handle, fp);
54420209230bSgjelinek 		}
54437c478bd9Sstevel@tonic-gate 		info_pool(handle, fp);
54440209230bSgjelinek 		if (!global_zone) {
5445ffbafc53Scomay 			info_limitpriv(handle, fp);
54460209230bSgjelinek 			info_sched(handle, fp);
5447f4b3ec61Sdh155122 			info_iptype(handle, fp);
54485679c89fSjv227347 			info_hostid(handle, fp);
54490fbb751dSJohn Levon 			info_fs_allowed(handle, fp);
54500209230bSgjelinek 		}
54510209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXLWPS);
5452ff19e029SMenno Lageman 		info_aliased_rctl(handle, fp, ALIAS_MAXPROCS);
54530209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSHMMEM);
54540209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSHMIDS);
54550209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXMSGIDS);
54560209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSEMIDS);
54570209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_SHARES);
54580209230bSgjelinek 		if (!global_zone) {
54597c478bd9Sstevel@tonic-gate 			info_fs(handle, fp, cmd);
54607c478bd9Sstevel@tonic-gate 			info_net(handle, fp, cmd);
54617c478bd9Sstevel@tonic-gate 			info_dev(handle, fp, cmd);
54620209230bSgjelinek 		}
54630209230bSgjelinek 		info_pset(handle, fp);
5464c97ad5cdSakolb 		info_pcap(fp);
54650209230bSgjelinek 		info_mcap(handle, fp);
54660209230bSgjelinek 		if (!global_zone) {
54677c478bd9Sstevel@tonic-gate 			info_attr(handle, fp, cmd);
5468fa9e4066Sahrens 			info_ds(handle, fp, cmd);
5469cb8a054bSGlenn Faden 			info_auth(handle, fp, cmd);
54700209230bSgjelinek 		}
54710209230bSgjelinek 		info_rctl(handle, fp, cmd);
54727c478bd9Sstevel@tonic-gate 		break;
5473087719fdSdp 	case RT_ZONENAME:
5474087719fdSdp 		info_zonename(handle, fp);
5475087719fdSdp 		break;
54767c478bd9Sstevel@tonic-gate 	case RT_ZONEPATH:
54777c478bd9Sstevel@tonic-gate 		info_zonepath(handle, fp);
54787c478bd9Sstevel@tonic-gate 		break;
54799acbbeafSnn35248 	case RT_BRAND:
54809acbbeafSnn35248 		info_brand(handle, fp);
54819acbbeafSnn35248 		break;
54827c478bd9Sstevel@tonic-gate 	case RT_AUTOBOOT:
54837c478bd9Sstevel@tonic-gate 		info_autoboot(handle, fp);
54847c478bd9Sstevel@tonic-gate 		break;
54857c478bd9Sstevel@tonic-gate 	case RT_POOL:
54867c478bd9Sstevel@tonic-gate 		info_pool(handle, fp);
54877c478bd9Sstevel@tonic-gate 		break;
5488ffbafc53Scomay 	case RT_LIMITPRIV:
5489ffbafc53Scomay 		info_limitpriv(handle, fp);
5490ffbafc53Scomay 		break;
54913f2f09c1Sdp 	case RT_BOOTARGS:
54923f2f09c1Sdp 		info_bootargs(handle, fp);
54933f2f09c1Sdp 		break;
54940209230bSgjelinek 	case RT_SCHED:
54950209230bSgjelinek 		info_sched(handle, fp);
54960209230bSgjelinek 		break;
5497f4b3ec61Sdh155122 	case RT_IPTYPE:
5498f4b3ec61Sdh155122 		info_iptype(handle, fp);
5499f4b3ec61Sdh155122 		break;
55000209230bSgjelinek 	case RT_MAXLWPS:
55010209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXLWPS);
55020209230bSgjelinek 		break;
5503ff19e029SMenno Lageman 	case RT_MAXPROCS:
5504ff19e029SMenno Lageman 		info_aliased_rctl(handle, fp, ALIAS_MAXPROCS);
5505ff19e029SMenno Lageman 		break;
55060209230bSgjelinek 	case RT_MAXSHMMEM:
55070209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSHMMEM);
55080209230bSgjelinek 		break;
55090209230bSgjelinek 	case RT_MAXSHMIDS:
55100209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSHMIDS);
55110209230bSgjelinek 		break;
55120209230bSgjelinek 	case RT_MAXMSGIDS:
55130209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXMSGIDS);
55140209230bSgjelinek 		break;
55150209230bSgjelinek 	case RT_MAXSEMIDS:
55160209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSEMIDS);
55170209230bSgjelinek 		break;
55180209230bSgjelinek 	case RT_SHARES:
55190209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_SHARES);
55200209230bSgjelinek 		break;
55217c478bd9Sstevel@tonic-gate 	case RT_FS:
55227c478bd9Sstevel@tonic-gate 		info_fs(handle, fp, cmd);
55237c478bd9Sstevel@tonic-gate 		break;
55247c478bd9Sstevel@tonic-gate 	case RT_NET:
55257c478bd9Sstevel@tonic-gate 		info_net(handle, fp, cmd);
55267c478bd9Sstevel@tonic-gate 		break;
55277c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
55287c478bd9Sstevel@tonic-gate 		info_dev(handle, fp, cmd);
55297c478bd9Sstevel@tonic-gate 		break;
55307c478bd9Sstevel@tonic-gate 	case RT_RCTL:
55317c478bd9Sstevel@tonic-gate 		info_rctl(handle, fp, cmd);
55327c478bd9Sstevel@tonic-gate 		break;
55337c478bd9Sstevel@tonic-gate 	case RT_ATTR:
55347c478bd9Sstevel@tonic-gate 		info_attr(handle, fp, cmd);
55357c478bd9Sstevel@tonic-gate 		break;
5536fa9e4066Sahrens 	case RT_DATASET:
5537fa9e4066Sahrens 		info_ds(handle, fp, cmd);
5538fa9e4066Sahrens 		break;
55390209230bSgjelinek 	case RT_DCPU:
55400209230bSgjelinek 		info_pset(handle, fp);
55410209230bSgjelinek 		break;
5542c97ad5cdSakolb 	case RT_PCAP:
5543c97ad5cdSakolb 		info_pcap(fp);
5544c97ad5cdSakolb 		break;
55450209230bSgjelinek 	case RT_MCAP:
55460209230bSgjelinek 		info_mcap(handle, fp);
55470209230bSgjelinek 		break;
55485679c89fSjv227347 	case RT_HOSTID:
55495679c89fSjv227347 		info_hostid(handle, fp);
55505679c89fSjv227347 		break;
5551cb8a054bSGlenn Faden 	case RT_ADMIN:
5552cb8a054bSGlenn Faden 		info_auth(handle, fp, cmd);
5553cb8a054bSGlenn Faden 		break;
55540fbb751dSJohn Levon 	case RT_FS_ALLOWED:
55550fbb751dSJohn Levon 		info_fs_allowed(handle, fp);
55560fbb751dSJohn Levon 		break;
55577c478bd9Sstevel@tonic-gate 	default:
55587c478bd9Sstevel@tonic-gate 		zone_perror(rt_to_str(cmd->cmd_res_type), Z_NO_RESOURCE_TYPE,
5559bbec428eSgjelinek 		    B_TRUE);
55607c478bd9Sstevel@tonic-gate 	}
55617c478bd9Sstevel@tonic-gate 
55627c478bd9Sstevel@tonic-gate cleanup:
55637c478bd9Sstevel@tonic-gate 	if (need_to_close)
55647c478bd9Sstevel@tonic-gate 		(void) pclose(fp);
55657c478bd9Sstevel@tonic-gate }
55667c478bd9Sstevel@tonic-gate 
5567087719fdSdp /*
5568087719fdSdp  * Helper function for verify-- checks that a required string property
5569087719fdSdp  * exists.
5570087719fdSdp  */
5571087719fdSdp static void
5572087719fdSdp check_reqd_prop(char *attr, int rt, int pt, int *ret_val)
55737c478bd9Sstevel@tonic-gate {
5574087719fdSdp 	if (strlen(attr) == 0) {
5575087719fdSdp 		zerr(gettext("%s: %s not specified"), rt_to_str(rt),
5576087719fdSdp 		    pt_to_str(pt));
5577bbec428eSgjelinek 		saw_error = B_TRUE;
5578087719fdSdp 		if (*ret_val == Z_OK)
5579087719fdSdp 			*ret_val = Z_REQD_PROPERTY_MISSING;
55807c478bd9Sstevel@tonic-gate 	}
55817c478bd9Sstevel@tonic-gate }
55827c478bd9Sstevel@tonic-gate 
55839acbbeafSnn35248 static int
55849acbbeafSnn35248 do_subproc(char *cmdbuf)
55859acbbeafSnn35248 {
55869acbbeafSnn35248 	char inbuf[MAX_CMD_LEN];
55879acbbeafSnn35248 	FILE *file;
55889acbbeafSnn35248 	int status;
55899acbbeafSnn35248 
55909acbbeafSnn35248 	file = popen(cmdbuf, "r");
55919acbbeafSnn35248 	if (file == NULL) {
55929acbbeafSnn35248 		zerr(gettext("Could not launch: %s"), cmdbuf);
55939acbbeafSnn35248 		return (-1);
55949acbbeafSnn35248 	}
55959acbbeafSnn35248 
55969acbbeafSnn35248 	while (fgets(inbuf, sizeof (inbuf), file) != NULL)
55979acbbeafSnn35248 		fprintf(stderr, "%s", inbuf);
55989acbbeafSnn35248 	status = pclose(file);
55999acbbeafSnn35248 
56009acbbeafSnn35248 	if (WIFSIGNALED(status)) {
56019acbbeafSnn35248 		zerr(gettext("%s unexpectedly terminated due to signal %d"),
56029acbbeafSnn35248 		    cmdbuf, WTERMSIG(status));
56039acbbeafSnn35248 		return (-1);
56049acbbeafSnn35248 	}
56059acbbeafSnn35248 	assert(WIFEXITED(status));
56069acbbeafSnn35248 	return (WEXITSTATUS(status));
56079acbbeafSnn35248 }
56089acbbeafSnn35248 
56099acbbeafSnn35248 static int
56109acbbeafSnn35248 brand_verify(zone_dochandle_t handle)
56119acbbeafSnn35248 {
56126e65f9afSnn35248 	char xml_file[32];
56139acbbeafSnn35248 	char cmdbuf[MAX_CMD_LEN];
5614123807fbSedp 	brand_handle_t bh;
56159acbbeafSnn35248 	char brand[MAXNAMELEN];
56169acbbeafSnn35248 	int err;
56179acbbeafSnn35248 
56189acbbeafSnn35248 	if (zonecfg_get_brand(handle, brand, sizeof (brand)) != Z_OK) {
56199acbbeafSnn35248 		zerr("%s: %s\n", zone, gettext("could not get zone brand"));
56209acbbeafSnn35248 		return (Z_INVALID_DOCUMENT);
56219acbbeafSnn35248 	}
5622123807fbSedp 	if ((bh = brand_open(brand)) == NULL) {
56239acbbeafSnn35248 		zerr("%s: %s\n", zone, gettext("unknown brand."));
56249acbbeafSnn35248 		return (Z_INVALID_DOCUMENT);
56259acbbeafSnn35248 	}
56269acbbeafSnn35248 
56279acbbeafSnn35248 	/*
56289acbbeafSnn35248 	 * Fetch the verify command, if any, from the brand configuration
56299acbbeafSnn35248 	 * and build the command line to execute it.
56309acbbeafSnn35248 	 */
56319acbbeafSnn35248 	strcpy(cmdbuf, EXEC_PREFIX);
5632123807fbSedp 	err = brand_get_verify_cfg(bh, cmdbuf + EXEC_LEN,
56339acbbeafSnn35248 	    sizeof (cmdbuf) - (EXEC_LEN + (strlen(xml_file) + 1)));
5634123807fbSedp 	brand_close(bh);
56359acbbeafSnn35248 	if (err != Z_OK) {
56369acbbeafSnn35248 		zerr("%s: %s\n", zone,
56379acbbeafSnn35248 		    gettext("could not get brand verification command"));
56389acbbeafSnn35248 		return (Z_INVALID_DOCUMENT);
56399acbbeafSnn35248 	}
56409acbbeafSnn35248 
56419acbbeafSnn35248 	/*
56429acbbeafSnn35248 	 * If the brand doesn't provide a verification routine, we just
56439acbbeafSnn35248 	 * return success.
56449acbbeafSnn35248 	 */
56459acbbeafSnn35248 	if (strlen(cmdbuf) == EXEC_LEN)
56469acbbeafSnn35248 		return (Z_OK);
56479acbbeafSnn35248 
56489acbbeafSnn35248 	/*
56499acbbeafSnn35248 	 * Dump the current config information for this zone to a file.
56509acbbeafSnn35248 	 */
56516e65f9afSnn35248 	strcpy(xml_file, "/tmp/zonecfg_verify.XXXXXX");
56529acbbeafSnn35248 	if (mkstemp(xml_file) == NULL)
56539acbbeafSnn35248 		return (Z_TEMP_FILE);
56549acbbeafSnn35248 	if ((err = zonecfg_verify_save(handle, xml_file)) != Z_OK) {
56559acbbeafSnn35248 		(void) unlink(xml_file);
56569acbbeafSnn35248 		return (err);
56579acbbeafSnn35248 	}
56589acbbeafSnn35248 
56599acbbeafSnn35248 	/*
56609acbbeafSnn35248 	 * Execute the verification command.
56619acbbeafSnn35248 	 */
56629acbbeafSnn35248 	if ((strlcat(cmdbuf, " ", MAX_CMD_LEN) >= MAX_CMD_LEN) ||
56639acbbeafSnn35248 	    (strlcat(cmdbuf, xml_file, MAX_CMD_LEN) >= MAX_CMD_LEN)) {
56649acbbeafSnn35248 		err = Z_BRAND_ERROR;
56659acbbeafSnn35248 	} else {
56669acbbeafSnn35248 		err = do_subproc(cmdbuf);
56679acbbeafSnn35248 	}
56689acbbeafSnn35248 
56699acbbeafSnn35248 	(void) unlink(xml_file);
56709acbbeafSnn35248 	return ((err == Z_OK) ? Z_OK : Z_BRAND_ERROR);
56719acbbeafSnn35248 }
56729acbbeafSnn35248 
56737c478bd9Sstevel@tonic-gate /*
5674550b6e40SSowmini Varadhan  * Track the network interfaces listed in zonecfg(1m) in a linked list
5675550b6e40SSowmini Varadhan  * so that we can later check that defrouter is specified for an exclusive IP
5676550b6e40SSowmini Varadhan  * zone if and only if at least one allowed-address has been specified.
5677550b6e40SSowmini Varadhan  */
5678550b6e40SSowmini Varadhan static boolean_t
5679550b6e40SSowmini Varadhan add_nwif(struct zone_nwiftab *nwif)
5680550b6e40SSowmini Varadhan {
5681550b6e40SSowmini Varadhan 	struct xif *tmp;
5682550b6e40SSowmini Varadhan 
5683550b6e40SSowmini Varadhan 	for (tmp = xif; tmp != NULL; tmp = tmp->xif_next) {
5684550b6e40SSowmini Varadhan 		if (strcmp(tmp->xif_name, nwif->zone_nwif_physical) == 0) {
5685550b6e40SSowmini Varadhan 			if (strlen(nwif->zone_nwif_allowed_address) > 0)
5686550b6e40SSowmini Varadhan 				tmp->xif_has_address = B_TRUE;
5687550b6e40SSowmini Varadhan 			if (strlen(nwif->zone_nwif_defrouter) > 0)
5688550b6e40SSowmini Varadhan 				tmp->xif_has_defrouter = B_TRUE;
5689550b6e40SSowmini Varadhan 			return (B_TRUE);
5690550b6e40SSowmini Varadhan 		}
5691550b6e40SSowmini Varadhan 	}
5692550b6e40SSowmini Varadhan 
5693550b6e40SSowmini Varadhan 	tmp = malloc(sizeof (*tmp));
5694550b6e40SSowmini Varadhan 	if (tmp == NULL) {
5695550b6e40SSowmini Varadhan 		zerr(gettext("memory allocation failed for %s"),
5696550b6e40SSowmini Varadhan 		    nwif->zone_nwif_physical);
5697550b6e40SSowmini Varadhan 		return (B_FALSE);
5698550b6e40SSowmini Varadhan 	}
5699550b6e40SSowmini Varadhan 	strlcpy(tmp->xif_name, nwif->zone_nwif_physical,
5700550b6e40SSowmini Varadhan 	    sizeof (tmp->xif_name));
5701550b6e40SSowmini Varadhan 	tmp->xif_has_defrouter = (strlen(nwif->zone_nwif_defrouter) > 0);
5702550b6e40SSowmini Varadhan 	tmp->xif_has_address = (strlen(nwif->zone_nwif_allowed_address) > 0);
5703550b6e40SSowmini Varadhan 	tmp->xif_next = xif;
5704550b6e40SSowmini Varadhan 	xif = tmp;
5705550b6e40SSowmini Varadhan 	return (B_TRUE);
5706550b6e40SSowmini Varadhan }
5707550b6e40SSowmini Varadhan 
5708550b6e40SSowmini Varadhan /*
57097c478bd9Sstevel@tonic-gate  * See the DTD for which attributes are required for which resources.
57107c478bd9Sstevel@tonic-gate  *
57117c478bd9Sstevel@tonic-gate  * This function can be called by commit_func(), which needs to save things,
57127c478bd9Sstevel@tonic-gate  * in addition to the general call from parse_and_run(), which doesn't need
57137c478bd9Sstevel@tonic-gate  * things saved.  Since the parameters are standardized, we distinguish by
57147c478bd9Sstevel@tonic-gate  * having commit_func() call here with cmd->cmd_arg set to "save" to indicate
57157c478bd9Sstevel@tonic-gate  * that a save is needed.
57167c478bd9Sstevel@tonic-gate  */
57177c478bd9Sstevel@tonic-gate void
57187c478bd9Sstevel@tonic-gate verify_func(cmd_t *cmd)
57197c478bd9Sstevel@tonic-gate {
57207c478bd9Sstevel@tonic-gate 	struct zone_nwiftab nwiftab;
57217c478bd9Sstevel@tonic-gate 	struct zone_fstab fstab;
57227c478bd9Sstevel@tonic-gate 	struct zone_attrtab attrtab;
57237c478bd9Sstevel@tonic-gate 	struct zone_rctltab rctltab;
5724fa9e4066Sahrens 	struct zone_dstab dstab;
57250209230bSgjelinek 	struct zone_psettab psettab;
5726cb8a054bSGlenn Faden 	struct zone_admintab admintab;
57277c478bd9Sstevel@tonic-gate 	char zonepath[MAXPATHLEN];
57280209230bSgjelinek 	char sched[MAXNAMELEN];
57299acbbeafSnn35248 	char brand[MAXNAMELEN];
57305679c89fSjv227347 	char hostidp[HW_HOSTID_LEN];
57310fbb751dSJohn Levon 	char fsallowedp[ZONE_FS_ALLOWED_MAX];
57327c478bd9Sstevel@tonic-gate 	int err, ret_val = Z_OK, arg;
5733c97ad5cdSakolb 	int pset_res;
5734bbec428eSgjelinek 	boolean_t save = B_FALSE;
5735bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
5736f4b3ec61Sdh155122 	zone_iptype_t iptype;
57370209230bSgjelinek 	boolean_t has_cpu_shares = B_FALSE;
5738c97ad5cdSakolb 	boolean_t has_cpu_cap = B_FALSE;
5739550b6e40SSowmini Varadhan 	struct xif *tmp;
57407c478bd9Sstevel@tonic-gate 
57417c478bd9Sstevel@tonic-gate 	optind = 0;
57427ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
57437c478bd9Sstevel@tonic-gate 		switch (arg) {
57447c478bd9Sstevel@tonic-gate 		case '?':
57457c478bd9Sstevel@tonic-gate 			longer_usage(CMD_VERIFY);
5746bbec428eSgjelinek 			arg_err = B_TRUE;
57477ec75eb8Sgjelinek 			break;
57487c478bd9Sstevel@tonic-gate 		default:
57497c478bd9Sstevel@tonic-gate 			short_usage(CMD_VERIFY);
5750bbec428eSgjelinek 			arg_err = B_TRUE;
57517ec75eb8Sgjelinek 			break;
57527ec75eb8Sgjelinek 		}
57537ec75eb8Sgjelinek 	}
57547ec75eb8Sgjelinek 	if (arg_err)
57557c478bd9Sstevel@tonic-gate 		return;
57567ec75eb8Sgjelinek 
57577c478bd9Sstevel@tonic-gate 	if (optind > cmd->cmd_argc) {
57587c478bd9Sstevel@tonic-gate 		short_usage(CMD_VERIFY);
57597c478bd9Sstevel@tonic-gate 		return;
57607c478bd9Sstevel@tonic-gate 	}
57617c478bd9Sstevel@tonic-gate 
57627c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_VERIFY))
57637c478bd9Sstevel@tonic-gate 		return;
57647c478bd9Sstevel@tonic-gate 
57657c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
57667c478bd9Sstevel@tonic-gate 
57677c478bd9Sstevel@tonic-gate 	if (cmd->cmd_argc > 0 && (strcmp(cmd->cmd_argv[0], "save") == 0))
5768bbec428eSgjelinek 		save = B_TRUE;
5769bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
57707c478bd9Sstevel@tonic-gate 		return;
57717c478bd9Sstevel@tonic-gate 
57720209230bSgjelinek 	if (zonecfg_get_zonepath(handle, zonepath, sizeof (zonepath)) != Z_OK &&
57730209230bSgjelinek 	    !global_zone) {
5774087719fdSdp 		zerr(gettext("%s not specified"), pt_to_str(PT_ZONEPATH));
57757c478bd9Sstevel@tonic-gate 		ret_val = Z_REQD_RESOURCE_MISSING;
5776bbec428eSgjelinek 		saw_error = B_TRUE;
57777c478bd9Sstevel@tonic-gate 	}
57780209230bSgjelinek 	if (strlen(zonepath) == 0 && !global_zone) {
5779087719fdSdp 		zerr(gettext("%s cannot be empty."), pt_to_str(PT_ZONEPATH));
57807c478bd9Sstevel@tonic-gate 		ret_val = Z_REQD_RESOURCE_MISSING;
5781bbec428eSgjelinek 		saw_error = B_TRUE;
57827c478bd9Sstevel@tonic-gate 	}
57837c478bd9Sstevel@tonic-gate 
57849acbbeafSnn35248 	if ((err = zonecfg_get_brand(handle, brand, sizeof (brand))) != Z_OK) {
5785bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
57869acbbeafSnn35248 		return;
57879acbbeafSnn35248 	}
57889acbbeafSnn35248 	if ((err = brand_verify(handle)) != Z_OK) {
5789bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
57909acbbeafSnn35248 		return;
57919acbbeafSnn35248 	}
57929acbbeafSnn35248 
5793f4b3ec61Sdh155122 	if (zonecfg_get_iptype(handle, &iptype) != Z_OK) {
5794f4b3ec61Sdh155122 		zerr("%s %s", gettext("cannot get"), pt_to_str(PT_IPTYPE));
5795f4b3ec61Sdh155122 		ret_val = Z_REQD_RESOURCE_MISSING;
5796bbec428eSgjelinek 		saw_error = B_TRUE;
5797f4b3ec61Sdh155122 	}
57987c478bd9Sstevel@tonic-gate 
57990fbb751dSJohn Levon 	if (zonecfg_get_hostid(handle, hostidp,
58000fbb751dSJohn Levon 	    sizeof (hostidp)) == Z_INVALID_PROPERTY) {
58010fbb751dSJohn Levon 		zerr(gettext("%s: invalid hostid: %s"),
58020fbb751dSJohn Levon 		    zone, hostidp);
58030fbb751dSJohn Levon 		return;
58040fbb751dSJohn Levon 	}
58050fbb751dSJohn Levon 
58060fbb751dSJohn Levon 	if (zonecfg_get_fs_allowed(handle, fsallowedp,
58070fbb751dSJohn Levon 	    sizeof (fsallowedp)) == Z_INVALID_PROPERTY) {
58080fbb751dSJohn Levon 		zerr(gettext("%s: invalid fs-allowed: %s"),
58090fbb751dSJohn Levon 		    zone, fsallowedp);
58105679c89fSjv227347 		return;
58115679c89fSjv227347 	}
58125679c89fSjv227347 
58137c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setfsent(handle)) != Z_OK) {
5814bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
58157c478bd9Sstevel@tonic-gate 		return;
58167c478bd9Sstevel@tonic-gate 	}
58177c478bd9Sstevel@tonic-gate 	while (zonecfg_getfsent(handle, &fstab) == Z_OK) {
5818087719fdSdp 		check_reqd_prop(fstab.zone_fs_dir, RT_FS, PT_DIR, &ret_val);
5819087719fdSdp 		check_reqd_prop(fstab.zone_fs_special, RT_FS, PT_SPECIAL,
5820087719fdSdp 		    &ret_val);
5821087719fdSdp 		check_reqd_prop(fstab.zone_fs_type, RT_FS, PT_TYPE, &ret_val);
5822087719fdSdp 
58237c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(fstab.zone_fs_options);
58247c478bd9Sstevel@tonic-gate 	}
58257c478bd9Sstevel@tonic-gate 	(void) zonecfg_endfsent(handle);
58267c478bd9Sstevel@tonic-gate 
58277c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setnwifent(handle)) != Z_OK) {
5828bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
58297c478bd9Sstevel@tonic-gate 		return;
58307c478bd9Sstevel@tonic-gate 	}
58317c478bd9Sstevel@tonic-gate 	while (zonecfg_getnwifent(handle, &nwiftab) == Z_OK) {
5832f4b3ec61Sdh155122 		/*
5833f4b3ec61Sdh155122 		 * physical is required in all cases.
5834de860bd9Sgfaden 		 * A shared IP requires an address,
5835de860bd9Sgfaden 		 * and may include a default router, while
5836de860bd9Sgfaden 		 * an exclusive IP must have neither an address
5837de860bd9Sgfaden 		 * nor a default router.
583801b4bc23Sjv227347 		 * The physical interface name must be valid in all cases.
5839f4b3ec61Sdh155122 		 */
5840087719fdSdp 		check_reqd_prop(nwiftab.zone_nwif_physical, RT_NET,
5841087719fdSdp 		    PT_PHYSICAL, &ret_val);
584201b4bc23Sjv227347 		if (validate_net_physical_syntax(nwiftab.zone_nwif_physical) !=
584301b4bc23Sjv227347 		    Z_OK) {
584401b4bc23Sjv227347 			saw_error = B_TRUE;
584501b4bc23Sjv227347 			if (ret_val == Z_OK)
584601b4bc23Sjv227347 				ret_val = Z_INVAL;
584701b4bc23Sjv227347 		}
5848f4b3ec61Sdh155122 
5849f4b3ec61Sdh155122 		switch (iptype) {
5850f4b3ec61Sdh155122 		case ZS_SHARED:
5851f4b3ec61Sdh155122 			check_reqd_prop(nwiftab.zone_nwif_address, RT_NET,
5852f4b3ec61Sdh155122 			    PT_ADDRESS, &ret_val);
5853550b6e40SSowmini Varadhan 			if (strlen(nwiftab.zone_nwif_allowed_address) > 0) {
5854550b6e40SSowmini Varadhan 				zerr(gettext("%s: %s cannot be specified "
5855550b6e40SSowmini Varadhan 				    "for a shared IP type"),
5856550b6e40SSowmini Varadhan 				    rt_to_str(RT_NET),
5857550b6e40SSowmini Varadhan 				    pt_to_str(PT_ALLOWED_ADDRESS));
5858550b6e40SSowmini Varadhan 				saw_error = B_TRUE;
5859550b6e40SSowmini Varadhan 				if (ret_val == Z_OK)
5860550b6e40SSowmini Varadhan 					ret_val = Z_INVAL;
5861550b6e40SSowmini Varadhan 			}
5862f4b3ec61Sdh155122 			break;
5863f4b3ec61Sdh155122 		case ZS_EXCLUSIVE:
5864f4b3ec61Sdh155122 			if (strlen(nwiftab.zone_nwif_address) > 0) {
5865f4b3ec61Sdh155122 				zerr(gettext("%s: %s cannot be specified "
5866f4b3ec61Sdh155122 				    "for an exclusive IP type"),
5867f4b3ec61Sdh155122 				    rt_to_str(RT_NET), pt_to_str(PT_ADDRESS));
5868bbec428eSgjelinek 				saw_error = B_TRUE;
5869f4b3ec61Sdh155122 				if (ret_val == Z_OK)
5870f4b3ec61Sdh155122 					ret_val = Z_INVAL;
5871550b6e40SSowmini Varadhan 			} else {
5872550b6e40SSowmini Varadhan 				if (!add_nwif(&nwiftab)) {
5873bbec428eSgjelinek 					saw_error = B_TRUE;
5874de860bd9Sgfaden 					if (ret_val == Z_OK)
5875de860bd9Sgfaden 						ret_val = Z_INVAL;
5876de860bd9Sgfaden 				}
5877550b6e40SSowmini Varadhan 			}
5878f4b3ec61Sdh155122 			break;
5879f4b3ec61Sdh155122 		}
58807c478bd9Sstevel@tonic-gate 	}
5881550b6e40SSowmini Varadhan 	for (tmp = xif; tmp != NULL; tmp = tmp->xif_next) {
5882550b6e40SSowmini Varadhan 		if (!tmp->xif_has_address && tmp->xif_has_defrouter) {
5883550b6e40SSowmini Varadhan 			zerr(gettext("%s: %s for %s cannot be specified "
5884550b6e40SSowmini Varadhan 			    "without %s for an exclusive IP type"),
5885550b6e40SSowmini Varadhan 			    rt_to_str(RT_NET), pt_to_str(PT_DEFROUTER),
5886550b6e40SSowmini Varadhan 			    tmp->xif_name, pt_to_str(PT_ALLOWED_ADDRESS));
5887550b6e40SSowmini Varadhan 			saw_error = B_TRUE;
5888550b6e40SSowmini Varadhan 			ret_val = Z_INVAL;
5889550b6e40SSowmini Varadhan 		}
5890550b6e40SSowmini Varadhan 	}
5891550b6e40SSowmini Varadhan 	free(xif);
5892550b6e40SSowmini Varadhan 	xif = NULL;
58937c478bd9Sstevel@tonic-gate 	(void) zonecfg_endnwifent(handle);
58947c478bd9Sstevel@tonic-gate 
58957c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setrctlent(handle)) != Z_OK) {
5896bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
58977c478bd9Sstevel@tonic-gate 		return;
58987c478bd9Sstevel@tonic-gate 	}
58997c478bd9Sstevel@tonic-gate 	while (zonecfg_getrctlent(handle, &rctltab) == Z_OK) {
5900087719fdSdp 		check_reqd_prop(rctltab.zone_rctl_name, RT_RCTL, PT_NAME,
5901087719fdSdp 		    &ret_val);
5902087719fdSdp 
59030209230bSgjelinek 		if (strcmp(rctltab.zone_rctl_name, "zone.cpu-shares") == 0)
59040209230bSgjelinek 			has_cpu_shares = B_TRUE;
59050209230bSgjelinek 
5906c97ad5cdSakolb 		if (strcmp(rctltab.zone_rctl_name, "zone.cpu-cap") == 0)
5907c97ad5cdSakolb 			has_cpu_cap = B_TRUE;
5908c97ad5cdSakolb 
59097c478bd9Sstevel@tonic-gate 		if (rctltab.zone_rctl_valptr == NULL) {
59107c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: no %s specified"),
59117c478bd9Sstevel@tonic-gate 			    rt_to_str(RT_RCTL), pt_to_str(PT_VALUE));
5912bbec428eSgjelinek 			saw_error = B_TRUE;
59137c478bd9Sstevel@tonic-gate 			if (ret_val == Z_OK)
59147c478bd9Sstevel@tonic-gate 				ret_val = Z_REQD_PROPERTY_MISSING;
59157c478bd9Sstevel@tonic-gate 		} else {
59167c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(rctltab.zone_rctl_valptr);
59177c478bd9Sstevel@tonic-gate 		}
59187c478bd9Sstevel@tonic-gate 	}
59197c478bd9Sstevel@tonic-gate 	(void) zonecfg_endrctlent(handle);
59207c478bd9Sstevel@tonic-gate 
5921c97ad5cdSakolb 	if ((pset_res = zonecfg_lookup_pset(handle, &psettab)) == Z_OK &&
5922c97ad5cdSakolb 	    has_cpu_shares) {
59230209230bSgjelinek 		zerr(gettext("%s zone.cpu-shares and %s are incompatible."),
59240209230bSgjelinek 		    rt_to_str(RT_RCTL), rt_to_str(RT_DCPU));
5925bbec428eSgjelinek 		saw_error = B_TRUE;
59260209230bSgjelinek 		if (ret_val == Z_OK)
59270209230bSgjelinek 			ret_val = Z_INCOMPATIBLE;
59280209230bSgjelinek 	}
59290209230bSgjelinek 
59300209230bSgjelinek 	if (has_cpu_shares && zonecfg_get_sched_class(handle, sched,
59310209230bSgjelinek 	    sizeof (sched)) == Z_OK && strlen(sched) > 0 &&
59320209230bSgjelinek 	    strcmp(sched, "FSS") != 0) {
59330209230bSgjelinek 		zerr(gettext("WARNING: %s zone.cpu-shares and %s=%s are "
59340209230bSgjelinek 		    "incompatible"),
59350209230bSgjelinek 		    rt_to_str(RT_RCTL), rt_to_str(RT_SCHED), sched);
5936bbec428eSgjelinek 		saw_error = B_TRUE;
59370209230bSgjelinek 		if (ret_val == Z_OK)
59380209230bSgjelinek 			ret_val = Z_INCOMPATIBLE;
59390209230bSgjelinek 	}
59400209230bSgjelinek 
5941c97ad5cdSakolb 	if (pset_res == Z_OK && has_cpu_cap) {
5942c97ad5cdSakolb 		zerr(gettext("%s zone.cpu-cap and the %s are incompatible."),
5943c97ad5cdSakolb 		    rt_to_str(RT_RCTL), rt_to_str(RT_DCPU));
5944bbec428eSgjelinek 		saw_error = B_TRUE;
5945c97ad5cdSakolb 		if (ret_val == Z_OK)
5946c97ad5cdSakolb 			ret_val = Z_INCOMPATIBLE;
5947c97ad5cdSakolb 	}
5948c97ad5cdSakolb 
59497c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setattrent(handle)) != Z_OK) {
5950bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
59517c478bd9Sstevel@tonic-gate 		return;
59527c478bd9Sstevel@tonic-gate 	}
59537c478bd9Sstevel@tonic-gate 	while (zonecfg_getattrent(handle, &attrtab) == Z_OK) {
5954087719fdSdp 		check_reqd_prop(attrtab.zone_attr_name, RT_ATTR, PT_NAME,
5955087719fdSdp 		    &ret_val);
5956087719fdSdp 		check_reqd_prop(attrtab.zone_attr_type, RT_ATTR, PT_TYPE,
5957087719fdSdp 		    &ret_val);
5958087719fdSdp 		check_reqd_prop(attrtab.zone_attr_value, RT_ATTR, PT_VALUE,
5959087719fdSdp 		    &ret_val);
59607c478bd9Sstevel@tonic-gate 	}
59617c478bd9Sstevel@tonic-gate 	(void) zonecfg_endattrent(handle);
59627c478bd9Sstevel@tonic-gate 
5963fa9e4066Sahrens 	if ((err = zonecfg_setdsent(handle)) != Z_OK) {
5964bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
5965fa9e4066Sahrens 		return;
5966fa9e4066Sahrens 	}
5967fa9e4066Sahrens 	while (zonecfg_getdsent(handle, &dstab) == Z_OK) {
5968fa9e4066Sahrens 		if (strlen(dstab.zone_dataset_name) == 0) {
5969fa9e4066Sahrens 			zerr("%s: %s %s", rt_to_str(RT_DATASET),
5970fa9e4066Sahrens 			    pt_to_str(PT_NAME), gettext("not specified"));
5971bbec428eSgjelinek 			saw_error = B_TRUE;
5972fa9e4066Sahrens 			if (ret_val == Z_OK)
5973fa9e4066Sahrens 				ret_val = Z_REQD_PROPERTY_MISSING;
5974fa9e4066Sahrens 		} else if (!zfs_name_valid(dstab.zone_dataset_name,
5975fa9e4066Sahrens 		    ZFS_TYPE_FILESYSTEM)) {
5976fa9e4066Sahrens 			zerr("%s: %s %s", rt_to_str(RT_DATASET),
5977fa9e4066Sahrens 			    pt_to_str(PT_NAME), gettext("invalid"));
5978bbec428eSgjelinek 			saw_error = B_TRUE;
5979fa9e4066Sahrens 			if (ret_val == Z_OK)
5980fa9e4066Sahrens 				ret_val = Z_BAD_PROPERTY;
5981fa9e4066Sahrens 		}
5982fa9e4066Sahrens 
5983fa9e4066Sahrens 	}
5984fa9e4066Sahrens 	(void) zonecfg_enddsent(handle);
5985fa9e4066Sahrens 
5986cb8a054bSGlenn Faden 	if ((err = zonecfg_setadminent(handle)) != Z_OK) {
5987cb8a054bSGlenn Faden 		zone_perror(zone, err, B_TRUE);
5988cb8a054bSGlenn Faden 		return;
5989cb8a054bSGlenn Faden 	}
5990cb8a054bSGlenn Faden 	while (zonecfg_getadminent(handle, &admintab) == Z_OK) {
5991cb8a054bSGlenn Faden 		check_reqd_prop(admintab.zone_admin_user, RT_ADMIN,
5992cb8a054bSGlenn Faden 		    PT_USER, &ret_val);
5993cb8a054bSGlenn Faden 		check_reqd_prop(admintab.zone_admin_auths, RT_ADMIN,
5994cb8a054bSGlenn Faden 		    PT_AUTHS, &ret_val);
5995cb8a054bSGlenn Faden 		if ((ret_val == Z_OK) && (getpwnam(admintab.zone_admin_user)
5996cb8a054bSGlenn Faden 		    == NULL)) {
5997cb8a054bSGlenn Faden 			zerr(gettext("%s %s is not a valid username"),
5998cb8a054bSGlenn Faden 			    pt_to_str(PT_USER),
5999cb8a054bSGlenn Faden 			    admintab.zone_admin_user);
6000cb8a054bSGlenn Faden 			ret_val = Z_BAD_PROPERTY;
6001cb8a054bSGlenn Faden 		}
6002cb8a054bSGlenn Faden 		if ((ret_val == Z_OK) && (!zonecfg_valid_auths(
6003cb8a054bSGlenn Faden 		    admintab.zone_admin_auths, zone))) {
6004cb8a054bSGlenn Faden 			ret_val = Z_BAD_PROPERTY;
6005cb8a054bSGlenn Faden 		}
6006cb8a054bSGlenn Faden 	}
6007cb8a054bSGlenn Faden 	(void) zonecfg_endadminent(handle);
6008cb8a054bSGlenn Faden 
60097c478bd9Sstevel@tonic-gate 	if (!global_scope) {
60107c478bd9Sstevel@tonic-gate 		zerr(gettext("resource specification incomplete"));
6011bbec428eSgjelinek 		saw_error = B_TRUE;
60127c478bd9Sstevel@tonic-gate 		if (ret_val == Z_OK)
60137c478bd9Sstevel@tonic-gate 			ret_val = Z_INSUFFICIENT_SPEC;
60147c478bd9Sstevel@tonic-gate 	}
60157c478bd9Sstevel@tonic-gate 
60167c478bd9Sstevel@tonic-gate 	if (save) {
6017087719fdSdp 		if (ret_val == Z_OK) {
6018087719fdSdp 			if ((ret_val = zonecfg_save(handle)) == Z_OK) {
6019bbec428eSgjelinek 				need_to_commit = B_FALSE;
6020087719fdSdp 				(void) strlcpy(revert_zone, zone,
6021087719fdSdp 				    sizeof (revert_zone));
6022087719fdSdp 			}
6023087719fdSdp 		} else {
6024087719fdSdp 			zerr(gettext("Zone %s failed to verify"), zone);
6025087719fdSdp 		}
60267c478bd9Sstevel@tonic-gate 	}
60277c478bd9Sstevel@tonic-gate 	if (ret_val != Z_OK)
6028bbec428eSgjelinek 		zone_perror(zone, ret_val, B_TRUE);
60297c478bd9Sstevel@tonic-gate }
60307c478bd9Sstevel@tonic-gate 
60317c478bd9Sstevel@tonic-gate void
60327c478bd9Sstevel@tonic-gate cancel_func(cmd_t *cmd)
60337c478bd9Sstevel@tonic-gate {
60347c478bd9Sstevel@tonic-gate 	int arg;
6035bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
60367c478bd9Sstevel@tonic-gate 
60377c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
60387c478bd9Sstevel@tonic-gate 
60397c478bd9Sstevel@tonic-gate 	optind = 0;
60407ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
60417c478bd9Sstevel@tonic-gate 		switch (arg) {
60427c478bd9Sstevel@tonic-gate 		case '?':
60437c478bd9Sstevel@tonic-gate 			longer_usage(CMD_CANCEL);
6044bbec428eSgjelinek 			arg_err = B_TRUE;
60457ec75eb8Sgjelinek 			break;
60467c478bd9Sstevel@tonic-gate 		default:
60477c478bd9Sstevel@tonic-gate 			short_usage(CMD_CANCEL);
6048bbec428eSgjelinek 			arg_err = B_TRUE;
60497ec75eb8Sgjelinek 			break;
60507ec75eb8Sgjelinek 		}
60517ec75eb8Sgjelinek 	}
60527ec75eb8Sgjelinek 	if (arg_err)
60537c478bd9Sstevel@tonic-gate 		return;
60547ec75eb8Sgjelinek 
60557c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
60567c478bd9Sstevel@tonic-gate 		short_usage(CMD_CANCEL);
60577c478bd9Sstevel@tonic-gate 		return;
60587c478bd9Sstevel@tonic-gate 	}
60597c478bd9Sstevel@tonic-gate 
60607c478bd9Sstevel@tonic-gate 	if (global_scope)
60617c478bd9Sstevel@tonic-gate 		scope_usage(CMD_CANCEL);
6062bbec428eSgjelinek 	global_scope = B_TRUE;
60637c478bd9Sstevel@tonic-gate 	zonecfg_free_fs_option_list(in_progress_fstab.zone_fs_options);
60647c478bd9Sstevel@tonic-gate 	bzero(&in_progress_fstab, sizeof (in_progress_fstab));
60657c478bd9Sstevel@tonic-gate 	bzero(&in_progress_nwiftab, sizeof (in_progress_nwiftab));
60667c478bd9Sstevel@tonic-gate 	bzero(&in_progress_devtab, sizeof (in_progress_devtab));
60677c478bd9Sstevel@tonic-gate 	zonecfg_free_rctl_value_list(in_progress_rctltab.zone_rctl_valptr);
60687c478bd9Sstevel@tonic-gate 	bzero(&in_progress_rctltab, sizeof (in_progress_rctltab));
60697c478bd9Sstevel@tonic-gate 	bzero(&in_progress_attrtab, sizeof (in_progress_attrtab));
6070fa9e4066Sahrens 	bzero(&in_progress_dstab, sizeof (in_progress_dstab));
60717c478bd9Sstevel@tonic-gate }
60727c478bd9Sstevel@tonic-gate 
60737c478bd9Sstevel@tonic-gate static int
60747c478bd9Sstevel@tonic-gate validate_attr_name(char *name)
60757c478bd9Sstevel@tonic-gate {
60767c478bd9Sstevel@tonic-gate 	int i;
60777c478bd9Sstevel@tonic-gate 
60787c478bd9Sstevel@tonic-gate 	if (!isalnum(name[0])) {
60797c478bd9Sstevel@tonic-gate 		zerr(gettext("Invalid %s %s %s: must start with an alpha-"
60807c478bd9Sstevel@tonic-gate 		    "numeric character."), rt_to_str(RT_ATTR),
60817c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_NAME), name);
60827c478bd9Sstevel@tonic-gate 		return (Z_INVAL);
60837c478bd9Sstevel@tonic-gate 	}
60847c478bd9Sstevel@tonic-gate 	for (i = 1; name[i]; i++)
60857c478bd9Sstevel@tonic-gate 		if (!isalnum(name[i]) && name[i] != '-' && name[i] != '.') {
60867c478bd9Sstevel@tonic-gate 			zerr(gettext("Invalid %s %s %s: can only contain "
60877c478bd9Sstevel@tonic-gate 			    "alpha-numeric characters, plus '-' and '.'."),
60887c478bd9Sstevel@tonic-gate 			    rt_to_str(RT_ATTR), pt_to_str(PT_NAME), name);
60897c478bd9Sstevel@tonic-gate 			return (Z_INVAL);
60907c478bd9Sstevel@tonic-gate 		}
60917c478bd9Sstevel@tonic-gate 	return (Z_OK);
60927c478bd9Sstevel@tonic-gate }
60937c478bd9Sstevel@tonic-gate 
60947c478bd9Sstevel@tonic-gate static int
60957c478bd9Sstevel@tonic-gate validate_attr_type_val(struct zone_attrtab *attrtab)
60967c478bd9Sstevel@tonic-gate {
60977c478bd9Sstevel@tonic-gate 	boolean_t boolval;
60987c478bd9Sstevel@tonic-gate 	int64_t intval;
60997c478bd9Sstevel@tonic-gate 	char strval[MAXNAMELEN];
61007c478bd9Sstevel@tonic-gate 	uint64_t uintval;
61017c478bd9Sstevel@tonic-gate 
61027c478bd9Sstevel@tonic-gate 	if (strcmp(attrtab->zone_attr_type, "boolean") == 0) {
61037c478bd9Sstevel@tonic-gate 		if (zonecfg_get_attr_boolean(attrtab, &boolval) == Z_OK)
61047c478bd9Sstevel@tonic-gate 			return (Z_OK);
61057c478bd9Sstevel@tonic-gate 		zerr(gettext("invalid %s value for %s=%s"),
61067c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR), pt_to_str(PT_TYPE), "boolean");
61077c478bd9Sstevel@tonic-gate 		return (Z_ERR);
61087c478bd9Sstevel@tonic-gate 	}
61097c478bd9Sstevel@tonic-gate 
61107c478bd9Sstevel@tonic-gate 	if (strcmp(attrtab->zone_attr_type, "int") == 0) {
61117c478bd9Sstevel@tonic-gate 		if (zonecfg_get_attr_int(attrtab, &intval) == Z_OK)
61127c478bd9Sstevel@tonic-gate 			return (Z_OK);
61137c478bd9Sstevel@tonic-gate 		zerr(gettext("invalid %s value for %s=%s"),
61147c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR), pt_to_str(PT_TYPE), "int");
61157c478bd9Sstevel@tonic-gate 		return (Z_ERR);
61167c478bd9Sstevel@tonic-gate 	}
61177c478bd9Sstevel@tonic-gate 
61187c478bd9Sstevel@tonic-gate 	if (strcmp(attrtab->zone_attr_type, "string") == 0) {
61197c478bd9Sstevel@tonic-gate 		if (zonecfg_get_attr_string(attrtab, strval,
61207c478bd9Sstevel@tonic-gate 		    sizeof (strval)) == Z_OK)
61217c478bd9Sstevel@tonic-gate 			return (Z_OK);
61227c478bd9Sstevel@tonic-gate 		zerr(gettext("invalid %s value for %s=%s"),
61237c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR), pt_to_str(PT_TYPE), "string");
61247c478bd9Sstevel@tonic-gate 		return (Z_ERR);
61257c478bd9Sstevel@tonic-gate 	}
61267c478bd9Sstevel@tonic-gate 
61277c478bd9Sstevel@tonic-gate 	if (strcmp(attrtab->zone_attr_type, "uint") == 0) {
61287c478bd9Sstevel@tonic-gate 		if (zonecfg_get_attr_uint(attrtab, &uintval) == Z_OK)
61297c478bd9Sstevel@tonic-gate 			return (Z_OK);
61307c478bd9Sstevel@tonic-gate 		zerr(gettext("invalid %s value for %s=%s"),
61317c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR), pt_to_str(PT_TYPE), "uint");
61327c478bd9Sstevel@tonic-gate 		return (Z_ERR);
61337c478bd9Sstevel@tonic-gate 	}
61347c478bd9Sstevel@tonic-gate 
61357c478bd9Sstevel@tonic-gate 	zerr(gettext("invalid %s %s '%s'"), rt_to_str(RT_ATTR),
61367c478bd9Sstevel@tonic-gate 	    pt_to_str(PT_TYPE), attrtab->zone_attr_type);
61377c478bd9Sstevel@tonic-gate 	return (Z_ERR);
61387c478bd9Sstevel@tonic-gate }
61397c478bd9Sstevel@tonic-gate 
6140087719fdSdp /*
6141087719fdSdp  * Helper function for end_func-- checks the existence of a given property
6142087719fdSdp  * and emits a message if not specified.
6143087719fdSdp  */
6144087719fdSdp static int
6145bbec428eSgjelinek end_check_reqd(char *attr, int pt, boolean_t *validation_failed)
6146087719fdSdp {
6147087719fdSdp 	if (strlen(attr) == 0) {
6148bbec428eSgjelinek 		*validation_failed = B_TRUE;
6149087719fdSdp 		zerr(gettext("%s not specified"), pt_to_str(pt));
6150087719fdSdp 		return (Z_ERR);
6151087719fdSdp 	}
6152087719fdSdp 	return (Z_OK);
6153087719fdSdp }
6154087719fdSdp 
6155550b6e40SSowmini Varadhan static void
6156550b6e40SSowmini Varadhan net_exists_error(struct zone_nwiftab nwif)
6157550b6e40SSowmini Varadhan {
6158550b6e40SSowmini Varadhan 	if (strlen(nwif.zone_nwif_address) > 0) {
6159550b6e40SSowmini Varadhan 		zerr(gettext("A %s resource with the %s '%s', "
6160550b6e40SSowmini Varadhan 		    "and %s '%s' already exists."),
6161550b6e40SSowmini Varadhan 		    rt_to_str(RT_NET),
6162550b6e40SSowmini Varadhan 		    pt_to_str(PT_PHYSICAL),
6163550b6e40SSowmini Varadhan 		    nwif.zone_nwif_physical,
6164550b6e40SSowmini Varadhan 		    pt_to_str(PT_ADDRESS),
6165550b6e40SSowmini Varadhan 		    in_progress_nwiftab.zone_nwif_address);
6166550b6e40SSowmini Varadhan 	} else {
6167550b6e40SSowmini Varadhan 		zerr(gettext("A %s resource with the %s '%s', "
6168550b6e40SSowmini Varadhan 		    "and %s '%s' already exists."),
6169550b6e40SSowmini Varadhan 		    rt_to_str(RT_NET),
6170550b6e40SSowmini Varadhan 		    pt_to_str(PT_PHYSICAL),
6171550b6e40SSowmini Varadhan 		    nwif.zone_nwif_physical,
6172550b6e40SSowmini Varadhan 		    pt_to_str(PT_ALLOWED_ADDRESS),
6173550b6e40SSowmini Varadhan 		    nwif.zone_nwif_allowed_address);
6174550b6e40SSowmini Varadhan 	}
6175550b6e40SSowmini Varadhan }
6176550b6e40SSowmini Varadhan 
61777c478bd9Sstevel@tonic-gate void
61787c478bd9Sstevel@tonic-gate end_func(cmd_t *cmd)
61797c478bd9Sstevel@tonic-gate {
6180bbec428eSgjelinek 	boolean_t validation_failed = B_FALSE;
6181bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
61827c478bd9Sstevel@tonic-gate 	struct zone_fstab tmp_fstab;
61837c478bd9Sstevel@tonic-gate 	struct zone_nwiftab tmp_nwiftab;
61847c478bd9Sstevel@tonic-gate 	struct zone_devtab tmp_devtab;
61857c478bd9Sstevel@tonic-gate 	struct zone_rctltab tmp_rctltab;
61867c478bd9Sstevel@tonic-gate 	struct zone_attrtab tmp_attrtab;
6187fa9e4066Sahrens 	struct zone_dstab tmp_dstab;
6188cb8a054bSGlenn Faden 	struct zone_admintab tmp_admintab;
61890209230bSgjelinek 	int err, arg, res1, res2, res3;
61900209230bSgjelinek 	uint64_t swap_limit;
61910209230bSgjelinek 	uint64_t locked_limit;
6192c97ad5cdSakolb 	uint64_t proc_cap;
61937c478bd9Sstevel@tonic-gate 
61947c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
61957c478bd9Sstevel@tonic-gate 
61967c478bd9Sstevel@tonic-gate 	optind = 0;
61977ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
61987c478bd9Sstevel@tonic-gate 		switch (arg) {
61997c478bd9Sstevel@tonic-gate 		case '?':
62007c478bd9Sstevel@tonic-gate 			longer_usage(CMD_END);
6201bbec428eSgjelinek 			arg_err = B_TRUE;
62027ec75eb8Sgjelinek 			break;
62037c478bd9Sstevel@tonic-gate 		default:
62047c478bd9Sstevel@tonic-gate 			short_usage(CMD_END);
6205bbec428eSgjelinek 			arg_err = B_TRUE;
62067ec75eb8Sgjelinek 			break;
62077ec75eb8Sgjelinek 		}
62087ec75eb8Sgjelinek 	}
62097ec75eb8Sgjelinek 	if (arg_err)
62107c478bd9Sstevel@tonic-gate 		return;
62117ec75eb8Sgjelinek 
62127c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
62137c478bd9Sstevel@tonic-gate 		short_usage(CMD_END);
62147c478bd9Sstevel@tonic-gate 		return;
62157c478bd9Sstevel@tonic-gate 	}
62167c478bd9Sstevel@tonic-gate 
62177c478bd9Sstevel@tonic-gate 	if (global_scope) {
62187c478bd9Sstevel@tonic-gate 		scope_usage(CMD_END);
62197c478bd9Sstevel@tonic-gate 		return;
62207c478bd9Sstevel@tonic-gate 	}
62217c478bd9Sstevel@tonic-gate 
62227c478bd9Sstevel@tonic-gate 	assert(end_op == CMD_ADD || end_op == CMD_SELECT);
62237c478bd9Sstevel@tonic-gate 
62247c478bd9Sstevel@tonic-gate 	switch (resource_scope) {
62257c478bd9Sstevel@tonic-gate 	case RT_FS:
62267c478bd9Sstevel@tonic-gate 		/* First make sure everything was filled in. */
6227087719fdSdp 		if (end_check_reqd(in_progress_fstab.zone_fs_dir,
6228087719fdSdp 		    PT_DIR, &validation_failed) == Z_OK) {
6229087719fdSdp 			if (in_progress_fstab.zone_fs_dir[0] != '/') {
6230087719fdSdp 				zerr(gettext("%s %s is not an absolute path."),
6231087719fdSdp 				    pt_to_str(PT_DIR),
6232087719fdSdp 				    in_progress_fstab.zone_fs_dir);
6233bbec428eSgjelinek 				validation_failed = B_TRUE;
62347c478bd9Sstevel@tonic-gate 			}
62357c478bd9Sstevel@tonic-gate 		}
6236087719fdSdp 
6237087719fdSdp 		(void) end_check_reqd(in_progress_fstab.zone_fs_special,
6238087719fdSdp 		    PT_SPECIAL, &validation_failed);
6239087719fdSdp 
62407c478bd9Sstevel@tonic-gate 		if (in_progress_fstab.zone_fs_raw[0] != '\0' &&
62417c478bd9Sstevel@tonic-gate 		    in_progress_fstab.zone_fs_raw[0] != '/') {
6242087719fdSdp 			zerr(gettext("%s %s is not an absolute path."),
6243087719fdSdp 			    pt_to_str(PT_RAW),
6244087719fdSdp 			    in_progress_fstab.zone_fs_raw);
6245bbec428eSgjelinek 			validation_failed = B_TRUE;
62467c478bd9Sstevel@tonic-gate 		}
6247087719fdSdp 
6248087719fdSdp 		(void) end_check_reqd(in_progress_fstab.zone_fs_type, PT_TYPE,
6249087719fdSdp 		    &validation_failed);
6250087719fdSdp 
6251087719fdSdp 		if (validation_failed) {
6252bbec428eSgjelinek 			saw_error = B_TRUE;
62537c478bd9Sstevel@tonic-gate 			return;
6254087719fdSdp 		}
6255087719fdSdp 
62567c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
62577c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
62587c478bd9Sstevel@tonic-gate 			bzero(&tmp_fstab, sizeof (tmp_fstab));
62597c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_fstab.zone_fs_dir,
62607c478bd9Sstevel@tonic-gate 			    in_progress_fstab.zone_fs_dir,
62617c478bd9Sstevel@tonic-gate 			    sizeof (tmp_fstab.zone_fs_dir));
62627c478bd9Sstevel@tonic-gate 			err = zonecfg_lookup_filesystem(handle, &tmp_fstab);
62637c478bd9Sstevel@tonic-gate 			zonecfg_free_fs_option_list(tmp_fstab.zone_fs_options);
62647c478bd9Sstevel@tonic-gate 			if (err == Z_OK) {
62657c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s resource "
62667c478bd9Sstevel@tonic-gate 				    "with the %s '%s' already exists."),
62677c478bd9Sstevel@tonic-gate 				    rt_to_str(RT_FS), pt_to_str(PT_DIR),
62687c478bd9Sstevel@tonic-gate 				    in_progress_fstab.zone_fs_dir);
6269bbec428eSgjelinek 				saw_error = B_TRUE;
62707c478bd9Sstevel@tonic-gate 				return;
62717c478bd9Sstevel@tonic-gate 			}
62727c478bd9Sstevel@tonic-gate 			err = zonecfg_add_filesystem(handle,
62737c478bd9Sstevel@tonic-gate 			    &in_progress_fstab);
62747c478bd9Sstevel@tonic-gate 		} else {
62757c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_filesystem(handle, &old_fstab,
62767c478bd9Sstevel@tonic-gate 			    &in_progress_fstab);
62777c478bd9Sstevel@tonic-gate 		}
62787c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(in_progress_fstab.zone_fs_options);
62797c478bd9Sstevel@tonic-gate 		in_progress_fstab.zone_fs_options = NULL;
62807c478bd9Sstevel@tonic-gate 		break;
6281087719fdSdp 
62827c478bd9Sstevel@tonic-gate 	case RT_NET:
6283f4b3ec61Sdh155122 		/*
6284f4b3ec61Sdh155122 		 * First make sure everything was filled in.
6285f4b3ec61Sdh155122 		 * Since we don't know whether IP will be shared
6286f4b3ec61Sdh155122 		 * or exclusive here, some checks are deferred until
6287f4b3ec61Sdh155122 		 * the verify command.
6288f4b3ec61Sdh155122 		 */
6289087719fdSdp 		(void) end_check_reqd(in_progress_nwiftab.zone_nwif_physical,
6290087719fdSdp 		    PT_PHYSICAL, &validation_failed);
6291087719fdSdp 
6292087719fdSdp 		if (validation_failed) {
6293bbec428eSgjelinek 			saw_error = B_TRUE;
62947c478bd9Sstevel@tonic-gate 			return;
6295087719fdSdp 		}
62967c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
62977c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
62987c478bd9Sstevel@tonic-gate 			bzero(&tmp_nwiftab, sizeof (tmp_nwiftab));
6299f4b3ec61Sdh155122 			(void) strlcpy(tmp_nwiftab.zone_nwif_physical,
6300f4b3ec61Sdh155122 			    in_progress_nwiftab.zone_nwif_physical,
6301f4b3ec61Sdh155122 			    sizeof (tmp_nwiftab.zone_nwif_physical));
63027c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_nwiftab.zone_nwif_address,
63037c478bd9Sstevel@tonic-gate 			    in_progress_nwiftab.zone_nwif_address,
63047c478bd9Sstevel@tonic-gate 			    sizeof (tmp_nwiftab.zone_nwif_address));
6305550b6e40SSowmini Varadhan 			(void) strlcpy(tmp_nwiftab.zone_nwif_allowed_address,
6306550b6e40SSowmini Varadhan 			    in_progress_nwiftab.zone_nwif_allowed_address,
6307550b6e40SSowmini Varadhan 			    sizeof (tmp_nwiftab.zone_nwif_allowed_address));
6308550b6e40SSowmini Varadhan 			(void) strlcpy(tmp_nwiftab.zone_nwif_defrouter,
6309550b6e40SSowmini Varadhan 			    in_progress_nwiftab.zone_nwif_defrouter,
6310550b6e40SSowmini Varadhan 			    sizeof (tmp_nwiftab.zone_nwif_defrouter));
63117c478bd9Sstevel@tonic-gate 			if (zonecfg_lookup_nwif(handle, &tmp_nwiftab) == Z_OK) {
6312550b6e40SSowmini Varadhan 				net_exists_error(in_progress_nwiftab);
6313bbec428eSgjelinek 				saw_error = B_TRUE;
63147c478bd9Sstevel@tonic-gate 				return;
63157c478bd9Sstevel@tonic-gate 			}
63167c478bd9Sstevel@tonic-gate 			err = zonecfg_add_nwif(handle, &in_progress_nwiftab);
63177c478bd9Sstevel@tonic-gate 		} else {
63187c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_nwif(handle, &old_nwiftab,
63197c478bd9Sstevel@tonic-gate 			    &in_progress_nwiftab);
63207c478bd9Sstevel@tonic-gate 		}
63217c478bd9Sstevel@tonic-gate 		break;
6322087719fdSdp 
63237c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
63247c478bd9Sstevel@tonic-gate 		/* First make sure everything was filled in. */
6325087719fdSdp 		(void) end_check_reqd(in_progress_devtab.zone_dev_match,
6326087719fdSdp 		    PT_MATCH, &validation_failed);
6327087719fdSdp 
6328087719fdSdp 		if (validation_failed) {
6329bbec428eSgjelinek 			saw_error = B_TRUE;
63307c478bd9Sstevel@tonic-gate 			return;
6331087719fdSdp 		}
6332087719fdSdp 
63337c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
63347c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
63357c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_devtab.zone_dev_match,
63367c478bd9Sstevel@tonic-gate 			    in_progress_devtab.zone_dev_match,
63377c478bd9Sstevel@tonic-gate 			    sizeof (tmp_devtab.zone_dev_match));
63387c478bd9Sstevel@tonic-gate 			if (zonecfg_lookup_dev(handle, &tmp_devtab) == Z_OK) {
63397c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s resource with the %s '%s' "
63407c478bd9Sstevel@tonic-gate 				    "already exists."), rt_to_str(RT_DEVICE),
63417c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_MATCH),
63427c478bd9Sstevel@tonic-gate 				    in_progress_devtab.zone_dev_match);
6343bbec428eSgjelinek 				saw_error = B_TRUE;
63447c478bd9Sstevel@tonic-gate 				return;
63457c478bd9Sstevel@tonic-gate 			}
63467c478bd9Sstevel@tonic-gate 			err = zonecfg_add_dev(handle, &in_progress_devtab);
63477c478bd9Sstevel@tonic-gate 		} else {
63487c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_dev(handle, &old_devtab,
63497c478bd9Sstevel@tonic-gate 			    &in_progress_devtab);
63507c478bd9Sstevel@tonic-gate 		}
63517c478bd9Sstevel@tonic-gate 		break;
6352087719fdSdp 
63537c478bd9Sstevel@tonic-gate 	case RT_RCTL:
63547c478bd9Sstevel@tonic-gate 		/* First make sure everything was filled in. */
6355087719fdSdp 		(void) end_check_reqd(in_progress_rctltab.zone_rctl_name,
6356087719fdSdp 		    PT_NAME, &validation_failed);
6357087719fdSdp 
63587c478bd9Sstevel@tonic-gate 		if (in_progress_rctltab.zone_rctl_valptr == NULL) {
63597c478bd9Sstevel@tonic-gate 			zerr(gettext("no %s specified"), pt_to_str(PT_VALUE));
6360bbec428eSgjelinek 			validation_failed = B_TRUE;
63617c478bd9Sstevel@tonic-gate 		}
6362087719fdSdp 
6363087719fdSdp 		if (validation_failed) {
6364bbec428eSgjelinek 			saw_error = B_TRUE;
63657c478bd9Sstevel@tonic-gate 			return;
6366087719fdSdp 		}
6367087719fdSdp 
63687c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
63697c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
63707c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_rctltab.zone_rctl_name,
63717c478bd9Sstevel@tonic-gate 			    in_progress_rctltab.zone_rctl_name,
63727c478bd9Sstevel@tonic-gate 			    sizeof (tmp_rctltab.zone_rctl_name));
63737c478bd9Sstevel@tonic-gate 			tmp_rctltab.zone_rctl_valptr = NULL;
63747c478bd9Sstevel@tonic-gate 			err = zonecfg_lookup_rctl(handle, &tmp_rctltab);
63757c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(
63767c478bd9Sstevel@tonic-gate 			    tmp_rctltab.zone_rctl_valptr);
63777c478bd9Sstevel@tonic-gate 			if (err == Z_OK) {
63787c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s resource "
63797c478bd9Sstevel@tonic-gate 				    "with the %s '%s' already exists."),
63807c478bd9Sstevel@tonic-gate 				    rt_to_str(RT_RCTL), pt_to_str(PT_NAME),
63817c478bd9Sstevel@tonic-gate 				    in_progress_rctltab.zone_rctl_name);
6382bbec428eSgjelinek 				saw_error = B_TRUE;
63837c478bd9Sstevel@tonic-gate 				return;
63847c478bd9Sstevel@tonic-gate 			}
63857c478bd9Sstevel@tonic-gate 			err = zonecfg_add_rctl(handle, &in_progress_rctltab);
63867c478bd9Sstevel@tonic-gate 		} else {
63877c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_rctl(handle, &old_rctltab,
63887c478bd9Sstevel@tonic-gate 			    &in_progress_rctltab);
63897c478bd9Sstevel@tonic-gate 		}
63907c478bd9Sstevel@tonic-gate 		if (err == Z_OK) {
63917c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(
63927c478bd9Sstevel@tonic-gate 			    in_progress_rctltab.zone_rctl_valptr);
63937c478bd9Sstevel@tonic-gate 			in_progress_rctltab.zone_rctl_valptr = NULL;
63947c478bd9Sstevel@tonic-gate 		}
63957c478bd9Sstevel@tonic-gate 		break;
6396087719fdSdp 
63977c478bd9Sstevel@tonic-gate 	case RT_ATTR:
63987c478bd9Sstevel@tonic-gate 		/* First make sure everything was filled in. */
6399087719fdSdp 		(void) end_check_reqd(in_progress_attrtab.zone_attr_name,
6400087719fdSdp 		    PT_NAME, &validation_failed);
6401087719fdSdp 		(void) end_check_reqd(in_progress_attrtab.zone_attr_type,
6402087719fdSdp 		    PT_TYPE, &validation_failed);
6403087719fdSdp 		(void) end_check_reqd(in_progress_attrtab.zone_attr_value,
6404087719fdSdp 		    PT_VALUE, &validation_failed);
6405087719fdSdp 
64067c478bd9Sstevel@tonic-gate 		if (validate_attr_name(in_progress_attrtab.zone_attr_name) !=
6407087719fdSdp 		    Z_OK)
6408bbec428eSgjelinek 			validation_failed = B_TRUE;
6409087719fdSdp 
6410087719fdSdp 		if (validate_attr_type_val(&in_progress_attrtab) != Z_OK)
6411bbec428eSgjelinek 			validation_failed = B_TRUE;
6412087719fdSdp 
6413087719fdSdp 		if (validation_failed) {
6414bbec428eSgjelinek 			saw_error = B_TRUE;
64157c478bd9Sstevel@tonic-gate 			return;
6416087719fdSdp 		}
64177c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
64187c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
64197c478bd9Sstevel@tonic-gate 			bzero(&tmp_attrtab, sizeof (tmp_attrtab));
64207c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_attrtab.zone_attr_name,
64217c478bd9Sstevel@tonic-gate 			    in_progress_attrtab.zone_attr_name,
64227c478bd9Sstevel@tonic-gate 			    sizeof (tmp_attrtab.zone_attr_name));
64237c478bd9Sstevel@tonic-gate 			if (zonecfg_lookup_attr(handle, &tmp_attrtab) == Z_OK) {
64247c478bd9Sstevel@tonic-gate 				zerr(gettext("An %s resource "
64257c478bd9Sstevel@tonic-gate 				    "with the %s '%s' already exists."),
64267c478bd9Sstevel@tonic-gate 				    rt_to_str(RT_ATTR), pt_to_str(PT_NAME),
64277c478bd9Sstevel@tonic-gate 				    in_progress_attrtab.zone_attr_name);
6428bbec428eSgjelinek 				saw_error = B_TRUE;
64297c478bd9Sstevel@tonic-gate 				return;
64307c478bd9Sstevel@tonic-gate 			}
64317c478bd9Sstevel@tonic-gate 			err = zonecfg_add_attr(handle, &in_progress_attrtab);
64327c478bd9Sstevel@tonic-gate 		} else {
64337c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_attr(handle, &old_attrtab,
64347c478bd9Sstevel@tonic-gate 			    &in_progress_attrtab);
64357c478bd9Sstevel@tonic-gate 		}
64367c478bd9Sstevel@tonic-gate 		break;
6437fa9e4066Sahrens 	case RT_DATASET:
6438fa9e4066Sahrens 		/* First make sure everything was filled in. */
6439fa9e4066Sahrens 		if (strlen(in_progress_dstab.zone_dataset_name) == 0) {
6440fa9e4066Sahrens 			zerr("%s %s", pt_to_str(PT_NAME),
6441fa9e4066Sahrens 			    gettext("not specified"));
6442bbec428eSgjelinek 			saw_error = B_TRUE;
6443bbec428eSgjelinek 			validation_failed = B_TRUE;
6444fa9e4066Sahrens 		}
6445fa9e4066Sahrens 		if (validation_failed)
6446fa9e4066Sahrens 			return;
6447fa9e4066Sahrens 		if (end_op == CMD_ADD) {
6448fa9e4066Sahrens 			/* Make sure there isn't already one like this. */
6449fa9e4066Sahrens 			bzero(&tmp_dstab, sizeof (tmp_dstab));
6450fa9e4066Sahrens 			(void) strlcpy(tmp_dstab.zone_dataset_name,
6451fa9e4066Sahrens 			    in_progress_dstab.zone_dataset_name,
6452fa9e4066Sahrens 			    sizeof (tmp_dstab.zone_dataset_name));
6453fa9e4066Sahrens 			err = zonecfg_lookup_ds(handle, &tmp_dstab);
6454fa9e4066Sahrens 			if (err == Z_OK) {
6455fa9e4066Sahrens 				zerr(gettext("A %s resource "
6456fa9e4066Sahrens 				    "with the %s '%s' already exists."),
6457fa9e4066Sahrens 				    rt_to_str(RT_DATASET), pt_to_str(PT_NAME),
6458fa9e4066Sahrens 				    in_progress_dstab.zone_dataset_name);
6459bbec428eSgjelinek 				saw_error = B_TRUE;
6460fa9e4066Sahrens 				return;
6461fa9e4066Sahrens 			}
6462fa9e4066Sahrens 			err = zonecfg_add_ds(handle, &in_progress_dstab);
6463fa9e4066Sahrens 		} else {
6464fa9e4066Sahrens 			err = zonecfg_modify_ds(handle, &old_dstab,
6465fa9e4066Sahrens 			    &in_progress_dstab);
6466fa9e4066Sahrens 		}
6467fa9e4066Sahrens 		break;
64680209230bSgjelinek 	case RT_DCPU:
64690209230bSgjelinek 		/* Make sure everything was filled in. */
64700209230bSgjelinek 		if (end_check_reqd(in_progress_psettab.zone_ncpu_min,
64710209230bSgjelinek 		    PT_NCPUS, &validation_failed) != Z_OK) {
6472bbec428eSgjelinek 			saw_error = B_TRUE;
64730209230bSgjelinek 			return;
64740209230bSgjelinek 		}
64750209230bSgjelinek 
64760209230bSgjelinek 		if (end_op == CMD_ADD) {
64770209230bSgjelinek 			err = zonecfg_add_pset(handle, &in_progress_psettab);
64780209230bSgjelinek 		} else {
64790209230bSgjelinek 			err = zonecfg_modify_pset(handle, &in_progress_psettab);
64800209230bSgjelinek 		}
64810209230bSgjelinek 		break;
6482c97ad5cdSakolb 	case RT_PCAP:
6483c97ad5cdSakolb 		/* Make sure everything was filled in. */
6484c97ad5cdSakolb 		if (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &proc_cap)
6485c97ad5cdSakolb 		    != Z_OK) {
6486c97ad5cdSakolb 			zerr(gettext("%s not specified"), pt_to_str(PT_NCPUS));
6487bbec428eSgjelinek 			saw_error = B_TRUE;
6488bbec428eSgjelinek 			validation_failed = B_TRUE;
6489c97ad5cdSakolb 			return;
6490c97ad5cdSakolb 		}
6491c97ad5cdSakolb 		err = Z_OK;
6492c97ad5cdSakolb 		break;
64930209230bSgjelinek 	case RT_MCAP:
64940209230bSgjelinek 		/* Make sure everything was filled in. */
64950209230bSgjelinek 		res1 = strlen(in_progress_mcaptab.zone_physmem_cap) == 0 ?
64960209230bSgjelinek 		    Z_ERR : Z_OK;
64970209230bSgjelinek 		res2 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP,
64980209230bSgjelinek 		    &swap_limit);
64990209230bSgjelinek 		res3 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM,
65000209230bSgjelinek 		    &locked_limit);
65010209230bSgjelinek 
65020209230bSgjelinek 		if (res1 != Z_OK && res2 != Z_OK && res3 != Z_OK) {
65030209230bSgjelinek 			zerr(gettext("No property was specified.  One of %s, "
65040209230bSgjelinek 			    "%s or %s is required."), pt_to_str(PT_PHYSICAL),
65050209230bSgjelinek 			    pt_to_str(PT_SWAP), pt_to_str(PT_LOCKED));
6506bbec428eSgjelinek 			saw_error = B_TRUE;
65070209230bSgjelinek 			return;
65080209230bSgjelinek 		}
65090209230bSgjelinek 
65100209230bSgjelinek 		/* if phys & locked are both set, verify locked <= phys */
65110209230bSgjelinek 		if (res1 == Z_OK && res3 == Z_OK) {
65120209230bSgjelinek 			uint64_t phys_limit;
65130209230bSgjelinek 			char *endp;
65140209230bSgjelinek 
65150209230bSgjelinek 			phys_limit = strtoull(
65160209230bSgjelinek 			    in_progress_mcaptab.zone_physmem_cap, &endp, 10);
65170209230bSgjelinek 			if (phys_limit < locked_limit) {
65180209230bSgjelinek 				zerr(gettext("The %s cap must be less than or "
65190209230bSgjelinek 				    "equal to the %s cap."),
65200209230bSgjelinek 				    pt_to_str(PT_LOCKED),
65210209230bSgjelinek 				    pt_to_str(PT_PHYSICAL));
6522bbec428eSgjelinek 				saw_error = B_TRUE;
65230209230bSgjelinek 				return;
65240209230bSgjelinek 			}
65250209230bSgjelinek 		}
65260209230bSgjelinek 
65270209230bSgjelinek 		err = Z_OK;
65280209230bSgjelinek 		if (res1 == Z_OK) {
65290209230bSgjelinek 			/*
65300209230bSgjelinek 			 * We could be ending from either an add operation
65310209230bSgjelinek 			 * or a select operation.  Since all of the properties
65320209230bSgjelinek 			 * within this resource are optional, we always use
65330209230bSgjelinek 			 * modify on the mcap entry.  zonecfg_modify_mcap()
65340209230bSgjelinek 			 * will handle both adding and modifying a memory cap.
65350209230bSgjelinek 			 */
65360209230bSgjelinek 			err = zonecfg_modify_mcap(handle, &in_progress_mcaptab);
65370209230bSgjelinek 		} else if (end_op == CMD_SELECT) {
65380209230bSgjelinek 			/*
65390209230bSgjelinek 			 * If we're ending from a select and the physical
65400209230bSgjelinek 			 * memory cap is empty then the user could have cleared
65410209230bSgjelinek 			 * the physical cap value, so try to delete the entry.
65420209230bSgjelinek 			 */
65430209230bSgjelinek 			(void) zonecfg_delete_mcap(handle);
65440209230bSgjelinek 		}
65450209230bSgjelinek 		break;
6546cb8a054bSGlenn Faden 	case RT_ADMIN:
6547cb8a054bSGlenn Faden 		/* First make sure everything was filled in. */
6548cb8a054bSGlenn Faden 		if (end_check_reqd(in_progress_admintab.zone_admin_user,
6549cb8a054bSGlenn Faden 		    PT_USER, &validation_failed) == Z_OK) {
6550cb8a054bSGlenn Faden 			if (getpwnam(in_progress_admintab.zone_admin_user)
6551cb8a054bSGlenn Faden 			    == NULL) {
6552cb8a054bSGlenn Faden 				zerr(gettext("%s %s is not a valid username"),
6553cb8a054bSGlenn Faden 				    pt_to_str(PT_USER),
6554cb8a054bSGlenn Faden 				    in_progress_admintab.zone_admin_user);
6555cb8a054bSGlenn Faden 				validation_failed = B_TRUE;
6556cb8a054bSGlenn Faden 			}
6557cb8a054bSGlenn Faden 		}
6558cb8a054bSGlenn Faden 
6559cb8a054bSGlenn Faden 		if (end_check_reqd(in_progress_admintab.zone_admin_auths,
6560cb8a054bSGlenn Faden 		    PT_AUTHS, &validation_failed) == Z_OK) {
6561cb8a054bSGlenn Faden 			if (!zonecfg_valid_auths(
6562cb8a054bSGlenn Faden 			    in_progress_admintab.zone_admin_auths,
6563cb8a054bSGlenn Faden 			    zone)) {
6564cb8a054bSGlenn Faden 				validation_failed = B_TRUE;
6565cb8a054bSGlenn Faden 			}
6566cb8a054bSGlenn Faden 		}
6567cb8a054bSGlenn Faden 
6568cb8a054bSGlenn Faden 		if (validation_failed) {
6569cb8a054bSGlenn Faden 			saw_error = B_TRUE;
6570cb8a054bSGlenn Faden 			return;
6571cb8a054bSGlenn Faden 		}
6572cb8a054bSGlenn Faden 
6573cb8a054bSGlenn Faden 		if (end_op == CMD_ADD) {
6574cb8a054bSGlenn Faden 			/* Make sure there isn't already one like this. */
6575cb8a054bSGlenn Faden 			bzero(&tmp_admintab, sizeof (tmp_admintab));
6576cb8a054bSGlenn Faden 			(void) strlcpy(tmp_admintab.zone_admin_user,
6577cb8a054bSGlenn Faden 			    in_progress_admintab.zone_admin_user,
6578cb8a054bSGlenn Faden 			    sizeof (tmp_admintab.zone_admin_user));
6579cb8a054bSGlenn Faden 			err = zonecfg_lookup_admin(
6580cb8a054bSGlenn Faden 			    handle, &tmp_admintab);
6581cb8a054bSGlenn Faden 			if (err == Z_OK) {
6582cb8a054bSGlenn Faden 				zerr(gettext("A %s resource "
6583cb8a054bSGlenn Faden 				    "with the %s '%s' already exists."),
6584cb8a054bSGlenn Faden 				    rt_to_str(RT_ADMIN),
6585cb8a054bSGlenn Faden 				    pt_to_str(PT_USER),
6586cb8a054bSGlenn Faden 				    in_progress_admintab.zone_admin_user);
6587cb8a054bSGlenn Faden 				saw_error = B_TRUE;
6588cb8a054bSGlenn Faden 				return;
6589cb8a054bSGlenn Faden 			}
6590cb8a054bSGlenn Faden 			err = zonecfg_add_admin(handle,
6591cb8a054bSGlenn Faden 			    &in_progress_admintab, zone);
6592cb8a054bSGlenn Faden 		} else {
6593cb8a054bSGlenn Faden 			err = zonecfg_modify_admin(handle,
6594cb8a054bSGlenn Faden 			    &old_admintab, &in_progress_admintab,
6595cb8a054bSGlenn Faden 			    zone);
6596cb8a054bSGlenn Faden 		}
6597cb8a054bSGlenn Faden 		break;
65987c478bd9Sstevel@tonic-gate 	default:
65997c478bd9Sstevel@tonic-gate 		zone_perror(rt_to_str(resource_scope), Z_NO_RESOURCE_TYPE,
6600bbec428eSgjelinek 		    B_TRUE);
6601bbec428eSgjelinek 		saw_error = B_TRUE;
66027c478bd9Sstevel@tonic-gate 		return;
66037c478bd9Sstevel@tonic-gate 	}
66047c478bd9Sstevel@tonic-gate 
66057c478bd9Sstevel@tonic-gate 	if (err != Z_OK) {
6606bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
66077c478bd9Sstevel@tonic-gate 	} else {
6608bbec428eSgjelinek 		need_to_commit = B_TRUE;
6609bbec428eSgjelinek 		global_scope = B_TRUE;
66107c478bd9Sstevel@tonic-gate 		end_op = -1;
66117c478bd9Sstevel@tonic-gate 	}
66127c478bd9Sstevel@tonic-gate }
66137c478bd9Sstevel@tonic-gate 
66147c478bd9Sstevel@tonic-gate void
66157c478bd9Sstevel@tonic-gate commit_func(cmd_t *cmd)
66167c478bd9Sstevel@tonic-gate {
66177c478bd9Sstevel@tonic-gate 	int arg;
6618bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
66197c478bd9Sstevel@tonic-gate 
66207c478bd9Sstevel@tonic-gate 	optind = 0;
66217ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
66227c478bd9Sstevel@tonic-gate 		switch (arg) {
66237c478bd9Sstevel@tonic-gate 		case '?':
66247c478bd9Sstevel@tonic-gate 			longer_usage(CMD_COMMIT);
6625bbec428eSgjelinek 			arg_err = B_TRUE;
66267ec75eb8Sgjelinek 			break;
66277c478bd9Sstevel@tonic-gate 		default:
66287c478bd9Sstevel@tonic-gate 			short_usage(CMD_COMMIT);
6629bbec428eSgjelinek 			arg_err = B_TRUE;
66307ec75eb8Sgjelinek 			break;
66317ec75eb8Sgjelinek 		}
66327ec75eb8Sgjelinek 	}
66337ec75eb8Sgjelinek 	if (arg_err)
66347c478bd9Sstevel@tonic-gate 		return;
66357ec75eb8Sgjelinek 
66367c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
66377c478bd9Sstevel@tonic-gate 		short_usage(CMD_COMMIT);
66387c478bd9Sstevel@tonic-gate 		return;
66397c478bd9Sstevel@tonic-gate 	}
66407c478bd9Sstevel@tonic-gate 
66417c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_COMMIT))
66427c478bd9Sstevel@tonic-gate 		return;
66437c478bd9Sstevel@tonic-gate 
66447c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
66457c478bd9Sstevel@tonic-gate 
66467c478bd9Sstevel@tonic-gate 	cmd->cmd_argc = 1;
66477c478bd9Sstevel@tonic-gate 	/*
66487c478bd9Sstevel@tonic-gate 	 * cmd_arg normally comes from a strdup() in the lexer, and the
66497c478bd9Sstevel@tonic-gate 	 * whole cmd structure and its (char *) attributes are freed at
66507c478bd9Sstevel@tonic-gate 	 * the completion of each command, so the strdup() below is needed
66517c478bd9Sstevel@tonic-gate 	 * to match this and prevent a core dump from trying to free()
66527c478bd9Sstevel@tonic-gate 	 * something that can't be.
66537c478bd9Sstevel@tonic-gate 	 */
66547c478bd9Sstevel@tonic-gate 	if ((cmd->cmd_argv[0] = strdup("save")) == NULL) {
6655bbec428eSgjelinek 		zone_perror(zone, Z_NOMEM, B_TRUE);
66567c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
66577c478bd9Sstevel@tonic-gate 	}
66587c478bd9Sstevel@tonic-gate 	cmd->cmd_argv[1] = NULL;
66597c478bd9Sstevel@tonic-gate 	verify_func(cmd);
66607c478bd9Sstevel@tonic-gate }
66617c478bd9Sstevel@tonic-gate 
66627c478bd9Sstevel@tonic-gate void
66637c478bd9Sstevel@tonic-gate revert_func(cmd_t *cmd)
66647c478bd9Sstevel@tonic-gate {
66657c478bd9Sstevel@tonic-gate 	char line[128];	/* enough to ask a question */
6666bbec428eSgjelinek 	boolean_t force = B_FALSE;
6667bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
66687c478bd9Sstevel@tonic-gate 	int err, arg, answer;
66697c478bd9Sstevel@tonic-gate 
66707c478bd9Sstevel@tonic-gate 	optind = 0;
66717c478bd9Sstevel@tonic-gate 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?F")) != EOF) {
66727c478bd9Sstevel@tonic-gate 		switch (arg) {
66737c478bd9Sstevel@tonic-gate 		case '?':
66747c478bd9Sstevel@tonic-gate 			longer_usage(CMD_REVERT);
6675bbec428eSgjelinek 			arg_err = B_TRUE;
66767ec75eb8Sgjelinek 			break;
66777c478bd9Sstevel@tonic-gate 		case 'F':
6678bbec428eSgjelinek 			force = B_TRUE;
66797c478bd9Sstevel@tonic-gate 			break;
66807c478bd9Sstevel@tonic-gate 		default:
66817c478bd9Sstevel@tonic-gate 			short_usage(CMD_REVERT);
6682bbec428eSgjelinek 			arg_err = B_TRUE;
66837ec75eb8Sgjelinek 			break;
66847ec75eb8Sgjelinek 		}
66857ec75eb8Sgjelinek 	}
66867ec75eb8Sgjelinek 	if (arg_err)
66877c478bd9Sstevel@tonic-gate 		return;
66887ec75eb8Sgjelinek 
66897c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
66907c478bd9Sstevel@tonic-gate 		short_usage(CMD_REVERT);
66917c478bd9Sstevel@tonic-gate 		return;
66927c478bd9Sstevel@tonic-gate 	}
66937c478bd9Sstevel@tonic-gate 
66947c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_REVERT))
66957c478bd9Sstevel@tonic-gate 		return;
66967c478bd9Sstevel@tonic-gate 
66975b418297Sjv227347 	if (!global_scope) {
66985b418297Sjv227347 		zerr(gettext("You can only use %s in the global scope.\nUse"
66995b418297Sjv227347 		    " '%s' to cancel changes to a resource specification."),
67005b418297Sjv227347 		    cmd_to_str(CMD_REVERT), cmd_to_str(CMD_CANCEL));
67015b418297Sjv227347 		saw_error = B_TRUE;
67025b418297Sjv227347 		return;
67035b418297Sjv227347 	}
67045b418297Sjv227347 
67057c478bd9Sstevel@tonic-gate 	if (zonecfg_check_handle(handle) != Z_OK) {
67067c478bd9Sstevel@tonic-gate 		zerr(gettext("No changes to revert."));
6707bbec428eSgjelinek 		saw_error = B_TRUE;
67087c478bd9Sstevel@tonic-gate 		return;
67097c478bd9Sstevel@tonic-gate 	}
67107c478bd9Sstevel@tonic-gate 
67117c478bd9Sstevel@tonic-gate 	if (!force) {
67127c478bd9Sstevel@tonic-gate 		(void) snprintf(line, sizeof (line),
67137c478bd9Sstevel@tonic-gate 		    gettext("Are you sure you want to revert"));
6714bbec428eSgjelinek 		if ((answer = ask_yesno(B_FALSE, line)) == -1) {
67157c478bd9Sstevel@tonic-gate 			zerr(gettext("Input not from terminal and -F not "
67167c478bd9Sstevel@tonic-gate 			    "specified:\n%s command ignored, exiting."),
67177c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_REVERT));
67187c478bd9Sstevel@tonic-gate 			exit(Z_ERR);
67197c478bd9Sstevel@tonic-gate 		}
67207c478bd9Sstevel@tonic-gate 		if (answer != 1)
67217c478bd9Sstevel@tonic-gate 			return;
67227c478bd9Sstevel@tonic-gate 	}
67237c478bd9Sstevel@tonic-gate 
67247c478bd9Sstevel@tonic-gate 	/*
6725cb8a054bSGlenn Faden 	 * Reset any pending admins that were
6726cb8a054bSGlenn Faden 	 * removed from the previous zone
6727cb8a054bSGlenn Faden 	 */
6728cb8a054bSGlenn Faden 	zonecfg_remove_userauths(handle, "", zone, B_FALSE);
6729cb8a054bSGlenn Faden 
6730cb8a054bSGlenn Faden 	/*
67317c478bd9Sstevel@tonic-gate 	 * Time for a new handle: finish the old one off first
67327c478bd9Sstevel@tonic-gate 	 * then get a new one properly to avoid leaks.
67337c478bd9Sstevel@tonic-gate 	 */
67347c478bd9Sstevel@tonic-gate 	zonecfg_fini_handle(handle);
67357c478bd9Sstevel@tonic-gate 	if ((handle = zonecfg_init_handle()) == NULL) {
6736bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
67377c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
67387c478bd9Sstevel@tonic-gate 	}
6739cb8a054bSGlenn Faden 
6740087719fdSdp 	if ((err = zonecfg_get_handle(revert_zone, handle)) != Z_OK) {
6741bbec428eSgjelinek 		saw_error = B_TRUE;
6742bbec428eSgjelinek 		got_handle = B_FALSE;
67437c478bd9Sstevel@tonic-gate 		if (err == Z_NO_ZONE)
67447c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: no such saved zone to revert to."),
6745087719fdSdp 			    revert_zone);
67467c478bd9Sstevel@tonic-gate 		else
6747bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
67487c478bd9Sstevel@tonic-gate 	}
6749087719fdSdp 	(void) strlcpy(zone, revert_zone, sizeof (zone));
67507c478bd9Sstevel@tonic-gate }
67517c478bd9Sstevel@tonic-gate 
67527c478bd9Sstevel@tonic-gate void
67537c478bd9Sstevel@tonic-gate help_func(cmd_t *cmd)
67547c478bd9Sstevel@tonic-gate {
67557c478bd9Sstevel@tonic-gate 	int i;
67567c478bd9Sstevel@tonic-gate 
67577c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
67587c478bd9Sstevel@tonic-gate 
67597c478bd9Sstevel@tonic-gate 	if (cmd->cmd_argc == 0) {
6760bbec428eSgjelinek 		usage(B_TRUE, global_scope ? HELP_SUBCMDS : HELP_RES_SCOPE);
67617c478bd9Sstevel@tonic-gate 		return;
67627c478bd9Sstevel@tonic-gate 	}
67637c478bd9Sstevel@tonic-gate 	if (strcmp(cmd->cmd_argv[0], "usage") == 0) {
6764bbec428eSgjelinek 		usage(B_TRUE, HELP_USAGE);
67657c478bd9Sstevel@tonic-gate 		return;
67667c478bd9Sstevel@tonic-gate 	}
67677c478bd9Sstevel@tonic-gate 	if (strcmp(cmd->cmd_argv[0], "commands") == 0) {
6768bbec428eSgjelinek 		usage(B_TRUE, HELP_SUBCMDS);
67697c478bd9Sstevel@tonic-gate 		return;
67707c478bd9Sstevel@tonic-gate 	}
67717c478bd9Sstevel@tonic-gate 	if (strcmp(cmd->cmd_argv[0], "syntax") == 0) {
6772bbec428eSgjelinek 		usage(B_TRUE, HELP_SYNTAX | HELP_RES_PROPS);
67737c478bd9Sstevel@tonic-gate 		return;
67747c478bd9Sstevel@tonic-gate 	}
67757c478bd9Sstevel@tonic-gate 	if (strcmp(cmd->cmd_argv[0], "-?") == 0) {
67767c478bd9Sstevel@tonic-gate 		longer_usage(CMD_HELP);
67777c478bd9Sstevel@tonic-gate 		return;
67787c478bd9Sstevel@tonic-gate 	}
67797c478bd9Sstevel@tonic-gate 
67807c478bd9Sstevel@tonic-gate 	for (i = 0; i <= CMD_MAX; i++) {
67817c478bd9Sstevel@tonic-gate 		if (strcmp(cmd->cmd_argv[0], cmd_to_str(i)) == 0) {
67827c478bd9Sstevel@tonic-gate 			longer_usage(i);
67837c478bd9Sstevel@tonic-gate 			return;
67847c478bd9Sstevel@tonic-gate 		}
67857c478bd9Sstevel@tonic-gate 	}
67867c478bd9Sstevel@tonic-gate 	/* We do not use zerr() here because we do not want its extra \n. */
67877c478bd9Sstevel@tonic-gate 	(void) fprintf(stderr, gettext("Unknown help subject %s.  "),
67887c478bd9Sstevel@tonic-gate 	    cmd->cmd_argv[0]);
6789bbec428eSgjelinek 	usage(B_FALSE, HELP_META);
67907c478bd9Sstevel@tonic-gate }
67917c478bd9Sstevel@tonic-gate 
67927c478bd9Sstevel@tonic-gate static int
67937c478bd9Sstevel@tonic-gate string_to_yyin(char *string)
67947c478bd9Sstevel@tonic-gate {
67957c478bd9Sstevel@tonic-gate 	if ((yyin = tmpfile()) == NULL) {
6796bbec428eSgjelinek 		zone_perror(execname, Z_TEMP_FILE, B_TRUE);
67977c478bd9Sstevel@tonic-gate 		return (Z_ERR);
67987c478bd9Sstevel@tonic-gate 	}
67997c478bd9Sstevel@tonic-gate 	if (fwrite(string, strlen(string), 1, yyin) != 1) {
6800bbec428eSgjelinek 		zone_perror(execname, Z_TEMP_FILE, B_TRUE);
68017c478bd9Sstevel@tonic-gate 		return (Z_ERR);
68027c478bd9Sstevel@tonic-gate 	}
68037c478bd9Sstevel@tonic-gate 	if (fseek(yyin, 0, SEEK_SET) != 0) {
6804bbec428eSgjelinek 		zone_perror(execname, Z_TEMP_FILE, B_TRUE);
68057c478bd9Sstevel@tonic-gate 		return (Z_ERR);
68067c478bd9Sstevel@tonic-gate 	}
68077c478bd9Sstevel@tonic-gate 	return (Z_OK);
68087c478bd9Sstevel@tonic-gate }
68097c478bd9Sstevel@tonic-gate 
68107c478bd9Sstevel@tonic-gate /* This is the back-end helper function for read_input() below. */
68117c478bd9Sstevel@tonic-gate 
68127c478bd9Sstevel@tonic-gate static int
68137c478bd9Sstevel@tonic-gate cleanup()
68147c478bd9Sstevel@tonic-gate {
68157c478bd9Sstevel@tonic-gate 	int answer;
68167c478bd9Sstevel@tonic-gate 	cmd_t *cmd;
68177c478bd9Sstevel@tonic-gate 
68187c478bd9Sstevel@tonic-gate 	if (!interactive_mode && !cmd_file_mode) {
68197c478bd9Sstevel@tonic-gate 		/*
68207c478bd9Sstevel@tonic-gate 		 * If we're not in interactive mode, and we're not in command
68217c478bd9Sstevel@tonic-gate 		 * file mode, then we must be in commands-from-the-command-line
68227c478bd9Sstevel@tonic-gate 		 * mode.  As such, we can't loop back and ask for more input.
68237c478bd9Sstevel@tonic-gate 		 * It was OK to prompt for such things as whether or not to
68247c478bd9Sstevel@tonic-gate 		 * really delete a zone in the command handler called from
68257c478bd9Sstevel@tonic-gate 		 * yyparse() above, but "really quit?" makes no sense in this
68267c478bd9Sstevel@tonic-gate 		 * context.  So disable prompting.
68277c478bd9Sstevel@tonic-gate 		 */
6828bbec428eSgjelinek 		ok_to_prompt = B_FALSE;
68297c478bd9Sstevel@tonic-gate 	}
68307c478bd9Sstevel@tonic-gate 	if (!global_scope) {
68317c478bd9Sstevel@tonic-gate 		if (!time_to_exit) {
68327c478bd9Sstevel@tonic-gate 			/*
68337c478bd9Sstevel@tonic-gate 			 * Just print a simple error message in the -1 case,
68347c478bd9Sstevel@tonic-gate 			 * since exit_func() already handles that case, and
68357c478bd9Sstevel@tonic-gate 			 * EOF means we are finished anyway.
68367c478bd9Sstevel@tonic-gate 			 */
6837bbec428eSgjelinek 			answer = ask_yesno(B_FALSE,
68387c478bd9Sstevel@tonic-gate 			    gettext("Resource incomplete; really quit"));
68397c478bd9Sstevel@tonic-gate 			if (answer == -1) {
68407c478bd9Sstevel@tonic-gate 				zerr(gettext("Resource incomplete."));
68417c478bd9Sstevel@tonic-gate 				return (Z_ERR);
68427c478bd9Sstevel@tonic-gate 			}
68437c478bd9Sstevel@tonic-gate 			if (answer != 1) {
68447c478bd9Sstevel@tonic-gate 				yyin = stdin;
68457c478bd9Sstevel@tonic-gate 				return (Z_REPEAT);
68467c478bd9Sstevel@tonic-gate 			}
68477c478bd9Sstevel@tonic-gate 		} else {
6848bbec428eSgjelinek 			saw_error = B_TRUE;
68497c478bd9Sstevel@tonic-gate 		}
68507c478bd9Sstevel@tonic-gate 	}
68517c478bd9Sstevel@tonic-gate 	/*
68527c478bd9Sstevel@tonic-gate 	 * Make sure we tried something and that the handle checks
68537c478bd9Sstevel@tonic-gate 	 * out, or we would get a false error trying to commit.
68547c478bd9Sstevel@tonic-gate 	 */
68557c478bd9Sstevel@tonic-gate 	if (need_to_commit && zonecfg_check_handle(handle) == Z_OK) {
68567c478bd9Sstevel@tonic-gate 		if ((cmd = alloc_cmd()) == NULL) {
6857bbec428eSgjelinek 			zone_perror(zone, Z_NOMEM, B_TRUE);
68587c478bd9Sstevel@tonic-gate 			return (Z_ERR);
68597c478bd9Sstevel@tonic-gate 		}
68607c478bd9Sstevel@tonic-gate 		cmd->cmd_argc = 0;
68617c478bd9Sstevel@tonic-gate 		cmd->cmd_argv[0] = NULL;
68627c478bd9Sstevel@tonic-gate 		commit_func(cmd);
68637c478bd9Sstevel@tonic-gate 		free_cmd(cmd);
68647c478bd9Sstevel@tonic-gate 		/*
68657c478bd9Sstevel@tonic-gate 		 * need_to_commit will get set back to FALSE if the
68667c478bd9Sstevel@tonic-gate 		 * configuration is saved successfully.
68677c478bd9Sstevel@tonic-gate 		 */
68687c478bd9Sstevel@tonic-gate 		if (need_to_commit) {
68697c478bd9Sstevel@tonic-gate 			if (force_exit) {
68707c478bd9Sstevel@tonic-gate 				zerr(gettext("Configuration not saved."));
68717c478bd9Sstevel@tonic-gate 				return (Z_ERR);
68727c478bd9Sstevel@tonic-gate 			}
6873bbec428eSgjelinek 			answer = ask_yesno(B_FALSE,
68747c478bd9Sstevel@tonic-gate 			    gettext("Configuration not saved; really quit"));
68757c478bd9Sstevel@tonic-gate 			if (answer == -1) {
68767c478bd9Sstevel@tonic-gate 				zerr(gettext("Configuration not saved."));
68777c478bd9Sstevel@tonic-gate 				return (Z_ERR);
68787c478bd9Sstevel@tonic-gate 			}
68797c478bd9Sstevel@tonic-gate 			if (answer != 1) {
6880bbec428eSgjelinek 				time_to_exit = B_FALSE;
68817c478bd9Sstevel@tonic-gate 				yyin = stdin;
68827c478bd9Sstevel@tonic-gate 				return (Z_REPEAT);
68837c478bd9Sstevel@tonic-gate 			}
68847c478bd9Sstevel@tonic-gate 		}
68857c478bd9Sstevel@tonic-gate 	}
68867c478bd9Sstevel@tonic-gate 	return ((need_to_commit || saw_error) ? Z_ERR : Z_OK);
68877c478bd9Sstevel@tonic-gate }
68887c478bd9Sstevel@tonic-gate 
68897c478bd9Sstevel@tonic-gate /*
68907c478bd9Sstevel@tonic-gate  * read_input() is the driver of this program.  It is a wrapper around
68917c478bd9Sstevel@tonic-gate  * yyparse(), printing appropriate prompts when needed, checking for
68927c478bd9Sstevel@tonic-gate  * exit conditions and reacting appropriately [the latter in its cleanup()
68937c478bd9Sstevel@tonic-gate  * helper function].
68947c478bd9Sstevel@tonic-gate  *
68957c478bd9Sstevel@tonic-gate  * Like most zonecfg functions, it returns Z_OK or Z_ERR, *or* Z_REPEAT
68967c478bd9Sstevel@tonic-gate  * so do_interactive() knows that we are not really done (i.e, we asked
68977c478bd9Sstevel@tonic-gate  * the user if we should really quit and the user said no).
68987c478bd9Sstevel@tonic-gate  */
68997c478bd9Sstevel@tonic-gate static int
69007c478bd9Sstevel@tonic-gate read_input()
69017c478bd9Sstevel@tonic-gate {
6902bbec428eSgjelinek 	boolean_t yyin_is_a_tty = isatty(fileno(yyin));
69037c478bd9Sstevel@tonic-gate 	/*
69047c478bd9Sstevel@tonic-gate 	 * The prompt is "e:z> " or "e:z:r> " where e is execname, z is zone
69057c478bd9Sstevel@tonic-gate 	 * and r is resource_scope: 5 is for the two ":"s + "> " + terminator.
69067c478bd9Sstevel@tonic-gate 	 */
69077c478bd9Sstevel@tonic-gate 	char prompt[MAXPATHLEN + ZONENAME_MAX + MAX_RT_STRLEN + 5], *line;
69087c478bd9Sstevel@tonic-gate 
69097c478bd9Sstevel@tonic-gate 	/* yyin should have been set to the appropriate (FILE *) if not stdin */
6910bbec428eSgjelinek 	newline_terminated = B_TRUE;
69117c478bd9Sstevel@tonic-gate 	for (;;) {
69127c478bd9Sstevel@tonic-gate 		if (yyin_is_a_tty) {
69137c478bd9Sstevel@tonic-gate 			if (newline_terminated) {
69147c478bd9Sstevel@tonic-gate 				if (global_scope)
69157c478bd9Sstevel@tonic-gate 					(void) snprintf(prompt, sizeof (prompt),
69167c478bd9Sstevel@tonic-gate 					    "%s:%s> ", execname, zone);
69177c478bd9Sstevel@tonic-gate 				else
69187c478bd9Sstevel@tonic-gate 					(void) snprintf(prompt, sizeof (prompt),
69197c478bd9Sstevel@tonic-gate 					    "%s:%s:%s> ", execname, zone,
69207c478bd9Sstevel@tonic-gate 					    rt_to_str(resource_scope));
69217c478bd9Sstevel@tonic-gate 			}
69227c478bd9Sstevel@tonic-gate 			/*
69237c478bd9Sstevel@tonic-gate 			 * If the user hits ^C then we want to catch it and
69247c478bd9Sstevel@tonic-gate 			 * start over.  If the user hits EOF then we want to
69257c478bd9Sstevel@tonic-gate 			 * bail out.
69267c478bd9Sstevel@tonic-gate 			 */
69277c478bd9Sstevel@tonic-gate 			line = gl_get_line(gl, prompt, NULL, -1);
69287c478bd9Sstevel@tonic-gate 			if (gl_return_status(gl) == GLR_SIGNAL) {
69297c478bd9Sstevel@tonic-gate 				gl_abandon_line(gl);
69307c478bd9Sstevel@tonic-gate 				continue;
69317c478bd9Sstevel@tonic-gate 			}
69327c478bd9Sstevel@tonic-gate 			if (line == NULL)
69337c478bd9Sstevel@tonic-gate 				break;
69347c478bd9Sstevel@tonic-gate 			(void) string_to_yyin(line);
69357c478bd9Sstevel@tonic-gate 			while (!feof(yyin))
69367c478bd9Sstevel@tonic-gate 				yyparse();
69377c478bd9Sstevel@tonic-gate 		} else {
69387c478bd9Sstevel@tonic-gate 			yyparse();
69397c478bd9Sstevel@tonic-gate 		}
69407c478bd9Sstevel@tonic-gate 		/* Bail out on an error in command file mode. */
69417c478bd9Sstevel@tonic-gate 		if (saw_error && cmd_file_mode && !interactive_mode)
6942bbec428eSgjelinek 			time_to_exit = B_TRUE;
69437c478bd9Sstevel@tonic-gate 		if (time_to_exit || (!yyin_is_a_tty && feof(yyin)))
69447c478bd9Sstevel@tonic-gate 			break;
69457c478bd9Sstevel@tonic-gate 	}
69467c478bd9Sstevel@tonic-gate 	return (cleanup());
69477c478bd9Sstevel@tonic-gate }
69487c478bd9Sstevel@tonic-gate 
69497c478bd9Sstevel@tonic-gate /*
69507c478bd9Sstevel@tonic-gate  * This function is used in the zonecfg-interactive-mode scenario: it just
69517c478bd9Sstevel@tonic-gate  * calls read_input() until we are done.
69527c478bd9Sstevel@tonic-gate  */
69537c478bd9Sstevel@tonic-gate 
69547c478bd9Sstevel@tonic-gate static int
69557c478bd9Sstevel@tonic-gate do_interactive(void)
69567c478bd9Sstevel@tonic-gate {
69577c478bd9Sstevel@tonic-gate 	int err;
69587c478bd9Sstevel@tonic-gate 
6959bbec428eSgjelinek 	interactive_mode = B_TRUE;
69607c478bd9Sstevel@tonic-gate 	if (!read_only_mode) {
69617c478bd9Sstevel@tonic-gate 		/*
69627c478bd9Sstevel@tonic-gate 		 * Try to set things up proactively in interactive mode, so
69637c478bd9Sstevel@tonic-gate 		 * that if the zone in question does not exist yet, we can
69647c478bd9Sstevel@tonic-gate 		 * provide the user with a clue.
69657c478bd9Sstevel@tonic-gate 		 */
6966bbec428eSgjelinek 		(void) initialize(B_FALSE);
69677c478bd9Sstevel@tonic-gate 	}
6968087719fdSdp 	do {
69697c478bd9Sstevel@tonic-gate 		err = read_input();
6970087719fdSdp 	} while (err == Z_REPEAT);
69717c478bd9Sstevel@tonic-gate 	return (err);
69727c478bd9Sstevel@tonic-gate }
69737c478bd9Sstevel@tonic-gate 
69747c478bd9Sstevel@tonic-gate /*
69757c478bd9Sstevel@tonic-gate  * cmd_file is slightly more complicated, as it has to open the command file
69767c478bd9Sstevel@tonic-gate  * and set yyin appropriately.  Once that is done, though, it just calls
69777c478bd9Sstevel@tonic-gate  * read_input(), and only once, since prompting is not possible.
69787c478bd9Sstevel@tonic-gate  */
69797c478bd9Sstevel@tonic-gate 
69807c478bd9Sstevel@tonic-gate static int
69817c478bd9Sstevel@tonic-gate cmd_file(char *file)
69827c478bd9Sstevel@tonic-gate {
69837c478bd9Sstevel@tonic-gate 	FILE *infile;
69847c478bd9Sstevel@tonic-gate 	int err;
69857c478bd9Sstevel@tonic-gate 	struct stat statbuf;
6986bbec428eSgjelinek 	boolean_t using_real_file = (strcmp(file, "-") != 0);
69877c478bd9Sstevel@tonic-gate 
69887c478bd9Sstevel@tonic-gate 	if (using_real_file) {
69897c478bd9Sstevel@tonic-gate 		/*
69907c478bd9Sstevel@tonic-gate 		 * zerr() prints a line number in cmd_file_mode, which we do
69917c478bd9Sstevel@tonic-gate 		 * not want here, so temporarily unset it.
69927c478bd9Sstevel@tonic-gate 		 */
6993bbec428eSgjelinek 		cmd_file_mode = B_FALSE;
69947c478bd9Sstevel@tonic-gate 		if ((infile = fopen(file, "r")) == NULL) {
69957c478bd9Sstevel@tonic-gate 			zerr(gettext("could not open file %s: %s"),
69967c478bd9Sstevel@tonic-gate 			    file, strerror(errno));
69977c478bd9Sstevel@tonic-gate 			return (Z_ERR);
69987c478bd9Sstevel@tonic-gate 		}
69997c478bd9Sstevel@tonic-gate 		if ((err = fstat(fileno(infile), &statbuf)) != 0) {
70007c478bd9Sstevel@tonic-gate 			zerr(gettext("could not stat file %s: %s"),
70017c478bd9Sstevel@tonic-gate 			    file, strerror(errno));
70027c478bd9Sstevel@tonic-gate 			err = Z_ERR;
70037c478bd9Sstevel@tonic-gate 			goto done;
70047c478bd9Sstevel@tonic-gate 		}
70057c478bd9Sstevel@tonic-gate 		if (!S_ISREG(statbuf.st_mode)) {
70067c478bd9Sstevel@tonic-gate 			zerr(gettext("%s is not a regular file."), file);
70077c478bd9Sstevel@tonic-gate 			err = Z_ERR;
70087c478bd9Sstevel@tonic-gate 			goto done;
70097c478bd9Sstevel@tonic-gate 		}
70107c478bd9Sstevel@tonic-gate 		yyin = infile;
7011bbec428eSgjelinek 		cmd_file_mode = B_TRUE;
7012bbec428eSgjelinek 		ok_to_prompt = B_FALSE;
70137c478bd9Sstevel@tonic-gate 	} else {
70147c478bd9Sstevel@tonic-gate 		/*
70157c478bd9Sstevel@tonic-gate 		 * "-f -" is essentially the same as interactive mode,
70167c478bd9Sstevel@tonic-gate 		 * so treat it that way.
70177c478bd9Sstevel@tonic-gate 		 */
7018bbec428eSgjelinek 		interactive_mode = B_TRUE;
70197c478bd9Sstevel@tonic-gate 	}
70207c478bd9Sstevel@tonic-gate 	/* Z_REPEAT is for interactive mode; treat it like Z_ERR here. */
70217c478bd9Sstevel@tonic-gate 	if ((err = read_input()) == Z_REPEAT)
70227c478bd9Sstevel@tonic-gate 		err = Z_ERR;
70237c478bd9Sstevel@tonic-gate done:
70247c478bd9Sstevel@tonic-gate 	if (using_real_file)
70257c478bd9Sstevel@tonic-gate 		(void) fclose(infile);
70267c478bd9Sstevel@tonic-gate 	return (err);
70277c478bd9Sstevel@tonic-gate }
70287c478bd9Sstevel@tonic-gate 
70297c478bd9Sstevel@tonic-gate /*
70307c478bd9Sstevel@tonic-gate  * Since yacc is based on reading from a (FILE *) whereas what we get from
70317c478bd9Sstevel@tonic-gate  * the command line is in argv format, we need to convert when the user
70327c478bd9Sstevel@tonic-gate  * gives us commands directly from the command line.  That is done here by
70337c478bd9Sstevel@tonic-gate  * concatenating the argv list into a space-separated string, writing it
70347c478bd9Sstevel@tonic-gate  * to a temp file, and rewinding the file so yyin can be set to it.  Then
70357c478bd9Sstevel@tonic-gate  * we call read_input(), and only once, since prompting about whether to
70367c478bd9Sstevel@tonic-gate  * continue or quit would make no sense in this context.
70377c478bd9Sstevel@tonic-gate  */
70387c478bd9Sstevel@tonic-gate 
70397c478bd9Sstevel@tonic-gate static int
70407c478bd9Sstevel@tonic-gate one_command_at_a_time(int argc, char *argv[])
70417c478bd9Sstevel@tonic-gate {
70427c478bd9Sstevel@tonic-gate 	char *command;
70437c478bd9Sstevel@tonic-gate 	size_t len = 2; /* terminal \n\0 */
70447c478bd9Sstevel@tonic-gate 	int i, err;
70457c478bd9Sstevel@tonic-gate 
70467c478bd9Sstevel@tonic-gate 	for (i = 0; i < argc; i++)
70477c478bd9Sstevel@tonic-gate 		len += strlen(argv[i]) + 1;
70487c478bd9Sstevel@tonic-gate 	if ((command = malloc(len)) == NULL) {
7049bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
70507c478bd9Sstevel@tonic-gate 		return (Z_ERR);
70517c478bd9Sstevel@tonic-gate 	}
70527c478bd9Sstevel@tonic-gate 	(void) strlcpy(command, argv[0], len);
70537c478bd9Sstevel@tonic-gate 	for (i = 1; i < argc; i++) {
70547c478bd9Sstevel@tonic-gate 		(void) strlcat(command, " ", len);
70557c478bd9Sstevel@tonic-gate 		(void) strlcat(command, argv[i], len);
70567c478bd9Sstevel@tonic-gate 	}
70577c478bd9Sstevel@tonic-gate 	(void) strlcat(command, "\n", len);
70587c478bd9Sstevel@tonic-gate 	err = string_to_yyin(command);
70597c478bd9Sstevel@tonic-gate 	free(command);
70607c478bd9Sstevel@tonic-gate 	if (err != Z_OK)
70617c478bd9Sstevel@tonic-gate 		return (err);
70627c478bd9Sstevel@tonic-gate 	while (!feof(yyin))
70637c478bd9Sstevel@tonic-gate 		yyparse();
70647c478bd9Sstevel@tonic-gate 	return (cleanup());
70657c478bd9Sstevel@tonic-gate }
70667c478bd9Sstevel@tonic-gate 
70677c478bd9Sstevel@tonic-gate static char *
70687c478bd9Sstevel@tonic-gate get_execbasename(char *execfullname)
70697c478bd9Sstevel@tonic-gate {
70707c478bd9Sstevel@tonic-gate 	char *last_slash, *execbasename;
70717c478bd9Sstevel@tonic-gate 
70727c478bd9Sstevel@tonic-gate 	/* guard against '/' at end of command invocation */
70737c478bd9Sstevel@tonic-gate 	for (;;) {
70747c478bd9Sstevel@tonic-gate 		last_slash = strrchr(execfullname, '/');
70757c478bd9Sstevel@tonic-gate 		if (last_slash == NULL) {
70767c478bd9Sstevel@tonic-gate 			execbasename = execfullname;
70777c478bd9Sstevel@tonic-gate 			break;
70787c478bd9Sstevel@tonic-gate 		} else {
70797c478bd9Sstevel@tonic-gate 			execbasename = last_slash + 1;
70807c478bd9Sstevel@tonic-gate 			if (*execbasename == '\0') {
70817c478bd9Sstevel@tonic-gate 				*last_slash = '\0';
70827c478bd9Sstevel@tonic-gate 				continue;
70837c478bd9Sstevel@tonic-gate 			}
70847c478bd9Sstevel@tonic-gate 			break;
70857c478bd9Sstevel@tonic-gate 		}
70867c478bd9Sstevel@tonic-gate 	}
70877c478bd9Sstevel@tonic-gate 	return (execbasename);
70887c478bd9Sstevel@tonic-gate }
70897c478bd9Sstevel@tonic-gate 
70907c478bd9Sstevel@tonic-gate int
70917c478bd9Sstevel@tonic-gate main(int argc, char *argv[])
70927c478bd9Sstevel@tonic-gate {
70937c478bd9Sstevel@tonic-gate 	int err, arg;
7094555afedfScarlsonj 	struct stat st;
70957c478bd9Sstevel@tonic-gate 
70967c478bd9Sstevel@tonic-gate 	/* This must be before anything goes to stdout. */
70977c478bd9Sstevel@tonic-gate 	setbuf(stdout, NULL);
70987c478bd9Sstevel@tonic-gate 
7099bbec428eSgjelinek 	saw_error = B_FALSE;
7100bbec428eSgjelinek 	cmd_file_mode = B_FALSE;
71017c478bd9Sstevel@tonic-gate 	execname = get_execbasename(argv[0]);
71027c478bd9Sstevel@tonic-gate 
71037c478bd9Sstevel@tonic-gate 	(void) setlocale(LC_ALL, "");
71047c478bd9Sstevel@tonic-gate 	(void) textdomain(TEXT_DOMAIN);
71057c478bd9Sstevel@tonic-gate 
71067c478bd9Sstevel@tonic-gate 	if (getzoneid() != GLOBAL_ZONEID) {
71077c478bd9Sstevel@tonic-gate 		zerr(gettext("%s can only be run from the global zone."),
71087c478bd9Sstevel@tonic-gate 		    execname);
71097c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
71107c478bd9Sstevel@tonic-gate 	}
71117c478bd9Sstevel@tonic-gate 
71127c478bd9Sstevel@tonic-gate 	if (argc < 2) {
7113bbec428eSgjelinek 		usage(B_FALSE, HELP_USAGE | HELP_SUBCMDS);
71147c478bd9Sstevel@tonic-gate 		exit(Z_USAGE);
71157c478bd9Sstevel@tonic-gate 	}
71167c478bd9Sstevel@tonic-gate 	if (strcmp(argv[1], cmd_to_str(CMD_HELP)) == 0) {
71177c478bd9Sstevel@tonic-gate 		(void) one_command_at_a_time(argc - 1, &(argv[1]));
71187c478bd9Sstevel@tonic-gate 		exit(Z_OK);
71197c478bd9Sstevel@tonic-gate 	}
71207c478bd9Sstevel@tonic-gate 
7121555afedfScarlsonj 	while ((arg = getopt(argc, argv, "?f:R:z:")) != EOF) {
71227c478bd9Sstevel@tonic-gate 		switch (arg) {
71237c478bd9Sstevel@tonic-gate 		case '?':
71247c478bd9Sstevel@tonic-gate 			if (optopt == '?')
7125bbec428eSgjelinek 				usage(B_TRUE, HELP_USAGE | HELP_SUBCMDS);
71267c478bd9Sstevel@tonic-gate 			else
7127bbec428eSgjelinek 				usage(B_FALSE, HELP_USAGE);
71287c478bd9Sstevel@tonic-gate 			exit(Z_USAGE);
71297c478bd9Sstevel@tonic-gate 			/* NOTREACHED */
71307c478bd9Sstevel@tonic-gate 		case 'f':
71317c478bd9Sstevel@tonic-gate 			cmd_file_name = optarg;
7132bbec428eSgjelinek 			cmd_file_mode = B_TRUE;
71337c478bd9Sstevel@tonic-gate 			break;
7134555afedfScarlsonj 		case 'R':
7135555afedfScarlsonj 			if (*optarg != '/') {
7136555afedfScarlsonj 				zerr(gettext("root path must be absolute: %s"),
7137555afedfScarlsonj 				    optarg);
7138555afedfScarlsonj 				exit(Z_USAGE);
7139555afedfScarlsonj 			}
7140555afedfScarlsonj 			if (stat(optarg, &st) == -1 || !S_ISDIR(st.st_mode)) {
7141555afedfScarlsonj 				zerr(gettext(
7142555afedfScarlsonj 				    "root path must be a directory: %s"),
7143555afedfScarlsonj 				    optarg);
7144555afedfScarlsonj 				exit(Z_USAGE);
7145555afedfScarlsonj 			}
7146555afedfScarlsonj 			zonecfg_set_root(optarg);
7147555afedfScarlsonj 			break;
71487c478bd9Sstevel@tonic-gate 		case 'z':
71490209230bSgjelinek 			if (strcmp(optarg, GLOBAL_ZONENAME) == 0) {
7150bbec428eSgjelinek 				global_zone = B_TRUE;
71510209230bSgjelinek 			} else if (zonecfg_validate_zonename(optarg) != Z_OK) {
7152bbec428eSgjelinek 				zone_perror(optarg, Z_BOGUS_ZONE_NAME, B_TRUE);
7153bbec428eSgjelinek 				usage(B_FALSE, HELP_SYNTAX);
7154087719fdSdp 				exit(Z_USAGE);
7155087719fdSdp 			}
7156087719fdSdp 			(void) strlcpy(zone, optarg, sizeof (zone));
7157087719fdSdp 			(void) strlcpy(revert_zone, optarg, sizeof (zone));
71587c478bd9Sstevel@tonic-gate 			break;
71597c478bd9Sstevel@tonic-gate 		default:
7160bbec428eSgjelinek 			usage(B_FALSE, HELP_USAGE);
71617c478bd9Sstevel@tonic-gate 			exit(Z_USAGE);
71627c478bd9Sstevel@tonic-gate 		}
71637c478bd9Sstevel@tonic-gate 	}
71647c478bd9Sstevel@tonic-gate 
7165087719fdSdp 	if (optind > argc || strcmp(zone, "") == 0) {
7166bbec428eSgjelinek 		usage(B_FALSE, HELP_USAGE);
71677c478bd9Sstevel@tonic-gate 		exit(Z_USAGE);
71687c478bd9Sstevel@tonic-gate 	}
71697c478bd9Sstevel@tonic-gate 
7170087719fdSdp 	if ((err = zonecfg_access(zone, W_OK)) == Z_OK) {
7171bbec428eSgjelinek 		read_only_mode = B_FALSE;
7172087719fdSdp 	} else if (err == Z_ACCES) {
7173bbec428eSgjelinek 		read_only_mode = B_TRUE;
71747c478bd9Sstevel@tonic-gate 		/* skip this message in one-off from command line mode */
71757c478bd9Sstevel@tonic-gate 		if (optind == argc)
71767c478bd9Sstevel@tonic-gate 			(void) fprintf(stderr, gettext("WARNING: you do not "
71777c478bd9Sstevel@tonic-gate 			    "have write access to this zone's configuration "
71787c478bd9Sstevel@tonic-gate 			    "file;\ngoing into read-only mode.\n"));
7179087719fdSdp 	} else {
7180087719fdSdp 		fprintf(stderr, "%s: Could not access zone configuration "
7181087719fdSdp 		    "store: %s\n", execname, zonecfg_strerror(err));
7182087719fdSdp 		exit(Z_ERR);
71837c478bd9Sstevel@tonic-gate 	}
71847c478bd9Sstevel@tonic-gate 
71857c478bd9Sstevel@tonic-gate 	if ((handle = zonecfg_init_handle()) == NULL) {
7186bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
71877c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
71887c478bd9Sstevel@tonic-gate 	}
71897c478bd9Sstevel@tonic-gate 
71907c478bd9Sstevel@tonic-gate 	/*
71917c478bd9Sstevel@tonic-gate 	 * This may get set back to FALSE again in cmd_file() if cmd_file_name
71927c478bd9Sstevel@tonic-gate 	 * is a "real" file as opposed to "-" (i.e. meaning use stdin).
71937c478bd9Sstevel@tonic-gate 	 */
71947c478bd9Sstevel@tonic-gate 	if (isatty(STDIN_FILENO))
7195bbec428eSgjelinek 		ok_to_prompt = B_TRUE;
71967c478bd9Sstevel@tonic-gate 	if ((gl = new_GetLine(MAX_LINE_LEN, MAX_CMD_HIST)) == NULL)
71977c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
71987c478bd9Sstevel@tonic-gate 	if (gl_customize_completion(gl, NULL, cmd_cpl_fn) != 0)
71997c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
72007c478bd9Sstevel@tonic-gate 	(void) sigset(SIGINT, SIG_IGN);
72017c478bd9Sstevel@tonic-gate 	if (optind == argc) {
72027c478bd9Sstevel@tonic-gate 		if (!cmd_file_mode)
72037c478bd9Sstevel@tonic-gate 			err = do_interactive();
72047c478bd9Sstevel@tonic-gate 		else
72057c478bd9Sstevel@tonic-gate 			err = cmd_file(cmd_file_name);
72067c478bd9Sstevel@tonic-gate 	} else {
72077c478bd9Sstevel@tonic-gate 		err = one_command_at_a_time(argc - optind, &(argv[optind]));
72087c478bd9Sstevel@tonic-gate 	}
72097c478bd9Sstevel@tonic-gate 	zonecfg_fini_handle(handle);
72109acbbeafSnn35248 	if (brand != NULL)
72119acbbeafSnn35248 		brand_close(brand);
72127c478bd9Sstevel@tonic-gate 	(void) del_GetLine(gl);
72137c478bd9Sstevel@tonic-gate 	return (err);
72147c478bd9Sstevel@tonic-gate }
7215