blob: ebdd413d13ec72ce60d25652863ea882b45837fe [file]
/* SPDX-License-Identifier: LGPL-2.1 OR MIT */
/*
* minimal stdio function definitions for NOLIBC
* Copyright (C) 2017-2021 Willy Tarreau <w@1wt.eu>
*/
/* make sure to include all global symbols */
#include "nolibc.h"
#ifndef _NOLIBC_STDIO_H
#define _NOLIBC_STDIO_H
#include "std.h"
#include "arch.h"
#include "errno.h"
#include "fcntl.h"
#include "types.h"
#include "sys.h"
#include "stdarg.h"
#include "stdlib.h"
#include "string.h"
#include "compiler.h"
static const char *strerror(int errnum);
#ifndef EOF
#define EOF (-1)
#endif
/* Buffering mode used by setvbuf. */
#define _IOFBF 0 /* Fully buffered. */
#define _IOLBF 1 /* Line buffered. */
#define _IONBF 2 /* No buffering. */
/* just define FILE as a non-empty type. The value of the pointer gives
* the FD: FILE=~fd for fd>=0 or NULL for fd<0. This way positive FILE
* are immediately identified as abnormal entries (i.e. possible copies
* of valid pointers to something else).
*/
typedef struct FILE {
char dummy[1];
} FILE;
static __attribute__((unused)) FILE* const stdin = (FILE*)(intptr_t)~STDIN_FILENO;
static __attribute__((unused)) FILE* const stdout = (FILE*)(intptr_t)~STDOUT_FILENO;
static __attribute__((unused)) FILE* const stderr = (FILE*)(intptr_t)~STDERR_FILENO;
/* provides a FILE* equivalent of fd. The mode is ignored. */
static __attribute__((unused))
FILE *fdopen(int fd, const char *mode __attribute__((unused)))
{
if (fd < 0) {
SET_ERRNO(EBADF);
return NULL;
}
return (FILE*)(intptr_t)~fd;
}
static __attribute__((unused))
FILE *fopen(const char *pathname, const char *mode)
{
int flags, fd;
switch (*mode) {
case 'r':
flags = O_RDONLY;
break;
case 'w':
flags = O_WRONLY | O_CREAT | O_TRUNC;
break;
case 'a':
flags = O_WRONLY | O_CREAT | O_APPEND;
break;
default:
SET_ERRNO(EINVAL); return NULL;
}
if (mode[1] == '+')
flags = (flags & ~(O_RDONLY | O_WRONLY)) | O_RDWR;
fd = open(pathname, flags, 0666);
return fdopen(fd, mode);
}
/* provides the fd of stream. */
static __attribute__((unused))
int fileno(FILE *stream)
{
intptr_t i = (intptr_t)stream;
if (i >= 0) {
SET_ERRNO(EBADF);
return -1;
}
return ~i;
}
/* flush a stream. */
static __attribute__((unused))
int fflush(FILE *stream)
{
intptr_t i = (intptr_t)stream;
/* NULL is valid here. */
if (i > 0) {
SET_ERRNO(EBADF);
return -1;
}
/* Don't do anything, nolibc does not support buffering. */
return 0;
}
/* flush a stream. */
static __attribute__((unused))
int fclose(FILE *stream)
{
intptr_t i = (intptr_t)stream;
if (i >= 0) {
SET_ERRNO(EBADF);
return -1;
}
if (close(~i))
return EOF;
return 0;
}
/* getc(), fgetc(), getchar() */
#define getc(stream) fgetc(stream)
static __attribute__((unused))
int fgetc(FILE* stream)
{
unsigned char ch;
if (read(fileno(stream), &ch, 1) <= 0)
return EOF;
return ch;
}
static __attribute__((unused))
int getchar(void)
{
return fgetc(stdin);
}
/* putc(), fputc(), putchar() */
#define putc(c, stream) fputc(c, stream)
static __attribute__((unused))
int fputc(int c, FILE* stream)
{
unsigned char ch = c;
if (write(fileno(stream), &ch, 1) <= 0)
return EOF;
return ch;
}
static __attribute__((unused))
int putchar(int c)
{
return fputc(c, stdout);
}
/* fwrite(), fread(), puts(), fputs(). Note that puts() emits '\n' but not fputs(). */
/* internal fwrite()-like function which only takes a size and returns 0 on
* success or EOF on error. It automatically retries on short writes.
*/
static __attribute__((unused))
int _fwrite(const void *buf, size_t size, FILE *stream)
{
ssize_t ret;
int fd = fileno(stream);
while (size) {
ret = write(fd, buf, size);
if (ret <= 0)
return EOF;
size -= ret;
buf += ret;
}
return 0;
}
static __attribute__((unused))
size_t fwrite(const void *s, size_t size, size_t nmemb, FILE *stream)
{
size_t written;
for (written = 0; written < nmemb; written++) {
if (_fwrite(s, size, stream) != 0)
break;
s += size;
}
return written;
}
/* internal fread()-like function which only takes a size and returns 0 on
* success or EOF on error. It automatically retries on short reads.
*/
static __attribute__((unused))
int _fread(void *buf, size_t size, FILE *stream)
{
int fd = fileno(stream);
ssize_t ret;
while (size) {
ret = read(fd, buf, size);
if (ret <= 0)
return EOF;
size -= ret;
buf += ret;
}
return 0;
}
static __attribute__((unused))
size_t fread(void *s, size_t size, size_t nmemb, FILE *stream)
{
size_t nread;
for (nread = 0; nread < nmemb; nread++) {
if (_fread(s, size, stream) != 0)
break;
s += size;
}
return nread;
}
static __attribute__((unused))
int fputs(const char *s, FILE *stream)
{
return _fwrite(s, strlen(s), stream);
}
static __attribute__((unused))
int puts(const char *s)
{
if (fputs(s, stdout) == EOF)
return EOF;
return putchar('\n');
}
/* fgets() */
static __attribute__((unused))
char *fgets(char *s, int size, FILE *stream)
{
int ofs;
int c;
for (ofs = 0; ofs + 1 < size;) {
c = fgetc(stream);
if (c == EOF)
break;
s[ofs++] = c;
if (c == '\n')
break;
}
if (ofs < size)
s[ofs] = 0;
return ofs ? s : NULL;
}
/* fseek */
static __attribute__((unused))
int fseek(FILE *stream, long offset, int whence)
{
int fd = fileno(stream);
off_t ret;
ret = lseek(fd, offset, whence);
/* lseek() and fseek() differ in that lseek returns the new
* position or -1, fseek() returns either 0 or -1.
*/
if (ret >= 0)
return 0;
return -1;
}
/* printf(). Supports most of the normal integer and string formats.
* - %[#0-+ ][width|*[.precision|*}][{l,t,z,ll,L,j,q}]{c,d,i,u,o,x,X,p,s,m,%}
* - %% generates a single %
* - %m outputs strerror(errno).
* - %X outputs a..f the same as %x.
* - No support for floating point or wide characters.
* - Invalid formats are copied to the output buffer.
*
* Called by vfprintf() and snprintf() to do the actual formatting.
* The callers provide a callback function to save the formatted data.
* The callback function is called multiple times:
* - for each group of literal characters in the format string.
* - for field padding.
* - for each conversion specifier.
* - with (NULL, 0) at the end of the __nolibc_printf.
* If the callback returns non-zero __nolibc_printf() immediately returns -1.
*/
typedef int (*__nolibc_printf_cb)(void *state, const char *buf, size_t size);
/* This code uses 'flag' variables that are indexed by the low 6 bits
* of characters to optimise checks for multiple characters.
*
* _NOLIBC_PF_FLAGS_CONTAIN(flags, 'a', 'b'. ...)
* returns non-zero if the bit for any of the specified characters is set.
*
* _NOLIBC_PF_CHAR_IS_ONE_OF(ch, 'a', 'b'. ...)
* returns the flag bit for ch if it is one of the specified characters.
* All the characters must be in the same 32 character block (non-alphabetic,
* upper case, or lower case) of the ASCII character set.
*/
#define _NOLIBC_PF_FLAG(ch) (1u << ((ch) & 0x1f))
#define _NOLIBC_PF_FLAG_NZ(ch) ((ch) ? _NOLIBC_PF_FLAG(ch) : 0)
#define _NOLIBC_PF_FLAG8(cmp_1, cmp_2, cmp_3, cmp_4, cmp_5, cmp_6, cmp_7, cmp_8, ...) \
(_NOLIBC_PF_FLAG_NZ(cmp_1) | _NOLIBC_PF_FLAG_NZ(cmp_2) | \
_NOLIBC_PF_FLAG_NZ(cmp_3) | _NOLIBC_PF_FLAG_NZ(cmp_4) | \
_NOLIBC_PF_FLAG_NZ(cmp_5) | _NOLIBC_PF_FLAG_NZ(cmp_6) | \
_NOLIBC_PF_FLAG_NZ(cmp_7) | _NOLIBC_PF_FLAG_NZ(cmp_8))
#define _NOLIBC_PF_FLAGS_CONTAIN(flags, ...) \
((flags) & _NOLIBC_PF_FLAG8(__VA_ARGS__, 0, 0, 0, 0, 0, 0, 0))
#define _NOLIBC_PF_CHAR_IS_ONE_OF(ch, cmp_1, ...) \
((unsigned int)(ch) - (cmp_1 & 0xe0) > 0x1f ? 0 : \
_NOLIBC_PF_FLAGS_CONTAIN(_NOLIBC_PF_FLAG(ch), cmp_1, __VA_ARGS__))
static __attribute__((unused, format(printf, 3, 0)))
int __nolibc_printf(__nolibc_printf_cb cb, void *state, const char *fmt, va_list args)
{
char ch;
unsigned long long v;
long long signed_v;
int written, width, precision, len;
unsigned int flags, ch_flag;
char outbuf[2 + 31 + 22 + 1];
char *out;
const char *outstr;
unsigned int sign_prefix;
int got_width;
written = 0;
while (1) {
outstr = fmt;
ch = *fmt++;
if (!ch)
break;
width = 0;
flags = 0;
if (ch != '%') {
while (*fmt && *fmt != '%')
fmt++;
/* Output characters from the format string. */
len = fmt - outstr;
goto do_output;
}
/* we're in a format sequence */
/* Conversion flag characters */
while (1) {
ch = *fmt++;
ch_flag = _NOLIBC_PF_CHAR_IS_ONE_OF(ch, ' ', '#', '+', '-', '0');
if (!ch_flag)
break;
flags |= ch_flag;
}
/* Width and precision */
for (got_width = 0;; ch = *fmt++) {
if (ch == '*') {
precision = va_arg(args, int);
ch = *fmt++;
} else {
for (precision = 0; ch >= '0' && ch <= '9'; ch = *fmt++)
precision = precision * 10 + (ch - '0');
}
if (got_width)
break;
width = precision;
if (ch != '.') {
/* Default precision for strings */
precision = -1;
break;
}
got_width = 1;
}
/* A negative width (e.g. from "%*s") requests left justify. */
if (width < 0) {
width = -width;
flags |= _NOLIBC_PF_FLAG('-');
}
/* Length modifier.
* They miss the conversion flags characters " #+-0" so can go into flags.
* Change both L and ll to j (all always 64bit).
*/
if (ch == 'L')
ch = 'j';
ch_flag = _NOLIBC_PF_CHAR_IS_ONE_OF(ch, 'l', 't', 'z', 'j', 'q');
if (ch_flag != 0) {
if (ch == 'l' && fmt[0] == 'l') {
fmt++;
ch_flag = _NOLIBC_PF_FLAG('j');
}
flags |= ch_flag;
ch = *fmt++;
}
/* Conversion specifiers. */
/* Numeric and pointer conversion specifiers.
*
* Use an explicit bound check (rather than _NOLIBC_PF_CHAR_IS_ONE_OF())
* so that 'X' can be allowed through.
* 'X' gets treated and 'x' because _NOLIBC_PF_FLAG() returns the same
* value for both.
*
* We need to check for "%p" or "%#x" later, merging here gives better code.
* But '#' collides with 'c' so shift right.
*/
ch_flag = _NOLIBC_PF_FLAG(ch) | (flags & _NOLIBC_PF_FLAG('#')) >> 1;
if (((ch >= 'a' && ch <= 'z') || ch == 'X') &&
_NOLIBC_PF_FLAGS_CONTAIN(ch_flag, 'c', 'd', 'i', 'u', 'o', 'x', 'p', 's')) {
/* 'long' is needed for pointer/string conversions and ltz lengths.
* A single test can be used provided 'p' (the same bit as '0')
* is masked from flags.
*/
if (_NOLIBC_PF_FLAGS_CONTAIN(ch_flag | (flags & ~_NOLIBC_PF_FLAG('p')),
'p', 's', 'l', 't', 'z')) {
v = va_arg(args, unsigned long);
signed_v = (long)v;
} else if (_NOLIBC_PF_FLAGS_CONTAIN(flags, 'j', 'q')) {
v = va_arg(args, unsigned long long);
signed_v = v;
} else {
v = va_arg(args, unsigned int);
signed_v = (int)v;
}
if (ch == 'c') {
/* "%c" - single character. */
outbuf[0] = v;
len = 1;
outstr = outbuf;
goto do_output;
}
if (ch == 's') {
/* "%s" - character string. */
outstr = (const char *)(uintptr_t)v;
if (!outstr) {
outstr = "(null)";
/* Match glibc, nothing output if precision too small */
len = precision < 0 || precision >= 6 ? 6 : 0;
goto do_output;
}
goto do_strlen_output;
}
/* The 'sign_prefix' can be zero, one or two ("0x") characters.
* Prepended least significant byte first stopping on a zero byte.
*/
sign_prefix = 0;
if (_NOLIBC_PF_FLAGS_CONTAIN(ch_flag, 'd', 'i')) {
/* "%d" and "%i" - signed decimal numbers. */
if (signed_v < 0) {
sign_prefix = '-';
v = -(signed_v + 1);
v++;
} else if (_NOLIBC_PF_FLAGS_CONTAIN(flags, '+')) {
sign_prefix = '+';
} else if (_NOLIBC_PF_FLAGS_CONTAIN(flags, ' ')) {
sign_prefix = ' ';
}
} else {
/* "#o" requires that the output always starts with a '0'.
* This needs another check after any zero padding to avoid
* adding an extra leading '0'.
*/
if (_NOLIBC_PF_FLAGS_CONTAIN(ch_flag, 'o') &&
_NOLIBC_PF_FLAGS_CONTAIN(ch_flag, '#' - 1))
sign_prefix = '0';
}
/* The value is converted offset into the buffer so that
* 31 zero pad characters and the sign/prefix can be added in front.
* The longest digit string is 22 + 1 for octal conversions.
*/
out = outbuf + 2 + 31;
if (v == 0) {
/* There are special rules for zero. */
if (_NOLIBC_PF_FLAGS_CONTAIN(ch_flag, 'p')) {
/* "%p" match glibc, precision is ignored */
outstr = "(nil)";
len = 5;
goto do_output;
}
if (!precision) {
/* Explicit %nn.0d, no digits output (except for %#.0o) */
len = 0;
goto prepend_sign;
}
/* All other formats (including "%#x") just output "0". */
out[0] = '0';
len = 1;
} else {
/* Convert the number to ascii in the required base. */
unsigned long long recip;
unsigned int base;
if (_NOLIBC_PF_FLAGS_CONTAIN(ch_flag, 'd', 'i', 'u')) {
base = 10;
recip = _NOLIBC_U64TOA_RECIP(10);
} else if (_NOLIBC_PF_FLAGS_CONTAIN(ch_flag, 'o')) {
base = 8;
recip = _NOLIBC_U64TOA_RECIP(8);
} else {
base = 16;
recip = _NOLIBC_U64TOA_RECIP(16);
if (_NOLIBC_PF_FLAGS_CONTAIN(ch_flag, 'p', '#' - 1)) {
/* "%p" and "%#x" need "0x" prepending. */
sign_prefix = '0' << 8 | 'x';
}
}
len = _nolibc_u64toa_base(v, out, base, recip);
}
/* Add zero padding */
if (precision < 0) {
/* No explicit precision (or negative from "%.*s"). */
if (!_NOLIBC_PF_FLAGS_CONTAIN(flags, '0'))
goto no_zero_padding;
if (_NOLIBC_PF_FLAGS_CONTAIN(flags, '-'))
/* Left justify overrides zero pad */
goto no_zero_padding;
/* eg "%05d", Zero pad to field width less sign.
* Note that precision can end up negative so all
* the variables have to be 'signed int'.
*/
precision = width;
if (sign_prefix) {
precision--;
if (sign_prefix >= 256)
precision--;
}
}
if (precision > 31)
/* Don't run off the start of outbuf[], arbitrary limit
* longer than the longest number field. */
precision = 31;
for (; len < precision; len++) {
/* Stop gcc generating horrid code and memset(). */
_NOLIBC_OPTIMIZER_HIDE_VAR(len);
*--out = '0';
}
no_zero_padding:
/* %#o has set sign_prefix to '0', but we don't want so add an extra
* leading zero here.
* Since the only other byte values of sign_prefix are ' ', '+' and '-'
* it is enough to check that out[] doesn't already start with sign_prefix.
*/
if (sign_prefix - *out) {
prepend_sign:
/* Add the 0, 1 or 2 ("0x") sign/prefix characters at the front. */
for (; sign_prefix; sign_prefix >>= 8) {
/* Force gcc to increment len inside the loop. */
_NOLIBC_OPTIMIZER_HIDE_VAR(len);
len++;
*--out = sign_prefix;
}
}
outstr = out;
goto do_output;
}
if (ch == 'm') {
#ifdef NOLIBC_IGNORE_ERRNO
outstr = "unknown error";
#else
outstr = strerror(errno);
#endif /* NOLIBC_IGNORE_ERRNO */
goto do_strlen_output;
}
if (ch != '%') {
/* Invalid format: back up to output the format characters */
fmt = outstr + 1;
/* and output a '%' now. */
}
/* %% is documented as a 'conversion specifier'.
* Any flags, precision or length modifier are ignored.
*/
len = 1;
width = 0;
outstr = fmt - 1;
goto do_output;
do_strlen_output:
/* Open coded strnlen() (slightly smaller). */
for (len = 0; precision < 0 || len < precision; len++)
if (!outstr[len])
break;
do_output:
written += len;
/* Stop gcc back-merging this code into one of the conditionals above. */
_NOLIBC_OPTIMIZER_HIDE_VAR(len);
/* Output the characters on the required side of any padding. */
width -= len;
flags = _NOLIBC_PF_FLAGS_CONTAIN(flags, '-');
if (flags && cb(state, outstr, len) != 0)
return -1;
while (width > 0) {
/* Output pad in 16 byte blocks with the small block first. */
int pad_len = ((width - 1) & 15) + 1;
width -= pad_len;
written += pad_len;
if (cb(state, " ", pad_len) != 0)
return -1;
}
if (!flags && cb(state, outstr, len) != 0)
return -1;
}
/* Request a final '\0' be added to the snprintf() output.
* This may be the only call of the cb() function.
*/
if (cb(state, NULL, 0) != 0)
return -1;
return written;
}
static int __nolibc_fprintf_cb(void *stream, const char *buf, size_t size)
{
return _fwrite(buf, size, stream);
}
static __attribute__((unused, format(printf, 2, 0)))
int vfprintf(FILE *stream, const char *fmt, va_list args)
{
return __nolibc_printf(__nolibc_fprintf_cb, stream, fmt, args);
}
static __attribute__((unused, format(printf, 1, 0)))
int vprintf(const char *fmt, va_list args)
{
return vfprintf(stdout, fmt, args);
}
static __attribute__((unused, format(printf, 2, 3)))
int fprintf(FILE *stream, const char *fmt, ...)
{
va_list args;
int ret;
va_start(args, fmt);
ret = vfprintf(stream, fmt, args);
va_end(args);
return ret;
}
static __attribute__((unused, format(printf, 1, 2)))
int printf(const char *fmt, ...)
{
va_list args;
int ret;
va_start(args, fmt);
ret = vfprintf(stdout, fmt, args);
va_end(args);
return ret;
}
static __attribute__((unused, format(printf, 2, 0)))
int vdprintf(int fd, const char *fmt, va_list args)
{
FILE *stream;
stream = fdopen(fd, NULL);
if (!stream)
return -1;
/* Technically 'stream' is leaked, but as it's only a wrapper around 'fd' that is fine */
return vfprintf(stream, fmt, args);
}
static __attribute__((unused, format(printf, 2, 3)))
int dprintf(int fd, const char *fmt, ...)
{
va_list args;
int ret;
va_start(args, fmt);
ret = vdprintf(fd, fmt, args);
va_end(args);
return ret;
}
struct __nolibc_sprintf_cb_state {
char *buf;
size_t space;
};
static int __nolibc_sprintf_cb(void *v_state, const char *buf, size_t size)
{
struct __nolibc_sprintf_cb_state *state = v_state;
size_t space = state->space;
char *tgt;
/* Truncate the request to fit in the output buffer space.
* The last byte is reserved for the terminating '\0'.
* state->space can only be zero for snprintf(NULL, 0, fmt, args)
* so this normally lets through calls with 'size == 0'.
*/
if (size >= space) {
if (space <= 1)
return 0;
size = space - 1;
}
tgt = state->buf;
/* __nolibc_printf() ends with cb(state, NULL, 0) to request the output
* buffer be '\0' terminated.
* That will be the only cb() call for, eg, snprintf(buf, sz, "").
* Zero lengths can occur at other times (eg "%s" for an empty string).
* Unconditionally write the '\0' byte to reduce code size, it is
* normally overwritten by the data being output.
* There is no point adding a '\0' after copied data - there is always
* another call.
*/
*tgt = '\0';
if (size) {
state->space = space - size;
state->buf = tgt + size;
memcpy(tgt, buf, size);
}
return 0;
}
static __attribute__((unused, format(printf, 3, 0)))
int vsnprintf(char *buf, size_t size, const char *fmt, va_list args)
{
struct __nolibc_sprintf_cb_state state = { .buf = buf, .space = size };
return __nolibc_printf(__nolibc_sprintf_cb, &state, fmt, args);
}
static __attribute__((unused, format(printf, 3, 4)))
int snprintf(char *buf, size_t size, const char *fmt, ...)
{
va_list args;
int ret;
va_start(args, fmt);
ret = vsnprintf(buf, size, fmt, args);
va_end(args);
return ret;
}
static __attribute__((unused, format(printf, 2, 0)))
int vsprintf(char *buf, const char *fmt, va_list args)
{
return vsnprintf(buf, SIZE_MAX, fmt, args);
}
static __attribute__((unused, format(printf, 2, 3)))
int sprintf(char *buf, const char *fmt, ...)
{
va_list args;
int ret;
va_start(args, fmt);
ret = vsprintf(buf, fmt, args);
va_end(args);
return ret;
}
static __attribute__((unused, format(printf, 2, 0)))
int __nolibc_vasprintf(char **strp, const char *fmt, va_list args1, va_list args2)
{
int len1, len2;
char *buf;
len1 = vsnprintf(NULL, 0, fmt, args1);
if (len1 < 0)
return -1;
buf = malloc(len1 + 1);
if (!buf)
return -1;
len2 = vsnprintf(buf, len1 + 1, fmt, args2);
if (len2 < 0) {
free(buf);
return -1;
}
*strp = buf;
return len1;
}
static __attribute__((unused, format(printf, 2, 0)))
int vasprintf(char **strp, const char *fmt, va_list args)
{
va_list args2;
int ret;
va_copy(args2, args);
ret = __nolibc_vasprintf(strp, fmt, args, args2);
va_end(args2);
return ret;
}
static __attribute__((unused, format(printf, 2, 3)))
int asprintf(char **strp, const char *fmt, ...)
{
va_list args;
int ret;
va_start(args, fmt);
ret = vasprintf(strp, fmt, args);
va_end(args);
return ret;
}
static __attribute__((unused))
int vsscanf(const char *str, const char *format, va_list args)
{
uintmax_t uval;
intmax_t ival;
int base;
char *endptr;
int matches;
int lpref;
matches = 0;
while (1) {
if (*format == '%') {
/* start of pattern */
lpref = 0;
format++;
if (*format == 'l') {
/* same as in printf() */
lpref = 1;
format++;
if (*format == 'l') {
lpref = 2;
format++;
}
}
if (*format == '%') {
/* literal % */
if ('%' != *str)
goto done;
str++;
format++;
continue;
} else if (*format == 'd') {
ival = strtoll(str, &endptr, 10);
if (lpref == 0)
*va_arg(args, int *) = ival;
else if (lpref == 1)
*va_arg(args, long *) = ival;
else if (lpref == 2)
*va_arg(args, long long *) = ival;
} else if (*format == 'u' || *format == 'x' || *format == 'X') {
base = *format == 'u' ? 10 : 16;
uval = strtoull(str, &endptr, base);
if (lpref == 0)
*va_arg(args, unsigned int *) = uval;
else if (lpref == 1)
*va_arg(args, unsigned long *) = uval;
else if (lpref == 2)
*va_arg(args, unsigned long long *) = uval;
} else if (*format == 'p') {
*va_arg(args, void **) = (void *)strtoul(str, &endptr, 16);
} else {
SET_ERRNO(EILSEQ);
goto done;
}
format++;
str = endptr;
matches++;
} else if (*format == '\0') {
goto done;
} else if (isspace(*format)) {
/* skip spaces in format and str */
while (isspace(*format))
format++;
while (isspace(*str))
str++;
} else if (*format == *str) {
/* literal match */
format++;
str++;
} else {
if (!matches)
matches = EOF;
goto done;
}
}
done:
return matches;
}
static __attribute__((unused, format(scanf, 2, 3)))
int sscanf(const char *str, const char *format, ...)
{
va_list args;
int ret;
va_start(args, format);
ret = vsscanf(str, format, args);
va_end(args);
return ret;
}
static __attribute__((unused))
void perror(const char *msg)
{
#ifdef NOLIBC_IGNORE_ERRNO
fprintf(stderr, "%s%sunknown error\n", (msg && *msg) ? msg : "", (msg && *msg) ? ": " : "");
#else
fprintf(stderr, "%s%serrno=%d\n", (msg && *msg) ? msg : "", (msg && *msg) ? ": " : "", errno);
#endif
}
static __attribute__((unused))
int setvbuf(FILE *stream __attribute__((unused)),
char *buf __attribute__((unused)),
int mode,
size_t size __attribute__((unused)))
{
/*
* nolibc does not support buffering so this is a nop. Just check mode
* is valid as required by the spec.
*/
switch (mode) {
case _IOFBF:
case _IOLBF:
case _IONBF:
break;
default:
return EOF;
}
return 0;
}
static __attribute__((unused))
int strerror_r(int errnum, char *buf, size_t buflen)
{
if (buflen < 18)
return ERANGE;
__builtin_memcpy(buf, "errno=", 6);
i64toa_r(errnum, buf + 6);
return 0;
}
static __attribute__((unused))
const char *strerror(int errnum)
{
static char buf[18];
char *b = buf;
/* Force gcc to use 'register offset' to access buf[]. */
_NOLIBC_OPTIMIZER_HIDE_VAR(b);
/* Use strerror_r() to avoid having the only .data in small programs. */
strerror_r(errnum, b, sizeof(buf));
return b;
}
#endif /* _NOLIBC_STDIO_H */