source: trunk/minix/commands/ash/error.c@ 11

Last change on this file since 11 was 9, checked in by Mattia Monga, 14 years ago

Minix 3.1.2a

File size: 6.2 KB
Line 
1/*-
2 * Copyright (c) 1991 The Regents of the University of California.
3 * All rights reserved.
4 *
5 * This code is derived from software contributed to Berkeley by
6 * Kenneth Almquist.
7 *
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions
10 * are met:
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
16 * 3. All advertising materials mentioning features or use of this software
17 * must display the following acknowledgement:
18 * This product includes software developed by the University of
19 * California, Berkeley and its contributors.
20 * 4. Neither the name of the University nor the names of its contributors
21 * may be used to endorse or promote products derived from this software
22 * without specific prior written permission.
23 *
24 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
25 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
26 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
27 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
28 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
29 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
30 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
31 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
32 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
33 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
34 * SUCH DAMAGE.
35 */
36
37#ifndef lint
38static char sccsid[] = "@(#)error.c 5.1 (Berkeley) 3/7/91";
39#endif /* not lint */
40
41/*
42 * Errors and exceptions.
43 */
44
45#include "shell.h"
46#include "main.h"
47#include "options.h"
48#include "output.h"
49#include "error.h"
50#include <sys/types.h>
51#include <signal.h>
52#ifdef __STDC__
53#include "stdarg.h"
54#else
55#include <varargs.h>
56#endif
57#include <errno.h>
58
59
60/*
61 * Code to handle exceptions in C.
62 */
63
64struct jmploc *handler;
65int exception;
66volatile int suppressint;
67volatile int intpending;
68char *commandname;
69
70
71/*
72 * Called to raise an exception. Since C doesn't include exceptions, we
73 * just do a longjmp to the exception handler. The type of exception is
74 * stored in the global variable "exception".
75 */
76
77void
78exraise(e) {
79 if (handler == NULL)
80 abort();
81 exception = e;
82 longjmp(handler->loc, 1);
83}
84
85
86/*
87 * Called from trap.c when a SIGINT is received. (If the user specifies
88 * that SIGINT is to be trapped or ignored using the trap builtin, then
89 * this routine is not called.) Suppressint is nonzero when interrupts
90 * are held using the INTOFF macro. The call to _exit is necessary because
91 * there is a short period after a fork before the signal handlers are
92 * set to the appropriate value for the child. (The test for iflag is
93 * just defensive programming.)
94 */
95
96void
97onint() {
98 if (suppressint) {
99 intpending++;
100 return;
101 }
102 intpending = 0;
103#ifdef BSD
104 sigsetmask(0);
105#endif
106 if (rootshell && iflag)
107 exraise(EXINT);
108 else
109 _exit(128 + SIGINT);
110}
111
112
113
114void
115error2(a, b)
116 char *a, *b;
117 {
118 error("%s: %s", a, b);
119}
120
121
122/*
123 * Error is called to raise the error exception. If the first argument
124 * is not NULL then error prints an error message using printf style
125 * formatting. It then raises the error exception.
126 */
127
128#ifdef __STDC__
129void
130error(char *msg, ...) {
131#else
132void
133error(va_alist)
134 va_dcl
135 {
136 char *msg;
137#endif
138 va_list ap;
139
140 CLEAR_PENDING_INT;
141 INTOFF;
142#ifdef __STDC__
143 va_start(ap, msg);
144#else
145 va_start(ap);
146 msg = va_arg(ap, char *);
147#endif
148#if DEBUG
149 if (msg)
150 TRACE(("error(\"%s\") pid=%d\n", msg, getpid()));
151 else
152 TRACE(("error(NULL) pid=%d\n", getpid()));
153#endif
154 if (msg) {
155 if (commandname)
156 outfmt(&errout, "%s: ", commandname);
157 doformat(&errout, msg, ap);
158 out2c('\n');
159 }
160 va_end(ap);
161 flushall();
162 exraise(EXERROR);
163}
164
165
166#ifdef notdef /* These strange error messages only confuse. -- kjb */
167/*
168 * Table of error messages.
169 */
170
171struct errname {
172 short errcode; /* error number */
173 short action; /* operation which encountered the error */
174 char *msg; /* text describing the error */
175};
176
177
178#define ALL (E_OPEN|E_CREAT|E_EXEC)
179
180STATIC const struct errname errormsg[] = {
181 EINTR, ALL, "interrupted",
182 EACCES, ALL, "permission denied",
183 EIO, ALL, "I/O error",
184 ENOENT, E_OPEN, "no such file",
185 ENOENT, E_CREAT, "directory nonexistent",
186 ENOENT, E_EXEC, "not found",
187 ENOTDIR, E_OPEN, "no such file",
188 ENOTDIR, E_CREAT, "directory nonexistent",
189 ENOTDIR, E_EXEC, "not found",
190 ENOEXEC, ALL, "not an executable",
191 EISDIR, ALL, "is a directory",
192/* EMFILE, ALL, "too many open files", */
193 ENFILE, ALL, "file table overflow",
194 ENOSPC, ALL, "file system full",
195#ifdef EDQUOT
196 EDQUOT, ALL, "disk quota exceeded",
197#endif
198#ifdef ENOSR
199 ENOSR, ALL, "no streams resources",
200#endif
201 ENXIO, ALL, "no such device or address",
202 EROFS, ALL, "read-only file system",
203 ETXTBSY, ALL, "text busy",
204#ifdef SYSV
205 EAGAIN, E_EXEC, "not enough memory",
206#endif
207 ENOMEM, ALL, "not enough memory",
208#ifdef ENOLINK
209 ENOLINK, ALL, "remote access failed",
210#endif
211#ifdef EMULTIHOP
212 EMULTIHOP, ALL, "remote access failed",
213#endif
214#ifdef ECOMM
215 ECOMM, ALL, "remote access failed",
216#endif
217#ifdef ESTALE
218 ESTALE, ALL, "remote access failed",
219#endif
220#ifdef ETIMEDOUT
221 ETIMEDOUT, ALL, "remote access failed",
222#endif
223#ifdef ELOOP
224 ELOOP, ALL, "symbolic link loop",
225#endif
226 E2BIG, E_EXEC, "argument list too long",
227#ifdef ELIBACC
228 ELIBACC, E_EXEC, "shared library missing",
229#endif
230 0, 0, NULL
231};
232
233
234/*
235 * Return a string describing an error. The returned string may be a
236 * pointer to a static buffer that will be overwritten on the next call.
237 * Action describes the operation that got the error.
238 */
239
240char *
241errmsg(e, action) {
242 const struct errname *ep;
243 static char buf[12];
244
245 for (ep = errormsg ; ep->errcode ; ep++) {
246 if (ep->errcode == e && (ep->action & action) != 0)
247 return ep->msg;
248 }
249 fmtstr(buf, sizeof buf, "error %d", e);
250 return buf;
251}
252#endif
Note: See TracBrowser for help on using the repository browser.