Sun May 17 16:23:56 1998 Uwe Bonnes <bon@elektron.ikp.physik.tu-darmstadt.de> * [file/profile.c] Fix the return value of PROFILE_GetSection * [misc/crtdll.c] Do _getdrive, fix _chdrive. * [misc/commdlg.c] First cut at ChooseColor[WA]. * [misc/network.c] Do something sensible for WNetGetDirectoryType16. Sun May 17 10:21:35 1998 Andreas Mohr <100.30936@germany.net> * [controls/menu.c] Fixed disabled sub menus with MF_BYPOSITION that were not disabled. * [misc/crtdll.c] [relay32/crtdll.spec] [include/winerror.h] Implemented fscanf, fsetpos, _access, _fpreset (thanks to Uwe Bonnes), and _ltoa. * [loader/task.c] MakeProcInstance: must use CURRENT_DS if hInst == NULL. * [misc/shell.c] SHELL_GetResourceTable, InternalExtractIcon: fixed broken .ICO handling * [windows/winpos.c] DeferWindowPos: removed "same parent" requirement. Which doc states that this is required ? Sat May 16 20:08:11 1998 Alexandre Julliard <julliard@lrc.epfl.ch> * [loader/module.c] [loader/ne/module.c] More NE module cleanups. * [loader/task.c] Fixed SwitchStackBack(). Fri May 15 10:04:27 1998 Marcus Meissner <marcus@jet.franken.de> * [configure.in][inlcude/acconfig.h] Fixed broken OSS check, added check for working sigaltstack, fixed broken statfs checks on some linux systems. * [files/directory.c][loader/pe_image.c][relay32/builtin.c] [loader/module.c] Added handling of win32 module pathnames. * [relay32/wnaspi32.spec] New file. * [misc/lzexpand.c] LZCopy auto-decompresses LZ compressed files, even if they are not specially flagged. Fixes some InstallShield problems. * [misc/registry.c] Some fixes for RegQueryInfoKey (reference program monkey.exe from Win32 SDK works now better). Probably still has faults. Fri May 15 08:58:58 1998 Martin Boehme <boehme@informatik.mu-luebeck.de> * [graphics/mapping.c] [include/dc.h] [include/gdi.h] [objects/dc.c] Reworked the way world transformations and mapping modes are handled so that both of these transformations can be computed in a single step. * [graphics/painting.c] [graphics/path.c] [include/path.h] More GDI path support. * [graphics/x11drv/graphics.c] Fixed the return value of GRAPH_DrawArc for the zero height / zero width case to reflect Windows' behaviour. * [include/windows.h] [relay32/gdi32.spec] [objects/dc.c] Implemented ModifyWorldTransform and CombineTransform. Tue May 14 18:03:46 1998 Eric Kohl <ekohl@abo.rhein-zeitung.de> * [controls/commctrl.c][relay32/comctl32.spec] [controls/*.c][include/*.h] Implemented InitCommonControlsEx (dll version 4.72 compatible). InitCommonControls calls ImageCommonControlsEx. Registering code of the common controls had to be changed (see XXXX_Register functions). * [controls/status.c][include/commctrl.h][include/status.h] Implemented most new features and fixed the look and feel. * [contols/commctrl.c][include/commctrl.h][relay32/comctl32.spec] Implemented MenuHelp (incomplete). * [controls/status.c][controls/progress.c] Changed allocation strategy for control specific memory. * [controls/header.c][include/header.h][include/commctrl.h] First implementation of header control. * [windows/defwnd.c][windows/syscolors.c] Fixed default control colors for Win95 look. * [windows/nonclient.c] Fixed off by one error for Win95 look. Top border of child windows should be visible. * [misc/imagelist.h] Improved documentation and fixed some bugs. Thu May 14 15:42:21 1998 Robert Wilhelm <robert@physiol.med.tu-muenchen.de> * [relay32/crtdll.spec] Added hypot,j0,j1,jn and ceil. Wed May 13 19:10:10 1998 Pascal Cuoq <pcuoq@ens-lyon.fr> * [controls/listbox.c] Item height is now exactly font height. Wine listboxes now behave like Windows' when they are created without WS_VSCROLL but the program subsequently calls ShowScrollBar or SetScrollInfo. Wed May 13 18:33:01 1998 Ulrich Weigand <weigand@informatik.uni-erlangen.de> * [relay32/relay386.c] Restore ES also in the non-debug case. * [windows/event.c] Bugfix: Blocking TSXNextEvent could deadlock Wine. * [win32/process.c] [windows/message.c] Silly stubs for MsgWaitForMultipleObjects / PostThreadMessage that make some programs run better. * [windows/winproc.c] WINPROC_MapMsg32Ato16/16To32A: added WM_NOTIFY. * [win32/kernel32.c] Added 16->32 thunking and improved 32->16 thunking functions. * [tools/build.c] Added new variant of CallFrom16 stub for use with Win95 thunks. * [if1632/kernel.spec] [if1632/builtin.c] [win32/kernel32.c] Added a few undocumented KERNEL functions. * [loader/ne/module.c] [loader/ne/segment.c] Call DllEntryPoint for 16-bit DLLs with subsystem >= 4.0. * [win32/kernel32.spec] [win32/wow32.spec] [win32/ordinals.c] Use names from the Oct 94 beta release for undoc. functions. Wed May 13 14:18:26 1998 Matthew Becker <mbecker@glasscity.net> * [misc/registry.c] Code cleanup. * [misc/cpu.c] Commented out the registry puts temporarily. * [programs/regtest/*] New registry testing program. Tue May 12 22:54:03 1998 Michael Mess <michael@kawo2.rwth-aachen.de> * [multimedia/audio.c] ioctl's do not commute in /dev/dsp initialization. Tue May 12 20:11:42 1998 Karl Garrison <karlos@eznet.net> * [win32/console.c] Implemented SetConsoleTextAttribute, FillConsoleOutputCharacter. Improved cursor positioning. This allows for text colors in an xterm, rxvt, or console. Tue May 12 17:57:52 1998 Petter Reinholdtsen <pere@td.org.uit.no> * [Makefile.in] Create prefix/{bin|lib} directories if missing during install. Sun May 10 19:37:51 1998 Jan Willamowius <jan@janhh.shnet.org> * [multimedia/mmio.c] Have mmioSetBuffer return success (0), so Corel Draw 4 keeps working. (IO is still unbuffered) Wed May 6 16:57:55 1998 James Juran <jrj120@psu.edu> * [Makefile.in] [Make.rules.in] Changed "make clean" to remove `textedit` backup files (*%) * [controls/menu.c][graphics/x11drv/xfont.c][include/libres.h] [loader/main.c][loader/ne/module.c][scheduler/synchro.c] [win32/time.c][windows/winpos.c][include/windows.h] Fixed miscellaneous compilation warnings. * [misc/main.c][miscemu/main.c][include/main.h] Moved prototypes to new include file main.h, various cleanups. Tue May 5 21:05:06 1998 Morten Welinder <terra@diku.dk> * [misc/winsock.c] Don't refer to __FreeBSD__ when HAVE_STRERROR is meant. * [misc/debugstr.c] For debug_dumpstrSend, send strings to stderr. Tue May 5 21:47:40 1998 Huw D M Davies <h.davies1@physics.oxford.ac.uk> * [objects/region.c] Fix for REGION_RegionOp() if newReg is one of the source regions. Tue May 5 18:27:32 1998 Jim Peterson <jspeter@roanoke.infi.net> * [misc/main.c] Add '-h/-help' option and print WINE_RELEASE_INFO with usage message. * [misc/spy.c] Realign trace messages. Tue May 5 15:46:47 1998 Donnie V. Savage <dsavage@cisco.com> * [graphics/ddraw.c] Fixed compile warnings * [misc/winsock.c] Warnings should not be errors. Tue May 5 13:40:42 1998 Jim Peterson <jspeter@roanoke.infi.net> * [*/*] Remove many warnings through explicit casts, added #include's, and corrected printf formats. Tue May 5 05:18:12 1998 Insomnia (Stea Greene) <insomnia@core.binghamton.edu> * [graphics/ddraw.c] Kept unchanged portion of old palette when changing only a few palette entries. Really should only deallocate the changed cells. This make StarCraft work almost perfectly (sound overflows still cause static). Mon May 4 15:04:57 1998 Alexander V. Lukyanov <lav@long.yar.ru> * [misc/lstr.c] FormatMessage: terminate string on %0, undo linefeed strip.
842 lines
21 KiB
C
842 lines
21 KiB
C
/*
|
|
* String functions
|
|
*
|
|
* Copyright 1993 Yngvi Sigurjonsson (yngvi@hafro.is)
|
|
* Copyright 1996 Marcus Meissner
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdio.h>
|
|
#include <stdarg.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <ctype.h>
|
|
|
|
#ifdef HAVE_WCTYPE_H
|
|
# include <wctype.h>
|
|
#else
|
|
# define towlower(c) tolower(c)
|
|
# define towupper(c) toupper(c)
|
|
# define iswalnum(c) isalnum(c)
|
|
# define iswalpha(c) isalpha(c)
|
|
# define iswupper(c) isupper(c)
|
|
# define iswlower(c) islower(c)
|
|
#endif /* HAVE_WCTYPE_H */
|
|
|
|
|
|
#include "windows.h"
|
|
#include "winnt.h" /* HEAP_ macros */
|
|
#include "task.h"
|
|
#include "heap.h"
|
|
#include "ldt.h"
|
|
#include "stackframe.h"
|
|
#include "module.h"
|
|
#include "debug.h"
|
|
|
|
/* Funny to divide them between user and kernel. */
|
|
|
|
/* be careful: always use functions from wctype.h if character > 255 */
|
|
|
|
/***********************************************************************
|
|
* IsCharAlpha (USER.433)
|
|
*/
|
|
BOOL16 WINAPI IsCharAlpha16(CHAR ch)
|
|
{
|
|
return isalpha(ch); /* This is probably not right for NLS */
|
|
}
|
|
|
|
/***********************************************************************
|
|
* IsCharAlphanumeric (USER.434)
|
|
*/
|
|
BOOL16 WINAPI IsCharAlphaNumeric16(CHAR ch)
|
|
{
|
|
return isalnum(ch);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* IsCharUpper (USER.435)
|
|
*/
|
|
BOOL16 WINAPI IsCharUpper16(CHAR ch)
|
|
{
|
|
return isupper(ch);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* IsCharLower (USER.436)
|
|
*/
|
|
BOOL16 WINAPI IsCharLower16(CHAR ch)
|
|
{
|
|
return islower(ch);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* AnsiUpper16 (USER.431)
|
|
*/
|
|
SEGPTR WINAPI AnsiUpper16( SEGPTR strOrChar )
|
|
{
|
|
/* I am not sure if the locale stuff works with toupper, but then again
|
|
I am not sure if the Linux libc locale stuffs works at all */
|
|
|
|
/* uppercase only one char if strOrChar < 0x10000 */
|
|
if (HIWORD(strOrChar))
|
|
{
|
|
char *s;
|
|
for (s = PTR_SEG_TO_LIN(strOrChar); *s; s++) *s = toupper(*s);
|
|
return strOrChar;
|
|
}
|
|
else return toupper((char)strOrChar);
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* AnsiUpperBuff16 (USER.437)
|
|
*/
|
|
UINT16 WINAPI AnsiUpperBuff16( LPSTR str, UINT16 len )
|
|
{
|
|
UINT32 count = len ? len : 65536;
|
|
for (; count; count--, str++) *str = toupper(*str);
|
|
return len;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* AnsiLower16 (USER.432)
|
|
*/
|
|
SEGPTR WINAPI AnsiLower16( SEGPTR strOrChar )
|
|
{
|
|
/* I am not sure if the locale stuff works with toupper, but then again
|
|
I am not sure if the Linux libc locale stuffs works at all */
|
|
|
|
/* lowercase only one char if strOrChar < 0x10000 */
|
|
if (HIWORD(strOrChar))
|
|
{
|
|
char *s;
|
|
for (s = PTR_SEG_TO_LIN( strOrChar ); *s; s++) *s = tolower( *s );
|
|
return strOrChar;
|
|
}
|
|
else return tolower((char)strOrChar);
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* AnsiLowerBuff16 (USER.438)
|
|
*/
|
|
UINT16 WINAPI AnsiLowerBuff16( LPSTR str, UINT16 len )
|
|
{
|
|
UINT32 count = len ? len : 65536;
|
|
for (; count; count--, str++) *str = tolower(*str);
|
|
return len;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* AnsiNext16 (USER.472)
|
|
*/
|
|
SEGPTR WINAPI AnsiNext16(SEGPTR current)
|
|
{
|
|
return (*(char *)PTR_SEG_TO_LIN(current)) ? current + 1 : current;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* AnsiPrev16 (USER.473)
|
|
*/
|
|
SEGPTR WINAPI AnsiPrev16( SEGPTR start, SEGPTR current )
|
|
{
|
|
return (current == start) ? start : current - 1;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* OutputDebugString16 (KERNEL.115)
|
|
*/
|
|
void WINAPI OutputDebugString16( LPCSTR str )
|
|
{
|
|
char module[10];
|
|
char *p, *buffer = HeapAlloc( GetProcessHeap(), 0, strlen(str)+2 );
|
|
/* Remove CRs */
|
|
for (p = buffer; *str; str++) if (*str != '\r') *p++ = *str;
|
|
*p = '\0';
|
|
if ((p > buffer) && (p[-1] == '\n')) p[1] = '\0'; /* Remove trailing \n */
|
|
if (!GetModuleName( GetCurrentTask(), module, sizeof(module) ))
|
|
strcpy( module, "???" );
|
|
TRACE(resource, "%s says '%s'\n", module, buffer );
|
|
HeapFree( GetProcessHeap(), 0, buffer );
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* OutputDebugString32A (KERNEL32
|
|
*/
|
|
void WINAPI OutputDebugString32A( LPCSTR str )
|
|
{
|
|
OutputDebugString16( str );
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
* OutputDebugString32W (KERNEL32
|
|
*/
|
|
void WINAPI OutputDebugString32W( LPCWSTR str )
|
|
{
|
|
LPSTR p = HEAP_strdupWtoA( GetProcessHeap(), 0, str );
|
|
OutputDebugString32A( p );
|
|
HeapFree( GetProcessHeap(), 0, p );
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
* CharNext32A (USER32.29)
|
|
*/
|
|
LPSTR WINAPI CharNext32A( LPCSTR ptr )
|
|
{
|
|
if (!*ptr) return (LPSTR)ptr;
|
|
if (IsDBCSLeadByte32( *ptr )) return (LPSTR)(ptr + 2);
|
|
return (LPSTR)(ptr + 1);
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* CharNextEx32A (USER32.30)
|
|
*/
|
|
LPSTR WINAPI CharNextEx32A( WORD codepage, LPCSTR ptr, DWORD flags )
|
|
{
|
|
if (!*ptr) return (LPSTR)ptr;
|
|
if (IsDBCSLeadByteEx( codepage, *ptr )) return (LPSTR)(ptr + 2);
|
|
return (LPSTR)(ptr + 1);
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* CharNextExW (USER32.31)
|
|
*/
|
|
LPWSTR WINAPI CharNextEx32W(WORD codepage,LPCWSTR x,DWORD flags)
|
|
{
|
|
/* FIXME: add DBCS / codepage stuff */
|
|
if (*x) return (LPWSTR)(x+1);
|
|
else return (LPWSTR)x;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* CharNextW (USER32.32)
|
|
*/
|
|
LPWSTR WINAPI CharNext32W(LPCWSTR x)
|
|
{
|
|
if (*x) return (LPWSTR)(x+1);
|
|
else return (LPWSTR)x;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* CharPrev32A (USER32.33)
|
|
*/
|
|
LPSTR WINAPI CharPrev32A( LPCSTR start, LPCSTR ptr )
|
|
{
|
|
while (*start && (start < ptr))
|
|
{
|
|
LPCSTR next = CharNext32A( start );
|
|
if (next >= ptr) break;
|
|
start = next;
|
|
}
|
|
return (LPSTR)start;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* CharPrevEx32A (USER32.34)
|
|
*/
|
|
LPSTR WINAPI CharPrevEx32A( WORD codepage, LPCSTR start, LPCSTR ptr, DWORD flags )
|
|
{
|
|
while (*start && (start < ptr))
|
|
{
|
|
LPCSTR next = CharNextEx32A( codepage, start, flags );
|
|
if (next > ptr) break;
|
|
start = next;
|
|
}
|
|
return (LPSTR)start;
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
* CharPrevExW (USER32.35)
|
|
*/
|
|
LPWSTR WINAPI CharPrevEx32W(WORD codepage,LPCWSTR start,LPCWSTR x,DWORD flags)
|
|
{
|
|
/* FIXME: add DBCS / codepage stuff */
|
|
if (x>start) return (LPWSTR)(x-1);
|
|
else return (LPWSTR)x;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* CharPrevW (USER32.36)
|
|
*/
|
|
LPWSTR WINAPI CharPrev32W(LPCWSTR start,LPCWSTR x)
|
|
{
|
|
if (x>start) return (LPWSTR)(x-1);
|
|
else return (LPWSTR)x;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* CharLowerA (USER32.25)
|
|
* FIXME: handle current locale
|
|
*/
|
|
LPSTR WINAPI CharLower32A(LPSTR x)
|
|
{
|
|
LPSTR s;
|
|
|
|
if (HIWORD(x))
|
|
{
|
|
s=x;
|
|
while (*s)
|
|
{
|
|
*s=tolower(*s);
|
|
s++;
|
|
}
|
|
return x;
|
|
}
|
|
else return (LPSTR)tolower((char)(int)x);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* CharLowerBuffA (USER32.26)
|
|
* FIXME: handle current locale
|
|
*/
|
|
DWORD WINAPI CharLowerBuff32A(LPSTR x,DWORD buflen)
|
|
{
|
|
DWORD done=0;
|
|
|
|
if (!x) return 0; /* YES */
|
|
while (*x && (buflen--))
|
|
{
|
|
*x=tolower(*x);
|
|
x++;
|
|
done++;
|
|
}
|
|
return done;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* CharLowerBuffW (USER32.27)
|
|
* FIXME: handle current locale
|
|
*/
|
|
DWORD WINAPI CharLowerBuff32W(LPWSTR x,DWORD buflen)
|
|
{
|
|
DWORD done=0;
|
|
|
|
if (!x) return 0; /* YES */
|
|
while (*x && (buflen--))
|
|
{
|
|
*x=towlower(*x);
|
|
x++;
|
|
done++;
|
|
}
|
|
return done;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* CharLowerW (USER32.28)
|
|
* FIXME: handle current locale
|
|
*/
|
|
LPWSTR WINAPI CharLower32W(LPWSTR x)
|
|
{
|
|
if (HIWORD(x))
|
|
{
|
|
LPWSTR s = x;
|
|
while (*s)
|
|
{
|
|
*s=towlower(*s);
|
|
s++;
|
|
}
|
|
return x;
|
|
}
|
|
else return (LPWSTR)towlower(LOWORD(x));
|
|
}
|
|
|
|
/***********************************************************************
|
|
* CharUpper32A (USER32.41)
|
|
* FIXME: handle current locale
|
|
*/
|
|
LPSTR WINAPI CharUpper32A(LPSTR x)
|
|
{
|
|
if (HIWORD(x))
|
|
{
|
|
LPSTR s = x;
|
|
while (*s)
|
|
{
|
|
*s=toupper(*s);
|
|
s++;
|
|
}
|
|
return x;
|
|
}
|
|
return (LPSTR)toupper((char)(int)x);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* CharUpperBuffA (USER32.42)
|
|
* FIXME: handle current locale
|
|
*/
|
|
DWORD WINAPI CharUpperBuff32A(LPSTR x,DWORD buflen)
|
|
{
|
|
DWORD done=0;
|
|
|
|
if (!x) return 0; /* YES */
|
|
while (*x && (buflen--))
|
|
{
|
|
*x=toupper(*x);
|
|
x++;
|
|
done++;
|
|
}
|
|
return done;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* CharUpperBuffW (USER32.43)
|
|
* FIXME: handle current locale
|
|
*/
|
|
DWORD WINAPI CharUpperBuff32W(LPWSTR x,DWORD buflen)
|
|
{
|
|
DWORD done=0;
|
|
|
|
if (!x) return 0; /* YES */
|
|
while (*x && (buflen--))
|
|
{
|
|
*x=towupper(*x);
|
|
x++;
|
|
done++;
|
|
}
|
|
return done;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* CharUpperW (USER32.44)
|
|
* FIXME: handle current locale
|
|
*/
|
|
LPWSTR WINAPI CharUpper32W(LPWSTR x)
|
|
{
|
|
if (HIWORD(x))
|
|
{
|
|
LPWSTR s = x;
|
|
while (*s)
|
|
{
|
|
*s=towupper(*s);
|
|
s++;
|
|
}
|
|
return x;
|
|
}
|
|
else return (LPWSTR)towupper(LOWORD(x));
|
|
}
|
|
|
|
/***********************************************************************
|
|
* IsCharAlphaA (USER32.331)
|
|
* FIXME: handle current locale
|
|
*/
|
|
BOOL32 WINAPI IsCharAlpha32A(CHAR x)
|
|
{
|
|
return isalpha(x);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* IsCharAlphaNumericA (USER32.332)
|
|
* FIXME: handle current locale
|
|
*/
|
|
BOOL32 WINAPI IsCharAlphaNumeric32A(CHAR x)
|
|
{
|
|
return isalnum(x);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* IsCharAlphaNumericW (USER32.333)
|
|
* FIXME: handle current locale
|
|
*/
|
|
BOOL32 WINAPI IsCharAlphaNumeric32W(WCHAR x)
|
|
{
|
|
return iswalnum(x);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* IsCharAlphaW (USER32.334)
|
|
* FIXME: handle current locale
|
|
*/
|
|
BOOL32 WINAPI IsCharAlpha32W(WCHAR x)
|
|
{
|
|
return iswalpha(x);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* IsCharLower32A (USER32.335)
|
|
* FIXME: handle current locale
|
|
*/
|
|
BOOL32 WINAPI IsCharLower32A(CHAR x)
|
|
{
|
|
return islower(x);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* IsCharLower32W (USER32.336)
|
|
* FIXME: handle current locale
|
|
*/
|
|
BOOL32 WINAPI IsCharLower32W(WCHAR x)
|
|
{
|
|
return iswlower(x);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* IsCharUpper32A (USER32.337)
|
|
* FIXME: handle current locale
|
|
*/
|
|
BOOL32 WINAPI IsCharUpper32A(CHAR x)
|
|
{
|
|
return isupper(x);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* IsCharUpper32W (USER32.338)
|
|
* FIXME: handle current locale
|
|
*/
|
|
BOOL32 WINAPI IsCharUpper32W(WCHAR x)
|
|
{
|
|
return iswupper(x);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* FormatMessage32A (KERNEL32.138)
|
|
* FIXME: missing wrap,FROM_SYSTEM message-loading,
|
|
*/
|
|
DWORD WINAPI FormatMessage32A(
|
|
DWORD dwFlags,
|
|
LPCVOID lpSource,
|
|
DWORD dwMessageId,
|
|
DWORD dwLanguageId,
|
|
LPSTR lpBuffer,
|
|
DWORD nSize,
|
|
LPDWORD args /* va_list *args */
|
|
) {
|
|
LPSTR target,t;
|
|
DWORD talloced;
|
|
LPSTR from,f;
|
|
DWORD width = dwFlags & FORMAT_MESSAGE_MAX_WIDTH_MASK;
|
|
DWORD nolinefeed = 0;
|
|
|
|
TRACE(resource, "(0x%lx,%p,%ld,0x%lx,%p,%ld,%p)\n",
|
|
dwFlags,lpSource,dwMessageId,dwLanguageId,lpBuffer,nSize,args);
|
|
if (width)
|
|
FIXME(resource,"line wrapping not supported.\n");
|
|
from = NULL;
|
|
if (dwFlags & FORMAT_MESSAGE_FROM_STRING)
|
|
from = HEAP_strdupA( GetProcessHeap(), 0, (LPSTR)lpSource);
|
|
if (dwFlags & FORMAT_MESSAGE_FROM_SYSTEM) {
|
|
from = HeapAlloc( GetProcessHeap(),0,200 );
|
|
sprintf(from,"Systemmessage, messageid = 0x%08lx\n",dwMessageId);
|
|
}
|
|
if (dwFlags & FORMAT_MESSAGE_FROM_HMODULE) {
|
|
INT32 bufsize;
|
|
|
|
dwMessageId &= 0xFFFF;
|
|
bufsize=LoadMessage32A((HMODULE32)lpSource,dwMessageId,dwLanguageId,NULL,100);
|
|
if (bufsize) {
|
|
from = HeapAlloc( GetProcessHeap(), 0, bufsize + 1 );
|
|
LoadMessage32A((HMODULE32)lpSource,dwMessageId,dwLanguageId,from,bufsize+1);
|
|
}
|
|
}
|
|
target = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, 100);
|
|
t = target;
|
|
talloced= 100;
|
|
|
|
#define ADD_TO_T(c) \
|
|
*t++=c;\
|
|
if (t-target == talloced) {\
|
|
target = (char*)HeapReAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,target,talloced*2);\
|
|
t = target+talloced;\
|
|
talloced*=2;\
|
|
}
|
|
|
|
if (from) {
|
|
f=from;
|
|
while (*f && !nolinefeed) {
|
|
if (*f=='%') {
|
|
int insertnr;
|
|
char *fmtstr,*sprintfbuf,*x,*lastf;
|
|
DWORD *argliststart;
|
|
|
|
fmtstr = NULL;
|
|
lastf = f;
|
|
f++;
|
|
if (!*f) {
|
|
ADD_TO_T('%');
|
|
continue;
|
|
}
|
|
switch (*f) {
|
|
case '1':case '2':case '3':case '4':case '5':
|
|
case '6':case '7':case '8':case '9':
|
|
insertnr=*f-'0';
|
|
switch (f[1]) {
|
|
case '0':case '1':case '2':case '3':
|
|
case '4':case '5':case '6':case '7':
|
|
case '8':case '9':
|
|
f++;
|
|
insertnr=insertnr*10+*f-'0';
|
|
f++;
|
|
break;
|
|
default:
|
|
f++;
|
|
break;
|
|
}
|
|
if (*f=='!') {
|
|
f++;
|
|
if (NULL!=(x=strchr(f,'!'))) {
|
|
*x='\0';
|
|
fmtstr=HeapAlloc(GetProcessHeap(),0,strlen(f)+2);
|
|
sprintf(fmtstr,"%%%s",f);
|
|
f=x+1;
|
|
} else {
|
|
fmtstr=HeapAlloc(GetProcessHeap(),0,strlen(f));
|
|
sprintf(fmtstr,"%%%s",f);
|
|
f+=strlen(f); /*at \0*/
|
|
}
|
|
} else
|
|
if(!args)
|
|
break;
|
|
else
|
|
fmtstr=HEAP_strdupA(GetProcessHeap(),0,"%s");
|
|
if (args) {
|
|
if (dwFlags & FORMAT_MESSAGE_ARGUMENT_ARRAY)
|
|
argliststart=args+insertnr-1;
|
|
else
|
|
argliststart=(*(DWORD**)args)+insertnr-1;
|
|
|
|
if (fmtstr[strlen(fmtstr)-1]=='s')
|
|
sprintfbuf=HeapAlloc(GetProcessHeap(),0,strlen((LPSTR)argliststart[0])+1);
|
|
else
|
|
sprintfbuf=HeapAlloc(GetProcessHeap(),0,100);
|
|
|
|
/* CMF - This makes a BIG assumption about va_list */
|
|
wvsprintf32A(sprintfbuf, fmtstr, (va_list) argliststart);
|
|
x=sprintfbuf;
|
|
while (*x) {
|
|
ADD_TO_T(*x++);
|
|
}
|
|
HeapFree(GetProcessHeap(),0,sprintfbuf);
|
|
} else {
|
|
/* NULL args - copy formatstr
|
|
* (probably wrong)
|
|
*/
|
|
while ((lastf<f)&&(*lastf)) {
|
|
ADD_TO_T(*lastf++);
|
|
}
|
|
}
|
|
HeapFree(GetProcessHeap(),0,fmtstr);
|
|
break;
|
|
case 'n':
|
|
/* FIXME: perhaps add \r too? */
|
|
ADD_TO_T('\n');
|
|
f++;
|
|
break;
|
|
case '0':
|
|
nolinefeed=1;
|
|
f++;
|
|
break;
|
|
default:ADD_TO_T(*f++)
|
|
break;
|
|
|
|
}
|
|
} else {
|
|
ADD_TO_T(*f++)
|
|
}
|
|
}
|
|
*t='\0';
|
|
}
|
|
if (!nolinefeed) {
|
|
/* add linefeed */
|
|
if(t==target || t[-1]!='\n')
|
|
ADD_TO_T('\n'); /* FIXME: perhaps add \r too? */
|
|
}
|
|
talloced = strlen(target)+1;
|
|
if (nSize && talloced<nSize) {
|
|
target = (char*)HeapReAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,target,nSize);
|
|
}
|
|
if (dwFlags & FORMAT_MESSAGE_ALLOCATE_BUFFER) {
|
|
/* nSize is the MINIMUM size */
|
|
*((LPVOID*)lpBuffer) = (LPVOID)LocalAlloc32(GMEM_ZEROINIT,talloced);
|
|
memcpy(*(LPSTR*)lpBuffer,target,talloced);
|
|
} else
|
|
strncpy(lpBuffer,target,nSize);
|
|
HeapFree(GetProcessHeap(),0,target);
|
|
if (from) HeapFree(GetProcessHeap(),0,from);
|
|
return (dwFlags & FORMAT_MESSAGE_ALLOCATE_BUFFER) ?
|
|
strlen(*(LPSTR*)lpBuffer):
|
|
strlen(lpBuffer);
|
|
}
|
|
#undef ADD_TO_T
|
|
|
|
|
|
/***********************************************************************
|
|
* FormatMessage32W (KERNEL32.138)
|
|
*/
|
|
DWORD WINAPI FormatMessage32W(
|
|
DWORD dwFlags,
|
|
LPCVOID lpSource,
|
|
DWORD dwMessageId,
|
|
DWORD dwLanguageId,
|
|
LPWSTR lpBuffer,
|
|
DWORD nSize,
|
|
LPDWORD args /* va_list *args */
|
|
) {
|
|
LPSTR target,t;
|
|
DWORD talloced;
|
|
LPSTR from,f;
|
|
DWORD width = dwFlags & FORMAT_MESSAGE_MAX_WIDTH_MASK;
|
|
DWORD nolinefeed = 0;
|
|
|
|
TRACE(resource, "(0x%lx,%p,%ld,0x%lx,%p,%ld,%p)\n",
|
|
dwFlags,lpSource,dwMessageId,dwLanguageId,lpBuffer,nSize,args);
|
|
if (width)
|
|
FIXME(resource,"line wrapping not supported.\n");
|
|
from = NULL;
|
|
if (dwFlags & FORMAT_MESSAGE_FROM_STRING)
|
|
from = HEAP_strdupWtoA(GetProcessHeap(),0,(LPWSTR)lpSource);
|
|
if (dwFlags & FORMAT_MESSAGE_FROM_SYSTEM) {
|
|
/* gather information from system message tables ... */
|
|
from = HeapAlloc( GetProcessHeap(),0,200 );
|
|
sprintf(from,"Systemmessage, messageid = 0x%08lx\n",dwMessageId);
|
|
}
|
|
if (dwFlags & FORMAT_MESSAGE_FROM_HMODULE) {
|
|
INT32 bufsize;
|
|
|
|
dwMessageId &= 0xFFFF;
|
|
bufsize=LoadMessage32A((HMODULE32)lpSource,dwMessageId,dwLanguageId,NULL,100);
|
|
if (bufsize)
|
|
{
|
|
from = HeapAlloc( GetProcessHeap(), 0, bufsize + 1 );
|
|
LoadMessage32A((HMODULE32)lpSource,dwMessageId,dwLanguageId,from,bufsize+1);
|
|
}
|
|
}
|
|
target = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, 100 );
|
|
t = target;
|
|
talloced= 100;
|
|
|
|
#define ADD_TO_T(c) \
|
|
*t++=c;\
|
|
if (t-target == talloced) {\
|
|
target = (char*)HeapReAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,target,talloced*2);\
|
|
t = target+talloced;\
|
|
talloced*=2;\
|
|
}
|
|
|
|
if (from) {
|
|
f=from;
|
|
while (*f && !nolinefeed) {
|
|
if (*f=='%') {
|
|
int insertnr;
|
|
char *fmtstr,*sprintfbuf,*x;
|
|
DWORD *argliststart;
|
|
|
|
fmtstr = NULL;
|
|
f++;
|
|
if (!*f) {
|
|
ADD_TO_T('%');
|
|
continue;
|
|
}
|
|
switch (*f) {
|
|
case '1':case '2':case '3':case '4':case '5':
|
|
case '6':case '7':case '8':case '9':
|
|
insertnr=*f-'0';
|
|
switch (f[1]) {
|
|
case '0':case '1':case '2':case '3':
|
|
case '4':case '5':case '6':case '7':
|
|
case '8':case '9':
|
|
f++;
|
|
insertnr=insertnr*10+*f-'0';
|
|
f++;
|
|
break;
|
|
default:
|
|
f++;
|
|
break;
|
|
}
|
|
if (*f=='!') {
|
|
f++;
|
|
if (NULL!=(x=strchr(f,'!')))
|
|
{
|
|
*x='\0';
|
|
fmtstr=HeapAlloc( GetProcessHeap(), 0, strlen(f)+2);
|
|
sprintf(fmtstr,"%%%s",f);
|
|
f=x+1;
|
|
} else {
|
|
fmtstr=HeapAlloc(GetProcessHeap(),0,strlen(f));
|
|
sprintf(fmtstr,"%%%s",f);
|
|
f+=strlen(f); /*at \0*/
|
|
}
|
|
} else
|
|
if(!args)
|
|
break;
|
|
else
|
|
fmtstr=HEAP_strdupA( GetProcessHeap(),0,"%s");
|
|
if (dwFlags & FORMAT_MESSAGE_ARGUMENT_ARRAY)
|
|
argliststart=args+insertnr-1;
|
|
else
|
|
argliststart=(*(DWORD**)args)+insertnr-1;
|
|
|
|
if (fmtstr[strlen(fmtstr)-1]=='s') {
|
|
DWORD xarr[3];
|
|
|
|
xarr[0]=(DWORD)HEAP_strdupWtoA(GetProcessHeap(),0,(LPWSTR)(*(argliststart+0)));
|
|
/* possible invalid pointers */
|
|
xarr[1]=*(argliststart+1);
|
|
xarr[2]=*(argliststart+2);
|
|
sprintfbuf=HeapAlloc(GetProcessHeap(),0,lstrlen32W((LPWSTR)argliststart[0])*2+1);
|
|
|
|
/* CMF - This makes a BIG assumption about va_list */
|
|
vsprintf(sprintfbuf, fmtstr, (va_list) xarr);
|
|
} else {
|
|
sprintfbuf=HeapAlloc(GetProcessHeap(),0,100);
|
|
|
|
/* CMF - This makes a BIG assumption about va_list */
|
|
wvsprintf32A(sprintfbuf, fmtstr, (va_list) argliststart);
|
|
}
|
|
x=sprintfbuf;
|
|
while (*x) {
|
|
ADD_TO_T(*x++);
|
|
}
|
|
HeapFree(GetProcessHeap(),0,sprintfbuf);
|
|
HeapFree(GetProcessHeap(),0,fmtstr);
|
|
break;
|
|
case 'n':
|
|
/* FIXME: perhaps add \r too? */
|
|
ADD_TO_T('\n');
|
|
f++;
|
|
break;
|
|
case '0':
|
|
nolinefeed=1;
|
|
f++;
|
|
break;
|
|
default:ADD_TO_T(*f++)
|
|
break;
|
|
|
|
}
|
|
} else {
|
|
ADD_TO_T(*f++)
|
|
}
|
|
}
|
|
*t='\0';
|
|
}
|
|
if (!nolinefeed) {
|
|
/* add linefeed */
|
|
if(t==target || t[-1]!='\n')
|
|
ADD_TO_T('\n'); /* FIXME: perhaps add \r too? */
|
|
}
|
|
talloced = strlen(target)+1;
|
|
if (nSize && talloced<nSize)
|
|
target = (char*)HeapReAlloc(GetProcessHeap(),HEAP_ZERO_MEMORY,target,nSize);
|
|
if (dwFlags & FORMAT_MESSAGE_ALLOCATE_BUFFER) {
|
|
/* nSize is the MINIMUM size */
|
|
*((LPVOID*)lpBuffer) = (LPVOID)LocalAlloc32(GMEM_ZEROINIT,talloced*2+2);
|
|
lstrcpynAtoW(*(LPWSTR*)lpBuffer,target,talloced);
|
|
} else
|
|
lstrcpynAtoW(lpBuffer,target,nSize);
|
|
HeapFree(GetProcessHeap(),0,target);
|
|
if (from) HeapFree(GetProcessHeap(),0,from);
|
|
return (dwFlags & FORMAT_MESSAGE_ALLOCATE_BUFFER) ?
|
|
lstrlen32W(*(LPWSTR*)lpBuffer):
|
|
lstrlen32W(lpBuffer);
|
|
}
|
|
#undef ADD_TO_T
|