Blame lib/error.c

Packit 33f14e
/* Error handler for noninteractive utilities
Packit 33f14e
   Copyright (C) 1990-1998, 2000-2007, 2009-2017 Free Software Foundation, Inc.
Packit 33f14e
   This file is part of the GNU C Library.
Packit 33f14e
Packit 33f14e
   This program is free software: you can redistribute it and/or modify
Packit 33f14e
   it under the terms of the GNU General Public License as published by
Packit 33f14e
   the Free Software Foundation; either version 3 of the License, or
Packit 33f14e
   (at your option) any later version.
Packit 33f14e
Packit 33f14e
   This program is distributed in the hope that it will be useful,
Packit 33f14e
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 33f14e
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 33f14e
   GNU General Public License for more details.
Packit 33f14e
Packit 33f14e
   You should have received a copy of the GNU General Public License
Packit 33f14e
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit 33f14e
Packit 33f14e
/* Written by David MacKenzie <djm@gnu.ai.mit.edu>.  */
Packit 33f14e
Packit 33f14e
#if !_LIBC
Packit 33f14e
# include <config.h>
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
#include "error.h"
Packit 33f14e
Packit 33f14e
#include <stdarg.h>
Packit 33f14e
#include <stdio.h>
Packit 33f14e
#include <stdlib.h>
Packit 33f14e
#include <string.h>
Packit 33f14e
Packit 33f14e
#if !_LIBC && ENABLE_NLS
Packit 33f14e
# include "gettext.h"
Packit 33f14e
# define _(msgid) gettext (msgid)
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
#ifdef _LIBC
Packit 33f14e
# include <libintl.h>
Packit 33f14e
# include <stdbool.h>
Packit 33f14e
# include <stdint.h>
Packit 33f14e
# include <wchar.h>
Packit 33f14e
# define mbsrtowcs __mbsrtowcs
Packit 33f14e
# define USE_UNLOCKED_IO 0
Packit 33f14e
# define _GL_ATTRIBUTE_FORMAT_PRINTF(a, b)
Packit 33f14e
# define _GL_ARG_NONNULL(a)
Packit 33f14e
#else
Packit 33f14e
# include "getprogname.h"
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
#if USE_UNLOCKED_IO
Packit 33f14e
# include "unlocked-io.h"
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
#ifndef _
Packit 33f14e
# define _(String) String
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
/* If NULL, error will flush stdout, then print on stderr the program
Packit 33f14e
   name, a colon and a space.  Otherwise, error will call this
Packit 33f14e
   function without parameters instead.  */
Packit 33f14e
void (*error_print_progname) (void);
Packit 33f14e
Packit 33f14e
/* This variable is incremented each time 'error' is called.  */
Packit 33f14e
unsigned int error_message_count;
Packit 33f14e
Packit 33f14e
#ifdef _LIBC
Packit 33f14e
/* In the GNU C library, there is a predefined variable for this.  */
Packit 33f14e
Packit 33f14e
# define program_name program_invocation_name
Packit 33f14e
# include <errno.h>
Packit 33f14e
# include <limits.h>
Packit 33f14e
# include <libio/libioP.h>
Packit 33f14e
Packit 33f14e
/* In GNU libc we want do not want to use the common name 'error' directly.
Packit 33f14e
   Instead make it a weak alias.  */
Packit 33f14e
extern void __error (int status, int errnum, const char *message, ...)
Packit 33f14e
     __attribute__ ((__format__ (__printf__, 3, 4)));
Packit 33f14e
extern void __error_at_line (int status, int errnum, const char *file_name,
Packit 33f14e
                             unsigned int line_number, const char *message,
Packit 33f14e
                             ...)
Packit 33f14e
     __attribute__ ((__format__ (__printf__, 5, 6)));
Packit 33f14e
# define error __error
Packit 33f14e
# define error_at_line __error_at_line
Packit 33f14e
Packit 33f14e
# include <libio/iolibio.h>
Packit 33f14e
# define fflush(s) _IO_fflush (s)
Packit 33f14e
# undef putc
Packit 33f14e
# define putc(c, fp) _IO_putc (c, fp)
Packit 33f14e
Packit 33f14e
# include <bits/libc-lock.h>
Packit 33f14e
Packit 33f14e
#else /* not _LIBC */
Packit 33f14e
Packit 33f14e
# include <fcntl.h>
Packit 33f14e
# include <unistd.h>
Packit 33f14e
Packit 33f14e
# if (defined _WIN32 || defined __WIN32__) && ! defined __CYGWIN__
Packit 33f14e
/* Get declarations of the native Windows API functions.  */
Packit 33f14e
#  define WIN32_LEAN_AND_MEAN
Packit 33f14e
#  include <windows.h>
Packit 33f14e
/* Get _get_osfhandle.  */
Packit 33f14e
#  if GNULIB_MSVC_NOTHROW
Packit 33f14e
#   include "msvc-nothrow.h"
Packit 33f14e
#  else
Packit 33f14e
#   include <io.h>
Packit 33f14e
#  endif
Packit 33f14e
# endif
Packit 33f14e
Packit 33f14e
/* The gnulib override of fcntl is not needed in this file.  */
Packit 33f14e
# undef fcntl
Packit 33f14e
Packit 33f14e
# if !(GNULIB_STRERROR_R_POSIX || HAVE_DECL_STRERROR_R)
Packit 33f14e
#  ifndef HAVE_DECL_STRERROR_R
Packit 33f14e
"this configure-time declaration test was not run"
Packit 33f14e
#  endif
Packit 33f14e
#  if STRERROR_R_CHAR_P
Packit 33f14e
char *strerror_r (int errnum, char *buf, size_t buflen);
Packit 33f14e
#  else
Packit 33f14e
int strerror_r (int errnum, char *buf, size_t buflen);
Packit 33f14e
#  endif
Packit 33f14e
# endif
Packit 33f14e
Packit 33f14e
#define program_name getprogname ()
Packit 33f14e
Packit 33f14e
# if GNULIB_STRERROR_R_POSIX || HAVE_STRERROR_R || defined strerror_r
Packit 33f14e
#  define __strerror_r strerror_r
Packit 33f14e
# endif /* GNULIB_STRERROR_R_POSIX || HAVE_STRERROR_R || defined strerror_r */
Packit 33f14e
#endif  /* not _LIBC */
Packit 33f14e
Packit 33f14e
#if !_LIBC
Packit 33f14e
/* Return non-zero if FD is open.  */
Packit 33f14e
static int
Packit 33f14e
is_open (int fd)
Packit 33f14e
{
Packit 33f14e
# if (defined _WIN32 || defined __WIN32__) && ! defined __CYGWIN__
Packit 33f14e
  /* On native Windows: The initial state of unassigned standard file
Packit 33f14e
     descriptors is that they are open but point to an INVALID_HANDLE_VALUE.
Packit 33f14e
     There is no fcntl, and the gnulib replacement fcntl does not support
Packit 33f14e
     F_GETFL.  */
Packit 33f14e
  return (HANDLE) _get_osfhandle (fd) != INVALID_HANDLE_VALUE;
Packit 33f14e
# else
Packit 33f14e
#  ifndef F_GETFL
Packit 33f14e
#   error Please port fcntl to your platform
Packit 33f14e
#  endif
Packit 33f14e
  return 0 <= fcntl (fd, F_GETFL);
Packit 33f14e
# endif
Packit 33f14e
}
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
static void
Packit 33f14e
flush_stdout (void)
Packit 33f14e
{
Packit 33f14e
#if !_LIBC
Packit 33f14e
  int stdout_fd;
Packit 33f14e
Packit 33f14e
# if GNULIB_FREOPEN_SAFER
Packit 33f14e
  /* Use of gnulib's freopen-safer module normally ensures that
Packit 33f14e
       fileno (stdout) == 1
Packit 33f14e
     whenever stdout is open.  */
Packit 33f14e
  stdout_fd = STDOUT_FILENO;
Packit 33f14e
# else
Packit 33f14e
  /* POSIX states that fileno (stdout) after fclose is unspecified.  But in
Packit 33f14e
     practice it is not a problem, because stdout is statically allocated and
Packit 33f14e
     the fd of a FILE stream is stored as a field in its allocated memory.  */
Packit 33f14e
  stdout_fd = fileno (stdout);
Packit 33f14e
# endif
Packit 33f14e
  /* POSIX states that fflush (stdout) after fclose is unspecified; it
Packit 33f14e
     is safe in glibc, but not on all other platforms.  fflush (NULL)
Packit 33f14e
     is always defined, but too draconian.  */
Packit 33f14e
  if (0 <= stdout_fd && is_open (stdout_fd))
Packit 33f14e
#endif
Packit 33f14e
    fflush (stdout);
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
static void
Packit 33f14e
print_errno_message (int errnum)
Packit 33f14e
{
Packit 33f14e
  char const *s;
Packit 33f14e
Packit 33f14e
#if _LIBC || GNULIB_STRERROR_R_POSIX || defined HAVE_STRERROR_R
Packit 33f14e
  char errbuf[1024];
Packit 33f14e
# if _LIBC || (!GNULIB_STRERROR_R_POSIX && STRERROR_R_CHAR_P)
Packit 33f14e
  s = __strerror_r (errnum, errbuf, sizeof errbuf);
Packit 33f14e
# else
Packit 33f14e
  if (__strerror_r (errnum, errbuf, sizeof errbuf) == 0)
Packit 33f14e
    s = errbuf;
Packit 33f14e
  else
Packit 33f14e
    s = 0;
Packit 33f14e
# endif
Packit 33f14e
#else
Packit 33f14e
  s = strerror (errnum);
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
#if !_LIBC
Packit 33f14e
  if (! s)
Packit 33f14e
    s = _("Unknown system error");
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
#if _LIBC
Packit 33f14e
  __fxprintf (NULL, ": %s", s);
Packit 33f14e
#else
Packit 33f14e
  fprintf (stderr, ": %s", s);
Packit 33f14e
#endif
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
static void _GL_ATTRIBUTE_FORMAT_PRINTF (3, 0) _GL_ARG_NONNULL ((3))
Packit 33f14e
error_tail (int status, int errnum, const char *message, va_list args)
Packit 33f14e
{
Packit 33f14e
#if _LIBC
Packit 33f14e
  if (_IO_fwide (stderr, 0) > 0)
Packit 33f14e
    {
Packit 33f14e
      size_t len = strlen (message) + 1;
Packit 33f14e
      wchar_t *wmessage = NULL;
Packit 33f14e
      mbstate_t st;
Packit 33f14e
      size_t res;
Packit 33f14e
      const char *tmp;
Packit 33f14e
      bool use_malloc = false;
Packit 33f14e
Packit 33f14e
      while (1)
Packit 33f14e
        {
Packit 33f14e
          if (__libc_use_alloca (len * sizeof (wchar_t)))
Packit 33f14e
            wmessage = (wchar_t *) alloca (len * sizeof (wchar_t));
Packit 33f14e
          else
Packit 33f14e
            {
Packit 33f14e
              if (!use_malloc)
Packit 33f14e
                wmessage = NULL;
Packit 33f14e
Packit 33f14e
              wchar_t *p = (wchar_t *) realloc (wmessage,
Packit 33f14e
                                                len * sizeof (wchar_t));
Packit 33f14e
              if (p == NULL)
Packit 33f14e
                {
Packit 33f14e
                  free (wmessage);
Packit 33f14e
                  fputws_unlocked (L"out of memory\n", stderr);
Packit 33f14e
                  return;
Packit 33f14e
                }
Packit 33f14e
              wmessage = p;
Packit 33f14e
              use_malloc = true;
Packit 33f14e
            }
Packit 33f14e
Packit 33f14e
          memset (&st, '\0', sizeof (st));
Packit 33f14e
          tmp = message;
Packit 33f14e
Packit 33f14e
          res = mbsrtowcs (wmessage, &tmp, len, &st);
Packit 33f14e
          if (res != len)
Packit 33f14e
            break;
Packit 33f14e
Packit 33f14e
          if (__builtin_expect (len >= SIZE_MAX / sizeof (wchar_t) / 2, 0))
Packit 33f14e
            {
Packit 33f14e
              /* This really should not happen if everything is fine.  */
Packit 33f14e
              res = (size_t) -1;
Packit 33f14e
              break;
Packit 33f14e
            }
Packit 33f14e
Packit 33f14e
          len *= 2;
Packit 33f14e
        }
Packit 33f14e
Packit 33f14e
      if (res == (size_t) -1)
Packit 33f14e
        {
Packit 33f14e
          /* The string cannot be converted.  */
Packit 33f14e
          if (use_malloc)
Packit 33f14e
            {
Packit 33f14e
              free (wmessage);
Packit 33f14e
              use_malloc = false;
Packit 33f14e
            }
Packit 33f14e
          wmessage = (wchar_t *) L"???";
Packit 33f14e
        }
Packit 33f14e
Packit 33f14e
      __vfwprintf (stderr, wmessage, args);
Packit 33f14e
Packit 33f14e
      if (use_malloc)
Packit 33f14e
        free (wmessage);
Packit 33f14e
    }
Packit 33f14e
  else
Packit 33f14e
#endif
Packit 33f14e
    vfprintf (stderr, message, args);
Packit 33f14e
  va_end (args);
Packit 33f14e
Packit 33f14e
  ++error_message_count;
Packit 33f14e
  if (errnum)
Packit 33f14e
    print_errno_message (errnum);
Packit 33f14e
#if _LIBC
Packit 33f14e
  __fxprintf (NULL, "\n");
Packit 33f14e
#else
Packit 33f14e
  putc ('\n', stderr);
Packit 33f14e
#endif
Packit 33f14e
  fflush (stderr);
Packit 33f14e
  if (status)
Packit 33f14e
    exit (status);
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
Packit 33f14e
/* Print the program name and error message MESSAGE, which is a printf-style
Packit 33f14e
   format string with optional args.
Packit 33f14e
   If ERRNUM is nonzero, print its corresponding system error message.
Packit 33f14e
   Exit with status STATUS if it is nonzero.  */
Packit 33f14e
void
Packit 33f14e
error (int status, int errnum, const char *message, ...)
Packit 33f14e
{
Packit 33f14e
  va_list args;
Packit 33f14e
Packit 33f14e
#if defined _LIBC && defined __libc_ptf_call
Packit 33f14e
  /* We do not want this call to be cut short by a thread
Packit 33f14e
     cancellation.  Therefore disable cancellation for now.  */
Packit 33f14e
  int state = PTHREAD_CANCEL_ENABLE;
Packit 33f14e
  __libc_ptf_call (pthread_setcancelstate, (PTHREAD_CANCEL_DISABLE, &state),
Packit 33f14e
                   0);
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
  flush_stdout ();
Packit 33f14e
#ifdef _LIBC
Packit 33f14e
  _IO_flockfile (stderr);
Packit 33f14e
#endif
Packit 33f14e
  if (error_print_progname)
Packit 33f14e
    (*error_print_progname) ();
Packit 33f14e
  else
Packit 33f14e
    {
Packit 33f14e
#if _LIBC
Packit 33f14e
      __fxprintf (NULL, "%s: ", program_name);
Packit 33f14e
#else
Packit 33f14e
      fprintf (stderr, "%s: ", program_name);
Packit 33f14e
#endif
Packit 33f14e
    }
Packit 33f14e
Packit 33f14e
  va_start (args, message);
Packit 33f14e
  error_tail (status, errnum, message, args);
Packit 33f14e
Packit 33f14e
#ifdef _LIBC
Packit 33f14e
  _IO_funlockfile (stderr);
Packit 33f14e
# ifdef __libc_ptf_call
Packit 33f14e
  __libc_ptf_call (pthread_setcancelstate, (state, NULL), 0);
Packit 33f14e
# endif
Packit 33f14e
#endif
Packit 33f14e
}
Packit 33f14e

Packit 33f14e
/* Sometimes we want to have at most one error per line.  This
Packit 33f14e
   variable controls whether this mode is selected or not.  */
Packit 33f14e
int error_one_per_line;
Packit 33f14e
Packit 33f14e
void
Packit 33f14e
error_at_line (int status, int errnum, const char *file_name,
Packit 33f14e
               unsigned int line_number, const char *message, ...)
Packit 33f14e
{
Packit 33f14e
  va_list args;
Packit 33f14e
Packit 33f14e
  if (error_one_per_line)
Packit 33f14e
    {
Packit 33f14e
      static const char *old_file_name;
Packit 33f14e
      static unsigned int old_line_number;
Packit 33f14e
Packit 33f14e
      if (old_line_number == line_number
Packit 33f14e
          && (file_name == old_file_name
Packit 33f14e
              || (old_file_name != NULL
Packit 33f14e
                  && file_name != NULL
Packit 33f14e
                  && strcmp (old_file_name, file_name) == 0)))
Packit 33f14e
Packit 33f14e
        /* Simply return and print nothing.  */
Packit 33f14e
        return;
Packit 33f14e
Packit 33f14e
      old_file_name = file_name;
Packit 33f14e
      old_line_number = line_number;
Packit 33f14e
    }
Packit 33f14e
Packit 33f14e
#if defined _LIBC && defined __libc_ptf_call
Packit 33f14e
  /* We do not want this call to be cut short by a thread
Packit 33f14e
     cancellation.  Therefore disable cancellation for now.  */
Packit 33f14e
  int state = PTHREAD_CANCEL_ENABLE;
Packit 33f14e
  __libc_ptf_call (pthread_setcancelstate, (PTHREAD_CANCEL_DISABLE, &state),
Packit 33f14e
                   0);
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
  flush_stdout ();
Packit 33f14e
#ifdef _LIBC
Packit 33f14e
  _IO_flockfile (stderr);
Packit 33f14e
#endif
Packit 33f14e
  if (error_print_progname)
Packit 33f14e
    (*error_print_progname) ();
Packit 33f14e
  else
Packit 33f14e
    {
Packit 33f14e
#if _LIBC
Packit 33f14e
      __fxprintf (NULL, "%s:", program_name);
Packit 33f14e
#else
Packit 33f14e
      fprintf (stderr, "%s:", program_name);
Packit 33f14e
#endif
Packit 33f14e
    }
Packit 33f14e
Packit 33f14e
#if _LIBC
Packit 33f14e
  __fxprintf (NULL, file_name != NULL ? "%s:%u: " : " ",
Packit 33f14e
              file_name, line_number);
Packit 33f14e
#else
Packit 33f14e
  fprintf (stderr, file_name != NULL ? "%s:%u: " : " ",
Packit 33f14e
           file_name, line_number);
Packit 33f14e
#endif
Packit 33f14e
Packit 33f14e
  va_start (args, message);
Packit 33f14e
  error_tail (status, errnum, message, args);
Packit 33f14e
Packit 33f14e
#ifdef _LIBC
Packit 33f14e
  _IO_funlockfile (stderr);
Packit 33f14e
# ifdef __libc_ptf_call
Packit 33f14e
  __libc_ptf_call (pthread_setcancelstate, (state, NULL), 0);
Packit 33f14e
# endif
Packit 33f14e
#endif
Packit 33f14e
}
Packit 33f14e
Packit 33f14e
#ifdef _LIBC
Packit 33f14e
/* Make the weak alias.  */
Packit 33f14e
# undef error
Packit 33f14e
# undef error_at_line
Packit 33f14e
weak_alias (__error, error)
Packit 33f14e
weak_alias (__error_at_line, error_at_line)
Packit 33f14e
#endif