diff options
author | Rich Felker <dalias@aerifal.cx> | 2013-06-05 16:53:26 -0400 |
---|---|---|
committer | Rich Felker <dalias@aerifal.cx> | 2013-06-05 16:53:26 -0400 |
commit | de80ea9f1c2821cbb4205533b86d5d17f9e8d376 (patch) | |
tree | 8bccf53da0eee966a1946e2fb710937a387a0a39 /src/stdio | |
parent | 1ab59de81e94e7802f85d314a709f8350a0e9b65 (diff) | |
download | musl-de80ea9f1c2821cbb4205533b86d5d17f9e8d376.tar.gz musl-de80ea9f1c2821cbb4205533b86d5d17f9e8d376.tar.xz musl-de80ea9f1c2821cbb4205533b86d5d17f9e8d376.zip |
refactor wide-char scanf string handling
this brings the wide version of the code into alignment with the byte-based version, in preparation for adding support for the m (malloc) modifier.
Diffstat (limited to 'src/stdio')
-rw-r--r-- | src/stdio/vfwscanf.c | 87 |
1 files changed, 32 insertions, 55 deletions
diff --git a/src/stdio/vfwscanf.c b/src/stdio/vfwscanf.c index 44910b4c..45de3372 100644 --- a/src/stdio/vfwscanf.c +++ b/src/stdio/vfwscanf.c @@ -102,6 +102,7 @@ int vfwscanf(FILE *restrict f, const wchar_t *restrict fmt, va_list ap) off_t pos = 0, cnt; static const char size_pfx[][3] = { "hh", "h", "", "l", "L", "ll" }; char tmp[3*sizeof(int)+10]; + const wchar_t *set; FLOCK(f); @@ -179,8 +180,11 @@ int vfwscanf(FILE *restrict f, const wchar_t *restrict fmt, va_list ap) t = *p; - /* Transform ls,lc -> S,C */ - if (size==SIZE_l && (t&15)==3) t&=~32; + /* Transform S,C -> ls,lc */ + if ((t&0x2f)==3) { + size = SIZE_l; + t |= 32; + } if (t != 'n') { if (t != '[' && (t|32) != 'c') @@ -197,63 +201,41 @@ int vfwscanf(FILE *restrict f, const wchar_t *restrict fmt, va_list ap) /* do not increment match count, etc! */ continue; - case 'c': - if (width < 1) width = 1; - s = dest; - for (; width && (c=getwc(f)) >= 0; width--) { - int l = wctomb(s?s:tmp, c); - if (l<0) goto input_fail; - if (s) s+=l; - pos++; - } - if (width) goto match_fail; - break; - - case 'C': - if (width < 1) width = 1; - wcs = dest; - for (; width && (c=getwc(f)) >= 0; width--) - pos++, wcs && (*wcs++ = c); - if (width) goto match_fail; - break; - case 's': - if (width < 1) width = -1; - s = dest; - while (width && !iswspace(c=getwc(f)) && c!=EOF) { - int l = wctomb(s?s:tmp, c); - if (l<0) goto input_fail; - if (s) s+=l; - pos++; - width-=(width>0); + case 'c': + case '[': + if (t == 'c') { + if (width<1) width = 1; + invert = 1; + set = L""; + } else if (t == 's') { + invert = 1; + set = (const wchar_t[]){ + ' ', '\t', '\n', '\r', 11, 12, 0x0085, + 0x2000, 0x2001, 0x2002, 0x2003, 0x2004, 0x2005, + 0x2006, 0x2008, 0x2009, 0x200a, + 0x2028, 0x2029, 0x205f, 0x3000, 0 }; + } else { + if (*++p == '^') p++, invert = 1; + else invert = 0; + set = p; + if (*p==']') p++; + while (*p!=']') { + if (!*p) goto fmt_fail; + p++; + } } - if (width) ungetwc(c, f); - if (s) *s = 0; - break; - - case 'S': - wcs = dest; - if (width < 1) width = -1; - while (width && !iswspace(c=getwc(f)) && c!=EOF) - width-=(width>0), pos++, *wcs++ = c; - if (width) ungetwc(c, f); - if (wcs) *wcs = 0; - break; - case '[': s = (size == SIZE_def) ? dest : 0; wcs = (size == SIZE_l) ? dest : 0; - if (*++p == '^') p++, invert = 1; - else invert = 0; - int gotmatch = 0; if (width < 1) width = -1; while (width) { if ((c=getwc(f))<0) break; - if (in_set(p, c) == invert) + if (in_set(set, c) == invert) break; if (wcs) { *wcs++ = c; @@ -266,14 +248,9 @@ int vfwscanf(FILE *restrict f, const wchar_t *restrict fmt, va_list ap) width-=(width>0); gotmatch=1; } - if (width) ungetwc(c, f); - - if (!gotmatch) goto match_fail; - - if (*p==']') p++; - while (*p!=']') { - if (!*p) goto fmt_fail; - p++; + if (width) { + ungetwc(c, f); + if (t == 'c' || !gotmatch) goto match_fail; } if (wcs) *wcs++ = 0; |