/* $NetBSD: iswctype_mb.c,v 1.11 2010/06/13 04:14:57 tnozaki Exp $ */ /*- * Copyright (c)2008 Citrus Project, * 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. */ #include #if defined(LIBC_SCCS) && !defined(lint) __RCSID("$NetBSD: iswctype_mb.c,v 1.11 2010/06/13 04:14:57 tnozaki Exp $"); #endif /* LIBC_SCCS and not lint */ #include "namespace.h" #include #include #include #define __SETLOCALE_SOURCE__ #include #include #include #include #include "setlocale_local.h" #include "runetype_local.h" #include "_wctype_local.h" #include "_wctrans_local.h" #define _RUNE_LOCALE() ((_RuneLocale const *) \ (*_current_locale())->part_impl[(size_t)LC_CTYPE]) #define _ISWCTYPE_FUNC(name, index) \ int \ isw##name(wint_t wc) \ { \ _RuneLocale const *rl; \ _WCTypeEntry const *te; \ \ rl = _RUNE_LOCALE(); \ te = &rl->rl_wctype[index]; \ return _iswctype_priv(rl, wc, te); \ } _ISWCTYPE_FUNC(alnum, _WCTYPE_INDEX_ALNUM) _ISWCTYPE_FUNC(alpha, _WCTYPE_INDEX_ALPHA) _ISWCTYPE_FUNC(blank, _WCTYPE_INDEX_BLANK) _ISWCTYPE_FUNC(cntrl, _WCTYPE_INDEX_CNTRL) _ISWCTYPE_FUNC(digit, _WCTYPE_INDEX_DIGIT) _ISWCTYPE_FUNC(graph, _WCTYPE_INDEX_GRAPH) _ISWCTYPE_FUNC(lower, _WCTYPE_INDEX_LOWER) _ISWCTYPE_FUNC(print, _WCTYPE_INDEX_PRINT) _ISWCTYPE_FUNC(punct, _WCTYPE_INDEX_PUNCT) _ISWCTYPE_FUNC(space, _WCTYPE_INDEX_SPACE) _ISWCTYPE_FUNC(upper, _WCTYPE_INDEX_UPPER) _ISWCTYPE_FUNC(xdigit, _WCTYPE_INDEX_XDIGIT) #define _TOWCTRANS_FUNC(name, index) \ wint_t \ tow##name(wint_t wc) \ { \ _RuneLocale const *rl; \ _WCTransEntry const *te; \ \ rl = _RUNE_LOCALE(); \ te = &rl->rl_wctrans[index]; \ return _towctrans_priv(wc, te); \ } _TOWCTRANS_FUNC(upper, _WCTRANS_INDEX_UPPER) _TOWCTRANS_FUNC(lower, _WCTRANS_INDEX_LOWER) wctype_t wctype(const char *charclass) { _RuneLocale const *rl; size_t i; rl = _RUNE_LOCALE(); for (i = 0; i < _WCTYPE_NINDEXES; ++i) { if (!strcmp(rl->rl_wctype[i].te_name, charclass)) return (wctype_t)__UNCONST(&rl->rl_wctype[i]); } return (wctype_t)NULL; } wctrans_t wctrans(const char *charmap) { _RuneLocale const *rl; size_t i; rl = _RUNE_LOCALE(); for (i = 0; i < _WCTRANS_NINDEXES; ++i) { _DIAGASSERT(rl->rl_wctrans[i].te_name != NULL); if (!strcmp(rl->rl_wctrans[i].te_name, charmap)) return (wctrans_t)__UNCONST(&rl->rl_wctype[i]); } return (wctrans_t)NULL; } int iswctype(wint_t wc, wctype_t charclass) { _RuneLocale const *rl; _WCTypeEntry const *te; if (charclass == NULL) { errno = EINVAL; return 0; } rl = _RUNE_LOCALE(); te = (_WCTypeEntry const *)(void *)charclass; return _iswctype_priv(rl, wc, te); } wint_t towctrans(wint_t wc, wctrans_t charmap) { _WCTransEntry const *te; if (charmap == NULL) { errno = EINVAL; return wc; } te = (_WCTransEntry const *)(void *)charmap; return _towctrans_priv(wc, te); } __weak_alias(wcwidth,_wcwidth) int wcwidth(wchar_t wc) { _RuneLocale const *rl; _RuneType x; if (wc == L'\0') return 0; rl = _RUNE_LOCALE(); x = _runetype_priv(rl, wc); if (x & _RUNETYPE_R) return ((unsigned)x & _RUNETYPE_SWM) >> _RUNETYPE_SWS; return -1; } int wcswidth(const wchar_t * __restrict ws, size_t wn) { _RuneLocale const *rl; _RuneType x; int width; _DIAGASSERT(ws != NULL); rl = _RUNE_LOCALE(); width = 0; while (wn > 0 && *ws != L'\0') { x = _runetype_priv(rl, *ws); if ((x & _RUNETYPE_R) == 0) return -1; width += ((unsigned)x & _RUNETYPE_SWM) >> _RUNETYPE_SWS; ++ws, --wn; } return width; }