Windows2003-3790/windows/feime/kor/ime2k/imm32/hanja.cpp

440 lines
14 KiB
C++
Raw Permalink Normal View History

2001-01-01 00:00:00 +01:00
/****************************************************************************
HANJA.CPP
Owner: cslim
Copyright (c) 1997-1999 Microsoft Corporation
Hanja conversion and dictionary lookup functions. Dictionary index is
stored as globally shared memory.
History:
14-JUL-1999 cslim Copied from IME98 source tree
*****************************************************************************/
#include "precomp.h"
#include "apientry.h"
#include "ui.h"
#include "debug.h"
#include "lexheader.h"
#include "hanja.h"
#include "immsec.h"
#include "winex.h"
#include "common.h"
#include <WINERROR.H>
// NT5 Globally shared memory.
const TCHAR IMEKR_LEX_SHAREDDATA_MUTEX_NAME[] = TEXT("{C5AFBBF9-8383-490c-AA9E-4FE93FA05512}");
const TCHAR IMEKR_LEX_SHAREDDATA_NAME[] = TEXT("ImeKrLexHangul2Hanja.SharedMemory");
UINT vuNumofK0=0, vuNumofK1=0;
WCHAR vwcHangul=0;
// Private data
PRIVATE BOOL vfLexOpen = fFalse;
PRIVATE HANDLE vhLex=0;
PRIVATE HANDLE vhLexIndexTbl=0;
PRIVATE UINT vuNumOfHangulEntry=0;
PRIVATE DWORD viBufferStart=0; // seek point
// Private functions
PRIVATE BOOL OpenLex();
//static VOID ClearHanjaSenseArray();
PRIVATE INT SearchHanjaIndex(WCHAR wHChar, _LexIndex *pLexIndexTbl);
/*
CHanja::CHanja()
{
vfLexOpen = fFalse;
vhLex = vhLexIndexTbl = vhLexIndexTbl = NULL;
vuNumOfHangulEntry = 0;
for (int i=0; i<MAX_CANDSTR; i++)
vprwszHanjaMeaning[i] = 0;
}
*/
BOOL EnsureHanjaLexLoaded()
{
_DictHeader *pLexHeader;
DWORD dwReadBytes;
CHAR szLexFileName[MAX_PATH], szLexPathExpanded[MAX_PATH];
HKEY hKey;
DWORD dwType, dwCb;
CIMEData ImeData;
if (vfLexOpen)
return fTrue;
// Get Lex file name with full path
szLexFileName[0] = 0;
szLexPathExpanded[0] = 0;
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, g_szIMERootKey, 0, KEY_READ, &hKey) == ERROR_SUCCESS)
{
dwCb = sizeof(szLexFileName);
dwType = REG_SZ;
if (RegQueryValueEx(hKey, g_szDictionary, NULL, &dwType, (LPBYTE)szLexFileName, &dwCb) == ERROR_SUCCESS)
ExpandEnvironmentStrings(szLexFileName, szLexPathExpanded, sizeof(szLexPathExpanded));
RegCloseKey(hKey);
}
DbgAssert(szLexPathExpanded[0] != 0);
if (szLexPathExpanded[0] == 0)
return fFalse;
vhLex = CreateFile(szLexPathExpanded, GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, FILE_FLAG_RANDOM_ACCESS, NULL);
if (vhLex==INVALID_HANDLE_VALUE)
{
DbgAssert(0);
return fFalse;
}
pLexHeader = new _DictHeader;
if (!pLexHeader)
return fFalse;
if (ReadFile(vhLex, pLexHeader, sizeof(_DictHeader), &dwReadBytes, 0) == 0 || (dwReadBytes != sizeof(_DictHeader)))
{
DbgAssert(0);
return fFalse;
}
// Set member vars
vuNumOfHangulEntry = pLexHeader->NumOfHangulEntry;
viBufferStart = pLexHeader->iBufferStart;
if (pLexHeader->Version < LEX_VERSION || pLexHeader->Version > LEX_COMPATIBLE_VERSION_LIMIT )
{
delete pLexHeader;
return fFalse;
}
if (lstrcmpA(pLexHeader->COPYRIGHT_HEADER, COPYRIGHT_STR))
{
delete pLexHeader;
return fFalse;
}
// Read Index table
SetFilePointer(vhLex, pLexHeader->Headersize, 0, FILE_BEGIN);
delete pLexHeader;
return OpenLex();
}
BOOL OpenLex()
{
BOOL fRet = fFalse;
_LexIndex *pLexIndexTbl;
HANDLE hMutex;
DWORD dwReadBytes;
///////////////////////////////////////////////////////////////////////////
// Mapping Lex file
// The dictionary index is shared data between all IME instance
hMutex=CreateMutex(GetIMESecurityAttributes(), fFalse, IMEKR_LEX_SHAREDDATA_MUTEX_NAME);
if (hMutex != NULL)
{
if (DoEnterCriticalSection(hMutex) == fFalse)
goto ExitOpenLexCritSection;
vhLexIndexTbl = OpenFileMapping(FILE_MAP_READ, fTrue, IMEKR_LEX_SHAREDDATA_NAME);
if(vhLexIndexTbl)
{
Dbg(DBGID_Hanja|DBGID_Mem, TEXT("CHanja::OpenLex() - File mapping already exists"));
fRet = fTrue;
}
else
{
// if no file mapping exist
vhLexIndexTbl = CreateFileMapping(INVALID_HANDLE_VALUE, GetIMESecurityAttributes(), PAGE_READWRITE,
0, sizeof(_LexIndex)*(vuNumOfHangulEntry),
IMEKR_LEX_SHAREDDATA_NAME);
if (vhLexIndexTbl)
{
Dbg(DBGID_Hanja|DBGID_Mem, TEXT("CHanja::OpenLex() - File mapping Created"));
pLexIndexTbl = (_LexIndex*)MapViewOfFile(vhLexIndexTbl, FILE_MAP_READ | FILE_MAP_WRITE, 0, 0, 0);
if (!pLexIndexTbl)
goto ExitOpenLexCritSection;
if (ReadFile(vhLex, pLexIndexTbl, vuNumOfHangulEntry*sizeof(_LexIndex), &dwReadBytes, 0) != 0 &&
dwReadBytes == vuNumOfHangulEntry*sizeof(_LexIndex))
{
fRet = fTrue;
}
else
{
fRet = fFalse;
}
UnmapViewOfFile(pLexIndexTbl);
}
#ifdef _DEBUG
else
DbgAssert(0);
#endif
}
ExitOpenLexCritSection:
ReleaseMutex(hMutex);
CloseHandle(hMutex);
}
FreeIMESecurityAttributes();
vfLexOpen = fRet;
return fRet;
}
BOOL CloseLex()
{
//ClearHanjaSenseArray();
if (vhLexIndexTbl)
{
CloseHandle(vhLexIndexTbl);
vhLexIndexTbl = 0;
}
if (vhLex)
{
CloseHandle(vhLex);
vhLex = 0;
}
vfLexOpen = fFalse;
return fTrue;
}
///////////////////////////////////////////////////////////////////////////////
BOOL GenerateHanjaCandList(PCIMECtx pImeCtx, WCHAR wcHangul)
{
WCHAR wcCandChar;
INT iMapCandStr;
UINT uNumOfCandStr;
_LexIndex *pLexIndexTbl;
WCHAR wszMeaning[MAX_SENSE_LENGTH+1];
BYTE cchMeaning = 0;
DWORD dwReadBytes;
BOOL fRet = fFalse;
Dbg(DBGID_Hanja, "GenerateHanjaCandList");
if (!EnsureHanjaLexLoaded())
{
MessageBeep(MB_ICONEXCLAMATION);
return fFalse;
}
// Get Current composition char
if (wcHangul == 0)
wcHangul = pImeCtx->GetCompBufStr();
if (wcHangul == 0)
return fFalse;
pLexIndexTbl = (_LexIndex*)MapViewOfFile(vhLexIndexTbl, FILE_MAP_READ, 0, 0, 0);
if (!pLexIndexTbl)
{
Dbg(DBGID_Hanja, TEXT("pLexIndexTbl==0"));
DbgAssert(0);
return fFalse;
}
if ((iMapCandStr = SearchHanjaIndex(wcHangul, pLexIndexTbl)) < 0)
MessageBeep(MB_ICONEXCLAMATION);
else
{
// Set member vars
vwcHangul = wcHangul;
vuNumofK0 = pLexIndexTbl[iMapCandStr].wNumOfK0;
// Is K1 Hanja enabled?
if (pImeCtx->GetGData() && pImeCtx->GetGData()->GetKSC5657Hanja() && (vpInstData->f16BitApps == fFalse) && !IsWin95())
vuNumofK1 = pLexIndexTbl[iMapCandStr].wNumOfK1;
else
vuNumofK1 = 0;
uNumOfCandStr = vuNumofK0 + vuNumofK1;
if (uNumOfCandStr == 0)
goto GenerateHanjaCandListExit;
Dbg(DBGID_Hanja, "Hangul = 0x%04X, K0=%d, K1=%d, iMapCandStr=%d", vwcHangul, vuNumofK0, vuNumofK1, iMapCandStr);
// Seek to mapping Hanja
SetFilePointer(vhLex, viBufferStart + pLexIndexTbl[iMapCandStr].iOffset, 0, FILE_BEGIN);
// Read all candidates
for (UINT i = 0; i < uNumOfCandStr; i++)
{
if (ReadFile(vhLex, &wcCandChar, sizeof(WCHAR), &dwReadBytes, 0) == 0)
goto GenerateHanjaCandListExit;
if (ReadFile(vhLex, &cchMeaning, sizeof(BYTE), &dwReadBytes, 0) == 0)
goto GenerateHanjaCandListExit;
if (wcCandChar && (cchMeaning < MAX_SENSE_LENGTH*sizeof(WCHAR)))
{
if (cchMeaning)
{
if (ReadFile(vhLex, wszMeaning, cchMeaning, &dwReadBytes, 0) == 0)
goto GenerateHanjaCandListExit;
}
wszMeaning[cchMeaning>>1] = L'\0';
Dbg(DBGID_Hanja, "Read Cand[%d], Hanja=0x%04X", i, wcCandChar);
pImeCtx->AppendCandidateStr(wcCandChar, wszMeaning);
}
}
pImeCtx->StoreCandidate();
fRet = fTrue;
}
GenerateHanjaCandListExit:
UnmapViewOfFile(pLexIndexTbl);
return fRet;
}
// For ImeConversionList.
DWORD GetConversionList(WCHAR wcReading, LPCANDIDATELIST lpCandList, DWORD dwBufLen)
{
_LexIndex *pLexIndexTbl;
INT iMaxCand;
INT i, iMapCandStr;
UINT uNumOfCandStr;
DWORD dwSize, readBytes, dwStartOfCandStr;
BYTE senseLen;
WCHAR szSense[MAX_SENSE_LENGTH];
CIMEData ImeData;
if (!EnsureHanjaLexLoaded())
return (0L);
// Calculate possible maximum candidates dwBufLen can contain.
if (dwBufLen)
{
iMaxCand = dwBufLen - sizeof(CANDIDATELIST) + sizeof(DWORD); // Minus header info.(unvariable part)
iMaxCand = iMaxCand / (sizeof(DWORD) + (sizeof(WCHAR)*2)); // DWORD: offset, WCHAR*2: 1 Character + null
}
else
iMaxCand = 0;
pLexIndexTbl = (_LexIndex*)MapViewOfFile(vhLexIndexTbl, FILE_MAP_READ, 0, 0, 0);
if (!pLexIndexTbl)
{
DbgAssert(0);
return (0L);
}
dwSize = 0;
if ((iMapCandStr = SearchHanjaIndex(wcReading, pLexIndexTbl)) < 0)
goto ConversionExit1;
else
{
vuNumofK0 = pLexIndexTbl[iMapCandStr].wNumOfK0;
if (ImeData->fKSC5657Hanja && (vpInstData->f16BitApps == fFalse) && !IsWin95())
vuNumofK1 = pLexIndexTbl[iMapCandStr].wNumOfK1;
else
vuNumofK1 = 0;
uNumOfCandStr = vuNumofK0 + vuNumofK1;
if (uNumOfCandStr == 0) // if no Hanja found
goto ConversionExit1;
dwSize = sizeof(CANDIDATELIST) + uNumOfCandStr*sizeof(DWORD)
+ uNumOfCandStr * sizeof(WCHAR) * 2;
// return required buffer size
if (dwBufLen == NULL)
goto ConversionExit1;
lpCandList->dwSize = dwSize;
lpCandList->dwStyle = IME_CAND_READ;
lpCandList->dwCount = uNumOfCandStr;
lpCandList->dwPageStart = lpCandList->dwSelection = 0;
lpCandList->dwPageSize = CAND_PAGE_SIZE;
//
SetFilePointer(vhLex, viBufferStart + pLexIndexTbl[iMapCandStr].iOffset, 0, FILE_BEGIN);
dwStartOfCandStr = sizeof(CANDIDATELIST)
+ sizeof(DWORD) * uNumOfCandStr; // for dwOffset array
for (i = 0; (i < (INT)uNumOfCandStr) && (i < iMaxCand); i++)
{
WCHAR wchHanja;
LPWSTR lpwchCand;
LPSTR lpchCand;
CHAR szCand[4] = ""; // one DBCS + one Null + one extra
lpCandList->dwOffset[i] = dwStartOfCandStr + (i<<2);
if (ReadFile(vhLex, &wchHanja, sizeof(WCHAR), &readBytes, 0) == 0)
goto ConversionExit1;
// if Unicode environment
if (vfUnicode)
{
lpwchCand = (LPWSTR)((LPSTR)lpCandList + lpCandList->dwOffset[i]);
*lpwchCand++ = wchHanja;
*lpwchCand = L'\0';
}
else
{
// Convert to ANSI
WideCharToMultiByte(CP_KOREA, 0,
&wchHanja, 1, (LPSTR)szCand, sizeof(szCand),
NULL, NULL);
lpchCand = (LPSTR)((LPSTR)lpCandList + lpCandList->dwOffset[i]);
*lpchCand++ = szCand[0];
*lpchCand++ = szCand[1];
*lpchCand = '\0';
}
// Skip meaning
if (ReadFile(vhLex, &senseLen, sizeof(BYTE), &readBytes, 0) == 0)
goto ConversionExit1;
if (senseLen < MAX_SENSE_LENGTH)
{
if (ReadFile(vhLex, szSense, senseLen, &readBytes, 0) == 0)
goto ConversionExit1;
}
}
}
// if buffer size too small to copy all conversion list info
if (i == iMaxCand && i < (INT)uNumOfCandStr)
{
lpCandList->dwSize = dwSize = (sizeof(CANDIDATELIST) - sizeof(DWORD))+ i*sizeof(DWORD) + i*sizeof(WCHAR)*2;
lpCandList->dwCount = (UINT)i;
}
ConversionExit1:
UnmapViewOfFile(pLexIndexTbl);
return (dwSize);
}
INT SearchHanjaIndex(WCHAR wHChar, _LexIndex *pLexIndexTbl)
{
int iHead = 0, iTail = vuNumOfHangulEntry-1, iMid;
while (iHead <= iTail)
{
iMid = (iHead + iTail) >> 1;
if (pLexIndexTbl[iMid].wcHangul > wHChar)
iTail = iMid - 1;
else
if (pLexIndexTbl[iMid].wcHangul < wHChar)
iHead = iMid + 1;
else
return (iMid);
}
return (-1);
}