[v4] libio: Fix wide stream backup buffer leak on fclose [BZ #33999]
Checks
| Context |
Check |
Description |
| redhat-pt-bot/TryBot-apply_patch |
success
|
Patch applied to master at the time it was sent
|
| linaro-tcwg-bot/tcwg_glibc_build--master-aarch64 |
success
|
Build passed
|
| linaro-tcwg-bot/tcwg_glibc_check--master-aarch64 |
success
|
Test passed
|
| linaro-tcwg-bot/tcwg_glibc_build--master-arm |
success
|
Build passed
|
| linaro-tcwg-bot/tcwg_glibc_check--master-arm |
success
|
Test passed
|
| redhat-pt-bot/TryBot-32bit |
success
|
Build for i686
|
Commit Message
From: Xiang Gao <gaoxiang@kylinos.cn>
This patch fixes a memory leak when ungetwc is used on a wide oriented stream.
The backup buffer was never freed on fclose, causing a memory leak per
ungetwc/fclose call.
The leak has two causes:
In iofclose.c, for wide streams (fp->mode > 0), _IO_new_fclose never calls
_IO_free_wbackup_area. Fixed by adding the missing call.
In wgenops.c, _IO_wdefault_finish checks fp->_IO_save_base (the narrow field,
always NULL for wide streams) instead of fp->_wide_data->_IO_save_base,
and uses a bare free() that leaves _IO_save_end and _IO_backup_base dangling.
Replace the hand-rolled cleanup with _IO_have_wbackup/_IO_free_wbackup_area,
which handles backup-mode switching and clears all three pointers.
This was independently reported by Rocket Ma [1], whose patch corrects the condition
but still uses the manual free path.
Apply the same _IO_have_backup condition in genops.c for consistency.
Tested by:
"make test t=libio/*" on x86_64-linux-gnu with no regressions.
[1] https://patchwork.sourceware.org/project/glibc/patch/20260323171742.1039768-1-marocketbd@gmail.com/
Signed-off-by: Xiang Gao <gaoxiang@kylinos.cn>
---
Changes since v3:
- Add rewind(fp) before ungetwc() to properly switch from output
to input mode as suggested.
---
libio/Makefile | 1 +
libio/genops.c | 7 ++----
libio/iofclose.c | 3 +++
libio/tst-wbackup-leak.c | 52 ++++++++++++++++++++++++++++++++++++++++
libio/wgenops.c | 7 ++----
5 files changed, 60 insertions(+), 10 deletions(-)
create mode 100644 libio/tst-wbackup-leak.c
@@ -135,6 +135,7 @@ tests = \
tst-swscanf \
tst-ungetwc1 \
tst-ungetwc2 \
+ tst-wbackup-leak \
tst-wfile-sync \
tst-wfiledoallocate-static \
tst-widetext \
@@ -636,11 +636,8 @@ _IO_default_finish (FILE *fp, int dummy)
for (mark = fp->_markers; mark != NULL; mark = mark->_next)
mark->_sbuf = NULL;
- if (fp->_IO_save_base)
- {
- _IO_free_backup_buf (fp, fp->_IO_save_base);
- fp->_IO_save_base = NULL;
- }
+ if (_IO_have_backup (fp))
+ _IO_free_backup_area (fp);
_IO_un_link ((struct _IO_FILE_plus *) fp);
@@ -67,6 +67,9 @@ _IO_new_fclose (FILE *fp)
_IO_FINISH (fp);
if (fp->_mode > 0)
{
+ if (_IO_have_wbackup (fp))
+ _IO_free_wbackup_area (fp);
+
/* This stream has a wide orientation. This means we have to free
the conversion functions. */
struct _IO_codecvt *cc = fp->_codecvt;
new file mode 100644
@@ -0,0 +1,52 @@
+/* Test _IO_wdefault_finish frees wide backup buffer [BZ #33999]. */
+
+#include <malloc.h>
+#include <stdio.h>
+#include <wchar.h>
+#include <support/check.h>
+
+static void
+one_round (void)
+{
+ wchar_t *buf = NULL;
+ size_t size = 0;
+
+ FILE *fp = open_wmemstream (&buf, &size);
+ TEST_VERIFY_EXIT (fp != NULL);
+ fputwc (L'A', fp);
+ fflush (fp);
+ /* Push back without prior read. rewind switches from output to
+ input mode. read_ptr == read_base, so _IO_wdefault_pbackfail
+ skips the buggy narrow read_ptr access (BZ #33998) and goes
+ straight to allocating a wide backup buffer at
+ fp->_wide_data->_IO_save_base.
+
+ Note: this testcase relies on the fact that open_wmemstream
+ does not set _IO_NO_READS on the stream. If that implementation
+ is changed, this test would need a different stream type to verify
+ the leak. */
+ rewind (fp);
+ ungetwc (L'Z', fp);
+ fclose (fp);
+ free (buf);
+}
+
+static int
+do_test (void)
+{
+ /* Warm up to stabilize allocator state. */
+ one_round ();
+
+ struct mallinfo2 before = mallinfo2 ();
+ for (int i = 0; i < 1000; i++)
+ one_round ();
+ struct mallinfo2 after = mallinfo2 ();
+
+ /* Each leak is 128 * sizeof(wchar_t) = 512 bytes.
+ 1000 iterations would leak ~512 KB. Allow 4 KB noise. */
+ TEST_VERIFY (after.uordblks - before.uordblks < 4096);
+
+ return 0;
+}
+
+#include <support/test-driver.c>
@@ -181,11 +181,8 @@ _IO_wdefault_finish (FILE *fp, int dummy)
for (mark = fp->_markers; mark != NULL; mark = mark->_next)
mark->_sbuf = NULL;
- if (fp->_IO_save_base)
- {
- free (fp->_wide_data->_IO_save_base);
- fp->_IO_save_base = NULL;
- }
+ if (_IO_have_wbackup (fp))
+ _IO_free_wbackup_area (fp);
#ifdef _IO_MTSAFE_IO
if (fp->_lock != NULL)