xref: /titanic_50/usr/src/cmd/devmgmt/cmds/devfree.c (revision 7c478bd95313f5f23a4c958a745db2134aa03244)
1*7c478bd9Sstevel@tonic-gate /*
2*7c478bd9Sstevel@tonic-gate  * CDDL HEADER START
3*7c478bd9Sstevel@tonic-gate  *
4*7c478bd9Sstevel@tonic-gate  * The contents of this file are subject to the terms of the
5*7c478bd9Sstevel@tonic-gate  * Common Development and Distribution License, Version 1.0 only
6*7c478bd9Sstevel@tonic-gate  * (the "License").  You may not use this file except in compliance
7*7c478bd9Sstevel@tonic-gate  * with the License.
8*7c478bd9Sstevel@tonic-gate  *
9*7c478bd9Sstevel@tonic-gate  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10*7c478bd9Sstevel@tonic-gate  * or http://www.opensolaris.org/os/licensing.
11*7c478bd9Sstevel@tonic-gate  * See the License for the specific language governing permissions
12*7c478bd9Sstevel@tonic-gate  * and limitations under the License.
13*7c478bd9Sstevel@tonic-gate  *
14*7c478bd9Sstevel@tonic-gate  * When distributing Covered Code, include this CDDL HEADER in each
15*7c478bd9Sstevel@tonic-gate  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16*7c478bd9Sstevel@tonic-gate  * If applicable, add the following below this CDDL HEADER, with the
17*7c478bd9Sstevel@tonic-gate  * fields enclosed by brackets "[]" replaced with your own identifying
18*7c478bd9Sstevel@tonic-gate  * information: Portions Copyright [yyyy] [name of copyright owner]
19*7c478bd9Sstevel@tonic-gate  *
20*7c478bd9Sstevel@tonic-gate  * CDDL HEADER END
21*7c478bd9Sstevel@tonic-gate  */
22*7c478bd9Sstevel@tonic-gate /*
23*7c478bd9Sstevel@tonic-gate  * Copyright 2003 Sun Microsystems, Inc.  All rights reserved.
24*7c478bd9Sstevel@tonic-gate  * Use is subject to license terms.
25*7c478bd9Sstevel@tonic-gate  */
26*7c478bd9Sstevel@tonic-gate 
27*7c478bd9Sstevel@tonic-gate /*    Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
28*7c478bd9Sstevel@tonic-gate /*      All Rights Reserved   */
29*7c478bd9Sstevel@tonic-gate 
30*7c478bd9Sstevel@tonic-gate 
31*7c478bd9Sstevel@tonic-gate #pragma ident	"%Z%%M%	%I%	%E% SMI"
32*7c478bd9Sstevel@tonic-gate 
33*7c478bd9Sstevel@tonic-gate /*
34*7c478bd9Sstevel@tonic-gate  *	devfree key [device [...]]
35*7c478bd9Sstevel@tonic-gate  */
36*7c478bd9Sstevel@tonic-gate 
37*7c478bd9Sstevel@tonic-gate #include	<sys/types.h>
38*7c478bd9Sstevel@tonic-gate #include	<sys/param.h>
39*7c478bd9Sstevel@tonic-gate #include	<stdio.h>
40*7c478bd9Sstevel@tonic-gate #include	<errno.h>
41*7c478bd9Sstevel@tonic-gate #include	<stdlib.h>
42*7c478bd9Sstevel@tonic-gate #include	<string.h>
43*7c478bd9Sstevel@tonic-gate #include	<fmtmsg.h>
44*7c478bd9Sstevel@tonic-gate #include	<devmgmt.h>
45*7c478bd9Sstevel@tonic-gate #include	<values.h>
46*7c478bd9Sstevel@tonic-gate #include	<devtab.h>
47*7c478bd9Sstevel@tonic-gate 
48*7c478bd9Sstevel@tonic-gate 
49*7c478bd9Sstevel@tonic-gate /*
50*7c478bd9Sstevel@tonic-gate  *  Local definitions
51*7c478bd9Sstevel@tonic-gate  *	TRUE		Boolean TRUE value
52*7c478bd9Sstevel@tonic-gate  *	FALSE		Boolean FALSE value
53*7c478bd9Sstevel@tonic-gate  */
54*7c478bd9Sstevel@tonic-gate #ifndef		TRUE
55*7c478bd9Sstevel@tonic-gate #define		TRUE		('t')
56*7c478bd9Sstevel@tonic-gate #endif
57*7c478bd9Sstevel@tonic-gate 
58*7c478bd9Sstevel@tonic-gate #ifndef		FALSE
59*7c478bd9Sstevel@tonic-gate #define		FALSE		0
60*7c478bd9Sstevel@tonic-gate #endif
61*7c478bd9Sstevel@tonic-gate 
62*7c478bd9Sstevel@tonic-gate 
63*7c478bd9Sstevel@tonic-gate /*
64*7c478bd9Sstevel@tonic-gate  *  Exit codes:
65*7c478bd9Sstevel@tonic-gate  *	EX_OK		Exit code for all went well
66*7c478bd9Sstevel@tonic-gate  *	EX_ERROR	Exit code for something failed
67*7c478bd9Sstevel@tonic-gate  *	EX_TBLERR	Exit code for errors relating to device or lock tables
68*7c478bd9Sstevel@tonic-gate  *	EX_NOFREE	Exit code for free failed
69*7c478bd9Sstevel@tonic-gate  */
70*7c478bd9Sstevel@tonic-gate 
71*7c478bd9Sstevel@tonic-gate #define		EX_OK		0
72*7c478bd9Sstevel@tonic-gate #define		EX_ERROR	1
73*7c478bd9Sstevel@tonic-gate #define		EX_TBLERR	2
74*7c478bd9Sstevel@tonic-gate #define		EX_NOFREE	3
75*7c478bd9Sstevel@tonic-gate 
76*7c478bd9Sstevel@tonic-gate 
77*7c478bd9Sstevel@tonic-gate /*
78*7c478bd9Sstevel@tonic-gate  * Messages
79*7c478bd9Sstevel@tonic-gate  *	M_USAGE		Usage error
80*7c478bd9Sstevel@tonic-gate  *	M_INVKEY	Invalid key specified
81*7c478bd9Sstevel@tonic-gate  *	M_NOTRSVD	Attempting to free something not alloc'd
82*7c478bd9Sstevel@tonic-gate  *	M_NOTONKEY	Attempting to free with wrong key
83*7c478bd9Sstevel@tonic-gate  *	M_DEVTAB	Error opening the device table
84*7c478bd9Sstevel@tonic-gate  *	M_RSVTAB	Error opening the device-reservation table
85*7c478bd9Sstevel@tonic-gate  *	M_ERROR		Some internal error
86*7c478bd9Sstevel@tonic-gate  */
87*7c478bd9Sstevel@tonic-gate 
88*7c478bd9Sstevel@tonic-gate #define		M_USAGE		"usage: devfree key [device [...]]"
89*7c478bd9Sstevel@tonic-gate #define		M_INVKEY	"Invalid key: %s"
90*7c478bd9Sstevel@tonic-gate #define		M_NOTRSVD	"Device not reserved: %s"
91*7c478bd9Sstevel@tonic-gate #define		M_NOTONKEY	"Cannot unreserve device: %s"
92*7c478bd9Sstevel@tonic-gate #define		M_DEVTAB	"Cannot open the device table: %s"
93*7c478bd9Sstevel@tonic-gate #define		M_RSVTAB	"Cannot open the device-reservation table: %s"
94*7c478bd9Sstevel@tonic-gate #define		M_ERROR		"Internal error, errno=%d"
95*7c478bd9Sstevel@tonic-gate 
96*7c478bd9Sstevel@tonic-gate 
97*7c478bd9Sstevel@tonic-gate /*
98*7c478bd9Sstevel@tonic-gate  *  Local functions and static data
99*7c478bd9Sstevel@tonic-gate  *	stdmsg(r,l,s,m)		Macro for standard message generation
100*7c478bd9Sstevel@tonic-gate  *				r	MM_NRECOV or MM_RECOV (recoverability)
101*7c478bd9Sstevel@tonic-gate  *				l	Label
102*7c478bd9Sstevel@tonic-gate  *				s	Severity
103*7c478bd9Sstevel@tonic-gate  *				m	Message
104*7c478bd9Sstevel@tonic-gate  *	lbl			Buffer for the label-component of a message.
105*7c478bd9Sstevel@tonic-gate  *	msg			Buffer for the text-component of a message.
106*7c478bd9Sstevel@tonic-gate  */
107*7c478bd9Sstevel@tonic-gate 
108*7c478bd9Sstevel@tonic-gate #define	stdmsg(r,l,s,m)	(void) fmtmsg(MM_PRINT|MM_UTIL|r,l,s,m,MM_NULLACT,MM_NULLTAG)
109*7c478bd9Sstevel@tonic-gate 
110*7c478bd9Sstevel@tonic-gate static	char	lbl[MM_MXLABELLN+1];
111*7c478bd9Sstevel@tonic-gate static	char	msg[MM_MXTXTLN+1];
112*7c478bd9Sstevel@tonic-gate 
113*7c478bd9Sstevel@tonic-gate /*
114*7c478bd9Sstevel@tonic-gate  *  devfree key [device [device [...]]]
115*7c478bd9Sstevel@tonic-gate  *
116*7c478bd9Sstevel@tonic-gate  *	This command frees devices that have been reserved using
117*7c478bd9Sstevel@tonic-gate  *	the devreserv command (or the devreserv() function).
118*7c478bd9Sstevel@tonic-gate  *
119*7c478bd9Sstevel@tonic-gate  *  Options:  None
120*7c478bd9Sstevel@tonic-gate  *
121*7c478bd9Sstevel@tonic-gate  *  Arguments:
122*7c478bd9Sstevel@tonic-gate  *	key		The key on which the device to free was allocated on.
123*7c478bd9Sstevel@tonic-gate  *			If omitted, all keys are assumed.
124*7c478bd9Sstevel@tonic-gate  *	device		The device to free.  If omitted, all devices allocated
125*7c478bd9Sstevel@tonic-gate  *			using the key are freed.
126*7c478bd9Sstevel@tonic-gate  *
127*7c478bd9Sstevel@tonic-gate  *  Command Values:
128*7c478bd9Sstevel@tonic-gate  *	EX_OK		0	Device(s) successfully freed
129*7c478bd9Sstevel@tonic-gate  *	EX_ERROR	1	A syntax error or other error occurred
130*7c478bd9Sstevel@tonic-gate  *	EX_TBLERR	2	A problem with device management tables
131*7c478bd9Sstevel@tonic-gate  *	EX_NOFREE	3	A requested device couldn't be freed
132*7c478bd9Sstevel@tonic-gate  */
133*7c478bd9Sstevel@tonic-gate 
134*7c478bd9Sstevel@tonic-gate main(argc, argv)
135*7c478bd9Sstevel@tonic-gate 	int		argc;		/* Arg count */
136*7c478bd9Sstevel@tonic-gate 	char	       *argv[];		/* Arg vector */
137*7c478bd9Sstevel@tonic-gate {
138*7c478bd9Sstevel@tonic-gate 	/* Automatics */
139*7c478bd9Sstevel@tonic-gate 	char		      **argp;		/* Ptr to current argument */
140*7c478bd9Sstevel@tonic-gate 	struct reservdev      **rsvd;		/* Ptr to list of locks */
141*7c478bd9Sstevel@tonic-gate 	struct reservdev      **plk;		/* Running ptr to locks */
142*7c478bd9Sstevel@tonic-gate 	char		       *devtab;		/* Ptr to device table name */
143*7c478bd9Sstevel@tonic-gate 	char		       *rsvtab;		/* Ptr to dev-rsv-tbl name */
144*7c478bd9Sstevel@tonic-gate 	char		       *p;		/* Temp char pointer */
145*7c478bd9Sstevel@tonic-gate 	int			argcount;	/* Number of args on cmd */
146*7c478bd9Sstevel@tonic-gate 	long			lkey;		/* Key for locking (long) */
147*7c478bd9Sstevel@tonic-gate 	int			key;		/* Key for locking */
148*7c478bd9Sstevel@tonic-gate 	int			halt;		/* TRUE if we need to stop */
149*7c478bd9Sstevel@tonic-gate 	int			sev;		/* Message severity */
150*7c478bd9Sstevel@tonic-gate 	int			exitcode;	/* Value of command */
151*7c478bd9Sstevel@tonic-gate 	int			syntaxerr;	/* Flag, TRUE if syntax error */
152*7c478bd9Sstevel@tonic-gate 	int			exitcd;		/* Value for exit() */
153*7c478bd9Sstevel@tonic-gate 	int			c;		/* Option character */
154*7c478bd9Sstevel@tonic-gate 
155*7c478bd9Sstevel@tonic-gate 
156*7c478bd9Sstevel@tonic-gate 	/*
157*7c478bd9Sstevel@tonic-gate 	 * Initializations
158*7c478bd9Sstevel@tonic-gate 	 */
159*7c478bd9Sstevel@tonic-gate 
160*7c478bd9Sstevel@tonic-gate 	/* Build a message label */
161*7c478bd9Sstevel@tonic-gate 	if (p = strrchr(argv[0], '/')) p++;
162*7c478bd9Sstevel@tonic-gate 	else p = argv[0];
163*7c478bd9Sstevel@tonic-gate 	(void) strlcat(strcpy(lbl, "UX:"), p, sizeof(lbl));
164*7c478bd9Sstevel@tonic-gate 
165*7c478bd9Sstevel@tonic-gate 	/* Make only the text component of messages appear (remove this in SVR4.1) */
166*7c478bd9Sstevel@tonic-gate 	(void) putenv("MSGVERB=text");
167*7c478bd9Sstevel@tonic-gate 
168*7c478bd9Sstevel@tonic-gate 
169*7c478bd9Sstevel@tonic-gate 	/*
170*7c478bd9Sstevel@tonic-gate 	 * Parse the options from the command line
171*7c478bd9Sstevel@tonic-gate 	 */
172*7c478bd9Sstevel@tonic-gate 
173*7c478bd9Sstevel@tonic-gate 	opterr = 0;
174*7c478bd9Sstevel@tonic-gate 	syntaxerr = FALSE;
175*7c478bd9Sstevel@tonic-gate 	while ((c = getopt(argc, argv, "")) != EOF) switch(c) {
176*7c478bd9Sstevel@tonic-gate 	default:
177*7c478bd9Sstevel@tonic-gate 	    syntaxerr = FALSE;
178*7c478bd9Sstevel@tonic-gate 	    break;
179*7c478bd9Sstevel@tonic-gate 	}
180*7c478bd9Sstevel@tonic-gate 
181*7c478bd9Sstevel@tonic-gate 
182*7c478bd9Sstevel@tonic-gate 	/* Argument initializations */
183*7c478bd9Sstevel@tonic-gate 	argp = &argv[optind];
184*7c478bd9Sstevel@tonic-gate 	if ((argcount = argc-optind) < 1) syntaxerr = TRUE;
185*7c478bd9Sstevel@tonic-gate 
186*7c478bd9Sstevel@tonic-gate 
187*7c478bd9Sstevel@tonic-gate 	/* If there's (an obvious) syntax error, write a message and quit */
188*7c478bd9Sstevel@tonic-gate 	if (syntaxerr) {
189*7c478bd9Sstevel@tonic-gate 	    stdmsg(MM_NRECOV, lbl, MM_ERROR, M_USAGE);
190*7c478bd9Sstevel@tonic-gate 	    exit(EX_ERROR);
191*7c478bd9Sstevel@tonic-gate 	}
192*7c478bd9Sstevel@tonic-gate 
193*7c478bd9Sstevel@tonic-gate 
194*7c478bd9Sstevel@tonic-gate 	/*
195*7c478bd9Sstevel@tonic-gate 	 *  devfree key
196*7c478bd9Sstevel@tonic-gate 	 *
197*7c478bd9Sstevel@tonic-gate 	 *  	Free all devices that have been reserved using the key "key".
198*7c478bd9Sstevel@tonic-gate 	 */
199*7c478bd9Sstevel@tonic-gate 
200*7c478bd9Sstevel@tonic-gate 	if (argcount == 1) {
201*7c478bd9Sstevel@tonic-gate 
202*7c478bd9Sstevel@tonic-gate 	    /* Extract the key from the command */
203*7c478bd9Sstevel@tonic-gate 	    lkey = strtol(*argp, &p, 10);
204*7c478bd9Sstevel@tonic-gate 	    if (*p || (lkey <= 0) || (lkey > MAXINT)) {
205*7c478bd9Sstevel@tonic-gate 		(void) snprintf(msg, sizeof (msg), M_INVKEY, *argp);
206*7c478bd9Sstevel@tonic-gate 		stdmsg(MM_NRECOV, lbl, MM_ERROR, msg);
207*7c478bd9Sstevel@tonic-gate 		exit(EX_ERROR);
208*7c478bd9Sstevel@tonic-gate 	    }
209*7c478bd9Sstevel@tonic-gate 	    key = (int) lkey;
210*7c478bd9Sstevel@tonic-gate 
211*7c478bd9Sstevel@tonic-gate 	    /* Get the list of devices currently reserved */
212*7c478bd9Sstevel@tonic-gate 	    if (rsvd = reservdev()) {
213*7c478bd9Sstevel@tonic-gate 		exitcd = EX_OK;
214*7c478bd9Sstevel@tonic-gate 		for (plk = rsvd ; *plk ; plk++) {
215*7c478bd9Sstevel@tonic-gate 		    if ((*plk)->key == key)
216*7c478bd9Sstevel@tonic-gate 			if (devfree(key, (*plk)->devname) != 0)
217*7c478bd9Sstevel@tonic-gate 			    exitcd = EX_NOFREE;
218*7c478bd9Sstevel@tonic-gate 		}
219*7c478bd9Sstevel@tonic-gate 	    } else {
220*7c478bd9Sstevel@tonic-gate 		if (((errno == ENOENT) || (errno == EACCES)) && (rsvtab = _rsvtabpath())) {
221*7c478bd9Sstevel@tonic-gate 		    (void) snprintf(msg, sizeof(msg), M_RSVTAB, rsvtab);
222*7c478bd9Sstevel@tonic-gate 		    exitcd = EX_TBLERR;
223*7c478bd9Sstevel@tonic-gate 		    sev = MM_ERROR;
224*7c478bd9Sstevel@tonic-gate 		} else {
225*7c478bd9Sstevel@tonic-gate 		    (void) snprintf(msg, sizeof (msg), M_ERROR, errno);
226*7c478bd9Sstevel@tonic-gate 		    exitcd = EX_ERROR;
227*7c478bd9Sstevel@tonic-gate 		    sev = MM_HALT;
228*7c478bd9Sstevel@tonic-gate 		}
229*7c478bd9Sstevel@tonic-gate 		stdmsg(MM_NRECOV, lbl, sev, msg);
230*7c478bd9Sstevel@tonic-gate 	    }
231*7c478bd9Sstevel@tonic-gate 
232*7c478bd9Sstevel@tonic-gate 	    /* Done */
233*7c478bd9Sstevel@tonic-gate 	    exit(exitcd);
234*7c478bd9Sstevel@tonic-gate 	}
235*7c478bd9Sstevel@tonic-gate 
236*7c478bd9Sstevel@tonic-gate 
237*7c478bd9Sstevel@tonic-gate 	/*
238*7c478bd9Sstevel@tonic-gate 	 *  devfree key device [...]
239*7c478bd9Sstevel@tonic-gate 	 *
240*7c478bd9Sstevel@tonic-gate 	 *	Free specific devices
241*7c478bd9Sstevel@tonic-gate 	 */
242*7c478bd9Sstevel@tonic-gate 
243*7c478bd9Sstevel@tonic-gate 	/* Open the device file (if there's one to be opened) */
244*7c478bd9Sstevel@tonic-gate 	if (!_opendevtab("r")) {
245*7c478bd9Sstevel@tonic-gate 	    if (devtab = _devtabpath()) {
246*7c478bd9Sstevel@tonic-gate 		(void) snprintf(msg, sizeof(msg), M_DEVTAB, devtab);
247*7c478bd9Sstevel@tonic-gate 		exitcd = EX_TBLERR;
248*7c478bd9Sstevel@tonic-gate 		sev = MM_ERROR;
249*7c478bd9Sstevel@tonic-gate 	    } else {
250*7c478bd9Sstevel@tonic-gate 		(void) snprintf(msg, sizeof (msg), M_ERROR, errno);
251*7c478bd9Sstevel@tonic-gate 		exitcd = EX_ERROR;
252*7c478bd9Sstevel@tonic-gate 		sev = MM_HALT;
253*7c478bd9Sstevel@tonic-gate 	    }
254*7c478bd9Sstevel@tonic-gate 	    stdmsg(MM_NRECOV, lbl, sev, msg);
255*7c478bd9Sstevel@tonic-gate 	    exit(exitcd);
256*7c478bd9Sstevel@tonic-gate 	}
257*7c478bd9Sstevel@tonic-gate 
258*7c478bd9Sstevel@tonic-gate 	/* Extract the key from the command */
259*7c478bd9Sstevel@tonic-gate 	lkey = strtol(*argp, &p, 10);
260*7c478bd9Sstevel@tonic-gate 	if (*p || (lkey <= 0) || (lkey > MAXINT)) {
261*7c478bd9Sstevel@tonic-gate 	    (void) snprintf(msg, sizeof(msg), M_INVKEY, *argp);
262*7c478bd9Sstevel@tonic-gate 	    stdmsg(MM_NRECOV, lbl, MM_ERROR, msg);
263*7c478bd9Sstevel@tonic-gate 	    exit(EX_ERROR);
264*7c478bd9Sstevel@tonic-gate 	}
265*7c478bd9Sstevel@tonic-gate 	key = (int) lkey;
266*7c478bd9Sstevel@tonic-gate 	argp++;
267*7c478bd9Sstevel@tonic-gate 
268*7c478bd9Sstevel@tonic-gate 	/* Loop through the list of devices to free */
269*7c478bd9Sstevel@tonic-gate 	exitcode = EX_OK;
270*7c478bd9Sstevel@tonic-gate 	halt = FALSE;
271*7c478bd9Sstevel@tonic-gate 	while (!halt && *argp) {
272*7c478bd9Sstevel@tonic-gate 
273*7c478bd9Sstevel@tonic-gate 	    /* Try to free the device */
274*7c478bd9Sstevel@tonic-gate 	    if (devfree(key, *argp) != 0) {
275*7c478bd9Sstevel@tonic-gate 		if ((errno == EACCES) || (errno == ENOENT)) {
276*7c478bd9Sstevel@tonic-gate 
277*7c478bd9Sstevel@tonic-gate 		    /* Can't get at reservation file */
278*7c478bd9Sstevel@tonic-gate 		    if (rsvtab = _rsvtabpath()) {
279*7c478bd9Sstevel@tonic-gate 			exitcode = EX_TBLERR;
280*7c478bd9Sstevel@tonic-gate 			(void) snprintf(msg, sizeof(msg), M_RSVTAB, rsvtab);
281*7c478bd9Sstevel@tonic-gate 			sev = MM_ERROR;
282*7c478bd9Sstevel@tonic-gate 		    }
283*7c478bd9Sstevel@tonic-gate 		    else {
284*7c478bd9Sstevel@tonic-gate 			exitcode = EX_ERROR;
285*7c478bd9Sstevel@tonic-gate 			(void) snprintf(msg, sizeof (msg), M_ERROR, errno);
286*7c478bd9Sstevel@tonic-gate 			sev = MM_HALT;
287*7c478bd9Sstevel@tonic-gate 		    }
288*7c478bd9Sstevel@tonic-gate 		    stdmsg(MM_NRECOV, lbl, sev, msg);
289*7c478bd9Sstevel@tonic-gate 		    halt = TRUE;
290*7c478bd9Sstevel@tonic-gate 		}
291*7c478bd9Sstevel@tonic-gate 	        else if (errno == EPERM) {
292*7c478bd9Sstevel@tonic-gate 
293*7c478bd9Sstevel@tonic-gate 		    /* Wrong key */
294*7c478bd9Sstevel@tonic-gate 		    (void) snprintf(msg, sizeof(msg), M_NOTONKEY, *argp);
295*7c478bd9Sstevel@tonic-gate 		    stdmsg(MM_NRECOV, lbl, MM_ERROR, msg);
296*7c478bd9Sstevel@tonic-gate 		    exitcode = EX_NOFREE;
297*7c478bd9Sstevel@tonic-gate 		}
298*7c478bd9Sstevel@tonic-gate 		else if (errno == EINVAL) {
299*7c478bd9Sstevel@tonic-gate 
300*7c478bd9Sstevel@tonic-gate 		    /* Device not reserved */
301*7c478bd9Sstevel@tonic-gate 		    (void) snprintf(msg, sizeof(msg), M_NOTRSVD, *argp);
302*7c478bd9Sstevel@tonic-gate 		    stdmsg(MM_NRECOV, lbl, MM_ERROR, msg);
303*7c478bd9Sstevel@tonic-gate 		    exitcode = EX_NOFREE;
304*7c478bd9Sstevel@tonic-gate 		}
305*7c478bd9Sstevel@tonic-gate 
306*7c478bd9Sstevel@tonic-gate 		else {
307*7c478bd9Sstevel@tonic-gate 
308*7c478bd9Sstevel@tonic-gate 		    /* Some other strange error occurred */
309*7c478bd9Sstevel@tonic-gate 		    (void) snprintf(msg, sizeof (msg), M_ERROR, errno);
310*7c478bd9Sstevel@tonic-gate 		    stdmsg(MM_NRECOV, lbl, MM_HALT, msg);
311*7c478bd9Sstevel@tonic-gate 		    exitcode = EX_ERROR;
312*7c478bd9Sstevel@tonic-gate 		    halt = TRUE;
313*7c478bd9Sstevel@tonic-gate 		}
314*7c478bd9Sstevel@tonic-gate 	    }
315*7c478bd9Sstevel@tonic-gate 	    argp++;
316*7c478bd9Sstevel@tonic-gate 	}
317*7c478bd9Sstevel@tonic-gate 
318*7c478bd9Sstevel@tonic-gate 
319*7c478bd9Sstevel@tonic-gate 	/* Exit with the appropriate code */
320*7c478bd9Sstevel@tonic-gate 	return(exitcode);
321*7c478bd9Sstevel@tonic-gate }
322