diff options
| author | Jerry Jelinek <jerry.jelinek@joyent.com> | 2011-02-14 16:32:12 -0700 |
|---|---|---|
| committer | Jerry Jelinek <jerry.jelinek@joyent.com> | 2011-02-14 16:32:12 -0700 |
| commit | 2e90f839a513a397c7bc8eb94b3b7e0139120cb4 (patch) | |
| tree | ace084a772f6bf1420d54798a6ea703b257209c1 /usr/src/cmd | |
| parent | 18a2608d74a29b0a9383e81186b5db6e7a7cd804 (diff) | |
| download | illumos-joyent-2e90f839a513a397c7bc8eb94b3b7e0139120cb4.tar.gz | |
OS-224 add more zonecfg net properties
Diffstat (limited to 'usr/src/cmd')
| -rw-r--r-- | usr/src/cmd/zoneadm/zoneadm.c | 2 | ||||
| -rw-r--r-- | usr/src/cmd/zoneadmd/zoneadmd.c | 72 | ||||
| -rw-r--r-- | usr/src/cmd/zonecfg/zonecfg.c | 200 | ||||
| -rw-r--r-- | usr/src/cmd/zonecfg/zonecfg.h | 3 | ||||
| -rw-r--r-- | usr/src/cmd/zonecfg/zonecfg_grammar.y | 5 | ||||
| -rw-r--r-- | usr/src/cmd/zonecfg/zonecfg_lex.l | 10 |
6 files changed, 263 insertions, 29 deletions
diff --git a/usr/src/cmd/zoneadm/zoneadm.c b/usr/src/cmd/zoneadm/zoneadm.c index 9931505a10..342cabb0f3 100644 --- a/usr/src/cmd/zoneadm/zoneadm.c +++ b/usr/src/cmd/zoneadm/zoneadm.c @@ -21,7 +21,7 @@ /* * Copyright (c) 2003, 2010, Oracle and/or its affiliates. All rights reserved. - * Copyright (c) 201l, Joyent Inc. All rights reserved. + * Copyright (c) 2011, Joyent Inc. All rights reserved. */ /* diff --git a/usr/src/cmd/zoneadmd/zoneadmd.c b/usr/src/cmd/zoneadmd/zoneadmd.c index 9b7c1247d8..9a28e6c3f5 100644 --- a/usr/src/cmd/zoneadmd/zoneadmd.c +++ b/usr/src/cmd/zoneadmd/zoneadmd.c @@ -740,6 +740,73 @@ mount_early_fs(void *data, const char *spec, const char *dir, } /* + * env variable name format + * _ZONECFG;{resource name};{identifying attr. name};{property name} + */ +static void +set_zonecfg_env(char *phys, char *name, char *val) +{ + char nm[MAXNAMELEN]; + + (void) snprintf(nm, sizeof (nm), "_ZONECFG;net;%s;%s", phys, name); + + (void) setenv(nm, val, 1); +} + +/* + * Export zonecfg network settings into environment for the boot and state + * change hooks. + * + * We could export more of the config in the future, as necessary. + */ +static int +setup_subproc_env() +{ + int res; + zone_dochandle_t handle; + struct zone_nwiftab ntab; + + if ((handle = zonecfg_init_handle()) == NULL) + exit(Z_NOMEM); + + if ((res = zonecfg_get_handle(zone_name, handle)) != Z_OK) + goto done; + + if ((res = zonecfg_setnwifent(handle)) != Z_OK) + goto done; + + while (zonecfg_getnwifent(handle, &ntab) == Z_OK) { + struct zone_nwif_attrtab *np; + char *phys; + + phys = ntab.zone_nwif_physical; + + set_zonecfg_env(phys, "physical", + ntab.zone_nwif_physical); + + set_zonecfg_env(phys, "address", ntab.zone_nwif_address); + set_zonecfg_env(phys, "allowed-address", + ntab.zone_nwif_allowed_address); + set_zonecfg_env(phys, "defrouter", ntab.zone_nwif_defrouter); + set_zonecfg_env(phys, "global-nic", ntab.zone_nwif_gnic); + set_zonecfg_env(phys, "mac-addr", ntab.zone_nwif_mac); + set_zonecfg_env(phys, "vlan-id", ntab.zone_nwif_vlan_id); + + for (np = ntab.zone_nwif_attrp; np != NULL; + np = np->zone_nwif_attr_next) + set_zonecfg_env(phys, np->zone_nwif_attr_name, + np->zone_nwif_attr_value); + } + + (void) zonecfg_endnwifent(handle); + res = Z_OK; + +done: + zonecfg_fini_handle(handle); + return (res); +} + +/* * If retstr is not NULL, the output of the subproc is returned in the str, * otherwise it is output using zerror(). Any memory allocated for retstr * should be freed by the caller. @@ -764,6 +831,11 @@ do_subproc(zlog_t *zlogp, char *cmdbuf, char **retstr) inbuf = buf; } + if (setup_subproc_env() != Z_OK) { + zerror(zlogp, B_FALSE, "failed to setup environment"); + return (-1); + } + file = popen(cmdbuf, "r"); if (file == NULL) { zerror(zlogp, B_TRUE, "could not launch: %s", cmdbuf); diff --git a/usr/src/cmd/zonecfg/zonecfg.c b/usr/src/cmd/zonecfg/zonecfg.c index 76a63fbb57..981a45b82b 100644 --- a/usr/src/cmd/zonecfg/zonecfg.c +++ b/usr/src/cmd/zonecfg/zonecfg.c @@ -238,6 +238,7 @@ char *prop_types[] = { "mac-addr", "vlan-id", "global-nic", + "property", NULL }; @@ -412,6 +413,13 @@ static const char *net_res_scope_cmds[] = { "exit", "help", "info", + "add property ", + "clear allowed-address", + "clear defrouter", + "clear global-nic", + "clear mac-addr", + "clear vlan-id", + "remove property ", "set address=", "set allowed-address=", "set defrouter=", @@ -419,11 +427,6 @@ static const char *net_res_scope_cmds[] = { "set mac-addr=", "set physical=", "set vlan-id=", - "clear allowed-address", - "clear defrouter", - "clear global-nic", - "clear mac-addr", - "clear vlan-id", NULL }; @@ -1009,6 +1012,9 @@ usage(boolean_t verbose, uint_t flags) (void) fprintf(fp, gettext("Valid commands:\n")); (void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET), pt_to_str(PT_ADDRESS), gettext("<IP-address>")); + (void) fprintf(fp, "\t%s %s (%s=<value>,%s=<value>)\n", + cmd_to_str(CMD_ADD), pt_to_str(PT_NPROP), + pt_to_str(PT_NAME), pt_to_str(PT_VALUE)); (void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET), pt_to_str(PT_ALLOWED_ADDRESS), gettext("<IP-address>")); @@ -1017,9 +1023,9 @@ usage(boolean_t verbose, uint_t flags) (void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET), pt_to_str(PT_MAC), gettext("<mac-address>")); (void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET), - pt_to_str(PT_VLANID), gettext("<vlan ID>")); - (void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET), pt_to_str(PT_GNIC), gettext("<global zone NIC>")); + (void) fprintf(fp, "\t%s %s=%s\n", cmd_to_str(CMD_SET), + pt_to_str(PT_VLANID), gettext("<vlan ID>")); (void) fprintf(fp, gettext("See ifconfig(1M) for " "details of the <interface> string.\n")); (void) fprintf(fp, gettext("%s %s is valid " @@ -1277,12 +1283,12 @@ usage(boolean_t verbose, uint_t flags) rt_to_str(RT_FS), pt_to_str(PT_DIR), pt_to_str(PT_SPECIAL), pt_to_str(PT_RAW), pt_to_str(PT_TYPE), pt_to_str(PT_OPTIONS)); - (void) fprintf(fp, "\t%s\t\t%s, %s, %s, %s, %s, %s|%s\n", + (void) fprintf(fp, "\t%s\t\t%s, %s, %s, %s, %s, %s, %s %s\n", rt_to_str(RT_NET), pt_to_str(PT_ADDRESS), pt_to_str(PT_ALLOWED_ADDRESS), - pt_to_str(PT_PHYSICAL), pt_to_str(PT_MAC), - pt_to_str(PT_VLANID), pt_to_str(PT_GNIC), - pt_to_str(PT_DEFROUTER)); + pt_to_str(PT_GNIC), pt_to_str(PT_MAC), + pt_to_str(PT_PHYSICAL), pt_to_str(PT_NPROP), + pt_to_str(PT_VLANID), pt_to_str(PT_DEFROUTER)); (void) fprintf(fp, "\t%s\t\t%s\n", rt_to_str(RT_DEVICE), pt_to_str(PT_MATCH)); (void) fprintf(fp, "\t%s\t\t%s, %s\n", rt_to_str(RT_RCTL), @@ -1641,7 +1647,7 @@ create_func(cmd_t *cmd) break; case 'X': (void) snprintf(zone_template, sizeof (zone_template), - "/etc/zones/%s.xml", zone); + "%s/%s.xml", ZONE_CONFIG_ROOT, zone); err = zonecfg_get_xml_handle(zone_template, handle); if (err != Z_OK) { zone_perror(execname, err, B_TRUE); @@ -1754,6 +1760,7 @@ export_func(cmd_t *cmd) struct zone_psettab psettab; struct zone_mcaptab mcaptab; struct zone_rctlvaltab *valptr; + struct zone_nwif_attrtab *nap; struct zone_admintab admintab; int err, arg; char zonepath[MAXPATHLEN], outfile[MAXPATHLEN], pool[MAXNAMELEN]; @@ -1927,6 +1934,13 @@ export_func(cmd_t *cmd) export_prop(of, PT_VLANID, nwiftab.zone_nwif_vlan_id); export_prop(of, PT_GNIC, nwiftab.zone_nwif_gnic); export_prop(of, PT_DEFROUTER, nwiftab.zone_nwif_defrouter); + for (nap = nwiftab.zone_nwif_attrp; nap != NULL; + nap = nap->zone_nwif_attr_next) { + fprintf(of, "%s %s (%s=%s,%s=\"%s\")\n", + cmd_to_str(CMD_ADD), pt_to_str(PT_NPROP), + pt_to_str(PT_NAME), nap->zone_nwif_attr_name, + pt_to_str(PT_VALUE), nap->zone_nwif_attr_value); + } (void) fprintf(of, "%s\n", cmd_to_str(CMD_END)); } (void) zonecfg_endnwifent(handle); @@ -2318,6 +2332,65 @@ bad: } static void +do_nwif_attr(complex_property_ptr_t cpp) +{ + complex_property_ptr_t cp; + struct zone_nwif_attrtab *np; + int err; + boolean_t seen_name = B_FALSE, seen_value = B_FALSE; + + if ((np = calloc(1, sizeof (struct zone_nwif_attrtab))) == NULL) { + zone_perror(zone, Z_NOMEM, B_TRUE); + exit(Z_ERR); + } + + for (cp = cpp; cp != NULL; cp = cp->cp_next) { + switch (cp->cp_type) { + case PT_NAME: + if (seen_name) { + zerr(gettext("%s already specified"), + pt_to_str(PT_NAME)); + goto bad; + } + (void) strlcpy(np->zone_nwif_attr_name, cp->cp_value, + sizeof (np->zone_nwif_attr_name)); + seen_name = B_TRUE; + break; + case PT_VALUE: + if (seen_value) { + zerr(gettext("%s already specified"), + pt_to_str(PT_VALUE)); + goto bad; + } + (void) strlcpy(np->zone_nwif_attr_value, cp->cp_value, + sizeof (np->zone_nwif_attr_value)); + seen_value = B_TRUE; + break; + default: + zone_perror(pt_to_str(PT_NPROP), Z_NO_PROPERTY_TYPE, + B_TRUE); + long_usage(CMD_ADD, B_TRUE); + usage(B_FALSE, HELP_PROPS); + zonecfg_free_nwif_attr_list(np); + return; + } + } + + if (!seen_name) + zerr(gettext("%s not specified"), pt_to_str(PT_NAME)); + if (!seen_value) + zerr(gettext("%s not specified"), pt_to_str(PT_VALUE)); + + err = zonecfg_add_nwif_attr(&in_progress_nwiftab, np); + if (err != Z_OK) + zone_perror(pt_to_str(PT_NPROP), err, B_TRUE); + return; + +bad: + zonecfg_free_nwif_attr_list(np); +} + +static void add_property(cmd_t *cmd) { char *prop_id; @@ -2384,6 +2457,24 @@ add_property(cmd_t *cmd) } } return; + case RT_NET: + if (prop_type != PT_NPROP) { + zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, + B_TRUE); + long_usage(CMD_ADD, B_TRUE); + usage(B_FALSE, HELP_PROPS); + return; + } + pp = cmd->cmd_property_ptr[0]; + if (pp->pv_type != PROP_VAL_COMPLEX) { + zerr(gettext("A %s value was expected here."), + pvt_to_str(PROP_VAL_COMPLEX)); + saw_error = B_TRUE; + return; + } + + do_nwif_attr(pp->pv_complex); + return; case RT_RCTL: if (prop_type != PT_VALUE) { zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, @@ -2488,8 +2579,9 @@ add_func(cmd_t *cmd) resource_scope = cmd->cmd_res_type; end_op = CMD_ADD; add_resource(cmd); - } else + } else { add_property(cmd); + } } /* @@ -3383,6 +3475,7 @@ remove_property(cmd_t *cmd) int err, res_type, prop_type; property_value_ptr_t pp; struct zone_rctlvaltab *rctlvaltab; + struct zone_nwif_attrtab *np; complex_property_ptr_t cx; res_type = resource_scope; @@ -3443,6 +3536,49 @@ remove_property(cmd_t *cmd) } } return; + case RT_NET: + if (prop_type != PT_NPROP) { + zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, + B_TRUE); + long_usage(CMD_REMOVE, B_TRUE); + usage(B_FALSE, HELP_PROPS); + return; + } + pp = cmd->cmd_property_ptr[0]; + if (pp->pv_type != PROP_VAL_COMPLEX) { + zerr(gettext("A %s value was expected here."), + pvt_to_str(PROP_VAL_COMPLEX)); + saw_error = B_TRUE; + return; + } + + np = alloca(sizeof (struct zone_nwif_attrtab)); + for (cx = pp->pv_complex; cx != NULL; cx = cx->cp_next) { + switch (cx->cp_type) { + case PT_NAME: + (void) strlcpy(np->zone_nwif_attr_name, + cx->cp_value, + sizeof (np->zone_nwif_attr_name)); + break; + case PT_VALUE: + (void) strlcpy(np->zone_nwif_attr_value, + cx->cp_value, + sizeof (np->zone_nwif_attr_value)); + break; + default: + zone_perror(pt_to_str(prop_type), + Z_NO_PROPERTY_TYPE, B_TRUE); + long_usage(CMD_REMOVE, B_TRUE); + usage(B_FALSE, HELP_PROPS); + return; + } + } + np->zone_nwif_attr_next = NULL; + + err = zonecfg_remove_nwif_attr(&in_progress_nwiftab, np); + if (err != Z_OK) + zone_perror(pt_to_str(prop_type), err, B_TRUE); + return; case RT_RCTL: if (prop_type != PT_VALUE) { zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, @@ -3495,18 +3631,6 @@ remove_property(cmd_t *cmd) zone_perror(pt_to_str(prop_type), err, B_TRUE); zonecfg_free_rctl_value_list(rctlvaltab); return; - case RT_NET: - if (prop_type != PT_DEFROUTER) { - zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, - B_TRUE); - long_usage(CMD_REMOVE, B_TRUE); - usage(B_FALSE, HELP_PROPS); - return; - } else { - bzero(&in_progress_nwiftab.zone_nwif_defrouter, - sizeof (in_progress_nwiftab.zone_nwif_defrouter)); - return; - } default: zone_perror(rt_to_str(res_type), Z_NO_RESOURCE_TYPE, B_TRUE); long_usage(CMD_REMOVE, B_TRUE); @@ -4221,10 +4345,12 @@ set_func(cmd_t *cmd) * A nasty expression but not that complicated: * 1. fs options are simple or list (tested below) * 2. rctl value's are complex or list (tested below) + * 3. net attr's are complex (tested below) * Anything else should be simple. */ if (!(res_type == RT_FS && prop_type == PT_OPTIONS) && !(res_type == RT_RCTL && prop_type == PT_VALUE) && + !(res_type == RT_NET && prop_type == PT_NPROP) && (pp->pv_type != PROP_VAL_SIMPLE || (prop_id = pp->pv_simple) == NULL)) { zerr(gettext("A %s value was expected here."), @@ -4512,6 +4638,20 @@ set_func(cmd_t *cmd) prop_id, sizeof (in_progress_nwiftab.zone_nwif_defrouter)); break; + case PT_NPROP: + if (pp->pv_type != PROP_VAL_COMPLEX) { + zerr(gettext("A %s value was expected here."), + pvt_to_str(PROP_VAL_COMPLEX)); + saw_error = B_TRUE; + return; + } + zonecfg_free_nwif_attr_list( + in_progress_nwiftab.zone_nwif_attrp); + in_progress_nwiftab.zone_nwif_attrp = NULL; + if (!(pp->pv_type == PROP_VAL_LIST && + pp->pv_list == NULL)) + add_property(cmd); + break; default: zone_perror(pt_to_str(prop_type), Z_NO_PROPERTY_TYPE, B_TRUE); @@ -5051,6 +5191,8 @@ loopend: static void output_net(FILE *fp, struct zone_nwiftab *nwiftab) { + struct zone_nwif_attrtab *np; + (void) fprintf(fp, "%s:\n", rt_to_str(RT_NET)); output_prop(fp, PT_ADDRESS, nwiftab->zone_nwif_address, B_TRUE); output_prop(fp, PT_ALLOWED_ADDRESS, @@ -5060,6 +5202,14 @@ output_net(FILE *fp, struct zone_nwiftab *nwiftab) output_prop(fp, PT_MAC, nwiftab->zone_nwif_mac, B_TRUE); output_prop(fp, PT_PHYSICAL, nwiftab->zone_nwif_physical, B_TRUE); output_prop(fp, PT_VLANID, nwiftab->zone_nwif_vlan_id, B_TRUE); + + for (np = nwiftab->zone_nwif_attrp; np != NULL; + np = np->zone_nwif_attr_next) { + fprintf(fp, "\t%s: (%s=%s,%s=\"%s\")\n", + pt_to_str(PT_NPROP), + pt_to_str(PT_NAME), np->zone_nwif_attr_name, + pt_to_str(PT_VALUE), np->zone_nwif_attr_value); + } } static void diff --git a/usr/src/cmd/zonecfg/zonecfg.h b/usr/src/cmd/zonecfg/zonecfg.h index d516949bac..960a35d8a1 100644 --- a/usr/src/cmd/zonecfg/zonecfg.h +++ b/usr/src/cmd/zonecfg/zonecfg.h @@ -143,9 +143,10 @@ extern "C" { #define PT_MAC 43 #define PT_VLANID 44 #define PT_GNIC 45 +#define PT_NPROP 46 #define PT_MIN PT_UNKNOWN -#define PT_MAX PT_GNIC +#define PT_MAX PT_NPROP #define MAX_EQ_PROP_PAIRS 3 diff --git a/usr/src/cmd/zonecfg/zonecfg_grammar.y b/usr/src/cmd/zonecfg/zonecfg_grammar.y index da393be79e..521953a909 100644 --- a/usr/src/cmd/zonecfg/zonecfg_grammar.y +++ b/usr/src/cmd/zonecfg/zonecfg_grammar.y @@ -137,7 +137,7 @@ complex_piece_func(int cp_type, const char *str, complex_property_ptr_t cp_next) %token OPEN_PAREN CLOSE_PAREN COMMA DATASET LIMITPRIV BOOTARGS BRAND PSET PCAP %token MCAP NCPUS IMPORTANCE SHARES MAXLWPS MAXSHMMEM MAXSHMIDS MAXMSGIDS %token MAXSEMIDS LOCKED SWAP SCHED CLEAR DEFROUTER ADMIN USER AUTHS MAXPROCS -%token ZFSPRI MAC VLANID GNIC +%token ZFSPRI MAC VLANID GNIC NPROP %type <strval> TOKEN EQUAL OPEN_SQ_BRACKET CLOSE_SQ_BRACKET property_value OPEN_PAREN CLOSE_PAREN COMMA simple_prop_val @@ -147,7 +147,7 @@ complex_piece_func(int cp_type, const char *str, complex_property_ptr_t cp_next) %type <ival> property_name SPECIAL RAW DIR OPTIONS TYPE ADDRESS PHYSICAL NAME MATCH ZONENAME ZONEPATH AUTOBOOT POOL LIMITPRIV BOOTARGS VALUE PRIV LIMIT ACTION BRAND SCHED IPTYPE DEFROUTER HOSTID USER AUTHS FS_ALLOWED - ALLOWED_ADDRESS MAC VLANID GNIC + ALLOWED_ADDRESS MAC VLANID GNIC NPROP %type <cmd> command %type <cmd> add_command ADD %type <cmd> cancel_command CANCEL @@ -990,6 +990,7 @@ property_name: SPECIAL { $$ = PT_SPECIAL; } | MAC { $$ = PT_MAC; } | VLANID { $$ = PT_VLANID; } | GNIC { $$ = PT_GNIC; } + | NPROP { $$ = PT_NPROP; } | NAME { $$ = PT_NAME; } | VALUE { $$ = PT_VALUE; } | MATCH { $$ = PT_MATCH; } diff --git a/usr/src/cmd/zonecfg/zonecfg_lex.l b/usr/src/cmd/zonecfg/zonecfg_lex.l index 603be43807..21a48f3ec3 100644 --- a/usr/src/cmd/zonecfg/zonecfg_lex.l +++ b/usr/src/cmd/zonecfg/zonecfg_lex.l @@ -247,6 +247,9 @@ static char *create_token(char *s); <TSTATE>global-nic { return GNIC; } <CSTATE>global-nic { return GNIC; } +<TSTATE>property { return NPROP; } +<CSTATE>property { return NPROP; } + <TSTATE>dir { return DIR; } <CSTATE>dir { return DIR; } @@ -371,6 +374,13 @@ static char *create_token(char *s); return TOKEN; } +<CSTATE>\"[^\"\n]*[\"\n] { + yylval.strval = create_token(yytext + 1); + if (yylval.strval[yyleng - 2] == '"') + yylval.strval[yyleng - 2] = 0; + return TOKEN; + } + <TSTATE>\"[^\"\n]*[\"\n] { yylval.strval = create_token(yytext + 1); if (yylval.strval[yyleng - 2] == '"') |
