| /* 4.4BSD utility functions for error messages. | 
 |    Copyright (C) 1995-2016 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 | 
 |    modify it under the terms of the GNU Lesser General Public | 
 |    License as published by the Free Software Foundation; either | 
 |    version 2.1 of the License, or (at your option) any later version. | 
 |  | 
 |    The GNU C Library is distributed in the hope that it will be useful, | 
 |    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
 |    Lesser General Public License for more details. | 
 |  | 
 |    You should have received a copy of the GNU Lesser General Public | 
 |    License along with the GNU C Library; if not, see | 
 |    <http://www.gnu.org/licenses/>.  */ | 
 |  | 
 | #include <stdarg.h> | 
 | #include <err.h> | 
 | #include <stdlib.h> | 
 | #include <errno.h> | 
 | #include <string.h> | 
 | #include <stdio.h> | 
 |  | 
 | #include <wchar.h> | 
 | #define flockfile(s) _IO_flockfile (s) | 
 | #define funlockfile(s) _IO_funlockfile (s) | 
 |  | 
 | extern char *__progname; | 
 |  | 
 | #define VA(call)							      \ | 
 | {									      \ | 
 |   va_list ap;								      \ | 
 |   va_start (ap, format);						      \ | 
 |   call;									      \ | 
 |   va_end (ap);								      \ | 
 | } | 
 |  | 
 | static void | 
 | convert_and_print (const char *format, __gnuc_va_list ap) | 
 | { | 
 | #define ALLOCA_LIMIT	2000 | 
 |   size_t len; | 
 |   wchar_t *wformat = NULL; | 
 |   mbstate_t st; | 
 |   size_t res; | 
 |   const char *tmp; | 
 |  | 
 |   if (format == NULL) | 
 |     return; | 
 |  | 
 |   len = strlen (format) + 1; | 
 |  | 
 |   do | 
 |     { | 
 |       if (len < ALLOCA_LIMIT) | 
 | 	wformat = (wchar_t *) alloca (len * sizeof (wchar_t)); | 
 |       else | 
 | 	{ | 
 | 	  if (wformat != NULL && len / 2 < ALLOCA_LIMIT) | 
 | 	    wformat = NULL; | 
 |  | 
 | 	  wformat = (wchar_t *) realloc (wformat, len * sizeof (wchar_t)); | 
 |  | 
 | 	  if (wformat == NULL) | 
 | 	    { | 
 | 	      fputws_unlocked (L"out of memory\n", stderr); | 
 | 	      return; | 
 | 	    } | 
 | 	} | 
 |  | 
 |       memset (&st, '\0', sizeof (st)); | 
 |       tmp =format; | 
 |     } | 
 |   while ((res = __mbsrtowcs (wformat, &tmp, len, &st)) == len); | 
 |  | 
 |   if (res == (size_t) -1) | 
 |     /* The string cannot be converted.  */ | 
 |     wformat = (wchar_t *) L"???"; | 
 |  | 
 |   __vfwprintf (stderr, wformat, ap); | 
 | } | 
 |  | 
 | void | 
 | vwarnx (const char *format, __gnuc_va_list ap) | 
 | { | 
 |   flockfile (stderr); | 
 |   if (_IO_fwide (stderr, 0) > 0) | 
 |     { | 
 |       __fwprintf (stderr, L"%s: ", __progname); | 
 |       convert_and_print (format, ap); | 
 |       putwc_unlocked (L'\n', stderr); | 
 |     } | 
 |   else | 
 |     { | 
 |       fprintf (stderr, "%s: ", __progname); | 
 |       if (format) | 
 | 	vfprintf (stderr, format, ap); | 
 |       putc_unlocked ('\n', stderr); | 
 |     } | 
 |   funlockfile (stderr); | 
 | } | 
 | libc_hidden_def (vwarnx) | 
 |  | 
 | void | 
 | vwarn (const char *format, __gnuc_va_list ap) | 
 | { | 
 |   int error = errno; | 
 |  | 
 |   flockfile (stderr); | 
 |   if (_IO_fwide (stderr, 0) > 0) | 
 |     { | 
 |       __fwprintf (stderr, L"%s: ", __progname); | 
 |       if (format) | 
 | 	{ | 
 | 	  convert_and_print (format, ap); | 
 | 	  fputws_unlocked (L": ", stderr); | 
 | 	} | 
 |       __set_errno (error); | 
 |       __fwprintf (stderr, L"%m\n"); | 
 |     } | 
 |   else | 
 |     { | 
 |       fprintf (stderr, "%s: ", __progname); | 
 |       if (format) | 
 | 	{ | 
 | 	  vfprintf (stderr, format, ap); | 
 | 	  fputs_unlocked (": ", stderr); | 
 | 	} | 
 |       __set_errno (error); | 
 |       fprintf (stderr, "%m\n"); | 
 |     } | 
 |   funlockfile (stderr); | 
 | } | 
 | libc_hidden_def (vwarn) | 
 |  | 
 |  | 
 | void | 
 | warn (const char *format, ...) | 
 | { | 
 |   VA (vwarn (format, ap)) | 
 | } | 
 | libc_hidden_def (warn) | 
 |  | 
 | void | 
 | warnx (const char *format, ...) | 
 | { | 
 |   VA (vwarnx (format, ap)) | 
 | } | 
 | libc_hidden_def (warnx) | 
 |  | 
 | void | 
 | verr (int status, const char *format, __gnuc_va_list ap) | 
 | { | 
 |   vwarn (format, ap); | 
 |   exit (status); | 
 | } | 
 | libc_hidden_def (verr) | 
 |  | 
 | void | 
 | verrx (int status, const char *format, __gnuc_va_list ap) | 
 | { | 
 |   vwarnx (format, ap); | 
 |   exit (status); | 
 | } | 
 | libc_hidden_def (verrx) | 
 |  | 
 | void | 
 | err (int status, const char *format, ...) | 
 | { | 
 |   VA (verr (status, format, ap)) | 
 | } | 
 |  | 
 | void | 
 | errx (int status, const char *format, ...) | 
 | { | 
 |   VA (verrx (status, format, ap)) | 
 | } |