1 /* Copyright (C) 1996-2022 Free Software Foundation, Inc.
2    This file is part of the GNU C Library.
3 
4    The GNU C Library is free software; you can redistribute it and/or
5    modify it under the terms of the GNU Lesser General Public
6    License as published by the Free Software Foundation; either
7    version 2.1 of the License, or (at your option) any later version.
8 
9    The GNU C Library is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12    Lesser General Public License for more details.
13 
14    You should have received a copy of the GNU Lesser General Public
15    License along with the GNU C Library; if not, see
16    <https://www.gnu.org/licenses/>.  */
17 
18 #include <ctype.h>
19 #include <dlfcn.h>
20 #include <gconv.h>
21 #include <stdio.h>
22 #include <string.h>
23 #include <wchar.h>
24 #include <wcsmbsload.h>
25 #include <limits.h>
26 
27 #include <sysdep.h>
28 
29 
30 wint_t
__btowc(int c)31 __btowc (int c)
32 {
33   const struct gconv_fcts *fcts;
34 
35   /* If the parameter does not fit into one byte or it is the EOF value
36      we can give the answer now.  */
37   if (c < SCHAR_MIN || c > UCHAR_MAX || c == EOF)
38     return WEOF;
39 
40   /* We know that only ASCII compatible encodings are used for the
41      locale and that the wide character encoding is ISO 10646.  */
42   if (isascii (c))
43     return (wint_t) c;
44 
45   /* Get the conversion functions.  */
46   fcts = get_gconv_fcts (_NL_CURRENT_DATA (LC_CTYPE));
47   __gconv_btowc_fct btowc_fct = fcts->towc->__btowc_fct;
48 #ifdef PTR_DEMANGLE
49   if (fcts->towc->__shlib_handle != NULL)
50     PTR_DEMANGLE (btowc_fct);
51 #endif
52 
53   if (__builtin_expect (fcts->towc_nsteps == 1, 1)
54       && __builtin_expect (btowc_fct != NULL, 1))
55     {
56       /* Use the shortcut function.  */
57       return DL_CALL_FCT (btowc_fct, (fcts->towc, (unsigned char) c));
58     }
59   else
60     {
61       /* Fall back to the slow but generic method.  */
62       wchar_t result;
63       struct __gconv_step_data data;
64       unsigned char inbuf[1];
65       const unsigned char *inptr = inbuf;
66       size_t dummy;
67       int status;
68 
69       /* Tell where we want the result.  */
70       data.__outbuf = (unsigned char *) &result;
71       data.__outbufend = data.__outbuf + sizeof (wchar_t);
72       data.__invocation_counter = 0;
73       data.__internal_use = 1;
74       data.__flags = __GCONV_IS_LAST;
75       data.__statep = &data.__state;
76 
77       /* Make sure we start in the initial state.  */
78       memset (&data.__state, '\0', sizeof (mbstate_t));
79 
80       /* Create the input string.  */
81       inbuf[0] = c;
82 
83       __gconv_fct fct = fcts->towc->__fct;
84 #ifdef PTR_DEMANGLE
85       if (fcts->towc->__shlib_handle != NULL)
86 	PTR_DEMANGLE (fct);
87 #endif
88       status = DL_CALL_FCT (fct, (fcts->towc, &data, &inptr, inptr + 1,
89 				  NULL, &dummy, 0, 1));
90 
91       if (status != __GCONV_OK && status != __GCONV_FULL_OUTPUT
92 	  && status != __GCONV_EMPTY_INPUT)
93 	/* The conversion failed.  */
94 	result = WEOF;
95 
96       return result;
97     }
98 }
99 weak_alias (__btowc, btowc)
100