Merge "Upgrade to current upstream scanf implementation."

This commit is contained in:
Elliott Hughes 2014-03-13 00:49:23 +00:00 committed by Gerrit Code Review
commit 01cd78854a
5 changed files with 219 additions and 40 deletions

View File

@ -80,10 +80,7 @@ libc_common_src_files := \
stdio/fvwrite.c \ stdio/fvwrite.c \
stdio/snprintf.c\ stdio/snprintf.c\
stdio/sprintf.c \ stdio/sprintf.c \
stdio/sscanf.c \
stdio/vfprintf.c \ stdio/vfprintf.c \
stdio/vfscanf.c \
stdio/vsscanf.c \
stdlib/atexit.c \ stdlib/atexit.c \
stdlib/ctype_.c \ stdlib/ctype_.c \
stdlib/getenv.c \ stdlib/getenv.c \
@ -362,15 +359,18 @@ libc_upstream_openbsd_src_files := \
upstream-openbsd/lib/libc/stdio/scanf.c \ upstream-openbsd/lib/libc/stdio/scanf.c \
upstream-openbsd/lib/libc/stdio/setbuf.c \ upstream-openbsd/lib/libc/stdio/setbuf.c \
upstream-openbsd/lib/libc/stdio/setbuffer.c \ upstream-openbsd/lib/libc/stdio/setbuffer.c \
upstream-openbsd/lib/libc/stdio/sscanf.c \
upstream-openbsd/lib/libc/stdio/stdio.c \ upstream-openbsd/lib/libc/stdio/stdio.c \
upstream-openbsd/lib/libc/stdio/tempnam.c \ upstream-openbsd/lib/libc/stdio/tempnam.c \
upstream-openbsd/lib/libc/stdio/tmpnam.c \ upstream-openbsd/lib/libc/stdio/tmpnam.c \
upstream-openbsd/lib/libc/stdio/ungetc.c \ upstream-openbsd/lib/libc/stdio/ungetc.c \
upstream-openbsd/lib/libc/stdio/vasprintf.c \ upstream-openbsd/lib/libc/stdio/vasprintf.c \
upstream-openbsd/lib/libc/stdio/vfscanf.c \
upstream-openbsd/lib/libc/stdio/vprintf.c \ upstream-openbsd/lib/libc/stdio/vprintf.c \
upstream-openbsd/lib/libc/stdio/vscanf.c \ upstream-openbsd/lib/libc/stdio/vscanf.c \
upstream-openbsd/lib/libc/stdio/vsnprintf.c \ upstream-openbsd/lib/libc/stdio/vsnprintf.c \
upstream-openbsd/lib/libc/stdio/vsprintf.c \ upstream-openbsd/lib/libc/stdio/vsprintf.c \
upstream-openbsd/lib/libc/stdio/vsscanf.c \
upstream-openbsd/lib/libc/stdio/wbuf.c \ upstream-openbsd/lib/libc/stdio/wbuf.c \
upstream-openbsd/lib/libc/stdlib/atoi.c \ upstream-openbsd/lib/libc/stdlib/atoi.c \
upstream-openbsd/lib/libc/stdlib/atol.c \ upstream-openbsd/lib/libc/stdlib/atol.c \

View File

@ -1,4 +1,4 @@
/* $OpenBSD: sscanf.c,v 1.12 2005/08/08 08:05:36 espie Exp $ */ /* $OpenBSD: sscanf.c,v 1.14 2011/11/08 18:30:42 guenther Exp $ */
/*- /*-
* Copyright (c) 1990, 1993 * Copyright (c) 1990, 1993
* The Regents of the University of California. All rights reserved. * The Regents of the University of California. All rights reserved.
@ -44,6 +44,7 @@ eofread(void *cookie, char *buf, int len)
return (0); return (0);
} }
/* SCANFLIKE2 */
int int
sscanf(const char *str, const char *fmt, ...) sscanf(const char *str, const char *fmt, ...)
{ {
@ -59,7 +60,7 @@ sscanf(const char *str, const char *fmt, ...)
f._read = eofread; f._read = eofread;
f._lb._base = NULL; f._lb._base = NULL;
va_start(ap, fmt); va_start(ap, fmt);
ret = vfscanf(&f, fmt, ap); ret = __svfscanf(&f, fmt, ap);
va_end(ap); va_end(ap);
return (ret); return (ret);
} }

View File

@ -1,4 +1,4 @@
/* $OpenBSD: vfscanf.c,v 1.21 2006/01/13 21:33:28 millert Exp $ */ /* $OpenBSD: vfscanf.c,v 1.30 2013/04/17 17:40:35 tedu Exp $ */
/*- /*-
* Copyright (c) 1990, 1993 * Copyright (c) 1990, 1993
* The Regents of the University of California. All rights reserved. * The Regents of the University of California. All rights reserved.
@ -32,11 +32,13 @@
*/ */
#include <ctype.h> #include <ctype.h>
#include <wctype.h>
#include <inttypes.h> #include <inttypes.h>
#include <stdarg.h> #include <stdarg.h>
#include <stddef.h> #include <stddef.h>
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
#include <string.h>
#include "local.h" #include "local.h"
#ifdef FLOATING_POINT #ifdef FLOATING_POINT
@ -49,7 +51,7 @@
* Flags used during conversion. * Flags used during conversion.
*/ */
#define LONG 0x00001 /* l: long or double */ #define LONG 0x00001 /* l: long or double */
#define LONGDBL 0x00002 /* L: long double; unimplemented */ #define LONGDBL 0x00002 /* L: long double */
#define SHORT 0x00004 /* h: short */ #define SHORT 0x00004 /* h: short */
#define SHORTSHORT 0x00008 /* hh: 8 bit integer */ #define SHORTSHORT 0x00008 /* hh: 8 bit integer */
#define LLONG 0x00010 /* ll: long long (+ deprecated q: quad) */ #define LLONG 0x00010 /* ll: long long (+ deprecated q: quad) */
@ -90,15 +92,11 @@
static u_char *__sccl(char *, u_char *); static u_char *__sccl(char *, u_char *);
#if !defined(VFSCANF)
#define VFSCANF vfscanf
#endif
/* /*
* vfscanf * Internal, unlocked version of vfscanf
*/ */
int int
VFSCANF(FILE *fp, const char *fmt0, __va_list ap) __svfscanf(FILE *fp, const char *fmt0, __va_list ap)
{ {
u_char *fmt = (u_char *)fmt0; u_char *fmt = (u_char *)fmt0;
int c; /* character from format, or conversion */ int c; /* character from format, or conversion */
@ -112,12 +110,16 @@ VFSCANF(FILE *fp, const char *fmt0, __va_list ap)
int base; /* base argument to strtoimax/strtouimax */ int base; /* base argument to strtoimax/strtouimax */
char ccltab[256]; /* character class table for %[...] */ char ccltab[256]; /* character class table for %[...] */
char buf[BUF]; /* buffer for numeric conversions */ char buf[BUF]; /* buffer for numeric conversions */
#ifdef SCANF_WIDE_CHAR
wchar_t *wcp; /* handy wide character pointer */
size_t nconv; /* length of multibyte sequence converted */
mbstate_t mbs;
#endif
/* `basefix' is used to avoid `if' tests in the integer scanner */ /* `basefix' is used to avoid `if' tests in the integer scanner */
static short basefix[17] = static short basefix[17] =
{ 10, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 }; { 10, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16 };
FLOCKFILE(fp);
_SET_ORIENTATION(fp, -1); _SET_ORIENTATION(fp, -1);
nassigned = 0; nassigned = 0;
@ -125,10 +127,8 @@ VFSCANF(FILE *fp, const char *fmt0, __va_list ap)
base = 0; /* XXX just to keep gcc happy */ base = 0; /* XXX just to keep gcc happy */
for (;;) { for (;;) {
c = *fmt++; c = *fmt++;
if (c == 0) { if (c == 0)
FUNLOCKFILE(fp);
return (nassigned); return (nassigned);
}
if (isspace(c)) { if (isspace(c)) {
while ((fp->_r > 0 || __srefill(fp) == 0) && while ((fp->_r > 0 || __srefill(fp) == 0) &&
isspace(*fp->_p)) isspace(*fp->_p))
@ -162,13 +162,7 @@ literal:
flags |= MAXINT; flags |= MAXINT;
goto again; goto again;
case 'L': case 'L':
flags |= flags |= LONGDBL;
(*fmt == 'd') ? LLONG :
(*fmt == 'i') ? LLONG :
(*fmt == 'o') ? LLONG :
(*fmt == 'u') ? LLONG :
(*fmt == 'x') ? LLONG :
LONGDBL;
goto again; goto again;
case 'h': case 'h':
if (*fmt == 'h') { if (*fmt == 'h') {
@ -245,11 +239,10 @@ literal:
break; break;
#ifdef FLOATING_POINT #ifdef FLOATING_POINT
case 'E': case 'e': case 'E':
case 'G': case 'f': case 'F':
case 'e': case 'g': case 'G':
case 'f': case 'a': case 'A':
case 'g':
c = CT_FLOAT; c = CT_FLOAT;
break; break;
#endif #endif
@ -301,7 +294,6 @@ literal:
* Disgusting backwards compatibility hacks. XXX * Disgusting backwards compatibility hacks. XXX
*/ */
case '\0': /* compat */ case '\0': /* compat */
FUNLOCKFILE(fp);
return (EOF); return (EOF);
default: /* compat */ default: /* compat */
@ -346,10 +338,52 @@ literal:
/* scan arbitrary characters (sets NOSKIP) */ /* scan arbitrary characters (sets NOSKIP) */
if (width == 0) if (width == 0)
width = 1; width = 1;
#ifdef SCANF_WIDE_CHAR
if (flags & LONG) {
if ((flags & SUPPRESS) == 0)
wcp = va_arg(ap, wchar_t *);
else
wcp = NULL;
n = 0;
while (width != 0) {
if (n == MB_CUR_MAX) {
fp->_flags |= __SERR;
goto input_failure;
}
buf[n++] = *fp->_p;
fp->_p++;
fp->_r--;
bzero(&mbs, sizeof(mbs));
nconv = mbrtowc(wcp, buf, n, &mbs);
if (nconv == (size_t)-1) {
fp->_flags |= __SERR;
goto input_failure;
}
if (nconv == 0 && !(flags & SUPPRESS))
*wcp = L'\0';
if (nconv != (size_t)-2) {
nread += n;
width--;
if (!(flags & SUPPRESS))
wcp++;
n = 0;
}
if (fp->_r <= 0 && __srefill(fp)) {
if (n != 0) {
fp->_flags |= __SERR;
goto input_failure;
}
break;
}
}
if (!(flags & SUPPRESS))
nassigned++;
} else
#endif /* SCANF_WIDE_CHAR */
if (flags & SUPPRESS) { if (flags & SUPPRESS) {
size_t sum = 0; size_t sum = 0;
for (;;) { for (;;) {
if ((n = fp->_r) < (int)width) { if ((n = fp->_r) < width) {
sum += n; sum += n;
width -= n; width -= n;
fp->_p += n; fp->_p += n;
@ -381,6 +415,72 @@ literal:
/* scan a (nonempty) character class (sets NOSKIP) */ /* scan a (nonempty) character class (sets NOSKIP) */
if (width == 0) if (width == 0)
width = (size_t)~0; /* `infinity' */ width = (size_t)~0; /* `infinity' */
#ifdef SCANF_WIDE_CHAR
/* take only those things in the class */
if (flags & LONG) {
wchar_t twc;
int nchars;
if ((flags & SUPPRESS) == 0)
wcp = va_arg(ap, wchar_t *);
else
wcp = &twc;
n = 0;
nchars = 0;
while (width != 0) {
if (n == MB_CUR_MAX) {
fp->_flags |= __SERR;
goto input_failure;
}
buf[n++] = *fp->_p;
fp->_p++;
fp->_r--;
bzero(&mbs, sizeof(mbs));
nconv = mbrtowc(wcp, buf, n, &mbs);
if (nconv == (size_t)-1) {
fp->_flags |= __SERR;
goto input_failure;
}
if (nconv == 0)
*wcp = L'\0';
if (nconv != (size_t)-2) {
if (wctob(*wcp) != EOF &&
!ccltab[wctob(*wcp)]) {
while (n != 0) {
n--;
ungetc(buf[n],
fp);
}
break;
}
nread += n;
width--;
if (!(flags & SUPPRESS))
wcp++;
nchars++;
n = 0;
}
if (fp->_r <= 0 && __srefill(fp)) {
if (n != 0) {
fp->_flags |= __SERR;
goto input_failure;
}
break;
}
}
if (n != 0) {
fp->_flags |= __SERR;
goto input_failure;
}
n = nchars;
if (n == 0)
goto match_failure;
if (!(flags & SUPPRESS)) {
*wcp = L'\0';
nassigned++;
}
} else
#endif /* SCANF_WIDE_CHAR */
/* take only those things in the class */ /* take only those things in the class */
if (flags & SUPPRESS) { if (flags & SUPPRESS) {
n = 0; n = 0;
@ -422,6 +522,60 @@ literal:
/* like CCL, but zero-length string OK, & no NOSKIP */ /* like CCL, but zero-length string OK, & no NOSKIP */
if (width == 0) if (width == 0)
width = (size_t)~0; width = (size_t)~0;
#ifdef SCANF_WIDE_CHAR
if (flags & LONG) {
wchar_t twc;
if ((flags & SUPPRESS) == 0)
wcp = va_arg(ap, wchar_t *);
else
wcp = &twc;
n = 0;
while (!isspace(*fp->_p) && width != 0) {
if (n == MB_CUR_MAX) {
fp->_flags |= __SERR;
goto input_failure;
}
buf[n++] = *fp->_p;
fp->_p++;
fp->_r--;
bzero(&mbs, sizeof(mbs));
nconv = mbrtowc(wcp, buf, n, &mbs);
if (nconv == (size_t)-1) {
fp->_flags |= __SERR;
goto input_failure;
}
if (nconv == 0)
*wcp = L'\0';
if (nconv != (size_t)-2) {
if (iswspace(*wcp)) {
while (n != 0) {
n--;
ungetc(buf[n],
fp);
}
break;
}
nread += n;
width--;
if (!(flags & SUPPRESS))
wcp++;
n = 0;
}
if (fp->_r <= 0 && __srefill(fp)) {
if (n != 0) {
fp->_flags |= __SERR;
goto input_failure;
}
break;
}
}
if (!(flags & SUPPRESS)) {
*wcp = L'\0';
nassigned++;
}
} else
#endif /* SCANF_WIDE_CHAR */
if (flags & SUPPRESS) { if (flags & SUPPRESS) {
n = 0; n = 0;
while (!isspace(*fp->_p)) { while (!isspace(*fp->_p)) {
@ -681,16 +835,18 @@ literal:
(void) ungetc(c, fp); (void) ungetc(c, fp);
} }
if ((flags & SUPPRESS) == 0) { if ((flags & SUPPRESS) == 0) {
double res;
*p = '\0'; *p = '\0';
res = strtod(buf, (char **) NULL); if (flags & LONGDBL) {
if (flags & LONGDBL) long double res = strtold(buf,
(char **)NULL);
*va_arg(ap, long double *) = res; *va_arg(ap, long double *) = res;
else if (flags & LONG) } else if (flags & LONG) {
double res = strtod(buf, (char **)NULL);
*va_arg(ap, double *) = res; *va_arg(ap, double *) = res;
else } else {
float res = strtof(buf, (char **)NULL);
*va_arg(ap, float *) = res; *va_arg(ap, float *) = res;
}
nassigned++; nassigned++;
} }
nread += p - buf; nread += p - buf;
@ -702,7 +858,6 @@ input_failure:
if (nassigned == 0) if (nassigned == 0)
nassigned = -1; nassigned = -1;
match_failure: match_failure:
FUNLOCKFILE(fp);
return (nassigned); return (nassigned);
} }
@ -801,3 +956,14 @@ doswitch:
} }
/* NOTREACHED */ /* NOTREACHED */
} }
int
vfscanf(FILE *fp, const char *fmt0, __va_list ap)
{
int r;
FLOCKFILE(fp);
r = __svfscanf(fp, fmt0, ap);
FUNLOCKFILE(fp);
return (r);
}

View File

@ -1,4 +1,4 @@
/* $OpenBSD: vsscanf.c,v 1.11 2006/01/06 18:53:04 millert Exp $ */ /* $OpenBSD: vsscanf.c,v 1.12 2011/11/08 18:30:42 guenther Exp $ */
/*- /*-
* Copyright (c) 1990, 1993 * Copyright (c) 1990, 1993
* The Regents of the University of California. All rights reserved. * The Regents of the University of California. All rights reserved.
@ -39,6 +39,7 @@
static int static int
eofread(void *cookie, char *buf, int len) eofread(void *cookie, char *buf, int len)
{ {
return (0); return (0);
} }
@ -54,5 +55,5 @@ vsscanf(const char *str, const char *fmt, __va_list ap)
f._bf._size = f._r = strlen(str); f._bf._size = f._r = strlen(str);
f._read = eofread; f._read = eofread;
f._lb._base = NULL; f._lb._base = NULL;
return (vfscanf(&f, fmt, ap)); return (__svfscanf(&f, fmt, ap));
} }

View File

@ -372,3 +372,14 @@ TEST(stdio, putc) {
ASSERT_EQ(EOF, putc('x', fp)); ASSERT_EQ(EOF, putc('x', fp));
fclose(fp); fclose(fp);
} }
TEST(stdio, sscanf) {
char s1[123];
int i1;
double d1;
char s2[123];
ASSERT_EQ(3, sscanf(" hello 123 1.23 ", "%s %i %lf %s", s1, &i1, &d1, s2));
ASSERT_STREQ("hello", s1);
ASSERT_EQ(123, i1);
ASSERT_EQ(1.23, d1);
}