2020-09-30 17:12:32 +02:00

869 lines
26 KiB
C++

// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1996 - 1997.
// File: LoadWC.cpp
// Contents: exe to load webcheck
// History: 12-12/96 rayen (Raymond Endres) Created
#include <windows.h>
#include <docobj.h>
#include <webcheck.h>
#include <shlguid.h>
#include <shlobj.h>
#include <shellp.h>
// Channels are enabled for the IE4 upgrades.
#define ENABLE_CHANNELS
#ifdef DEBUG
#define ASSERT(b) if (!(b)) DebugBreak();
#else
#define ASSERT(b)
#endif
#define MLUI_INIT
#include <mluisupp.h>
// NOTE: ActiveSetup relies on our window name and class name
// to shut us down properly in softboot. Do not change it.
const TCHAR c_szClassName[] = TEXT("LoadWC");
const TCHAR c_szWebCheckWindow[] = TEXT("MS_WebcheckMonitor");
const TCHAR c_szWebcheckKey[] = TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Webcheck");
// message send to dynamically start sens/lce (must not conflict with dialmon)
#define WM_START_SENSLCE (WM_USER+200)
// only used in debug code to grovel with shell service object
#ifdef DEBUG
const TCHAR c_szWebCheck[] = TEXT("WebCheck");
const TCHAR c_szShellReg[] = TEXT("SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\ShellServiceObjectDelayLoad");
#endif
typedef struct {
HINSTANCE hInstance; // handle to current instance
HWND hwnd; // main window handle
int nCmdShow; // hidden or not?
HINSTANCE hWebcheck; // handle to webcheck dll
BOOL fUninstallOnly; // TRUE -> run uninstall stubs only, then quit
BOOL fIntShellMode; // TRUE -> integrated shell mode, else browser-only
BOOL fStartSensLce;
} GLOBALS;
GLOBALS g;
// webcheck function we dynaload
typedef HRESULT (WINAPI *PFNSTART)(BOOL fForceExternals);
typedef HRESULT (WINAPI *PFNSTOP)(void);
// Code to run install/uninstall stubs, from shell\inc.
#define HINST_THISDLL g.hInstance
#include "resource.h"
//#include <stubsup.h>
#include <inststub.h>
int WINAPI WinMainT(HINSTANCE, HINSTANCE, LPSTR, int);
LRESULT APIENTRY WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam);
void vLoadWebCheck(void);
void vUnloadWebCheck(void);
BOOL bParseCommandLine(LPSTR lpCmdLine, int nCmdShow);
// ModuleEntry
extern "C" int _stdcall ModuleEntry(void)
{
int i;
STARTUPINFOA si;
LPTSTR pszCmdLine;
pszCmdLine = GetCommandLine();
// g_hProcessHeap = GetProcessHeap();
// We don't want the "No disk in drive X:" requesters, so we set
// the critical error mask such that calls will just silently fail
SetErrorMode(SEM_FAILCRITICALERRORS);
if ( *pszCmdLine == TEXT('\"') ) {
/*
* Scan, and skip over, subsequent characters until
* another double-quote or a null is encountered.
*/
while ( *++pszCmdLine && (*pszCmdLine
!= TEXT('\"')) );
/*
* If we stopped on a double-quote (usual case), skip
* over it.
*/
if ( *pszCmdLine == TEXT('\"') )
pszCmdLine++;
}
else {
while (*pszCmdLine > TEXT(' '))
pszCmdLine++;
}
/*
* Skip past any white space preceeding the second token.
*/
while (*pszCmdLine && (*pszCmdLine <= TEXT(' '))) {
pszCmdLine++;
}
si.dwFlags = 0;
GetStartupInfoA(&si);
i = WinMainT(GetModuleHandle(NULL), NULL, pszCmdLine,
si.dwFlags & STARTF_USESHOWWINDOW ? si.wShowWindow : SW_SHOWDEFAULT);
// Since we now have a way for an extension to tell us when it is finished,
// we will terminate all processes when the main thread goes away.
ExitProcess(i);
// DebugMsg(DM_TRACE, TEXT("c.me: Cabinet main thread exiting without ExitProcess."));
return i;
}
// Registry helper function
BOOL ReadRegValue(HKEY hkeyRoot, const TCHAR *pszKey, const TCHAR *pszValue,
void *pData, DWORD dwBytes)
{
long lResult;
HKEY hkey;
DWORD dwType;
lResult = RegOpenKey(hkeyRoot, pszKey, &hkey);
if (lResult != ERROR_SUCCESS) {
return FALSE;
}
lResult = RegQueryValueEx(hkey, pszValue, NULL, &dwType, (BYTE *)pData,
&dwBytes);
RegCloseKey(hkey);
if (lResult != ERROR_SUCCESS)
return FALSE;
return TRUE;
}
BOOL WriteRegValue(HKEY hkeyRoot, const TCHAR *pszKey, const TCHAR *pszValue,
DWORD dwType, void *pData, DWORD dwBytes)
{
HKEY hkey;
long lResult = RegOpenKey(hkeyRoot, pszKey, &hkey);
if (ERROR_SUCCESS == lResult)
{
lResult = RegSetValueEx(hkey, pszValue, 0, dwType, (BYTE *)pData, dwBytes);
RegCloseKey(hkey);
}
return ERROR_SUCCESS == lResult;
}
void MakeWindowsRootPath(LPSTR pszBuffer)
{
LPSTR pszEnd = NULL;
if (*pszBuffer == '\\' && *(pszBuffer+1) == '\\') {
pszEnd = pszBuffer + 2;
while (*pszEnd && (*pszEnd != '\\'))
pszEnd++;
if (*pszEnd) {
pszEnd++;
while (*pszEnd && (*pszEnd != '\\'))
pszEnd++;
if (*pszEnd)
pszEnd++;
}
}
else {
LPSTR pszNext = CharNext(pszBuffer);
if (*pszNext == ':' && *(pszNext+1) == '\\')
pszEnd = pszNext + 2;
}
if (pszEnd != NULL)
*pszEnd = '\0';
else {
/* ??? Windows dir is neither UNC nor a root path?
* Just make sure it ends in a backslash.
*/
LPSTR pszLast = pszBuffer;
if (*pszBuffer) {
pszLast = CharPrev(pszBuffer, pszBuffer + lstrlen(pszBuffer));
}
if (*pszLast != '\\')
lstrcat(pszLast, "\\");
}
}
// InitShellFolders
// More of making loadwc.exe a browser-only mode catch-all. This code makes
// sure that the Shell Folders key in the per-user registry is fully populated
// with the absolute paths to all the special folders for IE, even if
// shell32.dll doesn't understand all of them.
// As the shell would do in SHGetSpecialFolderLocation, we check the
// User Shell Folders key for a path, and if that's present we copy it to the
// Shell Folders key, expanding %USERPROFILE% if necessary. If the value is
// not present under User Shell Folders, we generate the default location
// (usually under the Windows directory) and store that location under
// Shell Folders.
struct FolderDescriptor {
UINT idsDirName; /* Resource ID for directory name */
LPCTSTR pszRegValue; /* Name of reg value to set path in */
BOOL fDefaultInRoot : 1; /* TRUE if default location is root directory */
BOOL fWriteToUSF : 1; /* TRUE if we should write to User Shell Folders to work around Win95 bug */
} aFolders[] = {
{ IDS_CSIDL_PERSONAL_L, TEXT("Personal"), TRUE, TRUE } ,
{ IDS_CSIDL_FAVORITES_L, TEXT("Favorites"), FALSE, TRUE },
{ IDS_CSIDL_APPDATA_L, TEXT("AppData"), FALSE, FALSE },
{ IDS_CSIDL_CACHE_L, TEXT("Cache"), FALSE, FALSE },
{ IDS_CSIDL_COOKIES_L, TEXT("Cookies"), FALSE, FALSE },
{ IDS_CSIDL_HISTORY_L, TEXT("History"), FALSE, FALSE },
};
void InitShellFolders(void)
{
LONG err;
HKEY hkeySF = NULL;
HKEY hkeyUSF = NULL;
TCHAR szDefaultDir[MAX_PATH];
TCHAR szRootDir[MAX_PATH+1]; // possible extra '\'
LPSTR pszPathEnd;
LPSTR pszRootEnd;
/* Get the windows directory and simulate PathAddBackslash (which we
* can't get out of shlwapi, because loadwc.exe also needs to be able
* to load after IE has been uninstalled and shlwapi deleted).
* Also build the root directory of the drive that the Windows directory
* is on, so we can put My Documents there if necessary.
*/
GetWindowsDirectory(szDefaultDir, ARRAYSIZE(szDefaultDir));
lstrcpy(szRootDir, szDefaultDir);
MakeWindowsRootPath(szRootDir);
pszRootEnd = szRootDir + lstrlen(szRootDir);
pszPathEnd = CharPrev(szDefaultDir, szDefaultDir + lstrlen(szDefaultDir));
if (*pszPathEnd != '\\') {
pszPathEnd = CharNext(pszPathEnd);
*(pszPathEnd++) = '\\';
}
// pszPathEnd now points to where we can append the relative path
UINT cchPathSpace = ARRAYSIZE(szDefaultDir) - (UINT)(pszPathEnd - szDefaultDir);
UINT cchRootSpace = ARRAYSIZE(szRootDir) - (UINT)(pszRootEnd - szRootDir);
err = RegOpenKeyEx(HKEY_CURRENT_USER, "Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Shell Folders", 0, KEY_QUERY_VALUE | KEY_SET_VALUE, &hkeySF);
if (err == ERROR_SUCCESS) {
err = RegOpenKeyEx(HKEY_CURRENT_USER, "Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\User Shell Folders", 0, KEY_QUERY_VALUE, &hkeyUSF);
if (err == ERROR_SUCCESS) {
for (UINT i=0; i<ARRAYSIZE(aFolders); i++) {
TCHAR szRawFolderName[MAX_PATH];
TCHAR szExpandedFolderName[MAX_PATH];
DWORD dwType;
DWORD cbData = sizeof(szRawFolderName);
LPSTR pszPath;
err = RegQueryValueEx(hkeyUSF, aFolders[i].pszRegValue, NULL, &dwType, (LPBYTE)szRawFolderName, &cbData);
if (err == ERROR_SUCCESS) {
if ((dwType != REG_EXPAND_SZ && dwType != REG_SZ) || !ExpandEnvironmentStrings(szRawFolderName, szExpandedFolderName, ARRAYSIZE(szExpandedFolderName)))
{
continue;
}
pszPath = szExpandedFolderName;
}
else {
if (!MLLoadString(aFolders[i].idsDirName, aFolders[i].fDefaultInRoot ? pszRootEnd : pszPathEnd, aFolders[i].fDefaultInRoot ? cchRootSpace : cchPathSpace)) {
continue;
}
if (aFolders[i].fDefaultInRoot)
pszPath = szRootDir;
else
pszPath = szDefaultDir;
if (GetFileAttributes(pszPath) == 0xffffffff)
CreateDirectory(pszPath, NULL);
/* The Win95 shell has a bug where for some shell folders,
* if there isn't a path recorded under User Shell Folders,
* the shell folder is assumed not to exist. So, for those
* folders only, we also write the default path to USF.
* We do not do this generically because no value under
* USF is supposed to mean "use the one in the Windows
* directory", whereas an absolute path means "use that
* path"; if there's a path under USF, it will be used
* literally, which is a problem if the folder is set up
* to use the shared folder location but roams to a machine
* with Windows installed in a different directory.
*/
if (aFolders[i].fWriteToUSF) {
RegSetValueEx(hkeyUSF, aFolders[i].pszRegValue, 0, REG_SZ, (LPBYTE)pszPath, lstrlen(pszPath)+1);
}
}
RegSetValueEx(hkeySF, aFolders[i].pszRegValue, 0, REG_SZ, (LPBYTE)pszPath, lstrlen(pszPath)+1);
}
RegCloseKey(hkeyUSF);
}
RegCloseKey(hkeySF);
}
}
/* Function to determine whether we're in integrated-shell mode or browser-only
* mode. The method for doing this (looking for DllGetVersion exported from
* shell32.dll) is taken from shdocvw. We don't actually call that entrypoint,
* we just look for it.
*/
BOOL IsIntegratedShellMode()
{
FARPROC pfnDllGetVersion = NULL;
HMODULE hmodShell = LoadLibrary("shell32.dll");
if (hmodShell != NULL) {
pfnDllGetVersion = GetProcAddress(hmodShell, "DllGetVersion");
FreeLibrary(hmodShell);
}
return (pfnDllGetVersion != NULL);
}
// Convert the string to a DWORD.
DWORD StringToDW(LPCTSTR psz)
{
DWORD dwRet = 0;
while (*psz >= TEXT('0') && *psz <= TEXT('9'))
{
dwRet = dwRet * 10 + *psz - TEXT('0');
*psz++;
}
return dwRet;
}
// Is the version string from IE4
BOOL IsVersionIE4(LPCTSTR pszVersion)
{
BOOL fRet = FALSE;
// IE3.0 is 4.70 and Ie4.0x is >= 4.71.1218.xxxx
if (pszVersion[0] == TEXT('4') && pszVersion[1] == TEXT('.'))
{
DWORD dw = StringToDW(pszVersion+2);
if (dw > 71 || (dw == 71 && pszVersion[4] == TEXT('.') &&
StringToDW(pszVersion+5) >= 1218))
{
fRet = TRUE;
}
}
return fRet;
}
// Determine if this is an IE4 upgrade.
BOOL IsIE4Upgrade()
{
BOOL fRet = FALSE;
TCHAR szVersion[MAX_PATH];
if (ReadRegValue(HKEY_LOCAL_MACHINE,
TEXT("Software\\Microsoft\\IE Setup\\Setup"),
TEXT("PreviousIESysFile"), (void *)szVersion,
sizeof(szVersion)))
{
fRet = IsVersionIE4(szVersion);
}
return fRet;
}
// This is an IE5 or later function. If the user's machine is running IE4 they
// have uninstalled back to IE4.
BOOL IsUninstallToIE4()
{
BOOL fRet = FALSE;
TCHAR szVersion[MAX_PATH];
if (ReadRegValue(HKEY_LOCAL_MACHINE,
TEXT("Software\\Microsoft\\Internet Explorer"),
TEXT("Version"), (void *)szVersion,
sizeof(szVersion)))
{
fRet = IsVersionIE4(szVersion);
}
return fRet;
}
/* Function to launch miscellaneous applications for browser only mode.
* We run IEXPLORE.EXE -channelband, looking in the registry for the path
* to IEXPLORE, and WELCOME.EXE /f, located in the same directory.
*/
#ifdef ENABLE_CHANNELS
const TCHAR c_szChanBarRegPath[] = TEXT("Software\\Microsoft\\Internet Explorer\\Main");
const TCHAR c_szChanBarKey[] = TEXT("Show_ChannelBand");
#endif
void LaunchBrowserOnlyApps()
{
TCHAR szPath[MAX_PATH];
/* Don't launch any of these guys if this is a "redist mode" install,
* i.e. if a game or something installed browser components silently
* without the user really realizing it's there.
*/
if (ReadRegValue(HKEY_LOCAL_MACHINE, "Software\\Microsoft\\IE Setup\\Setup",
"InstallMode", (void *)szPath, sizeof(szPath))
&& !lstrcmp(szPath, "R")) {
return;
}
LPTSTR pszPathEnd;
LONG cbPath = sizeof(szPath);
/* Get the default value from the App Paths\IEXPLORE.EXE reg key, which
* is the absolute path to the EXE.
*/
if (RegQueryValue(HKEY_LOCAL_MACHINE, "Software\\Microsoft\\Windows\\CurrentVersion\\App Paths\\IEXPLORE.EXE",
szPath, &cbPath) != ERROR_SUCCESS) {
pszPathEnd = szPath;
lstrcpy(szPath, "IEXPLORE.EXE"); /* can't get from reg, hope it's on the path */
}
else {
/* Find the last backslash in the path. This is a manual
* version of strrchr(szPath, '\\').
*/
LPTSTR pszLastBackslash = NULL;
for (pszPathEnd = szPath; *pszPathEnd; pszPathEnd = CharNext(pszPathEnd)) {
if (*pszPathEnd == '\\')
pszLastBackslash = pszPathEnd;
}
if (pszLastBackslash == NULL)
pszPathEnd = szPath;
else
pszPathEnd = pszLastBackslash + 1; /* point after last backslash */
}
#ifdef ENABLE_CHANNELS
/* Don't launch the channel band app if the user doesn't want it.
* They want it if the reg value is missing, or if it's "yes".
* On WinNT, we default to "no" for browser-only installs.
*/
TCHAR szValue[20];
BOOL fShowChannelBand=FALSE;
if (ReadRegValue(HKEY_CURRENT_USER, c_szChanBarRegPath,
c_szChanBarKey, (void *)szValue, sizeof(szValue)))
{
if (!lstrcmpi(szValue, "yes"))
{
fShowChannelBand=TRUE;
}
}
// In general, don't auto launch the channel bar post IE4.
// Exception: Show the channelband if there is no Show_ChannelBand key and
// the user upgraded over IE4 and this is W95 or W98. This is required
// because IE4 would launch a channel bar in this scenario w/o writting the
// Show_ChannelBand key. We don't want to turn of the channel bar for these
// users.
// Another exception: Loadwc doesn't get uninstalled when IE is
// uninstalled. If the user uninstalls IE5+ and goes back to IE4 we want
// this version of loadwc to revert to IE4 loadwc behavior.
else
{
OSVERSIONINFO vi;
vi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
GetVersionEx(&vi);
if (vi.dwPlatformId != VER_PLATFORM_WIN32_NT &&
(IsIE4Upgrade() || IsUninstallToIE4()))
{
fShowChannelBand=TRUE;
// Set the registry key so this code only runs once and upgrades to
// IE5 don't have to worry about this scenario.
WriteRegValue(HKEY_CURRENT_USER, c_szChanBarRegPath, c_szChanBarKey,
REG_SZ, TEXT("yes"), sizeof(TEXT("yes")));
}
else
{
WriteRegValue(HKEY_CURRENT_USER, c_szChanBarRegPath, c_szChanBarKey,
REG_SZ, TEXT("no"), sizeof(TEXT("no")));
}
}
if (fShowChannelBand)
{
int cLen = lstrlen(szPath);
lstrcpyn(szPath + cLen, " -channelband", ARRAYSIZE(szPath) - cLen);
WinExec(szPath, SW_SHOWNORMAL);
}
#endif
/* Check the registry to see if the welcome app should be launched.
* Again, only launch if value is missing or positive (non-zero dword, this time).
*/
DWORD dwShow = 0;
if (!ReadRegValue(HKEY_CURRENT_USER, "Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Tips", "ShowIE4", (void *)&dwShow, sizeof(dwShow)) || dwShow) {
lstrcpyn(pszPathEnd, "WELCOME.EXE /f", ARRAYSIZE(szPath) - (UINT)(pszPathEnd - szPath));
WinExec(szPath, SW_SHOWNORMAL);
}
}
int WINAPI WinMainT(
HINSTANCE hInstance, // handle to current instance
HINSTANCE hPrevInstance, // handle to previous instance
LPSTR lpCmdLine, // pointer to command line
int nCmdShow // show state of window
)
{
HWND hwndOtherInstance;
// Save the globals
g.hInstance = hInstance;
g.nCmdShow = SW_HIDE;
g.fUninstallOnly = FALSE;
g.fStartSensLce = FALSE;
g.fIntShellMode = IsIntegratedShellMode();
g.hWebcheck = NULL;
MLLoadResources(g.hInstance, TEXT("loadwclc.dll"));
// Parse the command line, for DEBUG options and for uninstall-only switch.
// Now also sets fStartSensLce
if (!bParseCommandLine(lpCmdLine, nCmdShow))
return 0;
// look for webcheck window. This is ultimately the guy we need to
// find to load sens/lce late in the game.
hwndOtherInstance = FindWindow(c_szWebCheckWindow, c_szWebCheckWindow);
if(NULL == hwndOtherInstance)
{
// can't find webcheck, look for loadwc. If we find him but not
// webcheck, we either don't have the MOP or we're still in the 15
// second delay. Send the messages to loadwc and he'll take care
// of it.
hwndOtherInstance = FindWindow(c_szClassName, c_szClassName);
}
if(hwndOtherInstance)
{
// an instance is already running. Tell it about Sens/LCE loading
// requirements and bail out
if(g.fStartSensLce)
{
PostMessage(hwndOtherInstance, WM_START_SENSLCE, 0, 0);
}
return 0;
}
// Set up the absolute paths for all the shell folders we care about,
// in case we're in browser-only mode and the shell doesn't support
// the new ones.
if (!g.fUninstallOnly)
InitShellFolders();
// Run all install/uninstall stubs for browser-only mode.
// If IE4 has been uninstalled, we'll be run with the -u switch; this
// means to run install/uninstall stubs only, no webcheck stuff.
if (!g.fIntShellMode) {
RunInstallUninstallStubs2(NULL);
}
if (g.fUninstallOnly)
return 0;
// Launch the channel bar and welcome apps in browser-only mode.
if (!g.fIntShellMode) {
LaunchBrowserOnlyApps();
}
// Register the window class for the main window.
WNDCLASS wc;
if (!hPrevInstance)
{
wc.style = 0;
wc.lpfnWndProc = (WNDPROC) WndProc;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = hInstance;
wc.hIcon = NULL;
wc.hCursor = NULL;
wc.hbrBackground = (HBRUSH) (COLOR_WINDOW + 1);
wc.lpszMenuName = NULL;
wc.lpszClassName = c_szClassName;
if (!RegisterClass(&wc))
return 0;
}
// Create the main window.
g.hwnd = CreateWindow(c_szClassName,
c_szClassName,
WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT,
CW_USEDEFAULT,
CW_USEDEFAULT,
CW_USEDEFAULT,
(HWND) NULL,
(HMENU) NULL,
hInstance,
(LPVOID) NULL);
if (!g.hwnd)
return 0;
// Show the window and paint its contents.
ShowWindow(g.hwnd, g.nCmdShow);
// Start the message loop
MSG msg;
while (GetMessage(&msg, (HWND) NULL, 0, 0)) {
TranslateMessage(&msg);
DispatchMessage(&msg);
}
MLFreeResources(g.hInstance);
// Return the exit code to Windows
return (int)msg.wParam;
}
LRESULT APIENTRY WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch (uMsg)
{
case WM_CREATE:
DWORD dwTime;
if (!ReadRegValue(HKEY_CURRENT_USER, c_szWebcheckKey, "DelayLoad", &dwTime, sizeof(DWORD)))
dwTime = 15;
SetTimer(hwnd, 1, 1000 * dwTime, NULL);
break;
case WM_START_SENSLCE:
g.fStartSensLce = TRUE;
break;
case WM_TIMER:
KillTimer(hwnd, 1);
vLoadWebCheck();
return 0;
case WM_ENDSESSION:
if (!wParam) // if not fEndSession, bail
break;
// else fall through to WM_DESTROY
case WM_DESTROY:
vUnloadWebCheck();
PostQuitMessage(0);
return 0;
default:
break;
}
return DefWindowProc(hwnd, uMsg, wParam, lParam);
}
void vLoadWebCheck(void)
{
if(g.hWebcheck)
return;
g.hWebcheck = LoadLibrary(TEXT("webcheck.dll"));
if(g.hWebcheck)
{
PFNSTART pfn = (PFNSTART)GetProcAddress(g.hWebcheck, (LPCSTR)7);
if(pfn)
{
pfn(g.fStartSensLce);
}
else
{
// clean up dll
FreeLibrary(g.hWebcheck);
g.hWebcheck = NULL;
}
}
}
void vUnloadWebCheck(void)
{
if (!g.hWebcheck)
return;
PFNSTOP pfn = (PFNSTOP)GetProcAddress(g.hWebcheck, (LPCSTR)8);
if(pfn)
{
pfn();
}
// [darrenmi] don't bother unloading webcheck. We only do this in
// response to a shut down so it's not a big deal. On NT screen saver
// proxy has a thread that wakes up after the call to StopService - if
// we've unloaded the dll before then, we're toast.
// clean up dll
//FreeLibrary(g.hWebcheck);
//g.hWebcheck = NULL;
}
// StringFromGuid
// returns an ANSI string from a CLSID or GUID
// Parameters:
// REFIID - [in] clsid to make string out of.
// LPSTR - [in] buffer in which to place resultant GUID.
// Output:
// int - number of chars written out.
#ifdef DEBUG
int StringFromGuid(
const CLSID* piid,
LPTSTR pszBuf
)
{
return wsprintf(pszBuf, TEXT("{%08lX-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}"), piid->Data1,
piid->Data2, piid->Data3, piid->Data4[0], piid->Data4[1], piid->Data4[2],
piid->Data4[3], piid->Data4[4], piid->Data4[5], piid->Data4[6], piid->Data4[7]);
}
#endif // DEBUG
// bParseCmdLine
// Parse the command line
// -u run install/uninstall stubs only, then quit
// DEBUG options:
// -v visible window (easy to shutdown)
// -a add webcheck to shell service object
// -r remove webcheck from shell service object
// -s fix shell folders only
// -? these options
BOOL bParseCommandLine(LPSTR lpCmdLine, int nCmdShow)
{
if (!lpCmdLine)
return TRUE;
CharUpper(lpCmdLine); /* easier to parse */
while (*lpCmdLine)
{
if (*lpCmdLine != '-' && *lpCmdLine != '/')
break;
lpCmdLine++;
switch (*(lpCmdLine++))
{
case 'E':
// ignore 'embedding' command line
break;
case 'L':
case 'M':
g.fStartSensLce = TRUE;
break;
case 'U':
g.fUninstallOnly = TRUE;
break;
#ifdef DEBUG
case 'V':
g.nCmdShow = nCmdShow;
break;
case 'A':
{
HKEY hkey;
DWORD dwRet;
if (ERROR_SUCCESS == RegCreateKeyEx(HKEY_LOCAL_MACHINE, c_szShellReg, 0, 0, 0, KEY_READ | KEY_WRITE, NULL, &hkey, &dwRet))
{
// NOTE: Does it matter if we set this as ANSI or Unicode? No, stored in Unicode.
// NOTE: Does sizeof include the terminating null? Yes.
int iRet;
TCHAR szCLSID[GUIDSTR_MAX];
iRet = StringFromGuid(&CLSID_WebCheck, szCLSID);
ASSERT(GUIDSTR_MAX == (iRet + 1));
if (ERROR_SUCCESS == RegSetValueEx(hkey, c_szWebCheck, 0, REG_SZ, (CONST BYTE*)szCLSID, sizeof(szCLSID)))
{
MessageBox(NULL, "Webcheck was added to the shell service object list", "LoadWC", MB_OK);
}
RegCloseKey(hkey);
}
return FALSE;
}
break;
case 'R':
{
HKEY hkey;
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE, c_szShellReg, 0, KEY_READ | KEY_WRITE, &hkey))
{
if (ERROR_SUCCESS == RegDeleteValue(hkey, c_szWebCheck))
MessageBox(NULL, "Webcheck was removed from the shell service object list", "LoadWC", MB_OK);
RegCloseKey(hkey);
}
return FALSE;
}
break;
case 'S':
InitShellFolders();
return FALSE;
case '?':
default:
MessageBox(NULL, "Command line options:\n-v\tvisible window\n-a\tadd webcheck as shell service\n-r\tremove webcheck as shell service\n-s\tfixup shell folders only", "LoadWC", MB_OK);
return FALSE;
#endif
}
while (*lpCmdLine == ' ' || *lpCmdLine == '\t') {
lpCmdLine++;
}
}
return TRUE;
}