tst-mbswcs4 \
tst-mbswcs5 \
tst-mbswcs6 \
+ tst-scanf-width-digit \
+ tst-scanf-width-point \
tst-setlocale \
tst-setlocale2 \
tst-setlocale3 \
--- /dev/null
+/* Verify multibyte digit extending beyond scanf field width.
+ Copyright (C) 2025 Free Software Foundation, Inc.
+ This file is part of the GNU C Library.
+
+ The GNU C Library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ The GNU C Library is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Lesser General Public
+ License along with the GNU C Library; if not, see
+ <https://www.gnu.org/licenses/>. */
+
+#include <locale.h>
+#include <stdio.h>
+#include <string.h>
+
+#include <libc-diag.h>
+#include <support/check.h>
+
+#define P1 "\xdb\xb1"
+#define P2 "\xdb\xb2"
+
+static int
+do_test (void)
+{
+ if (setlocale (LC_ALL, "fa_IR.UTF-8") == NULL)
+ FAIL_EXIT1 ("setlocale (LC_ALL, \"fa_IR.UTF-8\")");
+
+ char s[] = P1 P2;
+ FILE *f = fmemopen (s, strlen (s), "r");
+ if (f == NULL)
+ FAIL_EXIT1 ("fmemopen: %m");
+
+ /* Avoid: "warning: 'I' flag used with '%f' gnu_scanf format [-Wformat=]";
+ cf. GCC PR c/119514. */
+ DIAG_PUSH_NEEDS_COMMENT;
+ DIAG_IGNORE_NEEDS_COMMENT (4.9, "-Wformat");
+
+ /* This should succeed parsing a floating-point number, and leave '\xdb',
+ '\xb2' in the input. */
+ double d;
+ int c;
+ TEST_VERIFY_EXIT (fscanf (f, "%I3lf%n", &d, &c) == 1);
+ TEST_VERIFY_EXIT (d == 1.0);
+ TEST_VERIFY_EXIT (c == 2);
+ TEST_VERIFY_EXIT (fgetc (f) == 0xdb);
+ TEST_VERIFY_EXIT (fgetc (f) == 0xb2);
+
+ DIAG_POP_NEEDS_COMMENT;
+
+ return 0;
+}
+
+#include <support/test-driver.c>
--- /dev/null
+/* Verify multibyte decimal point extending beyond scanf field width.
+ Copyright (C) 2025 Free Software Foundation, Inc.
+ This file is part of the GNU C Library.
+
+ The GNU C Library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Lesser General Public
+ License as published by the Free Software Foundation; either
+ version 2.1 of the License, or (at your option) any later version.
+
+ The GNU C Library is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ Lesser General Public License for more details.
+
+ You should have received a copy of the GNU Lesser General Public
+ License along with the GNU C Library; if not, see
+ <https://www.gnu.org/licenses/>. */
+
+#include <locale.h>
+#include <stdio.h>
+#include <string.h>
+
+#include <libc-diag.h>
+#include <support/check.h>
+
+#define PD "\xd9\xab"
+
+static int
+do_test (void)
+{
+ if (setlocale (LC_ALL, "ps_AF.UTF-8") == NULL)
+ FAIL_EXIT1 ("setlocale (LC_ALL, \"ps_AF.UTF-8\")");
+
+ char s[] = "1" PD;
+ FILE *f = fmemopen (s, strlen (s), "r");
+ if (f == NULL)
+ FAIL_EXIT1 ("fmemopen: %m");
+
+ /* This should succeed parsing a floating-point number, and leave '\xd9',
+ '\xab' in the input. */
+ double d;
+ int c;
+ TEST_VERIFY_EXIT (fscanf (f, "%2lf%n", &d, &c) == 1);
+ TEST_VERIFY_EXIT (d == 1.0);
+ TEST_VERIFY_EXIT (c == 1);
+ TEST_VERIFY_EXIT (fgetc (f) == 0xd9);
+ TEST_VERIFY_EXIT (fgetc (f) == 0xab);
+
+ return 0;
+}
+
+#include <support/test-driver.c>
(void) (c != EOF \
? ++read_in \
: (size_t) (inchar_errno = errno)), c))
+/* Same as INCHAR, but stop upon field exhaustion according to AVAIL. */
+# define inchar_in_field(avail) \
+({ \
+ if (avail == 0) \
+ c = EOF; \
+ else \
+ inchar (); \
+ c; \
+})
# define ISSPACE(Ch) __isspace_l (Ch, loc)
# define ISDIGIT(Ch) __isdigit_l (Ch, loc)
# define ISXDIGIT(Ch) __isxdigit_l (Ch, loc)
++wcdigits[n];
#else
const char *cmpp;
- int avail = width > 0 ? width : INT_MAX;
+ int avail = width >= 0 ? width : INT_MAX;
if (__glibc_unlikely (map != NULL))
mbdigits[n] = digits_extended[n];
break;
else
{
- if (avail == 0 || inchar () == EOF)
+ if (inchar_in_field (avail) == EOF)
break;
--avail;
}
++wcdigits[n];
#else
const char *cmpp;
- int avail = width > 0 ? width : INT_MAX;
+ int avail = width >= 0 ? width : INT_MAX;
cmpp = mbdigits[n];
while ((unsigned char) *cmpp == c && avail >= 0)
break;
else
{
- if (avail == 0 || inchar () == EOF)
+ if (inchar_in_field (avail) == EOF)
break;
--avail;
}
break;
#else
const char *cmpp = thousands;
- int avail = width > 0 ? width : INT_MAX;
+ int avail = width >= 0 ? width : INT_MAX;
while ((unsigned char) *cmpp == c && avail >= 0)
{
break;
else
{
- if (avail == 0 || inchar () == EOF)
+ if (inchar_in_field (avail) == EOF)
break;
--avail;
}
break;
#else
const char *cmpp = thousands;
- int avail = width > 0 ? width : INT_MAX;
+ int avail = width >= 0 ? width : INT_MAX;
while ((unsigned char) *cmpp == c && avail >= 0)
{
break;
else
{
- if (avail == 0 || inchar () == EOF)
+ if (inchar_in_field (avail) == EOF)
break;
--avail;
}
}
#else
const char *cmpp = decimal;
- int avail = width > 0 ? width : INT_MAX;
+ int avail = width >= 0 ? width : INT_MAX;
if (! got_dot)
{
}
#else
const char *cmpp = mbdigits[n];
- int avail = width > 0 ? width : INT_MAX;
+ int avail = width >= 0 ? width : INT_MAX;
while ((unsigned char) *cmpp == c && avail >= 0)
if (*++cmpp == '\0')
break;
else
{
- if (avail == 0 || inchar () == EOF)
+ if (inchar_in_field (avail) == EOF)
break;
--avail;
}