xref: /netbsd-src/external/gpl3/binutils.old/dist/libiberty/xstrerror.c (revision 16dce51364ebe8aeafbae46bc5aa167b8115bc45)
1*16dce513Schristos /* xstrerror.c -- jacket routine for more robust strerror() usage.
2*16dce513Schristos    Fri Jun 16 18:30:00 1995  Pat Rankin  <rankin@eql.caltech.edu>
3*16dce513Schristos    This code is in the public domain.  */
4*16dce513Schristos 
5*16dce513Schristos /*
6*16dce513Schristos 
7*16dce513Schristos @deftypefn Replacement char* xstrerror (int @var{errnum})
8*16dce513Schristos 
9*16dce513Schristos Behaves exactly like the standard @code{strerror} function, but
10*16dce513Schristos will never return a @code{NULL} pointer.
11*16dce513Schristos 
12*16dce513Schristos @end deftypefn
13*16dce513Schristos 
14*16dce513Schristos */
15*16dce513Schristos 
16*16dce513Schristos #include <stdio.h>
17*16dce513Schristos 
18*16dce513Schristos #include "config.h"
19*16dce513Schristos #include "libiberty.h"
20*16dce513Schristos 
21*16dce513Schristos #ifdef VMS
22*16dce513Schristos #  include <errno.h>
23*16dce513Schristos #  if !defined (__STRICT_ANSI__) && !defined (__HIDE_FORBIDDEN_NAMES)
24*16dce513Schristos #    ifdef __cplusplus
25*16dce513Schristos extern "C" {
26*16dce513Schristos #    endif /* __cplusplus */
27*16dce513Schristos extern char *strerror (int,...);
28*16dce513Schristos #    define DONT_DECLARE_STRERROR
29*16dce513Schristos #    ifdef __cplusplus
30*16dce513Schristos }
31*16dce513Schristos #    endif /* __cplusplus */
32*16dce513Schristos #  endif
33*16dce513Schristos #endif  /* VMS */
34*16dce513Schristos 
35*16dce513Schristos 
36*16dce513Schristos #ifndef DONT_DECLARE_STRERROR
37*16dce513Schristos #  ifdef __cplusplus
38*16dce513Schristos extern "C" {
39*16dce513Schristos #  endif /* __cplusplus */
40*16dce513Schristos extern char *strerror (int);
41*16dce513Schristos #  ifdef __cplusplus
42*16dce513Schristos }
43*16dce513Schristos #  endif /* __cplusplus */
44*16dce513Schristos #endif
45*16dce513Schristos 
46*16dce513Schristos /* If strerror returns NULL, we'll format the number into a static buffer.  */
47*16dce513Schristos 
48*16dce513Schristos #define ERRSTR_FMT "undocumented error #%d"
49*16dce513Schristos static char xstrerror_buf[sizeof ERRSTR_FMT + 20];
50*16dce513Schristos 
51*16dce513Schristos /* Like strerror, but result is never a null pointer.  */
52*16dce513Schristos 
53*16dce513Schristos char *
xstrerror(int errnum)54*16dce513Schristos xstrerror (int errnum)
55*16dce513Schristos {
56*16dce513Schristos   char *errstr;
57*16dce513Schristos #ifdef VMS
58*16dce513Schristos   char *(*vmslib_strerror) (int,...);
59*16dce513Schristos 
60*16dce513Schristos   /* Override any possibly-conflicting declaration from system header.  */
61*16dce513Schristos   vmslib_strerror = (char *(*) (int,...)) strerror;
62*16dce513Schristos   /* Second argument matters iff first is EVMSERR, but it's simpler to
63*16dce513Schristos      pass it unconditionally.  `vaxc$errno' is declared in <errno.h>
64*16dce513Schristos      and maintained by the run-time library in parallel to `errno'.
65*16dce513Schristos      We assume that `errnum' corresponds to the last value assigned to
66*16dce513Schristos      errno by the run-time library, hence vaxc$errno will be relevant.  */
67*16dce513Schristos   errstr = (*vmslib_strerror) (errnum, vaxc$errno);
68*16dce513Schristos #else
69*16dce513Schristos   errstr = strerror (errnum);
70*16dce513Schristos #endif
71*16dce513Schristos 
72*16dce513Schristos   /* If `errnum' is out of range, result might be NULL.  We'll fix that.  */
73*16dce513Schristos   if (!errstr)
74*16dce513Schristos     {
75*16dce513Schristos       sprintf (xstrerror_buf, ERRSTR_FMT, errnum);
76*16dce513Schristos       errstr = xstrerror_buf;
77*16dce513Schristos     }
78*16dce513Schristos   return errstr;
79*16dce513Schristos }
80