mirror of
https://sourceware.org/git/glibc.git
synced 2025-07-30 22:43:12 +03:00
__vfscanf_internal: fix aliasing violation (bug 26690)
As noted in <https://gcc.gnu.org/bugzilla/show_bug.cgi?id=97264>, the cast in the call to the read_int function is an aliasing violation. Change the type of local variable f to a pointer to unsigned, which allows to eliminate most casts while only adding three new ones.
This commit is contained in:
@ -277,7 +277,7 @@ __vfscanf_internal (FILE *s, const char *format, va_list argptr,
|
|||||||
#endif
|
#endif
|
||||||
{
|
{
|
||||||
va_list arg;
|
va_list arg;
|
||||||
const CHAR_T *f = format;
|
const UCHAR_T *f = (const UCHAR_T *) format;
|
||||||
UCHAR_T fc; /* Current character of the format. */
|
UCHAR_T fc; /* Current character of the format. */
|
||||||
WINT_T done = 0; /* Assignments done. */
|
WINT_T done = 0; /* Assignments done. */
|
||||||
size_t read_in = 0; /* Chars read in. */
|
size_t read_in = 0; /* Chars read in. */
|
||||||
@ -415,10 +415,11 @@ __vfscanf_internal (FILE *s, const char *format, va_list argptr,
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef COMPILE_WSCANF
|
#ifndef COMPILE_WSCANF
|
||||||
if (!isascii ((unsigned char) *f))
|
if (!isascii (*f))
|
||||||
{
|
{
|
||||||
/* Non-ASCII, may be a multibyte. */
|
/* Non-ASCII, may be a multibyte. */
|
||||||
int len = __mbrlen (f, strlen (f), &state);
|
int len = __mbrlen ((const char *) f, strlen ((const char *) f),
|
||||||
|
&state);
|
||||||
if (len > 0)
|
if (len > 0)
|
||||||
{
|
{
|
||||||
do
|
do
|
||||||
@ -426,7 +427,7 @@ __vfscanf_internal (FILE *s, const char *format, va_list argptr,
|
|||||||
c = inchar ();
|
c = inchar ();
|
||||||
if (__glibc_unlikely (c == EOF))
|
if (__glibc_unlikely (c == EOF))
|
||||||
input_error ();
|
input_error ();
|
||||||
else if (c != (unsigned char) *f++)
|
else if (c != *f++)
|
||||||
{
|
{
|
||||||
ungetc_not_eof (c, s);
|
ungetc_not_eof (c, s);
|
||||||
conv_error ();
|
conv_error ();
|
||||||
@ -484,9 +485,9 @@ __vfscanf_internal (FILE *s, const char *format, va_list argptr,
|
|||||||
char_buffer_rewind (&charbuf);
|
char_buffer_rewind (&charbuf);
|
||||||
|
|
||||||
/* Check for a positional parameter specification. */
|
/* Check for a positional parameter specification. */
|
||||||
if (ISDIGIT ((UCHAR_T) *f))
|
if (ISDIGIT (*f))
|
||||||
{
|
{
|
||||||
argpos = read_int ((const UCHAR_T **) &f);
|
argpos = read_int (&f);
|
||||||
if (*f == L_('$'))
|
if (*f == L_('$'))
|
||||||
++f;
|
++f;
|
||||||
else
|
else
|
||||||
@ -521,8 +522,8 @@ __vfscanf_internal (FILE *s, const char *format, va_list argptr,
|
|||||||
|
|
||||||
/* Find the maximum field width. */
|
/* Find the maximum field width. */
|
||||||
width = 0;
|
width = 0;
|
||||||
if (ISDIGIT ((UCHAR_T) *f))
|
if (ISDIGIT (*f))
|
||||||
width = read_int ((const UCHAR_T **) &f);
|
width = read_int (&f);
|
||||||
got_width:
|
got_width:
|
||||||
if (width == 0)
|
if (width == 0)
|
||||||
width = -1;
|
width = -1;
|
||||||
@ -2522,12 +2523,11 @@ __vfscanf_internal (FILE *s, const char *format, va_list argptr,
|
|||||||
}
|
}
|
||||||
|
|
||||||
while ((fc = *f++) != '\0' && fc != ']')
|
while ((fc = *f++) != '\0' && fc != ']')
|
||||||
if (fc == '-' && *f != '\0' && *f != ']'
|
if (fc == '-' && *f != '\0' && *f != ']' && f[-2] <= *f)
|
||||||
&& (unsigned char) f[-2] <= (unsigned char) *f)
|
|
||||||
{
|
{
|
||||||
/* Add all characters from the one before the '-'
|
/* Add all characters from the one before the '-'
|
||||||
up to (but not including) the next format char. */
|
up to (but not including) the next format char. */
|
||||||
for (fc = (unsigned char) f[-2]; fc < (unsigned char) *f; ++fc)
|
for (fc = f[-2]; fc < *f; ++fc)
|
||||||
((char *)charbuf.scratch.data)[fc] = 1;
|
((char *)charbuf.scratch.data)[fc] = 1;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
Reference in New Issue
Block a user