2598 lines
73 KiB
C
2598 lines
73 KiB
C
/*++
|
|
|
|
Copyright (c) 1991-2000, Microsoft Corporation All rights reserved.
|
|
|
|
Module Name:
|
|
|
|
ansi.c
|
|
|
|
Abstract:
|
|
|
|
This file contains the ANSI versions of the NLS API functions.
|
|
|
|
APIs found in this file:
|
|
CompareStringA
|
|
LCMapStringA
|
|
GetLocaleInfoA
|
|
SetLocaleInfoA
|
|
GetCalendarInfoA
|
|
SetCalendarInfoA
|
|
GetTimeFormatA
|
|
GetDateFormatA
|
|
GetNumberFormatA
|
|
GetCurrencyFormatA
|
|
EnumCalendarInfoA
|
|
EnumCalendarInfoExA
|
|
EnumTimeFormatsA
|
|
EnumDateFormatsA
|
|
EnumDateFormatsExA
|
|
GetStringTypeExA
|
|
GetStringTypeA
|
|
FoldStringA
|
|
EnumSystemLanguageGroupsA
|
|
EnumLanguageGroupLocalesA
|
|
EnumUILanguagesA
|
|
EnumSystemLocalesA
|
|
EnumSystemCodePagesA
|
|
GetCPInfoExA
|
|
GetGeoInfoA
|
|
|
|
|
|
Revision History:
|
|
|
|
11-10-93 JulieB Created.
|
|
07-03-00 lguindon Began GEO API port
|
|
|
|
--*/
|
|
|
|
|
|
|
|
//
|
|
// Include Files.
|
|
//
|
|
|
|
#include "nls.h"
|
|
#include "nlssafe.h"
|
|
|
|
|
|
|
|
//
|
|
// Forward Declarations.
|
|
//
|
|
|
|
PCP_HASH
|
|
NlsGetACPFromLocale(
|
|
LCID Locale,
|
|
DWORD dwFlags);
|
|
|
|
BOOL
|
|
NlsAnsiToUnicode(
|
|
PCP_HASH pHashN,
|
|
DWORD dwFlags,
|
|
LPCSTR pAnsiBuffer,
|
|
int AnsiLength,
|
|
LPWSTR *ppUnicodeBuffer,
|
|
int *pUnicodeLength);
|
|
|
|
int
|
|
NlsUnicodeToAnsi(
|
|
PCP_HASH pHashN,
|
|
LPCWSTR pUnicodeBuffer,
|
|
int UnicodeLength,
|
|
LPSTR pAnsiBuffer,
|
|
int AnsiLength);
|
|
|
|
BOOL
|
|
NlsEnumUnicodeToAnsi(
|
|
PCP_HASH pHashN,
|
|
LPCWSTR pUnicodeBuffer,
|
|
LPSTR *ppAnsiBuffer);
|
|
|
|
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// API ROUTINES //
|
|
//-------------------------------------------------------------------------//
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// CompareStringA
|
|
//
|
|
// Compares two wide character strings of the same locale according to the
|
|
// supplied locale handle.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int WINAPI CompareStringA(
|
|
LCID Locale,
|
|
DWORD dwCmpFlags,
|
|
LPCSTR lpString1,
|
|
int cchCount1,
|
|
LPCSTR lpString2,
|
|
int cchCount2)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
WCHAR pSTmp1[MAX_STRING_LEN]; // tmp Unicode buffer (string 1)
|
|
WCHAR pSTmp2[MAX_STRING_LEN]; // tmp Unicode buffer (string 2)
|
|
LPWSTR pUnicode1; // ptr to unicode string 1
|
|
LPWSTR pUnicode2; // ptr to unicode string 2
|
|
int UnicodeLength1; // length of Unicode string 1
|
|
int UnicodeLength2; // length of Unicode string 2
|
|
int ResultLen; // result length
|
|
BOOL fUseNegCounts = (cchCount1 < 0 && cchCount2 < 0); // flag to use negative counts
|
|
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - Get the code page hash node for the given locale.
|
|
// - either string is null
|
|
//
|
|
pHashN = NlsGetACPFromLocale(Locale, dwCmpFlags);
|
|
if ((pHashN == NULL) ||
|
|
(lpString1 == NULL) || (lpString2 == NULL))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Invalid Flags Check:
|
|
// - invalid flags
|
|
//
|
|
if (dwCmpFlags & CS_INVALID_FLAG)
|
|
{
|
|
SetLastError(ERROR_INVALID_FLAGS);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Convert Ansi string 1 to Unicode.
|
|
//
|
|
pUnicode1 = pSTmp1;
|
|
if (!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpString1,
|
|
cchCount1,
|
|
&pUnicode1,
|
|
&UnicodeLength1 ))
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Convert Ansi string 2 to Unicode.
|
|
//
|
|
pUnicode2 = pSTmp2;
|
|
if (!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpString2,
|
|
cchCount2,
|
|
&pUnicode2,
|
|
&UnicodeLength2 ))
|
|
{
|
|
NLS_FREE_TMP_BUFFER(pUnicode1, pSTmp1);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
ResultLen = CompareStringW( Locale,
|
|
dwCmpFlags,
|
|
pUnicode1,
|
|
(fUseNegCounts) ? -1 : UnicodeLength1,
|
|
pUnicode2,
|
|
(fUseNegCounts) ? -1 : UnicodeLength2);
|
|
|
|
//
|
|
// Free the allocated source buffers (if they were allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pUnicode1, pSTmp1);
|
|
NLS_FREE_TMP_BUFFER(pUnicode2, pSTmp2);
|
|
|
|
//
|
|
// Return the result of the call to CompareStringW.
|
|
//
|
|
return (ResultLen);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// LCMapStringA
|
|
//
|
|
// Maps one wide character string to another performing the specified
|
|
// translation. This mapping routine only takes flags that are locale
|
|
// dependent.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int WINAPI LCMapStringA(
|
|
LCID Locale,
|
|
DWORD dwMapFlags,
|
|
LPCSTR lpSrcStr,
|
|
int cchSrc,
|
|
LPSTR lpDestStr,
|
|
int cchDest)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
LPWSTR pUnicode; // ptr to unicode string
|
|
int UnicodeLength; // length of Unicode string
|
|
WCHAR pSTmp[MAX_STRING_LEN]; // tmp Unicode buffer (source)
|
|
WCHAR pDTmp[MAX_STRING_LEN]; // tmp Unicode buffer (destination)
|
|
LPWSTR pBuf; // ptr to destination buffer
|
|
int ResultLen; // result length
|
|
|
|
|
|
//
|
|
// Get the code page hash node for the given locale.
|
|
//
|
|
pHashN = NlsGetACPFromLocale(Locale, dwMapFlags);
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - valid code page
|
|
// - destination buffer size is negative
|
|
// - length of dest string is NOT zero AND dest string is NULL
|
|
// - same buffer - src = destination
|
|
// if not UPPER or LOWER or
|
|
// UPPER or LOWER used with Japanese flags
|
|
//
|
|
if ((pHashN == NULL) ||
|
|
(cchDest < 0) ||
|
|
((cchDest != 0) && (lpDestStr == NULL)) ||
|
|
((lpSrcStr == lpDestStr) &&
|
|
((!(dwMapFlags & (LCMAP_UPPERCASE | LCMAP_LOWERCASE))) ||
|
|
(dwMapFlags & (LCMAP_HIRAGANA | LCMAP_KATAKANA |
|
|
LCMAP_HALFWIDTH | LCMAP_FULLWIDTH)))))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Convert Ansi string to Unicode.
|
|
//
|
|
pUnicode = pSTmp;
|
|
if (!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpSrcStr,
|
|
cchSrc,
|
|
&pUnicode,
|
|
&UnicodeLength ))
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Special case the sortkey flag, since the Unicode buffer does
|
|
// NOT need to be converted back to Ansi.
|
|
//
|
|
if (dwMapFlags & LCMAP_SORTKEY)
|
|
{
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
ResultLen = LCMapStringW( Locale,
|
|
dwMapFlags,
|
|
pUnicode,
|
|
UnicodeLength,
|
|
(LPWSTR)lpDestStr,
|
|
cchDest );
|
|
|
|
//
|
|
// Free the allocated source buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pUnicode, pSTmp);
|
|
|
|
//
|
|
// Return the result of LCMapStringW.
|
|
//
|
|
return (ResultLen);
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
pBuf = pDTmp;
|
|
ResultLen = MAX_STRING_LEN;
|
|
while (1)
|
|
{
|
|
ResultLen = LCMapStringW( Locale,
|
|
dwMapFlags,
|
|
pUnicode,
|
|
UnicodeLength,
|
|
pBuf,
|
|
ResultLen );
|
|
|
|
//
|
|
// Make sure the static buffer was large enough.
|
|
//
|
|
if ((ResultLen != 0) || (GetLastError() != ERROR_INSUFFICIENT_BUFFER))
|
|
{
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Get the size of the buffer needed for the mapping.
|
|
//
|
|
if (ResultLen = LCMapStringW( Locale,
|
|
dwMapFlags,
|
|
pUnicode,
|
|
UnicodeLength,
|
|
NULL,
|
|
0 ))
|
|
{
|
|
//
|
|
// Allocate a buffer of the appropriate size.
|
|
//
|
|
if ((pBuf = (LPWSTR)NLS_ALLOC_MEM(ResultLen * sizeof(WCHAR))) == NULL)
|
|
{
|
|
NLS_FREE_TMP_BUFFER(pUnicode, pSTmp);
|
|
SetLastError(ERROR_OUTOFMEMORY);
|
|
return (0);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Free the allocated source buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pUnicode, pSTmp);
|
|
|
|
//
|
|
// Convert the destination Unicode buffer to the given Ansi buffer.
|
|
//
|
|
if (ResultLen > 0)
|
|
{
|
|
ResultLen = NlsUnicodeToAnsi( pHashN,
|
|
pBuf,
|
|
ResultLen,
|
|
lpDestStr,
|
|
cchDest );
|
|
}
|
|
|
|
//
|
|
// Free the allocated destination buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pBuf, pDTmp);
|
|
|
|
//
|
|
// Return the result of the call to LCMapStringW.
|
|
//
|
|
return (ResultLen);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetLocaleInfoA
|
|
//
|
|
// Returns one of the various pieces of information about a particular
|
|
// locale by querying the configuration registry. This call also indicates
|
|
// how much memory is necessary to contain the desired information.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int WINAPI GetLocaleInfoA(
|
|
LCID Locale,
|
|
LCTYPE LCType,
|
|
LPSTR lpLCData,
|
|
int cchData)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
WCHAR pDTmp[MAX_STRING_LEN]; // tmp Unicode buffer (destination)
|
|
LPWSTR pBuf; // ptr to destination buffer
|
|
int ResultLen; // result length
|
|
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - count is negative
|
|
// - NULL data pointer AND count is not zero
|
|
//
|
|
if ((cchData < 0) ||
|
|
(lpLCData == NULL) && (cchData != 0))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
pBuf = pDTmp;
|
|
ResultLen = MAX_STRING_LEN;
|
|
while (1)
|
|
{
|
|
ResultLen = GetLocaleInfoW( Locale,
|
|
LCType,
|
|
pBuf,
|
|
ResultLen );
|
|
|
|
//
|
|
// Make sure the static buffer was large enough.
|
|
//
|
|
if ((ResultLen != 0) || (GetLastError() != ERROR_INSUFFICIENT_BUFFER))
|
|
{
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Get the size of the buffer needed for the mapping.
|
|
//
|
|
if (ResultLen = GetLocaleInfoW( Locale,
|
|
LCType,
|
|
NULL,
|
|
0 ))
|
|
{
|
|
//
|
|
// Allocate a buffer of the appropriate size.
|
|
//
|
|
if ((pBuf = (LPWSTR)NLS_ALLOC_MEM(ResultLen * sizeof(WCHAR))) == NULL)
|
|
{
|
|
SetLastError(ERROR_OUTOFMEMORY);
|
|
return (0);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Convert the destination Unicode buffer to the given Ansi buffer.
|
|
//
|
|
if (ResultLen > 0)
|
|
{
|
|
if ((LCType & LOCALE_RETURN_NUMBER) ||
|
|
(NLS_GET_LCTYPE_VALUE(LCType) == LOCALE_FONTSIGNATURE))
|
|
{
|
|
//
|
|
// For the font signature and number value, the result length
|
|
// will actually be twice the amount of the wide char version.
|
|
//
|
|
ResultLen *= 2;
|
|
|
|
//
|
|
// Make sure we can use the buffer.
|
|
//
|
|
if (cchData)
|
|
{
|
|
//
|
|
// Make sure the buffer is large enough.
|
|
//
|
|
if (cchData < ResultLen)
|
|
{
|
|
//
|
|
// The buffer is too small.
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pBuf, pDTmp);
|
|
SetLastError(ERROR_INSUFFICIENT_BUFFER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Convert the font signature or number value to its byte
|
|
// form. Since it's already byte reversed, just do a move
|
|
// memory.
|
|
//
|
|
RtlMoveMemory(lpLCData, pBuf, ResultLen);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// If this is LCTYPE == LOCALE_SLANGUAGE, then use the
|
|
// CP_ACP of the system. This is what Win9x has done,
|
|
// and we need to be compatible with this.
|
|
//
|
|
if (NLS_GET_LCTYPE_VALUE(LCType) == LOCALE_SLANGUAGE)
|
|
{
|
|
LCType |= LOCALE_USE_CP_ACP;
|
|
}
|
|
|
|
//
|
|
// Get the code page hash node for the given locale.
|
|
//
|
|
pHashN = NlsGetACPFromLocale(Locale, LCType);
|
|
if (pHashN == NULL)
|
|
{
|
|
ResultLen = 0;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Convert to Ansi.
|
|
//
|
|
ResultLen = NlsUnicodeToAnsi( pHashN,
|
|
pBuf,
|
|
ResultLen,
|
|
lpLCData,
|
|
cchData );
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Free the allocated destination buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pBuf, pDTmp);
|
|
|
|
//
|
|
// Return the result of the call to GetLocaleInfoW.
|
|
//
|
|
return (ResultLen);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// SetLocaleInfoA
|
|
//
|
|
// Sets one of the various pieces of information about a particular
|
|
// locale by making an entry in the user's portion of the configuration
|
|
// registry. This will only affect the user override portion of the locale
|
|
// settings. The system defaults will never be reset.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI SetLocaleInfoA(
|
|
LCID Locale,
|
|
LCTYPE LCType,
|
|
LPCSTR lpLCData)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
LPWSTR pUnicode; // ptr to unicode string
|
|
int UnicodeLength; // length of Unicode string
|
|
WCHAR pSTmp[MAX_STRING_LEN]; // tmp Unicode buffer (source)
|
|
BOOL Result; // result
|
|
|
|
|
|
//
|
|
// Get the code page hash node for the given locale.
|
|
//
|
|
pHashN = NlsGetACPFromLocale(Locale, LCType);
|
|
if (pHashN == NULL)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Convert Ansi string to Unicode.
|
|
//
|
|
pUnicode = pSTmp;
|
|
if (!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpLCData,
|
|
-1,
|
|
&pUnicode,
|
|
&UnicodeLength ))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
Result = SetLocaleInfoW( Locale,
|
|
LCType,
|
|
pUnicode );
|
|
|
|
//
|
|
// Free the allocated source buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pUnicode, pSTmp);
|
|
|
|
//
|
|
// Return the result of the call to SetLocaleInfoW.
|
|
//
|
|
return (Result);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetCalendarInfoA
|
|
//
|
|
// Returns one of the various pieces of information about a particular
|
|
// calendar by querying the configuration registry. This call also indicates
|
|
// how much memory is necessary to contain the desired information.
|
|
//
|
|
// 12-17-97 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int WINAPI GetCalendarInfoA(
|
|
LCID Locale,
|
|
CALID Calendar,
|
|
CALTYPE CalType,
|
|
LPSTR lpCalData,
|
|
int cchData,
|
|
LPDWORD lpValue)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
WCHAR pDTmp[MAX_STRING_LEN]; // tmp Unicode buffer (destination)
|
|
LPWSTR pBuf; // ptr to destination buffer
|
|
int ResultLen; // result length
|
|
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - count is negative
|
|
// - NULL data pointer AND count is not zero
|
|
//
|
|
if ((cchData < 0) ||
|
|
((lpCalData == NULL) && (cchData != 0)))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Need to check the parameters based on the CAL_RETURN_NUMBER
|
|
// CalType.
|
|
//
|
|
if (CalType & CAL_RETURN_NUMBER)
|
|
{
|
|
if ((lpCalData != NULL) || (cchData != 0) || (lpValue == NULL))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ((lpValue != NULL) ||
|
|
(cchData < 0) ||
|
|
((lpCalData == NULL) && (cchData != 0)))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
pBuf = pDTmp;
|
|
ResultLen = MAX_STRING_LEN;
|
|
while (1)
|
|
{
|
|
ResultLen = GetCalendarInfoW( Locale,
|
|
Calendar,
|
|
CalType,
|
|
lpCalData ? pBuf : NULL,
|
|
lpCalData ? ResultLen : 0,
|
|
lpValue );
|
|
|
|
//
|
|
// Make sure the static buffer was large enough.
|
|
//
|
|
if ((ResultLen != 0) ||
|
|
(lpValue != NULL) ||
|
|
(GetLastError() != ERROR_INSUFFICIENT_BUFFER))
|
|
{
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Get the size of the buffer needed for the mapping.
|
|
//
|
|
if (ResultLen = GetCalendarInfoW( Locale,
|
|
Calendar,
|
|
CalType,
|
|
NULL,
|
|
0,
|
|
NULL ))
|
|
{
|
|
//
|
|
// Allocate a buffer of the appropriate size.
|
|
//
|
|
if ((pBuf = (LPWSTR)NLS_ALLOC_MEM(ResultLen * sizeof(WCHAR))) == NULL)
|
|
{
|
|
SetLastError(ERROR_OUTOFMEMORY);
|
|
return (0);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Convert the destination Unicode buffer to the given Ansi buffer.
|
|
//
|
|
if (ResultLen > 0)
|
|
{
|
|
if (CalType & CAL_RETURN_NUMBER)
|
|
{
|
|
//
|
|
// For the number value, the result length will actually be
|
|
// twice the amount of the wide char version.
|
|
//
|
|
ResultLen *= 2;
|
|
|
|
//
|
|
// There is nothing else to do in this case, since the value
|
|
// has already been stored in lpValue.
|
|
//
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Get the code page hash node for the given locale.
|
|
//
|
|
pHashN = NlsGetACPFromLocale(Locale, CalType);
|
|
if (pHashN == NULL)
|
|
{
|
|
ResultLen = 0;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Convert to Ansi.
|
|
//
|
|
ResultLen = NlsUnicodeToAnsi( pHashN,
|
|
pBuf,
|
|
ResultLen,
|
|
lpCalData,
|
|
cchData );
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Free the allocated destination buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pBuf, pDTmp);
|
|
|
|
//
|
|
// Return the result of the call to GetCalendarInfoW.
|
|
//
|
|
return (ResultLen);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// SetCalendarInfoA
|
|
//
|
|
// Sets one of the various pieces of information about a particular
|
|
// calendar by making an entry in the user's portion of the configuration
|
|
// registry. This will only affect the user override portion of the
|
|
// calendar settings. The system defaults will never be reset.
|
|
//
|
|
// 12-17-97 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI SetCalendarInfoA(
|
|
LCID Locale,
|
|
CALID Calendar,
|
|
CALTYPE CalType,
|
|
LPCSTR lpCalData)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
LPWSTR pUnicode; // ptr to unicode string
|
|
int UnicodeLength; // length of Unicode string
|
|
WCHAR pSTmp[MAX_STRING_LEN]; // tmp Unicode buffer (source)
|
|
BOOL Result; // result
|
|
|
|
|
|
//
|
|
// Get the code page hash node for the given locale.
|
|
//
|
|
pHashN = NlsGetACPFromLocale(Locale, CalType);
|
|
if (pHashN == NULL)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Convert Ansi string to Unicode.
|
|
//
|
|
pUnicode = pSTmp;
|
|
if (!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpCalData,
|
|
-1,
|
|
&pUnicode,
|
|
&UnicodeLength ))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
Result = SetCalendarInfoW( Locale,
|
|
Calendar,
|
|
CalType,
|
|
pUnicode );
|
|
|
|
//
|
|
// Free the allocated source buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pUnicode, pSTmp);
|
|
|
|
//
|
|
// Return the result of the call to SetCalendarInfoW.
|
|
//
|
|
return (Result);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetTimeFormatA
|
|
//
|
|
// Returns a properly formatted time string for the given locale. It uses
|
|
// either the system time or the specified time. This call also indicates
|
|
// how much memory is necessary to contain the desired information.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int WINAPI GetTimeFormatA(
|
|
LCID Locale,
|
|
DWORD dwFlags,
|
|
CONST SYSTEMTIME *lpTime,
|
|
LPCSTR lpFormat,
|
|
LPSTR lpTimeStr,
|
|
int cchTime)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
LPWSTR pUnicode; // ptr to unicode string
|
|
int UnicodeLength; // length of Unicode string
|
|
WCHAR pSTmp[MAX_STRING_LEN]; // tmp Unicode buffer (source)
|
|
WCHAR pDTmp[MAX_STRING_LEN]; // tmp Unicode buffer (destination)
|
|
LPWSTR pBuf; // ptr to destination buffer
|
|
int ResultLen; // result length
|
|
|
|
|
|
//
|
|
// Get the code page hash node for the given locale.
|
|
//
|
|
pHashN = NlsGetACPFromLocale(Locale, dwFlags);
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - valid code page
|
|
// - count is negative
|
|
// - NULL data pointer AND count is not zero
|
|
//
|
|
if ((pHashN == NULL) ||
|
|
(cchTime < 0) ||
|
|
((lpTimeStr == NULL) && (cchTime != 0)))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Convert Ansi string to Unicode.
|
|
//
|
|
pUnicode = pSTmp;
|
|
if (!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpFormat,
|
|
-1,
|
|
&pUnicode,
|
|
&UnicodeLength ))
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
pBuf = pDTmp;
|
|
ResultLen = MAX_STRING_LEN;
|
|
while (1)
|
|
{
|
|
ResultLen = GetTimeFormatW( Locale,
|
|
dwFlags,
|
|
lpTime,
|
|
pUnicode,
|
|
pBuf,
|
|
ResultLen );
|
|
|
|
//
|
|
// Make sure the static buffer was large enough.
|
|
//
|
|
if ((ResultLen != 0) || (GetLastError() != ERROR_INSUFFICIENT_BUFFER))
|
|
{
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Get the size of the buffer needed for the mapping.
|
|
//
|
|
if (ResultLen = GetTimeFormatW( Locale,
|
|
dwFlags,
|
|
lpTime,
|
|
pUnicode,
|
|
NULL,
|
|
0 ))
|
|
{
|
|
//
|
|
// Allocate a buffer of the appropriate size.
|
|
//
|
|
if ((pBuf = (LPWSTR)NLS_ALLOC_MEM(ResultLen * sizeof(WCHAR))) == NULL)
|
|
{
|
|
NLS_FREE_TMP_BUFFER(pUnicode, pSTmp);
|
|
SetLastError(ERROR_OUTOFMEMORY);
|
|
return (0);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Free the allocated source buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pUnicode, pSTmp);
|
|
|
|
//
|
|
// Convert the destination Unicode buffer to the given Ansi buffer.
|
|
//
|
|
if (ResultLen > 0)
|
|
{
|
|
ResultLen = NlsUnicodeToAnsi( pHashN,
|
|
pBuf,
|
|
ResultLen,
|
|
lpTimeStr,
|
|
cchTime );
|
|
}
|
|
|
|
//
|
|
// Free the allocated destination buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pBuf, pDTmp);
|
|
|
|
//
|
|
// Return the result of the call to GetTimeFormatW.
|
|
//
|
|
return (ResultLen);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetDateFormatA
|
|
//
|
|
// Returns a properly formatted date string for the given locale. It uses
|
|
// either the system date or the specified date. The user may specify
|
|
// the short date format, the long date format, or the year/month format.
|
|
// This call also indicates how much memory is necessary to contain the
|
|
// desired information.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int WINAPI GetDateFormatA(
|
|
LCID Locale,
|
|
DWORD dwFlags,
|
|
CONST SYSTEMTIME *lpDate,
|
|
LPCSTR lpFormat,
|
|
LPSTR lpDateStr,
|
|
int cchDate)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
LPWSTR pUnicode; // ptr to unicode string
|
|
int UnicodeLength; // length of Unicode string
|
|
WCHAR pSTmp[MAX_STRING_LEN]; // tmp Unicode buffer (source)
|
|
WCHAR pDTmp[MAX_STRING_LEN]; // tmp Unicode buffer (destination)
|
|
LPWSTR pBuf; // ptr to destination buffer
|
|
int ResultLen; // result length
|
|
|
|
|
|
//
|
|
// Get the code page hash node for the given locale.
|
|
//
|
|
pHashN = NlsGetACPFromLocale(Locale, dwFlags);
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - valid code page
|
|
// - count is negative
|
|
// - NULL data pointer AND count is not zero
|
|
//
|
|
if ((pHashN == NULL) ||
|
|
(cchDate < 0) ||
|
|
((lpDateStr == NULL) && (cchDate != 0)))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Invalid Flags Check:
|
|
// - flags other than valid ones
|
|
// - more than one of either ltr reading or rtl reading
|
|
// - using LTR or RTL flag and not using cp 1255 or 1256
|
|
//
|
|
if ((dwFlags & GDF_INVALID_FLAG) ||
|
|
(MORE_THAN_ONE(dwFlags, GDF_SINGLE_FLAG)) ||
|
|
((dwFlags & (DATE_LTRREADING | DATE_RTLREADING)) &&
|
|
(pHashN->CodePage != 1255) && (pHashN->CodePage != 1256)))
|
|
{
|
|
SetLastError(ERROR_INVALID_FLAGS);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Convert Ansi string to Unicode.
|
|
//
|
|
pUnicode = pSTmp;
|
|
if (!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpFormat,
|
|
-1,
|
|
&pUnicode,
|
|
&UnicodeLength ))
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
pBuf = pDTmp;
|
|
ResultLen = MAX_STRING_LEN;
|
|
while (1)
|
|
{
|
|
ResultLen = GetDateFormatW( Locale,
|
|
dwFlags,
|
|
lpDate,
|
|
pUnicode,
|
|
pBuf,
|
|
ResultLen );
|
|
|
|
//
|
|
// Make sure the static buffer was large enough.
|
|
//
|
|
if ((ResultLen != 0) || (GetLastError() != ERROR_INSUFFICIENT_BUFFER))
|
|
{
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Get the size of the buffer needed for the mapping.
|
|
//
|
|
if (ResultLen = GetDateFormatW( Locale,
|
|
dwFlags,
|
|
lpDate,
|
|
pUnicode,
|
|
NULL,
|
|
0 ))
|
|
{
|
|
//
|
|
// Allocate a buffer of the appropriate size.
|
|
//
|
|
if ((pBuf = (LPWSTR)NLS_ALLOC_MEM(ResultLen * sizeof(WCHAR))) == NULL)
|
|
{
|
|
NLS_FREE_TMP_BUFFER(pUnicode, pSTmp);
|
|
SetLastError(ERROR_OUTOFMEMORY);
|
|
return (0);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Free the allocated source buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pUnicode, pSTmp);
|
|
|
|
//
|
|
// Convert the destination Unicode buffer to the given Ansi buffer.
|
|
//
|
|
if (ResultLen > 0)
|
|
{
|
|
ResultLen = NlsUnicodeToAnsi( pHashN,
|
|
pBuf,
|
|
ResultLen,
|
|
lpDateStr,
|
|
cchDate );
|
|
}
|
|
|
|
//
|
|
// Free the allocated destination buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pBuf, pDTmp);
|
|
|
|
//
|
|
// Return the result of the call to GetDateFormatW.
|
|
//
|
|
return (ResultLen);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetNumberFormatA
|
|
//
|
|
// Returns a properly formatted number string for the given locale.
|
|
// This call also indicates how much memory is necessary to contain
|
|
// the desired information.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int WINAPI GetNumberFormatA(
|
|
LCID Locale,
|
|
DWORD dwFlags,
|
|
LPCSTR lpValue,
|
|
CONST NUMBERFMTA *lpFormat,
|
|
LPSTR lpNumberStr,
|
|
int cchNumber)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
LPWSTR pValueU; // ptr to unicode string
|
|
int UnicodeLength; // length of Unicode string
|
|
WCHAR pSTmp[MAX_STRING_LEN]; // tmp Unicode buffer (source)
|
|
WCHAR pDTmp[MAX_STRING_LEN]; // tmp Unicode buffer (destination)
|
|
LPWSTR pBuf; // ptr to destination buffer
|
|
int ResultLen; // result length
|
|
NUMBERFMTW FormatU; // Unicode number format
|
|
LPNUMBERFMTW pFormatU = NULL; // ptr to Unicode number format
|
|
|
|
|
|
//
|
|
// Get the code page hash node for the given locale.
|
|
//
|
|
pHashN = NlsGetACPFromLocale(Locale, dwFlags);
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - valid code page
|
|
// - count is negative
|
|
// - NULL data pointer AND count is not zero
|
|
// - ptrs to string buffers same
|
|
//
|
|
if ((pHashN == NULL) ||
|
|
(cchNumber < 0) ||
|
|
((lpNumberStr == NULL) && (cchNumber != 0)) ||
|
|
(lpValue == lpNumberStr))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Convert Ansi string to Unicode.
|
|
//
|
|
pValueU = pSTmp;
|
|
if (!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpValue,
|
|
-1,
|
|
&pValueU,
|
|
&UnicodeLength ))
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// If the format structure exists, convert the strings
|
|
// in the structure.
|
|
//
|
|
if (lpFormat)
|
|
{
|
|
//
|
|
// Copy Ansi structure to Unicode structure.
|
|
//
|
|
FormatU = *(NUMBERFMTW *)lpFormat;
|
|
FormatU.lpDecimalSep = NULL;
|
|
FormatU.lpThousandSep = NULL;
|
|
|
|
//
|
|
// Convert Ansi strings in structure to Unicode strings.
|
|
//
|
|
if (!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpFormat->lpDecimalSep,
|
|
-1,
|
|
&(FormatU.lpDecimalSep),
|
|
&UnicodeLength ) ||
|
|
!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpFormat->lpThousandSep,
|
|
-1,
|
|
&(FormatU.lpThousandSep),
|
|
&UnicodeLength ))
|
|
{
|
|
NLS_FREE_TMP_BUFFER(pValueU, pSTmp);
|
|
NLS_FREE_MEM(FormatU.lpDecimalSep);
|
|
return (0);
|
|
}
|
|
|
|
pFormatU = &FormatU;
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
pBuf = pDTmp;
|
|
ResultLen = MAX_STRING_LEN;
|
|
while (1)
|
|
{
|
|
ResultLen = GetNumberFormatW( Locale,
|
|
dwFlags,
|
|
pValueU,
|
|
pFormatU,
|
|
pBuf,
|
|
ResultLen );
|
|
|
|
//
|
|
// Make sure the static buffer was large enough.
|
|
//
|
|
if ((ResultLen != 0) || (GetLastError() != ERROR_INSUFFICIENT_BUFFER))
|
|
{
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Get the size of the buffer needed for the mapping.
|
|
//
|
|
if (ResultLen = GetNumberFormatW( Locale,
|
|
dwFlags,
|
|
pValueU,
|
|
pFormatU,
|
|
NULL,
|
|
0 ))
|
|
{
|
|
//
|
|
// Allocate a buffer of the appropriate size.
|
|
//
|
|
if ((pBuf = (LPWSTR)NLS_ALLOC_MEM(ResultLen * sizeof(WCHAR))) == NULL)
|
|
{
|
|
SetLastError(ERROR_OUTOFMEMORY);
|
|
ResultLen = 0;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Free the allocated source buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pValueU, pSTmp);
|
|
if (lpFormat)
|
|
{
|
|
NLS_FREE_MEM(FormatU.lpDecimalSep);
|
|
NLS_FREE_MEM(FormatU.lpThousandSep);
|
|
}
|
|
|
|
//
|
|
// Convert the destination Unicode buffer to the given Ansi buffer.
|
|
//
|
|
if (ResultLen > 0)
|
|
{
|
|
ResultLen = NlsUnicodeToAnsi( pHashN,
|
|
pBuf,
|
|
ResultLen,
|
|
lpNumberStr,
|
|
cchNumber );
|
|
}
|
|
|
|
//
|
|
// Free the allocated destination buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pBuf, pDTmp);
|
|
|
|
//
|
|
// Return the result of the call to GetNumberFormatW.
|
|
//
|
|
return (ResultLen);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetCurrencyFormatA
|
|
//
|
|
// Returns a properly formatted currency string for the given locale.
|
|
// This call also indicates how much memory is necessary to contain
|
|
// the desired information.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int WINAPI GetCurrencyFormatA(
|
|
LCID Locale,
|
|
DWORD dwFlags,
|
|
LPCSTR lpValue,
|
|
CONST CURRENCYFMTA *lpFormat,
|
|
LPSTR lpCurrencyStr,
|
|
int cchCurrency)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
LPWSTR pValueU; // ptr to unicode string
|
|
int UnicodeLength; // length of Unicode string
|
|
WCHAR pSTmp[MAX_STRING_LEN]; // tmp Unicode buffer (source)
|
|
WCHAR pDTmp[MAX_STRING_LEN]; // tmp Unicode buffer (destination)
|
|
LPWSTR pBuf; // ptr to destination buffer
|
|
int ResultLen; // result length
|
|
CURRENCYFMTW FormatU; // Unicode currency format
|
|
LPCURRENCYFMTW pFormatU = NULL; // ptr to Unicode currency format
|
|
|
|
|
|
//
|
|
// Get the code page hash node for the given locale.
|
|
//
|
|
pHashN = NlsGetACPFromLocale(Locale, dwFlags);
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - count is negative
|
|
// - NULL data pointer AND count is not zero
|
|
// - ptrs to string buffers same
|
|
//
|
|
if ((pHashN == NULL) ||
|
|
(cchCurrency < 0) ||
|
|
((lpCurrencyStr == NULL) && (cchCurrency != 0)) ||
|
|
(lpValue == lpCurrencyStr))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Convert Ansi string to Unicode.
|
|
//
|
|
pValueU = pSTmp;
|
|
if (!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpValue,
|
|
-1,
|
|
&pValueU,
|
|
&UnicodeLength ))
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// If the format structure exists, convert the strings
|
|
// in the structure.
|
|
//
|
|
if (lpFormat)
|
|
{
|
|
//
|
|
// Copy Ansi structure to Unicode structure.
|
|
//
|
|
FormatU = *(CURRENCYFMTW *)lpFormat;
|
|
FormatU.lpDecimalSep = NULL;
|
|
FormatU.lpThousandSep = NULL;
|
|
FormatU.lpCurrencySymbol = NULL;
|
|
|
|
//
|
|
// Convert Ansi strings in structure to Unicode strings.
|
|
//
|
|
if (!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpFormat->lpDecimalSep,
|
|
-1,
|
|
&(FormatU.lpDecimalSep),
|
|
&UnicodeLength ) ||
|
|
!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpFormat->lpThousandSep,
|
|
-1,
|
|
&(FormatU.lpThousandSep),
|
|
&UnicodeLength ) ||
|
|
!NlsAnsiToUnicode( pHashN,
|
|
0,
|
|
lpFormat->lpCurrencySymbol,
|
|
-1,
|
|
&(FormatU.lpCurrencySymbol),
|
|
&UnicodeLength ))
|
|
{
|
|
NLS_FREE_TMP_BUFFER(pValueU, pSTmp);
|
|
NLS_FREE_MEM(FormatU.lpDecimalSep);
|
|
NLS_FREE_MEM(FormatU.lpThousandSep);
|
|
return (0);
|
|
}
|
|
|
|
pFormatU = &FormatU;
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
pBuf = pDTmp;
|
|
ResultLen = MAX_STRING_LEN;
|
|
while (1)
|
|
{
|
|
ResultLen = GetCurrencyFormatW( Locale,
|
|
dwFlags,
|
|
pValueU,
|
|
pFormatU,
|
|
pBuf,
|
|
ResultLen );
|
|
|
|
//
|
|
// Make sure the static buffer was large enough.
|
|
//
|
|
if ((ResultLen != 0) || (GetLastError() != ERROR_INSUFFICIENT_BUFFER))
|
|
{
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Get the size of the buffer needed for the mapping.
|
|
//
|
|
if (ResultLen = GetCurrencyFormatW( Locale,
|
|
dwFlags,
|
|
pValueU,
|
|
pFormatU,
|
|
NULL,
|
|
0 ))
|
|
{
|
|
//
|
|
// Allocate a buffer of the appropriate size.
|
|
//
|
|
if ((pBuf = (LPWSTR)NLS_ALLOC_MEM(ResultLen * sizeof(WCHAR))) == NULL)
|
|
{
|
|
SetLastError(ERROR_OUTOFMEMORY);
|
|
ResultLen = 0;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Free the allocated source buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pValueU, pSTmp);
|
|
if (lpFormat)
|
|
{
|
|
NLS_FREE_MEM(FormatU.lpDecimalSep);
|
|
NLS_FREE_MEM(FormatU.lpThousandSep);
|
|
NLS_FREE_MEM(FormatU.lpCurrencySymbol);
|
|
}
|
|
|
|
//
|
|
// Convert the destination Unicode buffer to the given Ansi buffer.
|
|
//
|
|
if (ResultLen > 0)
|
|
{
|
|
ResultLen = NlsUnicodeToAnsi( pHashN,
|
|
pBuf,
|
|
ResultLen,
|
|
lpCurrencyStr,
|
|
cchCurrency );
|
|
}
|
|
|
|
//
|
|
// Free the allocated destination buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pBuf, pDTmp);
|
|
|
|
//
|
|
// Return the result of the call to GetCurrencyFormatW.
|
|
//
|
|
return (ResultLen);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// EnumCalendarInfoA
|
|
//
|
|
// Enumerates the specified calendar information that is available for the
|
|
// specified locale, based on the CalType parameter. It does so by
|
|
// passing the pointer to the string buffer containing the calendar info
|
|
// to an application-defined callback function. It continues until the
|
|
// last calendar info is found or the callback function returns FALSE.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI EnumCalendarInfoA(
|
|
CALINFO_ENUMPROCA lpCalInfoEnumProc,
|
|
LCID Locale,
|
|
CALID Calendar,
|
|
CALTYPE CalType)
|
|
{
|
|
return (Internal_EnumCalendarInfo( (NLS_ENUMPROC)lpCalInfoEnumProc,
|
|
Locale,
|
|
Calendar,
|
|
CalType,
|
|
FALSE,
|
|
FALSE ));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// EnumCalendarInfoExA
|
|
//
|
|
// Enumerates the specified calendar information that is available for the
|
|
// specified locale, based on the CalType parameter. It does so by
|
|
// passing the pointer to the string buffer containing the calendar info
|
|
// and the calendar id to an application-defined callback function. It
|
|
// continues until the last calendar info is found or the callback function
|
|
// returns FALSE.
|
|
//
|
|
// 10-14-96 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI EnumCalendarInfoExA(
|
|
CALINFO_ENUMPROCEXA lpCalInfoEnumProcEx,
|
|
LCID Locale,
|
|
CALID Calendar,
|
|
CALTYPE CalType)
|
|
{
|
|
return (Internal_EnumCalendarInfo( (NLS_ENUMPROC)lpCalInfoEnumProcEx,
|
|
Locale,
|
|
Calendar,
|
|
CalType,
|
|
FALSE,
|
|
TRUE ));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// EnumTimeFormatsA
|
|
//
|
|
// Enumerates the time formats that are available for the
|
|
// specified locale, based on the dwFlags parameter. It does so by
|
|
// passing the pointer to the string buffer containing the time format
|
|
// to an application-defined callback function. It continues until the
|
|
// last time format is found or the callback function returns FALSE.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI EnumTimeFormatsA(
|
|
TIMEFMT_ENUMPROCA lpTimeFmtEnumProc,
|
|
LCID Locale,
|
|
DWORD dwFlags)
|
|
{
|
|
return (Internal_EnumTimeFormats( (NLS_ENUMPROC)lpTimeFmtEnumProc,
|
|
Locale,
|
|
dwFlags,
|
|
FALSE ));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// EnumDateFormatsA
|
|
//
|
|
// Enumerates the short date, long date, or year/month formats that are
|
|
// available for the specified locale, based on the dwFlags parameter.
|
|
// It does so by passing the pointer to the string buffer containing the
|
|
// date format to an application-defined callback function. It continues
|
|
// until the last date format is found or the callback function returns
|
|
// FALSE.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI EnumDateFormatsA(
|
|
DATEFMT_ENUMPROCA lpDateFmtEnumProc,
|
|
LCID Locale,
|
|
DWORD dwFlags)
|
|
{
|
|
return (Internal_EnumDateFormats( (NLS_ENUMPROC)lpDateFmtEnumProc,
|
|
Locale,
|
|
dwFlags,
|
|
FALSE,
|
|
FALSE ));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// EnumDateFormatsExA
|
|
//
|
|
// Enumerates the short date, long date, or year/month formats that are
|
|
// available for the specified locale, based on the dwFlags parameter.
|
|
// It does so by passing the pointer to the string buffer containing the
|
|
// date format and the calendar id to an application-defined callback
|
|
// function. It continues until the last date format is found or the
|
|
// callback function returns FALSE.
|
|
//
|
|
// 10-14-96 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI EnumDateFormatsExA(
|
|
DATEFMT_ENUMPROCEXA lpDateFmtEnumProcEx,
|
|
LCID Locale,
|
|
DWORD dwFlags)
|
|
{
|
|
return (Internal_EnumDateFormats( (NLS_ENUMPROC)lpDateFmtEnumProcEx,
|
|
Locale,
|
|
dwFlags,
|
|
FALSE,
|
|
TRUE ));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetStringTypeExA
|
|
//
|
|
// Returns character type information about a particular Ansi string.
|
|
//
|
|
// 01-18-94 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI GetStringTypeExA(
|
|
LCID Locale,
|
|
DWORD dwInfoType,
|
|
LPCSTR lpSrcStr,
|
|
int cchSrc,
|
|
LPWORD lpCharType)
|
|
{
|
|
return (GetStringTypeA( Locale,
|
|
dwInfoType,
|
|
lpSrcStr,
|
|
cchSrc,
|
|
lpCharType));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetStringTypeA
|
|
//
|
|
// Returns character type information about a particular Ansi string.
|
|
//
|
|
// NOTE: The number of parameters is different from GetStringTypeW.
|
|
// The 16-bit OLE product shipped this routine with the wrong
|
|
// parameters (ported from Chicago) and now we must support it.
|
|
//
|
|
// Use GetStringTypeEx to get the same set of parameters between
|
|
// the A and W version.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI GetStringTypeA(
|
|
LCID Locale,
|
|
DWORD dwInfoType,
|
|
LPCSTR lpSrcStr,
|
|
int cchSrc,
|
|
LPWORD lpCharType)
|
|
{
|
|
PCP_HASH pHashCP; // ptr to CP hash node
|
|
LPWSTR pUnicode; // ptr to unicode string
|
|
int UnicodeLength; // length of Unicode string
|
|
WCHAR pSTmp[MAX_STRING_LEN]; // tmp Unicode buffer (source)
|
|
BOOL Result; // result
|
|
|
|
|
|
//
|
|
// Get the code page hash node for the given locale.
|
|
// This will also return an error if the locale id is invalid,
|
|
// so there is no need to check the locale id separately.
|
|
//
|
|
pHashCP = NlsGetACPFromLocale(Locale, 0);
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - Validate LCID
|
|
// - valid code page
|
|
// - same buffer - src and destination
|
|
//
|
|
if ((pHashCP == NULL) ||
|
|
(lpSrcStr == (LPSTR)lpCharType))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Convert Ansi string to Unicode.
|
|
//
|
|
pUnicode = pSTmp;
|
|
if (!NlsAnsiToUnicode( pHashCP,
|
|
MB_INVALID_CHAR_CHECK,
|
|
lpSrcStr,
|
|
cchSrc,
|
|
&pUnicode,
|
|
&UnicodeLength ))
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
Result = GetStringTypeW( dwInfoType,
|
|
pUnicode,
|
|
UnicodeLength,
|
|
lpCharType );
|
|
|
|
//
|
|
// Free the allocated source buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pUnicode, pSTmp);
|
|
|
|
//
|
|
// Return the result of the call to GetStringTypeW.
|
|
//
|
|
return (Result);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// FoldStringA
|
|
//
|
|
// Maps one wide character string to another performing the specified
|
|
// translation. This mapping routine only takes flags that are locale
|
|
// independent.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int WINAPI FoldStringA(
|
|
DWORD dwMapFlags,
|
|
LPCSTR lpSrcStr,
|
|
int cchSrc,
|
|
LPSTR lpDestStr,
|
|
int cchDest)
|
|
{
|
|
LPWSTR pUnicode; // ptr to unicode string
|
|
int UnicodeLength; // length of Unicode string
|
|
WCHAR pSTmp[MAX_STRING_LEN]; // tmp Unicode buffer (source)
|
|
WCHAR pDTmp[MAX_STRING_LEN]; // tmp Unicode buffer (destination)
|
|
LPWSTR pBuf; // ptr to destination buffer
|
|
int ResultLen; // result length
|
|
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - dest buffer size is negative
|
|
// - length of dest string is NOT zero AND dest string is NULL
|
|
// - same buffer - src = destination
|
|
//
|
|
if ((cchDest < 0) ||
|
|
((cchDest != 0) && (lpDestStr == NULL)) ||
|
|
(lpSrcStr == lpDestStr))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
|
|
//
|
|
// Convert Ansi string to Unicode.
|
|
//
|
|
RtlZeroMemory(pSTmp, sizeof(pSTmp));
|
|
pUnicode = pSTmp;
|
|
if (!NlsAnsiToUnicode( gpACPHashN,
|
|
0,
|
|
lpSrcStr,
|
|
cchSrc,
|
|
&pUnicode,
|
|
&UnicodeLength ))
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
pBuf = pDTmp;
|
|
ResultLen = MAX_STRING_LEN;
|
|
while (1)
|
|
{
|
|
ResultLen = FoldStringW( dwMapFlags,
|
|
pUnicode,
|
|
UnicodeLength,
|
|
pBuf,
|
|
ResultLen );
|
|
|
|
//
|
|
// Make sure the static buffer was large enough.
|
|
//
|
|
if ((ResultLen != 0) || (GetLastError() != ERROR_INSUFFICIENT_BUFFER))
|
|
{
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Get the size of the buffer needed for the mapping.
|
|
//
|
|
if (ResultLen = FoldStringW( dwMapFlags,
|
|
pUnicode,
|
|
UnicodeLength,
|
|
NULL,
|
|
0 ))
|
|
{
|
|
//
|
|
// Allocate a buffer of the appropriate size.
|
|
//
|
|
if ((pBuf = (LPWSTR)NLS_ALLOC_MEM(ResultLen * sizeof(WCHAR))) == NULL)
|
|
{
|
|
NLS_FREE_TMP_BUFFER(pUnicode, pSTmp);
|
|
SetLastError(ERROR_OUTOFMEMORY);
|
|
return (0);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Free the allocated source buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pUnicode, pSTmp);
|
|
|
|
//
|
|
// Convert the destination Unicode buffer to the given Ansi buffer.
|
|
//
|
|
if (ResultLen > 0)
|
|
{
|
|
ResultLen = NlsUnicodeToAnsi( gpACPHashN,
|
|
pBuf,
|
|
ResultLen,
|
|
lpDestStr,
|
|
cchDest );
|
|
}
|
|
|
|
//
|
|
// Free the allocated destination buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pBuf, pDTmp);
|
|
|
|
//
|
|
// Return the result of the call to FoldStringW.
|
|
//
|
|
return (ResultLen);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// EnumSystemLanguageGroupsA
|
|
//
|
|
// Enumerates the system language groups that are installed or supported,
|
|
// based on the dwFlags parameter. It does so by passing the pointer to
|
|
// the string buffer containing the language group id to an
|
|
// application-defined callback function. It continues until the last
|
|
// language group id is found or the callback function returns FALSE.
|
|
//
|
|
// 03-10-98 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI EnumSystemLanguageGroupsA(
|
|
LANGUAGEGROUP_ENUMPROCA lpLanguageGroupEnumProc,
|
|
DWORD dwFlags,
|
|
LONG_PTR lParam)
|
|
{
|
|
return (Internal_EnumSystemLanguageGroups(
|
|
(NLS_ENUMPROC)lpLanguageGroupEnumProc,
|
|
dwFlags,
|
|
lParam,
|
|
FALSE ));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// EnumLanguageGroupLocalesA
|
|
//
|
|
// Enumerates the locales in a given language group. It does so by
|
|
// passing the appropriate information to an application-defined
|
|
// callback function. It continues until the last locale in the language
|
|
// group is found or the callback function returns FALSE.
|
|
//
|
|
// 03-10-98 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI EnumLanguageGroupLocalesA(
|
|
LANGGROUPLOCALE_ENUMPROCA lpLangGroupLocaleEnumProc,
|
|
LGRPID LanguageGroup,
|
|
DWORD dwFlags,
|
|
LONG_PTR lParam)
|
|
{
|
|
return (Internal_EnumLanguageGroupLocales(
|
|
(NLS_ENUMPROC)lpLangGroupLocaleEnumProc,
|
|
LanguageGroup,
|
|
dwFlags,
|
|
lParam,
|
|
FALSE ));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// EnumUILanguagesA
|
|
//
|
|
// Enumerates the system UI languages that are installed. It does so by
|
|
// passing the pointer to the string buffer containing the UI language id
|
|
// to an application-defined callback function. It continues until the
|
|
// last UI language id is found or the callback function returns FALSE.
|
|
//
|
|
// 03-10-98 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI EnumUILanguagesA(
|
|
UILANGUAGE_ENUMPROCA lpUILanguageEnumProc,
|
|
DWORD dwFlags,
|
|
LONG_PTR lParam)
|
|
{
|
|
return (Internal_EnumUILanguages( (NLS_ENUMPROC)lpUILanguageEnumProc,
|
|
dwFlags,
|
|
lParam,
|
|
FALSE ));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// EnumSystemLocalesA
|
|
//
|
|
// Enumerates the system locales that are installed or supported, based on
|
|
// the dwFlags parameter. It does so by passing the pointer to the string
|
|
// buffer containing the locale id to an application-defined callback
|
|
// function. It continues until the last locale id is found or the
|
|
// callback function returns FALSE.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI EnumSystemLocalesA(
|
|
LOCALE_ENUMPROCA lpLocaleEnumProc,
|
|
DWORD dwFlags)
|
|
{
|
|
return (Internal_EnumSystemLocales( (NLS_ENUMPROC)lpLocaleEnumProc,
|
|
dwFlags,
|
|
FALSE ));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// EnumSystemCodePagesA
|
|
//
|
|
// Enumerates the system code pages that are installed or supported, based on
|
|
// the dwFlags parameter. It does so by passing the pointer to the string
|
|
// buffer containing the code page id to an application-defined callback
|
|
// function. It continues until the last code page is found or the
|
|
// callback function returns FALSE.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI EnumSystemCodePagesA(
|
|
CODEPAGE_ENUMPROCA lpCodePageEnumProc,
|
|
DWORD dwFlags)
|
|
{
|
|
return (Internal_EnumSystemCodePages( (NLS_ENUMPROC)lpCodePageEnumProc,
|
|
dwFlags,
|
|
FALSE ));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetCPInfoExA
|
|
//
|
|
// Returns information about a given code page.
|
|
//
|
|
// 11-15-96 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI GetCPInfoExA(
|
|
UINT CodePage,
|
|
DWORD dwFlags,
|
|
LPCPINFOEXA lpCPInfoEx)
|
|
{
|
|
CPINFOEXW lpCPInfoExW;
|
|
BOOL rc;
|
|
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - lpCPInfoEx is NULL
|
|
//
|
|
if (lpCPInfoEx == NULL)
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Call the W version of the API.
|
|
//
|
|
rc = GetCPInfoExW(CodePage, dwFlags, &lpCPInfoExW);
|
|
|
|
//
|
|
// Convert the code page name from Unicode to Ansi.
|
|
//
|
|
if (rc == TRUE)
|
|
{
|
|
if (!NlsUnicodeToAnsi( gpACPHashN,
|
|
lpCPInfoExW.CodePageName,
|
|
-1,
|
|
lpCPInfoEx->CodePageName,
|
|
MAX_PATH ))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Copy the rest of the information from the Unicode buffer to the
|
|
// Ansi buffer.
|
|
//
|
|
RtlMoveMemory( lpCPInfoEx,
|
|
&lpCPInfoExW,
|
|
FIELD_OFFSET(CPINFOEXW, CodePageName) );
|
|
|
|
//
|
|
// Return the result.
|
|
//
|
|
return (rc);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetGeoInfoA
|
|
//
|
|
// Wrapper funtion of GetGeoInfoW for ANSI. This function return information
|
|
// about a geographical region.
|
|
//
|
|
// 11-20-99 WeiWu Created
|
|
// 07-03-00 lguindon Began GEO API port
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int WINAPI GetGeoInfoA(
|
|
GEOID GeoId,
|
|
DWORD GeoType,
|
|
LPSTR lpGeoData,
|
|
int cchData,
|
|
LANGID LangId)
|
|
{
|
|
int iRet = 0;
|
|
|
|
//
|
|
// Create buffer initialized to zero.
|
|
//
|
|
WCHAR wszBuffer[MAX_REG_VAL_SIZE] = {0};
|
|
|
|
//
|
|
// Sanity check.
|
|
//
|
|
if ((lpGeoData == NULL) && (cchData > 0))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (iRet);
|
|
}
|
|
|
|
//
|
|
// Call the unicode version.
|
|
//
|
|
iRet = GetGeoInfoW( GeoId,
|
|
GeoType,
|
|
wszBuffer,
|
|
sizeof(wszBuffer) / sizeof(WCHAR),
|
|
LangId );
|
|
|
|
//
|
|
// Convert to ANSI if we get something.
|
|
//
|
|
if (iRet)
|
|
{
|
|
iRet = WideCharToMultiByte( CP_ACP,
|
|
0,
|
|
wszBuffer,
|
|
iRet,
|
|
lpGeoData,
|
|
cchData,
|
|
NULL,
|
|
NULL );
|
|
}
|
|
|
|
return (iRet);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// INTERNAL ROUTINES //
|
|
//-------------------------------------------------------------------------//
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsGetACPFromLocale
|
|
//
|
|
// Gets the CP hash node for the default ACP of the given locale. If
|
|
// either the locale or the code page are invalid, then NULL is returned.
|
|
//
|
|
// 01-19-94 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
PCP_HASH NlsGetACPFromLocale(
|
|
LCID Locale,
|
|
DWORD dwFlags)
|
|
{
|
|
PLOC_HASH pHashN; // ptr to LOC hash node
|
|
PCP_HASH pHashCP; // ptr to CP hash node
|
|
UNICODE_STRING ObUnicodeStr; // value string
|
|
UINT CodePage; // code page value
|
|
|
|
|
|
//
|
|
// See if the system ACP should be used.
|
|
//
|
|
if (dwFlags & (LOCALE_USE_CP_ACP | LOCALE_RETURN_NUMBER))
|
|
{
|
|
return (gpACPHashN);
|
|
}
|
|
|
|
//
|
|
// Get the locale hash node.
|
|
//
|
|
VALIDATE_LOCALE(Locale, pHashN, FALSE);
|
|
if (pHashN == NULL)
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (NULL);
|
|
}
|
|
|
|
//
|
|
// Get the ACP code page. If it's equal to CP_ACP (0), then return
|
|
// the system ACP hash node.
|
|
//
|
|
CodePage = pHashN->pLocaleFixed->DefaultACP;
|
|
if (CodePage == CP_ACP)
|
|
{
|
|
return (gpACPHashN);
|
|
}
|
|
|
|
//
|
|
// Get the CP hash node for the code page.
|
|
//
|
|
pHashCP = GetCPHashNode(CodePage);
|
|
if (pHashCP == NULL)
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
}
|
|
|
|
//
|
|
// Return the code page hash node.
|
|
//
|
|
return (pHashCP);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsAnsiToUnicode
|
|
//
|
|
// Converts an Ansi string to a Unicode string.
|
|
//
|
|
// NOTE: The Unicode buffer is allocated if the routine succeeds, so the
|
|
// caller will need to free the buffer when it is no longer needed.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL NlsAnsiToUnicode(
|
|
PCP_HASH pHashN,
|
|
DWORD dwFlags,
|
|
LPCSTR pAnsiBuffer,
|
|
int AnsiLength,
|
|
LPWSTR *ppUnicodeBuffer,
|
|
int *pUnicodeLength)
|
|
{
|
|
LPWSTR pUnicode; // ptr to Unicode buffer
|
|
ULONG UnicodeLength; // length of the Unicode string
|
|
int ResultLength; // result length of Unicode string
|
|
|
|
|
|
//
|
|
// Make sure the pointer passed in is not null.
|
|
//
|
|
if (pAnsiBuffer == NULL)
|
|
{
|
|
*ppUnicodeBuffer = NULL;
|
|
*pUnicodeLength = 0;
|
|
return (TRUE);
|
|
}
|
|
|
|
//
|
|
// Make sure the Ansi length is set properly (in bytes).
|
|
//
|
|
if (AnsiLength < 0)
|
|
{
|
|
AnsiLength = strlen(pAnsiBuffer) + 1;
|
|
}
|
|
|
|
//
|
|
// See if the static buffer is big enough.
|
|
//
|
|
if ((*ppUnicodeBuffer == NULL) || (AnsiLength > (MAX_STRING_LEN - 1)))
|
|
{
|
|
//
|
|
// Get the size of the Unicode string, including the
|
|
// null terminator.
|
|
//
|
|
UnicodeLength = AnsiLength;
|
|
|
|
//
|
|
// Allocate the Unicode buffer.
|
|
//
|
|
if ((pUnicode = (LPWSTR)NLS_ALLOC_MEM(
|
|
(UnicodeLength + 1) * sizeof(WCHAR) )) == NULL)
|
|
{
|
|
SetLastError(ERROR_OUTOFMEMORY);
|
|
return (FALSE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
UnicodeLength = MAX_STRING_LEN - 1;
|
|
pUnicode = *ppUnicodeBuffer;
|
|
}
|
|
|
|
//
|
|
// Make sure the length of the Ansi string is not zero.
|
|
//
|
|
if (AnsiLength == 0)
|
|
{
|
|
pUnicode[0] = 0;
|
|
*ppUnicodeBuffer = pUnicode;
|
|
*pUnicodeLength = 0;
|
|
return (TRUE);
|
|
}
|
|
|
|
//
|
|
// Convert the Ansi string to a Unicode string.
|
|
//
|
|
ResultLength = SpecialMBToWC( pHashN,
|
|
dwFlags,
|
|
pAnsiBuffer,
|
|
AnsiLength,
|
|
pUnicode,
|
|
UnicodeLength );
|
|
if (ResultLength == 0)
|
|
{
|
|
//
|
|
// Free the allocated Unicode buffer (if one was allocated).
|
|
//
|
|
NLS_FREE_TMP_BUFFER(pUnicode, *ppUnicodeBuffer);
|
|
|
|
//
|
|
// See if the failure was due to insufficient buffer size.
|
|
//
|
|
if (GetLastError() == ERROR_INSUFFICIENT_BUFFER)
|
|
{
|
|
//
|
|
// Get the size of the buffer needed to hold the
|
|
// Unicode string.
|
|
//
|
|
UnicodeLength = SpecialMBToWC( pHashN,
|
|
dwFlags,
|
|
pAnsiBuffer,
|
|
AnsiLength,
|
|
NULL,
|
|
0 );
|
|
//
|
|
// Allocate the Unicode buffer.
|
|
//
|
|
if ((pUnicode = (LPWSTR)NLS_ALLOC_MEM(
|
|
(UnicodeLength + 1) * sizeof(WCHAR) )) == NULL)
|
|
{
|
|
SetLastError(ERROR_OUTOFMEMORY);
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Try the translation again.
|
|
//
|
|
ResultLength = SpecialMBToWC( pHashN,
|
|
dwFlags,
|
|
pAnsiBuffer,
|
|
AnsiLength,
|
|
pUnicode,
|
|
UnicodeLength );
|
|
}
|
|
|
|
//
|
|
// If there was still an error, return failure.
|
|
//
|
|
if (ResultLength == 0)
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (FALSE);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Make sure there is room in the buffer for the null terminator.
|
|
//
|
|
ASSERT(ResultLength <= (int)UnicodeLength);
|
|
|
|
//
|
|
// Null terminate the string.
|
|
//
|
|
pUnicode[ResultLength] = UNICODE_NULL;
|
|
|
|
//
|
|
// Return the Unicode buffer and success.
|
|
//
|
|
*ppUnicodeBuffer = pUnicode;
|
|
*pUnicodeLength = ResultLength;
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsUnicodeToAnsi
|
|
//
|
|
// Converts a Unicode string to an Ansi string.
|
|
//
|
|
// This routine does NOT allocate the Ansi buffer. Instead, it uses the
|
|
// Ansi buffer passed in (unless AnsiLength is 0) and checks for buffer
|
|
// overflow.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int NlsUnicodeToAnsi(
|
|
PCP_HASH pHashN,
|
|
LPCWSTR pUnicodeBuffer,
|
|
int UnicodeLength,
|
|
LPSTR pAnsiBuffer,
|
|
int AnsiLength)
|
|
{
|
|
//
|
|
// Convert the Unicode string to an Ansi string and return the
|
|
// result. The last error will be set appropriately by
|
|
// WideCharToMultiByte.
|
|
//
|
|
return (WideCharToMultiByte( pHashN->CodePage,
|
|
0,
|
|
pUnicodeBuffer,
|
|
UnicodeLength,
|
|
pAnsiBuffer,
|
|
AnsiLength,
|
|
NULL,
|
|
NULL ));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsEnumUnicodeToAnsi
|
|
//
|
|
// Converts a Unicode string to an Ansi string.
|
|
//
|
|
// NOTE: The Ansi buffer is allocated if the routine succeeds, so the
|
|
// caller will need to free the buffer when it is no longer needed.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL NlsEnumUnicodeToAnsi(
|
|
PCP_HASH pHashN,
|
|
LPCWSTR pUnicodeBuffer,
|
|
LPSTR *ppAnsiBuffer)
|
|
{
|
|
LPSTR pAnsi; // ptr to Ansi buffer
|
|
ULONG AnsiLength; // length of the Ansi string
|
|
ULONG UnicodeLength; // length of the Unicode string
|
|
ULONG ResultLength; // result length of Ansi string
|
|
|
|
|
|
//
|
|
// Get the length of the Unicode string (in bytes), including the
|
|
// null terminator.
|
|
//
|
|
UnicodeLength = NlsStrLenW(pUnicodeBuffer) + 1;
|
|
|
|
//
|
|
// Get the size of the Ansi string (in bytes), including the
|
|
// null terminator.
|
|
//
|
|
AnsiLength = UnicodeLength * sizeof(WCHAR);
|
|
|
|
//
|
|
// Allocate the Ansi buffer.
|
|
//
|
|
if ((pAnsi = (LPSTR)NLS_ALLOC_MEM(AnsiLength)) == NULL)
|
|
{
|
|
SetLastError(ERROR_OUTOFMEMORY);
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Convert the Unicode string to an Ansi string.
|
|
// It will already be null terminated.
|
|
//
|
|
ResultLength = WideCharToMultiByte( pHashN->CodePage,
|
|
0,
|
|
pUnicodeBuffer,
|
|
UnicodeLength,
|
|
pAnsi,
|
|
AnsiLength,
|
|
NULL,
|
|
NULL );
|
|
if (ResultLength == 0)
|
|
{
|
|
//
|
|
// Free the allocated Ansi buffer.
|
|
//
|
|
NLS_FREE_MEM(pAnsi);
|
|
|
|
//
|
|
// See if the failure was due to insufficient buffer size.
|
|
//
|
|
if (GetLastError() == ERROR_INSUFFICIENT_BUFFER)
|
|
{
|
|
//
|
|
// Get the size of the buffer needed to hold the
|
|
// ansi string.
|
|
//
|
|
AnsiLength = WideCharToMultiByte( pHashN->CodePage,
|
|
0,
|
|
pUnicodeBuffer,
|
|
UnicodeLength,
|
|
0,
|
|
0,
|
|
NULL,
|
|
NULL );
|
|
//
|
|
// Allocate the Ansi buffer.
|
|
//
|
|
if ((pAnsi = (LPSTR)NLS_ALLOC_MEM(AnsiLength)) == NULL)
|
|
{
|
|
SetLastError(ERROR_OUTOFMEMORY);
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Try the translation again.
|
|
//
|
|
ResultLength = WideCharToMultiByte( pHashN->CodePage,
|
|
0,
|
|
pUnicodeBuffer,
|
|
UnicodeLength,
|
|
pAnsi,
|
|
AnsiLength,
|
|
NULL,
|
|
NULL );
|
|
}
|
|
|
|
//
|
|
// If there was still an error, return failure.
|
|
//
|
|
if (ResultLength == 0)
|
|
{
|
|
NLS_FREE_MEM(pAnsi);
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (FALSE);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Return the Ansi buffer and success.
|
|
//
|
|
*ppAnsiBuffer = pAnsi;
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsDispatchAnsiEnumProc
|
|
//
|
|
// Converts a Unicode string to an Ansi string.
|
|
//
|
|
// 11-10-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL NlsDispatchAnsiEnumProc(
|
|
LCID Locale,
|
|
NLS_ENUMPROC pNlsEnumProc,
|
|
DWORD dwFlags,
|
|
LPWSTR pUnicodeBuffer1,
|
|
LPWSTR pUnicodeBuffer2,
|
|
DWORD dwValue1,
|
|
DWORD dwValue2,
|
|
LONG_PTR lParam,
|
|
BOOL fVersion)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
LPSTR pAnsiBuffer1 = NULL; // ptr to ansi buffer
|
|
LPSTR pAnsiBuffer2 = NULL; // ptr to ansi buffer
|
|
BOOL rc = FALSE; // return code
|
|
|
|
|
|
//
|
|
// Get the code page hash node for the given locale.
|
|
//
|
|
pHashN = NlsGetACPFromLocale(Locale, dwFlags);
|
|
if (pHashN == NULL)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Convert the null-terminated Unicode string to a
|
|
// null-terminated Ansi string.
|
|
//
|
|
if (!NlsEnumUnicodeToAnsi( pHashN,
|
|
pUnicodeBuffer1,
|
|
&pAnsiBuffer1 ))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
if ((pUnicodeBuffer2 != NULL) &&
|
|
(!NlsEnumUnicodeToAnsi( pHashN,
|
|
pUnicodeBuffer2,
|
|
&pAnsiBuffer2 )))
|
|
{
|
|
NLS_FREE_MEM(pAnsiBuffer1);
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Call the callback function.
|
|
//
|
|
switch (fVersion)
|
|
{
|
|
case ( 0 ) :
|
|
{
|
|
rc = (*pNlsEnumProc)(pAnsiBuffer1);
|
|
break;
|
|
}
|
|
case ( 1 ) :
|
|
{
|
|
rc = (*((NLS_ENUMPROCEX)pNlsEnumProc))(pAnsiBuffer1, dwValue1);
|
|
break;
|
|
}
|
|
case ( 2 ) :
|
|
{
|
|
rc = (*((NLS_ENUMPROC2)pNlsEnumProc))( dwValue1,
|
|
dwValue2,
|
|
pAnsiBuffer1,
|
|
lParam );
|
|
break;
|
|
}
|
|
case ( 3 ) :
|
|
{
|
|
rc = (*((NLS_ENUMPROC3)pNlsEnumProc))( dwValue1,
|
|
pAnsiBuffer1,
|
|
pAnsiBuffer2,
|
|
dwValue2,
|
|
lParam );
|
|
break;
|
|
}
|
|
case ( 4 ) :
|
|
{
|
|
rc = (*((NLS_ENUMPROC4)pNlsEnumProc))( pAnsiBuffer1,
|
|
lParam );
|
|
break;
|
|
|
|
}
|
|
}
|
|
|
|
//
|
|
// Free any allocated memory.
|
|
//
|
|
NLS_FREE_MEM(pAnsiBuffer1);
|
|
if (pAnsiBuffer2)
|
|
{
|
|
NLS_FREE_MEM(pAnsiBuffer2);
|
|
}
|
|
|
|
//
|
|
// Return the result.
|
|
//
|
|
return (rc);
|
|
}
|