2120 lines
61 KiB
C
2120 lines
61 KiB
C
/*++
|
|
|
|
Copyright (c) 1991-2000, Microsoft Corporation All rights reserved.
|
|
|
|
Module Name:
|
|
|
|
util.c
|
|
|
|
Abstract:
|
|
|
|
This file contains utility functions that are shared across NLS's code
|
|
modules, but are not necessarily part of any of the existing code
|
|
modules.
|
|
|
|
Private APIs found in this file:
|
|
NlsGetCacheUpdateCount
|
|
|
|
External Routines found in this file:
|
|
IsValidSeparatorString
|
|
IsValidGroupingString
|
|
IsValidCalendarType
|
|
IsValidCalendarTypeStr
|
|
GetUserInfo
|
|
GetPreComposedChar
|
|
GetCompositeChars
|
|
InsertPreComposedForm
|
|
InsertFullWidthPreComposedForm
|
|
InsertCompositeForm
|
|
NlsConvertIntegerToString
|
|
NlsConvertIntegerToHexStringW
|
|
NlsConvertStringToIntegerW
|
|
NlsStrLenW
|
|
NlsStrEqualW
|
|
NlsStrNEqualW
|
|
GetStringTableEntry
|
|
NlsIsDll
|
|
|
|
|
|
Revision History:
|
|
|
|
05-31-91 JulieB Created.
|
|
|
|
--*/
|
|
|
|
|
|
|
|
//
|
|
// Include Files.
|
|
//
|
|
|
|
#include "nls.h"
|
|
#include "nlssafe.h"
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// PRIVATE API ROUTINES //
|
|
//-------------------------------------------------------------------------//
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsGetCacheUpdateCount
|
|
//
|
|
// Returns the current cache update count. The cache update count is
|
|
// updated whenever the HKCU\Control Panel\International settings are
|
|
// modified. This count allows the caller to see if the cache has been
|
|
// updated since the last time this function was called.
|
|
//
|
|
// This private api is needed by the Complex Script Language Pack
|
|
// (CSLPK) to enable it to quickly see if the international section of
|
|
// the registry has been modified.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
ULONG WINAPI NlsGetCacheUpdateCount(void)
|
|
{
|
|
return (pNlsUserInfo->ulCacheUpdateCount);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// EXTERNAL ROUTINES //
|
|
//-------------------------------------------------------------------------//
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// IsValidSeparatorString
|
|
//
|
|
// Returns TRUE if the given string is valid. Otherwise, it returns FALSE.
|
|
//
|
|
// A valid string is one that does NOT contain any code points between
|
|
// L'0' and L'9', and does NOT have a length greater than the maximum.
|
|
//
|
|
// NOTE: The string must be a null terminated string.
|
|
//
|
|
// 10-12-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL IsValidSeparatorString(
|
|
LPCWSTR pString,
|
|
ULONG MaxLength,
|
|
BOOL fCheckZeroLen)
|
|
{
|
|
ULONG Length; // string length
|
|
LPWSTR pCur; // ptr to current position in string
|
|
|
|
|
|
//
|
|
// Search down the string to see if the chars are valid.
|
|
// Save the length of the string.
|
|
//
|
|
pCur = (LPWSTR)pString;
|
|
while (*pCur)
|
|
{
|
|
if ((*pCur >= NLS_CHAR_ZERO) && (*pCur <= NLS_CHAR_NINE))
|
|
{
|
|
//
|
|
// String is NOT valid.
|
|
//
|
|
return (FALSE);
|
|
}
|
|
pCur++;
|
|
}
|
|
Length = (ULONG)(pCur - (LPWSTR)pString);
|
|
|
|
//
|
|
// Make sure the length is not greater than the maximum allowed.
|
|
// Also, check for 0 length string (if appropriate).
|
|
//
|
|
if ((Length >= MaxLength) ||
|
|
((fCheckZeroLen) && (Length == 0)))
|
|
{
|
|
//
|
|
// String is NOT valid.
|
|
//
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// String is valid.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// IsValidGroupingString
|
|
//
|
|
// Returns TRUE if the given string is valid. Otherwise, it returns FALSE.
|
|
//
|
|
// A valid string is one that begins and ends with a number between
|
|
// L'0' and L'9', alternates between a number and a semicolon, and does
|
|
// NOT have a length greater than the maximum.
|
|
// (eg. 3;2;0 or 3;0 or 0 or 3)
|
|
//
|
|
// NOTE: The string must be a null terminated string.
|
|
//
|
|
// 01-05-98 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL IsValidGroupingString(
|
|
LPCWSTR pString,
|
|
ULONG MaxLength,
|
|
BOOL fCheckZeroLen)
|
|
{
|
|
ULONG Length; // string length
|
|
LPWSTR pCur; // ptr to current position in string
|
|
|
|
|
|
//
|
|
// Search down the string to see if the chars are valid.
|
|
// Save the length of the string.
|
|
//
|
|
pCur = (LPWSTR)pString;
|
|
while (*pCur)
|
|
{
|
|
if ((*pCur < NLS_CHAR_ZERO) || (*pCur > NLS_CHAR_NINE))
|
|
{
|
|
//
|
|
// String is NOT valid.
|
|
//
|
|
return (FALSE);
|
|
}
|
|
pCur++;
|
|
|
|
if (*pCur)
|
|
{
|
|
if ((*pCur != NLS_CHAR_SEMICOLON) || (*(pCur + 1) == 0))
|
|
{
|
|
//
|
|
// String is NOT valid.
|
|
//
|
|
return (FALSE);
|
|
}
|
|
pCur++;
|
|
}
|
|
}
|
|
Length = (ULONG)(pCur - (LPWSTR)pString);
|
|
|
|
//
|
|
// Make sure the length is not greater than the maximum allowed.
|
|
// Also, check for 0 length string (if appropriate).
|
|
//
|
|
if ((Length >= MaxLength) ||
|
|
((fCheckZeroLen) && (Length == 0)))
|
|
{
|
|
//
|
|
// String is NOT valid.
|
|
//
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// String is valid.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// IsValidCalendarType
|
|
//
|
|
// Returns the pointer to the optional calendar structure if the given
|
|
// calendar type is valid for the given locale. Otherwise, it returns
|
|
// NULL.
|
|
//
|
|
// 10-12-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
LPWORD IsValidCalendarType(
|
|
PLOC_HASH pHashN,
|
|
CALID CalId)
|
|
{
|
|
LPWORD pOptCal; // ptr to list of optional calendars
|
|
LPWORD pEndOptCal; // ptr to end of list of optional calendars
|
|
|
|
|
|
//
|
|
// Make sure the Cal Id is not zero, since that may be in the
|
|
// optional calendar section (meaning no optional calendars).
|
|
//
|
|
if (CalId == 0)
|
|
{
|
|
return (NULL);
|
|
}
|
|
|
|
//
|
|
// Search down the list of optional calendars.
|
|
//
|
|
pOptCal = (LPWORD)(pHashN->pLocaleHdr) + pHashN->pLocaleHdr->IOptionalCal;
|
|
pEndOptCal = (LPWORD)(pHashN->pLocaleHdr) + pHashN->pLocaleHdr->SDayName1;
|
|
while (pOptCal < pEndOptCal)
|
|
{
|
|
//
|
|
// Check the calendar ids.
|
|
//
|
|
if (CalId == ((POPT_CAL)pOptCal)->CalId)
|
|
{
|
|
//
|
|
// Calendar id is valid for the given locale.
|
|
//
|
|
return (pOptCal);
|
|
}
|
|
|
|
//
|
|
// Increment to the next optional calendar.
|
|
//
|
|
pOptCal += ((POPT_CAL)pOptCal)->Offset;
|
|
}
|
|
|
|
//
|
|
// Calendar id is NOT valid if this point is reached.
|
|
//
|
|
return (NULL);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// IsValidCalendarTypeStr
|
|
//
|
|
// Converts the calendar string to an integer and validates the calendar
|
|
// id for the given locale. It return a pointer to the optional calendar
|
|
// structure, or null if the calendar id was invalid.
|
|
//
|
|
// 10-19-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
LPWORD IsValidCalendarTypeStr(
|
|
PLOC_HASH pHashN,
|
|
LPCWSTR pCalStr)
|
|
{
|
|
UNICODE_STRING ObUnicodeStr; // value string
|
|
CALID CalNum; // calendar id
|
|
|
|
|
|
//
|
|
// Convert the string to an integer value.
|
|
//
|
|
RtlInitUnicodeString(&ObUnicodeStr, pCalStr);
|
|
if (RtlUnicodeStringToInteger(&ObUnicodeStr, 10, &CalNum))
|
|
{
|
|
return (NULL);
|
|
}
|
|
|
|
//
|
|
// Validate the calendar id and return the pointer to the
|
|
// optional calendar structure.
|
|
//
|
|
return (IsValidCalendarType(pHashN, CalNum));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetCPFileNameFromRegistry
|
|
//
|
|
// Gets the name of the code page file from the registry. If pResultBuf
|
|
// or Size == 0, then just return true if it exists in the registry, but
|
|
// don't return the actual value.
|
|
//
|
|
// 05-31-2002 ShawnSte Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL GetCPFileNameFromRegistry(
|
|
UINT CodePage,
|
|
LPWSTR pResultBuf,
|
|
UINT Size)
|
|
{
|
|
// Working things.
|
|
WCHAR pTmpBuf[MAX_SMALL_BUF_LEN]; // temp buffer
|
|
PKEY_VALUE_FULL_INFORMATION pKeyValueFull; // ptr to query info
|
|
BYTE pStatic[MAX_KEY_VALUE_FULLINFO]; // ptr to static buffer
|
|
|
|
//
|
|
// Convert value to unicode string.
|
|
//
|
|
if (!NT_SUCCESS(NlsConvertIntegerToString( CodePage,
|
|
10,
|
|
0,
|
|
pTmpBuf,
|
|
MAX_SMALL_BUF_LEN )))
|
|
{
|
|
// Didn't work. (Don't bother closing key though, its used globally)
|
|
return (FALSE);
|
|
}
|
|
|
|
// Open hCodePageKey, return false if it fails
|
|
OPEN_CODEPAGE_KEY(FALSE);
|
|
|
|
//
|
|
// Query the registry value for that code page.
|
|
//
|
|
pKeyValueFull = (PKEY_VALUE_FULL_INFORMATION)pStatic;
|
|
if ( NO_ERROR != QueryRegValue( hCodePageKey,
|
|
pTmpBuf,
|
|
&pKeyValueFull,
|
|
MAX_KEY_VALUE_FULLINFO,
|
|
NULL ) )
|
|
{
|
|
// Didn't work. (Don't bother closing key though, its used globally)
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Make sure there is data with this value.
|
|
//
|
|
if (GET_VALUE_DATA_PTR(pKeyValueFull)[0] == 0)
|
|
{
|
|
// Nope, no file name for this code page. (Not installed).
|
|
return (FALSE);
|
|
}
|
|
|
|
// It worked, see if that's all they wanted.
|
|
if (!pResultBuf || Size == 0)
|
|
{
|
|
// Caller didn't want the name, just to know if it was there
|
|
return (TRUE);
|
|
}
|
|
|
|
// Now we have to copy the name to their buffer for them.
|
|
if ( FAILED(StringCchCopyW(pResultBuf, Size, GET_VALUE_DATA_PTR(pKeyValueFull))))
|
|
{
|
|
// Couldn't make the string right, so fail
|
|
return (FALSE);
|
|
}
|
|
|
|
// Yea, it worked
|
|
return (TRUE);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetUserInfoFromRegistry
|
|
//
|
|
// Gets the information from the registry for the given value entry.
|
|
//
|
|
// 06-11-98 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL GetUserInfoFromRegistry(
|
|
LPWSTR pValue,
|
|
LPWSTR pOutput,
|
|
size_t cchOutput,
|
|
LCID Locale)
|
|
{
|
|
PKEY_VALUE_FULL_INFORMATION pKeyValueFull; // ptr to query info
|
|
BYTE pStatic[MAX_KEY_VALUE_FULLINFO]; // ptr to static buffer
|
|
HANDLE hKey = NULL; // handle to intl key
|
|
ULONG rc = 0L; // return code
|
|
|
|
|
|
//
|
|
// Open the Control Panel International registry key.
|
|
//
|
|
OPEN_CPANEL_INTL_KEY(hKey, FALSE, KEY_READ);
|
|
|
|
//
|
|
// Initialize the output string.
|
|
//
|
|
*pOutput = 0;
|
|
|
|
pKeyValueFull = (PKEY_VALUE_FULL_INFORMATION)pStatic;
|
|
|
|
//
|
|
// Check to be sure the current user is running in the given locale.
|
|
//
|
|
if (Locale)
|
|
{
|
|
if (NO_ERROR == QueryRegValue( hKey,
|
|
L"Locale",
|
|
&pKeyValueFull,
|
|
MAX_KEY_VALUE_FULLINFO,
|
|
NULL ))
|
|
{
|
|
UINT uiLocale;
|
|
|
|
if (NlsConvertStringToIntegerW(GET_VALUE_DATA_PTR(pKeyValueFull), 16, -1, &uiLocale) &&
|
|
uiLocale != Locale)
|
|
{
|
|
CLOSE_REG_KEY(hKey);
|
|
return FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Query the registry value.
|
|
//
|
|
rc = QueryRegValue( hKey,
|
|
pValue,
|
|
&pKeyValueFull,
|
|
MAX_KEY_VALUE_FULLINFO,
|
|
NULL );
|
|
|
|
//
|
|
// Close the registry key.
|
|
//
|
|
CLOSE_REG_KEY(hKey);
|
|
|
|
//
|
|
// If the query failed or if the output buffer is not large enough,
|
|
// then return failure.
|
|
//
|
|
if ((rc != NO_ERROR) ||
|
|
(pKeyValueFull->DataLength > (MAX_REG_VAL_SIZE * sizeof(WCHAR))))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Save the string in pOutput.
|
|
//
|
|
if(FAILED(StringCchCopyW(pOutput, cchOutput, GET_VALUE_DATA_PTR(pKeyValueFull))))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Return success.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetUserInfo
|
|
//
|
|
// Gets the information from the registry for the given locale and user
|
|
// value entry.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL GetUserInfo(
|
|
LCID Locale,
|
|
LCTYPE LCType,
|
|
SIZE_T CacheOffset,
|
|
LPWSTR pValue,
|
|
LPWSTR pOutput,
|
|
size_t cchOutput,
|
|
BOOL fCheckNull)
|
|
{
|
|
LCID UserLocale;
|
|
HRESULT hr; // return val for string copy
|
|
LPWSTR pCacheString;
|
|
|
|
|
|
//
|
|
// Check if the current thread/process is impersonating
|
|
// or running in the context of a user other than the
|
|
// interactive one.
|
|
//
|
|
if (NT_SUCCESS( NlsGetCurrentUserNlsInfo( Locale,
|
|
LCType,
|
|
pValue,
|
|
pOutput,
|
|
cchOutput,
|
|
FALSE )))
|
|
{
|
|
//
|
|
// See if we need to check for a null string.
|
|
//
|
|
if ((fCheckNull) && (*pOutput == 0))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
return (TRUE);
|
|
}
|
|
|
|
//
|
|
// Running in the same security context as the logged-on user.
|
|
//
|
|
|
|
|
|
RtlEnterCriticalSection(&gcsNlsProcessCache);
|
|
if (pNlsUserInfo->ulCacheUpdateCount != pServerNlsUserInfo->ulCacheUpdateCount)
|
|
{
|
|
//
|
|
// The cache content is out of date. Server has the latest copy of the cache, call server to update the
|
|
// the cache.
|
|
//
|
|
{
|
|
if (!NT_SUCCESS(CsrBasepNlsGetUserInfo(pNlsUserInfo, sizeof(NLS_USER_INFO))))
|
|
{
|
|
RtlLeaveCriticalSection(&gcsNlsProcessCache);
|
|
//
|
|
// The call to client failed, try to get the data from table.
|
|
return (FALSE);
|
|
}
|
|
}
|
|
//
|
|
// If the call to server side succeeds, now we garantee that we have a complete
|
|
// cache data, that is copied from the server side cache. It will have the same
|
|
// ulCacheUpdateCount in the time when the call to server side happens.
|
|
//
|
|
}
|
|
|
|
//
|
|
// We are in critical section here to check UserLocale to make sure that LCID and the data are in sync.
|
|
//
|
|
UserLocale = pNlsUserInfo->UserLocaleId;
|
|
|
|
//
|
|
// Check to be sure cached user locale is the same as the given locale.
|
|
//
|
|
if (Locale != UserLocale)
|
|
{
|
|
RtlLeaveCriticalSection(&gcsNlsProcessCache);
|
|
return (FALSE);
|
|
}
|
|
|
|
pCacheString = (LPWSTR)((LPBYTE)pNlsUserInfo + CacheOffset);
|
|
hr = StringCchCopyW(pOutput, MAX_REG_VAL_SIZE, pCacheString);
|
|
RtlLeaveCriticalSection(&gcsNlsProcessCache);
|
|
|
|
//
|
|
// Make sure the cache is valid.
|
|
//
|
|
// Also, check for an invalid entry. An invalid entry is marked
|
|
// with NLS_INVALID_INFO_CHAR in the first position of the string
|
|
// array.
|
|
//
|
|
if (FAILED(hr) || (*pOutput == NLS_INVALID_INFO_CHAR))
|
|
{
|
|
//
|
|
// The cache is invalid, so try getting the information directly
|
|
// from the registry.
|
|
//
|
|
return (GetUserInfoFromRegistry(pValue, pOutput, cchOutput, Locale));
|
|
}
|
|
|
|
//
|
|
// See if we need to check for a null string.
|
|
//
|
|
if ((fCheckNull) && (*pOutput == 0))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Return success.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetPreComposedChar
|
|
//
|
|
// Gets the precomposed character form of a given base character and
|
|
// nonspacing character. If there is no precomposed form for the given
|
|
// character, it returns 0.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
WCHAR FASTCALL GetPreComposedChar(
|
|
WCHAR wcNonSp,
|
|
WCHAR wcBase)
|
|
{
|
|
PCOMP_INFO pComp; // ptr to composite information
|
|
WORD BSOff = 0; // offset of base char in grid
|
|
WORD NSOff = 0; // offset of nonspace char in grid
|
|
int Index; // index into grid
|
|
|
|
|
|
//
|
|
// Store the ptr to the composite information. No need to check if
|
|
// it's a NULL pointer since all tables in the Unicode file are
|
|
// constructed during initialization.
|
|
//
|
|
pComp = pTblPtrs->pComposite;
|
|
|
|
//
|
|
// Traverse 8:4:4 table for Base character offset.
|
|
//
|
|
BSOff = TRAVERSE_844_W(pComp->pBase, wcBase);
|
|
if (!BSOff)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Traverse 8:4:4 table for NonSpace character offset.
|
|
//
|
|
NSOff = TRAVERSE_844_W(pComp->pNonSp, wcNonSp);
|
|
if (!NSOff)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Get wide character value out of 2D grid.
|
|
// If there is no precomposed character at the location in the
|
|
// grid, it will return 0.
|
|
//
|
|
Index = (BSOff - 1) * pComp->NumNonSp + (NSOff - 1);
|
|
return ((pComp->pGrid)[Index]);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetCompositeChars
|
|
//
|
|
// Gets the composite characters of a given wide character. If the
|
|
// composite form is found, it returns TRUE. Otherwise, it returns
|
|
// FALSE.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL FASTCALL GetCompositeChars(
|
|
WCHAR wch,
|
|
WCHAR *pNonSp,
|
|
WCHAR *pBase)
|
|
{
|
|
PPRECOMP pPreComp; // ptr to precomposed information
|
|
|
|
|
|
//
|
|
// Store the ptr to the precomposed information. No need to check if
|
|
// it's a NULL pointer since all tables in the Unicode file are
|
|
// constructed during initialization.
|
|
//
|
|
pPreComp = pTblPtrs->pPreComposed;
|
|
|
|
//
|
|
// Traverse 8:4:4 table for base and nonspace character translation.
|
|
//
|
|
TRAVERSE_844_D(pPreComp, wch, *pNonSp, *pBase);
|
|
|
|
//
|
|
// Return success if found. Otherwise, error.
|
|
//
|
|
return ((*pNonSp) && (*pBase));
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// InsertPreComposedForm
|
|
//
|
|
// Gets the precomposed form of a given wide character string, places it in
|
|
// the given wide character, and returns the number of composite characters
|
|
// used to form the precomposed form. If there is no precomposed form for
|
|
// the given character, nothing is written into pPreComp and it returns 1
|
|
// for the number of characters used.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int FASTCALL InsertPreComposedForm(
|
|
LPCWSTR pWCStr,
|
|
LPWSTR pEndWCStr,
|
|
LPWSTR pPreComp)
|
|
{
|
|
WCHAR wch; // precomposed character
|
|
LPWSTR pPos; // ptr to position in string
|
|
|
|
|
|
//
|
|
// If no precomposed form can be found, return 1 character used
|
|
// (base character).
|
|
//
|
|
if (((pWCStr + 1) >= pEndWCStr) ||
|
|
(!(wch = GetPreComposedChar(*(pWCStr + 1), *pWCStr))))
|
|
{
|
|
return (1);
|
|
}
|
|
|
|
//
|
|
// Get the precomposed character from the given wide character string.
|
|
// Must check for multiple nonspacing characters for the same
|
|
// precomposed character.
|
|
//
|
|
*pPreComp = wch;
|
|
pPos = (LPWSTR)pWCStr + 2;
|
|
while ((pPos < pEndWCStr) &&
|
|
(wch = GetPreComposedChar(*pPos, *pPreComp)))
|
|
{
|
|
*pPreComp = wch;
|
|
pPos++;
|
|
}
|
|
|
|
//
|
|
// Return the number of characters used to form the precomposed
|
|
// character.
|
|
//
|
|
return ((int)(pPos - (LPWSTR)pWCStr));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// InsertFullWidthPreComposedForm
|
|
//
|
|
// Gets the full width precomposed form of a given wide character string,
|
|
// places it in the given wide character, and returns the number of
|
|
// composite characters used to form the precomposed form. If there is
|
|
// no precomposed form for the given character, only the full width conversion
|
|
// of the first code point is written into pPreComp and it returns 1 for
|
|
// the number of characters used.
|
|
//
|
|
// 11-04-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int FASTCALL InsertFullWidthPreComposedForm(
|
|
LPCWSTR pWCStr,
|
|
LPWSTR pEndWCStr,
|
|
LPWSTR pPreComp,
|
|
PCASE pCase)
|
|
{
|
|
WCHAR wch; // nonspace character
|
|
LPWSTR pPos; // ptr to position in string
|
|
|
|
|
|
//
|
|
// Get the case (if necessary).
|
|
//
|
|
*pPreComp = (pCase) ? GET_LOWER_UPPER_CASE(pCase, *pWCStr) : *pWCStr;
|
|
|
|
//
|
|
// Get the full width.
|
|
//
|
|
*pPreComp = GET_FULL_WIDTH(pTblPtrs->pFullWidth, *pPreComp);
|
|
|
|
if ((pPos = ((LPWSTR)pWCStr + 1)) >= pEndWCStr)
|
|
{
|
|
return (1);
|
|
}
|
|
|
|
while (pPos < pEndWCStr)
|
|
{
|
|
wch = (pCase) ? GET_LOWER_UPPER_CASE(pCase, *pPos) : *pPos;
|
|
wch = GET_FULL_WIDTH(pTblPtrs->pFullWidth, wch);
|
|
if (wch = GetPreComposedChar(wch, *pPreComp))
|
|
{
|
|
*pPreComp = wch;
|
|
pPos++;
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Return the number of characters used to form the precomposed
|
|
// character.
|
|
//
|
|
return ((int)(pPos - (LPWSTR)pWCStr));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// InsertCompositeForm
|
|
//
|
|
// Gets the composite form of a given wide character, places it in the
|
|
// wide character string, and returns the number of characters written.
|
|
// If there is no composite form for the given character, the wide character
|
|
// string is not touched. It will return 1 for the number of characters
|
|
// written, since the base character was already written.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int FASTCALL InsertCompositeForm(
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr)
|
|
{
|
|
WCHAR Base; // base character
|
|
WCHAR NonSp; // non space character
|
|
int wcCount = 0; // number of wide characters written
|
|
LPWSTR pEndComp; // ptr to end of composite form
|
|
int ctr; // loop counter
|
|
|
|
|
|
//
|
|
// If no composite form can be found, return 1 for the base
|
|
// character that was already written.
|
|
//
|
|
if (!GetCompositeChars(*pWCStr, &NonSp, &Base))
|
|
{
|
|
return (1);
|
|
}
|
|
|
|
//
|
|
// Get the composite characters and write them to the pWCStr
|
|
// buffer. Must check for multiple breakdowns of the precomposed
|
|
// character into more than 2 characters (multiple nonspacing
|
|
// characters).
|
|
//
|
|
pEndComp = pWCStr;
|
|
do
|
|
{
|
|
//
|
|
// Make sure pWCStr is big enough to hold the nonspacing
|
|
// character.
|
|
//
|
|
if (pEndComp < (pEndWCStr - 1))
|
|
{
|
|
//
|
|
// Addition of next breakdown of nonspacing characters
|
|
// are to be added right after the base character. So,
|
|
// move all nonspacing characters ahead one position
|
|
// to make room for the next nonspacing character.
|
|
//
|
|
pEndComp++;
|
|
for (ctr = 0; ctr < wcCount; ctr++)
|
|
{
|
|
*(pEndComp - ctr) = *(pEndComp - (ctr + 1));
|
|
}
|
|
|
|
//
|
|
// Fill in the new base form and the new nonspacing character.
|
|
//
|
|
*pWCStr = Base;
|
|
*(pWCStr + 1) = NonSp;
|
|
wcCount++;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Make sure we don't get into an infinite loop if the
|
|
// destination buffer isn't large enough.
|
|
//
|
|
break;
|
|
}
|
|
} while (GetCompositeChars(*pWCStr, &NonSp, &Base));
|
|
|
|
//
|
|
// Return number of wide characters written. Add 1 to include the
|
|
// base character.
|
|
//
|
|
return (wcCount + 1);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsConvertIntegerToString
|
|
//
|
|
// This routine converts an integer to a Unicode string.
|
|
//
|
|
// 11-15-96 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
ULONG NlsConvertIntegerToString(
|
|
UINT Value,
|
|
UINT Base,
|
|
UINT Padding,
|
|
LPWSTR pResultBuf,
|
|
UINT Size)
|
|
{
|
|
UNICODE_STRING ObString; // value string
|
|
UINT ctr; // loop counter
|
|
LPWSTR pBufPtr; // ptr to result buffer
|
|
WCHAR pTmpBuf[MAX_PATH_LEN]; // ptr to temp buffer
|
|
ULONG rc = 0L; // return code
|
|
|
|
//
|
|
// Set up the Unicode string structure.
|
|
//
|
|
ObString.Length = (USHORT)(Size * sizeof(WCHAR));
|
|
ObString.MaximumLength = (USHORT)(Size * sizeof(WCHAR));
|
|
ObString.Buffer = pTmpBuf;
|
|
|
|
//
|
|
// Get the value as a string.
|
|
//
|
|
if (rc = RtlIntegerToUnicodeString(Value, Base, &ObString))
|
|
{
|
|
return (rc);
|
|
}
|
|
|
|
//
|
|
// Pad the string with the appropriate number of zeros.
|
|
//
|
|
pBufPtr = pResultBuf;
|
|
for (ctr = GET_WC_COUNT(ObString.Length);
|
|
ctr < Padding;
|
|
ctr++, pBufPtr++, Size--)
|
|
{
|
|
if( Size < 1 )
|
|
{
|
|
return(STATUS_UNSUCCESSFUL);
|
|
}
|
|
|
|
*pBufPtr = NLS_CHAR_ZERO;
|
|
}
|
|
|
|
if(FAILED(StringCchCopyW(pBufPtr, Size, ObString.Buffer)))
|
|
{
|
|
return(STATUS_UNSUCCESSFUL);
|
|
}
|
|
|
|
return(STATUS_SUCCESS);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsConvertIntegerToHexStringW
|
|
// Convert an integer value to an Unicode null-terminated string WITH
|
|
// leading zeros. E.g. 0x409 with Width 5 will be converted to L"0409".
|
|
// This function is faster than NlsConvertIntegerToString(), but it
|
|
// only supports hex numbers.
|
|
//
|
|
// Parameters:
|
|
// Value The number to be converted.
|
|
// UpperCase If TRUE, the hex digit will be uppercase.
|
|
// Str The buffer for the converted Unicode string.
|
|
// Width The character count of the buffer. The value should be the total
|
|
// heximal digit number plus one for null-terminiated.
|
|
// E.g. if the value is from 0x0000 - 0xffff, the Width should be 5.
|
|
//
|
|
// Return:
|
|
// TRUE if successful. FALSE if the width is not big enough to hold the converted string.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL FASTCALL NlsConvertIntegerToHexStringW(UINT Value, BOOL UpperCase, PWSTR Str, UINT CharCount)
|
|
{
|
|
int Digit;
|
|
PWSTR p;
|
|
|
|
if(Str == NULL)
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
p = Str + CharCount - 1;
|
|
*p-- = L'\0';
|
|
while (p >= Str)
|
|
{
|
|
Digit = Value & 0xf;
|
|
if (Digit < 10)
|
|
{
|
|
Digit = Digit + L'0';
|
|
}
|
|
else
|
|
{
|
|
Digit = Digit - 10 + (UpperCase ? L'A' : L'a');
|
|
}
|
|
*p-- = (WCHAR)Digit;
|
|
Value >>= 4;
|
|
}
|
|
|
|
if (Value > 0)
|
|
{
|
|
//
|
|
// There are still digit remaining.
|
|
//
|
|
return (FALSE);
|
|
}
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsConvertStringToIntegerW
|
|
//
|
|
// Parameters:
|
|
// Str the hex string to be converted.
|
|
// Base base
|
|
// CharCount
|
|
// the character count of the string (excluding the terminiated-null, if any).
|
|
// If the value is -1, this function assumes that
|
|
// Str is a null-terminated string.
|
|
// Result the pointer to the result.
|
|
//
|
|
// Result:
|
|
// TRUE if the operation is successful. FALSE if there is non-hex
|
|
// character in the string.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL FASTCALL NlsConvertStringToIntegerW(PWSTR Str, UINT Base, int CharCount, UINT* Result)
|
|
{
|
|
int i;
|
|
WCHAR Digit;
|
|
WCHAR c;
|
|
|
|
if (Str == NULL || Result == NULL)
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
*Result = 0;
|
|
|
|
if (CharCount == -1)
|
|
{
|
|
while (c = *Str)
|
|
{
|
|
c = *Str;
|
|
if (c >= L'0' && c <= L'9')
|
|
{
|
|
Digit = c - L'0';
|
|
}
|
|
else if(Base == 16)
|
|
{
|
|
if (c >= L'A' && c <= L'F')
|
|
{
|
|
Digit = c - L'A' + 10;
|
|
}
|
|
else if (c >= L'a' && c <= L'f')
|
|
{
|
|
Digit = c - L'a' + 10;
|
|
}
|
|
else
|
|
{
|
|
return (FALSE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
if (Base == 16)
|
|
{
|
|
*Result = (*Result << 4) | Digit;
|
|
}
|
|
else
|
|
{
|
|
*Result = *Result*10 + Digit;
|
|
}
|
|
|
|
Str++;
|
|
}
|
|
} else
|
|
{
|
|
for (i=0; i< CharCount; i++) {
|
|
c = *Str++;
|
|
if (c >= L'0' && c <= L'9')
|
|
{
|
|
Digit = c - L'0';
|
|
}
|
|
else if(Base == 16)
|
|
{
|
|
if (c >= L'A' && c <= L'F')
|
|
{
|
|
Digit = c - L'A' + 10;
|
|
}
|
|
else if (c >= L'a' && c <= L'f')
|
|
{
|
|
Digit = c - L'a' + 10;
|
|
}
|
|
else
|
|
{
|
|
return (FALSE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
if (Base == 16)
|
|
{
|
|
*Result = (*Result << 4) | Digit;
|
|
}
|
|
else
|
|
{
|
|
*Result = *Result*10 + Digit;
|
|
}
|
|
}
|
|
}
|
|
return (TRUE);
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsStrLenW
|
|
//
|
|
// This routine returns the length of the given wide character string.
|
|
// The length does NOT include the null terminator.
|
|
//
|
|
// NOTE: This routine is here to avoid any dependencies on other DLLs
|
|
// during initialization.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int FASTCALL NlsStrLenW(
|
|
LPCWSTR pwsz)
|
|
{
|
|
LPCWSTR pwszStart = pwsz; // ptr to beginning of string
|
|
|
|
loop:
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
if (*pwsz) pwsz++; else goto done;
|
|
|
|
goto loop;
|
|
|
|
done:
|
|
return ((int)(pwsz - pwszStart));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsStrEqualW
|
|
//
|
|
// This routine compares two strings to see if they are exactly identical.
|
|
// It returns 1 if they are identical, 0 if they are different.
|
|
//
|
|
// NOTE: This routine is here to avoid any dependencies on other DLLs
|
|
// during initialization.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int FASTCALL NlsStrEqualW(
|
|
LPCWSTR pwszFirst,
|
|
LPCWSTR pwszSecond)
|
|
{
|
|
loop:
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
|
|
goto loop;
|
|
|
|
error:
|
|
//
|
|
// Return error.
|
|
//
|
|
return (0);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsStrNEqualW
|
|
//
|
|
// This routine compares two strings to see if they are exactly identical
|
|
// for the count of characters given.
|
|
// It returns 1 if they are identical, 0 if they are different.
|
|
//
|
|
// NOTE: This routine is here to avoid any dependencies on other DLLs
|
|
// during initialization.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int FASTCALL NlsStrNEqualW(
|
|
LPCWSTR pwszFirst,
|
|
LPCWSTR pwszSecond,
|
|
int Count)
|
|
{
|
|
loop:
|
|
if (Count == 0) return (1);
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
Count--;
|
|
|
|
if (Count == 0) return (1);
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
Count--;
|
|
|
|
if (Count == 0) return (1);
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
Count--;
|
|
|
|
if (Count == 0) return (1);
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
Count--;
|
|
|
|
if (Count == 0) return (1);
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
Count--;
|
|
|
|
if (Count == 0) return (1);
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
Count--;
|
|
|
|
if (Count == 0) return (1);
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
Count--;
|
|
|
|
if (Count == 0) return (1);
|
|
if (*pwszFirst != *pwszSecond) goto error;
|
|
if (!*pwszFirst) return (1);
|
|
pwszFirst++;
|
|
pwszSecond++;
|
|
Count--;
|
|
|
|
goto loop;
|
|
|
|
error:
|
|
//
|
|
// Return error.
|
|
//
|
|
return (0);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetDefaultSortkeySize
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
ULONG GetDefaultSortkeySize(
|
|
PLARGE_INTEGER pSize)
|
|
{
|
|
*pSize = pTblPtrs->DefaultSortkeySize;
|
|
return (STATUS_SUCCESS);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetLinguistLangSize
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
ULONG GetLinguistLangSize(
|
|
PLARGE_INTEGER pSize)
|
|
{
|
|
*pSize = pTblPtrs->LinguistLangSize;
|
|
return (STATUS_SUCCESS);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// ValidateLocale
|
|
//
|
|
// Internal routine, called from server. Validates that a locale is
|
|
// present in the registry. This code comes from IsValidLocale, but
|
|
// does not check the internal data to prevent recursive calls to the
|
|
// server.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL ValidateLocale(
|
|
LCID Locale)
|
|
{
|
|
PKEY_VALUE_FULL_INFORMATION pKeyValueFull;
|
|
BYTE pStatic1[MAX_KEY_VALUE_FULLINFO];
|
|
BYTE pStatic2[MAX_KEY_VALUE_FULLINFO];
|
|
|
|
WCHAR pTmpBuf[MAX_PATH]; // temp buffer
|
|
UNICODE_STRING ObUnicodeStr; // registry data value string
|
|
DWORD Data; // registry data value
|
|
LPWSTR pData; // ptr to registry data
|
|
BOOL bResult = FALSE; // result value
|
|
|
|
//
|
|
// Invalid Locale Check.
|
|
//
|
|
if (IS_INVALID_LOCALE(Locale))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Open the Locale, the Alternate Sorts, and the Language Groups
|
|
// registry keys.
|
|
//
|
|
OPEN_LOCALE_KEY(FALSE);
|
|
OPEN_ALT_SORTS_KEY(FALSE);
|
|
OPEN_LANG_GROUPS_KEY(FALSE);
|
|
|
|
//
|
|
// Convert locale value to Unicode string.
|
|
//
|
|
if (NlsConvertIntegerToString(Locale, 16, 8, pTmpBuf, MAX_PATH))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Query the registry for the value.
|
|
//
|
|
pKeyValueFull = (PKEY_VALUE_FULL_INFORMATION)pStatic1;
|
|
if (((QueryRegValue( hLocaleKey,
|
|
pTmpBuf,
|
|
&pKeyValueFull,
|
|
MAX_KEY_VALUE_FULLINFO,
|
|
NULL ) == NO_ERROR) ||
|
|
(QueryRegValue( hAltSortsKey,
|
|
pTmpBuf,
|
|
&pKeyValueFull,
|
|
MAX_KEY_VALUE_FULLINFO,
|
|
NULL ) == NO_ERROR)) &&
|
|
(pKeyValueFull->DataLength > 2))
|
|
{
|
|
RtlInitUnicodeString(&ObUnicodeStr, GET_VALUE_DATA_PTR(pKeyValueFull));
|
|
if ((RtlUnicodeStringToInteger(&ObUnicodeStr, 16, &Data) == NO_ERROR) &&
|
|
(Data != 0))
|
|
{
|
|
pKeyValueFull = (PKEY_VALUE_FULL_INFORMATION)pStatic2;
|
|
if ((QueryRegValue( hLangGroupsKey,
|
|
ObUnicodeStr.Buffer,
|
|
&pKeyValueFull,
|
|
MAX_KEY_VALUE_FULLINFO,
|
|
NULL ) == NO_ERROR) &&
|
|
(pKeyValueFull->DataLength > 2))
|
|
{
|
|
pData = GET_VALUE_DATA_PTR(pKeyValueFull);
|
|
if ((pData[0] == L'1') && (pData[1] == 0))
|
|
{
|
|
bResult = TRUE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Return the result.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// ValidateLCType
|
|
//
|
|
// This routine is called from the server (and also from locale.c) in
|
|
// order to get a Registry key name and a field pointer in the NlsInfo
|
|
// structure given an LCType.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL ValidateLCType(
|
|
PNLS_USER_INFO pInfo,
|
|
LCTYPE LCType,
|
|
LPWSTR *ppwReg,
|
|
LPWSTR *ppwCache)
|
|
{
|
|
switch (LCType)
|
|
{
|
|
case ( LOCALE_IFIRSTWEEKOFYEAR ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_IFIRSTWEEKOFYEAR;
|
|
*ppwCache = pInfo->iFirstWeek;
|
|
break;
|
|
}
|
|
case ( LOCALE_IFIRSTDAYOFWEEK ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_IFIRSTDAYOFWEEK;
|
|
*ppwCache = pInfo->iFirstDay;
|
|
break;
|
|
}
|
|
case ( LOCALE_ICALENDARTYPE ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_ICALENDARTYPE;
|
|
*ppwCache = pInfo->iCalType;
|
|
break;
|
|
}
|
|
case ( LOCALE_SLONGDATE ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_SLONGDATE;
|
|
*ppwCache = pInfo->sLongDate;
|
|
break;
|
|
}
|
|
case ( LOCALE_SYEARMONTH ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_SYEARMONTH;
|
|
*ppwCache = pInfo->sYearMonth;
|
|
break;
|
|
}
|
|
case ( LOCALE_S1159 ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_S1159;
|
|
*ppwCache = pInfo->s1159;
|
|
break;
|
|
}
|
|
case ( LOCALE_SNEGATIVESIGN ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_SNEGATIVESIGN;
|
|
*ppwCache = pInfo->sNegSign;
|
|
break;
|
|
}
|
|
case ( LOCALE_SPOSITIVESIGN ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_SPOSITIVESIGN;
|
|
*ppwCache = pInfo->sPosSign;
|
|
break;
|
|
}
|
|
case ( LOCALE_INEGCURR ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_INEGCURR;
|
|
*ppwCache = pInfo->iNegCurr;
|
|
break;
|
|
}
|
|
case ( LOCALE_ICURRENCY ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_ICURRENCY;
|
|
*ppwCache = pInfo->iCurrency;
|
|
break;
|
|
}
|
|
case ( LOCALE_ICURRDIGITS ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_ICURRDIGITS;
|
|
*ppwCache = pInfo->iCurrDigits;
|
|
break;
|
|
}
|
|
case ( LOCALE_SMONGROUPING ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_SMONGROUPING;
|
|
*ppwCache = pInfo->sMonGrouping;
|
|
break;
|
|
}
|
|
case ( LOCALE_SMONTHOUSANDSEP ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_SMONTHOUSANDSEP;
|
|
*ppwCache = pInfo->sMonThouSep;
|
|
break;
|
|
}
|
|
case ( LOCALE_SMONDECIMALSEP ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_SMONDECIMALSEP;
|
|
*ppwCache = pInfo->sMonDecSep;
|
|
break;
|
|
}
|
|
case ( LOCALE_SCURRENCY ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_SCURRENCY;
|
|
*ppwCache = pInfo->sCurrency;
|
|
break;
|
|
}
|
|
case ( LOCALE_IDIGITSUBSTITUTION ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_IDIGITSUBST;
|
|
*ppwCache = pInfo->iDigitSubstitution;
|
|
break;
|
|
}
|
|
case ( LOCALE_SNATIVEDIGITS ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_SNATIVEDIGITS;
|
|
*ppwCache = pInfo->sNativeDigits;
|
|
break;
|
|
}
|
|
case ( LOCALE_INEGNUMBER ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_INEGNUMBER;
|
|
*ppwCache = pInfo->iNegNumber;
|
|
break;
|
|
}
|
|
case ( LOCALE_ILZERO ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_ILZERO;
|
|
*ppwCache = pInfo->iLZero;
|
|
break;
|
|
}
|
|
case ( LOCALE_IDIGITS ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_IDIGITS;
|
|
*ppwCache = pInfo->iDigits;
|
|
break;
|
|
}
|
|
case ( LOCALE_SGROUPING ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_SGROUPING;
|
|
*ppwCache = pInfo->sGrouping;
|
|
break;
|
|
}
|
|
case ( LOCALE_STHOUSAND ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_STHOUSAND;
|
|
*ppwCache = pInfo->sThousand;
|
|
break;
|
|
}
|
|
case ( LOCALE_SDECIMAL ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_SDECIMAL;
|
|
*ppwCache = pInfo->sDecimal;
|
|
break;
|
|
}
|
|
case ( LOCALE_IPAPERSIZE ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_IPAPERSIZE;
|
|
*ppwCache = pInfo->iPaperSize;
|
|
break;
|
|
}
|
|
case ( LOCALE_IMEASURE ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_IMEASURE;
|
|
*ppwCache = pInfo->iMeasure;
|
|
break;
|
|
}
|
|
case ( LOCALE_SLIST ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_SLIST;
|
|
*ppwCache = pInfo->sList;
|
|
break;
|
|
}
|
|
case ( LOCALE_S2359 ) :
|
|
{
|
|
*ppwReg = NLS_VALUE_S2359;
|
|
*ppwCache = pInfo->s2359;
|
|
break;
|
|
}
|
|
default :
|
|
{
|
|
return (FALSE);
|
|
}
|
|
}
|
|
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetStringTableEntry
|
|
//
|
|
// Returns the localized version of the strings for the given resource
|
|
// id. It gets the information from the resource file in the language that
|
|
// the current user is using.
|
|
//
|
|
// The string table contains a series of strings in the following order:
|
|
// Language Name
|
|
// Country Name
|
|
// Language Group Name
|
|
// Code Page Name (decimal values converted to hex values)
|
|
// Region (Geo) Friendly Name (decimal values converted to hex values)
|
|
// Region (Geo) Official Name (decimal values converted to hex values)
|
|
// Sorting Names (in order starting with 0, separated by $)
|
|
//
|
|
// Each string is separated by $. The final string is terminated with
|
|
// a null.
|
|
//
|
|
// The sorting names are in order of the sort ids, starting with 0.
|
|
//
|
|
// For example,
|
|
// "Language$Country$LangGrp$CodePage$Geo1$Geo2$Sort0$Sort1" or
|
|
// "Language$Country" or
|
|
// "$$LangGrp$CodePage" or
|
|
// "$$$CodePage" or
|
|
// "$$$$Geo1$Geo2"
|
|
//
|
|
// 11-17-00 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int GetStringTableEntry(
|
|
UINT ResourceID,
|
|
LANGID UILangId,
|
|
LPWSTR pBuffer,
|
|
int cchBuffer,
|
|
int WhichString)
|
|
{
|
|
HANDLE hFindRes; // handle from find resource
|
|
HANDLE hLoadRes; // handle from load resource
|
|
LPWSTR pSearch, pSearchEnd; // ptrs to search for correct string
|
|
LPWSTR pString; // ptr to final string
|
|
int cchCount = 0; // count of characters
|
|
|
|
|
|
//
|
|
// Make sure the buffer is ok.
|
|
//
|
|
if ((pBuffer == NULL) || (cchBuffer == 0))
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Make sure we're not hitting the GEO ID that is out of bounds.
|
|
//
|
|
// !!! NOTE !!! This is needed because the East Timor Geo Id
|
|
// is out of bounds and wraps to 0x60e7.
|
|
//
|
|
if (ResourceID == 0x60e7)
|
|
{
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Set the UI Language Id.
|
|
//
|
|
if (UILangId == 0)
|
|
{
|
|
UILangId = GetUserDefaultUILanguage();
|
|
}
|
|
|
|
//
|
|
// String Tables are broken up into 16 string segments. Find the
|
|
// resource containing the string we want.
|
|
//
|
|
if ((!(hFindRes = FindResourceExW( hModule,
|
|
RT_STRING,
|
|
(LPWSTR)UlongToPtr((ULONG)(((USHORT)ResourceID >> 4) + 1)),
|
|
(WORD)UILangId ))))
|
|
{
|
|
//
|
|
// Could not find resource. Try NEUTRAL language id.
|
|
//
|
|
if ((!(hFindRes = FindResourceExW( hModule,
|
|
RT_STRING,
|
|
(LPWSTR)UlongToPtr((ULONG)(((USHORT)ResourceID >> 4) + 1)),
|
|
(WORD)0 ))))
|
|
{
|
|
//
|
|
// Could not find resource. Return 0.
|
|
//
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Load the resource.
|
|
//
|
|
if (hLoadRes = LoadResource(hModule, hFindRes))
|
|
{
|
|
//
|
|
// Lock the resource. Store the found pointer in the given
|
|
// pointer.
|
|
//
|
|
if (pSearch = (LPWSTR)LockResource(hLoadRes))
|
|
{
|
|
//
|
|
// Move past the other strings in this segment.
|
|
// (16 strings in a segment -> & 0x0F)
|
|
//
|
|
ResourceID &= 0x0F;
|
|
|
|
//
|
|
// Find the correct string in this segment.
|
|
//
|
|
while (TRUE)
|
|
{
|
|
cchCount = *((WORD *)pSearch++);
|
|
if (ResourceID-- == 0)
|
|
{
|
|
break;
|
|
}
|
|
pSearch += cchCount;
|
|
}
|
|
|
|
//
|
|
// Mark the end of the resource string since it is not
|
|
// NULL terminated.
|
|
//
|
|
pSearchEnd = pSearch + cchCount;
|
|
|
|
//
|
|
// Get to the appropriate string.
|
|
//
|
|
while ((WhichString > 0) && (pSearch < pSearchEnd))
|
|
{
|
|
do
|
|
{
|
|
if (*pSearch == RC_STRING_SEPARATOR)
|
|
{
|
|
pSearch++;
|
|
break;
|
|
}
|
|
pSearch++;
|
|
|
|
} while (pSearch < pSearchEnd);
|
|
|
|
WhichString--;
|
|
}
|
|
|
|
//
|
|
// Count the number of characters for this string.
|
|
//
|
|
pString = pSearch;
|
|
cchCount = 0;
|
|
while ((pSearch < pSearchEnd) && (*pSearch != RC_STRING_SEPARATOR))
|
|
{
|
|
pSearch++;
|
|
cchCount++;
|
|
}
|
|
|
|
//
|
|
// See if there is anything to copy.
|
|
//
|
|
if (cchCount > 0)
|
|
{
|
|
//
|
|
// Don't copy more than the max allowed.
|
|
//
|
|
if (cchCount >= cchBuffer)
|
|
{
|
|
cchCount = cchBuffer - 1;
|
|
}
|
|
|
|
//
|
|
// Copy the string into the buffer and NULL terminate it.
|
|
//
|
|
CopyMemory(pBuffer, pString, cchCount * sizeof(WCHAR));
|
|
pBuffer[cchCount] = 0;
|
|
|
|
//
|
|
// Return the number of characters in the string, not
|
|
// including the NULL terminator.
|
|
//
|
|
return (cchCount);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Return failure.
|
|
//
|
|
return (0);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// NlsIsDll
|
|
//
|
|
// Check if file extension is DLL
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define DLL_SUFFIX_LENGTH 4 // (.XXX)
|
|
|
|
BOOL FASTCALL NlsIsDll(
|
|
LPCWSTR pFileName
|
|
)
|
|
{
|
|
BOOL bIsDll = FALSE;
|
|
|
|
|
|
if (pFileName)
|
|
{
|
|
size_t iLen = 0;
|
|
|
|
if(SUCCEEDED(StringCchLengthW(pFileName, MAX_PATH, &iLen)))
|
|
{
|
|
//
|
|
// Check DLL extension, save the trouble of calling lstricmpW
|
|
//
|
|
// REVIEW: lstricmpW would not be an appropriate function to
|
|
// call here anyway, since user locale collation
|
|
// semantics != file system collation semantics.
|
|
//
|
|
if (iLen > DLL_SUFFIX_LENGTH)
|
|
{
|
|
pFileName += iLen - DLL_SUFFIX_LENGTH;
|
|
|
|
//
|
|
// File names are lower case in setup, so optimize for that
|
|
// by putting them first.
|
|
//
|
|
if ((pFileName[0] == L'.') &&
|
|
(pFileName[1] == L'd' || pFileName[1] == L'D') &&
|
|
(pFileName[2] == L'l' || pFileName[2] == L'L') &&
|
|
(pFileName[3] == L'l' || pFileName[3] == L'L'))
|
|
{
|
|
bIsDll = TRUE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return bIsDll;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// IsCodePointDefined
|
|
//
|
|
// Check if the code point is defined.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL FASTCALL IsSortingCodePointDefined(
|
|
LPNLSVERSIONINFO lpVersionInformation,
|
|
LPCWSTR lpString,
|
|
INT cchStr
|
|
)
|
|
{
|
|
PNLSDEFINED pDefinedCodePoints = NULL;
|
|
LPCWSTR pStringEnd;
|
|
|
|
//
|
|
// Make sure the appropriate tables are available. If not,
|
|
// return an error.
|
|
//
|
|
if ((pTblPtrs->pDefinedVersion == NULL) ||
|
|
(pTblPtrs->pSortingTableFileBase == NULL) ||
|
|
(pTblPtrs->pDefaultSortkey == NULL))
|
|
{
|
|
KdPrint(("NLSAPI: Appropriate Tables (Defined, Base and/or Default) Not Loaded.\n"));
|
|
SetLastError(ERROR_FILE_NOT_FOUND);
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Get the version.
|
|
//
|
|
if (lpVersionInformation != NULL)
|
|
{
|
|
UINT idx;
|
|
|
|
//
|
|
// Buffer size check.
|
|
//
|
|
if (lpVersionInformation->dwNLSVersionInfoSize != sizeof(NLSVERSIONINFO))
|
|
{
|
|
SetLastError( ERROR_INSUFFICIENT_BUFFER );
|
|
return (FALSE);
|
|
}
|
|
|
|
if ((lpVersionInformation->dwDefinedVersion == 0L) ||
|
|
(lpVersionInformation->dwDefinedVersion == (pTblPtrs->pDefinedVersion)[0].Version))
|
|
{
|
|
// Use the current version.
|
|
// Do nothing here. We let pDefinedCodePoints to be NULL, so that current table is used.
|
|
}
|
|
else
|
|
{
|
|
if (lpVersionInformation->dwDefinedVersion < pTblPtrs->NumDefinedVersion) {
|
|
//
|
|
// Not the default version, get the the requested version.
|
|
//
|
|
pDefinedCodePoints = (PNLSDEFINED)(pTblPtrs->pSortingTableFileBase + (pTblPtrs->pDefinedVersion)[lpVersionInformation->dwDefinedVersion].dwOffset);
|
|
}
|
|
//
|
|
// Check if the version requested is valid.
|
|
//
|
|
if (pDefinedCodePoints == NULL)
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (FALSE);
|
|
}
|
|
}
|
|
}
|
|
|
|
pStringEnd = lpString + cchStr;
|
|
//
|
|
// Check if we deal with the current version.
|
|
//
|
|
if (pDefinedCodePoints == NULL)
|
|
{
|
|
//
|
|
// Use the default table.
|
|
//
|
|
// For each code point verify is they exist
|
|
// in the table.
|
|
//
|
|
while (lpString < pStringEnd)
|
|
{
|
|
//
|
|
// Check is the fist script member is defined for this codepoint.
|
|
//
|
|
if ((pTblPtrs->pDefaultSortkey)[*lpString].UW.SM_AW.Script == UNSORTABLE)
|
|
{
|
|
//
|
|
// Check for the NULL case and formatting characters case. Not
|
|
// defined but valid.
|
|
//
|
|
if ((*lpString == L'\x0000') ||
|
|
(*lpString == L'\x0640') ||
|
|
((*lpString >= L'\x180B') && (*lpString <= L'\x180E')) ||
|
|
((*lpString >= L'\x200C') && (*lpString <= L'\x200F')) ||
|
|
((*lpString >= L'\x202A') && (*lpString <= L'\x202E')) ||
|
|
((*lpString >= L'\x206A') && (*lpString <= L'\x206F')) ||
|
|
(*lpString == L'\xFEFF') ||
|
|
(*lpString == L'\xFFF9') ||
|
|
((*lpString >= L'\xFFFA') && (*lpString <= L'\xFFFD')))
|
|
{
|
|
lpString++;
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
return (FALSE);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Eliminate Private Use characters. They are defined but cannot be considered
|
|
// valid.
|
|
//
|
|
if ((*lpString >= L'\xE000') && (*lpString <= L'\xF8FF'))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Eliminate invalid surogates pairs or single surrogates.
|
|
//
|
|
if ((*lpString >= L'\xDC00') && (*lpString <= L'\xDFFF')) // Leading low surrogate
|
|
{
|
|
return (FALSE);
|
|
}
|
|
else if ((*lpString >= L'\xD800') && (*lpString <= L'\xDBFF')) // Leading high surrogate
|
|
{
|
|
if ( ((lpString + 1) < pStringEnd) && // Surrogates not the last character
|
|
(*(lpString + 1) >= L'\xDC00') && (*(lpString + 1) <= L'\xDFFF')) // Low surrogate
|
|
{
|
|
lpString++; // Valid surrogates pair, High followed by a low surrogate. Skip the pair!
|
|
}
|
|
else
|
|
{
|
|
return (FALSE);
|
|
}
|
|
}
|
|
|
|
lpString++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
WORD wIndex;
|
|
BYTE wMod32Val;
|
|
|
|
while (lpString < pStringEnd)
|
|
{
|
|
//
|
|
// Compute the modulo 32 of the code point value.
|
|
//
|
|
wMod32Val = (BYTE)(*lpString & 0x0000001f); // 0x1fff => 5 bits
|
|
|
|
//
|
|
// Compute the DWORD index that contain the desired code point.
|
|
//
|
|
wIndex = (WORD)(*lpString >> 5);
|
|
|
|
//
|
|
// Get the DWORD aligned entry that contain the desired code point.
|
|
//
|
|
// Note: We need to get a DWORD aligned value to make sure that we
|
|
// that we don't access memory outside the table especially at the
|
|
// end of the table.
|
|
//
|
|
|
|
//
|
|
// Shift the value to retrieve information about code point at the
|
|
// position 0.
|
|
//
|
|
//
|
|
// Check is the code point is defined or not.
|
|
//
|
|
if ((pDefinedCodePoints[wIndex] >> wMod32Val) == 0)
|
|
{
|
|
// NOTENOTE YSLin: In NLSTrans, make sure that we mark U+0000 as 1, instead of 0.
|
|
// NOTENOTE lguindon: In NLSTrans, make sure that we mark U+E000-U+F8FF as 0.
|
|
// NOTENOTE lguindon: In NLSTrans, make sure that we mark U+070F as 1.
|
|
// NOTENOTE lguindon: In NLSTrans, make sure that we mark U+0640 as 1.
|
|
// NOTENOTE lguindon: In NLSTrans, make sure that we mark U+180B-U+180E as 1.
|
|
// NOTENOTE lguindon: In NLSTrans, make sure that we mark U+200C-U+200F as 1.
|
|
// NOTENOTE lguindon: In NLSTrans, make sure that we mark U+202A-U+202E as 1.
|
|
// NOTENOTE lguindon: In NLSTrans, make sure that we mark U+206A-U+206F as 1.
|
|
// NOTENOTE lguindon: In NLSTrans, make sure that we mark U+FEFF as 1.
|
|
// NOTENOTE lguindon: In NLSTrans, make sure that we mark U+FFF9 as 1.
|
|
// NOTENOTE lguindon: In NLSTrans, make sure that we mark U+FFFA-U+FFFD as 1.
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Eliminate invalid surogates pairs or single surrogates.
|
|
//
|
|
if ((*lpString >= L'\xDC00') && (*lpString <= L'\xDFFF')) // Leading low surrogate
|
|
{
|
|
return (FALSE);
|
|
}
|
|
else if ((*lpString >= L'\xD800') && (*lpString <= L'\xDBFF')) // Leading high surrogate
|
|
{
|
|
if ( ((lpString + 1) < pStringEnd) && // Surrogates not the last character
|
|
(*(lpString + 1) >= L'\xDC00') && (*(lpString + 1) <= L'\xDFFF')) // Low surrogate
|
|
{
|
|
lpString++; // Valid surrogates pair, High followed by a low surrogate. Skip the pair!
|
|
}
|
|
else
|
|
{
|
|
return (FALSE);
|
|
}
|
|
}
|
|
|
|
lpString++;
|
|
}
|
|
}
|
|
return (TRUE);
|
|
}
|
|
|