| /* vi: set sw=4 ts=4: */ |
| /* |
| * Utility routines. |
| * |
| * Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org> |
| * |
| * Licensed under GPLv2 or later, see file LICENSE in this tarball for details. |
| */ |
| |
| #include <assert.h> |
| #include "libbb.h" |
| |
| int safe_strtod(const char *arg, double* value) |
| { |
| char *endptr; |
| int errno_save = errno; |
| |
| assert(arg!=NULL); |
| errno = 0; |
| *value = strtod(arg, &endptr); |
| if (errno != 0 || *endptr != '\0' || endptr == arg) { |
| return 1; |
| } |
| errno = errno_save; |
| return 0; |
| } |
| |
| int safe_strtoull(const char *arg, unsigned long long* value) |
| { |
| char *endptr; |
| int errno_save = errno; |
| |
| assert(arg!=NULL); |
| if (!isdigit(arg[0])) /* strtouXX takes minus signs w/o error! :( */ |
| return 1; |
| errno = 0; |
| *value = strtoull(arg, &endptr, 0); |
| if (errno != 0 || *endptr != '\0' || endptr == arg) { |
| return 1; |
| } |
| errno = errno_save; |
| return 0; |
| } |
| |
| int safe_strtoll(const char *arg, long long* value) |
| { |
| char *endptr; |
| int errno_save = errno; |
| |
| assert(arg!=NULL); |
| errno = 0; |
| *value = strtoll(arg, &endptr, 0); |
| if (errno != 0 || *endptr != '\0' || endptr == arg) { |
| return 1; |
| } |
| errno = errno_save; |
| return 0; |
| } |
| |
| int safe_strtoul(const char *arg, unsigned long* value) |
| { |
| char *endptr; |
| int errno_save = errno; |
| |
| assert(arg!=NULL); |
| if (!isdigit(arg[0])) /* strtouXX takes minus signs w/o error! :( */ |
| return 1; |
| errno = 0; |
| *value = strtoul(arg, &endptr, 0); |
| if (errno != 0 || *endptr != '\0' || endptr == arg) { |
| return 1; |
| } |
| errno = errno_save; |
| return 0; |
| } |
| |
| int safe_strtol(const char *arg, long* value) |
| { |
| char *endptr; |
| int errno_save = errno; |
| |
| assert(arg!=NULL); |
| errno = 0; |
| *value = strtol(arg, &endptr, 0); |
| if (errno != 0 || *endptr != '\0' || endptr == arg) { |
| return 1; |
| } |
| errno = errno_save; |
| return 0; |
| } |
| |
| /* TODO: This is what uclibc is doing. Try to do the same? */ |
| |
| #if 0 |
| #if defined __HAVE_ELF__ |
| |
| # define strong_alias(name, aliasname) _strong_alias(name, aliasname) |
| # define _strong_alias(name, aliasname) \ |
| extern __typeof (name) aliasname __attribute__ ((alias (#name))); |
| |
| #else /* !defined __HAVE_ELF__ */ |
| |
| # define strong_alias(name, aliasname) _strong_alias (name, aliasname) |
| # define _strong_alias(name, aliasname) \ |
| __asm__(".global " __C_SYMBOL_PREFIX__ #aliasname "\n" \ |
| ".set " __C_SYMBOL_PREFIX__ #aliasname "," __C_SYMBOL_PREFIX__ #name); |
| |
| #endif |
| #endif |
| |
| int safe_strtoi(const char *arg, int* value) |
| { |
| if (sizeof(long) == sizeof(int)) { |
| return safe_strtol(arg, (long*)value); |
| } else { |
| int error; |
| long lvalue = *value; |
| error = safe_strtol(arg, &lvalue); |
| if (lvalue < INT_MIN || lvalue > INT_MAX) |
| return 1; |
| *value = (int) lvalue; |
| return error; |
| } |
| } |
| |
| int safe_strtou(const char *arg, unsigned* value) |
| { |
| if (sizeof(unsigned long) == sizeof(unsigned)) { |
| return safe_strtoul(arg, (unsigned long*)value); |
| } else { |
| int error; |
| unsigned long lvalue = *value; |
| error = safe_strtoul(arg, &lvalue); |
| if (lvalue > UINT_MAX) |
| return 1; |
| *value = (unsigned) lvalue; |
| return error; |
| } |
| } |
| |
| int BUG_safe_strtou32_unimplemented(void); |
| int safe_strtou32(const char *arg, uint32_t* value) |
| { |
| if (sizeof(uint32_t) == sizeof(unsigned)) |
| return safe_strtou(arg, (unsigned*)value); |
| if (sizeof(uint32_t) == sizeof(unsigned long)) |
| return safe_strtoul(arg, (unsigned long*)value); |
| return BUG_safe_strtou32_unimplemented(); |
| } |