diff options
author | stevel@tonic-gate <none@none> | 2005-06-14 00:00:00 -0700 |
---|---|---|
committer | stevel@tonic-gate <none@none> | 2005-06-14 00:00:00 -0700 |
commit | 7c478bd95313f5f23a4c958a745db2134aa03244 (patch) | |
tree | c871e58545497667cbb4b0a4f2daf204743e1fe7 /usr/src/lib/libinetutil/common/ifspec.c | |
download | illumos-joyent-7c478bd95313f5f23a4c958a745db2134aa03244.tar.gz |
OpenSolaris Launch
Diffstat (limited to 'usr/src/lib/libinetutil/common/ifspec.c')
-rw-r--r-- | usr/src/lib/libinetutil/common/ifspec.c | 206 |
1 files changed, 206 insertions, 0 deletions
diff --git a/usr/src/lib/libinetutil/common/ifspec.c b/usr/src/lib/libinetutil/common/ifspec.c new file mode 100644 index 0000000000..58f7c5ae71 --- /dev/null +++ b/usr/src/lib/libinetutil/common/ifspec.c @@ -0,0 +1,206 @@ +/* + * CDDL HEADER START + * + * The contents of this file are subject to the terms of the + * Common Development and Distribution License, Version 1.0 only + * (the "License"). You may not use this file except in compliance + * with the License. + * + * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE + * or http://www.opensolaris.org/os/licensing. + * See the License for the specific language governing permissions + * and limitations under the License. + * + * When distributing Covered Code, include this CDDL HEADER in each + * file and include the License file at usr/src/OPENSOLARIS.LICENSE. + * If applicable, add the following below this CDDL HEADER, with the + * fields enclosed by brackets "[]" replaced with your own identifying + * information: Portions Copyright [yyyy] [name of copyright owner] + * + * CDDL HEADER END + */ +/* + * Copyright (c) 2001 by Sun Microsystems, Inc. + * All rights reserved. + */ + +#pragma ident "%Z%%M% %I% %E% SMI" + +/* + * This file contains a routine used to validate a ifconfig-style interface + * specification + */ + +#include <stdlib.h> +#include <ctype.h> +#include <alloca.h> +#include <errno.h> +#include <string.h> +#include <libinetutil.h> + +/* + * Given a token with a logical unit spec, return the logical unit converted + * to a uint_t. + * + * Returns: 0 for success, nonzero if an error occurred. errno is set if + * necessary. + */ +static int +getlun(const char *bp, int bpsize, uint_t *lun) +{ + char *ep = (char *)&bp[bpsize - 1]; + char *sp = strchr(bp, ':'), *tp; + + /* A logical unit spec looks like: <token>:<unsigned int>\0 */ + if (isdigit(*bp) || !isdigit(*ep) || sp == NULL || + strchr(sp + 1, ':') != NULL) { + errno = EINVAL; + return (-1); + } + + *sp++ = '\0'; + + /* Lun must be all digits */ + for (tp = sp; tp < ep && isdigit(*tp); tp++) + /* Null body */; + if (tp != ep) { + errno = EINVAL; + return (-1); + } + + *lun = atoi(sp); + return (0); +} + +/* + * Given a single token ending with a ppa spec, return the ppa spec converted + * to a uint_t. + * + * Returns: 0 for success, nonzero if an error occurred. errno is set if + * necessary. + */ +static int +getppa(const char *bp, int bpsize, uint_t *ppa) +{ + char *ep = (char *)&bp[bpsize - 1]; + char *tp; + + if (isdigit(*bp) || !isdigit(*ep)) { + errno = EINVAL; + return (-1); + } + + for (tp = ep; tp >= bp && isdigit(*tp); tp--) + /* Null body */; + + if (*tp == '.' || *tp == ':') { + errno = EINVAL; + return (-1); + } + + *ppa = atoi(tp + 1); + return (0); +} + +/* + * Given an ifconfig-style inet relative-path interface specification + * (e.g: hme0.foo.ip.udp:2), validate its form and decompose the contents + * into a dynamically allocated ifspec_t. + * + * Returns ifspec_t for success, NULL pointer if spec is malformed. + */ +boolean_t +ifparse_ifspec(const char *ifname, ifspec_t *ifsp) +{ + char *mp, *ep, *lp, *tp; + char *ifnamecp; + size_t iflen; + boolean_t have_ppa = B_FALSE; + + iflen = strlen(ifname); + if (iflen > LIFNAMSIZ) { + errno = EINVAL; + return (B_FALSE); + } + + /* snag a copy we can modify */ + ifnamecp = alloca(iflen + 1); + (void) strlcpy(ifnamecp, ifname, iflen + 1); + + ifsp->ifsp_lunvalid = B_FALSE; + + /* + * An interface name must have the format of: + * dev[ppa][.module[.module...][ppa]][:lun] + * + * where only one ppa may be specified e.g. ip0.foo.tun or ip.foo.tun0 + * + * lun - logical unit number. + * + * Produce substrings for each grouping, starting first with modules, + * then lun, devname, and finally ppa. + */ + + /* Any modules? */ + mp = strchr(ifnamecp, '.'); + + /* Any logical units? */ + lp = strchr(ifnamecp, ':'); + + if (lp != NULL && mp != NULL && lp < mp) { + errno = EINVAL; + return (B_FALSE); + } + + ifsp->ifsp_modcnt = 0; + if (mp != NULL) { + *mp++ = '\0'; + if (lp != NULL) + *lp = '\0'; + while (mp != NULL && ifsp->ifsp_modcnt <= IFSP_MAXMODS) { + if ((ep = strchr(mp, '.')) != NULL) + *ep++ = '\0'; + (void) strlcpy(ifsp->ifsp_mods[ifsp->ifsp_modcnt++], + mp, LIFNAMSIZ); + mp = ep; + } + if (lp != NULL) + *lp = ':'; + if (ifsp->ifsp_modcnt > IFSP_MAXMODS) { + errno = E2BIG; + return (B_FALSE); + } + } + + if (lp != NULL) { + if (getlun(lp, strlen(lp), &ifsp->ifsp_lun) != 0) + return (B_FALSE); + ifsp->ifsp_lunvalid = B_TRUE; + } + + (void) strlcpy(ifsp->ifsp_devnm, ifnamecp, LIFNAMSIZ); + + /* Find ppa - has to be part of devname or part of last module name */ + /* This should be changed to require the latter of the two */ + if (getppa(ifsp->ifsp_devnm, strlen(ifsp->ifsp_devnm), + &ifsp->ifsp_ppa) == 0) + have_ppa = B_TRUE; + if (ifsp->ifsp_modcnt != 0 && + getppa(ifsp->ifsp_mods[ifsp->ifsp_modcnt - 1], + strlen(ifsp->ifsp_mods[ifsp->ifsp_modcnt - 1]), + &ifsp->ifsp_ppa) == 0) { + if (!have_ppa) + have_ppa = B_TRUE; + else + return (B_FALSE); /* only one please */ + } + if (!have_ppa) + return (B_FALSE); + + /* strip the ppa off of the device name if present */ + for (tp = &ifsp->ifsp_devnm[strlen(ifsp->ifsp_devnm) - 1]; + tp >= ifsp->ifsp_devnm && isdigit(*tp); tp--) + *tp = '\0'; + + return (B_TRUE); +} |