From b3c20a361d75caf75f670b2fcd64ab8668c16ca5 Mon Sep 17 00:00:00 2001 From: Jakub Jelinek Date: Mon, 15 Nov 2004 09:47:23 +0000 Subject: Updated to fedora-glibc-20041115T0915 --- sysdeps/posix/libc_fatal.c | 125 +++++++++++++++++++++++++----- sysdeps/unix/sysv/linux/libc_fatal.c | 144 +++++++++++++++++++++++++++++------ 2 files changed, 227 insertions(+), 42 deletions(-) (limited to 'sysdeps') diff --git a/sysdeps/posix/libc_fatal.c b/sysdeps/posix/libc_fatal.c index 7e45174708..fac8cbd051 100644 --- a/sysdeps/posix/libc_fatal.c +++ b/sysdeps/posix/libc_fatal.c @@ -1,4 +1,4 @@ -/* Copyright (C) 1993, 1994, 1995, 1997, 2000 Free Software Foundation, Inc. +/* Copyright (C) 1993,1994,1995,1997,2000,2004 Free Software Foundation, Inc. This file is part of the GNU C Library. The GNU C Library is free software; you can redistribute it and/or @@ -16,44 +16,131 @@ Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. */ +#include +#include +#include +#include +#include #include #include -#include -#include -#include #include +#include +#include +#include +#include #ifdef FATAL_PREPARE_INCLUDE #include FATAL_PREPARE_INCLUDE #endif +struct str_list +{ + const char *str; + size_t len; + struct str_list *next; +}; + + /* Abort with an error message. */ void -__libc_fatal (message) - const char *message; +__libc_message (int do_abort, const char *fmt, ...) { - size_t len = strlen (message); + va_list ap; + va_list ap_copy; + int fd = -1; + + va_start (ap, fmt); + va_copy (ap_copy, ap); #ifdef FATAL_PREPARE FATAL_PREPARE; #endif - while (len > 0) + /* Open a descriptor for /dev/tty unless the user explicitly + requests errors on standard error. */ + const char *on_2 = __secure_getenv ("LIBC_FATAL_STDERR_"); + if (on_2 == NULL || *on_2 == '\0') + fd = open_not_cancel_2 (_PATH_TTY, O_RDWR | O_NOCTTY | O_NDELAY); + + if (fd == -1) + fd = STDERR_FILENO; + + struct str_list *list = NULL; + int nlist = 0; + + const char *cp = fmt; + while (*cp != '\0') { - register int count = __write (STDERR_FILENO, message, len); - if (count > 0) + /* Find the next "%s" or the end of the string. */ + char *next = cp; + while (next[0] != '%' || next[1] != 's') { - message += count; - len -= count; + next = __strchrnul (next + 1, '%'); + + if (next[0] == '\0') + break; } - else if (count < 0 -#ifdef EINTR - && errno != EINTR -#endif - ) - break; + + /* Determine what to print. */ + const char *str; + size_t len; + if (cp[0] == '%' && cp[1] == 's') + { + str = va_arg (ap, const char *); + len = strlen (str); + cp += 2; + } + else + { + str = cp; + len = next - cp; + cp = next; + } + + struct str_list *newp = alloca (sizeof (struct str_list)); + newp->str = str; + newp->len = len; + newp->next = list; + list = newp; + ++nlist; + } + + bool written = false; + if (nlist > 0) + { + struct iovec *iov = alloca (nlist * sizeof (struct iovec)); + ssize_t total = 0; + + for (int cnt = nlist - 1; cnt >= 0; --cnt) + { + iov[cnt].iov_base = list->str; + iov[cnt].iov_len = list->len; + total += list->len; + list = list->next; + } + + if (TEMP_FAILURE_RETRY (__writev (fd, iov, nlist)) == total) + written = true; } - abort (); + va_end (ap); + + /* If we had no success writing the message, use syslog. */ + if (! written) + vsyslog (LOG_ERR, fmt, ap_copy); + + va_end (ap_copy); + + if (do_abort() + /* Kill the application. */ + abort (); +} + + +void +__libc_fatal (message) + const char *message; +{ + __libc_message (1, "%s", message); } libc_hidden_def (__libc_fatal) diff --git a/sysdeps/unix/sysv/linux/libc_fatal.c b/sysdeps/unix/sysv/linux/libc_fatal.c index 46347bb20e..c0482d96f3 100644 --- a/sysdeps/unix/sysv/linux/libc_fatal.c +++ b/sysdeps/unix/sysv/linux/libc_fatal.c @@ -1,4 +1,4 @@ -/* Copyright (C) 1993-1995,1997,2000,2002, 2003 Free Software Foundation, Inc. +/* Copyright (C) 1993-1995,1997,2000,2002-2004 Free Software Foundation, Inc. This file is part of the GNU C Library. The GNU C Library is free software; you can redistribute it and/or @@ -16,45 +16,143 @@ Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. */ +#include +#include +#include +#include #include #include -#include -#include -#include #include -#include +#include +#include +#include #ifndef ABORT_INSTRUCTION /* No such instruction is available. */ # define ABORT_INSTRUCTION #endif +/* Abort with an error message. */ +#include + +#ifdef FATAL_PREPARE_INCLUDE +#include FATAL_PREPARE_INCLUDE +#endif + +struct str_list +{ + const char *str; + size_t len; + struct str_list *next; +}; + + /* Abort with an error message. */ void -__libc_fatal (message) - const char *message; +__libc_message (int do_abort, const char *fmt, ...) { - size_t len = strlen (message); + va_list ap; + va_list ap_copy; + int fd = -1; + + va_start (ap, fmt); + va_copy (ap_copy, ap); + +#ifdef FATAL_PREPARE + FATAL_PREPARE; +#endif - while (len > 0) + /* Open a descriptor for /dev/tty unless the user explicitly + requests errors on standard error. */ + const char *on_2 = __secure_getenv ("LIBC_FATAL_STDERR_"); + if (on_2 == NULL || *on_2 == '\0') + fd = open_not_cancel_2 (_PATH_TTY, O_RDWR | O_NOCTTY | O_NDELAY); + + if (fd == -1) + fd = STDERR_FILENO; + + struct str_list *list = NULL; + int nlist = 0; + + const char *cp = fmt; + while (*cp != '\0') { - INTERNAL_SYSCALL_DECL (err); - ssize_t count = INTERNAL_SYSCALL (write, err, 3, STDERR_FILENO, - message, len); - if (! INTERNAL_SYSCALL_ERROR_P (count, err)) + /* Find the next "%s" or the end of the string. */ + const char *next = cp; + while (next[0] != '%' || next[1] != 's') { - message += count; - len -= count; + next = __strchrnul (next + 1, '%'); + + if (next[0] == '\0') + break; } - else if (INTERNAL_SYSCALL_ERRNO (count, err) != EINTR) - break; + + /* Determine what to print. */ + const char *str; + size_t len; + if (cp[0] == '%' && cp[1] == 's') + { + str = va_arg (ap, const char *); + len = strlen (str); + cp += 2; + } + else + { + str = cp; + len = next - cp; + cp = next; + } + + struct str_list *newp = alloca (sizeof (struct str_list)); + newp->str = str; + newp->len = len; + newp->next = list; + list = newp; + ++nlist; } - /* Terminate the process. */ - _exit (127); + bool written = false; + if (nlist > 0) + { + struct iovec *iov = alloca (nlist * sizeof (struct iovec)); + ssize_t total = 0; + + for (int cnt = nlist - 1; cnt >= 0; --cnt) + { + iov[cnt].iov_base = (void *) list->str; + iov[cnt].iov_len = list->len; + total += list->len; + list = list->next; + } + + INTERNAL_SYSCALL_DECL (err); + ssize_t cnt; + do + cnt = INTERNAL_SYSCALL (writev, err, 3, fd, iov, nlist); + while (INTERNAL_SYSCALL_ERROR_P (cnt, err) + && INTERNAL_SYSCALL_ERRNO (cnt, err) == EINTR); - /* The previous call should never have returned. */ - while (1) - /* Try for ever and ever. */ - ABORT_INSTRUCTION; + if (cnt == total) + written = true; + } + + va_end (ap); + + /* If we had no success writing the message, use syslog. */ + if (! written) + vsyslog (LOG_ERR, fmt, ap_copy); + + va_end (ap_copy); + + if (do_abort) + /* Terminate the process. */ + abort (); +} + + +void +__libc_fatal (message) + const char *message; +{ + __libc_message (1, "%s", message); } libc_hidden_def (__libc_fatal) -- cgit v1.2.3