|
|
/*++
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"
//
// 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) { 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); }
|