/*- * Copyright (c) 2002-2004 Tim J. Robbins. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */ /* FUNCTION <>, <>, <>, <>---get a wide character from a file or stream INDEX fgetwc INDEX fgetwc_unlocked INDEX _fgetwc_r INDEX _fgetwc_unlocked_r INDEX getwc INDEX getwc_unlocked INDEX _getwc_r INDEX _getwc_unlocked_r SYNOPSIS #include #include wint_t fgetwc(FILE *<[fp]>); #define _GNU_SOURCE #include #include wint_t fgetwc_unlocked(FILE *<[fp]>); #include #include wint_t _fgetwc_r(struct _reent *<[ptr]>, FILE *<[fp]>); #include #include wint_t _fgetwc_unlocked_r(struct _reent *<[ptr]>, FILE *<[fp]>); #include #include wint_t getwc(FILE *<[fp]>); #define _GNU_SOURCE #include #include wint_t getwc_unlocked(FILE *<[fp]>); #include #include wint_t _getwc_r(struct _reent *<[ptr]>, FILE *<[fp]>); #include #include wint_t _getwc_unlocked_r(struct _reent *<[ptr]>, FILE *<[fp]>); DESCRIPTION Use <> to get the next wide character from the file or stream identified by <[fp]>. As a side effect, <> advances the file's current position indicator. <> is a non-thread-safe version of <>. <> may only safely be used within a scope protected by flockfile() (or ftrylockfile()) and funlockfile(). This function may safely be used in a multi-threaded program if and only if they are called while the invoking thread owns the (FILE *) object, as is the case after a successful call to the flockfile() or ftrylockfile() functions. If threads are disabled, then <> is equivalent to <>. The <> and <> functions or macros functions identically to <> and <>. It may be implemented as a macro, and may evaluate its argument more than once. There is no reason ever to use it. <<_fgetwc_r>>, <<_getwc_r>>, <<_fgetwc_unlocked_r>>, and <<_getwc_unlocked_r>> are simply reentrant versions of the above functions that are passed the additional reentrant structure pointer argument: <[ptr]>. RETURNS The next wide character cast to <>, unless there is no more data, or the host system reports a read error; in either of these situations, <> and <> return <>. You can distinguish the two situations that cause an <> result by using the <> and <> functions. PORTABILITY <> and <> are required by C99 and POSIX.1-2001. <> and <> are GNU extensions. */ #include <_ansi.h> #include #include #include #include #include #include "local.h" wint_t __fgetwc (struct _reent *ptr, register FILE *fp) { wchar_t wc; size_t nconv; if (fp->_r <= 0 && __srefill_r (ptr, fp)) return (WEOF); if (MB_CUR_MAX == 1) { /* Fast path for single-byte encodings. */ wc = *fp->_p++; fp->_r--; return (wc); } do { nconv = _mbrtowc_r (ptr, &wc, (char *) fp->_p, fp->_r, &fp->_mbstate); if (nconv == (size_t)-1) break; else if (nconv == (size_t)-2) continue; else if (nconv == 0) { /* * Assume that the only valid representation of * the null wide character is a single null byte. */ fp->_p++; fp->_r--; return (L'\0'); } else { fp->_p += nconv; fp->_r -= nconv; return (wc); } } while (__srefill_r(ptr, fp) == 0); fp->_flags |= __SERR; errno = EILSEQ; return (WEOF); } wint_t _fgetwc_r (struct _reent *ptr, register FILE *fp) { wint_t r; _newlib_flockfile_start (fp); if (ORIENT(fp, 1) != 1) r = WEOF; else r = __fgetwc (ptr, fp); _newlib_flockfile_end (fp); return r; } wint_t fgetwc (FILE *fp) { struct _reent *reent = _REENT; CHECK_INIT(reent, fp); return _fgetwc_r (reent, fp); }