/* $OpenBSD: strerror_r.c,v 1.6 2005/08/08 08:05:37 espie Exp $ */ /* Public Domain */ #define sys_siglist _sys_siglist #include #include #include #include typedef struct { int code; const char* msg; } CodeString; static const char* __code_string_lookup( const CodeString* strings, int code ) { int nn = 0; for (;;) { if (strings[nn].msg == NULL) break; if (strings[nn].code == code) return strings[nn].msg; nn++; } return NULL; } static const CodeString _sys_error_strings[] = { #define __BIONIC_ERRDEF(x,y,z) { y, z }, #include { 0, NULL } }; static size_t __digits10(unsigned int num) { size_t i = 0; do { num /= 10; i++; } while (num != 0); return i; } static int __itoa(int num, int sign, char *buffer, size_t start, size_t end) { size_t pos; unsigned int a; int neg; if (sign && num < 0) { a = -num; neg = 1; } else { a = num; neg = 0; } pos = start + __digits10(a); if (neg) pos++; if (pos < end) buffer[pos] = '\0'; else return ERANGE; pos--; do { buffer[pos] = (a % 10) + '0'; pos--; a /= 10; } while (a != 0); if (neg) buffer[pos] = '-'; return 0; } int strerror_r(int errnum, char *strerrbuf, size_t buflen) { int save_errno; int len, ret = 0; const char* msg; save_errno = errno; msg = __code_string_lookup( _sys_error_strings, errnum ); if (msg != NULL) { len = strlcpy(strerrbuf, msg, buflen); if ((size_t)len >= buflen) ret = ERANGE; } else { len = strlcpy(strerrbuf, "Unknown error: ", buflen); if ((size_t)len >= buflen) ret = ERANGE; else { int ret = __itoa(errnum, 1, strerrbuf, len, buflen); if (ret == 0) ret = EINVAL; } } return ret; } #if 0 static const CodeString _sys_signal_strings[] = { #define SIGDEF(x,y,z) { y, z }, #include }; static int __num2string(int num, int sign, int setid, char *buf, size_t buflen, char * list[], size_t max, const char *def) { int ret = 0; size_t len; if (0 <= num && num < max) { len = strlcpy(buf, def, buflen); if (len >= buflen) ret = ERANGE; } else { len = strlcpy(buf, def, buflen); if (len >= buflen) ret = ERANGE; else { ret = __itoa(num, sign, buf, len, buflen); if (ret == 0) ret = EINVAL; } } return ret; } #define USIGPREFIX "Unknown signal: " char * __strsignal(int num, char *buf) { __num2string(num, 0, 2, buf, NL_TEXTMAX, (char **)sys_siglist, NSIG, USIGPREFIX); return buf; } #endif