xref: /netbsd-src/lib/libc/stdio/fgetwc.c (revision 526d942790cc352b6ea7931f9884cc6bdbf20404)
1*526d9427Schristos /* $NetBSD: fgetwc.c,v 1.13 2012/03/15 18:22:30 christos Exp $ */
217f3654aSyamt 
317f3654aSyamt /*-
417f3654aSyamt  * Copyright (c)2001 Citrus Project,
517f3654aSyamt  * All rights reserved.
617f3654aSyamt  *
717f3654aSyamt  * Redistribution and use in source and binary forms, with or without
817f3654aSyamt  * modification, are permitted provided that the following conditions
917f3654aSyamt  * are met:
1017f3654aSyamt  * 1. Redistributions of source code must retain the above copyright
1117f3654aSyamt  *    notice, this list of conditions and the following disclaimer.
1217f3654aSyamt  * 2. Redistributions in binary form must reproduce the above copyright
1317f3654aSyamt  *    notice, this list of conditions and the following disclaimer in the
1417f3654aSyamt  *    documentation and/or other materials provided with the distribution.
1517f3654aSyamt  *
1617f3654aSyamt  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
1717f3654aSyamt  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
1817f3654aSyamt  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
1917f3654aSyamt  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
2017f3654aSyamt  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2117f3654aSyamt  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2217f3654aSyamt  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2317f3654aSyamt  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2417f3654aSyamt  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2517f3654aSyamt  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2617f3654aSyamt  * SUCH DAMAGE.
2717f3654aSyamt  *
2817f3654aSyamt  * $Citrus$
2917f3654aSyamt  */
3017f3654aSyamt 
3188c3eadbSlukem #include <sys/cdefs.h>
3288c3eadbSlukem #if defined(LIBC_SCCS) && !defined(lint)
33*526d9427Schristos __RCSID("$NetBSD: fgetwc.c,v 1.13 2012/03/15 18:22:30 christos Exp $");
3488c3eadbSlukem #endif /* LIBC_SCCS and not lint */
3588c3eadbSlukem 
3617f3654aSyamt #include <assert.h>
3717f3654aSyamt #include <errno.h>
3817f3654aSyamt #include <stdio.h>
3917f3654aSyamt #include <wchar.h>
4017f3654aSyamt #include "reentrant.h"
413fdac2b8Sthorpej #include "local.h"
4217f3654aSyamt 
4317f3654aSyamt wint_t
__fgetwc_unlock(FILE * fp)445abc4b4fStshiozak __fgetwc_unlock(FILE *fp)
4517f3654aSyamt {
4617f3654aSyamt 	struct wchar_io_data *wcio;
4717f3654aSyamt 	wchar_t wc;
4861c3ab2fStnozaki 	size_t nr;
4917f3654aSyamt 
5017f3654aSyamt 	_DIAGASSERT(fp != NULL);
5117f3654aSyamt 
5217f3654aSyamt 	_SET_ORIENTATION(fp, 1);
5317f3654aSyamt 	wcio = WCIO_GET(fp);
5461c3ab2fStnozaki 	_DIAGASSERT(wcio != NULL);
5517f3654aSyamt 
5617f3654aSyamt 	/* if there're ungetwc'ed wchars, use them */
5761c3ab2fStnozaki 	if (wcio->wcio_ungetwc_inbuf)
5861c3ab2fStnozaki 		return wcio->wcio_ungetwc_buf[--wcio->wcio_ungetwc_inbuf];
5917f3654aSyamt 
6061c3ab2fStnozaki 	if (fp->_r <= 0) {
6161c3ab2fStnozaki restart:
6261c3ab2fStnozaki 		if (__srefill(fp) != 0)
6317f3654aSyamt 			return WEOF;
6417f3654aSyamt 	}
6561c3ab2fStnozaki 	nr = mbrtowc(&wc, (const char *)fp->_p,
66cfbb35edSchristos 	    (size_t)fp->_r, &wcio->wcio_mbstate_in);
6761c3ab2fStnozaki 	if (nr == (size_t)-1) {
68c98a8494Stnozaki 		fp->_flags |= __SERR;
6917f3654aSyamt 		return WEOF;
7061c3ab2fStnozaki 	} else if (nr == (size_t)-2) {
7161c3ab2fStnozaki 		fp->_p += fp->_r;
7261c3ab2fStnozaki 		fp->_r = 0;
7361c3ab2fStnozaki 		goto restart;
7417f3654aSyamt 	}
754e9a5498Stnozaki 	if (wc == L'\0') {
76a2246766Stnozaki 		while (*fp->_p != '\0') {
77a2246766Stnozaki 			++fp->_p;
78a2246766Stnozaki 			--fp->_r;
79a2246766Stnozaki 		}
80a2246766Stnozaki 		nr = 1;
81a2246766Stnozaki 	}
8261c3ab2fStnozaki 	fp->_p += nr;
83c5e820caSchristos 	_DIAGASSERT(__type_fit(int, fp->_r - nr));
84c5e820caSchristos 	fp->_r -= (int)nr;
8517f3654aSyamt 
8617f3654aSyamt 	return wc;
8717f3654aSyamt }
885abc4b4fStshiozak 
895abc4b4fStshiozak wint_t
fgetwc(FILE * fp)905abc4b4fStshiozak fgetwc(FILE *fp)
915abc4b4fStshiozak {
925abc4b4fStshiozak 	wint_t r;
935abc4b4fStshiozak 
945abc4b4fStshiozak 	_DIAGASSERT(fp != NULL);
955abc4b4fStshiozak 
965abc4b4fStshiozak 	FLOCKFILE(fp);
975abc4b4fStshiozak 	r = __fgetwc_unlock(fp);
985abc4b4fStshiozak 	FUNLOCKFILE(fp);
995abc4b4fStshiozak 
100*526d9427Schristos 	return r;
1015abc4b4fStshiozak }
1025abc4b4fStshiozak 
103