]> git.saurik.com Git - apple/libc.git/blobdiff - gen/err-fbsd.c
Libc-498.tar.gz
[apple/libc.git] / gen / err-fbsd.c
diff --git a/gen/err-fbsd.c b/gen/err-fbsd.c
new file mode 100644 (file)
index 0000000..5137bb9
--- /dev/null
@@ -0,0 +1,287 @@
+/*-
+ * Copyright (c) 1993
+ *     The Regents of the University of California.  All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ * 1. Redistributions of source code must retain the above copyright
+ *    notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ *    notice, this list of conditions and the following disclaimer in the
+ *    documentation and/or other materials provided with the distribution.
+ * 3. All advertising materials mentioning features or use of this software
+ *    must display the following acknowledgement:
+ *     This product includes software developed by the University of
+ *     California, Berkeley and its contributors.
+ * 4. Neither the name of the University nor the names of its contributors
+ *    may be used to endorse or promote products derived from this software
+ *    without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
+ * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
+ * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
+ * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
+ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
+ * SUCH DAMAGE.
+ */
+
+#if defined(LIBC_SCCS) && !defined(lint)
+static char sccsid[] = "@(#)err.c      8.1 (Berkeley) 6/4/93";
+#endif /* LIBC_SCCS and not lint */
+#include <sys/cdefs.h>
+__FBSDID("$FreeBSD: src/lib/libc/gen/err.c,v 1.13 2002/03/29 22:43:41 markm Exp $");
+
+#include "namespace.h"
+#include <err.h>
+#include <errno.h>
+#include <stdarg.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <vis.h>
+#include "un-namespace.h"
+
+#include "libc_private.h"
+
+#ifdef BUILDING_VARIANT
+
+__private_extern__ FILE *_e_err_file; /* file to use for error output */
+__private_extern__ void (*_e_err_exit)(int);
+__private_extern__ void _e_visprintf(FILE * __restrict, const char * __restrict, va_list);
+
+#else /* !BUILDING_VARIANT */
+
+__private_extern__ FILE *_e_err_file = NULL; /* file to use for error output */
+__private_extern__ void (*_e_err_exit)(int) = NULL;
+
+/*
+ * zero means pass as is
+ * 255 means use \nnn (octal)
+ * otherwise use \x (x is value)
+ * (NUL isn't used)
+ */
+static unsigned char escape[256] = {
+     /* NUL  SOH  STX  ETX  EOT  ENQ  ACK  BEL */
+       0  , 255, 255, 255, 255, 255, 255, 'a',
+     /* BS   HT   NL   VT   NP   CR   SO   SI  */
+       'b', 't', 'n', 'v', 'f', 'r', 255, 255,
+     /* DLE  DC1  DC2  DC3  DC4  NAK  SYN  ETB */
+       255, 255, 255, 255, 255, 255, 255, 255,
+     /* CAN  EM   SUB  ESC  FS   GS   RS   US  */
+       255, 255, 255, 255, 255, 255, 255, 255,
+     /* the rest are zero */
+};
+
+/*
+ * Make characters visible.  If we can't allocate enough
+ * memory, we fall back on vfprintf().
+ */
+__private_extern__ void
+_e_visprintf(FILE * __restrict stream, const char * __restrict format, va_list ap)
+{
+       int failed = 0;
+       char *str, *visstr;
+       va_list backup;
+
+       va_copy(backup, ap);
+       vasprintf(&str, format, ap);
+       if (str != NULL) {
+               if ((visstr = malloc(4 * strlen(str) + 1)) != NULL) {
+                       unsigned char *fp = (unsigned char *)str;
+                       unsigned char *tp = (unsigned char *)visstr;
+                       while(*fp) {
+                               switch(escape[*fp]) {
+                               case 0:
+                                       *tp++ = *fp;
+                                       break;
+                               case 255:
+                                       sprintf(tp, "\\%03o", *fp);
+                                       tp += 4;
+                                       break;
+                               default:
+                                       *tp++ = '\\';
+                                       *tp++ = escape[*fp];
+                                       break;
+                               }
+                               fp++;
+                       }
+                       *tp = 0;
+                       fputs(visstr, stream);
+                       free(visstr);
+               } else
+                       failed = 1;
+               free(str);
+       } else
+               failed = 1;
+       if (failed)
+               vfprintf(stream, format, backup);
+       va_end(backup);
+}
+
+/*
+ * This is declared to take a `void *' so that the caller is not required
+ * to include <stdio.h> first.  However, it is really a `FILE *', and the
+ * manual page documents it as such.
+ */
+void
+err_set_file(void *fp)
+{
+       if (fp)
+               _e_err_file = fp;
+       else
+               _e_err_file = stderr;
+}
+
+void
+err_set_exit(void (*ef)(int))
+{
+       _e_err_exit = ef;
+}
+#endif /* !BUILDING_VARIANT */
+
+__weak_reference(_err, err);
+
+void
+_err(int eval, const char *fmt, ...)
+{
+       va_list ap;
+       va_start(ap, fmt);
+       verrc(eval, errno, fmt, ap);
+       va_end(ap);
+}
+
+void
+verr(eval, fmt, ap)
+       int eval;
+       const char *fmt;
+       va_list ap;
+{
+       verrc(eval, errno, fmt, ap);
+}
+
+void
+errc(int eval, int code, const char *fmt, ...)
+{
+       va_list ap;
+       va_start(ap, fmt);
+       verrc(eval, code, fmt, ap);
+       va_end(ap);
+}
+
+void
+verrc(eval, code, fmt, ap)
+       int eval;
+       int code;
+       const char *fmt;
+       va_list ap;
+{
+       if (_e_err_file == 0)
+               err_set_file((FILE *)0);
+       fprintf(_e_err_file, "%s: ", _getprogname());
+       if (fmt != NULL) {
+               _e_visprintf(_e_err_file, fmt, ap);
+               fprintf(_e_err_file, ": ");
+       }
+       fprintf(_e_err_file, "%s\n", strerror(code));
+       if (_e_err_exit)
+               _e_err_exit(eval);
+       exit(eval);
+}
+
+void
+errx(int eval, const char *fmt, ...)
+{
+       va_list ap;
+       va_start(ap, fmt);
+       verrx(eval, fmt, ap);
+       va_end(ap);
+}
+
+void
+verrx(eval, fmt, ap)
+       int eval;
+       const char *fmt;
+       va_list ap;
+{
+       if (_e_err_file == 0)
+               err_set_file((FILE *)0);
+       fprintf(_e_err_file, "%s: ", _getprogname());
+       if (fmt != NULL)
+               _e_visprintf(_e_err_file, fmt, ap);
+       fprintf(_e_err_file, "\n");
+       if (_e_err_exit)
+               _e_err_exit(eval);
+       exit(eval);
+}
+
+__weak_reference(_warn, warn);
+
+void
+_warn(const char *fmt, ...)
+{
+       va_list ap;
+       va_start(ap, fmt);
+       vwarnc(errno, fmt, ap);
+       va_end(ap);
+}
+
+void
+vwarn(fmt, ap)
+       const char *fmt;
+       va_list ap;
+{
+       vwarnc(errno, fmt, ap);
+}
+
+void
+warnc(int code, const char *fmt, ...)
+{
+       va_list ap;
+       va_start(ap, fmt);
+       vwarnc(code, fmt, ap);
+       va_end(ap);
+}
+
+void
+vwarnc(code, fmt, ap)
+       int code;
+       const char *fmt;
+       va_list ap;
+{
+       if (_e_err_file == 0)
+               err_set_file((FILE *)0);
+       fprintf(_e_err_file, "%s: ", _getprogname());
+       if (fmt != NULL) {
+               _e_visprintf(_e_err_file, fmt, ap);
+               fprintf(_e_err_file, ": ");
+       }
+       fprintf(_e_err_file, "%s\n", strerror(code));
+}
+
+void
+warnx(const char *fmt, ...)
+{
+       va_list ap;
+       va_start(ap, fmt);
+       vwarnx(fmt, ap);
+       va_end(ap);
+}
+
+void
+vwarnx(fmt, ap)
+       const char *fmt;
+       va_list ap;
+{
+       if (_e_err_file == 0)
+               err_set_file((FILE *)0);
+       fprintf(_e_err_file, "%s: ", _getprogname());
+       if (fmt != NULL)
+               _e_visprintf(_e_err_file, fmt, ap);
+       fprintf(_e_err_file, "\n");
+}