- support DirectInput 8 interfaces.
[wine/multimedia.git] / dlls / ntdll / wcstring.c
blobb366b5ed3624a7b37512d3e41bc24d34703feb14
1 /*
2 * NTDLL wide-char functions
4 * Copyright 2000 Alexandre Julliard
5 * Copyright 2000 Jon Griffiths
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 #include "config.h"
24 #include <ctype.h>
25 #include <limits.h>
26 #include <stdlib.h>
27 #include <string.h>
28 #include <stdio.h>
30 #include "ntddk.h"
31 #include "wine/unicode.h"
32 #include "wine/debug.h"
34 WINE_DEFAULT_DEBUG_CHANNEL(ntdll);
37 /*********************************************************************
38 * _wcsicmp (NTDLL.@)
40 INT __cdecl NTDLL__wcsicmp( LPCWSTR str1, LPCWSTR str2 )
42 return strcmpiW( str1, str2 );
46 /*********************************************************************
47 * _wcslwr (NTDLL.@)
49 LPWSTR __cdecl NTDLL__wcslwr( LPWSTR str )
51 return strlwrW( str );
55 /*********************************************************************
56 * _wcsnicmp (NTDLL.@)
58 INT __cdecl NTDLL__wcsnicmp( LPCWSTR str1, LPCWSTR str2, INT n )
60 return strncmpiW( str1, str2, n );
64 /*********************************************************************
65 * _wcsupr (NTDLL.@)
67 LPWSTR __cdecl NTDLL__wcsupr( LPWSTR str )
69 return struprW( str );
73 /*********************************************************************
74 * towlower (NTDLL.@)
76 WCHAR __cdecl NTDLL_towlower( WCHAR ch )
78 return tolowerW(ch);
82 /*********************************************************************
83 * towupper (NTDLL.@)
85 WCHAR __cdecl NTDLL_towupper( WCHAR ch )
87 return toupperW(ch);
91 /***********************************************************************
92 * wcscat (NTDLL.@)
94 LPWSTR __cdecl NTDLL_wcscat( LPWSTR dst, LPCWSTR src )
96 return strcatW( dst, src );
100 /*********************************************************************
101 * wcschr (NTDLL.@)
103 LPWSTR __cdecl NTDLL_wcschr( LPCWSTR str, WCHAR ch )
105 return strchrW( str, ch );
109 /*********************************************************************
110 * wcscmp (NTDLL.@)
112 INT __cdecl NTDLL_wcscmp( LPCWSTR str1, LPCWSTR str2 )
114 return strcmpW( str1, str2 );
118 /***********************************************************************
119 * wcscpy (NTDLL.@)
121 LPWSTR __cdecl NTDLL_wcscpy( LPWSTR dst, LPCWSTR src )
123 return strcpyW( dst, src );
127 /*********************************************************************
128 * wcscspn (NTDLL.@)
130 INT __cdecl NTDLL_wcscspn( LPCWSTR str, LPCWSTR reject )
132 LPCWSTR start = str;
133 while (*str)
135 LPCWSTR p = reject;
136 while (*p && (*p != *str)) p++;
137 if (*p) break;
138 str++;
140 return str - start;
144 /***********************************************************************
145 * wcslen (NTDLL.@)
147 INT __cdecl NTDLL_wcslen( LPCWSTR str )
149 return strlenW( str );
153 /*********************************************************************
154 * wcsncat (NTDLL.@)
156 LPWSTR __cdecl NTDLL_wcsncat( LPWSTR s1, LPCWSTR s2, INT n )
158 LPWSTR ret = s1;
159 while (*s1) s1++;
160 while (n-- > 0) if (!(*s1++ = *s2++)) return ret;
161 *s1 = 0;
162 return ret;
166 /*********************************************************************
167 * wcsncmp (NTDLL.@)
169 INT __cdecl NTDLL_wcsncmp( LPCWSTR str1, LPCWSTR str2, INT n )
171 return strncmpW( str1, str2, n );
175 /*********************************************************************
176 * wcsncpy (NTDLL.@)
178 LPWSTR __cdecl NTDLL_wcsncpy( LPWSTR s1, LPCWSTR s2, INT n )
180 return strncpyW( s1, s2, n );
184 /*********************************************************************
185 * wcspbrk (NTDLL.@)
187 LPWSTR __cdecl NTDLL_wcspbrk( LPCWSTR str, LPCWSTR accept )
189 LPCWSTR p;
190 while (*str)
192 for (p = accept; *p; p++) if (*p == *str) return (LPWSTR)str;
193 str++;
195 return NULL;
199 /*********************************************************************
200 * wcsrchr (NTDLL.@)
202 LPWSTR __cdecl NTDLL_wcsrchr( LPWSTR str, WCHAR ch )
204 LPWSTR last = NULL;
205 while (*str)
207 if (*str == ch) last = str;
208 str++;
210 return last;
214 /*********************************************************************
215 * wcsspn (NTDLL.@)
217 INT __cdecl NTDLL_wcsspn( LPCWSTR str, LPCWSTR accept )
219 LPCWSTR start = str;
220 while (*str)
222 LPCWSTR p = accept;
223 while (*p && (*p != *str)) p++;
224 if (!*p) break;
225 str++;
227 return str - start;
231 /*********************************************************************
232 * wcsstr (NTDLL.@)
234 LPWSTR __cdecl NTDLL_wcsstr( LPCWSTR str, LPCWSTR sub )
236 return strstrW( str, sub );
240 /*********************************************************************
241 * wcstok (NTDLL.@)
243 LPWSTR __cdecl NTDLL_wcstok( LPWSTR str, LPCWSTR delim )
245 static LPWSTR next = NULL;
246 LPWSTR ret;
248 if (!str)
249 if (!(str = next)) return NULL;
251 while (*str && NTDLL_wcschr( delim, *str )) str++;
252 if (!*str) return NULL;
253 ret = str++;
254 while (*str && !NTDLL_wcschr( delim, *str )) str++;
255 if (*str) *str++ = 0;
256 next = str;
257 return ret;
261 /*********************************************************************
262 * wcstombs (NTDLL.@)
264 INT __cdecl NTDLL_wcstombs( LPSTR dst, LPCWSTR src, INT n )
266 DWORD len;
268 if (!dst)
270 RtlUnicodeToMultiByteSize( &len, src, strlenW(src)*sizeof(WCHAR) );
271 return len;
273 else
275 if (n <= 0) return 0;
276 RtlUnicodeToMultiByteN( dst, n, &len, src, strlenW(src)*sizeof(WCHAR) );
277 if (len < n) dst[len] = 0;
279 return len;
283 /*********************************************************************
284 * mbstowcs (NTDLL.@)
286 INT __cdecl NTDLL_mbstowcs( LPWSTR dst, LPCSTR src, INT n )
288 DWORD len;
290 if (!dst)
292 RtlMultiByteToUnicodeSize( &len, src, strlen(src) );
294 else
296 if (n <= 0) return 0;
297 RtlMultiByteToUnicodeN( dst, n*sizeof(WCHAR), &len, src, strlen(src) );
298 if (len / sizeof(WCHAR) < n) dst[len / sizeof(WCHAR)] = 0;
300 return len / sizeof(WCHAR);
304 /*********************************************************************
305 * wcstol (NTDLL.@)
307 long __cdecl NTDLL_wcstol(LPCWSTR s,LPWSTR *end,INT base)
309 return strtolW( s, end, base );
313 /*********************************************************************
314 * wcstoul (NTDLL.@)
316 unsigned long __cdecl NTDLL_wcstoul(LPCWSTR s,LPWSTR *end,INT base)
318 return strtoulW( s, end, base );
322 /*********************************************************************
323 * iswctype (NTDLL.@)
325 INT __cdecl NTDLL_iswctype( WCHAR wc, WCHAR wct )
327 return (get_char_typeW(wc) & 0xfff) & wct;
331 /*********************************************************************
332 * iswalpha (NTDLL.@)
334 INT __cdecl NTDLL_iswalpha( WCHAR wc )
336 return get_char_typeW(wc) & C1_ALPHA;
340 /*********************************************************************
341 * _ultow (NTDLL.@)
342 * Like _ultoa, but for wide character strings.
344 LPWSTR __cdecl _ultow(ULONG value, LPWSTR string, INT radix)
346 WCHAR tmp[33];
347 LPWSTR tp = tmp;
348 LPWSTR sp;
349 LONG i;
350 ULONG v = value;
352 if (radix > 36 || radix <= 1)
353 return 0;
355 while (v || tp == tmp)
357 i = v % radix;
358 v = v / radix;
359 if (i < 10)
360 *tp++ = i + '0';
361 else
362 *tp++ = i + 'a' - 10;
365 sp = string;
366 while (tp > tmp)
367 *sp++ = *--tp;
368 *sp = 0;
369 return string;
372 /*********************************************************************
373 * _wtol (NTDLL.@)
374 * Like atol, but for wide character strings.
376 LONG __cdecl NTDLL__wtol(LPWSTR string)
378 return strtolW( string, NULL, 10 );
381 /*********************************************************************
382 * _wtoi (NTDLL.@)
384 INT __cdecl NTDLL__wtoi(LPWSTR string)
386 return NTDLL__wtol(string);
389 /* INTERNAL: Wide char snprintf
390 * If you fix a bug in this function, fix it in msvcrt/wcs.c also!
392 static int __cdecl NTDLL_vsnwprintf(WCHAR *str, unsigned int len,
393 const WCHAR *format, va_list valist)
395 unsigned int written = 0;
396 const WCHAR *iter = format;
397 char bufa[256], fmtbufa[64], *fmta;
399 TRACE("(%d,%s)\n",len,debugstr_w(format));
401 while (*iter)
403 while (*iter && *iter != (WCHAR)L'%')
405 if (written++ >= len)
406 return -1;
407 *str++ = *iter++;
409 if (*iter == (WCHAR)L'%')
411 fmta = fmtbufa;
412 *fmta++ = *iter++;
413 while (*iter == (WCHAR)L'0' ||
414 *iter == (WCHAR)L'+' ||
415 *iter == (WCHAR)L'-' ||
416 *iter == (WCHAR)L' ' ||
417 *iter == (WCHAR)L'0' ||
418 *iter == (WCHAR)L'*' ||
419 *iter == (WCHAR)L'#')
421 if (*iter == (WCHAR)L'*')
423 char *buffiter = bufa;
424 int fieldlen = va_arg(valist, int);
425 sprintf(buffiter, "%d", fieldlen);
426 while (*buffiter)
427 *fmta++ = *buffiter++;
429 else
430 *fmta++ = *iter;
431 iter++;
434 while (isdigit(*iter))
435 *fmta++ = *iter++;
437 if (*iter == (WCHAR)L'.')
439 *fmta++ = *iter++;
440 if (*iter == (WCHAR)L'*')
442 char *buffiter = bufa;
443 int fieldlen = va_arg(valist, int);
444 sprintf(buffiter, "%d", fieldlen);
445 while (*buffiter)
446 *fmta++ = *buffiter++;
448 else
449 while (isdigit(*iter))
450 *fmta++ = *iter++;
452 if (*iter == (WCHAR)L'h' ||
453 *iter == (WCHAR)L'l')
454 *fmta++ = *iter++;
456 switch (*iter)
458 case (WCHAR)L's':
460 static const WCHAR none[] = { '(', 'n', 'u', 'l', 'l', ')', 0 };
461 const WCHAR *wstr = va_arg(valist, const WCHAR *);
462 const WCHAR *striter = wstr ? wstr : none;
463 while (*striter)
465 if (written++ >= len)
466 return -1;
467 *str++ = *striter++;
469 iter++;
470 break;
473 case (WCHAR)L'c':
474 if (written++ >= len)
475 return -1;
476 *str++ = (WCHAR)va_arg(valist, int);
477 iter++;
478 break;
480 default:
482 /* For non wc types, use system sprintf and append to wide char output */
483 /* FIXME: for unrecognised types, should ignore % when printing */
484 char *bufaiter = bufa;
485 if (*iter == (WCHAR)L'p')
486 sprintf(bufaiter, "%08lX", va_arg(valist, long));
487 else
489 *fmta++ = *iter;
490 *fmta = '\0';
491 if (*iter == (WCHAR)L'f')
492 sprintf(bufaiter, fmtbufa, va_arg(valist, double));
493 else
494 sprintf(bufaiter, fmtbufa, va_arg(valist, void *));
496 while (*bufaiter)
498 if (written++ >= len)
499 return -1;
500 *str++ = *bufaiter++;
502 iter++;
503 break;
508 if (written >= len)
509 return -1;
510 *str++ = (WCHAR)L'\0';
511 return (int)written;
515 /***********************************************************************
516 * _snwprintf (NTDLL.@)
518 int __cdecl _snwprintf(WCHAR *str, unsigned int len, const WCHAR *format, ...)
520 int retval;
521 va_list valist;
522 va_start(valist, format);
523 retval = NTDLL_vsnwprintf(str, len, format, valist);
524 va_end(valist);
525 return retval;
529 /***********************************************************************
530 * swprintf (NTDLL.@)
532 int __cdecl NTDLL_swprintf(WCHAR *str, const WCHAR *format, ...)
534 int retval;
535 va_list valist;
536 va_start(valist, format);
537 retval = NTDLL_vsnwprintf(str, INT_MAX, format, valist);
538 va_end(valist);
539 return retval;