2 * NTDLL wide-char functions
4 * Copyright 2000 Alexandre Julliard
5 * Copyright 2000 Jon Griffiths
6 * Copyright 2003 Thomas Mertes
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2.1 of the License, or (at your option) any later version.
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with this library; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
33 #include "ntdll_misc.h"
35 static const unsigned short wctypes
[256] =
38 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020,
39 0x0020, 0x0068, 0x0028, 0x0028, 0x0028, 0x0028, 0x0020, 0x0020,
41 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020,
42 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020,
44 0x0048, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010,
45 0x0010, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010,
47 0x0084, 0x0084, 0x0084, 0x0084, 0x0084, 0x0084, 0x0084, 0x0084,
48 0x0084, 0x0084, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010,
50 0x0010, 0x0181, 0x0181, 0x0181, 0x0181, 0x0181, 0x0181, 0x0101,
51 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101,
53 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101,
54 0x0101, 0x0101, 0x0101, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010,
56 0x0010, 0x0182, 0x0182, 0x0182, 0x0182, 0x0182, 0x0182, 0x0102,
57 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102,
59 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102,
60 0x0102, 0x0102, 0x0102, 0x0010, 0x0010, 0x0010, 0x0010, 0x0020,
62 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020,
63 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020,
65 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020,
66 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020, 0x0020,
68 0x0048, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010,
69 0x0010, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010,
71 0x0010, 0x0010, 0x0014, 0x0014, 0x0010, 0x0010, 0x0010, 0x0010,
72 0x0010, 0x0014, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010, 0x0010,
74 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101,
75 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101,
77 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0010,
78 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0101, 0x0102,
80 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102,
81 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102,
83 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0010,
84 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102, 0x0102
88 /*********************************************************************
91 int __cdecl
_wcsicmp( LPCWSTR str1
, LPCWSTR str2
)
95 WCHAR ch1
= (*str1
>= 'A' && *str1
<= 'Z') ? *str1
+ 32 : *str1
;
96 WCHAR ch2
= (*str2
>= 'A' && *str2
<= 'Z') ? *str2
+ 32 : *str2
;
97 if (ch1
!= ch2
|| !*str1
) return ch1
- ch2
;
104 /*********************************************************************
107 LPWSTR __cdecl
_wcslwr( LPWSTR str
)
114 if (ch
>= 'A' && ch
<= 'Z') ch
+= 32;
121 /*********************************************************************
122 * _wcsnicmp (NTDLL.@)
124 int __cdecl
_wcsnicmp( LPCWSTR str1
, LPCWSTR str2
, size_t n
)
127 for ( ; n
> 0; n
--, str1
++, str2
++)
129 WCHAR ch1
= (*str1
>= 'A' && *str1
<= 'Z') ? *str1
+ 32 : *str1
;
130 WCHAR ch2
= (*str2
>= 'A' && *str2
<= 'Z') ? *str2
+ 32 : *str2
;
131 if ((ret
= ch1
- ch2
) || !*str1
) break;
137 /*********************************************************************
140 LPWSTR __cdecl
_wcsupr( LPWSTR str
)
147 if (ch
>= 'a' && ch
<= 'z') ch
-= 32;
154 /***********************************************************************
157 LPWSTR __cdecl
wcscpy( LPWSTR dst
, LPCWSTR src
)
160 while ((*p
++ = *src
++));
165 /***********************************************************************
168 size_t __cdecl
wcslen( LPCWSTR str
)
170 const WCHAR
*s
= str
;
176 /***********************************************************************
179 LPWSTR __cdecl
wcscat( LPWSTR dst
, LPCWSTR src
)
181 wcscpy( dst
+ wcslen(dst
), src
);
186 /*********************************************************************
189 LPWSTR __cdecl
wcschr( LPCWSTR str
, WCHAR ch
)
191 do { if (*str
== ch
) return (WCHAR
*)(ULONG_PTR
)str
; } while (*str
++);
196 /*********************************************************************
199 int __cdecl
wcscmp( LPCWSTR str1
, LPCWSTR str2
)
201 while (*str1
&& (*str1
== *str2
)) { str1
++; str2
++; }
202 return *str1
- *str2
;
206 /*********************************************************************
209 size_t __cdecl
wcscspn( LPCWSTR str
, LPCWSTR reject
)
212 for (ptr
= str
; *ptr
; ptr
++) if (wcschr( reject
, *ptr
)) break;
217 /*********************************************************************
220 LPWSTR __cdecl
wcsncat( LPWSTR s1
, LPCWSTR s2
, size_t n
)
224 while (n
-- > 0) if (!(*s1
++ = *s2
++)) return ret
;
230 /*********************************************************************
233 int __cdecl
wcsncmp( LPCWSTR str1
, LPCWSTR str2
, size_t n
)
235 if (n
<= 0) return 0;
236 while ((--n
> 0) && *str1
&& (*str1
== *str2
)) { str1
++; str2
++; }
237 return *str1
- *str2
;
241 /*********************************************************************
244 LPWSTR __cdecl
wcsncpy( LPWSTR s1
, LPCWSTR s2
, size_t n
)
247 for ( ; n
; n
--) if (!(*s1
++ = *s2
++)) break;
248 for ( ; n
; n
--) *s1
++ = 0;
253 /*********************************************************************
256 LPWSTR __cdecl
wcspbrk( LPCWSTR str
, LPCWSTR accept
)
258 for ( ; *str
; str
++) if (wcschr( accept
, *str
)) return (WCHAR
*)(ULONG_PTR
)str
;
263 /*********************************************************************
266 LPWSTR __cdecl
wcsrchr( LPCWSTR str
, WCHAR ch
)
269 do { if (*str
== ch
) ret
= (WCHAR
*)(ULONG_PTR
)str
; } while (*str
++);
274 /*********************************************************************
277 size_t __cdecl
wcsspn( LPCWSTR str
, LPCWSTR accept
)
280 for (ptr
= str
; *ptr
; ptr
++) if (!wcschr( accept
, *ptr
)) break;
285 /*********************************************************************
288 LPWSTR __cdecl
wcsstr( LPCWSTR str
, LPCWSTR sub
)
292 const WCHAR
*p1
= str
, *p2
= sub
;
293 while (*p1
&& *p2
&& *p1
== *p2
) { p1
++; p2
++; }
294 if (!*p2
) return (WCHAR
*)str
;
301 /*********************************************************************
304 LPWSTR __cdecl
wcstok( LPWSTR str
, LPCWSTR delim
)
306 static LPWSTR next
= NULL
;
310 if (!(str
= next
)) return NULL
;
312 while (*str
&& wcschr( delim
, *str
)) str
++;
313 if (!*str
) return NULL
;
315 while (*str
&& !wcschr( delim
, *str
)) str
++;
316 if (*str
) *str
++ = 0;
322 /*********************************************************************
325 size_t __cdecl
wcstombs( char *dst
, const WCHAR
*src
, size_t n
)
331 RtlUnicodeToMultiByteSize( &len
, src
, wcslen(src
) * sizeof(WCHAR
) );
336 if (n
<= 0) return 0;
337 RtlUnicodeToMultiByteN( dst
, n
, &len
, src
, wcslen(src
) * sizeof(WCHAR
) );
338 if (len
< n
) dst
[len
] = 0;
344 /*********************************************************************
347 size_t __cdecl
mbstowcs( WCHAR
*dst
, const char *src
, size_t n
)
353 RtlMultiByteToUnicodeSize( &len
, src
, strlen(src
) );
357 if (n
<= 0) return 0;
358 RtlMultiByteToUnicodeN( dst
, n
*sizeof(WCHAR
), &len
, src
, strlen(src
) );
359 if (len
/ sizeof(WCHAR
) < n
) dst
[len
/ sizeof(WCHAR
)] = 0;
361 return len
/ sizeof(WCHAR
);
365 /*********************************************************************
368 INT __cdecl
iswctype( WCHAR wc
, unsigned short type
)
370 if (wc
>= 256) return 0;
371 return wctypes
[wc
] & type
;
375 /*********************************************************************
378 INT __cdecl
iswalpha( WCHAR wc
)
380 if (wc
>= 256) return 0;
381 return wctypes
[wc
] & (C1_ALPHA
| C1_UPPER
| C1_LOWER
);
385 /*********************************************************************
388 * Checks if a unicode char wc is a digit
391 * TRUE: The unicode char wc is a digit.
394 INT __cdecl
iswdigit( WCHAR wc
)
396 if (wc
>= 256) return 0;
397 return wctypes
[wc
] & C1_DIGIT
;
401 /*********************************************************************
404 * Checks if a unicode char wc is a lower case letter
407 * TRUE: The unicode char wc is a lower case letter.
410 INT __cdecl
iswlower( WCHAR wc
)
412 if (wc
>= 256) return 0;
413 return wctypes
[wc
] & C1_LOWER
;
417 /*********************************************************************
420 * Checks if a unicode char wc is a white space character
423 * TRUE: The unicode char wc is a white space character.
426 INT __cdecl
iswspace( WCHAR wc
)
428 if (wc
>= 256) return 0;
429 return wctypes
[wc
] & C1_SPACE
;
433 /*********************************************************************
434 * iswxdigit (NTDLL.@)
436 * Checks if a unicode char wc is an extended digit
439 * TRUE: The unicode char wc is an extended digit.
442 INT __cdecl
iswxdigit( WCHAR wc
)
444 if (wc
>= 256) return 0;
445 return wctypes
[wc
] & C1_XDIGIT
;
449 static int wctoint( WCHAR c
)
451 /* NOTE: MAP_FOLDDIGITS supports too many things. */
452 /* Unicode points that contain digits 0-9; keep this sorted! */
453 static const WCHAR zeros
[] =
455 0x0660, 0x06f0, 0x0966, 0x09e6, 0x0a66, 0x0ae6, 0x0b66, 0x0c66, 0x0ce6,
456 0x0d66, 0x0e50, 0x0ed0, 0x0f20, 0x1040, 0x17e0, 0x1810, 0xff10
460 if ('0' <= c
&& c
<= '9') return c
- '0';
461 if ('A' <= c
&& c
<= 'Z') return c
- 'A' + 10;
462 if ('a' <= c
&& c
<= 'z') return c
- 'a' + 10;
463 for (i
= 0; i
< ARRAY_SIZE(zeros
) && c
>= zeros
[i
]; i
++)
464 if (zeros
[i
] <= c
&& c
<= zeros
[i
] + 9) return c
- zeros
[i
];
469 /*********************************************************************
472 __msvcrt_long __cdecl
wcstol(LPCWSTR s
, LPWSTR
*end
, INT base
)
474 BOOL negative
= FALSE
, empty
= TRUE
;
477 if (base
< 0 || base
== 1 || base
> 36) return 0;
478 if (end
) *end
= (WCHAR
*)s
;
479 while (iswspace(*s
)) s
++;
486 else if (*s
== '+') s
++;
488 if ((base
== 0 || base
== 16) && !wctoint( *s
) && (s
[1] == 'x' || s
[1] == 'X'))
493 if (base
== 0) base
= wctoint( *s
) ? 10 : 8;
497 int v
= wctoint( *s
);
498 if (v
< 0 || v
>= base
) break;
499 if (negative
) v
= -v
;
503 if (!negative
&& (ret
> MAXLONG
/ base
|| ret
* base
> MAXLONG
- v
))
505 else if (negative
&& (ret
< (LONG
)MINLONG
/ base
|| ret
* base
< (LONG
)(MINLONG
- v
)))
508 ret
= ret
* base
+ v
;
511 if (end
&& !empty
) *end
= (WCHAR
*)s
;
516 /*********************************************************************
519 __msvcrt_ulong __cdecl
wcstoul(LPCWSTR s
, LPWSTR
*end
, INT base
)
521 BOOL negative
= FALSE
, empty
= TRUE
;
524 if (base
< 0 || base
== 1 || base
> 36) return 0;
525 if (end
) *end
= (WCHAR
*)s
;
526 while (iswspace(*s
)) s
++;
533 else if (*s
== '+') s
++;
535 if ((base
== 0 || base
== 16) && !wctoint( *s
) && (s
[1] == 'x' || s
[1] == 'X'))
540 if (base
== 0) base
= wctoint( *s
) ? 10 : 8;
544 int v
= wctoint( *s
);
545 if (v
< 0 || v
>= base
) break;
549 if (ret
> MAXDWORD
/ base
|| ret
* base
> MAXDWORD
- v
)
552 ret
= ret
* base
+ v
;
555 if (end
&& !empty
) *end
= (WCHAR
*)s
;
556 return negative
? -ret
: ret
;
560 /*********************************************************************
563 * Converts an unsigned long integer to a unicode string.
566 * Always returns str.
569 * Converts value to a '\0' terminated wstring which is copied to str.
570 * The maximum length of the copied str is 33 bytes.
571 * Does not check if radix is in the range of 2 to 36.
572 * If str is NULL it just returns NULL.
574 LPWSTR __cdecl
_ultow( __msvcrt_ulong value
, LPWSTR str
, INT radix
)
584 digit
= value
% radix
;
585 value
= value
/ radix
;
587 *--pos
= '0' + digit
;
589 *--pos
= 'a' + digit
- 10;
591 } while (value
!= 0L);
594 memcpy(str
, pos
, (&buffer
[32] - pos
+ 1) * sizeof(WCHAR
));
600 /*********************************************************************
603 * Converts a long integer to a unicode string.
606 * Always returns str.
609 * Converts value to a '\0' terminated wstring which is copied to str.
610 * The maximum length of the copied str is 33 bytes. If radix
611 * is 10 and value is negative, the value is converted with sign.
612 * Does not check if radix is in the range of 2 to 36.
613 * If str is NULL it just returns NULL.
615 LPWSTR __cdecl
_ltow( __msvcrt_long value
, LPWSTR str
, INT radix
)
623 if (value
< 0 && radix
== 10) {
638 *--pos
= '0' + digit
;
640 *--pos
= 'a' + digit
- 10;
649 memcpy(str
, pos
, (&buffer
[32] - pos
+ 1) * sizeof(WCHAR
));
655 /*********************************************************************
658 * Converts an integer to a unicode string.
661 * Always returns str.
664 * Converts value to a '\0' terminated wstring which is copied to str.
665 * The maximum length of the copied str is 33 bytes. If radix
666 * is 10 and value is negative, the value is converted with sign.
667 * Does not check if radix is in the range of 2 to 36.
668 * If str is NULL it just returns NULL.
671 * - The native function crashes when the string is longer than 19 chars.
672 * This function does not have this bug.
674 LPWSTR __cdecl
_itow(
675 int value
, /* [I] Value to be converted */
676 LPWSTR str
, /* [O] Destination for the converted value */
677 INT radix
) /* [I] Number base for conversion */
679 return _ltow(value
, str
, radix
);
683 /*********************************************************************
686 * Converts a large unsigned integer to a unicode string.
689 * Always returns str.
692 * Converts value to a '\0' terminated wstring which is copied to str.
693 * The maximum length of the copied str is 33 bytes.
694 * Does not check if radix is in the range of 2 to 36.
695 * If str is NULL it just returns NULL.
698 * - This function does not exist in the native DLL (but in msvcrt).
699 * But since the maintenance of all these functions is better done
700 * in one place we implement it here.
702 LPWSTR __cdecl
_ui64tow(
703 ULONGLONG value
, /* [I] Value to be converted */
704 LPWSTR str
, /* [O] Destination for the converted value */
705 INT radix
) /* [I] Number base for conversion */
715 digit
= value
% radix
;
716 value
= value
/ radix
;
718 *--pos
= '0' + digit
;
720 *--pos
= 'a' + digit
- 10;
722 } while (value
!= 0L);
725 memcpy(str
, pos
, (&buffer
[64] - pos
+ 1) * sizeof(WCHAR
));
731 /*********************************************************************
734 * Converts a large integer to a unicode string.
737 * Always returns str.
740 * Converts value to a '\0' terminated wstring which is copied to str.
741 * The maximum length of the copied str is 33 bytes. If radix
742 * is 10 and value is negative, the value is converted with sign.
743 * Does not check if radix is in the range of 2 to 36.
744 * If str is NULL it just returns NULL.
747 * - The native DLL converts negative values (for base 10) wrong:
748 * -1 is converted to -18446744073709551615
749 * -2 is converted to -18446744073709551614
750 * -9223372036854775807 is converted to -9223372036854775809
751 * -9223372036854775808 is converted to -9223372036854775808
752 * The native msvcrt _i64tow function and our ntdll function do
755 LPWSTR __cdecl
_i64tow(
756 LONGLONG value
, /* [I] Value to be converted */
757 LPWSTR str
, /* [O] Destination for the converted value */
758 INT radix
) /* [I] Number base for conversion */
766 if (value
< 0 && radix
== 10) {
781 *--pos
= '0' + digit
;
783 *--pos
= 'a' + digit
- 10;
792 memcpy(str
, pos
, (&buffer
[64] - pos
+ 1) * sizeof(WCHAR
));
798 /*********************************************************************
801 * Converts a unicode string to a long integer.
804 * str [I] Wstring to be converted
807 * On success it returns the integer value otherwise it returns 0.
810 * Accepts: {whitespace} [+|-] {digits}
811 * No check is made for value overflow, only the lower 32 bits are assigned.
812 * If str is NULL it crashes, as the native function does.
814 __msvcrt_long __cdecl
_wtol( LPCWSTR str
)
816 ULONG RunningTotal
= 0;
819 while (iswspace(*str
)) str
++;
823 } else if (*str
== '-') {
828 while (*str
>= '0' && *str
<= '9') {
829 RunningTotal
= RunningTotal
* 10 + *str
- '0';
833 return bMinus
? -RunningTotal
: RunningTotal
;
837 /*********************************************************************
840 * Converts a unicode string to an integer.
843 * str [I] Wstring to be converted
846 * On success it returns the integer value otherwise it returns 0.
849 * Accepts: {whitespace} [+|-] {digits}
850 * No check is made for value overflow, only the lower 32 bits are assigned.
851 * If str is NULL it crashes, as the native function does.
853 int __cdecl
_wtoi( LPCWSTR str
)
859 /*********************************************************************
862 * Converts a unicode string to a large integer.
865 * str [I] Wstring to be converted
868 * On success it returns the integer value otherwise it returns 0.
871 * Accepts: {whitespace} [+|-] {digits}
872 * No check is made for value overflow, only the lower 64 bits are assigned.
873 * If str is NULL it crashes, as the native function does.
875 LONGLONG __cdecl
_wtoi64( LPCWSTR str
)
877 ULONGLONG RunningTotal
= 0;
880 while (iswspace(*str
)) str
++;
884 } else if (*str
== '-') {
889 while (*str
>= '0' && *str
<= '9') {
890 RunningTotal
= RunningTotal
* 10 + *str
- '0';
894 return bMinus
? -RunningTotal
: RunningTotal
;