Linux kernel mirror (for testing)
git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel
os
linux
1/* SPDX-License-Identifier: LGPL-2.1 OR MIT */
2/*
3 * stdlib function definitions for NOLIBC
4 * Copyright (C) 2017-2021 Willy Tarreau <w@1wt.eu>
5 */
6
7/* make sure to include all global symbols */
8#include "nolibc.h"
9
10#ifndef _NOLIBC_STDLIB_H
11#define _NOLIBC_STDLIB_H
12
13#include "std.h"
14#include "arch.h"
15#include "types.h"
16#include "sys.h"
17#include "string.h"
18#include <linux/auxvec.h>
19
20struct nolibc_heap {
21 size_t len;
22 char user_p[] __attribute__((__aligned__));
23};
24
25/* Buffer used to store int-to-ASCII conversions. Will only be implemented if
26 * any of the related functions is implemented. The area is large enough to
27 * store "18446744073709551615" or "-9223372036854775808" and the final zero.
28 */
29static __attribute__((unused)) char itoa_buffer[21];
30
31/*
32 * As much as possible, please keep functions alphabetically sorted.
33 */
34
35static __inline__
36int abs(int j)
37{
38 return j >= 0 ? j : -j;
39}
40
41static __inline__
42long labs(long j)
43{
44 return j >= 0 ? j : -j;
45}
46
47static __inline__
48long long llabs(long long j)
49{
50 return j >= 0 ? j : -j;
51}
52
53/* must be exported, as it's used by libgcc for various divide functions */
54void abort(void);
55__attribute__((weak,unused,noreturn,section(".text.nolibc_abort")))
56void abort(void)
57{
58 sys_kill(sys_getpid(), SIGABRT);
59 for (;;);
60}
61
62static __attribute__((unused))
63long atol(const char *s)
64{
65 unsigned long ret = 0;
66 unsigned long d;
67 int neg = 0;
68
69 if (*s == '-') {
70 neg = 1;
71 s++;
72 }
73
74 while (1) {
75 d = (*s++) - '0';
76 if (d > 9)
77 break;
78 ret *= 10;
79 ret += d;
80 }
81
82 return neg ? -ret : ret;
83}
84
85static __attribute__((unused))
86int atoi(const char *s)
87{
88 return atol(s);
89}
90
91static __attribute__((unused))
92void free(void *ptr)
93{
94 struct nolibc_heap *heap;
95
96 if (!ptr)
97 return;
98
99 heap = container_of(ptr, struct nolibc_heap, user_p);
100 munmap(heap, heap->len);
101}
102
103#ifndef NOLIBC_NO_RUNTIME
104/* getenv() tries to find the environment variable named <name> in the
105 * environment array pointed to by global variable "environ" which must be
106 * declared as a char **, and must be terminated by a NULL (it is recommended
107 * to set this variable to the "envp" argument of main()). If the requested
108 * environment variable exists its value is returned otherwise NULL is
109 * returned.
110 */
111static __attribute__((unused))
112char *getenv(const char *name)
113{
114 int idx, i;
115
116 if (environ) {
117 for (idx = 0; environ[idx]; idx++) {
118 for (i = 0; name[i] && name[i] == environ[idx][i];)
119 i++;
120 if (!name[i] && environ[idx][i] == '=')
121 return &environ[idx][i+1];
122 }
123 }
124 return NULL;
125}
126#endif /* NOLIBC_NO_RUNTIME */
127
128static __attribute__((unused))
129void *malloc(size_t len)
130{
131 struct nolibc_heap *heap;
132
133 /* Always allocate memory with size multiple of 4096. */
134 len = sizeof(*heap) + len;
135 len = (len + 4095UL) & -4096UL;
136 heap = mmap(NULL, len, PROT_READ|PROT_WRITE, MAP_ANONYMOUS|MAP_PRIVATE,
137 -1, 0);
138 if (__builtin_expect(heap == MAP_FAILED, 0))
139 return NULL;
140
141 heap->len = len;
142 return heap->user_p;
143}
144
145static __attribute__((unused))
146void *calloc(size_t size, size_t nmemb)
147{
148 size_t x = size * nmemb;
149
150 if (__builtin_expect(size && ((x / size) != nmemb), 0)) {
151 SET_ERRNO(ENOMEM);
152 return NULL;
153 }
154
155 /*
156 * No need to zero the heap, the MAP_ANONYMOUS in malloc()
157 * already does it.
158 */
159 return malloc(x);
160}
161
162static __attribute__((unused))
163void *realloc(void *old_ptr, size_t new_size)
164{
165 struct nolibc_heap *heap;
166 size_t user_p_len;
167 void *ret;
168
169 if (!old_ptr)
170 return malloc(new_size);
171
172 heap = container_of(old_ptr, struct nolibc_heap, user_p);
173 user_p_len = heap->len - sizeof(*heap);
174 /*
175 * Don't realloc() if @user_p_len >= @new_size, this block of
176 * memory is still enough to handle the @new_size. Just return
177 * the same pointer.
178 */
179 if (user_p_len >= new_size)
180 return old_ptr;
181
182 ret = malloc(new_size);
183 if (__builtin_expect(!ret, 0))
184 return NULL;
185
186 memcpy(ret, heap->user_p, user_p_len);
187 munmap(heap, heap->len);
188 return ret;
189}
190
191/* Converts the unsigned long integer <in> to its hex representation into
192 * buffer <buffer>, which must be long enough to store the number and the
193 * trailing zero (17 bytes for "ffffffffffffffff" or 9 for "ffffffff"). The
194 * buffer is filled from the first byte, and the number of characters emitted
195 * (not counting the trailing zero) is returned. The function is constructed
196 * in a way to optimize the code size and avoid any divide that could add a
197 * dependency on large external functions.
198 */
199static __attribute__((unused))
200int utoh_r(unsigned long in, char *buffer)
201{
202 signed char pos = (~0UL > 0xfffffffful) ? 60 : 28;
203 int digits = 0;
204 int dig;
205
206 do {
207 dig = in >> pos;
208 in -= (uint64_t)dig << pos;
209 pos -= 4;
210 if (dig || digits || pos < 0) {
211 if (dig > 9)
212 dig += 'a' - '0' - 10;
213 buffer[digits++] = '0' + dig;
214 }
215 } while (pos >= 0);
216
217 buffer[digits] = 0;
218 return digits;
219}
220
221/* converts unsigned long <in> to an hex string using the static itoa_buffer
222 * and returns the pointer to that string.
223 */
224static __inline__ __attribute__((unused))
225char *utoh(unsigned long in)
226{
227 utoh_r(in, itoa_buffer);
228 return itoa_buffer;
229}
230
231/* Converts the unsigned long integer <in> to its string representation into
232 * buffer <buffer>, which must be long enough to store the number and the
233 * trailing zero (21 bytes for 18446744073709551615 in 64-bit, 11 for
234 * 4294967295 in 32-bit). The buffer is filled from the first byte, and the
235 * number of characters emitted (not counting the trailing zero) is returned.
236 * The function is constructed in a way to optimize the code size and avoid
237 * any divide that could add a dependency on large external functions.
238 */
239static __attribute__((unused))
240int utoa_r(unsigned long in, char *buffer)
241{
242 unsigned long lim;
243 int digits = 0;
244 int pos = (~0UL > 0xfffffffful) ? 19 : 9;
245 int dig;
246
247 do {
248 for (dig = 0, lim = 1; dig < pos; dig++)
249 lim *= 10;
250
251 if (digits || in >= lim || !pos) {
252 for (dig = 0; in >= lim; dig++)
253 in -= lim;
254 buffer[digits++] = '0' + dig;
255 }
256 } while (pos--);
257
258 buffer[digits] = 0;
259 return digits;
260}
261
262/* Converts the signed long integer <in> to its string representation into
263 * buffer <buffer>, which must be long enough to store the number and the
264 * trailing zero (21 bytes for -9223372036854775808 in 64-bit, 12 for
265 * -2147483648 in 32-bit). The buffer is filled from the first byte, and the
266 * number of characters emitted (not counting the trailing zero) is returned.
267 */
268static __attribute__((unused))
269int itoa_r(long in, char *buffer)
270{
271 char *ptr = buffer;
272 int len = 0;
273
274 if (in < 0) {
275 in = -(unsigned long)in;
276 *(ptr++) = '-';
277 len++;
278 }
279 len += utoa_r(in, ptr);
280 return len;
281}
282
283/* for historical compatibility, same as above but returns the pointer to the
284 * buffer.
285 */
286static __inline__ __attribute__((unused))
287char *ltoa_r(long in, char *buffer)
288{
289 itoa_r(in, buffer);
290 return buffer;
291}
292
293/* converts long integer <in> to a string using the static itoa_buffer and
294 * returns the pointer to that string.
295 */
296static __inline__ __attribute__((unused))
297char *itoa(long in)
298{
299 itoa_r(in, itoa_buffer);
300 return itoa_buffer;
301}
302
303/* converts long integer <in> to a string using the static itoa_buffer and
304 * returns the pointer to that string. Same as above, for compatibility.
305 */
306static __inline__ __attribute__((unused))
307char *ltoa(long in)
308{
309 itoa_r(in, itoa_buffer);
310 return itoa_buffer;
311}
312
313/* converts unsigned long integer <in> to a string using the static itoa_buffer
314 * and returns the pointer to that string.
315 */
316static __inline__ __attribute__((unused))
317char *utoa(unsigned long in)
318{
319 utoa_r(in, itoa_buffer);
320 return itoa_buffer;
321}
322
323/* Converts the unsigned 64-bit integer <in> to its hex representation into
324 * buffer <buffer>, which must be long enough to store the number and the
325 * trailing zero (17 bytes for "ffffffffffffffff"). The buffer is filled from
326 * the first byte, and the number of characters emitted (not counting the
327 * trailing zero) is returned. The function is constructed in a way to optimize
328 * the code size and avoid any divide that could add a dependency on large
329 * external functions.
330 */
331static __attribute__((unused))
332int u64toh_r(uint64_t in, char *buffer)
333{
334 signed char pos = 60;
335 int digits = 0;
336 int dig;
337
338 do {
339 if (sizeof(long) >= 8) {
340 dig = (in >> pos) & 0xF;
341 } else {
342 /* 32-bit platforms: avoid a 64-bit shift */
343 uint32_t d = (pos >= 32) ? (in >> 32) : in;
344 dig = (d >> (pos & 31)) & 0xF;
345 }
346 if (dig > 9)
347 dig += 'a' - '0' - 10;
348 pos -= 4;
349 if (dig || digits || pos < 0)
350 buffer[digits++] = '0' + dig;
351 } while (pos >= 0);
352
353 buffer[digits] = 0;
354 return digits;
355}
356
357/* converts uint64_t <in> to an hex string using the static itoa_buffer and
358 * returns the pointer to that string.
359 */
360static __inline__ __attribute__((unused))
361char *u64toh(uint64_t in)
362{
363 u64toh_r(in, itoa_buffer);
364 return itoa_buffer;
365}
366
367/* Converts the unsigned 64-bit integer <in> to its string representation into
368 * buffer <buffer>, which must be long enough to store the number and the
369 * trailing zero (21 bytes for 18446744073709551615). The buffer is filled from
370 * the first byte, and the number of characters emitted (not counting the
371 * trailing zero) is returned. The function is constructed in a way to optimize
372 * the code size and avoid any divide that could add a dependency on large
373 * external functions.
374 */
375static __attribute__((unused))
376int u64toa_r(uint64_t in, char *buffer)
377{
378 unsigned long long lim;
379 int digits = 0;
380 int pos = 19; /* start with the highest possible digit */
381 int dig;
382
383 do {
384 for (dig = 0, lim = 1; dig < pos; dig++)
385 lim *= 10;
386
387 if (digits || in >= lim || !pos) {
388 for (dig = 0; in >= lim; dig++)
389 in -= lim;
390 buffer[digits++] = '0' + dig;
391 }
392 } while (pos--);
393
394 buffer[digits] = 0;
395 return digits;
396}
397
398/* Converts the signed 64-bit integer <in> to its string representation into
399 * buffer <buffer>, which must be long enough to store the number and the
400 * trailing zero (21 bytes for -9223372036854775808). The buffer is filled from
401 * the first byte, and the number of characters emitted (not counting the
402 * trailing zero) is returned.
403 */
404static __attribute__((unused))
405int i64toa_r(int64_t in, char *buffer)
406{
407 char *ptr = buffer;
408 int len = 0;
409
410 if (in < 0) {
411 in = -(uint64_t)in;
412 *(ptr++) = '-';
413 len++;
414 }
415 len += u64toa_r(in, ptr);
416 return len;
417}
418
419/* converts int64_t <in> to a string using the static itoa_buffer and returns
420 * the pointer to that string.
421 */
422static __inline__ __attribute__((unused))
423char *i64toa(int64_t in)
424{
425 i64toa_r(in, itoa_buffer);
426 return itoa_buffer;
427}
428
429/* converts uint64_t <in> to a string using the static itoa_buffer and returns
430 * the pointer to that string.
431 */
432static __inline__ __attribute__((unused))
433char *u64toa(uint64_t in)
434{
435 u64toa_r(in, itoa_buffer);
436 return itoa_buffer;
437}
438
439static __attribute__((unused))
440uintmax_t __strtox(const char *nptr, char **endptr, int base, intmax_t lower_limit, uintmax_t upper_limit)
441{
442 const char signed_ = lower_limit != 0;
443 unsigned char neg = 0, overflow = 0;
444 uintmax_t val = 0, limit, old_val;
445 char c;
446
447 if (base < 0 || base > 36) {
448 SET_ERRNO(EINVAL);
449 goto out;
450 }
451
452 while (isspace(*nptr))
453 nptr++;
454
455 if (*nptr == '+') {
456 nptr++;
457 } else if (*nptr == '-') {
458 neg = 1;
459 nptr++;
460 }
461
462 if (signed_ && neg)
463 limit = -(uintmax_t)lower_limit;
464 else
465 limit = upper_limit;
466
467 if ((base == 0 || base == 16) &&
468 (strncmp(nptr, "0x", 2) == 0 || strncmp(nptr, "0X", 2) == 0)) {
469 base = 16;
470 nptr += 2;
471 } else if (base == 0 && strncmp(nptr, "0", 1) == 0) {
472 base = 8;
473 nptr += 1;
474 } else if (base == 0) {
475 base = 10;
476 }
477
478 while (*nptr) {
479 c = *nptr;
480
481 if (c >= '0' && c <= '9')
482 c -= '0';
483 else if (c >= 'a' && c <= 'z')
484 c = c - 'a' + 10;
485 else if (c >= 'A' && c <= 'Z')
486 c = c - 'A' + 10;
487 else
488 goto out;
489
490 if (c >= base)
491 goto out;
492
493 nptr++;
494 old_val = val;
495 val *= base;
496 val += c;
497
498 if (val > limit || val < old_val)
499 overflow = 1;
500 }
501
502out:
503 if (overflow) {
504 SET_ERRNO(ERANGE);
505 val = limit;
506 }
507 if (endptr)
508 *endptr = (char *)nptr;
509 return neg ? -val : val;
510}
511
512static __attribute__((unused))
513long strtol(const char *nptr, char **endptr, int base)
514{
515 return __strtox(nptr, endptr, base, LONG_MIN, LONG_MAX);
516}
517
518static __attribute__((unused))
519unsigned long strtoul(const char *nptr, char **endptr, int base)
520{
521 return __strtox(nptr, endptr, base, 0, ULONG_MAX);
522}
523
524static __attribute__((unused))
525long long strtoll(const char *nptr, char **endptr, int base)
526{
527 return __strtox(nptr, endptr, base, LLONG_MIN, LLONG_MAX);
528}
529
530static __attribute__((unused))
531unsigned long long strtoull(const char *nptr, char **endptr, int base)
532{
533 return __strtox(nptr, endptr, base, 0, ULLONG_MAX);
534}
535
536static __attribute__((unused))
537intmax_t strtoimax(const char *nptr, char **endptr, int base)
538{
539 return __strtox(nptr, endptr, base, INTMAX_MIN, INTMAX_MAX);
540}
541
542static __attribute__((unused))
543uintmax_t strtoumax(const char *nptr, char **endptr, int base)
544{
545 return __strtox(nptr, endptr, base, 0, UINTMAX_MAX);
546}
547
548#endif /* _NOLIBC_STDLIB_H */