2000-06-23 18:16:30 +02:00
|
|
|
/*
|
|
|
|
* NTDLL wide-char functions
|
|
|
|
*
|
|
|
|
* Copyright 2000 Alexandre Julliard
|
2001-01-22 03:17:29 +01:00
|
|
|
* Copyright 2000 Jon Griffiths
|
2000-06-23 18:16:30 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <ctype.h>
|
2001-01-22 03:17:29 +01:00
|
|
|
#include <limits.h>
|
2000-06-23 18:16:30 +02:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2001-01-22 03:17:29 +01:00
|
|
|
#include <stdio.h>
|
2000-06-23 18:16:30 +02:00
|
|
|
|
|
|
|
#include "windef.h"
|
|
|
|
#include "winbase.h"
|
|
|
|
#include "winnls.h"
|
|
|
|
#include "wine/unicode.h"
|
|
|
|
#include "heap.h"
|
|
|
|
#include "debugtools.h"
|
|
|
|
|
|
|
|
DEFAULT_DEBUG_CHANNEL(ntdll);
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL__wcsicmp (NTDLL)
|
|
|
|
*/
|
|
|
|
INT __cdecl NTDLL__wcsicmp( LPCWSTR str1, LPCWSTR str2 )
|
|
|
|
{
|
|
|
|
return strcmpiW( str1, str2 );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL__wcslwr (NTDLL)
|
|
|
|
*/
|
|
|
|
LPWSTR __cdecl NTDLL__wcslwr( LPWSTR str )
|
|
|
|
{
|
|
|
|
return strlwrW( str );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL__wcsnicmp (NTDLL)
|
|
|
|
*/
|
|
|
|
INT __cdecl NTDLL__wcsnicmp( LPCWSTR str1, LPCWSTR str2, INT n )
|
|
|
|
{
|
|
|
|
return strncmpiW( str1, str2, n );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL__wcsupr (NTDLL)
|
|
|
|
*/
|
|
|
|
LPWSTR __cdecl NTDLL__wcsupr( LPWSTR str )
|
|
|
|
{
|
|
|
|
return struprW( str );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_towlower (NTDLL)
|
|
|
|
*/
|
|
|
|
WCHAR __cdecl NTDLL_towlower( WCHAR ch )
|
|
|
|
{
|
|
|
|
return tolowerW(ch);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_towupper (NTDLL)
|
|
|
|
*/
|
|
|
|
WCHAR __cdecl NTDLL_towupper( WCHAR ch )
|
|
|
|
{
|
|
|
|
return toupperW(ch);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* NTDLL_wcscat (NTDLL)
|
|
|
|
*/
|
|
|
|
LPWSTR __cdecl NTDLL_wcscat( LPWSTR dst, LPCWSTR src )
|
|
|
|
{
|
|
|
|
return strcatW( dst, src );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_wcschr (NTDLL)
|
|
|
|
*/
|
|
|
|
LPWSTR __cdecl NTDLL_wcschr( LPCWSTR str, WCHAR ch )
|
|
|
|
{
|
|
|
|
return strchrW( str, ch );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_wcscmp (NTDLL)
|
|
|
|
*/
|
|
|
|
INT __cdecl NTDLL_wcscmp( LPCWSTR str1, LPCWSTR str2 )
|
|
|
|
{
|
|
|
|
return strcmpW( str1, str2 );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* NTDLL_wcscpy (NTDLL)
|
|
|
|
*/
|
|
|
|
LPWSTR __cdecl NTDLL_wcscpy( LPWSTR dst, LPCWSTR src )
|
|
|
|
{
|
|
|
|
return strcpyW( dst, src );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_wcscspn (NTDLL)
|
|
|
|
*/
|
|
|
|
INT __cdecl NTDLL_wcscspn( LPCWSTR str, LPCWSTR reject )
|
|
|
|
{
|
|
|
|
LPCWSTR start = str;
|
|
|
|
while (*str)
|
|
|
|
{
|
|
|
|
LPCWSTR p = reject;
|
|
|
|
while (*p && (*p != *str)) p++;
|
|
|
|
if (*p) break;
|
|
|
|
str++;
|
|
|
|
}
|
|
|
|
return str - start;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* NTDLL_wcslen (NTDLL)
|
|
|
|
*/
|
|
|
|
INT __cdecl NTDLL_wcslen( LPCWSTR str )
|
|
|
|
{
|
|
|
|
return strlenW( str );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_wcsncat (NTDLL)
|
|
|
|
*/
|
|
|
|
LPWSTR __cdecl NTDLL_wcsncat( LPWSTR s1, LPCWSTR s2, INT n )
|
|
|
|
{
|
|
|
|
LPWSTR ret = s1;
|
|
|
|
while (*s1) s1++;
|
|
|
|
while (n-- > 0) if (!(*s1++ = *s2++)) return ret;
|
|
|
|
*s1 = 0;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_wcsncmp (NTDLL)
|
|
|
|
*/
|
|
|
|
INT __cdecl NTDLL_wcsncmp( LPCWSTR str1, LPCWSTR str2, INT n )
|
|
|
|
{
|
|
|
|
return strncmpW( str1, str2, n );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_wcsncpy (NTDLL)
|
|
|
|
*/
|
|
|
|
LPWSTR __cdecl NTDLL_wcsncpy( LPWSTR s1, LPCWSTR s2, INT n )
|
|
|
|
{
|
2000-08-14 16:41:19 +02:00
|
|
|
return strncpyW( s1, s2, n );
|
2000-06-23 18:16:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_wcspbrk (NTDLL)
|
|
|
|
*/
|
|
|
|
LPWSTR __cdecl NTDLL_wcspbrk( LPCWSTR str, LPCWSTR accept )
|
|
|
|
{
|
|
|
|
LPCWSTR p;
|
|
|
|
while (*str)
|
|
|
|
{
|
|
|
|
for (p = accept; *p; p++) if (*p == *str) return (LPWSTR)str;
|
|
|
|
str++;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_wcsrchr (NTDLL)
|
|
|
|
*/
|
|
|
|
LPWSTR __cdecl NTDLL_wcsrchr( LPWSTR str, WCHAR ch )
|
|
|
|
{
|
|
|
|
LPWSTR last = NULL;
|
|
|
|
while (*str)
|
|
|
|
{
|
|
|
|
if (*str == ch) last = str;
|
|
|
|
str++;
|
|
|
|
}
|
|
|
|
return last;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_wcsspn (NTDLL)
|
|
|
|
*/
|
|
|
|
INT __cdecl NTDLL_wcsspn( LPCWSTR str, LPCWSTR accept )
|
|
|
|
{
|
|
|
|
LPCWSTR start = str;
|
|
|
|
while (*str)
|
|
|
|
{
|
|
|
|
LPCWSTR p = accept;
|
|
|
|
while (*p && (*p != *str)) p++;
|
|
|
|
if (!*p) break;
|
|
|
|
str++;
|
|
|
|
}
|
|
|
|
return str - start;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_wcsstr (NTDLL)
|
|
|
|
*/
|
|
|
|
LPWSTR __cdecl NTDLL_wcsstr( LPCWSTR str, LPCWSTR sub )
|
|
|
|
{
|
|
|
|
return strstrW( str, sub );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_wcstok (NTDLL)
|
|
|
|
*/
|
|
|
|
LPWSTR __cdecl NTDLL_wcstok( LPWSTR str, LPCWSTR delim )
|
|
|
|
{
|
|
|
|
static LPWSTR next = NULL;
|
|
|
|
LPWSTR ret;
|
|
|
|
|
|
|
|
if (!str)
|
|
|
|
if (!(str = next)) return NULL;
|
|
|
|
|
|
|
|
while (*str && NTDLL_wcschr( delim, *str )) str++;
|
|
|
|
if (!*str) return NULL;
|
|
|
|
ret = str++;
|
|
|
|
while (*str && !NTDLL_wcschr( delim, *str )) str++;
|
|
|
|
if (*str) *str++ = 0;
|
|
|
|
next = str;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_wcstombs (NTDLL)
|
|
|
|
*/
|
|
|
|
INT __cdecl NTDLL_wcstombs( LPSTR dst, LPCWSTR src, INT n )
|
|
|
|
{
|
|
|
|
INT ret;
|
|
|
|
if (n <= 0) return 0;
|
|
|
|
ret = WideCharToMultiByte( CP_ACP, 0, src, -1, dst, dst ? n : 0, NULL, NULL );
|
|
|
|
if (!ret) return n; /* overflow */
|
|
|
|
return ret - 1; /* do not count terminating NULL */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_mbstowcs (NTDLL)
|
|
|
|
*/
|
|
|
|
INT __cdecl NTDLL_mbstowcs( LPWSTR dst, LPCSTR src, INT n )
|
|
|
|
{
|
|
|
|
INT ret;
|
|
|
|
if (n <= 0) return 0;
|
|
|
|
ret = MultiByteToWideChar( CP_ACP, 0, src, -1, dst, dst ? n : 0 );
|
|
|
|
if (!ret) return n; /* overflow */
|
|
|
|
return ret - 1; /* do not count terminating NULL */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* wcstol (NTDLL)
|
|
|
|
* Like strtol, but for wide character strings.
|
|
|
|
*/
|
|
|
|
INT __cdecl NTDLL_wcstol(LPWSTR s,LPWSTR *end,INT base)
|
|
|
|
{
|
|
|
|
LPSTR sA = HEAP_strdupWtoA(GetProcessHeap(),0,s),endA;
|
|
|
|
INT ret = strtol(sA,&endA,base);
|
|
|
|
|
|
|
|
HeapFree(GetProcessHeap(),0,sA);
|
|
|
|
if (end) *end = s+(endA-sA); /* pointer magic checked. */
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_iswctype (NTDLL)
|
|
|
|
*/
|
|
|
|
INT __cdecl NTDLL_iswctype( WCHAR wc, WCHAR wct )
|
|
|
|
{
|
2000-08-11 22:53:40 +02:00
|
|
|
return (get_char_typeW(wc) & 0xfff) & wct;
|
2000-06-23 18:16:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* NTDLL_iswalpha (NTDLL)
|
|
|
|
*/
|
|
|
|
INT __cdecl NTDLL_iswalpha( WCHAR wc )
|
|
|
|
{
|
2000-08-11 22:53:40 +02:00
|
|
|
return get_char_typeW(wc) & C1_ALPHA;
|
2000-06-23 18:16:30 +02:00
|
|
|
}
|
2000-12-13 22:28:15 +01:00
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************
|
|
|
|
* _ultow (NTDLL)
|
|
|
|
* Like _ultoa, but for wide character strings.
|
|
|
|
*/
|
|
|
|
LPWSTR __cdecl _ultow(ULONG value, LPWSTR string, INT radix)
|
|
|
|
{
|
|
|
|
WCHAR tmp[33];
|
|
|
|
LPWSTR tp = tmp;
|
|
|
|
LPWSTR sp;
|
|
|
|
LONG i;
|
|
|
|
ULONG v = value;
|
|
|
|
|
|
|
|
if (radix > 36 || radix <= 1)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
while (v || tp == tmp)
|
|
|
|
{
|
|
|
|
i = v % radix;
|
|
|
|
v = v / radix;
|
|
|
|
if (i < 10)
|
|
|
|
*tp++ = i + '0';
|
|
|
|
else
|
|
|
|
*tp++ = i + 'a' - 10;
|
|
|
|
}
|
|
|
|
|
|
|
|
sp = string;
|
|
|
|
while (tp > tmp)
|
|
|
|
*sp++ = *--tp;
|
|
|
|
*sp = 0;
|
|
|
|
return string;
|
|
|
|
}
|
2001-01-22 03:17:29 +01:00
|
|
|
|
|
|
|
|
|
|
|
/* INTERNAL: Wide char snprintf
|
|
|
|
* If you fix a bug in this function, fix it in msvcrt/wcs.c also!
|
|
|
|
*/
|
|
|
|
static int __cdecl NTDLL_vsnwprintf(WCHAR *str, unsigned int len,
|
|
|
|
const WCHAR *format, va_list valist)
|
|
|
|
{
|
|
|
|
unsigned int written = 0;
|
|
|
|
const WCHAR *iter = format;
|
|
|
|
char bufa[256], fmtbufa[64], *fmta;
|
|
|
|
|
|
|
|
TRACE("(%d,%s)\n",len,debugstr_w(format));
|
|
|
|
|
|
|
|
while (*iter)
|
|
|
|
{
|
|
|
|
while (*iter && *iter != (WCHAR)L'%')
|
|
|
|
{
|
|
|
|
if (written++ >= len)
|
|
|
|
return -1;
|
|
|
|
*str++ = *iter++;
|
|
|
|
}
|
|
|
|
if (*iter == (WCHAR)L'%')
|
|
|
|
{
|
|
|
|
fmta = fmtbufa;
|
|
|
|
*fmta++ = *iter++;
|
|
|
|
while (*iter == (WCHAR)L'0' ||
|
|
|
|
*iter == (WCHAR)L'+' ||
|
|
|
|
*iter == (WCHAR)L'-' ||
|
|
|
|
*iter == (WCHAR)L' ' ||
|
|
|
|
*iter == (WCHAR)L'0' ||
|
|
|
|
*iter == (WCHAR)L'*' ||
|
|
|
|
*iter == (WCHAR)L'#')
|
|
|
|
{
|
|
|
|
if (*iter == (WCHAR)L'*')
|
|
|
|
{
|
|
|
|
char *buffiter = bufa;
|
|
|
|
int fieldlen = va_arg(valist, int);
|
|
|
|
sprintf(buffiter, "%d", fieldlen);
|
|
|
|
while (*buffiter)
|
|
|
|
*fmta++ = *buffiter++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
*fmta++ = *iter;
|
|
|
|
iter++;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (isdigit(*iter))
|
|
|
|
*fmta++ = *iter++;
|
|
|
|
|
|
|
|
if (*iter == (WCHAR)L'.')
|
|
|
|
{
|
|
|
|
*fmta++ = *iter++;
|
|
|
|
if (*iter == (WCHAR)L'*')
|
|
|
|
{
|
|
|
|
char *buffiter = bufa;
|
|
|
|
int fieldlen = va_arg(valist, int);
|
|
|
|
sprintf(buffiter, "%d", fieldlen);
|
|
|
|
while (*buffiter)
|
|
|
|
*fmta++ = *buffiter++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
while (isdigit(*iter))
|
|
|
|
*fmta++ = *iter++;
|
|
|
|
}
|
|
|
|
if (*iter == (WCHAR)L'h' ||
|
|
|
|
*iter == (WCHAR)L'l')
|
|
|
|
{
|
|
|
|
*fmta++ = *iter++;
|
|
|
|
*fmta++ = *iter++;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (*iter)
|
|
|
|
{
|
|
|
|
case (WCHAR)L's':
|
|
|
|
{
|
|
|
|
static const WCHAR none[] = { '(', 'n', 'u', 'l', 'l', ')', 0 };
|
|
|
|
const WCHAR *wstr = va_arg(valist, const WCHAR *);
|
|
|
|
const WCHAR *striter = wstr ? wstr : none;
|
|
|
|
while (*striter)
|
|
|
|
{
|
|
|
|
if (written++ >= len)
|
|
|
|
return -1;
|
|
|
|
*str++ = *striter++;
|
|
|
|
}
|
|
|
|
iter++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case (WCHAR)L'c':
|
|
|
|
if (written++ >= len)
|
|
|
|
return -1;
|
2001-01-22 20:26:42 +01:00
|
|
|
*str++ = (WCHAR)va_arg(valist, int);
|
2001-01-22 03:17:29 +01:00
|
|
|
iter++;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
/* For non wc types, use system sprintf and append to wide char output */
|
|
|
|
/* FIXME: for unrecognised types, should ignore % when printing */
|
|
|
|
char *bufaiter = bufa;
|
|
|
|
if (*iter == (WCHAR)L'p')
|
|
|
|
sprintf(bufaiter, "%08lX", va_arg(valist, long));
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*fmta++ = *iter;
|
|
|
|
*fmta = '\0';
|
|
|
|
sprintf(bufaiter, fmtbufa, va_arg(valist, void *));
|
|
|
|
}
|
|
|
|
while (*bufaiter)
|
|
|
|
{
|
|
|
|
if (written++ >= len)
|
|
|
|
return -1;
|
|
|
|
*str++ = *bufaiter++;
|
|
|
|
}
|
|
|
|
iter++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (written >= len)
|
|
|
|
return -1;
|
|
|
|
*str++ = (WCHAR)L'\0';
|
|
|
|
return (int)written;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* _snwprintf (NTDLL)
|
|
|
|
*/
|
|
|
|
int __cdecl _snwprintf(WCHAR *str, unsigned int len, const WCHAR *format, ...)
|
|
|
|
{
|
|
|
|
int retval;
|
|
|
|
va_list valist;
|
|
|
|
va_start(valist, format);
|
|
|
|
retval = NTDLL_vsnwprintf(str, len, format, valist);
|
|
|
|
va_end(valist);
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
* NTDLL_swprintf (NTDLL)
|
|
|
|
*/
|
|
|
|
int __cdecl NTDLL_swprintf(WCHAR *str, const WCHAR *format, ...)
|
|
|
|
{
|
|
|
|
int retval;
|
|
|
|
va_list valist;
|
|
|
|
va_start(valist, format);
|
|
|
|
retval = NTDLL_vsnwprintf(str, INT_MAX, format, valist);
|
|
|
|
va_end(valist);
|
|
|
|
return retval;
|
|
|
|
}
|