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
57c478bd9Sstevel@tonic-gate * Common Development and Distribution License, Version 1.0 only
67c478bd9Sstevel@tonic-gate * (the "License"). You may not use this file except in compliance
77c478bd9Sstevel@tonic-gate * with the License.
87c478bd9Sstevel@tonic-gate *
97c478bd9Sstevel@tonic-gate * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
107c478bd9Sstevel@tonic-gate * or http://www.opensolaris.org/os/licensing.
117c478bd9Sstevel@tonic-gate * See the License for the specific language governing permissions
127c478bd9Sstevel@tonic-gate * and limitations under the License.
137c478bd9Sstevel@tonic-gate *
147c478bd9Sstevel@tonic-gate * When distributing Covered Code, include this CDDL HEADER in each
157c478bd9Sstevel@tonic-gate * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
167c478bd9Sstevel@tonic-gate * If applicable, add the following below this CDDL HEADER, with the
177c478bd9Sstevel@tonic-gate * fields enclosed by brackets "[]" replaced with your own identifying
187c478bd9Sstevel@tonic-gate * information: Portions Copyright [yyyy] [name of copyright owner]
197c478bd9Sstevel@tonic-gate *
207c478bd9Sstevel@tonic-gate * CDDL HEADER END
217c478bd9Sstevel@tonic-gate */
227c478bd9Sstevel@tonic-gate /*
23*113f4232Sakaplan * Copyright 2005 Sun Microsystems, Inc. All rights reserved.
247c478bd9Sstevel@tonic-gate * Use is subject to license terms.
257c478bd9Sstevel@tonic-gate */
267c478bd9Sstevel@tonic-gate
277c478bd9Sstevel@tonic-gate /* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
287c478bd9Sstevel@tonic-gate /* All Rights Reserved */
297c478bd9Sstevel@tonic-gate
307c478bd9Sstevel@tonic-gate
317c478bd9Sstevel@tonic-gate #pragma ident "%Z%%M% %I% %E% SMI"
327c478bd9Sstevel@tonic-gate
337c478bd9Sstevel@tonic-gate #include <sys/types.h>
347c478bd9Sstevel@tonic-gate #include <sys/param.h>
357c478bd9Sstevel@tonic-gate #include <stdio.h>
367c478bd9Sstevel@tonic-gate #include <errno.h>
377c478bd9Sstevel@tonic-gate #include <stdlib.h>
387c478bd9Sstevel@tonic-gate #include <string.h>
397c478bd9Sstevel@tonic-gate #include <fmtmsg.h>
407c478bd9Sstevel@tonic-gate #include <devmgmt.h>
417c478bd9Sstevel@tonic-gate #include <devtab.h>
427c478bd9Sstevel@tonic-gate #include <values.h>
437c478bd9Sstevel@tonic-gate
447c478bd9Sstevel@tonic-gate
457c478bd9Sstevel@tonic-gate /*
467c478bd9Sstevel@tonic-gate * Local definitions
477c478bd9Sstevel@tonic-gate * TRUE Boolean TRUE value
487c478bd9Sstevel@tonic-gate * FALSE Boolean FALSE value
497c478bd9Sstevel@tonic-gate * TOKDELIMS Char string of delimiters for lists
507c478bd9Sstevel@tonic-gate */
517c478bd9Sstevel@tonic-gate
527c478bd9Sstevel@tonic-gate #ifndef TRUE
537c478bd9Sstevel@tonic-gate #define TRUE ('t')
547c478bd9Sstevel@tonic-gate #endif
557c478bd9Sstevel@tonic-gate
567c478bd9Sstevel@tonic-gate #ifndef FALSE
577c478bd9Sstevel@tonic-gate #define FALSE 0
587c478bd9Sstevel@tonic-gate #endif
597c478bd9Sstevel@tonic-gate
607c478bd9Sstevel@tonic-gate #define TOKDELIMS ", \t\n"
617c478bd9Sstevel@tonic-gate
627c478bd9Sstevel@tonic-gate
637c478bd9Sstevel@tonic-gate /*
647c478bd9Sstevel@tonic-gate * Exit codes:
657c478bd9Sstevel@tonic-gate * EX_OK Exit code for all went well
667c478bd9Sstevel@tonic-gate * EX_ERROR Exit code for something failed
677c478bd9Sstevel@tonic-gate * EX_TABLES A table couldn't be accessed
687c478bd9Sstevel@tonic-gate * EX_NOALLOC Exit code for allocation failed
697c478bd9Sstevel@tonic-gate */
707c478bd9Sstevel@tonic-gate
717c478bd9Sstevel@tonic-gate #define EX_OK 0
727c478bd9Sstevel@tonic-gate #define EX_ERROR 1
737c478bd9Sstevel@tonic-gate #define EX_TABLES 2
747c478bd9Sstevel@tonic-gate #define EX_NOALLOC 3
757c478bd9Sstevel@tonic-gate
767c478bd9Sstevel@tonic-gate /*
777c478bd9Sstevel@tonic-gate * Messages:
787c478bd9Sstevel@tonic-gate * M_USAGE Usage error
797c478bd9Sstevel@tonic-gate * M_INVKEY Invalid key specified
807c478bd9Sstevel@tonic-gate * M_ERROR Some strange error
817c478bd9Sstevel@tonic-gate * M_UNABLE A list of devices is unavailable
827c478bd9Sstevel@tonic-gate * M_DEVTAB Can't access device table (for reading)
837c478bd9Sstevel@tonic-gate * M_RSVTAB Can't access device reservation table (for r/w)
847c478bd9Sstevel@tonic-gate * M_NODEV A list of devices is invalid
857c478bd9Sstevel@tonic-gate */
867c478bd9Sstevel@tonic-gate
877c478bd9Sstevel@tonic-gate #define M_USAGE "usage: devreserv [key [devicelist [...]]]"
887c478bd9Sstevel@tonic-gate #define M_INVKEY "Invalid key: %s"
897c478bd9Sstevel@tonic-gate #define M_ERROR "Internal error, errno=%d"
907c478bd9Sstevel@tonic-gate #define M_UNABLE "Cannot reserve devices"
917c478bd9Sstevel@tonic-gate #define M_DEVTAB "Cannot open the device table: %s"
927c478bd9Sstevel@tonic-gate #define M_RSVTAB "Cannot open the device-reservation table: %s"
937c478bd9Sstevel@tonic-gate #define M_NODEV M_UNABLE
947c478bd9Sstevel@tonic-gate
957c478bd9Sstevel@tonic-gate
967c478bd9Sstevel@tonic-gate /*
977c478bd9Sstevel@tonic-gate * Local functions and static data
987c478bd9Sstevel@tonic-gate *
997c478bd9Sstevel@tonic-gate * buildreqlist() Builds the list of requested devices for devreserv()
1007c478bd9Sstevel@tonic-gate * freereqlist() Free space allocated to the list of requested devices
1017c478bd9Sstevel@tonic-gate * ndevsin() Get number of elements in a list
1027c478bd9Sstevel@tonic-gate * stdmsg(r,l,s,m) Standard message generation
1037c478bd9Sstevel@tonic-gate * r Recoverability flag
1047c478bd9Sstevel@tonic-gate * l Label
1057c478bd9Sstevel@tonic-gate * s Severity
1067c478bd9Sstevel@tonic-gate * m Message
1077c478bd9Sstevel@tonic-gate *
1087c478bd9Sstevel@tonic-gate * lbl Buffer for the label-component of a message
1097c478bd9Sstevel@tonic-gate * txt Buffer for the text-component of a message
1107c478bd9Sstevel@tonic-gate */
1117c478bd9Sstevel@tonic-gate
1127c478bd9Sstevel@tonic-gate static char ***buildreqlist();
1137c478bd9Sstevel@tonic-gate static void freereqlist();
1147c478bd9Sstevel@tonic-gate static int ndevsin();
1157c478bd9Sstevel@tonic-gate
1167c478bd9Sstevel@tonic-gate #define stdmsg(r,l,s,m) (void) fmtmsg(MM_PRINT|MM_UTIL|r,l,s,m,MM_NULLACT,MM_NULLTAG)
1177c478bd9Sstevel@tonic-gate
1187c478bd9Sstevel@tonic-gate static char lbl[MM_MXLABELLN+1];
1197c478bd9Sstevel@tonic-gate static char txt[MM_MXTXTLN+1];
1207c478bd9Sstevel@tonic-gate
1217c478bd9Sstevel@tonic-gate /*
1227c478bd9Sstevel@tonic-gate * devreserv [key [devlist [devlist [...]]]]
1237c478bd9Sstevel@tonic-gate *
1247c478bd9Sstevel@tonic-gate * This command reserves sets of devices known to the OA&M device
1257c478bd9Sstevel@tonic-gate * management system. It reserves a device from each of the device
1267c478bd9Sstevel@tonic-gate * lists presented to it, reserving them on the key (<key>). If no
1277c478bd9Sstevel@tonic-gate * device-lists are provided, the command lists those devices reserved
1287c478bd9Sstevel@tonic-gate * on the given key (<key>). If no key (<key>) is provided, the
1297c478bd9Sstevel@tonic-gate * command lists all devices currently reserved.
1307c478bd9Sstevel@tonic-gate *
1317c478bd9Sstevel@tonic-gate * Options: None
1327c478bd9Sstevel@tonic-gate *
1337c478bd9Sstevel@tonic-gate * Arguments:
1347c478bd9Sstevel@tonic-gate * key Key to lock the devices on
1357c478bd9Sstevel@tonic-gate * devlist A comma-, space-, or tab-list containing devices
1367c478bd9Sstevel@tonic-gate * (pathnames or aliases). For typical shells, space-
1377c478bd9Sstevel@tonic-gate * and tab-lists should be quoted or the separator should
1387c478bd9Sstevel@tonic-gate * be somehow escaped.
1397c478bd9Sstevel@tonic-gate *
1407c478bd9Sstevel@tonic-gate * Command Values:
1417c478bd9Sstevel@tonic-gate * EX_OK 0 Device(s) successfully allocated
1427c478bd9Sstevel@tonic-gate * EX_ERROR 1 A syntax or other error occurred
1437c478bd9Sstevel@tonic-gate * EX_TABLES 2 Either the device-table or the device-
1447c478bd9Sstevel@tonic-gate * reservation table couldn't be opened as needed
1457c478bd9Sstevel@tonic-gate * EX_NOALLOC 3 The device-reservation request couldn't be
1467c478bd9Sstevel@tonic-gate * fulfilled.
1477c478bd9Sstevel@tonic-gate */
1487c478bd9Sstevel@tonic-gate
149*113f4232Sakaplan int
main(int argc,char * argv[])150*113f4232Sakaplan main(int argc, char *argv[])
1517c478bd9Sstevel@tonic-gate {
1527c478bd9Sstevel@tonic-gate
1537c478bd9Sstevel@tonic-gate /* Automatics */
1547c478bd9Sstevel@tonic-gate char ***reqlist; /* * to list of lists */
1557c478bd9Sstevel@tonic-gate char **argp; /* Ptr to current argument */
1567c478bd9Sstevel@tonic-gate char **alloclist; /* List of allocated devices */
1577c478bd9Sstevel@tonic-gate char **pp; /* Temp ptr to char ptrs */
1587c478bd9Sstevel@tonic-gate struct reservdev **rsvd; /* Ptr to list of rsvd devs */
1597c478bd9Sstevel@tonic-gate struct reservdev **plk; /* Running ptr to locks */
1607c478bd9Sstevel@tonic-gate char *p; /* Temp char ptr */
1617c478bd9Sstevel@tonic-gate char *devtab; /* Device table pathname */
1627c478bd9Sstevel@tonic-gate char *rsvtab; /* Dev-rsv tbl pathname */
1637c478bd9Sstevel@tonic-gate int argcount; /* Number of args on cmd */
1647c478bd9Sstevel@tonic-gate long lkey; /* Key for locking (long) */
1657c478bd9Sstevel@tonic-gate int key; /* Key for locking */
1667c478bd9Sstevel@tonic-gate int exitcode; /* Value to return */
1677c478bd9Sstevel@tonic-gate int sev; /* Message severity */
1687c478bd9Sstevel@tonic-gate int syntaxerr; /* Flag, TRUE if syntax error */
1697c478bd9Sstevel@tonic-gate int c; /* Option character */
1707c478bd9Sstevel@tonic-gate int i; /* Temp counter */
1717c478bd9Sstevel@tonic-gate
1727c478bd9Sstevel@tonic-gate
1737c478bd9Sstevel@tonic-gate /*
1747c478bd9Sstevel@tonic-gate * Initializations
1757c478bd9Sstevel@tonic-gate */
1767c478bd9Sstevel@tonic-gate
1777c478bd9Sstevel@tonic-gate /* Build a message label */
1787c478bd9Sstevel@tonic-gate if (p = strrchr(argv[0], '/')) p++;
1797c478bd9Sstevel@tonic-gate else p = argv[0];
1807c478bd9Sstevel@tonic-gate (void) strlcat(strcpy(lbl, "UX:"), p, sizeof(lbl));
1817c478bd9Sstevel@tonic-gate
1827c478bd9Sstevel@tonic-gate
1837c478bd9Sstevel@tonic-gate /*
1847c478bd9Sstevel@tonic-gate * Allow only the text component of messages to be written
1857c478bd9Sstevel@tonic-gate * (this will probably go away in SVR4.1)
1867c478bd9Sstevel@tonic-gate */
1877c478bd9Sstevel@tonic-gate
1887c478bd9Sstevel@tonic-gate (void) putenv("MSGVERB=text");
1897c478bd9Sstevel@tonic-gate
1907c478bd9Sstevel@tonic-gate
1917c478bd9Sstevel@tonic-gate /*
1927c478bd9Sstevel@tonic-gate * Parse the options from the command line
1937c478bd9Sstevel@tonic-gate */
1947c478bd9Sstevel@tonic-gate
1957c478bd9Sstevel@tonic-gate opterr = 0;
1967c478bd9Sstevel@tonic-gate syntaxerr = FALSE;
1977c478bd9Sstevel@tonic-gate while ((c = getopt(argc, argv, "")) != EOF) switch(c) {
1987c478bd9Sstevel@tonic-gate default:
1997c478bd9Sstevel@tonic-gate syntaxerr = FALSE;
2007c478bd9Sstevel@tonic-gate break;
2017c478bd9Sstevel@tonic-gate }
2027c478bd9Sstevel@tonic-gate
2037c478bd9Sstevel@tonic-gate /* If there's (an obvious) syntax error, write a message and quit */
2047c478bd9Sstevel@tonic-gate if (syntaxerr) {
2057c478bd9Sstevel@tonic-gate stdmsg(MM_NRECOV, lbl, MM_ERROR, M_USAGE);
2067c478bd9Sstevel@tonic-gate exit(EX_ERROR);
2077c478bd9Sstevel@tonic-gate }
2087c478bd9Sstevel@tonic-gate
2097c478bd9Sstevel@tonic-gate /* Argument initializations */
2107c478bd9Sstevel@tonic-gate argcount = argc - optind;
2117c478bd9Sstevel@tonic-gate argp = &argv[optind];
2127c478bd9Sstevel@tonic-gate
2137c478bd9Sstevel@tonic-gate
2147c478bd9Sstevel@tonic-gate /*
2157c478bd9Sstevel@tonic-gate * devreserv
2167c478bd9Sstevel@tonic-gate *
2177c478bd9Sstevel@tonic-gate * If euid == 0, write a list of all currently allocated devices.
2187c478bd9Sstevel@tonic-gate */
2197c478bd9Sstevel@tonic-gate
2207c478bd9Sstevel@tonic-gate if (argcount == 0) {
2217c478bd9Sstevel@tonic-gate
2227c478bd9Sstevel@tonic-gate /* Get the list of reserved devices */
2237c478bd9Sstevel@tonic-gate if (rsvd = reservdev()) {
2247c478bd9Sstevel@tonic-gate
2257c478bd9Sstevel@tonic-gate /* Write the list of reserved devices with the key
2267c478bd9Sstevel@tonic-gate * that the device was locked on. The key should go
2277c478bd9Sstevel@tonic-gate * in column 16, but separate it from the alias with at
2287c478bd9Sstevel@tonic-gate * least one space */
2297c478bd9Sstevel@tonic-gate
2307c478bd9Sstevel@tonic-gate exitcode = EX_OK;
2317c478bd9Sstevel@tonic-gate for (plk = rsvd ; *plk ; plk++) {
2327c478bd9Sstevel@tonic-gate if ((i = fputs((*plk)->devname, stdout)) >= 0) do
2337c478bd9Sstevel@tonic-gate (void) fputc(' ', stdout);
2347c478bd9Sstevel@tonic-gate while (++i < 16);
2357c478bd9Sstevel@tonic-gate (void) fprintf(stdout, "%ld\n", (*plk)->key);
2367c478bd9Sstevel@tonic-gate }
2377c478bd9Sstevel@tonic-gate
2387c478bd9Sstevel@tonic-gate } else {
2397c478bd9Sstevel@tonic-gate
2407c478bd9Sstevel@tonic-gate /* Problems getting the list of reserved devices */
2417c478bd9Sstevel@tonic-gate if (((errno == EINVAL) || (errno == EACCES)) && (rsvtab = _rsvtabpath())) {
2427c478bd9Sstevel@tonic-gate (void) snprintf(txt, sizeof(txt), M_RSVTAB, rsvtab);
2437c478bd9Sstevel@tonic-gate exitcode = EX_TABLES;
2447c478bd9Sstevel@tonic-gate sev = MM_ERROR;
2457c478bd9Sstevel@tonic-gate } else {
2467c478bd9Sstevel@tonic-gate (void) sprintf(txt, M_ERROR, errno);
2477c478bd9Sstevel@tonic-gate exitcode = EX_ERROR;
2487c478bd9Sstevel@tonic-gate sev = MM_HALT;
2497c478bd9Sstevel@tonic-gate }
2507c478bd9Sstevel@tonic-gate stdmsg(MM_NRECOV, lbl, sev, txt);
2517c478bd9Sstevel@tonic-gate }
2527c478bd9Sstevel@tonic-gate
2537c478bd9Sstevel@tonic-gate /* Finished */
2547c478bd9Sstevel@tonic-gate exit(exitcode);
2557c478bd9Sstevel@tonic-gate }
2567c478bd9Sstevel@tonic-gate
2577c478bd9Sstevel@tonic-gate
2587c478bd9Sstevel@tonic-gate /*
2597c478bd9Sstevel@tonic-gate * devreserv key
2607c478bd9Sstevel@tonic-gate *
2617c478bd9Sstevel@tonic-gate * Generate a list of the devices allocated on a specific key.
2627c478bd9Sstevel@tonic-gate */
2637c478bd9Sstevel@tonic-gate
2647c478bd9Sstevel@tonic-gate if (argcount == 1) {
2657c478bd9Sstevel@tonic-gate
2667c478bd9Sstevel@tonic-gate /* Extract the key from the command */
2677c478bd9Sstevel@tonic-gate lkey = strtol(*argp, &p, 10);
2687c478bd9Sstevel@tonic-gate if (*p || (lkey <= 0) || (lkey > MAXINT)) {
2697c478bd9Sstevel@tonic-gate
2707c478bd9Sstevel@tonic-gate /* <key> argument invalid */
2717c478bd9Sstevel@tonic-gate (void) snprintf(txt, sizeof(txt), M_INVKEY, *argp);
2727c478bd9Sstevel@tonic-gate stdmsg(MM_NRECOV, lbl, MM_ERROR, txt);
2737c478bd9Sstevel@tonic-gate exitcode = EX_ERROR;
2747c478bd9Sstevel@tonic-gate
2757c478bd9Sstevel@tonic-gate } else {
2767c478bd9Sstevel@tonic-gate
2777c478bd9Sstevel@tonic-gate key = (int) lkey;
2787c478bd9Sstevel@tonic-gate
2797c478bd9Sstevel@tonic-gate /* Get the list of reserved devices ... */
2807c478bd9Sstevel@tonic-gate if (rsvd = reservdev()) {
2817c478bd9Sstevel@tonic-gate
2827c478bd9Sstevel@tonic-gate /* For each reserved device, write the alias to stdout */
2837c478bd9Sstevel@tonic-gate exitcode = EX_OK;
2847c478bd9Sstevel@tonic-gate for (plk = rsvd ; *plk ; plk++) {
2857c478bd9Sstevel@tonic-gate if ((*plk)->key == key) (void) puts((*plk)->devname);
2867c478bd9Sstevel@tonic-gate }
2877c478bd9Sstevel@tonic-gate
2887c478bd9Sstevel@tonic-gate } else {
2897c478bd9Sstevel@tonic-gate
2907c478bd9Sstevel@tonic-gate /* Problems getting the list of reserved devices */
2917c478bd9Sstevel@tonic-gate if (((errno == EINVAL) || (errno == EACCES)) && (rsvtab = _rsvtabpath())) {
2927c478bd9Sstevel@tonic-gate (void) snprintf(txt, sizeof(txt), M_RSVTAB, rsvtab);
2937c478bd9Sstevel@tonic-gate exitcode = EX_TABLES;
2947c478bd9Sstevel@tonic-gate sev = MM_ERROR;
2957c478bd9Sstevel@tonic-gate } else {
2967c478bd9Sstevel@tonic-gate (void) sprintf(txt, M_ERROR, errno);
2977c478bd9Sstevel@tonic-gate exitcode = EX_ERROR;
2987c478bd9Sstevel@tonic-gate sev = MM_HALT;
2997c478bd9Sstevel@tonic-gate }
3007c478bd9Sstevel@tonic-gate stdmsg(MM_NRECOV, lbl, sev, txt);
3017c478bd9Sstevel@tonic-gate }
3027c478bd9Sstevel@tonic-gate }
3037c478bd9Sstevel@tonic-gate
3047c478bd9Sstevel@tonic-gate /* Finished */
3057c478bd9Sstevel@tonic-gate exit(exitcode);
3067c478bd9Sstevel@tonic-gate }
3077c478bd9Sstevel@tonic-gate
3087c478bd9Sstevel@tonic-gate
3097c478bd9Sstevel@tonic-gate /*
3107c478bd9Sstevel@tonic-gate * devreserv key devlist [...]
3117c478bd9Sstevel@tonic-gate *
3127c478bd9Sstevel@tonic-gate * Reserve specific devices
3137c478bd9Sstevel@tonic-gate */
3147c478bd9Sstevel@tonic-gate
3157c478bd9Sstevel@tonic-gate /* Open the device file (if there's one to be opened) */
3167c478bd9Sstevel@tonic-gate if (!_opendevtab("r")) {
3177c478bd9Sstevel@tonic-gate if (devtab = _devtabpath()) {
3187c478bd9Sstevel@tonic-gate (void) snprintf(txt, sizeof(txt), M_DEVTAB, devtab);
3197c478bd9Sstevel@tonic-gate exitcode = EX_TABLES;
3207c478bd9Sstevel@tonic-gate sev = MM_ERROR;
3217c478bd9Sstevel@tonic-gate } else {
3227c478bd9Sstevel@tonic-gate (void) sprintf(txt, M_ERROR, errno);
3237c478bd9Sstevel@tonic-gate exitcode = EX_ERROR;
3247c478bd9Sstevel@tonic-gate sev = MM_HALT;
3257c478bd9Sstevel@tonic-gate }
3267c478bd9Sstevel@tonic-gate stdmsg(MM_NRECOV, lbl, sev, txt);
3277c478bd9Sstevel@tonic-gate exit(exitcode);
3287c478bd9Sstevel@tonic-gate }
3297c478bd9Sstevel@tonic-gate
3307c478bd9Sstevel@tonic-gate /* Extract the key from the command */
3317c478bd9Sstevel@tonic-gate lkey = strtol(*argp, &p, 10);
3327c478bd9Sstevel@tonic-gate if (*p || (lkey <= 0) || (lkey > MAXINT)) {
3337c478bd9Sstevel@tonic-gate (void) snprintf(txt, sizeof(txt), M_INVKEY, *argp);
3347c478bd9Sstevel@tonic-gate stdmsg(MM_NRECOV, lbl, MM_ERROR, txt);
3357c478bd9Sstevel@tonic-gate exit(EX_ERROR);
3367c478bd9Sstevel@tonic-gate }
3377c478bd9Sstevel@tonic-gate
3387c478bd9Sstevel@tonic-gate key = (int) lkey;
3397c478bd9Sstevel@tonic-gate argp++;
3407c478bd9Sstevel@tonic-gate
3417c478bd9Sstevel@tonic-gate /* Build the device request list from the command arguments */
3427c478bd9Sstevel@tonic-gate if (reqlist = buildreqlist(argp)) {
3437c478bd9Sstevel@tonic-gate
3447c478bd9Sstevel@tonic-gate /* Attempt to allocate the devices */
3457c478bd9Sstevel@tonic-gate if (alloclist = devreserv(key, reqlist)) {
3467c478bd9Sstevel@tonic-gate
3477c478bd9Sstevel@tonic-gate /*
3487c478bd9Sstevel@tonic-gate * For each allocated device, write the alias to stdout
3497c478bd9Sstevel@tonic-gate * and free the space allocated for the string.
3507c478bd9Sstevel@tonic-gate */
3517c478bd9Sstevel@tonic-gate
3527c478bd9Sstevel@tonic-gate for (pp = alloclist; *pp; pp++) {
3537c478bd9Sstevel@tonic-gate (void) puts(*pp);
3547c478bd9Sstevel@tonic-gate free(*pp);
3557c478bd9Sstevel@tonic-gate }
3567c478bd9Sstevel@tonic-gate
3577c478bd9Sstevel@tonic-gate /* Free the list of allocated devices */
3587c478bd9Sstevel@tonic-gate free((char *) alloclist);
3597c478bd9Sstevel@tonic-gate exitcode = EX_OK;
3607c478bd9Sstevel@tonic-gate }
3617c478bd9Sstevel@tonic-gate else {
3627c478bd9Sstevel@tonic-gate /* Device allocation failed */
3637c478bd9Sstevel@tonic-gate if (errno == EAGAIN) {
3647c478bd9Sstevel@tonic-gate stdmsg(MM_NRECOV, lbl, MM_ERROR, M_UNABLE);
3657c478bd9Sstevel@tonic-gate exitcode = EX_NOALLOC;
3667c478bd9Sstevel@tonic-gate } else if (errno == ENODEV) {
3677c478bd9Sstevel@tonic-gate stdmsg(MM_NRECOV, lbl, MM_ERROR, M_NODEV);
3687c478bd9Sstevel@tonic-gate exitcode = EX_NOALLOC;
3697c478bd9Sstevel@tonic-gate } else {
3707c478bd9Sstevel@tonic-gate (void) sprintf(txt, M_ERROR, errno);
3717c478bd9Sstevel@tonic-gate stdmsg(MM_NRECOV, lbl, MM_HALT, txt);
3727c478bd9Sstevel@tonic-gate exitcode = EX_ERROR;
3737c478bd9Sstevel@tonic-gate }
3747c478bd9Sstevel@tonic-gate }
3757c478bd9Sstevel@tonic-gate freereqlist(reqlist);
3767c478bd9Sstevel@tonic-gate }
3777c478bd9Sstevel@tonic-gate
3787c478bd9Sstevel@tonic-gate
3797c478bd9Sstevel@tonic-gate /* Exit with the appropriate code */
3807c478bd9Sstevel@tonic-gate return(exitcode);
3817c478bd9Sstevel@tonic-gate }
3827c478bd9Sstevel@tonic-gate
3837c478bd9Sstevel@tonic-gate /*
3847c478bd9Sstevel@tonic-gate * char ***buildreqlist(args)
3857c478bd9Sstevel@tonic-gate * char **args
3867c478bd9Sstevel@tonic-gate *
3877c478bd9Sstevel@tonic-gate * Build the list of lists of devices to request, as described by the
3887c478bd9Sstevel@tonic-gate * arguments on the command line.
3897c478bd9Sstevel@tonic-gate *
3907c478bd9Sstevel@tonic-gate * Arguments:
3917c478bd9Sstevel@tonic-gate * char **args The address of the first argument of the list of
3927c478bd9Sstevel@tonic-gate * lists of devices to allocate. (This list is
3937c478bd9Sstevel@tonic-gate * terminated with a (char *) NULL.)
3947c478bd9Sstevel@tonic-gate *
3957c478bd9Sstevel@tonic-gate * Returns: char ***
3967c478bd9Sstevel@tonic-gate * A pointer to a list containing addresses of lists of pointers to
3977c478bd9Sstevel@tonic-gate * character-strings, as expected by "devreserv()"
3987c478bd9Sstevel@tonic-gate *
3997c478bd9Sstevel@tonic-gate * Notes:
4007c478bd9Sstevel@tonic-gate * - Assuming that strtok() won't return "". If it does, the
4017c478bd9Sstevel@tonic-gate * parsing algorithm needs to be enhanced a bit to eliminate
4027c478bd9Sstevel@tonic-gate * these cases.
4037c478bd9Sstevel@tonic-gate */
4047c478bd9Sstevel@tonic-gate
4057c478bd9Sstevel@tonic-gate static char ***
buildreqlist(args)4067c478bd9Sstevel@tonic-gate buildreqlist(args)
4077c478bd9Sstevel@tonic-gate char **args;
4087c478bd9Sstevel@tonic-gate {
4097c478bd9Sstevel@tonic-gate /* Local automatic data */
4107c478bd9Sstevel@tonic-gate char ***addrlist; /* Addr of space for ptrs to lists */
4117c478bd9Sstevel@tonic-gate char ***ppp; /* Pointer to pointers to pointers */
4127c478bd9Sstevel@tonic-gate char **pp; /* Pointer to pointers */
4137c478bd9Sstevel@tonic-gate char **qq; /* Pointer to pointers */
4147c478bd9Sstevel@tonic-gate int noerror; /* FLAG, TRUE if all's well */
4157c478bd9Sstevel@tonic-gate int i; /* Counter */
4167c478bd9Sstevel@tonic-gate int n; /* Another counter */
4177c478bd9Sstevel@tonic-gate
4187c478bd9Sstevel@tonic-gate
4197c478bd9Sstevel@tonic-gate /* Count the number of lists we have to work with */
4207c478bd9Sstevel@tonic-gate i = 1;
4217c478bd9Sstevel@tonic-gate for (pp = args ; *pp ; pp++) i++;
4227c478bd9Sstevel@tonic-gate
4237c478bd9Sstevel@tonic-gate
4247c478bd9Sstevel@tonic-gate /* If we can allocate space for the list of lists ... */
4257c478bd9Sstevel@tonic-gate if (addrlist = (char ***) malloc(i*sizeof(char **))) {
4267c478bd9Sstevel@tonic-gate
4277c478bd9Sstevel@tonic-gate /* Parse each list, putting that list in the list of lists */
4287c478bd9Sstevel@tonic-gate ppp = addrlist;
4297c478bd9Sstevel@tonic-gate noerror = TRUE;
4307c478bd9Sstevel@tonic-gate for (pp = args ; noerror && *pp ; pp++) {
4317c478bd9Sstevel@tonic-gate n = ndevsin(*pp, TOKDELIMS);
4327c478bd9Sstevel@tonic-gate if (*ppp = (char **) malloc((n+1)*sizeof(char *))) {
4337c478bd9Sstevel@tonic-gate qq = *ppp++;
4347c478bd9Sstevel@tonic-gate if (*qq++ = strtok(*pp, TOKDELIMS))
4357c478bd9Sstevel@tonic-gate while (*qq++ = strtok((char *) NULL, TOKDELIMS));
4367c478bd9Sstevel@tonic-gate } else noerror = FALSE;
4377c478bd9Sstevel@tonic-gate }
4387c478bd9Sstevel@tonic-gate
4397c478bd9Sstevel@tonic-gate /* If there was an error, clean up the malloc()s we've made */
4407c478bd9Sstevel@tonic-gate if (!noerror) {
4417c478bd9Sstevel@tonic-gate freereqlist(addrlist);
4427c478bd9Sstevel@tonic-gate addrlist = (char ***) NULL;
4437c478bd9Sstevel@tonic-gate }
4447c478bd9Sstevel@tonic-gate }
4457c478bd9Sstevel@tonic-gate
4467c478bd9Sstevel@tonic-gate /* Return ptr to the list of addresses of lists (or NULL if none) */
4477c478bd9Sstevel@tonic-gate return(addrlist);
4487c478bd9Sstevel@tonic-gate }
4497c478bd9Sstevel@tonic-gate
4507c478bd9Sstevel@tonic-gate /*
4517c478bd9Sstevel@tonic-gate * void freereqlist(list)
4527c478bd9Sstevel@tonic-gate * char ***list
4537c478bd9Sstevel@tonic-gate *
4547c478bd9Sstevel@tonic-gate * This function frees the space allocated to the list of lists
4557c478bd9Sstevel@tonic-gate * referenced by <list>
4567c478bd9Sstevel@tonic-gate *
4577c478bd9Sstevel@tonic-gate * Arguments:
4587c478bd9Sstevel@tonic-gate * char ***list Address of the list of lists
4597c478bd9Sstevel@tonic-gate *
4607c478bd9Sstevel@tonic-gate * Returns: void
4617c478bd9Sstevel@tonic-gate */
4627c478bd9Sstevel@tonic-gate
4637c478bd9Sstevel@tonic-gate static void
freereqlist(list)4647c478bd9Sstevel@tonic-gate freereqlist(list)
4657c478bd9Sstevel@tonic-gate char ***list;
4667c478bd9Sstevel@tonic-gate {
4677c478bd9Sstevel@tonic-gate char ***ppp;
4687c478bd9Sstevel@tonic-gate if (list) {
4697c478bd9Sstevel@tonic-gate for (ppp = list ; *ppp ; ppp++) free((char *) *ppp);
4707c478bd9Sstevel@tonic-gate free((char *) list);
4717c478bd9Sstevel@tonic-gate }
4727c478bd9Sstevel@tonic-gate }
4737c478bd9Sstevel@tonic-gate
4747c478bd9Sstevel@tonic-gate /*
4757c478bd9Sstevel@tonic-gate * int ndevsin(list, delims)
4767c478bd9Sstevel@tonic-gate * char *list
4777c478bd9Sstevel@tonic-gate * char *delims
4787c478bd9Sstevel@tonic-gate *
4797c478bd9Sstevel@tonic-gate * This function determines how many tokens are in the list <list>.
4807c478bd9Sstevel@tonic-gate * The tokens are delimited by fields of characters in the string
4817c478bd9Sstevel@tonic-gate * <delims>. It returns the number of tokens in the list.
4827c478bd9Sstevel@tonic-gate *
4837c478bd9Sstevel@tonic-gate * Arguments:
4847c478bd9Sstevel@tonic-gate * char *list The <delims>list of tokens to scan
4857c478bd9Sstevel@tonic-gate * char *delims The list of delimiters that define the list
4867c478bd9Sstevel@tonic-gate *
4877c478bd9Sstevel@tonic-gate * Returns: int
4887c478bd9Sstevel@tonic-gate * The number of elements in the list.
4897c478bd9Sstevel@tonic-gate *
4907c478bd9Sstevel@tonic-gate * Notes:
4917c478bd9Sstevel@tonic-gate * - This function does not recognize "null" elements. For example,
4927c478bd9Sstevel@tonic-gate * a,b,,,,c,,d contains 4 elememts (if delims contains a ',')
4937c478bd9Sstevel@tonic-gate */
4947c478bd9Sstevel@tonic-gate
4957c478bd9Sstevel@tonic-gate static int
ndevsin(list,delims)4967c478bd9Sstevel@tonic-gate ndevsin(list, delims)
4977c478bd9Sstevel@tonic-gate char *list; /* List to scan */
4987c478bd9Sstevel@tonic-gate char *delims; /* Delimiters */
4997c478bd9Sstevel@tonic-gate {
5007c478bd9Sstevel@tonic-gate char *p; /* Running character pointer */
5017c478bd9Sstevel@tonic-gate int count; /* Number of tokens seen so far */
5027c478bd9Sstevel@tonic-gate int tokflag; /* TRUE if we're parsing a token */
5037c478bd9Sstevel@tonic-gate
5047c478bd9Sstevel@tonic-gate count = 0; /* None seen yet */
5057c478bd9Sstevel@tonic-gate tokflag = FALSE; /* Not in a token */
5067c478bd9Sstevel@tonic-gate
5077c478bd9Sstevel@tonic-gate /* Scan the character-string containing the list of tokens */
5087c478bd9Sstevel@tonic-gate for (p = list ; *p ; p++) {
5097c478bd9Sstevel@tonic-gate
5107c478bd9Sstevel@tonic-gate /* If a delimiter, we're not in a token */
5117c478bd9Sstevel@tonic-gate if (strchr(delims, *p)) tokflag = FALSE;
5127c478bd9Sstevel@tonic-gate
5137c478bd9Sstevel@tonic-gate /* Otherwise, if we weren't in a token, we've found one */
5147c478bd9Sstevel@tonic-gate else if (!tokflag) {
5157c478bd9Sstevel@tonic-gate tokflag = TRUE;
5167c478bd9Sstevel@tonic-gate count++;
5177c478bd9Sstevel@tonic-gate }
5187c478bd9Sstevel@tonic-gate }
5197c478bd9Sstevel@tonic-gate
5207c478bd9Sstevel@tonic-gate /* Return the number of elements in the list */
5217c478bd9Sstevel@tonic-gate return(count);
5227c478bd9Sstevel@tonic-gate }
523