1*199767f8SToomas Soome /*-
2*199767f8SToomas Soome  * Copyright (c) 2011 The FreeBSD Foundation
3*199767f8SToomas Soome  * All rights reserved.
4*199767f8SToomas Soome  *
5*199767f8SToomas Soome  * This software was developed by David Chisnall under sponsorship from
6*199767f8SToomas Soome  * the FreeBSD Foundation.
7*199767f8SToomas Soome  *
8*199767f8SToomas Soome  * Redistribution and use in source and binary forms, with or without
9*199767f8SToomas Soome  * modification, are permitted provided that the following conditions
10*199767f8SToomas Soome  * are met:
11*199767f8SToomas Soome  * 1. Redistributions of source code must retain the above copyright
12*199767f8SToomas Soome  *    notice, this list of conditions and the following disclaimer.
13*199767f8SToomas Soome  * 2. Redistributions in binary form must reproduce the above copyright
14*199767f8SToomas Soome  *    notice, this list of conditions and the following disclaimer in the
15*199767f8SToomas Soome  *    documentation and/or other materials provided with the distribution.
16*199767f8SToomas Soome  *
17*199767f8SToomas Soome  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18*199767f8SToomas Soome  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19*199767f8SToomas Soome  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20*199767f8SToomas Soome  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21*199767f8SToomas Soome  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22*199767f8SToomas Soome  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23*199767f8SToomas Soome  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24*199767f8SToomas Soome  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25*199767f8SToomas Soome  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26*199767f8SToomas Soome  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27*199767f8SToomas Soome  * SUCH DAMAGE.
28*199767f8SToomas Soome  *
29*199767f8SToomas Soome  * $FreeBSD$
30*199767f8SToomas Soome  */
31*199767f8SToomas Soome 
32*199767f8SToomas Soome 
33*199767f8SToomas Soome #if	(defined(_XLOCALE_WCTYPES) && !defined(_XLOCALE_WCTYPE_H)) || \
34*199767f8SToomas Soome 	(!defined(_XLOCALE_WCTYPES) && !defined(_XLOCALE_CTYPE_H))
35*199767f8SToomas Soome 
36*199767f8SToomas Soome #ifdef _XLOCALE_WCTYPES
37*199767f8SToomas Soome #define _XLOCALE_WCTYPE_H
38*199767f8SToomas Soome #else
39*199767f8SToomas Soome #define _XLOCALE_CTYPE_H
40*199767f8SToomas Soome #endif
41*199767f8SToomas Soome 
42*199767f8SToomas Soome #ifndef _LOCALE_T_DEFINED
43*199767f8SToomas Soome #define _LOCALE_T_DEFINED
44*199767f8SToomas Soome typedef struct	_xlocale *locale_t;
45*199767f8SToomas Soome #endif
46*199767f8SToomas Soome 
47*199767f8SToomas Soome #ifndef _XLOCALE_RUN_FUNCTIONS_DEFINED
48*199767f8SToomas Soome #define _XLOCALE_RUN_FUNCTIONS_DEFINED 1
49*199767f8SToomas Soome unsigned long	 ___runetype_l(__ct_rune_t, locale_t) __pure;
50*199767f8SToomas Soome __ct_rune_t	 ___tolower_l(__ct_rune_t, locale_t) __pure;
51*199767f8SToomas Soome __ct_rune_t	 ___toupper_l(__ct_rune_t, locale_t) __pure;
52*199767f8SToomas Soome _RuneLocale	*__runes_for_locale(locale_t, int*);
53*199767f8SToomas Soome #endif
54*199767f8SToomas Soome 
55*199767f8SToomas Soome #ifndef _XLOCALE_INLINE
56*199767f8SToomas Soome #if defined(__GNUC__) && !defined(__GNUC_STDC_INLINE__)
57*199767f8SToomas Soome /* GNU89 inline has nonstandard semantics. */
58*199767f8SToomas Soome #define _XLOCALE_INLINE extern __inline
59*199767f8SToomas Soome #else
60*199767f8SToomas Soome /* Hack to work around people who define inline away */
61*199767f8SToomas Soome #ifdef inline
62*199767f8SToomas Soome #define _XLOCALE_INLINE static __inline
63*199767f8SToomas Soome #else
64*199767f8SToomas Soome /* Define with C++ / C99 compatible semantics */
65*199767f8SToomas Soome #define _XLOCALE_INLINE inline
66*199767f8SToomas Soome #endif
67*199767f8SToomas Soome #endif
68*199767f8SToomas Soome #endif /* _XLOCALE_INLINE */
69*199767f8SToomas Soome 
70*199767f8SToomas Soome #ifdef _XLOCALE_WCTYPES
71*199767f8SToomas Soome _XLOCALE_INLINE int
72*199767f8SToomas Soome __maskrune_l(__ct_rune_t __c, unsigned long __f, locale_t __loc);
73*199767f8SToomas Soome _XLOCALE_INLINE int
74*199767f8SToomas Soome __istype_l(__ct_rune_t __c, unsigned long __f, locale_t __loc);
75*199767f8SToomas Soome 
76*199767f8SToomas Soome _XLOCALE_INLINE int
__maskrune_l(__ct_rune_t __c,unsigned long __f,locale_t __loc)77*199767f8SToomas Soome __maskrune_l(__ct_rune_t __c, unsigned long __f, locale_t __loc)
78*199767f8SToomas Soome {
79*199767f8SToomas Soome 	int __limit;
80*199767f8SToomas Soome 	_RuneLocale *runes = __runes_for_locale(__loc, &__limit);
81*199767f8SToomas Soome 	return ((__c < 0 || __c >= _CACHED_RUNES) ? ___runetype_l(__c, __loc) :
82*199767f8SToomas Soome 	        runes->__runetype[__c]) & __f;
83*199767f8SToomas Soome }
84*199767f8SToomas Soome 
85*199767f8SToomas Soome _XLOCALE_INLINE int
__istype_l(__ct_rune_t __c,unsigned long __f,locale_t __loc)86*199767f8SToomas Soome __istype_l(__ct_rune_t __c, unsigned long __f, locale_t __loc)
87*199767f8SToomas Soome {
88*199767f8SToomas Soome 	return (!!__maskrune_l(__c, __f, __loc));
89*199767f8SToomas Soome }
90*199767f8SToomas Soome 
91*199767f8SToomas Soome #define XLOCALE_ISCTYPE(fname, cat) \
92*199767f8SToomas Soome 		_XLOCALE_INLINE int isw##fname##_l(int, locale_t);\
93*199767f8SToomas Soome 		_XLOCALE_INLINE int isw##fname##_l(int __c, locale_t __l)\
94*199767f8SToomas Soome 		{ return __istype_l(__c, cat, __l); }
95*199767f8SToomas Soome #else
96*199767f8SToomas Soome _XLOCALE_INLINE int
97*199767f8SToomas Soome __sbmaskrune_l(__ct_rune_t __c, unsigned long __f, locale_t __loc);
98*199767f8SToomas Soome _XLOCALE_INLINE int
99*199767f8SToomas Soome __sbistype_l(__ct_rune_t __c, unsigned long __f, locale_t __loc);
100*199767f8SToomas Soome 
101*199767f8SToomas Soome _XLOCALE_INLINE int
__sbmaskrune_l(__ct_rune_t __c,unsigned long __f,locale_t __loc)102*199767f8SToomas Soome __sbmaskrune_l(__ct_rune_t __c, unsigned long __f, locale_t __loc)
103*199767f8SToomas Soome {
104*199767f8SToomas Soome 	int __limit;
105*199767f8SToomas Soome 	_RuneLocale *runes = __runes_for_locale(__loc, &__limit);
106*199767f8SToomas Soome 	return (__c < 0 || __c >= __limit) ? 0 :
107*199767f8SToomas Soome 	       runes->__runetype[__c] & __f;
108*199767f8SToomas Soome }
109*199767f8SToomas Soome 
110*199767f8SToomas Soome _XLOCALE_INLINE int
__sbistype_l(__ct_rune_t __c,unsigned long __f,locale_t __loc)111*199767f8SToomas Soome __sbistype_l(__ct_rune_t __c, unsigned long __f, locale_t __loc)
112*199767f8SToomas Soome {
113*199767f8SToomas Soome 	return (!!__sbmaskrune_l(__c, __f, __loc));
114*199767f8SToomas Soome }
115*199767f8SToomas Soome 
116*199767f8SToomas Soome #define XLOCALE_ISCTYPE(__fname, __cat) \
117*199767f8SToomas Soome 		_XLOCALE_INLINE int is##__fname##_l(int, locale_t); \
118*199767f8SToomas Soome 		_XLOCALE_INLINE int is##__fname##_l(int __c, locale_t __l)\
119*199767f8SToomas Soome 		{ return __sbistype_l(__c, __cat, __l); }
120*199767f8SToomas Soome #endif
121*199767f8SToomas Soome 
122*199767f8SToomas Soome XLOCALE_ISCTYPE(alnum, _CTYPE_A|_CTYPE_D|_CTYPE_N)
123*199767f8SToomas Soome XLOCALE_ISCTYPE(alpha, _CTYPE_A)
124*199767f8SToomas Soome XLOCALE_ISCTYPE(blank, _CTYPE_B)
125*199767f8SToomas Soome XLOCALE_ISCTYPE(cntrl, _CTYPE_C)
126*199767f8SToomas Soome XLOCALE_ISCTYPE(digit, _CTYPE_D)
127*199767f8SToomas Soome XLOCALE_ISCTYPE(graph, _CTYPE_G)
128*199767f8SToomas Soome XLOCALE_ISCTYPE(hexnumber, _CTYPE_X)
129*199767f8SToomas Soome XLOCALE_ISCTYPE(ideogram, _CTYPE_I)
130*199767f8SToomas Soome XLOCALE_ISCTYPE(lower, _CTYPE_L)
131*199767f8SToomas Soome XLOCALE_ISCTYPE(number, _CTYPE_D|_CTYPE_N)
132*199767f8SToomas Soome XLOCALE_ISCTYPE(phonogram, _CTYPE_Q)
133*199767f8SToomas Soome XLOCALE_ISCTYPE(print, _CTYPE_R)
134*199767f8SToomas Soome XLOCALE_ISCTYPE(punct, _CTYPE_P)
135*199767f8SToomas Soome XLOCALE_ISCTYPE(rune, 0xFFFFFF00L)
136*199767f8SToomas Soome XLOCALE_ISCTYPE(space, _CTYPE_S)
137*199767f8SToomas Soome XLOCALE_ISCTYPE(special, _CTYPE_T)
138*199767f8SToomas Soome XLOCALE_ISCTYPE(upper, _CTYPE_U)
139*199767f8SToomas Soome XLOCALE_ISCTYPE(xdigit, _CTYPE_X)
140*199767f8SToomas Soome #undef XLOCALE_ISCTYPE
141*199767f8SToomas Soome 
142*199767f8SToomas Soome #ifdef _XLOCALE_WCTYPES
143*199767f8SToomas Soome _XLOCALE_INLINE int towlower_l(int, locale_t);
144*199767f8SToomas Soome _XLOCALE_INLINE int __wcwidth_l(__ct_rune_t, locale_t);
145*199767f8SToomas Soome _XLOCALE_INLINE int towupper_l(int, locale_t);
146*199767f8SToomas Soome 
towlower_l(int __c,locale_t __l)147*199767f8SToomas Soome _XLOCALE_INLINE int towlower_l(int __c, locale_t __l)
148*199767f8SToomas Soome {
149*199767f8SToomas Soome 	int __limit;
150*199767f8SToomas Soome 	_RuneLocale *__runes = __runes_for_locale(__l, &__limit);
151*199767f8SToomas Soome 	return (__c < 0 || __c >= _CACHED_RUNES) ? ___tolower_l(__c, __l) :
152*199767f8SToomas Soome 	       __runes->__maplower[__c];
153*199767f8SToomas Soome }
towupper_l(int __c,locale_t __l)154*199767f8SToomas Soome _XLOCALE_INLINE int towupper_l(int __c, locale_t __l)
155*199767f8SToomas Soome {
156*199767f8SToomas Soome 	int __limit;
157*199767f8SToomas Soome 	_RuneLocale *__runes = __runes_for_locale(__l, &__limit);
158*199767f8SToomas Soome 	return (__c < 0 || __c >= _CACHED_RUNES) ? ___toupper_l(__c, __l) :
159*199767f8SToomas Soome 	       __runes->__mapupper[__c];
160*199767f8SToomas Soome }
161*199767f8SToomas Soome _XLOCALE_INLINE int
__wcwidth_l(__ct_rune_t _c,locale_t __l)162*199767f8SToomas Soome __wcwidth_l(__ct_rune_t _c, locale_t __l)
163*199767f8SToomas Soome {
164*199767f8SToomas Soome 	unsigned int _x;
165*199767f8SToomas Soome 
166*199767f8SToomas Soome 	if (_c == 0)
167*199767f8SToomas Soome 		return (0);
168*199767f8SToomas Soome 	_x = (unsigned int)__maskrune_l(_c, _CTYPE_SWM|_CTYPE_R, __l);
169*199767f8SToomas Soome 	if ((_x & _CTYPE_SWM) != 0)
170*199767f8SToomas Soome 		return ((_x & _CTYPE_SWM) >> _CTYPE_SWS);
171*199767f8SToomas Soome 	return ((_x & _CTYPE_R) != 0 ? 1 : -1);
172*199767f8SToomas Soome }
173*199767f8SToomas Soome int iswctype_l(wint_t __wc, wctype_t __charclass, locale_t __l);
174*199767f8SToomas Soome wctype_t wctype_l(const char *property, locale_t __l);
175*199767f8SToomas Soome wint_t towctrans_l(wint_t __wc, wctrans_t desc, locale_t __l);
176*199767f8SToomas Soome wint_t nextwctype_l(wint_t __wc, wctype_t wct, locale_t __l);
177*199767f8SToomas Soome wctrans_t wctrans_l(const char *__charclass, locale_t __l);
178*199767f8SToomas Soome #undef _XLOCALE_WCTYPES
179*199767f8SToomas Soome #else
180*199767f8SToomas Soome _XLOCALE_INLINE int digittoint_l(int, locale_t);
181*199767f8SToomas Soome _XLOCALE_INLINE int tolower_l(int, locale_t);
182*199767f8SToomas Soome _XLOCALE_INLINE int toupper_l(int, locale_t);
183*199767f8SToomas Soome 
184*199767f8SToomas Soome _XLOCALE_INLINE int digittoint_l(int __c, locale_t __l)
185*199767f8SToomas Soome { return __sbmaskrune_l((__c), 0xFF, __l); }
186*199767f8SToomas Soome 
187*199767f8SToomas Soome _XLOCALE_INLINE int tolower_l(int __c, locale_t __l)
188*199767f8SToomas Soome {
189*199767f8SToomas Soome 	int __limit;
190*199767f8SToomas Soome 	_RuneLocale *__runes = __runes_for_locale(__l, &__limit);
191*199767f8SToomas Soome 	return (__c < 0 || __c >= __limit) ? __c :
192*199767f8SToomas Soome 	       __runes->__maplower[__c];
193*199767f8SToomas Soome }
194*199767f8SToomas Soome _XLOCALE_INLINE int toupper_l(int __c, locale_t __l)
195*199767f8SToomas Soome {
196*199767f8SToomas Soome 	int __limit;
197*199767f8SToomas Soome 	_RuneLocale *__runes = __runes_for_locale(__l, &__limit);
198*199767f8SToomas Soome 	return (__c < 0 || __c >= __limit) ? __c :
199*199767f8SToomas Soome 	       __runes->__mapupper[__c];
200*199767f8SToomas Soome }
201*199767f8SToomas Soome #endif
202*199767f8SToomas Soome #endif /* (defined(_XLOCALE_WCTYPES) && !defined(_XLOCALE_WCTYPE_H)) || \
203*199767f8SToomas Soome 	(!defined(_XLOCALE_WCTYPES) && !defined(_XLOCALE_CTYPE_H)) */
204