/* * Kernel string functions * * Copyright 1993 Yngvi Sigurjonsson * Copyright 1996 Alexandre Julliard * Copyright 2001 Dmitry Timoshkov for CodeWeavers * * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "config.h" #include "wine/port.h" #include <ctype.h> #include <stdarg.h> #include <string.h> #include "windef.h" #include "winbase.h" #include "excpt.h" #include "wine/winbase16.h" #include "wine/unicode.h" #include "wine/exception.h" static INT (WINAPI *pLoadStringA)(HINSTANCE, UINT, LPSTR, INT); static INT (WINAPI *pwvsprintfA)(LPSTR, LPCSTR, va_list); /* filter for page-fault exceptions */ static WINE_EXCEPTION_FILTER(page_fault) { if (GetExceptionCode() == EXCEPTION_ACCESS_VIOLATION) return EXCEPTION_EXECUTE_HANDLER; return EXCEPTION_CONTINUE_SEARCH; } /*********************************************************************** * Helper for k32 family functions */ static void *user32_proc_address(const char *proc_name) { static HMODULE hUser32; if(!hUser32) hUser32 = LoadLibraryA("user32.dll"); return GetProcAddress(hUser32, proc_name); } /*********************************************************************** * k32CharToOemBuffA (KERNEL32.11) */ BOOL WINAPI k32CharToOemBuffA(LPCSTR s, LPSTR d, DWORD len) { WCHAR *bufW; if ((bufW = HeapAlloc( GetProcessHeap(), 0, len * sizeof(WCHAR) ))) { MultiByteToWideChar( CP_ACP, 0, s, len, bufW, len ); WideCharToMultiByte( CP_OEMCP, 0, bufW, len, d, len, NULL, NULL ); HeapFree( GetProcessHeap(), 0, bufW ); } return TRUE; } /*********************************************************************** * k32CharToOemA (KERNEL32.10) */ BOOL WINAPI k32CharToOemA(LPCSTR s, LPSTR d) { if (!s || !d) return TRUE; return k32CharToOemBuffA( s, d, strlen(s) + 1 ); } /*********************************************************************** * k32OemToCharBuffA (KERNEL32.13) */ BOOL WINAPI k32OemToCharBuffA(LPCSTR s, LPSTR d, DWORD len) { WCHAR *bufW; if ((bufW = HeapAlloc( GetProcessHeap(), 0, len * sizeof(WCHAR) ))) { MultiByteToWideChar( CP_OEMCP, 0, s, len, bufW, len ); WideCharToMultiByte( CP_ACP, 0, bufW, len, d, len, NULL, NULL ); HeapFree( GetProcessHeap(), 0, bufW ); } return TRUE; } /*********************************************************************** * k32OemToCharA (KERNEL32.12) */ BOOL WINAPI k32OemToCharA(LPCSTR s, LPSTR d) { return k32OemToCharBuffA( s, d, strlen(s) + 1 ); } /********************************************************************** * k32LoadStringA (KERNEL32.14) */ INT WINAPI k32LoadStringA(HINSTANCE instance, UINT resource_id, LPSTR buffer, INT buflen) { if(!pLoadStringA) pLoadStringA = user32_proc_address("LoadStringA"); return pLoadStringA(instance, resource_id, buffer, buflen); } /*********************************************************************** * k32wvsprintfA (KERNEL32.16) */ INT WINAPI k32wvsprintfA(LPSTR buffer, LPCSTR spec, va_list args) { if(!pwvsprintfA) pwvsprintfA = user32_proc_address("wvsprintfA"); return (*pwvsprintfA)(buffer, spec, args); } /*********************************************************************** * k32wsprintfA (KERNEL32.15) */ INT WINAPIV k32wsprintfA(LPSTR buffer, LPCSTR spec, ...) { va_list args; INT res; va_start(args, spec); res = k32wvsprintfA(buffer, spec, args); va_end(args); return res; } /*********************************************************************** * hmemcpy (KERNEL.348) */ void WINAPI hmemcpy16( LPVOID dst, LPCVOID src, LONG count ) { memcpy( dst, src, count ); } /*********************************************************************** * lstrcat (KERNEL.89) */ SEGPTR WINAPI lstrcat16( SEGPTR dst, LPCSTR src ) { /* Windows does not check for NULL pointers here, so we don't either */ strcat( MapSL(dst), src ); return dst; } /*********************************************************************** * lstrcat (KERNEL32.@) * lstrcatA (KERNEL32.@) */ LPSTR WINAPI lstrcatA( LPSTR dst, LPCSTR src ) { __TRY { strcat( dst, src ); } __EXCEPT(page_fault) { SetLastError( ERROR_INVALID_PARAMETER ); return NULL; } __ENDTRY return dst; } /*********************************************************************** * lstrcatW (KERNEL32.@) */ LPWSTR WINAPI lstrcatW( LPWSTR dst, LPCWSTR src ) { __TRY { strcatW( dst, src ); } __EXCEPT(page_fault) { SetLastError( ERROR_INVALID_PARAMETER ); return NULL; } __ENDTRY return dst; } /*********************************************************************** * lstrcatn (KERNEL.352) */ SEGPTR WINAPI lstrcatn16( SEGPTR dst, LPCSTR src, INT16 n ) { LPSTR p = MapSL(dst); LPSTR start = p; while (*p) p++; if ((n -= (p - start)) <= 0) return dst; lstrcpynA( p, src, n ); return dst; } /*********************************************************************** * lstrcpy (KERNEL.88) */ SEGPTR WINAPI lstrcpy16( SEGPTR dst, LPCSTR src ) { if (!lstrcpyA( MapSL(dst), src )) dst = 0; return dst; } /*********************************************************************** * lstrcpy (KERNEL32.@) * lstrcpyA (KERNEL32.@) */ LPSTR WINAPI lstrcpyA( LPSTR dst, LPCSTR src ) { __TRY { /* this is how Windows does it */ memmove( dst, src, strlen(src)+1 ); } __EXCEPT(page_fault) { SetLastError( ERROR_INVALID_PARAMETER ); return NULL; } __ENDTRY return dst; } /*********************************************************************** * lstrcpyW (KERNEL32.@) */ LPWSTR WINAPI lstrcpyW( LPWSTR dst, LPCWSTR src ) { __TRY { strcpyW( dst, src ); } __EXCEPT(page_fault) { SetLastError( ERROR_INVALID_PARAMETER ); return NULL; } __ENDTRY return dst; } /*********************************************************************** * lstrcpyn (KERNEL.353) */ SEGPTR WINAPI lstrcpyn16( SEGPTR dst, LPCSTR src, INT16 n ) { lstrcpynA( MapSL(dst), src, n ); return dst; } /*********************************************************************** * lstrcpyn (KERNEL32.@) * lstrcpynA (KERNEL32.@) * * Note: this function differs from the UNIX strncpy, it _always_ writes * a terminating \0. * * Note: n is an INT but Windows treats it as unsigned, and will happily * copy a gazillion chars if n is negative. */ LPSTR WINAPI lstrcpynA( LPSTR dst, LPCSTR src, INT n ) { __TRY { LPSTR d = dst; LPCSTR s = src; UINT count = n; while ((count > 1) && *s) { count--; *d++ = *s++; } if (count) *d = 0; } __EXCEPT(page_fault) { SetLastError( ERROR_INVALID_PARAMETER ); return 0; } __ENDTRY return dst; } /*********************************************************************** * lstrcpynW (KERNEL32.@) * * Note: this function differs from the UNIX strncpy, it _always_ writes * a terminating \0 * * Note: n is an INT but Windows treats it as unsigned, and will happily * copy a gazillion chars if n is negative. */ LPWSTR WINAPI lstrcpynW( LPWSTR dst, LPCWSTR src, INT n ) { __TRY { LPWSTR d = dst; LPCWSTR s = src; UINT count = n; while ((count > 1) && *s) { count--; *d++ = *s++; } if (count) *d = 0; } __EXCEPT(page_fault) { SetLastError( ERROR_INVALID_PARAMETER ); return 0; } __ENDTRY return dst; } /*********************************************************************** * lstrlen (KERNEL.90) */ INT16 WINAPI lstrlen16( LPCSTR str ) { return (INT16)lstrlenA( str ); } /*********************************************************************** * lstrlen (KERNEL32.@) * lstrlenA (KERNEL32.@) */ INT WINAPI lstrlenA( LPCSTR str ) { INT ret; __TRY { ret = strlen(str); } __EXCEPT(page_fault) { SetLastError( ERROR_INVALID_PARAMETER ); return 0; } __ENDTRY return ret; } /*********************************************************************** * lstrlenW (KERNEL32.@) */ INT WINAPI lstrlenW( LPCWSTR str ) { INT ret; __TRY { ret = strlenW(str); } __EXCEPT(page_fault) { SetLastError( ERROR_INVALID_PARAMETER ); return 0; } __ENDTRY return ret; } /*********************************************************************** * UnicodeToAnsi (KERNEL.434) */ INT16 WINAPI UnicodeToAnsi16( LPCWSTR src, LPSTR dst, INT16 codepage ) { if ( codepage == -1 ) codepage = CP_ACP; return WideCharToMultiByte( codepage, 0, src, -1, dst, 0x7fffffff, NULL, NULL ); } /*************************************************************************** * * Win 2.x string functions now moved to USER * * We rather want to implement them here instead of doing Callouts */ /*********************************************************************** * Reserved1 (KERNEL.77) */ SEGPTR WINAPI KERNEL_AnsiNext16(SEGPTR current) { return (*(char *)MapSL(current)) ? current + 1 : current; } /*********************************************************************** * Reserved2(KERNEL.78) */ SEGPTR WINAPI KERNEL_AnsiPrev16( SEGPTR start, SEGPTR current ) { return (current==start)?start:current-1; } /*********************************************************************** * Reserved3 (KERNEL.79) */ SEGPTR WINAPI KERNEL_AnsiUpper16( SEGPTR strOrChar ) { /* uppercase only one char if strOrChar < 0x10000 */ if (HIWORD(strOrChar)) { char *s = MapSL(strOrChar); while (*s) { *s = toupper(*s); s++; } return strOrChar; } else return toupper((char)strOrChar); } /*********************************************************************** * Reserved4 (KERNEL.80) */ SEGPTR WINAPI KERNEL_AnsiLower16( SEGPTR strOrChar ) { /* lowercase only one char if strOrChar < 0x10000 */ if (HIWORD(strOrChar)) { char *s = MapSL(strOrChar); while (*s) { *s = tolower(*s); s++; } return strOrChar; } else return tolower((char)strOrChar); } /*********************************************************************** * Reserved5 (KERNEL.87) */ INT16 WINAPI KERNEL_lstrcmp16( LPCSTR str1, LPCSTR str2 ) { return (INT16)strcmp( str1, str2 ); }