mirror of
https://git.busybox.net/uClibc
synced 2025-05-08 23:02:28 +08:00

Unlike vsnprintf, vswprintf does not properly initialize locking elements of FILE structure, which in some unfortunate cases can result in lockups in _vfwprintf_internal. Interesting, the initialization code was removed in 2a915734a32c5aec9a6a76c13bcb074d30e64171 at the same time as it was added to vsnprintf. Signed-off-by: Maksim Rayskiy <mrayskiy@broadcom.com> Signed-off-by: Carmelo Amoroso <carmelo.amoroso@st.com>
76 lines
1.8 KiB
C
76 lines
1.8 KiB
C
/* Copyright (C) 2004 Manuel Novoa III <mjn3@codepoet.org>
|
|
*
|
|
* GNU Library General Public License (LGPL) version 2 or later.
|
|
*
|
|
* Dedicated to Toni. See uClibc/DEDICATION.mjn3 for details.
|
|
*/
|
|
|
|
#include "_stdio.h"
|
|
#include <stdarg.h>
|
|
#include <wchar.h>
|
|
|
|
|
|
/* NB: this file is not used if __USE_OLD_VFPRINTF__ */
|
|
|
|
#ifndef __STDIO_BUFFERS
|
|
#warning Skipping vswprintf since no buffering!
|
|
#else /* __STDIO_BUFFERS */
|
|
|
|
int vswprintf(wchar_t *__restrict buf, size_t size,
|
|
const wchar_t * __restrict format, va_list arg)
|
|
{
|
|
FILE f;
|
|
int rv;
|
|
|
|
/* __STDIO_STREAM_RESET_GCS(&f); */
|
|
#ifdef __UCLIBC_HAS_GLIBC_CUSTOM_STREAMS__
|
|
f.__cookie = &(f.__filedes);
|
|
f.__gcs.read = NULL;
|
|
f.__gcs.write = NULL;
|
|
f.__gcs.seek = NULL;
|
|
f.__gcs.close = NULL;
|
|
#endif
|
|
|
|
f.__filedes = __STDIO_STREAM_FAKE_VSWPRINTF_FILEDES;
|
|
f.__modeflags = (__FLAG_WIDE|__FLAG_WRITEONLY|__FLAG_WRITING);
|
|
|
|
f.__ungot_width[0] = 0;
|
|
#ifdef __STDIO_MBSTATE
|
|
__INIT_MBSTATE(&(f.__state));
|
|
#endif /* __STDIO_MBSTATE */
|
|
|
|
#ifdef __UCLIBC_HAS_THREADS__
|
|
f.__user_locking = 1; /* Set user locking. */
|
|
STDIO_INIT_MUTEX(f.__lock);
|
|
#endif /* __UCLIBC_HAS_THREADS__ */
|
|
|
|
f.__nextopen = NULL;
|
|
|
|
if (size > ((SIZE_MAX - (size_t) buf)/sizeof(wchar_t))) {
|
|
size = ((SIZE_MAX - (size_t) buf)/sizeof(wchar_t));
|
|
}
|
|
|
|
f.__bufstart = (unsigned char *) buf;
|
|
f.__bufend = (unsigned char *) (buf + size);
|
|
__STDIO_STREAM_INIT_BUFREAD_BUFPOS(&f);
|
|
__STDIO_STREAM_DISABLE_GETC(&f);
|
|
__STDIO_STREAM_DISABLE_PUTC(&f);
|
|
|
|
rv = _vfwprintf_internal(&f, format, arg);
|
|
|
|
/* NOTE: Return behaviour differs from snprintf... */
|
|
if (f.__bufpos == f.__bufend) {
|
|
rv = -1;
|
|
if (size) {
|
|
f.__bufpos = (unsigned char *) (((wchar_t *) f.__bufpos) - 1);
|
|
}
|
|
}
|
|
if (size) {
|
|
*((wchar_t *) f.__bufpos) = 0;
|
|
}
|
|
return rv;
|
|
}
|
|
libc_hidden_def(vswprintf)
|
|
|
|
#endif /* __STDIO_BUFFERS */
|