1 /* $NetBSD: iswctype_mb.c,v 1.13 2013/05/17 12:55:57 joerg Exp $ */
4 * Copyright (c)2008 Citrus Project,
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29 #include <sys/cdefs.h>
30 #if defined(LIBC_SCCS) && !defined(lint)
31 __RCSID("$NetBSD: iswctype_mb.c,v 1.13 2013/05/17 12:55:57 joerg Exp $");
32 #endif /* LIBC_SCCS and not lint */
34 #include "namespace.h"
35 #include <sys/types.h>
38 #define __SETLOCALE_SOURCE__
44 #include "setlocale_local.h"
46 #include "runetype_local.h"
47 #include "_wctype_local.h"
48 #include "_wctrans_local.h"
50 #define _RUNE_LOCALE(loc) ((_RuneLocale const *) \
51 (loc)->part_impl[(size_t)LC_CTYPE])
53 #define _ISWCTYPE_FUNC(name, index) \
55 isw##name##_l(wint_t wc, locale_t loc) \
57 _RuneLocale const *rl; \
58 _WCTypeEntry const *te; \
60 rl = _RUNE_LOCALE(loc); \
61 te = &rl->rl_wctype[index]; \
62 return _iswctype_priv(rl, wc, te); \
65 isw##name(wint_t wc) \
67 return isw##name##_l(wc, _current_locale()); \
70 _ISWCTYPE_FUNC(alnum
, _WCTYPE_INDEX_ALNUM
)
71 _ISWCTYPE_FUNC(alpha
, _WCTYPE_INDEX_ALPHA
)
72 _ISWCTYPE_FUNC(blank
, _WCTYPE_INDEX_BLANK
)
73 _ISWCTYPE_FUNC(cntrl
, _WCTYPE_INDEX_CNTRL
)
74 _ISWCTYPE_FUNC(digit
, _WCTYPE_INDEX_DIGIT
)
75 _ISWCTYPE_FUNC(graph
, _WCTYPE_INDEX_GRAPH
)
76 _ISWCTYPE_FUNC(lower
, _WCTYPE_INDEX_LOWER
)
77 _ISWCTYPE_FUNC(print
, _WCTYPE_INDEX_PRINT
)
78 _ISWCTYPE_FUNC(punct
, _WCTYPE_INDEX_PUNCT
)
79 _ISWCTYPE_FUNC(space
, _WCTYPE_INDEX_SPACE
)
80 _ISWCTYPE_FUNC(upper
, _WCTYPE_INDEX_UPPER
)
81 _ISWCTYPE_FUNC(xdigit
, _WCTYPE_INDEX_XDIGIT
)
83 #define _TOWCTRANS_FUNC(name, index) \
85 tow##name##_l(wint_t wc, locale_t loc) \
87 _RuneLocale const *rl; \
88 _WCTransEntry const *te; \
90 rl = _RUNE_LOCALE(loc); \
91 te = &rl->rl_wctrans[index]; \
92 return _towctrans_priv(wc, te); \
95 tow##name(wint_t wc) \
97 return tow##name##_l(wc, _current_locale()); \
99 _TOWCTRANS_FUNC(upper
, _WCTRANS_INDEX_UPPER
)
100 _TOWCTRANS_FUNC(lower
, _WCTRANS_INDEX_LOWER
)
103 wctype_l(const char *charclass
, locale_t loc
)
105 _RuneLocale
const *rl
;
108 rl
= _RUNE_LOCALE(loc
);
109 for (i
= 0; i
< _WCTYPE_NINDEXES
; ++i
) {
110 if (!strcmp(rl
->rl_wctype
[i
].te_name
, charclass
))
111 return (wctype_t)__UNCONST(&rl
->rl_wctype
[i
]);
113 return (wctype_t)NULL
;
117 wctype(const char *charclass
)
119 return wctype_l(charclass
, _current_locale());
123 wctrans_l(const char *charmap
, locale_t loc
)
125 _RuneLocale
const *rl
;
128 rl
= _RUNE_LOCALE(loc
);
129 for (i
= 0; i
< _WCTRANS_NINDEXES
; ++i
) {
130 _DIAGASSERT(rl
->rl_wctrans
[i
].te_name
!= NULL
);
131 if (!strcmp(rl
->rl_wctrans
[i
].te_name
, charmap
))
132 return (wctrans_t)__UNCONST(&rl
->rl_wctype
[i
]);
134 return (wctrans_t)NULL
;
138 wctrans(const char *charmap
)
140 return wctrans_l(charmap
, _current_locale());
144 iswctype_l(wint_t wc
, wctype_t charclass
, locale_t loc
)
146 _RuneLocale
const *rl
;
147 _WCTypeEntry
const *te
;
149 if (charclass
== NULL
) {
154 rl
= _RUNE_LOCALE(loc
);
155 te
= (_WCTypeEntry
const *)(void *)charclass
;
156 return _iswctype_priv(rl
, wc
, te
);
160 iswctype(wint_t wc
, wctype_t charclass
)
162 return iswctype_l(wc
, charclass
, _current_locale());
166 towctrans(wint_t wc
, wctrans_t charmap
)
168 _WCTransEntry
const *te
;
170 if (charmap
== NULL
) {
174 te
= (_WCTransEntry
const *)(void *)charmap
;
175 return _towctrans_priv(wc
, te
);
180 towctrans_l(wint_t wc
, wctrans_t charmap
, locale_t loc
)
182 return towctrans(wc
, charmap
);
185 __weak_alias(wcwidth
,_wcwidth
)
186 __weak_alias(wcwidth_l
,_wcwidth_l
)
189 wcwidth_l(wchar_t wc
, locale_t loc
)
191 _RuneLocale
const *rl
;
197 rl
= _RUNE_LOCALE(loc
);
198 x
= _runetype_priv(rl
, wc
);
200 return ((unsigned)x
& _RUNETYPE_SWM
) >> _RUNETYPE_SWS
;
207 return wcwidth_l(wc
, _current_locale());
211 wcswidth_l(const wchar_t * __restrict ws
, size_t wn
, locale_t loc
)
213 _RuneLocale
const *rl
;
217 _DIAGASSERT(ws
!= NULL
);
219 rl
= _RUNE_LOCALE(loc
);
221 while (wn
> 0 && *ws
!= L
'\0') {
222 x
= _runetype_priv(rl
, *ws
);
223 if ((x
& _RUNETYPE_R
) == 0)
225 width
+= ((unsigned)x
& _RUNETYPE_SWM
) >> _RUNETYPE_SWS
;
232 wcswidth(const wchar_t * __restrict ws
, size_t wn
)
234 return wcswidth_l(ws
, wn
, _current_locale());