2020-09-30 16:53:55 +02:00

2132 lines
76 KiB
C++

/**************************************************************************\
* Module Name: Winmain.cpp
*
* Copyright (C) 2000, Microsoft Corporation
*
* Korean IME 6.1 install utility
*
* History:
* 11-Dec-2000 CSLim Ported from Satori 8.1 code
\**************************************************************************/
#include "private.h"
#include <set>
#include "imkrinst.h"
#include "regtip.h"
#include "insert.h"
#include "..\tip\resource.h"
#include "..\version\verres.h"
// Safe String
#define STRSAFE_NO_DEPRECATE
#include "strsafe.h"
#define MEMALLOC(x) LocalAlloc(LMEM_FIXED, x)
#define MEMFREE(x) LocalFree(x)
// Profile reg key
#define SZTIPREG_LANGPROFILE_TEMPLATE TEXT("SOFTWARE\\Microsoft\\CTF\\TIP\\%s\\LanguageProfile\\0x00000412")
#if defined(_WIN64)
#define SZTIPREG_LANGPROFILE_TEMPLATE_WOW64 TEXT("SOFTWARE\\Wow6432Node\\Microsoft\\CTF\\TIP\\%s\\LanguageProfile\\0x00000412")
#endif
#ifdef DEBUG
CCicCriticalSectionStatic g_cs;
#endif
// IA64 IME does not support IME Pad. So we just need to take care Wow64 IME in IA64.
#if !defined(_WIN64)
// Pad Applet Registry location
#define SZPADSHARE TEXT("SOFTWARE\\Microsoft\\TIP Shared\\1.1\\IMEPad\\1042")
#define SZAPPLETCLSID TEXT("SOFTWARE\\Microsoft\\TIP Shared\\1.1\\IMEPad\\1042\\AppletCLSIDList")
#define SZAPPLETIID TEXT("SOFTWARE\\Microsoft\\TIP Shared\\1.1\\IMEPad\\1042\\AppletIIDList")
#else
#define SZPADSHARE TEXT("SOFTWARE\\Wow6432Node\\Microsoft\\TIP Shared\\1.1\\IMEPad\\1042")
#define SZAPPLETCLSID TEXT("SOFTWARE\\Wow6432Node\\Microsoft\\TIP Shared\\1.1\\IMEPad\\1042\\AppletCLSIDList")
#define SZAPPLETIID TEXT("SOFTWARE\\Wow6432Node\\Microsoft\\TIP Shared\\1.1\\IMEPad\\1042\\AppletIIDList")
#endif
/////////////////////////////////////////////////////////////////////////////
// Script run routines
BOOL CmdSetupDefaultParameters();
BOOL CmdSetVersion(LPCTSTR szParam);
BOOL CmdFileList(LPCTSTR szFormat);
BOOL CmdPreSetupCheck(LPCTSTR szParam);
BOOL CmdRenamePEFile(LPCTSTR szParam);
BOOL CmdRenameFile(LPCTSTR szParam);
BOOL CmdRegisterInterface(LPCTSTR szParam);
BOOL CmdRegisterInterfaceWow64(LPCTSTR szParam);
BOOL CmdRegisterIMEandTIP(LPCTSTR szParam);
BOOL CmdRegisterPackageVersion(void);
BOOL CmdRegisterPadOrder(void);
BOOL CmdAddToPreload(LPCTSTR szParam);
BOOL CmdPrepareMigration(LPCTSTR szParam);
BOOL CmdCreateDirectory(LPCTSTR szParam);
BOOL CmdRegisterHelpDirs();
/////////////////////////////////////////////////////////////////////////////
// Private Functions
// Utility functions
static DWORD WINAPI ProcessScriptFile();
static void RegisterIMEWithFixedHKL(LPCTSTR szHKL, LPCTSTR szIMEFileName, LPCTSTR szIMEName);
static void cdecl LogOutDCPrintf(LPCTSTR lpFmt, va_list va);
static void DebugLog(LPCTSTR szFormat, ...);
static void ErrorLog(LPCTSTR szFormat, ...);
static INT ParseEnvVar(LPTSTR szBuffer, const UINT arg_nLength);
static void TrimString(LPTSTR szString);
static BOOL fExistFile(LPCTSTR szFilePath);
static BOOL fOldIMEsExist();
static BOOL WINAPI ReplaceFileOnReboot (LPCTSTR pszExisting, LPCTSTR pszNew);
static void GetPEFileVersion(LPTSTR szFilePath, DWORD *pdwMajorVersion, DWORD *pdwMiddleVersion, DWORD *pdwMinorVersion, DWORD *pdwBuildNumber);
static BOOL ActRenameFile(LPCTSTR szSrcPath, LPCTSTR tszDstPath, DWORD dwFileAttributes);
static void RegisterTIP(LPCTSTR szTIPName);
static void RegisterTIPWow64(LPCTSTR szTIPName);
static BOOL RegisterRUNKey(LPCTSTR szParam);
static BOOL MakeSIDList();
static PSECURITY_DESCRIPTOR CreateSD();
static PSID MyCreateSid(DWORD dwSubAuthority);
// HKL Helpers
static void HKLHelpSetDefaultKeyboardLayout(HKEY hKeyHKCU, HKL hKL, BOOL fSetToDefault);
static BOOL HKLHelp412ExistInPreload(HKEY hKeyCU);
static HKL GetHKLfromHKLM(LPTSTR argszIMEFile);
static void RestoreMajorVersionRegistry();
/////////////////////////////////////////////////////////////////////////////
// Global variables
/////////////////////////////////////////////////////////////////////////////
static TCHAR g_szCurrentDirectory[MAX_PATH] = {0}; // Directory in where this EXE resides.
static TCHAR g_szScriptFile[MAX_PATH] = {0}; // File name (not full path) of script file.
static TCHAR g_szSetupProgram[MAX_PATH] = {0}; // File name (not full path) of setup program.
static TCHAR g_szSystemDirectory[MAX_PATH] = {0}; // System directory.
static TCHAR g_szErrorMessage[200] = {0}; // Global buffer for last error message.
static BOOL g_fDebugLog = FALSE; // Dump debug message when true.
static BOOL g_fErrorLog = FALSE; // Dump error message when true.
static DWORD g_dwMajorVersion = 0; // Package version of this installation.
static DWORD g_dwMiddleVersion = 0;
static DWORD g_dwMinorVersion = 0;
static DWORD g_dwBuildNumber = 0;
static std::set <FLE> g_FileList; // FileListSet. Used to store set of file paths given by "CmdFileList"
// script command.
TCHAR g_szVersionKeyCur[MAX_PATH] = {0};
BOOL g_fExistNewerVersion = FALSE;
/*---------------------------------------------------------------------------
WinMain
---------------------------------------------------------------------------*/
INT WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{
LPTSTR szHitPtr;
LPTSTR szCommandLine;
size_t cchCommandLine;
if (FAILED(StringCchLength(GetCommandLine(), MAX_PATH, &cchCommandLine)))
return (0);
szCommandLine = new TCHAR[cchCommandLine + 1];
if (szCommandLine == NULL)
return (0);
// szCommandLine contains all command line.
// Will be modified by _tcstok.
StringCchCopy(szCommandLine, cchCommandLine+1, GetCommandLine());
// TEMP Code
// LogOutDCPrintf(TEXT("WinMain CommandLine arg: %s"), szCommandLine);
CoInitialize(NULL);
szHitPtr = _tcstok(szCommandLine, TEXT("\""));
// g_szCurrentDirectory has path for the process. (IMKRINST.EXE)
StringCchCopy(g_szCurrentDirectory, ARRAYSIZE(g_szCurrentDirectory), szHitPtr);
szHitPtr = _tcsrchr(g_szCurrentDirectory, TEXT('\\'));
// if g_szCurrentDirectory contains full path,
if (szHitPtr != NULL)
{
*szHitPtr = 0; // terminate with the last '\' character to obtain current directory,
StringCchCopy(g_szSetupProgram, ARRAYSIZE(g_szSetupProgram), szHitPtr + 1); // then copy the rest (after last '\') to g_szScriptFile.
}
else
{
StringCchCopy(g_szSetupProgram, ARRAYSIZE(g_szSetupProgram), g_szCurrentDirectory); // Otherwise (g_szCurrentDirectory is not full path), copy entire
GetCurrentDirectory(MAX_PATH, g_szCurrentDirectory); // to g_szScriptFile and obtain current directory by GetCurrentDirectory.
}
StringCchCopy(g_szScriptFile, ARRAYSIZE(g_szScriptFile), g_szSetupProgram);
szHitPtr = _tcsrchr(g_szScriptFile, TEXT('.'));
if (szHitPtr != NULL) // If g_szScriptFile contains '.' character, find the last one
*szHitPtr = 0; // and terminate string with it, then concatenate ".inf" to it.
// Usually it results ".exe" -> ".inf" replacement. If g_szScriptFile
// doesn't have '.' character, just concatenate ".ini".
lstrcat(g_szScriptFile, TEXT(".ini"));
// Get system32 dir
GetSystemDirectory(g_szSystemDirectory, ARRAYSIZE(g_szSystemDirectory));
// g_szCurrentDirectory, g_szSetupProgram, g_szSystemDirectory and g_szScriptFile are prepared.
// szCommandLine will be no longer used.
// We can use these environment variable in the process.
SetEnvironmentVariable(TEXT("SETUPSOURCE"), g_szCurrentDirectory);
SetEnvironmentVariable(TEXT("SETUPEXE"), g_szSetupProgram);
SetEnvironmentVariable(TEXT("SETUPINF"), g_szScriptFile);
SetEnvironmentVariable(TEXT("SYSTEMDIR"), g_szSystemDirectory);
delete[] szCommandLine;
szCommandLine = NULL;
//////////////////////////////////////////////////////////////////////////
// Read and run Script file
switch (ProcessScriptFile())
{
case errNoError:
break;
case errPreSetupCheck:
{
#ifdef NEVER
for(set<FLE>::iterator itFLE = g_FileList.begin(); itFLE != g_FileList.end(); itFLE++){
DebugLog(TEXT("Cleanup: Deleting Source file: %s"), itFLE->szFileName);
DeleteFile(itFLE->szFileName);
}
#endif // NEVER
}
break;
default:
DebugLog(g_szErrorMessage);
}
#ifdef NEVER
for (set<FLE>::iterator itFLE = g_FileList.begin(); itFLE != g_FileList.end(); itFLE++)
{
ErrorLog(TEXT("Warning: File %s in CmdFileList will be removed without any processing"), itFLE->szFileName);
DeleteFile(itFLE->szFileName);
}
#endif // NEVER
CoUninitialize();
return(0);
}
/////////////////////////////////////////////////////////////////////////////
// Main Script Processing functions
/////////////////////////////////////////////////////////////////////////////
inline LPCTSTR GetParameter(LPTSTR szLineBuffer)
{
return(szLineBuffer + lstrlen(szLineBuffer) + 1);
}
/*---------------------------------------------------------------------------
ProcessScriptFile
Read script file. Dispatch commands for each line.
---------------------------------------------------------------------------*/
DWORD WINAPI ProcessScriptFile()
{
TCHAR szScriptFilePath[MAX_PATH];
FILE *fileScript;
wsprintf(szScriptFilePath, TEXT("%s\\%s"), g_szCurrentDirectory, g_szScriptFile);
fileScript = _tfopen(szScriptFilePath, TEXT("rt"));
if (fileScript != NULL)
{
TCHAR szLineBuffer[_cchBuffer];
LPTSTR szCommand;
// Parse command
// Command form <Command>:<Parameter>
while (_fgetts(szLineBuffer, _cchBuffer, fileScript) != NULL)
{
// Chop CR code.
szLineBuffer[lstrlen(szLineBuffer) - 1] = 0;
// Empty or Comment line
if (lstrlen(szLineBuffer) == 0 || (_tcsncmp(TEXT("//"), szLineBuffer, 2) == 0))
continue;
DebugLog(TEXT("Line: %s"), szLineBuffer);
szCommand = _tcstok(szLineBuffer, TEXT(":"));
if (szCommand == NULL)
{ // Dispatch each commands.
DebugLog(TEXT("Ignore line"));
}
else if (lstrcmpi(szCommand, TEXT("DebugLogOn")) == 0)
{
g_fDebugLog = TRUE;
g_fErrorLog = TRUE;
DebugLog(TEXT("Turn on DebugLog"));
}
else if (lstrcmpi(szCommand, TEXT("DebugLogOff")) == 0)
{
DebugLog(TEXT("Turn off DebugLog"));
g_fDebugLog = FALSE;
}
else if (lstrcmpi(szCommand, TEXT("ErrorLogOn")) == 0)
{
g_fErrorLog = TRUE;
DebugLog(TEXT("Turn on DebugLog"));
}
else if (lstrcmpi(szCommand, TEXT("ErrorLogOff")) == 0)
{
DebugLog(TEXT("Turn off DebugLog"));
g_fErrorLog = FALSE;
}
else if (lstrcmpi(szCommand, TEXT("FileList")) == 0)
{
if (!CmdFileList(GetParameter(szCommand)))
return(errFileList);
}
else if (lstrcmpi(szCommand, TEXT("SetupDefaultParameters")) == 0)
{
if (!CmdSetupDefaultParameters())
return(errSetDefaultParameters);
}
else if (lstrcmpi(szCommand, TEXT("SetVersion")) == 0)
{
if (!CmdSetVersion(GetParameter(szCommand)))
return(errSetVersion);
}
else if (lstrcmpi(szCommand, TEXT("PreSetupCheck")) == 0)
{
if (!CmdPreSetupCheck(GetParameter(szCommand)))
return(errPreSetupCheck);
}
else if (lstrcmpi(szCommand, TEXT("RenamePEFile")) == 0)
{
if (!CmdRenamePEFile(GetParameter(szCommand)))
return(errRenameFile);
}
else if (lstrcmpi(szCommand, TEXT("RenameFile")) == 0)
{
if (!CmdRenameFile(GetParameter(szCommand)))
return(errRenameFile);
}
else if (lstrcmpi(szCommand, TEXT("RegisterIMEandTIP")) == 0)
{
if (!CmdRegisterIMEandTIP(GetParameter(szCommand)))
return(errRegisterIMEandTIP);
}
else if (lstrcmpi(szCommand, TEXT("RegisterPackageVersion")) == 0)
{
if (!CmdRegisterPackageVersion())
return(errRegisterPackageVersion);
}
else if (lstrcmpi(szCommand, TEXT("RegisterInterface")) == 0)
{
if (!CmdRegisterInterface(GetParameter(szCommand)))
return(errRegisterInterface);
}
else if (lstrcmpi(szCommand, TEXT("RegisterInterfaceWow64")) == 0)
{
if (!CmdRegisterInterfaceWow64(GetParameter(szCommand)))
return(errRegisterInterfaceWow64);
}
else if (lstrcmpi(szCommand, TEXT("RegisterPadOrder")) == 0)
{
if (!CmdRegisterPadOrder())
return(errRegisterPadOrder);
}
else if (lstrcmpi(szCommand, TEXT("AddToPreload")) == 0)
{
if (!CmdAddToPreload(GetParameter(szCommand)))
return(errAddToPreload);
}
else if (lstrcmpi(szCommand, TEXT("PrepareMigration")) == 0)
{
if (!CmdPrepareMigration(GetParameter(szCommand)))
return(errPrepareMigration);
}
else if (lstrcmpi(szCommand, TEXT("CreateDirectory")) == 0)
{
if (!CmdCreateDirectory(GetParameter(szCommand)))
return(errCmdCreateDirectory);
}
else if (lstrcmpi(szCommand, TEXT("RegisterHelpDirs")) == 0)
{
if (!CmdRegisterHelpDirs())
return(errCmdRegisterHelpDirs);
}
else
DebugLog(TEXT("Ignore line"));
}
fclose(fileScript);
}
else
{
wsprintf(g_szErrorMessage, TEXT("Cannot open %s"), g_szScriptFile);
return(errNoFile);
}
return(errNoError);
}
/////////////////////////////////////////////////////////////////////////////
// Command handlers. Which are invoked from ProcessScriptFile
/////////////////////////////////////////////////////////////////////////////
/*---------------------------------------------------------------------------
CmdSetupDefaultParameters
Setup default parameters. For now, it only sets default ProductVersion value.
---------------------------------------------------------------------------*/
#define MAKE_STR(a) #a
#define MAKE_VERSTR(a, b, c, d) MAKE_STR(a.b.c.d)
#define VERRES_VERSIONSTR MAKE_VERSTR(VERRES_VERSION_MAJOR, VERRES_VERSION_MINOR, VERRES_VERSION_BUILD, VERRES_VERSION_REVISION)
BOOL CmdSetupDefaultParameters()
{
_stscanf(TEXT(VERRES_VERSIONSTR), TEXT("%d.%d.%d.%d"), &g_dwMajorVersion, &g_dwMiddleVersion, &g_dwMinorVersion, &g_dwBuildNumber);
wsprintf(g_szVersionKeyCur, "%s\\%d.%d\\%s", g_szVersionKey, g_dwMajorVersion, g_dwMiddleVersion, g_szVersion);
DebugLog(TEXT("CmdSetupDefaultParameters: Version Info : %d.%d.%d.%d"), g_dwMajorVersion, g_dwMiddleVersion, g_dwMinorVersion, g_dwBuildNumber);
return(TRUE);
}
/*---------------------------------------------------------------------------
CmdSetVersion
Overwrites default ProductVersion value with value provided in script file.
---------------------------------------------------------------------------*/
BOOL CmdSetVersion(LPCTSTR szParam)
{
int iNum = _stscanf(szParam, TEXT("%d.%d.%d.%d"), &g_dwMajorVersion, &g_dwMiddleVersion, &g_dwMinorVersion, &g_dwBuildNumber);
wsprintf(g_szVersionKeyCur, "%s\\%d.%d\\%s", g_szVersionKey, g_dwMajorVersion, g_dwMiddleVersion, g_szVersion);
if (iNum == 4)
{
DebugLog(TEXT("CmdSetVersion: Version Info : %d.%d.%d.%d"), g_dwMajorVersion, g_dwMiddleVersion, g_dwMinorVersion, g_dwBuildNumber);
return(TRUE);
}
else
{
ErrorLog(TEXT("CmdSetVersion: Failed to retrieve version number from string [%s]"), szParam);
wsprintf(g_szErrorMessage, TEXT("CmdSetVersion: Failed to retrieve version number from string [%s]"), szParam);
return(FALSE);
}
}
/*---------------------------------------------------------------------------
CmdFileList
Add file to the file list. Files in the file list will be deleted when they become no longer needed.
---------------------------------------------------------------------------*/
BOOL CmdFileList(LPCTSTR szFormat)
{
FLE flElem;
TCHAR szExpandedFileName[MAX_PATH];
flElem.fRemoved = FALSE;
StringCchCopy(flElem.szFileName, MAX_PATH, szFormat);
if (ExpandEnvironmentStrings(flElem.szFileName, szExpandedFileName, sizeof(szExpandedFileName)/sizeof(TCHAR)))
StringCchCopy(flElem.szFileName, MAX_PATH, szExpandedFileName);
// Add the element to file list set.
if (g_FileList.insert(flElem).second)
DebugLog(TEXT("Add to CmdFileList \"%s\" -> Added."), szFormat);
else
ErrorLog(TEXT("Add to CmdFileList \"%s\" -> Not added. Duplicate?"), szFormat);
return(TRUE);
}
/*---------------------------------------------------------------------------
CmdPreSetupCheck
Check whether the newer IME has been installed.
Return TRUE when we can proceed.
!!! FALSE will terminates setup. !!!
---------------------------------------------------------------------------*/
BOOL CmdPreSetupCheck(LPCTSTR szParam)
{
HKEY hKey;
TCHAR szInstalledVersionString[30];
DWORD cbInstalledVersionString = sizeof(szInstalledVersionString);
DWORD dwInstalledMajorVersion, dwInstalledMiddleVersion, dwInstalledMinorVersion, dwInstalledBuildNumber;
BOOL fResult = TRUE;
RestoreMajorVersionRegistry();
//
// root
//
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, g_szVersionKey, 0, KEY_READ, &hKey) == ERROR_SUCCESS )
{
cbInstalledVersionString = sizeof(szInstalledVersionString);
RegQueryValueEx(hKey, g_szVersion, NULL, NULL, (LPBYTE)szInstalledVersionString, &cbInstalledVersionString);
if (_stscanf(szInstalledVersionString, TEXT("%d.%d"), &dwInstalledMajorVersion, &dwInstalledMiddleVersion) == 2)
{
if (VersionComparison2(g_dwMajorVersion, g_dwMiddleVersion) < VersionComparison2(dwInstalledMajorVersion, dwInstalledMiddleVersion))
{
DebugLog(TEXT("Newer version of IME has been already installed."));
wsprintf(g_szErrorMessage, TEXT("Newer version of IME has been already installed. but, continue to setup"));
g_fExistNewerVersion = TRUE;
}
}
RegCloseKey(hKey);
}
//
// current
//
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, g_szVersionKeyCur, 0, KEY_READ, &hKey) == ERROR_SUCCESS)
{
cbInstalledVersionString = sizeof(szInstalledVersionString);
RegQueryValueEx(hKey, NULL, NULL, NULL, (LPBYTE)szInstalledVersionString, &cbInstalledVersionString);
if (_stscanf(szInstalledVersionString, TEXT("%d.%d.%d.%04d"),
&dwInstalledMajorVersion, &dwInstalledMiddleVersion, &dwInstalledMinorVersion, &dwInstalledBuildNumber) == 4)
{
if ( VersionComparison4(g_dwMajorVersion, g_dwMiddleVersion, g_dwMinorVersion, g_dwBuildNumber)
< VersionComparison4(dwInstalledMajorVersion, dwInstalledMiddleVersion, dwInstalledMinorVersion, dwInstalledBuildNumber))
{
DebugLog(TEXT("Newer version of IME has been already installed."));
wsprintf(g_szErrorMessage, TEXT("Newer version of IME has been already installed."));
fResult = FALSE;
}
}
RegCloseKey(hKey);
}
return(fResult);
}
/*---------------------------------------------------------------------------
CmdRenamePEFile
Rename file with PE format version comparison.
---------------------------------------------------------------------------*/
BOOL CmdRenamePEFile(LPCTSTR szParam)
{
TCHAR szSrc[MAX_PATH], szDst[MAX_PATH];
TCHAR szExpandedSrc[MAX_PATH], szExpandedDst[MAX_PATH];
LPTSTR szHitPtr;
DWORD dwFileAttributes = FILE_ATTRIBUTE_NORMAL;
DWORD dwSrcMajorVersion, dwSrcMiddleVersion, dwSrcMinorVersion, dwSrcBuildNumber;
DWORD dwDstMajorVersion, dwDstMiddleVersion, dwDstMinorVersion, dwDstBuildNumber;
szHitPtr = _tcschr(szParam, TEXT(','));
if (szHitPtr == NULL)
{
ErrorLog(TEXT("CmdRenamePEFile: Invalid parameters (%s)"), szParam);
wsprintf(g_szErrorMessage, TEXT("CmdRenamePEFile: Invalid parameters (%s)"), szParam);
return(FALSE);
}
*szHitPtr = 0;
StringCchCopy(szSrc, ARRAYSIZE(szSrc), szParam);
StringCchCopy(szDst, ARRAYSIZE(szDst), szHitPtr + 1); // Here, szDst may contain optional parameter.
szHitPtr = _tcschr(szDst, TEXT(','));
if (NULL != szHitPtr)
{
*szHitPtr = 0;
_stscanf(szHitPtr + 1, TEXT("%d"), &dwFileAttributes);
}
TrimString(szSrc);
TrimString(szDst);
ExpandEnvironmentStrings(szSrc, szExpandedSrc, sizeof(szExpandedSrc));
ExpandEnvironmentStrings(szDst, szExpandedDst, sizeof(szExpandedDst));
DebugLog(TEXT("CmdRenamePEFile: szExpandedSrc = %s, szExpandedDst = %s"), szExpandedSrc, szExpandedDst);
GetPEFileVersion(szExpandedSrc, &dwSrcMajorVersion, &dwSrcMiddleVersion, &dwSrcMinorVersion, &dwSrcBuildNumber);
GetPEFileVersion(szExpandedDst, &dwDstMajorVersion, &dwDstMiddleVersion, &dwDstMinorVersion, &dwDstBuildNumber);
DebugLog(TEXT("SrcVersion: (%d.%d.%d.%d), DstVersion: (%d.%d.%d.%d)"), dwSrcMajorVersion, dwSrcMiddleVersion, dwSrcMinorVersion, dwSrcBuildNumber,
dwDstMajorVersion, dwDstMiddleVersion, dwDstMinorVersion, dwDstBuildNumber);
if (VersionComparison4(0, 0, 0, 0) == VersionComparison4(dwSrcMajorVersion, dwSrcMinorVersion, dwSrcMiddleVersion, dwSrcBuildNumber))
ErrorLog(TEXT("Version of source file (%s) is 0.0.0.0. May be file not found."), szSrc);
if(VersionComparison4(dwSrcMajorVersion, dwSrcMiddleVersion, dwSrcMinorVersion, dwSrcBuildNumber) <
VersionComparison4(dwDstMajorVersion, dwDstMiddleVersion, dwDstMinorVersion, dwDstBuildNumber))
{
DebugLog(TEXT("CmdRenamePEFile: Source version is less than Destination. Copy skipped and Source will be deleted."));
if(DeleteFile(szSrc))
{
FLE fleKey;
StringCchCopy(fleKey.szFileName, MAX_PATH, szSrc);
g_FileList.erase(fleKey);
}
else
DebugLog(TEXT("CmdRenamePEFile: Failed to delete Source file (%s)"), szSrc);
}
else
{
DebugLog(TEXT("CmdRenamePEFile: Invoke ActRenameFile"));
ActRenameFile(szSrc, szDst, dwFileAttributes);
}
return(TRUE);
}
/*---------------------------------------------------------------------------
CmdRenameFile
Rename file without version comparison
---------------------------------------------------------------------------*/
BOOL CmdRenameFile(LPCTSTR szParam)
{
TCHAR szSrc[MAX_PATH], szDst[MAX_PATH];
TCHAR szExpandedSrc[MAX_PATH], szExpandedDst[MAX_PATH];
LPTSTR szHitPtr;
DWORD dwFileAttributes = FILE_ATTRIBUTE_NORMAL;
szHitPtr = _tcschr(szParam, TEXT(','));
if (szHitPtr == NULL)
{
ErrorLog(TEXT("CmdRenameFile: Invalid parameters (%s)"), szParam);
wsprintf(g_szErrorMessage, TEXT("CmdRenameFile: Invalid parameters (%s)"), szParam);
return(FALSE);
}
*szHitPtr = 0;
StringCchCopy(szSrc, ARRAYSIZE(szSrc), szParam);
StringCchCopy(szDst, ARRAYSIZE(szDst), szHitPtr + 1); // Here, szDst may contain optional parameter.
szHitPtr = _tcschr(szDst, TEXT(','));
if (szHitPtr != NULL)
{
*szHitPtr = 0;
_stscanf(szHitPtr + 1, TEXT("%d"), &dwFileAttributes);
}
TrimString(szSrc);
TrimString(szDst);
ExpandEnvironmentStrings(szSrc, szExpandedSrc, sizeof(szExpandedSrc));
ExpandEnvironmentStrings(szDst, szExpandedDst, sizeof(szExpandedDst));
DebugLog(TEXT("RanemeFile: szExpandedSrc = %s, szExpandedDst = %s"), szExpandedSrc, szExpandedDst);
ActRenameFile(szExpandedSrc, szExpandedDst, dwFileAttributes);
return(TRUE);
}
/*---------------------------------------------------------------------------
RegisterIMEWithFixedHKL
---------------------------------------------------------------------------*/
void RegisterIMEWithFixedHKL(LPCTSTR szHKL, LPCTSTR szIMEFileName, LPCTSTR szIMEName)
{
HKEY hKeyKbdLayout;
HKEY hKeyOneIME;
if (RegOpenKey(HKEY_LOCAL_MACHINE,
TEXT("SYSTEM\\CurrentControlSet\\Control\\Keyboard Layouts"),
&hKeyKbdLayout) != ERROR_SUCCESS)
return;
if (RegCreateKey(hKeyKbdLayout,
szHKL,
&hKeyOneIME) != ERROR_SUCCESS)
{
RegCloseKey(hKeyKbdLayout);
return;
}
if (RegSetValueEx(hKeyOneIME,
TEXT("Ime File"),
0,
REG_SZ,
(CONST BYTE*)szIMEFileName,
(lstrlen(szIMEFileName) + 1) * sizeof(TCHAR)) != ERROR_SUCCESS)
goto WriteImeLayoutFail;
if (RegSetValueEx(hKeyOneIME,
TEXT("Layout Text"),
0,
REG_SZ,
(CONST BYTE*)szIMEName,
(lstrlen(szIMEName) + 1) * sizeof(TCHAR)) != ERROR_SUCCESS)
goto WriteImeLayoutFail;
WriteImeLayoutFail:
RegCloseKey(hKeyOneIME);
RegCloseKey(hKeyKbdLayout);
}
/*---------------------------------------------------------------------------
CmdRegisterInterface
Invoke SelfReg. If given file is DLL, call DllRegisterServer export function. If given file is EXE, run it with "/RegServer"
command line option.
---------------------------------------------------------------------------*/
typedef HRESULT (STDAPICALLTYPE *pfnDllRegisterServerType)(void);
BOOL CmdRegisterInterface(LPCTSTR szParam)
{
TCHAR szExpandedModulePath[MAX_PATH];
HRESULT hr = S_FALSE;
ExpandEnvironmentStrings(szParam, szExpandedModulePath, sizeof(szExpandedModulePath));
TrimString(szExpandedModulePath);
INT iLen = 0;
iLen = lstrlen(szExpandedModulePath);
if (iLen < 4)
{
ErrorLog(TEXT("CmdRegisterInterface: Too short szExpandedModulePath (%s)"), szExpandedModulePath);
wsprintf(g_szErrorMessage, TEXT("CmdRegisterInterface: Invalid Parameters."));
return(FALSE);
}
if (lstrcmpi(TEXT(".dll"), &szExpandedModulePath[iLen - 4]) == 0)
{
DebugLog(TEXT("CmdRegisterInterface: DLL mode for %s"), szExpandedModulePath);
HINSTANCE hLib = LoadLibrary(szExpandedModulePath);
if (hLib != NULL)
{
pfnDllRegisterServerType pfnDllRegisterServer = (pfnDllRegisterServerType)GetProcAddress(hLib, "DllRegisterServer");
if (pfnDllRegisterServer != NULL)
{
hr = pfnDllRegisterServer();
ErrorLog(TEXT("CmdRegisterInterface: hResult=%x"), hr);
}
FreeLibrary(hLib);
}
}
else
{
if (lstrcmpi(TEXT(".exe"), &szExpandedModulePath[iLen - 4]) == 0)
{
DebugLog(TEXT("CmdRegisterInterface: EXE mode for %s"), szExpandedModulePath);
TCHAR szCommandBuffer[MAX_PATH * 2];
wsprintf(szCommandBuffer, TEXT("%s /RegServer"), szExpandedModulePath);
STARTUPINFO si;
ZeroMemory(&si, sizeof(si));
si.cb = sizeof(si);
si.dwFlags = STARTF_USESHOWWINDOW;
si.wShowWindow = SW_HIDE;
PROCESS_INFORMATION pi;
ZeroMemory(&pi, sizeof(pi));
if (CreateProcess(NULL, szCommandBuffer, NULL, NULL, FALSE, CREATE_DEFAULT_ERROR_MODE, NULL, NULL, &si, &pi))
{
DebugLog(TEXT("CmdRegisterInterface: Running \"%s\". Waiting for the process termination."), szCommandBuffer);
WaitForSingleObject(pi.hProcess, INFINITE);
DebugLog(TEXT("CmdRegisterInterface: \"%s\" terminates."), szCommandBuffer);
CloseHandle(pi.hThread);
CloseHandle(pi.hProcess);
hr = S_OK;
}
else
{
DWORD dwError = GetLastError();
ErrorLog(TEXT("CmdRegisterInterface: CreateProcess(\"%s\") failed with error code = %d(%x)"), szCommandBuffer, dwError, dwError);
}
}
else
ErrorLog(TEXT("Cannot detect module type for %s. Skipped."), szExpandedModulePath);
}
return(SUCCEEDED(hr));
}
/*---------------------------------------------------------------------------
CmdRegisterInterfaceWow64
Invoke SelfReg. If given file is DLL, call DllRegisterServer export function. If given file is EXE, run it with "/RegServer"
command line option.
---------------------------------------------------------------------------*/
BOOL CmdRegisterInterfaceWow64(LPCTSTR szParam)
{
#if defined(_WIN64)
TCHAR szExpandedModulePath[MAX_PATH];
TCHAR szCommandBuffer[MAX_PATH * 2];
STARTUPINFO si;
PROCESS_INFORMATION pi;
ExpandEnvironmentStrings(szParam, szExpandedModulePath, sizeof(szExpandedModulePath));
TrimString(szExpandedModulePath);
INT iLen = 0;
iLen = lstrlen(szExpandedModulePath);
if (iLen < 4)
{
ErrorLog(TEXT("CmdRegisterInterfaceWow64: Too short szExpandedModulePath (%s)"), szExpandedModulePath);
wsprintf(g_szErrorMessage, TEXT("CmdRegisterInterface: Invalid Parameters."));
return(FALSE);
}
if (lstrcmpi(TEXT(".dll"), &szExpandedModulePath[iLen - 4]) == 0)
{
// First get systemWow64Directory
TCHAR szSysWow64Dir[MAX_PATH] = TEXT("");
HMODULE hmod = GetModuleHandle(TEXT("kernel32.dll"));
DebugLog(TEXT("CmdRegisterInterfaceWow64: DLL mode for %s"), szExpandedModulePath);
if (hmod == NULL)
{
DebugLog(TEXT("CmdRegisterInterfaceWow64: Faile to load kernel32.dll"));
return (TRUE);
}
UINT (WINAPI* pfnGetSystemWow64Directory)(LPTSTR, UINT);
(FARPROC&)pfnGetSystemWow64Directory = GetProcAddress (hmod, "GetSystemWow64DirectoryA");
if (pfnGetSystemWow64Directory == NULL)
{
DebugLog(TEXT("CmdRegisterInterfaceWow64: Faile to load GetSystemWow64DirectoryA API"));
return (TRUE);
}
/*
* if GetSystemWow64Directory fails and sets the last error to
* ERROR_CALL_NOT_IMPLEMENTED, we're on a 32-bit OS
*/
if (((pfnGetSystemWow64Directory)(szSysWow64Dir, ARRAYSIZE(szSysWow64Dir)) == 0) &&
(GetLastError() == ERROR_CALL_NOT_IMPLEMENTED))
{
DebugLog(TEXT("CmdRegisterInterfaceWow64: Failed to load GetSystemWow64DirectoryA API"));
return (TRUE);
}
wsprintf(szCommandBuffer, TEXT("%s\\regsvr32.exe \"%s\" /s"), szSysWow64Dir, szExpandedModulePath);
}
else
if (lstrcmpi(TEXT(".exe"), &szExpandedModulePath[iLen - 4]) == 0)
{
DebugLog(TEXT("CmdRegisterInterfaceWow64: EXE mode for %s"), szExpandedModulePath);
wsprintf(szCommandBuffer, TEXT("\"%s\" /RegServer"), szExpandedModulePath);
}
else
{
ErrorLog(TEXT("Cannot detect module type for %s. Skipped."), szExpandedModulePath);
// Return true not to stop further processing.
return (TRUE);
}
ZeroMemory(&si, sizeof(si));
si.cb = sizeof(si);
si.dwFlags = STARTF_USESHOWWINDOW;
si.wShowWindow = SW_HIDE;
ZeroMemory(&pi, sizeof(pi));
if (CreateProcess(NULL, szCommandBuffer, NULL, NULL, FALSE, CREATE_DEFAULT_ERROR_MODE, NULL, NULL, &si, &pi))
{
DebugLog(TEXT("CmdRegisterInterfaceWow64: Running \"%s\". Waiting for the process termination."), szCommandBuffer);
WaitForSingleObject(pi.hProcess, INFINITE);
DebugLog(TEXT("CmdRegisterInterfaceWow64: \"%s\" terminates."), szCommandBuffer);
CloseHandle(pi.hThread);
CloseHandle(pi.hProcess);
}
else
{
DWORD dwError = GetLastError();
ErrorLog(TEXT("CmdRegisterInterfaceWow64: CreateProcess(\"%s\") failed with error code = %d(%x)"), szCommandBuffer, dwError, dwError);
}
#endif
return(TRUE);
}
/*---------------------------------------------------------------------------
CmdRegisterIMEandTIP
Register IME using IMM API and TIP
---------------------------------------------------------------------------*/
BOOL CmdRegisterIMEandTIP(LPCTSTR szParam)
{
TCHAR szIMEFileName[MAX_PATH], szTIPName[MAX_PATH], szTIPNameWow64[MAX_PATH];
TCHAR *szHitPtr;
TCHAR *szHitPtr2;
HKL hIME61KL = 0;
TCHAR szHKL[10];
TCHAR szNonFullPath[MAX_PATH];
HKEY hKey;
szHitPtr = _tcschr(szParam, TEXT(','));
szHitPtr2 = _tcschr(szHitPtr + 1, TEXT(',')); // because there are three parameters
if (szHitPtr2 == NULL)
{
ErrorLog(TEXT("CmdRegisterIMEandTIP: Invalid parameters (%s)"), szParam);
wsprintf(g_szErrorMessage, TEXT("CmdRegisterIMEandTIP: Invalid parameters (%s)"), szParam);
return(FALSE);
}
*szHitPtr = 0;
*szHitPtr2 = 0;
StringCchCopy(szIMEFileName, ARRAYSIZE(szIMEFileName), szParam);
StringCchCopy(szTIPName, ARRAYSIZE(szTIPName), szHitPtr + 1);
StringCchCopy(szTIPNameWow64, ARRAYSIZE(szTIPNameWow64), szHitPtr2 + 1);
TrimString(szIMEFileName);
TrimString(szTIPName);
TrimString(szTIPNameWow64);
ParseEnvVar(szIMEFileName, MAX_PATH);
ParseEnvVar(szTIPName, MAX_PATH);
ParseEnvVar(szTIPNameWow64, MAX_PATH);
DebugLog(TEXT("CmdRegisterIMEandTIP: IMEFileName = %s, TIPFileName = %s szTIPNameWow64 = %s"), szIMEFileName, szTIPName, szTIPNameWow64);
/////////////////////////////////////////////////////////////////////////////
// IME registration
if ((szHitPtr = _tcsrchr(szIMEFileName, TEXT('\\'))) != NULL)
szHitPtr++;
else
szHitPtr = szIMEFileName;
StringCchCopy(szNonFullPath, ARRAYSIZE(szNonFullPath), szHitPtr);
hIME61KL = GetHKLfromHKLM(szNonFullPath);
if (hIME61KL == (HKL)0)
DebugLog(TEXT("CmdRegisterIMEandTIP: hIME61KL is zero %x -- error"), hIME61KL);
//if (hKL && HKLHelp412ExistInPreload(HKEY_CURRENT_USER))
// {
// HKLHelpSetDefaultKeyboardLayout(HKEY_CURRENT_USER, hKL, FALSE);
//hKL = ImmInstallIME(szIMEFileName, szLayoutText);
// }
/////////////////////////////////////////////////////////////////////////////
// TIP registration
// Regster wow64 TIP first to avoid regstry overwrite problem.
RegisterTIPWow64(szTIPNameWow64);
RegisterTIP(szTIPName);
/////////////////////////////////////////////////////////////////////////////
// IME and TIP - make substitution
TCHAR szTIPGuid[MAX_PATH];
TCHAR szLangProfile[MAX_PATH];
CLSIDToStringA(CLSID_KorIMX, szTIPGuid);
DebugLog(TEXT("CmdRegisterIMEandTIP: CLSID_KorIMX guid=%s"), szTIPGuid);
// make a reg key
wsprintf(szLangProfile, SZTIPREG_LANGPROFILE_TEMPLATE, szTIPGuid);
/////////////////////////////////////////////////////////////////////////////
// Add Substitute HKL value to the TIP registry
if (hIME61KL != 0 && RegOpenKeyEx(HKEY_LOCAL_MACHINE, szLangProfile, 0, KEY_ALL_ACCESS, &hKey) == ERROR_SUCCESS)
{
TCHAR szSubKeyName[MAX_PATH], szHKL[MAX_PATH];
DWORD dwIndex;
HKEY hSubKey;
wsprintf(szHKL, TEXT("0x%x"), hIME61KL);
dwIndex = 0;
while (RegEnumKey(hKey, dwIndex, szSubKeyName, MAX_PATH) != ERROR_NO_MORE_ITEMS)
{
if (RegOpenKeyEx(hKey,szSubKeyName,0,KEY_ALL_ACCESS, &hSubKey) == ERROR_SUCCESS)
{
RegSetValueEx(hSubKey, TEXT("SubstituteLayout"), 0,REG_SZ,(BYTE *)szHKL, sizeof(TCHAR)*(lstrlen(szHKL)+1));
RegCloseKey(hSubKey);
}
dwIndex++;
}
RegCloseKey(hKey);
}
#if defined(_WIN64)
// make a reg key
wsprintf(szLangProfile, SZTIPREG_LANGPROFILE_TEMPLATE_WOW64, szTIPGuid);
/////////////////////////////////////////////////////////////////////////////
// Add Substitute HKL value to the TIP registry
if (hIME61KL != 0 && RegOpenKeyEx(HKEY_LOCAL_MACHINE, szLangProfile, 0, KEY_ALL_ACCESS, &hKey) == ERROR_SUCCESS)
{
TCHAR szSubKeyName[MAX_PATH], szHKL[MAX_PATH];
DWORD dwIndex;
HKEY hSubKey;
wsprintf(szHKL, TEXT("0x%x"), hIME61KL);
dwIndex = 0;
while (RegEnumKey(hKey, dwIndex, szSubKeyName, MAX_PATH) != ERROR_NO_MORE_ITEMS)
{
if (RegOpenKeyEx(hKey, szSubKeyName, 0, KEY_ALL_ACCESS, &hSubKey) == ERROR_SUCCESS)
{
RegSetValueEx(hSubKey, TEXT("SubstituteLayout"), 0, REG_SZ, (BYTE*)szHKL, sizeof(TCHAR)*(lstrlen(szHKL)+1));
RegCloseKey(hSubKey);
}
dwIndex++;
}
RegCloseKey(hKey);
}
#endif
return(TRUE);
}
/*---------------------------------------------------------------------------
CmdRegisterRUNKey
Register package version
---------------------------------------------------------------------------*/
BOOL CmdRegisterPackageVersion(void)
{
HKEY hKey;
TCHAR szVersionString[30];
// Write RootVersion reg only if this is latest IME.
if (g_fExistNewerVersion == FALSE)
{
if(ERROR_SUCCESS == RegCreateKeyEx(HKEY_LOCAL_MACHINE, g_szVersionKey, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKey, NULL))
{
wsprintf(szVersionString, TEXT("%d.%d"), g_dwMajorVersion, g_dwMiddleVersion);
RegSetValueEx(hKey, g_szVersion, 0, REG_SZ, (CONST BYTE *)szVersionString, (lstrlen(szVersionString) + 1) * sizeof(TCHAR));
RegCloseKey(hKey);
}
}
// Current
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, g_szVersionKeyCur, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKey, NULL) == ERROR_SUCCESS)
{
wsprintf(szVersionString, TEXT("%d.%d.%d.%d"), g_dwMajorVersion, g_dwMiddleVersion, g_dwMinorVersion, g_dwBuildNumber);
RegSetValueEx(hKey, NULL, 0, REG_SZ, (CONST BYTE *)szVersionString, (lstrlen(szVersionString) + 1) * sizeof(TCHAR));
RegCloseKey(hKey);
}
return(TRUE);
}
//
// Register Applet order
//
#define FE_KOREAN // need Korean stuff
#include "../fecommon/imembx/guids.h"
typedef
struct tagAPPLETCLSID
{
const GUID *pguidClsid;
BOOL fNoUIM;
} APPLETCLSID;
typedef
struct tagAPPLETIID
{
const GUID *pguidIID;
} APPLETIID;
/*---------------------------------------------------------------------------
CmdRegisterPadOrder
Not support WOW64.
---------------------------------------------------------------------------*/
BOOL CmdRegisterPadOrder(void)
{
HKEY hKey;
TCHAR szClsid[MAX_PATH];
TCHAR szKey[MAX_PATH];
static const APPLETCLSID appletClsid[] =
{
{ &CLSID_ImePadApplet_MultiBox, FALSE },
{0},
};
static const APPLETIID appletIID[] =
{
{ &IID_MultiBox },
{0},
};
//
// Applet clsid
//
for (INT i = 0; appletClsid[i].pguidClsid; i++)
{
CLSIDToStringA(*appletClsid[i].pguidClsid, szClsid);
wsprintf(szKey, TEXT("%s\\%s"), SZAPPLETCLSID, szClsid);
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, szKey, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKey, NULL) == ERROR_SUCCESS)
{
if(appletClsid[i].fNoUIM)
{
DWORD dw = 1;
RegSetValueEx(hKey, TEXT("nouim"), 0, NULL, (BYTE*)&dw, sizeof(DWORD));
}
RegCloseKey(hKey);
}
}
//
// Applet iid
//
TCHAR szSubKey[MAX_PATH];
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, SZAPPLETIID, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKey, NULL) == ERROR_SUCCESS)
{
for (INT i = 0; appletIID[i].pguidIID; i++)
{
CLSIDToStringA(*appletIID[i].pguidIID, szKey);
wsprintf(szSubKey, TEXT("%d"), i);
RegSetValueEx(hKey, szSubKey, 0, REG_SZ, (BYTE*)szKey, (lstrlen(szKey)+1)*sizeof(TCHAR));
}
RegCloseKey(hKey);
}
return(TRUE);
}
/*---------------------------------------------------------------------------
CmdCreateDirectory
---------------------------------------------------------------------------*/
BOOL CmdCreateDirectory(LPCTSTR szParam)
{
TCHAR szDirectory[MAX_PATH], szExpandedDirectory[MAX_PATH];
StringCchCopy(szDirectory, ARRAYSIZE(szDirectory), szParam);
TrimString(szDirectory);
ExpandEnvironmentStrings(szDirectory, szExpandedDirectory, sizeof(szExpandedDirectory)/sizeof(TCHAR));
CreateDirectory(szExpandedDirectory, NULL);
return(TRUE);
}
/*---------------------------------------------------------------------------
CmdAddToPreload
Add HKL for given IMEFile to current user's preload. The HKL won't become default IME.
---------------------------------------------------------------------------*/
BOOL CmdAddToPreload(LPCTSTR szParam)
{
TCHAR tszIMEFileName[MAX_PATH];
HKL hKL;
// If there is no Kor IME exist in preload, we shouldn't add Kor IME.
if (!HKLHelp412ExistInPreload(HKEY_CURRENT_USER))
{
DebugLog(TEXT("CmdAddToPreload: No 0412 HKL exist in HKCU\\Preload"));
return TRUE;
}
StringCchCopy(tszIMEFileName, ARRAYSIZE(tszIMEFileName), szParam);
TrimString(tszIMEFileName);
hKL = GetHKLfromHKLM(tszIMEFileName);
DebugLog(TEXT("CmdAddToPreload: Calling SetDefaultKeyboardLayout(HKEY_CURRENT_USER, %x, FALSE)"), hKL);
HKLHelpSetDefaultKeyboardLayout(HKEY_CURRENT_USER, hKL, FALSE);
return(TRUE);
}
/*---------------------------------------------------------------------------
fOldIMEsExist
Register Run regi onl if old IME exist in system.
---------------------------------------------------------------------------*/
static BOOL fOldIMEsExist()
{
HKL hKL;
static LPCSTR m_szOldIMEs[] =
{
"msime95.ime", // Win 95 IME
"msime95k.ime", // NT 4 IME
"imekr98u.ime", // IME98
"imekr.ime", // Office 10 IME
""
};
CHAR** ppch = (CHAR**)&m_szOldIMEs[0];
while (ppch && **ppch)
{
hKL = GetHKLfromHKLM(*ppch);
if (hKL)
return TRUE; // existing
ppch++;
}
return FALSE;
}
/*---------------------------------------------------------------------------
DisableTIP60ByDefault
---------------------------------------------------------------------------*/
VOID DisableTIP60ByDefault()
{
// KorIMX CLSID
// {766A2C14-B226-4fd6-B52A-867B3EBF38D2}
const static CLSID CLSID_KorTIP60 =
{
0x766A2C14,
0xB226,
0x4FD6,
{0xb5, 0x2a, 0x86, 0x7b, 0x3e, 0xbf, 0x38, 0xd2}
};
const static GUID guidProfile60 =
// {E47ABB1E-46AC-45f3-8A89-34F9D706DA83}
{
0xe47abb1e,
0x46ac,
0x45f3,
{0x8a, 0x89, 0x34, 0xf9, 0xd7, 0x6, 0xda, 0x83}
};
// Set default Tip as for Cicero.
CoInitialize(NULL);
ITfInputProcessorProfiles *pProfile;
HRESULT hr = CoCreateInstance(CLSID_TF_InputProcessorProfiles, NULL, CLSCTX_INPROC_SERVER,
IID_ITfInputProcessorProfiles, (void **) &pProfile);
if (SUCCEEDED(hr))
{
pProfile->EnableLanguageProfileByDefault(CLSID_KorTIP60,
MAKELANGID(LANG_KOREAN, SUBLANG_DEFAULT), guidProfile60, FALSE);
pProfile->Release();
}
else
{
OurEnableLanguageProfileByDefault(CLSID_KorTIP60, MAKELANGID(LANG_KOREAN, SUBLANG_DEFAULT), guidProfile60, FALSE);
}
CoUninitialize();
}
/*---------------------------------------------------------------------------
CmdPrepareMigration
---------------------------------------------------------------------------*/
BOOL CmdPrepareMigration(LPCTSTR szParam)
{
// Disable TIP 6.0 from HKLM by default
// This will handle Office 10 install after Whistler mig exe removed from run reg.
DisableTIP60ByDefault();
// First user SID list
if (MakeSIDList() == FALSE)
return FALSE;
//Register IMEKRMIG.EXE to run reg Key on "Software\Microsoft\Windows\CurrentVersion\Run"
return RegisterRUNKey(szParam);
}
/*---------------------------------------------------------------------------
CmdRegisterHelpDirs
---------------------------------------------------------------------------*/
BOOL CmdRegisterHelpDirs()
{
TCHAR szFileNameFullPath[MAX_PATH], szFileName[MAX_PATH];
LPTSTR szExtension, szFileNamePtr;
HKEY hKey;
for (std::set<FLE>::iterator itFLE = g_FileList.begin(); itFLE != g_FileList.end(); itFLE++)
{
StringCchCopy(szFileNameFullPath, ARRAYSIZE(szFileNameFullPath), itFLE->szFileName);
szExtension = _tcsrchr(szFileNameFullPath, TEXT('.'));
if (szExtension == NULL)
continue;
if (lstrcmpi(szExtension, TEXT(".CHM")) == 0)
{
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, TEXT("SOFTWARE\\Microsoft\\Windows\\HTML Help"), 0, NULL,
REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKey, NULL) == ERROR_SUCCESS)
{
szFileNamePtr = _tcsrchr(szFileNameFullPath, TEXT('\\'));
// Get file name
StringCchCopy(szFileName, ARRAYSIZE(szFileName), szFileNamePtr+1);
// Get rid of file name we only need path.
*(szFileNamePtr+1) = 0;
RegSetValueEx(hKey, szFileName, 0, REG_SZ, (LPBYTE)szFileNameFullPath, (lstrlen(szFileNameFullPath)+1)*sizeof(TCHAR));
}
}
else
if (lstrcmpi(szExtension, TEXT(".HLP")) == 0)
{
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, TEXT("SOFTWARE\\Microsoft\\Windows\\Help"), 0, NULL,
REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hKey, NULL) == ERROR_SUCCESS)
{
szFileNamePtr = _tcsrchr(szFileNameFullPath, TEXT('\\'));
// Get file name
StringCchCopy(szFileName, ARRAYSIZE(szFileName), szFileNamePtr+1);
// Get rid of file name we only need path.
*(szFileNamePtr+1) = 0;
RegSetValueEx(hKey, szFileName, 0, REG_SZ, (LPBYTE)szFileNameFullPath, (lstrlen(szFileNameFullPath)+1)*sizeof(TCHAR));
}
}
}
return(TRUE);
}
/////////////////////////////////////////////////////////////////////////////
// Private functions
/////////////////////////////////////////////////////////////////////////////
//
// Debug output routine.
//
void cdecl LogOutDCPrintf(LPCTSTR lpFmt, va_list va)
{
static INT DCLine = 0;
HDC hDC = GetDC((HWND)0);
TCHAR sz[512];
HANDLE hFile;
DWORD dwWrite;
wvsprintf(sz, lpFmt, va );
lstrcat(sz, TEXT("| "));
TextOut(hDC, 0, DCLine*16, sz, lstrlen(sz));
if (DCLine++ > 50)
DCLine = 0;
hFile = CreateFile(TEXT("\\IMKRINST.LOG"), GENERIC_WRITE, FILE_SHARE_READ, 0, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0);
if (hFile != INVALID_HANDLE_VALUE)
{
SetFilePointer(hFile, 0, NULL, FILE_END);
WriteFile(hFile, sz, lstrlen(sz), &dwWrite, NULL);
WriteFile(hFile, TEXT("\r\n"), 2, &dwWrite, NULL);
CloseHandle(hFile);
}
ReleaseDC((HWND)0, hDC);
}
void DebugLog(LPCTSTR szFormat, ...)
{
va_list va;
va_start(va, szFormat);
if (g_fDebugLog)
LogOutDCPrintf(szFormat, va);
va_end(va);
}
void ErrorLog(LPCTSTR szFormat, ...)
{
va_list va;
va_start(va, szFormat);
if (g_fErrorLog)
LogOutDCPrintf(szFormat, va);
va_end(va);
}
/*---------------------------------------------------------------------------
ParseEnvVar
Evaluate environment variable. Modiry given string.
---------------------------------------------------------------------------*/
INT ParseEnvVar(LPTSTR szBuffer, const UINT arg_nLength)
{
INT iTranslated=0, i, j;
TCHAR *pLParen, *pRParen, *pStart = szBuffer;
INT nLength = min(arg_nLength, MAX_PATH);
static TCHAR szInternalBuffer[MAX_PATH*2], szValue[MAX_PATH];
szInternalBuffer[0] = 0;
for (i=0; i<nLength; i++)
{
if (szBuffer[i] == 0)
break;
if (szBuffer[i] == '%')
{
pLParen = &(szBuffer[i]);
pRParen = NULL;
for (j=1; i+j<nLength; j++)
{
if (szBuffer[i+j] == 0)
break;
if (szBuffer[i+j] == TEXT('%'))
{
pRParen = &(szBuffer[i+j]);
break;
}
}
if (pRParen)
{
*pLParen = 0;
*pRParen = 0;
lstrcat(szInternalBuffer, pStart);
if (GetEnvironmentVariable(pLParen+1, szValue, sizeof(szValue)/sizeof(TCHAR)) == 0)
{
lstrcat(szInternalBuffer, TEXT("%"));
lstrcat(szInternalBuffer, pLParen+1);
lstrcat(szInternalBuffer, TEXT("%"));
}
else
{
lstrcat(szInternalBuffer, szValue);
iTranslated++;
}
pStart = pRParen+1;
i += j;
}
}
}
if (iTranslated)
{
lstrcat(szInternalBuffer, pStart);
lstrcpyn(szBuffer, szInternalBuffer, arg_nLength);
}
return(iTranslated);
}
/*---------------------------------------------------------------------------
TrimString
Chop head/tail white space from given string. Given string will be modified.
---------------------------------------------------------------------------*/
void TrimString(LPTSTR szString)
{
INT iBuffSize = lstrlen(szString) + 1;
LPTSTR szBuffer = new TCHAR[iBuffSize];
if (szBuffer != NULL)
{
INT iHeadSpace, iTailSpace;
StringCchCopy(szBuffer, MAX_PATH, szString);
iHeadSpace = (INT)_tcsspn(szBuffer, TEXT(" \t"));
_tcsrev(szBuffer);
iTailSpace = (INT)_tcsspn(szBuffer, TEXT(" \t"));
_tcsrev(szBuffer);
szBuffer[lstrlen(szBuffer) - iTailSpace] = 0;
StringCchCopy(szString, MAX_PATH, szBuffer + iHeadSpace);
}
if (szBuffer != NULL)
{
delete[] szBuffer;
szBuffer = NULL;
}
}
/*---------------------------------------------------------------------------
fExistFile
---------------------------------------------------------------------------*/
BOOL fExistFile(LPCTSTR szFilePath)
{
BOOL fResult = TRUE;
if (GetFileAttributes(szFilePath) == -1)
fResult = FALSE;
return(fResult);
}
/*---------------------------------------------------------------------------
ReplaceFileOnReboot
Writes wininit.ini rename section. Note that this function writes lines in reverse order (down to upper).
---------------------------------------------------------------------------*/
BOOL WINAPI ReplaceFileOnReboot(LPCTSTR pszExisting, LPCTSTR pszNew)
{
if (MoveFileEx(pszExisting, pszNew, MOVEFILE_DELAY_UNTIL_REBOOT))
return TRUE;
else
return FALSE;
}
/*---------------------------------------------------------------------------
GetPEFileVersion
Get version information from PE format.
---------------------------------------------------------------------------*/
void GetPEFileVersion(LPTSTR szFilePath, DWORD *pdwMajorVersion, DWORD *pdwMiddleVersion, DWORD *pdwMinorVersion, DWORD *pdwBuildNumber)
{
*pdwMajorVersion = *pdwMiddleVersion = *pdwMinorVersion = *pdwBuildNumber = 0;
DWORD dwDummy, dwVerResSize;
dwVerResSize = GetFileVersionInfoSize(szFilePath, &dwDummy);
if (dwVerResSize)
{
BYTE *pbData = new BYTE[dwVerResSize];
if (NULL != pbData)
{
if(GetFileVersionInfo(szFilePath, 0, dwVerResSize, pbData))
{
VS_FIXEDFILEINFO *pffiVersion;
UINT cbffiVersion;
if(VerQueryValue(pbData, TEXT("\\"), (LPVOID *)&pffiVersion, &cbffiVersion))
{
*pdwMajorVersion = HIWORD(pffiVersion->dwFileVersionMS);
*pdwMiddleVersion = LOWORD(pffiVersion->dwFileVersionMS);
*pdwMinorVersion = HIWORD(pffiVersion->dwFileVersionLS);
*pdwBuildNumber = LOWORD(pffiVersion->dwFileVersionLS);
}
}
}
if (NULL != pbData)
{
delete[] pbData;
pbData = NULL;
}
}
}
/*---------------------------------------------------------------------------
ActRenameFile
MoveFile. If destination file exists, it will be overwritten. If existing destination file cannot be
overwritten in this session, file replacement is reserved to be held after rebooting.
---------------------------------------------------------------------------*/
BOOL ActRenameFile(LPCTSTR szSrcPath, LPCTSTR tszDstPath, DWORD dwFileAttributes)
{
BOOL fReplaceAfterReboot = FALSE;
BOOL fResult = TRUE;
FLE fleKey;
StringCchCopy(fleKey.szFileName, MAX_PATH, szSrcPath);
if (g_FileList.end() == g_FileList.find(fleKey))
ErrorLog(TEXT("ActRenameFile: WARNING: Cannot find source file [%s] in CmdFileList"), szSrcPath);
if (!fExistFile(szSrcPath))
{
ErrorLog(TEXT("ActRenameFile: Source file [%s] doesn't exist."), szSrcPath);
wsprintf(g_szErrorMessage, TEXT("ActRenameFile: Source file [%s] doesn't exist."), szSrcPath);
return(FALSE);
}
if (fExistFile(tszDstPath))
{
SetFileAttributes(tszDstPath, FILE_ATTRIBUTE_NORMAL);
if(!DeleteFile(tszDstPath))
{
DWORD dwError = GetLastError();
fReplaceAfterReboot = TRUE;
DebugLog(TEXT("ActRenameFile: Cannot delete destination file [%s] with error code = %d(%x)"), tszDstPath, dwError, dwError);
}
}
if (!fReplaceAfterReboot)
{
if(MoveFile(szSrcPath, tszDstPath))
{
SetFileAttributes(szSrcPath, dwFileAttributes);
DebugLog(TEXT("ActRenameFile: MoveFile(%s, %s) succeeded."), szSrcPath, tszDstPath);
}
else
{
DWORD dwError = GetLastError();
DebugLog(TEXT("ActRenameFile: MoveFile(%s, %s) failed with error code = %d(%x)."), szSrcPath, tszDstPath, dwError, dwError);
DebugLog(TEXT("ActRenameFile: Try again with fReplaceAfterReboot."));
fReplaceAfterReboot = TRUE;
}
}
if (fReplaceAfterReboot)
{
SetFileAttributes(szSrcPath, dwFileAttributes); // In this case, change file attributes for Src path.
ReplaceFileOnReboot(szSrcPath, tszDstPath); // Since this function writes lines in reverse order, deletion of
DebugLog(TEXT("ActRenameFile: ReplaceFileOnReboot(%s, %s)."), szSrcPath, tszDstPath);
ReplaceFileOnReboot(tszDstPath, NULL); // tszDstPath will come first.
DebugLog(TEXT("ActRenameFile: ReplaceFileOnReboot(%s, NULL)."), tszDstPath);
}
if (fResult)
g_FileList.erase(fleKey);
return(fResult);
}
///////////////////////////////////////////////////////////////////////////////
// This should sync with SERVER.CPP in TIP folder
// TIP Categories to be added
const REGISTERCAT c_rgRegCat[] =
{
{&GUID_TFCAT_DISPLAYATTRIBUTEPROVIDER, &CLSID_KorIMX},
{&GUID_TFCAT_TIP_KEYBOARD, &CLSID_KorIMX},
{&GUID_TFCAT_PROPSTYLE_CUSTOM, &GUID_PROP_OVERTYPE},
{NULL, NULL}
};
// TIP Profile name
const REGTIPLANGPROFILE c_rgProf[] =
{
{ MAKELANGID(LANG_KOREAN, SUBLANG_DEFAULT), &GUID_Profile, SZ_TIPDISPNAME, SZ_TIPMODULENAME, (IDI_UNIKOR-IDI_ICONBASE), IDS_PROFILEDESC },
{0, &GUID_NULL, L"", L"", 0, 0}
};
//
///////////////////////////////////////////////////////////////////////////////
/*---------------------------------------------------------------------------
RegisterTIP
Write neccessary registry key and values for TIP
---------------------------------------------------------------------------*/
void RegisterTIP(LPCTSTR szTIPName)
{
HKEY hKey;
TCHAR szTIPGuid[MAX_PATH];
TCHAR szTIPProfileGuid[MAX_PATH];
TCHAR szSubKey[MAX_PATH];
DWORD dwValue;
DebugLog(TEXT("RegisterTIP: (%s)."), szTIPName);
// Run self reg
// If self reg fails, run custom TIP registration
if (!CmdRegisterInterface(szTIPName))
{
TCHAR szExpandedTIPPath[MAX_PATH];
DebugLog(TEXT("RegisterTIP: TIP self reg failed, Run custom reg"));
// Expand Env var
ExpandEnvironmentStrings(szTIPName, szExpandedTIPPath, sizeof(szExpandedTIPPath));
// Register TIP CLSID
if (!RegisterServer(CLSID_KorIMX, SZ_TIPSERVERNAME, szExpandedTIPPath, TEXT("Apartment"), NULL))
{
DebugLog(TEXT("RegisterTIP: RegisterServer failed"));
return;
}
if (!OurRegisterTIP(szExpandedTIPPath, CLSID_KorIMX, SZ_TIPNAME, c_rgProf))
{
DebugLog(TEXT("RegisterTIP: szExpandedTIPPath failed"));
return;
}
if (FAILED(OurRegisterCategories(CLSID_KorIMX, c_rgRegCat)))
{
DebugLog(TEXT("RegisterTIP: OurRegisterCategories failed"));
return;
}
}
// Get String format GUIDs
CLSIDToStringA(CLSID_KorIMX, szTIPGuid);
CLSIDToStringA(GUID_Profile, szTIPProfileGuid);
/////////////////////////////////////////////////////////////////////////////
// If no Kor IME is in .default user.
// Set HKLM [HKLM\Software\Microsoft\CTF\TIP\TIP classid\LanguageProfile\Language ID\Guid Profile]
// "Enable" = "0" (DWORD)
if (RegOpenKeyEx(HKEY_USERS, TEXT(".DEFAULT"), 0, KEY_ALL_ACCESS, &hKey) == ERROR_SUCCESS)
{
if (!HKLHelp412ExistInPreload(hKey))
{
HKEY hKProfRegKey;
// Create "Software\Microsoft\CTF\TIP\{CLSID_KorIMX}\LanguageProfile\0x00000412\{CLSID_INPUTPROFILE}"
wsprintf(szSubKey, TEXT("%s%s\\LanguageProfile\\0x00000412\\%s"), TEXT("SOFTWARE\\Microsoft\\CTF\\TIP\\"), szTIPGuid, szTIPProfileGuid);
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, szSubKey, 0, KEY_ALL_ACCESS, &hKProfRegKey) == ERROR_SUCCESS)
{
// Enabled
DebugLog(TEXT("RegisterTIP: IME HKL not exist in HKU\\.Default disable TIP"));
dwValue= 0;
RegSetValueEx(hKProfRegKey, TEXT("Enable"), 0, REG_DWORD, (BYTE*)&dwValue, sizeof(dwValue));
RegCloseKey(hKProfRegKey);
}
}
RegCloseKey(hKey);
}
}
/*---------------------------------------------------------------------------
RegisterTIPWow64
Write neccessary registry key and values for TIP
---------------------------------------------------------------------------*/
void RegisterTIPWow64(LPCTSTR szTIPName)
{
#if defined(_WIN64)
// Run just selfreg. Cicero doesn't use "HKLM\Software\Wow6432Node\Microsoft\CTF\TIP\"
CmdRegisterInterfaceWow64(szTIPName);
#endif
}
////////////////////////////////////////////////////////////////////////////
// HKL Helper functions
////////////////////////////////////////////////////////////////////////////
/*---------------------------------------------------------------------------
GetHKLfromHKLM
---------------------------------------------------------------------------*/
HKL GetHKLfromHKLM(LPTSTR argszIMEFile)
{
HKL hklAnswer = 0;
HKEY hKey, hSubKey;
DWORD i, cbSubKey, cbIMEFile;
TCHAR szSubKey[MAX_PATH], szIMEFile[MAX_PATH];
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, TEXT("System\\CurrentControlSet\\Control\\Keyboard Layouts"), 0, KEY_READ, &hKey) == ERROR_SUCCESS)
{
for (i=0; ;i++)
{
cbSubKey = MAX_PATH;
if (RegEnumKeyEx(hKey, i, szSubKey, &cbSubKey, NULL, NULL, NULL, NULL) == ERROR_NO_MORE_ITEMS)
break;
RegOpenKeyEx(hKey, szSubKey, 0, KEY_READ, &hSubKey);
cbIMEFile=MAX_PATH;
if (RegQueryValueEx(hSubKey, TEXT("IME File"), NULL, NULL, (LPBYTE)szIMEFile, &cbIMEFile) == ERROR_SUCCESS)
{
if (lstrcmpi(argszIMEFile, szIMEFile) == 0)
{
RegCloseKey(hSubKey);
_stscanf(szSubKey, TEXT("%08x"), &hklAnswer);
break;
}
}
RegCloseKey(hSubKey);
}
RegCloseKey(hKey);
}
return(hklAnswer);
}
/*---------------------------------------------------------------------------
HKLHelpSetDefaultKeyboardLayout
---------------------------------------------------------------------------*/
void HKLHelpSetDefaultKeyboardLayout(HKEY hKeyHKCU, HKL hKL, BOOL fSetToDefault)
{
TCHAR szKL[20];
BYTE Data[MAX_PATH];
DWORD cbData;
TCHAR szSubKey[MAX_PATH];
HKEY hKey,hSubKey;
DWORD NumKL;
wsprintf(szKL, TEXT("%08x"), hKL);
RegOpenKeyEx(hKeyHKCU, TEXT("Keyboard Layout\\Preload"), 0, KEY_ALL_ACCESS, &hKey);
RegQueryInfoKey(hKey, NULL, NULL, NULL, NULL, NULL, NULL, &NumKL, NULL, NULL, NULL, NULL);
for (DWORD i=0; i<NumKL; i++)
{
wsprintf(szSubKey, TEXT("%d"), i+1);
cbData = MAX_PATH;
RegQueryValueEx(hKey, szSubKey, NULL, NULL, Data, &cbData);
if (lstrcmpi((LPCTSTR)Data, szKL) == 0)
break;
}
// if hKL is not exist create it.
if (NumKL == i)
{
wsprintf(szSubKey, TEXT("%d"), i+1);
RegSetValueEx(hKey, szSubKey, 0, REG_SZ, (const LPBYTE)szKL, (lstrlen(szKL)+1)*sizeof(TCHAR));
NumKL++;
}
// Set hKL as first, Shift down other.
if(fSetToDefault)
{
for(int j=i; j>0; j--)
{
wsprintf(szSubKey, TEXT("%d"),j);
cbData = MAX_PATH;
RegQueryValueEx(hKey, szSubKey, NULL, NULL, Data, &cbData);
wsprintf(szSubKey, TEXT("%d"),j+1);
RegSetValueEx(hKey, szSubKey, 0, REG_SZ, Data, cbData);
}
RegSetValueEx(hKey, TEXT("1"), 0, REG_SZ, (const LPBYTE)szKL, (lstrlen(szKL)+1)*sizeof(TCHAR));
}
RegCloseKey(hKey);
(void)LoadKeyboardLayout(szKL, KLF_ACTIVATE);
// To activate IME2002 right now without reboot.
if(fSetToDefault)
(void)SystemParametersInfo(SPI_SETDEFAULTINPUTLANG, 0, &hKL, SPIF_SENDCHANGE);
}
#define MAX_NAME 100
/*---------------------------------------------------------------------------
HKLHelp412ExistInPreload
---------------------------------------------------------------------------*/
BOOL HKLHelp412ExistInPreload(HKEY hKeyCU)
{
HKEY hKey, hSubKey;
int i ,j;
DWORD cbName, cbData;
CHAR szName[MAX_NAME];
CHAR szData[MAX_NAME];
HKL hkl;
FILETIME ftLastWriteTime;
BOOL fResult = FALSE;
if (RegOpenKeyEx(hKeyCU, "Keyboard Layout\\Preload", 0, KEY_ALL_ACCESS, &hKey) == ERROR_SUCCESS)
{
for (j=0; cbName=MAX_NAME, cbData=MAX_NAME, RegEnumValue(hKey, j, szName, &cbName, NULL, NULL, (LPBYTE)szData, &cbData) != ERROR_NO_MORE_ITEMS; j++)
{
// See If Korean KL exist. Just compare last LCID part if it's 0x412.
// IME hkl set 0xE000 on hiword.
sscanf(szData, "%08x", &hkl);
if ((HIWORD(hkl) & 0xe000) && LOWORD(hkl) == 0x0412)
{
fResult = TRUE;
break;
}
}
RegCloseKey(hKey);
}
return(fResult);
}
/*---------------------------------------------------------------------------
RegisterRUNKey
Register IME using IMM API and TIP
---------------------------------------------------------------------------*/
BOOL RegisterRUNKey(LPCTSTR szParam)
{
TCHAR szKey[MAX_PATH];
TCHAR szFilename[MAX_PATH];
TCHAR *szHitPtr;
HKEY hRunKey;
szHitPtr = _tcschr(szParam, TEXT(','));
if (szHitPtr == NULL)
{
ErrorLog(TEXT("RegisterRUNKey: Invalid parameters (%s)"), szParam);
wsprintf(g_szErrorMessage, TEXT("RegisterRUNKey: Invalid parameters (%s)"), szParam);
return(FALSE);
}
*szHitPtr = 0;
StringCchCopy(szKey, ARRAYSIZE(szKey), szParam);
StringCchCopy(szFilename, ARRAYSIZE(szFilename), szHitPtr + 1);
TrimString(szKey);
TrimString(szFilename);
ParseEnvVar(szKey, MAX_PATH);
ParseEnvVar(szFilename, MAX_PATH);
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Run"), 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hRunKey, NULL) == ERROR_SUCCESS)
{
RegSetValueEx(hRunKey, szKey, 0, REG_SZ, (CONST BYTE *)szFilename, (lstrlen(szFilename)+1)*sizeof(TCHAR));
RegCloseKey(hRunKey);
}
return(TRUE);
}
/*---------------------------------------------------------------------------
MakeSIDList
Gets all users' SID and list that in the reg for migration
---------------------------------------------------------------------------*/
BOOL MakeSIDList()
{
HKEY hKey, hUserList;
DWORD i, cbName;
BOOL fNoMoreSID = FALSE;
TCHAR szMigRegKey[MAX_PATH], szName[500];
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, TEXT("SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\ProfileList"), 0, KEY_READ, &hKey) ==ERROR_SUCCESS)
{
StringCchCopy(szMigRegKey, ARRAYSIZE(g_szIMERootKey), g_szIMERootKey);
lstrcat(szMigRegKey, TEXT("\\MigrateUser"));
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, szMigRegKey, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hUserList, NULL) == ERROR_SUCCESS)
{
for (i=0; !fNoMoreSID; i++)
{
cbName = 500;
if (RegEnumKeyEx(hKey, i, szName, &cbName, NULL, NULL, NULL, NULL) == ERROR_NO_MORE_ITEMS)
fNoMoreSID = TRUE;
else
{
// Do not add Local Service and Network Service pid
if (lstrlen(szName) > 8)
RegSetValueEx(hUserList, szName, 0, REG_SZ, (BYTE *)TEXT(""), sizeof(TCHAR)*2);
}
}
//Change MigrateUser List security settings
PSECURITY_DESCRIPTOR pSD = CreateSD();
if (pSD)
{
RegSetKeySecurity(hUserList, DACL_SECURITY_INFORMATION, pSD);
MEMFREE(pSD);
}
RegCloseKey(hUserList);
}
RegCloseKey(hKey);
}
return (TRUE);
}
/*---------------------------------------------------------------------------
RestoreMajorVersionRegistry
Restore IME major version reg value.
It could be overwritten during Win9x to NT upgrade.
---------------------------------------------------------------------------*/
void RestoreMajorVersionRegistry()
{
HKEY hKey;
///////////////////////////////////////////////////////////////////////////
// Restore IME major version reg value.
// It could be overwritten during Win9x to NT upgrading.
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, g_szVersionKey, 0, KEY_ALL_ACCESS, &hKey) == ERROR_SUCCESS)
{
CHAR szVersion[MAX_PATH];
DWORD cbVersion = MAX_PATH;
CHAR szMaxVersion[MAX_PATH];
FILETIME time;
float flVersion, flMaxVersion;
StringCchCopy(szMaxVersion, ARRAYSIZE(szMaxVersion), "0");
for (int i=0; cbVersion = MAX_PATH, RegEnumKeyEx(hKey, i, szVersion, &cbVersion, NULL, NULL, NULL, &time) != ERROR_NO_MORE_ITEMS; i++)
{
if (lstrcmp(szVersion, szMaxVersion) > 0)
StringCchCopy(szMaxVersion, ARRAYSIZE(szMaxVersion), szVersion);
}
StringCchCopy(szVersion, ARRAYSIZE(szVersion), "0");
RegQueryValueEx(hKey, g_szVersion, NULL, NULL, (BYTE *)szVersion, &cbVersion);
flVersion = (float)atof(szVersion);
flMaxVersion = (float)atof(szMaxVersion);
if (flVersion < flMaxVersion)
RegSetValueEx(hKey, g_szVersion, 0, REG_SZ, (BYTE *)szMaxVersion, (sizeof(CHAR)*lstrlen(szMaxVersion)));
RegCloseKey(hKey);
}
///////////////////////////////////////////////////////////////////////////
}
/*---------------------------------------------------------------------------
CreateSecurityAttributes
---------------------------------------------------------------------------*/
PSECURITY_DESCRIPTOR CreateSD()
{
PSECURITY_DESCRIPTOR psd;
PACL pacl;
ULONG AclSize;
PSID psid1, psid2, psid3, psid4;
BOOL fResult;
psid1 = MyCreateSid(SECURITY_INTERACTIVE_RID);
if (psid1 == NULL)
return NULL;
psid2 = MyCreateSid(SECURITY_LOCAL_SYSTEM_RID);
if (psid2 == NULL)
goto Fail4;
psid3 = MyCreateSid(SECURITY_SERVICE_RID);
if (psid3 == NULL)
goto Fail3;
psid4 = MyCreateSid(SECURITY_NETWORK_RID);
if (psid4 == NULL)
goto Fail2;
//
// allocate and initialize an access control list (ACL) that will
// contain the SIDs we've just created.
//
AclSize = sizeof(ACL) +
(4 * (sizeof(ACCESS_ALLOWED_ACE) - sizeof(ULONG))) +
GetLengthSid(psid1) +
GetLengthSid(psid2) +
GetLengthSid(psid3) +
GetLengthSid(psid4);
//
// allocate and initialize a new security descriptor plus ACL
//
psd = MEMALLOC(SECURITY_DESCRIPTOR_MIN_LENGTH + AclSize);
if (psd == NULL)
{
return NULL;
}
pacl = (PACL)((LPBYTE)psd + SECURITY_DESCRIPTOR_MIN_LENGTH);
fResult = InitializeAcl(pacl, AclSize, ACL_REVISION);
if (!fResult)
{
goto Fail;
}
//
// adds an access-allowed ACE for interactive users to the ACL
//
fResult = AddAccessAllowedAce(pacl,
ACL_REVISION,
GENERIC_ALL,
psid1);
if (!fResult)
{
goto Fail;
}
//
// adds an access-allowed ACE for operating system to the ACL
//
fResult = AddAccessAllowedAce(pacl,
ACL_REVISION,
GENERIC_ALL,
psid2);
if (!fResult)
{
goto Fail;
}
//
// adds an access-allowed ACE for operating system to the ACL
//
fResult = AddAccessAllowedAce(pacl,
ACL_REVISION,
GENERIC_ALL,
psid3);
if (!fResult)
{
goto Fail;
}
//
// adds an access-allowed ACE for operating system to the ACL
//
fResult = AddAccessAllowedAce(pacl,
ACL_REVISION,
GENERIC_ALL,
psid4);
if (!fResult)
{
goto Fail;
}
//
// Let's make sure that our ACL is valid.
//
if (!IsValidAcl(pacl))
{
goto Fail;
}
if (!InitializeSecurityDescriptor(psd, SECURITY_DESCRIPTOR_REVISION))
{
goto Fail;
}
fResult = SetSecurityDescriptorDacl(psd, fTrue, pacl, fFalse );
// The discretionary ACL is referenced by, not copied
// into, the security descriptor. We shouldn't free up ACL
// after the SetSecurityDescriptorDacl call.
if (!fResult)
{
goto Fail;
}
if (!IsValidSecurityDescriptor(psd))
{
goto Fail;
}
//
// Those SIDs have been copied into the ACL. We don't need'em any more.
//
FreeSid(psid1);
FreeSid(psid2);
FreeSid(psid3);
FreeSid(psid4);
return psd;
Fail:
MEMFREE(psd);
FreeSid(psid4);
Fail2:
FreeSid(psid3);
Fail3:
FreeSid(psid2);
Fail4:
FreeSid(psid1);
return NULL;
}
PSID MyCreateSid(DWORD dwSubAuthority)
{
PSID psid;
BOOL fResult;
SID_IDENTIFIER_AUTHORITY SidAuthority = SECURITY_NT_AUTHORITY;
//
// allocate and initialize an SID
//
fResult = AllocateAndInitializeSid(&SidAuthority,
1,
dwSubAuthority,
0,0,0,0,0,0,0,
&psid );
if (!fResult)
{
return NULL;
}
if (!IsValidSid(psid))
{
FreeSid(psid);
return NULL;
}
return psid;
}