You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1002 lines
22 KiB
1002 lines
22 KiB
/*
|
|
* NMPGMGRP - Tiny program to add and remove items from the program group.
|
|
* Its initial purpose is to support Windows NT common program groups,
|
|
* which are not supported by GRPCONV.
|
|
*
|
|
* Usage:
|
|
*
|
|
* NMPGMGRP /add [/common] [/g:"<group name>"] /n:"<program name>"
|
|
* /p:"<program path>"
|
|
* NMPGMGRP /delete [/common] [/g:"<group name>"] /n:"<program name>"
|
|
*
|
|
* NMPGMGRP /i /n:"<src mnmdd.dll>" /p:"<dst mnmdd.dll>" INSTALL NT DD
|
|
* NMPGMGRP /u /n:"<src mnmdd.dll>" UNINSTALL NT DD
|
|
*
|
|
* NMPGMGRP /s [/q] /n:"<inf file>" /f"<friendly name>" SETUP
|
|
*
|
|
* /add is used to add a new program item.
|
|
* /delete is used to remove an existing program item.
|
|
*
|
|
* /common indicates that this item belong in the common (as opposed to
|
|
* per-user) program groups.
|
|
*
|
|
* <group name> is the name of the program group, expressed as a pathname
|
|
* relative to the Programs group. For items in the Programs group,
|
|
* this parameter should be omitted.
|
|
*
|
|
* <program name> is the name of the program, and is also used as the name
|
|
* of the shortcut file itself.
|
|
*
|
|
* <program path> is the full path name of the program.
|
|
*
|
|
* <inf file> is the name of the installation inf.
|
|
*
|
|
* <friendly name> is the text to be used for any message box title.
|
|
*
|
|
* Limitations:
|
|
*
|
|
* Because some of these strings may contain spaces, the group name, program
|
|
* name, and program path MUST be enclosed in quotes. Currently we do not
|
|
* support strings with quotes in them.
|
|
*
|
|
* Some of the system functions used in this program are Unicode
|
|
* specific, so this program will require some modifications to run on
|
|
* Windows 95.
|
|
*
|
|
* Author:
|
|
* DannyGl, 23 Mar 97
|
|
*/
|
|
|
|
#include "precomp.h"
|
|
#include "resource.h"
|
|
|
|
#include <nmremote.h>
|
|
|
|
#pragma intrinsic(memset)
|
|
|
|
#define ARRAYSIZE(a) (sizeof(a)/sizeof(a[0]))
|
|
|
|
// DEBUG only -- Define debug zone
|
|
#ifdef DEBUG
|
|
HDBGZONE ghZone = NULL; // Node Controller Zones
|
|
static PTCHAR rgZones[] = {
|
|
TEXT("NMPgmGrp")
|
|
};
|
|
#endif // DEBUG
|
|
|
|
|
|
// PROGRAM_ITEM_INFO structure:
|
|
//
|
|
// Intended to be passed as input to the CreateProgramItem and
|
|
// DeleteProgramItem functions. Fields are:
|
|
//
|
|
// pszProgramGroup - The full path of the program group in which the
|
|
// item is to be stored.
|
|
// pszProgramName - The name of the program item.
|
|
// pszProgramPath - The full path of the program.
|
|
|
|
typedef
|
|
struct tagProgramItemInfo
|
|
{
|
|
PTSTR pszProgramGroup;
|
|
PTSTR pszProgramName;
|
|
PTSTR pszProgramPath;
|
|
} PROGRAM_ITEM_INFO, *PPROGRAM_ITEM_INFO;
|
|
|
|
|
|
// Command line option data
|
|
enum tagGroupOperation
|
|
{
|
|
GRPOP_NONE = 0,
|
|
GRPOP_ADD,
|
|
GRPOP_DEL,
|
|
GRPOP_NTDDINSTALL,
|
|
GRPOP_NTDDUNINSTALL,
|
|
GRPOP_SETUP
|
|
} g_goAction;
|
|
|
|
BOOL g_fCommonGroup = FALSE;
|
|
PTSTR g_pszGroupName = NULL;
|
|
PTSTR g_pszProgramName = NULL;
|
|
PTSTR g_pszProgramPath = NULL;
|
|
PTSTR g_pszFriendlyName = NULL;
|
|
BOOL g_fQuietInstall = FALSE;
|
|
|
|
const TCHAR g_cszSetupDll[] = TEXT("advpack.dll");
|
|
const TCHAR g_cszSetupEntry[] = TEXT("LaunchINFSection");
|
|
typedef int (CALLBACK * PFNSETUPENTRY)(HWND hwnd, HINSTANCE hinst, LPTSTR lpszCmdLine, int nCmdShow);
|
|
|
|
|
|
// ProcessCommandLineArgs:
|
|
//
|
|
// Get the command line and parse it into individual parameters using the
|
|
// above global variables.
|
|
//
|
|
// Return: TRUE on success, FALSE if it could not parse the command line.
|
|
BOOL
|
|
ProcessCommandLineArgs(void)
|
|
{
|
|
PTSTR pszTemp;
|
|
|
|
pszTemp = GetCommandLine();
|
|
|
|
// Search for forward slashes
|
|
pszTemp = (PTSTR) _StrChr(pszTemp, TEXT('/'));
|
|
|
|
while (NULL != pszTemp)
|
|
{
|
|
PTSTR *ppszCurrentArg = NULL;
|
|
|
|
switch(*++pszTemp)
|
|
{
|
|
case TEXT('S'):
|
|
case TEXT('s'):
|
|
ASSERT(GRPOP_NONE == g_goAction); // Check for duplicate parameter
|
|
g_goAction = GRPOP_SETUP;
|
|
break;
|
|
|
|
case TEXT('I'):
|
|
case TEXT('i'):
|
|
//
|
|
// Install NT-specific display driver stuff
|
|
//
|
|
ASSERT(GRPOP_NONE == g_goAction); // Check for duplicate parameter
|
|
g_goAction = GRPOP_NTDDINSTALL;
|
|
break;
|
|
|
|
case TEXT('U'):
|
|
case TEXT('u'):
|
|
//
|
|
// Uninstall NT-specific display driver stuff
|
|
//
|
|
ASSERT(GRPOP_NONE == g_goAction); // Check for duplicate parameter
|
|
g_goAction = GRPOP_NTDDUNINSTALL;
|
|
break;
|
|
|
|
case TEXT('A'):
|
|
case TEXT('a'):
|
|
ASSERT(GRPOP_NONE == g_goAction); // Check for duplicate parameter
|
|
g_goAction = GRPOP_ADD;
|
|
break;
|
|
|
|
case TEXT('D'):
|
|
case TEXT('d'):
|
|
ASSERT(GRPOP_NONE == g_goAction); // Check for duplicate parameter
|
|
g_goAction = GRPOP_DEL;
|
|
break;
|
|
|
|
case TEXT('C'):
|
|
case TEXT('c'):
|
|
ASSERT(! g_fCommonGroup); // Check for duplicate parameter
|
|
g_fCommonGroup = TRUE;
|
|
break;
|
|
|
|
case TEXT('Q'):
|
|
case TEXT('q'):
|
|
g_fQuietInstall = TRUE;
|
|
break;
|
|
|
|
case TEXT('G'):
|
|
case TEXT('g'):
|
|
if (NULL == ppszCurrentArg)
|
|
{
|
|
ppszCurrentArg = &g_pszGroupName;
|
|
}
|
|
|
|
// NO break HERE -- fall through
|
|
|
|
case TEXT('N'):
|
|
case TEXT('n'):
|
|
if (NULL == ppszCurrentArg)
|
|
{
|
|
ppszCurrentArg = &g_pszProgramName;
|
|
}
|
|
|
|
// NO break HERE -- fall through
|
|
|
|
case TEXT('P'):
|
|
case TEXT('p'):
|
|
if (NULL == ppszCurrentArg)
|
|
{
|
|
ppszCurrentArg = &g_pszProgramPath;
|
|
}
|
|
|
|
// NO break HERE -- fall through
|
|
|
|
case TEXT('F'):
|
|
case TEXT('f'):
|
|
if (NULL == ppszCurrentArg)
|
|
{
|
|
ppszCurrentArg = &g_pszFriendlyName;
|
|
}
|
|
|
|
// ***** Processing for all string parameters *****
|
|
|
|
ASSERT(NULL == *ppszCurrentArg); // Check for duplicate parameter
|
|
|
|
// Save the string pointer after skipping past the colon and open quote
|
|
ASSERT(TEXT(':') == pszTemp[1] && TEXT('\"') == pszTemp[2]);
|
|
*ppszCurrentArg = pszTemp += 3;
|
|
|
|
// Find the closing quote and set it to null, then skip past it
|
|
// Note that we don't handle strings with quotes in them.
|
|
pszTemp = (PTSTR) _StrChr(pszTemp, TEXT('\"'));
|
|
ASSERT(NULL != pszTemp);
|
|
if (NULL != pszTemp)
|
|
{
|
|
*pszTemp++ = TEXT('\0');
|
|
}
|
|
else
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
ERROR_OUT(("Unknown parameter begins at %s", pszTemp));
|
|
return FALSE;
|
|
|
|
break;
|
|
}
|
|
|
|
// Find the next option flag
|
|
ASSERT(NULL != pszTemp);
|
|
pszTemp = (PTSTR) _StrChr(pszTemp, TEXT('/'));
|
|
}
|
|
|
|
// Return based on minimal parameter validation:
|
|
// 1) The program name must be specified.
|
|
// 2) Either add or delete must be specified.
|
|
// 3) If add is specified, the program path must be specified
|
|
switch (g_goAction)
|
|
{
|
|
case GRPOP_ADD:
|
|
case GRPOP_NTDDINSTALL:
|
|
return((NULL != g_pszProgramName) && (NULL != g_pszProgramPath));
|
|
|
|
case GRPOP_DEL:
|
|
case GRPOP_NTDDUNINSTALL:
|
|
return(NULL != g_pszProgramName);
|
|
|
|
case GRPOP_SETUP:
|
|
return((NULL != g_pszProgramName) && (NULL != g_pszFriendlyName));
|
|
|
|
default:
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
// GetFolderPathname:
|
|
//
|
|
// Use the official shell interfaces to retrieve the full pathname of a
|
|
// a programs folder.
|
|
//
|
|
// Input:
|
|
// ptstrPath, ccPath - The pointer to a size of the buffer in
|
|
// which to store the path.
|
|
// nFolder - The folder to locate, expressed as a CSIDL constant.
|
|
// See SHGetSpecialFolderLocation for details.
|
|
// pctstrSubFolder - A specific subfolder, can be NULL if not specified.
|
|
// If specified, this is appended (after a backslash) to the path.
|
|
//
|
|
// Returns:
|
|
// An HRESULT to indicate success or failure of the Shell methods.
|
|
// The path is returned in <ptstrPath>.
|
|
|
|
HRESULT
|
|
GetFolderPathname(
|
|
PTSTR ptstrPath,
|
|
UINT cchPath,
|
|
int nFolder,
|
|
LPCTSTR pctstrSubFolder)
|
|
{
|
|
HRESULT hr;
|
|
LPMALLOC pMalloc = NULL;
|
|
LPSHELLFOLDER pDesktopFolder = NULL;
|
|
LPITEMIDLIST pidlSpecialFolder = NULL;
|
|
|
|
// Get the allocator object
|
|
hr = CoGetMalloc(MEMCTX_TASK, &pMalloc);
|
|
|
|
// Get the desktop object
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = SHGetDesktopFolder(&pDesktopFolder);
|
|
}
|
|
|
|
// Get the special folder item ID
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = SHGetSpecialFolderLocation(
|
|
GetDesktopWindow(),
|
|
nFolder,
|
|
&pidlSpecialFolder);
|
|
}
|
|
|
|
// Retrieve the folder name
|
|
STRRET strFolder;
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
strFolder.uType = STRRET_WSTR;
|
|
|
|
hr = pDesktopFolder->GetDisplayNameOf(
|
|
pidlSpecialFolder,
|
|
SHGDN_FORPARSING,
|
|
&strFolder);
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
CUSTRING custrPath;
|
|
|
|
switch(strFolder.uType)
|
|
{
|
|
case STRRET_WSTR:
|
|
custrPath.AssignString(strFolder.pOleStr);
|
|
|
|
break;
|
|
|
|
case STRRET_OFFSET:
|
|
custrPath.AssignString(((LPSTR) pidlSpecialFolder) + strFolder.uOffset);
|
|
|
|
break;
|
|
|
|
case STRRET_CSTR:
|
|
custrPath.AssignString(strFolder.cStr);
|
|
|
|
break;
|
|
}
|
|
|
|
if(NULL != (PTSTR) custrPath)
|
|
lstrcpyn(ptstrPath, custrPath, cchPath);
|
|
else
|
|
*ptstrPath = _TEXT('\0');
|
|
|
|
if (STRRET_WSTR == strFolder.uType)
|
|
{
|
|
pMalloc->Free(strFolder.pOleStr);
|
|
}
|
|
|
|
}
|
|
|
|
// Append subgroup name, if it's specified
|
|
if (SUCCEEDED(hr) && NULL != pctstrSubFolder)
|
|
{
|
|
// BUGBUG - We don't create this folder if it doesn't already exist
|
|
|
|
int cchLen = lstrlen(ptstrPath);
|
|
|
|
ASSERT((UINT) cchLen < cchPath);
|
|
|
|
// Insert a path separator
|
|
ptstrPath[cchLen++] = TEXT('\\');
|
|
|
|
// Copy the subgroup
|
|
lstrcpyn(ptstrPath + cchLen, pctstrSubFolder, cchPath - cchLen);
|
|
}
|
|
|
|
// Release resources
|
|
if (pDesktopFolder)
|
|
{
|
|
pDesktopFolder->Release();
|
|
}
|
|
|
|
if (pMalloc)
|
|
{
|
|
if (pidlSpecialFolder)
|
|
{
|
|
pMalloc->Free(pidlSpecialFolder);
|
|
}
|
|
|
|
pMalloc->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
// BuildLinkFileName:
|
|
//
|
|
// Inline utility function to construct the full file name of a link given its
|
|
// directory name and item name.
|
|
inline void
|
|
BuildLinkFileName(
|
|
OUT LPWSTR wszOutputPath,
|
|
IN LPCTSTR pcszDirectory,
|
|
IN LPCTSTR pcszFile)
|
|
{
|
|
// The file name is of the form <directory>\<file>.LNK
|
|
|
|
#ifdef UNICODE
|
|
static const WCHAR wszFileFormat[] = L"%s\\%s.LNK";
|
|
#else // UNICODE
|
|
static const WCHAR wszFileFormat[] = L"%hs\\%hs.LNK";
|
|
#endif // UNICODE
|
|
int cchSize;
|
|
|
|
cchSize = wsprintfW(
|
|
wszOutputPath,
|
|
wszFileFormat,
|
|
pcszDirectory,
|
|
pcszFile);
|
|
|
|
ASSERT(cchSize > ARRAY_ELEMENTS(wszFileFormat) - 1 && cchSize < MAX_PATH);
|
|
}
|
|
|
|
|
|
// CreateProgramItem:
|
|
//
|
|
// Use the official shell interfaces to create a shortcut to a program.
|
|
//
|
|
// Input: A pointer to a PROGRAM_ITEM_INFO structure, defined above.
|
|
//
|
|
// Returns:
|
|
// An HRESULT to indicate success or failure of the Shell methods.
|
|
|
|
HRESULT
|
|
CreateProgramItem(
|
|
PPROGRAM_ITEM_INFO ppii)
|
|
{
|
|
HRESULT hr;
|
|
IShellLink *psl = NULL;
|
|
IPersistFile *ppf = NULL;
|
|
|
|
// Get the shell link object
|
|
hr = CoCreateInstance(
|
|
CLSID_ShellLink,
|
|
NULL,
|
|
CLSCTX_INPROC,
|
|
IID_IShellLink,
|
|
(LPVOID *) &psl);
|
|
|
|
// Fill in the fields of the program group item
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = psl->SetDescription(ppii->pszProgramName);
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = psl->SetPath(ppii->pszProgramPath);
|
|
}
|
|
|
|
// Save the link as a file
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = psl->QueryInterface(IID_IPersistFile, (LPVOID *) &ppf);
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
WCHAR wszFileName[MAX_PATH];
|
|
|
|
BuildLinkFileName(
|
|
wszFileName,
|
|
ppii->pszProgramGroup,
|
|
ppii->pszProgramName);
|
|
|
|
hr = ppf->Save(wszFileName, TRUE);
|
|
}
|
|
|
|
// Release the objects we used
|
|
if (ppf)
|
|
{
|
|
ppf->Release();
|
|
}
|
|
|
|
if (psl)
|
|
{
|
|
psl->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
// DeleteProgramItem:
|
|
//
|
|
// Delete a shortcut to a program.
|
|
//
|
|
// Input: A pointer to a PROGRAM_ITEM_INFO structure, defined above.
|
|
//
|
|
// Returns:
|
|
// An HRESULT to indicate success or failure of the Shell methods.
|
|
|
|
HRESULT
|
|
DeleteProgramItem(
|
|
PPROGRAM_ITEM_INFO ppii)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
WCHAR wszFileName[MAX_PATH];
|
|
|
|
BuildLinkFileName(
|
|
wszFileName,
|
|
ppii->pszProgramGroup,
|
|
ppii->pszProgramName);
|
|
|
|
if (! DeleteFileW(wszFileName))
|
|
{
|
|
WARNING_OUT(("DeleteFile failed"));
|
|
hr = E_FAIL;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//
|
|
// NtDDInstall()
|
|
// This does NT-specific display driver install stuff, which depends on
|
|
// whether it's NT4 or NT5
|
|
//
|
|
//
|
|
HRESULT NtDDInstall(LPTSTR pszOrigDd, LPTSTR pszNewDd)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
OSVERSIONINFO osvi;
|
|
RegEntry re(NM_NT_DISPLAY_DRIVER_KEY, HKEY_LOCAL_MACHINE, FALSE);
|
|
|
|
//
|
|
// If NT4, set service key to disabled
|
|
// If NT5, copy mnmdd.dll from NM dir to cur (system32) dir
|
|
//
|
|
osvi.dwOSVersionInfoSize = sizeof(osvi);
|
|
|
|
if (!GetVersionEx(&osvi))
|
|
{
|
|
ERROR_OUT(("GetVersionEx() failed"));
|
|
goto AllDone;
|
|
}
|
|
|
|
if ((osvi.dwPlatformId == VER_PLATFORM_WIN32s) ||
|
|
(osvi.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS))
|
|
{
|
|
WARNING_OUT(("NT setup running on non-NT platform!"));
|
|
goto AllDone;
|
|
}
|
|
|
|
if (osvi.dwMajorVersion >= 5)
|
|
{
|
|
//
|
|
// This is NT5. Always set the service key to enabled (in case
|
|
// the end user managed to munge it) and copy mnmdd.dll to the
|
|
// current (system) directory. For example, if somebody had a
|
|
// stand-alone version of a beta, uninstalled it, then installed
|
|
// NM 3.0 proper--or same for 2.11.
|
|
//
|
|
re.SetValue(REGVAL_NM_NT_DISPLAY_DRIVER_ENABLED, NT_DRIVER_START_SYSTEM);
|
|
|
|
if (!CopyFile(pszOrigDd, pszNewDd, FALSE))
|
|
{
|
|
WARNING_OUT(("CopyFile from %s to %s failed", pszOrigDd, pszNewDd));
|
|
goto AllDone;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// This is NT4. Set the disabled service key
|
|
re.SetValue(REGVAL_NM_NT_DISPLAY_DRIVER_ENABLED, NT_DRIVER_START_DISABLED);
|
|
}
|
|
|
|
hr = S_OK;
|
|
|
|
AllDone:
|
|
return(hr);
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// NtDDUninstall()
|
|
// This does NT-specific display driver uninstall stuff, which depends
|
|
// on whether it's NT4 or NT5
|
|
//
|
|
HRESULT NtDDUninstall(LPTSTR pszOrigFile)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
OSVERSIONINFO osvi;
|
|
|
|
//
|
|
// If NT4, set service key to disabled
|
|
// If NT5, delete mnmdd.dll from cur (system32) dir
|
|
//
|
|
osvi.dwOSVersionInfoSize = sizeof(osvi);
|
|
|
|
if (!GetVersionEx(&osvi))
|
|
{
|
|
ERROR_OUT(("GetVersionEx() failed"));
|
|
goto AllDone;
|
|
}
|
|
|
|
if ((osvi.dwPlatformId == VER_PLATFORM_WIN32s) ||
|
|
(osvi.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS))
|
|
{
|
|
ERROR_OUT(("NT setup running on non-NT platform!"));
|
|
goto AllDone;
|
|
}
|
|
|
|
if (osvi.dwMajorVersion >= 5)
|
|
{
|
|
// This is NT5. Delete mnmdd.dll from the current (system) directory
|
|
if (!DeleteFile(pszOrigFile))
|
|
{
|
|
WARNING_OUT(("DeleteFile of %s failed", pszOrigFile));
|
|
goto AllDone;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// This is NT4. Set the disabled service key
|
|
RegEntry re(NM_NT_DISPLAY_DRIVER_KEY, HKEY_LOCAL_MACHINE, FALSE);
|
|
|
|
re.SetValue(REGVAL_NM_NT_DISPLAY_DRIVER_ENABLED,
|
|
NT_DRIVER_START_DISABLED);
|
|
}
|
|
|
|
hr = S_OK;
|
|
|
|
AllDone:
|
|
return(hr);
|
|
}
|
|
|
|
UINT _MessageBox(HINSTANCE hInst, UINT uID, LPCTSTR lpCaption, UINT uType)
|
|
{
|
|
TCHAR szText[512];
|
|
|
|
if (0 != LoadString(hInst, uID, szText, CCHMAX(szText)))
|
|
{
|
|
return MessageBox(NULL, szText, lpCaption, uType);
|
|
}
|
|
|
|
return IDCANCEL;
|
|
}
|
|
|
|
#define CONF_INIT_EVENT TEXT("CONF:Init")
|
|
|
|
BOOL FIsNetMeetingRunning()
|
|
{
|
|
HANDLE hEvent;
|
|
|
|
hEvent = OpenEvent(EVENT_ALL_ACCESS, FALSE, SERVICE_STOP_EVENT);
|
|
if (hEvent)
|
|
{
|
|
CloseHandle(hEvent);
|
|
return TRUE;
|
|
}
|
|
|
|
hEvent = OpenEvent(EVENT_ALL_ACCESS, FALSE, CONF_INIT_EVENT);
|
|
if (hEvent)
|
|
{
|
|
CloseHandle(hEvent);
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL FIsNT5()
|
|
{
|
|
OSVERSIONINFO osvi;
|
|
|
|
osvi.dwOSVersionInfoSize = sizeof(osvi);
|
|
|
|
if (GetVersionEx(&osvi))
|
|
{
|
|
if ((osvi.dwPlatformId != VER_PLATFORM_WIN32s) &&
|
|
(osvi.dwPlatformId != VER_PLATFORM_WIN32_WINDOWS))
|
|
{
|
|
if (osvi.dwMajorVersion >= 5)
|
|
{
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ERROR_OUT(("GetVersionEx() failed"));
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
#define IE4_KEY TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Internet Settings\\Last Update\\IEXPLOREV4")
|
|
|
|
BOOL FIsIE4Installed()
|
|
{
|
|
RegEntry re(IE4_KEY, HKEY_LOCAL_MACHINE, FALSE);
|
|
if (ERROR_SUCCESS != re.GetError())
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
#define INTEL_KEY1 TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\App Paths\\vphone.exe")
|
|
#define INTEL_KEY2 TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\App Paths\\rvp.exe")
|
|
#define INTEL_KEY3 TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\App Paths\\vp30.exe")
|
|
#define INTEL_NM_VERSION_SZ TEXT("NetMeeting")
|
|
#define INTEL_NM_VERSION_DW 3
|
|
|
|
#define PANTHER_KEY TEXT("CLSID\\{690968D0-418C-11D1-8E0B-00A0C95A83DA}\\Version")
|
|
#define PANTHER_VERSION_VALUE TEXT("1.0")
|
|
|
|
#define TRANSPORTS_KEY TEXT("SOFTWARE\\Microsoft\\Conferencing\\Transports")
|
|
|
|
#define REGKEY_PSTN TEXT("PSTN")
|
|
#define REGKEY_TCPIP TEXT("TCPIP")
|
|
#define REGKEY_IPX TEXT("IPX")
|
|
#define REGKEY_NETBIOS TEXT("NETBIOS")
|
|
#define REGKEY_DIRCB TEXT("DIRCB")
|
|
|
|
|
|
long GetIntelVersion(LPCTSTR pszKey)
|
|
{
|
|
RegEntry re(pszKey, HKEY_LOCAL_MACHINE, FALSE);
|
|
if (ERROR_SUCCESS != re.GetError())
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
return re.GetNumber(INTEL_NM_VERSION_SZ, 0);
|
|
}
|
|
|
|
BOOL FAnyBadIntelApps()
|
|
{
|
|
long lVersion;
|
|
|
|
lVersion = GetIntelVersion(INTEL_KEY1);
|
|
if (0 > lVersion)
|
|
{
|
|
lVersion = GetIntelVersion(INTEL_KEY2);
|
|
if (0 > lVersion)
|
|
{
|
|
lVersion = GetIntelVersion(INTEL_KEY3);
|
|
}
|
|
}
|
|
|
|
return ((0 <= lVersion) && (3 > lVersion));
|
|
}
|
|
|
|
BOOL FAnyBadPantherApps()
|
|
{
|
|
RegEntry re(PANTHER_KEY, HKEY_CLASSES_ROOT, FALSE);
|
|
if (ERROR_SUCCESS != re.GetError())
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
LPCTSTR pszVersion = re.GetString(TEXT(""));
|
|
|
|
return 0 == lstrcmp(pszVersion, PANTHER_VERSION_VALUE);
|
|
}
|
|
|
|
BOOL FAnyUnknownTransports()
|
|
{
|
|
RegEntry TransportsKey(TRANSPORTS_KEY, HKEY_LOCAL_MACHINE, FALSE);
|
|
RegEnumSubKeys EnumTransports(&TransportsKey);
|
|
|
|
while( 0 == EnumTransports.Next() )
|
|
{
|
|
LPCTSTR pszName = EnumTransports.GetName();
|
|
|
|
if ((0 != lstrcmpi(pszName, REGKEY_PSTN)) &&
|
|
(0 != lstrcmpi(pszName, REGKEY_TCPIP)) &&
|
|
(0 != lstrcmpi(pszName, REGKEY_IPX)) &&
|
|
(0 != lstrcmpi(pszName, REGKEY_NETBIOS)) &&
|
|
(0 != lstrcmpi(pszName, REGKEY_DIRCB)))
|
|
{
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL FAnyIncompatibleApps()
|
|
{
|
|
return FAnyUnknownTransports() || FAnyBadIntelApps() || FAnyBadPantherApps();
|
|
}
|
|
|
|
HRESULT Setup(HINSTANCE hInst, LPTSTR pszInfFile, LPTSTR pszFriendlyName, BOOL fQuietInstall)
|
|
{
|
|
if (FIsNT5())
|
|
{
|
|
_MessageBox(hInst, IDS_SETUP_WIN2K, pszFriendlyName, MB_OK);
|
|
// if the SHFT-CTRL was pressed continue with the install, else exit
|
|
if ((0 == GetAsyncKeyState(VK_CONTROL)) ||
|
|
(0 == GetAsyncKeyState(VK_SHIFT)))
|
|
{
|
|
return S_FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
if (!FIsIE4Installed())
|
|
{
|
|
_MessageBox(hInst, IDS_SETUP_IE4, pszFriendlyName, MB_OK);
|
|
return S_FALSE;
|
|
}
|
|
|
|
while (FIsNetMeetingRunning())
|
|
{
|
|
if (IDCANCEL == _MessageBox(hInst, IDS_SETUP_RUNNING, pszFriendlyName, MB_OKCANCEL))
|
|
{
|
|
return S_FALSE;
|
|
}
|
|
}
|
|
|
|
if (!fQuietInstall)
|
|
{
|
|
if (FAnyIncompatibleApps())
|
|
{
|
|
if (IDNO == _MessageBox(hInst, IDS_SETUP_INCOMPATIBLE, pszFriendlyName, MB_YESNO))
|
|
{
|
|
return S_FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
HRESULT hr = S_FALSE;
|
|
|
|
HINSTANCE hLib = NmLoadLibrary(g_cszSetupDll,TRUE);
|
|
if (NULL != hLib)
|
|
{
|
|
PFNSETUPENTRY pfnEntry = (PFNSETUPENTRY)GetProcAddress(hLib, g_cszSetupEntry);
|
|
if (pfnEntry)
|
|
{
|
|
TCHAR szArgs[MAX_PATH];
|
|
lstrcpyn(szArgs, pszInfFile, ARRAYSIZE(szArgs));
|
|
if (fQuietInstall)
|
|
{
|
|
lstrcat(szArgs, TEXT(",,1,N"));
|
|
}
|
|
else
|
|
{
|
|
lstrcat(szArgs, TEXT(",,,N"));
|
|
}
|
|
|
|
int iRet = pfnEntry(NULL, GetModuleHandle(NULL), szArgs, SW_SHOWNORMAL);
|
|
if (0 == iRet)
|
|
{
|
|
if (!fQuietInstall)
|
|
{
|
|
_MessageBox(hInst, IDS_SETUP_SUCCESS, pszFriendlyName, MB_OK);
|
|
}
|
|
|
|
hr = S_OK;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ERROR_OUT(("Could not find setup DLL entry point"));
|
|
}
|
|
FreeLibrary(hLib);
|
|
}
|
|
else
|
|
{
|
|
ERROR_OUT(("Could not load setup DLL"));
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
// main:
|
|
//
|
|
// The entry point of the program, it pulls everything together using the
|
|
// above utility functions.
|
|
|
|
void __cdecl
|
|
main(
|
|
void)
|
|
{
|
|
HRESULT hr;
|
|
HINSTANCE hInstance;
|
|
BOOL fErrorReported = FALSE;
|
|
TCHAR szFolderPath[MAX_PATH];
|
|
|
|
// Initialization
|
|
hInstance = GetModuleHandle(NULL);
|
|
DBGINIT(&ghZone, rgZones);
|
|
DBG_INIT_MEMORY_TRACKING(hInstance);
|
|
|
|
hr = CoInitialize(NULL);
|
|
|
|
// Process the command line.
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = ProcessCommandLineArgs() ? S_OK : E_INVALIDARG;
|
|
}
|
|
else if (!fErrorReported)
|
|
{
|
|
ERROR_OUT(("CoInitialize fails"));
|
|
fErrorReported = TRUE;
|
|
}
|
|
|
|
// Retreive the path of the Programs folder
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if ((g_goAction != GRPOP_NTDDINSTALL) &&
|
|
(g_goAction != GRPOP_NTDDUNINSTALL) &&
|
|
(g_goAction != GRPOP_SETUP))
|
|
{
|
|
hr = GetFolderPathname(
|
|
szFolderPath,
|
|
CCHMAX(szFolderPath),
|
|
g_fCommonGroup ? CSIDL_COMMON_PROGRAMS : CSIDL_PROGRAMS,
|
|
g_pszGroupName);
|
|
}
|
|
}
|
|
else if (!fErrorReported)
|
|
{
|
|
ERROR_OUT(("Invalid command line parameters specified."));
|
|
fErrorReported = TRUE;
|
|
}
|
|
|
|
// Add or delete the program item, as appropriate
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
PROGRAM_ITEM_INFO pii;
|
|
|
|
switch(g_goAction)
|
|
{
|
|
case GRPOP_NTDDINSTALL:
|
|
//
|
|
// Hack: Use program name for source mnmdd.dll
|
|
// Use program path for dest mnmdd.dll
|
|
//
|
|
hr = NtDDInstall(g_pszProgramName, g_pszProgramPath);
|
|
break;
|
|
|
|
case GRPOP_NTDDUNINSTALL:
|
|
//
|
|
// Hack: Use program name for source mnmdd.dll
|
|
//
|
|
hr = NtDDUninstall(g_pszProgramName);
|
|
break;
|
|
|
|
case GRPOP_ADD:
|
|
pii.pszProgramGroup = szFolderPath;
|
|
pii.pszProgramName = g_pszProgramName;
|
|
pii.pszProgramPath = g_pszProgramPath;
|
|
|
|
hr = CreateProgramItem(&pii);
|
|
|
|
break;
|
|
|
|
case GRPOP_DEL:
|
|
pii.pszProgramGroup = szFolderPath;
|
|
pii.pszProgramName = g_pszProgramName;
|
|
|
|
hr = DeleteProgramItem(&pii);
|
|
|
|
break;
|
|
|
|
case GRPOP_SETUP:
|
|
hr = Setup(hInstance, g_pszProgramName, g_pszFriendlyName, g_fQuietInstall);
|
|
break;
|
|
|
|
default:
|
|
ERROR_OUT(("No operation type specified"));
|
|
hr = E_INVALIDARG;
|
|
|
|
break;
|
|
}
|
|
}
|
|
else if (!fErrorReported)
|
|
{
|
|
ERROR_OUT(("GetFolderPathname returns %lu", hr));
|
|
fErrorReported = TRUE;
|
|
}
|
|
|
|
|
|
// Process cleanup
|
|
CoUninitialize();
|
|
|
|
DBG_CHECK_MEMORY_TRACKING(hInstance);
|
|
DBGDEINIT(&ghZone);
|
|
|
|
ExitProcess(SUCCEEDED(hr) ? 0 : 2);
|
|
}
|