]>
Commit | Line | Data |
---|---|---|
e4c5383e SS |
1 | /* |
2 | * linux/lib/vsprintf.c | |
3 | * | |
4 | * Copyright (C) 1991, 1992 Linus Torvalds | |
5 | */ | |
6 | ||
7 | /* vsprintf.c -- Lars Wirzenius & Linus Torvalds. */ | |
8 | /* | |
9 | * Wirzenius wrote this portably, Torvalds fucked it up :-) | |
10 | */ | |
11 | ||
e4c5383e | 12 | #include <errno.h> |
3e96ed44 | 13 | #include <malloc.h> |
467382ca | 14 | #include <vsprintf.h> |
e4c5383e SS |
15 | #include <linux/ctype.h> |
16 | ||
2e794614 | 17 | /* from lib/kstrtox.c */ |
e6951139 | 18 | static const char *_parse_integer_fixup_radix(const char *s, uint *basep) |
2e794614 | 19 | { |
e6951139 SG |
20 | /* Look for a 0x prefix */ |
21 | if (s[0] == '0') { | |
22 | int ch = tolower(s[1]); | |
23 | ||
24 | if (ch == 'x') { | |
25 | *basep = 16; | |
26 | s += 2; | |
27 | } else if (!*basep) { | |
28 | /* Only select octal if we don't have a base */ | |
29 | *basep = 8; | |
30 | } | |
2e794614 | 31 | } |
e6951139 SG |
32 | |
33 | /* Use decimal by default */ | |
34 | if (!*basep) | |
35 | *basep = 10; | |
36 | ||
2e794614 RC |
37 | return s; |
38 | } | |
39 | ||
5a94546e SG |
40 | /** |
41 | * decode_digit() - Decode a single character into its numeric digit value | |
42 | * | |
43 | * This ignore case | |
44 | * | |
45 | * @ch: Character to convert (expects '0'..'9', 'a'..'f' or 'A'..'F') | |
185f812c | 46 | * Return: value of digit (0..0xf) or 255 if the character is invalid |
5a94546e SG |
47 | */ |
48 | static uint decode_digit(int ch) | |
49 | { | |
50 | if (!isxdigit(ch)) | |
51 | return 256; | |
52 | ||
53 | ch = tolower(ch); | |
54 | ||
55 | return ch <= '9' ? ch - '0' : ch - 'a' + 0xa; | |
56 | } | |
57 | ||
7e5f460e | 58 | ulong simple_strtoul(const char *cp, char **endp, uint base) |
e4c5383e | 59 | { |
7e5f460e | 60 | ulong result = 0; |
5a94546e | 61 | uint value; |
e4c5383e | 62 | |
2e794614 | 63 | cp = _parse_integer_fixup_radix(cp, &base); |
e4c5383e | 64 | |
5a94546e SG |
65 | while (value = decode_digit(*cp), value < base) { |
66 | result = result * base + value; | |
e4c5383e SS |
67 | cp++; |
68 | } | |
69 | ||
70 | if (endp) | |
71 | *endp = (char *)cp; | |
72 | ||
73 | return result; | |
74 | } | |
75 | ||
7e5f460e SG |
76 | ulong hextoul(const char *cp, char **endp) |
77 | { | |
78 | return simple_strtoul(cp, endp, 16); | |
79 | } | |
80 | ||
0b1284eb SG |
81 | ulong dectoul(const char *cp, char **endp) |
82 | { | |
83 | return simple_strtoul(cp, endp, 10); | |
84 | } | |
85 | ||
e4c5383e SS |
86 | int strict_strtoul(const char *cp, unsigned int base, unsigned long *res) |
87 | { | |
88 | char *tail; | |
89 | unsigned long val; | |
90 | size_t len; | |
91 | ||
92 | *res = 0; | |
93 | len = strlen(cp); | |
94 | if (len == 0) | |
95 | return -EINVAL; | |
96 | ||
97 | val = simple_strtoul(cp, &tail, base); | |
98 | if (tail == cp) | |
99 | return -EINVAL; | |
100 | ||
101 | if ((*tail == '\0') || | |
102 | ((len == (size_t)(tail - cp) + 1) && (*tail == '\n'))) { | |
103 | *res = val; | |
104 | return 0; | |
105 | } | |
106 | ||
107 | return -EINVAL; | |
108 | } | |
109 | ||
110 | long simple_strtol(const char *cp, char **endp, unsigned int base) | |
111 | { | |
112 | if (*cp == '-') | |
113 | return -simple_strtoul(cp + 1, endp, base); | |
114 | ||
115 | return simple_strtoul(cp, endp, base); | |
116 | } | |
117 | ||
118 | unsigned long ustrtoul(const char *cp, char **endp, unsigned int base) | |
119 | { | |
120 | unsigned long result = simple_strtoul(cp, endp, base); | |
a353e6aa MR |
121 | switch (tolower(**endp)) { |
122 | case 'g': | |
e4c5383e SS |
123 | result *= 1024; |
124 | /* fall through */ | |
a353e6aa | 125 | case 'm': |
e4c5383e SS |
126 | result *= 1024; |
127 | /* fall through */ | |
e4c5383e SS |
128 | case 'k': |
129 | result *= 1024; | |
b87b0d8d MR |
130 | (*endp)++; |
131 | if (**endp == 'i') | |
132 | (*endp)++; | |
133 | if (**endp == 'B') | |
134 | (*endp)++; | |
e4c5383e SS |
135 | } |
136 | return result; | |
137 | } | |
138 | ||
139 | unsigned long long ustrtoull(const char *cp, char **endp, unsigned int base) | |
140 | { | |
141 | unsigned long long result = simple_strtoull(cp, endp, base); | |
a353e6aa MR |
142 | switch (tolower(**endp)) { |
143 | case 'g': | |
e4c5383e SS |
144 | result *= 1024; |
145 | /* fall through */ | |
a353e6aa | 146 | case 'm': |
e4c5383e SS |
147 | result *= 1024; |
148 | /* fall through */ | |
e4c5383e SS |
149 | case 'k': |
150 | result *= 1024; | |
b87b0d8d MR |
151 | (*endp)++; |
152 | if (**endp == 'i') | |
153 | (*endp)++; | |
154 | if (**endp == 'B') | |
155 | (*endp)++; | |
e4c5383e SS |
156 | } |
157 | return result; | |
158 | } | |
159 | ||
160 | unsigned long long simple_strtoull(const char *cp, char **endp, | |
161 | unsigned int base) | |
162 | { | |
5a94546e SG |
163 | unsigned long long result = 0; |
164 | uint value; | |
e4c5383e | 165 | |
2e794614 | 166 | cp = _parse_integer_fixup_radix(cp, &base); |
e4c5383e | 167 | |
5a94546e | 168 | while (value = decode_digit(*cp), value < base) { |
e4c5383e SS |
169 | result = result * base + value; |
170 | cp++; | |
171 | } | |
172 | ||
173 | if (endp) | |
174 | *endp = (char *) cp; | |
175 | ||
176 | return result; | |
177 | } | |
178 | ||
0b016428 RG |
179 | long long simple_strtoll(const char *cp, char **endp, unsigned int base) |
180 | { | |
181 | if (*cp == '-') | |
182 | return -simple_strtoull(cp + 1, endp, base); | |
183 | ||
184 | return simple_strtoull(cp, endp, base); | |
185 | } | |
186 | ||
8565efd5 | 187 | long trailing_strtoln_end(const char *str, const char *end, char const **endp) |
e4c5383e SS |
188 | { |
189 | const char *p; | |
190 | ||
191 | if (!end) | |
192 | end = str + strlen(str); | |
d667a0d8 SG |
193 | p = end - 1; |
194 | if (p > str && isdigit(*p)) { | |
195 | do { | |
8565efd5 SG |
196 | if (!isdigit(p[-1])) { |
197 | if (endp) | |
198 | *endp = p; | |
d667a0d8 | 199 | return dectoul(p, NULL); |
8565efd5 | 200 | } |
d667a0d8 | 201 | } while (--p > str); |
e4c5383e | 202 | } |
8565efd5 SG |
203 | if (endp) |
204 | *endp = end; | |
e4c5383e SS |
205 | |
206 | return -1; | |
207 | } | |
208 | ||
8565efd5 SG |
209 | long trailing_strtoln(const char *str, const char *end) |
210 | { | |
211 | return trailing_strtoln_end(str, end, NULL); | |
212 | } | |
213 | ||
e4c5383e SS |
214 | long trailing_strtol(const char *str) |
215 | { | |
216 | return trailing_strtoln(str, NULL); | |
217 | } | |
fdc79a6b SG |
218 | |
219 | void str_to_upper(const char *in, char *out, size_t len) | |
220 | { | |
221 | for (; len > 0 && *in; len--) | |
222 | *out++ = toupper(*in++); | |
223 | if (len) | |
224 | *out = '\0'; | |
225 | } | |
3e96ed44 SG |
226 | |
227 | const char **str_to_list(const char *instr) | |
228 | { | |
229 | const char **ptr; | |
230 | char *str, *p; | |
231 | int count, i; | |
232 | ||
233 | /* don't allocate if the string is empty */ | |
234 | str = *instr ? strdup(instr) : (char *)instr; | |
235 | if (!str) | |
236 | return NULL; | |
237 | ||
238 | /* count the number of space-separated strings */ | |
947aafde | 239 | for (count = 0, p = str; *p; p++) { |
3e96ed44 SG |
240 | if (*p == ' ') { |
241 | count++; | |
242 | *p = '\0'; | |
243 | } | |
244 | } | |
947aafde SG |
245 | if (p != str && p[-1]) |
246 | count++; | |
3e96ed44 SG |
247 | |
248 | /* allocate the pointer array, allowing for a NULL terminator */ | |
249 | ptr = calloc(count + 1, sizeof(char *)); | |
250 | if (!ptr) { | |
251 | if (*str) | |
252 | free(str); | |
253 | return NULL; | |
254 | } | |
255 | ||
256 | for (i = 0, p = str; i < count; p += strlen(p) + 1, i++) | |
257 | ptr[i] = p; | |
258 | ||
259 | return ptr; | |
260 | } | |
261 | ||
262 | void str_free_list(const char **ptr) | |
263 | { | |
264 | if (ptr) | |
265 | free((char *)ptr[0]); | |
266 | free(ptr); | |
267 | } |