2270 lines
63 KiB
C++
2270 lines
63 KiB
C++
#include <cdlpch.h>
|
|
#pragma hdrstop
|
|
|
|
#include "verp.h"
|
|
#include <mstask.h>
|
|
#include <pkgguid.h>
|
|
|
|
extern LCID g_lcidBrowser; // default to english
|
|
extern char g_szBrowserPrimaryLang[];
|
|
// global that encapsulates delay-loaded version.dll
|
|
CVersion g_versiondll;
|
|
extern BOOL g_bRunOnWin95;
|
|
|
|
HRESULT NeedForceLanguageCheck(HKEY hkeyCLSID, CLocalComponentInfo* plci);
|
|
BOOL SniffStringFileInfo(LPSTR szFileName, LPCTSTR lpszSubblock, DWORD* pdw = NULL);
|
|
HRESULT IsDistUnitLocallyInstalledZI(LPCWSTR, LPCWSTR, DWORD, DWORD, CLocalComponentInfo*);
|
|
HRESULT IsDistUnitLocallyInstalledSxS(LPCWSTR wszDistUnit, LPCWSTR wszClsid, DWORD dwFileVersionMS, DWORD dwFileVersionLS, CLocalComponentInfo* plci);
|
|
void ExtractVersion(char* pszDistUnit, DWORD* pdwVerMS, DWORD* pdwVerLS);
|
|
void GetLatestZIVersion(const WCHAR* pwzDistUnit, DWORD* pdwVerMS, DWORD* pdwVerLS);
|
|
|
|
|
|
// %%Function: CLocalComponentInfo::CLocalComponentInfo
|
|
|
|
CLocalComponentInfo::CLocalComponentInfo()
|
|
{
|
|
szExistingFileName[0] = '\0';
|
|
pBaseExistingFileName = NULL;
|
|
lpDestDir = NULL;
|
|
dwLocFVMS = 0;
|
|
dwLocFVLS = 0;
|
|
ftLastModified.dwLowDateTime = 0;
|
|
ftLastModified.dwHighDateTime = 0;
|
|
lcid = g_lcidBrowser;
|
|
|
|
bForceLangGetLatest = FALSE;
|
|
|
|
pbEtag = NULL;
|
|
|
|
dwAvailMS = 0;
|
|
dwAvailLS = 0;
|
|
|
|
} // CLocalComponentInfo
|
|
|
|
|
|
// %%Function: CLocalComponentInfo::~CLocalComponentInfo
|
|
|
|
CLocalComponentInfo::~CLocalComponentInfo()
|
|
{
|
|
SAFEDELETE(lpDestDir);
|
|
|
|
SAFEDELETE(pbEtag);
|
|
|
|
} // ~CLocalComponentInfo
|
|
|
|
|
|
// %%Function: CLocalComponentInfo::MakeDestDir
|
|
|
|
HRESULT
|
|
CLocalComponentInfo::MakeDestDir()
|
|
{
|
|
|
|
HRESULT hr = S_OK;
|
|
|
|
Assert(pBaseExistingFileName);
|
|
|
|
if (szExistingFileName[0]) {
|
|
|
|
DWORD cbLen = (DWORD)(pBaseExistingFileName - szExistingFileName);
|
|
|
|
|
|
lpDestDir = new char[cbLen + 1];
|
|
|
|
if (lpDestDir) {
|
|
StrNCpy(lpDestDir, szExistingFileName, cbLen);
|
|
} else {
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
|
|
} // ~MakeDestDir
|
|
|
|
|
|
|
|
// %%Function: CModuleUsage::CModuleUsage
|
|
// CModuleUsage is the basic download obj.
|
|
|
|
CModuleUsage::CModuleUsage(LPCSTR szFileName, DWORD muflags, HRESULT* phr)
|
|
{
|
|
|
|
m_szFileName = NULL;
|
|
|
|
if (szFileName) {
|
|
CHAR szCanonical[MAX_PATH];
|
|
|
|
if (CDLGetLongPathNameA(szCanonical, szFileName, MAX_PATH) != 0)
|
|
m_szFileName = new char[lstrlen(szCanonical) + 1];
|
|
|
|
if (m_szFileName) {
|
|
lstrcpy(m_szFileName, szCanonical);
|
|
} else {
|
|
// CDLGetLongPathName failed, so we are on a platform that
|
|
// doesn't support it, and we can't guess what the right long
|
|
// pathname is. Just use the short one
|
|
m_szFileName = new char[lstrlen(szFileName) + 1];
|
|
|
|
if (m_szFileName) {
|
|
lstrcpy(m_szFileName, szFileName);
|
|
} else {
|
|
*phr = E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
}
|
|
|
|
m_dwFlags = muflags;
|
|
|
|
} // CModuleUsage
|
|
|
|
|
|
// %%Function: CModuleUsage::~CModuleUsage
|
|
|
|
CModuleUsage::~CModuleUsage()
|
|
{
|
|
if (m_szFileName)
|
|
SAFEDELETE(m_szFileName);
|
|
|
|
} // ~CModuleUsage
|
|
|
|
|
|
// %%Function: CModuleUsage::Update(LPCSTR lpClientName)
|
|
|
|
HRESULT
|
|
CModuleUsage::Update(LPCSTR lpClientName)
|
|
{
|
|
|
|
return ::UpdateModuleUsage(m_szFileName,
|
|
lpClientName, NULL,
|
|
m_dwFlags);
|
|
} // ~CModuleUsage
|
|
|
|
HRESULT
|
|
GetVersionHint(HKEY hkeyVersion, DWORD* pdwVersionMS, DWORD* pdwVersionLS)
|
|
{
|
|
DWORD Size = MAX_PATH;
|
|
char szVersionBuf[MAX_PATH];
|
|
DWORD dwType;
|
|
HRESULT hr = S_OK;
|
|
|
|
*pdwVersionMS = 0;
|
|
*pdwVersionLS = 0;
|
|
|
|
DWORD lResult = ::RegQueryValueEx(hkeyVersion, NULL, NULL, &dwType,
|
|
(unsigned char*)szVersionBuf, &Size);
|
|
|
|
if (lResult != ERROR_SUCCESS) {
|
|
|
|
// value not found, consider this is always with bad/low version
|
|
// of InstalledVersion
|
|
|
|
hr = S_FALSE;
|
|
goto Exit;
|
|
}
|
|
|
|
if (FAILED(GetVersionFromString(szVersionBuf, pdwVersionMS, pdwVersionLS))) {
|
|
hr = S_FALSE;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/*
|
|
|
|
NAME: CheckInstalledVersionHint
|
|
|
|
SYNOPSIS: Checks for key InstalledVersion under {...}
|
|
If no key then we fail
|
|
once key is present we check version numbers
|
|
S_OK: local version is good enough
|
|
S_FALSE: need update:
|
|
ERROR: not applicable, caller proceeds with InProcServer32
|
|
check.
|
|
|
|
[HKCR:clsid]
|
|
[{...}]
|
|
[InstalledVersion]
|
|
Deafult "1,0,0,1"
|
|
Path "c:\foo\foo.ocx"
|
|
|
|
The Path is optional, if find we will req update id
|
|
file pointed to by path is missing on client. This is a
|
|
way to robustify if user deletes the file on disk but
|
|
not the regsitry (unclean uninstall)
|
|
In this case update is needed
|
|
|
|
*/
|
|
HRESULT
|
|
CheckInstalledVersionHint(
|
|
HKEY hKeyEmbedding,
|
|
CLocalComponentInfo* plci,
|
|
DWORD dwFileVersionMS,
|
|
DWORD dwFileVersionLS)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD Size = MAX_PATH;
|
|
DWORD dwType = 0;
|
|
LONG lResult = ERROR_SUCCESS;
|
|
|
|
const static char* szInstalledVersion = "InstalledVersion";
|
|
const static char* szAvailableVersion = "AvailableVersion";
|
|
const static char* szPATH = "Path";
|
|
const static char* szLASTMODIFIED = "LastModified";
|
|
const static char* szETAG = "Etag";
|
|
|
|
char szVersionBuf[MAX_PATH];
|
|
char szFileName[MAX_PATH];
|
|
HKEY hKeyVersion = 0;
|
|
HKEY hkeyAvailableVersion = 0;
|
|
|
|
DWORD dwLocFVMS = 0;
|
|
DWORD dwLocFVLS = 0;
|
|
char szLastMod[INTERNET_RFC1123_BUFSIZE + 1];
|
|
|
|
if (RegOpenKeyEx(hKeyEmbedding, szAvailableVersion, 0,
|
|
KEY_READ, &hkeyAvailableVersion) == ERROR_SUCCESS) {
|
|
|
|
DWORD dwAvailMS = 0;
|
|
DWORD dwAvailLS = 0;
|
|
|
|
if (GetVersionHint(hkeyAvailableVersion, &dwAvailMS, &dwAvailLS) == S_OK) {
|
|
|
|
plci->dwAvailMS = dwAvailMS;
|
|
plci->dwAvailLS = dwAvailLS;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
lResult = ::RegOpenKeyEx(hKeyEmbedding, szInstalledVersion, 0,
|
|
KEY_READ, &hKeyVersion);
|
|
|
|
if (lResult != ERROR_SUCCESS) {
|
|
|
|
// key not found, consider this is regular ActiveX with no hint
|
|
// of InstalledVersion
|
|
|
|
hr = HRESULT_FROM_WIN32(lResult);
|
|
goto Exit;
|
|
}
|
|
|
|
if (GetVersionHint(hKeyVersion, &dwLocFVMS, &dwLocFVLS) == S_FALSE) {
|
|
hr = S_FALSE;
|
|
goto Exit;
|
|
}
|
|
|
|
plci->dwLocFVMS = dwLocFVMS;
|
|
plci->dwLocFVLS = dwLocFVLS;
|
|
|
|
Size = INTERNET_RFC1123_BUFSIZE + 1;
|
|
lResult = ::RegQueryValueEx(hKeyVersion, szLASTMODIFIED, NULL, &dwType,
|
|
(unsigned char*)szLastMod, &Size);
|
|
|
|
if (lResult == ERROR_SUCCESS) {
|
|
|
|
SYSTEMTIME st;
|
|
FILETIME ft;
|
|
// convert intenet time to file time
|
|
if (InternetTimeToSystemTime(szLastMod, &st, 0) &&
|
|
SystemTimeToFileTime(&st, &ft)) {
|
|
|
|
memcpy(&(plci->ftLastModified), &ft, sizeof(FILETIME));
|
|
|
|
}
|
|
}
|
|
|
|
Size = 0;
|
|
lResult = ::RegQueryValueEx(hKeyVersion, szETAG, NULL, &dwType,
|
|
(unsigned char*)NULL, &Size);
|
|
if (lResult == ERROR_SUCCESS) {
|
|
char* pbEtag = new char[Size];
|
|
if (pbEtag) {
|
|
lResult = ::RegQueryValueEx(hKeyVersion, szETAG, NULL, &dwType,
|
|
(unsigned char*)pbEtag, &Size);
|
|
if (lResult == ERROR_SUCCESS)
|
|
plci->pbEtag = pbEtag;
|
|
else
|
|
delete pbEtag;
|
|
}
|
|
}
|
|
|
|
// check file versions
|
|
if ((dwFileVersionMS > dwLocFVMS) ||
|
|
((dwFileVersionMS == dwLocFVMS) &&
|
|
(dwFileVersionLS > dwLocFVLS)))
|
|
hr = S_FALSE;
|
|
|
|
|
|
if (hr == S_OK) {
|
|
|
|
// if we seem to have the right version
|
|
// check if the file physically exists on disk
|
|
// the software can specify this by having a PATH="c:\foo\foo.class"
|
|
|
|
// if present we will check for physical file existance
|
|
|
|
dwType = 0;
|
|
Size = MAX_PATH;
|
|
lResult = ::SHQueryValueEx(hKeyVersion, szPATH, NULL, &dwType,
|
|
(unsigned char*)szFileName, &Size);
|
|
|
|
if (lResult != ERROR_SUCCESS)
|
|
goto Exit;
|
|
|
|
// value present, check if file is present
|
|
if (GetFileAttributes(szFileName) == -1) {
|
|
|
|
// if file is not physically present then clear out our
|
|
// local file version. this comes in the way of doing
|
|
// get latest. (ie get latest will assume that if a local file
|
|
// is present then do If-Modified-Since
|
|
plci->dwLocFVMS = 0;
|
|
plci->dwLocFVLS = 0;
|
|
|
|
hr = S_FALSE;
|
|
}
|
|
}
|
|
|
|
Exit:
|
|
|
|
if (hKeyVersion)
|
|
::RegCloseKey(hKeyVersion);
|
|
if (hkeyAvailableVersion)
|
|
::RegCloseKey(hkeyAvailableVersion);
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT
|
|
CreateJavaPackageManager(IJavaPackageManager** ppPackageManager)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
Assert(ppPackageManager);
|
|
|
|
if (!(*ppPackageManager)) {
|
|
ICreateJavaPackageMgr* picjpm;
|
|
|
|
hr = CoCreateInstance(CLSID_JavaPackageManager, NULL, (CLSCTX_INPROC_SERVER), IID_ICreateJavaPackageMgr, (LPVOID*)&picjpm);
|
|
if (SUCCEEDED(hr)) {
|
|
hr = picjpm->GetPackageManager(ppPackageManager);
|
|
picjpm->Release();
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT
|
|
IsPackageLocallyInstalled(IJavaPackageManager** ppPackageManager, LPCWSTR szPackageName, LPCWSTR szNameSpace, DWORD dwVersionMS, DWORD dwVersionLS)
|
|
{
|
|
HRESULT hr = S_OK; // assume Ok!
|
|
IJavaPackage* pJavaPkg = NULL;
|
|
DWORD dwLocMS = 0;
|
|
DWORD dwLocLS = 0;
|
|
|
|
Assert(ppPackageManager);
|
|
|
|
if (!(*ppPackageManager)) {
|
|
hr = CreateJavaPackageManager(ppPackageManager);
|
|
|
|
if (FAILED(hr))
|
|
goto Exit;
|
|
}
|
|
|
|
|
|
if (SUCCEEDED((*ppPackageManager)->GetPackage(szPackageName, szNameSpace, &pJavaPkg))) {
|
|
|
|
Assert(pJavaPkg);
|
|
|
|
pJavaPkg->GetVersion(&dwLocMS, &dwLocLS);
|
|
|
|
if ((dwVersionMS > dwLocMS) ||
|
|
((dwVersionMS == dwLocMS) &&
|
|
(dwVersionLS > dwLocLS)))
|
|
hr = S_FALSE;
|
|
|
|
|
|
BSTR bstrFileName = NULL;
|
|
if (SUCCEEDED(pJavaPkg->GetFilePath(&bstrFileName))) {
|
|
|
|
// check if file really exists
|
|
LPSTR szFileName = NULL;
|
|
|
|
if (SUCCEEDED(Unicode2Ansi(bstrFileName, &szFileName))) {
|
|
if (GetFileAttributes(szFileName) == -1)
|
|
hr = S_FALSE;
|
|
} else {
|
|
hr = S_FALSE;
|
|
}
|
|
} else {
|
|
hr = S_FALSE;
|
|
}
|
|
|
|
SAFESYSFREESTRING(bstrFileName);
|
|
SAFERELEASE(pJavaPkg);
|
|
|
|
} else {
|
|
hr = S_FALSE;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT
|
|
AreNameSpacePackagesIntact(HKEY hkeyJava, LPCWSTR lpwszNameSpace, IJavaPackageManager** ppPackageManager)
|
|
{
|
|
int iValue = 0;
|
|
DWORD dwType = REG_SZ;
|
|
DWORD dwValueSize = MAX_PATH;
|
|
char szPkgName[MAX_PATH];
|
|
HRESULT hr = S_OK;
|
|
|
|
while (
|
|
RegEnumValue(hkeyJava, iValue++, szPkgName, &dwValueSize, 0,
|
|
&dwType, NULL, NULL) == ERROR_SUCCESS) {
|
|
|
|
LPWSTR lpwszPkgName = NULL;
|
|
|
|
dwValueSize = MAX_PATH; // reset
|
|
|
|
if ((Ansi2Unicode(szPkgName, &lpwszPkgName) == S_OK)
|
|
&& ((IsPackageLocallyInstalled
|
|
(ppPackageManager, lpwszPkgName, lpwszNameSpace, 0, 0) != S_OK))) {
|
|
|
|
hr = S_FALSE;
|
|
SAFEDELETE(lpwszPkgName);
|
|
break;
|
|
}
|
|
SAFEDELETE(lpwszPkgName);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT
|
|
ArePackagesIntact(HKEY hkeyContains)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
HKEY hkeyJava = 0;
|
|
const static char* szJava = "Java";
|
|
IJavaPackageManager* pPackageManager = NULL;
|
|
DWORD iSubKey = 0;
|
|
DWORD dwSize = MAX_PATH;
|
|
DWORD lResult;
|
|
char szNameSpace[MAX_PATH];
|
|
|
|
// first validate pkgs in the global namespace
|
|
if (RegOpenKeyEx(hkeyContains, szJava,
|
|
0, KEY_READ, &hkeyJava) == ERROR_SUCCESS) {
|
|
|
|
hr = AreNameSpacePackagesIntact(hkeyJava, NULL, &pPackageManager);
|
|
|
|
if (hr != S_OK)
|
|
goto Exit;
|
|
} else {
|
|
goto Exit;
|
|
}
|
|
|
|
// validate pkgs in each of other namespaces, if any
|
|
while ((lResult = RegEnumKeyEx(hkeyJava, iSubKey++, szNameSpace, &dwSize, NULL, NULL, NULL, NULL)) == ERROR_SUCCESS) {
|
|
|
|
dwSize = MAX_PATH;
|
|
HKEY hkeyNameSpace = 0;
|
|
|
|
if (RegOpenKeyEx(hkeyJava, szNameSpace,
|
|
0, KEY_READ, &hkeyNameSpace) == ERROR_SUCCESS) {
|
|
|
|
LPWSTR lpwszNameSpace = NULL;
|
|
|
|
if (Ansi2Unicode(szNameSpace, &lpwszNameSpace) == S_OK)
|
|
hr = AreNameSpacePackagesIntact(hkeyNameSpace, lpwszNameSpace,
|
|
&pPackageManager);
|
|
|
|
SAFEDELETE(lpwszNameSpace);
|
|
RegCloseKey(hkeyNameSpace);
|
|
|
|
if (hr != S_OK)
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
|
|
if (lResult != ERROR_NO_MORE_ITEMS) {
|
|
hr = HRESULT_FROM_WIN32(lResult);
|
|
//FALLTHRU goto Exit;
|
|
}
|
|
|
|
Exit:
|
|
|
|
SAFERELEASE(pPackageManager);
|
|
|
|
if (hkeyJava)
|
|
RegCloseKey(hkeyJava);
|
|
|
|
return hr;
|
|
}
|
|
|
|
/*
|
|
|
|
NAME: IsDistUnitLocallyInstalled
|
|
|
|
SYNOPSIS: S_OK - distribution is installed correctly
|
|
S_FALSE - distribution unit entry exists, but not installed
|
|
or installed incorrectly
|
|
E_FAIL - distribution unit doesn't exit (no entry for it)
|
|
|
|
*/
|
|
|
|
HRESULT
|
|
IsDistUnitLocallyInstalled(
|
|
LPCWSTR szDistUnit,
|
|
DWORD dwFileVersionMS,
|
|
DWORD dwFileVersionLS,
|
|
CLocalComponentInfo* plci,
|
|
LPSTR szDestDirHint,
|
|
LPBOOL pbParanoidCheck,
|
|
DWORD flags)
|
|
{
|
|
LPSTR pszDist = NULL;
|
|
HRESULT hr = S_FALSE;
|
|
HKEY hkeyDist, hkeyThisDist = 0;
|
|
HKEY hkeyContains = 0;
|
|
HKEY hkeyFiles = 0;
|
|
HKEY hkeyDepends = 0;
|
|
const static char* szContains = "Contains";
|
|
const static char* szFiles = "Files";
|
|
const static char* szDistUnitStr = "Distribution Units";
|
|
LONG lResult = ERROR_SUCCESS;
|
|
char szFileName[MAX_PATH];
|
|
ULONG cbSize = MAX_PATH;
|
|
DWORD dwType = REG_SZ;
|
|
|
|
if (pbParanoidCheck) {
|
|
*pbParanoidCheck = FALSE;
|
|
}
|
|
|
|
lResult = RegOpenKeyEx(HKEY_LOCAL_MACHINE, REGSTR_PATH_DIST_UNITS,
|
|
0, KEY_READ, &hkeyDist);
|
|
|
|
if (lResult != ERROR_SUCCESS) {
|
|
hr = HRESULT_FROM_WIN32(REGDB_E_KEYMISSING);
|
|
goto Exit;
|
|
}
|
|
|
|
if (FAILED((hr = ::Unicode2Ansi(szDistUnit, &pszDist)))) {
|
|
goto Exit;
|
|
}
|
|
|
|
hr = S_FALSE; // reset to NOT found
|
|
|
|
// Open the key for this embedding:
|
|
|
|
lResult = ::RegOpenKeyEx(hkeyDist, pszDist, 0, KEY_READ,
|
|
&hkeyThisDist);
|
|
|
|
if (lResult == ERROR_SUCCESS) {
|
|
|
|
hr = CheckInstalledVersionHint(hkeyThisDist, plci,
|
|
dwFileVersionMS, dwFileVersionLS);
|
|
|
|
} else {
|
|
hr = HRESULT_FROM_WIN32(REGDB_E_KEYMISSING);
|
|
goto Exit;
|
|
}
|
|
|
|
if (hr == S_OK || (SUCCEEDED(hr) && dwFileVersionMS == -1 && dwFileVersionLS == -1)) {
|
|
|
|
if (RegOpenKeyEx(hkeyThisDist, szContains,
|
|
0, KEY_READ, &hkeyContains) == ERROR_SUCCESS) {
|
|
|
|
if (pbParanoidCheck) {
|
|
*pbParanoidCheck = TRUE;
|
|
}
|
|
|
|
// BUGBUG: only do if paranoid flag on?
|
|
|
|
// assert dependency state installed correctly on machine
|
|
// this is where we would have to walk the dependecy tree
|
|
// as well as make sure the contained packages and files
|
|
// are indeed availabel on the client machine
|
|
// BUGBUG: maybe we should only do this in paranoid mode
|
|
// instead of all the time.
|
|
|
|
if (RegOpenKeyEx(hkeyContains, szDistUnitStr,
|
|
0, KEY_READ, &hkeyDepends) == ERROR_SUCCESS) {
|
|
|
|
int iSubKey = 0;
|
|
while (RegEnumValue(hkeyDepends, iSubKey++,
|
|
szFileName, &cbSize, NULL, NULL, NULL, NULL) == ERROR_SUCCESS) {
|
|
|
|
CLocalComponentInfo lci;
|
|
LPWSTR wszFileName = 0;
|
|
CLSID clsidTemp;
|
|
|
|
if (FAILED(Ansi2Unicode(szFileName, &wszFileName)))
|
|
break;
|
|
|
|
clsidTemp = CLSID_NULL;
|
|
|
|
if (wszFileName)
|
|
CLSIDFromString(wszFileName, &clsidTemp);
|
|
// if above call fails DistUnit is not clsid
|
|
|
|
if (IsControlLocallyInstalled(NULL, &clsidTemp, wszFileName, 0, 0, &lci, NULL) != S_OK) {
|
|
SAFEDELETE(wszFileName);
|
|
|
|
plci->dwLocFVMS = 0;
|
|
plci->dwLocFVLS = 0;
|
|
|
|
hr = S_FALSE;
|
|
goto Exit;
|
|
}
|
|
|
|
SAFEDELETE(wszFileName);
|
|
cbSize = MAX_PATH;
|
|
}
|
|
}
|
|
|
|
if (RegOpenKeyEx(hkeyContains, szFiles,
|
|
0, KEY_READ, &hkeyFiles) == ERROR_SUCCESS) {
|
|
|
|
int iValue = 0;
|
|
DWORD dwType = REG_SZ;
|
|
DWORD dwValueSize = MAX_PATH;
|
|
|
|
while (RegEnumValue(hkeyFiles, iValue++,
|
|
szFileName, &dwValueSize, 0, &dwType, NULL, NULL) == ERROR_SUCCESS) {
|
|
|
|
dwValueSize = MAX_PATH; // reset
|
|
|
|
if (GetFileAttributes(szFileName) == -1) {
|
|
|
|
// if file is not physically present then clear out our
|
|
// local file version. this comes in the way of doing
|
|
// get latest. (get latest will assume that if a local
|
|
// file is present then do If-Modified-Since
|
|
|
|
plci->dwLocFVMS = 0;
|
|
plci->dwLocFVLS = 0;
|
|
|
|
hr = S_FALSE;
|
|
goto Exit;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ArePackagesIntact(hkeyContains) == S_FALSE) {
|
|
plci->dwLocFVMS = 0;
|
|
plci->dwLocFVLS = 0;
|
|
hr = S_FALSE;
|
|
goto Exit;
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
hr = S_FALSE; // mark not present, don't error out
|
|
}
|
|
|
|
|
|
Exit:
|
|
if (pszDist)
|
|
delete pszDist;
|
|
|
|
if (hkeyDepends)
|
|
::RegCloseKey(hkeyDepends);
|
|
|
|
if (hkeyFiles)
|
|
::RegCloseKey(hkeyFiles);
|
|
|
|
if (hkeyContains)
|
|
::RegCloseKey(hkeyContains);
|
|
|
|
if (hkeyThisDist)
|
|
::RegCloseKey(hkeyThisDist);
|
|
|
|
if (hkeyDist)
|
|
::RegCloseKey(hkeyDist);
|
|
|
|
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
IsFileLocallyInstalled(
|
|
LPSTR lpCurCode,
|
|
DWORD dwFileVersionMS,
|
|
DWORD dwFileVersionLS,
|
|
CLocalComponentInfo* plci,
|
|
LPSTR szDestDirHint,
|
|
BOOL bExactVersion
|
|
)
|
|
{
|
|
HRESULT hr = S_FALSE;
|
|
|
|
// no clsid, but we have a file name
|
|
|
|
// first check for a file of the same name in DestDirHint.
|
|
// This is the directory that the main OCX file existed on
|
|
// and so should be checked first.
|
|
|
|
// In case this is a new install this will point to the
|
|
// suggested destination dir for the DLL.
|
|
|
|
if (szDestDirHint) {
|
|
|
|
if (SearchPath(szDestDirHint,
|
|
lpCurCode, NULL, MAX_PATH,
|
|
plci->szExistingFileName, &(plci->pBaseExistingFileName))) {
|
|
|
|
// check fileversion to see if update reqd.
|
|
hr = LocalVersionOK(0, plci, dwFileVersionMS, dwFileVersionLS, bExactVersion);
|
|
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
// file not found in suggested destination. Look in system searchpath
|
|
// SearchPath for this filename
|
|
|
|
if (!(SearchPath(NULL, lpCurCode, NULL, MAX_PATH,
|
|
plci->szExistingFileName, &(plci->pBaseExistingFileName)))) {
|
|
hr = S_FALSE;
|
|
goto Exit;
|
|
}
|
|
|
|
// check fileversion to see if update reqd.
|
|
hr = LocalVersionOK(0, plci, dwFileVersionMS, dwFileVersionLS, bExactVersion);
|
|
|
|
Exit:
|
|
|
|
return hr;
|
|
}
|
|
|
|
BOOL GetEXEName(LPSTR szCmdLine)
|
|
{
|
|
|
|
Assert(szCmdLine);
|
|
|
|
LPSTR pchStartBaseName = szCmdLine;
|
|
BOOL bFullyQualified = FALSE;
|
|
BOOL bHasSpaces = FALSE;
|
|
char* pch;
|
|
|
|
if (*szCmdLine == '"') {
|
|
szCmdLine++;
|
|
char* pszEnd = StrStrA(szCmdLine, "\"");
|
|
|
|
ASSERT(pszEnd);
|
|
*pszEnd = '\0';
|
|
|
|
if (GetFileAttributes(szCmdLine) != -1) {
|
|
//found the EXE name, but got to get rid of the first quote
|
|
|
|
szCmdLine--; // step back to the "
|
|
while (*szCmdLine) {
|
|
*szCmdLine = *(szCmdLine + 1);
|
|
szCmdLine++;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
szCmdLine--; // step back to the "
|
|
while (*szCmdLine) {
|
|
*szCmdLine = *(szCmdLine + 1);
|
|
szCmdLine++;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
// skip past the directory if fully qualified.
|
|
for (pch = szCmdLine; *pch; pch++) {
|
|
|
|
if (*pch == '\\')
|
|
pchStartBaseName = pch;
|
|
|
|
if ((*pch == ' ') || (*pch == '\t'))
|
|
bHasSpaces = TRUE;
|
|
}
|
|
|
|
if (!bHasSpaces) {
|
|
if (GetFileAttributes(szCmdLine) != -1) {
|
|
//found the EXE name, it is already in szCmdLine
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
// pchStartBaseName now points at the last '\\' if any
|
|
if (*pchStartBaseName == '\\') {
|
|
pchStartBaseName++;
|
|
|
|
bFullyQualified = TRUE;
|
|
}
|
|
|
|
// pchStartBaseName no points at the base name of the EXE.
|
|
|
|
// Now look for spaces. When we find a space we will
|
|
// replace with a '\0' and check if the cmd line is valid.
|
|
// if valid, this must be the EXE name, return
|
|
// if not valid, march on to do the same. when we finish
|
|
// examining all of the cmd line, or no more spaces, likely the
|
|
// EXE is missing.
|
|
|
|
for (pch = pchStartBaseName; *pch != '\0'; pch++) {
|
|
|
|
if ((*pch == ' ') || (*pch == '\t')) {
|
|
|
|
char chTemp = *pch; // sacve the white spc char
|
|
|
|
*pch = '\0'; // stomp the spc with nul. now we have in szCmdLine
|
|
// what could be the full EXE name
|
|
|
|
if (bFullyQualified) {
|
|
|
|
if (GetFileAttributes(szCmdLine) != -1) {
|
|
//found the EXE name, it is already in szCmdLine
|
|
return TRUE;
|
|
}
|
|
|
|
} else {
|
|
|
|
char szBuf[MAX_PATH];
|
|
LPSTR pBaseFileName;
|
|
|
|
if (SearchPath(NULL, szCmdLine, NULL, MAX_PATH,
|
|
szBuf, &pBaseFileName)) {
|
|
|
|
//found the EXE name, it is already in szCmdLine
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
*pch = chTemp; // restore the while spc and move past.
|
|
}
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL
|
|
AdviseForceDownload(const LPCLSID lpclsid, DWORD dwClsContext)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
BOOL bNullClsid = lpclsid ? IsEqualGUID(*lpclsid, CLSID_NULL) : TRUE;
|
|
HKEY hKeyClsid = 0;
|
|
HKEY hKeyEmbedding = 0;
|
|
HKEY hkeyDist = 0;
|
|
BOOL bForceDownload = TRUE;
|
|
LPOLESTR pwcsClsid = NULL;
|
|
DWORD dwType;
|
|
LONG lResult = ERROR_SUCCESS;
|
|
static char* szAppID = "AppID";
|
|
LPSTR pszClsid = NULL;
|
|
CLocalComponentInfo lci;
|
|
static char* szInprocServer32 = "InProcServer32";
|
|
static char* szLocalServer32 = "LocalServer32";
|
|
|
|
if (bNullClsid)
|
|
goto Exit;
|
|
|
|
// return if we can't get a valid string representation of the CLSID
|
|
if (FAILED((hr = StringFromCLSID(*lpclsid, &pwcsClsid))))
|
|
goto Exit;
|
|
|
|
Assert(pwcsClsid != NULL);
|
|
|
|
|
|
if (FAILED((hr = ::Unicode2Ansi(pwcsClsid, &pszClsid)))) {
|
|
goto Exit;
|
|
}
|
|
|
|
// Open root HKEY_CLASSES_ROOT\CLSID key
|
|
lResult = ::RegOpenKeyEx(HKEY_CLASSES_ROOT, "CLSID", 0, KEY_READ, &hKeyClsid);
|
|
|
|
|
|
if (lResult == ERROR_SUCCESS) {
|
|
|
|
// Open the key for this embedding:
|
|
lResult = ::RegOpenKeyEx(hKeyClsid, pszClsid, 0, KEY_READ,
|
|
&hKeyEmbedding);
|
|
|
|
if (lResult == ERROR_SUCCESS) {
|
|
|
|
// check for hint of FileVersion before actually getting FileVersion
|
|
// This way non-PE files, like Java, random data etc. can be
|
|
// accomodated with CODEBASE= with proper version checking.
|
|
// If they are not really COM object then they can't be
|
|
// instantiated by COM or us (if object needed). But if they
|
|
// use AsyncGetClassBits(no object instantiated) or from the
|
|
// browser give height=0 width=0, the browser will not
|
|
// report placeholders with errors, so they can work gracefully
|
|
// overloading the object tag to do version checking and
|
|
// random stuff
|
|
|
|
hr = CheckInstalledVersionHint(hKeyEmbedding, &lci,
|
|
0, 0);
|
|
|
|
// if the key is found and
|
|
// if the latest version is not available then
|
|
// return now with false, if right version is already
|
|
// present return. If the key is missing then we
|
|
// proceed with checks for InprocServer32/LocalServer32
|
|
if (SUCCEEDED(hr)) {
|
|
|
|
// if using installed version hint and the verdict is
|
|
// local good enough
|
|
if (hr == S_OK)
|
|
bForceDownload = FALSE;
|
|
|
|
goto Exit;
|
|
}
|
|
|
|
|
|
hr = S_OK; // reset
|
|
|
|
|
|
// ckeck if DCOM
|
|
HKEY hKeyAppID;
|
|
lResult = ::RegOpenKeyEx(hKeyEmbedding, szAppID, 0,
|
|
KEY_READ, &hKeyAppID);
|
|
if (lResult == ERROR_SUCCESS) {
|
|
|
|
// DCOM
|
|
// just assume that this is the latest version already
|
|
// we never attempt code download for dcom
|
|
|
|
bForceDownload = FALSE; // no force download for DCOM
|
|
|
|
RegCloseKey(hKeyAppID);
|
|
goto Exit;
|
|
}
|
|
|
|
HKEY hKeyInProc;
|
|
lResult = ::RegOpenKeyEx(hKeyEmbedding, szInprocServer32, 0,
|
|
KEY_READ, &hKeyInProc);
|
|
|
|
if (lResult != ERROR_SUCCESS) {
|
|
if (RegOpenKeyEx(hKeyEmbedding, szLocalServer32, 0,
|
|
KEY_READ, &hKeyInProc) == ERROR_SUCCESS) {
|
|
|
|
// specific look for vb doc obj hack where they just use
|
|
// the OBJECT tag to do code download, but not hosting
|
|
|
|
// no inproc but we have localserver
|
|
// we could have failed because we pass the wrong clsctx
|
|
|
|
RegCloseKey(hKeyInProc);
|
|
if (!(dwClsContext & CLSCTX_LOCAL_SERVER))
|
|
bForceDownload = FALSE;
|
|
}
|
|
|
|
} else {
|
|
RegCloseKey(hKeyInProc);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
lResult = RegOpenKeyEx(HKEY_LOCAL_MACHINE, REGSTR_PATH_DIST_UNITS,
|
|
0, KEY_READ, &hkeyDist);
|
|
|
|
if (lResult == ERROR_SUCCESS) {
|
|
|
|
// Open the key for this embedding:
|
|
HKEY hkeyThisDist = 0;
|
|
|
|
if (RegOpenKeyEx(hkeyDist, pszClsid, 0, KEY_READ,
|
|
&hkeyThisDist) == ERROR_SUCCESS) {
|
|
|
|
HKEY hkeyJava = 0;
|
|
if (RegOpenKeyEx(hkeyThisDist, "Contains\\Java", 0, KEY_READ,
|
|
&hkeyJava) == ERROR_SUCCESS) {
|
|
bForceDownload = FALSE;
|
|
RegCloseKey(hkeyJava);
|
|
|
|
}
|
|
|
|
RegCloseKey(hkeyThisDist);
|
|
}
|
|
|
|
RegCloseKey(hkeyDist);
|
|
}
|
|
|
|
|
|
|
|
Exit:
|
|
if (pwcsClsid)
|
|
delete pwcsClsid;
|
|
|
|
if (pszClsid)
|
|
delete pszClsid;
|
|
|
|
if (hKeyClsid)
|
|
::RegCloseKey(hKeyClsid);
|
|
if (hKeyEmbedding)
|
|
::RegCloseKey(hKeyEmbedding);
|
|
|
|
|
|
return bForceDownload;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
NAME: IsControlLocallyInstalled
|
|
|
|
SYNOPSIS: Indicates whether the provided CLSID represents an
|
|
OLE control.
|
|
If no clsid provided then checks to see if lpCurCode
|
|
exists in system and checks file version to verify if
|
|
update is needed
|
|
|
|
*/
|
|
HRESULT
|
|
IsControlLocallyInstalled(LPSTR lpCurCode, const LPCLSID lpclsid,
|
|
LPCWSTR szDistUnit,
|
|
DWORD dwFileVersionMS, DWORD dwFileVersionLS,
|
|
CLocalComponentInfo* plci,
|
|
LPSTR szDestDirHint,
|
|
BOOL bExactVersion)
|
|
{
|
|
HRESULT hr1, hr2, hrResult, hr = S_FALSE;
|
|
BOOL bNullClsid = lpclsid ? IsEqualGUID(*lpclsid, CLSID_NULL) : TRUE;
|
|
BOOL bParanoidCheck = FALSE;
|
|
#ifdef _ZEROIMPACT
|
|
LPOLESTR wszClsid = NULL;
|
|
StringFromCLSID(*lpclsid, &wszClsid);
|
|
#endif
|
|
|
|
if (bNullClsid && (lpCurCode == NULL) && (szDistUnit == NULL)) {
|
|
|
|
hr = E_INVALIDARG;
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
// hr1: HRESULT for whether the dist unit is installed
|
|
// hr2: HRESULT for whether the particular clsid is installed
|
|
// hrResult: HRESULT for whether the particular clsid is present (but not necessarily installed correctly)
|
|
|
|
#ifdef _ZEROIMPACT
|
|
// check for an exact match in the ZeroImpact installations first
|
|
if (szDistUnit) {
|
|
hr = IsDistUnitLocallyInstalledZI(szDistUnit, wszClsid, dwFileVersionMS, dwFileVersionLS, plci);
|
|
}
|
|
|
|
if (hr == S_OK) {
|
|
goto Exit;
|
|
}
|
|
|
|
plci->m_bIsZI = FALSE;
|
|
#endif
|
|
|
|
if (szDistUnit) {
|
|
hr1 = IsDistUnitLocallyInstalled(szDistUnit, dwFileVersionMS, dwFileVersionLS, plci, szDestDirHint, &bParanoidCheck, 0);
|
|
} else {
|
|
|
|
if (bNullClsid) {
|
|
hr1 = IsFileLocallyInstalled(lpCurCode, dwFileVersionMS, dwFileVersionLS, plci, szDestDirHint, bExactVersion);
|
|
|
|
// if no dist unit name or clsid and this is
|
|
// clearly just checking for dependent file then
|
|
// no need to fall thru and check the com br as well
|
|
|
|
hr = hr1;
|
|
goto Exit;
|
|
} else {
|
|
hr1 = E_FAIL;
|
|
}
|
|
|
|
}
|
|
|
|
hr2 = IsCLSIDLocallyInstalled(lpCurCode, lpclsid, szDistUnit, dwFileVersionMS, dwFileVersionLS, plci, szDestDirHint, &hrResult, bExactVersion);
|
|
|
|
if (hr2 != S_OK) {
|
|
// if HKLM\CLSID\{CLSID} existed, but control wasn't there, we fail with that error
|
|
// otherwise we fail with hr1.
|
|
if (SUCCEEDED(hrResult)) {
|
|
hr = hr2;
|
|
} else {
|
|
// if DU check returned S_FALSE or S_OK we return that, otherwise return from CLSID check.
|
|
|
|
if (SUCCEEDED(hr1)) {
|
|
hr = hr1;
|
|
} else {
|
|
hr = hr2;
|
|
}
|
|
}
|
|
} else {
|
|
if (hr1 == S_FALSE) {
|
|
// COM branch says we are OK, but Distribution unit says we are lacking.
|
|
// if we did paranoid checking and then failed the DU then
|
|
// really fail. But if we just looked at the InstalledVersion
|
|
// in the registry and concluded that our DU is no good then we
|
|
// should go by the COM br and succeed the call as the user
|
|
// could have obtained a newer version thru a mechanism other than
|
|
// code download and so we have no business trying to update this
|
|
|
|
// BUGBUG: do we at this point try to correct our registry
|
|
// record of the version? need to ship tomorrow!
|
|
|
|
if (bParanoidCheck)
|
|
hr = hr1;
|
|
else
|
|
hr = hr2;
|
|
|
|
} else {
|
|
hr = hr2;
|
|
}
|
|
}
|
|
|
|
Exit:
|
|
#ifdef _ZEROIMPACT
|
|
if (wszClsid)
|
|
delete wszClsid;
|
|
#endif
|
|
return hr;
|
|
}
|
|
|
|
/*
|
|
|
|
NAME: IsCLSIDLocallyInstalled
|
|
|
|
SYNOPSIS: Indicates whether the provided CLSID represents an
|
|
OLE control.
|
|
If no clsid provided then checks to see if lpCurCode
|
|
exists in system and checks file version to verify if
|
|
update is needed
|
|
|
|
*/
|
|
HRESULT
|
|
IsCLSIDLocallyInstalled(LPSTR lpCurCode, const LPCLSID lpclsid,
|
|
LPCWSTR szDistUnit,
|
|
DWORD dwFileVersionMS, DWORD dwFileVersionLS,
|
|
CLocalComponentInfo* plci,
|
|
LPSTR szDestDirHint,
|
|
HRESULT* pHrExtra,
|
|
BOOL bExactVersion
|
|
)
|
|
|
|
|
|
{
|
|
LPSTR pszClsid = NULL;
|
|
LPOLESTR pwcsClsid = NULL;
|
|
HRESULT hr = S_FALSE;
|
|
DWORD dwType;
|
|
LONG lResult = ERROR_SUCCESS;
|
|
static char* szInprocServer32 = "InProcServer32";
|
|
static char* szLocalServer32 = "LocalServer32";
|
|
static char* szAppID = "AppID";
|
|
HKEY hKeyClsid = 0;
|
|
DWORD Size = MAX_PATH;
|
|
|
|
if (pHrExtra)
|
|
*pHrExtra = E_FAIL;
|
|
|
|
// return if we can't get a valid string representation of the CLSID
|
|
if (FAILED((hr = StringFromCLSID(*lpclsid, &pwcsClsid))))
|
|
goto Exit;
|
|
|
|
Assert(pwcsClsid != NULL);
|
|
|
|
// Open root HKEY_CLASSES_ROOT\CLSID key
|
|
lResult = ::RegOpenKeyEx(HKEY_CLASSES_ROOT, "CLSID", 0, KEY_READ, &hKeyClsid);
|
|
|
|
if (lResult == ERROR_SUCCESS) {
|
|
if (FAILED((hr = ::Unicode2Ansi(pwcsClsid, &pszClsid)))) {
|
|
goto Exit;
|
|
}
|
|
|
|
// Open the key for this embedding:
|
|
HKEY hKeyEmbedding;
|
|
HKEY hKeyInProc;
|
|
|
|
lResult = ::RegOpenKeyEx(hKeyClsid, pszClsid, 0, KEY_READ,
|
|
&hKeyEmbedding);
|
|
|
|
if (lResult == ERROR_SUCCESS) {
|
|
|
|
// check for hint of FileVersion before actually getting FileVersion
|
|
// This way non-PE files, like Java, random data etc. can be
|
|
// accomodated with CODEBASE= with proper version checking.
|
|
// If they are not really COM object then they can't be
|
|
// instantiated by COM or us (if object needed). But if they
|
|
// use AsyncGetClassBits(no object instantiated) or from the
|
|
// browser give height=0 width=0, the browser will not
|
|
// report placeholders with errors, so they can work gracefully
|
|
// overloading the object tag to do version checking and
|
|
// random stuff
|
|
|
|
if (pHrExtra) {
|
|
// indicate that CLSID reg key exists, so any failures after this
|
|
// imply the control is not registered correctly
|
|
*pHrExtra = S_OK;
|
|
}
|
|
|
|
hr = CheckInstalledVersionHint(hKeyEmbedding, plci,
|
|
dwFileVersionMS, dwFileVersionLS);
|
|
|
|
// if the key is found and
|
|
// if the latest version is not available then
|
|
// return now with false, if right version is already
|
|
// present return. If the key is missing then we
|
|
// proceed with checks for InprocServer32/LocalServer32
|
|
if (SUCCEEDED(hr))
|
|
goto finish_all;
|
|
|
|
hr = S_OK; // reset
|
|
|
|
// ckeck if DCOM
|
|
HKEY hKeyAppID;
|
|
lResult = ::RegOpenKeyEx(hKeyEmbedding, szAppID, 0,
|
|
KEY_READ, &hKeyAppID);
|
|
if (lResult == ERROR_SUCCESS) {
|
|
|
|
// DCOM
|
|
// just assume that this is the latest version already
|
|
// we never attempt code download for dcom
|
|
|
|
::RegCloseKey(hKeyAppID);
|
|
goto finish_all;
|
|
}
|
|
|
|
|
|
lResult = ::RegOpenKeyEx(hKeyEmbedding, szInprocServer32, 0,
|
|
KEY_READ, &hKeyInProc);
|
|
|
|
if (lResult == ERROR_SUCCESS) {
|
|
|
|
Size = MAX_PATH;
|
|
lResult = ::SHQueryValueEx(hKeyInProc, NULL, NULL, &dwType,
|
|
(unsigned char*)plci->szExistingFileName, &Size);
|
|
|
|
if (lResult == ERROR_SUCCESS) {
|
|
|
|
if (!(SearchPath(NULL,
|
|
plci->szExistingFileName, NULL, MAX_PATH,
|
|
plci->szExistingFileName, &(plci->pBaseExistingFileName)))) {
|
|
hr = S_FALSE;
|
|
goto finish_verchecks;
|
|
}
|
|
|
|
// check fileversion to see if update reqd.
|
|
hr = LocalVersionOK(hKeyEmbedding, plci,
|
|
dwFileVersionMS, dwFileVersionLS, bExactVersion);
|
|
|
|
if (plci->bForceLangGetLatest) {
|
|
hr = NeedForceLanguageCheck(hKeyEmbedding, plci);
|
|
}
|
|
|
|
goto finish_verchecks;
|
|
|
|
} else {
|
|
hr = S_FALSE; // problem: can't locate file
|
|
goto finish_verchecks;
|
|
}
|
|
|
|
} else {
|
|
|
|
lResult = ::RegOpenKeyEx(hKeyEmbedding, szLocalServer32, 0,
|
|
KEY_READ, &hKeyInProc);
|
|
|
|
if (lResult != ERROR_SUCCESS) {
|
|
hr = S_FALSE; // problem :have a clsid but, can't locate it
|
|
goto finish_all;
|
|
}
|
|
|
|
Size = MAX_PATH;
|
|
lResult = ::SHQueryValueEx(hKeyInProc, NULL, NULL, &dwType,
|
|
(unsigned char*)plci->szExistingFileName, &Size);
|
|
|
|
if (lResult == ERROR_SUCCESS) {
|
|
|
|
// strip out args if any for this localserver32
|
|
// and extract only the EXE name
|
|
|
|
GetEXEName(plci->szExistingFileName);
|
|
|
|
if (!(SearchPath(NULL,
|
|
plci->szExistingFileName, NULL, MAX_PATH,
|
|
plci->szExistingFileName, &(plci->pBaseExistingFileName)))) {
|
|
hr = S_FALSE;
|
|
goto finish_verchecks;
|
|
}
|
|
|
|
// check fileversion to see if update reqd.
|
|
hr = LocalVersionOK(hKeyEmbedding, plci, dwFileVersionMS, dwFileVersionLS, bExactVersion);
|
|
|
|
if (plci->bForceLangGetLatest)
|
|
hr = NeedForceLanguageCheck(hKeyEmbedding, plci);
|
|
|
|
goto finish_verchecks;
|
|
|
|
} else {
|
|
hr = S_FALSE; // problem: can't locate file
|
|
goto finish_verchecks;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
finish_verchecks:
|
|
::RegCloseKey(hKeyInProc);
|
|
|
|
finish_all:
|
|
|
|
::RegCloseKey(hKeyEmbedding);
|
|
|
|
} else {
|
|
// here if we could not find the embedding in HKCR\CLSID
|
|
|
|
hr = S_FALSE;
|
|
|
|
}
|
|
|
|
} else
|
|
hr = S_FALSE;
|
|
|
|
Exit:
|
|
|
|
// release the string allocated by StringFromCLSID
|
|
if (pwcsClsid)
|
|
delete pwcsClsid;
|
|
|
|
if (pszClsid)
|
|
delete pszClsid;
|
|
|
|
if (hKeyClsid)
|
|
::RegCloseKey(hKeyClsid);
|
|
|
|
return hr;
|
|
}
|
|
|
|
BOOL SupportsSelfRegister(LPSTR szFileName)
|
|
{
|
|
return SniffStringFileInfo(szFileName, TEXT("OLESelfRegister"));
|
|
}
|
|
|
|
BOOL WantsAutoExpire(LPSTR szFileName, DWORD* pnExpireDays)
|
|
{
|
|
return SniffStringFileInfo(szFileName, TEXT("Expire"), pnExpireDays);
|
|
}
|
|
|
|
|
|
HRESULT GetFileVersion(CLocalComponentInfo* plci, LPDWORD pdwFileVersionMS, LPDWORD pdwFileVersionLS)
|
|
{
|
|
DWORD handle;
|
|
UINT uiInfoSize;
|
|
UINT uiVerSize;
|
|
UINT uiSize;
|
|
BYTE* pbData = NULL;
|
|
VS_FIXEDFILEINFO* lpVSInfo;;
|
|
HRESULT hr = S_OK;
|
|
LPVOID lpVerBuffer = NULL;
|
|
|
|
#ifdef UNIX
|
|
// We don't have version.dll
|
|
DebugBreak();
|
|
return E_INVALIDARG;
|
|
#endif
|
|
|
|
if (!pdwFileVersionMS || !pdwFileVersionLS) {
|
|
hr = E_INVALIDARG;
|
|
goto Exit;
|
|
}
|
|
|
|
*pdwFileVersionMS = 0;
|
|
*pdwFileVersionLS = 0;
|
|
|
|
// Get the size of the version information.
|
|
uiInfoSize = g_versiondll.GetFileVersionInfoSize((char*)plci->szExistingFileName, &handle);
|
|
|
|
if (uiInfoSize == 0) {
|
|
hr = S_FALSE;
|
|
goto Exit;
|
|
}
|
|
|
|
// Allocate a buffer for the version information.
|
|
pbData = new BYTE[uiInfoSize];
|
|
if (!pbData)
|
|
return E_OUTOFMEMORY;
|
|
|
|
// Fill the buffer with the version information.
|
|
if (!g_versiondll.GetFileVersionInfo((char*)plci->szExistingFileName, handle, uiInfoSize, pbData)) {
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
goto Exit;
|
|
}
|
|
|
|
// Get the translation information.
|
|
if (!g_versiondll.VerQueryValue(pbData, "\\", (void**)&lpVSInfo, &uiVerSize)) {
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
goto Exit;
|
|
}
|
|
|
|
if (!uiVerSize) {
|
|
hr = E_FAIL;
|
|
goto Exit;
|
|
}
|
|
|
|
*pdwFileVersionMS = lpVSInfo->dwFileVersionMS;
|
|
*pdwFileVersionLS = lpVSInfo->dwFileVersionLS;
|
|
|
|
// Get the translation information.
|
|
if (!g_versiondll.VerQueryValue(pbData, "\\VarFileInfo\\Translation", &lpVerBuffer, &uiVerSize)) {
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
goto Exit;
|
|
}
|
|
|
|
if (!uiVerSize) {
|
|
hr = E_FAIL;
|
|
goto Exit;
|
|
}
|
|
|
|
plci->lcid = LOWORD(*((DWORD*)lpVerBuffer)); // Language ID
|
|
|
|
Exit:
|
|
|
|
if (pbData)
|
|
delete[] pbData;
|
|
|
|
return hr;
|
|
}
|
|
|
|
DWORD
|
|
GetLanguageCheckInterval(HKEY hkeyCheckPeriod)
|
|
{
|
|
DWORD dwMagicDays = 30;
|
|
|
|
DWORD dwType = REG_DWORD;
|
|
DWORD dwSize = sizeof(DWORD);
|
|
char szLangString[MAX_PATH];
|
|
|
|
LCID lcidPriOverride = PRIMARYLANGID(LANGIDFROMLCID(GetThreadLocale()));
|
|
LCID lcidPriBrowser = PRIMARYLANGID(LANGIDFROMLCID(g_lcidBrowser));
|
|
|
|
if (SUCCEEDED(GetLangString(lcidPriOverride, szLangString))) {
|
|
|
|
if (RegQueryValueEx(hkeyCheckPeriod, szLangString, NULL, &dwType,
|
|
(unsigned char*)&dwMagicDays, &dwSize) == ERROR_SUCCESS) {
|
|
|
|
return dwMagicDays;
|
|
}
|
|
}
|
|
|
|
if ((lcidPriOverride != lcidPriBrowser) &&
|
|
SUCCEEDED(GetLangString(lcidPriBrowser, szLangString))) {
|
|
|
|
if (RegQueryValueEx(hkeyCheckPeriod, szLangString, NULL, &dwType,
|
|
(unsigned char*)&dwMagicDays, &dwSize) == ERROR_SUCCESS) {
|
|
|
|
return dwMagicDays;
|
|
}
|
|
}
|
|
|
|
return dwMagicDays;
|
|
}
|
|
|
|
// returns:
|
|
// S_OK: local version OK or local version lang check not reqd now
|
|
// S_FALSE: localversion not of right lang force lang check now
|
|
// ERROR: fail
|
|
|
|
HRESULT NeedForceLanguageCheck(HKEY hkeyCLSID, CLocalComponentInfo* plci)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD lResult;
|
|
const char* szCHECKPERIOD = "LanguageCheckPeriod";
|
|
const char* szLASTCHECKEDHI = "LastCheckedHi";
|
|
DWORD dwMagicPerDay = 201;
|
|
DWORD dwMagicDays;
|
|
DWORD dwType;
|
|
DWORD dwSize;
|
|
FILETIME ftlast, ftnow;
|
|
SYSTEMTIME st;
|
|
HKEY hkeyCheckPeriod = 0;
|
|
char szLangEnable[MAX_PATH];
|
|
|
|
|
|
if (!plci->bForceLangGetLatest)
|
|
return hr;
|
|
|
|
// lang is mismatched for this browser
|
|
// check when was the last time we checked for the right lang
|
|
|
|
if ((lResult = RegOpenKeyEx(hkeyCLSID, szCHECKPERIOD,
|
|
0, KEY_READ, &hkeyCheckPeriod)) != ERROR_SUCCESS) {
|
|
plci->bForceLangGetLatest = FALSE;
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
szLangEnable[0] = '\0';
|
|
dwType = REG_SZ;
|
|
dwSize = MAX_PATH;
|
|
if ((RegQueryValueEx(hkeyCheckPeriod, NULL, NULL, &dwType,
|
|
(unsigned char*)szLangEnable, &dwSize) != ERROR_SUCCESS) ||
|
|
lstrcmpi(szLangEnable, "Enabled") != 0) {
|
|
|
|
plci->bForceLangGetLatest = FALSE;
|
|
goto Exit;
|
|
}
|
|
|
|
// see if lang check interval is specified for this lang
|
|
dwMagicDays = GetLanguageCheckInterval(hkeyCheckPeriod);
|
|
|
|
GetSystemTime(&st);
|
|
SystemTimeToFileTime(&st, &ftnow);
|
|
ftnow.dwLowDateTime = 0;
|
|
|
|
memset(&ftlast, 0, sizeof(FILETIME));
|
|
|
|
dwType = REG_DWORD;
|
|
dwSize = sizeof(DWORD);
|
|
if (RegQueryValueEx(hkeyCheckPeriod, szLASTCHECKEDHI, NULL, &dwType,
|
|
(unsigned char*)&ftlast.dwHighDateTime, &dwSize) == ERROR_SUCCESS) {
|
|
|
|
ftlast.dwHighDateTime += (dwMagicPerDay * dwMagicDays);
|
|
}
|
|
|
|
if (CompareFileTime(&ftlast, &ftnow) > 0) {
|
|
plci->bForceLangGetLatest = FALSE;
|
|
}
|
|
|
|
|
|
Exit:
|
|
|
|
SAFEREGCLOSEKEY(hkeyCheckPeriod);
|
|
|
|
if (FAILED(hr))
|
|
plci->bForceLangGetLatest = FALSE;
|
|
|
|
return plci->bForceLangGetLatest ? S_FALSE : S_OK;
|
|
}
|
|
|
|
HRESULT IsRightLanguageLocallyInstalled(CLocalComponentInfo* plci)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
LCID lcidLocalVersion;
|
|
LCID lcidNeeded;
|
|
|
|
if (!plci->lcid) // lang neutral?
|
|
goto Exit;
|
|
|
|
// make sure that the browser locale and lang strings are
|
|
// initialized at this point
|
|
|
|
hr = InitBrowserLangStrings();
|
|
|
|
if (FAILED(hr))
|
|
goto Exit;
|
|
|
|
// BUGBUG: we are using threadlocale here instead of the
|
|
// bindopts from the bindctx passed in
|
|
if (plci->lcid == GetThreadLocale()) // full match with override?
|
|
goto Exit;
|
|
if (plci->lcid == g_lcidBrowser) // full match with browser?
|
|
goto Exit;
|
|
|
|
// get primary lang of local version
|
|
lcidLocalVersion = PRIMARYLANGID(LANGIDFROMLCID(plci->lcid));
|
|
|
|
// check with primary language of override
|
|
lcidNeeded = PRIMARYLANGID(LANGIDFROMLCID(GetThreadLocale()));
|
|
|
|
if (lcidLocalVersion == lcidNeeded) // same primary lang?
|
|
goto Exit;
|
|
|
|
// check with primary language of browser
|
|
lcidNeeded = PRIMARYLANGID(LANGIDFROMLCID(g_lcidBrowser));
|
|
|
|
if (lcidLocalVersion == lcidNeeded) // same primary lang?
|
|
goto Exit;
|
|
|
|
|
|
// BUGBUG: how to detect language neutral or multiligual OCX
|
|
|
|
// we have a mismatch
|
|
// check when was the last time we check and if past the
|
|
// interval to check for new language availability
|
|
// force a download now.
|
|
|
|
hr = S_FALSE;
|
|
|
|
plci->bForceLangGetLatest = TRUE;
|
|
|
|
Exit:
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT LocalVersionOK(HKEY hkeyCLSID, CLocalComponentInfo* plci, DWORD dwFileVersionMS, DWORD dwFileVersionLS, BOOL bExactVersion)
|
|
{
|
|
DWORD handle;
|
|
HRESULT hr = S_OK; // assume local version OK.
|
|
DWORD dwLocFVMS = 0;
|
|
DWORD dwLocFVLS = 0;
|
|
HKEY hkeyCheckPeriod = 0;
|
|
const char* szCHECKPERIOD = "LanguageCheckPeriod";
|
|
|
|
if (FAILED(hr = plci->MakeDestDir())) {
|
|
goto Exit;
|
|
}
|
|
|
|
#ifdef UNIX
|
|
return S_OK;
|
|
#endif
|
|
|
|
if ((dwFileVersionMS == 0) && (dwFileVersionLS == 0)) {
|
|
|
|
// for dlls that don't require lang support and have no version req
|
|
// don't check version numbers
|
|
// this is to boost perf on system/IE dlls
|
|
|
|
// One can also avoid such checks
|
|
// by adding a [InstalledVersion] key under the clsid
|
|
|
|
if (!hkeyCLSID || RegOpenKeyEx(hkeyCLSID, szCHECKPERIOD,
|
|
0, KEY_READ, &hkeyCheckPeriod) != ERROR_SUCCESS) {
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
hr = GetFileVersion(plci, &dwLocFVMS, &dwLocFVLS);
|
|
|
|
if (hr == S_OK) {
|
|
|
|
plci->dwLocFVMS = dwLocFVMS;
|
|
plci->dwLocFVLS = dwLocFVLS;
|
|
|
|
if (bExactVersion) {
|
|
if (dwFileVersionMS != dwLocFVMS || dwFileVersionLS != dwLocFVLS) {
|
|
hr = S_FALSE;
|
|
} else {
|
|
// check language
|
|
// sets the plci->bForcelangGetLatest if reqd
|
|
IsRightLanguageLocallyInstalled(plci);
|
|
}
|
|
} else {
|
|
if ((dwFileVersionMS > dwLocFVMS) ||
|
|
((dwFileVersionMS == dwLocFVMS) &&
|
|
(dwFileVersionLS > dwLocFVLS))) {
|
|
hr = S_FALSE;
|
|
} else {
|
|
// check language
|
|
// sets the plci->bForcelangGetLatest if reqd
|
|
IsRightLanguageLocallyInstalled(plci);
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((dwFileVersionMS == 0) && (dwFileVersionLS == 0)) {
|
|
hr = S_OK;
|
|
}
|
|
|
|
if ((dwFileVersionMS == -1) && (dwFileVersionLS == -1)) {
|
|
hr = S_FALSE;
|
|
}
|
|
|
|
|
|
Exit:
|
|
|
|
if (hkeyCheckPeriod)
|
|
RegCloseKey(hkeyCheckPeriod);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
/*
|
|
*
|
|
* UpdateSharedDlls
|
|
*
|
|
* the SharedDlls section looks like this
|
|
*
|
|
* [SharedDlls]
|
|
* C:\Windows\System\foo.ocx = <ref count>
|
|
*
|
|
* Parameters:
|
|
*
|
|
* szFileName full file name of module we want to use
|
|
*
|
|
* Returns:
|
|
*
|
|
* S_OK incremented tge shared dlls ref count.
|
|
*
|
|
* Error the error encountered
|
|
*/
|
|
|
|
HRESULT
|
|
UpdateSharedDlls(LPCSTR szFileName)
|
|
{
|
|
HKEY hKeySD = NULL;
|
|
HRESULT hr = S_OK;
|
|
DWORD dwType;
|
|
DWORD dwRef = 1;
|
|
DWORD dwSize = sizeof(DWORD);
|
|
LONG lResult;
|
|
|
|
// get the main SHAREDDLLS key ready; this is never freed!
|
|
|
|
|
|
if ((lResult = RegOpenKeyEx(HKEY_LOCAL_MACHINE, REGSTR_PATH_SHAREDDLLS,
|
|
0, KEY_ALL_ACCESS, &hKeySD)) != ERROR_SUCCESS) {
|
|
if ((lResult = RegCreateKey(HKEY_LOCAL_MACHINE,
|
|
REGSTR_PATH_SHAREDDLLS, &hKeySD)) != ERROR_SUCCESS) {
|
|
hKeySD = NULL;
|
|
hr = HRESULT_FROM_WIN32(lResult);
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
|
|
// now look for szFileName
|
|
lResult = SHQueryValueEx(hKeySD, szFileName, NULL, &dwType,
|
|
(unsigned char*)&dwRef, &dwSize);
|
|
|
|
if (lResult == ERROR_SUCCESS)
|
|
dwRef++;
|
|
|
|
// does not exist. Create one and initialize to 1
|
|
|
|
if ((lResult = RegSetValueEx(hKeySD, szFileName, 0, REG_DWORD,
|
|
(unsigned char*)&dwRef,
|
|
sizeof(DWORD))) != ERROR_SUCCESS) {
|
|
hr = HRESULT_FROM_WIN32(lResult);
|
|
goto Exit;
|
|
}
|
|
|
|
Exit:
|
|
|
|
if (hKeySD)
|
|
RegCloseKey(hKeySD);
|
|
|
|
return hr;
|
|
}
|
|
|
|
/*
|
|
*
|
|
* UpdateModuleUsage
|
|
*
|
|
* the module usage section in the regitry looks like this
|
|
*
|
|
* [ModuleUsage]
|
|
* [c:/windows/occache/foo.ocx]
|
|
* Owner = Internet Code Downloader
|
|
* FileVersion = <optional text version of 64-bit fileversion of ocx
|
|
* [clients]
|
|
* Internet Code Downloader = <this client's ref count>
|
|
* To allow for full path names without using the backslash we convert
|
|
* backslahes to forward slashes.
|
|
*
|
|
*
|
|
* Parameters:
|
|
*
|
|
* szFileName full file name of module we want to use
|
|
*
|
|
* szClientName name of or stringfromclsid of client.
|
|
*
|
|
* szClientPath optional (if present we can detect if client is gone)
|
|
*
|
|
* muFlags:
|
|
* MU_CLIENT mark ourselves client
|
|
* MU_OWNER mark ourselves owner
|
|
*
|
|
* Returns:
|
|
* S_OK we updated the module usage section,
|
|
* and if that was previously absent then we also upped the
|
|
* shared dlls count.
|
|
*
|
|
* Error the error encountered
|
|
*/
|
|
HRESULT
|
|
UpdateModuleUsage(
|
|
LPCSTR szFileName,
|
|
LPCSTR szClientName,
|
|
LPCSTR szClientPath,
|
|
LONG muFlags)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
LONG lResult = 0;
|
|
BOOL fUpdateSharedDlls = TRUE;
|
|
|
|
DWORD dwType;
|
|
HKEY hKeyMod = NULL;
|
|
|
|
DWORD dwSize = MAX_PATH;
|
|
char szBuf[MAX_PATH];
|
|
|
|
const char* pchSrc;
|
|
char* pchDest;
|
|
static const LPCSTR szCLIENTPATHDEFAULT = "";
|
|
LPCSTR lpClientPath = (szClientPath) ? szClientPath : szCLIENTPATHDEFAULT;
|
|
|
|
HKEY hKeyMU = NULL;
|
|
static const char szOWNER[] = ".Owner";
|
|
static const char szUNKNOWN[] = "Unknown Owner";
|
|
|
|
Assert(szClientName);
|
|
DWORD cbClientName = lstrlen(szClientName);
|
|
DWORD cbUnknown = sizeof(szUNKNOWN);
|
|
|
|
char szShortFileName[MAX_PATH];
|
|
|
|
#ifdef SHORTEN
|
|
if (!GetShortPathName(szFileName, szShortFileName, MAX_PATH)) {
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
goto Exit;
|
|
}
|
|
#else
|
|
lstrcpy(szShortFileName, szFileName);
|
|
#endif
|
|
|
|
if (g_bRunOnWin95) {
|
|
char szCharFileName[MAX_PATH];
|
|
|
|
OemToChar(szShortFileName, szCharFileName);
|
|
lstrcpy(szShortFileName, szCharFileName);
|
|
}
|
|
|
|
// get the main MODULEUSAGE key ready; this is never freed!
|
|
|
|
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, REGSTR_PATH_MODULE_USAGE,
|
|
0, KEY_ALL_ACCESS, &hKeyMU) != ERROR_SUCCESS)
|
|
if ((lResult = RegCreateKey(HKEY_LOCAL_MACHINE,
|
|
REGSTR_PATH_MODULE_USAGE, &hKeyMU)) != ERROR_SUCCESS) {
|
|
hKeyMU = NULL;
|
|
hr = HRESULT_FROM_WIN32(lResult);
|
|
goto Exit;
|
|
}
|
|
|
|
|
|
// check if Usage section is present for this dll
|
|
// open the file's section we are concerned with
|
|
// if absent create it
|
|
|
|
|
|
// BUGBUG: win95 registry bug does not allow keys to be > 255
|
|
// MAX_PATH for filename is 260
|
|
pchDest = szBuf;
|
|
for (pchSrc = szShortFileName; *pchSrc != '\0'; pchSrc++, pchDest++) {
|
|
if ((*pchDest = *pchSrc) == '\\')
|
|
*pchDest = '/';
|
|
}
|
|
*pchDest = '\0'; // null terminate
|
|
szBuf[256] = '\0'; // truncate if longer than 255 ude to win95 registry bug
|
|
|
|
if (RegOpenKeyEx(hKeyMU, szBuf,
|
|
0, KEY_ALL_ACCESS, &hKeyMod) != ERROR_SUCCESS) {
|
|
if ((lResult = RegCreateKey(hKeyMU,
|
|
szBuf, &hKeyMod)) != ERROR_SUCCESS) {
|
|
hr = HRESULT_FROM_WIN32(lResult);
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
|
|
// now look for '.Owner='
|
|
dwSize = MAX_PATH;
|
|
szBuf[0] = '\0';
|
|
if (RegQueryValueEx(hKeyMod, szOWNER, NULL, &dwType,
|
|
(unsigned char*)szBuf, &dwSize) == ERROR_SUCCESS) {
|
|
if ((lstrcmpi(szBuf, szClientName) != 0) && (muFlags & MU_OWNER)) {
|
|
// if we are the not the owner we can't make ourselves the owner
|
|
hr = E_INVALIDARG;
|
|
goto Exit;
|
|
}
|
|
|
|
} else {
|
|
|
|
// '.Owner =' does not exist. Create one and initialize to us
|
|
// if muFlags & MU_OWNER
|
|
|
|
if (((lResult = RegSetValueEx(hKeyMod, szOWNER, 0, REG_SZ,
|
|
(UCHAR*)((muFlags & MU_OWNER) ? szClientName : szUNKNOWN),
|
|
((muFlags & MU_OWNER) ? cbClientName : cbUnknown)))) != ERROR_SUCCESS) {
|
|
hr = HRESULT_FROM_WIN32(lResult);
|
|
goto Exit;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
// look for szClientName already marked as a client
|
|
|
|
dwSize = MAX_PATH;
|
|
if (SHQueryValueEx(hKeyMod, szClientName, NULL, &dwType,
|
|
(unsigned char*)szBuf, &dwSize) == ERROR_SUCCESS) {
|
|
|
|
// signal that we have already registered as a
|
|
// client and so don't up ref count in shareddlls
|
|
|
|
fUpdateSharedDlls = FALSE;
|
|
|
|
|
|
} else {
|
|
|
|
// add ourselves as a client
|
|
|
|
if ((lResult = RegSetValueEx(hKeyMod, szClientName, 0, REG_SZ,
|
|
(unsigned char*)lpClientPath, lstrlen(lpClientPath) + 1)) != ERROR_SUCCESS) {
|
|
hr = HRESULT_FROM_WIN32(lResult);
|
|
goto Exit;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
Exit:
|
|
|
|
if (hKeyMod)
|
|
RegCloseKey(hKeyMod);
|
|
|
|
if (hKeyMU)
|
|
RegCloseKey(hKeyMU);
|
|
|
|
// Update ref count in SharedDlls only if usage section was not
|
|
// already updated. This will ensure that the code downloader has
|
|
// just one ref count represented in SharedDlls
|
|
|
|
if (fUpdateSharedDlls)
|
|
hr = UpdateSharedDlls(szShortFileName);
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
// Name: SniffStringFileInfo
|
|
// Parameters:
|
|
// szFileName - full path to file whose StringFileInfo we want to sniff
|
|
// lpszSubblock - sub block of the StringFileInfo to look for
|
|
// pdw - place to put the string, interpreted as a number
|
|
// Returns:
|
|
// TRUE - if the subblock is present.
|
|
// FALSE - if the subblock is absent.
|
|
// Notes:
|
|
// This function implements stuff common to SupportsSelfRegister and WantsAutoExpire
|
|
|
|
BOOL SniffStringFileInfo(LPSTR szFileName, LPCTSTR lpszSubblock, DWORD* pdw)
|
|
{
|
|
BOOL bResult = FALSE;
|
|
DWORD handle;
|
|
UINT uiInfoSize;
|
|
UINT uiVerSize;
|
|
UINT uiSize;
|
|
BYTE* pbData = NULL;
|
|
DWORD* lpBuffer;
|
|
TCHAR szName[512];
|
|
LPTSTR szExpire;
|
|
|
|
if (pdw)
|
|
*pdw = 0;
|
|
|
|
#ifdef UNIX
|
|
// Don't have version.dll
|
|
DebugBreak();
|
|
return FALSE;
|
|
#endif
|
|
// Get the size of the version information.
|
|
uiInfoSize = g_versiondll.GetFileVersionInfoSize(szFileName, &handle);
|
|
|
|
if (uiInfoSize == 0) return FALSE;
|
|
|
|
// Allocate a buffer for the version information.
|
|
pbData = new BYTE[uiInfoSize];
|
|
|
|
if (!pbData)
|
|
return TRUE; // nothing nasty, just quirky
|
|
|
|
// Fill the buffer with the version information.
|
|
bResult = g_versiondll.GetFileVersionInfo(szFileName, handle, uiInfoSize, pbData);
|
|
|
|
if (!bResult) goto Exit;
|
|
|
|
// Get the translation information.
|
|
bResult = g_versiondll.VerQueryValue(pbData, "\\VarFileInfo\\Translation",
|
|
(void**)&lpBuffer, &uiVerSize);
|
|
|
|
if (!bResult) goto Exit;
|
|
|
|
if (!uiVerSize) goto Exit;
|
|
|
|
// Build the path to the OLESelfRegister key
|
|
// using the translation information.
|
|
wsprintf(szName, "\\StringFileInfo\\%04hX%04hX\\%s",
|
|
LOWORD(*lpBuffer), HIWORD(*lpBuffer), lpszSubblock);
|
|
|
|
// Search for the key.
|
|
bResult = g_versiondll.VerQueryValue(pbData, szName, (void**)&szExpire, &uiSize);
|
|
|
|
// If there's a string there, we need to convert it to a count of days.
|
|
if (bResult && pdw && uiSize) {
|
|
DWORD dwExpire = 0;
|
|
|
|
for (; *szExpire; szExpire++) {
|
|
if ((*szExpire >= TEXT('0') && *szExpire <= TEXT('9')))
|
|
dwExpire = dwExpire * 10 + *szExpire - TEXT('0');
|
|
else
|
|
break;
|
|
}
|
|
|
|
if (dwExpire > MAX_EXPIRE_DAYS)
|
|
dwExpire = MAX_EXPIRE_DAYS;
|
|
|
|
*pdw = dwExpire;
|
|
}
|
|
|
|
Exit:
|
|
delete[] pbData;
|
|
return bResult;
|
|
}
|
|
|
|
|
|
// returns S_OK on dist unit(at version number) installed zeroimpactly, S_FALSE on not installed
|
|
// zeroimpactly, or error codes
|
|
HRESULT
|
|
IsDistUnitLocallyInstalledZI(LPCWSTR wszDistUnit, LPCWSTR wszClsid, DWORD dwFileVersionMS, DWORD dwFileVersionLS,
|
|
CLocalComponentInfo* plci)
|
|
{
|
|
HRESULT hrNormalCheck = S_OK;
|
|
HRESULT hrZICheck = S_OK;
|
|
HRESULT hrDll = S_OK;
|
|
int cBufLen = MAX_PATH;
|
|
int iLenDU = 0;
|
|
int iLenVer = 0;
|
|
TCHAR szDir[MAX_PATH];
|
|
TCHAR szFile[MAX_PATH];
|
|
TCHAR szDllName[MAX_PATH];
|
|
WCHAR wszDistDotVersion[MAX_PATH];
|
|
TCHAR szVersion[MAX_PATH];
|
|
WCHAR wszVersion[MAX_PATH];
|
|
BOOL bParanoid = TRUE;
|
|
|
|
ASSERT(wszDistUnit);
|
|
|
|
if (!dwFileVersionMS && !dwFileVersionLS) {
|
|
GetLatestZIVersion(wszDistUnit, &dwFileVersionMS, &dwFileVersionLS);
|
|
}
|
|
|
|
if (GetStringFromVersion(szVersion, dwFileVersionMS, dwFileVersionLS, '_')) {
|
|
if (MultiByteToWideChar(CP_ACP, 0, szVersion, -1, wszVersion, MAX_PATH) != 0) {
|
|
// buffer overflow guard
|
|
iLenDU = lstrlenW(wszDistUnit);
|
|
iLenVer = lstrlenW(wszVersion);
|
|
if (iLenDU + iLenVer /*dot and null*/ + 2 > MAX_PATH)
|
|
return E_UNEXPECTED;
|
|
StrCpyW(wszDistDotVersion, wszDistUnit);
|
|
wszDistDotVersion[iLenDU++] = L'!';
|
|
StrCpyW(wszDistDotVersion + iLenDU, wszVersion);
|
|
}
|
|
}
|
|
|
|
|
|
// Use the ZI api first; this has the final say on whether the control is installed ZI;
|
|
// It also fills in szDir
|
|
hrZICheck = ZIGetInstallationDir(wszDistUnit, &dwFileVersionMS, &dwFileVersionLS, szDir, &cBufLen);
|
|
|
|
// Call the normal IsDistUnitLocallyInstalled function; this will fill in plci
|
|
hrNormalCheck = IsDistUnitLocallyInstalled(wszDistDotVersion, dwFileVersionMS,
|
|
dwFileVersionLS, plci, NULL, &bParanoid, 0);
|
|
|
|
|
|
if (FAILED(hrZICheck)) {
|
|
if (hrZICheck == HRESULT_FROM_WIN32(ERROR_MORE_DATA))
|
|
return E_UNEXPECTED;
|
|
else
|
|
return hrZICheck;
|
|
}
|
|
|
|
// hrZICheck == S_OK: directory was found; hrZICheck == S_FALSE: directory was not found
|
|
if (hrZICheck == S_OK) {
|
|
cBufLen = MAX_PATH;
|
|
|
|
// Found the dir, now look for a dll name to report up
|
|
hrDll = ZIGetDllName(szDir, wszDistUnit, wszClsid, dwFileVersionMS, dwFileVersionLS, szFile, &cBufLen);
|
|
if (hrDll == S_OK) {
|
|
wsprintf(szDllName, TEXT("%s\\%s"), szDir, szFile);
|
|
if (MultiByteToWideChar(CP_ACP, 0, szDllName, -1, plci->wszDllName, MAX_PATH) == 0)
|
|
plci->wszDllName[0] = L'\0';
|
|
} else {
|
|
plci->wszDllName[0] = L'\0';
|
|
}
|
|
|
|
// if IsDistUnit...lled already filled this in, don't overwrite
|
|
if (hrNormalCheck != S_OK) {
|
|
plci->dwLocFVMS = dwFileVersionMS;
|
|
plci->dwLocFVLS = dwFileVersionLS;
|
|
}
|
|
plci->m_bIsZI = TRUE;
|
|
} else {
|
|
plci->dwLocFVMS = 0;
|
|
plci->dwLocFVLS = 0;
|
|
plci->m_bIsZI = FALSE;
|
|
}
|
|
|
|
return hrZICheck;
|
|
}
|
|
|
|
void GetLatestZIVersion(const WCHAR* pwzDistUnit, DWORD* pdwVerMS,
|
|
DWORD* pdwVerLS)
|
|
{
|
|
HKEY hkeyZI = 0;
|
|
HANDLE hFile = INVALID_HANDLE_VALUE;
|
|
DWORD dwDistVerMS = 0;
|
|
DWORD dwDistVerLS = 0;
|
|
DWORD lResult;
|
|
DWORD iSubKey;
|
|
DWORD dwSize = MAX_REGSTR_LEN;
|
|
char szDistUnitCur[MAX_REGSTR_LEN];
|
|
char* szPtr = NULL;
|
|
char* pszDistUnit = NULL;
|
|
int iLen = 0;
|
|
|
|
if (Unicode2Ansi(pwzDistUnit, &pszDistUnit)) {
|
|
goto Exit;
|
|
}
|
|
|
|
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, REGKEY_ZEROIMPACT_DIRS, 0, KEY_READ,
|
|
&hkeyZI) == ERROR_SUCCESS) {
|
|
iSubKey = 0;
|
|
while ((lResult = RegEnumKeyEx(hkeyZI, iSubKey++, szDistUnitCur,
|
|
&dwSize, NULL, NULL, NULL,
|
|
NULL)) == ERROR_SUCCESS) {
|
|
szPtr = StrStrI(szDistUnitCur, pszDistUnit);
|
|
|
|
if (szPtr) {
|
|
ExtractVersion(szDistUnitCur, &dwDistVerMS, &dwDistVerLS);
|
|
|
|
if (dwDistVerMS > * pdwVerMS ||
|
|
((dwDistVerMS == *pdwVerMS) && dwDistVerLS > * pdwVerLS)) {
|
|
*pdwVerMS = dwDistVerMS;
|
|
*pdwVerLS = dwDistVerLS;
|
|
}
|
|
}
|
|
|
|
|
|
dwSize = MAX_REGSTR_LEN;
|
|
}
|
|
}
|
|
|
|
if (!*pdwVerMS && !*pdwVerLS) {
|
|
LPTSTR szDir = (LPTSTR)GetZeroImpactRootDir();
|
|
TCHAR szQualifiedPath[MAX_PATH];
|
|
WIN32_FIND_DATA wfd;
|
|
|
|
// Still haven't found a version. Look in the file system
|
|
|
|
if (szDir) {
|
|
StrCpyN(szQualifiedPath, szDir, MAX_PATH);
|
|
|
|
Assert(lstrlen(szQualifiedPath) + 4 < MAX_PATH);
|
|
|
|
StrCat(szQualifiedPath, TEXT("\\*.*"));
|
|
|
|
hFile = FindFirstFile(szQualifiedPath, &wfd);
|
|
|
|
iLen = lstrlen(pszDistUnit);
|
|
|
|
if (hFile != INVALID_HANDLE_VALUE) {
|
|
if (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
|
|
if (!StrCmpNI(wfd.cFileName, pszDistUnit, iLen)) {
|
|
ExtractVersion(wfd.cFileName, &dwDistVerMS,
|
|
&dwDistVerLS);
|
|
|
|
if (dwDistVerMS > * pdwVerMS ||
|
|
((dwDistVerMS == *pdwVerMS) &&
|
|
dwDistVerLS > * pdwVerLS)) {
|
|
*pdwVerMS = dwDistVerMS;
|
|
*pdwVerLS = dwDistVerLS;
|
|
}
|
|
}
|
|
}
|
|
|
|
while (FindNextFile(hFile, &wfd)) {
|
|
if (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {
|
|
if (!StrCmpNI(wfd.cFileName, pszDistUnit, iLen)) {
|
|
ExtractVersion(wfd.cFileName, &dwDistVerMS,
|
|
&dwDistVerLS);
|
|
|
|
if (dwDistVerMS > * pdwVerMS ||
|
|
((dwDistVerMS == *pdwVerMS) &&
|
|
dwDistVerLS > * pdwVerLS)) {
|
|
*pdwVerMS = dwDistVerMS;
|
|
*pdwVerLS = dwDistVerLS;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Exit:
|
|
|
|
if (hkeyZI) {
|
|
RegCloseKey(hkeyZI);
|
|
}
|
|
|
|
if (hFile != INVALID_HANDLE_VALUE) {
|
|
FindClose(hFile);
|
|
}
|
|
|
|
SAFEDELETE(pszDistUnit);
|
|
}
|
|
|
|
void ExtractVersion(char* pszDistUnit, DWORD* pdwVerMS, DWORD* pdwVerLS)
|
|
{
|
|
char* pszCopy = NULL;
|
|
char* pszPtr = NULL;
|
|
int iLen = 0;
|
|
|
|
if (!pszDistUnit) {
|
|
return;
|
|
}
|
|
|
|
iLen = lstrlen(pszDistUnit) + 1;
|
|
pszCopy = new char[iLen];
|
|
|
|
if (!pszCopy) {
|
|
return;
|
|
}
|
|
|
|
StrNCpy(pszCopy, pszDistUnit, iLen);
|
|
|
|
pszPtr = pszCopy;
|
|
|
|
// Convert _ to , for GetVersionFromString()
|
|
|
|
while (*pszPtr) {
|
|
if (*pszPtr == '_') {
|
|
*pszPtr = ',';
|
|
}
|
|
|
|
pszPtr++;
|
|
}
|
|
|
|
pszPtr = StrStrA(pszCopy, "!");
|
|
if (pszPtr) {
|
|
pszPtr++;
|
|
GetVersionFromString(pszPtr, pdwVerMS, pdwVerLS);
|
|
}
|
|
|
|
SAFEDELETE(pszCopy);
|
|
}
|