Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/lib/libc/stdio Improvements to fgetwln(3) from Andrey Chernov:



details:   https://anonhg.NetBSD.org/src/rev/d434027ce358
branches:  trunk
changeset: 347479:d434027ce358
user:      christos <christos%NetBSD.org@localhost>
date:      Sat Aug 27 13:15:48 2016 +0000

description:
Improvements to fgetwln(3) from Andrey Chernov:

1) Set the stdio error indicator on __slbexpand() failure.

2) fgetwc(3) may succeed even when ferror(3) is already set
   (for example, consider a program using SIG_IGN on SIGTTIN,
   reading from the tty while in the background, getting EIO,
   then coming to the foreground and reading again).
   So do not force fgetwln(3) to fail in such a case either.
   (Yes, the program should probably clearerr(3) before
   reading again, but let's be nicer in case it forgets.)

diffstat:

 lib/libc/stdio/fgetwln.c |  20 ++++++++++++++++----
 1 files changed, 16 insertions(+), 4 deletions(-)

diffs (46 lines):

diff -r ead6fa2e0dd9 -r d434027ce358 lib/libc/stdio/fgetwln.c
--- a/lib/libc/stdio/fgetwln.c  Sat Aug 27 12:08:14 2016 +0000
+++ b/lib/libc/stdio/fgetwln.c  Sat Aug 27 13:15:48 2016 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: fgetwln.c,v 1.7 2016/08/22 07:41:10 christos Exp $     */
+/*     $NetBSD: fgetwln.c,v 1.8 2016/08/27 13:15:48 christos Exp $     */
 
 /*-
  * Copyright (c) 2002-2004 Tim J. Robbins.
@@ -31,7 +31,7 @@
 #if 0
 __FBSDID("$FreeBSD: src/lib/libc/stdio/fgetwln.c,v 1.2 2004/08/06 17:00:09 tjr Exp $");
 #else
-__RCSID("$NetBSD: fgetwln.c,v 1.7 2016/08/22 07:41:10 christos Exp $");
+__RCSID("$NetBSD: fgetwln.c,v 1.8 2016/08/27 13:15:48 christos Exp $");
 #endif
 #endif /* LIBC_SCCS and not lint */
 
@@ -88,13 +88,25 @@
        while ((wc = __fgetwc_unlock(fp)) != WEOF) {
 #define        GROW    512
                if (len * sizeof(wchar_t) >= _EXT(fp)->_fgetstr_len &&
-                   __slbexpand(fp, (len + GROW) * sizeof(wchar_t)))
+                   __slbexpand(fp, (len + GROW) * sizeof(wchar_t))) {
+                       fp->_flags |= __SERR;
                        goto error;
+               }
                *((wchar_t *)(void *)_EXT(fp)->_fgetstr_buf + len++) = wc;
                if (wc == L'\n')
                        break;
        }
-       if (len == 0 || fp->_flags & __SERR)
+
+       /*
+        * The following test assumes that fgetwc() fails when
+        * feof() is already set, and that fgetwc() will never
+        * set feof() in the same call where it also sets ferror()
+        * or returns non-WEOF.
+        * Testing ferror() would not be better because fgetwc()
+        * may succeed even when ferror() is already set.
+        */
+
+       if (len == 0 || (wc == WEOF && !__sfeof(fp)))
                goto error;
 
        FUNLOCKFILE(fp);



Home | Main Index | Thread Index | Old Index