| /* Copyright (C) 1993-2018 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 <libintl.h> |
| #include <stdio.h> |
| #include <string.h> |
| #include <mach/error.h> |
| #include <errorlib.h> |
| #include <sys/param.h> |
| #include <_itoa.h> |
| |
| /* It is critical here that we always use the `dcgettext' function for |
| the message translation. Since <libintl.h> only defines the macro |
| `dgettext' to use `dcgettext' for optimizing programs this is not |
| always guaranteed. */ |
| #ifndef dgettext |
| # include <locale.h> /* We need LC_MESSAGES. */ |
| # define dgettext(domainname, msgid) dcgettext (domainname, msgid, LC_MESSAGES) |
| #endif |
| |
| /* Return a string describing the errno code in ERRNUM. */ |
| char * |
| __strerror_r (int errnum, char *buf, size_t buflen) |
| { |
| int system; |
| int sub; |
| int code; |
| const struct error_system *es; |
| extern void __mach_error_map_compat (int *); |
| |
| __mach_error_map_compat (&errnum); |
| |
| system = err_get_system (errnum); |
| sub = err_get_sub (errnum); |
| code = err_get_code (errnum); |
| |
| if (system > err_max_system || ! __mach_error_systems[system].bad_sub) |
| { |
| /* Buffer we use to print the number in. For a maximum size for |
| `int' of 8 bytes we never need more than 20 digits. */ |
| char numbuf[21]; |
| const char *unk = _("Error in unknown error system: "); |
| const size_t unklen = strlen (unk); |
| char *p, *q; |
| |
| numbuf[20] = '\0'; |
| p = _itoa_word (errnum, &numbuf[20], 16, 1); |
| |
| /* Now construct the result while taking care for the destination |
| buffer size. */ |
| q = __mempcpy (buf, unk, MIN (unklen, buflen)); |
| if (unklen < buflen) |
| memcpy (q, p, MIN (&numbuf[21] - p, buflen - unklen)); |
| |
| /* Terminate the string in any case. */ |
| if (buflen > 0) |
| buf[buflen - 1] = '\0'; |
| |
| return buf; |
| } |
| |
| es = &__mach_error_systems[system]; |
| |
| if (sub >= es->max_sub) |
| return (char *) es->bad_sub; |
| |
| if (code >= es->subsystem[sub].max_code) |
| { |
| /* Buffer we use to print the number in. For a maximum size for |
| `int' of 8 bytes we never need more than 20 digits. */ |
| char numbuf[21]; |
| const char *unk = _("Unknown error "); |
| const size_t unklen = strlen (unk); |
| char *p, *q; |
| size_t len = strlen (es->subsystem[sub].subsys_name); |
| |
| numbuf[20] = '\0'; |
| p = _itoa_word (errnum, &numbuf[20], 10, 0); |
| |
| /* Now construct the result while taking care for the destination |
| buffer size. */ |
| q = __mempcpy (buf, unk, MIN (unklen, buflen)); |
| if (unklen < buflen) |
| { |
| q = __mempcpy (q, es->subsystem[sub].subsys_name, |
| MIN (len, buflen - unklen)); |
| if (unklen + len < buflen) |
| { |
| *q++ = ' '; |
| if (unklen + len + 1 < buflen) |
| memcpy (q, p, |
| MIN (&numbuf[21] - p, buflen - unklen - len - 1)); |
| } |
| } |
| |
| /* Terminate the string in any case. */ |
| if (buflen > 0) |
| buf[buflen - 1] = '\0'; |
| |
| return buf; |
| } |
| |
| return (char *) _(es->subsystem[sub].codes[code]); |
| } |
| libc_hidden_def (__strerror_r) |
| weak_alias (__strerror_r, strerror_r) |