xref: /illumos-gate/usr/src/cmd/zonecfg/zonecfg.c (revision bbf21555)
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.
24f93d2c19SAlexander Eremin  * Copyright 2014 Nexenta Systems, Inc.  All rights reserved.
25944b13ecSGary Mills  * Copyright 2014 Gary Mills
26c8236ea6SBrian Bennett  * Copyright 2019 Joyent, Inc.
277c478bd9Sstevel@tonic-gate  */
287c478bd9Sstevel@tonic-gate 
297c478bd9Sstevel@tonic-gate /*
307c478bd9Sstevel@tonic-gate  * zonecfg is a lex/yacc based command interpreter used to manage zone
317c478bd9Sstevel@tonic-gate  * configurations.  The lexer (see zonecfg_lex.l) builds up tokens, which
327c478bd9Sstevel@tonic-gate  * the grammar (see zonecfg_grammar.y) builds up into commands, some of
337c478bd9Sstevel@tonic-gate  * which takes resources and/or properties as arguments.  See the block
347c478bd9Sstevel@tonic-gate  * comments near the end of zonecfg_grammar.y for how the data structures
357c478bd9Sstevel@tonic-gate  * which keep track of these resources and properties are built up.
367c478bd9Sstevel@tonic-gate  *
377c478bd9Sstevel@tonic-gate  * The resource/property data structures are inserted into a command
387c478bd9Sstevel@tonic-gate  * structure (see zonecfg.h), which also keeps track of command names,
397c478bd9Sstevel@tonic-gate  * miscellaneous arguments, and function handlers.  The grammar selects
407c478bd9Sstevel@tonic-gate  * the appropriate function handler, each of which takes a pointer to a
417c478bd9Sstevel@tonic-gate  * command structure as its sole argument, and invokes it.  The grammar
427c478bd9Sstevel@tonic-gate  * itself is "entered" (a la the Matrix) by yyparse(), which is called
437c478bd9Sstevel@tonic-gate  * from read_input(), our main driving function.  That in turn is called
447c478bd9Sstevel@tonic-gate  * by one of do_interactive(), cmd_file() or one_command_at_a_time(), each
457c478bd9Sstevel@tonic-gate  * of which is called from main() depending on how the program was invoked.
467c478bd9Sstevel@tonic-gate  *
477c478bd9Sstevel@tonic-gate  * The rest of this module consists of the various function handlers and
487c478bd9Sstevel@tonic-gate  * their helper functions.  Some of these functions, particularly the
497c478bd9Sstevel@tonic-gate  * X_to_str() functions, which maps command, resource and property numbers
507c478bd9Sstevel@tonic-gate  * to strings, are used quite liberally, as doing so results in a better
517c478bd9Sstevel@tonic-gate  * program w/rt I18N, reducing the need for translation notes.
527c478bd9Sstevel@tonic-gate  */
537c478bd9Sstevel@tonic-gate 
547c478bd9Sstevel@tonic-gate #include <sys/mntent.h>
557c478bd9Sstevel@tonic-gate #include <sys/varargs.h>
567c478bd9Sstevel@tonic-gate #include <sys/sysmacros.h>
57d2a70789SRichard Lowe #include <sys/secflags.h>
587c478bd9Sstevel@tonic-gate 
597c478bd9Sstevel@tonic-gate #include <errno.h>
609acbbeafSnn35248 #include <fcntl.h>
617c478bd9Sstevel@tonic-gate #include <strings.h>
627c478bd9Sstevel@tonic-gate #include <unistd.h>
637c478bd9Sstevel@tonic-gate #include <ctype.h>
647c478bd9Sstevel@tonic-gate #include <stdlib.h>
657c478bd9Sstevel@tonic-gate #include <assert.h>
667c478bd9Sstevel@tonic-gate #include <sys/stat.h>
677c478bd9Sstevel@tonic-gate #include <zone.h>
687c478bd9Sstevel@tonic-gate #include <arpa/inet.h>
697c478bd9Sstevel@tonic-gate #include <netdb.h>
707c478bd9Sstevel@tonic-gate #include <locale.h>
717c478bd9Sstevel@tonic-gate #include <libintl.h>
727c478bd9Sstevel@tonic-gate #include <alloca.h>
737c478bd9Sstevel@tonic-gate #include <signal.h>
749acbbeafSnn35248 #include <wait.h>
757c478bd9Sstevel@tonic-gate #include <libtecla.h>
76fa9e4066Sahrens #include <libzfs.h>
779acbbeafSnn35248 #include <sys/brand.h>
789acbbeafSnn35248 #include <libbrand.h>
795679c89fSjv227347 #include <sys/systeminfo.h>
80c9f134eaSjv227347 #include <libdladm.h>
81c9f134eaSjv227347 #include <libinetutil.h>
82cb8a054bSGlenn Faden #include <pwd.h>
83550b6e40SSowmini Varadhan #include <inet/ip.h>
847c478bd9Sstevel@tonic-gate 
857c478bd9Sstevel@tonic-gate #include <libzonecfg.h>
867c478bd9Sstevel@tonic-gate #include "zonecfg.h"
877c478bd9Sstevel@tonic-gate 
887c478bd9Sstevel@tonic-gate #if !defined(TEXT_DOMAIN)		/* should be defined by cc -D */
897c478bd9Sstevel@tonic-gate #define	TEXT_DOMAIN	"SYS_TEST"	/* Use this only if it wasn't */
907c478bd9Sstevel@tonic-gate #endif
917c478bd9Sstevel@tonic-gate 
927c478bd9Sstevel@tonic-gate #define	PAGER	"/usr/bin/more"
939acbbeafSnn35248 #define	EXEC_PREFIX	"exec "
949acbbeafSnn35248 #define	EXEC_LEN	(strlen(EXEC_PREFIX))
957c478bd9Sstevel@tonic-gate 
967c478bd9Sstevel@tonic-gate struct help {
977c478bd9Sstevel@tonic-gate 	uint_t	cmd_num;
987c478bd9Sstevel@tonic-gate 	char	*cmd_name;
997c478bd9Sstevel@tonic-gate 	uint_t	flags;
1007c478bd9Sstevel@tonic-gate 	char	*short_usage;
1017c478bd9Sstevel@tonic-gate };
1027c478bd9Sstevel@tonic-gate 
1037c478bd9Sstevel@tonic-gate extern int yyparse(void);
1047c478bd9Sstevel@tonic-gate extern int lex_lineno;
1057c478bd9Sstevel@tonic-gate 
1067c478bd9Sstevel@tonic-gate #define	MAX_LINE_LEN	1024
1077c478bd9Sstevel@tonic-gate #define	MAX_CMD_HIST	1024
1089acbbeafSnn35248 #define	MAX_CMD_LEN	1024
1097c478bd9Sstevel@tonic-gate 
1100209230bSgjelinek #define	ONE_MB		1048576
1110209230bSgjelinek 
1127c478bd9Sstevel@tonic-gate /*
1137c478bd9Sstevel@tonic-gate  * Each SHELP_ should be a simple string.
1147c478bd9Sstevel@tonic-gate  */
1157c478bd9Sstevel@tonic-gate 
1167c478bd9Sstevel@tonic-gate #define	SHELP_ADD	"add <resource-type>\n\t(global scope)\n" \
1177c478bd9Sstevel@tonic-gate 	"add <property-name> <property-value>\n\t(resource scope)"
1187c478bd9Sstevel@tonic-gate #define	SHELP_CANCEL	"cancel"
1190209230bSgjelinek #define	SHELP_CLEAR	"clear <property-name>"
1207c478bd9Sstevel@tonic-gate #define	SHELP_COMMIT	"commit"
121ee519a1fSgjelinek #define	SHELP_CREATE	"create [-F] [ -a <path> | -b | -t <template> ]"
1227c478bd9Sstevel@tonic-gate #define	SHELP_DELETE	"delete [-F]"
1237c478bd9Sstevel@tonic-gate #define	SHELP_END	"end"
1247c478bd9Sstevel@tonic-gate #define	SHELP_EXIT	"exit [-F]"
1257c478bd9Sstevel@tonic-gate #define	SHELP_EXPORT	"export [-f output-file]"
1267c478bd9Sstevel@tonic-gate #define	SHELP_HELP	"help [commands] [syntax] [usage] [<command-name>]"
1277c478bd9Sstevel@tonic-gate #define	SHELP_INFO	"info [<resource-type> [property-name=property-value]*]"
1280209230bSgjelinek #define	SHELP_REMOVE	"remove [-F] <resource-type> " \
1290209230bSgjelinek 	"[ <property-name>=<property-value> ]*\n" \
1300209230bSgjelinek 	"\t(global scope)\n" \
1310209230bSgjelinek 	"remove <property-name> <property-value>\n" \
1320209230bSgjelinek 	"\t(resource scope)"
1337c478bd9Sstevel@tonic-gate #define	SHELP_REVERT	"revert [-F]"
1347c478bd9Sstevel@tonic-gate #define	SHELP_SELECT	"select <resource-type> { <property-name>=" \
1357c478bd9Sstevel@tonic-gate 	"<property-value> }"
1367c478bd9Sstevel@tonic-gate #define	SHELP_SET	"set <property-name>=<property-value>"
1377c478bd9Sstevel@tonic-gate #define	SHELP_VERIFY	"verify"
1387c478bd9Sstevel@tonic-gate 
1397c478bd9Sstevel@tonic-gate static struct help helptab[] = {
1407c478bd9Sstevel@tonic-gate 	{ CMD_ADD,	"add",		HELP_RES_PROPS,	SHELP_ADD, },
1417c478bd9Sstevel@tonic-gate 	{ CMD_CANCEL,	"cancel",	0,		SHELP_CANCEL, },
1420209230bSgjelinek 	{ CMD_CLEAR,	"clear",	HELP_PROPS,	SHELP_CLEAR, },
1437c478bd9Sstevel@tonic-gate 	{ CMD_COMMIT,	"commit",	0,		SHELP_COMMIT, },
1447c478bd9Sstevel@tonic-gate 	{ CMD_CREATE,	"create",	0,		SHELP_CREATE, },
1457c478bd9Sstevel@tonic-gate 	{ CMD_DELETE,	"delete",	0,		SHELP_DELETE, },
1467c478bd9Sstevel@tonic-gate 	{ CMD_END,	"end",		0,		SHELP_END, },
1477c478bd9Sstevel@tonic-gate 	{ CMD_EXIT,	"exit",		0,		SHELP_EXIT, },
1487c478bd9Sstevel@tonic-gate 	{ CMD_EXPORT,	"export",	0,		SHELP_EXPORT, },
1497c478bd9Sstevel@tonic-gate 	{ CMD_HELP,	"help",		0,		SHELP_HELP },
1507c478bd9Sstevel@tonic-gate 	{ CMD_INFO,	"info",		HELP_RES_PROPS,	SHELP_INFO, },
1517c478bd9Sstevel@tonic-gate 	{ CMD_REMOVE,	"remove",	HELP_RES_PROPS,	SHELP_REMOVE, },
1527c478bd9Sstevel@tonic-gate 	{ CMD_REVERT,	"revert",	0,		SHELP_REVERT, },
1537c478bd9Sstevel@tonic-gate 	{ CMD_SELECT,	"select",	HELP_RES_PROPS,	SHELP_SELECT, },
1547c478bd9Sstevel@tonic-gate 	{ CMD_SET,	"set",		HELP_PROPS,	SHELP_SET, },
1557c478bd9Sstevel@tonic-gate 	{ CMD_VERIFY,	"verify",	0,		SHELP_VERIFY, },
1567c478bd9Sstevel@tonic-gate 	{ 0 },
1577c478bd9Sstevel@tonic-gate };
1587c478bd9Sstevel@tonic-gate 
1597c478bd9Sstevel@tonic-gate #define	MAX_RT_STRLEN	16
1607c478bd9Sstevel@tonic-gate 
1617c478bd9Sstevel@tonic-gate /* These *must* match the order of the RT_ define's from zonecfg.h */
162c94c1ef0Sjv227347 char *res_types[] = {
1637c478bd9Sstevel@tonic-gate 	"unknown",
164087719fdSdp 	"zonename",
1657c478bd9Sstevel@tonic-gate 	"zonepath",
1667c478bd9Sstevel@tonic-gate 	"autoboot",
1677c478bd9Sstevel@tonic-gate 	"pool",
1687c478bd9Sstevel@tonic-gate 	"fs",
1697c478bd9Sstevel@tonic-gate 	"net",
1707c478bd9Sstevel@tonic-gate 	"device",
1717c478bd9Sstevel@tonic-gate 	"rctl",
1727c478bd9Sstevel@tonic-gate 	"attr",
173fa9e4066Sahrens 	"dataset",
174ffbafc53Scomay 	"limitpriv",
1753f2f09c1Sdp 	"bootargs",
1769acbbeafSnn35248 	"brand",
1770209230bSgjelinek 	"dedicated-cpu",
1780209230bSgjelinek 	"capped-memory",
1790209230bSgjelinek 	ALIAS_MAXLWPS,
1800209230bSgjelinek 	ALIAS_MAXSHMMEM,
1810209230bSgjelinek 	ALIAS_MAXSHMIDS,
1820209230bSgjelinek 	ALIAS_MAXMSGIDS,
1830209230bSgjelinek 	ALIAS_MAXSEMIDS,
1840209230bSgjelinek 	ALIAS_SHARES,
1850209230bSgjelinek 	"scheduling-class",
186f4b3ec61Sdh155122 	"ip-type",
187c97ad5cdSakolb 	"capped-cpu",
1885679c89fSjv227347 	"hostid",
189cb8a054bSGlenn Faden 	"admin",
1900fbb751dSJohn Levon 	"fs-allowed",
191ff19e029SMenno Lageman 	ALIAS_MAXPROCS,
192d2a70789SRichard Lowe 	"security-flags",
1937c478bd9Sstevel@tonic-gate 	NULL
1947c478bd9Sstevel@tonic-gate };
1957c478bd9Sstevel@tonic-gate 
1967c478bd9Sstevel@tonic-gate /* These *must* match the order of the PT_ define's from zonecfg.h */
197c94c1ef0Sjv227347 char *prop_types[] = {
1987c478bd9Sstevel@tonic-gate 	"unknown",
199087719fdSdp 	"zonename",
2007c478bd9Sstevel@tonic-gate 	"zonepath",
2017c478bd9Sstevel@tonic-gate 	"autoboot",
2027c478bd9Sstevel@tonic-gate 	"pool",
2037c478bd9Sstevel@tonic-gate 	"dir",
2047c478bd9Sstevel@tonic-gate 	"special",
2057c478bd9Sstevel@tonic-gate 	"type",
2067c478bd9Sstevel@tonic-gate 	"options",
2077c478bd9Sstevel@tonic-gate 	"address",
2087c478bd9Sstevel@tonic-gate 	"physical",
2097c478bd9Sstevel@tonic-gate 	"name",
2107c478bd9Sstevel@tonic-gate 	"value",
2117c478bd9Sstevel@tonic-gate 	"match",
2127c478bd9Sstevel@tonic-gate 	"priv",
2137c478bd9Sstevel@tonic-gate 	"limit",
2147c478bd9Sstevel@tonic-gate 	"action",
2157c478bd9Sstevel@tonic-gate 	"raw",
216ffbafc53Scomay 	"limitpriv",
2173f2f09c1Sdp 	"bootargs",
2189acbbeafSnn35248 	"brand",
2190209230bSgjelinek 	"ncpus",
2200209230bSgjelinek 	"importance",
2210209230bSgjelinek 	"swap",
2220209230bSgjelinek 	"locked",
2230209230bSgjelinek 	ALIAS_SHARES,
2240209230bSgjelinek 	ALIAS_MAXLWPS,
2250209230bSgjelinek 	ALIAS_MAXSHMMEM,
2260209230bSgjelinek 	ALIAS_MAXSHMIDS,
2270209230bSgjelinek 	ALIAS_MAXMSGIDS,
2280209230bSgjelinek 	ALIAS_MAXSEMIDS,
2290209230bSgjelinek 	ALIAS_MAXLOCKEDMEM,
2300209230bSgjelinek 	ALIAS_MAXSWAP,
2310209230bSgjelinek 	"scheduling-class",
232f4b3ec61Sdh155122 	"ip-type",
233de860bd9Sgfaden 	"defrouter",
2345679c89fSjv227347 	"hostid",
235cb8a054bSGlenn Faden 	"user",
236cb8a054bSGlenn Faden 	"auths",
2370fbb751dSJohn Levon 	"fs-allowed",
238ff19e029SMenno Lageman 	ALIAS_MAXPROCS,
239550b6e40SSowmini Varadhan 	"allowed-address",
240d2a70789SRichard Lowe 	"default",
241d2a70789SRichard Lowe 	"lower",
242d2a70789SRichard Lowe 	"upper",
2437c478bd9Sstevel@tonic-gate 	NULL
2447c478bd9Sstevel@tonic-gate };
2457c478bd9Sstevel@tonic-gate 
246ffbafc53Scomay /* These *must* match the order of the PROP_VAL_ define's from zonecfg.h */
2477c478bd9Sstevel@tonic-gate static char *prop_val_types[] = {
2487c478bd9Sstevel@tonic-gate 	"simple",
2497c478bd9Sstevel@tonic-gate 	"complex",
2507c478bd9Sstevel@tonic-gate 	"list",
2517c478bd9Sstevel@tonic-gate };
2527c478bd9Sstevel@tonic-gate 
2537c478bd9Sstevel@tonic-gate /*
2547c478bd9Sstevel@tonic-gate  * The various _cmds[] lists below are for command tab-completion.
2557c478bd9Sstevel@tonic-gate  */
2567c478bd9Sstevel@tonic-gate 
2577c478bd9Sstevel@tonic-gate /*
2587c478bd9Sstevel@tonic-gate  * remove has a space afterwards because it has qualifiers; the other commands
2590209230bSgjelinek  * that have qualifiers (add, select, etc.) don't need a space here because
2607c478bd9Sstevel@tonic-gate  * they have their own _cmds[] lists below.
2617c478bd9Sstevel@tonic-gate  */
2627c478bd9Sstevel@tonic-gate static const char *global_scope_cmds[] = {
2637c478bd9Sstevel@tonic-gate 	"add",
2640209230bSgjelinek 	"clear",
2657c478bd9Sstevel@tonic-gate 	"commit",
2667c478bd9Sstevel@tonic-gate 	"create",
2677c478bd9Sstevel@tonic-gate 	"delete",
2687c478bd9Sstevel@tonic-gate 	"exit",
2697c478bd9Sstevel@tonic-gate 	"export",
2707c478bd9Sstevel@tonic-gate 	"help",
2717c478bd9Sstevel@tonic-gate 	"info",
2727c478bd9Sstevel@tonic-gate 	"remove ",
2737c478bd9Sstevel@tonic-gate 	"revert",
2747c478bd9Sstevel@tonic-gate 	"select",
2757c478bd9Sstevel@tonic-gate 	"set",
2767c478bd9Sstevel@tonic-gate 	"verify",
2777c478bd9Sstevel@tonic-gate 	NULL
2787c478bd9Sstevel@tonic-gate };
2797c478bd9Sstevel@tonic-gate 
2807c478bd9Sstevel@tonic-gate static const char *add_cmds[] = {
2817c478bd9Sstevel@tonic-gate 	"add fs",
2827c478bd9Sstevel@tonic-gate 	"add net",
2837c478bd9Sstevel@tonic-gate 	"add device",
2847c478bd9Sstevel@tonic-gate 	"add rctl",
2857c478bd9Sstevel@tonic-gate 	"add attr",
286fa9e4066Sahrens 	"add dataset",
2870209230bSgjelinek 	"add dedicated-cpu",
288c97ad5cdSakolb 	"add capped-cpu",
2890209230bSgjelinek 	"add capped-memory",
290cb8a054bSGlenn Faden 	"add admin",
291d2a70789SRichard Lowe 	"add security-flags",
2920209230bSgjelinek 	NULL
2930209230bSgjelinek };
2940209230bSgjelinek 
2950209230bSgjelinek static const char *clear_cmds[] = {
2960209230bSgjelinek 	"clear autoboot",
2970209230bSgjelinek 	"clear pool",
2980209230bSgjelinek 	"clear limitpriv",
2990209230bSgjelinek 	"clear bootargs",
3000209230bSgjelinek 	"clear scheduling-class",
301f4b3ec61Sdh155122 	"clear ip-type",
3020209230bSgjelinek 	"clear " ALIAS_MAXLWPS,
3030209230bSgjelinek 	"clear " ALIAS_MAXSHMMEM,
3040209230bSgjelinek 	"clear " ALIAS_MAXSHMIDS,
3050209230bSgjelinek 	"clear " ALIAS_MAXMSGIDS,
3060209230bSgjelinek 	"clear " ALIAS_MAXSEMIDS,
3070209230bSgjelinek 	"clear " ALIAS_SHARES,
308ff19e029SMenno Lageman 	"clear " ALIAS_MAXPROCS,
3097c478bd9Sstevel@tonic-gate 	NULL
3107c478bd9Sstevel@tonic-gate };
3117c478bd9Sstevel@tonic-gate 
3129e7542f4Sdp static const char *remove_cmds[] = {
3139e7542f4Sdp 	"remove fs ",
3149e7542f4Sdp 	"remove net ",
3159e7542f4Sdp 	"remove device ",
3169e7542f4Sdp 	"remove rctl ",
3179e7542f4Sdp 	"remove attr ",
3189e7542f4Sdp 	"remove dataset ",
3190209230bSgjelinek 	"remove dedicated-cpu ",
320c97ad5cdSakolb 	"remove capped-cpu ",
3210209230bSgjelinek 	"remove capped-memory ",
322cb8a054bSGlenn Faden 	"remove admin ",
323d2a70789SRichard Lowe 	"remove security-flags",
3249e7542f4Sdp 	NULL
3259e7542f4Sdp };
3269e7542f4Sdp 
3277c478bd9Sstevel@tonic-gate static const char *select_cmds[] = {
3287c478bd9Sstevel@tonic-gate 	"select fs ",
3297c478bd9Sstevel@tonic-gate 	"select net ",
3307c478bd9Sstevel@tonic-gate 	"select device ",
3317c478bd9Sstevel@tonic-gate 	"select rctl ",
3327c478bd9Sstevel@tonic-gate 	"select attr ",
333fa9e4066Sahrens 	"select dataset ",
3340209230bSgjelinek 	"select dedicated-cpu",
335c97ad5cdSakolb 	"select capped-cpu",
3360209230bSgjelinek 	"select capped-memory",
337cb8a054bSGlenn Faden 	"select admin",
338d2a70789SRichard Lowe 	"select security-flags",
3397c478bd9Sstevel@tonic-gate 	NULL
3407c478bd9Sstevel@tonic-gate };
3417c478bd9Sstevel@tonic-gate 
3427c478bd9Sstevel@tonic-gate static const char *set_cmds[] = {
343087719fdSdp 	"set zonename=",
344087719fdSdp 	"set zonepath=",
3459acbbeafSnn35248 	"set brand=",
346087719fdSdp 	"set autoboot=",
347087719fdSdp 	"set pool=",
348ffbafc53Scomay 	"set limitpriv=",
3493f2f09c1Sdp 	"set bootargs=",
3500209230bSgjelinek 	"set scheduling-class=",
351f4b3ec61Sdh155122 	"set ip-type=",
3520209230bSgjelinek 	"set " ALIAS_MAXLWPS "=",
3530209230bSgjelinek 	"set " ALIAS_MAXSHMMEM "=",
3540209230bSgjelinek 	"set " ALIAS_MAXSHMIDS "=",
3550209230bSgjelinek 	"set " ALIAS_MAXMSGIDS "=",
3560209230bSgjelinek 	"set " ALIAS_MAXSEMIDS "=",
3570209230bSgjelinek 	"set " ALIAS_SHARES "=",
3585679c89fSjv227347 	"set hostid=",
3590fbb751dSJohn Levon 	"set fs-allowed=",
360ff19e029SMenno Lageman 	"set " ALIAS_MAXPROCS "=",
3617c478bd9Sstevel@tonic-gate 	NULL
3627c478bd9Sstevel@tonic-gate };
3637c478bd9Sstevel@tonic-gate 
3649e7542f4Sdp static const char *info_cmds[] = {
3659e7542f4Sdp 	"info fs ",
3669e7542f4Sdp 	"info net ",
3679e7542f4Sdp 	"info device ",
3689e7542f4Sdp 	"info rctl ",
3699e7542f4Sdp 	"info attr ",
3709e7542f4Sdp 	"info dataset ",
3710209230bSgjelinek 	"info capped-memory",
3720209230bSgjelinek 	"info dedicated-cpu",
373c97ad5cdSakolb 	"info capped-cpu",
374d2a70789SRichard Lowe 	"info security-flags",
3759e7542f4Sdp 	"info zonename",
3769e7542f4Sdp 	"info zonepath",
3779e7542f4Sdp 	"info autoboot",
3789e7542f4Sdp 	"info pool",
3799e7542f4Sdp 	"info limitpriv",
3809e7542f4Sdp 	"info bootargs",
3810209230bSgjelinek 	"info brand",
3820209230bSgjelinek 	"info scheduling-class",
383f4b3ec61Sdh155122 	"info ip-type",
3840209230bSgjelinek 	"info max-lwps",
3850209230bSgjelinek 	"info max-shm-memory",
3860209230bSgjelinek 	"info max-shm-ids",
3870209230bSgjelinek 	"info max-msg-ids",
3880209230bSgjelinek 	"info max-sem-ids",
3890209230bSgjelinek 	"info cpu-shares",
3905679c89fSjv227347 	"info hostid",
391cb8a054bSGlenn Faden 	"info admin",
3920fbb751dSJohn Levon 	"info fs-allowed",
393ff19e029SMenno Lageman 	"info max-processes",
3949e7542f4Sdp 	NULL
3959e7542f4Sdp };
3969e7542f4Sdp 
3977c478bd9Sstevel@tonic-gate static const char *fs_res_scope_cmds[] = {
3987c478bd9Sstevel@tonic-gate 	"add options ",
3997c478bd9Sstevel@tonic-gate 	"cancel",
4007c478bd9Sstevel@tonic-gate 	"end",
4017c478bd9Sstevel@tonic-gate 	"exit",
4027c478bd9Sstevel@tonic-gate 	"help",
4037c478bd9Sstevel@tonic-gate 	"info",
404ffbafc53Scomay 	"remove options ",
4057c478bd9Sstevel@tonic-gate 	"set dir=",
4067c478bd9Sstevel@tonic-gate 	"set raw=",
4077c478bd9Sstevel@tonic-gate 	"set special=",
4087c478bd9Sstevel@tonic-gate 	"set type=",
4090209230bSgjelinek 	"clear raw",
4107c478bd9Sstevel@tonic-gate 	NULL
4117c478bd9Sstevel@tonic-gate };
4127c478bd9Sstevel@tonic-gate 
4137c478bd9Sstevel@tonic-gate static const char *net_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 address=",
4203a1c41deSPeter Tribble 	"set allowed-address=",
4217c478bd9Sstevel@tonic-gate 	"set physical=",
4221b3281c0SGerald Jelinek 	"set defrouter=",
4237c478bd9Sstevel@tonic-gate 	NULL
4247c478bd9Sstevel@tonic-gate };
4257c478bd9Sstevel@tonic-gate 
4267c478bd9Sstevel@tonic-gate static const char *device_res_scope_cmds[] = {
4277c478bd9Sstevel@tonic-gate 	"cancel",
4287c478bd9Sstevel@tonic-gate 	"end",
4297c478bd9Sstevel@tonic-gate 	"exit",
4307c478bd9Sstevel@tonic-gate 	"help",
4317c478bd9Sstevel@tonic-gate 	"info",
4327c478bd9Sstevel@tonic-gate 	"set match=",
4337c478bd9Sstevel@tonic-gate 	NULL
4347c478bd9Sstevel@tonic-gate };
4357c478bd9Sstevel@tonic-gate 
4367c478bd9Sstevel@tonic-gate static const char *attr_res_scope_cmds[] = {
4377c478bd9Sstevel@tonic-gate 	"cancel",
4387c478bd9Sstevel@tonic-gate 	"end",
4397c478bd9Sstevel@tonic-gate 	"exit",
4407c478bd9Sstevel@tonic-gate 	"help",
4417c478bd9Sstevel@tonic-gate 	"info",
4427c478bd9Sstevel@tonic-gate 	"set name=",
4437c478bd9Sstevel@tonic-gate 	"set type=",
4447c478bd9Sstevel@tonic-gate 	"set value=",
4457c478bd9Sstevel@tonic-gate 	NULL
4467c478bd9Sstevel@tonic-gate };
4477c478bd9Sstevel@tonic-gate 
4487c478bd9Sstevel@tonic-gate static const char *rctl_res_scope_cmds[] = {
4497c478bd9Sstevel@tonic-gate 	"add value ",
4507c478bd9Sstevel@tonic-gate 	"cancel",
4517c478bd9Sstevel@tonic-gate 	"end",
4527c478bd9Sstevel@tonic-gate 	"exit",
4537c478bd9Sstevel@tonic-gate 	"help",
4547c478bd9Sstevel@tonic-gate 	"info",
455ffbafc53Scomay 	"remove value ",
4567c478bd9Sstevel@tonic-gate 	"set name=",
4577c478bd9Sstevel@tonic-gate 	NULL
4587c478bd9Sstevel@tonic-gate };
4597c478bd9Sstevel@tonic-gate 
460fa9e4066Sahrens static const char *dataset_res_scope_cmds[] = {
461fa9e4066Sahrens 	"cancel",
462fa9e4066Sahrens 	"end",
463fa9e4066Sahrens 	"exit",
464fa9e4066Sahrens 	"help",
465fa9e4066Sahrens 	"info",
466fa9e4066Sahrens 	"set name=",
467fa9e4066Sahrens 	NULL
468fa9e4066Sahrens };
469fa9e4066Sahrens 
4700209230bSgjelinek static const char *pset_res_scope_cmds[] = {
4710209230bSgjelinek 	"cancel",
4720209230bSgjelinek 	"end",
4730209230bSgjelinek 	"exit",
4740209230bSgjelinek 	"help",
4750209230bSgjelinek 	"info",
4760209230bSgjelinek 	"set ncpus=",
4770209230bSgjelinek 	"set importance=",
4780209230bSgjelinek 	"clear importance",
4790209230bSgjelinek 	NULL
4800209230bSgjelinek };
4810209230bSgjelinek 
482c97ad5cdSakolb static const char *pcap_res_scope_cmds[] = {
483c97ad5cdSakolb 	"cancel",
484c97ad5cdSakolb 	"end",
485c97ad5cdSakolb 	"exit",
486c97ad5cdSakolb 	"help",
487c97ad5cdSakolb 	"info",
488c97ad5cdSakolb 	"set ncpus=",
489c97ad5cdSakolb 	NULL
490c97ad5cdSakolb };
491c97ad5cdSakolb 
4920209230bSgjelinek static const char *mcap_res_scope_cmds[] = {
4930209230bSgjelinek 	"cancel",
4940209230bSgjelinek 	"end",
4950209230bSgjelinek 	"exit",
4960209230bSgjelinek 	"help",
4970209230bSgjelinek 	"info",
4980209230bSgjelinek 	"set physical=",
4990209230bSgjelinek 	"set swap=",
5000209230bSgjelinek 	"set locked=",
5010209230bSgjelinek 	"clear physical",
5020209230bSgjelinek 	"clear swap",
5030209230bSgjelinek 	"clear locked",
5040209230bSgjelinek 	NULL
5050209230bSgjelinek };
5060209230bSgjelinek 
507cb8a054bSGlenn Faden static const char *admin_res_scope_cmds[] = {
508cb8a054bSGlenn Faden 	"cancel",
509cb8a054bSGlenn Faden 	"end",
510cb8a054bSGlenn Faden 	"exit",
511cb8a054bSGlenn Faden 	"help",
512cb8a054bSGlenn Faden 	"info",
513cb8a054bSGlenn Faden 	"set user=",
514cb8a054bSGlenn Faden 	"set auths=",
515cb8a054bSGlenn Faden 	NULL
516cb8a054bSGlenn Faden };
517cb8a054bSGlenn Faden 
518d2a70789SRichard Lowe static const char *secflags_res_scope_cmds[] = {
519d2a70789SRichard Lowe 	"cancel",
520d2a70789SRichard Lowe 	"end",
521d2a70789SRichard Lowe 	"exit",
522d2a70789SRichard Lowe 	"set default=",
523d2a70789SRichard Lowe 	"set lower=",
524d2a70789SRichard Lowe 	"set upper=",
525d2a70789SRichard Lowe 	NULL
526d2a70789SRichard Lowe };
527d2a70789SRichard Lowe 
528550b6e40SSowmini Varadhan struct xif {
529550b6e40SSowmini Varadhan 	struct xif	*xif_next;
530550b6e40SSowmini Varadhan 	char		xif_name[LIFNAMSIZ];
531550b6e40SSowmini Varadhan 	boolean_t	xif_has_address;
532550b6e40SSowmini Varadhan 	boolean_t	xif_has_defrouter;
533550b6e40SSowmini Varadhan };
534550b6e40SSowmini Varadhan 
5357c478bd9Sstevel@tonic-gate /* Global variables */
5367c478bd9Sstevel@tonic-gate 
537550b6e40SSowmini Varadhan /* list of network interfaces specified for exclusive IP zone */
538550b6e40SSowmini Varadhan struct xif *xif;
539550b6e40SSowmini Varadhan 
5407c478bd9Sstevel@tonic-gate /* set early in main(), never modified thereafter, used all over the place */
5417c478bd9Sstevel@tonic-gate static char *execname;
5427c478bd9Sstevel@tonic-gate 
5437c478bd9Sstevel@tonic-gate /* set in main(), used all over the place */
5447c478bd9Sstevel@tonic-gate static zone_dochandle_t handle;
5457c478bd9Sstevel@tonic-gate 
5467c478bd9Sstevel@tonic-gate /* used all over the place */
547087719fdSdp static char zone[ZONENAME_MAX];
548087719fdSdp static char revert_zone[ZONENAME_MAX];
5497c478bd9Sstevel@tonic-gate 
5509acbbeafSnn35248 /* global brand operations */
551123807fbSedp static brand_handle_t brand;
5529acbbeafSnn35248 
5537c478bd9Sstevel@tonic-gate /* set in modifying functions, checked in read_input() */
554bbec428eSgjelinek static boolean_t need_to_commit = B_FALSE;
555bbec428eSgjelinek boolean_t saw_error;
5567c478bd9Sstevel@tonic-gate 
5577c478bd9Sstevel@tonic-gate /* set in yacc parser, checked in read_input() */
558bbec428eSgjelinek boolean_t newline_terminated;
5597c478bd9Sstevel@tonic-gate 
5607c478bd9Sstevel@tonic-gate /* set in main(), checked in lex error handler */
561bbec428eSgjelinek boolean_t cmd_file_mode;
5627c478bd9Sstevel@tonic-gate 
5637c478bd9Sstevel@tonic-gate /* set in exit_func(), checked in read_input() */
564bbec428eSgjelinek static boolean_t time_to_exit = B_FALSE, force_exit = B_FALSE;
5657c478bd9Sstevel@tonic-gate 
5667c478bd9Sstevel@tonic-gate /* used in short_usage() and zerr() */
5677c478bd9Sstevel@tonic-gate static char *cmd_file_name = NULL;
5687c478bd9Sstevel@tonic-gate 
5697c478bd9Sstevel@tonic-gate /* checked in read_input() and other places */
570bbec428eSgjelinek static boolean_t ok_to_prompt = B_FALSE;
5717c478bd9Sstevel@tonic-gate 
5727c478bd9Sstevel@tonic-gate /* set and checked in initialize() */
573bbec428eSgjelinek static boolean_t got_handle = B_FALSE;
5747c478bd9Sstevel@tonic-gate 
5757c478bd9Sstevel@tonic-gate /* initialized in do_interactive(), checked in initialize() */
576bbec428eSgjelinek static boolean_t interactive_mode;
5777c478bd9Sstevel@tonic-gate 
5780209230bSgjelinek /* set if configuring the global zone */
579bbec428eSgjelinek static boolean_t global_zone = B_FALSE;
5800209230bSgjelinek 
5817c478bd9Sstevel@tonic-gate /* set in main(), checked in multiple places */
582bbec428eSgjelinek static boolean_t read_only_mode;
5837c478bd9Sstevel@tonic-gate 
584bbec428eSgjelinek /* scope is outer/global or inner/resource */
585bbec428eSgjelinek static boolean_t global_scope = B_TRUE;
5867c478bd9Sstevel@tonic-gate static int resource_scope;	/* should be in the RT_ list from zonecfg.h */
5877c478bd9Sstevel@tonic-gate static int end_op = -1;		/* operation on end is either add or modify */
5887c478bd9Sstevel@tonic-gate 
5897c478bd9Sstevel@tonic-gate int num_prop_vals;		/* for grammar */
5907c478bd9Sstevel@tonic-gate 
5917c478bd9Sstevel@tonic-gate /*
5927c478bd9Sstevel@tonic-gate  * These are for keeping track of resources as they are specified as part of
5937c478bd9Sstevel@tonic-gate  * the multi-step process.  They should be initialized by add_resource() or
5947c478bd9Sstevel@tonic-gate  * select_func() and filled in by add_property() or set_func().
5957c478bd9Sstevel@tonic-gate  */
5967c478bd9Sstevel@tonic-gate static struct zone_fstab	old_fstab, in_progress_fstab;
5977c478bd9Sstevel@tonic-gate static struct zone_nwiftab	old_nwiftab, in_progress_nwiftab;
5987c478bd9Sstevel@tonic-gate static struct zone_devtab	old_devtab, in_progress_devtab;
5997c478bd9Sstevel@tonic-gate static struct zone_rctltab	old_rctltab, in_progress_rctltab;
6007c478bd9Sstevel@tonic-gate static struct zone_attrtab	old_attrtab, in_progress_attrtab;
601fa9e4066Sahrens static struct zone_dstab	old_dstab, in_progress_dstab;
6020209230bSgjelinek static struct zone_psettab	old_psettab, in_progress_psettab;
6030209230bSgjelinek static struct zone_mcaptab	old_mcaptab, in_progress_mcaptab;
604cb8a054bSGlenn Faden static struct zone_admintab	old_admintab, in_progress_admintab;
605d2a70789SRichard Lowe static struct zone_secflagstab	old_secflagstab, in_progress_secflagstab;
6067c478bd9Sstevel@tonic-gate 
6077c478bd9Sstevel@tonic-gate static GetLine *gl;	/* The gl_get_line() resource object */
6087c478bd9Sstevel@tonic-gate 
6090209230bSgjelinek static void bytes_to_units(char *str, char *buf, int bufsize);
6100209230bSgjelinek 
6117c478bd9Sstevel@tonic-gate /* Functions begin here */
6127c478bd9Sstevel@tonic-gate 
613bbec428eSgjelinek static boolean_t
initial_match(const char * line1,const char * line2,int word_end)6147c478bd9Sstevel@tonic-gate initial_match(const char *line1, const char *line2, int word_end)
6157c478bd9Sstevel@tonic-gate {
6167c478bd9Sstevel@tonic-gate 	if (word_end <= 0)
617bbec428eSgjelinek 		return (B_TRUE);
6187c478bd9Sstevel@tonic-gate 	return (strncmp(line1, line2, word_end) == 0);
6197c478bd9Sstevel@tonic-gate }
6207c478bd9Sstevel@tonic-gate 
6217c478bd9Sstevel@tonic-gate static int
add_stuff(WordCompletion * cpl,const char * line1,const char ** list,int word_end)6227c478bd9Sstevel@tonic-gate add_stuff(WordCompletion *cpl, const char *line1, const char **list,
6237c478bd9Sstevel@tonic-gate     int word_end)
6247c478bd9Sstevel@tonic-gate {
6257c478bd9Sstevel@tonic-gate 	int i, err;
6267c478bd9Sstevel@tonic-gate 
6277c478bd9Sstevel@tonic-gate 	for (i = 0; list[i] != NULL; i++) {
6287c478bd9Sstevel@tonic-gate 		if (initial_match(line1, list[i], word_end)) {
6297c478bd9Sstevel@tonic-gate 			err = cpl_add_completion(cpl, line1, 0, word_end,
6307c478bd9Sstevel@tonic-gate 			    list[i] + word_end, "", "");
6317c478bd9Sstevel@tonic-gate 			if (err != 0)
6327c478bd9Sstevel@tonic-gate 				return (err);
6337c478bd9Sstevel@tonic-gate 		}
6347c478bd9Sstevel@tonic-gate 	}
6357c478bd9Sstevel@tonic-gate 	return (0);
6367c478bd9Sstevel@tonic-gate }
6377c478bd9Sstevel@tonic-gate 
6387c478bd9Sstevel@tonic-gate static
6397c478bd9Sstevel@tonic-gate /* ARGSUSED */
CPL_MATCH_FN(cmd_cpl_fn)6407c478bd9Sstevel@tonic-gate CPL_MATCH_FN(cmd_cpl_fn)
6417c478bd9Sstevel@tonic-gate {
6427c478bd9Sstevel@tonic-gate 	if (global_scope) {
6437c478bd9Sstevel@tonic-gate 		/*
6447c478bd9Sstevel@tonic-gate 		 * The MAX/MIN tests below are to make sure we have at least
6457c478bd9Sstevel@tonic-gate 		 * enough characters to distinguish from other prefixes (MAX)
6467c478bd9Sstevel@tonic-gate 		 * but only check MIN(what we have, what we're checking).
6477c478bd9Sstevel@tonic-gate 		 */
6487c478bd9Sstevel@tonic-gate 		if (strncmp(line, "add ", MAX(MIN(word_end, 4), 1)) == 0)
6497c478bd9Sstevel@tonic-gate 			return (add_stuff(cpl, line, add_cmds, word_end));
6500209230bSgjelinek 		if (strncmp(line, "clear ", MAX(MIN(word_end, 6), 2)) == 0)
6510209230bSgjelinek 			return (add_stuff(cpl, line, clear_cmds, word_end));
6527c478bd9Sstevel@tonic-gate 		if (strncmp(line, "select ", MAX(MIN(word_end, 7), 3)) == 0)
6537c478bd9Sstevel@tonic-gate 			return (add_stuff(cpl, line, select_cmds, word_end));
6547c478bd9Sstevel@tonic-gate 		if (strncmp(line, "set ", MAX(MIN(word_end, 4), 3)) == 0)
6557c478bd9Sstevel@tonic-gate 			return (add_stuff(cpl, line, set_cmds, word_end));
6569e7542f4Sdp 		if (strncmp(line, "remove ", MAX(MIN(word_end, 7), 1)) == 0)
6579e7542f4Sdp 			return (add_stuff(cpl, line, remove_cmds, word_end));
6589e7542f4Sdp 		if (strncmp(line, "info ", MAX(MIN(word_end, 5), 1)) == 0)
6599e7542f4Sdp 			return (add_stuff(cpl, line, info_cmds, word_end));
6607c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, global_scope_cmds, word_end));
6617c478bd9Sstevel@tonic-gate 	}
6627c478bd9Sstevel@tonic-gate 	switch (resource_scope) {
6637c478bd9Sstevel@tonic-gate 	case RT_FS:
6647c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, fs_res_scope_cmds, word_end));
6657c478bd9Sstevel@tonic-gate 	case RT_NET:
6667c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, net_res_scope_cmds, word_end));
6677c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
6687c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, device_res_scope_cmds, word_end));
6697c478bd9Sstevel@tonic-gate 	case RT_RCTL:
6707c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, rctl_res_scope_cmds, word_end));
6717c478bd9Sstevel@tonic-gate 	case RT_ATTR:
6727c478bd9Sstevel@tonic-gate 		return (add_stuff(cpl, line, attr_res_scope_cmds, word_end));
673fa9e4066Sahrens 	case RT_DATASET:
674fa9e4066Sahrens 		return (add_stuff(cpl, line, dataset_res_scope_cmds, word_end));
6750209230bSgjelinek 	case RT_DCPU:
6760209230bSgjelinek 		return (add_stuff(cpl, line, pset_res_scope_cmds, word_end));
677c97ad5cdSakolb 	case RT_PCAP:
678c97ad5cdSakolb 		return (add_stuff(cpl, line, pcap_res_scope_cmds, word_end));
6790209230bSgjelinek 	case RT_MCAP:
6800209230bSgjelinek 		return (add_stuff(cpl, line, mcap_res_scope_cmds, word_end));
681cb8a054bSGlenn Faden 	case RT_ADMIN:
682cb8a054bSGlenn Faden 		return (add_stuff(cpl, line, admin_res_scope_cmds, word_end));
683d2a70789SRichard Lowe 	case RT_SECFLAGS:
684d2a70789SRichard Lowe 		return (add_stuff(cpl, line, secflags_res_scope_cmds,
685d2a70789SRichard Lowe 		    word_end));
686d2a70789SRichard Lowe 
6877c478bd9Sstevel@tonic-gate 	}
6887c478bd9Sstevel@tonic-gate 	return (0);
6897c478bd9Sstevel@tonic-gate }
6907c478bd9Sstevel@tonic-gate 
6917c478bd9Sstevel@tonic-gate /*
6927c478bd9Sstevel@tonic-gate  * For the main CMD_func() functions below, several of them call getopt()
6937c478bd9Sstevel@tonic-gate  * then check optind against argc to make sure an extra parameter was not
6947c478bd9Sstevel@tonic-gate  * passed in.  The reason this is not caught in the grammar is that the
6957c478bd9Sstevel@tonic-gate  * grammar just checks for a miscellaneous TOKEN, which is *expected* to
6967c478bd9Sstevel@tonic-gate  * be "-F" (for example), but could be anything.  So (for example) this
6977c478bd9Sstevel@tonic-gate  * check will prevent "create bogus".
6987c478bd9Sstevel@tonic-gate  */
6997c478bd9Sstevel@tonic-gate 
7007c478bd9Sstevel@tonic-gate cmd_t *
alloc_cmd(void)7017c478bd9Sstevel@tonic-gate alloc_cmd(void)
7027c478bd9Sstevel@tonic-gate {
7037c478bd9Sstevel@tonic-gate 	return (calloc(1, sizeof (cmd_t)));
7047c478bd9Sstevel@tonic-gate }
7057c478bd9Sstevel@tonic-gate 
7067c478bd9Sstevel@tonic-gate void
free_cmd(cmd_t * cmd)7077c478bd9Sstevel@tonic-gate free_cmd(cmd_t *cmd)
7087c478bd9Sstevel@tonic-gate {
7097c478bd9Sstevel@tonic-gate 	int i;
7107c478bd9Sstevel@tonic-gate 
7117c478bd9Sstevel@tonic-gate 	for (i = 0; i < MAX_EQ_PROP_PAIRS; i++)
7127c478bd9Sstevel@tonic-gate 		if (cmd->cmd_property_ptr[i] != NULL) {
7137c478bd9Sstevel@tonic-gate 			property_value_ptr_t pp = cmd->cmd_property_ptr[i];
7147c478bd9Sstevel@tonic-gate 
7157c478bd9Sstevel@tonic-gate 			switch (pp->pv_type) {
7167c478bd9Sstevel@tonic-gate 			case PROP_VAL_SIMPLE:
7177c478bd9Sstevel@tonic-gate 				free(pp->pv_simple);
7187c478bd9Sstevel@tonic-gate 				break;
7197c478bd9Sstevel@tonic-gate 			case PROP_VAL_COMPLEX:
7207c478bd9Sstevel@tonic-gate 				free_complex(pp->pv_complex);
7217c478bd9Sstevel@tonic-gate 				break;
7227c478bd9Sstevel@tonic-gate 			case PROP_VAL_LIST:
7237c478bd9Sstevel@tonic-gate 				free_list(pp->pv_list);
7247c478bd9Sstevel@tonic-gate 				break;
7257c478bd9Sstevel@tonic-gate 			}
7267c478bd9Sstevel@tonic-gate 		}
7277c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_argc; i++)
7287c478bd9Sstevel@tonic-gate 		free(cmd->cmd_argv[i]);
7297c478bd9Sstevel@tonic-gate 	free(cmd);
7307c478bd9Sstevel@tonic-gate }
7317c478bd9Sstevel@tonic-gate 
7327c478bd9Sstevel@tonic-gate complex_property_ptr_t
alloc_complex(void)7337c478bd9Sstevel@tonic-gate alloc_complex(void)
7347c478bd9Sstevel@tonic-gate {
7357c478bd9Sstevel@tonic-gate 	return (calloc(1, sizeof (complex_property_t)));
7367c478bd9Sstevel@tonic-gate }
7377c478bd9Sstevel@tonic-gate 
7387c478bd9Sstevel@tonic-gate void
free_complex(complex_property_ptr_t complex)7397c478bd9Sstevel@tonic-gate free_complex(complex_property_ptr_t complex)
7407c478bd9Sstevel@tonic-gate {
7417c478bd9Sstevel@tonic-gate 	if (complex == NULL)
7427c478bd9Sstevel@tonic-gate 		return;
7437c478bd9Sstevel@tonic-gate 	free_complex(complex->cp_next);
7447c478bd9Sstevel@tonic-gate 	if (complex->cp_value != NULL)
7457c478bd9Sstevel@tonic-gate 		free(complex->cp_value);
7467c478bd9Sstevel@tonic-gate 	free(complex);
7477c478bd9Sstevel@tonic-gate }
7487c478bd9Sstevel@tonic-gate 
7497c478bd9Sstevel@tonic-gate list_property_ptr_t
alloc_list(void)7507c478bd9Sstevel@tonic-gate alloc_list(void)
7517c478bd9Sstevel@tonic-gate {
7527c478bd9Sstevel@tonic-gate 	return (calloc(1, sizeof (list_property_t)));
7537c478bd9Sstevel@tonic-gate }
7547c478bd9Sstevel@tonic-gate 
7557c478bd9Sstevel@tonic-gate void
free_list(list_property_ptr_t list)7567c478bd9Sstevel@tonic-gate free_list(list_property_ptr_t list)
7577c478bd9Sstevel@tonic-gate {
7587c478bd9Sstevel@tonic-gate 	if (list == NULL)
7597c478bd9Sstevel@tonic-gate 		return;
7607c478bd9Sstevel@tonic-gate 	if (list->lp_simple != NULL)
7617c478bd9Sstevel@tonic-gate 		free(list->lp_simple);
7627c478bd9Sstevel@tonic-gate 	free_complex(list->lp_complex);
7637c478bd9Sstevel@tonic-gate 	free_list(list->lp_next);
7647c478bd9Sstevel@tonic-gate 	free(list);
7657c478bd9Sstevel@tonic-gate }
7667c478bd9Sstevel@tonic-gate 
7677c478bd9Sstevel@tonic-gate void
free_outer_list(list_property_ptr_t list)7687c478bd9Sstevel@tonic-gate free_outer_list(list_property_ptr_t list)
7697c478bd9Sstevel@tonic-gate {
7707c478bd9Sstevel@tonic-gate 	if (list == NULL)
7717c478bd9Sstevel@tonic-gate 		return;
7727c478bd9Sstevel@tonic-gate 	free_outer_list(list->lp_next);
7737c478bd9Sstevel@tonic-gate 	free(list);
7747c478bd9Sstevel@tonic-gate }
7757c478bd9Sstevel@tonic-gate 
7767c478bd9Sstevel@tonic-gate static struct zone_rctlvaltab *
alloc_rctlvaltab(void)7777c478bd9Sstevel@tonic-gate alloc_rctlvaltab(void)
7787c478bd9Sstevel@tonic-gate {
7797c478bd9Sstevel@tonic-gate 	return (calloc(1, sizeof (struct zone_rctlvaltab)));
7807c478bd9Sstevel@tonic-gate }
7817c478bd9Sstevel@tonic-gate 
7827c478bd9Sstevel@tonic-gate static char *
rt_to_str(int res_type)7837c478bd9Sstevel@tonic-gate rt_to_str(int res_type)
7847c478bd9Sstevel@tonic-gate {
7857c478bd9Sstevel@tonic-gate 	assert(res_type >= RT_MIN && res_type <= RT_MAX);
7867c478bd9Sstevel@tonic-gate 	return (res_types[res_type]);
7877c478bd9Sstevel@tonic-gate }
7887c478bd9Sstevel@tonic-gate 
7897c478bd9Sstevel@tonic-gate static char *
pt_to_str(int prop_type)7907c478bd9Sstevel@tonic-gate pt_to_str(int prop_type)
7917c478bd9Sstevel@tonic-gate {
7927c478bd9Sstevel@tonic-gate 	assert(prop_type >= PT_MIN && prop_type <= PT_MAX);
7937c478bd9Sstevel@tonic-gate 	return (prop_types[prop_type]);
7947c478bd9Sstevel@tonic-gate }
7957c478bd9Sstevel@tonic-gate 
7967c478bd9Sstevel@tonic-gate static char *
pvt_to_str(int pv_type)7977c478bd9Sstevel@tonic-gate pvt_to_str(int pv_type)
7987c478bd9Sstevel@tonic-gate {
7997c478bd9Sstevel@tonic-gate 	assert(pv_type >= PROP_VAL_MIN && pv_type <= PROP_VAL_MAX);
8007c478bd9Sstevel@tonic-gate 	return (prop_val_types[pv_type]);
8017c478bd9Sstevel@tonic-gate }
8027c478bd9Sstevel@tonic-gate 
8037c478bd9Sstevel@tonic-gate static char *
cmd_to_str(int cmd_num)8047c478bd9Sstevel@tonic-gate cmd_to_str(int cmd_num)
8057c478bd9Sstevel@tonic-gate {
8067c478bd9Sstevel@tonic-gate 	assert(cmd_num >= CMD_MIN && cmd_num <= CMD_MAX);
8077c478bd9Sstevel@tonic-gate 	return (helptab[cmd_num].cmd_name);
8087c478bd9Sstevel@tonic-gate }
8097c478bd9Sstevel@tonic-gate 
8103042b8b5Sbatschul /* PRINTFLIKE1 */
8113042b8b5Sbatschul static void
zerr(const char * fmt,...)8123042b8b5Sbatschul zerr(const char *fmt, ...)
8133042b8b5Sbatschul {
8143042b8b5Sbatschul 	va_list alist;
8153042b8b5Sbatschul 	static int last_lineno;
8163042b8b5Sbatschul 
8173042b8b5Sbatschul 	/* lex_lineno has already been incremented in the lexer; compensate */
8183042b8b5Sbatschul 	if (cmd_file_mode && lex_lineno > last_lineno) {
8193042b8b5Sbatschul 		if (strcmp(cmd_file_name, "-") == 0)
8203042b8b5Sbatschul 			(void) fprintf(stderr, gettext("On line %d:\n"),
8213042b8b5Sbatschul 			    lex_lineno - 1);
8223042b8b5Sbatschul 		else
8233042b8b5Sbatschul 			(void) fprintf(stderr, gettext("On line %d of %s:\n"),
8243042b8b5Sbatschul 			    lex_lineno - 1, cmd_file_name);
8253042b8b5Sbatschul 		last_lineno = lex_lineno;
8263042b8b5Sbatschul 	}
8273042b8b5Sbatschul 	va_start(alist, fmt);
8283042b8b5Sbatschul 	(void) vfprintf(stderr, fmt, alist);
8293042b8b5Sbatschul 	(void) fprintf(stderr, "\n");
8303042b8b5Sbatschul 	va_end(alist);
8313042b8b5Sbatschul }
8323042b8b5Sbatschul 
8337c478bd9Sstevel@tonic-gate /*
8347c478bd9Sstevel@tonic-gate  * This is a separate function rather than a set of define's because of the
8357c478bd9Sstevel@tonic-gate  * gettext() wrapping.
8367c478bd9Sstevel@tonic-gate  */
8377c478bd9Sstevel@tonic-gate 
8387c478bd9Sstevel@tonic-gate /*
8397c478bd9Sstevel@tonic-gate  * TRANSLATION_NOTE
8407c478bd9Sstevel@tonic-gate  * Each string below should have \t follow \n whenever needed; the
8417c478bd9Sstevel@tonic-gate  * initial \t and the terminal \n will be provided by the calling function.
8427c478bd9Sstevel@tonic-gate  */
8437c478bd9Sstevel@tonic-gate 
8447c478bd9Sstevel@tonic-gate static char *
long_help(int cmd_num)8457c478bd9Sstevel@tonic-gate long_help(int cmd_num)
8467c478bd9Sstevel@tonic-gate {
8477c478bd9Sstevel@tonic-gate 	static char line[1024];	/* arbitrary large amount */
8487c478bd9Sstevel@tonic-gate 
8497c478bd9Sstevel@tonic-gate 	assert(cmd_num >= CMD_MIN && cmd_num <= CMD_MAX);
8507c478bd9Sstevel@tonic-gate 	switch (cmd_num) {
8517c478bd9Sstevel@tonic-gate 		case CMD_HELP:
8527c478bd9Sstevel@tonic-gate 			return (gettext("Prints help message."));
8537c478bd9Sstevel@tonic-gate 		case CMD_CREATE:
8547c478bd9Sstevel@tonic-gate 			(void) snprintf(line, sizeof (line),
8557c478bd9Sstevel@tonic-gate 			    gettext("Creates a configuration for the "
8567c478bd9Sstevel@tonic-gate 			    "specified zone.  %s should be\n\tused to "
8577c478bd9Sstevel@tonic-gate 			    "begin configuring a new zone.  If overwriting an "
8587c478bd9Sstevel@tonic-gate 			    "existing\n\tconfiguration, the -F flag can be "
8597c478bd9Sstevel@tonic-gate 			    "used to force the action.  If\n\t-t template is "
8607c478bd9Sstevel@tonic-gate 			    "given, creates a configuration identical to the\n"
8617c478bd9Sstevel@tonic-gate 			    "\tspecified template, except that the zone name "
8629e518655Sgjelinek 			    "is changed from\n\ttemplate to zonename.  '%s -a' "
8639e518655Sgjelinek 			    "creates a configuration from a\n\tdetached "
8649e518655Sgjelinek 			    "zonepath.  '%s -b' results in a blank "
8659e518655Sgjelinek 			    "configuration.\n\t'%s' with no arguments applies "
8669e518655Sgjelinek 			    "the Sun default settings."),
8677c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_CREATE), cmd_to_str(CMD_CREATE),
8689e518655Sgjelinek 			    cmd_to_str(CMD_CREATE), cmd_to_str(CMD_CREATE));
8697c478bd9Sstevel@tonic-gate 			return (line);
8707c478bd9Sstevel@tonic-gate 		case CMD_EXIT:
8717c478bd9Sstevel@tonic-gate 			return (gettext("Exits the program.  The -F flag can "
8727c478bd9Sstevel@tonic-gate 			    "be used to force the action."));
8737c478bd9Sstevel@tonic-gate 		case CMD_EXPORT:
8747c478bd9Sstevel@tonic-gate 			return (gettext("Prints configuration to standard "
8757c478bd9Sstevel@tonic-gate 			    "output, or to output-file if\n\tspecified, in "
8767c478bd9Sstevel@tonic-gate 			    "a form suitable for use in a command-file."));
8777c478bd9Sstevel@tonic-gate 		case CMD_ADD:
8787c478bd9Sstevel@tonic-gate 			return (gettext("Add specified resource to "
8797c478bd9Sstevel@tonic-gate 			    "configuration."));
8807c478bd9Sstevel@tonic-gate 		case CMD_DELETE:
8817c478bd9Sstevel@tonic-gate 			return (gettext("Deletes the specified zone.  The -F "
8827c478bd9Sstevel@tonic-gate 			    "flag can be used to force the\n\taction."));
8837c478bd9Sstevel@tonic-gate 		case CMD_REMOVE:
8847c478bd9Sstevel@tonic-gate 			return (gettext("Remove specified resource from "
8850209230bSgjelinek 			    "configuration.  The -F flag can be used\n\tto "
8860209230bSgjelinek 			    "force the action."));
8877c478bd9Sstevel@tonic-gate 		case CMD_SELECT:
8887c478bd9Sstevel@tonic-gate 			(void) snprintf(line, sizeof (line),
8897c478bd9Sstevel@tonic-gate 			    gettext("Selects a resource to modify.  "
8907c478bd9Sstevel@tonic-gate 			    "Resource modification is completed\n\twith the "
8917c478bd9Sstevel@tonic-gate 			    "command \"%s\".  The property name/value pairs "
8927c478bd9Sstevel@tonic-gate 			    "must uniquely\n\tidentify a resource.  Note that "
8937c478bd9Sstevel@tonic-gate 			    "the curly braces ('{', '}') mean one\n\tor more "
8947c478bd9Sstevel@tonic-gate 			    "of whatever is between them."),
8957c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_END));
8967c478bd9Sstevel@tonic-gate 			return (line);
8977c478bd9Sstevel@tonic-gate 		case CMD_SET:
8987c478bd9Sstevel@tonic-gate 			return (gettext("Sets property values."));
8990209230bSgjelinek 		case CMD_CLEAR:
9000209230bSgjelinek 			return (gettext("Clears property values."));
9017c478bd9Sstevel@tonic-gate 		case CMD_INFO:
9027c478bd9Sstevel@tonic-gate 			return (gettext("Displays information about the "
9037c478bd9Sstevel@tonic-gate 			    "current configuration.  If resource\n\ttype is "
9047c478bd9Sstevel@tonic-gate 			    "specified, displays only information about "
9057c478bd9Sstevel@tonic-gate 			    "resources of\n\tthe relevant type.  If resource "
9067c478bd9Sstevel@tonic-gate 			    "id is specified, displays only\n\tinformation "
9077c478bd9Sstevel@tonic-gate 			    "about that resource."));
9087c478bd9Sstevel@tonic-gate 		case CMD_VERIFY:
9097c478bd9Sstevel@tonic-gate 			return (gettext("Verifies current configuration "
9107c478bd9Sstevel@tonic-gate 			    "for correctness (some resource types\n\thave "
9117c478bd9Sstevel@tonic-gate 			    "required properties)."));
9127c478bd9Sstevel@tonic-gate 		case CMD_COMMIT:
9137c478bd9Sstevel@tonic-gate 			(void) snprintf(line, sizeof (line),
9147c478bd9Sstevel@tonic-gate 			    gettext("Commits current configuration.  "
9157c478bd9Sstevel@tonic-gate 			    "Configuration must be committed to\n\tbe used by "
9167c478bd9Sstevel@tonic-gate 			    "%s.  Until the configuration is committed, "
9177c478bd9Sstevel@tonic-gate 			    "changes \n\tcan be removed with the %s "
9187c478bd9Sstevel@tonic-gate 			    "command.  This operation is\n\tattempted "
9197c478bd9Sstevel@tonic-gate 			    "automatically upon completion of a %s "
9207c478bd9Sstevel@tonic-gate 			    "session."), "zoneadm", cmd_to_str(CMD_REVERT),
9217c478bd9Sstevel@tonic-gate 			    "zonecfg");
9227c478bd9Sstevel@tonic-gate 			return (line);
9237c478bd9Sstevel@tonic-gate 		case CMD_REVERT:
9247c478bd9Sstevel@tonic-gate 			return (gettext("Reverts configuration back to the "
9257c478bd9Sstevel@tonic-gate 			    "last committed state.  The -F flag\n\tcan be "
9267c478bd9Sstevel@tonic-gate 			    "used to force the action."));
9277c478bd9Sstevel@tonic-gate 		case CMD_CANCEL:
9287c478bd9Sstevel@tonic-gate 			return (gettext("Cancels resource/property "
9297c478bd9Sstevel@tonic-gate 			    "specification."));
9307c478bd9Sstevel@tonic-gate 		case CMD_END:
9317c478bd9Sstevel@tonic-gate 			return (gettext("Ends resource/property "
9327c478bd9Sstevel@tonic-gate 			    "specification."));
9337c478bd9Sstevel@tonic-gate 	}
9347c478bd9Sstevel@tonic-gate 	/* NOTREACHED */
9357e362f58Scomay 	return (NULL);
9367c478bd9Sstevel@tonic-gate }
9377c478bd9Sstevel@tonic-gate 
9387c478bd9Sstevel@tonic-gate /*
939944b13ecSGary Mills  * Return the input filename appended to each component of the path
940944b13ecSGary Mills  * or the filename itself if it is absolute.
941944b13ecSGary Mills  * Parameters: path string, file name, output string.
942944b13ecSGary Mills  */
943944b13ecSGary Mills /* Copied almost verbatim from libtnfctl/prb_findexec.c */
944944b13ecSGary Mills static const char *
exec_cat(const char * s1,const char * s2,char * si)945944b13ecSGary Mills exec_cat(const char *s1, const char *s2, char *si)
946944b13ecSGary Mills {
947944b13ecSGary Mills 	char		   *s;
948944b13ecSGary Mills 	/* Number of remaining characters in s */
949944b13ecSGary Mills 	int			 cnt = PATH_MAX + 1;
950944b13ecSGary Mills 
951944b13ecSGary Mills 	s = si;
952944b13ecSGary Mills 	while (*s1 && *s1 != ':') { /* Copy first component of path to si */
953944b13ecSGary Mills 		if (cnt > 0) {
954944b13ecSGary Mills 			*s++ = *s1++;
955944b13ecSGary Mills 			cnt--;
956944b13ecSGary Mills 		} else {
957944b13ecSGary Mills 			s1++;
958944b13ecSGary Mills 		}
959944b13ecSGary Mills 	}
960944b13ecSGary Mills 	if (si != s && cnt > 0) { /* Add slash if s2 is not absolute */
961944b13ecSGary Mills 		*s++ = '/';
962944b13ecSGary Mills 		cnt--;
963944b13ecSGary Mills 	}
964944b13ecSGary Mills 	while (*s2 && cnt > 0) { /* Copy s2 to si */
965944b13ecSGary Mills 		*s++ = *s2++;
966944b13ecSGary Mills 		cnt--;
967944b13ecSGary Mills 	}
968944b13ecSGary Mills 	*s = '\0';  /* Terminate the output string */
969944b13ecSGary Mills 	return (*s1 ? ++s1 : NULL);  /* Return next path component or NULL */
970944b13ecSGary Mills }
971944b13ecSGary Mills 
972944b13ecSGary Mills /* Determine that a name exists in PATH */
973944b13ecSGary Mills /* Copied with changes from libtnfctl/prb_findexec.c */
974944b13ecSGary Mills static int
path_find(const char * name)975944b13ecSGary Mills path_find(const char *name)
976944b13ecSGary Mills {
977944b13ecSGary Mills 	const char	 *pathstr;
978944b13ecSGary Mills 	char		fname[PATH_MAX + 2];
979944b13ecSGary Mills 	const char	 *cp;
980944b13ecSGary Mills 	struct stat	 stat_buf;
981944b13ecSGary Mills 
982944b13ecSGary Mills 	if ((pathstr = getenv("PATH")) == NULL) {
983944b13ecSGary Mills 		if (geteuid() == 0 || getuid() == 0)
984944b13ecSGary Mills 			pathstr = "/usr/sbin:/usr/bin";
985944b13ecSGary Mills 		else
986944b13ecSGary Mills 			pathstr = "/usr/bin:";
987944b13ecSGary Mills 	}
988944b13ecSGary Mills 	cp = strchr(name, '/') ? (const char *) "" : pathstr;
989944b13ecSGary Mills 
990944b13ecSGary Mills 	do {
991944b13ecSGary Mills 		cp = exec_cat(cp, name, fname);
992944b13ecSGary Mills 		if (stat(fname, &stat_buf) != -1) {
993944b13ecSGary Mills 			/* successful find of the file */
994944b13ecSGary Mills 			return (0);
995944b13ecSGary Mills 		}
996944b13ecSGary Mills 	} while (cp != NULL);
997944b13ecSGary Mills 
998944b13ecSGary Mills 	return (-1);
999944b13ecSGary Mills }
1000944b13ecSGary Mills 
1001944b13ecSGary Mills static FILE *
pager_open(void)1002d2a70789SRichard Lowe pager_open(void)
1003d2a70789SRichard Lowe {
1004944b13ecSGary Mills 	FILE *newfp;
1005944b13ecSGary Mills 	char *pager, *space;
1006944b13ecSGary Mills 
1007944b13ecSGary Mills 	pager = getenv("PAGER");
1008944b13ecSGary Mills 	if (pager == NULL || *pager == '\0')
1009944b13ecSGary Mills 		pager = PAGER;
1010944b13ecSGary Mills 
1011944b13ecSGary Mills 	space = strchr(pager, ' ');
1012944b13ecSGary Mills 	if (space)
1013944b13ecSGary Mills 		*space = '\0';
1014944b13ecSGary Mills 	if (path_find(pager) == 0) {
1015944b13ecSGary Mills 		if (space)
1016944b13ecSGary Mills 			*space = ' ';
1017944b13ecSGary Mills 		if ((newfp = popen(pager, "w")) == NULL)
1018944b13ecSGary Mills 			zerr(gettext("PAGER open failed (%s)."),
1019944b13ecSGary Mills 			    strerror(errno));
1020944b13ecSGary Mills 		return (newfp);
1021944b13ecSGary Mills 	} else {
1022944b13ecSGary Mills 		zerr(gettext("PAGER %s does not exist (%s)."),
1023944b13ecSGary Mills 		    pager, strerror(errno));
1024944b13ecSGary Mills 	}
1025944b13ecSGary Mills 	return (NULL);
1026944b13ecSGary Mills }
1027944b13ecSGary Mills 
1028944b13ecSGary Mills static void
pager_close(FILE * fp)1029d2a70789SRichard Lowe pager_close(FILE *fp)
1030d2a70789SRichard Lowe {
1031944b13ecSGary Mills 	int status;
1032944b13ecSGary Mills 
1033944b13ecSGary Mills 	status = pclose(fp);
1034944b13ecSGary Mills 	if (status == -1)
1035944b13ecSGary Mills 		zerr(gettext("PAGER close failed (%s)."),
1036944b13ecSGary Mills 		    strerror(errno));
1037944b13ecSGary Mills }
1038944b13ecSGary Mills 
1039944b13ecSGary Mills /*
10407c478bd9Sstevel@tonic-gate  * Called with verbose TRUE when help is explicitly requested, FALSE for
10417c478bd9Sstevel@tonic-gate  * unexpected errors.
10427c478bd9Sstevel@tonic-gate  */
10437c478bd9Sstevel@tonic-gate 
10447c478bd9Sstevel@tonic-gate void
usage(boolean_t verbose,uint_t flags)1045bbec428eSgjelinek usage(boolean_t verbose, uint_t flags)
10467c478bd9Sstevel@tonic-gate {
10473042b8b5Sbatschul 	FILE *fp = verbose ? stdout : stderr;
10483042b8b5Sbatschul 	FILE *newfp;
1049bbec428eSgjelinek 	boolean_t need_to_close = B_FALSE;
10507c478bd9Sstevel@tonic-gate 	int i;
10517c478bd9Sstevel@tonic-gate 
10527c478bd9Sstevel@tonic-gate 	/* don't page error output */
10537c478bd9Sstevel@tonic-gate 	if (verbose && interactive_mode) {
1054944b13ecSGary Mills 		if ((newfp = pager_open()) != NULL) {
1055bbec428eSgjelinek 			need_to_close = B_TRUE;
10567c478bd9Sstevel@tonic-gate 			fp = newfp;
10577c478bd9Sstevel@tonic-gate 		}
10583042b8b5Sbatschul 	}
10593042b8b5Sbatschul 
10607c478bd9Sstevel@tonic-gate 	if (flags & HELP_META) {
10617c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("More help is available for the "
10627c478bd9Sstevel@tonic-gate 		    "following:\n"));
10637c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\n\tcommands ('%s commands')\n",
10647c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_HELP));
10657c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\tsyntax ('%s syntax')\n",
10667c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_HELP));
10677c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\tusage ('%s usage')\n\n",
10687c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_HELP));
10697c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("You may also obtain help on any "
10707c478bd9Sstevel@tonic-gate 		    "command by typing '%s <command-name>.'\n"),
10717c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_HELP));
10727c478bd9Sstevel@tonic-gate 	}
10737c478bd9Sstevel@tonic-gate 	if (flags & HELP_RES_SCOPE) {
10747c478bd9Sstevel@tonic-gate 		switch (resource_scope) {
10757c478bd9Sstevel@tonic-gate 		case RT_FS:
10767c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
10777c478bd9Sstevel@tonic-gate 			    "used to configure a file-system.\n"),
10787c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
10797c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
10807c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10817c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_DIR), gettext("<path>"));
10827c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10837c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_SPECIAL), gettext("<path>"));
10847c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10857c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_RAW), gettext("<raw-device>"));
10867c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
10877c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_TYPE), gettext("<file-system type>"));
10887c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s %s\n", cmd_to_str(CMD_ADD),
10897c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_OPTIONS),
10907c478bd9Sstevel@tonic-gate 			    gettext("<file-system options>"));
1091ffbafc53Scomay 			(void) fprintf(fp, "\t%s %s %s\n",
1092ffbafc53Scomay 			    cmd_to_str(CMD_REMOVE), pt_to_str(PT_OPTIONS),
1093ffbafc53Scomay 			    gettext("<file-system options>"));
10947c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Consult the file-system "
1095*bbf21555SRichard Lowe 			    "specific manual page, such as mount_ufs(8), "
10967c478bd9Sstevel@tonic-gate 			    "for\ndetails about file-system options.  Note "
10977c478bd9Sstevel@tonic-gate 			    "that any file-system options with an\nembedded "
10987c478bd9Sstevel@tonic-gate 			    "'=' character must be enclosed in double quotes, "
10997c478bd9Sstevel@tonic-gate 			    /*CSTYLED*/
11007c478bd9Sstevel@tonic-gate 			    "such as \"%s=5\".\n"), MNTOPT_RETRY);
11017c478bd9Sstevel@tonic-gate 			break;
11027c478bd9Sstevel@tonic-gate 		case RT_NET:
11037c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
11047c478bd9Sstevel@tonic-gate 			    "used to configure a network interface.\n"),
11057c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
11067c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
11077c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
11087c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_ADDRESS), gettext("<IP-address>"));
11097c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1110550b6e40SSowmini Varadhan 			    pt_to_str(PT_ALLOWED_ADDRESS),
1111550b6e40SSowmini Varadhan 			    gettext("<IP-address>"));
1112550b6e40SSowmini Varadhan 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
11137c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_PHYSICAL), gettext("<interface>"));
1114*bbf21555SRichard Lowe 			(void) fprintf(fp, gettext("See ifconfig(8) for "
11157c478bd9Sstevel@tonic-gate 			    "details of the <interface> string.\n"));
1116550b6e40SSowmini Varadhan 			(void) fprintf(fp, gettext("%s %s is valid "
1117550b6e40SSowmini Varadhan 			    "if the %s property is set to %s, otherwise it "
1118de860bd9Sgfaden 			    "must not be set.\n"),
1119f4b3ec61Sdh155122 			    cmd_to_str(CMD_SET), pt_to_str(PT_ADDRESS),
1120550b6e40SSowmini Varadhan 			    pt_to_str(PT_IPTYPE), gettext("shared"));
1121550b6e40SSowmini Varadhan 			(void) fprintf(fp, gettext("%s %s is valid "
1122550b6e40SSowmini Varadhan 			    "if the %s property is set to %s, otherwise it "
1123550b6e40SSowmini Varadhan 			    "must not be set.\n"),
1124550b6e40SSowmini Varadhan 			    cmd_to_str(CMD_SET), pt_to_str(PT_ALLOWED_ADDRESS),
1125550b6e40SSowmini Varadhan 			    pt_to_str(PT_IPTYPE), gettext("exclusive"));
1126550b6e40SSowmini Varadhan 			(void) fprintf(fp, gettext("\t%s %s=%s\n%s %s "
1127550b6e40SSowmini Varadhan 			    "is valid if the %s or %s property is set, "
1128550b6e40SSowmini Varadhan 			    "otherwise it must not be set\n"),
1129550b6e40SSowmini Varadhan 			    cmd_to_str(CMD_SET),
1130550b6e40SSowmini Varadhan 			    pt_to_str(PT_DEFROUTER), gettext("<IP-address>"),
1131de860bd9Sgfaden 			    cmd_to_str(CMD_SET), pt_to_str(PT_DEFROUTER),
1132550b6e40SSowmini Varadhan 			    gettext(pt_to_str(PT_ADDRESS)),
1133550b6e40SSowmini Varadhan 			    gettext(pt_to_str(PT_ALLOWED_ADDRESS)));
11347c478bd9Sstevel@tonic-gate 			break;
11357c478bd9Sstevel@tonic-gate 		case RT_DEVICE:
11367c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
11377c478bd9Sstevel@tonic-gate 			    "used to configure a device node.\n"),
11387c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
11397c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
11407c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
11417c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_MATCH), gettext("<device-path>"));
11427c478bd9Sstevel@tonic-gate 			break;
11437c478bd9Sstevel@tonic-gate 		case RT_RCTL:
11447c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
11457c478bd9Sstevel@tonic-gate 			    "used to configure a resource control.\n"),
11467c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
11477c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
11487c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
11497c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_NAME), gettext("<string>"));
11507c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s (%s=%s,%s=%s,%s=%s)\n",
11517c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_ADD), pt_to_str(PT_VALUE),
11527c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_PRIV), gettext("<priv-value>"),
11537c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_LIMIT), gettext("<number>"),
11547c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_ACTION), gettext("<action-value>"));
1155ffbafc53Scomay 			(void) fprintf(fp, "\t%s %s (%s=%s,%s=%s,%s=%s)\n",
1156ffbafc53Scomay 			    cmd_to_str(CMD_REMOVE), pt_to_str(PT_VALUE),
1157ffbafc53Scomay 			    pt_to_str(PT_PRIV), gettext("<priv-value>"),
1158ffbafc53Scomay 			    pt_to_str(PT_LIMIT), gettext("<number>"),
1159ffbafc53Scomay 			    pt_to_str(PT_ACTION), gettext("<action-value>"));
11607c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "%s\n\t%s := privileged\n"
11617c478bd9Sstevel@tonic-gate 			    "\t%s := none | deny\n", gettext("Where"),
11627c478bd9Sstevel@tonic-gate 			    gettext("<priv-value>"), gettext("<action-value>"));
11637c478bd9Sstevel@tonic-gate 			break;
11647c478bd9Sstevel@tonic-gate 		case RT_ATTR:
11657c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("The '%s' resource scope is "
11667c478bd9Sstevel@tonic-gate 			    "used to configure a generic attribute.\n"),
11677c478bd9Sstevel@tonic-gate 			    rt_to_str(resource_scope));
11687c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("Valid commands:\n"));
11697c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
11707c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_NAME), gettext("<name>"));
11717c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=boolean\n",
11727c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_SET), pt_to_str(PT_TYPE));
11737c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=true | false\n",
11747c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_SET), pt_to_str(PT_VALUE));
11757c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("or\n"));
11767c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=int\n", cmd_to_str(CMD_SET),
11777c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_TYPE));
11787c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
11797c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_VALUE), gettext("<integer>"));
11807c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("or\n"));
11817c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=string\n",
11827c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_SET), pt_to_str(PT_TYPE));
11837c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
11847c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_VALUE), gettext("<string>"));
11857c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, gettext("or\n"));
11867c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=uint\n",
11877c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_SET), pt_to_str(PT_TYPE));
11887c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
11897c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_VALUE), gettext("<unsigned integer>"));
11907c478bd9Sstevel@tonic-gate 			break;
1191fa9e4066Sahrens 		case RT_DATASET:
1192fa9e4066Sahrens 			(void) fprintf(fp, gettext("The '%s' resource scope is "
1193fa9e4066Sahrens 			    "used to export ZFS datasets.\n"),
1194fa9e4066Sahrens 			    rt_to_str(resource_scope));
1195fa9e4066Sahrens 			(void) fprintf(fp, gettext("Valid commands:\n"));
1196fa9e4066Sahrens 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1197fa9e4066Sahrens 			    pt_to_str(PT_NAME), gettext("<name>"));
1198fa9e4066Sahrens 			break;
11990209230bSgjelinek 		case RT_DCPU:
12000209230bSgjelinek 			(void) fprintf(fp, gettext("The '%s' resource scope "
12010209230bSgjelinek 			    "configures the 'pools' facility to dedicate\na "
12020209230bSgjelinek 			    "subset of the system's processors to this zone "
12030209230bSgjelinek 			    "while it is running.\n"),
12040209230bSgjelinek 			    rt_to_str(resource_scope));
12050209230bSgjelinek 			(void) fprintf(fp, gettext("Valid commands:\n"));
12060209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
12070209230bSgjelinek 			    pt_to_str(PT_NCPUS),
12080209230bSgjelinek 			    gettext("<unsigned integer | range>"));
12090209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
12100209230bSgjelinek 			    pt_to_str(PT_IMPORTANCE),
12110209230bSgjelinek 			    gettext("<unsigned integer>"));
12120209230bSgjelinek 			break;
1213c97ad5cdSakolb 		case RT_PCAP:
1214c97ad5cdSakolb 			(void) fprintf(fp, gettext("The '%s' resource scope is "
1215c97ad5cdSakolb 			    "used to set an upper limit (a cap) on the\n"
1216c97ad5cdSakolb 			    "percentage of CPU that can be used by this zone.  "
1217c97ad5cdSakolb 			    "A '%s' value of 1\ncorresponds to one cpu.  The "
1218c97ad5cdSakolb 			    "value can be set higher than 1, up to the total\n"
1219c97ad5cdSakolb 			    "number of CPUs on the system.  The value can "
1220c97ad5cdSakolb 			    "also be less than 1,\nrepresenting a fraction of "
1221c97ad5cdSakolb 			    "a cpu.\n"),
1222c97ad5cdSakolb 			    rt_to_str(resource_scope), pt_to_str(PT_NCPUS));
1223c97ad5cdSakolb 			(void) fprintf(fp, gettext("Valid commands:\n"));
1224c97ad5cdSakolb 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1225c97ad5cdSakolb 			    pt_to_str(PT_NCPUS), gettext("<unsigned decimal>"));
1226c97ad5cdSakolb 			break;
12270209230bSgjelinek 		case RT_MCAP:
12280209230bSgjelinek 			(void) fprintf(fp, gettext("The '%s' resource scope is "
12290209230bSgjelinek 			    "used to set an upper limit (a cap) on the\n"
12300209230bSgjelinek 			    "amount of physical memory, swap space and locked "
12310209230bSgjelinek 			    "memory that can be used by\nthis zone.\n"),
12320209230bSgjelinek 			    rt_to_str(resource_scope));
12330209230bSgjelinek 			(void) fprintf(fp, gettext("Valid commands:\n"));
12340209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
12350209230bSgjelinek 			    pt_to_str(PT_PHYSICAL),
12360209230bSgjelinek 			    gettext("<qualified unsigned decimal>"));
12370209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
12380209230bSgjelinek 			    pt_to_str(PT_SWAP),
12390209230bSgjelinek 			    gettext("<qualified unsigned decimal>"));
12400209230bSgjelinek 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
12410209230bSgjelinek 			    pt_to_str(PT_LOCKED),
12420209230bSgjelinek 			    gettext("<qualified unsigned decimal>"));
12430209230bSgjelinek 			break;
1244cb8a054bSGlenn Faden 		case RT_ADMIN:
1245cb8a054bSGlenn Faden 			(void) fprintf(fp, gettext("The '%s' resource scope is "
1246cb8a054bSGlenn Faden 			    "used to delegate specific zone management\n"
1247cb8a054bSGlenn Faden 			    "rights to users and roles. These rights are "
1248cb8a054bSGlenn Faden 			    "only applicable to this zone.\n"),
1249cb8a054bSGlenn Faden 			    rt_to_str(resource_scope));
1250cb8a054bSGlenn Faden 			(void) fprintf(fp, gettext("Valid commands:\n"));
1251cb8a054bSGlenn Faden 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1252cb8a054bSGlenn Faden 			    pt_to_str(PT_USER),
1253cb8a054bSGlenn Faden 			    gettext("<single user or role name>"));
1254cb8a054bSGlenn Faden 			(void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET),
1255cb8a054bSGlenn Faden 			    pt_to_str(PT_AUTHS),
1256cb8a054bSGlenn Faden 			    gettext("<comma separated list>"));
1257cb8a054bSGlenn Faden 			break;
1258d2a70789SRichard Lowe 		case RT_SECFLAGS:
1259d2a70789SRichard Lowe 			(void) fprintf(fp, gettext("The '%s' resource scope is "
1260d2a70789SRichard Lowe 			    "used to specify the default security-flags\n"
1261d2a70789SRichard Lowe 			    "of this zone, and their upper and lower bound.\n"),
1262d2a70789SRichard Lowe 			    rt_to_str(resource_scope));
1263d2a70789SRichard Lowe 			(void) fprintf(fp, "\t%s %s=%s\n",
1264d2a70789SRichard Lowe 			    cmd_to_str(CMD_SET), pt_to_str(PT_DEFAULT),
1265d2a70789SRichard Lowe 			    gettext("<security flags>"));
1266d2a70789SRichard Lowe 			(void) fprintf(fp, "\t%s %s=%s\n",
1267d2a70789SRichard Lowe 			    cmd_to_str(CMD_SET), pt_to_str(PT_LOWER),
1268d2a70789SRichard Lowe 			    gettext("<security flags>"));
1269d2a70789SRichard Lowe 			(void) fprintf(fp, "\t%s %s=%s\n",
1270d2a70789SRichard Lowe 			    cmd_to_str(CMD_SET), pt_to_str(PT_UPPER),
1271d2a70789SRichard Lowe 			    gettext("<security flags>"));
1272d2a70789SRichard Lowe 			break;
12737c478bd9Sstevel@tonic-gate 		}
12747c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("And from any resource scope, you "
12757c478bd9Sstevel@tonic-gate 		    "can:\n"));
12767c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", cmd_to_str(CMD_END),
12777c478bd9Sstevel@tonic-gate 		    gettext("(to conclude this operation)"));
12787c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", cmd_to_str(CMD_CANCEL),
12797c478bd9Sstevel@tonic-gate 		    gettext("(to cancel this operation)"));
12807c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", cmd_to_str(CMD_EXIT),
12817c478bd9Sstevel@tonic-gate 		    gettext("(to exit the zonecfg utility)"));
12827c478bd9Sstevel@tonic-gate 	}
12837c478bd9Sstevel@tonic-gate 	if (flags & HELP_USAGE) {
12847c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s:\t%s %s\n", gettext("usage"),
12857c478bd9Sstevel@tonic-gate 		    execname, cmd_to_str(CMD_HELP));
12867c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s -z <zone>\t\t\t(%s)\n",
12877c478bd9Sstevel@tonic-gate 		    execname, gettext("interactive"));
12887c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s -z <zone> <command>\n", execname);
12897c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s -z <zone> -f <command-file>\n",
12907c478bd9Sstevel@tonic-gate 		    execname);
12917c478bd9Sstevel@tonic-gate 	}
12927c478bd9Sstevel@tonic-gate 	if (flags & HELP_SUBCMDS) {
12937c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s:\n\n", gettext("Commands"));
12947c478bd9Sstevel@tonic-gate 		for (i = 0; i <= CMD_MAX; i++) {
12957c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "%s\n", helptab[i].short_usage);
12967c478bd9Sstevel@tonic-gate 			if (verbose)
12977c478bd9Sstevel@tonic-gate 				(void) fprintf(fp, "\t%s\n\n", long_help(i));
12987c478bd9Sstevel@tonic-gate 		}
12997c478bd9Sstevel@tonic-gate 	}
13007c478bd9Sstevel@tonic-gate 	if (flags & HELP_SYNTAX) {
13017c478bd9Sstevel@tonic-gate 		if (!verbose)
13027c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\n");
13037c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "<zone> := [A-Za-z0-9][A-Za-z0-9_.-]*\n");
13047c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("\t(except the reserved words "
13057c478bd9Sstevel@tonic-gate 		    "'%s' and anything starting with '%s')\n"), "global",
13067c478bd9Sstevel@tonic-gate 		    "SUNW");
13077c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
13087c478bd9Sstevel@tonic-gate 		    gettext("\tName must be less than %d characters.\n"),
13097c478bd9Sstevel@tonic-gate 		    ZONENAME_MAX);
13107c478bd9Sstevel@tonic-gate 		if (verbose)
13117c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\n");
13127c478bd9Sstevel@tonic-gate 	}
13137c478bd9Sstevel@tonic-gate 	if (flags & HELP_NETADDR) {
13147c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("\n<net-addr> :="));
13157c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
13167c478bd9Sstevel@tonic-gate 		    gettext("\t<IPv4-address>[/<IPv4-prefix-length>] |\n"));
13177c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
13187c478bd9Sstevel@tonic-gate 		    gettext("\t\t<IPv6-address>/<IPv6-prefix-length> |\n"));
13197c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
13207c478bd9Sstevel@tonic-gate 		    gettext("\t\t<hostname>[/<IPv4-prefix-length>]\n"));
13217c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("See inet(3SOCKET) for IPv4 and "
13227c478bd9Sstevel@tonic-gate 		    "IPv6 address syntax.\n"));
13237c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("<IPv4-prefix-length> := [0-32]\n"));
13247c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
13257c478bd9Sstevel@tonic-gate 		    gettext("<IPv6-prefix-length> := [0-128]\n"));
13267c478bd9Sstevel@tonic-gate 		(void) fprintf(fp,
13277c478bd9Sstevel@tonic-gate 		    gettext("<hostname> := [A-Za-z0-9][A-Za-z0-9-.]*\n"));
13287c478bd9Sstevel@tonic-gate 	}
13297c478bd9Sstevel@tonic-gate 	if (flags & HELP_RESOURCES) {
13306e1ae2a3SGary Pennington 		(void) fprintf(fp, "<%s> := %s | %s | %s | %s | %s |\n\t"
1331cb8a054bSGlenn Faden 		    "%s | %s | %s | %s | %s\n\n",
13327c478bd9Sstevel@tonic-gate 		    gettext("resource type"), rt_to_str(RT_FS),
13336e1ae2a3SGary Pennington 		    rt_to_str(RT_NET), rt_to_str(RT_DEVICE),
13349e7542f4Sdp 		    rt_to_str(RT_RCTL), rt_to_str(RT_ATTR),
13350209230bSgjelinek 		    rt_to_str(RT_DATASET), rt_to_str(RT_DCPU),
1336cb8a054bSGlenn Faden 		    rt_to_str(RT_PCAP), rt_to_str(RT_MCAP),
1337d2a70789SRichard Lowe 		    rt_to_str(RT_ADMIN), rt_to_str(RT_SECFLAGS));
13387c478bd9Sstevel@tonic-gate 	}
13397c478bd9Sstevel@tonic-gate 	if (flags & HELP_PROPS) {
13407c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, gettext("For resource type ... there are "
13417c478bd9Sstevel@tonic-gate 		    "property types ...:\n"));
13427c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
1343087719fdSdp 		    pt_to_str(PT_ZONENAME));
1344087719fdSdp 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13457c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_ZONEPATH));
13467c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13479acbbeafSnn35248 		    pt_to_str(PT_BRAND));
13489acbbeafSnn35248 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13497c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_AUTOBOOT));
13507c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13513f2f09c1Sdp 		    pt_to_str(PT_BOOTARGS));
13523f2f09c1Sdp 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13537c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_POOL));
1354ffbafc53Scomay 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
1355ffbafc53Scomay 		    pt_to_str(PT_LIMITPRIV));
13560209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13570209230bSgjelinek 		    pt_to_str(PT_SCHED));
13580209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
1359f4b3ec61Sdh155122 		    pt_to_str(PT_IPTYPE));
1360f4b3ec61Sdh155122 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13615679c89fSjv227347 		    pt_to_str(PT_HOSTID));
13625679c89fSjv227347 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13630fbb751dSJohn Levon 		    pt_to_str(PT_FS_ALLOWED));
13640fbb751dSJohn Levon 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13650209230bSgjelinek 		    pt_to_str(PT_MAXLWPS));
13660209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
1367ff19e029SMenno Lageman 		    pt_to_str(PT_MAXPROCS));
1368ff19e029SMenno Lageman 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13690209230bSgjelinek 		    pt_to_str(PT_MAXSHMMEM));
13700209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13710209230bSgjelinek 		    pt_to_str(PT_MAXSHMIDS));
13720209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13730209230bSgjelinek 		    pt_to_str(PT_MAXMSGIDS));
13740209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13750209230bSgjelinek 		    pt_to_str(PT_MAXSEMIDS));
13760209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s\n", gettext("(global)"),
13770209230bSgjelinek 		    pt_to_str(PT_SHARES));
13786d4d1c0dSbatschul 		(void) fprintf(fp, "\t%s\t\t%s, %s, %s, %s, %s\n",
13796d4d1c0dSbatschul 		    rt_to_str(RT_FS), pt_to_str(PT_DIR),
13806d4d1c0dSbatschul 		    pt_to_str(PT_SPECIAL), pt_to_str(PT_RAW),
13816d4d1c0dSbatschul 		    pt_to_str(PT_TYPE), pt_to_str(PT_OPTIONS));
1382550b6e40SSowmini Varadhan 		(void) fprintf(fp, "\t%s\t\t%s, %s, %s|%s\n", rt_to_str(RT_NET),
1383550b6e40SSowmini Varadhan 		    pt_to_str(PT_ADDRESS), pt_to_str(PT_ALLOWED_ADDRESS),
1384550b6e40SSowmini Varadhan 		    pt_to_str(PT_PHYSICAL), pt_to_str(PT_DEFROUTER));
13857c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t\t%s\n", rt_to_str(RT_DEVICE),
13867c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_MATCH));
13877c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t\t%s, %s\n", rt_to_str(RT_RCTL),
13887c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_NAME), pt_to_str(PT_VALUE));
13897c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "\t%s\t\t%s, %s, %s\n", rt_to_str(RT_ATTR),
13907c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_NAME), pt_to_str(PT_TYPE),
13917c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_VALUE));
1392fa9e4066Sahrens 		(void) fprintf(fp, "\t%s\t\t%s\n", rt_to_str(RT_DATASET),
1393fa9e4066Sahrens 		    pt_to_str(PT_NAME));
13940209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s, %s\n", rt_to_str(RT_DCPU),
13950209230bSgjelinek 		    pt_to_str(PT_NCPUS), pt_to_str(PT_IMPORTANCE));
1396c97ad5cdSakolb 		(void) fprintf(fp, "\t%s\t%s\n", rt_to_str(RT_PCAP),
1397c97ad5cdSakolb 		    pt_to_str(PT_NCPUS));
13980209230bSgjelinek 		(void) fprintf(fp, "\t%s\t%s, %s, %s\n", rt_to_str(RT_MCAP),
13990209230bSgjelinek 		    pt_to_str(PT_PHYSICAL), pt_to_str(PT_SWAP),
14000209230bSgjelinek 		    pt_to_str(PT_LOCKED));
1401cb8a054bSGlenn Faden 		(void) fprintf(fp, "\t%s\t\t%s, %s\n", rt_to_str(RT_ADMIN),
1402cb8a054bSGlenn Faden 		    pt_to_str(PT_USER), pt_to_str(PT_AUTHS));
1403d2a70789SRichard Lowe 		(void) fprintf(fp, "\t%s\t\t%s, %s, %s\n",
1404d2a70789SRichard Lowe 		    rt_to_str(RT_SECFLAGS), pt_to_str(PT_DEFAULT),
1405d2a70789SRichard Lowe 		    pt_to_str(PT_LOWER), pt_to_str(PT_UPPER));
14067c478bd9Sstevel@tonic-gate 	}
14077c478bd9Sstevel@tonic-gate 	if (need_to_close)
1408944b13ecSGary Mills 		(void) pager_close(fp);
14097c478bd9Sstevel@tonic-gate }
14107c478bd9Sstevel@tonic-gate 
14117c478bd9Sstevel@tonic-gate static void
zone_perror(char * prefix,int err,boolean_t set_saw)1412bbec428eSgjelinek zone_perror(char *prefix, int err, boolean_t set_saw)
14137c478bd9Sstevel@tonic-gate {
14147c478bd9Sstevel@tonic-gate 	zerr("%s: %s", prefix, zonecfg_strerror(err));
14157c478bd9Sstevel@tonic-gate 	if (set_saw)
1416bbec428eSgjelinek 		saw_error = B_TRUE;
14177c478bd9Sstevel@tonic-gate }
14187c478bd9Sstevel@tonic-gate 
14197c478bd9Sstevel@tonic-gate /*
14207c478bd9Sstevel@tonic-gate  * zone_perror() expects a single string, but for remove and select
14217c478bd9Sstevel@tonic-gate  * we have both the command and the resource type, so this wrapper
14227c478bd9Sstevel@tonic-gate  * function serves the same purpose in a slightly different way.
14237c478bd9Sstevel@tonic-gate  */
14247c478bd9Sstevel@tonic-gate 
14257c478bd9Sstevel@tonic-gate static void
z_cmd_rt_perror(int cmd_num,int res_num,int err,boolean_t set_saw)1426bbec428eSgjelinek z_cmd_rt_perror(int cmd_num, int res_num, int err, boolean_t set_saw)
14277c478bd9Sstevel@tonic-gate {
14287c478bd9Sstevel@tonic-gate 	zerr("%s %s: %s", cmd_to_str(cmd_num), rt_to_str(res_num),
14297c478bd9Sstevel@tonic-gate 	    zonecfg_strerror(err));
14307c478bd9Sstevel@tonic-gate 	if (set_saw)
1431bbec428eSgjelinek 		saw_error = B_TRUE;
14327c478bd9Sstevel@tonic-gate }
14337c478bd9Sstevel@tonic-gate 
14347c478bd9Sstevel@tonic-gate /* returns Z_OK if successful, Z_foo from <libzonecfg.h> otherwise */
14357c478bd9Sstevel@tonic-gate static int
initialize(boolean_t handle_expected)1436bbec428eSgjelinek initialize(boolean_t handle_expected)
14377c478bd9Sstevel@tonic-gate {
14387c478bd9Sstevel@tonic-gate 	int err;
14399acbbeafSnn35248 	char brandname[MAXNAMELEN];
14407c478bd9Sstevel@tonic-gate 
14417c478bd9Sstevel@tonic-gate 	if (zonecfg_check_handle(handle) != Z_OK) {
14427c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_get_handle(zone, handle)) == Z_OK) {
1443bbec428eSgjelinek 			got_handle = B_TRUE;
14449acbbeafSnn35248 			if (zonecfg_get_brand(handle, brandname,
14459acbbeafSnn35248 			    sizeof (brandname)) != Z_OK) {
14469acbbeafSnn35248 				zerr("Zone %s is inconsistent: missing "
14479acbbeafSnn35248 				    "brand attribute", zone);
14489acbbeafSnn35248 				exit(Z_ERR);
14499acbbeafSnn35248 			}
14509acbbeafSnn35248 			if ((brand = brand_open(brandname)) == NULL) {
14519acbbeafSnn35248 				zerr("Zone %s uses non-existent brand \"%s\"."
14529acbbeafSnn35248 				    "  Unable to continue", zone, brandname);
14539acbbeafSnn35248 				exit(Z_ERR);
14549acbbeafSnn35248 			}
1455cb8a054bSGlenn Faden 			/*
1456cb8a054bSGlenn Faden 			 * If the user_attr file is newer than
1457cb8a054bSGlenn Faden 			 * the zone config file, the admins
1458cb8a054bSGlenn Faden 			 * may need to be updated since the
1459cb8a054bSGlenn Faden 			 * RBAC files are authoritative for
1460cb8a054bSGlenn Faden 			 * authorization checks.
1461cb8a054bSGlenn Faden 			 */
1462cb8a054bSGlenn Faden 			err = zonecfg_update_userauths(handle, zone);
1463cb8a054bSGlenn Faden 			if (err == Z_OK) {
1464cb8a054bSGlenn Faden 				zerr(gettext("The administrative rights "
1465cb8a054bSGlenn Faden 				    "were updated to match "
1466cb8a054bSGlenn Faden 				    "the current RBAC configuration.\n"
1467cb8a054bSGlenn Faden 				    "Use \"info admin\" and \"revert\" to "
1468cb8a054bSGlenn Faden 				    "compare with the previous settings."));
1469cb8a054bSGlenn Faden 				need_to_commit = B_TRUE;
1470cb8a054bSGlenn Faden 			} else if (err != Z_NO_ENTRY) {
1471cb8a054bSGlenn Faden 				zerr(gettext("failed to update "
1472cb8a054bSGlenn Faden 				    "admin  rights."));
1473cb8a054bSGlenn Faden 				exit(Z_ERR);
1474cb8a054bSGlenn Faden 			} else if (need_to_commit) {
1475cb8a054bSGlenn Faden 				zerr(gettext("admin rights were updated "
1476cb8a054bSGlenn Faden 				    "to match RBAC configuration."));
1477cb8a054bSGlenn Faden 			}
1478cb8a054bSGlenn Faden 
14790209230bSgjelinek 		} else if (global_zone && err == Z_NO_ZONE && !got_handle &&
14800209230bSgjelinek 		    !read_only_mode) {
14810209230bSgjelinek 			/*
14820209230bSgjelinek 			 * We implicitly create the global zone config if it
14830209230bSgjelinek 			 * doesn't exist.
14840209230bSgjelinek 			 */
14850209230bSgjelinek 			zone_dochandle_t tmphandle;
14860209230bSgjelinek 
14870209230bSgjelinek 			if ((tmphandle = zonecfg_init_handle()) == NULL) {
1488bbec428eSgjelinek 				zone_perror(execname, Z_NOMEM, B_TRUE);
14890209230bSgjelinek 				exit(Z_ERR);
14900209230bSgjelinek 			}
14910209230bSgjelinek 
14920209230bSgjelinek 			err = zonecfg_get_template_handle("SUNWblank", zone,
14930209230bSgjelinek 			    tmphandle);
14940209230bSgjelinek 
14950209230bSgjelinek 			if (err != Z_OK) {
14960209230bSgjelinek 				zonecfg_fini_handle(tmphandle);
1497bbec428eSgjelinek 				zone_perror("SUNWblank", err, B_TRUE);
14980209230bSgjelinek 				return (err);
14990209230bSgjelinek 			}
15000209230bSgjelinek 
1501bbec428eSgjelinek 			need_to_commit = B_TRUE;
15020209230bSgjelinek 			zonecfg_fini_handle(handle);
15030209230bSgjelinek 			handle = tmphandle;
1504bbec428eSgjelinek 			got_handle = B_TRUE;
15050209230bSgjelinek 
15067c478bd9Sstevel@tonic-gate 		} else {
15077c478bd9Sstevel@tonic-gate 			zone_perror(zone, err, handle_expected || got_handle);
15087c478bd9Sstevel@tonic-gate 			if (err == Z_NO_ZONE && !got_handle &&
15097c478bd9Sstevel@tonic-gate 			    interactive_mode && !read_only_mode)
15107c478bd9Sstevel@tonic-gate 				(void) printf(gettext("Use '%s' to begin "
15117c478bd9Sstevel@tonic-gate 				    "configuring a new zone.\n"),
15127c478bd9Sstevel@tonic-gate 				    cmd_to_str(CMD_CREATE));
15137c478bd9Sstevel@tonic-gate 			return (err);
15147c478bd9Sstevel@tonic-gate 		}
15157c478bd9Sstevel@tonic-gate 	}
15167c478bd9Sstevel@tonic-gate 	return (Z_OK);
15177c478bd9Sstevel@tonic-gate }
15187c478bd9Sstevel@tonic-gate 
1519bbec428eSgjelinek static boolean_t
state_atleast(zone_state_t state)1520087719fdSdp state_atleast(zone_state_t state)
1521087719fdSdp {
1522087719fdSdp 	zone_state_t state_num;
1523087719fdSdp 	int err;
1524087719fdSdp 
1525087719fdSdp 	if ((err = zone_get_state(zone, &state_num)) != Z_OK) {
1526087719fdSdp 		/* all states are greater than "non-existent" */
1527087719fdSdp 		if (err == Z_NO_ZONE)
1528087719fdSdp 			return (B_FALSE);
1529087719fdSdp 		zerr(gettext("Unexpectedly failed to determine state "
1530087719fdSdp 		    "of zone %s: %s"), zone, zonecfg_strerror(err));
1531087719fdSdp 		exit(Z_ERR);
1532087719fdSdp 	}
1533087719fdSdp 	return (state_num >= state);
1534087719fdSdp }
1535087719fdSdp 
15367c478bd9Sstevel@tonic-gate /*
15377c478bd9Sstevel@tonic-gate  * short_usage() is for bad syntax: getopt() issues, too many arguments, etc.
15387c478bd9Sstevel@tonic-gate  */
15397c478bd9Sstevel@tonic-gate 
15407c478bd9Sstevel@tonic-gate void
short_usage(int command)15417c478bd9Sstevel@tonic-gate short_usage(int command)
15427c478bd9Sstevel@tonic-gate {
15437c478bd9Sstevel@tonic-gate 	/* lex_lineno has already been incremented in the lexer; compensate */
15447c478bd9Sstevel@tonic-gate 	if (cmd_file_mode) {
15457c478bd9Sstevel@tonic-gate 		if (strcmp(cmd_file_name, "-") == 0)
15467c478bd9Sstevel@tonic-gate 			(void) fprintf(stderr,
15477c478bd9Sstevel@tonic-gate 			    gettext("syntax error on line %d\n"),
15487c478bd9Sstevel@tonic-gate 			    lex_lineno - 1);
15497c478bd9Sstevel@tonic-gate 		else
15507c478bd9Sstevel@tonic-gate 			(void) fprintf(stderr,
15517c478bd9Sstevel@tonic-gate 			    gettext("syntax error on line %d of %s\n"),
15527c478bd9Sstevel@tonic-gate 			    lex_lineno - 1, cmd_file_name);
15537c478bd9Sstevel@tonic-gate 	}
15547c478bd9Sstevel@tonic-gate 	(void) fprintf(stderr, "%s:\n%s\n", gettext("usage"),
15557c478bd9Sstevel@tonic-gate 	    helptab[command].short_usage);
1556bbec428eSgjelinek 	saw_error = B_TRUE;
15577c478bd9Sstevel@tonic-gate }
15587c478bd9Sstevel@tonic-gate 
15597c478bd9Sstevel@tonic-gate /*
15607c478bd9Sstevel@tonic-gate  * long_usage() is for bad semantics: e.g., wrong property type for a given
15617c478bd9Sstevel@tonic-gate  * resource type.  It is also used by longer_usage() below.
15627c478bd9Sstevel@tonic-gate  */
15637c478bd9Sstevel@tonic-gate 
15647c478bd9Sstevel@tonic-gate void
long_usage(uint_t cmd_num,boolean_t set_saw)1565bbec428eSgjelinek long_usage(uint_t cmd_num, boolean_t set_saw)
15667c478bd9Sstevel@tonic-gate {
15677c478bd9Sstevel@tonic-gate 	(void) fprintf(set_saw ? stderr : stdout, "%s:\n%s\n", gettext("usage"),
15687c478bd9Sstevel@tonic-gate 	    helptab[cmd_num].short_usage);
15697c478bd9Sstevel@tonic-gate 	(void) fprintf(set_saw ? stderr : stdout, "\t%s\n", long_help(cmd_num));
15707c478bd9Sstevel@tonic-gate 	if (set_saw)
1571bbec428eSgjelinek 		saw_error = B_TRUE;
15727c478bd9Sstevel@tonic-gate }
15737c478bd9Sstevel@tonic-gate 
15747c478bd9Sstevel@tonic-gate /*
15757c478bd9Sstevel@tonic-gate  * longer_usage() is for 'help foo' and 'foo -?': call long_usage() and also
15767c478bd9Sstevel@tonic-gate  * any extra usage() flags as appropriate for whatever command.
15777c478bd9Sstevel@tonic-gate  */
15787c478bd9Sstevel@tonic-gate 
15797c478bd9Sstevel@tonic-gate void
longer_usage(uint_t cmd_num)15807c478bd9Sstevel@tonic-gate longer_usage(uint_t cmd_num)
15817c478bd9Sstevel@tonic-gate {
1582bbec428eSgjelinek 	long_usage(cmd_num, B_FALSE);
15837c478bd9Sstevel@tonic-gate 	if (helptab[cmd_num].flags != 0) {
15847c478bd9Sstevel@tonic-gate 		(void) printf("\n");
1585bbec428eSgjelinek 		usage(B_TRUE, helptab[cmd_num].flags);
15867c478bd9Sstevel@tonic-gate 	}
15877c478bd9Sstevel@tonic-gate }
15887c478bd9Sstevel@tonic-gate 
15897c478bd9Sstevel@tonic-gate /*
15907c478bd9Sstevel@tonic-gate  * scope_usage() is simply used when a command is called from the wrong scope.
15917c478bd9Sstevel@tonic-gate  */
15927c478bd9Sstevel@tonic-gate 
15937c478bd9Sstevel@tonic-gate static void
scope_usage(uint_t cmd_num)15947c478bd9Sstevel@tonic-gate scope_usage(uint_t cmd_num)
15957c478bd9Sstevel@tonic-gate {
15967c478bd9Sstevel@tonic-gate 	zerr(gettext("The %s command only makes sense in the %s scope."),
15977c478bd9Sstevel@tonic-gate 	    cmd_to_str(cmd_num),
15987c478bd9Sstevel@tonic-gate 	    global_scope ?  gettext("resource") : gettext("global"));
1599bbec428eSgjelinek 	saw_error = B_TRUE;
16007c478bd9Sstevel@tonic-gate }
16017c478bd9Sstevel@tonic-gate 
16027c478bd9Sstevel@tonic-gate /*
1603bbec428eSgjelinek  * On input, B_TRUE => yes, B_FALSE => no.
1604bbec428eSgjelinek  * On return, B_TRUE => 1, B_FALSE => no, could not ask => -1.
16057c478bd9Sstevel@tonic-gate  */
16067c478bd9Sstevel@tonic-gate 
16077c478bd9Sstevel@tonic-gate static int
ask_yesno(boolean_t default_answer,const char * question)1608bbec428eSgjelinek ask_yesno(boolean_t default_answer, const char *question)
16097c478bd9Sstevel@tonic-gate {
16107c478bd9Sstevel@tonic-gate 	char line[64];	/* should be enough to answer yes or no */
16117c478bd9Sstevel@tonic-gate 
16127c478bd9Sstevel@tonic-gate 	if (!ok_to_prompt) {
1613bbec428eSgjelinek 		saw_error = B_TRUE;
16147c478bd9Sstevel@tonic-gate 		return (-1);
16157c478bd9Sstevel@tonic-gate 	}
16167c478bd9Sstevel@tonic-gate 	for (;;) {
1617087719fdSdp 		if (printf("%s (%s)? ", question,
1618087719fdSdp 		    default_answer ? "[y]/n" : "y/[n]") < 0)
1619087719fdSdp 			return (-1);
1620087719fdSdp 		if (fgets(line, sizeof (line), stdin) == NULL)
1621087719fdSdp 			return (-1);
1622087719fdSdp 
1623087719fdSdp 		if (line[0] == '\n')
16247c478bd9Sstevel@tonic-gate 			return (default_answer ? 1 : 0);
16257c478bd9Sstevel@tonic-gate 		if (tolower(line[0]) == 'y')
16267c478bd9Sstevel@tonic-gate 			return (1);
16277c478bd9Sstevel@tonic-gate 		if (tolower(line[0]) == 'n')
16287c478bd9Sstevel@tonic-gate 			return (0);
16297c478bd9Sstevel@tonic-gate 	}
16307c478bd9Sstevel@tonic-gate }
16317c478bd9Sstevel@tonic-gate 
16327c478bd9Sstevel@tonic-gate /*
16337c478bd9Sstevel@tonic-gate  * Prints warning if zone already exists.
16347c478bd9Sstevel@tonic-gate  * In interactive mode, prompts if we should continue anyway and returns Z_OK
16357c478bd9Sstevel@tonic-gate  * if so, Z_ERR if not.  In non-interactive mode, exits with Z_ERR.
16367c478bd9Sstevel@tonic-gate  *
16377c478bd9Sstevel@tonic-gate  * Note that if a zone exists and its state is >= INSTALLED, an error message
16387c478bd9Sstevel@tonic-gate  * will be printed and this function will return Z_ERR regardless of mode.
16397c478bd9Sstevel@tonic-gate  */
16407c478bd9Sstevel@tonic-gate 
16417c478bd9Sstevel@tonic-gate static int
check_if_zone_already_exists(boolean_t force)1642bbec428eSgjelinek check_if_zone_already_exists(boolean_t force)
16437c478bd9Sstevel@tonic-gate {
16447c478bd9Sstevel@tonic-gate 	char line[ZONENAME_MAX + 128];	/* enough to ask a question */
16457c478bd9Sstevel@tonic-gate 	zone_dochandle_t tmphandle;
16467c478bd9Sstevel@tonic-gate 	int res, answer;
16477c478bd9Sstevel@tonic-gate 
16487c478bd9Sstevel@tonic-gate 	if ((tmphandle = zonecfg_init_handle()) == NULL) {
1649bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
16507c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
16517c478bd9Sstevel@tonic-gate 	}
16527c478bd9Sstevel@tonic-gate 	res = zonecfg_get_handle(zone, tmphandle);
16537c478bd9Sstevel@tonic-gate 	zonecfg_fini_handle(tmphandle);
1654087719fdSdp 	if (res != Z_OK)
16557c478bd9Sstevel@tonic-gate 		return (Z_OK);
1656087719fdSdp 
1657087719fdSdp 	if (state_atleast(ZONE_STATE_INSTALLED)) {
16587c478bd9Sstevel@tonic-gate 		zerr(gettext("Zone %s already installed; %s not allowed."),
16597c478bd9Sstevel@tonic-gate 		    zone, cmd_to_str(CMD_CREATE));
16607c478bd9Sstevel@tonic-gate 		return (Z_ERR);
16617c478bd9Sstevel@tonic-gate 	}
16627c478bd9Sstevel@tonic-gate 
16637c478bd9Sstevel@tonic-gate 	if (force) {
16647c478bd9Sstevel@tonic-gate 		(void) printf(gettext("Zone %s already exists; overwriting.\n"),
16657c478bd9Sstevel@tonic-gate 		    zone);
16667c478bd9Sstevel@tonic-gate 		return (Z_OK);
16677c478bd9Sstevel@tonic-gate 	}
16687c478bd9Sstevel@tonic-gate 	(void) snprintf(line, sizeof (line),
16697c478bd9Sstevel@tonic-gate 	    gettext("Zone %s already exists; %s anyway"), zone,
16707c478bd9Sstevel@tonic-gate 	    cmd_to_str(CMD_CREATE));
1671bbec428eSgjelinek 	if ((answer = ask_yesno(B_FALSE, line)) == -1) {
16727c478bd9Sstevel@tonic-gate 		zerr(gettext("Zone exists, input not from terminal and -F not "
16737c478bd9Sstevel@tonic-gate 		    "specified:\n%s command ignored, exiting."),
16747c478bd9Sstevel@tonic-gate 		    cmd_to_str(CMD_CREATE));
16757c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
16767c478bd9Sstevel@tonic-gate 	}
16777c478bd9Sstevel@tonic-gate 	return (answer == 1 ? Z_OK : Z_ERR);
16787c478bd9Sstevel@tonic-gate }
16797c478bd9Sstevel@tonic-gate 
1680bbec428eSgjelinek static boolean_t
zone_is_read_only(int cmd_num)16817c478bd9Sstevel@tonic-gate zone_is_read_only(int cmd_num)
16827c478bd9Sstevel@tonic-gate {
16837c478bd9Sstevel@tonic-gate 	if (strncmp(zone, "SUNW", 4) == 0) {
16847c478bd9Sstevel@tonic-gate 		zerr(gettext("%s: zones beginning with SUNW are read-only."),
16857c478bd9Sstevel@tonic-gate 		    zone);
1686bbec428eSgjelinek 		saw_error = B_TRUE;
1687bbec428eSgjelinek 		return (B_TRUE);
16887c478bd9Sstevel@tonic-gate 	}
16897c478bd9Sstevel@tonic-gate 	if (read_only_mode) {
16907c478bd9Sstevel@tonic-gate 		zerr(gettext("%s: cannot %s in read-only mode."), zone,
16917c478bd9Sstevel@tonic-gate 		    cmd_to_str(cmd_num));
1692bbec428eSgjelinek 		saw_error = B_TRUE;
1693bbec428eSgjelinek 		return (B_TRUE);
16947c478bd9Sstevel@tonic-gate 	}
1695bbec428eSgjelinek 	return (B_FALSE);
16967c478bd9Sstevel@tonic-gate }
16977c478bd9Sstevel@tonic-gate 
16987c478bd9Sstevel@tonic-gate /*
16997c478bd9Sstevel@tonic-gate  * Create a new configuration.
17007c478bd9Sstevel@tonic-gate  */
17017c478bd9Sstevel@tonic-gate void
create_func(cmd_t * cmd)17027c478bd9Sstevel@tonic-gate create_func(cmd_t *cmd)
17037c478bd9Sstevel@tonic-gate {
17047c478bd9Sstevel@tonic-gate 	int err, arg;
17057c478bd9Sstevel@tonic-gate 	char zone_template[ZONENAME_MAX];
1706ee519a1fSgjelinek 	char attach_path[MAXPATHLEN];
17077c478bd9Sstevel@tonic-gate 	zone_dochandle_t tmphandle;
1708bbec428eSgjelinek 	boolean_t force = B_FALSE;
1709bbec428eSgjelinek 	boolean_t attach = B_FALSE;
1710bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
17117c478bd9Sstevel@tonic-gate 
17127c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
17137c478bd9Sstevel@tonic-gate 
17147c478bd9Sstevel@tonic-gate 	/* This is the default if no arguments are given. */
17157c478bd9Sstevel@tonic-gate 	(void) strlcpy(zone_template, "SUNWdefault", sizeof (zone_template));
17167c478bd9Sstevel@tonic-gate 
17177c478bd9Sstevel@tonic-gate 	optind = 0;
17189acbbeafSnn35248 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?a:bFt:"))
17199acbbeafSnn35248 	    != EOF) {
17207c478bd9Sstevel@tonic-gate 		switch (arg) {
17217c478bd9Sstevel@tonic-gate 		case '?':
17227c478bd9Sstevel@tonic-gate 			if (optopt == '?')
17237c478bd9Sstevel@tonic-gate 				longer_usage(CMD_CREATE);
17247c478bd9Sstevel@tonic-gate 			else
17257c478bd9Sstevel@tonic-gate 				short_usage(CMD_CREATE);
1726bbec428eSgjelinek 			arg_err = B_TRUE;
17277ec75eb8Sgjelinek 			break;
1728ee519a1fSgjelinek 		case 'a':
1729ee519a1fSgjelinek 			(void) strlcpy(attach_path, optarg,
1730ee519a1fSgjelinek 			    sizeof (attach_path));
1731bbec428eSgjelinek 			attach = B_TRUE;
1732ee519a1fSgjelinek 			break;
17337c478bd9Sstevel@tonic-gate 		case 'b':
17347c478bd9Sstevel@tonic-gate 			(void) strlcpy(zone_template, "SUNWblank",
17357c478bd9Sstevel@tonic-gate 			    sizeof (zone_template));
17367c478bd9Sstevel@tonic-gate 			break;
17377c478bd9Sstevel@tonic-gate 		case 'F':
1738bbec428eSgjelinek 			force = B_TRUE;
17397c478bd9Sstevel@tonic-gate 			break;
17407c478bd9Sstevel@tonic-gate 		case 't':
17417c478bd9Sstevel@tonic-gate 			(void) strlcpy(zone_template, optarg,
17427c478bd9Sstevel@tonic-gate 			    sizeof (zone_template));
17437c478bd9Sstevel@tonic-gate 			break;
17447c478bd9Sstevel@tonic-gate 		default:
17457c478bd9Sstevel@tonic-gate 			short_usage(CMD_CREATE);
1746bbec428eSgjelinek 			arg_err = B_TRUE;
17477ec75eb8Sgjelinek 			break;
17487ec75eb8Sgjelinek 		}
17497ec75eb8Sgjelinek 	}
17507ec75eb8Sgjelinek 	if (arg_err)
17517c478bd9Sstevel@tonic-gate 		return;
17527ec75eb8Sgjelinek 
17537c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
17547c478bd9Sstevel@tonic-gate 		short_usage(CMD_CREATE);
17557c478bd9Sstevel@tonic-gate 		return;
17567c478bd9Sstevel@tonic-gate 	}
17577c478bd9Sstevel@tonic-gate 
17587c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_CREATE))
17597c478bd9Sstevel@tonic-gate 		return;
17607c478bd9Sstevel@tonic-gate 
17617c478bd9Sstevel@tonic-gate 	if (check_if_zone_already_exists(force) != Z_OK)
17627c478bd9Sstevel@tonic-gate 		return;
17637c478bd9Sstevel@tonic-gate 
17647c478bd9Sstevel@tonic-gate 	/*
17657c478bd9Sstevel@tonic-gate 	 * Get a temporary handle first.  If that fails, the old handle
17667c478bd9Sstevel@tonic-gate 	 * will not be lost.  Then finish whichever one we don't need,
17677c478bd9Sstevel@tonic-gate 	 * to avoid leaks.  Then get the handle for zone_template, and
17687c478bd9Sstevel@tonic-gate 	 * set the name to zone: this "copy, rename" method is how
17697c478bd9Sstevel@tonic-gate 	 * create -[b|t] works.
17707c478bd9Sstevel@tonic-gate 	 */
17717c478bd9Sstevel@tonic-gate 	if ((tmphandle = zonecfg_init_handle()) == NULL) {
1772bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
17737c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
17747c478bd9Sstevel@tonic-gate 	}
1775ee519a1fSgjelinek 
1776ee519a1fSgjelinek 	if (attach)
177716ab8c7bSgjelinek 		err = zonecfg_get_attach_handle(attach_path, ZONE_DETACHED,
177816ab8c7bSgjelinek 		    zone, B_FALSE, tmphandle);
1779ee519a1fSgjelinek 	else
1780ee519a1fSgjelinek 		err = zonecfg_get_template_handle(zone_template, zone,
1781ee519a1fSgjelinek 		    tmphandle);
1782ee519a1fSgjelinek 
1783ee519a1fSgjelinek 	if (err != Z_OK) {
17847c478bd9Sstevel@tonic-gate 		zonecfg_fini_handle(tmphandle);
1785ee519a1fSgjelinek 		if (attach && err == Z_NO_ZONE)
1786ee519a1fSgjelinek 			(void) fprintf(stderr, gettext("invalid path to "
1787ee519a1fSgjelinek 			    "detached zone\n"));
1788ee519a1fSgjelinek 		else if (attach && err == Z_INVALID_DOCUMENT)
1789ee519a1fSgjelinek 			(void) fprintf(stderr, gettext("Cannot attach to an "
1790ee519a1fSgjelinek 			    "earlier release of the operating system\n"));
1791ee519a1fSgjelinek 		else
1792bbec428eSgjelinek 			zone_perror(zone_template, err, B_TRUE);
17937c478bd9Sstevel@tonic-gate 		return;
17947c478bd9Sstevel@tonic-gate 	}
1795087719fdSdp 
1796bbec428eSgjelinek 	need_to_commit = B_TRUE;
17977c478bd9Sstevel@tonic-gate 	zonecfg_fini_handle(handle);
17987c478bd9Sstevel@tonic-gate 	handle = tmphandle;
1799bbec428eSgjelinek 	got_handle = B_TRUE;
18007c478bd9Sstevel@tonic-gate }
18017c478bd9Sstevel@tonic-gate 
18027c478bd9Sstevel@tonic-gate /*
18037c478bd9Sstevel@tonic-gate  * This malloc()'s memory, which must be freed by the caller.
18047c478bd9Sstevel@tonic-gate  */
18057c478bd9Sstevel@tonic-gate static char *
quoteit(char * instr)18067c478bd9Sstevel@tonic-gate quoteit(char *instr)
18077c478bd9Sstevel@tonic-gate {
18087c478bd9Sstevel@tonic-gate 	char *outstr;
18097c478bd9Sstevel@tonic-gate 	size_t outstrsize = strlen(instr) + 3;	/* 2 quotes + '\0' */
18107c478bd9Sstevel@tonic-gate 
18117c478bd9Sstevel@tonic-gate 	if ((outstr = malloc(outstrsize)) == NULL) {
1812bbec428eSgjelinek 		zone_perror(zone, Z_NOMEM, B_FALSE);
18137c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
18147c478bd9Sstevel@tonic-gate 	}
18157c478bd9Sstevel@tonic-gate 	if (strchr(instr, ' ') == NULL) {
18167c478bd9Sstevel@tonic-gate 		(void) strlcpy(outstr, instr, outstrsize);
18177c478bd9Sstevel@tonic-gate 		return (outstr);
18187c478bd9Sstevel@tonic-gate 	}
18197c478bd9Sstevel@tonic-gate 	(void) snprintf(outstr, outstrsize, "\"%s\"", instr);
18207c478bd9Sstevel@tonic-gate 	return (outstr);
18217c478bd9Sstevel@tonic-gate }
18227c478bd9Sstevel@tonic-gate 
18237c478bd9Sstevel@tonic-gate static void
export_prop(FILE * of,int prop_num,char * prop_id)18247c478bd9Sstevel@tonic-gate export_prop(FILE *of, int prop_num, char *prop_id)
18257c478bd9Sstevel@tonic-gate {
18267c478bd9Sstevel@tonic-gate 	if (strlen(prop_id) == 0)
18277c478bd9Sstevel@tonic-gate 		return;
1828c8236ea6SBrian Bennett 	/*
1829c8236ea6SBrian Bennett 	 * We're going to explicitly quote all strings on export.
1830c8236ea6SBrian Bennett 	 * This should be fine since it seems that no amount of escaping
1831c8236ea6SBrian Bennett 	 * will coerce zonecfg to properly parse a double quote as
1832c8236ea6SBrian Bennett 	 * part of the string value.
1833c8236ea6SBrian Bennett 	 */
1834c8236ea6SBrian Bennett 	(void) fprintf(of, "%s %s=\"%s\"\n", cmd_to_str(CMD_SET),
1835c8236ea6SBrian Bennett 	    pt_to_str(prop_num), prop_id);
18367c478bd9Sstevel@tonic-gate }
18377c478bd9Sstevel@tonic-gate 
18387c478bd9Sstevel@tonic-gate void
export_func(cmd_t * cmd)18397c478bd9Sstevel@tonic-gate export_func(cmd_t *cmd)
18407c478bd9Sstevel@tonic-gate {
18417c478bd9Sstevel@tonic-gate 	struct zone_nwiftab nwiftab;
18427c478bd9Sstevel@tonic-gate 	struct zone_fstab fstab;
18437c478bd9Sstevel@tonic-gate 	struct zone_devtab devtab;
18447c478bd9Sstevel@tonic-gate 	struct zone_attrtab attrtab;
18457c478bd9Sstevel@tonic-gate 	struct zone_rctltab rctltab;
1846fa9e4066Sahrens 	struct zone_dstab dstab;
18470209230bSgjelinek 	struct zone_psettab psettab;
18480209230bSgjelinek 	struct zone_mcaptab mcaptab;
18497c478bd9Sstevel@tonic-gate 	struct zone_rctlvaltab *valptr;
1850cb8a054bSGlenn Faden 	struct zone_admintab admintab;
1851d2a70789SRichard Lowe 	struct zone_secflagstab secflagstab;
18527c478bd9Sstevel@tonic-gate 	int err, arg;
18537c478bd9Sstevel@tonic-gate 	char zonepath[MAXPATHLEN], outfile[MAXPATHLEN], pool[MAXNAMELEN];
18543f2f09c1Sdp 	char bootargs[BOOTARGS_MAX];
18550209230bSgjelinek 	char sched[MAXNAMELEN];
18569acbbeafSnn35248 	char brand[MAXNAMELEN];
18575679c89fSjv227347 	char hostidp[HW_HOSTID_LEN];
18580fbb751dSJohn Levon 	char fsallowedp[ZONE_FS_ALLOWED_MAX];
1859ffbafc53Scomay 	char *limitpriv;
18607c478bd9Sstevel@tonic-gate 	FILE *of;
18617c478bd9Sstevel@tonic-gate 	boolean_t autoboot;
1862f4b3ec61Sdh155122 	zone_iptype_t iptype;
1863bbec428eSgjelinek 	boolean_t need_to_close = B_FALSE;
1864bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
18657c478bd9Sstevel@tonic-gate 
18667c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
18677c478bd9Sstevel@tonic-gate 
18687c478bd9Sstevel@tonic-gate 	outfile[0] = '\0';
18697c478bd9Sstevel@tonic-gate 	optind = 0;
18707c478bd9Sstevel@tonic-gate 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?f:")) != EOF) {
18717c478bd9Sstevel@tonic-gate 		switch (arg) {
18727c478bd9Sstevel@tonic-gate 		case '?':
18737c478bd9Sstevel@tonic-gate 			if (optopt == '?')
18747c478bd9Sstevel@tonic-gate 				longer_usage(CMD_EXPORT);
18757c478bd9Sstevel@tonic-gate 			else
18767c478bd9Sstevel@tonic-gate 				short_usage(CMD_EXPORT);
1877bbec428eSgjelinek 			arg_err = B_TRUE;
18787ec75eb8Sgjelinek 			break;
18797c478bd9Sstevel@tonic-gate 		case 'f':
18807c478bd9Sstevel@tonic-gate 			(void) strlcpy(outfile, optarg, sizeof (outfile));
18817c478bd9Sstevel@tonic-gate 			break;
18827c478bd9Sstevel@tonic-gate 		default:
18837c478bd9Sstevel@tonic-gate 			short_usage(CMD_EXPORT);
1884bbec428eSgjelinek 			arg_err = B_TRUE;
18857ec75eb8Sgjelinek 			break;
18867ec75eb8Sgjelinek 		}
18877ec75eb8Sgjelinek 	}
18887ec75eb8Sgjelinek 	if (arg_err)
18897c478bd9Sstevel@tonic-gate 		return;
18907ec75eb8Sgjelinek 
18917c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
18927c478bd9Sstevel@tonic-gate 		short_usage(CMD_EXPORT);
18937c478bd9Sstevel@tonic-gate 		return;
18947c478bd9Sstevel@tonic-gate 	}
18957c478bd9Sstevel@tonic-gate 	if (strlen(outfile) == 0) {
18967c478bd9Sstevel@tonic-gate 		of = stdout;
18977c478bd9Sstevel@tonic-gate 	} else {
18987c478bd9Sstevel@tonic-gate 		if ((of = fopen(outfile, "w")) == NULL) {
18997c478bd9Sstevel@tonic-gate 			zerr(gettext("opening file %s: %s"),
19007c478bd9Sstevel@tonic-gate 			    outfile, strerror(errno));
19017c478bd9Sstevel@tonic-gate 			goto done;
19027c478bd9Sstevel@tonic-gate 		}
19037c478bd9Sstevel@tonic-gate 		setbuf(of, NULL);
1904bbec428eSgjelinek 		need_to_close = B_TRUE;
19057c478bd9Sstevel@tonic-gate 	}
19067c478bd9Sstevel@tonic-gate 
1907bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
19087c478bd9Sstevel@tonic-gate 		goto done;
19097c478bd9Sstevel@tonic-gate 
19107c478bd9Sstevel@tonic-gate 	(void) fprintf(of, "%s -b\n", cmd_to_str(CMD_CREATE));
19117c478bd9Sstevel@tonic-gate 
19127c478bd9Sstevel@tonic-gate 	if (zonecfg_get_zonepath(handle, zonepath, sizeof (zonepath)) == Z_OK &&
19137c478bd9Sstevel@tonic-gate 	    strlen(zonepath) > 0)
19147c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19157c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_ZONEPATH), zonepath);
19167c478bd9Sstevel@tonic-gate 
19179acbbeafSnn35248 	if ((zone_get_brand(zone, brand, sizeof (brand)) == Z_OK) &&
19189acbbeafSnn35248 	    (strcmp(brand, NATIVE_BRAND_NAME) != 0))
19199acbbeafSnn35248 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19209acbbeafSnn35248 		    pt_to_str(PT_BRAND), brand);
19219acbbeafSnn35248 
19227c478bd9Sstevel@tonic-gate 	if (zonecfg_get_autoboot(handle, &autoboot) == Z_OK)
19237c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19247c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_AUTOBOOT), autoboot ? "true" : "false");
19257c478bd9Sstevel@tonic-gate 
19263f2f09c1Sdp 	if (zonecfg_get_bootargs(handle, bootargs, sizeof (bootargs)) == Z_OK &&
19273f2f09c1Sdp 	    strlen(bootargs) > 0) {
19283f2f09c1Sdp 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19293f2f09c1Sdp 		    pt_to_str(PT_BOOTARGS), bootargs);
19303f2f09c1Sdp 	}
19313f2f09c1Sdp 
19327c478bd9Sstevel@tonic-gate 	if (zonecfg_get_pool(handle, pool, sizeof (pool)) == Z_OK &&
19337c478bd9Sstevel@tonic-gate 	    strlen(pool) > 0)
19347c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19357c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_POOL), pool);
19367c478bd9Sstevel@tonic-gate 
1937ffbafc53Scomay 	if (zonecfg_get_limitpriv(handle, &limitpriv) == Z_OK &&
1938ffbafc53Scomay 	    strlen(limitpriv) > 0) {
1939ffbafc53Scomay 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
1940ffbafc53Scomay 		    pt_to_str(PT_LIMITPRIV), limitpriv);
1941ffbafc53Scomay 		free(limitpriv);
1942ffbafc53Scomay 	}
1943ffbafc53Scomay 
19440209230bSgjelinek 	if (zonecfg_get_sched_class(handle, sched, sizeof (sched)) == Z_OK &&
19450209230bSgjelinek 	    strlen(sched) > 0)
19460209230bSgjelinek 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19470209230bSgjelinek 		    pt_to_str(PT_SCHED), sched);
19483f2f09c1Sdp 
1949f4b3ec61Sdh155122 	if (zonecfg_get_iptype(handle, &iptype) == Z_OK) {
1950f4b3ec61Sdh155122 		switch (iptype) {
1951f4b3ec61Sdh155122 		case ZS_SHARED:
1952f4b3ec61Sdh155122 			(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
1953f4b3ec61Sdh155122 			    pt_to_str(PT_IPTYPE), "shared");
1954f4b3ec61Sdh155122 			break;
1955f4b3ec61Sdh155122 		case ZS_EXCLUSIVE:
1956f4b3ec61Sdh155122 			(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
1957f4b3ec61Sdh155122 			    pt_to_str(PT_IPTYPE), "exclusive");
1958f4b3ec61Sdh155122 			break;
1959f4b3ec61Sdh155122 		}
1960f4b3ec61Sdh155122 	}
1961f4b3ec61Sdh155122 
19625679c89fSjv227347 	if (zonecfg_get_hostid(handle, hostidp, sizeof (hostidp)) == Z_OK) {
19635679c89fSjv227347 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19645679c89fSjv227347 		    pt_to_str(PT_HOSTID), hostidp);
19655679c89fSjv227347 	}
19665679c89fSjv227347 
19670fbb751dSJohn Levon 	if (zonecfg_get_fs_allowed(handle, fsallowedp,
19680fbb751dSJohn Levon 	    sizeof (fsallowedp)) == Z_OK) {
19690fbb751dSJohn Levon 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
19700fbb751dSJohn Levon 		    pt_to_str(PT_FS_ALLOWED), fsallowedp);
19710fbb751dSJohn Levon 	}
19720fbb751dSJohn Levon 
19737c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setfsent(handle)) != Z_OK) {
1974bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
19757c478bd9Sstevel@tonic-gate 		goto done;
19767c478bd9Sstevel@tonic-gate 	}
19777c478bd9Sstevel@tonic-gate 	while (zonecfg_getfsent(handle, &fstab) == Z_OK) {
19787c478bd9Sstevel@tonic-gate 		zone_fsopt_t *optptr;
19797c478bd9Sstevel@tonic-gate 
19807c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
19817c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_FS));
19827c478bd9Sstevel@tonic-gate 		export_prop(of, PT_DIR, fstab.zone_fs_dir);
19837c478bd9Sstevel@tonic-gate 		export_prop(of, PT_SPECIAL, fstab.zone_fs_special);
19847c478bd9Sstevel@tonic-gate 		export_prop(of, PT_RAW, fstab.zone_fs_raw);
19857c478bd9Sstevel@tonic-gate 		export_prop(of, PT_TYPE, fstab.zone_fs_type);
19867c478bd9Sstevel@tonic-gate 		for (optptr = fstab.zone_fs_options; optptr != NULL;
19877c478bd9Sstevel@tonic-gate 		    optptr = optptr->zone_fsopt_next) {
19887c478bd9Sstevel@tonic-gate 			/*
19897c478bd9Sstevel@tonic-gate 			 * Simple property values with embedded equal signs
19907c478bd9Sstevel@tonic-gate 			 * need to be quoted to prevent the lexer from
19917c478bd9Sstevel@tonic-gate 			 * mis-parsing them as complex name=value pairs.
19927c478bd9Sstevel@tonic-gate 			 */
19937c478bd9Sstevel@tonic-gate 			if (strchr(optptr->zone_fsopt_opt, '='))
19947c478bd9Sstevel@tonic-gate 				(void) fprintf(of, "%s %s \"%s\"\n",
19957c478bd9Sstevel@tonic-gate 				    cmd_to_str(CMD_ADD),
19967c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_OPTIONS),
19977c478bd9Sstevel@tonic-gate 				    optptr->zone_fsopt_opt);
19987c478bd9Sstevel@tonic-gate 			else
19997c478bd9Sstevel@tonic-gate 				(void) fprintf(of, "%s %s %s\n",
20007c478bd9Sstevel@tonic-gate 				    cmd_to_str(CMD_ADD),
20017c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_OPTIONS),
20027c478bd9Sstevel@tonic-gate 				    optptr->zone_fsopt_opt);
20037c478bd9Sstevel@tonic-gate 		}
20047c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
20057c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(fstab.zone_fs_options);
20067c478bd9Sstevel@tonic-gate 	}
20077c478bd9Sstevel@tonic-gate 	(void) zonecfg_endfsent(handle);
20087c478bd9Sstevel@tonic-gate 
20097c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setnwifent(handle)) != Z_OK) {
2010bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
20117c478bd9Sstevel@tonic-gate 		goto done;
20127c478bd9Sstevel@tonic-gate 	}
20137c478bd9Sstevel@tonic-gate 	while (zonecfg_getnwifent(handle, &nwiftab) == Z_OK) {
20147c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
20157c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_NET));
20167c478bd9Sstevel@tonic-gate 		export_prop(of, PT_ADDRESS, nwiftab.zone_nwif_address);
2017550b6e40SSowmini Varadhan 		export_prop(of, PT_ALLOWED_ADDRESS,
2018550b6e40SSowmini Varadhan 		    nwiftab.zone_nwif_allowed_address);
20197c478bd9Sstevel@tonic-gate 		export_prop(of, PT_PHYSICAL, nwiftab.zone_nwif_physical);
2020de860bd9Sgfaden 		export_prop(of, PT_DEFROUTER, nwiftab.zone_nwif_defrouter);
20217c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
20227c478bd9Sstevel@tonic-gate 	}
20237c478bd9Sstevel@tonic-gate 	(void) zonecfg_endnwifent(handle);
20247c478bd9Sstevel@tonic-gate 
20257c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setdevent(handle)) != Z_OK) {
2026bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
20277c478bd9Sstevel@tonic-gate 		goto done;
20287c478bd9Sstevel@tonic-gate 	}
20297c478bd9Sstevel@tonic-gate 	while (zonecfg_getdevent(handle, &devtab) == Z_OK) {
20307c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
20317c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_DEVICE));
20327c478bd9Sstevel@tonic-gate 		export_prop(of, PT_MATCH, devtab.zone_dev_match);
20337c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
20347c478bd9Sstevel@tonic-gate 	}
20357c478bd9Sstevel@tonic-gate 	(void) zonecfg_enddevent(handle);
20367c478bd9Sstevel@tonic-gate 
20371b3281c0SGerald Jelinek 	if (zonecfg_getmcapent(handle, &mcaptab) == Z_OK) {
20381b3281c0SGerald Jelinek 		char buf[128];
20391b3281c0SGerald Jelinek 
20401b3281c0SGerald Jelinek 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
20411b3281c0SGerald Jelinek 		    rt_to_str(RT_MCAP));
20421b3281c0SGerald Jelinek 		bytes_to_units(mcaptab.zone_physmem_cap, buf, sizeof (buf));
20431b3281c0SGerald Jelinek 		(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
20441b3281c0SGerald Jelinek 		    pt_to_str(PT_PHYSICAL), buf);
20451b3281c0SGerald Jelinek 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
20461b3281c0SGerald Jelinek 	}
20471b3281c0SGerald Jelinek 
20487c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setrctlent(handle)) != Z_OK) {
2049bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
20507c478bd9Sstevel@tonic-gate 		goto done;
20517c478bd9Sstevel@tonic-gate 	}
20527c478bd9Sstevel@tonic-gate 	while (zonecfg_getrctlent(handle, &rctltab) == Z_OK) {
20537c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s rctl\n", cmd_to_str(CMD_ADD));
20547c478bd9Sstevel@tonic-gate 		export_prop(of, PT_NAME, rctltab.zone_rctl_name);
20557c478bd9Sstevel@tonic-gate 		for (valptr = rctltab.zone_rctl_valptr; valptr != NULL;
20567c478bd9Sstevel@tonic-gate 		    valptr = valptr->zone_rctlval_next) {
20577c478bd9Sstevel@tonic-gate 			fprintf(of, "%s %s (%s=%s,%s=%s,%s=%s)\n",
20587c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_ADD), pt_to_str(PT_VALUE),
20597c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_PRIV), valptr->zone_rctlval_priv,
20607c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_LIMIT), valptr->zone_rctlval_limit,
20617c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_ACTION), valptr->zone_rctlval_action);
20627c478bd9Sstevel@tonic-gate 		}
20637c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
20647c478bd9Sstevel@tonic-gate 		zonecfg_free_rctl_value_list(rctltab.zone_rctl_valptr);
20657c478bd9Sstevel@tonic-gate 	}
20667c478bd9Sstevel@tonic-gate 	(void) zonecfg_endrctlent(handle);
20677c478bd9Sstevel@tonic-gate 
20687c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setattrent(handle)) != Z_OK) {
2069bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
20707c478bd9Sstevel@tonic-gate 		goto done;
20717c478bd9Sstevel@tonic-gate 	}
20727c478bd9Sstevel@tonic-gate 	while (zonecfg_getattrent(handle, &attrtab) == Z_OK) {
20737c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
20747c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR));
20757c478bd9Sstevel@tonic-gate 		export_prop(of, PT_NAME, attrtab.zone_attr_name);
20767c478bd9Sstevel@tonic-gate 		export_prop(of, PT_TYPE, attrtab.zone_attr_type);
20777c478bd9Sstevel@tonic-gate 		export_prop(of, PT_VALUE, attrtab.zone_attr_value);
20787c478bd9Sstevel@tonic-gate 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
20797c478bd9Sstevel@tonic-gate 	}
20807c478bd9Sstevel@tonic-gate 	(void) zonecfg_endattrent(handle);
20817c478bd9Sstevel@tonic-gate 
2082fa9e4066Sahrens 	if ((err = zonecfg_setdsent(handle)) != Z_OK) {
2083bbec428eSgjelinek 		zone_perror(zone, err, B_FALSE);
2084fa9e4066Sahrens 		goto done;
2085fa9e4066Sahrens 	}
2086fa9e4066Sahrens 	while (zonecfg_getdsent(handle, &dstab) == Z_OK) {
2087fa9e4066Sahrens 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
2088fa9e4066Sahrens 		    rt_to_str(RT_DATASET));
2089fa9e4066Sahrens 		export_prop(of, PT_NAME, dstab.zone_dataset_name);
2090fa9e4066Sahrens 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
2091fa9e4066Sahrens 	}
2092fa9e4066Sahrens 	(void) zonecfg_enddsent(handle);
2093fa9e4066Sahrens 
20940209230bSgjelinek 	if (zonecfg_getpsetent(handle, &psettab) == Z_OK) {
20950209230bSgjelinek 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
20960209230bSgjelinek 		    rt_to_str(RT_DCPU));
20970209230bSgjelinek 		if (strcmp(psettab.zone_ncpu_min, psettab.zone_ncpu_max) == 0)
20980209230bSgjelinek 			(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
20990209230bSgjelinek 			    pt_to_str(PT_NCPUS), psettab.zone_ncpu_max);
21000209230bSgjelinek 		else
21010209230bSgjelinek 			(void) fprintf(of, "%s %s=%s-%s\n", cmd_to_str(CMD_SET),
21020209230bSgjelinek 			    pt_to_str(PT_NCPUS), psettab.zone_ncpu_min,
21030209230bSgjelinek 			    psettab.zone_ncpu_max);
21040209230bSgjelinek 		if (psettab.zone_importance[0] != '\0')
21050209230bSgjelinek 			(void) fprintf(of, "%s %s=%s\n", cmd_to_str(CMD_SET),
21060209230bSgjelinek 			    pt_to_str(PT_IMPORTANCE), psettab.zone_importance);
21070209230bSgjelinek 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
21080209230bSgjelinek 	}
21090209230bSgjelinek 
2110cb8a054bSGlenn Faden 	if ((err = zonecfg_setadminent(handle)) != Z_OK) {
2111cb8a054bSGlenn Faden 		zone_perror(zone, err, B_FALSE);
2112cb8a054bSGlenn Faden 		goto done;
2113cb8a054bSGlenn Faden 	}
2114cb8a054bSGlenn Faden 	while (zonecfg_getadminent(handle, &admintab) == Z_OK) {
2115cb8a054bSGlenn Faden 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
2116cb8a054bSGlenn Faden 		    rt_to_str(RT_ADMIN));
2117cb8a054bSGlenn Faden 		export_prop(of, PT_USER, admintab.zone_admin_user);
2118cb8a054bSGlenn Faden 		export_prop(of, PT_AUTHS, admintab.zone_admin_auths);
2119cb8a054bSGlenn Faden 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
2120cb8a054bSGlenn Faden 	}
2121d2a70789SRichard Lowe 
2122cb8a054bSGlenn Faden 	(void) zonecfg_endadminent(handle);
2123cb8a054bSGlenn Faden 
2124d2a70789SRichard Lowe 	if (zonecfg_getsecflagsent(handle, &secflagstab) == Z_OK) {
2125d2a70789SRichard Lowe 		(void) fprintf(of, "%s %s\n", cmd_to_str(CMD_ADD),
2126d2a70789SRichard Lowe 		    rt_to_str(RT_SECFLAGS));
2127d2a70789SRichard Lowe 		export_prop(of, PT_DEFAULT, secflagstab.zone_secflags_default);
2128d2a70789SRichard Lowe 		export_prop(of, PT_LOWER, secflagstab.zone_secflags_lower);
2129d2a70789SRichard Lowe 		export_prop(of, PT_UPPER, secflagstab.zone_secflags_upper);
2130d2a70789SRichard Lowe 		(void) fprintf(of, "%s\n", cmd_to_str(CMD_END));
2131d2a70789SRichard Lowe 	}
2132d2a70789SRichard Lowe 
2133c97ad5cdSakolb 	/*
2134c97ad5cdSakolb 	 * There is nothing to export for pcap since this resource is just
2135c97ad5cdSakolb 	 * a container for an rctl alias.
2136c97ad5cdSakolb 	 */
2137c97ad5cdSakolb 
21387c478bd9Sstevel@tonic-gate done:
21397c478bd9Sstevel@tonic-gate 	if (need_to_close)
21407c478bd9Sstevel@tonic-gate 		(void) fclose(of);
21417c478bd9Sstevel@tonic-gate }
21427c478bd9Sstevel@tonic-gate 
21437c478bd9Sstevel@tonic-gate void
exit_func(cmd_t * cmd)21447c478bd9Sstevel@tonic-gate exit_func(cmd_t *cmd)
21457c478bd9Sstevel@tonic-gate {
21467c478bd9Sstevel@tonic-gate 	int arg, answer;
2147bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
21487c478bd9Sstevel@tonic-gate 
21497c478bd9Sstevel@tonic-gate 	optind = 0;
21507c478bd9Sstevel@tonic-gate 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?F")) != EOF) {
21517c478bd9Sstevel@tonic-gate 		switch (arg) {
21527c478bd9Sstevel@tonic-gate 		case '?':
21537c478bd9Sstevel@tonic-gate 			longer_usage(CMD_EXIT);
2154bbec428eSgjelinek 			arg_err = B_TRUE;
21557ec75eb8Sgjelinek 			break;
21567c478bd9Sstevel@tonic-gate 		case 'F':
2157bbec428eSgjelinek 			force_exit = B_TRUE;
21587c478bd9Sstevel@tonic-gate 			break;
21597c478bd9Sstevel@tonic-gate 		default:
21607c478bd9Sstevel@tonic-gate 			short_usage(CMD_EXIT);
2161bbec428eSgjelinek 			arg_err = B_TRUE;
21627ec75eb8Sgjelinek 			break;
21637ec75eb8Sgjelinek 		}
21647ec75eb8Sgjelinek 	}
21657ec75eb8Sgjelinek 	if (arg_err)
21667c478bd9Sstevel@tonic-gate 		return;
21677ec75eb8Sgjelinek 
21687c478bd9Sstevel@tonic-gate 	if (optind < cmd->cmd_argc) {
21697c478bd9Sstevel@tonic-gate 		short_usage(CMD_EXIT);
21707c478bd9Sstevel@tonic-gate 		return;
21717c478bd9Sstevel@tonic-gate 	}
21727c478bd9Sstevel@tonic-gate 
21737c478bd9Sstevel@tonic-gate 	if (global_scope || force_exit) {
2174bbec428eSgjelinek 		time_to_exit = B_TRUE;
21757c478bd9Sstevel@tonic-gate 		return;
21767c478bd9Sstevel@tonic-gate 	}
21777c478bd9Sstevel@tonic-gate 
2178bbec428eSgjelinek 	answer = ask_yesno(B_FALSE, "Resource incomplete; really quit");
21797c478bd9Sstevel@tonic-gate 	if (answer == -1) {
21807c478bd9Sstevel@tonic-gate 		zerr(gettext("Resource incomplete, input "
21817c478bd9Sstevel@tonic-gate 		    "not from terminal and -F not specified:\n%s command "
21827c478bd9Sstevel@tonic-gate 		    "ignored, but exiting anyway."), cmd_to_str(CMD_EXIT));
21837c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
21847c478bd9Sstevel@tonic-gate 	} else if (answer == 1) {
2185bbec428eSgjelinek 		time_to_exit = B_TRUE;
21867c478bd9Sstevel@tonic-gate 	}
21877c478bd9Sstevel@tonic-gate 	/* (answer == 0) => just return */
21887c478bd9Sstevel@tonic-gate }
21897c478bd9Sstevel@tonic-gate 
21907c478bd9Sstevel@tonic-gate static int
validate_zonepath_syntax(char * path)21917c478bd9Sstevel@tonic-gate validate_zonepath_syntax(char *path)
21927c478bd9Sstevel@tonic-gate {
21937c478bd9Sstevel@tonic-gate 	if (path[0] != '/') {
21947c478bd9Sstevel@tonic-gate 		zerr(gettext("%s is not an absolute path."), path);
21957c478bd9Sstevel@tonic-gate 		return (Z_ERR);
21967c478bd9Sstevel@tonic-gate 	}
21976fb06a2bSSusan Kamm-Worrell 	/* If path is all slashes, then fail */
21986fb06a2bSSusan Kamm-Worrell 	if (strspn(path, "/") == strlen(path)) {
21997c478bd9Sstevel@tonic-gate 		zerr(gettext("/ is not allowed as a %s."),
22007c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_ZONEPATH));
22017c478bd9Sstevel@tonic-gate 		return (Z_ERR);
22027c478bd9Sstevel@tonic-gate 	}
22037c478bd9Sstevel@tonic-gate 	return (Z_OK);
22047c478bd9Sstevel@tonic-gate }
22057c478bd9Sstevel@tonic-gate 
22067c478bd9Sstevel@tonic-gate static void
add_resource(cmd_t * cmd)22077c478bd9Sstevel@tonic-gate add_resource(cmd_t *cmd)
22087c478bd9Sstevel@tonic-gate {
22097c478bd9Sstevel@tonic-gate 	int type;
22100209230bSgjelinek 	struct zone_psettab tmp_psettab;
22110209230bSgjelinek 	struct zone_mcaptab tmp_mcaptab;
2212d2a70789SRichard Lowe 	struct zone_secflagstab tmp_secflagstab;
2213c97ad5cdSakolb 	uint64_t tmp;
22140209230bSgjelinek 	uint64_t tmp_mcap;
22150209230bSgjelinek 	char pool[MAXNAMELEN];
22167c478bd9Sstevel@tonic-gate 
22177c478bd9Sstevel@tonic-gate 	if ((type = cmd->cmd_res_type) == RT_UNKNOWN) {
2218bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
22197c478bd9Sstevel@tonic-gate 		goto bad;
22207c478bd9Sstevel@tonic-gate 	}
22217c478bd9Sstevel@tonic-gate 
22227c478bd9Sstevel@tonic-gate 	switch (type) {
22237c478bd9Sstevel@tonic-gate 	case RT_FS:
22247c478bd9Sstevel@tonic-gate 		bzero(&in_progress_fstab, sizeof (in_progress_fstab));
22257c478bd9Sstevel@tonic-gate 		return;
22267c478bd9Sstevel@tonic-gate 	case RT_NET:
22277c478bd9Sstevel@tonic-gate 		bzero(&in_progress_nwiftab, sizeof (in_progress_nwiftab));
22287c478bd9Sstevel@tonic-gate 		return;
22297c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
22307c478bd9Sstevel@tonic-gate 		bzero(&in_progress_devtab, sizeof (in_progress_devtab));
22317c478bd9Sstevel@tonic-gate 		return;
22327c478bd9Sstevel@tonic-gate 	case RT_RCTL:
22330209230bSgjelinek 		if (global_zone)
22340209230bSgjelinek 			zerr(gettext("WARNING: Setting a global zone resource "
22350209230bSgjelinek 			    "control too low could deny\nservice "
22360209230bSgjelinek 			    "to even the root user; "
22370209230bSgjelinek 			    "this could render the system impossible\n"
22380209230bSgjelinek 			    "to administer.  Please use caution."));
22397c478bd9Sstevel@tonic-gate 		bzero(&in_progress_rctltab, sizeof (in_progress_rctltab));
22407c478bd9Sstevel@tonic-gate 		return;
22417c478bd9Sstevel@tonic-gate 	case RT_ATTR:
22427c478bd9Sstevel@tonic-gate 		bzero(&in_progress_attrtab, sizeof (in_progress_attrtab));
22437c478bd9Sstevel@tonic-gate 		return;
2244fa9e4066Sahrens 	case RT_DATASET:
2245fa9e4066Sahrens 		bzero(&in_progress_dstab, sizeof (in_progress_dstab));
2246fa9e4066Sahrens 		return;
22470209230bSgjelinek 	case RT_DCPU:
2248c97ad5cdSakolb 		/* Make sure there isn't already a cpu-set or cpu-cap entry. */
22490209230bSgjelinek 		if (zonecfg_lookup_pset(handle, &tmp_psettab) == Z_OK) {
22500209230bSgjelinek 			zerr(gettext("The %s resource already exists."),
22510209230bSgjelinek 			    rt_to_str(RT_DCPU));
22520209230bSgjelinek 			goto bad;
22530209230bSgjelinek 		}
2254c97ad5cdSakolb 		if (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &tmp) !=
2255c97ad5cdSakolb 		    Z_NO_ENTRY) {
2256c97ad5cdSakolb 			zerr(gettext("The %s resource already exists."),
2257c97ad5cdSakolb 			    rt_to_str(RT_PCAP));
2258c97ad5cdSakolb 			goto bad;
2259c97ad5cdSakolb 		}
22600209230bSgjelinek 
22610209230bSgjelinek 		/* Make sure the pool property isn't set. */
22620209230bSgjelinek 		if (zonecfg_get_pool(handle, pool, sizeof (pool)) == Z_OK &&
22630209230bSgjelinek 		    strlen(pool) > 0) {
22640209230bSgjelinek 			zerr(gettext("The %s property is already set.  "
22650209230bSgjelinek 			    "A persistent pool is incompatible with\nthe %s "
22660209230bSgjelinek 			    "resource."),
22670209230bSgjelinek 			    pt_to_str(PT_POOL), rt_to_str(RT_DCPU));
22680209230bSgjelinek 			goto bad;
22690209230bSgjelinek 		}
22700209230bSgjelinek 
22710209230bSgjelinek 		bzero(&in_progress_psettab, sizeof (in_progress_psettab));
22720209230bSgjelinek 		return;
2273c97ad5cdSakolb 	case RT_PCAP:
2274c97ad5cdSakolb 		/*
2275c97ad5cdSakolb 		 * Make sure there isn't already a cpu-set or incompatible
2276c97ad5cdSakolb 		 * cpu-cap rctls.
2277c97ad5cdSakolb 		 */
2278c97ad5cdSakolb 		if (zonecfg_lookup_pset(handle, &tmp_psettab) == Z_OK) {
2279c97ad5cdSakolb 			zerr(gettext("The %s resource already exists."),
2280c97ad5cdSakolb 			    rt_to_str(RT_DCPU));
2281c97ad5cdSakolb 			goto bad;
2282c97ad5cdSakolb 		}
2283c97ad5cdSakolb 
2284c97ad5cdSakolb 		switch (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &tmp)) {
2285c97ad5cdSakolb 		case Z_ALIAS_DISALLOW:
2286c97ad5cdSakolb 			zone_perror(rt_to_str(RT_PCAP), Z_ALIAS_DISALLOW,
2287bbec428eSgjelinek 			    B_FALSE);
2288c97ad5cdSakolb 			goto bad;
2289c97ad5cdSakolb 
2290c97ad5cdSakolb 		case Z_OK:
2291c97ad5cdSakolb 			zerr(gettext("The %s resource already exists."),
2292c97ad5cdSakolb 			    rt_to_str(RT_PCAP));
2293c97ad5cdSakolb 			goto bad;
2294c97ad5cdSakolb 
2295c97ad5cdSakolb 		default:
2296c97ad5cdSakolb 			break;
2297c97ad5cdSakolb 		}
2298c97ad5cdSakolb 		return;
22990209230bSgjelinek 	case RT_MCAP:
23000209230bSgjelinek 		/*
23010209230bSgjelinek 		 * Make sure there isn't already a mem-cap entry or max-swap
23020209230bSgjelinek 		 * or max-locked rctl.
23030209230bSgjelinek 		 */
23040209230bSgjelinek 		if (zonecfg_lookup_mcap(handle, &tmp_mcaptab) == Z_OK ||
23050209230bSgjelinek 		    zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP, &tmp_mcap)
23060209230bSgjelinek 		    == Z_OK ||
23070209230bSgjelinek 		    zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM,
23080209230bSgjelinek 		    &tmp_mcap) == Z_OK) {
23090209230bSgjelinek 			zerr(gettext("The %s resource or a related resource "
23100209230bSgjelinek 			    "control already exists."), rt_to_str(RT_MCAP));
23110209230bSgjelinek 			goto bad;
23120209230bSgjelinek 		}
23130209230bSgjelinek 		if (global_zone)
23140209230bSgjelinek 			zerr(gettext("WARNING: Setting a global zone memory "
23150209230bSgjelinek 			    "cap too low could deny\nservice "
23160209230bSgjelinek 			    "to even the root user; "
23170209230bSgjelinek 			    "this could render the system impossible\n"
23180209230bSgjelinek 			    "to administer.  Please use caution."));
23190209230bSgjelinek 		bzero(&in_progress_mcaptab, sizeof (in_progress_mcaptab));
23200209230bSgjelinek 		return;
2321cb8a054bSGlenn Faden 	case RT_ADMIN:
2322cb8a054bSGlenn Faden 		bzero(&in_progress_admintab, sizeof (in_progress_admintab));
2323cb8a054bSGlenn Faden 		return;
2324d2a70789SRichard Lowe 	case RT_SECFLAGS:
2325d2a70789SRichard Lowe 		/* Make sure we haven't already set this */
2326d2a70789SRichard Lowe 		if (zonecfg_lookup_secflags(handle, &tmp_secflagstab) == Z_OK)
2327d2a70789SRichard Lowe 			zerr(gettext("The %s resource already exists."),
2328d2a70789SRichard Lowe 			    rt_to_str(RT_SECFLAGS));
2329d2a70789SRichard Lowe 		bzero(&in_progress_secflagstab,
2330d2a70789SRichard Lowe 		    sizeof (in_progress_secflagstab));
2331d2a70789SRichard Lowe 		return;
23327c478bd9Sstevel@tonic-gate 	default:
2333bbec428eSgjelinek 		zone_perror(rt_to_str(type), Z_NO_RESOURCE_TYPE, B_TRUE);
2334bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
2335bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
23367c478bd9Sstevel@tonic-gate 	}
23377c478bd9Sstevel@tonic-gate bad:
2338bbec428eSgjelinek 	global_scope = B_TRUE;
23397c478bd9Sstevel@tonic-gate 	end_op = -1;
23407c478bd9Sstevel@tonic-gate }
23417c478bd9Sstevel@tonic-gate 
23427c478bd9Sstevel@tonic-gate static void
do_complex_rctl_val(complex_property_ptr_t cp)23437c478bd9Sstevel@tonic-gate do_complex_rctl_val(complex_property_ptr_t cp)
23447c478bd9Sstevel@tonic-gate {
23457c478bd9Sstevel@tonic-gate 	struct zone_rctlvaltab *rctlvaltab;
23467c478bd9Sstevel@tonic-gate 	complex_property_ptr_t cx;
2347bbec428eSgjelinek 	boolean_t seen_priv = B_FALSE, seen_limit = B_FALSE,
2348bbec428eSgjelinek 	    seen_action = B_FALSE;
23497c478bd9Sstevel@tonic-gate 	rctlblk_t *rctlblk;
23507c478bd9Sstevel@tonic-gate 	int err;
23517c478bd9Sstevel@tonic-gate 
23527c478bd9Sstevel@tonic-gate 	if ((rctlvaltab = alloc_rctlvaltab()) == NULL) {
2353bbec428eSgjelinek 		zone_perror(zone, Z_NOMEM, B_TRUE);
23547c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
23557c478bd9Sstevel@tonic-gate 	}
23567c478bd9Sstevel@tonic-gate 	for (cx = cp; cx != NULL; cx = cx->cp_next) {
23577c478bd9Sstevel@tonic-gate 		switch (cx->cp_type) {
23587c478bd9Sstevel@tonic-gate 		case PT_PRIV:
23597c478bd9Sstevel@tonic-gate 			if (seen_priv) {
23607c478bd9Sstevel@tonic-gate 				zerr(gettext("%s already specified"),
23617c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_PRIV));
23627c478bd9Sstevel@tonic-gate 				goto bad;
23637c478bd9Sstevel@tonic-gate 			}
23647c478bd9Sstevel@tonic-gate 			(void) strlcpy(rctlvaltab->zone_rctlval_priv,
23657c478bd9Sstevel@tonic-gate 			    cx->cp_value,
23667c478bd9Sstevel@tonic-gate 			    sizeof (rctlvaltab->zone_rctlval_priv));
2367bbec428eSgjelinek 			seen_priv = B_TRUE;
23687c478bd9Sstevel@tonic-gate 			break;
23697c478bd9Sstevel@tonic-gate 		case PT_LIMIT:
23707c478bd9Sstevel@tonic-gate 			if (seen_limit) {
23717c478bd9Sstevel@tonic-gate 				zerr(gettext("%s already specified"),
23727c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_LIMIT));
23737c478bd9Sstevel@tonic-gate 				goto bad;
23747c478bd9Sstevel@tonic-gate 			}
23757c478bd9Sstevel@tonic-gate 			(void) strlcpy(rctlvaltab->zone_rctlval_limit,
23767c478bd9Sstevel@tonic-gate 			    cx->cp_value,
23777c478bd9Sstevel@tonic-gate 			    sizeof (rctlvaltab->zone_rctlval_limit));
2378bbec428eSgjelinek 			seen_limit = B_TRUE;
23797c478bd9Sstevel@tonic-gate 			break;
23807c478bd9Sstevel@tonic-gate 		case PT_ACTION:
23817c478bd9Sstevel@tonic-gate 			if (seen_action) {
23827c478bd9Sstevel@tonic-gate 				zerr(gettext("%s already specified"),
23837c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_ACTION));
23847c478bd9Sstevel@tonic-gate 				goto bad;
23857c478bd9Sstevel@tonic-gate 			}
23867c478bd9Sstevel@tonic-gate 			(void) strlcpy(rctlvaltab->zone_rctlval_action,
23877c478bd9Sstevel@tonic-gate 			    cx->cp_value,
23887c478bd9Sstevel@tonic-gate 			    sizeof (rctlvaltab->zone_rctlval_action));
2389bbec428eSgjelinek 			seen_action = B_TRUE;
23907c478bd9Sstevel@tonic-gate 			break;
23917c478bd9Sstevel@tonic-gate 		default:
23927c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(PT_VALUE),
2393bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
2394bbec428eSgjelinek 			long_usage(CMD_ADD, B_TRUE);
2395bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
23967c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(rctlvaltab);
23977c478bd9Sstevel@tonic-gate 			return;
23987c478bd9Sstevel@tonic-gate 		}
23997c478bd9Sstevel@tonic-gate 	}
24007c478bd9Sstevel@tonic-gate 	if (!seen_priv)
24017c478bd9Sstevel@tonic-gate 		zerr(gettext("%s not specified"), pt_to_str(PT_PRIV));
24027c478bd9Sstevel@tonic-gate 	if (!seen_limit)
24037c478bd9Sstevel@tonic-gate 		zerr(gettext("%s not specified"), pt_to_str(PT_LIMIT));
24047c478bd9Sstevel@tonic-gate 	if (!seen_action)
24057c478bd9Sstevel@tonic-gate 		zerr(gettext("%s not specified"), pt_to_str(PT_ACTION));
24067c478bd9Sstevel@tonic-gate 	if (!seen_priv || !seen_limit || !seen_action)
24077c478bd9Sstevel@tonic-gate 		goto bad;
24087c478bd9Sstevel@tonic-gate 	rctlvaltab->zone_rctlval_next = NULL;
24097c478bd9Sstevel@tonic-gate 	rctlblk = alloca(rctlblk_size());
24107c478bd9Sstevel@tonic-gate 	/*
24117c478bd9Sstevel@tonic-gate 	 * Make sure the rctl value looks roughly correct; we won't know if
24127c478bd9Sstevel@tonic-gate 	 * it's truly OK until we verify the configuration on the target
24137c478bd9Sstevel@tonic-gate 	 * system.
24147c478bd9Sstevel@tonic-gate 	 */
24157c478bd9Sstevel@tonic-gate 	if (zonecfg_construct_rctlblk(rctlvaltab, rctlblk) != Z_OK ||
24167c478bd9Sstevel@tonic-gate 	    !zonecfg_valid_rctlblk(rctlblk)) {
24177c478bd9Sstevel@tonic-gate 		zerr(gettext("Invalid %s %s specification"), rt_to_str(RT_RCTL),
24187c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_VALUE));
24197c478bd9Sstevel@tonic-gate 		goto bad;
24207c478bd9Sstevel@tonic-gate 	}
24217c478bd9Sstevel@tonic-gate 	err = zonecfg_add_rctl_value(&in_progress_rctltab, rctlvaltab);
24227c478bd9Sstevel@tonic-gate 	if (err != Z_OK)
2423bbec428eSgjelinek 		zone_perror(pt_to_str(PT_VALUE), err, B_TRUE);
24247c478bd9Sstevel@tonic-gate 	return;
24257c478bd9Sstevel@tonic-gate 
24267c478bd9Sstevel@tonic-gate bad:
24277c478bd9Sstevel@tonic-gate 	zonecfg_free_rctl_value_list(rctlvaltab);
24287c478bd9Sstevel@tonic-gate }
24297c478bd9Sstevel@tonic-gate 
24307c478bd9Sstevel@tonic-gate static void
add_property(cmd_t * cmd)24317c478bd9Sstevel@tonic-gate add_property(cmd_t *cmd)
24327c478bd9Sstevel@tonic-gate {
24337c478bd9Sstevel@tonic-gate 	char *prop_id;
24347c478bd9Sstevel@tonic-gate 	int err, res_type, prop_type;
24357c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
24367c478bd9Sstevel@tonic-gate 	list_property_ptr_t l;
24377c478bd9Sstevel@tonic-gate 
24387c478bd9Sstevel@tonic-gate 	res_type = resource_scope;
24397c478bd9Sstevel@tonic-gate 	prop_type = cmd->cmd_prop_name[0];
24407c478bd9Sstevel@tonic-gate 	if (res_type == RT_UNKNOWN || prop_type == PT_UNKNOWN) {
2441bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
24427c478bd9Sstevel@tonic-gate 		return;
24437c478bd9Sstevel@tonic-gate 	}
24447c478bd9Sstevel@tonic-gate 
24457c478bd9Sstevel@tonic-gate 	if (cmd->cmd_prop_nv_pairs != 1) {
2446bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
24477c478bd9Sstevel@tonic-gate 		return;
24487c478bd9Sstevel@tonic-gate 	}
24497c478bd9Sstevel@tonic-gate 
2450bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
24517c478bd9Sstevel@tonic-gate 		return;
24527c478bd9Sstevel@tonic-gate 
24537c478bd9Sstevel@tonic-gate 	switch (res_type) {
24547c478bd9Sstevel@tonic-gate 	case RT_FS:
24557c478bd9Sstevel@tonic-gate 		if (prop_type != PT_OPTIONS) {
24567c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
2457bbec428eSgjelinek 			    B_TRUE);
2458bbec428eSgjelinek 			long_usage(CMD_ADD, B_TRUE);
2459bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
24607c478bd9Sstevel@tonic-gate 			return;
24617c478bd9Sstevel@tonic-gate 		}
24627c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[0];
24637c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE &&
24647c478bd9Sstevel@tonic-gate 		    pp->pv_type != PROP_VAL_LIST) {
24657c478bd9Sstevel@tonic-gate 			zerr(gettext("A %s or %s value was expected here."),
24667c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_SIMPLE),
24677c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_LIST));
2468bbec428eSgjelinek 			saw_error = B_TRUE;
24697c478bd9Sstevel@tonic-gate 			return;
24707c478bd9Sstevel@tonic-gate 		}
24717c478bd9Sstevel@tonic-gate 		if (pp->pv_type == PROP_VAL_SIMPLE) {
24727c478bd9Sstevel@tonic-gate 			if (pp->pv_simple == NULL) {
2473bbec428eSgjelinek 				long_usage(CMD_ADD, B_TRUE);
24747c478bd9Sstevel@tonic-gate 				return;
24757c478bd9Sstevel@tonic-gate 			}
24767c478bd9Sstevel@tonic-gate 			prop_id = pp->pv_simple;
24777c478bd9Sstevel@tonic-gate 			err = zonecfg_add_fs_option(&in_progress_fstab,
24787c478bd9Sstevel@tonic-gate 			    prop_id);
24797c478bd9Sstevel@tonic-gate 			if (err != Z_OK)
2480bbec428eSgjelinek 				zone_perror(pt_to_str(prop_type), err, B_TRUE);
24817c478bd9Sstevel@tonic-gate 		} else {
24827c478bd9Sstevel@tonic-gate 			list_property_ptr_t list;
24837c478bd9Sstevel@tonic-gate 
24847c478bd9Sstevel@tonic-gate 			for (list = pp->pv_list; list != NULL;
24857c478bd9Sstevel@tonic-gate 			    list = list->lp_next) {
24867c478bd9Sstevel@tonic-gate 				prop_id = list->lp_simple;
24877c478bd9Sstevel@tonic-gate 				if (prop_id == NULL)
24887c478bd9Sstevel@tonic-gate 					break;
24897c478bd9Sstevel@tonic-gate 				err = zonecfg_add_fs_option(
24907c478bd9Sstevel@tonic-gate 				    &in_progress_fstab, prop_id);
24917c478bd9Sstevel@tonic-gate 				if (err != Z_OK)
24927c478bd9Sstevel@tonic-gate 					zone_perror(pt_to_str(prop_type), err,
2493bbec428eSgjelinek 					    B_TRUE);
24947c478bd9Sstevel@tonic-gate 			}
24957c478bd9Sstevel@tonic-gate 		}
24967c478bd9Sstevel@tonic-gate 		return;
24977c478bd9Sstevel@tonic-gate 	case RT_RCTL:
24987c478bd9Sstevel@tonic-gate 		if (prop_type != PT_VALUE) {
24997c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
2500bbec428eSgjelinek 			    B_TRUE);
2501bbec428eSgjelinek 			long_usage(CMD_ADD, B_TRUE);
2502bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
25037c478bd9Sstevel@tonic-gate 			return;
25047c478bd9Sstevel@tonic-gate 		}
25057c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[0];
25067c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_COMPLEX &&
25077c478bd9Sstevel@tonic-gate 		    pp->pv_type != PROP_VAL_LIST) {
25087c478bd9Sstevel@tonic-gate 			zerr(gettext("A %s or %s value was expected here."),
25097c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_COMPLEX),
25107c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_LIST));
2511bbec428eSgjelinek 			saw_error = B_TRUE;
25127c478bd9Sstevel@tonic-gate 			return;
25137c478bd9Sstevel@tonic-gate 		}
25147c478bd9Sstevel@tonic-gate 		if (pp->pv_type == PROP_VAL_COMPLEX) {
25157c478bd9Sstevel@tonic-gate 			do_complex_rctl_val(pp->pv_complex);
25167c478bd9Sstevel@tonic-gate 			return;
25177c478bd9Sstevel@tonic-gate 		}
25187c478bd9Sstevel@tonic-gate 		for (l = pp->pv_list; l != NULL; l = l->lp_next)
25197c478bd9Sstevel@tonic-gate 			do_complex_rctl_val(l->lp_complex);
25207c478bd9Sstevel@tonic-gate 		return;
25217c478bd9Sstevel@tonic-gate 	default:
2522bbec428eSgjelinek 		zone_perror(rt_to_str(res_type), Z_NO_RESOURCE_TYPE, B_TRUE);
2523bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
2524bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
25257c478bd9Sstevel@tonic-gate 		return;
25267c478bd9Sstevel@tonic-gate 	}
25277c478bd9Sstevel@tonic-gate }
25287c478bd9Sstevel@tonic-gate 
25290209230bSgjelinek static boolean_t
gz_invalid_resource(int type)25300209230bSgjelinek gz_invalid_resource(int type)
25310209230bSgjelinek {
25326e1ae2a3SGary Pennington 	return (global_zone && (type == RT_FS ||
25330209230bSgjelinek 	    type == RT_NET || type == RT_DEVICE || type == RT_ATTR ||
25340209230bSgjelinek 	    type == RT_DATASET));
25350209230bSgjelinek }
25360209230bSgjelinek 
25370209230bSgjelinek static boolean_t
gz_invalid_rt_property(int type)25380209230bSgjelinek gz_invalid_rt_property(int type)
25390209230bSgjelinek {
25400209230bSgjelinek 	return (global_zone && (type == RT_ZONENAME || type == RT_ZONEPATH ||
25410209230bSgjelinek 	    type == RT_AUTOBOOT || type == RT_LIMITPRIV ||
2542f4b3ec61Sdh155122 	    type == RT_BOOTARGS || type == RT_BRAND || type == RT_SCHED ||
25430fbb751dSJohn Levon 	    type == RT_IPTYPE || type == RT_HOSTID || type == RT_FS_ALLOWED));
25440209230bSgjelinek }
25450209230bSgjelinek 
25460209230bSgjelinek static boolean_t
gz_invalid_property(int type)25470209230bSgjelinek gz_invalid_property(int type)
25480209230bSgjelinek {
25490209230bSgjelinek 	return (global_zone && (type == PT_ZONENAME || type == PT_ZONEPATH ||
25500209230bSgjelinek 	    type == PT_AUTOBOOT || type == PT_LIMITPRIV ||
2551f4b3ec61Sdh155122 	    type == PT_BOOTARGS || type == PT_BRAND || type == PT_SCHED ||
25520fbb751dSJohn Levon 	    type == PT_IPTYPE || type == PT_HOSTID || type == PT_FS_ALLOWED));
25530209230bSgjelinek }
25540209230bSgjelinek 
25557c478bd9Sstevel@tonic-gate void
add_func(cmd_t * cmd)25567c478bd9Sstevel@tonic-gate add_func(cmd_t *cmd)
25577c478bd9Sstevel@tonic-gate {
25587c478bd9Sstevel@tonic-gate 	int arg;
2559bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
25607c478bd9Sstevel@tonic-gate 
25617c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
25627c478bd9Sstevel@tonic-gate 
25637c478bd9Sstevel@tonic-gate 	optind = 0;
25647ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
25657c478bd9Sstevel@tonic-gate 		switch (arg) {
25667c478bd9Sstevel@tonic-gate 		case '?':
25677c478bd9Sstevel@tonic-gate 			longer_usage(CMD_ADD);
2568bbec428eSgjelinek 			arg_err = B_TRUE;
25697ec75eb8Sgjelinek 			break;
25707c478bd9Sstevel@tonic-gate 		default:
25717c478bd9Sstevel@tonic-gate 			short_usage(CMD_ADD);
2572bbec428eSgjelinek 			arg_err = B_TRUE;
25737ec75eb8Sgjelinek 			break;
25747ec75eb8Sgjelinek 		}
25757ec75eb8Sgjelinek 	}
25767ec75eb8Sgjelinek 	if (arg_err)
25777c478bd9Sstevel@tonic-gate 		return;
25787ec75eb8Sgjelinek 
25797c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
25807c478bd9Sstevel@tonic-gate 		short_usage(CMD_ADD);
25817c478bd9Sstevel@tonic-gate 		return;
25827c478bd9Sstevel@tonic-gate 	}
25837c478bd9Sstevel@tonic-gate 
25847c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_ADD))
25857c478bd9Sstevel@tonic-gate 		return;
25867c478bd9Sstevel@tonic-gate 
2587bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
25887c478bd9Sstevel@tonic-gate 		return;
25897c478bd9Sstevel@tonic-gate 	if (global_scope) {
25900209230bSgjelinek 		if (gz_invalid_resource(cmd->cmd_res_type)) {
25910209230bSgjelinek 			zerr(gettext("Cannot add a %s resource to the "
25920209230bSgjelinek 			    "global zone."), rt_to_str(cmd->cmd_res_type));
2593bbec428eSgjelinek 			saw_error = B_TRUE;
25940209230bSgjelinek 			return;
25950209230bSgjelinek 		}
25960209230bSgjelinek 
2597bbec428eSgjelinek 		global_scope = B_FALSE;
25987c478bd9Sstevel@tonic-gate 		resource_scope = cmd->cmd_res_type;
25997c478bd9Sstevel@tonic-gate 		end_op = CMD_ADD;
26007c478bd9Sstevel@tonic-gate 		add_resource(cmd);
26017c478bd9Sstevel@tonic-gate 	} else
26027c478bd9Sstevel@tonic-gate 		add_property(cmd);
26037c478bd9Sstevel@tonic-gate }
26047c478bd9Sstevel@tonic-gate 
2605087719fdSdp /*
2606087719fdSdp  * This routine has an unusual implementation, because it tries very
2607087719fdSdp  * hard to succeed in the face of a variety of failure modes.
2608087719fdSdp  * The most common and most vexing occurs when the index file and
2609087719fdSdp  * the /etc/zones/<zonename.xml> file are not both present.  In
2610087719fdSdp  * this case, delete must eradicate as much of the zone state as is left
2611087719fdSdp  * so that the user can later create a new zone with the same name.
2612087719fdSdp  */
26137c478bd9Sstevel@tonic-gate void
delete_func(cmd_t * cmd)26147c478bd9Sstevel@tonic-gate delete_func(cmd_t *cmd)
26157c478bd9Sstevel@tonic-gate {
26167c478bd9Sstevel@tonic-gate 	int err, arg, answer;
26177c478bd9Sstevel@tonic-gate 	char line[ZONENAME_MAX + 128];	/* enough to ask a question */
2618bbec428eSgjelinek 	boolean_t force = B_FALSE;
2619bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
26207c478bd9Sstevel@tonic-gate 
26217c478bd9Sstevel@tonic-gate 	optind = 0;
26227c478bd9Sstevel@tonic-gate 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?F")) != EOF) {
26237c478bd9Sstevel@tonic-gate 		switch (arg) {
26247c478bd9Sstevel@tonic-gate 		case '?':
26257c478bd9Sstevel@tonic-gate 			longer_usage(CMD_DELETE);
2626bbec428eSgjelinek 			arg_err = B_TRUE;
26277ec75eb8Sgjelinek 			break;
26287c478bd9Sstevel@tonic-gate 		case 'F':
2629bbec428eSgjelinek 			force = B_TRUE;
26307c478bd9Sstevel@tonic-gate 			break;
26317c478bd9Sstevel@tonic-gate 		default:
26327c478bd9Sstevel@tonic-gate 			short_usage(CMD_DELETE);
2633bbec428eSgjelinek 			arg_err = B_TRUE;
26347ec75eb8Sgjelinek 			break;
26357ec75eb8Sgjelinek 		}
26367ec75eb8Sgjelinek 	}
26377ec75eb8Sgjelinek 	if (arg_err)
26387c478bd9Sstevel@tonic-gate 		return;
26397ec75eb8Sgjelinek 
26407c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
26417c478bd9Sstevel@tonic-gate 		short_usage(CMD_DELETE);
26427c478bd9Sstevel@tonic-gate 		return;
26437c478bd9Sstevel@tonic-gate 	}
26447c478bd9Sstevel@tonic-gate 
26457c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_DELETE))
26467c478bd9Sstevel@tonic-gate 		return;
26477c478bd9Sstevel@tonic-gate 
2648087719fdSdp 	if (!force) {
2649087719fdSdp 		/*
2650087719fdSdp 		 * Initialize sets up the global called "handle" and warns the
2651087719fdSdp 		 * user if the zone is not configured.  In force mode, we don't
2652087719fdSdp 		 * trust that evaluation, and hence skip it.  (We don't need the
2653087719fdSdp 		 * handle to be loaded anyway, since zonecfg_destroy is done by
2654087719fdSdp 		 * zonename). However, we also have to take care to emulate the
2655087719fdSdp 		 * messages spit out by initialize; see below.
2656087719fdSdp 		 */
2657bbec428eSgjelinek 		if (initialize(B_TRUE) != Z_OK)
26587c478bd9Sstevel@tonic-gate 			return;
26597c478bd9Sstevel@tonic-gate 
26607c478bd9Sstevel@tonic-gate 		(void) snprintf(line, sizeof (line),
26617c478bd9Sstevel@tonic-gate 		    gettext("Are you sure you want to delete zone %s"), zone);
2662bbec428eSgjelinek 		if ((answer = ask_yesno(B_FALSE, line)) == -1) {
2663087719fdSdp 			zerr(gettext("Input not from terminal and -F not "
2664087719fdSdp 			    "specified:\n%s command ignored, exiting."),
2665087719fdSdp 			    cmd_to_str(CMD_DELETE));
26667c478bd9Sstevel@tonic-gate 			exit(Z_ERR);
26677c478bd9Sstevel@tonic-gate 		}
26687c478bd9Sstevel@tonic-gate 		if (answer != 1)
26697c478bd9Sstevel@tonic-gate 			return;
26707c478bd9Sstevel@tonic-gate 	}
26717c478bd9Sstevel@tonic-gate 
2672cb8a054bSGlenn Faden 	/*
2673cb8a054bSGlenn Faden 	 * This function removes the authorizations from user_attr
2674cb8a054bSGlenn Faden 	 * that correspond to those specified in the configuration
2675cb8a054bSGlenn Faden 	 */
2676cb8a054bSGlenn Faden 	if (initialize(B_TRUE) == Z_OK) {
2677cb8a054bSGlenn Faden 		(void) zonecfg_deauthorize_users(handle, zone);
2678cb8a054bSGlenn Faden 	}
2679087719fdSdp 	if ((err = zonecfg_destroy(zone, force)) != Z_OK) {
2680087719fdSdp 		if ((err == Z_BAD_ZONE_STATE) && !force) {
2681087719fdSdp 			zerr(gettext("Zone %s not in %s state; %s not "
2682087719fdSdp 			    "allowed.  Use -F to force %s."),
2683087719fdSdp 			    zone, zone_state_str(ZONE_STATE_CONFIGURED),
2684087719fdSdp 			    cmd_to_str(CMD_DELETE), cmd_to_str(CMD_DELETE));
2685087719fdSdp 		} else {
2686bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
26877c478bd9Sstevel@tonic-gate 		}
2688087719fdSdp 	}
2689bbec428eSgjelinek 	need_to_commit = B_FALSE;
2690087719fdSdp 
2691087719fdSdp 	/*
2692087719fdSdp 	 * Emulate initialize's messaging; if there wasn't a valid handle to
2693087719fdSdp 	 * begin with, then user had typed delete (or delete -F) multiple
2694087719fdSdp 	 * times.  So we emit a message.
2695087719fdSdp 	 *
2696087719fdSdp 	 * We only do this in the 'force' case because normally, initialize()
2697087719fdSdp 	 * takes care of this for us.
2698087719fdSdp 	 */
2699087719fdSdp 	if (force && zonecfg_check_handle(handle) != Z_OK && interactive_mode)
2700087719fdSdp 		(void) printf(gettext("Use '%s' to begin "
2701087719fdSdp 		    "configuring a new zone.\n"), cmd_to_str(CMD_CREATE));
27027c478bd9Sstevel@tonic-gate 
27037c478bd9Sstevel@tonic-gate 	/*
27047c478bd9Sstevel@tonic-gate 	 * Time for a new handle: finish the old one off first
27057c478bd9Sstevel@tonic-gate 	 * then get a new one properly to avoid leaks.
27067c478bd9Sstevel@tonic-gate 	 */
2707087719fdSdp 	if (got_handle) {
27087c478bd9Sstevel@tonic-gate 		zonecfg_fini_handle(handle);
27097c478bd9Sstevel@tonic-gate 		if ((handle = zonecfg_init_handle()) == NULL) {
2710bbec428eSgjelinek 			zone_perror(execname, Z_NOMEM, B_TRUE);
27117c478bd9Sstevel@tonic-gate 			exit(Z_ERR);
27127c478bd9Sstevel@tonic-gate 		}
27137c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_get_handle(zone, handle)) != Z_OK) {
27147c478bd9Sstevel@tonic-gate 			/* If there was no zone before, that's OK */
27157c478bd9Sstevel@tonic-gate 			if (err != Z_NO_ZONE)
2716bbec428eSgjelinek 				zone_perror(zone, err, B_TRUE);
2717bbec428eSgjelinek 			got_handle = B_FALSE;
27187c478bd9Sstevel@tonic-gate 		}
27197c478bd9Sstevel@tonic-gate 	}
2720087719fdSdp }
27217c478bd9Sstevel@tonic-gate 
27227c478bd9Sstevel@tonic-gate static int
fill_in_fstab(cmd_t * cmd,struct zone_fstab * fstab,boolean_t fill_in_only)2723bbec428eSgjelinek fill_in_fstab(cmd_t *cmd, struct zone_fstab *fstab, boolean_t fill_in_only)
27247c478bd9Sstevel@tonic-gate {
27257c478bd9Sstevel@tonic-gate 	int err, i;
27267c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
27277c478bd9Sstevel@tonic-gate 
2728bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
27297c478bd9Sstevel@tonic-gate 		return (err);
27307c478bd9Sstevel@tonic-gate 
2731e193d1e6Svp157776 	bzero(fstab, sizeof (*fstab));
27327c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
27337c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
27347c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
27357c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2736bbec428eSgjelinek 			saw_error = B_TRUE;
27377c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
27387c478bd9Sstevel@tonic-gate 		}
27397c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
27407c478bd9Sstevel@tonic-gate 		case PT_DIR:
27417c478bd9Sstevel@tonic-gate 			(void) strlcpy(fstab->zone_fs_dir, pp->pv_simple,
27427c478bd9Sstevel@tonic-gate 			    sizeof (fstab->zone_fs_dir));
27437c478bd9Sstevel@tonic-gate 			break;
27447c478bd9Sstevel@tonic-gate 		case PT_SPECIAL:
27457c478bd9Sstevel@tonic-gate 			(void) strlcpy(fstab->zone_fs_special, pp->pv_simple,
27467c478bd9Sstevel@tonic-gate 			    sizeof (fstab->zone_fs_special));
27477c478bd9Sstevel@tonic-gate 			break;
27487c478bd9Sstevel@tonic-gate 		case PT_RAW:
27497c478bd9Sstevel@tonic-gate 			(void) strlcpy(fstab->zone_fs_raw, pp->pv_simple,
27507c478bd9Sstevel@tonic-gate 			    sizeof (fstab->zone_fs_raw));
27517c478bd9Sstevel@tonic-gate 			break;
27527c478bd9Sstevel@tonic-gate 		case PT_TYPE:
27537c478bd9Sstevel@tonic-gate 			(void) strlcpy(fstab->zone_fs_type, pp->pv_simple,
27547c478bd9Sstevel@tonic-gate 			    sizeof (fstab->zone_fs_type));
27557c478bd9Sstevel@tonic-gate 			break;
27567c478bd9Sstevel@tonic-gate 		default:
27577c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2758bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
27597c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
27607c478bd9Sstevel@tonic-gate 		}
27617c478bd9Sstevel@tonic-gate 	}
27627c478bd9Sstevel@tonic-gate 	if (fill_in_only)
27637c478bd9Sstevel@tonic-gate 		return (Z_OK);
27647c478bd9Sstevel@tonic-gate 	return (zonecfg_lookup_filesystem(handle, fstab));
27657c478bd9Sstevel@tonic-gate }
27667c478bd9Sstevel@tonic-gate 
27677c478bd9Sstevel@tonic-gate static int
fill_in_nwiftab(cmd_t * cmd,struct zone_nwiftab * nwiftab,boolean_t fill_in_only)2768bbec428eSgjelinek fill_in_nwiftab(cmd_t *cmd, struct zone_nwiftab *nwiftab,
2769bbec428eSgjelinek     boolean_t fill_in_only)
27707c478bd9Sstevel@tonic-gate {
27717c478bd9Sstevel@tonic-gate 	int err, i;
27727c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
27737c478bd9Sstevel@tonic-gate 
2774bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
27757c478bd9Sstevel@tonic-gate 		return (err);
27767c478bd9Sstevel@tonic-gate 
2777e193d1e6Svp157776 	bzero(nwiftab, sizeof (*nwiftab));
27787c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
27797c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
27807c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
27817c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2782bbec428eSgjelinek 			saw_error = B_TRUE;
27837c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
27847c478bd9Sstevel@tonic-gate 		}
27857c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
27867c478bd9Sstevel@tonic-gate 		case PT_ADDRESS:
27877c478bd9Sstevel@tonic-gate 			(void) strlcpy(nwiftab->zone_nwif_address,
27887c478bd9Sstevel@tonic-gate 			    pp->pv_simple, sizeof (nwiftab->zone_nwif_address));
27897c478bd9Sstevel@tonic-gate 			break;
2790550b6e40SSowmini Varadhan 		case PT_ALLOWED_ADDRESS:
2791550b6e40SSowmini Varadhan 			(void) strlcpy(nwiftab->zone_nwif_allowed_address,
2792550b6e40SSowmini Varadhan 			    pp->pv_simple,
2793550b6e40SSowmini Varadhan 			    sizeof (nwiftab->zone_nwif_allowed_address));
2794550b6e40SSowmini Varadhan 			break;
27957c478bd9Sstevel@tonic-gate 		case PT_PHYSICAL:
27967c478bd9Sstevel@tonic-gate 			(void) strlcpy(nwiftab->zone_nwif_physical,
27977c478bd9Sstevel@tonic-gate 			    pp->pv_simple,
27987c478bd9Sstevel@tonic-gate 			    sizeof (nwiftab->zone_nwif_physical));
27997c478bd9Sstevel@tonic-gate 			break;
2800de860bd9Sgfaden 		case PT_DEFROUTER:
2801de860bd9Sgfaden 			(void) strlcpy(nwiftab->zone_nwif_defrouter,
2802de860bd9Sgfaden 			    pp->pv_simple,
2803de860bd9Sgfaden 			    sizeof (nwiftab->zone_nwif_defrouter));
2804de860bd9Sgfaden 			break;
28057c478bd9Sstevel@tonic-gate 		default:
28067c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2807bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
28087c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
28097c478bd9Sstevel@tonic-gate 		}
28107c478bd9Sstevel@tonic-gate 	}
28117c478bd9Sstevel@tonic-gate 	if (fill_in_only)
28127c478bd9Sstevel@tonic-gate 		return (Z_OK);
28137c478bd9Sstevel@tonic-gate 	err = zonecfg_lookup_nwif(handle, nwiftab);
28147c478bd9Sstevel@tonic-gate 	return (err);
28157c478bd9Sstevel@tonic-gate }
28167c478bd9Sstevel@tonic-gate 
28177c478bd9Sstevel@tonic-gate static int
fill_in_devtab(cmd_t * cmd,struct zone_devtab * devtab,boolean_t fill_in_only)2818bbec428eSgjelinek fill_in_devtab(cmd_t *cmd, struct zone_devtab *devtab, boolean_t fill_in_only)
28197c478bd9Sstevel@tonic-gate {
28207c478bd9Sstevel@tonic-gate 	int err, i;
28217c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
28227c478bd9Sstevel@tonic-gate 
2823bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
28247c478bd9Sstevel@tonic-gate 		return (err);
28257c478bd9Sstevel@tonic-gate 
2826e193d1e6Svp157776 	bzero(devtab, sizeof (*devtab));
28277c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
28287c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
28297c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
28307c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2831bbec428eSgjelinek 			saw_error = B_TRUE;
28327c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
28337c478bd9Sstevel@tonic-gate 		}
28347c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
28357c478bd9Sstevel@tonic-gate 		case PT_MATCH:
28367c478bd9Sstevel@tonic-gate 			(void) strlcpy(devtab->zone_dev_match, pp->pv_simple,
28377c478bd9Sstevel@tonic-gate 			    sizeof (devtab->zone_dev_match));
28387c478bd9Sstevel@tonic-gate 			break;
28397c478bd9Sstevel@tonic-gate 		default:
28407c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2841bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
28427c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
28437c478bd9Sstevel@tonic-gate 		}
28447c478bd9Sstevel@tonic-gate 	}
28457c478bd9Sstevel@tonic-gate 	if (fill_in_only)
28467c478bd9Sstevel@tonic-gate 		return (Z_OK);
28477c478bd9Sstevel@tonic-gate 	err = zonecfg_lookup_dev(handle, devtab);
28487c478bd9Sstevel@tonic-gate 	return (err);
28497c478bd9Sstevel@tonic-gate }
28507c478bd9Sstevel@tonic-gate 
28517c478bd9Sstevel@tonic-gate static int
fill_in_rctltab(cmd_t * cmd,struct zone_rctltab * rctltab,boolean_t fill_in_only)2852bbec428eSgjelinek fill_in_rctltab(cmd_t *cmd, struct zone_rctltab *rctltab,
2853bbec428eSgjelinek     boolean_t fill_in_only)
28547c478bd9Sstevel@tonic-gate {
28557c478bd9Sstevel@tonic-gate 	int err, i;
28567c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
28577c478bd9Sstevel@tonic-gate 
2858bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
28597c478bd9Sstevel@tonic-gate 		return (err);
28607c478bd9Sstevel@tonic-gate 
2861e193d1e6Svp157776 	bzero(rctltab, sizeof (*rctltab));
28627c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
28637c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
28647c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
28657c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2866bbec428eSgjelinek 			saw_error = B_TRUE;
28677c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
28687c478bd9Sstevel@tonic-gate 		}
28697c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
28707c478bd9Sstevel@tonic-gate 		case PT_NAME:
28717c478bd9Sstevel@tonic-gate 			(void) strlcpy(rctltab->zone_rctl_name, pp->pv_simple,
28727c478bd9Sstevel@tonic-gate 			    sizeof (rctltab->zone_rctl_name));
28737c478bd9Sstevel@tonic-gate 			break;
28747c478bd9Sstevel@tonic-gate 		default:
28757c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2876bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
28777c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
28787c478bd9Sstevel@tonic-gate 		}
28797c478bd9Sstevel@tonic-gate 	}
28807c478bd9Sstevel@tonic-gate 	if (fill_in_only)
28817c478bd9Sstevel@tonic-gate 		return (Z_OK);
28827c478bd9Sstevel@tonic-gate 	err = zonecfg_lookup_rctl(handle, rctltab);
28837c478bd9Sstevel@tonic-gate 	return (err);
28847c478bd9Sstevel@tonic-gate }
28857c478bd9Sstevel@tonic-gate 
28867c478bd9Sstevel@tonic-gate static int
fill_in_attrtab(cmd_t * cmd,struct zone_attrtab * attrtab,boolean_t fill_in_only)2887bbec428eSgjelinek fill_in_attrtab(cmd_t *cmd, struct zone_attrtab *attrtab,
2888bbec428eSgjelinek     boolean_t fill_in_only)
28897c478bd9Sstevel@tonic-gate {
28907c478bd9Sstevel@tonic-gate 	int err, i;
28917c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
28927c478bd9Sstevel@tonic-gate 
2893bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
28947c478bd9Sstevel@tonic-gate 		return (err);
28957c478bd9Sstevel@tonic-gate 
2896e193d1e6Svp157776 	bzero(attrtab, sizeof (*attrtab));
28977c478bd9Sstevel@tonic-gate 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
28987c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[i];
28997c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
29007c478bd9Sstevel@tonic-gate 			zerr(gettext("A simple value was expected here."));
2901bbec428eSgjelinek 			saw_error = B_TRUE;
29027c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
29037c478bd9Sstevel@tonic-gate 		}
29047c478bd9Sstevel@tonic-gate 		switch (cmd->cmd_prop_name[i]) {
29057c478bd9Sstevel@tonic-gate 		case PT_NAME:
29067c478bd9Sstevel@tonic-gate 			(void) strlcpy(attrtab->zone_attr_name, pp->pv_simple,
29077c478bd9Sstevel@tonic-gate 			    sizeof (attrtab->zone_attr_name));
29087c478bd9Sstevel@tonic-gate 			break;
29097c478bd9Sstevel@tonic-gate 		case PT_TYPE:
29107c478bd9Sstevel@tonic-gate 			(void) strlcpy(attrtab->zone_attr_type, pp->pv_simple,
29117c478bd9Sstevel@tonic-gate 			    sizeof (attrtab->zone_attr_type));
29127c478bd9Sstevel@tonic-gate 			break;
29137c478bd9Sstevel@tonic-gate 		case PT_VALUE:
29147c478bd9Sstevel@tonic-gate 			(void) strlcpy(attrtab->zone_attr_value, pp->pv_simple,
29157c478bd9Sstevel@tonic-gate 			    sizeof (attrtab->zone_attr_value));
29167c478bd9Sstevel@tonic-gate 			break;
29177c478bd9Sstevel@tonic-gate 		default:
29187c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2919bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
29207c478bd9Sstevel@tonic-gate 			return (Z_INSUFFICIENT_SPEC);
29217c478bd9Sstevel@tonic-gate 		}
29227c478bd9Sstevel@tonic-gate 	}
29237c478bd9Sstevel@tonic-gate 	if (fill_in_only)
29247c478bd9Sstevel@tonic-gate 		return (Z_OK);
29257c478bd9Sstevel@tonic-gate 	err = zonecfg_lookup_attr(handle, attrtab);
29267c478bd9Sstevel@tonic-gate 	return (err);
29277c478bd9Sstevel@tonic-gate }
29287c478bd9Sstevel@tonic-gate 
2929fa9e4066Sahrens static int
fill_in_dstab(cmd_t * cmd,struct zone_dstab * dstab,boolean_t fill_in_only)2930bbec428eSgjelinek fill_in_dstab(cmd_t *cmd, struct zone_dstab *dstab, boolean_t fill_in_only)
2931fa9e4066Sahrens {
2932fa9e4066Sahrens 	int err, i;
2933fa9e4066Sahrens 	property_value_ptr_t pp;
2934fa9e4066Sahrens 
2935bbec428eSgjelinek 	if ((err = initialize(B_TRUE)) != Z_OK)
2936fa9e4066Sahrens 		return (err);
2937fa9e4066Sahrens 
2938fa9e4066Sahrens 	dstab->zone_dataset_name[0] = '\0';
2939fa9e4066Sahrens 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
2940fa9e4066Sahrens 		pp = cmd->cmd_property_ptr[i];
2941fa9e4066Sahrens 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
2942fa9e4066Sahrens 			zerr(gettext("A simple value was expected here."));
2943bbec428eSgjelinek 			saw_error = B_TRUE;
2944fa9e4066Sahrens 			return (Z_INSUFFICIENT_SPEC);
2945fa9e4066Sahrens 		}
2946fa9e4066Sahrens 		switch (cmd->cmd_prop_name[i]) {
2947fa9e4066Sahrens 		case PT_NAME:
2948fa9e4066Sahrens 			(void) strlcpy(dstab->zone_dataset_name, pp->pv_simple,
2949fa9e4066Sahrens 			    sizeof (dstab->zone_dataset_name));
2950fa9e4066Sahrens 			break;
2951fa9e4066Sahrens 		default:
2952fa9e4066Sahrens 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2953bbec428eSgjelinek 			    Z_NO_PROPERTY_TYPE, B_TRUE);
2954fa9e4066Sahrens 			return (Z_INSUFFICIENT_SPEC);
2955fa9e4066Sahrens 		}
2956fa9e4066Sahrens 	}
2957fa9e4066Sahrens 	if (fill_in_only)
2958fa9e4066Sahrens 		return (Z_OK);
2959fa9e4066Sahrens 	return (zonecfg_lookup_ds(handle, dstab));
2960fa9e4066Sahrens }
2961fa9e4066Sahrens 
2962cb8a054bSGlenn Faden static int
fill_in_admintab(cmd_t * cmd,struct zone_admintab * admintab,boolean_t fill_in_only)2963cb8a054bSGlenn Faden fill_in_admintab(cmd_t *cmd, struct zone_admintab *admintab,
2964cb8a054bSGlenn Faden     boolean_t fill_in_only)
2965cb8a054bSGlenn Faden {
2966cb8a054bSGlenn Faden 	int err, i;
2967cb8a054bSGlenn Faden 	property_value_ptr_t pp;
2968cb8a054bSGlenn Faden 
2969cb8a054bSGlenn Faden 	if ((err = initialize(B_TRUE)) != Z_OK)
2970cb8a054bSGlenn Faden 		return (err);
2971cb8a054bSGlenn Faden 
2972cb8a054bSGlenn Faden 	bzero(admintab, sizeof (*admintab));
2973cb8a054bSGlenn Faden 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
2974cb8a054bSGlenn Faden 		pp = cmd->cmd_property_ptr[i];
2975cb8a054bSGlenn Faden 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
2976cb8a054bSGlenn Faden 			zerr(gettext("A simple value was expected here."));
2977cb8a054bSGlenn Faden 			saw_error = B_TRUE;
2978cb8a054bSGlenn Faden 			return (Z_INSUFFICIENT_SPEC);
2979cb8a054bSGlenn Faden 		}
2980cb8a054bSGlenn Faden 		switch (cmd->cmd_prop_name[i]) {
2981cb8a054bSGlenn Faden 		case PT_USER:
2982cb8a054bSGlenn Faden 			(void) strlcpy(admintab->zone_admin_user, pp->pv_simple,
2983cb8a054bSGlenn Faden 			    sizeof (admintab->zone_admin_user));
2984cb8a054bSGlenn Faden 			break;
2985cb8a054bSGlenn Faden 		case PT_AUTHS:
2986cb8a054bSGlenn Faden 			(void) strlcpy(admintab->zone_admin_auths,
2987cb8a054bSGlenn Faden 			    pp->pv_simple, sizeof (admintab->zone_admin_auths));
2988cb8a054bSGlenn Faden 			break;
2989cb8a054bSGlenn Faden 		default:
2990cb8a054bSGlenn Faden 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
2991cb8a054bSGlenn Faden 			    Z_NO_PROPERTY_TYPE, B_TRUE);
2992cb8a054bSGlenn Faden 			return (Z_INSUFFICIENT_SPEC);
2993cb8a054bSGlenn Faden 		}
2994cb8a054bSGlenn Faden 	}
2995cb8a054bSGlenn Faden 	if (fill_in_only)
2996cb8a054bSGlenn Faden 		return (Z_OK);
2997cb8a054bSGlenn Faden 	err = zonecfg_lookup_admin(handle, admintab);
2998cb8a054bSGlenn Faden 	return (err);
2999cb8a054bSGlenn Faden }
3000cb8a054bSGlenn Faden 
3001d2a70789SRichard Lowe static int
fill_in_secflagstab(cmd_t * cmd,struct zone_secflagstab * secflagstab,boolean_t fill_in_only)3002d2a70789SRichard Lowe fill_in_secflagstab(cmd_t *cmd, struct zone_secflagstab *secflagstab,
3003d2a70789SRichard Lowe     boolean_t fill_in_only)
3004d2a70789SRichard Lowe {
3005d2a70789SRichard Lowe 	int err, i;
3006d2a70789SRichard Lowe 	property_value_ptr_t pp;
3007d2a70789SRichard Lowe 
3008d2a70789SRichard Lowe 	if ((err = initialize(B_TRUE)) != Z_OK)
3009d2a70789SRichard Lowe 		return (err);
3010d2a70789SRichard Lowe 
3011d2a70789SRichard Lowe 	bzero(secflagstab, sizeof (*secflagstab));
3012d2a70789SRichard Lowe 	for (i = 0; i < cmd->cmd_prop_nv_pairs; i++) {
3013d2a70789SRichard Lowe 		pp = cmd->cmd_property_ptr[i];
3014d2a70789SRichard Lowe 		if (pp->pv_type != PROP_VAL_SIMPLE || pp->pv_simple == NULL) {
3015d2a70789SRichard Lowe 			zerr(gettext("A simple value was expected here."));
3016d2a70789SRichard Lowe 			saw_error = B_TRUE;
3017d2a70789SRichard Lowe 			return (Z_INSUFFICIENT_SPEC);
3018d2a70789SRichard Lowe 		}
3019d2a70789SRichard Lowe 		switch (cmd->cmd_prop_name[i]) {
3020d2a70789SRichard Lowe 		case PT_DEFAULT:
3021d2a70789SRichard Lowe 			(void) strlcpy(secflagstab->zone_secflags_default,
3022d2a70789SRichard Lowe 			    pp->pv_simple,
3023d2a70789SRichard Lowe 			    sizeof (secflagstab->zone_secflags_default));
3024d2a70789SRichard Lowe 			break;
3025d2a70789SRichard Lowe 		case PT_LOWER:
3026d2a70789SRichard Lowe 			(void) strlcpy(secflagstab->zone_secflags_lower,
3027d2a70789SRichard Lowe 			    pp->pv_simple,
3028d2a70789SRichard Lowe 			    sizeof (secflagstab->zone_secflags_lower));
3029d2a70789SRichard Lowe 			break;
3030d2a70789SRichard Lowe 		case PT_UPPER:
3031d2a70789SRichard Lowe 			(void) strlcpy(secflagstab->zone_secflags_upper,
3032d2a70789SRichard Lowe 			    pp->pv_simple,
3033d2a70789SRichard Lowe 			    sizeof (secflagstab->zone_secflags_upper));
3034d2a70789SRichard Lowe 			break;
3035d2a70789SRichard Lowe 		default:
3036d2a70789SRichard Lowe 			zone_perror(pt_to_str(cmd->cmd_prop_name[i]),
3037d2a70789SRichard Lowe 			    Z_NO_PROPERTY_TYPE, B_TRUE);
3038d2a70789SRichard Lowe 			return (Z_INSUFFICIENT_SPEC);
3039d2a70789SRichard Lowe 		}
3040d2a70789SRichard Lowe 	}
3041d2a70789SRichard Lowe 	if (fill_in_only)
3042d2a70789SRichard Lowe 		return (Z_OK);
3043d2a70789SRichard Lowe 
3044d2a70789SRichard Lowe 	err = zonecfg_lookup_secflags(handle, secflagstab);
3045d2a70789SRichard Lowe 
3046d2a70789SRichard Lowe 	return (err);
3047d2a70789SRichard Lowe }
3048d2a70789SRichard Lowe 
30497c478bd9Sstevel@tonic-gate static void
remove_aliased_rctl(int type,char * name)30500209230bSgjelinek remove_aliased_rctl(int type, char *name)
30517c478bd9Sstevel@tonic-gate {
30520209230bSgjelinek 	int err;
30530209230bSgjelinek 	uint64_t tmp;
30547c478bd9Sstevel@tonic-gate 
30550209230bSgjelinek 	if ((err = zonecfg_get_aliased_rctl(handle, name, &tmp)) != Z_OK) {
30560209230bSgjelinek 		zerr("%s %s: %s", cmd_to_str(CMD_CLEAR), pt_to_str(type),
30570209230bSgjelinek 		    zonecfg_strerror(err));
3058bbec428eSgjelinek 		saw_error = B_TRUE;
30597c478bd9Sstevel@tonic-gate 		return;
30607c478bd9Sstevel@tonic-gate 	}
30610209230bSgjelinek 	if ((err = zonecfg_rm_aliased_rctl(handle, name)) != Z_OK) {
30620209230bSgjelinek 		zerr("%s %s: %s", cmd_to_str(CMD_CLEAR), pt_to_str(type),
30630209230bSgjelinek 		    zonecfg_strerror(err));
3064bbec428eSgjelinek 		saw_error = B_TRUE;
30650209230bSgjelinek 	} else {
3066bbec428eSgjelinek 		need_to_commit = B_TRUE;
30670209230bSgjelinek 	}
30680209230bSgjelinek }
30697c478bd9Sstevel@tonic-gate 
30700209230bSgjelinek static boolean_t
prompt_remove_resource(cmd_t * cmd,char * rsrc)30710209230bSgjelinek prompt_remove_resource(cmd_t *cmd, char *rsrc)
30720209230bSgjelinek {
30730209230bSgjelinek 	int num;
30740209230bSgjelinek 	int answer;
30750209230bSgjelinek 	int arg;
30760209230bSgjelinek 	boolean_t force = B_FALSE;
30770209230bSgjelinek 	char prompt[128];
3078bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
30797c478bd9Sstevel@tonic-gate 
30800209230bSgjelinek 	optind = 0;
30810209230bSgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "F")) != EOF) {
30820209230bSgjelinek 		switch (arg) {
30830209230bSgjelinek 		case 'F':
30840209230bSgjelinek 			force = B_TRUE;
30850209230bSgjelinek 			break;
30860209230bSgjelinek 		default:
3087bbec428eSgjelinek 			arg_err = B_TRUE;
30887ec75eb8Sgjelinek 			break;
30897ec75eb8Sgjelinek 		}
30907ec75eb8Sgjelinek 	}
30917ec75eb8Sgjelinek 	if (arg_err)
30920209230bSgjelinek 		return (B_FALSE);
30937ec75eb8Sgjelinek 
30940209230bSgjelinek 
30950209230bSgjelinek 	num = zonecfg_num_resources(handle, rsrc);
30960209230bSgjelinek 
30970209230bSgjelinek 	if (num == 0) {
30980209230bSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, cmd->cmd_res_type, Z_NO_ENTRY,
3099bbec428eSgjelinek 		    B_TRUE);
31000209230bSgjelinek 		return (B_FALSE);
31010209230bSgjelinek 	}
31020209230bSgjelinek 	if (num > 1 && !force) {
31030209230bSgjelinek 		if (!interactive_mode) {
31040209230bSgjelinek 			zerr(gettext("There are multiple instances of this "
31050209230bSgjelinek 			    "resource.  Either qualify the resource to\n"
31060209230bSgjelinek 			    "remove a single instance or use the -F option to "
31070209230bSgjelinek 			    "remove all instances."));
3108bbec428eSgjelinek 			saw_error = B_TRUE;
31090209230bSgjelinek 			return (B_FALSE);
31100209230bSgjelinek 		}
31110209230bSgjelinek 		(void) snprintf(prompt, sizeof (prompt), gettext(
31120209230bSgjelinek 		    "Are you sure you want to remove ALL '%s' resources"),
31130209230bSgjelinek 		    rsrc);
3114bbec428eSgjelinek 		answer = ask_yesno(B_FALSE, prompt);
31150209230bSgjelinek 		if (answer == -1) {
31160209230bSgjelinek 			zerr(gettext("Resource incomplete."));
31170209230bSgjelinek 			return (B_FALSE);
31180209230bSgjelinek 		}
31190209230bSgjelinek 		if (answer != 1)
31200209230bSgjelinek 			return (B_FALSE);
31210209230bSgjelinek 	}
31220209230bSgjelinek 	return (B_TRUE);
31230209230bSgjelinek }
31240209230bSgjelinek 
31250209230bSgjelinek static void
remove_fs(cmd_t * cmd)31260209230bSgjelinek remove_fs(cmd_t *cmd)
31270209230bSgjelinek {
31280209230bSgjelinek 	int err;
31290209230bSgjelinek 
31300209230bSgjelinek 	/* traditional, qualified fs removal */
31310209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
31320209230bSgjelinek 		struct zone_fstab fstab;
31330209230bSgjelinek 
3134bbec428eSgjelinek 		if ((err = fill_in_fstab(cmd, &fstab, B_FALSE)) != Z_OK) {
3135bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_FS, err, B_TRUE);
31367c478bd9Sstevel@tonic-gate 			return;
31377c478bd9Sstevel@tonic-gate 		}
31387c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_delete_filesystem(handle, &fstab)) != Z_OK)
3139bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_FS, err, B_TRUE);
31407c478bd9Sstevel@tonic-gate 		else
3141bbec428eSgjelinek 			need_to_commit = B_TRUE;
31427c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(fstab.zone_fs_options);
31437c478bd9Sstevel@tonic-gate 		return;
31440209230bSgjelinek 	}
31450209230bSgjelinek 
31460209230bSgjelinek 	/*
31470209230bSgjelinek 	 * unqualified fs removal.  remove all fs's but prompt if more
31480209230bSgjelinek 	 * than one.
31490209230bSgjelinek 	 */
31500209230bSgjelinek 	if (!prompt_remove_resource(cmd, "fs"))
31510209230bSgjelinek 		return;
31520209230bSgjelinek 
31530209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "fs")) != Z_OK)
3154bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_FS, err, B_TRUE);
31550209230bSgjelinek 	else
3156bbec428eSgjelinek 		need_to_commit = B_TRUE;
31570209230bSgjelinek }
31580209230bSgjelinek 
31590209230bSgjelinek static void
remove_net(cmd_t * cmd)31600209230bSgjelinek remove_net(cmd_t *cmd)
31610209230bSgjelinek {
31620209230bSgjelinek 	int err;
31630209230bSgjelinek 
31640209230bSgjelinek 	/* traditional, qualified net removal */
31650209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
31660209230bSgjelinek 		struct zone_nwiftab nwiftab;
31670209230bSgjelinek 
3168bbec428eSgjelinek 		if ((err = fill_in_nwiftab(cmd, &nwiftab, B_FALSE)) != Z_OK) {
3169bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_NET, err, B_TRUE);
31707c478bd9Sstevel@tonic-gate 			return;
31717c478bd9Sstevel@tonic-gate 		}
31727c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_delete_nwif(handle, &nwiftab)) != Z_OK)
3173bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_NET, err, B_TRUE);
31747c478bd9Sstevel@tonic-gate 		else
3175bbec428eSgjelinek 			need_to_commit = B_TRUE;
31767c478bd9Sstevel@tonic-gate 		return;
31770209230bSgjelinek 	}
31780209230bSgjelinek 
31790209230bSgjelinek 	/*
31800209230bSgjelinek 	 * unqualified net removal.  remove all nets but prompt if more
31810209230bSgjelinek 	 * than one.
31820209230bSgjelinek 	 */
31830209230bSgjelinek 	if (!prompt_remove_resource(cmd, "net"))
31840209230bSgjelinek 		return;
31850209230bSgjelinek 
31860209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "net")) != Z_OK)
3187bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_NET, err, B_TRUE);
31880209230bSgjelinek 	else
3189bbec428eSgjelinek 		need_to_commit = B_TRUE;
31900209230bSgjelinek }
31910209230bSgjelinek 
31920209230bSgjelinek static void
remove_device(cmd_t * cmd)31930209230bSgjelinek remove_device(cmd_t *cmd)
31940209230bSgjelinek {
31950209230bSgjelinek 	int err;
31960209230bSgjelinek 
31970209230bSgjelinek 	/* traditional, qualified device removal */
31980209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
31990209230bSgjelinek 		struct zone_devtab devtab;
32000209230bSgjelinek 
3201bbec428eSgjelinek 		if ((err = fill_in_devtab(cmd, &devtab, B_FALSE)) != Z_OK) {
3202bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_DEVICE, err, B_TRUE);
32037c478bd9Sstevel@tonic-gate 			return;
32047c478bd9Sstevel@tonic-gate 		}
32057c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_delete_dev(handle, &devtab)) != Z_OK)
3206bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_DEVICE, err, B_TRUE);
32077c478bd9Sstevel@tonic-gate 		else
3208bbec428eSgjelinek 			need_to_commit = B_TRUE;
32097c478bd9Sstevel@tonic-gate 		return;
32100209230bSgjelinek 	}
32110209230bSgjelinek 
32120209230bSgjelinek 	/*
32130209230bSgjelinek 	 * unqualified device removal.  remove all devices but prompt if more
32140209230bSgjelinek 	 * than one.
32150209230bSgjelinek 	 */
32160209230bSgjelinek 	if (!prompt_remove_resource(cmd, "device"))
32170209230bSgjelinek 		return;
32180209230bSgjelinek 
32190209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "device")) != Z_OK)
3220bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_DEVICE, err, B_TRUE);
32210209230bSgjelinek 	else
3222bbec428eSgjelinek 		need_to_commit = B_TRUE;
32230209230bSgjelinek }
32240209230bSgjelinek 
32250209230bSgjelinek static void
remove_attr(cmd_t * cmd)32260209230bSgjelinek remove_attr(cmd_t *cmd)
32270209230bSgjelinek {
32280209230bSgjelinek 	int err;
32290209230bSgjelinek 
32300209230bSgjelinek 	/* traditional, qualified attr removal */
32310209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
32320209230bSgjelinek 		struct zone_attrtab attrtab;
32330209230bSgjelinek 
3234bbec428eSgjelinek 		if ((err = fill_in_attrtab(cmd, &attrtab, B_FALSE)) != Z_OK) {
3235bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_ATTR, err, B_TRUE);
32360209230bSgjelinek 			return;
32370209230bSgjelinek 		}
32380209230bSgjelinek 		if ((err = zonecfg_delete_attr(handle, &attrtab)) != Z_OK)
3239bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_ATTR, err, B_TRUE);
32400209230bSgjelinek 		else
3241bbec428eSgjelinek 			need_to_commit = B_TRUE;
32420209230bSgjelinek 		return;
32430209230bSgjelinek 	}
32440209230bSgjelinek 
32450209230bSgjelinek 	/*
32460209230bSgjelinek 	 * unqualified attr removal.  remove all attrs but prompt if more
32470209230bSgjelinek 	 * than one.
32480209230bSgjelinek 	 */
32490209230bSgjelinek 	if (!prompt_remove_resource(cmd, "attr"))
32500209230bSgjelinek 		return;
32510209230bSgjelinek 
32520209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "attr")) != Z_OK)
3253bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_ATTR, err, B_TRUE);
32540209230bSgjelinek 	else
3255bbec428eSgjelinek 		need_to_commit = B_TRUE;
32560209230bSgjelinek }
32570209230bSgjelinek 
32580209230bSgjelinek static void
remove_dataset(cmd_t * cmd)32590209230bSgjelinek remove_dataset(cmd_t *cmd)
32600209230bSgjelinek {
32610209230bSgjelinek 	int err;
32620209230bSgjelinek 
32630209230bSgjelinek 	/* traditional, qualified dataset removal */
32640209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
32650209230bSgjelinek 		struct zone_dstab dstab;
32660209230bSgjelinek 
3267bbec428eSgjelinek 		if ((err = fill_in_dstab(cmd, &dstab, B_FALSE)) != Z_OK) {
3268bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_DATASET, err, B_TRUE);
32690209230bSgjelinek 			return;
32700209230bSgjelinek 		}
32710209230bSgjelinek 		if ((err = zonecfg_delete_ds(handle, &dstab)) != Z_OK)
3272bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_DATASET, err, B_TRUE);
32730209230bSgjelinek 		else
3274bbec428eSgjelinek 			need_to_commit = B_TRUE;
32750209230bSgjelinek 		return;
32760209230bSgjelinek 	}
32770209230bSgjelinek 
32780209230bSgjelinek 	/*
32790209230bSgjelinek 	 * unqualified dataset removal.  remove all datasets but prompt if more
32800209230bSgjelinek 	 * than one.
32810209230bSgjelinek 	 */
32820209230bSgjelinek 	if (!prompt_remove_resource(cmd, "dataset"))
32830209230bSgjelinek 		return;
32840209230bSgjelinek 
32850209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "dataset")) != Z_OK)
3286bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_DATASET, err, B_TRUE);
32870209230bSgjelinek 	else
3288bbec428eSgjelinek 		need_to_commit = B_TRUE;
32890209230bSgjelinek }
32900209230bSgjelinek 
32910209230bSgjelinek static void
remove_rctl(cmd_t * cmd)32920209230bSgjelinek remove_rctl(cmd_t *cmd)
32930209230bSgjelinek {
32940209230bSgjelinek 	int err;
32950209230bSgjelinek 
32960209230bSgjelinek 	/* traditional, qualified rctl removal */
32970209230bSgjelinek 	if (cmd->cmd_prop_nv_pairs > 0) {
32980209230bSgjelinek 		struct zone_rctltab rctltab;
32990209230bSgjelinek 
3300bbec428eSgjelinek 		if ((err = fill_in_rctltab(cmd, &rctltab, B_FALSE)) != Z_OK) {
3301bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_RCTL, err, B_TRUE);
33027c478bd9Sstevel@tonic-gate 			return;
33037c478bd9Sstevel@tonic-gate 		}
33047c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_delete_rctl(handle, &rctltab)) != Z_OK)
3305bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_RCTL, err, B_TRUE);
33067c478bd9Sstevel@tonic-gate 		else
3307bbec428eSgjelinek 			need_to_commit = B_TRUE;
33087c478bd9Sstevel@tonic-gate 		zonecfg_free_rctl_value_list(rctltab.zone_rctl_valptr);
33097c478bd9Sstevel@tonic-gate 		return;
33107c478bd9Sstevel@tonic-gate 	}
33110209230bSgjelinek 
33120209230bSgjelinek 	/*
33130209230bSgjelinek 	 * unqualified rctl removal.  remove all rctls but prompt if more
33140209230bSgjelinek 	 * than one.
33150209230bSgjelinek 	 */
33160209230bSgjelinek 	if (!prompt_remove_resource(cmd, "rctl"))
33170209230bSgjelinek 		return;
33180209230bSgjelinek 
33190209230bSgjelinek 	if ((err = zonecfg_del_all_resources(handle, "rctl")) != Z_OK)
3320bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_RCTL, err, B_TRUE);
33217c478bd9Sstevel@tonic-gate 	else
3322bbec428eSgjelinek 		need_to_commit = B_TRUE;
33230209230bSgjelinek }
33240209230bSgjelinek 
33250209230bSgjelinek static void
remove_pset()33260209230bSgjelinek remove_pset()
33270209230bSgjelinek {
33280209230bSgjelinek 	int err;
33290209230bSgjelinek 	struct zone_psettab psettab;
33300209230bSgjelinek 
33310209230bSgjelinek 	if ((err = zonecfg_lookup_pset(handle, &psettab)) != Z_OK) {
3332bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_DCPU, err, B_TRUE);
33330209230bSgjelinek 		return;
33340209230bSgjelinek 	}
33350209230bSgjelinek 	if ((err = zonecfg_delete_pset(handle)) != Z_OK)
3336bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_DCPU, err, B_TRUE);
33370209230bSgjelinek 	else
3338bbec428eSgjelinek 		need_to_commit = B_TRUE;
33390209230bSgjelinek }
33400209230bSgjelinek 
33410209230bSgjelinek static void
remove_pcap()3342c97ad5cdSakolb remove_pcap()
3343c97ad5cdSakolb {
3344c97ad5cdSakolb 	int err;
3345c97ad5cdSakolb 	uint64_t tmp;
3346c97ad5cdSakolb 
3347c97ad5cdSakolb 	if (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &tmp) != Z_OK) {
3348c97ad5cdSakolb 		zerr("%s %s: %s", cmd_to_str(CMD_REMOVE), rt_to_str(RT_PCAP),
3349c97ad5cdSakolb 		    zonecfg_strerror(Z_NO_RESOURCE_TYPE));
3350bbec428eSgjelinek 		saw_error = B_TRUE;
3351c97ad5cdSakolb 		return;
3352c97ad5cdSakolb 	}
3353c97ad5cdSakolb 
3354c97ad5cdSakolb 	if ((err = zonecfg_rm_aliased_rctl(handle, ALIAS_CPUCAP)) != Z_OK)
3355bbec428eSgjelinek 		z_cmd_rt_perror(CMD_REMOVE, RT_PCAP, err, B_TRUE);
3356c97ad5cdSakolb 	else
3357bbec428eSgjelinek 		need_to_commit = B_TRUE;
3358c97ad5cdSakolb }
3359c97ad5cdSakolb 
3360c97ad5cdSakolb static void
remove_mcap()33610209230bSgjelinek remove_mcap()
33620209230bSgjelinek {
33630209230bSgjelinek 	int err, res1, res2, res3;
33640209230bSgjelinek 	uint64_t tmp;
33650209230bSgjelinek 	struct zone_mcaptab mcaptab;
33660209230bSgjelinek 	boolean_t revert = B_FALSE;
33670209230bSgjelinek 
33680209230bSgjelinek 	res1 = zonecfg_lookup_mcap(handle, &mcaptab);
33690209230bSgjelinek 	res2 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP, &tmp);
33700209230bSgjelinek 	res3 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM, &tmp);
33710209230bSgjelinek 
33720209230bSgjelinek 	/* if none of these exist, there is no resource to remove */
33730209230bSgjelinek 	if (res1 != Z_OK && res2 != Z_OK && res3 != Z_OK) {
33740209230bSgjelinek 		zerr("%s %s: %s", cmd_to_str(CMD_REMOVE), rt_to_str(RT_MCAP),
33750209230bSgjelinek 		    zonecfg_strerror(Z_NO_RESOURCE_TYPE));
3376bbec428eSgjelinek 		saw_error = B_TRUE;
33770209230bSgjelinek 		return;
33780209230bSgjelinek 	}
33790209230bSgjelinek 	if (res1 == Z_OK) {
33800209230bSgjelinek 		if ((err = zonecfg_delete_mcap(handle)) != Z_OK) {
3381bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_MCAP, err, B_TRUE);
33820209230bSgjelinek 			revert = B_TRUE;
33830209230bSgjelinek 		} else {
3384bbec428eSgjelinek 			need_to_commit = B_TRUE;
33850209230bSgjelinek 		}
33860209230bSgjelinek 	}
33870209230bSgjelinek 	if (res2 == Z_OK) {
33880209230bSgjelinek 		if ((err = zonecfg_rm_aliased_rctl(handle, ALIAS_MAXSWAP))
33890209230bSgjelinek 		    != Z_OK) {
3390bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_MCAP, err, B_TRUE);
33910209230bSgjelinek 			revert = B_TRUE;
33920209230bSgjelinek 		} else {
3393bbec428eSgjelinek 			need_to_commit = B_TRUE;
33940209230bSgjelinek 		}
33950209230bSgjelinek 	}
33960209230bSgjelinek 	if (res3 == Z_OK) {
33970209230bSgjelinek 		if ((err = zonecfg_rm_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM))
33980209230bSgjelinek 		    != Z_OK) {
3399bbec428eSgjelinek 			z_cmd_rt_perror(CMD_REMOVE, RT_MCAP, err, B_TRUE);
34000209230bSgjelinek 			revert = B_TRUE;
34010209230bSgjelinek 		} else {
3402bbec428eSgjelinek 			need_to_commit = B_TRUE;
34030209230bSgjelinek 		}
34040209230bSgjelinek 	}
34050209230bSgjelinek 
34060209230bSgjelinek 	if (revert)
3407bbec428eSgjelinek 		need_to_commit = B_FALSE;
34080209230bSgjelinek }
34090209230bSgjelinek 
34100209230bSgjelinek static void
remove_admin(cmd_t * cmd)3411cb8a054bSGlenn Faden remove_admin(cmd_t *cmd)
3412cb8a054bSGlenn Faden {
3413cb8a054bSGlenn Faden 	int err;
3414cb8a054bSGlenn Faden 
3415cb8a054bSGlenn Faden 	/* traditional, qualified attr removal */
3416cb8a054bSGlenn Faden 	if (cmd->cmd_prop_nv_pairs > 0) {
3417cb8a054bSGlenn Faden 		struct zone_admintab admintab;
3418cb8a054bSGlenn Faden 
3419cb8a054bSGlenn Faden 		if ((err = fill_in_admintab(cmd, &admintab, B_FALSE)) != Z_OK) {
3420cb8a054bSGlenn Faden 			z_cmd_rt_perror(CMD_REMOVE, RT_ADMIN,
3421cb8a054bSGlenn Faden 			    err, B_TRUE);
3422cb8a054bSGlenn Faden 			return;
3423cb8a054bSGlenn Faden 		}
3424cb8a054bSGlenn Faden 		if ((err = zonecfg_delete_admin(handle, &admintab,
3425cb8a054bSGlenn Faden 		    zone))
3426cb8a054bSGlenn Faden 		    != Z_OK)
3427cb8a054bSGlenn Faden 			z_cmd_rt_perror(CMD_REMOVE, RT_ADMIN,
3428cb8a054bSGlenn Faden 			    err, B_TRUE);
3429cb8a054bSGlenn Faden 		else
3430cb8a054bSGlenn Faden 			need_to_commit = B_TRUE;
3431cb8a054bSGlenn Faden 		return;
3432cb8a054bSGlenn Faden 	} else {
3433cb8a054bSGlenn Faden 		/*
3434cb8a054bSGlenn Faden 		 * unqualified admin removal.
3435cb8a054bSGlenn Faden 		 * remove all admins but prompt if more
3436cb8a054bSGlenn Faden 		 * than one.
3437cb8a054bSGlenn Faden 		 */
3438cb8a054bSGlenn Faden 		if (!prompt_remove_resource(cmd, "admin"))
3439cb8a054bSGlenn Faden 			return;
3440cb8a054bSGlenn Faden 
3441cb8a054bSGlenn Faden 		if ((err = zonecfg_delete_admins(handle, zone))
3442cb8a054bSGlenn Faden 		    != Z_OK)
3443cb8a054bSGlenn Faden 			z_cmd_rt_perror(CMD_REMOVE, RT_ADMIN,
3444cb8a054bSGlenn Faden 			    err, B_TRUE);
3445cb8a054bSGlenn Faden 		else
3446cb8a054bSGlenn Faden 			need_to_commit = B_TRUE;
3447cb8a054bSGlenn Faden 	}
3448cb8a054bSGlenn Faden }
3449cb8a054bSGlenn Faden 
3450cb8a054bSGlenn Faden static void
remove_secflags()3451d2a70789SRichard Lowe remove_secflags()
3452d2a70789SRichard Lowe {
3453d2a70789SRichard Lowe 	int err;
3454d2a70789SRichard Lowe 	struct zone_secflagstab sectab = { 0 };
3455d2a70789SRichard Lowe 
3456d2a70789SRichard Lowe 	if (zonecfg_lookup_secflags(handle, &sectab) != Z_OK) {
3457d2a70789SRichard Lowe 		zerr("%s %s: %s", cmd_to_str(CMD_REMOVE),
3458d2a70789SRichard Lowe 		    rt_to_str(RT_SECFLAGS),
3459d2a70789SRichard Lowe 		    zonecfg_strerror(Z_NO_RESOURCE_TYPE));
3460d2a70789SRichard Lowe 		return;
3461d2a70789SRichard Lowe 	}
3462d2a70789SRichard Lowe 
3463d2a70789SRichard Lowe 	if ((err = zonecfg_delete_secflags(handle, &sectab)) != Z_OK) {
3464d2a70789SRichard Lowe 		z_cmd_rt_perror(CMD_REMOVE, RT_SECFLAGS, err, B_TRUE);
3465d2a70789SRichard Lowe 		return;
3466d2a70789SRichard Lowe 	}
3467d2a70789SRichard Lowe 
3468d2a70789SRichard Lowe 	need_to_commit = B_TRUE;
3469d2a70789SRichard Lowe }
3470d2a70789SRichard Lowe 
3471d2a70789SRichard Lowe static void
remove_resource(cmd_t * cmd)34720209230bSgjelinek remove_resource(cmd_t *cmd)
34730209230bSgjelinek {
34740209230bSgjelinek 	int type;
34750209230bSgjelinek 	int arg;
3476bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
34770209230bSgjelinek 
34780209230bSgjelinek 	if ((type = cmd->cmd_res_type) == RT_UNKNOWN) {
3479bbec428eSgjelinek 		long_usage(CMD_REMOVE, B_TRUE);
34800209230bSgjelinek 		return;
34810209230bSgjelinek 	}
34820209230bSgjelinek 
34830209230bSgjelinek 	optind = 0;
34840209230bSgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?F")) != EOF) {
34850209230bSgjelinek 		switch (arg) {
34860209230bSgjelinek 		case '?':
34870209230bSgjelinek 			longer_usage(CMD_REMOVE);
3488bbec428eSgjelinek 			arg_err = B_TRUE;
34897ec75eb8Sgjelinek 			break;
34900209230bSgjelinek 		case 'F':
34910209230bSgjelinek 			break;
34920209230bSgjelinek 		default:
34930209230bSgjelinek 			short_usage(CMD_REMOVE);
3494bbec428eSgjelinek 			arg_err = B_TRUE;
34957ec75eb8Sgjelinek 			break;
34967ec75eb8Sgjelinek 		}
34977ec75eb8Sgjelinek 	}
34987ec75eb8Sgjelinek 	if (arg_err)
34990209230bSgjelinek 		return;
35000209230bSgjelinek 
3501bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
35020209230bSgjelinek 		return;
35030209230bSgjelinek 
35040209230bSgjelinek 	switch (type) {
35050209230bSgjelinek 	case RT_FS:
35060209230bSgjelinek 		remove_fs(cmd);
35070209230bSgjelinek 		return;
35080209230bSgjelinek 	case RT_NET:
35090209230bSgjelinek 		remove_net(cmd);
35100209230bSgjelinek 		return;
35110209230bSgjelinek 	case RT_DEVICE:
35120209230bSgjelinek 		remove_device(cmd);
35130209230bSgjelinek 		return;
35140209230bSgjelinek 	case RT_RCTL:
35150209230bSgjelinek 		remove_rctl(cmd);
35160209230bSgjelinek 		return;
35170209230bSgjelinek 	case RT_ATTR:
35180209230bSgjelinek 		remove_attr(cmd);
35197c478bd9Sstevel@tonic-gate 		return;
3520fa9e4066Sahrens 	case RT_DATASET:
35210209230bSgjelinek 		remove_dataset(cmd);
3522fa9e4066Sahrens 		return;
35230209230bSgjelinek 	case RT_DCPU:
35240209230bSgjelinek 		remove_pset();
35250209230bSgjelinek 		return;
3526c97ad5cdSakolb 	case RT_PCAP:
3527c97ad5cdSakolb 		remove_pcap();
3528c97ad5cdSakolb 		return;
35290209230bSgjelinek 	case RT_MCAP:
35300209230bSgjelinek 		remove_mcap();
3531fa9e4066Sahrens 		return;
3532cb8a054bSGlenn Faden 	case RT_ADMIN:
3533cb8a054bSGlenn Faden 		remove_admin(cmd);
3534cb8a054bSGlenn Faden 		return;
3535d2a70789SRichard Lowe 	case RT_SECFLAGS:
3536d2a70789SRichard Lowe 		remove_secflags();
3537d2a70789SRichard Lowe 		return;
35387c478bd9Sstevel@tonic-gate 	default:
3539bbec428eSgjelinek 		zone_perror(rt_to_str(type), Z_NO_RESOURCE_TYPE, B_TRUE);
3540bbec428eSgjelinek 		long_usage(CMD_REMOVE, B_TRUE);
3541bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
35427c478bd9Sstevel@tonic-gate 		return;
35437c478bd9Sstevel@tonic-gate 	}
35447c478bd9Sstevel@tonic-gate }
35457c478bd9Sstevel@tonic-gate 
35467c478bd9Sstevel@tonic-gate static void
remove_property(cmd_t * cmd)35477c478bd9Sstevel@tonic-gate remove_property(cmd_t *cmd)
35487c478bd9Sstevel@tonic-gate {
35497c478bd9Sstevel@tonic-gate 	char *prop_id;
35507c478bd9Sstevel@tonic-gate 	int err, res_type, prop_type;
35517c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
35527c478bd9Sstevel@tonic-gate 	struct zone_rctlvaltab *rctlvaltab;
35537c478bd9Sstevel@tonic-gate 	complex_property_ptr_t cx;
35547c478bd9Sstevel@tonic-gate 
35557c478bd9Sstevel@tonic-gate 	res_type = resource_scope;
35567c478bd9Sstevel@tonic-gate 	prop_type = cmd->cmd_prop_name[0];
35577c478bd9Sstevel@tonic-gate 	if (res_type == RT_UNKNOWN || prop_type == PT_UNKNOWN) {
3558bbec428eSgjelinek 		long_usage(CMD_REMOVE, B_TRUE);
35597c478bd9Sstevel@tonic-gate 		return;
35607c478bd9Sstevel@tonic-gate 	}
35617c478bd9Sstevel@tonic-gate 
35627c478bd9Sstevel@tonic-gate 	if (cmd->cmd_prop_nv_pairs != 1) {
3563bbec428eSgjelinek 		long_usage(CMD_ADD, B_TRUE);
35647c478bd9Sstevel@tonic-gate 		return;
35657c478bd9Sstevel@tonic-gate 	}
35667c478bd9Sstevel@tonic-gate 
3567bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
35687c478bd9Sstevel@tonic-gate 		return;
35697c478bd9Sstevel@tonic-gate 
35707c478bd9Sstevel@tonic-gate 	switch (res_type) {
35717c478bd9Sstevel@tonic-gate 	case RT_FS:
35727c478bd9Sstevel@tonic-gate 		if (prop_type != PT_OPTIONS) {
35737c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
3574bbec428eSgjelinek 			    B_TRUE);
3575bbec428eSgjelinek 			long_usage(CMD_REMOVE, B_TRUE);
3576bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
35777c478bd9Sstevel@tonic-gate 			return;
35787c478bd9Sstevel@tonic-gate 		}
35797c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[0];
35807c478bd9Sstevel@tonic-gate 		if (pp->pv_type == PROP_VAL_COMPLEX) {
35817c478bd9Sstevel@tonic-gate 			zerr(gettext("A %s or %s value was expected here."),
35827c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_SIMPLE),
35837c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_LIST));
3584bbec428eSgjelinek 			saw_error = B_TRUE;
35857c478bd9Sstevel@tonic-gate 			return;
35867c478bd9Sstevel@tonic-gate 		}
35877c478bd9Sstevel@tonic-gate 		if (pp->pv_type == PROP_VAL_SIMPLE) {
35887c478bd9Sstevel@tonic-gate 			if (pp->pv_simple == NULL) {
3589bbec428eSgjelinek 				long_usage(CMD_ADD, B_TRUE);
35907c478bd9Sstevel@tonic-gate 				return;
35917c478bd9Sstevel@tonic-gate 			}
35927c478bd9Sstevel@tonic-gate 			prop_id = pp->pv_simple;
35937c478bd9Sstevel@tonic-gate 			err = zonecfg_remove_fs_option(&in_progress_fstab,
35947c478bd9Sstevel@tonic-gate 			    prop_id);
35957c478bd9Sstevel@tonic-gate 			if (err != Z_OK)
3596bbec428eSgjelinek 				zone_perror(pt_to_str(prop_type), err, B_TRUE);
35977c478bd9Sstevel@tonic-gate 		} else {
35987c478bd9Sstevel@tonic-gate 			list_property_ptr_t list;
35997c478bd9Sstevel@tonic-gate 
36007c478bd9Sstevel@tonic-gate 			for (list = pp->pv_list; list != NULL;
36017c478bd9Sstevel@tonic-gate 			    list = list->lp_next) {
36027c478bd9Sstevel@tonic-gate 				prop_id = list->lp_simple;
36037c478bd9Sstevel@tonic-gate 				if (prop_id == NULL)
36047c478bd9Sstevel@tonic-gate 					break;
36057c478bd9Sstevel@tonic-gate 				err = zonecfg_remove_fs_option(
36067c478bd9Sstevel@tonic-gate 				    &in_progress_fstab, prop_id);
36077c478bd9Sstevel@tonic-gate 				if (err != Z_OK)
36087c478bd9Sstevel@tonic-gate 					zone_perror(pt_to_str(prop_type), err,
3609bbec428eSgjelinek 					    B_TRUE);
36107c478bd9Sstevel@tonic-gate 			}
36117c478bd9Sstevel@tonic-gate 		}
36127c478bd9Sstevel@tonic-gate 		return;
36137c478bd9Sstevel@tonic-gate 	case RT_RCTL:
36147c478bd9Sstevel@tonic-gate 		if (prop_type != PT_VALUE) {
36157c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
3616bbec428eSgjelinek 			    B_TRUE);
3617bbec428eSgjelinek 			long_usage(CMD_REMOVE, B_TRUE);
3618bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
36197c478bd9Sstevel@tonic-gate 			return;
36207c478bd9Sstevel@tonic-gate 		}
36217c478bd9Sstevel@tonic-gate 		pp = cmd->cmd_property_ptr[0];
36227c478bd9Sstevel@tonic-gate 		if (pp->pv_type != PROP_VAL_COMPLEX) {
36237c478bd9Sstevel@tonic-gate 			zerr(gettext("A %s value was expected here."),
36247c478bd9Sstevel@tonic-gate 			    pvt_to_str(PROP_VAL_COMPLEX));
3625bbec428eSgjelinek 			saw_error = B_TRUE;
36267c478bd9Sstevel@tonic-gate 			return;
36277c478bd9Sstevel@tonic-gate 		}
36287c478bd9Sstevel@tonic-gate 		if ((rctlvaltab = alloc_rctlvaltab()) == NULL) {
3629bbec428eSgjelinek 			zone_perror(zone, Z_NOMEM, B_TRUE);
36307c478bd9Sstevel@tonic-gate 			exit(Z_ERR);
36317c478bd9Sstevel@tonic-gate 		}
36327c478bd9Sstevel@tonic-gate 		for (cx = pp->pv_complex; cx != NULL; cx = cx->cp_next) {
36337c478bd9Sstevel@tonic-gate 			switch (cx->cp_type) {
36347c478bd9Sstevel@tonic-gate 			case PT_PRIV:
36357c478bd9Sstevel@tonic-gate 				(void) strlcpy(rctlvaltab->zone_rctlval_priv,
36367c478bd9Sstevel@tonic-gate 				    cx->cp_value,
36377c478bd9Sstevel@tonic-gate 				    sizeof (rctlvaltab->zone_rctlval_priv));
36387c478bd9Sstevel@tonic-gate 				break;
36397c478bd9Sstevel@tonic-gate 			case PT_LIMIT:
36407c478bd9Sstevel@tonic-gate 				(void) strlcpy(rctlvaltab->zone_rctlval_limit,
36417c478bd9Sstevel@tonic-gate 				    cx->cp_value,
36427c478bd9Sstevel@tonic-gate 				    sizeof (rctlvaltab->zone_rctlval_limit));
36437c478bd9Sstevel@tonic-gate 				break;
36447c478bd9Sstevel@tonic-gate 			case PT_ACTION:
36457c478bd9Sstevel@tonic-gate 				(void) strlcpy(rctlvaltab->zone_rctlval_action,
36467c478bd9Sstevel@tonic-gate 				    cx->cp_value,
36477c478bd9Sstevel@tonic-gate 				    sizeof (rctlvaltab->zone_rctlval_action));
36487c478bd9Sstevel@tonic-gate 				break;
36497c478bd9Sstevel@tonic-gate 			default:
36507c478bd9Sstevel@tonic-gate 				zone_perror(pt_to_str(prop_type),
3651bbec428eSgjelinek 				    Z_NO_PROPERTY_TYPE, B_TRUE);
3652bbec428eSgjelinek 				long_usage(CMD_ADD, B_TRUE);
3653bbec428eSgjelinek 				usage(B_FALSE, HELP_PROPS);
36547c478bd9Sstevel@tonic-gate 				zonecfg_free_rctl_value_list(rctlvaltab);
36557c478bd9Sstevel@tonic-gate 				return;
36567c478bd9Sstevel@tonic-gate 			}
36577c478bd9Sstevel@tonic-gate 		}
36587c478bd9Sstevel@tonic-gate 		rctlvaltab->zone_rctlval_next = NULL;
36597c478bd9Sstevel@tonic-gate 		err = zonecfg_remove_rctl_value(&in_progress_rctltab,
36607c478bd9Sstevel@tonic-gate 		    rctlvaltab);
36617c478bd9Sstevel@tonic-gate 		if (err != Z_OK)
3662bbec428eSgjelinek 			zone_perror(pt_to_str(prop_type), err, B_TRUE);
36637c478bd9Sstevel@tonic-gate 		zonecfg_free_rctl_value_list(rctlvaltab);
36647c478bd9Sstevel@tonic-gate 		return;
3665de860bd9Sgfaden 	case RT_NET:
3666de860bd9Sgfaden 		if (prop_type != PT_DEFROUTER) {
3667de860bd9Sgfaden 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
3668bbec428eSgjelinek 			    B_TRUE);
3669bbec428eSgjelinek 			long_usage(CMD_REMOVE, B_TRUE);
3670bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
3671de860bd9Sgfaden 			return;
3672de860bd9Sgfaden 		} else {
3673de860bd9Sgfaden 			bzero(&in_progress_nwiftab.zone_nwif_defrouter,
3674de860bd9Sgfaden 			    sizeof (in_progress_nwiftab.zone_nwif_defrouter));
3675de860bd9Sgfaden 			return;
3676de860bd9Sgfaden 		}
36777c478bd9Sstevel@tonic-gate 	default:
3678bbec428eSgjelinek 		zone_perror(rt_to_str(res_type), Z_NO_RESOURCE_TYPE, B_TRUE);
3679bbec428eSgjelinek 		long_usage(CMD_REMOVE, B_TRUE);
3680bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
36817c478bd9Sstevel@tonic-gate 		return;
36827c478bd9Sstevel@tonic-gate 	}
36837c478bd9Sstevel@tonic-gate }
36847c478bd9Sstevel@tonic-gate 
36857c478bd9Sstevel@tonic-gate void
remove_func(cmd_t * cmd)36867c478bd9Sstevel@tonic-gate remove_func(cmd_t *cmd)
36877c478bd9Sstevel@tonic-gate {
36887c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_REMOVE))
36897c478bd9Sstevel@tonic-gate 		return;
36907c478bd9Sstevel@tonic-gate 
36917c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
36927c478bd9Sstevel@tonic-gate 
36930209230bSgjelinek 	if (global_scope) {
36940209230bSgjelinek 		if (gz_invalid_resource(cmd->cmd_res_type)) {
36950209230bSgjelinek 			zerr(gettext("%s is not a valid resource for the "
36960209230bSgjelinek 			    "global zone."), rt_to_str(cmd->cmd_res_type));
3697bbec428eSgjelinek 			saw_error = B_TRUE;
36980209230bSgjelinek 			return;
36990209230bSgjelinek 		}
37007c478bd9Sstevel@tonic-gate 		remove_resource(cmd);
37010209230bSgjelinek 	} else {
37027c478bd9Sstevel@tonic-gate 		remove_property(cmd);
37037c478bd9Sstevel@tonic-gate 	}
37040209230bSgjelinek }
37050209230bSgjelinek 
37060209230bSgjelinek static void
clear_property(cmd_t * cmd)37070209230bSgjelinek clear_property(cmd_t *cmd)
37080209230bSgjelinek {
37090209230bSgjelinek 	int res_type, prop_type;
37100209230bSgjelinek 
37110209230bSgjelinek 	res_type = resource_scope;
37120209230bSgjelinek 	prop_type = cmd->cmd_res_type;
37130209230bSgjelinek 	if (res_type == RT_UNKNOWN || prop_type == PT_UNKNOWN) {
3714bbec428eSgjelinek 		long_usage(CMD_CLEAR, B_TRUE);
37150209230bSgjelinek 		return;
37160209230bSgjelinek 	}
37170209230bSgjelinek 
3718bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
37190209230bSgjelinek 		return;
37200209230bSgjelinek 
37210209230bSgjelinek 	switch (res_type) {
37220209230bSgjelinek 	case RT_FS:
37230209230bSgjelinek 		if (prop_type == PT_RAW) {
37240209230bSgjelinek 			in_progress_fstab.zone_fs_raw[0] = '\0';
3725bbec428eSgjelinek 			need_to_commit = B_TRUE;
37260209230bSgjelinek 			return;
37270209230bSgjelinek 		}
37280209230bSgjelinek 		break;
37290209230bSgjelinek 	case RT_DCPU:
37300209230bSgjelinek 		if (prop_type == PT_IMPORTANCE) {
37310209230bSgjelinek 			in_progress_psettab.zone_importance[0] = '\0';
3732bbec428eSgjelinek 			need_to_commit = B_TRUE;
37330209230bSgjelinek 			return;
37340209230bSgjelinek 		}
37350209230bSgjelinek 		break;
37360209230bSgjelinek 	case RT_MCAP:
37370209230bSgjelinek 		switch (prop_type) {
37380209230bSgjelinek 		case PT_PHYSICAL:
37390209230bSgjelinek 			in_progress_mcaptab.zone_physmem_cap[0] = '\0';
3740bbec428eSgjelinek 			need_to_commit = B_TRUE;
37410209230bSgjelinek 			return;
37420209230bSgjelinek 		case PT_SWAP:
37430209230bSgjelinek 			remove_aliased_rctl(PT_SWAP, ALIAS_MAXSWAP);
37440209230bSgjelinek 			return;
37450209230bSgjelinek 		case PT_LOCKED:
37460209230bSgjelinek 			remove_aliased_rctl(PT_LOCKED, ALIAS_MAXLOCKEDMEM);
37470209230bSgjelinek 			return;
37480209230bSgjelinek 		}
37490209230bSgjelinek 		break;
3750d2a70789SRichard Lowe 	case RT_SECFLAGS:
3751d2a70789SRichard Lowe 		switch (prop_type) {
3752d2a70789SRichard Lowe 		case PT_LOWER:
3753d2a70789SRichard Lowe 			in_progress_secflagstab.zone_secflags_lower[0] = '\0';
3754d2a70789SRichard Lowe 			need_to_commit = B_TRUE;
3755d2a70789SRichard Lowe 			return;
3756d2a70789SRichard Lowe 		case PT_DEFAULT:
3757d2a70789SRichard Lowe 			in_progress_secflagstab.zone_secflags_default[0] = '\0';
3758d2a70789SRichard Lowe 			need_to_commit = B_TRUE;
3759d2a70789SRichard Lowe 			return;
3760d2a70789SRichard Lowe 		case PT_UPPER:
3761d2a70789SRichard Lowe 			in_progress_secflagstab.zone_secflags_upper[0] = '\0';
3762d2a70789SRichard Lowe 			need_to_commit = B_TRUE;
3763d2a70789SRichard Lowe 			return;
3764d2a70789SRichard Lowe 		}
3765d2a70789SRichard Lowe 		break;
37660209230bSgjelinek 	default:
37670209230bSgjelinek 		break;
37680209230bSgjelinek 	}
37690209230bSgjelinek 
3770bbec428eSgjelinek 	zone_perror(pt_to_str(prop_type), Z_CLEAR_DISALLOW, B_TRUE);
37710209230bSgjelinek }
37720209230bSgjelinek 
37730209230bSgjelinek static void
clear_global(cmd_t * cmd)37740209230bSgjelinek clear_global(cmd_t *cmd)
37750209230bSgjelinek {
37760209230bSgjelinek 	int err, type;
37770209230bSgjelinek 
37780209230bSgjelinek 	if ((type = cmd->cmd_res_type) == RT_UNKNOWN) {
3779bbec428eSgjelinek 		long_usage(CMD_CLEAR, B_TRUE);
37800209230bSgjelinek 		return;
37810209230bSgjelinek 	}
37820209230bSgjelinek 
3783bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
37840209230bSgjelinek 		return;
37850209230bSgjelinek 
37860209230bSgjelinek 	switch (type) {
37870209230bSgjelinek 	case PT_ZONENAME:
37880209230bSgjelinek 		/* FALLTHRU */
37890209230bSgjelinek 	case PT_ZONEPATH:
37900209230bSgjelinek 		/* FALLTHRU */
37910209230bSgjelinek 	case PT_BRAND:
3792bbec428eSgjelinek 		zone_perror(pt_to_str(type), Z_CLEAR_DISALLOW, B_TRUE);
37930209230bSgjelinek 		return;
37940209230bSgjelinek 	case PT_AUTOBOOT:
37950209230bSgjelinek 		/* false is default; we'll treat as equivalent to clearing */
37960209230bSgjelinek 		if ((err = zonecfg_set_autoboot(handle, B_FALSE)) != Z_OK)
3797bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_AUTOBOOT, err, B_TRUE);
37980209230bSgjelinek 		else
3799bbec428eSgjelinek 			need_to_commit = B_TRUE;
38000209230bSgjelinek 		return;
38010209230bSgjelinek 	case PT_POOL:
38020209230bSgjelinek 		if ((err = zonecfg_set_pool(handle, NULL)) != Z_OK)
3803bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_POOL, err, B_TRUE);
38040209230bSgjelinek 		else
3805bbec428eSgjelinek 			need_to_commit = B_TRUE;
38060209230bSgjelinek 		return;
38070209230bSgjelinek 	case PT_LIMITPRIV:
38080209230bSgjelinek 		if ((err = zonecfg_set_limitpriv(handle, NULL)) != Z_OK)
3809bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_LIMITPRIV, err, B_TRUE);
38100209230bSgjelinek 		else
3811bbec428eSgjelinek 			need_to_commit = B_TRUE;
38120209230bSgjelinek 		return;
38130209230bSgjelinek 	case PT_BOOTARGS:
38140209230bSgjelinek 		if ((err = zonecfg_set_bootargs(handle, NULL)) != Z_OK)
3815bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_BOOTARGS, err, B_TRUE);
38160209230bSgjelinek 		else
3817bbec428eSgjelinek 			need_to_commit = B_TRUE;
38180209230bSgjelinek 		return;
38190209230bSgjelinek 	case PT_SCHED:
38200209230bSgjelinek 		if ((err = zonecfg_set_sched(handle, NULL)) != Z_OK)
3821bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_SCHED, err, B_TRUE);
38220209230bSgjelinek 		else
3823bbec428eSgjelinek 			need_to_commit = B_TRUE;
38240209230bSgjelinek 		return;
3825f4b3ec61Sdh155122 	case PT_IPTYPE:
3826f4b3ec61Sdh155122 		/* shared is default; we'll treat as equivalent to clearing */
3827f4b3ec61Sdh155122 		if ((err = zonecfg_set_iptype(handle, ZS_SHARED)) != Z_OK)
3828bbec428eSgjelinek 			z_cmd_rt_perror(CMD_CLEAR, RT_IPTYPE, err, B_TRUE);
3829f4b3ec61Sdh155122 		else
3830bbec428eSgjelinek 			need_to_commit = B_TRUE;
3831f4b3ec61Sdh155122 		return;
38320209230bSgjelinek 	case PT_MAXLWPS:
38330209230bSgjelinek 		remove_aliased_rctl(PT_MAXLWPS, ALIAS_MAXLWPS);
38340209230bSgjelinek 		return;
3835ff19e029SMenno Lageman 	case PT_MAXPROCS:
3836ff19e029SMenno Lageman 		remove_aliased_rctl(PT_MAXPROCS, ALIAS_MAXPROCS);
3837ff19e029SMenno Lageman 		return;
38380209230bSgjelinek 	case PT_MAXSHMMEM:
38390209230bSgjelinek 		remove_aliased_rctl(PT_MAXSHMMEM, ALIAS_MAXSHMMEM);
38400209230bSgjelinek 		return;
38410209230bSgjelinek 	case PT_MAXSHMIDS:
38420209230bSgjelinek 		remove_aliased_rctl(PT_MAXSHMIDS, ALIAS_MAXSHMIDS);
38430209230bSgjelinek 		return;
38440209230bSgjelinek 	case PT_MAXMSGIDS:
38450209230bSgjelinek 		remove_aliased_rctl(PT_MAXMSGIDS, ALIAS_MAXMSGIDS);
38460209230bSgjelinek 		return;
38470209230bSgjelinek 	case PT_MAXSEMIDS:
38480209230bSgjelinek 		remove_aliased_rctl(PT_MAXSEMIDS, ALIAS_MAXSEMIDS);
38490209230bSgjelinek 		return;
38500209230bSgjelinek 	case PT_SHARES:
38510209230bSgjelinek 		remove_aliased_rctl(PT_SHARES, ALIAS_SHARES);
38520209230bSgjelinek 		return;
38535679c89fSjv227347 	case PT_HOSTID:
38545679c89fSjv227347 		if ((err = zonecfg_set_hostid(handle, NULL)) != Z_OK)
38555679c89fSjv227347 			z_cmd_rt_perror(CMD_CLEAR, RT_HOSTID, err, B_TRUE);
38565679c89fSjv227347 		else
38575679c89fSjv227347 			need_to_commit = B_TRUE;
38585679c89fSjv227347 		return;
38590fbb751dSJohn Levon 	case PT_FS_ALLOWED:
38600fbb751dSJohn Levon 		if ((err = zonecfg_set_fs_allowed(handle, NULL)) != Z_OK)
38610fbb751dSJohn Levon 			z_cmd_rt_perror(CMD_CLEAR, RT_FS_ALLOWED, err, B_TRUE);
38620fbb751dSJohn Levon 		else
38630fbb751dSJohn Levon 			need_to_commit = B_TRUE;
38640fbb751dSJohn Levon 		return;
38650209230bSgjelinek 	default:
3866bbec428eSgjelinek 		zone_perror(pt_to_str(type), Z_NO_PROPERTY_TYPE, B_TRUE);
3867bbec428eSgjelinek 		long_usage(CMD_CLEAR, B_TRUE);
3868bbec428eSgjelinek 		usage(B_FALSE, HELP_PROPS);
38690209230bSgjelinek 		return;
38700209230bSgjelinek 	}
38710209230bSgjelinek }
38720209230bSgjelinek 
38730209230bSgjelinek void
clear_func(cmd_t * cmd)38740209230bSgjelinek clear_func(cmd_t *cmd)
38750209230bSgjelinek {
38760209230bSgjelinek 	if (zone_is_read_only(CMD_CLEAR))
38770209230bSgjelinek 		return;
38780209230bSgjelinek 
38790209230bSgjelinek 	assert(cmd != NULL);
38800209230bSgjelinek 
38810209230bSgjelinek 	if (global_scope) {
38820209230bSgjelinek 		if (gz_invalid_property(cmd->cmd_res_type)) {
38830209230bSgjelinek 			zerr(gettext("%s is not a valid property for the "
38840209230bSgjelinek 			    "global zone."), pt_to_str(cmd->cmd_res_type));
3885bbec428eSgjelinek 			saw_error = B_TRUE;
38860209230bSgjelinek 			return;
38870209230bSgjelinek 		}
38880209230bSgjelinek 
38890209230bSgjelinek 		clear_global(cmd);
38900209230bSgjelinek 	} else {
38910209230bSgjelinek 		clear_property(cmd);
38920209230bSgjelinek 	}
38930209230bSgjelinek }
38947c478bd9Sstevel@tonic-gate 
38957c478bd9Sstevel@tonic-gate void
select_func(cmd_t * cmd)38967c478bd9Sstevel@tonic-gate select_func(cmd_t *cmd)
38977c478bd9Sstevel@tonic-gate {
38980209230bSgjelinek 	int type, err, res;
38990209230bSgjelinek 	uint64_t limit;
3900c97ad5cdSakolb 	uint64_t tmp;
39017c478bd9Sstevel@tonic-gate 
39027c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_SELECT))
39037c478bd9Sstevel@tonic-gate 		return;
39047c478bd9Sstevel@tonic-gate 
39057c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
39067c478bd9Sstevel@tonic-gate 
39077c478bd9Sstevel@tonic-gate 	if (global_scope) {
3908bbec428eSgjelinek 		global_scope = B_FALSE;
39097c478bd9Sstevel@tonic-gate 		resource_scope = cmd->cmd_res_type;
39107c478bd9Sstevel@tonic-gate 		end_op = CMD_SELECT;
39117c478bd9Sstevel@tonic-gate 	} else {
39127c478bd9Sstevel@tonic-gate 		scope_usage(CMD_SELECT);
39137c478bd9Sstevel@tonic-gate 		return;
39147c478bd9Sstevel@tonic-gate 	}
39157c478bd9Sstevel@tonic-gate 
39167c478bd9Sstevel@tonic-gate 	if ((type = cmd->cmd_res_type) == RT_UNKNOWN) {
3917bbec428eSgjelinek 		long_usage(CMD_SELECT, B_TRUE);
39187c478bd9Sstevel@tonic-gate 		return;
39197c478bd9Sstevel@tonic-gate 	}
39207c478bd9Sstevel@tonic-gate 
3921bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
39227c478bd9Sstevel@tonic-gate 		return;
39237c478bd9Sstevel@tonic-gate 
39247c478bd9Sstevel@tonic-gate 	switch (type) {
39257c478bd9Sstevel@tonic-gate 	case RT_FS:
3926bbec428eSgjelinek 		if ((err = fill_in_fstab(cmd, &old_fstab, B_FALSE)) != Z_OK) {
3927bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_FS, err, B_TRUE);
3928bbec428eSgjelinek 			global_scope = B_TRUE;
39297c478bd9Sstevel@tonic-gate 		}
39307c478bd9Sstevel@tonic-gate 		bcopy(&old_fstab, &in_progress_fstab,
39317c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_fstab));
39327c478bd9Sstevel@tonic-gate 		return;
39337c478bd9Sstevel@tonic-gate 	case RT_NET:
3934bbec428eSgjelinek 		if ((err = fill_in_nwiftab(cmd, &old_nwiftab, B_FALSE))
3935bbec428eSgjelinek 		    != Z_OK) {
3936bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_NET, err, B_TRUE);
3937bbec428eSgjelinek 			global_scope = B_TRUE;
39387c478bd9Sstevel@tonic-gate 		}
39397c478bd9Sstevel@tonic-gate 		bcopy(&old_nwiftab, &in_progress_nwiftab,
39407c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_nwiftab));
39417c478bd9Sstevel@tonic-gate 		return;
39427c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
3943bbec428eSgjelinek 		if ((err = fill_in_devtab(cmd, &old_devtab, B_FALSE)) != Z_OK) {
3944bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_DEVICE, err, B_TRUE);
3945bbec428eSgjelinek 			global_scope = B_TRUE;
39467c478bd9Sstevel@tonic-gate 		}
39477c478bd9Sstevel@tonic-gate 		bcopy(&old_devtab, &in_progress_devtab,
39487c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_devtab));
39497c478bd9Sstevel@tonic-gate 		return;
39507c478bd9Sstevel@tonic-gate 	case RT_RCTL:
3951bbec428eSgjelinek 		if ((err = fill_in_rctltab(cmd, &old_rctltab, B_FALSE))
3952bbec428eSgjelinek 		    != Z_OK) {
3953bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_RCTL, err, B_TRUE);
3954bbec428eSgjelinek 			global_scope = B_TRUE;
39557c478bd9Sstevel@tonic-gate 		}
39567c478bd9Sstevel@tonic-gate 		bcopy(&old_rctltab, &in_progress_rctltab,
39577c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_rctltab));
39587c478bd9Sstevel@tonic-gate 		return;
39597c478bd9Sstevel@tonic-gate 	case RT_ATTR:
3960bbec428eSgjelinek 		if ((err = fill_in_attrtab(cmd, &old_attrtab, B_FALSE))
3961bbec428eSgjelinek 		    != Z_OK) {
3962bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_ATTR, err, B_TRUE);
3963bbec428eSgjelinek 			global_scope = B_TRUE;
39647c478bd9Sstevel@tonic-gate 		}
39657c478bd9Sstevel@tonic-gate 		bcopy(&old_attrtab, &in_progress_attrtab,
39667c478bd9Sstevel@tonic-gate 		    sizeof (struct zone_attrtab));
39677c478bd9Sstevel@tonic-gate 		return;
3968fa9e4066Sahrens 	case RT_DATASET:
3969bbec428eSgjelinek 		if ((err = fill_in_dstab(cmd, &old_dstab, B_FALSE)) != Z_OK) {
3970bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_DATASET, err, B_TRUE);
3971bbec428eSgjelinek 			global_scope = B_TRUE;
3972fa9e4066Sahrens 		}
3973fa9e4066Sahrens 		bcopy(&old_dstab, &in_progress_dstab,
3974fa9e4066Sahrens 		    sizeof (struct zone_dstab));
3975fa9e4066Sahrens 		return;
39760209230bSgjelinek 	case RT_DCPU:
39770209230bSgjelinek 		if ((err = zonecfg_lookup_pset(handle, &old_psettab)) != Z_OK) {
3978bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_DCPU, err, B_TRUE);
3979bbec428eSgjelinek 			global_scope = B_TRUE;
39800209230bSgjelinek 		}
39810209230bSgjelinek 		bcopy(&old_psettab, &in_progress_psettab,
39820209230bSgjelinek 		    sizeof (struct zone_psettab));
39830209230bSgjelinek 		return;
3984c97ad5cdSakolb 	case RT_PCAP:
3985c97ad5cdSakolb 		if ((err = zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &tmp))
3986c97ad5cdSakolb 		    != Z_OK) {
3987bbec428eSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_PCAP, err, B_TRUE);
3988bbec428eSgjelinek 			global_scope = B_TRUE;
3989c97ad5cdSakolb 		}
3990c97ad5cdSakolb 		return;
39910209230bSgjelinek 	case RT_MCAP:
39920209230bSgjelinek 		/* if none of these exist, there is no resource to select */
39930209230bSgjelinek 		if ((res = zonecfg_lookup_mcap(handle, &old_mcaptab)) != Z_OK &&
39940209230bSgjelinek 		    zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP, &limit)
39950209230bSgjelinek 		    != Z_OK &&
39960209230bSgjelinek 		    zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM, &limit)
39970209230bSgjelinek 		    != Z_OK) {
39980209230bSgjelinek 			z_cmd_rt_perror(CMD_SELECT, RT_MCAP, Z_NO_RESOURCE_TYPE,
3999bbec428eSgjelinek 			    B_TRUE);
4000bbec428eSgjelinek 			global_scope = B_TRUE;
40010209230bSgjelinek 		}
40020209230bSgjelinek 		if (res == Z_OK)
40030209230bSgjelinek 			bcopy(&old_mcaptab, &in_progress_mcaptab,
40040209230bSgjelinek 			    sizeof (struct zone_mcaptab));
40050209230bSgjelinek 		else
40060209230bSgjelinek 			bzero(&in_progress_mcaptab,
40070209230bSgjelinek 			    sizeof (in_progress_mcaptab));
40080209230bSgjelinek 		return;
4009cb8a054bSGlenn Faden 	case RT_ADMIN:
4010cb8a054bSGlenn Faden 		if ((err = fill_in_admintab(cmd, &old_admintab, B_FALSE))
4011cb8a054bSGlenn Faden 		    != Z_OK) {
4012cb8a054bSGlenn Faden 			z_cmd_rt_perror(CMD_SELECT, RT_ADMIN, err,
4013cb8a054bSGlenn Faden 			    B_TRUE);
4014cb8a054bSGlenn Faden 			global_scope = B_TRUE;
4015cb8a054bSGlenn Faden 		}
4016cb8a054bSGlenn Faden 		bcopy(&old_admintab, &in_progress_admintab,
4017cb8a054bSGlenn Faden 		    sizeof (struct zone_admintab));
4018cb8a054bSGlenn Faden 		return;
4019d2a70789SRichard Lowe 	case RT_SECFLAGS:
4020d2a70789SRichard Lowe 		if ((err = fill_in_secflagstab(cmd, &old_secflagstab, B_FALSE))
4021d2a70789SRichard Lowe 		    != Z_OK) {
4022d2a70789SRichard Lowe 			z_cmd_rt_perror(CMD_SELECT, RT_SECFLAGS, err,
4023d2a70789SRichard Lowe 			    B_TRUE);
4024d2a70789SRichard Lowe 			global_scope = B_TRUE;
4025d2a70789SRichard Lowe 		}
4026d2a70789SRichard Lowe 		bcopy(&old_secflagstab, &in_progress_secflagstab,
4027d2a70789SRichard Lowe 		    sizeof (struct zone_secflagstab));
4028d2a70789SRichard Lowe 		return;
40297c478bd9Sstevel@tonic-gate 	default:
4030bbec428eSgjelinek 		zone_perror(rt_to_str(type), Z_NO_RESOURCE_TYPE, B_TRUE);
4031bbec428eSgjelinek 		long_usage(CMD_SELECT, B_TRUE);
4032bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
40337c478bd9Sstevel@tonic-gate 		return;
40347c478bd9Sstevel@tonic-gate 	}
40357c478bd9Sstevel@tonic-gate }
40367c478bd9Sstevel@tonic-gate 
40377c478bd9Sstevel@tonic-gate /*
40387c478bd9Sstevel@tonic-gate  * Network "addresses" can be one of the following forms:
40397c478bd9Sstevel@tonic-gate  *	<IPv4 address>
40407c478bd9Sstevel@tonic-gate  *	<IPv4 address>/<prefix length>
40417c478bd9Sstevel@tonic-gate  *	<IPv6 address>/<prefix length>
40427c478bd9Sstevel@tonic-gate  *	<host name>
40437c478bd9Sstevel@tonic-gate  *	<host name>/<prefix length>
40447c478bd9Sstevel@tonic-gate  * In other words, the "/" followed by a prefix length is allowed but not
40457c478bd9Sstevel@tonic-gate  * required for IPv4 addresses and host names, and required for IPv6 addresses.
40467c478bd9Sstevel@tonic-gate  * If a prefix length is given, it must be in the allowable range: 0 to 32 for
40477c478bd9Sstevel@tonic-gate  * IPv4 addresses and host names, 0 to 128 for IPv6 addresses.
40487c478bd9Sstevel@tonic-gate  * Host names must start with an alpha-numeric character, and all subsequent
40497c478bd9Sstevel@tonic-gate  * characters must be either alpha-numeric or "-".
4050550b6e40SSowmini Varadhan  *
4051550b6e40SSowmini Varadhan  * In some cases, e.g., the nexthop for the defrouter, the context indicates
4052550b6e40SSowmini Varadhan  * that this is the IPV4_ABITS or IPV6_ABITS netmask, in which case we don't
4053550b6e40SSowmini Varadhan  * require the /<prefix length> (and should ignore it if provided).
40547c478bd9Sstevel@tonic-gate  */
40557c478bd9Sstevel@tonic-gate 
40567c478bd9Sstevel@tonic-gate static int
validate_net_address_syntax(char * address,boolean_t ishost)4057550b6e40SSowmini Varadhan validate_net_address_syntax(char *address, boolean_t ishost)
40587c478bd9Sstevel@tonic-gate {
40597c478bd9Sstevel@tonic-gate 	char *slashp, part1[MAXHOSTNAMELEN];
40607c478bd9Sstevel@tonic-gate 	struct in6_addr in6;
40617c478bd9Sstevel@tonic-gate 	struct in_addr in4;
40627c478bd9Sstevel@tonic-gate 	int prefixlen, i;
40637c478bd9Sstevel@tonic-gate 
40647c478bd9Sstevel@tonic-gate 	/*
40657c478bd9Sstevel@tonic-gate 	 * Copy the part before any '/' into part1 or copy the whole
40667c478bd9Sstevel@tonic-gate 	 * thing if there is no '/'.
40677c478bd9Sstevel@tonic-gate 	 */
40687c478bd9Sstevel@tonic-gate 	if ((slashp = strchr(address, '/')) != NULL) {
40697c478bd9Sstevel@tonic-gate 		*slashp = '\0';
40707c478bd9Sstevel@tonic-gate 		(void) strlcpy(part1, address, sizeof (part1));
40717c478bd9Sstevel@tonic-gate 		*slashp = '/';
40727c478bd9Sstevel@tonic-gate 		prefixlen = atoi(++slashp);
40737c478bd9Sstevel@tonic-gate 	} else {
40747c478bd9Sstevel@tonic-gate 		(void) strlcpy(part1, address, sizeof (part1));
40757c478bd9Sstevel@tonic-gate 	}
40767c478bd9Sstevel@tonic-gate 
4077550b6e40SSowmini Varadhan 	if (ishost && slashp != NULL) {
4078550b6e40SSowmini Varadhan 		zerr(gettext("Warning: prefix length in %s is not required and "
4079550b6e40SSowmini Varadhan 		    "will be ignored. The default host-prefix length "
4080550b6e40SSowmini Varadhan 		    "will be used"), address);
4081550b6e40SSowmini Varadhan 	}
4082550b6e40SSowmini Varadhan 
4083550b6e40SSowmini Varadhan 
40847c478bd9Sstevel@tonic-gate 	if (inet_pton(AF_INET6, part1, &in6) == 1) {
4085550b6e40SSowmini Varadhan 		if (ishost) {
4086550b6e40SSowmini Varadhan 			prefixlen = IPV6_ABITS;
4087550b6e40SSowmini Varadhan 		} else if (slashp == NULL) {
40887c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: IPv6 addresses "
40897c478bd9Sstevel@tonic-gate 			    "require /prefix-length suffix."), address);
40907c478bd9Sstevel@tonic-gate 			return (Z_ERR);
40917c478bd9Sstevel@tonic-gate 		}
40927c478bd9Sstevel@tonic-gate 		if (prefixlen < 0 || prefixlen > 128) {
40937c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: IPv6 address "
40947c478bd9Sstevel@tonic-gate 			    "prefix lengths must be 0 - 128."), address);
40957c478bd9Sstevel@tonic-gate 			return (Z_ERR);
40967c478bd9Sstevel@tonic-gate 		}
40977c478bd9Sstevel@tonic-gate 		return (Z_OK);
40987c478bd9Sstevel@tonic-gate 	}
40997c478bd9Sstevel@tonic-gate 
41007c478bd9Sstevel@tonic-gate 	/* At this point, any /prefix must be for IPv4. */
4101550b6e40SSowmini Varadhan 	if (ishost)
4102550b6e40SSowmini Varadhan 		prefixlen = IPV4_ABITS;
4103550b6e40SSowmini Varadhan 	else if (slashp != NULL) {
41047c478bd9Sstevel@tonic-gate 		if (prefixlen < 0 || prefixlen > 32) {
41057c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: IPv4 address "
41067c478bd9Sstevel@tonic-gate 			    "prefix lengths must be 0 - 32."), address);
41077c478bd9Sstevel@tonic-gate 			return (Z_ERR);
41087c478bd9Sstevel@tonic-gate 		}
41097c478bd9Sstevel@tonic-gate 	}
4110550b6e40SSowmini Varadhan 
41117c478bd9Sstevel@tonic-gate 	if (inet_pton(AF_INET, part1, &in4) == 1)
41127c478bd9Sstevel@tonic-gate 		return (Z_OK);
41137c478bd9Sstevel@tonic-gate 
41147c478bd9Sstevel@tonic-gate 	/* address may also be a host name */
41157c478bd9Sstevel@tonic-gate 	if (!isalnum(part1[0])) {
41167c478bd9Sstevel@tonic-gate 		zerr(gettext("%s: bogus host name or network address syntax"),
41177c478bd9Sstevel@tonic-gate 		    part1);
4118bbec428eSgjelinek 		saw_error = B_TRUE;
4119bbec428eSgjelinek 		usage(B_FALSE, HELP_NETADDR);
41207c478bd9Sstevel@tonic-gate 		return (Z_ERR);
41217c478bd9Sstevel@tonic-gate 	}
41227c478bd9Sstevel@tonic-gate 	for (i = 1; part1[i]; i++)
41237c478bd9Sstevel@tonic-gate 		if (!isalnum(part1[i]) && part1[i] != '-' && part1[i] != '.') {
41247c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: bogus host name or "
41257c478bd9Sstevel@tonic-gate 			    "network address syntax"), part1);
4126bbec428eSgjelinek 			saw_error = B_TRUE;
4127bbec428eSgjelinek 			usage(B_FALSE, HELP_NETADDR);
41287c478bd9Sstevel@tonic-gate 			return (Z_ERR);
41297c478bd9Sstevel@tonic-gate 		}
41307c478bd9Sstevel@tonic-gate 	return (Z_OK);
41317c478bd9Sstevel@tonic-gate }
41327c478bd9Sstevel@tonic-gate 
41337c478bd9Sstevel@tonic-gate static int
validate_net_physical_syntax(const char * ifname)4134c9f134eaSjv227347 validate_net_physical_syntax(const char *ifname)
41357c478bd9Sstevel@tonic-gate {
4136c9f134eaSjv227347 	ifspec_t ifnameprop;
4137c9f134eaSjv227347 	zone_iptype_t iptype;
4138c9f134eaSjv227347 
413937b210dcSjv227347 	if (zonecfg_get_iptype(handle, &iptype) != Z_OK) {
4140c9f134eaSjv227347 		zerr(gettext("zone configuration has an invalid or nonexistent "
4141c9f134eaSjv227347 		    "ip-type property"));
41427c478bd9Sstevel@tonic-gate 		return (Z_ERR);
41437c478bd9Sstevel@tonic-gate 	}
4144c9f134eaSjv227347 	switch (iptype) {
4145c9f134eaSjv227347 	case ZS_SHARED:
4146c9f134eaSjv227347 		if (ifparse_ifspec(ifname, &ifnameprop) == B_FALSE) {
4147c9f134eaSjv227347 			zerr(gettext("%s: invalid physical interface name"),
4148c9f134eaSjv227347 			    ifname);
4149c9f134eaSjv227347 			return (Z_ERR);
4150c9f134eaSjv227347 		}
4151c9f134eaSjv227347 		if (ifnameprop.ifsp_lunvalid) {
4152c9f134eaSjv227347 			zerr(gettext("%s: LUNs not allowed in physical "
4153c9f134eaSjv227347 			    "interface names"), ifname);
4154c9f134eaSjv227347 			return (Z_ERR);
4155c9f134eaSjv227347 		}
4156c9f134eaSjv227347 		break;
4157c9f134eaSjv227347 	case ZS_EXCLUSIVE:
4158c9f134eaSjv227347 		if (dladm_valid_linkname(ifname) == B_FALSE) {
4159c9f134eaSjv227347 			if (strchr(ifname, ':') != NULL)
4160c9f134eaSjv227347 				zerr(gettext("%s: physical interface name "
4161c9f134eaSjv227347 				    "required; logical interface name not "
4162c9f134eaSjv227347 				    "allowed"), ifname);
4163c9f134eaSjv227347 			else
4164c9f134eaSjv227347 				zerr(gettext("%s: invalid physical interface "
4165c9f134eaSjv227347 				    "name"), ifname);
4166c9f134eaSjv227347 			return (Z_ERR);
4167c9f134eaSjv227347 		}
4168c9f134eaSjv227347 		break;
4169c9f134eaSjv227347 	}
4170c9f134eaSjv227347 	return (Z_OK);
4171c9f134eaSjv227347 }
41727c478bd9Sstevel@tonic-gate 
41737c478bd9Sstevel@tonic-gate static boolean_t
valid_fs_type(const char * type)41747c478bd9Sstevel@tonic-gate valid_fs_type(const char *type)
41757c478bd9Sstevel@tonic-gate {
41767c478bd9Sstevel@tonic-gate 	/*
41777c478bd9Sstevel@tonic-gate 	 * Is this a valid path component?
41787c478bd9Sstevel@tonic-gate 	 */
41797c478bd9Sstevel@tonic-gate 	if (strlen(type) + 1 > MAXNAMELEN)
41807c478bd9Sstevel@tonic-gate 		return (B_FALSE);
41817c478bd9Sstevel@tonic-gate 	/*
41827c478bd9Sstevel@tonic-gate 	 * Make sure a bad value for "type" doesn't make
41837c478bd9Sstevel@tonic-gate 	 * /usr/lib/fs/<type>/mount turn into something else.
41847c478bd9Sstevel@tonic-gate 	 */
41857c478bd9Sstevel@tonic-gate 	if (strchr(type, '/') != NULL || type[0] == '\0' ||
41867c478bd9Sstevel@tonic-gate 	    strcmp(type, ".") == 0 || strcmp(type, "..") == 0)
41877c478bd9Sstevel@tonic-gate 		return (B_FALSE);
41887c478bd9Sstevel@tonic-gate 	/*
4189*bbf21555SRichard Lowe 	 * More detailed verification happens later by zoneadm(8).
41907c478bd9Sstevel@tonic-gate 	 */
41917c478bd9Sstevel@tonic-gate 	return (B_TRUE);
41927c478bd9Sstevel@tonic-gate }
41937c478bd9Sstevel@tonic-gate 
4194f4b3ec61Sdh155122 static boolean_t
allow_exclusive()4195f4b3ec61Sdh155122 allow_exclusive()
4196f4b3ec61Sdh155122 {
4197f4b3ec61Sdh155122 	brand_handle_t	bh;
4198f4b3ec61Sdh155122 	char		brand[MAXNAMELEN];
4199f4b3ec61Sdh155122 	boolean_t	ret;
4200f4b3ec61Sdh155122 
4201f4b3ec61Sdh155122 	if (zonecfg_get_brand(handle, brand, sizeof (brand)) != Z_OK) {
4202f4b3ec61Sdh155122 		zerr("%s: %s\n", zone, gettext("could not get zone brand"));
4203f4b3ec61Sdh155122 		return (B_FALSE);
4204f4b3ec61Sdh155122 	}
4205f4b3ec61Sdh155122 	if ((bh = brand_open(brand)) == NULL) {
4206f4b3ec61Sdh155122 		zerr("%s: %s\n", zone, gettext("unknown brand."));
4207f4b3ec61Sdh155122 		return (B_FALSE);
4208f4b3ec61Sdh155122 	}
4209f4b3ec61Sdh155122 	ret = brand_allow_exclusive_ip(bh);
4210f4b3ec61Sdh155122 	brand_close(bh);
4211f4b3ec61Sdh155122 	if (!ret)
4212f4b3ec61Sdh155122 		zerr(gettext("%s cannot be '%s' when %s is '%s'."),
4213f4b3ec61Sdh155122 		    pt_to_str(PT_IPTYPE), "exclusive",
4214f4b3ec61Sdh155122 		    pt_to_str(PT_BRAND), brand);
4215f4b3ec61Sdh155122 	return (ret);
4216f4b3ec61Sdh155122 }
4217f4b3ec61Sdh155122 
42180209230bSgjelinek static void
set_aliased_rctl(char * alias,int prop_type,char * s)42190209230bSgjelinek set_aliased_rctl(char *alias, int prop_type, char *s)
42200209230bSgjelinek {
42210209230bSgjelinek 	uint64_t limit;
42220209230bSgjelinek 	int err;
42230209230bSgjelinek 	char tmp[128];
42240209230bSgjelinek 
42250209230bSgjelinek 	if (global_zone && strcmp(alias, ALIAS_SHARES) != 0)
42260209230bSgjelinek 		zerr(gettext("WARNING: Setting a global zone resource "
42270209230bSgjelinek 		    "control too low could deny\nservice "
42280209230bSgjelinek 		    "to even the root user; "
42290209230bSgjelinek 		    "this could render the system impossible\n"
42300209230bSgjelinek 		    "to administer.  Please use caution."));
42310209230bSgjelinek 
42320209230bSgjelinek 	/* convert memory based properties */
42330209230bSgjelinek 	if (prop_type == PT_MAXSHMMEM) {
42340209230bSgjelinek 		if (!zonecfg_valid_memlimit(s, &limit)) {
42350209230bSgjelinek 			zerr(gettext("A non-negative number with a required "
42360209230bSgjelinek 			    "scale suffix (K, M, G or T) was expected\nhere."));
4237bbec428eSgjelinek 			saw_error = B_TRUE;
42380209230bSgjelinek 			return;
42390209230bSgjelinek 		}
42400209230bSgjelinek 
42410209230bSgjelinek 		(void) snprintf(tmp, sizeof (tmp), "%llu", limit);
42420209230bSgjelinek 		s = tmp;
42430209230bSgjelinek 	}
42440209230bSgjelinek 
42450209230bSgjelinek 	if (!zonecfg_aliased_rctl_ok(handle, alias)) {
4246bbec428eSgjelinek 		zone_perror(pt_to_str(prop_type), Z_ALIAS_DISALLOW, B_FALSE);
4247bbec428eSgjelinek 		saw_error = B_TRUE;
42480209230bSgjelinek 	} else if (!zonecfg_valid_alias_limit(alias, s, &limit)) {
42490209230bSgjelinek 		zerr(gettext("%s property is out of range."),
42500209230bSgjelinek 		    pt_to_str(prop_type));
4251bbec428eSgjelinek 		saw_error = B_TRUE;
42520209230bSgjelinek 	} else if ((err = zonecfg_set_aliased_rctl(handle, alias, limit))
42530209230bSgjelinek 	    != Z_OK) {
4254bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
4255bbec428eSgjelinek 		saw_error = B_TRUE;
42560209230bSgjelinek 	} else {
4257bbec428eSgjelinek 		need_to_commit = B_TRUE;
42580209230bSgjelinek 	}
42590209230bSgjelinek }
42600209230bSgjelinek 
4261550b6e40SSowmini Varadhan static void
set_in_progress_nwiftab_address(char * prop_id,int prop_type)4262550b6e40SSowmini Varadhan set_in_progress_nwiftab_address(char *prop_id, int prop_type)
4263550b6e40SSowmini Varadhan {
4264550b6e40SSowmini Varadhan 	if (prop_type == PT_ADDRESS) {
4265550b6e40SSowmini Varadhan 		(void) strlcpy(in_progress_nwiftab.zone_nwif_address, prop_id,
4266550b6e40SSowmini Varadhan 		    sizeof (in_progress_nwiftab.zone_nwif_address));
4267550b6e40SSowmini Varadhan 	} else {
4268550b6e40SSowmini Varadhan 		assert(prop_type == PT_ALLOWED_ADDRESS);
4269550b6e40SSowmini Varadhan 		(void) strlcpy(in_progress_nwiftab.zone_nwif_allowed_address,
4270550b6e40SSowmini Varadhan 		    prop_id,
4271550b6e40SSowmini Varadhan 		    sizeof (in_progress_nwiftab.zone_nwif_allowed_address));
4272550b6e40SSowmini Varadhan 	}
4273550b6e40SSowmini Varadhan }
4274550b6e40SSowmini Varadhan 
42757c478bd9Sstevel@tonic-gate void
set_func(cmd_t * cmd)42767c478bd9Sstevel@tonic-gate set_func(cmd_t *cmd)
42777c478bd9Sstevel@tonic-gate {
42787c478bd9Sstevel@tonic-gate 	char *prop_id;
4279555afedfScarlsonj 	int arg, err, res_type, prop_type;
42807c478bd9Sstevel@tonic-gate 	property_value_ptr_t pp;
42817c478bd9Sstevel@tonic-gate 	boolean_t autoboot;
4282f4b3ec61Sdh155122 	zone_iptype_t iptype;
4283bbec428eSgjelinek 	boolean_t force_set = B_FALSE;
42840209230bSgjelinek 	size_t physmem_size = sizeof (in_progress_mcaptab.zone_physmem_cap);
42850209230bSgjelinek 	uint64_t mem_cap, mem_limit;
4286c97ad5cdSakolb 	float cap;
4287c97ad5cdSakolb 	char *unitp;
42880209230bSgjelinek 	struct zone_psettab tmp_psettab;
4289bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
42907c478bd9Sstevel@tonic-gate 
42917c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_SET))
42927c478bd9Sstevel@tonic-gate 		return;
42937c478bd9Sstevel@tonic-gate 
42947c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
42957c478bd9Sstevel@tonic-gate 
4296555afedfScarlsonj 	optind = opterr = 0;
4297555afedfScarlsonj 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "F")) != EOF) {
4298555afedfScarlsonj 		switch (arg) {
4299555afedfScarlsonj 		case 'F':
4300bbec428eSgjelinek 			force_set = B_TRUE;
4301555afedfScarlsonj 			break;
4302555afedfScarlsonj 		default:
4303555afedfScarlsonj 			if (optopt == '?')
4304555afedfScarlsonj 				longer_usage(CMD_SET);
4305555afedfScarlsonj 			else
4306555afedfScarlsonj 				short_usage(CMD_SET);
4307bbec428eSgjelinek 			arg_err = B_TRUE;
43087ec75eb8Sgjelinek 			break;
43097ec75eb8Sgjelinek 		}
43107ec75eb8Sgjelinek 	}
43117ec75eb8Sgjelinek 	if (arg_err)
4312555afedfScarlsonj 		return;
4313555afedfScarlsonj 
43147c478bd9Sstevel@tonic-gate 	prop_type = cmd->cmd_prop_name[0];
43157c478bd9Sstevel@tonic-gate 	if (global_scope) {
43160209230bSgjelinek 		if (gz_invalid_property(prop_type)) {
43170209230bSgjelinek 			zerr(gettext("%s is not a valid property for the "
43180209230bSgjelinek 			    "global zone."), pt_to_str(prop_type));
4319bbec428eSgjelinek 			saw_error = B_TRUE;
43200209230bSgjelinek 			return;
43210209230bSgjelinek 		}
43220209230bSgjelinek 
4323087719fdSdp 		if (prop_type == PT_ZONENAME) {
4324087719fdSdp 			res_type = RT_ZONENAME;
4325087719fdSdp 		} else if (prop_type == PT_ZONEPATH) {
43267c478bd9Sstevel@tonic-gate 			res_type = RT_ZONEPATH;
43277c478bd9Sstevel@tonic-gate 		} else if (prop_type == PT_AUTOBOOT) {
43287c478bd9Sstevel@tonic-gate 			res_type = RT_AUTOBOOT;
43299acbbeafSnn35248 		} else if (prop_type == PT_BRAND) {
43309acbbeafSnn35248 			res_type = RT_BRAND;
43317c478bd9Sstevel@tonic-gate 		} else if (prop_type == PT_POOL) {
43327c478bd9Sstevel@tonic-gate 			res_type = RT_POOL;
4333ffbafc53Scomay 		} else if (prop_type == PT_LIMITPRIV) {
4334ffbafc53Scomay 			res_type = RT_LIMITPRIV;
43353f2f09c1Sdp 		} else if (prop_type == PT_BOOTARGS) {
43363f2f09c1Sdp 			res_type = RT_BOOTARGS;
43370209230bSgjelinek 		} else if (prop_type == PT_SCHED) {
43380209230bSgjelinek 			res_type = RT_SCHED;
4339f4b3ec61Sdh155122 		} else if (prop_type == PT_IPTYPE) {
4340f4b3ec61Sdh155122 			res_type = RT_IPTYPE;
43410209230bSgjelinek 		} else if (prop_type == PT_MAXLWPS) {
43420209230bSgjelinek 			res_type = RT_MAXLWPS;
4343ff19e029SMenno Lageman 		} else if (prop_type == PT_MAXPROCS) {
4344ff19e029SMenno Lageman 			res_type = RT_MAXPROCS;
43450209230bSgjelinek 		} else if (prop_type == PT_MAXSHMMEM) {
43460209230bSgjelinek 			res_type = RT_MAXSHMMEM;
43470209230bSgjelinek 		} else if (prop_type == PT_MAXSHMIDS) {
43480209230bSgjelinek 			res_type = RT_MAXSHMIDS;
43490209230bSgjelinek 		} else if (prop_type == PT_MAXMSGIDS) {
43500209230bSgjelinek 			res_type = RT_MAXMSGIDS;
43510209230bSgjelinek 		} else if (prop_type == PT_MAXSEMIDS) {
43520209230bSgjelinek 			res_type = RT_MAXSEMIDS;
43530209230bSgjelinek 		} else if (prop_type == PT_SHARES) {
43540209230bSgjelinek 			res_type = RT_SHARES;
43555679c89fSjv227347 		} else if (prop_type == PT_HOSTID) {
43565679c89fSjv227347 			res_type = RT_HOSTID;
43570fbb751dSJohn Levon 		} else if (prop_type == PT_FS_ALLOWED) {
43580fbb751dSJohn Levon 			res_type = RT_FS_ALLOWED;
43597c478bd9Sstevel@tonic-gate 		} else {
43607c478bd9Sstevel@tonic-gate 			zerr(gettext("Cannot set a resource-specific property "
43617c478bd9Sstevel@tonic-gate 			    "from the global scope."));
4362bbec428eSgjelinek 			saw_error = B_TRUE;
43637c478bd9Sstevel@tonic-gate 			return;
43647c478bd9Sstevel@tonic-gate 		}
43657c478bd9Sstevel@tonic-gate 	} else {
43667c478bd9Sstevel@tonic-gate 		res_type = resource_scope;
43677c478bd9Sstevel@tonic-gate 	}
43687c478bd9Sstevel@tonic-gate 
4369555afedfScarlsonj 	if (force_set) {
4370555afedfScarlsonj 		if (res_type != RT_ZONEPATH) {
4371555afedfScarlsonj 			zerr(gettext("Only zonepath setting can be forced."));
4372bbec428eSgjelinek 			saw_error = B_TRUE;
4373555afedfScarlsonj 			return;
4374555afedfScarlsonj 		}
4375555afedfScarlsonj 		if (!zonecfg_in_alt_root()) {
4376555afedfScarlsonj 			zerr(gettext("Zonepath is changeable only in an "
4377555afedfScarlsonj 			    "alternate root."));
4378bbec428eSgjelinek 			saw_error = B_TRUE;
4379555afedfScarlsonj 			return;
4380555afedfScarlsonj 		}
4381555afedfScarlsonj 	}
4382555afedfScarlsonj 
43837c478bd9Sstevel@tonic-gate 	pp = cmd->cmd_property_ptr[0];
43847c478bd9Sstevel@tonic-gate 	/*
43857c478bd9Sstevel@tonic-gate 	 * A nasty expression but not that complicated:
43867c478bd9Sstevel@tonic-gate 	 * 1. fs options are simple or list (tested below)
43877c478bd9Sstevel@tonic-gate 	 * 2. rctl value's are complex or list (tested below)
43887c478bd9Sstevel@tonic-gate 	 * Anything else should be simple.
43897c478bd9Sstevel@tonic-gate 	 */
43907c478bd9Sstevel@tonic-gate 	if (!(res_type == RT_FS && prop_type == PT_OPTIONS) &&
43917c478bd9Sstevel@tonic-gate 	    !(res_type == RT_RCTL && prop_type == PT_VALUE) &&
43927c478bd9Sstevel@tonic-gate 	    (pp->pv_type != PROP_VAL_SIMPLE ||
43937c478bd9Sstevel@tonic-gate 	    (prop_id = pp->pv_simple) == NULL)) {
43947c478bd9Sstevel@tonic-gate 		zerr(gettext("A %s value was expected here."),
43957c478bd9Sstevel@tonic-gate 		    pvt_to_str(PROP_VAL_SIMPLE));
4396bbec428eSgjelinek 		saw_error = B_TRUE;
43977c478bd9Sstevel@tonic-gate 		return;
43987c478bd9Sstevel@tonic-gate 	}
43997c478bd9Sstevel@tonic-gate 	if (prop_type == PT_UNKNOWN) {
4400bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
44017c478bd9Sstevel@tonic-gate 		return;
44027c478bd9Sstevel@tonic-gate 	}
44037c478bd9Sstevel@tonic-gate 
4404087719fdSdp 	/*
4405087719fdSdp 	 * Special case: the user can change the zone name prior to 'create';
4406087719fdSdp 	 * if the zone already exists, we fall through letting initialize()
4407087719fdSdp 	 * and the rest of the logic run.
4408087719fdSdp 	 */
4409bbec428eSgjelinek 	if (res_type == RT_ZONENAME && got_handle == B_FALSE &&
4410087719fdSdp 	    !state_atleast(ZONE_STATE_CONFIGURED)) {
4411fb03efaaSdp 		if ((err = zonecfg_validate_zonename(prop_id)) != Z_OK) {
4412bbec428eSgjelinek 			zone_perror(prop_id, err, B_TRUE);
4413bbec428eSgjelinek 			usage(B_FALSE, HELP_SYNTAX);
4414fb03efaaSdp 			return;
4415fb03efaaSdp 		}
4416087719fdSdp 		(void) strlcpy(zone, prop_id, sizeof (zone));
4417087719fdSdp 		return;
4418087719fdSdp 	}
4419087719fdSdp 
4420bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
44217c478bd9Sstevel@tonic-gate 		return;
44227c478bd9Sstevel@tonic-gate 
44237c478bd9Sstevel@tonic-gate 	switch (res_type) {
4424087719fdSdp 	case RT_ZONENAME:
4425087719fdSdp 		if ((err = zonecfg_set_name(handle, prop_id)) != Z_OK) {
4426087719fdSdp 			/*
4427087719fdSdp 			 * Use prop_id instead of 'zone' here, since we're
4428087719fdSdp 			 * reporting a problem about the *new* zonename.
4429087719fdSdp 			 */
4430bbec428eSgjelinek 			zone_perror(prop_id, err, B_TRUE);
4431bbec428eSgjelinek 			usage(B_FALSE, HELP_SYNTAX);
4432087719fdSdp 		} else {
4433bbec428eSgjelinek 			need_to_commit = B_TRUE;
4434087719fdSdp 			(void) strlcpy(zone, prop_id, sizeof (zone));
4435087719fdSdp 		}
4436087719fdSdp 		return;
44377c478bd9Sstevel@tonic-gate 	case RT_ZONEPATH:
4438555afedfScarlsonj 		if (!force_set && state_atleast(ZONE_STATE_INSTALLED)) {
44397c478bd9Sstevel@tonic-gate 			zerr(gettext("Zone %s already installed; %s %s not "
44407c478bd9Sstevel@tonic-gate 			    "allowed."), zone, cmd_to_str(CMD_SET),
44417c478bd9Sstevel@tonic-gate 			    rt_to_str(RT_ZONEPATH));
44427c478bd9Sstevel@tonic-gate 			return;
44437c478bd9Sstevel@tonic-gate 		}
44447c478bd9Sstevel@tonic-gate 		if (validate_zonepath_syntax(prop_id) != Z_OK) {
4445bbec428eSgjelinek 			saw_error = B_TRUE;
44467c478bd9Sstevel@tonic-gate 			return;
44477c478bd9Sstevel@tonic-gate 		}
44487c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_set_zonepath(handle, prop_id)) != Z_OK)
4449bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
44507c478bd9Sstevel@tonic-gate 		else
4451bbec428eSgjelinek 			need_to_commit = B_TRUE;
44527c478bd9Sstevel@tonic-gate 		return;
44539acbbeafSnn35248 	case RT_BRAND:
44549acbbeafSnn35248 		if (state_atleast(ZONE_STATE_INSTALLED)) {
44559acbbeafSnn35248 			zerr(gettext("Zone %s already installed; %s %s not "
44569acbbeafSnn35248 			    "allowed."), zone, cmd_to_str(CMD_SET),
44579acbbeafSnn35248 			    rt_to_str(RT_BRAND));
44589acbbeafSnn35248 			return;
44599acbbeafSnn35248 		}
44609acbbeafSnn35248 		if ((err = zonecfg_set_brand(handle, prop_id)) != Z_OK)
4461bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
44629acbbeafSnn35248 		else
4463bbec428eSgjelinek 			need_to_commit = B_TRUE;
44649acbbeafSnn35248 		return;
44657c478bd9Sstevel@tonic-gate 	case RT_AUTOBOOT:
44667c478bd9Sstevel@tonic-gate 		if (strcmp(prop_id, "true") == 0) {
44677c478bd9Sstevel@tonic-gate 			autoboot = B_TRUE;
44687c478bd9Sstevel@tonic-gate 		} else if (strcmp(prop_id, "false") == 0) {
44697c478bd9Sstevel@tonic-gate 			autoboot = B_FALSE;
44707c478bd9Sstevel@tonic-gate 		} else {
44717c478bd9Sstevel@tonic-gate 			zerr(gettext("%s value must be '%s' or '%s'."),
44727c478bd9Sstevel@tonic-gate 			    pt_to_str(PT_AUTOBOOT), "true", "false");
4473bbec428eSgjelinek 			saw_error = B_TRUE;
44747c478bd9Sstevel@tonic-gate 			return;
44757c478bd9Sstevel@tonic-gate 		}
44767c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_set_autoboot(handle, autoboot)) != Z_OK)
4477bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
44787c478bd9Sstevel@tonic-gate 		else
4479bbec428eSgjelinek 			need_to_commit = B_TRUE;
44807c478bd9Sstevel@tonic-gate 		return;
44817c478bd9Sstevel@tonic-gate 	case RT_POOL:
44820209230bSgjelinek 		/* don't allow use of the reserved temporary pool names */
44830209230bSgjelinek 		if (strncmp("SUNW", prop_id, 4) == 0) {
44840209230bSgjelinek 			zerr(gettext("pool names starting with SUNW are "
44850209230bSgjelinek 			    "reserved."));
4486bbec428eSgjelinek 			saw_error = B_TRUE;
44870209230bSgjelinek 			return;
44880209230bSgjelinek 		}
44890209230bSgjelinek 
44900209230bSgjelinek 		/* can't set pool if dedicated-cpu exists */
44910209230bSgjelinek 		if (zonecfg_lookup_pset(handle, &tmp_psettab) == Z_OK) {
44920209230bSgjelinek 			zerr(gettext("The %s resource already exists.  "
44930209230bSgjelinek 			    "A persistent pool is incompatible\nwith the %s "
44940209230bSgjelinek 			    "resource."), rt_to_str(RT_DCPU),
44950209230bSgjelinek 			    rt_to_str(RT_DCPU));
4496bbec428eSgjelinek 			saw_error = B_TRUE;
44970209230bSgjelinek 			return;
44980209230bSgjelinek 		}
44990209230bSgjelinek 
45007c478bd9Sstevel@tonic-gate 		if ((err = zonecfg_set_pool(handle, prop_id)) != Z_OK)
4501bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
45027c478bd9Sstevel@tonic-gate 		else
4503bbec428eSgjelinek 			need_to_commit = B_TRUE;
45047c478bd9Sstevel@tonic-gate 		return;
4505ffbafc53Scomay 	case RT_LIMITPRIV:
4506ffbafc53Scomay 		if ((err = zonecfg_set_limitpriv(handle, prop_id)) != Z_OK)
4507bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
4508ffbafc53Scomay 		else
4509bbec428eSgjelinek 			need_to_commit = B_TRUE;
4510ffbafc53Scomay 		return;
45113f2f09c1Sdp 	case RT_BOOTARGS:
45123f2f09c1Sdp 		if ((err = zonecfg_set_bootargs(handle, prop_id)) != Z_OK)
4513bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
45143f2f09c1Sdp 		else
4515bbec428eSgjelinek 			need_to_commit = B_TRUE;
45163f2f09c1Sdp 		return;
45170209230bSgjelinek 	case RT_SCHED:
45180209230bSgjelinek 		if ((err = zonecfg_set_sched(handle, prop_id)) != Z_OK)
4519bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
45200209230bSgjelinek 		else
4521bbec428eSgjelinek 			need_to_commit = B_TRUE;
45220209230bSgjelinek 		return;
4523f4b3ec61Sdh155122 	case RT_IPTYPE:
4524f4b3ec61Sdh155122 		if (strcmp(prop_id, "shared") == 0) {
4525f4b3ec61Sdh155122 			iptype = ZS_SHARED;
4526f4b3ec61Sdh155122 		} else if (strcmp(prop_id, "exclusive") == 0) {
4527f4b3ec61Sdh155122 			iptype = ZS_EXCLUSIVE;
4528f4b3ec61Sdh155122 		} else {
4529f4b3ec61Sdh155122 			zerr(gettext("%s value must be '%s' or '%s'."),
4530f4b3ec61Sdh155122 			    pt_to_str(PT_IPTYPE), "shared", "exclusive");
4531bbec428eSgjelinek 			saw_error = B_TRUE;
4532f4b3ec61Sdh155122 			return;
4533f4b3ec61Sdh155122 		}
4534f4b3ec61Sdh155122 		if (iptype == ZS_EXCLUSIVE && !allow_exclusive()) {
4535bbec428eSgjelinek 			saw_error = B_TRUE;
4536f4b3ec61Sdh155122 			return;
4537f4b3ec61Sdh155122 		}
4538f4b3ec61Sdh155122 		if ((err = zonecfg_set_iptype(handle, iptype)) != Z_OK)
4539bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
4540f4b3ec61Sdh155122 		else
4541bbec428eSgjelinek 			need_to_commit = B_TRUE;
4542f4b3ec61Sdh155122 		return;
45430209230bSgjelinek 	case RT_MAXLWPS:
45440209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXLWPS, prop_type, prop_id);
45450209230bSgjelinek 		return;
4546ff19e029SMenno Lageman 	case RT_MAXPROCS:
4547ff19e029SMenno Lageman 		set_aliased_rctl(ALIAS_MAXPROCS, prop_type, prop_id);
4548ff19e029SMenno Lageman 		return;
45490209230bSgjelinek 	case RT_MAXSHMMEM:
45500209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXSHMMEM, prop_type, prop_id);
45510209230bSgjelinek 		return;
45520209230bSgjelinek 	case RT_MAXSHMIDS:
45530209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXSHMIDS, prop_type, prop_id);
45540209230bSgjelinek 		return;
45550209230bSgjelinek 	case RT_MAXMSGIDS:
45560209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXMSGIDS, prop_type, prop_id);
45570209230bSgjelinek 		return;
45580209230bSgjelinek 	case RT_MAXSEMIDS:
45590209230bSgjelinek 		set_aliased_rctl(ALIAS_MAXSEMIDS, prop_type, prop_id);
45600209230bSgjelinek 		return;
45610209230bSgjelinek 	case RT_SHARES:
45620209230bSgjelinek 		set_aliased_rctl(ALIAS_SHARES, prop_type, prop_id);
45630209230bSgjelinek 		return;
45645679c89fSjv227347 	case RT_HOSTID:
45655679c89fSjv227347 		if ((err = zonecfg_set_hostid(handle, prop_id)) != Z_OK) {
45665679c89fSjv227347 			if (err == Z_TOO_BIG) {
45675679c89fSjv227347 				zerr(gettext("hostid string is too large: %s"),
45685679c89fSjv227347 				    prop_id);
45695679c89fSjv227347 				saw_error = B_TRUE;
45705679c89fSjv227347 			} else {
45715679c89fSjv227347 				zone_perror(pt_to_str(prop_type), err, B_TRUE);
45725679c89fSjv227347 			}
45735679c89fSjv227347 			return;
45745679c89fSjv227347 		}
45755679c89fSjv227347 		need_to_commit = B_TRUE;
45765679c89fSjv227347 		return;
45770fbb751dSJohn Levon 	case RT_FS_ALLOWED:
45780fbb751dSJohn Levon 		if ((err = zonecfg_set_fs_allowed(handle, prop_id)) != Z_OK)
45790fbb751dSJohn Levon 			zone_perror(zone, err, B_TRUE);
45800fbb751dSJohn Levon 		else
45810fbb751dSJohn Levon 			need_to_commit = B_TRUE;
45820fbb751dSJohn Levon 		return;
45837c478bd9Sstevel@tonic-gate 	case RT_FS:
45847c478bd9Sstevel@tonic-gate 		switch (prop_type) {
45857c478bd9Sstevel@tonic-gate 		case PT_DIR:
45867c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_fstab.zone_fs_dir, prop_id,
45877c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_fstab.zone_fs_dir));
45887c478bd9Sstevel@tonic-gate 			return;
45897c478bd9Sstevel@tonic-gate 		case PT_SPECIAL:
45907c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_fstab.zone_fs_special,
45917c478bd9Sstevel@tonic-gate 			    prop_id,
45927c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_fstab.zone_fs_special));
45937c478bd9Sstevel@tonic-gate 			return;
45947c478bd9Sstevel@tonic-gate 		case PT_RAW:
45957c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_fstab.zone_fs_raw,
45967c478bd9Sstevel@tonic-gate 			    prop_id, sizeof (in_progress_fstab.zone_fs_raw));
45977c478bd9Sstevel@tonic-gate 			return;
45987c478bd9Sstevel@tonic-gate 		case PT_TYPE:
45997c478bd9Sstevel@tonic-gate 			if (!valid_fs_type(prop_id)) {
46007c478bd9Sstevel@tonic-gate 				zerr(gettext("\"%s\" is not a valid %s."),
46017c478bd9Sstevel@tonic-gate 				    prop_id, pt_to_str(PT_TYPE));
4602bbec428eSgjelinek 				saw_error = B_TRUE;
46037c478bd9Sstevel@tonic-gate 				return;
46047c478bd9Sstevel@tonic-gate 			}
46057c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_fstab.zone_fs_type, prop_id,
46067c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_fstab.zone_fs_type));
46077c478bd9Sstevel@tonic-gate 			return;
46087c478bd9Sstevel@tonic-gate 		case PT_OPTIONS:
46097c478bd9Sstevel@tonic-gate 			if (pp->pv_type != PROP_VAL_SIMPLE &&
46107c478bd9Sstevel@tonic-gate 			    pp->pv_type != PROP_VAL_LIST) {
46117c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s or %s value was expected "
46127c478bd9Sstevel@tonic-gate 				    "here."), pvt_to_str(PROP_VAL_SIMPLE),
46137c478bd9Sstevel@tonic-gate 				    pvt_to_str(PROP_VAL_LIST));
4614bbec428eSgjelinek 				saw_error = B_TRUE;
46157c478bd9Sstevel@tonic-gate 				return;
46167c478bd9Sstevel@tonic-gate 			}
46177c478bd9Sstevel@tonic-gate 			zonecfg_free_fs_option_list(
46187c478bd9Sstevel@tonic-gate 			    in_progress_fstab.zone_fs_options);
46197c478bd9Sstevel@tonic-gate 			in_progress_fstab.zone_fs_options = NULL;
46207c478bd9Sstevel@tonic-gate 			if (!(pp->pv_type == PROP_VAL_LIST &&
46217c478bd9Sstevel@tonic-gate 			    pp->pv_list == NULL))
46227c478bd9Sstevel@tonic-gate 				add_property(cmd);
46237c478bd9Sstevel@tonic-gate 			return;
46247c478bd9Sstevel@tonic-gate 		default:
46257c478bd9Sstevel@tonic-gate 			break;
46267c478bd9Sstevel@tonic-gate 		}
4627bbec428eSgjelinek 		zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, B_TRUE);
4628bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
4629bbec428eSgjelinek 		usage(B_FALSE, HELP_PROPS);
46307c478bd9Sstevel@tonic-gate 		return;
46317c478bd9Sstevel@tonic-gate 	case RT_NET:
46327c478bd9Sstevel@tonic-gate 		switch (prop_type) {
46337c478bd9Sstevel@tonic-gate 		case PT_ADDRESS:
4634550b6e40SSowmini Varadhan 		case PT_ALLOWED_ADDRESS:
4635550b6e40SSowmini Varadhan 			if (validate_net_address_syntax(prop_id, B_FALSE)
4636550b6e40SSowmini Varadhan 			    != Z_OK) {
4637bbec428eSgjelinek 				saw_error = B_TRUE;
46387c478bd9Sstevel@tonic-gate 				return;
46397c478bd9Sstevel@tonic-gate 			}
4640550b6e40SSowmini Varadhan 			set_in_progress_nwiftab_address(prop_id, prop_type);
46417c478bd9Sstevel@tonic-gate 			break;
46427c478bd9Sstevel@tonic-gate 		case PT_PHYSICAL:
46437c478bd9Sstevel@tonic-gate 			if (validate_net_physical_syntax(prop_id) != Z_OK) {
4644bbec428eSgjelinek 				saw_error = B_TRUE;
46457c478bd9Sstevel@tonic-gate 				return;
46467c478bd9Sstevel@tonic-gate 			}
46477c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_nwiftab.zone_nwif_physical,
46487c478bd9Sstevel@tonic-gate 			    prop_id,
46497c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_nwiftab.zone_nwif_physical));
46507c478bd9Sstevel@tonic-gate 			break;
4651de860bd9Sgfaden 		case PT_DEFROUTER:
4652550b6e40SSowmini Varadhan 			if (validate_net_address_syntax(prop_id, B_TRUE)
4653550b6e40SSowmini Varadhan 			    != Z_OK) {
4654bbec428eSgjelinek 				saw_error = B_TRUE;
4655de860bd9Sgfaden 				return;
4656de860bd9Sgfaden 			}
4657de860bd9Sgfaden 			(void) strlcpy(in_progress_nwiftab.zone_nwif_defrouter,
4658de860bd9Sgfaden 			    prop_id,
4659de860bd9Sgfaden 			    sizeof (in_progress_nwiftab.zone_nwif_defrouter));
4660de860bd9Sgfaden 			break;
46617c478bd9Sstevel@tonic-gate 		default:
46627c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4663bbec428eSgjelinek 			    B_TRUE);
4664bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4665bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
46667c478bd9Sstevel@tonic-gate 			return;
46677c478bd9Sstevel@tonic-gate 		}
46687c478bd9Sstevel@tonic-gate 		return;
46697c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
46707c478bd9Sstevel@tonic-gate 		switch (prop_type) {
46717c478bd9Sstevel@tonic-gate 		case PT_MATCH:
46727c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_devtab.zone_dev_match,
46737c478bd9Sstevel@tonic-gate 			    prop_id,
46747c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_devtab.zone_dev_match));
46757c478bd9Sstevel@tonic-gate 			break;
46767c478bd9Sstevel@tonic-gate 		default:
46777c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4678bbec428eSgjelinek 			    B_TRUE);
4679bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4680bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
46817c478bd9Sstevel@tonic-gate 			return;
46827c478bd9Sstevel@tonic-gate 		}
46837c478bd9Sstevel@tonic-gate 		return;
46847c478bd9Sstevel@tonic-gate 	case RT_RCTL:
46857c478bd9Sstevel@tonic-gate 		switch (prop_type) {
46867c478bd9Sstevel@tonic-gate 		case PT_NAME:
46877c478bd9Sstevel@tonic-gate 			if (!zonecfg_valid_rctlname(prop_id)) {
46887c478bd9Sstevel@tonic-gate 				zerr(gettext("'%s' is not a valid zone %s "
46897c478bd9Sstevel@tonic-gate 				    "name."), prop_id, rt_to_str(RT_RCTL));
46907c478bd9Sstevel@tonic-gate 				return;
46917c478bd9Sstevel@tonic-gate 			}
46927c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_rctltab.zone_rctl_name,
46937c478bd9Sstevel@tonic-gate 			    prop_id,
46947c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_rctltab.zone_rctl_name));
46957c478bd9Sstevel@tonic-gate 			break;
46967c478bd9Sstevel@tonic-gate 		case PT_VALUE:
46977c478bd9Sstevel@tonic-gate 			if (pp->pv_type != PROP_VAL_COMPLEX &&
46987c478bd9Sstevel@tonic-gate 			    pp->pv_type != PROP_VAL_LIST) {
46997c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s or %s value was expected "
47007c478bd9Sstevel@tonic-gate 				    "here."), pvt_to_str(PROP_VAL_COMPLEX),
47017c478bd9Sstevel@tonic-gate 				    pvt_to_str(PROP_VAL_LIST));
4702bbec428eSgjelinek 				saw_error = B_TRUE;
47037c478bd9Sstevel@tonic-gate 				return;
47047c478bd9Sstevel@tonic-gate 			}
47057c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(
47067c478bd9Sstevel@tonic-gate 			    in_progress_rctltab.zone_rctl_valptr);
47077c478bd9Sstevel@tonic-gate 			in_progress_rctltab.zone_rctl_valptr = NULL;
47087c478bd9Sstevel@tonic-gate 			if (!(pp->pv_type == PROP_VAL_LIST &&
47097c478bd9Sstevel@tonic-gate 			    pp->pv_list == NULL))
47107c478bd9Sstevel@tonic-gate 				add_property(cmd);
47117c478bd9Sstevel@tonic-gate 			break;
47127c478bd9Sstevel@tonic-gate 		default:
47137c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4714bbec428eSgjelinek 			    B_TRUE);
4715bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4716bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
47177c478bd9Sstevel@tonic-gate 			return;
47187c478bd9Sstevel@tonic-gate 		}
47197c478bd9Sstevel@tonic-gate 		return;
47207c478bd9Sstevel@tonic-gate 	case RT_ATTR:
47217c478bd9Sstevel@tonic-gate 		switch (prop_type) {
47227c478bd9Sstevel@tonic-gate 		case PT_NAME:
47237c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_attrtab.zone_attr_name,
47247c478bd9Sstevel@tonic-gate 			    prop_id,
47257c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_attrtab.zone_attr_name));
47267c478bd9Sstevel@tonic-gate 			break;
47277c478bd9Sstevel@tonic-gate 		case PT_TYPE:
47287c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_attrtab.zone_attr_type,
47297c478bd9Sstevel@tonic-gate 			    prop_id,
47307c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_attrtab.zone_attr_type));
47317c478bd9Sstevel@tonic-gate 			break;
47327c478bd9Sstevel@tonic-gate 		case PT_VALUE:
47337c478bd9Sstevel@tonic-gate 			(void) strlcpy(in_progress_attrtab.zone_attr_value,
47347c478bd9Sstevel@tonic-gate 			    prop_id,
47357c478bd9Sstevel@tonic-gate 			    sizeof (in_progress_attrtab.zone_attr_value));
47367c478bd9Sstevel@tonic-gate 			break;
47377c478bd9Sstevel@tonic-gate 		default:
47387c478bd9Sstevel@tonic-gate 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4739bbec428eSgjelinek 			    B_TRUE);
4740bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4741bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
47427c478bd9Sstevel@tonic-gate 			return;
47437c478bd9Sstevel@tonic-gate 		}
47447c478bd9Sstevel@tonic-gate 		return;
4745fa9e4066Sahrens 	case RT_DATASET:
4746fa9e4066Sahrens 		switch (prop_type) {
4747fa9e4066Sahrens 		case PT_NAME:
4748fa9e4066Sahrens 			(void) strlcpy(in_progress_dstab.zone_dataset_name,
4749fa9e4066Sahrens 			    prop_id,
4750fa9e4066Sahrens 			    sizeof (in_progress_dstab.zone_dataset_name));
4751fa9e4066Sahrens 			return;
4752fa9e4066Sahrens 		default:
4753fa9e4066Sahrens 			break;
4754fa9e4066Sahrens 		}
4755bbec428eSgjelinek 		zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, B_TRUE);
4756bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
4757bbec428eSgjelinek 		usage(B_FALSE, HELP_PROPS);
4758fa9e4066Sahrens 		return;
47590209230bSgjelinek 	case RT_DCPU:
47600209230bSgjelinek 		switch (prop_type) {
47610209230bSgjelinek 		char *lowp, *highp;
47620209230bSgjelinek 
47630209230bSgjelinek 		case PT_NCPUS:
47640209230bSgjelinek 			lowp = prop_id;
47650209230bSgjelinek 			if ((highp = strchr(prop_id, '-')) != NULL)
47660209230bSgjelinek 				*highp++ = '\0';
47670209230bSgjelinek 			else
47680209230bSgjelinek 				highp = lowp;
47690209230bSgjelinek 
47700209230bSgjelinek 			/* Make sure the input makes sense. */
47710209230bSgjelinek 			if (!zonecfg_valid_ncpus(lowp, highp)) {
47720209230bSgjelinek 				zerr(gettext("%s property is out of range."),
47730209230bSgjelinek 				    pt_to_str(PT_NCPUS));
4774bbec428eSgjelinek 				saw_error = B_TRUE;
47750209230bSgjelinek 				return;
47760209230bSgjelinek 			}
47770209230bSgjelinek 
47780209230bSgjelinek 			(void) strlcpy(
47790209230bSgjelinek 			    in_progress_psettab.zone_ncpu_min, lowp,
47800209230bSgjelinek 			    sizeof (in_progress_psettab.zone_ncpu_min));
47810209230bSgjelinek 			(void) strlcpy(
47820209230bSgjelinek 			    in_progress_psettab.zone_ncpu_max, highp,
47830209230bSgjelinek 			    sizeof (in_progress_psettab.zone_ncpu_max));
47840209230bSgjelinek 			return;
47850209230bSgjelinek 		case PT_IMPORTANCE:
47860209230bSgjelinek 			/* Make sure the value makes sense. */
47870209230bSgjelinek 			if (!zonecfg_valid_importance(prop_id)) {
47880209230bSgjelinek 				zerr(gettext("%s property is out of range."),
47890209230bSgjelinek 				    pt_to_str(PT_IMPORTANCE));
4790bbec428eSgjelinek 				saw_error = B_TRUE;
47910209230bSgjelinek 				return;
47920209230bSgjelinek 			}
47930209230bSgjelinek 
47940209230bSgjelinek 			(void) strlcpy(in_progress_psettab.zone_importance,
47950209230bSgjelinek 			    prop_id,
47960209230bSgjelinek 			    sizeof (in_progress_psettab.zone_importance));
47970209230bSgjelinek 			return;
47980209230bSgjelinek 		default:
47990209230bSgjelinek 			break;
48000209230bSgjelinek 		}
4801bbec428eSgjelinek 		zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, B_TRUE);
4802bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
4803bbec428eSgjelinek 		usage(B_FALSE, HELP_PROPS);
48040209230bSgjelinek 		return;
4805c97ad5cdSakolb 	case RT_PCAP:
4806c97ad5cdSakolb 		if (prop_type != PT_NCPUS) {
4807c97ad5cdSakolb 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4808bbec428eSgjelinek 			    B_TRUE);
4809bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4810bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
4811c97ad5cdSakolb 			return;
4812c97ad5cdSakolb 		}
4813c97ad5cdSakolb 
4814c97ad5cdSakolb 		/*
4815c97ad5cdSakolb 		 * We already checked that an rctl alias is allowed in
4816c97ad5cdSakolb 		 * the add_resource() function.
4817c97ad5cdSakolb 		 */
4818c97ad5cdSakolb 
4819c97ad5cdSakolb 		if ((cap = strtof(prop_id, &unitp)) <= 0 || *unitp != '\0' ||
4820c97ad5cdSakolb 		    (int)(cap * 100) < 1) {
4821c97ad5cdSakolb 			zerr(gettext("%s property is out of range."),
4822c97ad5cdSakolb 			    pt_to_str(PT_NCPUS));
4823bbec428eSgjelinek 			saw_error = B_TRUE;
4824c97ad5cdSakolb 			return;
4825c97ad5cdSakolb 		}
4826c97ad5cdSakolb 
4827c97ad5cdSakolb 		if ((err = zonecfg_set_aliased_rctl(handle, ALIAS_CPUCAP,
4828c97ad5cdSakolb 		    (int)(cap * 100))) != Z_OK)
4829bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
4830c97ad5cdSakolb 		else
4831bbec428eSgjelinek 			need_to_commit = B_TRUE;
4832c97ad5cdSakolb 		return;
48330209230bSgjelinek 	case RT_MCAP:
48340209230bSgjelinek 		switch (prop_type) {
48350209230bSgjelinek 		case PT_PHYSICAL:
48360209230bSgjelinek 			if (!zonecfg_valid_memlimit(prop_id, &mem_cap)) {
48370209230bSgjelinek 				zerr(gettext("A positive number with a "
48380209230bSgjelinek 				    "required scale suffix (K, M, G or T) was "
48390209230bSgjelinek 				    "expected here."));
4840bbec428eSgjelinek 				saw_error = B_TRUE;
48410209230bSgjelinek 			} else if (mem_cap < ONE_MB) {
48420209230bSgjelinek 				zerr(gettext("%s value is too small.  It must "
48430209230bSgjelinek 				    "be at least 1M."), pt_to_str(PT_PHYSICAL));
4844bbec428eSgjelinek 				saw_error = B_TRUE;
48450209230bSgjelinek 			} else {
48460209230bSgjelinek 				snprintf(in_progress_mcaptab.zone_physmem_cap,
48470209230bSgjelinek 				    physmem_size, "%llu", mem_cap);
48480209230bSgjelinek 			}
48490209230bSgjelinek 			break;
48500209230bSgjelinek 		case PT_SWAP:
48510209230bSgjelinek 			/*
48520209230bSgjelinek 			 * We have to check if an rctl is allowed here since
48530209230bSgjelinek 			 * there might already be a rctl defined that blocks
48540209230bSgjelinek 			 * the alias.
48550209230bSgjelinek 			 */
48560209230bSgjelinek 			if (!zonecfg_aliased_rctl_ok(handle, ALIAS_MAXSWAP)) {
48570209230bSgjelinek 				zone_perror(pt_to_str(PT_MAXSWAP),
4858bbec428eSgjelinek 				    Z_ALIAS_DISALLOW, B_FALSE);
4859bbec428eSgjelinek 				saw_error = B_TRUE;
48600209230bSgjelinek 				return;
48610209230bSgjelinek 			}
48620209230bSgjelinek 
48630209230bSgjelinek 			if (global_zone)
48640209230bSgjelinek 				mem_limit = ONE_MB * 100;
48650209230bSgjelinek 			else
48660209230bSgjelinek 				mem_limit = ONE_MB * 50;
48670209230bSgjelinek 
48680209230bSgjelinek 			if (!zonecfg_valid_memlimit(prop_id, &mem_cap)) {
48690209230bSgjelinek 				zerr(gettext("A positive number with a "
48700209230bSgjelinek 				    "required scale suffix (K, M, G or T) was "
48710209230bSgjelinek 				    "expected here."));
4872bbec428eSgjelinek 				saw_error = B_TRUE;
48730209230bSgjelinek 			} else if (mem_cap < mem_limit) {
48740209230bSgjelinek 				char buf[128];
48750209230bSgjelinek 
48760209230bSgjelinek 				(void) snprintf(buf, sizeof (buf), "%llu",
48770209230bSgjelinek 				    mem_limit);
48780209230bSgjelinek 				bytes_to_units(buf, buf, sizeof (buf));
48790209230bSgjelinek 				zerr(gettext("%s value is too small.  It must "
48800209230bSgjelinek 				    "be at least %s."), pt_to_str(PT_SWAP),
48810209230bSgjelinek 				    buf);
4882bbec428eSgjelinek 				saw_error = B_TRUE;
48830209230bSgjelinek 			} else {
48840209230bSgjelinek 				if ((err = zonecfg_set_aliased_rctl(handle,
48850209230bSgjelinek 				    ALIAS_MAXSWAP, mem_cap)) != Z_OK)
4886bbec428eSgjelinek 					zone_perror(zone, err, B_TRUE);
48870209230bSgjelinek 				else
4888bbec428eSgjelinek 					need_to_commit = B_TRUE;
48890209230bSgjelinek 			}
48900209230bSgjelinek 			break;
48910209230bSgjelinek 		case PT_LOCKED:
48920209230bSgjelinek 			/*
48930209230bSgjelinek 			 * We have to check if an rctl is allowed here since
48940209230bSgjelinek 			 * there might already be a rctl defined that blocks
48950209230bSgjelinek 			 * the alias.
48960209230bSgjelinek 			 */
48970209230bSgjelinek 			if (!zonecfg_aliased_rctl_ok(handle,
48980209230bSgjelinek 			    ALIAS_MAXLOCKEDMEM)) {
48990209230bSgjelinek 				zone_perror(pt_to_str(PT_LOCKED),
4900bbec428eSgjelinek 				    Z_ALIAS_DISALLOW, B_FALSE);
4901bbec428eSgjelinek 				saw_error = B_TRUE;
49020209230bSgjelinek 				return;
49030209230bSgjelinek 			}
49040209230bSgjelinek 
49050209230bSgjelinek 			if (!zonecfg_valid_memlimit(prop_id, &mem_cap)) {
49060209230bSgjelinek 				zerr(gettext("A non-negative number with a "
49070209230bSgjelinek 				    "required scale suffix (K, M, G or T) was "
49080209230bSgjelinek 				    "expected\nhere."));
4909bbec428eSgjelinek 				saw_error = B_TRUE;
49100209230bSgjelinek 			} else {
49110209230bSgjelinek 				if ((err = zonecfg_set_aliased_rctl(handle,
49120209230bSgjelinek 				    ALIAS_MAXLOCKEDMEM, mem_cap)) != Z_OK)
4913bbec428eSgjelinek 					zone_perror(zone, err, B_TRUE);
49140209230bSgjelinek 				else
4915bbec428eSgjelinek 					need_to_commit = B_TRUE;
49160209230bSgjelinek 			}
49170209230bSgjelinek 			break;
49180209230bSgjelinek 		default:
49190209230bSgjelinek 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4920bbec428eSgjelinek 			    B_TRUE);
4921bbec428eSgjelinek 			long_usage(CMD_SET, B_TRUE);
4922bbec428eSgjelinek 			usage(B_FALSE, HELP_PROPS);
49230209230bSgjelinek 			return;
49240209230bSgjelinek 		}
49250209230bSgjelinek 		return;
4926cb8a054bSGlenn Faden 	case RT_ADMIN:
4927cb8a054bSGlenn Faden 		switch (prop_type) {
4928cb8a054bSGlenn Faden 		case PT_USER:
4929cb8a054bSGlenn Faden 			(void) strlcpy(in_progress_admintab.zone_admin_user,
4930cb8a054bSGlenn Faden 			    prop_id,
4931cb8a054bSGlenn Faden 			    sizeof (in_progress_admintab.zone_admin_user));
4932cb8a054bSGlenn Faden 			return;
4933cb8a054bSGlenn Faden 		case PT_AUTHS:
4934cb8a054bSGlenn Faden 			(void) strlcpy(in_progress_admintab.zone_admin_auths,
4935cb8a054bSGlenn Faden 			    prop_id,
4936cb8a054bSGlenn Faden 			    sizeof (in_progress_admintab.zone_admin_auths));
4937cb8a054bSGlenn Faden 			return;
4938cb8a054bSGlenn Faden 		default:
4939cb8a054bSGlenn Faden 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4940cb8a054bSGlenn Faden 			    B_TRUE);
4941cb8a054bSGlenn Faden 			long_usage(CMD_SET, B_TRUE);
4942cb8a054bSGlenn Faden 			usage(B_FALSE, HELP_PROPS);
4943cb8a054bSGlenn Faden 			return;
4944cb8a054bSGlenn Faden 		}
4945d2a70789SRichard Lowe 	case RT_SECFLAGS: {
4946d2a70789SRichard Lowe 		char *propstr;
4947d2a70789SRichard Lowe 
4948d2a70789SRichard Lowe 		switch (prop_type) {
4949d2a70789SRichard Lowe 		case PT_DEFAULT:
4950d2a70789SRichard Lowe 			propstr = in_progress_secflagstab.zone_secflags_default;
4951d2a70789SRichard Lowe 			break;
4952d2a70789SRichard Lowe 		case PT_UPPER:
4953d2a70789SRichard Lowe 			propstr = in_progress_secflagstab.zone_secflags_upper;
4954d2a70789SRichard Lowe 			break;
4955d2a70789SRichard Lowe 		case PT_LOWER:
4956d2a70789SRichard Lowe 			propstr = in_progress_secflagstab.zone_secflags_lower;
4957d2a70789SRichard Lowe 			break;
4958d2a70789SRichard Lowe 		default:
4959d2a70789SRichard Lowe 			zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE,
4960d2a70789SRichard Lowe 			    B_TRUE);
4961d2a70789SRichard Lowe 			long_usage(CMD_SET, B_TRUE);
4962d2a70789SRichard Lowe 			usage(B_FALSE, HELP_PROPS);
4963d2a70789SRichard Lowe 			return;
4964d2a70789SRichard Lowe 		}
4965d2a70789SRichard Lowe 		(void) strlcpy(propstr, prop_id, ZONECFG_SECFLAGS_MAX);
4966d2a70789SRichard Lowe 		return;
4967d2a70789SRichard Lowe 	}
49687c478bd9Sstevel@tonic-gate 	default:
4969bbec428eSgjelinek 		zone_perror(rt_to_str(res_type), Z_NO_RESOURCE_TYPE, B_TRUE);
4970bbec428eSgjelinek 		long_usage(CMD_SET, B_TRUE);
4971bbec428eSgjelinek 		usage(B_FALSE, HELP_RESOURCES);
49727c478bd9Sstevel@tonic-gate 		return;
49737c478bd9Sstevel@tonic-gate 	}
49747c478bd9Sstevel@tonic-gate }
49757c478bd9Sstevel@tonic-gate 
49767c478bd9Sstevel@tonic-gate static void
output_prop(FILE * fp,int pnum,char * pval,boolean_t print_notspec)4977bbec428eSgjelinek output_prop(FILE *fp, int pnum, char *pval, boolean_t print_notspec)
49787c478bd9Sstevel@tonic-gate {
49797c478bd9Sstevel@tonic-gate 	char *qstr;
49807c478bd9Sstevel@tonic-gate 
49817c478bd9Sstevel@tonic-gate 	if (*pval != '\0') {
49827c478bd9Sstevel@tonic-gate 		qstr = quoteit(pval);
49830209230bSgjelinek 		if (pnum == PT_SWAP || pnum == PT_LOCKED)
49840209230bSgjelinek 			(void) fprintf(fp, "\t[%s: %s]\n", pt_to_str(pnum),
49850209230bSgjelinek 			    qstr);
49860209230bSgjelinek 		else
49877c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\t%s: %s\n", pt_to_str(pnum), qstr);
49887c478bd9Sstevel@tonic-gate 		free(qstr);
49897c478bd9Sstevel@tonic-gate 	} else if (print_notspec)
4990087719fdSdp 		(void) fprintf(fp, gettext("\t%s not specified\n"),
4991087719fdSdp 		    pt_to_str(pnum));
4992087719fdSdp }
4993087719fdSdp 
4994087719fdSdp static void
info_zonename(zone_dochandle_t handle,FILE * fp)4995087719fdSdp info_zonename(zone_dochandle_t handle, FILE *fp)
4996087719fdSdp {
4997087719fdSdp 	char zonename[ZONENAME_MAX];
4998087719fdSdp 
4999087719fdSdp 	if (zonecfg_get_name(handle, zonename, sizeof (zonename)) == Z_OK)
5000087719fdSdp 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_ZONENAME),
5001087719fdSdp 		    zonename);
5002087719fdSdp 	else
5003087719fdSdp 		(void) fprintf(fp, gettext("%s not specified\n"),
5004087719fdSdp 		    pt_to_str(PT_ZONENAME));
50057c478bd9Sstevel@tonic-gate }
50067c478bd9Sstevel@tonic-gate 
50077c478bd9Sstevel@tonic-gate static void
info_zonepath(zone_dochandle_t handle,FILE * fp)50087c478bd9Sstevel@tonic-gate info_zonepath(zone_dochandle_t handle, FILE *fp)
50097c478bd9Sstevel@tonic-gate {
50107c478bd9Sstevel@tonic-gate 	char zonepath[MAXPATHLEN];
50117c478bd9Sstevel@tonic-gate 
50127c478bd9Sstevel@tonic-gate 	if (zonecfg_get_zonepath(handle, zonepath, sizeof (zonepath)) == Z_OK)
50137c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_ZONEPATH),
50147c478bd9Sstevel@tonic-gate 		    zonepath);
5015087719fdSdp 	else {
5016087719fdSdp 		(void) fprintf(fp, gettext("%s not specified\n"),
5017087719fdSdp 		    pt_to_str(PT_ZONEPATH));
5018087719fdSdp 	}
50197c478bd9Sstevel@tonic-gate }
50207c478bd9Sstevel@tonic-gate 
50217c478bd9Sstevel@tonic-gate static void
info_brand(zone_dochandle_t handle,FILE * fp)50229acbbeafSnn35248 info_brand(zone_dochandle_t handle, FILE *fp)
50239acbbeafSnn35248 {
50249acbbeafSnn35248 	char brand[MAXNAMELEN];
50259acbbeafSnn35248 
50269acbbeafSnn35248 	if (zonecfg_get_brand(handle, brand, sizeof (brand)) == Z_OK)
50279acbbeafSnn35248 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_BRAND),
50289acbbeafSnn35248 		    brand);
50299acbbeafSnn35248 	else
50309acbbeafSnn35248 		(void) fprintf(fp, "%s %s\n", pt_to_str(PT_BRAND),
50319acbbeafSnn35248 		    gettext("not specified"));
50329acbbeafSnn35248 }
50339acbbeafSnn35248 
50349acbbeafSnn35248 static void
info_autoboot(zone_dochandle_t handle,FILE * fp)50357c478bd9Sstevel@tonic-gate info_autoboot(zone_dochandle_t handle, FILE *fp)
50367c478bd9Sstevel@tonic-gate {
50377c478bd9Sstevel@tonic-gate 	boolean_t autoboot;
50387c478bd9Sstevel@tonic-gate 	int err;
50397c478bd9Sstevel@tonic-gate 
50407c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_get_autoboot(handle, &autoboot)) == Z_OK)
50417c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_AUTOBOOT),
50427c478bd9Sstevel@tonic-gate 		    autoboot ? "true" : "false");
50437c478bd9Sstevel@tonic-gate 	else
5044bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
50457c478bd9Sstevel@tonic-gate }
50467c478bd9Sstevel@tonic-gate 
50477c478bd9Sstevel@tonic-gate static void
info_pool(zone_dochandle_t handle,FILE * fp)50487c478bd9Sstevel@tonic-gate info_pool(zone_dochandle_t handle, FILE *fp)
50497c478bd9Sstevel@tonic-gate {
50507c478bd9Sstevel@tonic-gate 	char pool[MAXNAMELEN];
50517c478bd9Sstevel@tonic-gate 	int err;
50527c478bd9Sstevel@tonic-gate 
50537c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_get_pool(handle, pool, sizeof (pool))) == Z_OK)
50547c478bd9Sstevel@tonic-gate 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_POOL), pool);
50557c478bd9Sstevel@tonic-gate 	else
5056bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
50577c478bd9Sstevel@tonic-gate }
50587c478bd9Sstevel@tonic-gate 
50597c478bd9Sstevel@tonic-gate static void
info_limitpriv(zone_dochandle_t handle,FILE * fp)5060ffbafc53Scomay info_limitpriv(zone_dochandle_t handle, FILE *fp)
5061ffbafc53Scomay {
5062ffbafc53Scomay 	char *limitpriv;
5063ffbafc53Scomay 	int err;
5064ffbafc53Scomay 
5065ffbafc53Scomay 	if ((err = zonecfg_get_limitpriv(handle, &limitpriv)) == Z_OK) {
5066ffbafc53Scomay 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_LIMITPRIV),
5067ffbafc53Scomay 		    limitpriv);
5068ffbafc53Scomay 		free(limitpriv);
5069ffbafc53Scomay 	} else {
5070bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
5071ffbafc53Scomay 	}
5072ffbafc53Scomay }
5073ffbafc53Scomay 
5074ffbafc53Scomay static void
info_bootargs(zone_dochandle_t handle,FILE * fp)50753f2f09c1Sdp info_bootargs(zone_dochandle_t handle, FILE *fp)
50763f2f09c1Sdp {
50773f2f09c1Sdp 	char bootargs[BOOTARGS_MAX];
50783f2f09c1Sdp 	int err;
50793f2f09c1Sdp 
50803f2f09c1Sdp 	if ((err = zonecfg_get_bootargs(handle, bootargs,
50813f2f09c1Sdp 	    sizeof (bootargs))) == Z_OK) {
50823f2f09c1Sdp 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_BOOTARGS),
50833f2f09c1Sdp 		    bootargs);
50843f2f09c1Sdp 	} else {
5085bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
50863f2f09c1Sdp 	}
50873f2f09c1Sdp }
50883f2f09c1Sdp 
50893f2f09c1Sdp static void
info_sched(zone_dochandle_t handle,FILE * fp)50900209230bSgjelinek info_sched(zone_dochandle_t handle, FILE *fp)
50910209230bSgjelinek {
50920209230bSgjelinek 	char sched[MAXNAMELEN];
50930209230bSgjelinek 	int err;
50940209230bSgjelinek 
50950209230bSgjelinek 	if ((err = zonecfg_get_sched_class(handle, sched, sizeof (sched)))
50960209230bSgjelinek 	    == Z_OK) {
50970209230bSgjelinek 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_SCHED), sched);
50980209230bSgjelinek 	} else {
5099bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
51000209230bSgjelinek 	}
51010209230bSgjelinek }
51020209230bSgjelinek 
51030209230bSgjelinek static void
info_iptype(zone_dochandle_t handle,FILE * fp)5104f4b3ec61Sdh155122 info_iptype(zone_dochandle_t handle, FILE *fp)
5105f4b3ec61Sdh155122 {
5106f4b3ec61Sdh155122 	zone_iptype_t iptype;
5107f4b3ec61Sdh155122 	int err;
5108f4b3ec61Sdh155122 
5109f4b3ec61Sdh155122 	if ((err = zonecfg_get_iptype(handle, &iptype)) == Z_OK) {
5110f4b3ec61Sdh155122 		switch (iptype) {
5111f4b3ec61Sdh155122 		case ZS_SHARED:
5112f4b3ec61Sdh155122 			(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_IPTYPE),
5113f4b3ec61Sdh155122 			    "shared");
5114f4b3ec61Sdh155122 			break;
5115f4b3ec61Sdh155122 		case ZS_EXCLUSIVE:
5116f4b3ec61Sdh155122 			(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_IPTYPE),
5117f4b3ec61Sdh155122 			    "exclusive");
5118f4b3ec61Sdh155122 			break;
5119f4b3ec61Sdh155122 		}
5120f4b3ec61Sdh155122 	} else {
5121bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
5122f4b3ec61Sdh155122 	}
5123f4b3ec61Sdh155122 }
5124f4b3ec61Sdh155122 
5125f4b3ec61Sdh155122 static void
info_hostid(zone_dochandle_t handle,FILE * fp)51265679c89fSjv227347 info_hostid(zone_dochandle_t handle, FILE *fp)
51275679c89fSjv227347 {
51285679c89fSjv227347 	char hostidp[HW_HOSTID_LEN];
51290fbb751dSJohn Levon 	int err;
51305679c89fSjv227347 
51310fbb751dSJohn Levon 	if ((err = zonecfg_get_hostid(handle, hostidp,
51320fbb751dSJohn Levon 	    sizeof (hostidp))) == Z_OK) {
51335679c89fSjv227347 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_HOSTID), hostidp);
51340fbb751dSJohn Levon 	} else if (err == Z_BAD_PROPERTY) {
51350fbb751dSJohn Levon 		(void) fprintf(fp, "%s: \n", pt_to_str(PT_HOSTID));
51360fbb751dSJohn Levon 	} else {
51370fbb751dSJohn Levon 		zone_perror(zone, err, B_TRUE);
51380fbb751dSJohn Levon 	}
51390fbb751dSJohn Levon }
51400fbb751dSJohn Levon 
51410fbb751dSJohn Levon static void
info_fs_allowed(zone_dochandle_t handle,FILE * fp)51420fbb751dSJohn Levon info_fs_allowed(zone_dochandle_t handle, FILE *fp)
51430fbb751dSJohn Levon {
51440fbb751dSJohn Levon 	char fsallowedp[ZONE_FS_ALLOWED_MAX];
51450fbb751dSJohn Levon 	int err;
51460fbb751dSJohn Levon 
51470fbb751dSJohn Levon 	if ((err = zonecfg_get_fs_allowed(handle, fsallowedp,
51480fbb751dSJohn Levon 	    sizeof (fsallowedp))) == Z_OK) {
51490fbb751dSJohn Levon 		(void) fprintf(fp, "%s: %s\n", pt_to_str(PT_FS_ALLOWED),
51500fbb751dSJohn Levon 		    fsallowedp);
51510fbb751dSJohn Levon 	} else if (err == Z_BAD_PROPERTY) {
51520fbb751dSJohn Levon 		(void) fprintf(fp, "%s: \n", pt_to_str(PT_FS_ALLOWED));
51530fbb751dSJohn Levon 	} else {
51540fbb751dSJohn Levon 		zone_perror(zone, err, B_TRUE);
51550fbb751dSJohn Levon 	}
51565679c89fSjv227347 }
51575679c89fSjv227347 
51585679c89fSjv227347 static void
output_fs(FILE * fp,struct zone_fstab * fstab)51597c478bd9Sstevel@tonic-gate output_fs(FILE *fp, struct zone_fstab *fstab)
51607c478bd9Sstevel@tonic-gate {
51617c478bd9Sstevel@tonic-gate 	zone_fsopt_t *this;
51627c478bd9Sstevel@tonic-gate 
51637c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_FS));
51647c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_DIR, fstab->zone_fs_dir, B_TRUE);
51657c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_SPECIAL, fstab->zone_fs_special, B_TRUE);
51667c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_RAW, fstab->zone_fs_raw, B_TRUE);
51677c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_TYPE, fstab->zone_fs_type, B_TRUE);
51687c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "\t%s: [", pt_to_str(PT_OPTIONS));
51697c478bd9Sstevel@tonic-gate 	for (this = fstab->zone_fs_options; this != NULL;
51707c478bd9Sstevel@tonic-gate 	    this = this->zone_fsopt_next) {
51717c478bd9Sstevel@tonic-gate 		if (strchr(this->zone_fsopt_opt, '='))
51727c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "\"%s\"", this->zone_fsopt_opt);
51737c478bd9Sstevel@tonic-gate 		else
51747c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, "%s", this->zone_fsopt_opt);
51757c478bd9Sstevel@tonic-gate 		if (this->zone_fsopt_next != NULL)
51767c478bd9Sstevel@tonic-gate 			(void) fprintf(fp, ",");
51777c478bd9Sstevel@tonic-gate 	}
51787c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "]\n");
51797c478bd9Sstevel@tonic-gate }
51807c478bd9Sstevel@tonic-gate 
51817c478bd9Sstevel@tonic-gate static void
info_fs(zone_dochandle_t handle,FILE * fp,cmd_t * cmd)51827c478bd9Sstevel@tonic-gate info_fs(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
51837c478bd9Sstevel@tonic-gate {
51847c478bd9Sstevel@tonic-gate 	struct zone_fstab lookup, user;
5185bbec428eSgjelinek 	boolean_t output = B_FALSE;
51867c478bd9Sstevel@tonic-gate 
51877c478bd9Sstevel@tonic-gate 	if (zonecfg_setfsent(handle) != Z_OK)
51887c478bd9Sstevel@tonic-gate 		return;
51897c478bd9Sstevel@tonic-gate 	while (zonecfg_getfsent(handle, &lookup) == Z_OK) {
51907c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
51917c478bd9Sstevel@tonic-gate 			output_fs(fp, &lookup);
51927c478bd9Sstevel@tonic-gate 			goto loopend;
51937c478bd9Sstevel@tonic-gate 		}
5194bbec428eSgjelinek 		if (fill_in_fstab(cmd, &user, B_TRUE) != Z_OK)
51957c478bd9Sstevel@tonic-gate 			goto loopend;
51967c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_fs_dir) > 0 &&
51977c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_fs_dir, lookup.zone_fs_dir) != 0)
51987c478bd9Sstevel@tonic-gate 			goto loopend;	/* no match */
51997c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_fs_special) > 0 &&
52007c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_fs_special, lookup.zone_fs_special) != 0)
52017c478bd9Sstevel@tonic-gate 			goto loopend;	/* no match */
52027c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_fs_type) > 0 &&
52037c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_fs_type, lookup.zone_fs_type) != 0)
52047c478bd9Sstevel@tonic-gate 			goto loopend;	/* no match */
52057c478bd9Sstevel@tonic-gate 		output_fs(fp, &lookup);
5206bbec428eSgjelinek 		output = B_TRUE;
52077c478bd9Sstevel@tonic-gate loopend:
52087c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(lookup.zone_fs_options);
52097c478bd9Sstevel@tonic-gate 	}
52107c478bd9Sstevel@tonic-gate 	(void) zonecfg_endfsent(handle);
52117c478bd9Sstevel@tonic-gate 	/*
52127c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
52137c478bd9Sstevel@tonic-gate 	 * nothing to output.
52147c478bd9Sstevel@tonic-gate 	 */
52157c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
52167c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
52177c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_FS));
52187c478bd9Sstevel@tonic-gate }
52197c478bd9Sstevel@tonic-gate 
52207c478bd9Sstevel@tonic-gate static void
output_net(FILE * fp,struct zone_nwiftab * nwiftab)52217c478bd9Sstevel@tonic-gate output_net(FILE *fp, struct zone_nwiftab *nwiftab)
52227c478bd9Sstevel@tonic-gate {
52237c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_NET));
52247c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_ADDRESS, nwiftab->zone_nwif_address, B_TRUE);
5225550b6e40SSowmini Varadhan 	output_prop(fp, PT_ALLOWED_ADDRESS,
5226550b6e40SSowmini Varadhan 	    nwiftab->zone_nwif_allowed_address, B_TRUE);
52277c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_PHYSICAL, nwiftab->zone_nwif_physical, B_TRUE);
5228de860bd9Sgfaden 	output_prop(fp, PT_DEFROUTER, nwiftab->zone_nwif_defrouter, B_TRUE);
52297c478bd9Sstevel@tonic-gate }
52307c478bd9Sstevel@tonic-gate 
52317c478bd9Sstevel@tonic-gate static void
info_net(zone_dochandle_t handle,FILE * fp,cmd_t * cmd)52327c478bd9Sstevel@tonic-gate info_net(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
52337c478bd9Sstevel@tonic-gate {
52347c478bd9Sstevel@tonic-gate 	struct zone_nwiftab lookup, user;
5235bbec428eSgjelinek 	boolean_t output = B_FALSE;
52367c478bd9Sstevel@tonic-gate 
52377c478bd9Sstevel@tonic-gate 	if (zonecfg_setnwifent(handle) != Z_OK)
52387c478bd9Sstevel@tonic-gate 		return;
52397c478bd9Sstevel@tonic-gate 	while (zonecfg_getnwifent(handle, &lookup) == Z_OK) {
52407c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
52417c478bd9Sstevel@tonic-gate 			output_net(fp, &lookup);
52427c478bd9Sstevel@tonic-gate 			continue;
52437c478bd9Sstevel@tonic-gate 		}
5244bbec428eSgjelinek 		if (fill_in_nwiftab(cmd, &user, B_TRUE) != Z_OK)
52457c478bd9Sstevel@tonic-gate 			continue;
52467c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_nwif_physical) > 0 &&
52477c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_nwif_physical,
52487c478bd9Sstevel@tonic-gate 		    lookup.zone_nwif_physical) != 0)
52497c478bd9Sstevel@tonic-gate 			continue;	/* no match */
5250f4b3ec61Sdh155122 		/* If present make sure it matches */
52517c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_nwif_address) > 0 &&
52527c478bd9Sstevel@tonic-gate 		    !zonecfg_same_net_address(user.zone_nwif_address,
52537c478bd9Sstevel@tonic-gate 		    lookup.zone_nwif_address))
52547c478bd9Sstevel@tonic-gate 			continue;	/* no match */
52557c478bd9Sstevel@tonic-gate 		output_net(fp, &lookup);
5256bbec428eSgjelinek 		output = B_TRUE;
52577c478bd9Sstevel@tonic-gate 	}
52587c478bd9Sstevel@tonic-gate 	(void) zonecfg_endnwifent(handle);
52597c478bd9Sstevel@tonic-gate 	/*
52607c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
52617c478bd9Sstevel@tonic-gate 	 * nothing to output.
52627c478bd9Sstevel@tonic-gate 	 */
52637c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
52647c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
52657c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_NET));
52667c478bd9Sstevel@tonic-gate }
52677c478bd9Sstevel@tonic-gate 
52687c478bd9Sstevel@tonic-gate static void
output_dev(FILE * fp,struct zone_devtab * devtab)52697c478bd9Sstevel@tonic-gate output_dev(FILE *fp, struct zone_devtab *devtab)
52707c478bd9Sstevel@tonic-gate {
527127e6fb21Sdp 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_DEVICE));
52727c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_MATCH, devtab->zone_dev_match, B_TRUE);
52737c478bd9Sstevel@tonic-gate }
52747c478bd9Sstevel@tonic-gate 
52757c478bd9Sstevel@tonic-gate static void
info_dev(zone_dochandle_t handle,FILE * fp,cmd_t * cmd)52767c478bd9Sstevel@tonic-gate info_dev(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
52777c478bd9Sstevel@tonic-gate {
52787c478bd9Sstevel@tonic-gate 	struct zone_devtab lookup, user;
5279bbec428eSgjelinek 	boolean_t output = B_FALSE;
52807c478bd9Sstevel@tonic-gate 
52817c478bd9Sstevel@tonic-gate 	if (zonecfg_setdevent(handle) != Z_OK)
52827c478bd9Sstevel@tonic-gate 		return;
52837c478bd9Sstevel@tonic-gate 	while (zonecfg_getdevent(handle, &lookup) == Z_OK) {
52847c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
52857c478bd9Sstevel@tonic-gate 			output_dev(fp, &lookup);
52867c478bd9Sstevel@tonic-gate 			continue;
52877c478bd9Sstevel@tonic-gate 		}
5288bbec428eSgjelinek 		if (fill_in_devtab(cmd, &user, B_TRUE) != Z_OK)
52897c478bd9Sstevel@tonic-gate 			continue;
52907c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_dev_match) > 0 &&
52917c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_dev_match, lookup.zone_dev_match) != 0)
52927c478bd9Sstevel@tonic-gate 			continue;	/* no match */
52937c478bd9Sstevel@tonic-gate 		output_dev(fp, &lookup);
5294bbec428eSgjelinek 		output = B_TRUE;
52957c478bd9Sstevel@tonic-gate 	}
52967c478bd9Sstevel@tonic-gate 	(void) zonecfg_enddevent(handle);
52977c478bd9Sstevel@tonic-gate 	/*
52987c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
52997c478bd9Sstevel@tonic-gate 	 * nothing to output.
53007c478bd9Sstevel@tonic-gate 	 */
53017c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
53027c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
53037c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_DEVICE));
53047c478bd9Sstevel@tonic-gate }
53057c478bd9Sstevel@tonic-gate 
53067c478bd9Sstevel@tonic-gate static void
output_rctl(FILE * fp,struct zone_rctltab * rctltab)53077c478bd9Sstevel@tonic-gate output_rctl(FILE *fp, struct zone_rctltab *rctltab)
53087c478bd9Sstevel@tonic-gate {
53097c478bd9Sstevel@tonic-gate 	struct zone_rctlvaltab *valptr;
53107c478bd9Sstevel@tonic-gate 
53117c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_RCTL));
53127c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_NAME, rctltab->zone_rctl_name, B_TRUE);
53137c478bd9Sstevel@tonic-gate 	for (valptr = rctltab->zone_rctl_valptr; valptr != NULL;
53147c478bd9Sstevel@tonic-gate 	    valptr = valptr->zone_rctlval_next) {
53157c478bd9Sstevel@tonic-gate 		fprintf(fp, "\t%s: (%s=%s,%s=%s,%s=%s)\n",
53167c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_VALUE),
53177c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_PRIV), valptr->zone_rctlval_priv,
53187c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_LIMIT), valptr->zone_rctlval_limit,
53197c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_ACTION), valptr->zone_rctlval_action);
53207c478bd9Sstevel@tonic-gate 	}
53217c478bd9Sstevel@tonic-gate }
53227c478bd9Sstevel@tonic-gate 
53237c478bd9Sstevel@tonic-gate static void
info_rctl(zone_dochandle_t handle,FILE * fp,cmd_t * cmd)53247c478bd9Sstevel@tonic-gate info_rctl(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
53257c478bd9Sstevel@tonic-gate {
53267c478bd9Sstevel@tonic-gate 	struct zone_rctltab lookup, user;
5327bbec428eSgjelinek 	boolean_t output = B_FALSE;
53287c478bd9Sstevel@tonic-gate 
53297c478bd9Sstevel@tonic-gate 	if (zonecfg_setrctlent(handle) != Z_OK)
53307c478bd9Sstevel@tonic-gate 		return;
53317c478bd9Sstevel@tonic-gate 	while (zonecfg_getrctlent(handle, &lookup) == Z_OK) {
53327c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
53337c478bd9Sstevel@tonic-gate 			output_rctl(fp, &lookup);
5334bbec428eSgjelinek 		} else if (fill_in_rctltab(cmd, &user, B_TRUE) == Z_OK &&
53357c478bd9Sstevel@tonic-gate 		    (strlen(user.zone_rctl_name) == 0 ||
53367c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_rctl_name, lookup.zone_rctl_name) == 0)) {
53377c478bd9Sstevel@tonic-gate 			output_rctl(fp, &lookup);
5338bbec428eSgjelinek 			output = B_TRUE;
53397c478bd9Sstevel@tonic-gate 		}
53407c478bd9Sstevel@tonic-gate 		zonecfg_free_rctl_value_list(lookup.zone_rctl_valptr);
53417c478bd9Sstevel@tonic-gate 	}
53427c478bd9Sstevel@tonic-gate 	(void) zonecfg_endrctlent(handle);
53437c478bd9Sstevel@tonic-gate 	/*
53447c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
53457c478bd9Sstevel@tonic-gate 	 * nothing to output.
53467c478bd9Sstevel@tonic-gate 	 */
53477c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
53487c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
53497c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_RCTL));
53507c478bd9Sstevel@tonic-gate }
53517c478bd9Sstevel@tonic-gate 
53527c478bd9Sstevel@tonic-gate static void
output_attr(FILE * fp,struct zone_attrtab * attrtab)53537c478bd9Sstevel@tonic-gate output_attr(FILE *fp, struct zone_attrtab *attrtab)
53547c478bd9Sstevel@tonic-gate {
53557c478bd9Sstevel@tonic-gate 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_ATTR));
53567c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_NAME, attrtab->zone_attr_name, B_TRUE);
53577c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_TYPE, attrtab->zone_attr_type, B_TRUE);
53587c478bd9Sstevel@tonic-gate 	output_prop(fp, PT_VALUE, attrtab->zone_attr_value, B_TRUE);
53597c478bd9Sstevel@tonic-gate }
53607c478bd9Sstevel@tonic-gate 
53617c478bd9Sstevel@tonic-gate static void
info_attr(zone_dochandle_t handle,FILE * fp,cmd_t * cmd)53627c478bd9Sstevel@tonic-gate info_attr(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
53637c478bd9Sstevel@tonic-gate {
53647c478bd9Sstevel@tonic-gate 	struct zone_attrtab lookup, user;
5365bbec428eSgjelinek 	boolean_t output = B_FALSE;
53667c478bd9Sstevel@tonic-gate 
53677c478bd9Sstevel@tonic-gate 	if (zonecfg_setattrent(handle) != Z_OK)
53687c478bd9Sstevel@tonic-gate 		return;
53697c478bd9Sstevel@tonic-gate 	while (zonecfg_getattrent(handle, &lookup) == Z_OK) {
53707c478bd9Sstevel@tonic-gate 		if (cmd->cmd_prop_nv_pairs == 0) {
53717c478bd9Sstevel@tonic-gate 			output_attr(fp, &lookup);
53727c478bd9Sstevel@tonic-gate 			continue;
53737c478bd9Sstevel@tonic-gate 		}
5374bbec428eSgjelinek 		if (fill_in_attrtab(cmd, &user, B_TRUE) != Z_OK)
53757c478bd9Sstevel@tonic-gate 			continue;
53767c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_attr_name) > 0 &&
53777c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_attr_name, lookup.zone_attr_name) != 0)
53787c478bd9Sstevel@tonic-gate 			continue;	/* no match */
53797c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_attr_type) > 0 &&
53807c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_attr_type, lookup.zone_attr_type) != 0)
53817c478bd9Sstevel@tonic-gate 			continue;	/* no match */
53827c478bd9Sstevel@tonic-gate 		if (strlen(user.zone_attr_value) > 0 &&
53837c478bd9Sstevel@tonic-gate 		    strcmp(user.zone_attr_value, lookup.zone_attr_value) != 0)
53847c478bd9Sstevel@tonic-gate 			continue;	/* no match */
53857c478bd9Sstevel@tonic-gate 		output_attr(fp, &lookup);
5386bbec428eSgjelinek 		output = B_TRUE;
53877c478bd9Sstevel@tonic-gate 	}
53887c478bd9Sstevel@tonic-gate 	(void) zonecfg_endattrent(handle);
53897c478bd9Sstevel@tonic-gate 	/*
53907c478bd9Sstevel@tonic-gate 	 * If a property n/v pair was specified, warn the user if there was
53917c478bd9Sstevel@tonic-gate 	 * nothing to output.
53927c478bd9Sstevel@tonic-gate 	 */
53937c478bd9Sstevel@tonic-gate 	if (!output && cmd->cmd_prop_nv_pairs > 0)
53947c478bd9Sstevel@tonic-gate 		(void) printf(gettext("No such %s resource.\n"),
53957c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR));
53967c478bd9Sstevel@tonic-gate }
53977c478bd9Sstevel@tonic-gate 
5398fa9e4066Sahrens static void
output_ds(FILE * fp,struct zone_dstab * dstab)5399fa9e4066Sahrens output_ds(FILE *fp, struct zone_dstab *dstab)
5400fa9e4066Sahrens {
5401fa9e4066Sahrens 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_DATASET));
5402fa9e4066Sahrens 	output_prop(fp, PT_NAME, dstab->zone_dataset_name, B_TRUE);
5403fa9e4066Sahrens }
5404fa9e4066Sahrens 
5405fa9e4066Sahrens static void
info_ds(zone_dochandle_t handle,FILE * fp,cmd_t * cmd)5406fa9e4066Sahrens info_ds(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
5407fa9e4066Sahrens {
5408fa9e4066Sahrens 	struct zone_dstab lookup, user;
5409bbec428eSgjelinek 	boolean_t output = B_FALSE;
5410fa9e4066Sahrens 
54110209230bSgjelinek 	if (zonecfg_setdsent(handle) != Z_OK)
5412fa9e4066Sahrens 		return;
5413fa9e4066Sahrens 	while (zonecfg_getdsent(handle, &lookup) == Z_OK) {
5414fa9e4066Sahrens 		if (cmd->cmd_prop_nv_pairs == 0) {
5415fa9e4066Sahrens 			output_ds(fp, &lookup);
5416fa9e4066Sahrens 			continue;
5417fa9e4066Sahrens 		}
5418bbec428eSgjelinek 		if (fill_in_dstab(cmd, &user, B_TRUE) != Z_OK)
5419fa9e4066Sahrens 			continue;
5420fa9e4066Sahrens 		if (strlen(user.zone_dataset_name) > 0 &&
5421fa9e4066Sahrens 		    strcmp(user.zone_dataset_name,
5422fa9e4066Sahrens 		    lookup.zone_dataset_name) != 0)
5423fa9e4066Sahrens 			continue;	/* no match */
5424fa9e4066Sahrens 		output_ds(fp, &lookup);
5425bbec428eSgjelinek 		output = B_TRUE;
5426fa9e4066Sahrens 	}
5427fa9e4066Sahrens 	(void) zonecfg_enddsent(handle);
5428fa9e4066Sahrens 	/*
5429fa9e4066Sahrens 	 * If a property n/v pair was specified, warn the user if there was
5430fa9e4066Sahrens 	 * nothing to output.
5431fa9e4066Sahrens 	 */
5432fa9e4066Sahrens 	if (!output && cmd->cmd_prop_nv_pairs > 0)
5433fa9e4066Sahrens 		(void) printf(gettext("No such %s resource.\n"),
5434fa9e4066Sahrens 		    rt_to_str(RT_DATASET));
5435fa9e4066Sahrens }
5436fa9e4066Sahrens 
54370209230bSgjelinek static void
output_pset(FILE * fp,struct zone_psettab * psettab)54380209230bSgjelinek output_pset(FILE *fp, struct zone_psettab *psettab)
54390209230bSgjelinek {
54400209230bSgjelinek 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_DCPU));
54410209230bSgjelinek 	if (strcmp(psettab->zone_ncpu_min, psettab->zone_ncpu_max) == 0)
54420209230bSgjelinek 		(void) fprintf(fp, "\t%s: %s\n", pt_to_str(PT_NCPUS),
54430209230bSgjelinek 		    psettab->zone_ncpu_max);
54440209230bSgjelinek 	else
54450209230bSgjelinek 		(void) fprintf(fp, "\t%s: %s-%s\n", pt_to_str(PT_NCPUS),
54460209230bSgjelinek 		    psettab->zone_ncpu_min, psettab->zone_ncpu_max);
54470209230bSgjelinek 	if (psettab->zone_importance[0] != '\0')
54480209230bSgjelinek 		(void) fprintf(fp, "\t%s: %s\n", pt_to_str(PT_IMPORTANCE),
54490209230bSgjelinek 		    psettab->zone_importance);
54500209230bSgjelinek }
54510209230bSgjelinek 
54520209230bSgjelinek static void
info_pset(zone_dochandle_t handle,FILE * fp)54530209230bSgjelinek info_pset(zone_dochandle_t handle, FILE *fp)
54540209230bSgjelinek {
54550209230bSgjelinek 	struct zone_psettab lookup;
54560209230bSgjelinek 
54570209230bSgjelinek 	if (zonecfg_getpsetent(handle, &lookup) == Z_OK)
54580209230bSgjelinek 		output_pset(fp, &lookup);
54590209230bSgjelinek }
54600209230bSgjelinek 
54610209230bSgjelinek static void
output_pcap(FILE * fp)5462c97ad5cdSakolb output_pcap(FILE *fp)
5463c97ad5cdSakolb {
5464c97ad5cdSakolb 	uint64_t cap;
5465c97ad5cdSakolb 
5466c97ad5cdSakolb 	if (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &cap) == Z_OK) {
5467c97ad5cdSakolb 		float scaled = (float)cap / 100;
5468c97ad5cdSakolb 		(void) fprintf(fp, "%s:\n", rt_to_str(RT_PCAP));
5469c97ad5cdSakolb 		(void) fprintf(fp, "\t[%s: %.2f]\n", pt_to_str(PT_NCPUS),
5470c97ad5cdSakolb 		    scaled);
5471c97ad5cdSakolb 	}
5472c97ad5cdSakolb }
5473c97ad5cdSakolb 
5474c97ad5cdSakolb static void
info_pcap(FILE * fp)5475c97ad5cdSakolb info_pcap(FILE *fp)
5476c97ad5cdSakolb {
5477c97ad5cdSakolb 	output_pcap(fp);
5478c97ad5cdSakolb }
5479c97ad5cdSakolb 
5480c97ad5cdSakolb 
5481c97ad5cdSakolb static void
info_aliased_rctl(zone_dochandle_t handle,FILE * fp,char * alias)54820209230bSgjelinek info_aliased_rctl(zone_dochandle_t handle, FILE *fp, char *alias)
54830209230bSgjelinek {
54840209230bSgjelinek 	uint64_t limit;
54850209230bSgjelinek 
54860209230bSgjelinek 	if (zonecfg_get_aliased_rctl(handle, alias, &limit) == Z_OK) {
54870209230bSgjelinek 		/* convert memory based properties */
54880209230bSgjelinek 		if (strcmp(alias, ALIAS_MAXSHMMEM) == 0) {
54890209230bSgjelinek 			char buf[128];
54900209230bSgjelinek 
54910209230bSgjelinek 			(void) snprintf(buf, sizeof (buf), "%llu", limit);
54920209230bSgjelinek 			bytes_to_units(buf, buf, sizeof (buf));
54930209230bSgjelinek 			(void) fprintf(fp, "[%s: %s]\n", alias, buf);
54940209230bSgjelinek 			return;
54950209230bSgjelinek 		}
54960209230bSgjelinek 
54970209230bSgjelinek 		(void) fprintf(fp, "[%s: %llu]\n", alias, limit);
54980209230bSgjelinek 	}
54990209230bSgjelinek }
55000209230bSgjelinek 
55010209230bSgjelinek static void
bytes_to_units(char * str,char * buf,int bufsize)55020209230bSgjelinek bytes_to_units(char *str, char *buf, int bufsize)
55030209230bSgjelinek {
55040209230bSgjelinek 	unsigned long long num;
55050209230bSgjelinek 	unsigned long long save = 0;
55060209230bSgjelinek 	char *units = "BKMGT";
55070209230bSgjelinek 	char *up = units;
55080209230bSgjelinek 
55090209230bSgjelinek 	num = strtoll(str, NULL, 10);
55100209230bSgjelinek 
55110209230bSgjelinek 	if (num < 1024) {
55120209230bSgjelinek 		(void) snprintf(buf, bufsize, "%llu", num);
55130209230bSgjelinek 		return;
55140209230bSgjelinek 	}
55150209230bSgjelinek 
55160209230bSgjelinek 	while ((num >= 1024) && (*up != 'T')) {
55170209230bSgjelinek 		up++; /* next unit of measurement */
55180209230bSgjelinek 		save = num;
55190209230bSgjelinek 		num = (num + 512) >> 10;
55200209230bSgjelinek 	}
55210209230bSgjelinek 
55220209230bSgjelinek 	/* check if we should output a fraction.  snprintf will round for us */
55230209230bSgjelinek 	if (save % 1024 != 0 && ((save >> 10) < 10))
55240209230bSgjelinek 		(void) snprintf(buf, bufsize, "%2.1f%c", ((float)save / 1024),
55250209230bSgjelinek 		    *up);
55260209230bSgjelinek 	else
55270209230bSgjelinek 		(void) snprintf(buf, bufsize, "%llu%c", num, *up);
55280209230bSgjelinek }
55290209230bSgjelinek 
55300209230bSgjelinek static void
output_mcap(FILE * fp,struct zone_mcaptab * mcaptab,int showswap,uint64_t maxswap,int showlocked,uint64_t maxlocked)55310209230bSgjelinek output_mcap(FILE *fp, struct zone_mcaptab *mcaptab, int showswap,
55320209230bSgjelinek     uint64_t maxswap, int showlocked, uint64_t maxlocked)
55330209230bSgjelinek {
55340209230bSgjelinek 	char buf[128];
55350209230bSgjelinek 
55360209230bSgjelinek 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_MCAP));
55370209230bSgjelinek 	if (mcaptab->zone_physmem_cap[0] != '\0') {
55380209230bSgjelinek 		bytes_to_units(mcaptab->zone_physmem_cap, buf, sizeof (buf));
55390209230bSgjelinek 		output_prop(fp, PT_PHYSICAL, buf, B_TRUE);
55400209230bSgjelinek 	}
55410209230bSgjelinek 
55420209230bSgjelinek 	if (showswap == Z_OK) {
55430209230bSgjelinek 		(void) snprintf(buf, sizeof (buf), "%llu", maxswap);
55440209230bSgjelinek 		bytes_to_units(buf, buf, sizeof (buf));
55450209230bSgjelinek 		output_prop(fp, PT_SWAP, buf, B_TRUE);
55460209230bSgjelinek 	}
55470209230bSgjelinek 
55480209230bSgjelinek 	if (showlocked == Z_OK) {
55490209230bSgjelinek 		(void) snprintf(buf, sizeof (buf), "%llu", maxlocked);
55500209230bSgjelinek 		bytes_to_units(buf, buf, sizeof (buf));
55510209230bSgjelinek 		output_prop(fp, PT_LOCKED, buf, B_TRUE);
55520209230bSgjelinek 	}
55530209230bSgjelinek }
55540209230bSgjelinek 
55550209230bSgjelinek static void
info_mcap(zone_dochandle_t handle,FILE * fp)55560209230bSgjelinek info_mcap(zone_dochandle_t handle, FILE *fp)
55570209230bSgjelinek {
55580209230bSgjelinek 	int res1, res2, res3;
55590209230bSgjelinek 	uint64_t swap_limit;
55600209230bSgjelinek 	uint64_t locked_limit;
55610209230bSgjelinek 	struct zone_mcaptab lookup;
55620209230bSgjelinek 
55630209230bSgjelinek 	bzero(&lookup, sizeof (lookup));
55640209230bSgjelinek 	res1 = zonecfg_getmcapent(handle, &lookup);
55650209230bSgjelinek 	res2 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP, &swap_limit);
55660209230bSgjelinek 	res3 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM,
55670209230bSgjelinek 	    &locked_limit);
55680209230bSgjelinek 
55690209230bSgjelinek 	if (res1 == Z_OK || res2 == Z_OK || res3 == Z_OK)
55700209230bSgjelinek 		output_mcap(fp, &lookup, res2, swap_limit, res3, locked_limit);
55710209230bSgjelinek }
55720209230bSgjelinek 
5573cb8a054bSGlenn Faden static void
output_auth(FILE * fp,struct zone_admintab * admintab)5574cb8a054bSGlenn Faden output_auth(FILE *fp, struct zone_admintab *admintab)
5575cb8a054bSGlenn Faden {
5576cb8a054bSGlenn Faden 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_ADMIN));
5577cb8a054bSGlenn Faden 	output_prop(fp, PT_USER, admintab->zone_admin_user, B_TRUE);
5578cb8a054bSGlenn Faden 	output_prop(fp, PT_AUTHS, admintab->zone_admin_auths, B_TRUE);
5579cb8a054bSGlenn Faden }
5580cb8a054bSGlenn Faden 
5581cb8a054bSGlenn Faden static void
output_secflags(FILE * fp,struct zone_secflagstab * sftab)5582d2a70789SRichard Lowe output_secflags(FILE *fp, struct zone_secflagstab *sftab)
5583d2a70789SRichard Lowe {
5584d2a70789SRichard Lowe 	(void) fprintf(fp, "%s:\n", rt_to_str(RT_SECFLAGS));
5585d2a70789SRichard Lowe 	output_prop(fp, PT_DEFAULT, sftab->zone_secflags_default, B_TRUE);
5586d2a70789SRichard Lowe 	output_prop(fp, PT_LOWER, sftab->zone_secflags_lower, B_TRUE);
5587d2a70789SRichard Lowe 	output_prop(fp, PT_UPPER, sftab->zone_secflags_upper, B_TRUE);
5588d2a70789SRichard Lowe }
5589d2a70789SRichard Lowe 
5590d2a70789SRichard Lowe static void
info_auth(zone_dochandle_t handle,FILE * fp,cmd_t * cmd)5591cb8a054bSGlenn Faden info_auth(zone_dochandle_t handle, FILE *fp, cmd_t *cmd)
5592cb8a054bSGlenn Faden {
5593cb8a054bSGlenn Faden 	struct zone_admintab lookup, user;
5594cb8a054bSGlenn Faden 	boolean_t output = B_FALSE;
5595cb8a054bSGlenn Faden 	int err;
5596cb8a054bSGlenn Faden 
5597cb8a054bSGlenn Faden 	if ((err = zonecfg_setadminent(handle)) != Z_OK) {
5598cb8a054bSGlenn Faden 		zone_perror(zone, err, B_TRUE);
5599cb8a054bSGlenn Faden 		return;
5600cb8a054bSGlenn Faden 	}
5601cb8a054bSGlenn Faden 	while (zonecfg_getadminent(handle, &lookup) == Z_OK) {
5602cb8a054bSGlenn Faden 		if (cmd->cmd_prop_nv_pairs == 0) {
5603cb8a054bSGlenn Faden 			output_auth(fp, &lookup);
5604cb8a054bSGlenn Faden 			continue;
5605cb8a054bSGlenn Faden 		}
5606cb8a054bSGlenn Faden 		if (fill_in_admintab(cmd, &user, B_TRUE) != Z_OK)
5607cb8a054bSGlenn Faden 			continue;
5608cb8a054bSGlenn Faden 		if (strlen(user.zone_admin_user) > 0 &&
5609cb8a054bSGlenn Faden 		    strcmp(user.zone_admin_user, lookup.zone_admin_user) != 0)
5610cb8a054bSGlenn Faden 			continue;	/* no match */
5611cb8a054bSGlenn Faden 		output_auth(fp, &lookup);
5612cb8a054bSGlenn Faden 		output = B_TRUE;
5613cb8a054bSGlenn Faden 	}
5614cb8a054bSGlenn Faden 	(void) zonecfg_endadminent(handle);
5615cb8a054bSGlenn Faden 	/*
5616cb8a054bSGlenn Faden 	 * If a property n/v pair was specified, warn the user if there was
5617cb8a054bSGlenn Faden 	 * nothing to output.
5618cb8a054bSGlenn Faden 	 */
5619cb8a054bSGlenn Faden 	if (!output && cmd->cmd_prop_nv_pairs > 0)
5620cb8a054bSGlenn Faden 		(void) printf(gettext("No such %s resource.\n"),
5621cb8a054bSGlenn Faden 		    rt_to_str(RT_ADMIN));
5622cb8a054bSGlenn Faden }
5623cb8a054bSGlenn Faden 
5624d2a70789SRichard Lowe static void
info_secflags(zone_dochandle_t handle,FILE * fp)5625d2a70789SRichard Lowe info_secflags(zone_dochandle_t handle, FILE *fp)
5626d2a70789SRichard Lowe {
5627d2a70789SRichard Lowe 	struct zone_secflagstab sftab;
5628d2a70789SRichard Lowe 
5629d2a70789SRichard Lowe 	if (zonecfg_lookup_secflags(handle, &sftab) == Z_OK) {
5630d2a70789SRichard Lowe 		output_secflags(fp, &sftab);
5631d2a70789SRichard Lowe 	}
5632d2a70789SRichard Lowe }
5633d2a70789SRichard Lowe 
56347c478bd9Sstevel@tonic-gate void
info_func(cmd_t * cmd)56357c478bd9Sstevel@tonic-gate info_func(cmd_t *cmd)
56367c478bd9Sstevel@tonic-gate {
56377c478bd9Sstevel@tonic-gate 	FILE *fp = stdout;
5638bbec428eSgjelinek 	boolean_t need_to_close = B_FALSE;
56390209230bSgjelinek 	int type;
56400209230bSgjelinek 	int res1, res2;
56410209230bSgjelinek 	uint64_t swap_limit;
56420209230bSgjelinek 	uint64_t locked_limit;
56437c478bd9Sstevel@tonic-gate 
56447c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
56457c478bd9Sstevel@tonic-gate 
5646bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
56477c478bd9Sstevel@tonic-gate 		return;
56487c478bd9Sstevel@tonic-gate 
56497c478bd9Sstevel@tonic-gate 	/* don't page error output */
56507c478bd9Sstevel@tonic-gate 	if (interactive_mode) {
5651944b13ecSGary Mills 		if ((fp = pager_open()) != NULL)
5652bbec428eSgjelinek 			need_to_close = B_TRUE;
56533042b8b5Sbatschul 		else
56543042b8b5Sbatschul 			fp = stdout;
56553042b8b5Sbatschul 
56567c478bd9Sstevel@tonic-gate 		setbuf(fp, NULL);
56577c478bd9Sstevel@tonic-gate 	}
56587c478bd9Sstevel@tonic-gate 
56597c478bd9Sstevel@tonic-gate 	if (!global_scope) {
56607c478bd9Sstevel@tonic-gate 		switch (resource_scope) {
56617c478bd9Sstevel@tonic-gate 		case RT_FS:
56627c478bd9Sstevel@tonic-gate 			output_fs(fp, &in_progress_fstab);
56637c478bd9Sstevel@tonic-gate 			break;
56647c478bd9Sstevel@tonic-gate 		case RT_NET:
56657c478bd9Sstevel@tonic-gate 			output_net(fp, &in_progress_nwiftab);
56667c478bd9Sstevel@tonic-gate 			break;
56677c478bd9Sstevel@tonic-gate 		case RT_DEVICE:
56687c478bd9Sstevel@tonic-gate 			output_dev(fp, &in_progress_devtab);
56697c478bd9Sstevel@tonic-gate 			break;
56707c478bd9Sstevel@tonic-gate 		case RT_RCTL:
56717c478bd9Sstevel@tonic-gate 			output_rctl(fp, &in_progress_rctltab);
56727c478bd9Sstevel@tonic-gate 			break;
56737c478bd9Sstevel@tonic-gate 		case RT_ATTR:
56747c478bd9Sstevel@tonic-gate 			output_attr(fp, &in_progress_attrtab);
56757c478bd9Sstevel@tonic-gate 			break;
5676fa9e4066Sahrens 		case RT_DATASET:
5677fa9e4066Sahrens 			output_ds(fp, &in_progress_dstab);
5678fa9e4066Sahrens 			break;
56790209230bSgjelinek 		case RT_DCPU:
56800209230bSgjelinek 			output_pset(fp, &in_progress_psettab);
56810209230bSgjelinek 			break;
5682c97ad5cdSakolb 		case RT_PCAP:
5683c97ad5cdSakolb 			output_pcap(fp);
5684c97ad5cdSakolb 			break;
56850209230bSgjelinek 		case RT_MCAP:
56860209230bSgjelinek 			res1 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP,
56870209230bSgjelinek 			    &swap_limit);
56880209230bSgjelinek 			res2 = zonecfg_get_aliased_rctl(handle,
56890209230bSgjelinek 			    ALIAS_MAXLOCKEDMEM, &locked_limit);
56900209230bSgjelinek 			output_mcap(fp, &in_progress_mcaptab, res1, swap_limit,
56910209230bSgjelinek 			    res2, locked_limit);
56920209230bSgjelinek 			break;
5693cb8a054bSGlenn Faden 		case RT_ADMIN:
5694cb8a054bSGlenn Faden 			output_auth(fp, &in_progress_admintab);
5695cb8a054bSGlenn Faden 			break;
5696d2a70789SRichard Lowe 		case RT_SECFLAGS:
5697d2a70789SRichard Lowe 			output_secflags(fp, &in_progress_secflagstab);
5698d2a70789SRichard Lowe 			break;
56997c478bd9Sstevel@tonic-gate 		}
57007c478bd9Sstevel@tonic-gate 		goto cleanup;
57017c478bd9Sstevel@tonic-gate 	}
57027c478bd9Sstevel@tonic-gate 
57030209230bSgjelinek 	type = cmd->cmd_res_type;
57040209230bSgjelinek 
57050209230bSgjelinek 	if (gz_invalid_rt_property(type)) {
57060209230bSgjelinek 		zerr(gettext("%s is not a valid property for the global zone."),
57070209230bSgjelinek 		    rt_to_str(type));
57080209230bSgjelinek 		goto cleanup;
57090209230bSgjelinek 	}
57100209230bSgjelinek 
57110209230bSgjelinek 	if (gz_invalid_resource(type)) {
57120209230bSgjelinek 		zerr(gettext("%s is not a valid resource for the global zone."),
57130209230bSgjelinek 		    rt_to_str(type));
57140209230bSgjelinek 		goto cleanup;
57150209230bSgjelinek 	}
57160209230bSgjelinek 
57177c478bd9Sstevel@tonic-gate 	switch (cmd->cmd_res_type) {
57187c478bd9Sstevel@tonic-gate 	case RT_UNKNOWN:
5719087719fdSdp 		info_zonename(handle, fp);
57200209230bSgjelinek 		if (!global_zone) {
57217c478bd9Sstevel@tonic-gate 			info_zonepath(handle, fp);
57229acbbeafSnn35248 			info_brand(handle, fp);
57237c478bd9Sstevel@tonic-gate 			info_autoboot(handle, fp);
57243f2f09c1Sdp 			info_bootargs(handle, fp);
57250209230bSgjelinek 		}
57267c478bd9Sstevel@tonic-gate 		info_pool(handle, fp);
57270209230bSgjelinek 		if (!global_zone) {
5728ffbafc53Scomay 			info_limitpriv(handle, fp);
57290209230bSgjelinek 			info_sched(handle, fp);
5730f4b3ec61Sdh155122 			info_iptype(handle, fp);
57315679c89fSjv227347 			info_hostid(handle, fp);
57320fbb751dSJohn Levon 			info_fs_allowed(handle, fp);
57330209230bSgjelinek 		}
57340209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXLWPS);
5735ff19e029SMenno Lageman 		info_aliased_rctl(handle, fp, ALIAS_MAXPROCS);
57360209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSHMMEM);
57370209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSHMIDS);
57380209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXMSGIDS);
57390209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSEMIDS);
57400209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_SHARES);
57410209230bSgjelinek 		if (!global_zone) {
57427c478bd9Sstevel@tonic-gate 			info_fs(handle, fp, cmd);
57437c478bd9Sstevel@tonic-gate 			info_net(handle, fp, cmd);
57447c478bd9Sstevel@tonic-gate 			info_dev(handle, fp, cmd);
57450209230bSgjelinek 		}
57460209230bSgjelinek 		info_pset(handle, fp);
5747c97ad5cdSakolb 		info_pcap(fp);
57480209230bSgjelinek 		info_mcap(handle, fp);
57490209230bSgjelinek 		if (!global_zone) {
57507c478bd9Sstevel@tonic-gate 			info_attr(handle, fp, cmd);
5751fa9e4066Sahrens 			info_ds(handle, fp, cmd);
5752cb8a054bSGlenn Faden 			info_auth(handle, fp, cmd);
57530209230bSgjelinek 		}
57540209230bSgjelinek 		info_rctl(handle, fp, cmd);
5755d2a70789SRichard Lowe 		info_secflags(handle, fp);
57567c478bd9Sstevel@tonic-gate 		break;
5757087719fdSdp 	case RT_ZONENAME:
5758087719fdSdp 		info_zonename(handle, fp);
5759087719fdSdp 		break;
57607c478bd9Sstevel@tonic-gate 	case RT_ZONEPATH:
57617c478bd9Sstevel@tonic-gate 		info_zonepath(handle, fp);
57627c478bd9Sstevel@tonic-gate 		break;
57639acbbeafSnn35248 	case RT_BRAND:
57649acbbeafSnn35248 		info_brand(handle, fp);
57659acbbeafSnn35248 		break;
57667c478bd9Sstevel@tonic-gate 	case RT_AUTOBOOT:
57677c478bd9Sstevel@tonic-gate 		info_autoboot(handle, fp);
57687c478bd9Sstevel@tonic-gate 		break;
57697c478bd9Sstevel@tonic-gate 	case RT_POOL:
57707c478bd9Sstevel@tonic-gate 		info_pool(handle, fp);
57717c478bd9Sstevel@tonic-gate 		break;
5772ffbafc53Scomay 	case RT_LIMITPRIV:
5773ffbafc53Scomay 		info_limitpriv(handle, fp);
5774ffbafc53Scomay 		break;
57753f2f09c1Sdp 	case RT_BOOTARGS:
57763f2f09c1Sdp 		info_bootargs(handle, fp);
57773f2f09c1Sdp 		break;
57780209230bSgjelinek 	case RT_SCHED:
57790209230bSgjelinek 		info_sched(handle, fp);
57800209230bSgjelinek 		break;
5781f4b3ec61Sdh155122 	case RT_IPTYPE:
5782f4b3ec61Sdh155122 		info_iptype(handle, fp);
5783f4b3ec61Sdh155122 		break;
57840209230bSgjelinek 	case RT_MAXLWPS:
57850209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXLWPS);
57860209230bSgjelinek 		break;
5787ff19e029SMenno Lageman 	case RT_MAXPROCS:
5788ff19e029SMenno Lageman 		info_aliased_rctl(handle, fp, ALIAS_MAXPROCS);
5789ff19e029SMenno Lageman 		break;
57900209230bSgjelinek 	case RT_MAXSHMMEM:
57910209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSHMMEM);
57920209230bSgjelinek 		break;
57930209230bSgjelinek 	case RT_MAXSHMIDS:
57940209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSHMIDS);
57950209230bSgjelinek 		break;
57960209230bSgjelinek 	case RT_MAXMSGIDS:
57970209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXMSGIDS);
57980209230bSgjelinek 		break;
57990209230bSgjelinek 	case RT_MAXSEMIDS:
58000209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_MAXSEMIDS);
58010209230bSgjelinek 		break;
58020209230bSgjelinek 	case RT_SHARES:
58030209230bSgjelinek 		info_aliased_rctl(handle, fp, ALIAS_SHARES);
58040209230bSgjelinek 		break;
58057c478bd9Sstevel@tonic-gate 	case RT_FS:
58067c478bd9Sstevel@tonic-gate 		info_fs(handle, fp, cmd);
58077c478bd9Sstevel@tonic-gate 		break;
58087c478bd9Sstevel@tonic-gate 	case RT_NET:
58097c478bd9Sstevel@tonic-gate 		info_net(handle, fp, cmd);
58107c478bd9Sstevel@tonic-gate 		break;
58117c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
58127c478bd9Sstevel@tonic-gate 		info_dev(handle, fp, cmd);
58137c478bd9Sstevel@tonic-gate 		break;
58147c478bd9Sstevel@tonic-gate 	case RT_RCTL:
58157c478bd9Sstevel@tonic-gate 		info_rctl(handle, fp, cmd);
58167c478bd9Sstevel@tonic-gate 		break;
58177c478bd9Sstevel@tonic-gate 	case RT_ATTR:
58187c478bd9Sstevel@tonic-gate 		info_attr(handle, fp, cmd);
58197c478bd9Sstevel@tonic-gate 		break;
5820fa9e4066Sahrens 	case RT_DATASET:
5821fa9e4066Sahrens 		info_ds(handle, fp, cmd);
5822fa9e4066Sahrens 		break;
58230209230bSgjelinek 	case RT_DCPU:
58240209230bSgjelinek 		info_pset(handle, fp);
58250209230bSgjelinek 		break;
5826c97ad5cdSakolb 	case RT_PCAP:
5827c97ad5cdSakolb 		info_pcap(fp);
5828c97ad5cdSakolb 		break;
58290209230bSgjelinek 	case RT_MCAP:
58300209230bSgjelinek 		info_mcap(handle, fp);
58310209230bSgjelinek 		break;
58325679c89fSjv227347 	case RT_HOSTID:
58335679c89fSjv227347 		info_hostid(handle, fp);
58345679c89fSjv227347 		break;
5835cb8a054bSGlenn Faden 	case RT_ADMIN:
5836cb8a054bSGlenn Faden 		info_auth(handle, fp, cmd);
5837cb8a054bSGlenn Faden 		break;
58380fbb751dSJohn Levon 	case RT_FS_ALLOWED:
58390fbb751dSJohn Levon 		info_fs_allowed(handle, fp);
58400fbb751dSJohn Levon 		break;
5841d2a70789SRichard Lowe 	case RT_SECFLAGS:
5842d2a70789SRichard Lowe 		info_secflags(handle, fp);
5843d2a70789SRichard Lowe 		break;
58447c478bd9Sstevel@tonic-gate 	default:
58457c478bd9Sstevel@tonic-gate 		zone_perror(rt_to_str(cmd->cmd_res_type), Z_NO_RESOURCE_TYPE,
5846bbec428eSgjelinek 		    B_TRUE);
58477c478bd9Sstevel@tonic-gate 	}
58487c478bd9Sstevel@tonic-gate 
58497c478bd9Sstevel@tonic-gate cleanup:
58507c478bd9Sstevel@tonic-gate 	if (need_to_close)
5851944b13ecSGary Mills 		(void) pager_close(fp);
58527c478bd9Sstevel@tonic-gate }
58537c478bd9Sstevel@tonic-gate 
5854087719fdSdp /*
5855087719fdSdp  * Helper function for verify-- checks that a required string property
5856087719fdSdp  * exists.
5857087719fdSdp  */
5858087719fdSdp static void
check_reqd_prop(char * attr,int rt,int pt,int * ret_val)5859087719fdSdp check_reqd_prop(char *attr, int rt, int pt, int *ret_val)
58607c478bd9Sstevel@tonic-gate {
5861087719fdSdp 	if (strlen(attr) == 0) {
5862087719fdSdp 		zerr(gettext("%s: %s not specified"), rt_to_str(rt),
5863087719fdSdp 		    pt_to_str(pt));
5864bbec428eSgjelinek 		saw_error = B_TRUE;
5865087719fdSdp 		if (*ret_val == Z_OK)
5866087719fdSdp 			*ret_val = Z_REQD_PROPERTY_MISSING;
58677c478bd9Sstevel@tonic-gate 	}
58687c478bd9Sstevel@tonic-gate }
58697c478bd9Sstevel@tonic-gate 
58709acbbeafSnn35248 static int
do_subproc(char * cmdbuf)58719acbbeafSnn35248 do_subproc(char *cmdbuf)
58729acbbeafSnn35248 {
58739acbbeafSnn35248 	char inbuf[MAX_CMD_LEN];
58749acbbeafSnn35248 	FILE *file;
58759acbbeafSnn35248 	int status;
58769acbbeafSnn35248 
58779acbbeafSnn35248 	file = popen(cmdbuf, "r");
58789acbbeafSnn35248 	if (file == NULL) {
58799acbbeafSnn35248 		zerr(gettext("Could not launch: %s"), cmdbuf);
58809acbbeafSnn35248 		return (-1);
58819acbbeafSnn35248 	}
58829acbbeafSnn35248 
58839acbbeafSnn35248 	while (fgets(inbuf, sizeof (inbuf), file) != NULL)
58849acbbeafSnn35248 		fprintf(stderr, "%s", inbuf);
58859acbbeafSnn35248 	status = pclose(file);
58869acbbeafSnn35248 
58879acbbeafSnn35248 	if (WIFSIGNALED(status)) {
58889acbbeafSnn35248 		zerr(gettext("%s unexpectedly terminated due to signal %d"),
58899acbbeafSnn35248 		    cmdbuf, WTERMSIG(status));
58909acbbeafSnn35248 		return (-1);
58919acbbeafSnn35248 	}
58929acbbeafSnn35248 	assert(WIFEXITED(status));
58939acbbeafSnn35248 	return (WEXITSTATUS(status));
58949acbbeafSnn35248 }
58959acbbeafSnn35248 
58969acbbeafSnn35248 static int
brand_verify(zone_dochandle_t handle)58979acbbeafSnn35248 brand_verify(zone_dochandle_t handle)
58989acbbeafSnn35248 {
58996e65f9afSnn35248 	char xml_file[32];
59009acbbeafSnn35248 	char cmdbuf[MAX_CMD_LEN];
5901123807fbSedp 	brand_handle_t bh;
59029acbbeafSnn35248 	char brand[MAXNAMELEN];
59039acbbeafSnn35248 	int err;
59049acbbeafSnn35248 
59059acbbeafSnn35248 	if (zonecfg_get_brand(handle, brand, sizeof (brand)) != Z_OK) {
59069acbbeafSnn35248 		zerr("%s: %s\n", zone, gettext("could not get zone brand"));
59079acbbeafSnn35248 		return (Z_INVALID_DOCUMENT);
59089acbbeafSnn35248 	}
5909123807fbSedp 	if ((bh = brand_open(brand)) == NULL) {
59109acbbeafSnn35248 		zerr("%s: %s\n", zone, gettext("unknown brand."));
59119acbbeafSnn35248 		return (Z_INVALID_DOCUMENT);
59129acbbeafSnn35248 	}
59139acbbeafSnn35248 
59149acbbeafSnn35248 	/*
59159acbbeafSnn35248 	 * Fetch the verify command, if any, from the brand configuration
59169acbbeafSnn35248 	 * and build the command line to execute it.
59179acbbeafSnn35248 	 */
59189acbbeafSnn35248 	strcpy(cmdbuf, EXEC_PREFIX);
5919123807fbSedp 	err = brand_get_verify_cfg(bh, cmdbuf + EXEC_LEN,
59209acbbeafSnn35248 	    sizeof (cmdbuf) - (EXEC_LEN + (strlen(xml_file) + 1)));
5921123807fbSedp 	brand_close(bh);
59229acbbeafSnn35248 	if (err != Z_OK) {
59239acbbeafSnn35248 		zerr("%s: %s\n", zone,
59249acbbeafSnn35248 		    gettext("could not get brand verification command"));
59259acbbeafSnn35248 		return (Z_INVALID_DOCUMENT);
59269acbbeafSnn35248 	}
59279acbbeafSnn35248 
59289acbbeafSnn35248 	/*
59299acbbeafSnn35248 	 * If the brand doesn't provide a verification routine, we just
59309acbbeafSnn35248 	 * return success.
59319acbbeafSnn35248 	 */
59329acbbeafSnn35248 	if (strlen(cmdbuf) == EXEC_LEN)
59339acbbeafSnn35248 		return (Z_OK);
59349acbbeafSnn35248 
59359acbbeafSnn35248 	/*
59369acbbeafSnn35248 	 * Dump the current config information for this zone to a file.
59379acbbeafSnn35248 	 */
59386e65f9afSnn35248 	strcpy(xml_file, "/tmp/zonecfg_verify.XXXXXX");
5939eaf5a31cSToomas Soome 	if (mkstemp(xml_file) == -1)
59409acbbeafSnn35248 		return (Z_TEMP_FILE);
59419acbbeafSnn35248 	if ((err = zonecfg_verify_save(handle, xml_file)) != Z_OK) {
59429acbbeafSnn35248 		(void) unlink(xml_file);
59439acbbeafSnn35248 		return (err);
59449acbbeafSnn35248 	}
59459acbbeafSnn35248 
59469acbbeafSnn35248 	/*
59479acbbeafSnn35248 	 * Execute the verification command.
59489acbbeafSnn35248 	 */
59499acbbeafSnn35248 	if ((strlcat(cmdbuf, " ", MAX_CMD_LEN) >= MAX_CMD_LEN) ||
59509acbbeafSnn35248 	    (strlcat(cmdbuf, xml_file, MAX_CMD_LEN) >= MAX_CMD_LEN)) {
59519acbbeafSnn35248 		err = Z_BRAND_ERROR;
59529acbbeafSnn35248 	} else {
59539acbbeafSnn35248 		err = do_subproc(cmdbuf);
59549acbbeafSnn35248 	}
59559acbbeafSnn35248 
59569acbbeafSnn35248 	(void) unlink(xml_file);
59579acbbeafSnn35248 	return ((err == Z_OK) ? Z_OK : Z_BRAND_ERROR);
59589acbbeafSnn35248 }
59599acbbeafSnn35248 
59607c478bd9Sstevel@tonic-gate /*
5961*bbf21555SRichard Lowe  * Track the network interfaces listed in zonecfg(8) in a linked list
5962550b6e40SSowmini Varadhan  * so that we can later check that defrouter is specified for an exclusive IP
5963550b6e40SSowmini Varadhan  * zone if and only if at least one allowed-address has been specified.
5964550b6e40SSowmini Varadhan  */
5965550b6e40SSowmini Varadhan static boolean_t
add_nwif(struct zone_nwiftab * nwif)5966550b6e40SSowmini Varadhan add_nwif(struct zone_nwiftab *nwif)
5967550b6e40SSowmini Varadhan {
5968550b6e40SSowmini Varadhan 	struct xif *tmp;
5969550b6e40SSowmini Varadhan 
5970550b6e40SSowmini Varadhan 	for (tmp = xif; tmp != NULL; tmp = tmp->xif_next) {
5971550b6e40SSowmini Varadhan 		if (strcmp(tmp->xif_name, nwif->zone_nwif_physical) == 0) {
5972550b6e40SSowmini Varadhan 			if (strlen(nwif->zone_nwif_allowed_address) > 0)
5973550b6e40SSowmini Varadhan 				tmp->xif_has_address = B_TRUE;
5974550b6e40SSowmini Varadhan 			if (strlen(nwif->zone_nwif_defrouter) > 0)
5975550b6e40SSowmini Varadhan 				tmp->xif_has_defrouter = B_TRUE;
5976550b6e40SSowmini Varadhan 			return (B_TRUE);
5977550b6e40SSowmini Varadhan 		}
5978550b6e40SSowmini Varadhan 	}
5979550b6e40SSowmini Varadhan 
5980550b6e40SSowmini Varadhan 	tmp = malloc(sizeof (*tmp));
5981550b6e40SSowmini Varadhan 	if (tmp == NULL) {
5982550b6e40SSowmini Varadhan 		zerr(gettext("memory allocation failed for %s"),
5983550b6e40SSowmini Varadhan 		    nwif->zone_nwif_physical);
5984550b6e40SSowmini Varadhan 		return (B_FALSE);
5985550b6e40SSowmini Varadhan 	}
5986550b6e40SSowmini Varadhan 	strlcpy(tmp->xif_name, nwif->zone_nwif_physical,
5987550b6e40SSowmini Varadhan 	    sizeof (tmp->xif_name));
5988550b6e40SSowmini Varadhan 	tmp->xif_has_defrouter = (strlen(nwif->zone_nwif_defrouter) > 0);
5989550b6e40SSowmini Varadhan 	tmp->xif_has_address = (strlen(nwif->zone_nwif_allowed_address) > 0);
5990550b6e40SSowmini Varadhan 	tmp->xif_next = xif;
5991550b6e40SSowmini Varadhan 	xif = tmp;
5992550b6e40SSowmini Varadhan 	return (B_TRUE);
5993550b6e40SSowmini Varadhan }
5994550b6e40SSowmini Varadhan 
5995d2a70789SRichard Lowe boolean_t
verify_secflags(struct zone_secflagstab * tab)5996d2a70789SRichard Lowe verify_secflags(struct zone_secflagstab *tab)
5997d2a70789SRichard Lowe {
5998d2a70789SRichard Lowe 	secflagdelta_t def = {0};
5999d2a70789SRichard Lowe 	secflagdelta_t upper = {0};
6000d2a70789SRichard Lowe 	secflagdelta_t lower = {0};
6001d2a70789SRichard Lowe 	boolean_t def_set = B_FALSE;
6002d2a70789SRichard Lowe 	boolean_t upper_set = B_FALSE;
6003d2a70789SRichard Lowe 	boolean_t lower_set = B_FALSE;
6004d2a70789SRichard Lowe 	boolean_t ret = B_TRUE;
6005d2a70789SRichard Lowe 
6006d2a70789SRichard Lowe 	if (strlen(tab->zone_secflags_default) > 0) {
6007d2a70789SRichard Lowe 		def_set = B_TRUE;
6008d2a70789SRichard Lowe 		if (secflags_parse(NULL, tab->zone_secflags_default,
6009d2a70789SRichard Lowe 		    &def) == -1) {
6010d2a70789SRichard Lowe 			zerr(gettext("default security flags '%s' are invalid"),
6011d2a70789SRichard Lowe 			    tab->zone_secflags_default);
6012d2a70789SRichard Lowe 			ret = B_FALSE;
6013d2a70789SRichard Lowe 		}
6014d2a70789SRichard Lowe 	} else {
6015d2a70789SRichard Lowe 		secflags_zero(&def.psd_assign);
6016d2a70789SRichard Lowe 		def.psd_ass_active = B_TRUE;
6017d2a70789SRichard Lowe 	}
6018d2a70789SRichard Lowe 
6019d2a70789SRichard Lowe 	if (strlen(tab->zone_secflags_upper) > 0) {
6020d2a70789SRichard Lowe 		upper_set = B_TRUE;
6021d2a70789SRichard Lowe 		if (secflags_parse(NULL, tab->zone_secflags_upper,
6022d2a70789SRichard Lowe 		    &upper) == -1) {
6023d2a70789SRichard Lowe 			zerr(gettext("upper security flags '%s' are invalid"),
6024d2a70789SRichard Lowe 			    tab->zone_secflags_upper);
6025d2a70789SRichard Lowe 			ret = B_FALSE;
6026d2a70789SRichard Lowe 		}
6027d2a70789SRichard Lowe 	} else {
6028d2a70789SRichard Lowe 		secflags_fullset(&upper.psd_assign);
6029d2a70789SRichard Lowe 		upper.psd_ass_active = B_TRUE;
6030d2a70789SRichard Lowe 	}
6031d2a70789SRichard Lowe 
6032d2a70789SRichard Lowe 	if (strlen(tab->zone_secflags_lower) > 0) {
6033d2a70789SRichard Lowe 		lower_set = B_TRUE;
6034d2a70789SRichard Lowe 		if (secflags_parse(NULL, tab->zone_secflags_lower,
6035d2a70789SRichard Lowe 		    &lower) == -1) {
6036d2a70789SRichard Lowe 			zerr(gettext("lower security flags '%s' are invalid"),
6037d2a70789SRichard Lowe 			    tab->zone_secflags_lower);
6038d2a70789SRichard Lowe 			ret = B_FALSE;
6039d2a70789SRichard Lowe 		}
6040d2a70789SRichard Lowe 	} else {
6041d2a70789SRichard Lowe 		secflags_zero(&lower.psd_assign);
6042d2a70789SRichard Lowe 		lower.psd_ass_active = B_TRUE;
6043d2a70789SRichard Lowe 	}
6044d2a70789SRichard Lowe 
6045d2a70789SRichard Lowe 	if (def_set && !def.psd_ass_active) {
6046d2a70789SRichard Lowe 		zerr(gettext("only assignment of security flags is "
6047d2a70789SRichard Lowe 		    "allowed (default: %s)"), tab->zone_secflags_default);
6048d2a70789SRichard Lowe 	}
6049d2a70789SRichard Lowe 
6050d2a70789SRichard Lowe 	if (lower_set && !lower.psd_ass_active) {
6051d2a70789SRichard Lowe 		zerr(gettext("only assignment of security flags is "
6052d2a70789SRichard Lowe 		    "allowed (lower: %s)"), tab->zone_secflags_lower);
6053d2a70789SRichard Lowe 	}
6054d2a70789SRichard Lowe 
6055d2a70789SRichard Lowe 	if (upper_set && !upper.psd_ass_active) {
6056d2a70789SRichard Lowe 		zerr(gettext("only assignment of security flags is "
6057d2a70789SRichard Lowe 		    "allowed (upper: %s)"), tab->zone_secflags_upper);
6058d2a70789SRichard Lowe 	}
6059d2a70789SRichard Lowe 
6060d2a70789SRichard Lowe 	if (def.psd_assign & ~upper.psd_assign)	{ /* In default but not upper */
6061d2a70789SRichard Lowe 		zerr(gettext("default secflags must be within the "
6062d2a70789SRichard Lowe 		    "upper limit"));
6063d2a70789SRichard Lowe 		ret = B_FALSE;
6064d2a70789SRichard Lowe 	}
6065d2a70789SRichard Lowe 	if (lower.psd_assign & ~def.psd_assign) { /* In lower but not default */
6066d2a70789SRichard Lowe 		zerr(gettext("default secflags must be above the lower limit"));
6067d2a70789SRichard Lowe 		ret = B_FALSE;
6068d2a70789SRichard Lowe 	}
6069d2a70789SRichard Lowe 	if (lower.psd_assign & ~upper.psd_assign) { /* In lower but not upper */
6070d2a70789SRichard Lowe 		zerr(gettext("lower secflags must be within the upper limit"));
6071d2a70789SRichard Lowe 		ret = B_FALSE;
6072d2a70789SRichard Lowe 	}
6073d2a70789SRichard Lowe 
6074d2a70789SRichard Lowe 	return (ret);
6075d2a70789SRichard Lowe }
6076d2a70789SRichard Lowe 
6077550b6e40SSowmini Varadhan /*
60787c478bd9Sstevel@tonic-gate  * See the DTD for which attributes are required for which resources.
60797c478bd9Sstevel@tonic-gate  *
60807c478bd9Sstevel@tonic-gate  * This function can be called by commit_func(), which needs to save things,
60817c478bd9Sstevel@tonic-gate  * in addition to the general call from parse_and_run(), which doesn't need
60827c478bd9Sstevel@tonic-gate  * things saved.  Since the parameters are standardized, we distinguish by
60837c478bd9Sstevel@tonic-gate  * having commit_func() call here with cmd->cmd_arg set to "save" to indicate
60847c478bd9Sstevel@tonic-gate  * that a save is needed.
60857c478bd9Sstevel@tonic-gate  */
60867c478bd9Sstevel@tonic-gate void
verify_func(cmd_t * cmd)60877c478bd9Sstevel@tonic-gate verify_func(cmd_t *cmd)
60887c478bd9Sstevel@tonic-gate {
60897c478bd9Sstevel@tonic-gate 	struct zone_nwiftab nwiftab;
60907c478bd9Sstevel@tonic-gate 	struct zone_fstab fstab;
60917c478bd9Sstevel@tonic-gate 	struct zone_attrtab attrtab;
60927c478bd9Sstevel@tonic-gate 	struct zone_rctltab rctltab;
6093fa9e4066Sahrens 	struct zone_dstab dstab;
60940209230bSgjelinek 	struct zone_psettab psettab;
6095cb8a054bSGlenn Faden 	struct zone_admintab admintab;
6096d2a70789SRichard Lowe 	struct zone_secflagstab secflagstab;
60977c478bd9Sstevel@tonic-gate 	char zonepath[MAXPATHLEN];
60980209230bSgjelinek 	char sched[MAXNAMELEN];
60999acbbeafSnn35248 	char brand[MAXNAMELEN];
61005679c89fSjv227347 	char hostidp[HW_HOSTID_LEN];
61010fbb751dSJohn Levon 	char fsallowedp[ZONE_FS_ALLOWED_MAX];
6102f93d2c19SAlexander Eremin 	priv_set_t *privs;
6103f93d2c19SAlexander Eremin 	char *privname = NULL;
61047c478bd9Sstevel@tonic-gate 	int err, ret_val = Z_OK, arg;
6105c97ad5cdSakolb 	int pset_res;
6106bbec428eSgjelinek 	boolean_t save = B_FALSE;
6107bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
6108f4b3ec61Sdh155122 	zone_iptype_t iptype;
61090209230bSgjelinek 	boolean_t has_cpu_shares = B_FALSE;
6110c97ad5cdSakolb 	boolean_t has_cpu_cap = B_FALSE;
6111550b6e40SSowmini Varadhan 	struct xif *tmp;
61127c478bd9Sstevel@tonic-gate 
61137c478bd9Sstevel@tonic-gate 	optind = 0;
61147ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
61157c478bd9Sstevel@tonic-gate 		switch (arg) {
61167c478bd9Sstevel@tonic-gate 		case '?':
61177c478bd9Sstevel@tonic-gate 			longer_usage(CMD_VERIFY);
6118bbec428eSgjelinek 			arg_err = B_TRUE;
61197ec75eb8Sgjelinek 			break;
61207c478bd9Sstevel@tonic-gate 		default:
61217c478bd9Sstevel@tonic-gate 			short_usage(CMD_VERIFY);
6122bbec428eSgjelinek 			arg_err = B_TRUE;
61237ec75eb8Sgjelinek 			break;
61247ec75eb8Sgjelinek 		}
61257ec75eb8Sgjelinek 	}
61267ec75eb8Sgjelinek 	if (arg_err)
61277c478bd9Sstevel@tonic-gate 		return;
61287ec75eb8Sgjelinek 
61297c478bd9Sstevel@tonic-gate 	if (optind > cmd->cmd_argc) {
61307c478bd9Sstevel@tonic-gate 		short_usage(CMD_VERIFY);
61317c478bd9Sstevel@tonic-gate 		return;
61327c478bd9Sstevel@tonic-gate 	}
61337c478bd9Sstevel@tonic-gate 
61347c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_VERIFY))
61357c478bd9Sstevel@tonic-gate 		return;
61367c478bd9Sstevel@tonic-gate 
61377c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
61387c478bd9Sstevel@tonic-gate 
61397c478bd9Sstevel@tonic-gate 	if (cmd->cmd_argc > 0 && (strcmp(cmd->cmd_argv[0], "save") == 0))
6140bbec428eSgjelinek 		save = B_TRUE;
6141bbec428eSgjelinek 	if (initialize(B_TRUE) != Z_OK)
61427c478bd9Sstevel@tonic-gate 		return;
61437c478bd9Sstevel@tonic-gate 
61440209230bSgjelinek 	if (zonecfg_get_zonepath(handle, zonepath, sizeof (zonepath)) != Z_OK &&
61450209230bSgjelinek 	    !global_zone) {
6146087719fdSdp 		zerr(gettext("%s not specified"), pt_to_str(PT_ZONEPATH));
61477c478bd9Sstevel@tonic-gate 		ret_val = Z_REQD_RESOURCE_MISSING;
6148bbec428eSgjelinek 		saw_error = B_TRUE;
61497c478bd9Sstevel@tonic-gate 	}
61500209230bSgjelinek 	if (strlen(zonepath) == 0 && !global_zone) {
6151087719fdSdp 		zerr(gettext("%s cannot be empty."), pt_to_str(PT_ZONEPATH));
61527c478bd9Sstevel@tonic-gate 		ret_val = Z_REQD_RESOURCE_MISSING;
6153bbec428eSgjelinek 		saw_error = B_TRUE;
61547c478bd9Sstevel@tonic-gate 	}
61557c478bd9Sstevel@tonic-gate 
61569acbbeafSnn35248 	if ((err = zonecfg_get_brand(handle, brand, sizeof (brand))) != Z_OK) {
6157bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
61589acbbeafSnn35248 		return;
61599acbbeafSnn35248 	}
61609acbbeafSnn35248 	if ((err = brand_verify(handle)) != Z_OK) {
6161bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
61629acbbeafSnn35248 		return;
61639acbbeafSnn35248 	}
61649acbbeafSnn35248 
6165f4b3ec61Sdh155122 	if (zonecfg_get_iptype(handle, &iptype) != Z_OK) {
6166f4b3ec61Sdh155122 		zerr("%s %s", gettext("cannot get"), pt_to_str(PT_IPTYPE));
6167f4b3ec61Sdh155122 		ret_val = Z_REQD_RESOURCE_MISSING;
6168bbec428eSgjelinek 		saw_error = B_TRUE;
6169f4b3ec61Sdh155122 	}
61707c478bd9Sstevel@tonic-gate 
6171f93d2c19SAlexander Eremin 	if ((privs = priv_allocset()) == NULL) {
6172f93d2c19SAlexander Eremin 		zerr(gettext("%s: priv_allocset failed"), zone);
6173f93d2c19SAlexander Eremin 		return;
6174f93d2c19SAlexander Eremin 	}
6175f93d2c19SAlexander Eremin 	if (zonecfg_get_privset(handle, privs, &privname) != Z_OK) {
6176f93d2c19SAlexander Eremin 		zerr(gettext("%s: invalid privilege: %s"), zone, privname);
6177f93d2c19SAlexander Eremin 		priv_freeset(privs);
6178f93d2c19SAlexander Eremin 		free(privname);
6179f93d2c19SAlexander Eremin 		return;
6180f93d2c19SAlexander Eremin 	}
6181f93d2c19SAlexander Eremin 	priv_freeset(privs);
6182f93d2c19SAlexander Eremin 
61830fbb751dSJohn Levon 	if (zonecfg_get_hostid(handle, hostidp,
61840fbb751dSJohn Levon 	    sizeof (hostidp)) == Z_INVALID_PROPERTY) {
61850fbb751dSJohn Levon 		zerr(gettext("%s: invalid hostid: %s"),
61860fbb751dSJohn Levon 		    zone, hostidp);
61870fbb751dSJohn Levon 		return;
61880fbb751dSJohn Levon 	}
61890fbb751dSJohn Levon 
61900fbb751dSJohn Levon 	if (zonecfg_get_fs_allowed(handle, fsallowedp,
61910fbb751dSJohn Levon 	    sizeof (fsallowedp)) == Z_INVALID_PROPERTY) {
61920fbb751dSJohn Levon 		zerr(gettext("%s: invalid fs-allowed: %s"),
61930fbb751dSJohn Levon 		    zone, fsallowedp);
61945679c89fSjv227347 		return;
61955679c89fSjv227347 	}
61965679c89fSjv227347 
61977c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setfsent(handle)) != Z_OK) {
6198bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
61997c478bd9Sstevel@tonic-gate 		return;
62007c478bd9Sstevel@tonic-gate 	}
62017c478bd9Sstevel@tonic-gate 	while (zonecfg_getfsent(handle, &fstab) == Z_OK) {
6202087719fdSdp 		check_reqd_prop(fstab.zone_fs_dir, RT_FS, PT_DIR, &ret_val);
6203087719fdSdp 		check_reqd_prop(fstab.zone_fs_special, RT_FS, PT_SPECIAL,
6204087719fdSdp 		    &ret_val);
6205087719fdSdp 		check_reqd_prop(fstab.zone_fs_type, RT_FS, PT_TYPE, &ret_val);
6206087719fdSdp 
62077c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(fstab.zone_fs_options);
62087c478bd9Sstevel@tonic-gate 	}
62097c478bd9Sstevel@tonic-gate 	(void) zonecfg_endfsent(handle);
62107c478bd9Sstevel@tonic-gate 
62117c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setnwifent(handle)) != Z_OK) {
6212bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
62137c478bd9Sstevel@tonic-gate 		return;
62147c478bd9Sstevel@tonic-gate 	}
62157c478bd9Sstevel@tonic-gate 	while (zonecfg_getnwifent(handle, &nwiftab) == Z_OK) {
6216f4b3ec61Sdh155122 		/*
6217f4b3ec61Sdh155122 		 * physical is required in all cases.
6218de860bd9Sgfaden 		 * A shared IP requires an address,
6219de860bd9Sgfaden 		 * and may include a default router, while
6220de860bd9Sgfaden 		 * an exclusive IP must have neither an address
6221de860bd9Sgfaden 		 * nor a default router.
622201b4bc23Sjv227347 		 * The physical interface name must be valid in all cases.
6223f4b3ec61Sdh155122 		 */
6224087719fdSdp 		check_reqd_prop(nwiftab.zone_nwif_physical, RT_NET,
6225087719fdSdp 		    PT_PHYSICAL, &ret_val);
622601b4bc23Sjv227347 		if (validate_net_physical_syntax(nwiftab.zone_nwif_physical) !=
622701b4bc23Sjv227347 		    Z_OK) {
622801b4bc23Sjv227347 			saw_error = B_TRUE;
622901b4bc23Sjv227347 			if (ret_val == Z_OK)
623001b4bc23Sjv227347 				ret_val = Z_INVAL;
623101b4bc23Sjv227347 		}
6232f4b3ec61Sdh155122 
6233f4b3ec61Sdh155122 		switch (iptype) {
6234f4b3ec61Sdh155122 		case ZS_SHARED:
6235f4b3ec61Sdh155122 			check_reqd_prop(nwiftab.zone_nwif_address, RT_NET,
6236f4b3ec61Sdh155122 			    PT_ADDRESS, &ret_val);
6237550b6e40SSowmini Varadhan 			if (strlen(nwiftab.zone_nwif_allowed_address) > 0) {
6238550b6e40SSowmini Varadhan 				zerr(gettext("%s: %s cannot be specified "
6239550b6e40SSowmini Varadhan 				    "for a shared IP type"),
6240550b6e40SSowmini Varadhan 				    rt_to_str(RT_NET),
6241550b6e40SSowmini Varadhan 				    pt_to_str(PT_ALLOWED_ADDRESS));
6242550b6e40SSowmini Varadhan 				saw_error = B_TRUE;
6243550b6e40SSowmini Varadhan 				if (ret_val == Z_OK)
6244550b6e40SSowmini Varadhan 					ret_val = Z_INVAL;
6245550b6e40SSowmini Varadhan 			}
6246f4b3ec61Sdh155122 			break;
6247f4b3ec61Sdh155122 		case ZS_EXCLUSIVE:
6248f4b3ec61Sdh155122 			if (strlen(nwiftab.zone_nwif_address) > 0) {
6249f4b3ec61Sdh155122 				zerr(gettext("%s: %s cannot be specified "
6250f4b3ec61Sdh155122 				    "for an exclusive IP type"),
6251f4b3ec61Sdh155122 				    rt_to_str(RT_NET), pt_to_str(PT_ADDRESS));
6252bbec428eSgjelinek 				saw_error = B_TRUE;
6253f4b3ec61Sdh155122 				if (ret_val == Z_OK)
6254f4b3ec61Sdh155122 					ret_val = Z_INVAL;
6255550b6e40SSowmini Varadhan 			} else {
6256550b6e40SSowmini Varadhan 				if (!add_nwif(&nwiftab)) {
6257bbec428eSgjelinek 					saw_error = B_TRUE;
6258de860bd9Sgfaden 					if (ret_val == Z_OK)
6259de860bd9Sgfaden 						ret_val = Z_INVAL;
6260de860bd9Sgfaden 				}
6261550b6e40SSowmini Varadhan 			}
6262f4b3ec61Sdh155122 			break;
6263f4b3ec61Sdh155122 		}
62647c478bd9Sstevel@tonic-gate 	}
6265550b6e40SSowmini Varadhan 	for (tmp = xif; tmp != NULL; tmp = tmp->xif_next) {
6266550b6e40SSowmini Varadhan 		if (!tmp->xif_has_address && tmp->xif_has_defrouter) {
6267550b6e40SSowmini Varadhan 			zerr(gettext("%s: %s for %s cannot be specified "
6268550b6e40SSowmini Varadhan 			    "without %s for an exclusive IP type"),
6269550b6e40SSowmini Varadhan 			    rt_to_str(RT_NET), pt_to_str(PT_DEFROUTER),
6270550b6e40SSowmini Varadhan 			    tmp->xif_name, pt_to_str(PT_ALLOWED_ADDRESS));
6271550b6e40SSowmini Varadhan 			saw_error = B_TRUE;
6272550b6e40SSowmini Varadhan 			ret_val = Z_INVAL;
6273550b6e40SSowmini Varadhan 		}
6274550b6e40SSowmini Varadhan 	}
6275550b6e40SSowmini Varadhan 	free(xif);
6276550b6e40SSowmini Varadhan 	xif = NULL;
62777c478bd9Sstevel@tonic-gate 	(void) zonecfg_endnwifent(handle);
62787c478bd9Sstevel@tonic-gate 
62797c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setrctlent(handle)) != Z_OK) {
6280bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
62817c478bd9Sstevel@tonic-gate 		return;
62827c478bd9Sstevel@tonic-gate 	}
62837c478bd9Sstevel@tonic-gate 	while (zonecfg_getrctlent(handle, &rctltab) == Z_OK) {
6284087719fdSdp 		check_reqd_prop(rctltab.zone_rctl_name, RT_RCTL, PT_NAME,
6285087719fdSdp 		    &ret_val);
6286087719fdSdp 
62870209230bSgjelinek 		if (strcmp(rctltab.zone_rctl_name, "zone.cpu-shares") == 0)
62880209230bSgjelinek 			has_cpu_shares = B_TRUE;
62890209230bSgjelinek 
6290c97ad5cdSakolb 		if (strcmp(rctltab.zone_rctl_name, "zone.cpu-cap") == 0)
6291c97ad5cdSakolb 			has_cpu_cap = B_TRUE;
6292c97ad5cdSakolb 
62937c478bd9Sstevel@tonic-gate 		if (rctltab.zone_rctl_valptr == NULL) {
62947c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: no %s specified"),
62957c478bd9Sstevel@tonic-gate 			    rt_to_str(RT_RCTL), pt_to_str(PT_VALUE));
6296bbec428eSgjelinek 			saw_error = B_TRUE;
62977c478bd9Sstevel@tonic-gate 			if (ret_val == Z_OK)
62987c478bd9Sstevel@tonic-gate 				ret_val = Z_REQD_PROPERTY_MISSING;
62997c478bd9Sstevel@tonic-gate 		} else {
63007c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(rctltab.zone_rctl_valptr);
63017c478bd9Sstevel@tonic-gate 		}
63027c478bd9Sstevel@tonic-gate 	}
63037c478bd9Sstevel@tonic-gate 	(void) zonecfg_endrctlent(handle);
63047c478bd9Sstevel@tonic-gate 
6305c97ad5cdSakolb 	if ((pset_res = zonecfg_lookup_pset(handle, &psettab)) == Z_OK &&
6306c97ad5cdSakolb 	    has_cpu_shares) {
63070209230bSgjelinek 		zerr(gettext("%s zone.cpu-shares and %s are incompatible."),
63080209230bSgjelinek 		    rt_to_str(RT_RCTL), rt_to_str(RT_DCPU));
6309bbec428eSgjelinek 		saw_error = B_TRUE;
63100209230bSgjelinek 		if (ret_val == Z_OK)
63110209230bSgjelinek 			ret_val = Z_INCOMPATIBLE;
63120209230bSgjelinek 	}
63130209230bSgjelinek 
63140209230bSgjelinek 	if (has_cpu_shares && zonecfg_get_sched_class(handle, sched,
63150209230bSgjelinek 	    sizeof (sched)) == Z_OK && strlen(sched) > 0 &&
63160209230bSgjelinek 	    strcmp(sched, "FSS") != 0) {
63170209230bSgjelinek 		zerr(gettext("WARNING: %s zone.cpu-shares and %s=%s are "
63180209230bSgjelinek 		    "incompatible"),
63190209230bSgjelinek 		    rt_to_str(RT_RCTL), rt_to_str(RT_SCHED), sched);
6320bbec428eSgjelinek 		saw_error = B_TRUE;
63210209230bSgjelinek 		if (ret_val == Z_OK)
63220209230bSgjelinek 			ret_val = Z_INCOMPATIBLE;
63230209230bSgjelinek 	}
63240209230bSgjelinek 
6325c97ad5cdSakolb 	if (pset_res == Z_OK && has_cpu_cap) {
6326c97ad5cdSakolb 		zerr(gettext("%s zone.cpu-cap and the %s are incompatible."),
6327c97ad5cdSakolb 		    rt_to_str(RT_RCTL), rt_to_str(RT_DCPU));
6328bbec428eSgjelinek 		saw_error = B_TRUE;
6329c97ad5cdSakolb 		if (ret_val == Z_OK)
6330c97ad5cdSakolb 			ret_val = Z_INCOMPATIBLE;
6331c97ad5cdSakolb 	}
6332c97ad5cdSakolb 
63337c478bd9Sstevel@tonic-gate 	if ((err = zonecfg_setattrent(handle)) != Z_OK) {
6334bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
63357c478bd9Sstevel@tonic-gate 		return;
63367c478bd9Sstevel@tonic-gate 	}
63377c478bd9Sstevel@tonic-gate 	while (zonecfg_getattrent(handle, &attrtab) == Z_OK) {
6338087719fdSdp 		check_reqd_prop(attrtab.zone_attr_name, RT_ATTR, PT_NAME,
6339087719fdSdp 		    &ret_val);
6340087719fdSdp 		check_reqd_prop(attrtab.zone_attr_type, RT_ATTR, PT_TYPE,
6341087719fdSdp 		    &ret_val);
6342087719fdSdp 		check_reqd_prop(attrtab.zone_attr_value, RT_ATTR, PT_VALUE,
6343087719fdSdp 		    &ret_val);
63447c478bd9Sstevel@tonic-gate 	}
63457c478bd9Sstevel@tonic-gate 	(void) zonecfg_endattrent(handle);
63467c478bd9Sstevel@tonic-gate 
6347fa9e4066Sahrens 	if ((err = zonecfg_setdsent(handle)) != Z_OK) {
6348bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
6349fa9e4066Sahrens 		return;
6350fa9e4066Sahrens 	}
6351fa9e4066Sahrens 	while (zonecfg_getdsent(handle, &dstab) == Z_OK) {
6352fa9e4066Sahrens 		if (strlen(dstab.zone_dataset_name) == 0) {
6353fa9e4066Sahrens 			zerr("%s: %s %s", rt_to_str(RT_DATASET),
6354fa9e4066Sahrens 			    pt_to_str(PT_NAME), gettext("not specified"));
6355bbec428eSgjelinek 			saw_error = B_TRUE;
6356fa9e4066Sahrens 			if (ret_val == Z_OK)
6357fa9e4066Sahrens 				ret_val = Z_REQD_PROPERTY_MISSING;
6358fa9e4066Sahrens 		} else if (!zfs_name_valid(dstab.zone_dataset_name,
6359fa9e4066Sahrens 		    ZFS_TYPE_FILESYSTEM)) {
6360fa9e4066Sahrens 			zerr("%s: %s %s", rt_to_str(RT_DATASET),
6361fa9e4066Sahrens 			    pt_to_str(PT_NAME), gettext("invalid"));
6362bbec428eSgjelinek 			saw_error = B_TRUE;
6363fa9e4066Sahrens 			if (ret_val == Z_OK)
6364fa9e4066Sahrens 				ret_val = Z_BAD_PROPERTY;
6365fa9e4066Sahrens 		}
6366fa9e4066Sahrens 
6367fa9e4066Sahrens 	}
6368fa9e4066Sahrens 	(void) zonecfg_enddsent(handle);
6369fa9e4066Sahrens 
6370cb8a054bSGlenn Faden 	if ((err = zonecfg_setadminent(handle)) != Z_OK) {
6371cb8a054bSGlenn Faden 		zone_perror(zone, err, B_TRUE);
6372cb8a054bSGlenn Faden 		return;
6373cb8a054bSGlenn Faden 	}
6374cb8a054bSGlenn Faden 	while (zonecfg_getadminent(handle, &admintab) == Z_OK) {
6375cb8a054bSGlenn Faden 		check_reqd_prop(admintab.zone_admin_user, RT_ADMIN,
6376cb8a054bSGlenn Faden 		    PT_USER, &ret_val);
6377cb8a054bSGlenn Faden 		check_reqd_prop(admintab.zone_admin_auths, RT_ADMIN,
6378cb8a054bSGlenn Faden 		    PT_AUTHS, &ret_val);
6379cb8a054bSGlenn Faden 		if ((ret_val == Z_OK) && (getpwnam(admintab.zone_admin_user)
6380cb8a054bSGlenn Faden 		    == NULL)) {
6381cb8a054bSGlenn Faden 			zerr(gettext("%s %s is not a valid username"),
6382cb8a054bSGlenn Faden 			    pt_to_str(PT_USER),
6383cb8a054bSGlenn Faden 			    admintab.zone_admin_user);
6384cb8a054bSGlenn Faden 			ret_val = Z_BAD_PROPERTY;
6385cb8a054bSGlenn Faden 		}
6386cb8a054bSGlenn Faden 		if ((ret_val == Z_OK) && (!zonecfg_valid_auths(
6387cb8a054bSGlenn Faden 		    admintab.zone_admin_auths, zone))) {
6388cb8a054bSGlenn Faden 			ret_val = Z_BAD_PROPERTY;
6389cb8a054bSGlenn Faden 		}
6390cb8a054bSGlenn Faden 	}
6391cb8a054bSGlenn Faden 	(void) zonecfg_endadminent(handle);
6392cb8a054bSGlenn Faden 
6393d2a70789SRichard Lowe 	if (zonecfg_getsecflagsent(handle, &secflagstab) == Z_OK) {
6394d2a70789SRichard Lowe 		/*
6395d2a70789SRichard Lowe 		 * No properties are required, but any specified should be
6396d2a70789SRichard Lowe 		 * valid
6397d2a70789SRichard Lowe 		 */
6398d2a70789SRichard Lowe 		if (verify_secflags(&secflagstab) != B_TRUE) {
6399d2a70789SRichard Lowe 			/* Error is reported from verify_secflags */
6400d2a70789SRichard Lowe 			ret_val = Z_BAD_PROPERTY;
6401d2a70789SRichard Lowe 		}
6402d2a70789SRichard Lowe 	}
6403d2a70789SRichard Lowe 
64047c478bd9Sstevel@tonic-gate 	if (!global_scope) {
64057c478bd9Sstevel@tonic-gate 		zerr(gettext("resource specification incomplete"));
6406bbec428eSgjelinek 		saw_error = B_TRUE;
64077c478bd9Sstevel@tonic-gate 		if (ret_val == Z_OK)
64087c478bd9Sstevel@tonic-gate 			ret_val = Z_INSUFFICIENT_SPEC;
64097c478bd9Sstevel@tonic-gate 	}
64107c478bd9Sstevel@tonic-gate 
64117c478bd9Sstevel@tonic-gate 	if (save) {
6412087719fdSdp 		if (ret_val == Z_OK) {
6413087719fdSdp 			if ((ret_val = zonecfg_save(handle)) == Z_OK) {
6414bbec428eSgjelinek 				need_to_commit = B_FALSE;
6415087719fdSdp 				(void) strlcpy(revert_zone, zone,
6416087719fdSdp 				    sizeof (revert_zone));
6417087719fdSdp 			}
6418087719fdSdp 		} else {
6419087719fdSdp 			zerr(gettext("Zone %s failed to verify"), zone);
6420087719fdSdp 		}
64217c478bd9Sstevel@tonic-gate 	}
64227c478bd9Sstevel@tonic-gate 	if (ret_val != Z_OK)
6423bbec428eSgjelinek 		zone_perror(zone, ret_val, B_TRUE);
64247c478bd9Sstevel@tonic-gate }
64257c478bd9Sstevel@tonic-gate 
64267c478bd9Sstevel@tonic-gate void
cancel_func(cmd_t * cmd)64277c478bd9Sstevel@tonic-gate cancel_func(cmd_t *cmd)
64287c478bd9Sstevel@tonic-gate {
64297c478bd9Sstevel@tonic-gate 	int arg;
6430bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
64317c478bd9Sstevel@tonic-gate 
64327c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
64337c478bd9Sstevel@tonic-gate 
64347c478bd9Sstevel@tonic-gate 	optind = 0;
64357ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
64367c478bd9Sstevel@tonic-gate 		switch (arg) {
64377c478bd9Sstevel@tonic-gate 		case '?':
64387c478bd9Sstevel@tonic-gate 			longer_usage(CMD_CANCEL);
6439bbec428eSgjelinek 			arg_err = B_TRUE;
64407ec75eb8Sgjelinek 			break;
64417c478bd9Sstevel@tonic-gate 		default:
64427c478bd9Sstevel@tonic-gate 			short_usage(CMD_CANCEL);
6443bbec428eSgjelinek 			arg_err = B_TRUE;
64447ec75eb8Sgjelinek 			break;
64457ec75eb8Sgjelinek 		}
64467ec75eb8Sgjelinek 	}
64477ec75eb8Sgjelinek 	if (arg_err)
64487c478bd9Sstevel@tonic-gate 		return;
64497ec75eb8Sgjelinek 
64507c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
64517c478bd9Sstevel@tonic-gate 		short_usage(CMD_CANCEL);
64527c478bd9Sstevel@tonic-gate 		return;
64537c478bd9Sstevel@tonic-gate 	}
64547c478bd9Sstevel@tonic-gate 
64557c478bd9Sstevel@tonic-gate 	if (global_scope)
64567c478bd9Sstevel@tonic-gate 		scope_usage(CMD_CANCEL);
6457bbec428eSgjelinek 	global_scope = B_TRUE;
64587c478bd9Sstevel@tonic-gate 	zonecfg_free_fs_option_list(in_progress_fstab.zone_fs_options);
64597c478bd9Sstevel@tonic-gate 	bzero(&in_progress_fstab, sizeof (in_progress_fstab));
64607c478bd9Sstevel@tonic-gate 	bzero(&in_progress_nwiftab, sizeof (in_progress_nwiftab));
64617c478bd9Sstevel@tonic-gate 	bzero(&in_progress_devtab, sizeof (in_progress_devtab));
64627c478bd9Sstevel@tonic-gate 	zonecfg_free_rctl_value_list(in_progress_rctltab.zone_rctl_valptr);
64637c478bd9Sstevel@tonic-gate 	bzero(&in_progress_rctltab, sizeof (in_progress_rctltab));
64647c478bd9Sstevel@tonic-gate 	bzero(&in_progress_attrtab, sizeof (in_progress_attrtab));
6465fa9e4066Sahrens 	bzero(&in_progress_dstab, sizeof (in_progress_dstab));
64667c478bd9Sstevel@tonic-gate }
64677c478bd9Sstevel@tonic-gate 
64687c478bd9Sstevel@tonic-gate static int
validate_attr_name(char * name)64697c478bd9Sstevel@tonic-gate validate_attr_name(char *name)
64707c478bd9Sstevel@tonic-gate {
64717c478bd9Sstevel@tonic-gate 	int i;
64727c478bd9Sstevel@tonic-gate 
64737c478bd9Sstevel@tonic-gate 	if (!isalnum(name[0])) {
64747c478bd9Sstevel@tonic-gate 		zerr(gettext("Invalid %s %s %s: must start with an alpha-"
64757c478bd9Sstevel@tonic-gate 		    "numeric character."), rt_to_str(RT_ATTR),
64767c478bd9Sstevel@tonic-gate 		    pt_to_str(PT_NAME), name);
64777c478bd9Sstevel@tonic-gate 		return (Z_INVAL);
64787c478bd9Sstevel@tonic-gate 	}
64797c478bd9Sstevel@tonic-gate 	for (i = 1; name[i]; i++)
64807c478bd9Sstevel@tonic-gate 		if (!isalnum(name[i]) && name[i] != '-' && name[i] != '.') {
64817c478bd9Sstevel@tonic-gate 			zerr(gettext("Invalid %s %s %s: can only contain "
64827c478bd9Sstevel@tonic-gate 			    "alpha-numeric characters, plus '-' and '.'."),
64837c478bd9Sstevel@tonic-gate 			    rt_to_str(RT_ATTR), pt_to_str(PT_NAME), name);
64847c478bd9Sstevel@tonic-gate 			return (Z_INVAL);
64857c478bd9Sstevel@tonic-gate 		}
64867c478bd9Sstevel@tonic-gate 	return (Z_OK);
64877c478bd9Sstevel@tonic-gate }
64887c478bd9Sstevel@tonic-gate 
64897c478bd9Sstevel@tonic-gate static int
validate_attr_type_val(struct zone_attrtab * attrtab)64907c478bd9Sstevel@tonic-gate validate_attr_type_val(struct zone_attrtab *attrtab)
64917c478bd9Sstevel@tonic-gate {
64927c478bd9Sstevel@tonic-gate 	boolean_t boolval;
64937c478bd9Sstevel@tonic-gate 	int64_t intval;
64947c478bd9Sstevel@tonic-gate 	char strval[MAXNAMELEN];
64957c478bd9Sstevel@tonic-gate 	uint64_t uintval;
64967c478bd9Sstevel@tonic-gate 
64977c478bd9Sstevel@tonic-gate 	if (strcmp(attrtab->zone_attr_type, "boolean") == 0) {
64987c478bd9Sstevel@tonic-gate 		if (zonecfg_get_attr_boolean(attrtab, &boolval) == Z_OK)
64997c478bd9Sstevel@tonic-gate 			return (Z_OK);
65007c478bd9Sstevel@tonic-gate 		zerr(gettext("invalid %s value for %s=%s"),
65017c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR), pt_to_str(PT_TYPE), "boolean");
65027c478bd9Sstevel@tonic-gate 		return (Z_ERR);
65037c478bd9Sstevel@tonic-gate 	}
65047c478bd9Sstevel@tonic-gate 
65057c478bd9Sstevel@tonic-gate 	if (strcmp(attrtab->zone_attr_type, "int") == 0) {
65067c478bd9Sstevel@tonic-gate 		if (zonecfg_get_attr_int(attrtab, &intval) == Z_OK)
65077c478bd9Sstevel@tonic-gate 			return (Z_OK);
65087c478bd9Sstevel@tonic-gate 		zerr(gettext("invalid %s value for %s=%s"),
65097c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR), pt_to_str(PT_TYPE), "int");
65107c478bd9Sstevel@tonic-gate 		return (Z_ERR);
65117c478bd9Sstevel@tonic-gate 	}
65127c478bd9Sstevel@tonic-gate 
65137c478bd9Sstevel@tonic-gate 	if (strcmp(attrtab->zone_attr_type, "string") == 0) {
65147c478bd9Sstevel@tonic-gate 		if (zonecfg_get_attr_string(attrtab, strval,
65157c478bd9Sstevel@tonic-gate 		    sizeof (strval)) == Z_OK)
65167c478bd9Sstevel@tonic-gate 			return (Z_OK);
65177c478bd9Sstevel@tonic-gate 		zerr(gettext("invalid %s value for %s=%s"),
65187c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR), pt_to_str(PT_TYPE), "string");
65197c478bd9Sstevel@tonic-gate 		return (Z_ERR);
65207c478bd9Sstevel@tonic-gate 	}
65217c478bd9Sstevel@tonic-gate 
65227c478bd9Sstevel@tonic-gate 	if (strcmp(attrtab->zone_attr_type, "uint") == 0) {
65237c478bd9Sstevel@tonic-gate 		if (zonecfg_get_attr_uint(attrtab, &uintval) == Z_OK)
65247c478bd9Sstevel@tonic-gate 			return (Z_OK);
65257c478bd9Sstevel@tonic-gate 		zerr(gettext("invalid %s value for %s=%s"),
65267c478bd9Sstevel@tonic-gate 		    rt_to_str(RT_ATTR), pt_to_str(PT_TYPE), "uint");
65277c478bd9Sstevel@tonic-gate 		return (Z_ERR);
65287c478bd9Sstevel@tonic-gate 	}
65297c478bd9Sstevel@tonic-gate 
65307c478bd9Sstevel@tonic-gate 	zerr(gettext("invalid %s %s '%s'"), rt_to_str(RT_ATTR),
65317c478bd9Sstevel@tonic-gate 	    pt_to_str(PT_TYPE), attrtab->zone_attr_type);
65327c478bd9Sstevel@tonic-gate 	return (Z_ERR);
65337c478bd9Sstevel@tonic-gate }
65347c478bd9Sstevel@tonic-gate 
6535087719fdSdp /*
6536087719fdSdp  * Helper function for end_func-- checks the existence of a given property
6537087719fdSdp  * and emits a message if not specified.
6538087719fdSdp  */
6539087719fdSdp static int
end_check_reqd(char * attr,int pt,boolean_t * validation_failed)6540bbec428eSgjelinek end_check_reqd(char *attr, int pt, boolean_t *validation_failed)
6541087719fdSdp {
6542087719fdSdp 	if (strlen(attr) == 0) {
6543bbec428eSgjelinek 		*validation_failed = B_TRUE;
6544087719fdSdp 		zerr(gettext("%s not specified"), pt_to_str(pt));
6545087719fdSdp 		return (Z_ERR);
6546087719fdSdp 	}
6547087719fdSdp 	return (Z_OK);
6548087719fdSdp }
6549087719fdSdp 
6550550b6e40SSowmini Varadhan static void
net_exists_error(struct zone_nwiftab nwif)6551550b6e40SSowmini Varadhan net_exists_error(struct zone_nwiftab nwif)
6552550b6e40SSowmini Varadhan {
6553550b6e40SSowmini Varadhan 	if (strlen(nwif.zone_nwif_address) > 0) {
6554550b6e40SSowmini Varadhan 		zerr(gettext("A %s resource with the %s '%s', "
6555550b6e40SSowmini Varadhan 		    "and %s '%s' already exists."),
6556550b6e40SSowmini Varadhan 		    rt_to_str(RT_NET),
6557550b6e40SSowmini Varadhan 		    pt_to_str(PT_PHYSICAL),
6558550b6e40SSowmini Varadhan 		    nwif.zone_nwif_physical,
6559550b6e40SSowmini Varadhan 		    pt_to_str(PT_ADDRESS),
6560550b6e40SSowmini Varadhan 		    in_progress_nwiftab.zone_nwif_address);
6561550b6e40SSowmini Varadhan 	} else {
6562550b6e40SSowmini Varadhan 		zerr(gettext("A %s resource with the %s '%s', "
6563550b6e40SSowmini Varadhan 		    "and %s '%s' already exists."),
6564550b6e40SSowmini Varadhan 		    rt_to_str(RT_NET),
6565550b6e40SSowmini Varadhan 		    pt_to_str(PT_PHYSICAL),
6566550b6e40SSowmini Varadhan 		    nwif.zone_nwif_physical,
6567550b6e40SSowmini Varadhan 		    pt_to_str(PT_ALLOWED_ADDRESS),
6568550b6e40SSowmini Varadhan 		    nwif.zone_nwif_allowed_address);
6569550b6e40SSowmini Varadhan 	}
6570550b6e40SSowmini Varadhan }
6571550b6e40SSowmini Varadhan 
65727c478bd9Sstevel@tonic-gate void
end_func(cmd_t * cmd)65737c478bd9Sstevel@tonic-gate end_func(cmd_t *cmd)
65747c478bd9Sstevel@tonic-gate {
6575bbec428eSgjelinek 	boolean_t validation_failed = B_FALSE;
6576bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
65777c478bd9Sstevel@tonic-gate 	struct zone_fstab tmp_fstab;
65787c478bd9Sstevel@tonic-gate 	struct zone_nwiftab tmp_nwiftab;
65797c478bd9Sstevel@tonic-gate 	struct zone_devtab tmp_devtab;
65807c478bd9Sstevel@tonic-gate 	struct zone_rctltab tmp_rctltab;
65817c478bd9Sstevel@tonic-gate 	struct zone_attrtab tmp_attrtab;
6582fa9e4066Sahrens 	struct zone_dstab tmp_dstab;
6583cb8a054bSGlenn Faden 	struct zone_admintab tmp_admintab;
65840209230bSgjelinek 	int err, arg, res1, res2, res3;
65850209230bSgjelinek 	uint64_t swap_limit;
65860209230bSgjelinek 	uint64_t locked_limit;
6587c97ad5cdSakolb 	uint64_t proc_cap;
65887c478bd9Sstevel@tonic-gate 
65897c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
65907c478bd9Sstevel@tonic-gate 
65917c478bd9Sstevel@tonic-gate 	optind = 0;
65927ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
65937c478bd9Sstevel@tonic-gate 		switch (arg) {
65947c478bd9Sstevel@tonic-gate 		case '?':
65957c478bd9Sstevel@tonic-gate 			longer_usage(CMD_END);
6596bbec428eSgjelinek 			arg_err = B_TRUE;
65977ec75eb8Sgjelinek 			break;
65987c478bd9Sstevel@tonic-gate 		default:
65997c478bd9Sstevel@tonic-gate 			short_usage(CMD_END);
6600bbec428eSgjelinek 			arg_err = B_TRUE;
66017ec75eb8Sgjelinek 			break;
66027ec75eb8Sgjelinek 		}
66037ec75eb8Sgjelinek 	}
66047ec75eb8Sgjelinek 	if (arg_err)
66057c478bd9Sstevel@tonic-gate 		return;
66067ec75eb8Sgjelinek 
66077c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
66087c478bd9Sstevel@tonic-gate 		short_usage(CMD_END);
66097c478bd9Sstevel@tonic-gate 		return;
66107c478bd9Sstevel@tonic-gate 	}
66117c478bd9Sstevel@tonic-gate 
66127c478bd9Sstevel@tonic-gate 	if (global_scope) {
66137c478bd9Sstevel@tonic-gate 		scope_usage(CMD_END);
66147c478bd9Sstevel@tonic-gate 		return;
66157c478bd9Sstevel@tonic-gate 	}
66167c478bd9Sstevel@tonic-gate 
66177c478bd9Sstevel@tonic-gate 	assert(end_op == CMD_ADD || end_op == CMD_SELECT);
66187c478bd9Sstevel@tonic-gate 
66197c478bd9Sstevel@tonic-gate 	switch (resource_scope) {
66207c478bd9Sstevel@tonic-gate 	case RT_FS:
66217c478bd9Sstevel@tonic-gate 		/* First make sure everything was filled in. */
6622087719fdSdp 		if (end_check_reqd(in_progress_fstab.zone_fs_dir,
6623087719fdSdp 		    PT_DIR, &validation_failed) == Z_OK) {
6624087719fdSdp 			if (in_progress_fstab.zone_fs_dir[0] != '/') {
6625087719fdSdp 				zerr(gettext("%s %s is not an absolute path."),
6626087719fdSdp 				    pt_to_str(PT_DIR),
6627087719fdSdp 				    in_progress_fstab.zone_fs_dir);
6628bbec428eSgjelinek 				validation_failed = B_TRUE;
66297c478bd9Sstevel@tonic-gate 			}
66307c478bd9Sstevel@tonic-gate 		}
6631087719fdSdp 
6632087719fdSdp 		(void) end_check_reqd(in_progress_fstab.zone_fs_special,
6633087719fdSdp 		    PT_SPECIAL, &validation_failed);
6634087719fdSdp 
66357c478bd9Sstevel@tonic-gate 		if (in_progress_fstab.zone_fs_raw[0] != '\0' &&
66367c478bd9Sstevel@tonic-gate 		    in_progress_fstab.zone_fs_raw[0] != '/') {
6637087719fdSdp 			zerr(gettext("%s %s is not an absolute path."),
6638087719fdSdp 			    pt_to_str(PT_RAW),
6639087719fdSdp 			    in_progress_fstab.zone_fs_raw);
6640bbec428eSgjelinek 			validation_failed = B_TRUE;
66417c478bd9Sstevel@tonic-gate 		}
6642087719fdSdp 
6643087719fdSdp 		(void) end_check_reqd(in_progress_fstab.zone_fs_type, PT_TYPE,
6644087719fdSdp 		    &validation_failed);
6645087719fdSdp 
6646087719fdSdp 		if (validation_failed) {
6647bbec428eSgjelinek 			saw_error = B_TRUE;
66487c478bd9Sstevel@tonic-gate 			return;
6649087719fdSdp 		}
6650087719fdSdp 
66517c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
66527c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
66537c478bd9Sstevel@tonic-gate 			bzero(&tmp_fstab, sizeof (tmp_fstab));
66547c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_fstab.zone_fs_dir,
66557c478bd9Sstevel@tonic-gate 			    in_progress_fstab.zone_fs_dir,
66567c478bd9Sstevel@tonic-gate 			    sizeof (tmp_fstab.zone_fs_dir));
66577c478bd9Sstevel@tonic-gate 			err = zonecfg_lookup_filesystem(handle, &tmp_fstab);
66587c478bd9Sstevel@tonic-gate 			zonecfg_free_fs_option_list(tmp_fstab.zone_fs_options);
66597c478bd9Sstevel@tonic-gate 			if (err == Z_OK) {
66607c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s resource "
66617c478bd9Sstevel@tonic-gate 				    "with the %s '%s' already exists."),
66627c478bd9Sstevel@tonic-gate 				    rt_to_str(RT_FS), pt_to_str(PT_DIR),
66637c478bd9Sstevel@tonic-gate 				    in_progress_fstab.zone_fs_dir);
6664bbec428eSgjelinek 				saw_error = B_TRUE;
66657c478bd9Sstevel@tonic-gate 				return;
66667c478bd9Sstevel@tonic-gate 			}
66677c478bd9Sstevel@tonic-gate 			err = zonecfg_add_filesystem(handle,
66687c478bd9Sstevel@tonic-gate 			    &in_progress_fstab);
66697c478bd9Sstevel@tonic-gate 		} else {
66707c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_filesystem(handle, &old_fstab,
66717c478bd9Sstevel@tonic-gate 			    &in_progress_fstab);
66727c478bd9Sstevel@tonic-gate 		}
66737c478bd9Sstevel@tonic-gate 		zonecfg_free_fs_option_list(in_progress_fstab.zone_fs_options);
66747c478bd9Sstevel@tonic-gate 		in_progress_fstab.zone_fs_options = NULL;
66757c478bd9Sstevel@tonic-gate 		break;
6676087719fdSdp 
66777c478bd9Sstevel@tonic-gate 	case RT_NET:
6678f4b3ec61Sdh155122 		/*
6679f4b3ec61Sdh155122 		 * First make sure everything was filled in.
6680f4b3ec61Sdh155122 		 * Since we don't know whether IP will be shared
6681f4b3ec61Sdh155122 		 * or exclusive here, some checks are deferred until
6682f4b3ec61Sdh155122 		 * the verify command.
6683f4b3ec61Sdh155122 		 */
6684087719fdSdp 		(void) end_check_reqd(in_progress_nwiftab.zone_nwif_physical,
6685087719fdSdp 		    PT_PHYSICAL, &validation_failed);
6686087719fdSdp 
6687087719fdSdp 		if (validation_failed) {
6688bbec428eSgjelinek 			saw_error = B_TRUE;
66897c478bd9Sstevel@tonic-gate 			return;
6690087719fdSdp 		}
66917c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
66927c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
66937c478bd9Sstevel@tonic-gate 			bzero(&tmp_nwiftab, sizeof (tmp_nwiftab));
6694f4b3ec61Sdh155122 			(void) strlcpy(tmp_nwiftab.zone_nwif_physical,
6695f4b3ec61Sdh155122 			    in_progress_nwiftab.zone_nwif_physical,
6696f4b3ec61Sdh155122 			    sizeof (tmp_nwiftab.zone_nwif_physical));
66977c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_nwiftab.zone_nwif_address,
66987c478bd9Sstevel@tonic-gate 			    in_progress_nwiftab.zone_nwif_address,
66997c478bd9Sstevel@tonic-gate 			    sizeof (tmp_nwiftab.zone_nwif_address));
6700550b6e40SSowmini Varadhan 			(void) strlcpy(tmp_nwiftab.zone_nwif_allowed_address,
6701550b6e40SSowmini Varadhan 			    in_progress_nwiftab.zone_nwif_allowed_address,
6702550b6e40SSowmini Varadhan 			    sizeof (tmp_nwiftab.zone_nwif_allowed_address));
6703550b6e40SSowmini Varadhan 			(void) strlcpy(tmp_nwiftab.zone_nwif_defrouter,
6704550b6e40SSowmini Varadhan 			    in_progress_nwiftab.zone_nwif_defrouter,
6705550b6e40SSowmini Varadhan 			    sizeof (tmp_nwiftab.zone_nwif_defrouter));
67067c478bd9Sstevel@tonic-gate 			if (zonecfg_lookup_nwif(handle, &tmp_nwiftab) == Z_OK) {
6707550b6e40SSowmini Varadhan 				net_exists_error(in_progress_nwiftab);
6708bbec428eSgjelinek 				saw_error = B_TRUE;
67097c478bd9Sstevel@tonic-gate 				return;
67107c478bd9Sstevel@tonic-gate 			}
67117c478bd9Sstevel@tonic-gate 			err = zonecfg_add_nwif(handle, &in_progress_nwiftab);
67127c478bd9Sstevel@tonic-gate 		} else {
67137c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_nwif(handle, &old_nwiftab,
67147c478bd9Sstevel@tonic-gate 			    &in_progress_nwiftab);
67157c478bd9Sstevel@tonic-gate 		}
67167c478bd9Sstevel@tonic-gate 		break;
6717087719fdSdp 
67187c478bd9Sstevel@tonic-gate 	case RT_DEVICE:
67197c478bd9Sstevel@tonic-gate 		/* First make sure everything was filled in. */
6720087719fdSdp 		(void) end_check_reqd(in_progress_devtab.zone_dev_match,
6721087719fdSdp 		    PT_MATCH, &validation_failed);
6722087719fdSdp 
6723087719fdSdp 		if (validation_failed) {
6724bbec428eSgjelinek 			saw_error = B_TRUE;
67257c478bd9Sstevel@tonic-gate 			return;
6726087719fdSdp 		}
6727087719fdSdp 
67287c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
67297c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
67307c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_devtab.zone_dev_match,
67317c478bd9Sstevel@tonic-gate 			    in_progress_devtab.zone_dev_match,
67327c478bd9Sstevel@tonic-gate 			    sizeof (tmp_devtab.zone_dev_match));
67337c478bd9Sstevel@tonic-gate 			if (zonecfg_lookup_dev(handle, &tmp_devtab) == Z_OK) {
67347c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s resource with the %s '%s' "
67357c478bd9Sstevel@tonic-gate 				    "already exists."), rt_to_str(RT_DEVICE),
67367c478bd9Sstevel@tonic-gate 				    pt_to_str(PT_MATCH),
67377c478bd9Sstevel@tonic-gate 				    in_progress_devtab.zone_dev_match);
6738bbec428eSgjelinek 				saw_error = B_TRUE;
67397c478bd9Sstevel@tonic-gate 				return;
67407c478bd9Sstevel@tonic-gate 			}
67417c478bd9Sstevel@tonic-gate 			err = zonecfg_add_dev(handle, &in_progress_devtab);
67427c478bd9Sstevel@tonic-gate 		} else {
67437c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_dev(handle, &old_devtab,
67447c478bd9Sstevel@tonic-gate 			    &in_progress_devtab);
67457c478bd9Sstevel@tonic-gate 		}
67467c478bd9Sstevel@tonic-gate 		break;
6747087719fdSdp 
67487c478bd9Sstevel@tonic-gate 	case RT_RCTL:
67497c478bd9Sstevel@tonic-gate 		/* First make sure everything was filled in. */
6750087719fdSdp 		(void) end_check_reqd(in_progress_rctltab.zone_rctl_name,
6751087719fdSdp 		    PT_NAME, &validation_failed);
6752087719fdSdp 
67537c478bd9Sstevel@tonic-gate 		if (in_progress_rctltab.zone_rctl_valptr == NULL) {
67547c478bd9Sstevel@tonic-gate 			zerr(gettext("no %s specified"), pt_to_str(PT_VALUE));
6755bbec428eSgjelinek 			validation_failed = B_TRUE;
67567c478bd9Sstevel@tonic-gate 		}
6757087719fdSdp 
6758087719fdSdp 		if (validation_failed) {
6759bbec428eSgjelinek 			saw_error = B_TRUE;
67607c478bd9Sstevel@tonic-gate 			return;
6761087719fdSdp 		}
6762087719fdSdp 
67637c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
67647c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
67657c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_rctltab.zone_rctl_name,
67667c478bd9Sstevel@tonic-gate 			    in_progress_rctltab.zone_rctl_name,
67677c478bd9Sstevel@tonic-gate 			    sizeof (tmp_rctltab.zone_rctl_name));
67687c478bd9Sstevel@tonic-gate 			tmp_rctltab.zone_rctl_valptr = NULL;
67697c478bd9Sstevel@tonic-gate 			err = zonecfg_lookup_rctl(handle, &tmp_rctltab);
67707c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(
67717c478bd9Sstevel@tonic-gate 			    tmp_rctltab.zone_rctl_valptr);
67727c478bd9Sstevel@tonic-gate 			if (err == Z_OK) {
67737c478bd9Sstevel@tonic-gate 				zerr(gettext("A %s resource "
67747c478bd9Sstevel@tonic-gate 				    "with the %s '%s' already exists."),
67757c478bd9Sstevel@tonic-gate 				    rt_to_str(RT_RCTL), pt_to_str(PT_NAME),
67767c478bd9Sstevel@tonic-gate 				    in_progress_rctltab.zone_rctl_name);
6777bbec428eSgjelinek 				saw_error = B_TRUE;
67787c478bd9Sstevel@tonic-gate 				return;
67797c478bd9Sstevel@tonic-gate 			}
67807c478bd9Sstevel@tonic-gate 			err = zonecfg_add_rctl(handle, &in_progress_rctltab);
67817c478bd9Sstevel@tonic-gate 		} else {
67827c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_rctl(handle, &old_rctltab,
67837c478bd9Sstevel@tonic-gate 			    &in_progress_rctltab);
67847c478bd9Sstevel@tonic-gate 		}
67857c478bd9Sstevel@tonic-gate 		if (err == Z_OK) {
67867c478bd9Sstevel@tonic-gate 			zonecfg_free_rctl_value_list(
67877c478bd9Sstevel@tonic-gate 			    in_progress_rctltab.zone_rctl_valptr);
67887c478bd9Sstevel@tonic-gate 			in_progress_rctltab.zone_rctl_valptr = NULL;
67897c478bd9Sstevel@tonic-gate 		}
67907c478bd9Sstevel@tonic-gate 		break;
6791087719fdSdp 
67927c478bd9Sstevel@tonic-gate 	case RT_ATTR:
67937c478bd9Sstevel@tonic-gate 		/* First make sure everything was filled in. */
6794087719fdSdp 		(void) end_check_reqd(in_progress_attrtab.zone_attr_name,
6795087719fdSdp 		    PT_NAME, &validation_failed);
6796087719fdSdp 		(void) end_check_reqd(in_progress_attrtab.zone_attr_type,
6797087719fdSdp 		    PT_TYPE, &validation_failed);
6798087719fdSdp 		(void) end_check_reqd(in_progress_attrtab.zone_attr_value,
6799087719fdSdp 		    PT_VALUE, &validation_failed);
6800087719fdSdp 
68017c478bd9Sstevel@tonic-gate 		if (validate_attr_name(in_progress_attrtab.zone_attr_name) !=
6802087719fdSdp 		    Z_OK)
6803bbec428eSgjelinek 			validation_failed = B_TRUE;
6804087719fdSdp 
6805087719fdSdp 		if (validate_attr_type_val(&in_progress_attrtab) != Z_OK)
6806bbec428eSgjelinek 			validation_failed = B_TRUE;
6807087719fdSdp 
6808087719fdSdp 		if (validation_failed) {
6809bbec428eSgjelinek 			saw_error = B_TRUE;
68107c478bd9Sstevel@tonic-gate 			return;
6811087719fdSdp 		}
68127c478bd9Sstevel@tonic-gate 		if (end_op == CMD_ADD) {
68137c478bd9Sstevel@tonic-gate 			/* Make sure there isn't already one like this. */
68147c478bd9Sstevel@tonic-gate 			bzero(&tmp_attrtab, sizeof (tmp_attrtab));
68157c478bd9Sstevel@tonic-gate 			(void) strlcpy(tmp_attrtab.zone_attr_name,
68167c478bd9Sstevel@tonic-gate 			    in_progress_attrtab.zone_attr_name,
68177c478bd9Sstevel@tonic-gate 			    sizeof (tmp_attrtab.zone_attr_name));
68187c478bd9Sstevel@tonic-gate 			if (zonecfg_lookup_attr(handle, &tmp_attrtab) == Z_OK) {
68197c478bd9Sstevel@tonic-gate 				zerr(gettext("An %s resource "
68207c478bd9Sstevel@tonic-gate 				    "with the %s '%s' already exists."),
68217c478bd9Sstevel@tonic-gate 				    rt_to_str(RT_ATTR), pt_to_str(PT_NAME),
68227c478bd9Sstevel@tonic-gate 				    in_progress_attrtab.zone_attr_name);
6823bbec428eSgjelinek 				saw_error = B_TRUE;
68247c478bd9Sstevel@tonic-gate 				return;
68257c478bd9Sstevel@tonic-gate 			}
68267c478bd9Sstevel@tonic-gate 			err = zonecfg_add_attr(handle, &in_progress_attrtab);
68277c478bd9Sstevel@tonic-gate 		} else {
68287c478bd9Sstevel@tonic-gate 			err = zonecfg_modify_attr(handle, &old_attrtab,
68297c478bd9Sstevel@tonic-gate 			    &in_progress_attrtab);
68307c478bd9Sstevel@tonic-gate 		}
68317c478bd9Sstevel@tonic-gate 		break;
6832fa9e4066Sahrens 	case RT_DATASET:
6833fa9e4066Sahrens 		/* First make sure everything was filled in. */
6834fa9e4066Sahrens 		if (strlen(in_progress_dstab.zone_dataset_name) == 0) {
6835fa9e4066Sahrens 			zerr("%s %s", pt_to_str(PT_NAME),
6836fa9e4066Sahrens 			    gettext("not specified"));
6837bbec428eSgjelinek 			saw_error = B_TRUE;
6838bbec428eSgjelinek 			validation_failed = B_TRUE;
6839fa9e4066Sahrens 		}
6840fa9e4066Sahrens 		if (validation_failed)
6841fa9e4066Sahrens 			return;
6842fa9e4066Sahrens 		if (end_op == CMD_ADD) {
6843fa9e4066Sahrens 			/* Make sure there isn't already one like this. */
6844fa9e4066Sahrens 			bzero(&tmp_dstab, sizeof (tmp_dstab));
6845fa9e4066Sahrens 			(void) strlcpy(tmp_dstab.zone_dataset_name,
6846fa9e4066Sahrens 			    in_progress_dstab.zone_dataset_name,
6847fa9e4066Sahrens 			    sizeof (tmp_dstab.zone_dataset_name));
6848fa9e4066Sahrens 			err = zonecfg_lookup_ds(handle, &tmp_dstab);
6849fa9e4066Sahrens 			if (err == Z_OK) {
6850fa9e4066Sahrens 				zerr(gettext("A %s resource "
6851fa9e4066Sahrens 				    "with the %s '%s' already exists."),
6852fa9e4066Sahrens 				    rt_to_str(RT_DATASET), pt_to_str(PT_NAME),
6853fa9e4066Sahrens 				    in_progress_dstab.zone_dataset_name);
6854bbec428eSgjelinek 				saw_error = B_TRUE;
6855fa9e4066Sahrens 				return;
6856fa9e4066Sahrens 			}
6857fa9e4066Sahrens 			err = zonecfg_add_ds(handle, &in_progress_dstab);
6858fa9e4066Sahrens 		} else {
6859fa9e4066Sahrens 			err = zonecfg_modify_ds(handle, &old_dstab,
6860fa9e4066Sahrens 			    &in_progress_dstab);
6861fa9e4066Sahrens 		}
6862fa9e4066Sahrens 		break;
68630209230bSgjelinek 	case RT_DCPU:
68640209230bSgjelinek 		/* Make sure everything was filled in. */
68650209230bSgjelinek 		if (end_check_reqd(in_progress_psettab.zone_ncpu_min,
68660209230bSgjelinek 		    PT_NCPUS, &validation_failed) != Z_OK) {
6867bbec428eSgjelinek 			saw_error = B_TRUE;
68680209230bSgjelinek 			return;
68690209230bSgjelinek 		}
68700209230bSgjelinek 
68710209230bSgjelinek 		if (end_op == CMD_ADD) {
68720209230bSgjelinek 			err = zonecfg_add_pset(handle, &in_progress_psettab);
68730209230bSgjelinek 		} else {
68740209230bSgjelinek 			err = zonecfg_modify_pset(handle, &in_progress_psettab);
68750209230bSgjelinek 		}
68760209230bSgjelinek 		break;
6877c97ad5cdSakolb 	case RT_PCAP:
6878c97ad5cdSakolb 		/* Make sure everything was filled in. */
6879c97ad5cdSakolb 		if (zonecfg_get_aliased_rctl(handle, ALIAS_CPUCAP, &proc_cap)
6880c97ad5cdSakolb 		    != Z_OK) {
6881c97ad5cdSakolb 			zerr(gettext("%s not specified"), pt_to_str(PT_NCPUS));
6882bbec428eSgjelinek 			saw_error = B_TRUE;
6883bbec428eSgjelinek 			validation_failed = B_TRUE;
6884c97ad5cdSakolb 			return;
6885c97ad5cdSakolb 		}
6886c97ad5cdSakolb 		err = Z_OK;
6887c97ad5cdSakolb 		break;
68880209230bSgjelinek 	case RT_MCAP:
68890209230bSgjelinek 		/* Make sure everything was filled in. */
68900209230bSgjelinek 		res1 = strlen(in_progress_mcaptab.zone_physmem_cap) == 0 ?
68910209230bSgjelinek 		    Z_ERR : Z_OK;
68920209230bSgjelinek 		res2 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXSWAP,
68930209230bSgjelinek 		    &swap_limit);
68940209230bSgjelinek 		res3 = zonecfg_get_aliased_rctl(handle, ALIAS_MAXLOCKEDMEM,
68950209230bSgjelinek 		    &locked_limit);
68960209230bSgjelinek 
68970209230bSgjelinek 		if (res1 != Z_OK && res2 != Z_OK && res3 != Z_OK) {
68980209230bSgjelinek 			zerr(gettext("No property was specified.  One of %s, "
68990209230bSgjelinek 			    "%s or %s is required."), pt_to_str(PT_PHYSICAL),
69000209230bSgjelinek 			    pt_to_str(PT_SWAP), pt_to_str(PT_LOCKED));
6901bbec428eSgjelinek 			saw_error = B_TRUE;
69020209230bSgjelinek 			return;
69030209230bSgjelinek 		}
69040209230bSgjelinek 
69050209230bSgjelinek 		/* if phys & locked are both set, verify locked <= phys */
69060209230bSgjelinek 		if (res1 == Z_OK && res3 == Z_OK) {
69070209230bSgjelinek 			uint64_t phys_limit;
69080209230bSgjelinek 			char *endp;
69090209230bSgjelinek 
69100209230bSgjelinek 			phys_limit = strtoull(
69110209230bSgjelinek 			    in_progress_mcaptab.zone_physmem_cap, &endp, 10);
69120209230bSgjelinek 			if (phys_limit < locked_limit) {
69130209230bSgjelinek 				zerr(gettext("The %s cap must be less than or "
69140209230bSgjelinek 				    "equal to the %s cap."),
69150209230bSgjelinek 				    pt_to_str(PT_LOCKED),
69160209230bSgjelinek 				    pt_to_str(PT_PHYSICAL));
6917bbec428eSgjelinek 				saw_error = B_TRUE;
69180209230bSgjelinek 				return;
69190209230bSgjelinek 			}
69200209230bSgjelinek 		}
69210209230bSgjelinek 
69220209230bSgjelinek 		err = Z_OK;
69230209230bSgjelinek 		if (res1 == Z_OK) {
69240209230bSgjelinek 			/*
69250209230bSgjelinek 			 * We could be ending from either an add operation
69260209230bSgjelinek 			 * or a select operation.  Since all of the properties
69270209230bSgjelinek 			 * within this resource are optional, we always use
69280209230bSgjelinek 			 * modify on the mcap entry.  zonecfg_modify_mcap()
69290209230bSgjelinek 			 * will handle both adding and modifying a memory cap.
69300209230bSgjelinek 			 */
69310209230bSgjelinek 			err = zonecfg_modify_mcap(handle, &in_progress_mcaptab);
69320209230bSgjelinek 		} else if (end_op == CMD_SELECT) {
69330209230bSgjelinek 			/*
69340209230bSgjelinek 			 * If we're ending from a select and the physical
69350209230bSgjelinek 			 * memory cap is empty then the user could have cleared
69360209230bSgjelinek 			 * the physical cap value, so try to delete the entry.
69370209230bSgjelinek 			 */
69380209230bSgjelinek 			(void) zonecfg_delete_mcap(handle);
69390209230bSgjelinek 		}
69400209230bSgjelinek 		break;
6941cb8a054bSGlenn Faden 	case RT_ADMIN:
6942cb8a054bSGlenn Faden 		/* First make sure everything was filled in. */
6943cb8a054bSGlenn Faden 		if (end_check_reqd(in_progress_admintab.zone_admin_user,
6944cb8a054bSGlenn Faden 		    PT_USER, &validation_failed) == Z_OK) {
6945cb8a054bSGlenn Faden 			if (getpwnam(in_progress_admintab.zone_admin_user)
6946cb8a054bSGlenn Faden 			    == NULL) {
6947cb8a054bSGlenn Faden 				zerr(gettext("%s %s is not a valid username"),
6948cb8a054bSGlenn Faden 				    pt_to_str(PT_USER),
6949cb8a054bSGlenn Faden 				    in_progress_admintab.zone_admin_user);
6950cb8a054bSGlenn Faden 				validation_failed = B_TRUE;
6951cb8a054bSGlenn Faden 			}
6952cb8a054bSGlenn Faden 		}
6953cb8a054bSGlenn Faden 
6954cb8a054bSGlenn Faden 		if (end_check_reqd(in_progress_admintab.zone_admin_auths,
6955cb8a054bSGlenn Faden 		    PT_AUTHS, &validation_failed) == Z_OK) {
6956cb8a054bSGlenn Faden 			if (!zonecfg_valid_auths(
6957cb8a054bSGlenn Faden 			    in_progress_admintab.zone_admin_auths,
6958cb8a054bSGlenn Faden 			    zone)) {
6959cb8a054bSGlenn Faden 				validation_failed = B_TRUE;
6960cb8a054bSGlenn Faden 			}
6961cb8a054bSGlenn Faden 		}
6962cb8a054bSGlenn Faden 
6963cb8a054bSGlenn Faden 		if (validation_failed) {
6964cb8a054bSGlenn Faden 			saw_error = B_TRUE;
6965cb8a054bSGlenn Faden 			return;
6966cb8a054bSGlenn Faden 		}
6967cb8a054bSGlenn Faden 
6968cb8a054bSGlenn Faden 		if (end_op == CMD_ADD) {
6969cb8a054bSGlenn Faden 			/* Make sure there isn't already one like this. */
6970cb8a054bSGlenn Faden 			bzero(&tmp_admintab, sizeof (tmp_admintab));
6971cb8a054bSGlenn Faden 			(void) strlcpy(tmp_admintab.zone_admin_user,
6972cb8a054bSGlenn Faden 			    in_progress_admintab.zone_admin_user,
6973cb8a054bSGlenn Faden 			    sizeof (tmp_admintab.zone_admin_user));
6974cb8a054bSGlenn Faden 			err = zonecfg_lookup_admin(
6975cb8a054bSGlenn Faden 			    handle, &tmp_admintab);
6976cb8a054bSGlenn Faden 			if (err == Z_OK) {
6977cb8a054bSGlenn Faden 				zerr(gettext("A %s resource "
6978cb8a054bSGlenn Faden 				    "with the %s '%s' already exists."),
6979cb8a054bSGlenn Faden 				    rt_to_str(RT_ADMIN),
6980cb8a054bSGlenn Faden 				    pt_to_str(PT_USER),
6981cb8a054bSGlenn Faden 				    in_progress_admintab.zone_admin_user);
6982cb8a054bSGlenn Faden 				saw_error = B_TRUE;
6983cb8a054bSGlenn Faden 				return;
6984cb8a054bSGlenn Faden 			}
6985cb8a054bSGlenn Faden 			err = zonecfg_add_admin(handle,
6986cb8a054bSGlenn Faden 			    &in_progress_admintab, zone);
6987cb8a054bSGlenn Faden 		} else {
6988cb8a054bSGlenn Faden 			err = zonecfg_modify_admin(handle,
6989cb8a054bSGlenn Faden 			    &old_admintab, &in_progress_admintab,
6990cb8a054bSGlenn Faden 			    zone);
6991cb8a054bSGlenn Faden 		}
6992cb8a054bSGlenn Faden 		break;
6993d2a70789SRichard Lowe 	case RT_SECFLAGS:
6994d2a70789SRichard Lowe 		if (verify_secflags(&in_progress_secflagstab) != B_TRUE) {
6995d2a70789SRichard Lowe 			saw_error = B_TRUE;
6996d2a70789SRichard Lowe 			return;
6997d2a70789SRichard Lowe 		}
6998d2a70789SRichard Lowe 
6999d2a70789SRichard Lowe 		if (end_op == CMD_ADD) {
7000d2a70789SRichard Lowe 			err = zonecfg_add_secflags(handle,
7001d2a70789SRichard Lowe 			    &in_progress_secflagstab);
7002d2a70789SRichard Lowe 		} else {
7003d2a70789SRichard Lowe 			err = zonecfg_modify_secflags(handle,
7004d2a70789SRichard Lowe 			    &old_secflagstab, &in_progress_secflagstab);
7005d2a70789SRichard Lowe 		}
7006d2a70789SRichard Lowe 		break;
70077c478bd9Sstevel@tonic-gate 	default:
70087c478bd9Sstevel@tonic-gate 		zone_perror(rt_to_str(resource_scope), Z_NO_RESOURCE_TYPE,
7009bbec428eSgjelinek 		    B_TRUE);
7010bbec428eSgjelinek 		saw_error = B_TRUE;
70117c478bd9Sstevel@tonic-gate 		return;
70127c478bd9Sstevel@tonic-gate 	}
70137c478bd9Sstevel@tonic-gate 
70147c478bd9Sstevel@tonic-gate 	if (err != Z_OK) {
7015bbec428eSgjelinek 		zone_perror(zone, err, B_TRUE);
70167c478bd9Sstevel@tonic-gate 	} else {
7017bbec428eSgjelinek 		need_to_commit = B_TRUE;
7018bbec428eSgjelinek 		global_scope = B_TRUE;
70197c478bd9Sstevel@tonic-gate 		end_op = -1;
70207c478bd9Sstevel@tonic-gate 	}
70217c478bd9Sstevel@tonic-gate }
70227c478bd9Sstevel@tonic-gate 
70237c478bd9Sstevel@tonic-gate void
commit_func(cmd_t * cmd)70247c478bd9Sstevel@tonic-gate commit_func(cmd_t *cmd)
70257c478bd9Sstevel@tonic-gate {
70267c478bd9Sstevel@tonic-gate 	int arg;
7027bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
70287c478bd9Sstevel@tonic-gate 
70297c478bd9Sstevel@tonic-gate 	optind = 0;
70307ec75eb8Sgjelinek 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?")) != EOF) {
70317c478bd9Sstevel@tonic-gate 		switch (arg) {
70327c478bd9Sstevel@tonic-gate 		case '?':
70337c478bd9Sstevel@tonic-gate 			longer_usage(CMD_COMMIT);
7034bbec428eSgjelinek 			arg_err = B_TRUE;
70357ec75eb8Sgjelinek 			break;
70367c478bd9Sstevel@tonic-gate 		default:
70377c478bd9Sstevel@tonic-gate 			short_usage(CMD_COMMIT);
7038bbec428eSgjelinek 			arg_err = B_TRUE;
70397ec75eb8Sgjelinek 			break;
70407ec75eb8Sgjelinek 		}
70417ec75eb8Sgjelinek 	}
70427ec75eb8Sgjelinek 	if (arg_err)
70437c478bd9Sstevel@tonic-gate 		return;
70447ec75eb8Sgjelinek 
70457c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
70467c478bd9Sstevel@tonic-gate 		short_usage(CMD_COMMIT);
70477c478bd9Sstevel@tonic-gate 		return;
70487c478bd9Sstevel@tonic-gate 	}
70497c478bd9Sstevel@tonic-gate 
70507c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_COMMIT))
70517c478bd9Sstevel@tonic-gate 		return;
70527c478bd9Sstevel@tonic-gate 
70537c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
70547c478bd9Sstevel@tonic-gate 
70557c478bd9Sstevel@tonic-gate 	cmd->cmd_argc = 1;
70567c478bd9Sstevel@tonic-gate 	/*
70577c478bd9Sstevel@tonic-gate 	 * cmd_arg normally comes from a strdup() in the lexer, and the
70587c478bd9Sstevel@tonic-gate 	 * whole cmd structure and its (char *) attributes are freed at
70597c478bd9Sstevel@tonic-gate 	 * the completion of each command, so the strdup() below is needed
70607c478bd9Sstevel@tonic-gate 	 * to match this and prevent a core dump from trying to free()
70617c478bd9Sstevel@tonic-gate 	 * something that can't be.
70627c478bd9Sstevel@tonic-gate 	 */
70637c478bd9Sstevel@tonic-gate 	if ((cmd->cmd_argv[0] = strdup("save")) == NULL) {
7064bbec428eSgjelinek 		zone_perror(zone, Z_NOMEM, B_TRUE);
70657c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
70667c478bd9Sstevel@tonic-gate 	}
70677c478bd9Sstevel@tonic-gate 	cmd->cmd_argv[1] = NULL;
70687c478bd9Sstevel@tonic-gate 	verify_func(cmd);
70697c478bd9Sstevel@tonic-gate }
70707c478bd9Sstevel@tonic-gate 
70717c478bd9Sstevel@tonic-gate void
revert_func(cmd_t * cmd)70727c478bd9Sstevel@tonic-gate revert_func(cmd_t *cmd)
70737c478bd9Sstevel@tonic-gate {
70747c478bd9Sstevel@tonic-gate 	char line[128];	/* enough to ask a question */
7075bbec428eSgjelinek 	boolean_t force = B_FALSE;
7076bbec428eSgjelinek 	boolean_t arg_err = B_FALSE;
70777c478bd9Sstevel@tonic-gate 	int err, arg, answer;
70787c478bd9Sstevel@tonic-gate 
70797c478bd9Sstevel@tonic-gate 	optind = 0;
70807c478bd9Sstevel@tonic-gate 	while ((arg = getopt(cmd->cmd_argc, cmd->cmd_argv, "?F")) != EOF) {
70817c478bd9Sstevel@tonic-gate 		switch (arg) {
70827c478bd9Sstevel@tonic-gate 		case '?':
70837c478bd9Sstevel@tonic-gate 			longer_usage(CMD_REVERT);
7084bbec428eSgjelinek 			arg_err = B_TRUE;
70857ec75eb8Sgjelinek 			break;
70867c478bd9Sstevel@tonic-gate 		case 'F':
7087bbec428eSgjelinek 			force = B_TRUE;
70887c478bd9Sstevel@tonic-gate 			break;
70897c478bd9Sstevel@tonic-gate 		default:
70907c478bd9Sstevel@tonic-gate 			short_usage(CMD_REVERT);
7091bbec428eSgjelinek 			arg_err = B_TRUE;
70927ec75eb8Sgjelinek 			break;
70937ec75eb8Sgjelinek 		}
70947ec75eb8Sgjelinek 	}
70957ec75eb8Sgjelinek 	if (arg_err)
70967c478bd9Sstevel@tonic-gate 		return;
70977ec75eb8Sgjelinek 
70987c478bd9Sstevel@tonic-gate 	if (optind != cmd->cmd_argc) {
70997c478bd9Sstevel@tonic-gate 		short_usage(CMD_REVERT);
71007c478bd9Sstevel@tonic-gate 		return;
71017c478bd9Sstevel@tonic-gate 	}
71027c478bd9Sstevel@tonic-gate 
71037c478bd9Sstevel@tonic-gate 	if (zone_is_read_only(CMD_REVERT))
71047c478bd9Sstevel@tonic-gate 		return;
71057c478bd9Sstevel@tonic-gate 
71065b418297Sjv227347 	if (!global_scope) {
71075b418297Sjv227347 		zerr(gettext("You can only use %s in the global scope.\nUse"
71085b418297Sjv227347 		    " '%s' to cancel changes to a resource specification."),
71095b418297Sjv227347 		    cmd_to_str(CMD_REVERT), cmd_to_str(CMD_CANCEL));
71105b418297Sjv227347 		saw_error = B_TRUE;
71115b418297Sjv227347 		return;
71125b418297Sjv227347 	}
71135b418297Sjv227347 
71147c478bd9Sstevel@tonic-gate 	if (zonecfg_check_handle(handle) != Z_OK) {
71157c478bd9Sstevel@tonic-gate 		zerr(gettext("No changes to revert."));
7116bbec428eSgjelinek 		saw_error = B_TRUE;
71177c478bd9Sstevel@tonic-gate 		return;
71187c478bd9Sstevel@tonic-gate 	}
71197c478bd9Sstevel@tonic-gate 
71207c478bd9Sstevel@tonic-gate 	if (!force) {
71217c478bd9Sstevel@tonic-gate 		(void) snprintf(line, sizeof (line),
71227c478bd9Sstevel@tonic-gate 		    gettext("Are you sure you want to revert"));
7123bbec428eSgjelinek 		if ((answer = ask_yesno(B_FALSE, line)) == -1) {
71247c478bd9Sstevel@tonic-gate 			zerr(gettext("Input not from terminal and -F not "
71257c478bd9Sstevel@tonic-gate 			    "specified:\n%s command ignored, exiting."),
71267c478bd9Sstevel@tonic-gate 			    cmd_to_str(CMD_REVERT));
71277c478bd9Sstevel@tonic-gate 			exit(Z_ERR);
71287c478bd9Sstevel@tonic-gate 		}
71297c478bd9Sstevel@tonic-gate 		if (answer != 1)
71307c478bd9Sstevel@tonic-gate 			return;
71317c478bd9Sstevel@tonic-gate 	}
71327c478bd9Sstevel@tonic-gate 
71337c478bd9Sstevel@tonic-gate 	/*
7134cb8a054bSGlenn Faden 	 * Reset any pending admins that were
7135cb8a054bSGlenn Faden 	 * removed from the previous zone
7136cb8a054bSGlenn Faden 	 */
7137cb8a054bSGlenn Faden 	zonecfg_remove_userauths(handle, "", zone, B_FALSE);
7138cb8a054bSGlenn Faden 
7139cb8a054bSGlenn Faden 	/*
71407c478bd9Sstevel@tonic-gate 	 * Time for a new handle: finish the old one off first
71417c478bd9Sstevel@tonic-gate 	 * then get a new one properly to avoid leaks.
71427c478bd9Sstevel@tonic-gate 	 */
71437c478bd9Sstevel@tonic-gate 	zonecfg_fini_handle(handle);
71447c478bd9Sstevel@tonic-gate 	if ((handle = zonecfg_init_handle()) == NULL) {
7145bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
71467c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
71477c478bd9Sstevel@tonic-gate 	}
7148cb8a054bSGlenn Faden 
7149087719fdSdp 	if ((err = zonecfg_get_handle(revert_zone, handle)) != Z_OK) {
7150bbec428eSgjelinek 		saw_error = B_TRUE;
7151bbec428eSgjelinek 		got_handle = B_FALSE;
71527c478bd9Sstevel@tonic-gate 		if (err == Z_NO_ZONE)
71537c478bd9Sstevel@tonic-gate 			zerr(gettext("%s: no such saved zone to revert to."),
7154087719fdSdp 			    revert_zone);
71557c478bd9Sstevel@tonic-gate 		else
7156bbec428eSgjelinek 			zone_perror(zone, err, B_TRUE);
71577c478bd9Sstevel@tonic-gate 	}
7158087719fdSdp 	(void) strlcpy(zone, revert_zone, sizeof (zone));
71597c478bd9Sstevel@tonic-gate }
71607c478bd9Sstevel@tonic-gate 
71617c478bd9Sstevel@tonic-gate void
help_func(cmd_t * cmd)71627c478bd9Sstevel@tonic-gate help_func(cmd_t *cmd)
71637c478bd9Sstevel@tonic-gate {
71647c478bd9Sstevel@tonic-gate 	int i;
71657c478bd9Sstevel@tonic-gate 
71667c478bd9Sstevel@tonic-gate 	assert(cmd != NULL);
71677c478bd9Sstevel@tonic-gate 
71687c478bd9Sstevel@tonic-gate 	if (cmd->cmd_argc == 0) {
7169bbec428eSgjelinek 		usage(B_TRUE, global_scope ? HELP_SUBCMDS : HELP_RES_SCOPE);
71707c478bd9Sstevel@tonic-gate 		return;
71717c478bd9Sstevel@tonic-gate 	}
71727c478bd9Sstevel@tonic-gate 	if (strcmp(cmd->cmd_argv[0], "usage") == 0) {
7173bbec428eSgjelinek 		usage(B_TRUE, HELP_USAGE);
71747c478bd9Sstevel@tonic-gate 		return;
71757c478bd9Sstevel@tonic-gate 	}
71767c478bd9Sstevel@tonic-gate 	if (strcmp(cmd->cmd_argv[0], "commands") == 0) {
7177bbec428eSgjelinek 		usage(B_TRUE, HELP_SUBCMDS);
71787c478bd9Sstevel@tonic-gate 		return;
71797c478bd9Sstevel@tonic-gate 	}
71807c478bd9Sstevel@tonic-gate 	if (strcmp(cmd->cmd_argv[0], "syntax") == 0) {
7181bbec428eSgjelinek 		usage(B_TRUE, HELP_SYNTAX | HELP_RES_PROPS);
71827c478bd9Sstevel@tonic-gate 		return;
71837c478bd9Sstevel@tonic-gate 	}
71847c478bd9Sstevel@tonic-gate 	if (strcmp(cmd->cmd_argv[0], "-?") == 0) {
71857c478bd9Sstevel@tonic-gate 		longer_usage(CMD_HELP);
71867c478bd9Sstevel@tonic-gate 		return;
71877c478bd9Sstevel@tonic-gate 	}
71887c478bd9Sstevel@tonic-gate 
71897c478bd9Sstevel@tonic-gate 	for (i = 0; i <= CMD_MAX; i++) {
71907c478bd9Sstevel@tonic-gate 		if (strcmp(cmd->cmd_argv[0], cmd_to_str(i)) == 0) {
71917c478bd9Sstevel@tonic-gate 			longer_usage(i);
71927c478bd9Sstevel@tonic-gate 			return;
71937c478bd9Sstevel@tonic-gate 		}
71947c478bd9Sstevel@tonic-gate 	}
71957c478bd9Sstevel@tonic-gate 	/* We do not use zerr() here because we do not want its extra \n. */
71967c478bd9Sstevel@tonic-gate 	(void) fprintf(stderr, gettext("Unknown help subject %s.  "),
71977c478bd9Sstevel@tonic-gate 	    cmd->cmd_argv[0]);
7198bbec428eSgjelinek 	usage(B_FALSE, HELP_META);
71997c478bd9Sstevel@tonic-gate }
72007c478bd9Sstevel@tonic-gate 
72017c478bd9Sstevel@tonic-gate static int
string_to_yyin(char * string)72027c478bd9Sstevel@tonic-gate string_to_yyin(char *string)
72037c478bd9Sstevel@tonic-gate {
72047c478bd9Sstevel@tonic-gate 	if ((yyin = tmpfile()) == NULL) {
7205bbec428eSgjelinek 		zone_perror(execname, Z_TEMP_FILE, B_TRUE);
72067c478bd9Sstevel@tonic-gate 		return (Z_ERR);
72077c478bd9Sstevel@tonic-gate 	}
72087c478bd9Sstevel@tonic-gate 	if (fwrite(string, strlen(string), 1, yyin) != 1) {
7209bbec428eSgjelinek 		zone_perror(execname, Z_TEMP_FILE, B_TRUE);
72107c478bd9Sstevel@tonic-gate 		return (Z_ERR);
72117c478bd9Sstevel@tonic-gate 	}
72127c478bd9Sstevel@tonic-gate 	if (fseek(yyin, 0, SEEK_SET) != 0) {
7213bbec428eSgjelinek 		zone_perror(execname, Z_TEMP_FILE, B_TRUE);
72147c478bd9Sstevel@tonic-gate 		return (Z_ERR);
72157c478bd9Sstevel@tonic-gate 	}
72167c478bd9Sstevel@tonic-gate 	return (Z_OK);
72177c478bd9Sstevel@tonic-gate }
72187c478bd9Sstevel@tonic-gate 
72197c478bd9Sstevel@tonic-gate /* This is the back-end helper function for read_input() below. */
72207c478bd9Sstevel@tonic-gate 
72217c478bd9Sstevel@tonic-gate static int
cleanup()72227c478bd9Sstevel@tonic-gate cleanup()
72237c478bd9Sstevel@tonic-gate {
72247c478bd9Sstevel@tonic-gate 	int answer;
72257c478bd9Sstevel@tonic-gate 	cmd_t *cmd;
72267c478bd9Sstevel@tonic-gate 
72277c478bd9Sstevel@tonic-gate 	if (!interactive_mode && !cmd_file_mode) {
72287c478bd9Sstevel@tonic-gate 		/*
72297c478bd9Sstevel@tonic-gate 		 * If we're not in interactive mode, and we're not in command
72307c478bd9Sstevel@tonic-gate 		 * file mode, then we must be in commands-from-the-command-line
72317c478bd9Sstevel@tonic-gate 		 * mode.  As such, we can't loop back and ask for more input.
72327c478bd9Sstevel@tonic-gate 		 * It was OK to prompt for such things as whether or not to
72337c478bd9Sstevel@tonic-gate 		 * really delete a zone in the command handler called from
72347c478bd9Sstevel@tonic-gate 		 * yyparse() above, but "really quit?" makes no sense in this
72357c478bd9Sstevel@tonic-gate 		 * context.  So disable prompting.
72367c478bd9Sstevel@tonic-gate 		 */
7237bbec428eSgjelinek 		ok_to_prompt = B_FALSE;
72387c478bd9Sstevel@tonic-gate 	}
72397c478bd9Sstevel@tonic-gate 	if (!global_scope) {
72407c478bd9Sstevel@tonic-gate 		if (!time_to_exit) {
72417c478bd9Sstevel@tonic-gate 			/*
72427c478bd9Sstevel@tonic-gate 			 * Just print a simple error message in the -1 case,
72437c478bd9Sstevel@tonic-gate 			 * since exit_func() already handles that case, and
72447c478bd9Sstevel@tonic-gate 			 * EOF means we are finished anyway.
72457c478bd9Sstevel@tonic-gate 			 */
7246bbec428eSgjelinek 			answer = ask_yesno(B_FALSE,
72477c478bd9Sstevel@tonic-gate 			    gettext("Resource incomplete; really quit"));
72487c478bd9Sstevel@tonic-gate 			if (answer == -1) {
72497c478bd9Sstevel@tonic-gate 				zerr(gettext("Resource incomplete."));
72507c478bd9Sstevel@tonic-gate 				return (Z_ERR);
72517c478bd9Sstevel@tonic-gate 			}
72527c478bd9Sstevel@tonic-gate 			if (answer != 1) {
72537c478bd9Sstevel@tonic-gate 				yyin = stdin;
72547c478bd9Sstevel@tonic-gate 				return (Z_REPEAT);
72557c478bd9Sstevel@tonic-gate 			}
72567c478bd9Sstevel@tonic-gate 		} else {
7257bbec428eSgjelinek 			saw_error = B_TRUE;
72587c478bd9Sstevel@tonic-gate 		}
72597c478bd9Sstevel@tonic-gate 	}
72607c478bd9Sstevel@tonic-gate 	/*
72617c478bd9Sstevel@tonic-gate 	 * Make sure we tried something and that the handle checks
72627c478bd9Sstevel@tonic-gate 	 * out, or we would get a false error trying to commit.
72637c478bd9Sstevel@tonic-gate 	 */
72647c478bd9Sstevel@tonic-gate 	if (need_to_commit && zonecfg_check_handle(handle) == Z_OK) {
72657c478bd9Sstevel@tonic-gate 		if ((cmd = alloc_cmd()) == NULL) {
7266bbec428eSgjelinek 			zone_perror(zone, Z_NOMEM, B_TRUE);
72677c478bd9Sstevel@tonic-gate 			return (Z_ERR);
72687c478bd9Sstevel@tonic-gate 		}
72697c478bd9Sstevel@tonic-gate 		cmd->cmd_argc = 0;
72707c478bd9Sstevel@tonic-gate 		cmd->cmd_argv[0] = NULL;
72717c478bd9Sstevel@tonic-gate 		commit_func(cmd);
72727c478bd9Sstevel@tonic-gate 		free_cmd(cmd);
72737c478bd9Sstevel@tonic-gate 		/*
72747c478bd9Sstevel@tonic-gate 		 * need_to_commit will get set back to FALSE if the
72757c478bd9Sstevel@tonic-gate 		 * configuration is saved successfully.
72767c478bd9Sstevel@tonic-gate 		 */
72777c478bd9Sstevel@tonic-gate 		if (need_to_commit) {
72787c478bd9Sstevel@tonic-gate 			if (force_exit) {
72797c478bd9Sstevel@tonic-gate 				zerr(gettext("Configuration not saved."));
72807c478bd9Sstevel@tonic-gate 				return (Z_ERR);
72817c478bd9Sstevel@tonic-gate 			}
7282bbec428eSgjelinek 			answer = ask_yesno(B_FALSE,
72837c478bd9Sstevel@tonic-gate 			    gettext("Configuration not saved; really quit"));
72847c478bd9Sstevel@tonic-gate 			if (answer == -1) {
72857c478bd9Sstevel@tonic-gate 				zerr(gettext("Configuration not saved."));
72867c478bd9Sstevel@tonic-gate 				return (Z_ERR);
72877c478bd9Sstevel@tonic-gate 			}
72887c478bd9Sstevel@tonic-gate 			if (answer != 1) {
7289bbec428eSgjelinek 				time_to_exit = B_FALSE;
72907c478bd9Sstevel@tonic-gate 				yyin = stdin;
72917c478bd9Sstevel@tonic-gate 				return (Z_REPEAT);
72927c478bd9Sstevel@tonic-gate 			}
72937c478bd9Sstevel@tonic-gate 		}
72947c478bd9Sstevel@tonic-gate 	}
72957c478bd9Sstevel@tonic-gate 	return ((need_to_commit || saw_error) ? Z_ERR : Z_OK);
72967c478bd9Sstevel@tonic-gate }
72977c478bd9Sstevel@tonic-gate 
72987c478bd9Sstevel@tonic-gate /*
72997c478bd9Sstevel@tonic-gate  * read_input() is the driver of this program.  It is a wrapper around
73007c478bd9Sstevel@tonic-gate  * yyparse(), printing appropriate prompts when needed, checking for
73017c478bd9Sstevel@tonic-gate  * exit conditions and reacting appropriately [the latter in its cleanup()
73027c478bd9Sstevel@tonic-gate  * helper function].
73037c478bd9Sstevel@tonic-gate  *
73047c478bd9Sstevel@tonic-gate  * Like most zonecfg functions, it returns Z_OK or Z_ERR, *or* Z_REPEAT
73057c478bd9Sstevel@tonic-gate  * so do_interactive() knows that we are not really done (i.e, we asked
73067c478bd9Sstevel@tonic-gate  * the user if we should really quit and the user said no).
73077c478bd9Sstevel@tonic-gate  */
73087c478bd9Sstevel@tonic-gate static int
read_input()73097c478bd9Sstevel@tonic-gate read_input()
73107c478bd9Sstevel@tonic-gate {
7311bbec428eSgjelinek 	boolean_t yyin_is_a_tty = isatty(fileno(yyin));
73127c478bd9Sstevel@tonic-gate 	/*
73137c478bd9Sstevel@tonic-gate 	 * The prompt is "e:z> " or "e:z:r> " where e is execname, z is zone
73147c478bd9Sstevel@tonic-gate 	 * and r is resource_scope: 5 is for the two ":"s + "> " + terminator.
73157c478bd9Sstevel@tonic-gate 	 */
73167c478bd9Sstevel@tonic-gate 	char prompt[MAXPATHLEN + ZONENAME_MAX + MAX_RT_STRLEN + 5], *line;
73177c478bd9Sstevel@tonic-gate 
73187c478bd9Sstevel@tonic-gate 	/* yyin should have been set to the appropriate (FILE *) if not stdin */
7319bbec428eSgjelinek 	newline_terminated = B_TRUE;
73207c478bd9Sstevel@tonic-gate 	for (;;) {
73217c478bd9Sstevel@tonic-gate 		if (yyin_is_a_tty) {
73227c478bd9Sstevel@tonic-gate 			if (newline_terminated) {
73237c478bd9Sstevel@tonic-gate 				if (global_scope)
73247c478bd9Sstevel@tonic-gate 					(void) snprintf(prompt, sizeof (prompt),
73257c478bd9Sstevel@tonic-gate 					    "%s:%s> ", execname, zone);
73267c478bd9Sstevel@tonic-gate 				else
73277c478bd9Sstevel@tonic-gate 					(void) snprintf(prompt, sizeof (prompt),
73287c478bd9Sstevel@tonic-gate 					    "%s:%s:%s> ", execname, zone,
73297c478bd9Sstevel@tonic-gate 					    rt_to_str(resource_scope));
73307c478bd9Sstevel@tonic-gate 			}
73317c478bd9Sstevel@tonic-gate 			/*
73327c478bd9Sstevel@tonic-gate 			 * If the user hits ^C then we want to catch it and
73337c478bd9Sstevel@tonic-gate 			 * start over.  If the user hits EOF then we want to
73347c478bd9Sstevel@tonic-gate 			 * bail out.
73357c478bd9Sstevel@tonic-gate 			 */
73367c478bd9Sstevel@tonic-gate 			line = gl_get_line(gl, prompt, NULL, -1);
73377c478bd9Sstevel@tonic-gate 			if (gl_return_status(gl) == GLR_SIGNAL) {
73387c478bd9Sstevel@tonic-gate 				gl_abandon_line(gl);
73397c478bd9Sstevel@tonic-gate 				continue;
73407c478bd9Sstevel@tonic-gate 			}
73417c478bd9Sstevel@tonic-gate 			if (line == NULL)
73427c478bd9Sstevel@tonic-gate 				break;
73437c478bd9Sstevel@tonic-gate 			(void) string_to_yyin(line);
73447c478bd9Sstevel@tonic-gate 			while (!feof(yyin))
73457c478bd9Sstevel@tonic-gate 				yyparse();
73467c478bd9Sstevel@tonic-gate 		} else {
73477c478bd9Sstevel@tonic-gate 			yyparse();
73487c478bd9Sstevel@tonic-gate 		}
73497c478bd9Sstevel@tonic-gate 		/* Bail out on an error in command file mode. */
73507c478bd9Sstevel@tonic-gate 		if (saw_error && cmd_file_mode && !interactive_mode)
7351bbec428eSgjelinek 			time_to_exit = B_TRUE;
73527c478bd9Sstevel@tonic-gate 		if (time_to_exit || (!yyin_is_a_tty && feof(yyin)))
73537c478bd9Sstevel@tonic-gate 			break;
73547c478bd9Sstevel@tonic-gate 	}
73557c478bd9Sstevel@tonic-gate 	return (cleanup());
73567c478bd9Sstevel@tonic-gate }
73577c478bd9Sstevel@tonic-gate 
73587c478bd9Sstevel@tonic-gate /*
73597c478bd9Sstevel@tonic-gate  * This function is used in the zonecfg-interactive-mode scenario: it just
73607c478bd9Sstevel@tonic-gate  * calls read_input() until we are done.
73617c478bd9Sstevel@tonic-gate  */
73627c478bd9Sstevel@tonic-gate 
73637c478bd9Sstevel@tonic-gate static int
do_interactive(void)73647c478bd9Sstevel@tonic-gate do_interactive(void)
73657c478bd9Sstevel@tonic-gate {
73667c478bd9Sstevel@tonic-gate 	int err;
73677c478bd9Sstevel@tonic-gate 
7368bbec428eSgjelinek 	interactive_mode = B_TRUE;
73697c478bd9Sstevel@tonic-gate 	if (!read_only_mode) {
73707c478bd9Sstevel@tonic-gate 		/*
73717c478bd9Sstevel@tonic-gate 		 * Try to set things up proactively in interactive mode, so
73727c478bd9Sstevel@tonic-gate 		 * that if the zone in question does not exist yet, we can
73737c478bd9Sstevel@tonic-gate 		 * provide the user with a clue.
73747c478bd9Sstevel@tonic-gate 		 */
7375bbec428eSgjelinek 		(void) initialize(B_FALSE);
73767c478bd9Sstevel@tonic-gate 	}
7377087719fdSdp 	do {
73787c478bd9Sstevel@tonic-gate 		err = read_input();
7379087719fdSdp 	} while (err == Z_REPEAT);
73807c478bd9Sstevel@tonic-gate 	return (err);
73817c478bd9Sstevel@tonic-gate }
73827c478bd9Sstevel@tonic-gate 
73837c478bd9Sstevel@tonic-gate /*
73847c478bd9Sstevel@tonic-gate  * cmd_file is slightly more complicated, as it has to open the command file
73857c478bd9Sstevel@tonic-gate  * and set yyin appropriately.  Once that is done, though, it just calls
73867c478bd9Sstevel@tonic-gate  * read_input(), and only once, since prompting is not possible.
73877c478bd9Sstevel@tonic-gate  */
73887c478bd9Sstevel@tonic-gate 
73897c478bd9Sstevel@tonic-gate static int
cmd_file(char * file)73907c478bd9Sstevel@tonic-gate cmd_file(char *file)
73917c478bd9Sstevel@tonic-gate {
73927c478bd9Sstevel@tonic-gate 	FILE *infile;
73937c478bd9Sstevel@tonic-gate 	int err;
73947c478bd9Sstevel@tonic-gate 	struct stat statbuf;
7395bbec428eSgjelinek 	boolean_t using_real_file = (strcmp(file, "-") != 0);
73967c478bd9Sstevel@tonic-gate 
73977c478bd9Sstevel@tonic-gate 	if (using_real_file) {
73987c478bd9Sstevel@tonic-gate 		/*
73997c478bd9Sstevel@tonic-gate 		 * zerr() prints a line number in cmd_file_mode, which we do
74007c478bd9Sstevel@tonic-gate 		 * not want here, so temporarily unset it.
74017c478bd9Sstevel@tonic-gate 		 */
7402bbec428eSgjelinek 		cmd_file_mode = B_FALSE;
74037c478bd9Sstevel@tonic-gate 		if ((infile = fopen(file, "r")) == NULL) {
74047c478bd9Sstevel@tonic-gate 			zerr(gettext("could not open file %s: %s"),
74057c478bd9Sstevel@tonic-gate 			    file, strerror(errno));
74067c478bd9Sstevel@tonic-gate 			return (Z_ERR);
74077c478bd9Sstevel@tonic-gate 		}
74087c478bd9Sstevel@tonic-gate 		if ((err = fstat(fileno(infile), &statbuf)) != 0) {
74097c478bd9Sstevel@tonic-gate 			zerr(gettext("could not stat file %s: %s"),
74107c478bd9Sstevel@tonic-gate 			    file, strerror(errno));
74117c478bd9Sstevel@tonic-gate 			err = Z_ERR;
74127c478bd9Sstevel@tonic-gate 			goto done;
74137c478bd9Sstevel@tonic-gate 		}
74147c478bd9Sstevel@tonic-gate 		if (!S_ISREG(statbuf.st_mode)) {
74157c478bd9Sstevel@tonic-gate 			zerr(gettext("%s is not a regular file."), file);
74167c478bd9Sstevel@tonic-gate 			err = Z_ERR;
74177c478bd9Sstevel@tonic-gate 			goto done;
74187c478bd9Sstevel@tonic-gate 		}
74197c478bd9Sstevel@tonic-gate 		yyin = infile;
7420bbec428eSgjelinek 		cmd_file_mode = B_TRUE;
7421bbec428eSgjelinek 		ok_to_prompt = B_FALSE;
74227c478bd9Sstevel@tonic-gate 	} else {
74237c478bd9Sstevel@tonic-gate 		/*
74247c478bd9Sstevel@tonic-gate 		 * "-f -" is essentially the same as interactive mode,
74257c478bd9Sstevel@tonic-gate 		 * so treat it that way.
74267c478bd9Sstevel@tonic-gate 		 */
7427bbec428eSgjelinek 		interactive_mode = B_TRUE;
74287c478bd9Sstevel@tonic-gate 	}
74297c478bd9Sstevel@tonic-gate 	/* Z_REPEAT is for interactive mode; treat it like Z_ERR here. */
74307c478bd9Sstevel@tonic-gate 	if ((err = read_input()) == Z_REPEAT)
74317c478bd9Sstevel@tonic-gate 		err = Z_ERR;
74327c478bd9Sstevel@tonic-gate done:
74337c478bd9Sstevel@tonic-gate 	if (using_real_file)
74347c478bd9Sstevel@tonic-gate 		(void) fclose(infile);
74357c478bd9Sstevel@tonic-gate 	return (err);
74367c478bd9Sstevel@tonic-gate }
74377c478bd9Sstevel@tonic-gate 
74387c478bd9Sstevel@tonic-gate /*
74397c478bd9Sstevel@tonic-gate  * Since yacc is based on reading from a (FILE *) whereas what we get from
74407c478bd9Sstevel@tonic-gate  * the command line is in argv format, we need to convert when the user
74417c478bd9Sstevel@tonic-gate  * gives us commands directly from the command line.  That is done here by
74427c478bd9Sstevel@tonic-gate  * concatenating the argv list into a space-separated string, writing it
74437c478bd9Sstevel@tonic-gate  * to a temp file, and rewinding the file so yyin can be set to it.  Then
74447c478bd9Sstevel@tonic-gate  * we call read_input(), and only once, since prompting about whether to
74457c478bd9Sstevel@tonic-gate  * continue or quit would make no sense in this context.
74467c478bd9Sstevel@tonic-gate  */
74477c478bd9Sstevel@tonic-gate 
74487c478bd9Sstevel@tonic-gate static int
one_command_at_a_time(int argc,char * argv[])74497c478bd9Sstevel@tonic-gate one_command_at_a_time(int argc, char *argv[])
74507c478bd9Sstevel@tonic-gate {
74517c478bd9Sstevel@tonic-gate 	char *command;
74527c478bd9Sstevel@tonic-gate 	size_t len = 2; /* terminal \n\0 */
74537c478bd9Sstevel@tonic-gate 	int i, err;
74547c478bd9Sstevel@tonic-gate 
74557c478bd9Sstevel@tonic-gate 	for (i = 0; i < argc; i++)
74567c478bd9Sstevel@tonic-gate 		len += strlen(argv[i]) + 1;
74577c478bd9Sstevel@tonic-gate 	if ((command = malloc(len)) == NULL) {
7458bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
74597c478bd9Sstevel@tonic-gate 		return (Z_ERR);
74607c478bd9Sstevel@tonic-gate 	}
74617c478bd9Sstevel@tonic-gate 	(void) strlcpy(command, argv[0], len);
74627c478bd9Sstevel@tonic-gate 	for (i = 1; i < argc; i++) {
74637c478bd9Sstevel@tonic-gate 		(void) strlcat(command, " ", len);
74647c478bd9Sstevel@tonic-gate 		(void) strlcat(command, argv[i], len);
74657c478bd9Sstevel@tonic-gate 	}
74667c478bd9Sstevel@tonic-gate 	(void) strlcat(command, "\n", len);
74677c478bd9Sstevel@tonic-gate 	err = string_to_yyin(command);
74687c478bd9Sstevel@tonic-gate 	free(command);
74697c478bd9Sstevel@tonic-gate 	if (err != Z_OK)
74707c478bd9Sstevel@tonic-gate 		return (err);
74717c478bd9Sstevel@tonic-gate 	while (!feof(yyin))
74727c478bd9Sstevel@tonic-gate 		yyparse();
74737c478bd9Sstevel@tonic-gate 	return (cleanup());
74747c478bd9Sstevel@tonic-gate }
74757c478bd9Sstevel@tonic-gate 
74767c478bd9Sstevel@tonic-gate static char *
get_execbasename(char * execfullname)74777c478bd9Sstevel@tonic-gate get_execbasename(char *execfullname)
74787c478bd9Sstevel@tonic-gate {
74797c478bd9Sstevel@tonic-gate 	char *last_slash, *execbasename;
74807c478bd9Sstevel@tonic-gate 
74817c478bd9Sstevel@tonic-gate 	/* guard against '/' at end of command invocation */
74827c478bd9Sstevel@tonic-gate 	for (;;) {
74837c478bd9Sstevel@tonic-gate 		last_slash = strrchr(execfullname, '/');
74847c478bd9Sstevel@tonic-gate 		if (last_slash == NULL) {
74857c478bd9Sstevel@tonic-gate 			execbasename = execfullname;
74867c478bd9Sstevel@tonic-gate 			break;
74877c478bd9Sstevel@tonic-gate 		} else {
74887c478bd9Sstevel@tonic-gate 			execbasename = last_slash + 1;
74897c478bd9Sstevel@tonic-gate 			if (*execbasename == '\0') {
74907c478bd9Sstevel@tonic-gate 				*last_slash = '\0';
74917c478bd9Sstevel@tonic-gate 				continue;
74927c478bd9Sstevel@tonic-gate 			}
74937c478bd9Sstevel@tonic-gate 			break;
74947c478bd9Sstevel@tonic-gate 		}
74957c478bd9Sstevel@tonic-gate 	}
74967c478bd9Sstevel@tonic-gate 	return (execbasename);
74977c478bd9Sstevel@tonic-gate }
74987c478bd9Sstevel@tonic-gate 
74997c478bd9Sstevel@tonic-gate int
main(int argc,char * argv[])75007c478bd9Sstevel@tonic-gate main(int argc, char *argv[])
75017c478bd9Sstevel@tonic-gate {
75027c478bd9Sstevel@tonic-gate 	int err, arg;
7503555afedfScarlsonj 	struct stat st;
75047c478bd9Sstevel@tonic-gate 
75057c478bd9Sstevel@tonic-gate 	/* This must be before anything goes to stdout. */
75067c478bd9Sstevel@tonic-gate 	setbuf(stdout, NULL);
75077c478bd9Sstevel@tonic-gate 
7508bbec428eSgjelinek 	saw_error = B_FALSE;
7509bbec428eSgjelinek 	cmd_file_mode = B_FALSE;
75107c478bd9Sstevel@tonic-gate 	execname = get_execbasename(argv[0]);
75117c478bd9Sstevel@tonic-gate 
75127c478bd9Sstevel@tonic-gate 	(void) setlocale(LC_ALL, "");
75137c478bd9Sstevel@tonic-gate 	(void) textdomain(TEXT_DOMAIN);
75147c478bd9Sstevel@tonic-gate 
75157c478bd9Sstevel@tonic-gate 	if (getzoneid() != GLOBAL_ZONEID) {
75167c478bd9Sstevel@tonic-gate 		zerr(gettext("%s can only be run from the global zone."),
75177c478bd9Sstevel@tonic-gate 		    execname);
75187c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
75197c478bd9Sstevel@tonic-gate 	}
75207c478bd9Sstevel@tonic-gate 
75217c478bd9Sstevel@tonic-gate 	if (argc < 2) {
7522bbec428eSgjelinek 		usage(B_FALSE, HELP_USAGE | HELP_SUBCMDS);
75237c478bd9Sstevel@tonic-gate 		exit(Z_USAGE);
75247c478bd9Sstevel@tonic-gate 	}
75257c478bd9Sstevel@tonic-gate 	if (strcmp(argv[1], cmd_to_str(CMD_HELP)) == 0) {
75267c478bd9Sstevel@tonic-gate 		(void) one_command_at_a_time(argc - 1, &(argv[1]));
75277c478bd9Sstevel@tonic-gate 		exit(Z_OK);
75287c478bd9Sstevel@tonic-gate 	}
75297c478bd9Sstevel@tonic-gate 
7530555afedfScarlsonj 	while ((arg = getopt(argc, argv, "?f:R:z:")) != EOF) {
75317c478bd9Sstevel@tonic-gate 		switch (arg) {
75327c478bd9Sstevel@tonic-gate 		case '?':
75337c478bd9Sstevel@tonic-gate 			if (optopt == '?')
7534bbec428eSgjelinek 				usage(B_TRUE, HELP_USAGE | HELP_SUBCMDS);
75357c478bd9Sstevel@tonic-gate 			else
7536bbec428eSgjelinek 				usage(B_FALSE, HELP_USAGE);
75377c478bd9Sstevel@tonic-gate 			exit(Z_USAGE);
75387c478bd9Sstevel@tonic-gate 			/* NOTREACHED */
75397c478bd9Sstevel@tonic-gate 		case 'f':
75407c478bd9Sstevel@tonic-gate 			cmd_file_name = optarg;
7541bbec428eSgjelinek 			cmd_file_mode = B_TRUE;
75427c478bd9Sstevel@tonic-gate 			break;
7543555afedfScarlsonj 		case 'R':
7544555afedfScarlsonj 			if (*optarg != '/') {
7545555afedfScarlsonj 				zerr(gettext("root path must be absolute: %s"),
7546555afedfScarlsonj 				    optarg);
7547555afedfScarlsonj 				exit(Z_USAGE);
7548555afedfScarlsonj 			}
7549555afedfScarlsonj 			if (stat(optarg, &st) == -1 || !S_ISDIR(st.st_mode)) {
7550555afedfScarlsonj 				zerr(gettext(
7551555afedfScarlsonj 				    "root path must be a directory: %s"),
7552555afedfScarlsonj 				    optarg);
7553555afedfScarlsonj 				exit(Z_USAGE);
7554555afedfScarlsonj 			}
7555555afedfScarlsonj 			zonecfg_set_root(optarg);
7556555afedfScarlsonj 			break;
75577c478bd9Sstevel@tonic-gate 		case 'z':
75580209230bSgjelinek 			if (strcmp(optarg, GLOBAL_ZONENAME) == 0) {
7559bbec428eSgjelinek 				global_zone = B_TRUE;
75600209230bSgjelinek 			} else if (zonecfg_validate_zonename(optarg) != Z_OK) {
7561bbec428eSgjelinek 				zone_perror(optarg, Z_BOGUS_ZONE_NAME, B_TRUE);
7562bbec428eSgjelinek 				usage(B_FALSE, HELP_SYNTAX);
7563087719fdSdp 				exit(Z_USAGE);
7564087719fdSdp 			}
7565087719fdSdp 			(void) strlcpy(zone, optarg, sizeof (zone));
7566087719fdSdp 			(void) strlcpy(revert_zone, optarg, sizeof (zone));
75677c478bd9Sstevel@tonic-gate 			break;
75687c478bd9Sstevel@tonic-gate 		default:
7569bbec428eSgjelinek 			usage(B_FALSE, HELP_USAGE);
75707c478bd9Sstevel@tonic-gate 			exit(Z_USAGE);
75717c478bd9Sstevel@tonic-gate 		}
75727c478bd9Sstevel@tonic-gate 	}
75737c478bd9Sstevel@tonic-gate 
7574087719fdSdp 	if (optind > argc || strcmp(zone, "") == 0) {
7575bbec428eSgjelinek 		usage(B_FALSE, HELP_USAGE);
75767c478bd9Sstevel@tonic-gate 		exit(Z_USAGE);
75777c478bd9Sstevel@tonic-gate 	}
75787c478bd9Sstevel@tonic-gate 
7579087719fdSdp 	if ((err = zonecfg_access(zone, W_OK)) == Z_OK) {
7580bbec428eSgjelinek 		read_only_mode = B_FALSE;
7581087719fdSdp 	} else if (err == Z_ACCES) {
7582bbec428eSgjelinek 		read_only_mode = B_TRUE;
75837c478bd9Sstevel@tonic-gate 		/* skip this message in one-off from command line mode */
75847c478bd9Sstevel@tonic-gate 		if (optind == argc)
75857c478bd9Sstevel@tonic-gate 			(void) fprintf(stderr, gettext("WARNING: you do not "
75867c478bd9Sstevel@tonic-gate 			    "have write access to this zone's configuration "
75877c478bd9Sstevel@tonic-gate 			    "file;\ngoing into read-only mode.\n"));
7588087719fdSdp 	} else {
7589087719fdSdp 		fprintf(stderr, "%s: Could not access zone configuration "
7590087719fdSdp 		    "store: %s\n", execname, zonecfg_strerror(err));
7591087719fdSdp 		exit(Z_ERR);
75927c478bd9Sstevel@tonic-gate 	}
75937c478bd9Sstevel@tonic-gate 
75947c478bd9Sstevel@tonic-gate 	if ((handle = zonecfg_init_handle()) == NULL) {
7595bbec428eSgjelinek 		zone_perror(execname, Z_NOMEM, B_TRUE);
75967c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
75977c478bd9Sstevel@tonic-gate 	}
75987c478bd9Sstevel@tonic-gate 
75997c478bd9Sstevel@tonic-gate 	/*
76007c478bd9Sstevel@tonic-gate 	 * This may get set back to FALSE again in cmd_file() if cmd_file_name
76017c478bd9Sstevel@tonic-gate 	 * is a "real" file as opposed to "-" (i.e. meaning use stdin).
76027c478bd9Sstevel@tonic-gate 	 */
76037c478bd9Sstevel@tonic-gate 	if (isatty(STDIN_FILENO))
7604bbec428eSgjelinek 		ok_to_prompt = B_TRUE;
76057c478bd9Sstevel@tonic-gate 	if ((gl = new_GetLine(MAX_LINE_LEN, MAX_CMD_HIST)) == NULL)
76067c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
76077c478bd9Sstevel@tonic-gate 	if (gl_customize_completion(gl, NULL, cmd_cpl_fn) != 0)
76087c478bd9Sstevel@tonic-gate 		exit(Z_ERR);
76097c478bd9Sstevel@tonic-gate 	(void) sigset(SIGINT, SIG_IGN);
76107c478bd9Sstevel@tonic-gate 	if (optind == argc) {
76117c478bd9Sstevel@tonic-gate 		if (!cmd_file_mode)
76127c478bd9Sstevel@tonic-gate 			err = do_interactive();
76137c478bd9Sstevel@tonic-gate 		else
76147c478bd9Sstevel@tonic-gate 			err = cmd_file(cmd_file_name);
76157c478bd9Sstevel@tonic-gate 	} else {
76167c478bd9Sstevel@tonic-gate 		err = one_command_at_a_time(argc - optind, &(argv[optind]));
76177c478bd9Sstevel@tonic-gate 	}
76187c478bd9Sstevel@tonic-gate 	zonecfg_fini_handle(handle);
76199acbbeafSnn35248 	if (brand != NULL)
76209acbbeafSnn35248 		brand_close(brand);
76217c478bd9Sstevel@tonic-gate 	(void) del_GetLine(gl);
76227c478bd9Sstevel@tonic-gate 	return (err);
76237c478bd9Sstevel@tonic-gate }
7624