mirror of
https://git.busybox.net/busybox.git
synced 2024-11-27 23:53:29 +08:00
150 lines
3.2 KiB
C
150 lines
3.2 KiB
C
/* 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();
|
|
}
|