|
Message-Id: <ae4f840bb48367d0a9aabeb5f12102caef2c71e9.1685536319.git.Jens.Gustedt@inria.fr> Date: Wed, 31 May 2023 16:05:43 +0200 From: Jens Gustedt <Jens.Gustedt@...ia.fr> To: musl@...ts.openwall.com Subject: [C23 scanf 3/3] C23: implement wN and wfN specifiers for scanf functions --- src/stdio/vfscanf.c | 30 ++++++++++++++++++++++++++++++ src/stdio/vfwscanf.c | 30 ++++++++++++++++++++++++++++++ 2 files changed, 60 insertions(+) diff --git a/src/stdio/vfscanf.c b/src/stdio/vfscanf.c index f72ac9c9..f8ace92f 100644 --- a/src/stdio/vfscanf.c +++ b/src/stdio/vfscanf.c @@ -41,6 +41,15 @@ static void store_int(void *dest, int size, unsigned long long i) } } +static int getint(unsigned char const**s) { + int i=0; + if (**s != '0') for (; isdigit(**s); (*s)++) { + if (i > INT_MAX/10U || **s-'0' > INT_MAX-10*i) i = -1; + else i = 10*i + (**s-'0'); + } + return i; +} + static void *arg_n(va_list ap, unsigned int n) { void *p; @@ -57,6 +66,7 @@ int vfscanf(FILE *restrict f, const char *restrict fmt, va_list ap) { int width; int size; + int fast16; int alloc = 0; int base; const unsigned char *p; @@ -150,6 +160,26 @@ int vfscanf(FILE *restrict f, const char *restrict fmt, va_list ap) case 'L': size = SIZE_L; break; + case 'w': + // See if "fast" is requested. Difference is only + // relevant for a fast type of minimum width 16. + fast16 = SIZE_h; + if (*p == 'f') { + fast16 = SIZE_def; + ++p; + } + switch (getint(&p)) { + default: goto fmt_fail; + case 8: size = SIZE_hh; break; + case 16: size = fast16; break; + case 32: size = SIZE_def; break; +#if UINTPTR_MAX >= UINT64_MAX + case 64: size = SIZE_l; break; +#else + case 64: size = SIZE_ll; break; +#endif + } + break; case 'b': case 'd': case 'i': case 'o': case 'u': case 'x': case 'a': case 'e': case 'f': case 'g': case 'A': case 'E': case 'F': case 'G': case 'X': diff --git a/src/stdio/vfwscanf.c b/src/stdio/vfwscanf.c index 17f5a2f9..6fe2749c 100644 --- a/src/stdio/vfwscanf.c +++ b/src/stdio/vfwscanf.c @@ -41,6 +41,15 @@ static void store_int(void *dest, int size, unsigned long long i) } } +static int getint(wchar_t const**s) { + int i=0; + if (**s != L'0') for (i=0; iswdigit(**s); (*s)++) { + if (i > INT_MAX/10U || **s-L'0' > INT_MAX-10*i) i = -1; + else i = 10*i + (**s-L'0'); + } + return i; +} + static void *arg_n(va_list ap, unsigned int n) { void *p; @@ -87,6 +96,7 @@ int vfwscanf(FILE *restrict f, const wchar_t *restrict fmt, va_list ap) { int width; int size; + int fast16; int alloc; const wchar_t *p; int c, t; @@ -173,6 +183,26 @@ int vfwscanf(FILE *restrict f, const wchar_t *restrict fmt, va_list ap) case 'L': size = SIZE_L; break; + case 'w': + // See if "fast" is requested. Difference is only + // relevant for a fast type of minimum width 16. + fast16 = SIZE_h; + if (*p == 'f') { + fast16 = SIZE_def; + ++p; + } + switch (getint(&p)) { + default: goto fmt_fail; + case 8: size = SIZE_hh; break; + case 16: size = fast16; break; + case 32: size = SIZE_def; break; +#if UINTPTR_MAX >= UINT64_MAX + case 64: size = SIZE_l; break; +#else + case 64: size = SIZE_ll; break; +#endif + } + break; case 'b': case 'd': case 'i': case 'o': case 'u': case 'x': case 'a': case 'e': case 'f': case 'g': case 'A': case 'E': case 'F': case 'G': case 'X': -- 2.34.1
Powered by blists - more mailing lists
Confused about mailing lists and their use? Read about mailing lists on Wikipedia and check out these guidelines on proper formatting of your messages.