blob: 49ddced508237ff0043b6d8380f01b3a20fb8aa3 [file] [log] [blame]
Denis Vlasenkoed836cd2006-11-25 14:44:13 +00001/* vi: set sw=4 ts=4: */
2/*
3 * ascii-to-numbers implementations for busybox
4 *
5 * Copyright (C) 2003 Manuel Novoa III <mjn3@codepoet.org>
6 *
7 * Licensed under GPLv2, see file LICENSE in this tarball for details.
8 */
9
10/* Provides extern declarations of functions */
11#define DECLARE_STR_CONV(type, T, UT) \
12\
13unsigned type xstrto##UT##_range_sfx(const char *str, int b, unsigned type l, unsigned type u, const struct suffix_mult *sfx); \
14unsigned type xstrto##UT##_range(const char *str, int b, unsigned type l, unsigned type u); \
15unsigned type xstrto##UT##_sfx(const char *str, int b, const struct suffix_mult *sfx); \
16unsigned type xstrto##UT(const char *str, int b); \
17unsigned type xato##UT##_range_sfx(const char *str, unsigned type l, unsigned type u, const struct suffix_mult *sfx); \
18unsigned type xato##UT##_range(const char *str, unsigned type l, unsigned type u); \
19unsigned type xato##UT##_sfx(const char *str, const struct suffix_mult *sfx); \
20unsigned type xato##UT(const char *str); \
Denis Vlasenkob71c6682007-07-21 15:08:09 +000021type xstrto##T##_range_sfx(const char *str, int b, type l, type u, const struct suffix_mult *sfx); \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000022type xstrto##T##_range(const char *str, int b, type l, type u); \
23type xato##T##_range_sfx(const char *str, type l, type u, const struct suffix_mult *sfx); \
24type xato##T##_range(const char *str, type l, type u); \
25type xato##T##_sfx(const char *str, const struct suffix_mult *sfx); \
26type xato##T(const char *str); \
27
28/* Unsigned long long functions always exist */
29DECLARE_STR_CONV(long long, ll, ull)
30
31
Denis Vlasenkoa597aad2006-12-16 23:48:13 +000032/* Provides inline definitions of functions */
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000033/* (useful for mapping them to the type of the same width) */
34#define DEFINE_EQUIV_STR_CONV(narrow, N, W, UN, UW) \
35\
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000036static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000037unsigned narrow xstrto##UN##_range_sfx(const char *str, int b, unsigned narrow l, unsigned narrow u, const struct suffix_mult *sfx) \
38{ return xstrto##UW##_range_sfx(str, b, l, u, sfx); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000039static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000040unsigned narrow xstrto##UN##_range(const char *str, int b, unsigned narrow l, unsigned narrow u) \
41{ return xstrto##UW##_range(str, b, l, u); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000042static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000043unsigned narrow xstrto##UN##_sfx(const char *str, int b, const struct suffix_mult *sfx) \
44{ return xstrto##UW##_sfx(str, b, sfx); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000045static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000046unsigned narrow xstrto##UN(const char *str, int b) \
47{ return xstrto##UW(str, b); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000048static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000049unsigned narrow xato##UN##_range_sfx(const char *str, unsigned narrow l, unsigned narrow u, const struct suffix_mult *sfx) \
50{ return xato##UW##_range_sfx(str, l, u, sfx); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000051static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000052unsigned narrow xato##UN##_range(const char *str, unsigned narrow l, unsigned narrow u) \
53{ return xato##UW##_range(str, l, u); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000054static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000055unsigned narrow xato##UN##_sfx(const char *str, const struct suffix_mult *sfx) \
56{ return xato##UW##_sfx(str, sfx); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000057static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000058unsigned narrow xato##UN(const char *str) \
59{ return xato##UW(str); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000060static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000061narrow xstrto##N##_range_sfx(const char *str, int b, narrow l, narrow u, const struct suffix_mult *sfx) \
62{ return xstrto##W##_range_sfx(str, b, l, u, sfx); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000063static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000064narrow xstrto##N##_range(const char *str, int b, narrow l, narrow u) \
65{ return xstrto##W##_range(str, b, l, u); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000066static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000067narrow xato##N##_range_sfx(const char *str, narrow l, narrow u, const struct suffix_mult *sfx) \
68{ return xato##W##_range_sfx(str, l, u, sfx); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000069static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000070narrow xato##N##_range(const char *str, narrow l, narrow u) \
71{ return xato##W##_range(str, l, u); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000072static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000073narrow xato##N##_sfx(const char *str, const struct suffix_mult *sfx) \
74{ return xato##W##_sfx(str, sfx); } \
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000075static ALWAYS_INLINE \
Denis Vlasenkoed836cd2006-11-25 14:44:13 +000076narrow xato##N(const char *str) \
77{ return xato##W(str); } \
78
79/* If long == long long, then just map them one-to-one */
80#if ULONG_MAX == ULLONG_MAX
81DEFINE_EQUIV_STR_CONV(long, l, ll, ul, ull)
82#else
83/* Else provide extern defs */
84DECLARE_STR_CONV(long, l, ul)
85#endif
86
87/* Same for int -> [long] long */
88#if UINT_MAX == ULLONG_MAX
89DEFINE_EQUIV_STR_CONV(int, i, ll, u, ull)
90#elif UINT_MAX == ULONG_MAX
91DEFINE_EQUIV_STR_CONV(int, i, l, u, ul)
92#else
93DECLARE_STR_CONV(int, i, u)
94#endif
Denis Vlasenko43bddf32006-11-25 14:49:04 +000095
96/* Specialized */
97
98int BUG_xatou32_unimplemented(void);
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000099static ALWAYS_INLINE uint32_t xatou32(const char *numstr)
Denis Vlasenko43bddf32006-11-25 14:49:04 +0000100{
101 if (UINT_MAX == 0xffffffff)
102 return xatou(numstr);
103 if (ULONG_MAX == 0xffffffff)
104 return xatoul(numstr);
105 return BUG_xatou32_unimplemented();
106}
Denis Vlasenkod686a042006-11-27 14:43:21 +0000107
Denis Vlasenkof74194e2007-10-18 12:54:39 +0000108/* Non-aborting kind of convertors: bb_strto[u][l]l */
109
110/* On exit: errno = 0 only if there was non-empty, '\0' terminated value
111 * errno = EINVAL if value was not '\0' terminated, but othervise ok
112 * Return value is still valid, caller should just check whether end[0]
113 * is a valid terminating char for particular case. OTOH, if caller
114 * requires '\0' terminated input, [s]he can just check errno == 0.
115 * errno = ERANGE if value had alphanumeric terminating char ("1234abcg").
116 * errno = ERANGE if value is out of range, missing, etc.
117 * errno = ERANGE if value had minus sign for strtouXX (even "-0" is not ok )
118 * return value is all-ones in this case.
119 */
Denis Vlasenkod686a042006-11-27 14:43:21 +0000120
121unsigned long long bb_strtoull(const char *arg, char **endp, int base);
122long long bb_strtoll(const char *arg, char **endp, int base);
123
124#if ULONG_MAX == ULLONG_MAX
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +0000125static ALWAYS_INLINE
Denis Vlasenkod686a042006-11-27 14:43:21 +0000126unsigned long bb_strtoul(const char *arg, char **endp, int base)
127{ return bb_strtoull(arg, endp, base); }
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +0000128static ALWAYS_INLINE
Denis Vlasenkoa597aad2006-12-16 23:48:13 +0000129long bb_strtol(const char *arg, char **endp, int base)
Denis Vlasenkod686a042006-11-27 14:43:21 +0000130{ return bb_strtoll(arg, endp, base); }
131#else
132unsigned long bb_strtoul(const char *arg, char **endp, int base);
133long bb_strtol(const char *arg, char **endp, int base);
134#endif
135
136#if UINT_MAX == ULLONG_MAX
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +0000137static ALWAYS_INLINE
Denis Vlasenkoa597aad2006-12-16 23:48:13 +0000138unsigned bb_strtou(const char *arg, char **endp, int base)
Denis Vlasenkod686a042006-11-27 14:43:21 +0000139{ return bb_strtoull(arg, endp, base); }
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +0000140static ALWAYS_INLINE
Denis Vlasenkoa597aad2006-12-16 23:48:13 +0000141int bb_strtoi(const char *arg, char **endp, int base)
Denis Vlasenkod686a042006-11-27 14:43:21 +0000142{ return bb_strtoll(arg, endp, base); }
143#elif UINT_MAX == ULONG_MAX
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +0000144static ALWAYS_INLINE
Denis Vlasenkoa597aad2006-12-16 23:48:13 +0000145unsigned bb_strtou(const char *arg, char **endp, int base)
Denis Vlasenkod686a042006-11-27 14:43:21 +0000146{ return bb_strtoul(arg, endp, base); }
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +0000147static ALWAYS_INLINE
Denis Vlasenkoa597aad2006-12-16 23:48:13 +0000148int bb_strtoi(const char *arg, char **endp, int base)
Denis Vlasenkod686a042006-11-27 14:43:21 +0000149{ return bb_strtol(arg, endp, base); }
150#else
Denis Vlasenkoa597aad2006-12-16 23:48:13 +0000151unsigned bb_strtou(const char *arg, char **endp, int base);
152int bb_strtoi(const char *arg, char **endp, int base);
Denis Vlasenkod686a042006-11-27 14:43:21 +0000153#endif
154
155int BUG_bb_strtou32_unimplemented(void);
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +0000156static ALWAYS_INLINE
Denis Vlasenkod686a042006-11-27 14:43:21 +0000157uint32_t bb_strtou32(const char *arg, char **endp, int base)
158{
159 if (sizeof(uint32_t) == sizeof(unsigned))
160 return bb_strtou(arg, endp, base);
161 if (sizeof(uint32_t) == sizeof(unsigned long))
162 return bb_strtoul(arg, endp, base);
163 return BUG_bb_strtou32_unimplemented();
164}
165
166/* Floating point */
167
168/* double bb_strtod(const char *arg, char **endp); */