4019 lines
147 KiB
C
4019 lines
147 KiB
C
/*++
|
|
|
|
Copyright (c) 1991-2000, Microsoft Corporation All rights reserved.
|
|
|
|
Module Name:
|
|
|
|
mbcs.c
|
|
|
|
Abstract:
|
|
|
|
This file contains functions that convert multibyte character strings
|
|
to wide character strings, convert wide character strings to multibyte
|
|
character strings, convert a multibyte character string from one code
|
|
page to a multibyte character string of another code page, and get the
|
|
DBCS leadbyte ranges for a given code page.
|
|
|
|
APIs found in this file:
|
|
IsValidCodePage
|
|
GetACP
|
|
GetOEMCP
|
|
GetCPInfo
|
|
GetCPInfoExW
|
|
IsDBCSLeadByte
|
|
IsDBCSLeadByteEx
|
|
MultiByteToWideChar
|
|
WideCharToMultiByte
|
|
|
|
Revision History:
|
|
|
|
05-31-91 JulieB Created.
|
|
|
|
--*/
|
|
|
|
|
|
|
|
//
|
|
// Include Files.
|
|
//
|
|
|
|
#include "nls.h"
|
|
#include "nlssafe.h"
|
|
|
|
|
|
|
|
|
|
//
|
|
// Forward Declarations.
|
|
//
|
|
|
|
int
|
|
GetWCCompSB(
|
|
PMB_TABLE pMBTbl,
|
|
LPBYTE pMBStr,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr);
|
|
|
|
int
|
|
GetWCCompMB(
|
|
PCP_HASH pHashN,
|
|
PMB_TABLE pMBTbl,
|
|
LPBYTE pMBStr,
|
|
LPBYTE pEndMBStr,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr,
|
|
int *pmbIncr);
|
|
|
|
int
|
|
GetWCCompSBErr(
|
|
PCP_HASH pHashN,
|
|
PMB_TABLE pMBTbl,
|
|
LPBYTE pMBStr,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr);
|
|
|
|
int
|
|
GetWCCompMBErr(
|
|
PCP_HASH pHashN,
|
|
PMB_TABLE pMBTbl,
|
|
LPBYTE pMBStr,
|
|
LPBYTE pEndMBStr,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr,
|
|
int *pmbIncr);
|
|
|
|
int
|
|
GetMBNoDefault(
|
|
PCP_HASH pHashN,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr,
|
|
LPBYTE pMBStr,
|
|
int cbMultiByte,
|
|
DWORD dwFlags);
|
|
|
|
int
|
|
GetMBDefault(
|
|
PCP_HASH pHashN,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr,
|
|
LPBYTE pMBStr,
|
|
int cbMultiByte,
|
|
WORD wDefault,
|
|
LPBOOL pUsedDef,
|
|
DWORD dwFlags);
|
|
|
|
int
|
|
GetMBDefaultComp(
|
|
PCP_HASH pHashN,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr,
|
|
LPBYTE pMBStr,
|
|
int cbMultiByte,
|
|
WORD wDefault,
|
|
LPBOOL pUsedDef,
|
|
DWORD dwFlags);
|
|
|
|
int
|
|
GetMBCompSB(
|
|
PCP_HASH pHashN,
|
|
DWORD dwFlags,
|
|
LPWSTR pWCStr,
|
|
LPBYTE pMBStr,
|
|
int mbCount,
|
|
WORD wDefault,
|
|
LPBOOL pUsedDef);
|
|
|
|
int
|
|
GetMBCompMB(
|
|
PCP_HASH pHashN,
|
|
DWORD dwFlags,
|
|
LPWSTR pWCStr,
|
|
LPBYTE pMBStr,
|
|
int mbCount,
|
|
WORD wDefault,
|
|
LPBOOL pUsedDef,
|
|
BOOL *fError,
|
|
BOOL fOnlyOne);
|
|
|
|
UINT
|
|
GetMacCodePage(void);
|
|
|
|
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// INTERNAL MACROS //
|
|
//-------------------------------------------------------------------------//
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// CHECK_DBCS_LEAD_BYTE
|
|
//
|
|
// Returns the offset to the DBCS table for the given leadbyte character.
|
|
// If the given character is not a leadbyte, then it returns zero (table
|
|
// value).
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define CHECK_DBCS_LEAD_BYTE(pDBCSOff, Ch) \
|
|
(pDBCSOff ? ((WORD)(pDBCSOff[Ch])) : ((WORD)0))
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// CHECK_ERROR_WC_SINGLE
|
|
//
|
|
// Checks to see if the default character was used due to an invalid
|
|
// character. Sets last error and returns 0 characters written if an
|
|
// invalid character was used.
|
|
//
|
|
// NOTE: This macro may return if an error is encountered.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define CHECK_ERROR_WC_SINGLE( pHashN, \
|
|
wch, \
|
|
Ch ) \
|
|
{ \
|
|
if ( ( (wch == pHashN->pCPInfo->wUniDefaultChar) && \
|
|
(Ch != pHashN->pCPInfo->wTransUniDefaultChar) ) || \
|
|
( (wch >= PRIVATE_USE_BEGIN) && (wch <= PRIVATE_USE_END) ) ) \
|
|
{ \
|
|
SetLastError(ERROR_NO_UNICODE_TRANSLATION); \
|
|
return (0); \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// CHECK_ERROR_WC_MULTI
|
|
//
|
|
// Checks to see if the default character was used due to an invalid
|
|
// character. Sets last error and returns 0 characters written if an
|
|
// invalid character was used.
|
|
//
|
|
// NOTE: This macro may return if an error is encountered.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define CHECK_ERROR_WC_MULTI( pHashN, \
|
|
wch, \
|
|
lead, \
|
|
trail ) \
|
|
{ \
|
|
if ((wch == pHashN->pCPInfo->wUniDefaultChar) && \
|
|
(MAKEWORD(trail, lead) != pHashN->pCPInfo->wTransUniDefaultChar)) \
|
|
{ \
|
|
SetLastError(ERROR_NO_UNICODE_TRANSLATION); \
|
|
return (0); \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// CHECK_ERROR_WC_MULTI_SPECIAL
|
|
//
|
|
// Checks to see if the default character was used due to an invalid
|
|
// character. Sets it to 0xffff if invalid.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 08-21-95 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define CHECK_ERROR_WC_MULTI_SPECIAL( pHashN, \
|
|
pWCStr, \
|
|
lead, \
|
|
trail ) \
|
|
{ \
|
|
if ((*pWCStr == pHashN->pCPInfo->wUniDefaultChar) && \
|
|
(MAKEWORD(trail, lead) != pHashN->pCPInfo->wTransUniDefaultChar)) \
|
|
{ \
|
|
*pWCStr = 0xffff; \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GET_WC_SINGLE
|
|
//
|
|
// Fills in pWCStr with the wide character(s) for the corresponding single
|
|
// byte character from the appropriate translation table.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define GET_WC_SINGLE( pMBTbl, \
|
|
pMBStr, \
|
|
pWCStr ) \
|
|
{ \
|
|
*pWCStr = pMBTbl[*pMBStr]; \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GET_WC_SINGLE_SPECIAL
|
|
//
|
|
// Fills in pWCStr with the wide character(s) for the corresponding single
|
|
// byte character from the appropriate translation table. Also checks for
|
|
// invalid characters - if invalid, it fills in 0xffff instead.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 08-21-95 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define GET_WC_SINGLE_SPECIAL( pHashN, \
|
|
pMBTbl, \
|
|
pMBStr, \
|
|
pWCStr ) \
|
|
{ \
|
|
*pWCStr = pMBTbl[*pMBStr]; \
|
|
\
|
|
if ( ( (*pWCStr == pHashN->pCPInfo->wUniDefaultChar) && \
|
|
(*pMBStr != pHashN->pCPInfo->wTransUniDefaultChar) ) || \
|
|
( (*pWCStr >= PRIVATE_USE_BEGIN) && \
|
|
(*pWCStr <= PRIVATE_USE_END) ) ) \
|
|
{ \
|
|
*pWCStr = 0xffff; \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GET_WC_MULTI
|
|
//
|
|
// Fills in pWCStr with the wide character(s) for the corresponding multibyte
|
|
// character from the appropriate translation table. The number of bytes
|
|
// used from the pMBStr buffer (single byte or double byte) is stored in
|
|
// the mbIncr parameter.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define GET_WC_MULTI( pHashN, \
|
|
pMBTbl, \
|
|
pMBStr, \
|
|
pEndMBStr, \
|
|
pWCStr, \
|
|
pEndWCStr, \
|
|
mbIncr ) \
|
|
{ \
|
|
WORD Offset; /* offset to DBCS table for range */ \
|
|
\
|
|
\
|
|
if (Offset = CHECK_DBCS_LEAD_BYTE(pHashN->pDBCSOffsets, *pMBStr)) \
|
|
{ \
|
|
/* \
|
|
* DBCS Lead Byte. Make sure there is a trail byte with the \
|
|
* lead byte. \
|
|
*/ \
|
|
if (pMBStr + 1 == pEndMBStr) \
|
|
{ \
|
|
/* \
|
|
* There is no trail byte with the lead byte. The lead byte \
|
|
* is the LAST character in the string. Translate to NULL. \
|
|
*/ \
|
|
*pWCStr = (WCHAR)0; \
|
|
mbIncr = 1; \
|
|
} \
|
|
else if (*(pMBStr + 1) == 0) \
|
|
{ \
|
|
/* \
|
|
* There is no trail byte with the lead byte. The lead byte \
|
|
* is followed by a NULL. Translate to NULL. \
|
|
* \
|
|
* Increment by 2 so that the null is not counted twice. \
|
|
*/ \
|
|
*pWCStr = (WCHAR)0; \
|
|
mbIncr = 2; \
|
|
} \
|
|
else \
|
|
{ \
|
|
/* \
|
|
* Fill in the wide character translation from the double \
|
|
* byte character table. \
|
|
*/ \
|
|
*pWCStr = (pHashN->pDBCSOffsets + Offset)[*(pMBStr + 1)]; \
|
|
mbIncr = 2; \
|
|
} \
|
|
} \
|
|
else \
|
|
{ \
|
|
/* \
|
|
* Not DBCS Lead Byte. Fill in the wide character translation \
|
|
* from the single byte character table. \
|
|
*/ \
|
|
*pWCStr = pMBTbl[*pMBStr]; \
|
|
mbIncr = 1; \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GET_WC_MULTI_ERR
|
|
//
|
|
// Fills in pWCStr with the wide character(s) for the corresponding multibyte
|
|
// character from the appropriate translation table. The number of bytes
|
|
// used from the pMBStr buffer (single byte or double byte) is stored in
|
|
// the mbIncr parameter.
|
|
//
|
|
// Once the character has been translated, it checks to be sure the
|
|
// character was valid. If not, it sets last error and return 0 characters
|
|
// written.
|
|
//
|
|
// NOTE: This macro may return if an error is encountered.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 09-01-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define GET_WC_MULTI_ERR( pHashN, \
|
|
pMBTbl, \
|
|
pMBStr, \
|
|
pEndMBStr, \
|
|
pWCStr, \
|
|
pEndWCStr, \
|
|
mbIncr ) \
|
|
{ \
|
|
WORD Offset; /* offset to DBCS table for range */ \
|
|
\
|
|
\
|
|
if (Offset = CHECK_DBCS_LEAD_BYTE(pHashN->pDBCSOffsets, *pMBStr)) \
|
|
{ \
|
|
/* \
|
|
* DBCS Lead Byte. Make sure there is a trail byte with the \
|
|
* lead byte. \
|
|
*/ \
|
|
if ((pMBStr + 1 == pEndMBStr) || (*(pMBStr + 1) == 0)) \
|
|
{ \
|
|
/* \
|
|
* There is no trail byte with the lead byte. Return error. \
|
|
*/ \
|
|
SetLastError(ERROR_NO_UNICODE_TRANSLATION); \
|
|
return (0); \
|
|
} \
|
|
\
|
|
/* \
|
|
* Fill in the wide character translation from the double \
|
|
* byte character table. \
|
|
*/ \
|
|
*pWCStr = (pHashN->pDBCSOffsets + Offset)[*(pMBStr + 1)]; \
|
|
mbIncr = 2; \
|
|
\
|
|
/* \
|
|
* Make sure an invalid character was not translated to \
|
|
* the default char. Return an error if invalid. \
|
|
*/ \
|
|
CHECK_ERROR_WC_MULTI( pHashN, \
|
|
*pWCStr, \
|
|
*pMBStr, \
|
|
*(pMBStr + 1) ); \
|
|
} \
|
|
else \
|
|
{ \
|
|
/* \
|
|
* Not DBCS Lead Byte. Fill in the wide character translation \
|
|
* from the single byte character table. \
|
|
*/ \
|
|
*pWCStr = pMBTbl[*pMBStr]; \
|
|
mbIncr = 1; \
|
|
\
|
|
/* \
|
|
* Make sure an invalid character was not translated to \
|
|
* the default char. Return an error if invalid. \
|
|
*/ \
|
|
CHECK_ERROR_WC_SINGLE( pHashN, \
|
|
*pWCStr, \
|
|
*pMBStr ); \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GET_WC_MULTI_ERR_SPECIAL
|
|
//
|
|
// Fills in pWCStr with the wide character(s) for the corresponding multibyte
|
|
// character from the appropriate translation table. The number of bytes
|
|
// used from the pMBStr buffer (single byte or double byte) is stored in
|
|
// the mbIncr parameter.
|
|
//
|
|
// Once the character has been translated, it checks to be sure the
|
|
// character was valid. If not, it fills in 0xffff.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 08-21-95 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define GET_WC_MULTI_ERR_SPECIAL( pHashN, \
|
|
pMBTbl, \
|
|
pMBStr, \
|
|
pEndMBStr, \
|
|
pWCStr, \
|
|
pEndWCStr, \
|
|
mbIncr ) \
|
|
{ \
|
|
WORD Offset; /* offset to DBCS table for range */ \
|
|
\
|
|
\
|
|
if (Offset = CHECK_DBCS_LEAD_BYTE(pHashN->pDBCSOffsets, *pMBStr)) \
|
|
{ \
|
|
/* \
|
|
* DBCS Lead Byte. Make sure there is a trail byte with the \
|
|
* lead byte. \
|
|
*/ \
|
|
if ((pMBStr + 1 == pEndMBStr) || (*(pMBStr + 1) == 0)) \
|
|
{ \
|
|
/* \
|
|
* There is no trail byte with the lead byte. The lead byte \
|
|
* is the LAST character in the string. Translate to 0xffff. \
|
|
*/ \
|
|
*pWCStr = (WCHAR)0xffff; \
|
|
mbIncr = 1; \
|
|
} \
|
|
else \
|
|
{ \
|
|
/* \
|
|
* Fill in the wide character translation from the double \
|
|
* byte character table. \
|
|
*/ \
|
|
*pWCStr = (pHashN->pDBCSOffsets + Offset)[*(pMBStr + 1)]; \
|
|
mbIncr = 2; \
|
|
\
|
|
/* \
|
|
* Make sure an invalid character was not translated to \
|
|
* the default char. Translate to 0xffff if invalid. \
|
|
*/ \
|
|
CHECK_ERROR_WC_MULTI_SPECIAL( pHashN, \
|
|
pWCStr, \
|
|
*pMBStr, \
|
|
*(pMBStr + 1) ); \
|
|
} \
|
|
} \
|
|
else \
|
|
{ \
|
|
/* \
|
|
* Not DBCS Lead Byte. Fill in the wide character translation \
|
|
* from the single byte character table. \
|
|
* Make sure an invalid character was not translated to \
|
|
* the default char. Return an error if invalid. \
|
|
*/ \
|
|
GET_WC_SINGLE_SPECIAL( pHashN, \
|
|
pMBTbl, \
|
|
pMBStr, \
|
|
pWCStr ); \
|
|
mbIncr = 1; \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// COPY_MB_CHAR
|
|
//
|
|
// Copies a multibyte character to the given string buffer. If the
|
|
// high byte of the multibyte word is zero, then it is a single byte
|
|
// character and the number of characters written (returned) is 1.
|
|
// Otherwise, it is a double byte character and the number of characters
|
|
// written (returned) is 2.
|
|
//
|
|
// NumByte will be 0 if the buffer is too small for the translation.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define COPY_MB_CHAR( mbChar, \
|
|
pMBStr, \
|
|
NumByte, \
|
|
fOnlyOne ) \
|
|
{ \
|
|
if (HIBYTE(mbChar)) \
|
|
{ \
|
|
/* \
|
|
* Make sure there is enough room in the buffer for both bytes. \
|
|
*/ \
|
|
if (fOnlyOne) \
|
|
{ \
|
|
NumByte = 0; \
|
|
} \
|
|
else \
|
|
{ \
|
|
/* \
|
|
* High Byte is NOT zero, so it's a DOUBLE byte char. \
|
|
* Return 2 characters written. \
|
|
*/ \
|
|
*pMBStr = HIBYTE(mbChar); \
|
|
*(pMBStr + 1) = LOBYTE(mbChar); \
|
|
NumByte = 2; \
|
|
} \
|
|
} \
|
|
else \
|
|
{ \
|
|
/* \
|
|
* High Byte IS zero, so it's a SINGLE byte char. \
|
|
* Return 1 character written. \
|
|
*/ \
|
|
*pMBStr = LOBYTE(mbChar); \
|
|
NumByte = 1; \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GET_SB
|
|
//
|
|
// Fills in pMBStr with the single byte character for the corresponding
|
|
// wide character from the appropriate translation table.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define GET_SB( pWC, \
|
|
wChar, \
|
|
pMBStr ) \
|
|
{ \
|
|
*pMBStr = ((BYTE *)(pWC))[wChar]; \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GET_MB
|
|
//
|
|
// Fills in pMBStr with the multi byte character for the corresponding
|
|
// wide character from the appropriate translation table.
|
|
//
|
|
// mbCount will be 0 if the buffer is too small for the translation.
|
|
//
|
|
// Broken Down Version:
|
|
// --------------------
|
|
// mbChar = ((WORD *)(pHashN->pWC))[wChar];
|
|
// COPY_MB_CHAR(mbChar, pMBStr, mbCount);
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define GET_MB( pWC, \
|
|
wChar, \
|
|
pMBStr, \
|
|
mbCount, \
|
|
fOnlyOne ) \
|
|
{ \
|
|
COPY_MB_CHAR( ((WORD *)(pWC))[wChar], \
|
|
pMBStr, \
|
|
mbCount, \
|
|
fOnlyOne ); \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// ELIMINATE_BEST_FIT_SB
|
|
//
|
|
// Checks to see if a single byte Best Fit character was used. If so,
|
|
// it replaces it with a single byte default character.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define ELIMINATE_BEST_FIT_SB( pHashN, \
|
|
wChar, \
|
|
pMBStr ) \
|
|
{ \
|
|
if ((pHashN->pMBTbl)[*pMBStr] != wChar) \
|
|
{ \
|
|
*pMBStr = LOBYTE(pHashN->pCPInfo->wDefaultChar); \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// ELIMINATE_BEST_FIT_MB
|
|
//
|
|
// Checks to see if a multi byte Best Fit character was used. If so,
|
|
// it replaces it with a multi byte default character.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define ELIMINATE_BEST_FIT_MB( pHashN, \
|
|
wChar, \
|
|
pMBStr, \
|
|
mbCount, \
|
|
fOnlyOne ) \
|
|
{ \
|
|
WORD Offset; \
|
|
WORD wDefault; \
|
|
\
|
|
if (((mbCount == 1) && ((pHashN->pMBTbl)[*pMBStr] != wChar)) || \
|
|
((mbCount == 2) && \
|
|
(Offset = CHECK_DBCS_LEAD_BYTE(pHashN->pDBCSOffsets, *pMBStr)) && \
|
|
(((pHashN->pDBCSOffsets + Offset)[*(pMBStr + 1)]) != wChar))) \
|
|
{ \
|
|
wDefault = pHashN->pCPInfo->wDefaultChar; \
|
|
if (HIBYTE(wDefault)) \
|
|
{ \
|
|
if (fOnlyOne) \
|
|
{ \
|
|
mbCount = 0; \
|
|
} \
|
|
else \
|
|
{ \
|
|
*pMBStr = HIBYTE(wDefault); \
|
|
*(pMBStr + 1) = LOBYTE(wDefault); \
|
|
mbCount = 2; \
|
|
} \
|
|
} \
|
|
else \
|
|
{ \
|
|
*pMBStr = LOBYTE(wDefault); \
|
|
mbCount = 1; \
|
|
} \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GET_DEFAULT_WORD
|
|
//
|
|
// Takes a pointer to a character string (either one or two characters),
|
|
// and converts it to a WORD value. If the character is not DBCS, then it
|
|
// zero extends the high byte.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define GET_DEFAULT_WORD(pOff, pDefault) \
|
|
(CHECK_DBCS_LEAD_BYTE(pOff, *pDefault) \
|
|
? MAKEWORD(*(pDefault + 1), *pDefault) \
|
|
: MAKEWORD(*pDefault, 0))
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// DEFAULT_CHAR_CHECK_SB
|
|
//
|
|
// Checks to see if the default character is used. If it is, it sets
|
|
// pUsedDef to TRUE (if non-null). If the user specified a default, then
|
|
// the user's default character is used.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define DEFAULT_CHAR_CHECK_SB( pHashN, \
|
|
wch, \
|
|
pMBStr, \
|
|
wDefChar, \
|
|
pUsedDef ) \
|
|
{ \
|
|
WORD wSysDefChar = pHashN->pCPInfo->wDefaultChar; \
|
|
\
|
|
\
|
|
/* \
|
|
* Check for default character being used. \
|
|
*/ \
|
|
if ((*pMBStr == (BYTE)wSysDefChar) && \
|
|
(wch != pHashN->pCPInfo->wTransDefaultChar)) \
|
|
{ \
|
|
/* \
|
|
* Default was used. Set the pUsedDef parameter to TRUE. \
|
|
*/ \
|
|
*pUsedDef = TRUE; \
|
|
\
|
|
/* \
|
|
* If the user specified a different default character than \
|
|
* the system default, use that character instead. \
|
|
*/ \
|
|
if (wSysDefChar != wDefChar) \
|
|
{ \
|
|
*pMBStr = LOBYTE(wDefChar); \
|
|
} \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// DEFAULT_CHAR_CHECK_MB
|
|
//
|
|
// Checks to see if the default character is used. If it is, it sets
|
|
// pUsedDef to TRUE (if non-null). If the user specified a default, then
|
|
// the user's default character is used. The number of bytes written to
|
|
// the buffer is returned.
|
|
//
|
|
// NumByte will be -1 if the buffer is too small for the translation.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define DEFAULT_CHAR_CHECK_MB( pHashN, \
|
|
wch, \
|
|
pMBStr, \
|
|
wDefChar, \
|
|
pUsedDef, \
|
|
NumByte, \
|
|
fOnlyOne ) \
|
|
{ \
|
|
WORD wSysDefChar = pHashN->pCPInfo->wDefaultChar; \
|
|
\
|
|
\
|
|
/* \
|
|
* Set NumByte to zero for return (zero bytes written). \
|
|
*/ \
|
|
NumByte = 0; \
|
|
\
|
|
/* \
|
|
* Check for default character being used. \
|
|
*/ \
|
|
if ((*pMBStr == (BYTE)wSysDefChar) && \
|
|
(wch != pHashN->pCPInfo->wTransDefaultChar)) \
|
|
{ \
|
|
/* \
|
|
* Default was used. Set the pUsedDef parameter to TRUE. \
|
|
*/ \
|
|
*pUsedDef = TRUE; \
|
|
\
|
|
/* \
|
|
* If the user specified a different default character than \
|
|
* the system default, use that character instead. \
|
|
*/ \
|
|
if (wSysDefChar != wDefChar) \
|
|
{ \
|
|
COPY_MB_CHAR( wDefChar, \
|
|
pMBStr, \
|
|
NumByte, \
|
|
fOnlyOne ); \
|
|
if (NumByte == 0) \
|
|
{ \
|
|
NumByte = -1; \
|
|
} \
|
|
} \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GET_WC_TRANSLATION_SB
|
|
//
|
|
// Gets the 1:1 translation of a given wide character. It fills in the
|
|
// string pointer with the single byte character.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define GET_WC_TRANSLATION_SB( pHashN, \
|
|
wch, \
|
|
pMBStr, \
|
|
wDefault, \
|
|
pUsedDef, \
|
|
dwFlags ) \
|
|
{ \
|
|
GET_SB( pHashN->pWC, \
|
|
wch, \
|
|
pMBStr ); \
|
|
if (dwFlags & WC_NO_BEST_FIT_CHARS) \
|
|
{ \
|
|
ELIMINATE_BEST_FIT_SB( pHashN, \
|
|
wch, \
|
|
pMBStr ); \
|
|
} \
|
|
DEFAULT_CHAR_CHECK_SB( pHashN, \
|
|
wch, \
|
|
pMBStr, \
|
|
wDefault, \
|
|
pUsedDef ); \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GET_WC_TRANSLATION_MB
|
|
//
|
|
// Gets the 1:1 translation of a given wide character. It fills in the
|
|
// appropriate number of characters for the multibyte character and then
|
|
// returns the number of characters written to the multibyte string.
|
|
//
|
|
// mbCnt will be 0 if the buffer is too small for the translation.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define GET_WC_TRANSLATION_MB( pHashN, \
|
|
wch, \
|
|
pMBStr, \
|
|
wDefault, \
|
|
pUsedDef, \
|
|
mbCnt, \
|
|
fOnlyOne, \
|
|
dwFlags ) \
|
|
{ \
|
|
int mbCnt2; /* number of characters written */ \
|
|
\
|
|
\
|
|
GET_MB( pHashN->pWC, \
|
|
wch, \
|
|
pMBStr, \
|
|
mbCnt, \
|
|
fOnlyOne ); \
|
|
if (dwFlags & WC_NO_BEST_FIT_CHARS) \
|
|
{ \
|
|
ELIMINATE_BEST_FIT_MB( pHashN, \
|
|
wch, \
|
|
pMBStr, \
|
|
mbCnt, \
|
|
fOnlyOne ); \
|
|
} \
|
|
if (mbCnt) \
|
|
{ \
|
|
DEFAULT_CHAR_CHECK_MB( pHashN, \
|
|
wch, \
|
|
pMBStr, \
|
|
wDefault, \
|
|
pUsedDef, \
|
|
mbCnt2, \
|
|
fOnlyOne ); \
|
|
if (mbCnt2 == -1) \
|
|
{ \
|
|
mbCnt = 0; \
|
|
} \
|
|
else if (mbCnt2) \
|
|
{ \
|
|
mbCnt = mbCnt2; \
|
|
} \
|
|
} \
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GET_CP_HASH_NODE
|
|
//
|
|
// Sets the code page value (if a special value is passed in) and the
|
|
// hash node pointer. If the code page value is invalid, the pointer
|
|
// to the hash node will be set to NULL.
|
|
//
|
|
// DEFINED AS A MACRO.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define GET_CP_HASH_NODE( CodePage, \
|
|
pHashN ) \
|
|
{ \
|
|
PLOC_HASH pHashLoc; \
|
|
\
|
|
\
|
|
/* \
|
|
* Check for the ACP, OEMCP, or MACCP. Fill in the appropriate \
|
|
* value for the code page if one of these values is given. \
|
|
* Otherwise, just get the hash node for the given code page. \
|
|
*/ \
|
|
if (CodePage == gAnsiCodePage) \
|
|
{ \
|
|
pHashN = gpACPHashN; \
|
|
} \
|
|
else if (CodePage == gOemCodePage) \
|
|
{ \
|
|
pHashN = gpOEMCPHashN; \
|
|
} \
|
|
else if (CodePage == CP_ACP) \
|
|
{ \
|
|
CodePage = gAnsiCodePage; \
|
|
pHashN = gpACPHashN; \
|
|
} \
|
|
else if (CodePage == CP_OEMCP) \
|
|
{ \
|
|
CodePage = gOemCodePage; \
|
|
pHashN = gpOEMCPHashN; \
|
|
} \
|
|
else if (CodePage == CP_SYMBOL) \
|
|
{ \
|
|
pHashN = NULL; \
|
|
} \
|
|
else if (CodePage == CP_MACCP) \
|
|
{ \
|
|
CodePage = GetMacCodePage(); \
|
|
pHashN = gpMACCPHashN; \
|
|
} \
|
|
else if (CodePage == CP_THREAD_ACP) \
|
|
{ \
|
|
VALIDATE_LOCALE(NtCurrentTeb()->CurrentLocale, pHashLoc, FALSE); \
|
|
if (pHashLoc != NULL) \
|
|
{ \
|
|
CodePage = pHashLoc->pLocaleFixed->DefaultACP; \
|
|
} \
|
|
if (CodePage == CP_ACP) \
|
|
{ \
|
|
CodePage = gAnsiCodePage; \
|
|
pHashN = gpACPHashN; \
|
|
} \
|
|
else if (CodePage == CP_OEMCP) \
|
|
{ \
|
|
CodePage = gOemCodePage; \
|
|
pHashN = gpOEMCPHashN; \
|
|
} \
|
|
else if (CodePage == CP_MACCP) \
|
|
{ \
|
|
CodePage = GetMacCodePage(); \
|
|
pHashN = gpMACCPHashN; \
|
|
} \
|
|
else \
|
|
{ \
|
|
pHashN = GetCPHashNode(CodePage); \
|
|
} \
|
|
} \
|
|
else \
|
|
{ \
|
|
pHashN = GetCPHashNode(CodePage); \
|
|
} \
|
|
}
|
|
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// API ROUTINES //
|
|
//-------------------------------------------------------------------------//
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// IsValidCodePage
|
|
//
|
|
// Checks that the given code page is a valid one. It does so by querying
|
|
// the registry. If the code page is found, then TRUE is returned.
|
|
// Otherwise, FALSE is returned.
|
|
//
|
|
// 05-31-1991 JulieB Created.
|
|
// 05-31-2002 ShawnSte Make it not force the loading of the code page
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI IsValidCodePage(
|
|
UINT CodePage)
|
|
{
|
|
WCHAR wszFileName[MAX_SMALL_BUF_LEN]; // file name (Actually l2 chars is max: c_nlsXXXXX.nls\0
|
|
WCHAR wszFilePath[MAX_PATH_LEN]; // ptr to full path
|
|
|
|
//
|
|
// Do not allow special code page values to be valid here.
|
|
// (CP_ACP, CP_OEMCP, CP_MACCP, CP_THREAD_ACP, CP_SYMBOL are invalid)
|
|
//
|
|
|
|
//
|
|
// Do the quick check for the code page value equal to either
|
|
// the Ansi code page value or the OEM code page value.
|
|
//
|
|
if ((CodePage == gAnsiCodePage) || (CodePage == gOemCodePage) ||
|
|
(CodePage == CP_UTF7) || (CodePage == CP_UTF8))
|
|
{
|
|
//
|
|
// Return success.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
//
|
|
// Check for other code page values.
|
|
//
|
|
|
|
// If a node already exists, then we're OK
|
|
if (IsCPHashNodeLoaded(CodePage) == TRUE)
|
|
{
|
|
//
|
|
// Return success.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
//
|
|
// Hash node doesn't exist. Have to look in the registry.
|
|
// True if this works, false if it doesn't
|
|
//
|
|
|
|
if (FALSE == GetCPFileNameFromRegistry(CodePage, wszFileName, MAX_SMALL_BUF_LEN))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
// Guess we need a full path
|
|
if((0 == GetSystemDirectoryW(wszFilePath, MAX_PATH_LEN)) ||
|
|
FAILED(StringCchCatW(wszFilePath, MAX_PATH_LEN, L"\\")) ||
|
|
FAILED(StringCchCatW(wszFilePath, MAX_PATH_LEN, wszFileName)))
|
|
{
|
|
// Best we can do.
|
|
return FALSE;
|
|
}
|
|
|
|
if (INVALID_FILE_ATTRIBUTES == GetFileAttributesW(wszFilePath))
|
|
{
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetACP
|
|
//
|
|
// Returns the ANSI code page for the system. If the registry value is
|
|
// not readable, then the chosen default ACP is used (NLS_DEFAULT_ACP).
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
UINT WINAPI GetACP()
|
|
{
|
|
//
|
|
// Return the ACP stored in the cache.
|
|
//
|
|
return (gAnsiCodePage);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// SetCPGlobal
|
|
//
|
|
// Sets the code page global, used by Setup to force the code page into
|
|
// the correct value during GUI mode.
|
|
//
|
|
// 02-15-99 JimSchm Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
UINT
|
|
WINAPI
|
|
SetCPGlobal (
|
|
IN UINT NewAcp
|
|
)
|
|
{
|
|
UINT oldVal;
|
|
|
|
|
|
oldVal = gAnsiCodePage;
|
|
|
|
//
|
|
// Sets the ACP global. This is a private exported routine, not an API.
|
|
//
|
|
gAnsiCodePage = NewAcp;
|
|
return oldVal;
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetOEMCP
|
|
//
|
|
// Returns the OEM code page for the system. If the registry value is
|
|
// not readable, then the chosen default ACP is used (NLS_DEFAULT_OEMCP).
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
UINT WINAPI GetOEMCP()
|
|
{
|
|
//
|
|
// Return the OEMCP stored in the cache.
|
|
//
|
|
return (gOemCodePage);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetCPInfo
|
|
//
|
|
// Returns information about a given code page.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI GetCPInfo(
|
|
UINT CodePage,
|
|
LPCPINFO lpCPInfo)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
PCP_TABLE pInfo; // ptr to CP information in file
|
|
WORD wDefChar; // default character
|
|
BYTE *pLeadBytes; // ptr to lead byte ranges
|
|
UINT Ctr; // loop counter
|
|
|
|
|
|
//
|
|
// See if it's a special code page value for UTF translations.
|
|
//
|
|
if (CodePage >= NLS_CP_ALGORITHM_RANGE)
|
|
{
|
|
return (UTFCPInfo(CodePage, lpCPInfo, FALSE));
|
|
}
|
|
|
|
//
|
|
// Get the code page value and the appropriate hash node.
|
|
//
|
|
GET_CP_HASH_NODE(CodePage, pHashN);
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - validate code page - get hash node containing translation tables
|
|
// - lpCPInfo is NULL
|
|
//
|
|
if ( (pHashN == NULL) ||
|
|
((pHashN->pCPInfo == NULL) && (pHashN->pfnCPProc == NULL)) ||
|
|
(lpCPInfo == NULL) )
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// See if the given code page is in the DLL range.
|
|
//
|
|
if (pHashN->pfnCPProc)
|
|
{
|
|
//
|
|
// Call the DLL to get the code page information.
|
|
//
|
|
return ( (*(pHashN->pfnCPProc))( CodePage,
|
|
NLS_CP_CPINFO,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
0,
|
|
lpCPInfo ) );
|
|
}
|
|
|
|
//
|
|
// Fill in the CPINFO structure with the appropriate information.
|
|
//
|
|
pInfo = pHashN->pCPInfo;
|
|
|
|
//
|
|
// Get the max char size.
|
|
//
|
|
lpCPInfo->MaxCharSize = (UINT)((WORD)pInfo->MaxCharSize);
|
|
|
|
//
|
|
// Get the default character.
|
|
//
|
|
wDefChar = pInfo->wDefaultChar;
|
|
if (HIBYTE(wDefChar))
|
|
{
|
|
(lpCPInfo->DefaultChar)[0] = HIBYTE(wDefChar);
|
|
(lpCPInfo->DefaultChar)[1] = LOBYTE(wDefChar);
|
|
}
|
|
else
|
|
{
|
|
(lpCPInfo->DefaultChar)[0] = LOBYTE(wDefChar);
|
|
(lpCPInfo->DefaultChar)[1] = (BYTE)0;
|
|
}
|
|
|
|
//
|
|
// Get the leadbytes.
|
|
//
|
|
pLeadBytes = pInfo->LeadByte;
|
|
for (Ctr = 0; Ctr < MAX_LEADBYTES; Ctr++)
|
|
{
|
|
(lpCPInfo->LeadByte)[Ctr] = pLeadBytes[Ctr];
|
|
}
|
|
|
|
//
|
|
// Return success.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetCPInfoExW
|
|
//
|
|
// Returns information about a given code page.
|
|
//
|
|
// 11-15-96 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI GetCPInfoExW(
|
|
UINT CodePage,
|
|
DWORD dwFlags,
|
|
LPCPINFOEXW lpCPInfoEx)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
PCP_TABLE pInfo; // ptr to CP information in file
|
|
WORD wDefChar; // default character
|
|
BYTE *pLeadBytes; // ptr to lead byte ranges
|
|
UINT Ctr; // loop counter
|
|
|
|
|
|
//
|
|
// See if it's a special code page value for UTF translations.
|
|
//
|
|
if (CodePage >= NLS_CP_ALGORITHM_RANGE)
|
|
{
|
|
if (UTFCPInfo(CodePage, (LPCPINFO)lpCPInfoEx, TRUE))
|
|
{
|
|
if (GetStringTableEntry( CodePage,
|
|
0,
|
|
lpCPInfoEx->CodePageName,
|
|
MAX_PATH,
|
|
RC_CODE_PAGE_NAME ) != 0)
|
|
{
|
|
return (TRUE);
|
|
}
|
|
}
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Get the code page value and the appropriate hash node.
|
|
//
|
|
GET_CP_HASH_NODE(CodePage, pHashN);
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - validate code page - get hash node containing translation tables
|
|
// - lpCPInfoEx is NULL
|
|
//
|
|
if ( (pHashN == NULL) ||
|
|
((pHashN->pCPInfo == NULL) && (pHashN->pfnCPProc == NULL)) ||
|
|
(lpCPInfoEx == NULL) )
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Invalid Flags Check:
|
|
// - flags not 0
|
|
//
|
|
if (dwFlags != 0)
|
|
{
|
|
SetLastError(ERROR_INVALID_FLAGS);
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// See if the given code page is in the DLL range.
|
|
//
|
|
if (pHashN->pfnCPProc)
|
|
{
|
|
//
|
|
// Call the DLL to get the code page information.
|
|
//
|
|
if (((*(pHashN->pfnCPProc))( CodePage,
|
|
NLS_CP_CPINFOEX,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
0,
|
|
(LPCPINFO)lpCPInfoEx )) == TRUE)
|
|
{
|
|
return (TRUE);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// See if the CPINFO will succeed. If so, then add the
|
|
// default CPINFOEX info to the structure.
|
|
//
|
|
if (((*(pHashN->pfnCPProc))( CodePage,
|
|
NLS_CP_CPINFO,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
0,
|
|
(LPCPINFO)lpCPInfoEx )) == TRUE)
|
|
{
|
|
//
|
|
// Fill in the Ex version info.
|
|
//
|
|
lpCPInfoEx->UnicodeDefaultChar = L'?';
|
|
lpCPInfoEx->CodePage = CodePage;
|
|
GetStringTableEntry( CodePage,
|
|
0,
|
|
lpCPInfoEx->CodePageName,
|
|
MAX_PATH,
|
|
RC_CODE_PAGE_NAME );
|
|
|
|
SetLastError(NO_ERROR);
|
|
return (TRUE);
|
|
}
|
|
|
|
return (FALSE);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Fill in the CPINFO structure with the appropriate information.
|
|
//
|
|
pInfo = pHashN->pCPInfo;
|
|
|
|
//
|
|
// Get the max char size.
|
|
//
|
|
lpCPInfoEx->MaxCharSize = (UINT)((WORD)pInfo->MaxCharSize);
|
|
|
|
//
|
|
// Get the default character.
|
|
//
|
|
wDefChar = pInfo->wDefaultChar;
|
|
if (HIBYTE(wDefChar))
|
|
{
|
|
(lpCPInfoEx->DefaultChar)[0] = HIBYTE(wDefChar);
|
|
(lpCPInfoEx->DefaultChar)[1] = LOBYTE(wDefChar);
|
|
}
|
|
else
|
|
{
|
|
(lpCPInfoEx->DefaultChar)[0] = LOBYTE(wDefChar);
|
|
(lpCPInfoEx->DefaultChar)[1] = (BYTE)0;
|
|
}
|
|
|
|
//
|
|
// Get the leadbytes.
|
|
//
|
|
pLeadBytes = pInfo->LeadByte;
|
|
for (Ctr = 0; Ctr < MAX_LEADBYTES; Ctr++)
|
|
{
|
|
(lpCPInfoEx->LeadByte)[Ctr] = pLeadBytes[Ctr];
|
|
}
|
|
|
|
//
|
|
// Get the Unicode default character.
|
|
//
|
|
lpCPInfoEx->UnicodeDefaultChar = pInfo->wUniDefaultChar;
|
|
|
|
//
|
|
// Get the code page id.
|
|
//
|
|
lpCPInfoEx->CodePage = CodePage;
|
|
|
|
//
|
|
// Get the code page name.
|
|
//
|
|
if (GetStringTableEntry( CodePage,
|
|
0,
|
|
lpCPInfoEx->CodePageName,
|
|
MAX_PATH,
|
|
RC_CODE_PAGE_NAME ) == 0)
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Return success.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// IsDBCSLeadByte
|
|
//
|
|
// Checks to see if a given character is a DBCS lead byte in the ACP.
|
|
// Returns TRUE if it is, FALSE if it is not.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI IsDBCSLeadByte(
|
|
BYTE TestChar)
|
|
{
|
|
//
|
|
// Get the hash node for the ACP.
|
|
//
|
|
if (gpACPHashN == NULL)
|
|
{
|
|
SetLastError(ERROR_FILE_NOT_FOUND);
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// See if the given character is a DBCS lead byte.
|
|
//
|
|
if (CHECK_DBCS_LEAD_BYTE(gpACPHashN->pDBCSOffsets, TestChar))
|
|
{
|
|
//
|
|
// Return success - IS a DBCS lead byte.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
//
|
|
// Return failure - is NOT a DBCS lead byte.
|
|
//
|
|
return (FALSE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// IsDBCSLeadByteEx
|
|
//
|
|
// Checks to see if a given character is a DBCS lead byte in the given
|
|
// code page. Returns TRUE if it is, FALSE if it is not.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL WINAPI IsDBCSLeadByteEx(
|
|
UINT CodePage,
|
|
BYTE TestChar)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
|
|
//
|
|
// See if it's a special code page value for UTF translations.
|
|
//
|
|
if (CodePage >= NLS_CP_ALGORITHM_RANGE)
|
|
{
|
|
if (CodePage != CP_UTF8 && CodePage != CP_UTF7)
|
|
{
|
|
// NOTE: This condition has to be updated if we have more codepages in
|
|
// the NLS_CP_ALGORITHM_RANGE.
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
}
|
|
//
|
|
// Return that it's not a DBCS leadbyte.
|
|
//
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Get the code page value and the appropriate hash node.
|
|
//
|
|
GET_CP_HASH_NODE(CodePage, pHashN);
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - validate code page
|
|
//
|
|
if (pHashN == NULL)
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (FALSE);
|
|
}
|
|
|
|
|
|
//
|
|
// See if the given character is a DBCS lead byte.
|
|
//
|
|
if (CHECK_DBCS_LEAD_BYTE(pHashN->pDBCSOffsets, TestChar))
|
|
{
|
|
//
|
|
// Return success - IS a DBCS lead byte.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
//
|
|
// Return failure - is NOT a DBCS lead byte.
|
|
//
|
|
return (FALSE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// MultiByteToWideChar
|
|
//
|
|
// Maps a multibyte character string to its wide character string
|
|
// counterpart.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
// 09-01-93 JulieB Add support for MB_ERR_INVALID_CHARS flag.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int WINAPI MultiByteToWideChar(
|
|
UINT CodePage,
|
|
DWORD dwFlags,
|
|
LPCSTR lpMultiByteStr,
|
|
int cbMultiByte,
|
|
LPWSTR lpWideCharStr,
|
|
int cchWideChar)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
register LPBYTE pMBStr; // ptr to search through MB string
|
|
register LPWSTR pWCStr; // ptr to search through WC string
|
|
LPBYTE pEndMBStr; // ptr to end of MB search string
|
|
LPWSTR pEndWCStr; // ptr to end of WC string buffer
|
|
int wcIncr; // amount to increment pWCStr
|
|
int mbIncr; // amount to increment pMBStr
|
|
int wcCount = 0; // count of wide chars written
|
|
int CompSet; // if MB_COMPOSITE flag is set
|
|
PMB_TABLE pMBTbl; // ptr to correct MB table (MB or GLYPH)
|
|
int ctr; // loop counter
|
|
|
|
|
|
//
|
|
// See if it's a special code page value for UTF translations.
|
|
//
|
|
if (CodePage >= NLS_CP_ALGORITHM_RANGE)
|
|
{
|
|
return (UTFToUnicode( CodePage,
|
|
dwFlags,
|
|
lpMultiByteStr,
|
|
cbMultiByte,
|
|
lpWideCharStr,
|
|
cchWideChar ));
|
|
}
|
|
|
|
//
|
|
// Get the code page value and the appropriate hash node.
|
|
//
|
|
GET_CP_HASH_NODE(CodePage, pHashN);
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - length of MB string is 0
|
|
// - wide char buffer size is negative
|
|
// - MB string is NULL
|
|
// - length of WC string is NOT zero AND
|
|
// (WC string is NULL OR src and dest pointers equal)
|
|
//
|
|
if ( (cbMultiByte == 0) || (cchWideChar < 0) ||
|
|
(lpMultiByteStr == NULL) ||
|
|
((cchWideChar != 0) &&
|
|
((lpWideCharStr == NULL) ||
|
|
(lpMultiByteStr == (LPSTR)lpWideCharStr))) )
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// If cbMultiByte is -1, then the string is null terminated and we
|
|
// need to get the length of the string. Add one to the length to
|
|
// include the null termination. (This will always be at least 1.)
|
|
//
|
|
if (cbMultiByte <= -1)
|
|
{
|
|
cbMultiByte = strlen(lpMultiByteStr) + 1;
|
|
}
|
|
|
|
//
|
|
// Check for valid code page.
|
|
//
|
|
if (pHashN == NULL)
|
|
{
|
|
//
|
|
// Special case the CP_SYMBOL code page.
|
|
//
|
|
if ((CodePage == CP_SYMBOL) && (dwFlags == 0))
|
|
{
|
|
//
|
|
// If the caller just wants the size of the buffer needed
|
|
// to do this translation, return the size of the MB string.
|
|
//
|
|
if (cchWideChar == 0)
|
|
{
|
|
return (cbMultiByte);
|
|
}
|
|
|
|
//
|
|
// Make sure the buffer is large enough.
|
|
//
|
|
if (cchWideChar < cbMultiByte)
|
|
{
|
|
SetLastError(ERROR_INSUFFICIENT_BUFFER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Translate SB char xx to Unicode f0xx.
|
|
// 0x00->0x1f map to 0x0000->0x001f
|
|
// 0x20->0xff map to 0xf020->0xf0ff
|
|
//
|
|
for (ctr = 0; ctr < cbMultiByte; ctr++)
|
|
{
|
|
lpWideCharStr[ctr] = ((BYTE)(lpMultiByteStr[ctr]) < 0x20)
|
|
? (WCHAR)lpMultiByteStr[ctr]
|
|
: MAKEWORD(lpMultiByteStr[ctr], 0xf0);
|
|
}
|
|
return (cbMultiByte);
|
|
}
|
|
else
|
|
{
|
|
SetLastError(((CodePage == CP_SYMBOL) && (dwFlags != 0))
|
|
? ERROR_INVALID_FLAGS
|
|
: ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
//
|
|
// See if the given code page is in the DLL range.
|
|
//
|
|
if (pHashN->pfnCPProc)
|
|
{
|
|
//
|
|
// Invalid Flags Check:
|
|
// - flags not 0
|
|
//
|
|
if (dwFlags != 0)
|
|
{
|
|
SetLastError(ERROR_INVALID_FLAGS);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Call the DLL to do the translation.
|
|
//
|
|
return ( (*(pHashN->pfnCPProc))( CodePage,
|
|
NLS_CP_MBTOWC,
|
|
(LPSTR)lpMultiByteStr,
|
|
cbMultiByte,
|
|
(LPWSTR)lpWideCharStr,
|
|
cchWideChar,
|
|
NULL ) );
|
|
}
|
|
|
|
//
|
|
// Invalid Flags Check:
|
|
// - flags other than valid ones
|
|
// - composite and precomposed both set
|
|
//
|
|
if ( (dwFlags & MB_INVALID_FLAG) ||
|
|
((dwFlags & MB_PRECOMPOSED) && (dwFlags & MB_COMPOSITE)) )
|
|
{
|
|
SetLastError(ERROR_INVALID_FLAGS);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Initialize multibyte character loop pointers.
|
|
//
|
|
pMBStr = (LPBYTE)lpMultiByteStr;
|
|
pEndMBStr = pMBStr + cbMultiByte;
|
|
CompSet = dwFlags & MB_COMPOSITE;
|
|
|
|
//
|
|
// Get the correct MB table (MB or GLYPH).
|
|
//
|
|
if ((dwFlags & MB_USEGLYPHCHARS) && (pHashN->pGlyphTbl != NULL))
|
|
{
|
|
pMBTbl = pHashN->pGlyphTbl;
|
|
}
|
|
else
|
|
{
|
|
pMBTbl = pHashN->pMBTbl;
|
|
}
|
|
|
|
//
|
|
// If cchWideChar is 0, then we can't use lpWideCharStr. In this
|
|
// case, we simply want to count the number of characters that would
|
|
// be written to the buffer.
|
|
//
|
|
if (cchWideChar == 0)
|
|
{
|
|
WCHAR pTempStr[MAX_COMPOSITE]; // tmp buffer - max for composite
|
|
|
|
//
|
|
// For each multibyte char, translate it to its corresponding
|
|
// wide char and increment the wide character count.
|
|
//
|
|
pEndWCStr = pTempStr + MAX_COMPOSITE;
|
|
if (IS_SBCS_CP(pHashN))
|
|
{
|
|
//
|
|
// Single Byte Character Code Page.
|
|
//
|
|
if (CompSet)
|
|
{
|
|
//
|
|
// Composite flag is set.
|
|
//
|
|
if (dwFlags & MB_ERR_INVALID_CHARS)
|
|
{
|
|
//
|
|
// Error check flag is set.
|
|
//
|
|
while (pMBStr < pEndMBStr)
|
|
{
|
|
if (!(wcIncr = GetWCCompSBErr( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pTempStr,
|
|
pEndWCStr )))
|
|
{
|
|
return (0);
|
|
}
|
|
pMBStr++;
|
|
wcCount += wcIncr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Error check flag is NOT set.
|
|
//
|
|
while (pMBStr < pEndMBStr)
|
|
{
|
|
wcCount += GetWCCompSB( pMBTbl,
|
|
pMBStr,
|
|
pTempStr,
|
|
pEndWCStr );
|
|
pMBStr++;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Composite flag is NOT set.
|
|
//
|
|
if (dwFlags & MB_ERR_INVALID_CHARS)
|
|
{
|
|
//
|
|
// Error check flag is set.
|
|
//
|
|
wcCount = (int)(pEndMBStr - pMBStr);
|
|
while (pMBStr < pEndMBStr)
|
|
{
|
|
GET_WC_SINGLE( pMBTbl,
|
|
pMBStr,
|
|
pTempStr );
|
|
CHECK_ERROR_WC_SINGLE( pHashN,
|
|
*pTempStr,
|
|
*pMBStr );
|
|
pMBStr++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Error check flag is NOT set.
|
|
//
|
|
// Just return the size of the MB string, since
|
|
// it's a 1:1 translation.
|
|
//
|
|
wcCount = (int)(pEndMBStr - pMBStr);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Multi Byte Character Code Page.
|
|
//
|
|
if (CompSet)
|
|
{
|
|
//
|
|
// Composite flag is set.
|
|
//
|
|
if (dwFlags & MB_ERR_INVALID_CHARS)
|
|
{
|
|
//
|
|
// Error check flag is set.
|
|
//
|
|
while (pMBStr < pEndMBStr)
|
|
{
|
|
if (!(wcIncr = GetWCCompMBErr( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pTempStr,
|
|
pEndWCStr,
|
|
&mbIncr )))
|
|
{
|
|
return (0);
|
|
}
|
|
pMBStr += mbIncr;
|
|
wcCount += wcIncr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Error check flag is NOT set.
|
|
//
|
|
while (pMBStr < pEndMBStr)
|
|
{
|
|
wcCount += GetWCCompMB( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pTempStr,
|
|
pEndWCStr,
|
|
&mbIncr );
|
|
pMBStr += mbIncr;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Composite flag is NOT set.
|
|
//
|
|
if (dwFlags & MB_ERR_INVALID_CHARS)
|
|
{
|
|
//
|
|
// Error check flag is set.
|
|
//
|
|
while (pMBStr < pEndMBStr)
|
|
{
|
|
GET_WC_MULTI_ERR( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pTempStr,
|
|
pEndWCStr,
|
|
mbIncr );
|
|
pMBStr += mbIncr;
|
|
wcCount++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Error check flag is NOT set.
|
|
//
|
|
while (pMBStr < pEndMBStr)
|
|
{
|
|
GET_WC_MULTI( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pTempStr,
|
|
pEndWCStr,
|
|
mbIncr );
|
|
pMBStr += mbIncr;
|
|
wcCount++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Initialize wide character loop pointers.
|
|
//
|
|
pWCStr = lpWideCharStr;
|
|
pEndWCStr = pWCStr + cchWideChar;
|
|
|
|
//
|
|
// For each multibyte char, translate it to its corresponding
|
|
// wide char, store it in lpWideCharStr, and increment the wide
|
|
// character count.
|
|
//
|
|
if (IS_SBCS_CP(pHashN))
|
|
{
|
|
//
|
|
// Single Byte Character Code Page.
|
|
//
|
|
if (CompSet)
|
|
{
|
|
//
|
|
// Composite flag is set.
|
|
//
|
|
if (dwFlags & MB_ERR_INVALID_CHARS)
|
|
{
|
|
//
|
|
// Error check flag is set.
|
|
//
|
|
while ((pMBStr < pEndMBStr) && (pWCStr < pEndWCStr))
|
|
{
|
|
if (!(wcIncr = GetWCCompSBErr( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pWCStr,
|
|
pEndWCStr )))
|
|
{
|
|
return (0);
|
|
}
|
|
pMBStr++;
|
|
pWCStr += wcIncr;
|
|
}
|
|
wcCount = (int)(pWCStr - lpWideCharStr);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Error check flag is NOT set.
|
|
//
|
|
while ((pMBStr < pEndMBStr) && (pWCStr < pEndWCStr))
|
|
{
|
|
pWCStr += GetWCCompSB( pMBTbl,
|
|
pMBStr,
|
|
pWCStr,
|
|
pEndWCStr );
|
|
pMBStr++;
|
|
}
|
|
wcCount = (int)(pWCStr - lpWideCharStr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Composite flag is NOT set.
|
|
//
|
|
if (dwFlags & MB_ERR_INVALID_CHARS)
|
|
{
|
|
//
|
|
// Error check flag is set.
|
|
//
|
|
wcCount = (int)(pEndMBStr - pMBStr);
|
|
if ((pEndWCStr - pWCStr) < wcCount)
|
|
{
|
|
wcCount = (int)(pEndWCStr - pWCStr);
|
|
}
|
|
for (ctr = wcCount; ctr > 0; ctr--)
|
|
{
|
|
GET_WC_SINGLE( pMBTbl,
|
|
pMBStr,
|
|
pWCStr );
|
|
CHECK_ERROR_WC_SINGLE( pHashN,
|
|
*pWCStr,
|
|
*pMBStr );
|
|
pMBStr++;
|
|
pWCStr++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Error check flag is NOT set.
|
|
//
|
|
wcCount = (int)(pEndMBStr - pMBStr);
|
|
if ((pEndWCStr - pWCStr) < wcCount)
|
|
{
|
|
wcCount = (int)(pEndWCStr - pWCStr);
|
|
}
|
|
for (ctr = wcCount; ctr > 0; ctr--)
|
|
{
|
|
GET_WC_SINGLE( pMBTbl,
|
|
pMBStr,
|
|
pWCStr );
|
|
pMBStr++;
|
|
pWCStr++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Multi Byte Character Code Page.
|
|
//
|
|
if (CompSet)
|
|
{
|
|
//
|
|
// Composite flag is set.
|
|
//
|
|
if (dwFlags & MB_ERR_INVALID_CHARS)
|
|
{
|
|
//
|
|
// Error check flag is set.
|
|
//
|
|
while ((pMBStr < pEndMBStr) && (pWCStr < pEndWCStr))
|
|
{
|
|
if (!(wcIncr = GetWCCompMBErr( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pWCStr,
|
|
pEndWCStr,
|
|
&mbIncr )))
|
|
{
|
|
return (0);
|
|
}
|
|
pMBStr += mbIncr;
|
|
pWCStr += wcIncr;
|
|
}
|
|
wcCount = (int)(pWCStr - lpWideCharStr);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Error check flag is NOT set.
|
|
//
|
|
while ((pMBStr < pEndMBStr) && (pWCStr < pEndWCStr))
|
|
{
|
|
pWCStr += GetWCCompMB( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pWCStr,
|
|
pEndWCStr,
|
|
&mbIncr );
|
|
pMBStr += mbIncr;
|
|
}
|
|
wcCount = (int)(pWCStr - lpWideCharStr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Composite flag is NOT set.
|
|
//
|
|
if (dwFlags & MB_ERR_INVALID_CHARS)
|
|
{
|
|
//
|
|
// Error check flag is set.
|
|
//
|
|
while ((pMBStr < pEndMBStr) && (pWCStr < pEndWCStr))
|
|
{
|
|
GET_WC_MULTI_ERR( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pWCStr,
|
|
pEndWCStr,
|
|
mbIncr );
|
|
pMBStr += mbIncr;
|
|
pWCStr++;
|
|
}
|
|
wcCount = (int)(pWCStr - lpWideCharStr);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Error check flag is NOT set.
|
|
//
|
|
while ((pMBStr < pEndMBStr) && (pWCStr < pEndWCStr))
|
|
{
|
|
GET_WC_MULTI( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pWCStr,
|
|
pEndWCStr,
|
|
mbIncr );
|
|
pMBStr += mbIncr;
|
|
pWCStr++;
|
|
}
|
|
wcCount = (int)(pWCStr - lpWideCharStr);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Make sure wide character buffer was large enough.
|
|
//
|
|
if (pMBStr < pEndMBStr)
|
|
{
|
|
SetLastError(ERROR_INSUFFICIENT_BUFFER);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Return the number of characters written (or that would have
|
|
// been written) to the buffer.
|
|
//
|
|
return (wcCount);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// WideCharToMultiByte
|
|
//
|
|
// Maps a wide character string to its multibyte character string
|
|
// counterpart.
|
|
//
|
|
// NOTE: Most significant bit of dwFlags parameter is used by this routine
|
|
// to indicate that the caller only wants the count of the number of
|
|
// characters written, not the string (ie. do not back up in buffer).
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int WINAPI WideCharToMultiByte(
|
|
UINT CodePage,
|
|
DWORD dwFlags,
|
|
LPCWSTR lpWideCharStr,
|
|
int cchWideChar,
|
|
LPSTR lpMultiByteStr,
|
|
int cbMultiByte,
|
|
LPCSTR lpDefaultChar,
|
|
LPBOOL lpUsedDefaultChar)
|
|
{
|
|
PCP_HASH pHashN; // ptr to CP hash node
|
|
LPWSTR pWCStr; // ptr to search through WC string
|
|
LPWSTR pEndWCStr; // ptr to end of WC string buffer
|
|
WORD wDefault = 0; // default character as a word
|
|
int IfNoDefault; // if default check is to be made
|
|
int IfCompositeChk; // if check for composite
|
|
BOOL TmpUsed; // temp storage for default used
|
|
int ctr; // loop counter
|
|
|
|
|
|
//
|
|
// See if it's a special code page value for UTF translations.
|
|
//
|
|
if (CodePage >= NLS_CP_ALGORITHM_RANGE)
|
|
{
|
|
return (UnicodeToUTF( CodePage,
|
|
dwFlags,
|
|
lpWideCharStr,
|
|
cchWideChar,
|
|
lpMultiByteStr,
|
|
cbMultiByte,
|
|
lpDefaultChar,
|
|
lpUsedDefaultChar ));
|
|
}
|
|
|
|
//
|
|
// Get the code page value and the appropriate hash node.
|
|
//
|
|
GET_CP_HASH_NODE(CodePage, pHashN);
|
|
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - length of WC string is 0
|
|
// - multibyte buffer size is negative
|
|
// - WC string is NULL
|
|
// - length of WC string is NOT zero AND
|
|
// (MB string is NULL OR src and dest pointers equal)
|
|
//
|
|
if ( (cchWideChar == 0) || (cbMultiByte < 0) ||
|
|
(lpWideCharStr == NULL) ||
|
|
((cbMultiByte != 0) &&
|
|
((lpMultiByteStr == NULL) ||
|
|
(lpWideCharStr == (LPWSTR)lpMultiByteStr))) )
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// If cchWideChar is -1, then the string is null terminated and we
|
|
// need to get the length of the string. Add one to the length to
|
|
// include the null termination. (This will always be at least 1.)
|
|
//
|
|
if (cchWideChar <= -1)
|
|
{
|
|
cchWideChar = NlsStrLenW(lpWideCharStr) + 1;
|
|
}
|
|
|
|
//
|
|
// Check for valid code page.
|
|
//
|
|
if (pHashN == NULL)
|
|
{
|
|
//
|
|
// Special case the CP_SYMBOL code page.
|
|
//
|
|
if ((CodePage == CP_SYMBOL) && (dwFlags == 0) &&
|
|
(lpDefaultChar == NULL) && (lpUsedDefaultChar == NULL))
|
|
{
|
|
//
|
|
// If the caller just wants the size of the buffer needed
|
|
// to do this translation, return the size of the MB string.
|
|
//
|
|
if (cbMultiByte == 0)
|
|
{
|
|
return (cchWideChar);
|
|
}
|
|
|
|
//
|
|
// Make sure the buffer is large enough.
|
|
//
|
|
if (cbMultiByte < cchWideChar)
|
|
{
|
|
SetLastError(ERROR_INSUFFICIENT_BUFFER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Translate Unicode char f0xx to SB xx.
|
|
// 0x0000->0x001f map to 0x00->0x1f
|
|
// 0xf020->0xf0ff map to 0x20->0xff
|
|
//
|
|
for (ctr = 0; ctr < cchWideChar; ctr++)
|
|
{
|
|
if ((lpWideCharStr[ctr] >= 0x0020) &&
|
|
((lpWideCharStr[ctr] < 0xf020) ||
|
|
(lpWideCharStr[ctr] > 0xf0ff)))
|
|
{
|
|
SetLastError(ERROR_NO_UNICODE_TRANSLATION); \
|
|
return (0);
|
|
}
|
|
lpMultiByteStr[ctr] = (BYTE)lpWideCharStr[ctr];
|
|
}
|
|
return (cchWideChar);
|
|
}
|
|
else
|
|
{
|
|
SetLastError(((CodePage == CP_SYMBOL) && (dwFlags != 0))
|
|
? ERROR_INVALID_FLAGS
|
|
: ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
//
|
|
// See if the given code page is in the DLL range.
|
|
//
|
|
if (pHashN->pfnCPProc)
|
|
{
|
|
//
|
|
// Invalid Parameter Check:
|
|
// - lpDefaultChar not NULL
|
|
// - lpUsedDefaultChar not NULL
|
|
//
|
|
if ((lpDefaultChar != NULL) || (lpUsedDefaultChar != NULL))
|
|
{
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Invalid Flags Check:
|
|
// - flags not 0
|
|
//
|
|
if (dwFlags != 0)
|
|
{
|
|
SetLastError(ERROR_INVALID_FLAGS);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Call the DLL to do the translation.
|
|
//
|
|
return ( (*(pHashN->pfnCPProc))( CodePage,
|
|
NLS_CP_WCTOMB,
|
|
(LPSTR)lpMultiByteStr,
|
|
cbMultiByte,
|
|
(LPWSTR)lpWideCharStr,
|
|
cchWideChar,
|
|
NULL ) );
|
|
}
|
|
|
|
//
|
|
// Invalid Flags Check:
|
|
// - compositechk flag is not set AND any of comp flags are set
|
|
// - flags other than valid ones
|
|
//
|
|
if ( ((!(IfCompositeChk = (dwFlags & WC_COMPOSITECHECK))) &&
|
|
(dwFlags & WC_COMPCHK_FLAGS)) ||
|
|
(dwFlags & WC_INVALID_FLAG) )
|
|
{
|
|
SetLastError(ERROR_INVALID_FLAGS);
|
|
return (0);
|
|
}
|
|
|
|
//
|
|
// Initialize wide character loop pointers.
|
|
//
|
|
pWCStr = (LPWSTR)lpWideCharStr;
|
|
pEndWCStr = pWCStr + cchWideChar;
|
|
|
|
//
|
|
// Set the IfNoDefault parameter to TRUE if both lpDefaultChar and
|
|
// lpUsedDefaultChar are NULL.
|
|
//
|
|
IfNoDefault = ((lpDefaultChar == NULL) && (lpUsedDefaultChar == NULL));
|
|
|
|
//
|
|
// If the composite check flag is NOT set AND both of the default
|
|
// parameters (lpDefaultChar and lpUsedDefaultChar) are null, then
|
|
// do the quick translation.
|
|
//
|
|
if (IfNoDefault && !IfCompositeChk)
|
|
{
|
|
//
|
|
// Translate WC string to MB string, ignoring default chars.
|
|
//
|
|
return (GetMBNoDefault( pHashN,
|
|
pWCStr,
|
|
pEndWCStr,
|
|
(LPBYTE)lpMultiByteStr,
|
|
cbMultiByte,
|
|
dwFlags ));
|
|
}
|
|
|
|
//
|
|
// Set the system default character.
|
|
//
|
|
wDefault = pHashN->pCPInfo->wDefaultChar;
|
|
|
|
//
|
|
// See if the default check is needed.
|
|
//
|
|
if (!IfNoDefault)
|
|
{
|
|
//
|
|
// If lpDefaultChar is NULL, then use the system default.
|
|
// Form a word out of the default character. Single byte
|
|
// characters are zero extended, DBCS characters are as is.
|
|
//
|
|
if (lpDefaultChar != NULL)
|
|
{
|
|
wDefault = GET_DEFAULT_WORD( pHashN->pDBCSOffsets,
|
|
(LPBYTE)lpDefaultChar );
|
|
}
|
|
|
|
//
|
|
// If lpUsedDefaultChar is NULL, then it won't be used later
|
|
// on if a default character is detected. Otherwise, we need
|
|
// to initialize it.
|
|
//
|
|
if (lpUsedDefaultChar == NULL)
|
|
{
|
|
lpUsedDefaultChar = &TmpUsed;
|
|
}
|
|
*lpUsedDefaultChar = FALSE;
|
|
|
|
//
|
|
// Check for "composite check" flag.
|
|
//
|
|
if (!IfCompositeChk)
|
|
{
|
|
//
|
|
// Translate WC string to MB string, checking for the use of the
|
|
// default character.
|
|
//
|
|
return (GetMBDefault( pHashN,
|
|
pWCStr,
|
|
pEndWCStr,
|
|
(LPBYTE)lpMultiByteStr,
|
|
cbMultiByte,
|
|
wDefault,
|
|
lpUsedDefaultChar,
|
|
dwFlags ));
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Translate WC string to MB string, checking for the use of the
|
|
// default character.
|
|
//
|
|
return (GetMBDefaultComp( pHashN,
|
|
pWCStr,
|
|
pEndWCStr,
|
|
(LPBYTE)lpMultiByteStr,
|
|
cbMultiByte,
|
|
wDefault,
|
|
lpUsedDefaultChar,
|
|
dwFlags ));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// The only case left here is that the Composite check
|
|
// flag IS set and the default check flag is NOT set.
|
|
//
|
|
// Translate WC string to MB string, checking for the use of the
|
|
// default character.
|
|
//
|
|
return (GetMBDefaultComp( pHashN,
|
|
pWCStr,
|
|
pEndWCStr,
|
|
(LPBYTE)lpMultiByteStr,
|
|
cbMultiByte,
|
|
wDefault,
|
|
&TmpUsed,
|
|
dwFlags ));
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
//-------------------------------------------------------------------------//
|
|
// INTERNAL ROUTINES //
|
|
//-------------------------------------------------------------------------//
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetWCCompSB
|
|
//
|
|
// Fills in pWCStr with the wide character(s) for the corresponding single
|
|
// byte character from the appropriate translation table and returns the
|
|
// number of wide characters written. This routine should only be called
|
|
// when the precomposed forms need to be translated to composite.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int GetWCCompSB(
|
|
PMB_TABLE pMBTbl,
|
|
LPBYTE pMBStr,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr)
|
|
{
|
|
//
|
|
// Get the single byte to wide character translation.
|
|
//
|
|
GET_WC_SINGLE(pMBTbl, pMBStr, pWCStr);
|
|
|
|
//
|
|
// Fill in the composite form of the character (if one exists)
|
|
// and return the number of wide characters written.
|
|
//
|
|
return (InsertCompositeForm(pWCStr, pEndWCStr));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetWCCompMB
|
|
//
|
|
// Fills in pWCStr with the wide character(s) for the corresponding multibyte
|
|
// character from the appropriate translation table and returns the number
|
|
// of wide characters written. The number of bytes used from the pMBStr
|
|
// buffer (single byte or double byte) is returned in the mbIncr parameter.
|
|
// This routine should only be called when the precomposed forms need to be
|
|
// translated to composite.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int GetWCCompMB(
|
|
PCP_HASH pHashN,
|
|
PMB_TABLE pMBTbl,
|
|
LPBYTE pMBStr,
|
|
LPBYTE pEndMBStr,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr,
|
|
int *pmbIncr)
|
|
{
|
|
//
|
|
// Get the multibyte to wide char translation.
|
|
//
|
|
GET_WC_MULTI( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pWCStr,
|
|
pEndWCStr,
|
|
*pmbIncr );
|
|
|
|
//
|
|
// Fill in the composite form of the character (if one exists)
|
|
// and return the number of wide characters written.
|
|
//
|
|
return (InsertCompositeForm(pWCStr, pEndWCStr));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetWCCompSBErr
|
|
//
|
|
// Fills in pWCStr with the wide character(s) for the corresponding single
|
|
// byte character from the appropriate translation table and returns the
|
|
// number of wide characters written. This routine should only be called
|
|
// when the precomposed forms need to be translated to composite.
|
|
//
|
|
// Checks to be sure an invalid character is not translated to the default
|
|
// character. If so, it sets last error and returns 0 characters written.
|
|
//
|
|
// 09-01-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int GetWCCompSBErr(
|
|
PCP_HASH pHashN,
|
|
PMB_TABLE pMBTbl,
|
|
LPBYTE pMBStr,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr)
|
|
{
|
|
//
|
|
// Get the single byte to wide character translation.
|
|
//
|
|
GET_WC_SINGLE(pMBTbl, pMBStr, pWCStr);
|
|
|
|
//
|
|
// Make sure an invalid character was not translated to the
|
|
// default char. If it was, set last error and return 0
|
|
// characters written.
|
|
//
|
|
CHECK_ERROR_WC_SINGLE(pHashN, *pWCStr, *pMBStr);
|
|
|
|
//
|
|
// Fill in the composite form of the character (if one exists)
|
|
// and return the number of wide characters written.
|
|
//
|
|
return (InsertCompositeForm(pWCStr, pEndWCStr));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetWCCompMBErr
|
|
//
|
|
// Fills in pWCStr with the wide character(s) for the corresponding multibyte
|
|
// character from the appropriate translation table and returns the number
|
|
// of wide characters written. The number of bytes used from the pMBStr
|
|
// buffer (single byte or double byte) is returned in the mbIncr parameter.
|
|
// This routine should only be called when the precomposed forms need to be
|
|
// translated to composite.
|
|
//
|
|
// Checks to be sure an invalid character is not translated to the default
|
|
// character. If so, it sets last error and returns 0 characters written.
|
|
//
|
|
// 09-01-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int GetWCCompMBErr(
|
|
PCP_HASH pHashN,
|
|
PMB_TABLE pMBTbl,
|
|
LPBYTE pMBStr,
|
|
LPBYTE pEndMBStr,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr,
|
|
int *pmbIncr)
|
|
{
|
|
//
|
|
// Get the multibyte to wide char translation.
|
|
//
|
|
// Make sure an invalid character was not translated to the
|
|
// default char. If it was, set last error and return 0
|
|
// characters written.
|
|
//
|
|
GET_WC_MULTI_ERR( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pWCStr,
|
|
pEndWCStr,
|
|
*pmbIncr );
|
|
|
|
//
|
|
// Fill in the composite form of the character (if one exists)
|
|
// and return the number of wide characters written.
|
|
//
|
|
return (InsertCompositeForm(pWCStr, pEndWCStr));
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetMBNoDefault
|
|
//
|
|
// Translates the wide character string to a multibyte string and returns
|
|
// the number of bytes written.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int GetMBNoDefault(
|
|
PCP_HASH pHashN,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr,
|
|
LPBYTE pMBStr,
|
|
int cbMultiByte,
|
|
DWORD dwFlags)
|
|
{
|
|
int mbIncr; // amount to increment pMBStr
|
|
int mbCount = 0; // count of multibyte chars written
|
|
LPBYTE pEndMBStr; // ptr to end of MB string buffer
|
|
PWC_TABLE pWC = pHashN->pWC; // ptr to WC table
|
|
int ctr; // loop counter
|
|
|
|
|
|
//
|
|
// If cbMultiByte is 0, then we can't use pMBStr. In this
|
|
// case, we simply want to count the number of characters that
|
|
// would be written to the buffer.
|
|
//
|
|
if (cbMultiByte == 0)
|
|
{
|
|
BYTE pTempStr[2]; // tmp buffer - 2 bytes for DBCS
|
|
|
|
//
|
|
// For each wide char, translate it to its corresponding multibyte
|
|
// char and increment the multibyte character count.
|
|
//
|
|
if (IS_SBCS_CP(pHashN))
|
|
{
|
|
//
|
|
// Single Byte Character Code Page.
|
|
//
|
|
// Just return the count of characters - it will be the
|
|
// same number of characters as the source string.
|
|
//
|
|
mbCount = (int)(pEndWCStr - pWCStr);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Multi Byte Character Code Page.
|
|
//
|
|
if (dwFlags & WC_NO_BEST_FIT_CHARS)
|
|
{
|
|
while (pWCStr < pEndWCStr)
|
|
{
|
|
GET_MB( pWC,
|
|
*pWCStr,
|
|
pTempStr,
|
|
mbIncr,
|
|
FALSE );
|
|
ELIMINATE_BEST_FIT_MB( pHashN,
|
|
*pWCStr,
|
|
pTempStr,
|
|
mbIncr,
|
|
FALSE );
|
|
pWCStr++;
|
|
mbCount += mbIncr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
while (pWCStr < pEndWCStr)
|
|
{
|
|
GET_MB( pWC,
|
|
*pWCStr,
|
|
pTempStr,
|
|
mbIncr,
|
|
FALSE );
|
|
pWCStr++;
|
|
mbCount += mbIncr;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Initialize multibyte loop pointers.
|
|
//
|
|
pEndMBStr = pMBStr + cbMultiByte;
|
|
|
|
//
|
|
// For each wide char, translate it to its corresponding
|
|
// multibyte char, store it in pMBStr, and increment the
|
|
// multibyte character count.
|
|
//
|
|
if (IS_SBCS_CP(pHashN))
|
|
{
|
|
//
|
|
// Single Byte Character Code Page.
|
|
//
|
|
mbCount = (int)(pEndWCStr - pWCStr);
|
|
if ((pEndMBStr - pMBStr) < mbCount)
|
|
{
|
|
mbCount = (int)(pEndMBStr - pMBStr);
|
|
}
|
|
if (dwFlags & WC_NO_BEST_FIT_CHARS)
|
|
{
|
|
for (ctr = mbCount; ctr > 0; ctr--)
|
|
{
|
|
GET_SB( pWC,
|
|
*pWCStr,
|
|
pMBStr );
|
|
ELIMINATE_BEST_FIT_SB( pHashN,
|
|
*pWCStr,
|
|
pMBStr );
|
|
pWCStr++;
|
|
pMBStr++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (ctr = mbCount; ctr > 0; ctr--)
|
|
{
|
|
GET_SB( pWC,
|
|
*pWCStr,
|
|
pMBStr );
|
|
pWCStr++;
|
|
pMBStr++;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Multi Byte Character Code Page.
|
|
//
|
|
if (dwFlags & WC_NO_BEST_FIT_CHARS)
|
|
{
|
|
while ((pWCStr < pEndWCStr) && (pMBStr < pEndMBStr))
|
|
{
|
|
GET_MB( pWC,
|
|
*pWCStr,
|
|
pMBStr,
|
|
mbIncr,
|
|
((pMBStr + 1) < pEndMBStr) ? FALSE : TRUE );
|
|
ELIMINATE_BEST_FIT_MB( pHashN,
|
|
*pWCStr,
|
|
pMBStr,
|
|
mbIncr,
|
|
((pMBStr + 1) < pEndMBStr) ? FALSE : TRUE );
|
|
if (mbIncr == 0)
|
|
{
|
|
//
|
|
// Not enough space in buffer.
|
|
//
|
|
break;
|
|
}
|
|
|
|
pWCStr++;
|
|
mbCount += mbIncr;
|
|
pMBStr += mbIncr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
while ((pWCStr < pEndWCStr) && (pMBStr < pEndMBStr))
|
|
{
|
|
GET_MB( pWC,
|
|
*pWCStr,
|
|
pMBStr,
|
|
mbIncr,
|
|
((pMBStr + 1) < pEndMBStr) ? FALSE : TRUE );
|
|
if (mbIncr == 0)
|
|
{
|
|
//
|
|
// Not enough space in buffer.
|
|
//
|
|
break;
|
|
}
|
|
|
|
pWCStr++;
|
|
mbCount += mbIncr;
|
|
pMBStr += mbIncr;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Make sure multibyte character buffer was large enough.
|
|
//
|
|
if (pWCStr < pEndWCStr)
|
|
{
|
|
SetLastError(ERROR_INSUFFICIENT_BUFFER);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Return the number of characters written (or that would have
|
|
// been written) to the buffer.
|
|
//
|
|
return (mbCount);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetMBDefault
|
|
//
|
|
// Translates the wide character string to a multibyte string and returns
|
|
// the number of bytes written. This also checks for the use of the default
|
|
// character, so the translation is slower.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int GetMBDefault(
|
|
PCP_HASH pHashN,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr,
|
|
LPBYTE pMBStr,
|
|
int cbMultiByte,
|
|
WORD wDefault,
|
|
LPBOOL pUsedDef,
|
|
DWORD dwFlags)
|
|
{
|
|
int mbIncr; // amount to increment pMBStr
|
|
int mbIncr2; // amount to increment pMBStr
|
|
int mbCount = 0; // count of multibyte chars written
|
|
LPBYTE pEndMBStr; // ptr to end of MB string buffer
|
|
PWC_TABLE pWC = pHashN->pWC; // ptr to WC table
|
|
int ctr; // loop counter
|
|
|
|
|
|
//
|
|
// If cbMultiByte is 0, then we can't use pMBStr. In this
|
|
// case, we simply want to count the number of characters that
|
|
// would be written to the buffer.
|
|
//
|
|
if (cbMultiByte == 0)
|
|
{
|
|
BYTE pTempStr[2]; // tmp buffer - 2 bytes for DBCS
|
|
|
|
//
|
|
// For each wide char, translate it to its corresponding multibyte
|
|
// char and increment the multibyte character count.
|
|
//
|
|
if (IS_SBCS_CP(pHashN))
|
|
{
|
|
//
|
|
// Single Byte Character Code Page.
|
|
//
|
|
mbCount = (int)(pEndWCStr - pWCStr);
|
|
if (dwFlags & WC_NO_BEST_FIT_CHARS)
|
|
{
|
|
while (pWCStr < pEndWCStr)
|
|
{
|
|
GET_SB( pWC,
|
|
*pWCStr,
|
|
pTempStr );
|
|
ELIMINATE_BEST_FIT_SB( pHashN,
|
|
*pWCStr,
|
|
pTempStr );
|
|
DEFAULT_CHAR_CHECK_SB( pHashN,
|
|
*pWCStr,
|
|
pTempStr,
|
|
wDefault,
|
|
pUsedDef );
|
|
pWCStr++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
while (pWCStr < pEndWCStr)
|
|
{
|
|
GET_SB( pWC,
|
|
*pWCStr,
|
|
pTempStr );
|
|
DEFAULT_CHAR_CHECK_SB( pHashN,
|
|
*pWCStr,
|
|
pTempStr,
|
|
wDefault,
|
|
pUsedDef );
|
|
pWCStr++;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Multi Byte Character Code Page.
|
|
//
|
|
if (dwFlags & WC_NO_BEST_FIT_CHARS)
|
|
{
|
|
while (pWCStr < pEndWCStr)
|
|
{
|
|
GET_MB( pWC,
|
|
*pWCStr,
|
|
pTempStr,
|
|
mbIncr,
|
|
FALSE );
|
|
ELIMINATE_BEST_FIT_MB( pHashN,
|
|
*pWCStr,
|
|
pTempStr,
|
|
mbIncr,
|
|
FALSE );
|
|
DEFAULT_CHAR_CHECK_MB( pHashN,
|
|
*pWCStr,
|
|
pTempStr,
|
|
wDefault,
|
|
pUsedDef,
|
|
mbIncr2,
|
|
FALSE );
|
|
mbCount += (mbIncr2) ? (mbIncr2) : (mbIncr);
|
|
pWCStr++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
while (pWCStr < pEndWCStr)
|
|
{
|
|
GET_MB( pWC,
|
|
*pWCStr,
|
|
pTempStr,
|
|
mbIncr,
|
|
FALSE );
|
|
DEFAULT_CHAR_CHECK_MB( pHashN,
|
|
*pWCStr,
|
|
pTempStr,
|
|
wDefault,
|
|
pUsedDef,
|
|
mbIncr2,
|
|
FALSE );
|
|
mbCount += (mbIncr2) ? (mbIncr2) : (mbIncr);
|
|
pWCStr++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Initialize multibyte loop pointers.
|
|
//
|
|
pEndMBStr = pMBStr + cbMultiByte;
|
|
|
|
//
|
|
// For each wide char, translate it to its corresponding
|
|
// multibyte char, store it in pMBStr, and increment the
|
|
// multibyte character count.
|
|
//
|
|
if (IS_SBCS_CP(pHashN))
|
|
{
|
|
//
|
|
// Single Byte Character Code Page.
|
|
//
|
|
mbCount = (int)(pEndWCStr - pWCStr);
|
|
if ((pEndMBStr - pMBStr) < mbCount)
|
|
{
|
|
mbCount = (int)(pEndMBStr - pMBStr);
|
|
}
|
|
if (dwFlags & WC_NO_BEST_FIT_CHARS)
|
|
{
|
|
for (ctr = mbCount; ctr > 0; ctr--)
|
|
{
|
|
GET_SB( pWC,
|
|
*pWCStr,
|
|
pMBStr );
|
|
ELIMINATE_BEST_FIT_SB( pHashN,
|
|
*pWCStr,
|
|
pMBStr );
|
|
DEFAULT_CHAR_CHECK_SB( pHashN,
|
|
*pWCStr,
|
|
pMBStr,
|
|
wDefault,
|
|
pUsedDef );
|
|
pWCStr++;
|
|
pMBStr++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (ctr = mbCount; ctr > 0; ctr--)
|
|
{
|
|
GET_SB( pWC,
|
|
*pWCStr,
|
|
pMBStr );
|
|
DEFAULT_CHAR_CHECK_SB( pHashN,
|
|
*pWCStr,
|
|
pMBStr,
|
|
wDefault,
|
|
pUsedDef );
|
|
pWCStr++;
|
|
pMBStr++;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Multi Byte Character Code Page.
|
|
//
|
|
if (dwFlags & WC_NO_BEST_FIT_CHARS)
|
|
{
|
|
while ((pWCStr < pEndWCStr) && (pMBStr < pEndMBStr))
|
|
{
|
|
GET_MB( pWC,
|
|
*pWCStr,
|
|
pMBStr,
|
|
mbIncr,
|
|
((pMBStr + 1) < pEndMBStr) ? FALSE : TRUE );
|
|
ELIMINATE_BEST_FIT_MB( pHashN,
|
|
*pWCStr,
|
|
pMBStr,
|
|
mbIncr,
|
|
((pMBStr + 1) < pEndMBStr) ? FALSE : TRUE );
|
|
DEFAULT_CHAR_CHECK_MB( pHashN,
|
|
*pWCStr,
|
|
pMBStr,
|
|
wDefault,
|
|
pUsedDef,
|
|
mbIncr2,
|
|
((pMBStr + 1) < pEndMBStr) ? FALSE : TRUE );
|
|
if ((mbIncr == 0) || (mbIncr2 == -1))
|
|
{
|
|
//
|
|
// Not enough room in buffer.
|
|
//
|
|
break;
|
|
}
|
|
|
|
mbCount += (mbIncr2) ? (mbIncr2) : (mbIncr);
|
|
pWCStr++;
|
|
pMBStr += mbIncr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
while ((pWCStr < pEndWCStr) && (pMBStr < pEndMBStr))
|
|
{
|
|
GET_MB( pWC,
|
|
*pWCStr,
|
|
pMBStr,
|
|
mbIncr,
|
|
((pMBStr + 1) < pEndMBStr) ? FALSE : TRUE );
|
|
DEFAULT_CHAR_CHECK_MB( pHashN,
|
|
*pWCStr,
|
|
pMBStr,
|
|
wDefault,
|
|
pUsedDef,
|
|
mbIncr2,
|
|
((pMBStr + 1) < pEndMBStr) ? FALSE : TRUE );
|
|
if ((mbIncr == 0) || (mbIncr2 == -1))
|
|
{
|
|
//
|
|
// Not enough room in buffer.
|
|
//
|
|
break;
|
|
}
|
|
|
|
mbCount += (mbIncr2) ? (mbIncr2) : (mbIncr);
|
|
pWCStr++;
|
|
pMBStr += mbIncr;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Make sure multibyte character buffer was large enough.
|
|
//
|
|
if (pWCStr < pEndWCStr)
|
|
{
|
|
SetLastError(ERROR_INSUFFICIENT_BUFFER);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Return the number of characters written (or that would have
|
|
// been written) to the buffer.
|
|
//
|
|
return (mbCount);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetMBDefaultComp
|
|
//
|
|
// Translates the wide character string to a multibyte string and returns
|
|
// the number of bytes written. This also checks for the use of the default
|
|
// character and tries to convert composite forms to precomposed forms, so
|
|
// the translation is a lot slower.
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int GetMBDefaultComp(
|
|
PCP_HASH pHashN,
|
|
LPWSTR pWCStr,
|
|
LPWSTR pEndWCStr,
|
|
LPBYTE pMBStr,
|
|
int cbMultiByte,
|
|
WORD wDefault,
|
|
LPBOOL pUsedDef,
|
|
DWORD dwFlags)
|
|
{
|
|
int mbIncr; // amount to increment pMBStr
|
|
int mbCount = 0; // count of multibyte chars written
|
|
LPBYTE pEndMBStr; // ptr to end of MB string buffer
|
|
BOOL fError; // if error during MB conversion
|
|
|
|
|
|
//
|
|
// If cbMultiByte is 0, then we can't use pMBStr. In this
|
|
// case, we simply want to count the number of characters that
|
|
// would be written to the buffer.
|
|
//
|
|
if (cbMultiByte == 0)
|
|
{
|
|
BYTE pTempStr[2]; // tmp buffer - 2 bytes for DBCS
|
|
|
|
//
|
|
// Set most significant bit of flags to indicate to the
|
|
// GetMBComp routine that it's using a temporary storage
|
|
// area, so don't back up in the buffer.
|
|
//
|
|
SET_MSB(dwFlags);
|
|
|
|
//
|
|
// For each wide char, translate it to its corresponding multibyte
|
|
// char and increment the multibyte character count.
|
|
//
|
|
if (IS_SBCS_CP(pHashN))
|
|
{
|
|
//
|
|
// Single Byte Character Code Page.
|
|
//
|
|
while (pWCStr < pEndWCStr)
|
|
{
|
|
//
|
|
// Get the translation.
|
|
//
|
|
mbCount += GetMBCompSB( pHashN,
|
|
dwFlags,
|
|
pWCStr,
|
|
pTempStr,
|
|
mbCount,
|
|
wDefault,
|
|
pUsedDef );
|
|
pWCStr++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Multi Byte Character Code Page.
|
|
//
|
|
while (pWCStr < pEndWCStr)
|
|
{
|
|
//
|
|
// Get the translation.
|
|
//
|
|
mbCount += GetMBCompMB( pHashN,
|
|
dwFlags,
|
|
pWCStr,
|
|
pTempStr,
|
|
mbCount,
|
|
wDefault,
|
|
pUsedDef,
|
|
&fError,
|
|
FALSE );
|
|
pWCStr++;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Initialize multibyte loop pointers.
|
|
//
|
|
pEndMBStr = pMBStr + cbMultiByte;
|
|
|
|
//
|
|
// For each wide char, translate it to its corresponding
|
|
// multibyte char, store it in pMBStr, and increment the
|
|
// multibyte character count.
|
|
//
|
|
if (IS_SBCS_CP(pHashN))
|
|
{
|
|
//
|
|
// Single Byte Character Code Page.
|
|
//
|
|
while ((pWCStr < pEndWCStr) && (pMBStr < pEndMBStr))
|
|
{
|
|
//
|
|
// Get the translation.
|
|
//
|
|
mbIncr = GetMBCompSB( pHashN,
|
|
dwFlags,
|
|
pWCStr,
|
|
pMBStr,
|
|
mbCount,
|
|
wDefault,
|
|
pUsedDef );
|
|
pWCStr++;
|
|
mbCount += mbIncr;
|
|
pMBStr += mbIncr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Multi Byte Character Code Page.
|
|
//
|
|
while ((pWCStr < pEndWCStr) && (pMBStr < pEndMBStr))
|
|
{
|
|
//
|
|
// Get the translation.
|
|
//
|
|
mbIncr = GetMBCompMB( pHashN,
|
|
dwFlags,
|
|
pWCStr,
|
|
pMBStr,
|
|
mbCount,
|
|
wDefault,
|
|
pUsedDef,
|
|
&fError,
|
|
((pMBStr + 1) < pEndMBStr) ? FALSE : TRUE );
|
|
if (fError)
|
|
{
|
|
//
|
|
// Not enough room in the buffer.
|
|
//
|
|
break;
|
|
}
|
|
|
|
pWCStr++;
|
|
mbCount += mbIncr;
|
|
pMBStr += mbIncr;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Make sure multibyte character buffer was large enough.
|
|
//
|
|
if (pWCStr < pEndWCStr)
|
|
{
|
|
SetLastError(ERROR_INSUFFICIENT_BUFFER);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Return the number of characters written (or that would have
|
|
// been written) to the buffer.
|
|
//
|
|
return (mbCount);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetMBCompSB
|
|
//
|
|
// Fills in pMBStr with the byte character(s) for the corresponding wide
|
|
// character from the appropriate translation table and returns the number
|
|
// of byte characters written to pMBStr. This routine is only called if
|
|
// the defaultcheck and compositecheck flags were both set.
|
|
//
|
|
// NOTE: Most significant bit of dwFlags parameter is used by this routine
|
|
// to indicate that the caller only wants the count of the number of
|
|
// characters written, not the string (ie. do not back up in buffer).
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int GetMBCompSB(
|
|
PCP_HASH pHashN,
|
|
DWORD dwFlags,
|
|
LPWSTR pWCStr,
|
|
LPBYTE pMBStr,
|
|
int mbCount,
|
|
WORD wDefault,
|
|
LPBOOL pUsedDef)
|
|
{
|
|
WCHAR PreComp; // precomposed wide character
|
|
|
|
|
|
if ((pTblPtrs->pDefaultSortkey == NULL) ||
|
|
(!IS_NONSPACE_ONLY(pTblPtrs->pDefaultSortkey, *pWCStr)))
|
|
{
|
|
//
|
|
// Get the 1:1 translation from wide char to single byte.
|
|
//
|
|
GET_WC_TRANSLATION_SB( pHashN,
|
|
*pWCStr,
|
|
pMBStr,
|
|
wDefault,
|
|
pUsedDef,
|
|
dwFlags );
|
|
return (1);
|
|
}
|
|
else
|
|
{
|
|
if (mbCount < 1)
|
|
{
|
|
//
|
|
// Need to handle the nonspace character by itself, since
|
|
// it is the first character in the string.
|
|
//
|
|
if (dwFlags & WC_DISCARDNS)
|
|
{
|
|
//
|
|
// Discard the non-spacing char, so just return with
|
|
// zero chars written.
|
|
//
|
|
return (0);
|
|
}
|
|
else if (dwFlags & WC_DEFAULTCHAR)
|
|
{
|
|
//
|
|
// Need to replace the nonspace character with the default
|
|
// character and return the number of characters written
|
|
// to the multibyte string.
|
|
//
|
|
*pUsedDef = TRUE;
|
|
*pMBStr = LOBYTE(wDefault);
|
|
return (1);
|
|
}
|
|
else // WC_SEPCHARS - default
|
|
{
|
|
//
|
|
// Get the 1:1 translation from wide char to multibyte
|
|
// of the non-spacing char and return the number of
|
|
// characters written to the multibyte string.
|
|
//
|
|
GET_WC_TRANSLATION_SB( pHashN,
|
|
*pWCStr,
|
|
pMBStr,
|
|
wDefault,
|
|
pUsedDef,
|
|
dwFlags );
|
|
return (1);
|
|
}
|
|
}
|
|
else if (PreComp = GetPreComposedChar(*pWCStr, *(pWCStr - 1)))
|
|
{
|
|
//
|
|
// Back up in the single byte string and write the
|
|
// precomposed char.
|
|
//
|
|
if (!IS_MSB(dwFlags))
|
|
{
|
|
pMBStr--;
|
|
}
|
|
|
|
GET_WC_TRANSLATION_SB( pHashN,
|
|
PreComp,
|
|
pMBStr,
|
|
wDefault,
|
|
pUsedDef,
|
|
dwFlags );
|
|
return (0);
|
|
}
|
|
else
|
|
{
|
|
if (dwFlags & WC_DISCARDNS)
|
|
{
|
|
//
|
|
// Discard the non-spacing char, so just return with
|
|
// zero chars written.
|
|
//
|
|
return (0);
|
|
}
|
|
else if (dwFlags & WC_DEFAULTCHAR)
|
|
{
|
|
//
|
|
// Need to replace the base character with the default
|
|
// character. Since we've already written the base
|
|
// translation char in the single byte string, we need to
|
|
// back up in the single byte string and write the default
|
|
// char.
|
|
//
|
|
if (!IS_MSB(dwFlags))
|
|
{
|
|
pMBStr--;
|
|
}
|
|
|
|
*pUsedDef = TRUE;
|
|
*pMBStr = LOBYTE(wDefault);
|
|
return (0);
|
|
}
|
|
else // WC_SEPCHARS - default
|
|
{
|
|
//
|
|
// Get the 1:1 translation from wide char to multibyte
|
|
// of the non-spacing char and return the number of
|
|
// characters written to the multibyte string.
|
|
//
|
|
GET_WC_TRANSLATION_SB( pHashN,
|
|
*pWCStr,
|
|
pMBStr,
|
|
wDefault,
|
|
pUsedDef,
|
|
dwFlags );
|
|
return (1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetMBCompMB
|
|
//
|
|
// Fills in pMBStr with the byte character(s) for the corresponding wide
|
|
// character from the appropriate translation table and returns the number
|
|
// of byte characters written to pMBStr. This routine is only called if
|
|
// the defaultcheck and compositecheck flags were both set.
|
|
//
|
|
// If the buffer was too small, the fError flag will be set to TRUE.
|
|
//
|
|
// NOTE: Most significant bit of dwFlags parameter is used by this routine
|
|
// to indicate that the caller only wants the count of the number of
|
|
// characters written, not the string (ie. do not back up in buffer).
|
|
//
|
|
// 05-31-91 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int GetMBCompMB(
|
|
PCP_HASH pHashN,
|
|
DWORD dwFlags,
|
|
LPWSTR pWCStr,
|
|
LPBYTE pMBStr,
|
|
int mbCount,
|
|
WORD wDefault,
|
|
LPBOOL pUsedDef,
|
|
BOOL *fError,
|
|
BOOL fOnlyOne)
|
|
{
|
|
WCHAR PreComp; // precomposed wide character
|
|
BYTE pTmpSp[2]; // temp space - 2 bytes for DBCS
|
|
int nCnt; // number of characters written
|
|
|
|
|
|
*fError = FALSE;
|
|
if ((pTblPtrs->pDefaultSortkey == NULL) ||
|
|
(!IS_NONSPACE_ONLY(pTblPtrs->pDefaultSortkey, *pWCStr)))
|
|
{
|
|
//
|
|
// Get the 1:1 translation from wide char to multibyte.
|
|
// This also handles DBCS and returns the number of characters
|
|
// written to the multibyte string.
|
|
//
|
|
GET_WC_TRANSLATION_MB( pHashN,
|
|
*pWCStr,
|
|
pMBStr,
|
|
wDefault,
|
|
pUsedDef,
|
|
nCnt,
|
|
fOnlyOne,
|
|
dwFlags );
|
|
if (nCnt == 0)
|
|
{
|
|
*fError = TRUE;
|
|
}
|
|
return (nCnt);
|
|
}
|
|
else
|
|
{
|
|
if (mbCount < 1)
|
|
{
|
|
//
|
|
// Need to handle the nonspace character by itself, since
|
|
// it is the first character in the string.
|
|
//
|
|
if (dwFlags & WC_DISCARDNS)
|
|
{
|
|
//
|
|
// Discard the non-spacing char, so just return with
|
|
// zero chars written.
|
|
//
|
|
return (0);
|
|
}
|
|
else if (dwFlags & WC_DEFAULTCHAR)
|
|
{
|
|
//
|
|
// Need to replace the nonspace character with the default
|
|
// character and return the number of characters written
|
|
// to the multibyte string.
|
|
//
|
|
*pUsedDef = TRUE;
|
|
COPY_MB_CHAR( wDefault,
|
|
pMBStr,
|
|
nCnt,
|
|
fOnlyOne );
|
|
if (nCnt == 0)
|
|
{
|
|
*fError = TRUE;
|
|
}
|
|
return (nCnt);
|
|
}
|
|
else // WC_SEPCHARS - default
|
|
{
|
|
//
|
|
// Get the 1:1 translation from wide char to multibyte
|
|
// of the non-spacing char and return the number of
|
|
// characters written to the multibyte string.
|
|
//
|
|
GET_WC_TRANSLATION_MB( pHashN,
|
|
*pWCStr,
|
|
pMBStr,
|
|
wDefault,
|
|
pUsedDef,
|
|
nCnt,
|
|
fOnlyOne,
|
|
dwFlags );
|
|
if (nCnt == 0)
|
|
{
|
|
*fError = TRUE;
|
|
}
|
|
return (nCnt);
|
|
}
|
|
|
|
}
|
|
else if (PreComp = GetPreComposedChar(*pWCStr, *(pWCStr - 1)))
|
|
{
|
|
//
|
|
// Get the 1:1 translation from wide char to multibyte
|
|
// of the precomposed char, back up in the multibyte string,
|
|
// write the precomposed char, and return the DIFFERENCE of
|
|
// the number of characters written to the the multibyte
|
|
// string.
|
|
//
|
|
GET_WC_TRANSLATION_MB( pHashN,
|
|
*(pWCStr - 1),
|
|
pTmpSp,
|
|
wDefault,
|
|
pUsedDef,
|
|
nCnt,
|
|
fOnlyOne,
|
|
dwFlags );
|
|
if (nCnt == 0)
|
|
{
|
|
*fError = TRUE;
|
|
return (nCnt);
|
|
}
|
|
|
|
if (!IS_MSB(dwFlags))
|
|
{
|
|
pMBStr -= nCnt;
|
|
}
|
|
|
|
GET_WC_TRANSLATION_MB( pHashN,
|
|
PreComp,
|
|
pMBStr,
|
|
wDefault,
|
|
pUsedDef,
|
|
mbCount,
|
|
fOnlyOne,
|
|
dwFlags );
|
|
if (mbCount == 0)
|
|
{
|
|
*fError = TRUE;
|
|
}
|
|
return (mbCount - nCnt);
|
|
}
|
|
else
|
|
{
|
|
if (dwFlags & WC_DISCARDNS)
|
|
{
|
|
//
|
|
// Discard the non-spacing char, so just return with
|
|
// zero chars written.
|
|
//
|
|
return (0);
|
|
}
|
|
else if (dwFlags & WC_DEFAULTCHAR)
|
|
{
|
|
//
|
|
// Need to replace the base character with the default
|
|
// character. Since we've already written the base
|
|
// translation char in the multibyte string, we need to
|
|
// back up in the multibyte string and return the
|
|
// DIFFERENCE of the number of characters written
|
|
// (could be negative).
|
|
//
|
|
|
|
//
|
|
// If the previous character written is the default
|
|
// character, then the base character for this nonspace
|
|
// character has already been replaced. Simply throw
|
|
// this character away and return zero chars written.
|
|
//
|
|
if (!IS_MSB(dwFlags))
|
|
{
|
|
//
|
|
// Not using a temporary buffer, so find out if the
|
|
// previous character translated was the default char.
|
|
//
|
|
if ((MAKEWORD(*(pMBStr - 1), 0) == wDefault) ||
|
|
((mbCount > 1) &&
|
|
(MAKEWORD(*(pMBStr - 1), *(pMBStr - 2)) == wDefault)))
|
|
{
|
|
return (0);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Using a temporary buffer. The temp buffer is 2 bytes
|
|
// in length and contains the previous character written.
|
|
//
|
|
if ((MAKEWORD(*pMBStr, 0) == wDefault) ||
|
|
((mbCount > 1) &&
|
|
(MAKEWORD(*pMBStr, *(pMBStr + 1)) == wDefault)))
|
|
{
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Get the 1:1 translation from wide char to multibyte
|
|
// of the base char, back up in the multibyte string,
|
|
// write the default char, and return the DIFFERENCE of
|
|
// the number of characters written to the the multibyte
|
|
// string.
|
|
//
|
|
GET_WC_TRANSLATION_MB( pHashN,
|
|
*(pWCStr - 1),
|
|
pTmpSp,
|
|
wDefault,
|
|
pUsedDef,
|
|
nCnt,
|
|
fOnlyOne,
|
|
dwFlags );
|
|
if (nCnt == 0)
|
|
{
|
|
*fError = TRUE;
|
|
return (nCnt);
|
|
}
|
|
|
|
if (!IS_MSB(dwFlags))
|
|
{
|
|
pMBStr -= nCnt;
|
|
}
|
|
|
|
*pUsedDef = TRUE;
|
|
COPY_MB_CHAR( wDefault,
|
|
pMBStr,
|
|
mbCount,
|
|
fOnlyOne );
|
|
if (mbCount == 0)
|
|
{
|
|
*fError = TRUE;
|
|
}
|
|
return (mbCount - nCnt);
|
|
}
|
|
else // WC_SEPCHARS - default
|
|
{
|
|
//
|
|
// Get the 1:1 translation from wide char to multibyte
|
|
// of the non-spacing char and return the number of
|
|
// characters written to the multibyte string.
|
|
//
|
|
GET_WC_TRANSLATION_MB( pHashN,
|
|
*pWCStr,
|
|
pMBStr,
|
|
wDefault,
|
|
pUsedDef,
|
|
nCnt,
|
|
fOnlyOne,
|
|
dwFlags );
|
|
if (nCnt == 0)
|
|
{
|
|
*fError = TRUE;
|
|
}
|
|
return (nCnt);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// GetMacCodePage
|
|
//
|
|
// Returns the system default Mac code page.
|
|
//
|
|
// 09-22-93 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
UINT GetMacCodePage()
|
|
{
|
|
PKEY_VALUE_FULL_INFORMATION pKeyValueFull; // ptr to query information
|
|
BYTE pStatic[MAX_KEY_VALUE_FULLINFO]; // ptr to static buffer
|
|
UNICODE_STRING ObUnicodeStr; // unicode string
|
|
UINT CodePage; // code page value
|
|
PCP_HASH pHashN; // ptr to hash node
|
|
|
|
|
|
//
|
|
// See if the Mac code page globals have been initialized yet.
|
|
// If they have, return the mac code page value.
|
|
//
|
|
if (gMacCodePage != 0)
|
|
{
|
|
return (gMacCodePage);
|
|
}
|
|
|
|
//
|
|
// Make sure code page key is open.
|
|
//
|
|
OPEN_CODEPAGE_KEY(NLS_DEFAULT_MACCP);
|
|
|
|
//
|
|
// Query the registry for the Mac CP value.
|
|
//
|
|
CodePage = 0;
|
|
pKeyValueFull = (PKEY_VALUE_FULL_INFORMATION)pStatic;
|
|
if ((QueryRegValue( hCodePageKey,
|
|
NLS_VALUE_MACCP,
|
|
&pKeyValueFull,
|
|
MAX_KEY_VALUE_FULLINFO,
|
|
NULL )) == NO_ERROR)
|
|
{
|
|
//
|
|
// Convert the value to an integer.
|
|
//
|
|
RtlInitUnicodeString(&ObUnicodeStr, GET_VALUE_DATA_PTR(pKeyValueFull));
|
|
if (RtlUnicodeStringToInteger(&ObUnicodeStr, 10, (PULONG)&CodePage))
|
|
{
|
|
CodePage = 0;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Make sure the CodePage value was set.
|
|
//
|
|
if (CodePage == 0)
|
|
{
|
|
//
|
|
// Registry value is corrupt, so use default Mac code page.
|
|
//
|
|
CodePage = NLS_DEFAULT_MACCP;
|
|
}
|
|
|
|
//
|
|
// Get the hash node for the Mac code page.
|
|
//
|
|
pHashN = GetCPHashNode(CodePage);
|
|
|
|
//
|
|
// Make sure the Mac hash node is valid.
|
|
//
|
|
if (pHashN == NULL)
|
|
{
|
|
//
|
|
// Invalid hash node, which means either the registry is
|
|
// corrupt, or setup failed to install a file. Use the
|
|
// Ansi code page values.
|
|
//
|
|
CodePage = gAnsiCodePage;
|
|
pHashN = gpACPHashN;
|
|
}
|
|
|
|
//
|
|
// Set the final MAC CP values.
|
|
//
|
|
RtlEnterCriticalSection(&gcsTblPtrs);
|
|
|
|
if (gMacCodePage == 0)
|
|
{
|
|
gpMACCPHashN = pHashN;
|
|
gMacCodePage = CodePage;
|
|
}
|
|
|
|
RtlLeaveCriticalSection(&gcsTblPtrs);
|
|
|
|
//
|
|
// Return the Mac code page value.
|
|
//
|
|
return (gMacCodePage);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// SpecialMBToWC
|
|
//
|
|
// Maps a multibyte character string to its wide character string
|
|
// counterpart.
|
|
//
|
|
// 08-21-95 JulieB Created.
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
int SpecialMBToWC(
|
|
PCP_HASH pHashN,
|
|
DWORD dwFlags,
|
|
LPCSTR lpMultiByteStr,
|
|
int cbMultiByte,
|
|
LPWSTR lpWideCharStr,
|
|
int cchWideChar)
|
|
{
|
|
register LPBYTE pMBStr; // ptr to search through MB string
|
|
register LPWSTR pWCStr; // ptr to search through WC string
|
|
LPBYTE pEndMBStr; // ptr to end of MB search string
|
|
LPWSTR pEndWCStr; // ptr to end of WC string buffer
|
|
int mbIncr; // amount to increment pMBStr
|
|
int wcCount = 0; // count of wide chars written
|
|
PMB_TABLE pMBTbl; // ptr to MB table
|
|
int ctr; // loop counter
|
|
|
|
|
|
//
|
|
// Initialize multibyte character loop pointers.
|
|
//
|
|
pMBStr = (LPBYTE)lpMultiByteStr;
|
|
pEndMBStr = pMBStr + cbMultiByte;
|
|
|
|
//
|
|
// Get the MB table.
|
|
//
|
|
pMBTbl = pHashN->pMBTbl;
|
|
|
|
//
|
|
// If cchWideChar is 0, then we can't use lpWideCharStr. In this
|
|
// case, we simply want to count the number of characters that would
|
|
// be written to the buffer.
|
|
//
|
|
if (cchWideChar == 0)
|
|
{
|
|
//
|
|
// For each multibyte char, translate it to its corresponding
|
|
// wide char and increment the wide character count.
|
|
//
|
|
if (IS_SBCS_CP(pHashN))
|
|
{
|
|
//
|
|
// Single Byte Character Code Page.
|
|
//
|
|
wcCount = (int)(pEndMBStr - pMBStr);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Multi Byte Character Code Page.
|
|
//
|
|
WCHAR pTempStr[MAX_COMPOSITE]; // tmp buffer
|
|
|
|
pEndWCStr = pTempStr + MAX_COMPOSITE;
|
|
while (pMBStr < pEndMBStr)
|
|
{
|
|
GET_WC_MULTI( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pTempStr,
|
|
pEndWCStr,
|
|
mbIncr );
|
|
pMBStr += mbIncr;
|
|
wcCount++;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Initialize wide character loop pointers.
|
|
//
|
|
pWCStr = lpWideCharStr;
|
|
pEndWCStr = pWCStr + cchWideChar;
|
|
|
|
//
|
|
// For each multibyte char, translate it to its corresponding
|
|
// wide char, store it in lpWideCharStr, and increment the wide
|
|
// character count.
|
|
//
|
|
if (IS_SBCS_CP(pHashN))
|
|
{
|
|
//
|
|
// Single Byte Character Code Page.
|
|
//
|
|
wcCount = (int)(pEndMBStr - pMBStr);
|
|
if ((pEndWCStr - pWCStr) < wcCount)
|
|
{
|
|
wcCount = (int)(pEndWCStr - pWCStr);
|
|
}
|
|
|
|
if (dwFlags & MB_INVALID_CHAR_CHECK)
|
|
{
|
|
//
|
|
// Error check flag is set.
|
|
//
|
|
for (ctr = wcCount; ctr > 0; ctr--)
|
|
{
|
|
GET_WC_SINGLE_SPECIAL( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pWCStr );
|
|
pMBStr++;
|
|
pWCStr++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Error check flag is NOT set.
|
|
//
|
|
for (ctr = wcCount; ctr > 0; ctr--)
|
|
{
|
|
GET_WC_SINGLE( pMBTbl,
|
|
pMBStr,
|
|
pWCStr );
|
|
pMBStr++;
|
|
pWCStr++;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Multi Byte Character Code Page.
|
|
//
|
|
if (dwFlags & MB_INVALID_CHAR_CHECK)
|
|
{
|
|
//
|
|
// Error check flag is set.
|
|
//
|
|
while ((pMBStr < pEndMBStr) && (pWCStr < pEndWCStr))
|
|
{
|
|
GET_WC_MULTI_ERR_SPECIAL( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pWCStr,
|
|
pEndWCStr,
|
|
mbIncr );
|
|
pMBStr += mbIncr;
|
|
pWCStr++;
|
|
}
|
|
wcCount = (int)(pWCStr - lpWideCharStr);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Error check flag is NOT set.
|
|
//
|
|
while ((pMBStr < pEndMBStr) && (pWCStr < pEndWCStr))
|
|
{
|
|
GET_WC_MULTI( pHashN,
|
|
pMBTbl,
|
|
pMBStr,
|
|
pEndMBStr,
|
|
pWCStr,
|
|
pEndWCStr,
|
|
mbIncr );
|
|
pMBStr += mbIncr;
|
|
pWCStr++;
|
|
}
|
|
wcCount = (int)(pWCStr - lpWideCharStr);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Make sure wide character buffer was large enough.
|
|
//
|
|
if (pMBStr < pEndMBStr)
|
|
{
|
|
SetLastError(ERROR_INSUFFICIENT_BUFFER);
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Return the number of characters written (or that would have
|
|
// been written) to the buffer.
|
|
//
|
|
return (wcCount);
|
|
}
|