summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--libc/misc/wchar/wstdio.c185
1 files changed, 28 insertions, 157 deletions
diff --git a/libc/misc/wchar/wstdio.c b/libc/misc/wchar/wstdio.c
index 79cf1dfa1..a4984f3c1 100644
--- a/libc/misc/wchar/wstdio.c
+++ b/libc/misc/wchar/wstdio.c
@@ -1,3 +1,31 @@
+/* Copyright (C) 2002 Manuel Novoa III
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Library General Public
+ * License as published by the Free Software Foundation; either
+ * version 2 of the License, or (at your option) any later version.
+ *
+ * This 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
+ * Library General Public License for more details.
+ *
+ * You should have received a copy of the GNU Library General Public
+ * License along with this library; if not, write to the Free
+ * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+ */
+
+/* ATTENTION! ATTENTION! ATTENTION! ATTENTION! ATTENTION!
+ *
+ * Besides uClibc, I'm using this code in my libc for elks, which is
+ * a 16-bit environment with a fairly limited compiler. It would make
+ * things much easier for me if this file isn't modified unnecessarily.
+ * In particular, please put any new or replacement functions somewhere
+ * else, and modify the makefile to use your version instead.
+ * Thanks. Manuel
+ *
+ * ATTENTION! ATTENTION! ATTENTION! ATTENTION! ATTENTION! */
+
/*
* ANSI/ISO C99 says
@@ -260,41 +288,6 @@ UNLOCKED(wchar_t *,fgetws,(wchar_t *__restrict ws, int n,
UNLOCKED(wint_t,fputwc,(wchar_t wc, FILE *stream),(wc, stream))
{
-#if 0
- size_t r;
- char buf[MB_LEN_MAX];
-
- if (stream->modeflags & __FLAG_NARROW) {
- stream->modeflags |= __FLAG_ERROR;
- __set_errno(EBADF);
- return WEOF;
- }
- stream->modeflags |= __FLAG_WIDE;
-
- /* TODO:
- * If stream is in reading state with bad mbstate object, what to do?
- * Should we check the state first? Should we check error indicator?
- * Should we check reading or even read-only?
- */
- /* It looks like the only ANSI/ISO C99 - blessed way of manipulating
- * the stream's mbstate object is through fgetpos/fsetpos. */
- r = wcrtomb(buf, wc, &stream->state);
-
- return (r != ((size_t) -1) && (r == _stdio_fwrite(buf, r, stream)))
- ? wc : WEOF;
-
-#elif 0
-
- /* this is broken if wc == 0 !!! */
- wchar_t wbuf[2];
-
- wbuf[0] = wc;
- wbuf[1] = 0;
-
- return (fputws_unlocked(wbuf, stream) > 0) ? wc : WEOF;
-
-#else
-
size_t n;
char buf[MB_LEN_MAX];
@@ -308,8 +301,6 @@ UNLOCKED(wint_t,fputwc,(wchar_t wc, FILE *stream),(wc, stream))
return (((n = wcrtomb(buf, wc, &stream->state)) != ((size_t)-1)) /* EILSEQ */
&& (_stdio_fwrite(buf, n, stream) != n))/* Didn't write everything. */
? wc : WEOF;
-
-#endif
}
strong_alias(fputwc_unlocked,putwc_unlocked);
@@ -333,7 +324,6 @@ UNLOCKED_STREAM(wint_t,putwchar,(wchar_t wc),(wc),stdout)
UNLOCKED(int,fputws,(const wchar_t *__restrict ws,
register FILE *__restrict stream),(ws, stream))
{
-#if 1
size_t n;
char buf[64];
@@ -357,125 +347,6 @@ UNLOCKED(int,fputws,(const wchar_t *__restrict ws,
}
return 1;
-
-
-
-
-#elif 1
- int result;
- size_t n;
- size_t len;
- register char *s;
- unsigned char *bufend;
- char sbuf[MB_LEN_MAX];
-
- if (stream->modeflags & __FLAG_NARROW) {
- RETURN_BADF:
- stream->modeflags |= __FLAG_ERROR;
- __set_errno(EBADF);
- return -1;
- }
- stream->modeflags |= __FLAG_WIDE;
-
- /* Note: What follows is setup grabbed from _stdio_fwrite and modified
- * slightly. Since this is a wide stream, we can ignore bufgetc and
- * bufputc if present. They always == bufstart.
- * It is unfortunate that we need to duplicate so much code here, but
- * we need to do the stream setup before starting the wc->mb conversion. */
-
- if ((stream->modeflags & __FLAG_READONLY)
-#ifndef __STDIO_AUTO_RW_TRANSITION
- /* ANSI/ISO requires either at EOF or currently not reading. */
- || ((stream->modeflags & (__FLAG_READING|__FLAG_EOF))
- == __FLAG_READING)
-#endif /* __STDIO_AUTO_RW_TRANSITION */
- ) {
- /* TODO: This is for posix behavior if readonly. To save space, we
- * use this errno for write attempt while reading, as no errno is
- * specified by posix for this case, even though the restriction is
- * mentioned in fopen(). */
- goto RETURN_BADF;
- }
-
-#ifdef __STDIO_AUTO_RW_TRANSITION
- /* If reading, deal with ungots and read-buffered chars. */
- if (stream->modeflags & __FLAG_READING) {
- if (((stream->bufpos < stream->bufread)
- || (stream->modeflags & __MASK_UNGOT))
- /* If appending, we might as well seek to end to save a seek. */
- /* TODO: set EOF in fseek when appropriate? */
- && fseek(stream, 0L,
- ((stream->modeflags & __FLAG_APPEND)
- ? SEEK_END : SEEK_CUR))
- ) {
- /* Note: This differs from glibc's apparent behavior of
- not setting the error flag and discarding the buffered
- read data. */
- stream->modeflags |= __FLAG_ERROR; /* fseek may not set this. */
- return -1; /* Fail if we need to fseek but can't. */
- }
- /* Always reset even if fseek called (saves a test). */
- stream->bufpos = stream->bufread = stream->bufstart;
- stream->modeflags &= ~__FLAG_READING;
- }
-#endif
-
- /* Ok, the boilerplate from _stdio_fwrite is done. */
-
- if (stream->bufpos > stream->bufstart) { /* Pending writes.. */
- /* This is a performance penalty, but it simplifies the code below.
- * If this is removed, the buffer sharing and while loop condition
- * need to be modified below (at least). We at least save a little
- * on the overhead by calling _stdio_fwrite directly instead of
- * fflush_unlocked. */
- if (_stdio_fwrite(NULL, 0, stream) > 0) {/* fflush incomplete! */
- return -1;
- }
- }
-
- stream->modeflags |= __FLAG_WRITING; /* Ensure Writing flag is set. */
-
- /* Next, we "steal" the stream's buffer and do the wc->mb conversion
- * straight into it. This will cause the equivalent of an fflush
- * for each string write. :-( */
-
- bufend = NULL;
- s = stream->bufstart;
-
- if ((len = stream->bufend - stream->bufstart) < MB_LEN_MAX) {
- /* Stream is unbuffered or buffer is too small, so deactivate. */
- bufend = stream->bufend;
- stream->bufend = stream->bufstart;
- s = sbuf;
- len = MB_LEN_MAX;
- }
-
- result = 1; /* Assume success. */
- while (ws && (n = wcsrtombs(s, &ws, len, &stream->state)) != 0) {
- if ((n == ((size_t) -1)) /* Encoding error! */
- /* TODO - maybe call write directly? but what about custom streams? */
- || (_stdio_fwrite(s, n, stream) != n)/* Didn't write everything. */
- ) {
- result = -1;
- break;
- }
- }
-
- if (bufend) { /* If deactivated stream buffer, renable it. */
- stream->bufend = bufend;
- }
-
- return result;
-
-#else /* slow, dumb version */
- while (*ws) {
- if (fputwc_unlocked(*ws, stream) == WEOF) {
- return -1;
- }
- ++ws;
- }
- return 1;
-#endif
}
#endif