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

1329 lines
35 KiB
C++

// File: regzone.cxx
// Contents: Registry management for a single zone.
// Classes: CRegZone
// Functions:
// History:
#include "zonepch.h"
// Max # of chars to the root of the zones tree (SZZONES, SZTEMPLATE,...)
#define MAX_REGZONE_ROOT 100
// Value names in the registry
#define SZZONEINDEX __TEXT("ZoneIndex")
#define SZTEMPLATEINDEX __TEXT("TemplateIndex")
#define SZDISPLAYNAME __TEXT("DisplayName")
#define SZICON __TEXT("Icon")
#define SZDESCRIPTION __TEXT("Description")
#define SZFLAGS __TEXT("Flags")
// Registry key names for the template policies
#define SZLOW __TEXT("Low")
#define SZMEDLOW __TEXT("MedLow")
#define SZMEDIUM __TEXT("Medium")
#define SZHIGH __TEXT("High")
CRegZone::CRegZoneCache CRegZone::s_rzcache;
HANDLE CRegZone::CRegZoneCache::s_hMutexCounter;
// Array of Value Names corresponding to zone attributes.
// These values will not be copied when doing a mass copy
// from a template zone (HIGH, MED, LOW) to a zone.
static LPCTSTR rgszAttributeNames [ ] =
{
__TEXT(""), // The default value is excluded as well.
SZZONEINDEX,
SZTEMPLATEINDEX,
SZDISPLAYNAME,
SZDESCRIPTION,
SZICON,
SZFLAGS,
SZMINLEVEL,
SZRECLEVEL,
SZCURRLEVEL,
};
struct templateNameIdxMap
{
URLTEMPLATE index;
LPCTSTR pszName;
};
static templateNameIdxMap
TemplateNameIdxMap [ ] =
{
{ URLTEMPLATE_LOW, SZLOW },
{ URLTEMPLATE_MEDLOW, SZMEDLOW},
{ URLTEMPLATE_MEDIUM, SZMEDIUM},
{ URLTEMPLATE_HIGH, SZHIGH}
};
// CRegZone implementation.
CRegZone::CRegZone()
{
// defaults
m_dwZoneId = ZONEID_INVALID;
m_dwZoneFlags = ZAFLAGS_ADD_SITES; // BUGBUG: what is the right default here.
m_lpZoneName = NULL;
m_lpZonePath = NULL;
m_bStandard = TRUE;
m_bZoneLockOut = FALSE;
m_bHKLMOnly = TRUE;
}
CRegZone::~CRegZone()
{
LocalFree((HLOCAL)m_lpZoneName);
LocalFree((HLOCAL)m_lpZonePath);
}
// Sets up the CRegZone object a given string.
// If the setting is in the Zones key the string passed in is the actual
// zone index. Otherwise it is one of the "High", "Medium", "Low" strings which indicates
// a template policy.
BOOL CRegZone::Init(LPCWSTR lpwStr, BOOL bUseHKLMOnly, REGZONEUSE regZoneUse, BOOL bCreate /*=TRUE*/)
{
TransAssert(lpwStr != NULL);
if (lpwStr == NULL)
{
return FALSE;
}
m_bHKLMOnly = bUseHKLMOnly;
m_regZoneUse = regZoneUse;
TCHAR szTemp[MAX_REGZONE_ROOT + MAX_ZONE_NAME];
StrCpyW(szTemp, (regZoneUse == REGZONEUSEZONES? SZZONES : SZTEMPLATE));
StrCatW(szTemp, lpwStr);
m_lpZonePath = StrDup(szTemp);
m_lpZoneName = StrDup(lpwStr);
CRegKey regKey(bUseHKLMOnly);
if (regKey.Open(NULL, m_lpZonePath, KEY_READ) != ERROR_SUCCESS)
{
// BUGBUG:: We have to be able to deal with this. situation and not just bail.
// Possibilities: Setup defaults here if we can create and write to the key
return FALSE;
}
// Add code here to
DWORD dwZoneId = ZONEID_INVALID;
// Get the Zone Index.
if ( regZoneUse == REGZONEUSEZONES )
{
// The Zone Id for the string is the same as the key name.
// Just convert the string
m_dwZoneId = StrToInt(m_lpZoneName);
}
else if (regZoneUse == REGZONEUSETEMPLATE )
{
if (regKey.QueryValue(&dwZoneId, SZTEMPLATEINDEX) == ERROR_SUCCESS)
m_dwZoneId = dwZoneId;
else
{
// Could happen if the registry is messed up.
TransAssert(FALSE);
}
}
else
{
TransAssert(FALSE);
}
// Get the zone flags
if (regKey.QueryValue(&m_dwZoneFlags, SZFLAGS) != ERROR_SUCCESS)
{
m_dwZoneFlags = ZAFLAGS_ADD_SITES; // What is the right value here.
}
else
{
// return value from UpdateZoneMapFlags ignored.
UpdateZoneMapFlags( );
}
// Check and make sure the zone Id's are within range.
// Assert that the zone ID's are in the appropriate user or standard range.
return TRUE;
}
// This updates the flags in the ZoneMap part of the registry which correspond to the
// ZAFLAGS_. For convenience the UI will only update the ZAFLAGS.
BOOL CRegZone::UpdateZoneMapFlags( )
{
// ProxyByPass is current controlled by the ProxyByPass flag, not the zoneAttrib.
if (m_dwZoneId == URLZONE_INTRANET)
{
// If we are updating zonemap flags we have to invalidate any url to zone caches
CSecurityManager::IncrementGlobalCounter( );
CRegKey regZoneMap;
if (ERROR_SUCCESS == regZoneMap.Open(NULL, SZZONEMAP, KEY_READ | KEY_WRITE))
{
if (m_dwZoneFlags & ZAFLAGS_INCLUDE_PROXY_OVERRIDE)
{
// We will succeed even if this fails.
regZoneMap.SetValue(m_dwZoneId, SZPROXYBYPASS);
}
else
{
regZoneMap.DeleteValue(SZPROXYBYPASS);
}
if (m_dwZoneFlags & ZAFLAGS_INCLUDE_INTRANET_SITES)
{
// We will succeed even if this fails.
regZoneMap.SetValue(m_dwZoneId, SZINTRANETNAME);
}
else
{
regZoneMap.DeleteValue(SZINTRANETNAME);
}
DWORD dwUncAsIntranet = (m_dwZoneFlags & ZAFLAGS_UNC_AS_INTRANET) ? 1 : 0 ;
regZoneMap.SetValue(dwUncAsIntranet, SZUNCASINTRANET);
}
}
return TRUE;
}
// Static functions.
VOID
CRegZone::IncrementGlobalCounter( )
{
CRegZone::CRegZoneCache::IncrementGlobalCounter( );
}
BOOL CRegZone::IsAttributeName(LPCTSTR psz)
{
DWORD dwMaxIndex = sizeof(rgszAttributeNames)/sizeof(rgszAttributeNames[0]);
for ( DWORD dwIndex = 0 ; dwIndex < dwMaxIndex ; dwIndex++ )
{
#ifndef UNIX
if (0 == StrCmpW(psz, rgszAttributeNames[dwIndex]))
#else
if (0 == lstrcmpi(psz, rgszAttributeNames[dwIndex]))
#endif
return TRUE;
}
return FALSE;
}
LPCTSTR CRegZone::GetTemplateNameFromIndex(URLTEMPLATE urlTemplateIndex)
{
DWORD dwMaxIndex = sizeof(TemplateNameIdxMap) / sizeof(TemplateNameIdxMap[0]);
for (DWORD dwIndex = 0 ; dwIndex < dwMaxIndex ; dwIndex++ )
{
if (TemplateNameIdxMap[dwIndex].index == urlTemplateIndex)
return TemplateNameIdxMap[dwIndex].pszName;
}
return NULL;
}
// These are static functions to deal with aggregate policies.
// Because of the discrepancy between the UI and the actions defined,
// there are cases where the security manager munges the policies for certain actions.
inline void CRegZone::KludgeMapAggregatePolicy(DWORD dwAction, LPDWORD pdwPolicy)
{
TransAssert(pdwPolicy != NULL);
switch (dwAction)
{
case URLACTION_ACTIVEX_OVERRIDE_DATA_SAFETY:
case URLACTION_ACTIVEX_OVERRIDE_SCRIPT_SAFETY:
case URLACTION_SCRIPT_OVERRIDE_SAFETY:
{
if (GetUrlPolicyPermissions(*pdwPolicy) == URLPOLICY_QUERY)
SetUrlPolicyPermissions(*pdwPolicy, URLPOLICY_DISALLOW);
break;
}
}
}
// Call this function to determine if an action is aggregated by some
// other action.
// RETURNS : TRUE if there is an aggregate action corr to dwAction.
// also returns the action in pdwAggregate.
// FALSE: if this action is not aggregated by some other action.
// pdwAggregate is unchanged in this case.
inline BOOL CRegZone::GetAggregateAction(DWORD dwAction, LPDWORD pdwAggregate)
{
DWORD dwAggregate = 0;
BOOL bReturn = FALSE;
TransAssert(dwAction >= URLACTION_MIN);
switch(dwAction)
{
case URLACTION_ACTIVEX_OVERRIDE_DATA_SAFETY:
case URLACTION_ACTIVEX_OVERRIDE_SCRIPT_SAFETY:
case URLACTION_ACTIVEX_CONFIRM_NOOBJECTSAFETY:
case URLACTION_SCRIPT_OVERRIDE_SAFETY:
bReturn = TRUE;
dwAggregate = URLACTION_ACTIVEX_OVERRIDE_OBJECT_SAFETY;
break;
case URLACTION_HTML_SUBMIT_FORMS_FROM:
case URLACTION_HTML_SUBMIT_FORMS_TO:
bReturn = TRUE;
dwAggregate = URLACTION_HTML_SUBMIT_FORMS;
break;
}
if (bReturn && pdwAggregate)
*pdwAggregate = dwAggregate;
return bReturn;
}
// Functions corresponding to IInternetZoneManager functionality.
STDMETHODIMP CRegZone::GetZoneAttributes(ZONEATTRIBUTES& zoneAttrib)
{
if (!IsValid())
{
return E_FAIL;
}
CRegKey regKey(m_bHKLMOnly);
if (regKey.Open(NULL, m_lpZonePath, KEY_READ) != ERROR_SUCCESS)
{
// BUGBUG:: We have to be able to deal with this. situation and not just bail.
// Possibilities: Setup defaults here if we can create and write to the key
return E_FAIL;
}
// Since this is the first rev, we should have enough memory
// to fill in the ZONEATTRIBUTES structure. If we need to extend
// the structure this code will have to be modified.
TransAssert(zoneAttrib.cbSize >= sizeof(ZONEATTRIBUTES));
// Amount of information we will copy.
zoneAttrib.cbSize = sizeof(ZONEATTRIBUTES);
TransAssert(regKey!= NULL);
DWORD dwCount;
LONG lRet;
// BUGBUG deal with values exceeding size limit.
// We would have to allocate memory ourself and
// truncate the resulting string down.
// Read DisplayName.
dwCount = sizeof(zoneAttrib.szDisplayName);
lRet = regKey.QueryValue(zoneAttrib.szDisplayName, SZDISPLAYNAME, &dwCount);
TransAssert(ERROR_MORE_DATA != lRet);
if (NO_ERROR != lRet)
zoneAttrib.szDisplayName[0] = __TEXT('\0');
// Read Description
dwCount = sizeof(zoneAttrib.szDescription);
regKey.QueryValue(zoneAttrib.szDescription, SZDESCRIPTION, &dwCount);
TransAssert(ERROR_MORE_DATA != lRet);
if (NO_ERROR != lRet)
zoneAttrib.szDescription[0] = __TEXT('\0');
// Read Icon.
dwCount = sizeof(zoneAttrib.szIconPath);
regKey.QueryValue(zoneAttrib.szIconPath, SZICON, &dwCount);
TransAssert(ERROR_MORE_DATA != lRet);
if (NO_ERROR != lRet)
zoneAttrib.szIconPath[0] = __TEXT('\0');
// Read Current, Recommended and Min Settings.
QueryTemplatePolicyIndex(regKey, SZMINLEVEL, &zoneAttrib.dwTemplateMinLevel);
QueryTemplatePolicyIndex(regKey, SZRECLEVEL, &zoneAttrib.dwTemplateRecommended);
QueryTemplatePolicyIndex(regKey, SZCURRLEVEL, &zoneAttrib.dwTemplateCurrentLevel);
// Re-read the flags in case someone else updated it in an independent process.
DWORD dwZoneFlags;
if (regKey.QueryValue(&dwZoneFlags, SZFLAGS) == ERROR_SUCCESS)
{
m_dwZoneFlags = dwZoneFlags;
UpdateZoneMapFlags();
}
zoneAttrib.dwFlags = m_dwZoneFlags;
return S_OK;
}
STDMETHODIMP CRegZone::SetZoneAttributes(const ZONEATTRIBUTES& zoneAttrib)
{
if (!IsValid())
{
return E_FAIL;
}
// Check if the attributes we are trying to set are valid.
if (!IsValidTemplateIndex(zoneAttrib.dwTemplateMinLevel) ||
!IsValidTemplateIndex(zoneAttrib.dwTemplateCurrentLevel) ||
!IsValidTemplateIndex(zoneAttrib.dwTemplateRecommended))
{
return E_INVALIDARG;
}
CRegKey regKey(m_bHKLMOnly);
if (regKey.Open(NULL, m_lpZonePath, KEY_WRITE | KEY_READ) != ERROR_SUCCESS)
{
// BUGBUG:: We have to be able to deal with this. situation and not just bail.
// Possibilities: Setup defaults here if we can create and write to the key
return E_FAIL;
}
// Write the descriptive strings.
// These should almost never be changed by this call.
if (zoneAttrib.szDisplayName[0] != TEXT('\0'))
regKey.SetValue(zoneAttrib.szDisplayName, SZDISPLAYNAME);
if (zoneAttrib.szDescription[0] != TEXT('\0'))
regKey.SetValue(zoneAttrib.szDescription, SZDESCRIPTION);
if (zoneAttrib.szIconPath[0] != TEXT('\0'))
regKey.SetValue(zoneAttrib.szIconPath, SZICON);
// Write the Template Indicies.
SetTemplatePolicyIndex(regKey, SZMINLEVEL, zoneAttrib.dwTemplateMinLevel);
SetTemplatePolicyIndex(regKey, SZRECLEVEL, zoneAttrib.dwTemplateRecommended);
DWORD dwTemplateCurrentLevel;
// When the caller is setting the "CurrentLevel" to "Custom" it is assumed
// that the caller has already changed the underlying policies.
if (zoneAttrib.dwTemplateCurrentLevel == URLTEMPLATE_CUSTOM)
{
SetTemplatePolicyIndex(regKey, SZCURRLEVEL, zoneAttrib.dwTemplateCurrentLevel);
}
else
{
CopyTemplatePolicies(zoneAttrib.dwTemplateCurrentLevel);
}
// Finally write the flags value.
regKey.SetValue(zoneAttrib.dwFlags, SZFLAGS);
m_dwZoneFlags = zoneAttrib.dwFlags;
UpdateZoneMapFlags();
IncrementGlobalCounter(); // increment the count to invalidate the zone policy cache
return S_OK;
}
STDMETHODIMP CRegZone::GetActionPolicy(DWORD dwAction, URLZONEREG urlZoneReg, DWORD& dwPolicy) const
{
if (!IsValid())
return E_FAIL;
DWORD dwActionUse;
// If the action is aggregated by some other action, then we should
// actually check the policy for the aggregate action. If the function
if (!GetAggregateAction(dwAction, &dwActionUse))
dwActionUse = dwAction;
// If it is a hard-coded zone get the policy from internal tables.
// Don't look up the registry for these.
if (IsHardCodedZone() && GetHardCodedZonePolicy(dwActionUse, dwPolicy))
{
// dwPolicy should have the policy now.
}
else
{
if(!s_rzcache.Lookup(m_dwZoneId, m_lpZonePath, dwActionUse, UseHKLM(urlZoneReg), &dwPolicy))
{
return E_FAIL;
}
// For some special aggregate policies we have to modify the policy value.
KludgeMapAggregatePolicy(dwAction, &dwPolicy);
}
return S_OK;
}
STDMETHODIMP CRegZone::SetActionPolicy(DWORD dwAction, URLZONEREG urlZoneReg, DWORD dwPolicy)
{
if (!IsValid())
return E_FAIL;
CRegKey regKey(UseHKLM(urlZoneReg));
if (regKey.Open(NULL, m_lpZonePath, KEY_WRITE) != ERROR_SUCCESS)
{
// BUGBUG:: We have to be able to deal with this. situation and not just bail.
// Possibilities: Setup defaults here if we can create and write to the key
return E_FAIL;
}
// Policies cannot be set on Actions that are aggregate's.
// They can be only be set on the aggregator policy.
if (IsHardCodedZone())
{
TransAssert(FALSE);
return E_FAIL;
}
DWORD dwActionUse;
// If the action is aggregated by some other action, then we should
// actually use the policy for the aggregate action.
if (!GetAggregateAction(dwAction, &dwActionUse))
dwActionUse = dwAction;
// Convert the Action to a string.
#ifndef unix
TCHAR wsz[9]; // FFFFFFFF\0
#else
TCHAR wsz[(sizeof(DWORD)+1)*sizeof(WCHAR)];
#endif /* unix */
if (!DwToWchar(dwActionUse, wsz, 16))
{
TransAssert(FALSE);
return E_UNEXPECTED;
}
regKey.SetValue(dwPolicy, wsz);
s_rzcache.Add(m_dwZoneId, dwActionUse, UseHKLM(urlZoneReg), dwPolicy, URLZONE_FINDCACHEENTRY);
return S_OK;
}
STDMETHODIMP CRegZone::GetCustomPolicy (REFGUID guid, URLZONEREG urlZoneReg, BYTE** ppByte, DWORD *pcb) const
{
if (!IsValid())
return E_FAIL;
CRegKey regKey(UseHKLM(urlZoneReg));
if (regKey.Open(NULL, m_lpZonePath, KEY_READ) != ERROR_SUCCESS)
{
// BUGBUG:: We have to be able to deal with this. situation and not just bail.
// Possibilities: Setup defaults here if we can create and write to the key
return E_FAIL;
}
// Convert the Action to a string.
TCHAR sz[40]; // {8CC49940-3146-11CF-97A1-00AA00424A9F}\0
SHStringFromGUID(guid, sz, sizeof(sz));
*pcb = 0;
// First figure out the amount of memory required.
if (regKey.QueryBinaryValue(NULL, sz, pcb) != ERROR_SUCCESS)
{
return HRESULT_FROM_WIN32(ERROR_NOT_FOUND);
}
// Memory will be freed by caller.
*ppByte = (BYTE *)CoTaskMemAlloc(*pcb);
if ( *ppByte == NULL)
{
return E_OUTOFMEMORY;
}
// Actually query the registry for the value.
if (regKey.QueryBinaryValue(*ppByte, sz, pcb) != ERROR_SUCCESS)
{
return HRESULT_FROM_WIN32(ERROR_NOT_FOUND);
}
return S_OK;
}
STDMETHODIMP CRegZone::SetCustomPolicy (REFGUID guid, URLZONEREG urlZoneReg, BYTE* pByte, DWORD cb)
{
if (!IsValid())
return E_FAIL;
CRegKey regKey(UseHKLM(urlZoneReg));
if (regKey.Open(NULL, m_lpZonePath, KEY_WRITE) != ERROR_SUCCESS)
{
// BUGBUG:: We have to be able to deal with this. situation and not just bail.
// Possibilities: Setup defaults here if we can create and write to the key
return E_FAIL;
}
// Convert the Action to a string.
TCHAR sz[40]; // {8CC49940-3146-11CF-97A1-00AA00424A9F}\0
SHStringFromGUID(guid, sz, sizeof(sz));
DWORD dwError = ERROR_SUCCESS;
if ((dwError = regKey.SetBinaryValue(pByte, sz, cb)) != ERROR_SUCCESS)
{
return HRESULT_FROM_WIN32(dwError);
}
return S_OK;
}
STDMETHODIMP CRegZone::CopyTemplatePolicies(DWORD dwTemplate)
// Copy the policies from a predefined template into the current zone
{
HRESULT hr = E_FAIL;
// First check if we can get a name back for the template.
LPCTSTR szTemplateName = GetTemplateNameFromIndex((URLTEMPLATE)dwTemplate);
if (NULL == szTemplateName )
return E_INVALIDARG;
// Create a CRegZone for the template.
CRegZone regTemplate;
if (regTemplate.Init(szTemplateName, TRUE /* templates are stored in HKLM only */, REGZONEUSETEMPLATE))
{
CRegKey regZoneKey(m_bHKLMOnly);
CRegKey regTemplateKey(TRUE);
if ((NO_ERROR == regTemplateKey.Open(NULL, regTemplate.m_lpZonePath, KEY_READ)) &&
(NO_ERROR == regZoneKey.Open(NULL, m_lpZonePath, KEY_WRITE)))
{
TCHAR szValueName[MAX_VALUE_NAME];
DWORD dwNameLen = sizeof(szValueName)/sizeof(TCHAR);
DWORD dwBufLen = 2048;
DWORD dwActualLen = 2048;
BYTE * buffer = new BYTE[dwBufLen];
DWORD dwEnumIndex = 0;
DWORD dwType;
LONG lRet;
while ((lRet = regTemplateKey.EnumValue (dwEnumIndex, szValueName, &dwNameLen, &dwType, buffer, &dwActualLen))
!= ERROR_NO_MORE_ITEMS)
{
// Need more memory, allocate and re-try.
if (lRet == ERROR_MORE_DATA && dwActualLen > dwBufLen)
{
dwBufLen = dwActualLen;
delete [] buffer;
buffer = new BYTE[dwBufLen];
dwNameLen = sizeof(szValueName)/sizeof(TCHAR);
// Try with the bigger buffer.
lRet = regTemplateKey.EnumValue(dwEnumIndex, szValueName, &dwNameLen, &dwType, buffer, &dwActualLen);
}
// dwActualLen contains the actual size of the data to be written.
if (lRet == NO_ERROR && !IsAttributeName(szValueName))
{
// Copy the value over.
regZoneKey.SetValueOfType(buffer, szValueName, dwActualLen, dwType);
}
dwEnumIndex++;
dwActualLen = dwBufLen;
dwNameLen = sizeof(szValueName)/sizeof(TCHAR);
}
// Set the "CurrentLevel" value to the Template Index.
if (regZoneKey.SetValue(dwTemplate, SZCURRLEVEL) == NO_ERROR)
hr = S_OK;
delete [] buffer;
}
}
return hr;
}
// CRegZoneContainer methods.
CRegZoneContainer::CRegZoneContainer()
{
m_ppRegZones = NULL;
m_cZones = 0;
m_bHKLMOnly = FALSE;
m_pZoneEnumList = NULL;
m_dwNextEnum = 0;
InitializeCriticalSection(&m_csect);
}
CRegZoneContainer::~CRegZoneContainer()
{
Detach();
DeleteCriticalSection(&m_csect);
};
// This functions goes through the registry and creates the CRegZone objects corresponding
// to the zones currently in the registry.
BOOL CRegZoneContainer::Attach(BOOL bUseHKLM, REGZONEUSE regZoneUse /* = REGZONEUSEZONES */)
{
// If this assert fires you probably forgot to call Detach.
TransAssert(m_cZones == 0);
TransAssert(m_ppRegZones == NULL);
// recover if we are screwed up.
Detach();
m_bHKLMOnly = bUseHKLM ;
TCHAR sz[MAX_REGZONE_ROOT];
StrCpyW(sz, (regZoneUse == REGZONEUSEZONES ? SZZONES : SZTEMPLATE ));
// Make sure we have the minimal set of zones required and we self-heal if there is a problem.
// even if self-heal fails we ignore the error code and try to initialize the zones anyway.
if (regZoneUse == REGZONEUSEZONES )
SelfHeal(bUseHKLM);
CRegKey regKey(m_bHKLMOnly);
if (regKey.Open(NULL, sz, KEY_READ) != ERROR_SUCCESS)
{
// Hosed setup right defaults here.
return FALSE;
}
TransAssert(regKey.m_hKey != NULL);
// For each entry in the registry
DWORD dwIndex = 0;
DWORD dwCount = 0;
DWORD lRes;
TCHAR szZoneName[MAX_ZONE_NAME];
DWORD dwSize = MAX_ZONE_NAME;
CRegListElem *pElemStart = NULL;
for (; (lRes = regKey.EnumKey(dwIndex, szZoneName, &dwSize)) != ERROR_NO_MORE_ITEMS; dwIndex++)
{
if (lRes != ERROR_SUCCESS)
{
break;
}
dwSize = MAX_ZONE_NAME;
CRegZone *pRegZone = new CRegZone();
if (pRegZone == NULL)
{
m_cZones = 0;
// Out of memory -- change all error codes to return HRESULT's
break;
}
if (!pRegZone->Init(szZoneName, m_bHKLMOnly))
{
continue; // can't create the zone for some reason.
}
m_cZones++;
CRegListElem * pRegListElem = new CRegListElem();
if ( pRegListElem == NULL)
{
m_cZones = 0;
break; // Out of memory.
}
pRegListElem->pRegZone = pRegZone;
pRegListElem->dwZoneIndex = pRegZone->GetZoneId();
// Insert list into sorted position.
if (pElemStart == NULL)
{
pElemStart = pRegListElem;
pRegListElem->next = NULL;
}
else if (pElemStart->dwZoneIndex > pRegListElem->dwZoneIndex)
{
// Insert to the head of the list.
pRegListElem->next = pElemStart;
pElemStart = pRegListElem;
}
else
{
// Insert in the correct position.
CRegListElem *pElemCurr = pElemStart;
while (pElemCurr->next != NULL &&
pElemCurr->next->dwZoneIndex < pRegListElem->dwZoneIndex )
{
pElemCurr = pElemCurr->next;
}
TransAssert(pElemCurr != NULL);
pRegListElem->next = pElemCurr->next;
pElemCurr->next = pRegListElem;
}
}
// Now that we have all the RegZones collected we will just store them in
// sorted order in an array.
if (m_cZones)
m_ppRegZones = new LPREGZONE[m_cZones];
else
m_ppRegZones = NULL;
if (m_ppRegZones == NULL)
{
// Out of memory
m_cZones = 0;
return FALSE;
}
for (dwIndex = 0; dwIndex < m_cZones ; dwIndex++)
{
TransAssert(pElemStart != NULL);
m_ppRegZones[dwIndex] = pElemStart->pRegZone;
CRegListElem * pElemDelete = pElemStart;
pElemStart = pElemStart->next;
delete pElemDelete;
}
return TRUE;
}
BOOL CRegZoneContainer::Detach()
{
// First free all the CRegZone entries we are holding on to.
DWORD dwIndex = 0;
for (; dwIndex < m_cZones; dwIndex++)
{
delete m_ppRegZones[dwIndex];
}
delete [] m_ppRegZones;
m_ppRegZones = NULL;
m_cZones = 0;
m_bHKLMOnly = TRUE;
// Zone enumerator cleanup.
// This ASSERT will fire if you forget to call DestroyZoneEnumerator before
// freeing the object.
TransAssert(m_pZoneEnumList == NULL);
CZoneEnumList * pNextEnum = m_pZoneEnumList;
while (pNextEnum != NULL)
{
CZoneEnumList * pEnumListDelete = pNextEnum;
pNextEnum = pNextEnum->next;
delete pEnumListDelete;
}
return TRUE;
}
// This function makes sure that the minimal set of zones are in the registry If things are missing
// it calls the self-registration entry point and re-creates the zone key.
#define WIN2KSETUP TEXT("System\\Setup")
#define INPROGRESS TEXT("SystemSetupInProgress")
BOOL CRegZoneContainer::SelfHeal(BOOL bUseHKLM)
{
HKEY hKeyZones = NULL;
OSVERSIONINFO osvi;
osvi.dwOSVersionInfoSize = sizeof(osvi);
if (GetVersionEx(&osvi) && osvi.dwPlatformId == VER_PLATFORM_WIN32_NT)
{
// If we are called in here as part of Win2K Setup, do not run the SelfHeal code.
HKEY hWin2kSetup = NULL;
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, WIN2KSETUP, 0, KEY_READ, &hWin2kSetup) == ERROR_SUCCESS)
{
DWORD dwValue = 0; // Default is "Setup not in progress"
DWORD dwSize = sizeof(dwValue);
// If we cannnot read the value, fall back to default.
if (RegQueryValueEx(hWin2kSetup, INPROGRESS, NULL, NULL, (LPBYTE)&dwValue, &dwSize) != ERROR_SUCCESS)
dwValue = 0;
// First close the open handle
RegCloseKey(hWin2kSetup);
// Now compare the Setup flag.
// Anything other than 0 means we are in Win2k Setup. If so, ignore SelfHeal
if (dwValue != 0)
{
return TRUE;
}
}
}
if (RegOpenKeyEx((bUseHKLM ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER), SZZONES, 0, KEY_READ, &hKeyZones) == ERROR_SUCCESS)
{
// Strings corresponding to the five pre-defined zones.
TCHAR * rgszZones[] = { TEXT("0"), TEXT("1"), TEXT("2"), TEXT("3"), TEXT("4") };
int i;
for (i = 0 ; i < ARRAYSIZE(rgszZones) ; i++ )
{
HKEY hKey;
DWORD dwError = RegOpenKeyEx(hKeyZones, rgszZones[i], 0, KEY_READ, &hKey);
if (dwError != ERROR_SUCCESS)
break;
else
RegCloseKey(hKey);
}
RegCloseKey(hKeyZones);
// If we succesfully opened all the zones.
if (i == ARRAYSIZE(rgszZones))
{
return TRUE;
}
}
// If we reached here we were not able to open atleast one of the keys.
// Note that we use L"" because ZonesDllInstall takes a LPCWSTR
BOOL bRet;
HRESULT hr = ZonesDllInstall(TRUE, bUseHKLM ? L"HKLM" : L"HKCU");
if (SUCCEEDED(hr))
{
CRegKey regKey(bUseHKLM);
// Keep track of how many times we self heal for diagnostic purposes..
DWORD dwSelfHealCount;
TCHAR *pszSelfHealCount = TEXT("SelfHealCount");
if (regKey.Open(NULL, SZZONES, KEY_READ | KEY_WRITE) == ERROR_SUCCESS)
{
if (regKey.QueryValue(&dwSelfHealCount, pszSelfHealCount) != ERROR_SUCCESS)
dwSelfHealCount = 0;
dwSelfHealCount++;
regKey.SetValue(dwSelfHealCount, pszSelfHealCount);
}
bRet = TRUE;
}
else
bRet = FALSE;
return bRet;
}
CRegZone * CRegZoneContainer::GetRegZoneByName(LPCTSTR lpName) const
{
DWORD dwIndex = 0;
CRegZone *pRegZone = NULL;
for ( ; dwIndex < m_cZones; dwIndex++ )
{
pRegZone = m_ppRegZones[dwIndex] ;
if (pRegZone && StrCmpIW(pRegZone->GetZoneName(), lpName) == 0)
break;
}
return pRegZone;
}
CRegZone * CRegZoneContainer::GetRegZoneById(DWORD dwZoneId) const
{
DWORD dwIndex = 0;
CRegZone * pReturnZone = NULL;
for (; dwIndex < m_cZones ; dwIndex++ )
{
CRegZone * pRegZone = m_ppRegZones[dwIndex];
if (pRegZone == NULL)
{
// This shouldn't happen but a safety check doesn't hurt.
break;
}
else if (pRegZone->GetZoneId() == dwZoneId)
{
pReturnZone = pRegZone;
break; // Got it
}
else if (pRegZone->GetZoneId() > dwZoneId)
{
break;
}
}
return pReturnZone;
}
// Zone Enumeration functions.
BOOL CRegZoneContainer::VerifyZoneEnum(DWORD dwEnum ) const
{
BOOL bFound = FALSE;
CZoneEnumList *pNext = m_pZoneEnumList;
while (pNext)
{
if (pNext->dwEnum == dwEnum)
{
bFound = TRUE;
break;
}
pNext = pNext->next;
}
return bFound;
}
STDMETHODIMP CRegZoneContainer::CreateZoneEnumerator(DWORD* pdwEnum, DWORD *pdwCount)
{
if (pdwEnum == NULL || pdwCount == NULL)
return E_INVALIDARG;
if (m_cZones == 0)
{
return E_FAIL;
}
CZoneEnumList *pEnumListElem = new CZoneEnumList;
if (pEnumListElem == NULL)
return E_OUTOFMEMORY;
pEnumListElem->dwEnum = m_dwNextEnum++;
EnterCriticalSection(&m_csect);
if (m_pZoneEnumList == NULL)
{
pEnumListElem->next = NULL;
m_pZoneEnumList = pEnumListElem;
}
else
{
pEnumListElem->next = m_pZoneEnumList;
m_pZoneEnumList = pEnumListElem;
}
*pdwEnum = m_pZoneEnumList->dwEnum;
*pdwCount = m_cZones;
TransAssert(VerifyZoneEnum(*pdwEnum));
LeaveCriticalSection(&m_csect);
return S_OK;
}
STDMETHODIMP CRegZoneContainer::GetZoneAt(DWORD dwEnum, DWORD dwIndex, DWORD *pdwZone)
{
if (!VerifyZoneEnum(dwEnum) || dwIndex >= m_cZones)
{
return E_INVALIDARG;
}
if (m_ppRegZones && m_ppRegZones[dwIndex])
{
*pdwZone = m_ppRegZones[dwIndex]->GetZoneId();
return S_OK;
}
else
{
return E_OUTOFMEMORY;
}
}
STDMETHODIMP CRegZoneContainer::DestroyZoneEnumerator(DWORD dwEnum)
{
HRESULT hr = S_OK;
CZoneEnumList *pDelete = NULL;
EnterCriticalSection(&m_csect);
if (m_pZoneEnumList == NULL)
{
}
else if (m_pZoneEnumList->dwEnum == dwEnum)
{
pDelete = m_pZoneEnumList;
m_pZoneEnumList = pDelete->next;
}
else {
CZoneEnumList *pCurr = m_pZoneEnumList;
while (pCurr != NULL)
{
if (pCurr->next && pCurr->next->dwEnum == dwEnum)
{
pDelete = pCurr->next;
pCurr->next = pDelete->next;
break;
}
pCurr = pCurr->next;
}
}
if (pDelete == NULL)
{
// Didn't find the entry must be an invalid Enumerator.
hr = E_INVALIDARG;
}
else
{
delete pDelete;
hr = S_OK;
}
LeaveCriticalSection(&m_csect);
return hr;
}
// CRegZoneCache methods
CRegZone::CRegZoneCache::CRegZoneCache(void)
{
InitializeCriticalSection(&m_csectZoneCache);
// single static object, so this only gets inited once per
// process.
s_hMutexCounter = CreateMutexA(NULL, FALSE, "ZonesCacheCounterMutex");
m_iAdd = 0;
}
CRegZone::CRegZoneCache::~CRegZoneCache(void)
{
Flush();
DeleteCriticalSection(&m_csectZoneCache) ;
CloseHandle(s_hMutexCounter);
}
BOOL
CRegZone::CRegZoneCache::Lookup(DWORD dwZone, LPTSTR lpZonePath, DWORD dwAction, BOOL fUseHKLM, DWORD *pdwPolicy)
{
BOOL fFound = FALSE;
int iEntry = URLZONE_FINDCACHEENTRY;
TransAssert(iEntry < MAX_REG_ZONE_CACHE);
EnterCriticalSection(&m_csectZoneCache);
if ( !IsCounterEqual() )
Flush();
fFound = FindCacheEntry(dwZone, dwAction, fUseHKLM, iEntry );
if (fFound)
{
if (pdwPolicy)
{
*pdwPolicy = m_arzce[iEntry].m_dwPolicy;
}
}
else
{
// Convert the Action to a string.
#ifndef unix
TCHAR wsz[9]; // FFFFFFFF\0
#else
TCHAR wsz[(sizeof(DWORD)+1)*sizeof(WCHAR)];
#endif /* unix */
if (DwToWchar(dwAction, wsz, 16))
{
CRegKey regKey(fUseHKLM);
if (regKey.Open(NULL, lpZonePath, KEY_READ) == ERROR_SUCCESS)
{
if (regKey.QueryValue(pdwPolicy, wsz) == ERROR_SUCCESS)
{
fFound = TRUE;
Add(dwZone, dwAction, fUseHKLM, *pdwPolicy, iEntry);
}
}
else
{
// BUGBUG:: We have to be able to deal with this situation and not just bail.
// Possibilities: Setup defaults here if we can create and write to the key
TransAssert(FALSE);
}
}
else
{
TransAssert(FALSE);
}
}
LeaveCriticalSection(&m_csectZoneCache);
return fFound;
}
void
CRegZone::CRegZoneCache::Add(DWORD dwZone, DWORD dwAction, BOOL fUseHKLM, DWORD dwPolicy, int iEntry)
{
BOOL fFound;
TransAssert(iEntry < MAX_REG_ZONE_CACHE);
EnterCriticalSection(&m_csectZoneCache);
if ( !IsCounterEqual() )
Flush();
if(iEntry == URLZONE_FINDCACHEENTRY) // using optional param that indicates the entry we want to add so don't bother doing a find.
fFound = FindCacheEntry(dwZone, dwAction, fUseHKLM, iEntry ); // found or not, iEntry will be the right place to set it.
m_arzce[iEntry].Set(dwZone, dwAction, fUseHKLM, dwPolicy);
SetToCurrentCounter(); // validate this cache.
LeaveCriticalSection(&m_csectZoneCache);
}
void
CRegZone::CRegZoneCache::Flush(void)
{
int i;
EnterCriticalSection(&m_csectZoneCache);
for ( i = 0; i < MAX_REG_ZONE_CACHE; i++ )
m_arzce[i].Flush();
m_iAdd = 0;
LeaveCriticalSection(&m_csectZoneCache);
}
// Is the counter we saved with the cache entry, equal to the current counter.
BOOL
CRegZone::CRegZoneCache::IsCounterEqual( ) const
{
CExclusiveLock lock(s_hMutexCounter);
LPDWORD lpdwCounter = (LPDWORD)g_SharedMem.GetPtr(SM_REGZONECHANGE_COUNTER);
// If we couldn't create the shared memory for some reason, we just assume our cache is up to date.
if (lpdwCounter == NULL)
return TRUE;
return (m_dwPrevCounter == *lpdwCounter);
}
VOID
CRegZone::CRegZoneCache::SetToCurrentCounter( )
{
CExclusiveLock lock(s_hMutexCounter);
LPDWORD lpdwCounter = (LPDWORD)g_SharedMem.GetPtr(SM_REGZONECHANGE_COUNTER);
if (lpdwCounter == NULL)
return;
m_dwPrevCounter = *lpdwCounter;
}
VOID
CRegZone::CRegZoneCache::IncrementGlobalCounter( )
{
CExclusiveLock lock(s_hMutexCounter);
LPDWORD lpdwCounter = (LPDWORD)g_SharedMem.GetPtr(SM_REGZONECHANGE_COUNTER);
if (lpdwCounter == NULL)
return;
(*lpdwCounter)++;
}
BOOL
CRegZone::CRegZoneCache::FindCacheEntry(DWORD dwZone, DWORD dwAction, BOOL fUseHKLM, int& riEntry )
{
BOOL fFound = FALSE;
for ( riEntry = 0; (m_arzce[riEntry].m_dwZone != ZONEID_INVALID) && (riEntry < MAX_REG_ZONE_CACHE); riEntry++ )
{
if ( m_arzce[riEntry].m_dwZone == dwZone &&
m_arzce[riEntry].m_dwAction == dwAction &&
m_arzce[riEntry].m_fUseHKLM == fUseHKLM )
{
fFound = TRUE;
break;
}
}
if(!fFound)
{
riEntry = m_iAdd;
m_iAdd = (m_iAdd + 1) % MAX_REG_ZONE_CACHE; // next index to add an entry that's not found
}
return fFound;
}
void
CRegZone::CRegZoneCache::CRegZoneCacheEntry::Set(DWORD dwZone, DWORD dwAction, BOOL fUseHKLM, DWORD dwPolicy)
{
m_dwZone = dwZone;
m_dwAction = dwAction;
m_fUseHKLM = fUseHKLM;
m_dwPolicy = dwPolicy;
}
void
CRegZone::CRegZoneCache::CRegZoneCacheEntry::Flush(void)
{
m_dwZone = ZONEID_INVALID;
}