/* * msvcrt.dll ctype functions * * Copyright 2000 Jon Griffiths * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include "msvcrt.h" #include "winnls.h" #include "wine/unicode.h" /* Some abbreviations to make the following table readable */ #define _C_ MSVCRT__CONTROL #define _S_ MSVCRT__SPACE #define _P_ MSVCRT__PUNCT #define _D_ MSVCRT__DIGIT #define _H_ MSVCRT__HEX #define _U_ MSVCRT__UPPER #define _L_ MSVCRT__LOWER WORD MSVCRT__ctype [257] = { 0, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _S_|_C_, _S_|_C_, _S_|_C_, _S_|_C_, _S_|_C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _C_, _S_|MSVCRT__BLANK, _P_, _P_, _P_, _P_, _P_, _P_, _P_, _P_, _P_, _P_, _P_, _P_, _P_, _P_, _P_, _D_|_H_, _D_|_H_, _D_|_H_, _D_|_H_, _D_|_H_, _D_|_H_, _D_|_H_, _D_|_H_, _D_|_H_, _D_|_H_, _P_, _P_, _P_, _P_, _P_, _P_, _P_, _U_|_H_, _U_|_H_, _U_|_H_, _U_|_H_, _U_|_H_, _U_|_H_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _U_, _P_, _P_, _P_, _P_, _P_, _P_, _L_|_H_, _L_|_H_, _L_|_H_, _L_|_H_, _L_|_H_, _L_|_H_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _L_, _P_, _P_, _P_, _P_, _C_, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; /* pctype is used by macros in the Win32 headers. It must point * To a table of flags exactly like ctype. To allow locale * changes to affect ctypes (i.e. isleadbyte), we use a second table * and update its flags whenever the current locale changes. */ unsigned short *MSVCRT__pctype = NULL; /********************************************************************* * __pctype_func (MSVCRT.@) */ const unsigned short* CDECL MSVCRT___pctype_func(void) { return get_locale()->locinfo->pctype; } /********************************************************************* * _isctype_l (MSVCRT.@) */ int CDECL _isctype_l(int c, int type, MSVCRT__locale_t locale) { if(!locale) locale = get_locale(); if (c >= -1 && c <= 255) return locale->locinfo->pctype[c] & type; if (locale->locinfo->mb_cur_max != 1 && c > 0) { /* FIXME: Is there a faster way to do this? */ WORD typeInfo; char convert[3], *pconv = convert; if (locale->locinfo->pctype[(UINT)c >> 8] & MSVCRT__LEADBYTE) *pconv++ = (UINT)c >> 8; *pconv++ = c & 0xff; *pconv = 0; if (GetStringTypeExA(locale->locinfo->lc_handle[MSVCRT_LC_CTYPE], CT_CTYPE1, convert, convert[1] ? 2 : 1, &typeInfo)) return typeInfo & type; } return 0; } /********************************************************************* * _isctype (MSVCRT.@) */ int CDECL _isctype(int c, int type) { return _isctype_l(c, type, NULL); } /********************************************************************* * _isalnum_l (MSVCRT.@) */ int CDECL MSVCRT__isalnum_l(int c, MSVCRT__locale_t locale) { return _isctype_l( c, MSVCRT__ALPHA | MSVCRT__DIGIT, locale ); } /********************************************************************* * isalnum (MSVCRT.@) */ int CDECL MSVCRT_isalnum(int c) { return _isctype( c, MSVCRT__ALPHA | MSVCRT__DIGIT ); } /********************************************************************* * _isalpha_l (MSVCRT.@) */ int CDECL MSVCRT__isalpha_l(int c, MSVCRT__locale_t locale) { return _isctype_l( c, MSVCRT__ALPHA, locale ); } /********************************************************************* * isalpha (MSVCRT.@) */ int CDECL MSVCRT_isalpha(int c) { return _isctype( c, MSVCRT__ALPHA ); } /********************************************************************* * _iscntrl_l (MSVCRT.@) */ int CDECL MSVCRT__iscntrl_l(int c, MSVCRT__locale_t locale) { return _isctype_l( c, MSVCRT__CONTROL, locale ); } /********************************************************************* * iscntrl (MSVCRT.@) */ int CDECL MSVCRT_iscntrl(int c) { return _isctype( c, MSVCRT__CONTROL ); } /********************************************************************* * _isdigit_l (MSVCRT.@) */ int CDECL MSVCRT__isdigit_l(int c, MSVCRT__locale_t locale) { return _isctype_l( c, MSVCRT__DIGIT, locale ); } /********************************************************************* * isdigit (MSVCRT.@) */ int CDECL MSVCRT_isdigit(int c) { return _isctype( c, MSVCRT__DIGIT ); } /********************************************************************* * _isgraph_l (MSVCRT.@) */ int CDECL MSVCRT__isgraph_l(int c, MSVCRT__locale_t locale) { return _isctype_l( c, MSVCRT__ALPHA | MSVCRT__DIGIT | MSVCRT__PUNCT, locale ); } /********************************************************************* * isgraph (MSVCRT.@) */ int CDECL MSVCRT_isgraph(int c) { return _isctype( c, MSVCRT__ALPHA | MSVCRT__DIGIT | MSVCRT__PUNCT ); } /********************************************************************* * _isleadbyte_l (MSVCRT.@) */ int CDECL MSVCRT__isleadbyte_l(int c, MSVCRT__locale_t locale) { return _isctype_l( c, MSVCRT__LEADBYTE, locale ); } /********************************************************************* * isleadbyte (MSVCRT.@) */ int CDECL MSVCRT_isleadbyte(int c) { return _isctype( c, MSVCRT__LEADBYTE ); } /********************************************************************* * _islower_l (MSVCRT.@) */ int CDECL MSVCRT__islower_l(int c, MSVCRT__locale_t locale) { return _isctype_l( c, MSVCRT__LOWER, locale ); } /********************************************************************* * islower (MSVCRT.@) */ int CDECL MSVCRT_islower(int c) { return _isctype( c, MSVCRT__LOWER ); } /********************************************************************* * _isprint_l (MSVCRT.@) */ int CDECL MSVCRT__isprint_l(int c, MSVCRT__locale_t locale) { return _isctype_l( c, MSVCRT__ALPHA | MSVCRT__DIGIT | MSVCRT__BLANK | MSVCRT__PUNCT, locale ); } /********************************************************************* * isprint (MSVCRT.@) */ int CDECL MSVCRT_isprint(int c) { return _isctype( c, MSVCRT__ALPHA | MSVCRT__DIGIT | MSVCRT__BLANK | MSVCRT__PUNCT ); } /********************************************************************* * ispunct (MSVCRT.@) */ int CDECL MSVCRT_ispunct(int c) { return _isctype( c, MSVCRT__PUNCT ); } /********************************************************************* * _isspace_l (MSVCRT.@) */ int CDECL MSVCRT__isspace_l(int c, MSVCRT__locale_t locale) { return _isctype_l( c, MSVCRT__SPACE, locale ); } /********************************************************************* * isspace (MSVCRT.@) */ int CDECL MSVCRT_isspace(int c) { return _isctype( c, MSVCRT__SPACE ); } /********************************************************************* * _isupper_l (MSVCRT.@) */ int CDECL MSVCRT__isupper_l(int c, MSVCRT__locale_t locale) { return _isctype_l( c, MSVCRT__UPPER, locale ); } /********************************************************************* * isupper (MSVCRT.@) */ int CDECL MSVCRT_isupper(int c) { return _isctype( c, MSVCRT__UPPER ); } /********************************************************************* * _isxdigit_l (MSVCRT.@) */ int CDECL MSVCRT__isxdigit_l(int c, MSVCRT__locale_t locale) { return _isctype_l( c, MSVCRT__HEX, locale ); } /********************************************************************* * isxdigit (MSVCRT.@) */ int CDECL MSVCRT_isxdigit(int c) { return _isctype( c, MSVCRT__HEX ); } /********************************************************************* * __isascii (MSVCRT.@) */ int CDECL MSVCRT___isascii(int c) { return isascii((unsigned)c); } /********************************************************************* * __toascii (MSVCRT.@) */ int CDECL MSVCRT___toascii(int c) { return (unsigned)c & 0x7f; } /********************************************************************* * iswascii (MSVCRT.@) * */ int CDECL MSVCRT_iswascii(MSVCRT_wchar_t c) { return ((unsigned)c < 0x80); } /********************************************************************* * __iscsym (MSVCRT.@) */ int CDECL MSVCRT___iscsym(int c) { return (c < 127 && (isalnum(c) || c == '_')); } /********************************************************************* * __iscsymf (MSVCRT.@) */ int CDECL MSVCRT___iscsymf(int c) { return (c < 127 && (isalpha(c) || c == '_')); } /********************************************************************* * _toupper_l (MSVCRT.@) */ int CDECL MSVCRT__toupper_l(int c, MSVCRT__locale_t locale) { if(!locale) locale = get_locale(); if(c < 256) return locale->locinfo->pcumap[c]; if(locale->locinfo->pctype[(c>>8)&255] & MSVCRT__LEADBYTE) { WCHAR wide, upper; char str[2], *p = str; *p++ = (c>>8) & 255; *p++ = c & 255; if(!MultiByteToWideChar(locale->locinfo->lc_codepage, MB_ERR_INVALID_CHARS, str, 2, &wide, 1)) return c; upper = toupperW(wide); if(upper == wide) return c; WideCharToMultiByte(locale->locinfo->lc_codepage, 0, &upper, 1, str, 2, NULL, NULL); return str[0] + (str[1]<<8); } return c; } /********************************************************************* * toupper (MSVCRT.@) */ int CDECL MSVCRT_toupper(int c) { return MSVCRT__toupper_l(c, NULL); } /********************************************************************* * _toupper (MSVCRT.@) */ int CDECL MSVCRT__toupper(int c) { return c - 0x20; /* sic */ } /********************************************************************* * _tolower_l (MSVCRT.@) */ int CDECL MSVCRT__tolower_l(int c, MSVCRT__locale_t locale) { if(!locale) locale = get_locale(); if(c < 256) return locale->locinfo->pclmap[c]; if(locale->locinfo->pctype[(c>>8)&255] & MSVCRT__LEADBYTE) { WCHAR wide, upper; char str[2], *p = str; *p++ = (c>>8) & 255; *p++ = c & 255; if(!MultiByteToWideChar(locale->locinfo->lc_codepage, MB_ERR_INVALID_CHARS, str, 2, &wide, 1)) return c; upper = tolowerW(wide); if(upper == wide) return c; WideCharToMultiByte(locale->locinfo->lc_codepage, 0, &upper, 1, str, 2, NULL, NULL); return str[0] + (str[1]<<8); } return c; } /********************************************************************* * tolower (MSVCRT.@) */ int CDECL MSVCRT_tolower(int c) { return MSVCRT__tolower_l(c, NULL); } /********************************************************************* * _tolower (MSVCRT.@) */ int CDECL MSVCRT__tolower(int c) { return c + 0x20; /* sic */ }