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.
4690 lines
141 KiB
4690 lines
141 KiB
//+-------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1998.
|
|
//
|
|
// File: Reg.cpp
|
|
//
|
|
// Contents: Registration routines
|
|
//
|
|
// Classes:
|
|
//
|
|
// Notes:
|
|
//
|
|
// History: 05-Nov-97 rogerg Created.
|
|
// 11-18-97 susia Added Autosync and user reg key functions
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
|
|
#include "precomp.h"
|
|
|
|
#ifdef _SENS
|
|
#include <eventsys.h> // include event system
|
|
#include <sens.h>
|
|
#include <sensevts.h>
|
|
#endif // _SENS
|
|
|
|
// temporariy define new mstask flag in case hasn't
|
|
// propogated to sdk\inc
|
|
|
|
#ifndef TASK_FLAG_RUN_ONLY_IF_LOGGED_ON
|
|
#define TASK_FLAG_RUN_ONLY_IF_LOGGED_ON (0x2000)
|
|
#endif // TASK_FLAG_RUN_ONLY_IF_LOGGED_ON
|
|
extern HINSTANCE g_hmodThisDll; // Handle to this DLL itself.
|
|
extern CRITICAL_SECTION g_DllCriticalSection; // Global Critical Section for this DLL
|
|
|
|
// only return success on NT 5.0
|
|
|
|
BOOL GetUserDefaultSecurityAttribs(SECURITY_ATTRIBUTES *psa
|
|
,PSECURITY_DESCRIPTOR psd,
|
|
PACL *ppOutAcl)
|
|
{
|
|
BOOL bRetVal;
|
|
int cbAcl;
|
|
PACL pAcl = NULL;
|
|
PSID pInteractiveUserSid = NULL;
|
|
PSID pLocalSystemSid = NULL;
|
|
PSID pAdminsSid = NULL;
|
|
SID_IDENTIFIER_AUTHORITY LocalSystemAuthority = SECURITY_NT_AUTHORITY;
|
|
|
|
*ppOutAcl = NULL;
|
|
|
|
bRetVal = FALSE;
|
|
|
|
// in the structure.
|
|
|
|
bRetVal = InitializeSecurityDescriptor(
|
|
psd, // Pointer to SD
|
|
SECURITY_DESCRIPTOR_REVISION // SD revision
|
|
);
|
|
|
|
if (!bRetVal)
|
|
{
|
|
AssertSz(0,"Unable to Init SecurityDescriptor");
|
|
goto errRtn;
|
|
}
|
|
|
|
// setup acls.
|
|
|
|
bRetVal = AllocateAndInitializeSid(
|
|
&LocalSystemAuthority, // Pointer to identifier authority
|
|
1, // Count of subauthority
|
|
SECURITY_INTERACTIVE_RID, // Subauthority 0
|
|
0, // Subauthority 1
|
|
0, // Subauthority 2
|
|
0, // Subauthority 3
|
|
0, // Subauthority 4
|
|
0, // Subauthority 5
|
|
0, // Subauthority 6
|
|
0, // Subauthority 7
|
|
&pInteractiveUserSid // pointer to pointer to SID
|
|
);
|
|
|
|
|
|
if (!bRetVal)
|
|
{
|
|
AssertSz(0,"Alocate sid failed");
|
|
goto errRtn;
|
|
}
|
|
|
|
bRetVal = AllocateAndInitializeSid(
|
|
&LocalSystemAuthority, // Pointer to identifier authority
|
|
2,
|
|
SECURITY_BUILTIN_DOMAIN_RID,
|
|
DOMAIN_ALIAS_RID_ADMINS,
|
|
0,
|
|
0, // Subauthority 3
|
|
0, // Subauthority 4
|
|
0, // Subauthority 5
|
|
0, // Subauthority 6
|
|
0, // Subauthority 7
|
|
&pAdminsSid // pointer to pointer to SID
|
|
);
|
|
|
|
if (!bRetVal)
|
|
{
|
|
AssertSz(0,"Alocate sid failed");
|
|
goto errRtn;
|
|
}
|
|
|
|
|
|
bRetVal = AllocateAndInitializeSid(
|
|
&LocalSystemAuthority,// Pointer to identifier authority
|
|
1, // Count of subauthority
|
|
SECURITY_LOCAL_SYSTEM_RID, // Subauthority 0
|
|
0, // Subauthority 1
|
|
0, // Subauthority 2
|
|
0, // Subauthority 3
|
|
0, // Subauthority 4
|
|
0, // Subauthority 5
|
|
0, // Subauthority 6
|
|
0, // Subauthority 7
|
|
&pLocalSystemSid // pointer to pointer to SID
|
|
);
|
|
|
|
if (!bRetVal)
|
|
{
|
|
AssertSz(0,"Alocate sid failed");
|
|
goto errRtn;
|
|
}
|
|
|
|
cbAcl = sizeof (ACL)
|
|
+ 3 * sizeof (ACCESS_ALLOWED_ACE)
|
|
+ GetLengthSid(pInteractiveUserSid)
|
|
+ GetLengthSid(pLocalSystemSid)
|
|
+ GetLengthSid(pAdminsSid);
|
|
|
|
pAcl = (PACL) new char[cbAcl];
|
|
|
|
if (NULL == pAcl)
|
|
{
|
|
bRetVal = FALSE;
|
|
|
|
AssertSz(0,"unable to alloc ACL");
|
|
goto errRtn;
|
|
}
|
|
|
|
bRetVal = InitializeAcl(
|
|
pAcl, // Pointer to the ACL
|
|
cbAcl, // Size of ACL
|
|
ACL_REVISION // Revision level of ACL
|
|
);
|
|
|
|
if (!bRetVal)
|
|
{
|
|
AssertSz(0,"InitAcl failed");
|
|
goto errRtn;
|
|
}
|
|
|
|
|
|
bRetVal = AddAccessAllowedAce(
|
|
pAcl, // Pointer to the ACL
|
|
ACL_REVISION, // ACL revision level
|
|
SPECIFIC_RIGHTS_ALL | GENERIC_READ | DELETE , // Access Mask
|
|
pInteractiveUserSid // Pointer to SID
|
|
);
|
|
|
|
if (!bRetVal)
|
|
{
|
|
AssertSz(0,"AddAccessAllowed Failed");
|
|
goto errRtn;
|
|
}
|
|
|
|
|
|
bRetVal = AddAccessAllowedAce(
|
|
pAcl, // Pointer to the ACL
|
|
ACL_REVISION, // ACL revision level
|
|
GENERIC_ALL, // Access Mask
|
|
pAdminsSid // Pointer to SID
|
|
);
|
|
|
|
if (!bRetVal)
|
|
{
|
|
AssertSz(0,"AddAccessAllowed Failed");
|
|
goto errRtn;
|
|
}
|
|
|
|
bRetVal = AddAccessAllowedAce(
|
|
pAcl, // Pointer to the ACL
|
|
ACL_REVISION, // ACL revision level
|
|
GENERIC_ALL, // Access Mask
|
|
pLocalSystemSid // Pointer to SID
|
|
);
|
|
|
|
if (!bRetVal)
|
|
{
|
|
AssertSz(0,"AddAccessAllowed Failed");
|
|
goto errRtn;
|
|
}
|
|
|
|
bRetVal = SetSecurityDescriptorDacl(psd,TRUE,pAcl,FALSE);
|
|
|
|
if (!bRetVal)
|
|
{
|
|
AssertSz(0,"SetSecurityDescriptorDacl Failed");
|
|
goto errRtn;
|
|
}
|
|
|
|
psa->nLength = sizeof(SECURITY_ATTRIBUTES);
|
|
psa->lpSecurityDescriptor = psd;
|
|
psa->bInheritHandle = FALSE;
|
|
|
|
errRtn:
|
|
|
|
if (pInteractiveUserSid)
|
|
{
|
|
FreeSid(pInteractiveUserSid);
|
|
}
|
|
|
|
if (pLocalSystemSid)
|
|
{
|
|
FreeSid(pLocalSystemSid);
|
|
}
|
|
|
|
if (pAdminsSid)
|
|
{
|
|
FreeSid(pAdminsSid);
|
|
}
|
|
|
|
//
|
|
// On failure, we clean the ACL up. On success, the caller cleans
|
|
// it up after using it.
|
|
//
|
|
if (FALSE == bRetVal)
|
|
{
|
|
if (pAcl)
|
|
{
|
|
delete[] pAcl;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Assert(pAcl);
|
|
*ppOutAcl = pAcl;
|
|
}
|
|
|
|
return bRetVal;
|
|
}
|
|
|
|
const WCHAR SZ_USERSIDKEY[] = TEXT("SID");
|
|
|
|
// calls regOpen or create based on fCreate param
|
|
|
|
|
|
LONG RegGetKeyHelper(HKEY hkey,LPCWSTR pszKey,REGSAM samDesired,BOOL fCreate,
|
|
HKEY *phkResult,DWORD *pdwDisposition)
|
|
{
|
|
LONG lRet = -1;
|
|
|
|
Assert(pdwDisposition);
|
|
|
|
*pdwDisposition = 0;
|
|
|
|
if (fCreate)
|
|
{
|
|
lRet = RegCreateKeyEx(hkey,pszKey,0,NULL,REG_OPTION_NON_VOLATILE,
|
|
samDesired,NULL,phkResult,pdwDisposition);
|
|
}
|
|
else
|
|
{
|
|
lRet = RegOpenKeyEx(hkey,pszKey,0,samDesired,phkResult);
|
|
|
|
if (ERROR_SUCCESS == lRet)
|
|
{
|
|
*pdwDisposition = REG_OPENED_EXISTING_KEY;
|
|
}
|
|
|
|
}
|
|
|
|
return lRet;
|
|
}
|
|
|
|
|
|
|
|
// called to create a new UserKey or subkey
|
|
LONG RegCreateUserSubKey(
|
|
HKEY hKey,
|
|
LPCWSTR lpSubKey,
|
|
REGSAM samDesired,
|
|
PHKEY phkResult)
|
|
{
|
|
LONG lRet;
|
|
DWORD dwDisposition;
|
|
|
|
lRet = RegCreateKeyEx(hKey,lpSubKey,0,NULL,REG_OPTION_NON_VOLATILE,
|
|
samDesired,NULL,phkResult,&dwDisposition);
|
|
|
|
// !! if subkey contains \\ don't traverse back through the list
|
|
|
|
if ( (ERROR_SUCCESS == lRet) && (REG_CREATED_NEW_KEY == dwDisposition))
|
|
{
|
|
HKEY hKeySecurity;
|
|
SECURITY_ATTRIBUTES sa;
|
|
SECURITY_DESCRIPTOR sd;
|
|
PACL pOutAcl;
|
|
|
|
if (ERROR_SUCCESS == RegOpenKeyEx(hKey,
|
|
lpSubKey,
|
|
REG_OPTION_OPEN_LINK, WRITE_DAC,&hKeySecurity) )
|
|
{
|
|
if (GetUserDefaultSecurityAttribs(&sa,&sd,&pOutAcl))
|
|
{
|
|
RegSetKeySecurity(hKeySecurity,
|
|
(SECURITY_INFORMATION) DACL_SECURITY_INFORMATION,
|
|
&sd);
|
|
|
|
delete[] pOutAcl;
|
|
}
|
|
|
|
RegCloseKey(hKeySecurity);
|
|
}
|
|
}
|
|
return lRet;
|
|
}
|
|
|
|
|
|
STDAPI_(HKEY) RegOpenUserKey(HKEY hkeyParent,REGSAM samDesired,BOOL fCreate,BOOL fCleanReg)
|
|
{
|
|
TCHAR pszDomainAndUser[MAX_DOMANDANDMACHINENAMESIZE];
|
|
HKEY hKeyUser;
|
|
BOOL fSetUserSid = FALSE;
|
|
WCHAR szUserSID[MAX_PATH + 1];
|
|
DWORD dwDisposition;
|
|
LONG ret;
|
|
|
|
|
|
GetDefaultDomainAndUserName(pszDomainAndUser,TEXT("_"),ARRAYSIZE(pszDomainAndUser));
|
|
|
|
// If suppose to clean the settings for the user/ delete the key
|
|
if (fCleanReg)
|
|
{
|
|
RegDeleteKeyNT(hkeyParent,pszDomainAndUser);
|
|
}
|
|
|
|
|
|
if (ERROR_SUCCESS != (ret = RegGetKeyHelper(hkeyParent,pszDomainAndUser,samDesired,fCreate,
|
|
&hKeyUser,&dwDisposition)))
|
|
{
|
|
hKeyUser = NULL;
|
|
}
|
|
|
|
if (hKeyUser)
|
|
{
|
|
WCHAR szRegSID[MAX_PATH + 1];
|
|
DWORD cbRegSID = sizeof(szRegSID);
|
|
|
|
if (ERROR_SUCCESS !=SHRegGetValue(hKeyUser,NULL,
|
|
SZ_USERSIDKEY,SRRF_RT_REG_SZ | SRRF_NOEXPAND, NULL,
|
|
(LPBYTE) szRegSID, &cbRegSID))
|
|
{
|
|
fSetUserSid = TRUE;
|
|
|
|
// if have to set the sid need to make sure openned
|
|
// with set Value and if didn't close key and
|
|
// let create re-open it with the desired access.
|
|
|
|
if (!(samDesired & KEY_SET_VALUE))
|
|
{
|
|
RegCloseKey(hKeyUser);
|
|
hKeyUser = NULL;
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
if (GetUserTextualSid(szUserSID, ARRAYSIZE(szUserSID)))
|
|
{
|
|
if (lstrcmp(szRegSID, szUserSID))
|
|
{
|
|
// if don't have access privledges
|
|
// to delete the User this will fail.
|
|
// may want a call into SENS to delete the
|
|
// User Key on Failure.
|
|
RegCloseKey(hKeyUser);
|
|
hKeyUser = NULL; // set to NULL so check below fails.
|
|
RegDeleteKeyNT(hkeyParent,pszDomainAndUser);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
|
|
if (ERROR_SUCCESS != (ret = RegGetKeyHelper(hkeyParent,pszDomainAndUser,
|
|
samDesired,fCreate,
|
|
&hKeyUser,&dwDisposition)))
|
|
{
|
|
hKeyUser = NULL;
|
|
}
|
|
else
|
|
{
|
|
if (REG_CREATED_NEW_KEY == dwDisposition)
|
|
{
|
|
fSetUserSid = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
// on creation setup the security
|
|
if ( (ERROR_SUCCESS == ret) && (REG_CREATED_NEW_KEY == dwDisposition))
|
|
{
|
|
|
|
HKEY hKeySecurity;
|
|
SECURITY_ATTRIBUTES sa;
|
|
SECURITY_DESCRIPTOR sd;
|
|
PACL pOutAcl;
|
|
|
|
// !! should have own call for sync type key security
|
|
if (ERROR_SUCCESS == RegOpenKeyEx(hkeyParent,pszDomainAndUser,
|
|
REG_OPTION_OPEN_LINK, WRITE_DAC,&hKeySecurity) )
|
|
{
|
|
if (GetUserDefaultSecurityAttribs(&sa,&sd,&pOutAcl))
|
|
{
|
|
RegSetKeySecurity(hKeySecurity,
|
|
(SECURITY_INFORMATION) DACL_SECURITY_INFORMATION,
|
|
&sd);
|
|
|
|
delete[] pOutAcl;
|
|
}
|
|
|
|
RegCloseKey(hKeySecurity);
|
|
}
|
|
}
|
|
|
|
// setup the User sid.
|
|
// depends on key being openned with KEY_SET_VALUE
|
|
if (hKeyUser && fSetUserSid && (samDesired & KEY_SET_VALUE))
|
|
{
|
|
if (GetUserTextualSid(szUserSID, ARRAYSIZE(szUserSID)))
|
|
{
|
|
DWORD dwType = REG_SZ;
|
|
|
|
RegSetValueEx (hKeyUser,SZ_USERSIDKEY,NULL,
|
|
dwType,
|
|
(LPBYTE) szUserSID,
|
|
(lstrlen(szUserSID) + 1)*sizeof(WCHAR));
|
|
}
|
|
|
|
}
|
|
|
|
|
|
return hKeyUser;
|
|
|
|
}
|
|
|
|
STDAPI_(HKEY) RegGetSyncTypeKey(DWORD dwSyncType,REGSAM samDesired,BOOL fCreate)
|
|
{
|
|
HKEY hKeySyncType;
|
|
LPCWSTR pszKey;
|
|
LONG ret;
|
|
DWORD dwDisposition;
|
|
|
|
// get appropriate key to open based on sync type
|
|
|
|
switch(dwSyncType)
|
|
{
|
|
case SYNCTYPE_MANUAL:
|
|
pszKey = MANUALSYNC_REGKEY;
|
|
break;
|
|
case SYNCTYPE_AUTOSYNC:
|
|
pszKey = AUTOSYNC_REGKEY;
|
|
break;
|
|
case SYNCTYPE_IDLE:
|
|
pszKey = IDLESYNC_REGKEY;
|
|
break;
|
|
case SYNCTYPE_SCHEDULED:
|
|
pszKey = SCHEDSYNC_REGKEY;
|
|
break;
|
|
case SYNCTYPE_PROGRESS:
|
|
pszKey = PROGRESS_REGKEY;
|
|
break;
|
|
default:
|
|
AssertSz(0,"Unknown SyncType");
|
|
pszKey = NULL;
|
|
break;
|
|
}
|
|
|
|
if (NULL == pszKey)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
// first try to open the existing key
|
|
if (ERROR_SUCCESS != (ret = RegGetKeyHelper(HKEY_LOCAL_MACHINE,pszKey,samDesired,fCreate,
|
|
&hKeySyncType,&dwDisposition)))
|
|
{
|
|
// if can't open key try to create
|
|
|
|
if (ERROR_ACCESS_DENIED == ret )
|
|
{
|
|
// if access denied, call sens to reset
|
|
// the security on the topelevel keys.
|
|
SyncMgrExecCmd_ResetRegSecurity();
|
|
|
|
ret = RegGetKeyHelper(HKEY_LOCAL_MACHINE,pszKey,samDesired,fCreate,
|
|
&hKeySyncType,&dwDisposition);
|
|
}
|
|
|
|
if (ERROR_SUCCESS != ret)
|
|
{
|
|
hKeySyncType = NULL;
|
|
}
|
|
}
|
|
if ( (ERROR_SUCCESS == ret) && (REG_CREATED_NEW_KEY == dwDisposition))
|
|
{
|
|
HKEY hKeySecurity;
|
|
SECURITY_ATTRIBUTES sa;
|
|
SECURITY_DESCRIPTOR sd;
|
|
PACL pOutAcl;
|
|
|
|
// !! should have own call for sync type key security
|
|
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE,pszKey,
|
|
REG_OPTION_OPEN_LINK, WRITE_DAC,&hKeySecurity) )
|
|
{
|
|
if (GetUserDefaultSecurityAttribs(&sa,&sd,&pOutAcl))
|
|
{
|
|
RegSetKeySecurity(hKeySecurity,
|
|
(SECURITY_INFORMATION) DACL_SECURITY_INFORMATION,
|
|
&sd);
|
|
|
|
delete[] pOutAcl;
|
|
}
|
|
|
|
RegCloseKey(hKeySecurity);
|
|
}
|
|
}
|
|
|
|
return hKeySyncType;
|
|
}
|
|
|
|
|
|
STDAPI_(HKEY) RegGetCurrentUserKey(DWORD dwSyncType,REGSAM samDesired,BOOL fCreate)
|
|
{
|
|
HKEY hKeySyncType;
|
|
HKEY hKeyUser = NULL;
|
|
|
|
hKeySyncType = RegGetSyncTypeKey(dwSyncType,samDesired ,fCreate);
|
|
|
|
if (hKeySyncType)
|
|
{
|
|
hKeyUser = RegOpenUserKey(hKeySyncType,samDesired,fCreate,FALSE /* fClean */);
|
|
RegCloseKey(hKeySyncType);
|
|
}
|
|
|
|
return hKeyUser;
|
|
}
|
|
|
|
|
|
// tries to open and set security if necessary on the handler keys.
|
|
|
|
STDAPI_(HKEY) RegGetHandlerTopLevelKey(REGSAM samDesired)
|
|
{
|
|
HKEY hKeyTopLevel;
|
|
LONG ret;
|
|
DWORD dwDisposition;
|
|
|
|
// if open failed then try a create.
|
|
if (ERROR_SUCCESS != (ret = RegCreateKeyEx (HKEY_LOCAL_MACHINE,HANDLERS_REGKEY,0, NULL,
|
|
REG_OPTION_NON_VOLATILE,samDesired,NULL,
|
|
&hKeyTopLevel,
|
|
&dwDisposition)))
|
|
{
|
|
|
|
// if got an access denige on the handlers key
|
|
// call sens to reset.
|
|
if (ERROR_ACCESS_DENIED == ret )
|
|
{
|
|
// if access denied, call sens to reset
|
|
// the security on the topelevel keys.
|
|
// and try again
|
|
SyncMgrExecCmd_ResetRegSecurity();
|
|
|
|
ret = RegCreateKeyEx (HKEY_LOCAL_MACHINE,HANDLERS_REGKEY,0, NULL,
|
|
REG_OPTION_NON_VOLATILE,samDesired,NULL,
|
|
&hKeyTopLevel,
|
|
&dwDisposition);
|
|
}
|
|
|
|
if (ERROR_SUCCESS != ret)
|
|
{
|
|
hKeyTopLevel = NULL;
|
|
}
|
|
}
|
|
|
|
if ( (ERROR_SUCCESS == ret) && (REG_CREATED_NEW_KEY == dwDisposition))
|
|
{
|
|
HKEY hKeySecurity;
|
|
SECURITY_ATTRIBUTES sa;
|
|
SECURITY_DESCRIPTOR sd;
|
|
PACL pOutAcl;
|
|
|
|
// !! should have own call for toplevel handler key security
|
|
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE,
|
|
HANDLERS_REGKEY,
|
|
REG_OPTION_OPEN_LINK, WRITE_DAC,&hKeySecurity) )
|
|
{
|
|
if (GetUserDefaultSecurityAttribs(&sa,&sd,&pOutAcl))
|
|
{
|
|
RegSetKeySecurity(hKeySecurity,
|
|
(SECURITY_INFORMATION) DACL_SECURITY_INFORMATION,
|
|
&sd);
|
|
|
|
delete[] pOutAcl;
|
|
}
|
|
|
|
RegCloseKey(hKeySecurity);
|
|
}
|
|
}
|
|
|
|
return hKeyTopLevel;
|
|
|
|
}
|
|
|
|
STDAPI_(HKEY) RegGetHandlerKey(HKEY hkeyParent,LPCWSTR pszHandlerClsid,REGSAM samDesired,
|
|
BOOL fCreate)
|
|
{
|
|
HKEY hKeyHandler = NULL;
|
|
LRESULT lRet;
|
|
DWORD dwDisposition;
|
|
|
|
|
|
if (ERROR_SUCCESS != (lRet = RegGetKeyHelper(hkeyParent,pszHandlerClsid,samDesired,
|
|
fCreate,&hKeyHandler,&dwDisposition)))
|
|
{
|
|
hKeyHandler = NULL;
|
|
}
|
|
|
|
if (NULL == hKeyHandler)
|
|
{
|
|
|
|
// if got an access denied call sens to unlock
|
|
if (ERROR_ACCESS_DENIED == lRet )
|
|
{
|
|
// if access denied, call sens to reset
|
|
// the security on the topelevel keys.
|
|
// and try again
|
|
SyncMgrExecCmd_ResetRegSecurity();
|
|
|
|
lRet = RegGetKeyHelper(hkeyParent,pszHandlerClsid,samDesired,
|
|
fCreate,&hKeyHandler,&dwDisposition);
|
|
}
|
|
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
hKeyHandler = NULL;
|
|
}
|
|
|
|
}
|
|
|
|
if ( (ERROR_SUCCESS == lRet) && (REG_CREATED_NEW_KEY == dwDisposition))
|
|
{
|
|
HKEY hKeySecurity;
|
|
SECURITY_ATTRIBUTES sa;
|
|
SECURITY_DESCRIPTOR sd;
|
|
PACL pOutAcl;
|
|
|
|
// !! should have own call for handler key security
|
|
if (ERROR_SUCCESS == RegOpenKeyEx(hkeyParent,
|
|
pszHandlerClsid,
|
|
REG_OPTION_OPEN_LINK, WRITE_DAC,&hKeySecurity) )
|
|
{
|
|
if (GetUserDefaultSecurityAttribs(&sa,&sd,&pOutAcl))
|
|
{
|
|
|
|
RegSetKeySecurity(hKeySecurity,
|
|
(SECURITY_INFORMATION) DACL_SECURITY_INFORMATION,
|
|
&sd);
|
|
|
|
delete[] pOutAcl;
|
|
}
|
|
|
|
RegCloseKey(hKeySecurity);
|
|
}
|
|
}
|
|
|
|
return hKeyHandler;
|
|
}
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: DWORD RegDeleteKeyNT(HKEY hStartKey , LPTSTR pKeyName )
|
|
|
|
Summary: Recursively delete a key on NT
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
|
|
STDAPI_(DWORD) RegDeleteKeyNT(HKEY hStartKey , LPCWSTR pKeyName )
|
|
{
|
|
DWORD dwRtn, dwSubKeyLength;
|
|
LPTSTR pSubKey = NULL;
|
|
TCHAR szSubKey[MAX_KEY_LENGTH]; // (256) this should be dynamic.
|
|
HKEY hkey;
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
// do not allow NULL or empty key name
|
|
if ( pKeyName && lstrlen(pKeyName))
|
|
{
|
|
if( (dwRtn=RegOpenKeyEx(hStartKey,pKeyName,
|
|
0, KEY_ENUMERATE_SUB_KEYS | DELETE, &hkey )) == ERROR_SUCCESS)
|
|
{
|
|
while (dwRtn == ERROR_SUCCESS )
|
|
{
|
|
dwSubKeyLength = MAX_KEY_LENGTH;
|
|
dwRtn=RegEnumKeyEx(
|
|
hkey,
|
|
0, // always index zero
|
|
szSubKey,
|
|
&dwSubKeyLength,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL
|
|
);
|
|
|
|
if(dwRtn == ERROR_NO_MORE_ITEMS)
|
|
{
|
|
dwRtn = RegDeleteKey(hStartKey, pKeyName);
|
|
break;
|
|
}
|
|
else if(dwRtn == ERROR_SUCCESS)
|
|
dwRtn=RegDeleteKeyNT(hkey, szSubKey);
|
|
}
|
|
RegCloseKey(hkey);
|
|
// Do not save return code because error
|
|
// has already occurred
|
|
}
|
|
}
|
|
else
|
|
dwRtn = ERROR_BADKEY;
|
|
|
|
CMutexRegistry.Leave();
|
|
return dwRtn;
|
|
}
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegGetProgressDetailsState()
|
|
|
|
Summary: Gets the expanded or collapsed user preference for the progress dialog
|
|
and the pushpin preference
|
|
|
|
Returns: Returns TRUE if succeeded, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
STDAPI_(BOOL) RegGetProgressDetailsState(REFCLSID clsidDlg,BOOL *pfPushPin, BOOL *pfExpanded)
|
|
{
|
|
SCODE sc = S_FALSE;
|
|
HKEY hkeyUserProgress,hkeyclsidDlg;
|
|
WCHAR wszCLSID[GUID_SIZE + 1];
|
|
|
|
if (0 == StringFromGUID2(clsidDlg, wszCLSID, ARRAYSIZE(wszCLSID)))
|
|
{
|
|
AssertSz(0,"Unable to make Guid a String");
|
|
return FALSE;
|
|
}
|
|
|
|
//Prgress dialog defaults to collapsed, pushpin out
|
|
*pfExpanded = FALSE;
|
|
*pfPushPin = FALSE;
|
|
|
|
hkeyUserProgress = RegGetCurrentUserKey(SYNCTYPE_PROGRESS,KEY_READ,FALSE);
|
|
|
|
if (hkeyUserProgress)
|
|
{
|
|
|
|
if (ERROR_SUCCESS == RegOpenKeyEx(hkeyUserProgress,wszCLSID,0,KEY_READ,
|
|
&hkeyclsidDlg))
|
|
{
|
|
DWORD cbDataSize = sizeof(*pfExpanded);
|
|
if (ERROR_SUCCESS == SHRegGetValue(hkeyclsidDlg,NULL,TEXT("Expanded"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) pfExpanded, &cbDataSize))
|
|
{
|
|
cbDataSize = sizeof(*pfPushPin);
|
|
|
|
if (ERROR_SUCCESS == SHRegGetValue(hkeyclsidDlg,NULL,TEXT("PushPin"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) pfPushPin, &cbDataSize))
|
|
{
|
|
sc = S_OK;
|
|
}
|
|
}
|
|
RegCloseKey(hkeyclsidDlg);
|
|
}
|
|
RegCloseKey(hkeyUserProgress);
|
|
}
|
|
|
|
if (sc == S_OK)
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegSetProgressDetailsState(BOOL fExpanded)
|
|
|
|
Summary: Sets the expanded or collapsed user preference for the progress dialog
|
|
|
|
Returns: Returns TRUE if succeeded, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
STDAPI_(BOOL) RegSetProgressDetailsState(REFCLSID clsidDlg,BOOL fPushPin, BOOL fExpanded)
|
|
{
|
|
BOOL fResult = FALSE;
|
|
HKEY hkeyUserProgress,hkeyclsidDlg;
|
|
WCHAR wszCLSID[GUID_SIZE + 1];
|
|
|
|
|
|
if (0 == StringFromGUID2(clsidDlg, wszCLSID, ARRAYSIZE(wszCLSID)))
|
|
{
|
|
AssertSz(0,"Unable to make Guid a String");
|
|
return FALSE;
|
|
}
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
hkeyUserProgress = RegGetCurrentUserKey(SYNCTYPE_PROGRESS,KEY_WRITE | KEY_READ,TRUE);
|
|
|
|
if (hkeyUserProgress)
|
|
{
|
|
|
|
if (ERROR_SUCCESS == RegCreateUserSubKey(hkeyUserProgress,wszCLSID,
|
|
KEY_WRITE | KEY_READ,
|
|
&hkeyclsidDlg))
|
|
{
|
|
|
|
fResult = TRUE;
|
|
|
|
if (ERROR_SUCCESS != RegSetValueEx(hkeyclsidDlg,TEXT("Expanded"),NULL, REG_DWORD,
|
|
(LPBYTE) &(fExpanded),
|
|
sizeof(fExpanded)))
|
|
{
|
|
fResult = FALSE;
|
|
}
|
|
|
|
|
|
if (ERROR_SUCCESS == RegSetValueEx(hkeyclsidDlg,TEXT("PushPin"),NULL, REG_DWORD,
|
|
(LPBYTE) &(fPushPin),
|
|
sizeof(fPushPin)))
|
|
{
|
|
fResult = FALSE;
|
|
}
|
|
|
|
RegCloseKey(hkeyclsidDlg);
|
|
}
|
|
|
|
RegCloseKey(hkeyUserProgress);
|
|
|
|
}
|
|
|
|
CMutexRegistry.Leave();
|
|
return fResult;
|
|
|
|
}
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegGetSyncItemSettings( DWORD dwSyncType,
|
|
CLSID clsidHandler,
|
|
SYNCMGRITEMID ItemId,
|
|
const TCHAR *pszConnectionName,
|
|
DWORD *pdwCheckState,
|
|
DWORD dwDefaultCheckState,
|
|
TCHAR *pszSchedName)
|
|
|
|
Summary: Gets the settings per handler, itemID, and connection name.
|
|
If no selections on this connection, the default is ?
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
STDAPI_(BOOL) RegGetSyncItemSettings( DWORD dwSyncType,
|
|
CLSID clsidHandler,
|
|
SYNCMGRITEMID ItemId,
|
|
const TCHAR *pszConnectionName,
|
|
DWORD *pdwCheckState,
|
|
DWORD dwDefaultCheckState,
|
|
TCHAR *pszSchedName)
|
|
{
|
|
HKEY hKeyUser;
|
|
|
|
*pdwCheckState = dwDefaultCheckState;
|
|
|
|
// special case schedule
|
|
if (SYNCTYPE_SCHEDULED == dwSyncType)
|
|
{
|
|
// items are always turned off by default and if no schedule name
|
|
// don't bother.
|
|
*pdwCheckState = FALSE;
|
|
if (!pszSchedName)
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
// open the user key for the type
|
|
hKeyUser = RegGetCurrentUserKey(dwSyncType,KEY_READ,FALSE);
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
// for a schedule we need to go ahead and open up the schedule name
|
|
// key
|
|
|
|
HKEY hKeySchedule = NULL;
|
|
|
|
// Review if want GetCurrentUserKey to Handle this.
|
|
if (SYNCTYPE_SCHEDULED == dwSyncType)
|
|
{
|
|
if (ERROR_SUCCESS != RegOpenKeyEx((hKeyUser),
|
|
pszSchedName,0,KEY_READ,
|
|
&hKeySchedule))
|
|
{
|
|
hKeySchedule = NULL;
|
|
RegCloseKey(hKeyUser);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
BOOL fResult;
|
|
|
|
fResult = RegLookupSettings(hKeySchedule ? hKeySchedule : hKeyUser,
|
|
clsidHandler,
|
|
ItemId,
|
|
pszConnectionName,
|
|
pdwCheckState);
|
|
|
|
if (hKeySchedule)
|
|
{
|
|
RegCloseKey(hKeySchedule);
|
|
}
|
|
|
|
if (hKeyUser)
|
|
{
|
|
RegCloseKey(hKeyUser);
|
|
}
|
|
|
|
return fResult;
|
|
}
|
|
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegSetSyncItemSettings( DWORD dwSyncType,
|
|
CLSID clsidHandler,
|
|
SYNCMGRITEMID ItemId,
|
|
const TCHAR *pszConnectionName,
|
|
DWORD dwCheckState,
|
|
TCHAR *pszSchedName)
|
|
|
|
Summary: Sets the settings per handler, itemID, and connection name.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegSetSyncItemSettings( DWORD dwSyncType,
|
|
CLSID clsidHandler,
|
|
SYNCMGRITEMID ItemId,
|
|
const TCHAR *pszConnectionName,
|
|
DWORD dwCheckState,
|
|
TCHAR *pszSchedName)
|
|
{
|
|
HKEY hKeyUser;
|
|
|
|
|
|
if (SYNCTYPE_SCHEDULED == dwSyncType)
|
|
{
|
|
if (NULL == pszSchedName)
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
// open the user key for the type
|
|
hKeyUser = RegGetCurrentUserKey(dwSyncType,KEY_WRITE | KEY_READ,TRUE);
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
// for a schedule we need to go ahead and open up the schedule name
|
|
// key
|
|
|
|
HKEY hKeySchedule = NULL;
|
|
|
|
if (SYNCTYPE_SCHEDULED == dwSyncType)
|
|
{
|
|
if (ERROR_SUCCESS != RegCreateUserSubKey(hKeyUser,
|
|
pszSchedName,KEY_WRITE | KEY_READ,
|
|
&hKeySchedule))
|
|
{
|
|
hKeySchedule = NULL;
|
|
RegCloseKey(hKeyUser);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
BOOL fResult;
|
|
|
|
fResult = RegWriteOutSettings(hKeySchedule ? hKeySchedule : hKeyUser,
|
|
clsidHandler,
|
|
ItemId,
|
|
pszConnectionName,
|
|
dwCheckState);
|
|
|
|
if (hKeySchedule)
|
|
{
|
|
RegCloseKey(hKeySchedule);
|
|
}
|
|
|
|
if (hKeyUser)
|
|
{
|
|
RegCloseKey(hKeyUser);
|
|
}
|
|
|
|
return fResult;
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegQueryLoadHandlerOnEvent( )
|
|
|
|
Summary: Determines if there is any reason to load this handler
|
|
for the specified event and Connection
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegQueryLoadHandlerOnEvent(TCHAR *pszClsid,DWORD dwSyncFlags,
|
|
TCHAR *pConnectionName)
|
|
{
|
|
BOOL fLoadHandler = FALSE;
|
|
DWORD dwSyncType;
|
|
|
|
switch(dwSyncFlags & SYNCMGRFLAG_EVENTMASK)
|
|
{
|
|
case SYNCMGRFLAG_CONNECT:
|
|
case SYNCMGRFLAG_PENDINGDISCONNECT:
|
|
{
|
|
dwSyncType = SYNCTYPE_AUTOSYNC;
|
|
}
|
|
break;
|
|
case SYNCMGRFLAG_IDLE:
|
|
{
|
|
dwSyncType = SYNCTYPE_IDLE;
|
|
}
|
|
break;
|
|
default:
|
|
AssertSz(0,"Unknown SyncType");
|
|
return FALSE;
|
|
break;
|
|
}
|
|
|
|
// walk done the list openning keys.
|
|
HKEY hkeySyncType;
|
|
|
|
if (hkeySyncType = RegGetCurrentUserKey(dwSyncType,KEY_READ,FALSE))
|
|
{
|
|
HKEY hkeyConnectionName;
|
|
|
|
if (ERROR_SUCCESS == RegOpenKeyEx(hkeySyncType,pConnectionName,0,KEY_READ,&hkeyConnectionName))
|
|
{
|
|
HKEY hkeyClsid;
|
|
|
|
if (ERROR_SUCCESS == RegOpenKeyEx(hkeyConnectionName,pszClsid,0,KEY_READ,&hkeyClsid))
|
|
{
|
|
DWORD fQueryResult;
|
|
DWORD cbDataSize = sizeof(fQueryResult);
|
|
|
|
|
|
if (ERROR_SUCCESS == SHRegGetValue(hkeyClsid,NULL,TEXT("ItemsChecked"),SRRF_RT_REG_DWORD,
|
|
NULL, (LPBYTE) &fQueryResult, &cbDataSize))
|
|
{
|
|
fLoadHandler = fQueryResult;
|
|
}
|
|
|
|
RegCloseKey(hkeyClsid);
|
|
}
|
|
|
|
RegCloseKey(hkeyConnectionName);
|
|
}
|
|
|
|
RegCloseKey(hkeySyncType);
|
|
}
|
|
|
|
|
|
return fLoadHandler;
|
|
}
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegSetSyncHandlerSettings( )
|
|
|
|
Summary: Sets the handler settings for syncType and Connection.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegSetSyncHandlerSettings(DWORD dwSyncType,
|
|
const TCHAR *pszConnectionName,
|
|
CLSID clsidHandler,
|
|
BOOL fItemsChecked)
|
|
{
|
|
HKEY hKeyUser;
|
|
|
|
hKeyUser = RegGetCurrentUserKey(dwSyncType,KEY_WRITE | KEY_READ,TRUE);
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
SCODE sc;
|
|
HKEY hkeyConnection;
|
|
HKEY hkeyCLSID;
|
|
TCHAR szCLSID[(GUID_SIZE + 1)];
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
smChkTo(EH_Err2,RegCreateUserSubKey(hKeyUser,
|
|
pszConnectionName,
|
|
KEY_WRITE | KEY_READ,
|
|
&hkeyConnection));
|
|
|
|
|
|
StringFromGUID2(clsidHandler, szCLSID, ARRAYSIZE(szCLSID));
|
|
|
|
// Write entries under CLSID.
|
|
smChkTo(EH_Err3,RegCreateUserSubKey(hkeyConnection,
|
|
szCLSID,KEY_WRITE | KEY_READ,
|
|
&hkeyCLSID));
|
|
|
|
RegSetValueEx(hkeyCLSID,TEXT("ItemsChecked"),NULL, REG_DWORD,
|
|
(LPBYTE) &fItemsChecked,
|
|
sizeof(fItemsChecked));
|
|
|
|
RegCloseKey(hKeyUser);
|
|
RegCloseKey(hkeyConnection);
|
|
RegCloseKey(hkeyCLSID);
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
|
|
EH_Err3:
|
|
RegCloseKey(hkeyConnection);
|
|
EH_Err2:
|
|
RegCloseKey(hKeyUser);
|
|
|
|
CMutexRegistry.Leave();
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegLookupSettings(const TCHAR *hkeyName,
|
|
CLSID clsidHandler,
|
|
SYNCMGRITEMID ItemID,
|
|
const TCHAR *pszConnectionName,
|
|
DWORD pdwCheckState)
|
|
|
|
Summary: Gets the settings per handler, itemID, and connection name.
|
|
|
|
Returns: Returns TRUE if there are settings for this item and
|
|
connection, flase otherwise.
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegLookupSettings(HKEY hKeyUser,
|
|
CLSID clsidHandler,
|
|
SYNCMGRITEMID ItemID,
|
|
const TCHAR *pszConnectionName,
|
|
DWORD *pdwCheckState)
|
|
{
|
|
SCODE sc;
|
|
HKEY hkeyConnection;
|
|
HKEY hkeyItem;
|
|
HKEY hKeyHandler;
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
TCHAR szID[GUID_SIZE+1];
|
|
TCHAR szCLSID[2*(GUID_SIZE+1)];
|
|
|
|
|
|
smChkTo(EH_Err2,RegOpenKeyEx(hKeyUser,
|
|
pszConnectionName,0,KEY_READ,
|
|
&hkeyConnection));
|
|
|
|
|
|
StringFromGUID2(clsidHandler, szCLSID, ARRAYSIZE(szCLSID));
|
|
StringFromGUID2(ItemID, szID, ARRAYSIZE(szID));
|
|
|
|
// Read entries under CLSID.
|
|
smChkTo(EH_Err3,RegOpenKeyEx((hkeyConnection),
|
|
szCLSID, 0, KEY_READ,
|
|
&hKeyHandler));
|
|
|
|
smChkTo(EH_Err4,RegOpenKeyEx((hKeyHandler),
|
|
szID, 0, KEY_READ,
|
|
&hkeyItem));
|
|
|
|
DWORD cbDataSize = sizeof(*pdwCheckState);
|
|
|
|
smChkTo(EH_Err4,SHRegGetValue(hkeyItem,NULL,TEXT("CheckState"),SRRF_RT_REG_DWORD, NULL, (LPBYTE)pdwCheckState, &cbDataSize));
|
|
|
|
RegCloseKey(hkeyConnection);
|
|
RegCloseKey(hkeyItem);
|
|
RegCloseKey(hKeyHandler);
|
|
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
|
|
EH_Err4:
|
|
RegCloseKey(hKeyHandler);
|
|
EH_Err3:
|
|
RegCloseKey(hkeyConnection);
|
|
EH_Err2:
|
|
CMutexRegistry.Leave();
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegWriteOutSettings(HKEY hKeyUser,
|
|
CLSID clsidHandler,
|
|
SYNCMGRITEMID ItemID,
|
|
const TCHAR *pszConnectionName,
|
|
DWORD dwCheckState)
|
|
|
|
Summary: Sets the settings per handler, itemID, and connection name.
|
|
|
|
Returns: Returns TRUE if we can set them, FALSE if there is an error
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegWriteOutSettings(HKEY hKeyUser,
|
|
CLSID clsidHandler,
|
|
SYNCMGRITEMID ItemID,
|
|
const TCHAR *pszConnectionName,
|
|
DWORD dwCheckState)
|
|
{
|
|
SCODE sc;
|
|
HKEY hkeyConnection;
|
|
HKEY hKeyHandler;
|
|
HKEY hkeyItem;
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
TCHAR szID[GUID_SIZE+1];
|
|
TCHAR szCLSID[2*(GUID_SIZE+1)];
|
|
|
|
smChkTo(EH_Err2,RegCreateUserSubKey(hKeyUser,
|
|
pszConnectionName,KEY_WRITE | KEY_READ,
|
|
&hkeyConnection));
|
|
|
|
|
|
StringFromGUID2(clsidHandler, szCLSID, ARRAYSIZE(szCLSID));
|
|
StringFromGUID2(ItemID, szID, ARRAYSIZE(szID));
|
|
|
|
|
|
smChkTo(EH_Err3,RegCreateUserSubKey(hkeyConnection,
|
|
szCLSID,KEY_WRITE | KEY_READ,
|
|
&hKeyHandler));
|
|
|
|
|
|
// Write entries under CLSID.
|
|
smChkTo(EH_Err4,RegCreateUserSubKey(hKeyHandler,
|
|
szID,KEY_WRITE | KEY_READ,
|
|
&hkeyItem));
|
|
|
|
|
|
smChkTo(EH_Err4,RegSetValueEx(hkeyItem,TEXT("CheckState"),NULL, REG_DWORD,
|
|
(LPBYTE) &dwCheckState,
|
|
sizeof(dwCheckState)));
|
|
|
|
RegCloseKey(hkeyConnection);
|
|
RegCloseKey(hkeyItem);
|
|
RegCloseKey(hKeyHandler);
|
|
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
|
|
EH_Err4:
|
|
RegCloseKey(hKeyHandler);
|
|
EH_Err3:
|
|
RegCloseKey(hkeyConnection);
|
|
EH_Err2:
|
|
CMutexRegistry.Leave();
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
AutoSync Registry Functions
|
|
|
|
***************************************************************************F-F*/
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegGetAutoSyncSettings( LPCONNECTIONSETTINGS lpConnectionSettings)
|
|
|
|
Summary: Gets the logon, logoff and prompt me first user selections.
|
|
If no selections on this connection, the default is ?
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
STDAPI_(BOOL) RegGetAutoSyncSettings(LPCONNECTIONSETTINGS lpConnectionSettings)
|
|
{
|
|
SCODE sc;
|
|
HKEY hkeyConnection;
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
|
|
// Set these first to default values, in case there are no current
|
|
// user preferences in the registry
|
|
lpConnectionSettings->dwLogon = FALSE;
|
|
lpConnectionSettings->dwLogoff = FALSE;
|
|
lpConnectionSettings->dwPromptMeFirst = FALSE;
|
|
|
|
HKEY hKeyUser;
|
|
|
|
hKeyUser = RegGetCurrentUserKey(SYNCTYPE_AUTOSYNC,KEY_READ,FALSE);
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
goto EH_Err;
|
|
}
|
|
|
|
DWORD cbDataSize = sizeof(lpConnectionSettings->dwLogon);
|
|
|
|
smChkTo(EH_Err3,RegOpenKeyEx(hKeyUser, lpConnectionSettings->pszConnectionName,0,KEY_READ,
|
|
&hkeyConnection));
|
|
|
|
smChkTo(EH_Err4,SHRegGetValue(hkeyConnection,NULL,TEXT("Logon"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->dwLogon),
|
|
&cbDataSize));
|
|
|
|
cbDataSize = sizeof(lpConnectionSettings->dwLogoff);
|
|
|
|
smChkTo(EH_Err4,SHRegGetValue(hkeyConnection,NULL,TEXT("Logoff"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->dwLogoff),
|
|
&cbDataSize));
|
|
|
|
cbDataSize = sizeof(lpConnectionSettings->dwPromptMeFirst);
|
|
|
|
smChkTo(EH_Err4,SHRegGetValue(hkeyConnection,NULL,TEXT("PromptMeFirst"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->dwPromptMeFirst),
|
|
&cbDataSize));
|
|
|
|
RegCloseKey(hkeyConnection);
|
|
RegCloseKey(hKeyUser);
|
|
|
|
CMutexRegistry.Leave();
|
|
|
|
return TRUE;
|
|
|
|
EH_Err4:
|
|
RegCloseKey(hkeyConnection);
|
|
EH_Err3:
|
|
RegCloseKey(hKeyUser);
|
|
EH_Err:
|
|
CMutexRegistry.Leave();
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegUpdateUserAutosyncKey
|
|
|
|
Summary: given an Autosync User Key makes sure it is in the latest
|
|
format and if not updates it.
|
|
|
|
Returns:
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
void RegUpdateUserAutosyncKey(HKEY hkeyUser,BOOL fForce)
|
|
{
|
|
DWORD dwUserLogonLogoff;
|
|
DWORD cbDataSize = sizeof(dwUserLogonLogoff);
|
|
DWORD dwIndex = 0;
|
|
TCHAR lpName[MAX_PATH];
|
|
DWORD cbName = MAX_PATH;
|
|
|
|
DWORD dwLogon = 0;
|
|
DWORD dwLogoff = 0;
|
|
|
|
if (!fForce && (ERROR_SUCCESS == SHRegGetValue(hkeyUser,NULL,TEXT("Logon"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwUserLogonLogoff,
|
|
&cbDataSize)) )
|
|
{
|
|
// if can open Logon Key this is up to date.
|
|
return;
|
|
}
|
|
|
|
// need to enum connection names and update the toplevel information.
|
|
while ( ERROR_SUCCESS == RegEnumKey(hkeyUser,dwIndex,
|
|
lpName,cbName) )
|
|
{
|
|
LONG lRet;
|
|
HKEY hKeyConnection;
|
|
|
|
lRet = RegOpenKeyEx( hkeyUser,
|
|
lpName,
|
|
NULL,
|
|
KEY_READ,
|
|
&hKeyConnection );
|
|
|
|
if (ERROR_SUCCESS == lRet)
|
|
{
|
|
|
|
cbDataSize = sizeof(dwUserLogonLogoff);
|
|
|
|
if (ERROR_SUCCESS == SHRegGetValue(hKeyConnection,NULL,TEXT("Logon"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwUserLogonLogoff,
|
|
&cbDataSize) )
|
|
{
|
|
dwLogon |= dwUserLogonLogoff;
|
|
}
|
|
|
|
cbDataSize = sizeof(dwUserLogonLogoff);
|
|
|
|
if (ERROR_SUCCESS == SHRegGetValue(hKeyConnection,NULL,TEXT("Logoff"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwUserLogonLogoff,
|
|
&cbDataSize) )
|
|
{
|
|
dwLogoff |= dwUserLogonLogoff;
|
|
}
|
|
|
|
RegCloseKey(hKeyConnection);
|
|
}
|
|
|
|
dwIndex++;
|
|
}
|
|
|
|
// write out new flags even if errors occured. work thing that happens is
|
|
// we don't set up someones autosync automatically.
|
|
RegSetValueEx(hkeyUser,TEXT("Logon"),NULL, REG_DWORD,
|
|
(LPBYTE) &(dwLogon), sizeof(dwLogon));
|
|
RegSetValueEx(hkeyUser,TEXT("Logoff"),NULL, REG_DWORD,
|
|
(LPBYTE) &(dwLogoff), sizeof(dwLogoff));
|
|
|
|
RegWriteTimeStamp(hkeyUser);
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegUpdateAutoSyncKeyValueSettings
|
|
|
|
Summary: walks through the UserList Updating the AutoSync
|
|
|
|
Returns:
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
void RegUpdateAutoSyncKeyValue(HKEY hkeyAutoSync,DWORD dwLogonDefault,DWORD dwLogoffDefault)
|
|
{
|
|
DWORD dwIndex = 0;
|
|
WCHAR lpName[MAX_PATH];
|
|
DWORD cbName = MAX_PATH;
|
|
LONG lRet;
|
|
DWORD dwLogon = 0;
|
|
DWORD dwLogoff = 0;
|
|
BOOL fSetLogon,fSetLogoff;
|
|
|
|
// need to walk the autosync user key and set the top level information
|
|
// based on whther someone logon/logoff
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
while ( ERROR_SUCCESS == (lRet = RegEnumKey(hkeyAutoSync,dwIndex,
|
|
lpName,cbName) ))
|
|
{
|
|
DWORD dwUserLogonLogoff;
|
|
DWORD cbDataSize = sizeof(dwUserLogonLogoff);
|
|
HKEY hKeyDomainUser;
|
|
|
|
lRet = RegOpenKeyEx( hkeyAutoSync,
|
|
lpName,
|
|
NULL,
|
|
KEY_READ,
|
|
&hKeyDomainUser );
|
|
|
|
if (ERROR_SUCCESS == lRet)
|
|
{
|
|
|
|
// If Query fails don't want to count this as a failed to enum
|
|
// error so don't set lRet
|
|
if (ERROR_SUCCESS == (SHRegGetValue(hKeyDomainUser,NULL,TEXT("Logon"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwUserLogonLogoff,
|
|
&cbDataSize) ))
|
|
{
|
|
dwLogon |= dwUserLogonLogoff;
|
|
}
|
|
|
|
cbDataSize = sizeof(dwUserLogonLogoff);
|
|
|
|
if (ERROR_SUCCESS == lRet)
|
|
{
|
|
if (ERROR_SUCCESS == (SHRegGetValue(hKeyDomainUser,NULL,TEXT("Logoff"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwUserLogonLogoff,
|
|
&cbDataSize) ) )
|
|
{
|
|
dwLogoff |= dwUserLogonLogoff;
|
|
}
|
|
}
|
|
|
|
RegCloseKey(hKeyDomainUser);
|
|
}
|
|
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
break;
|
|
}
|
|
|
|
dwIndex++;
|
|
}
|
|
|
|
fSetLogon = FALSE;
|
|
fSetLogoff = FALSE;
|
|
|
|
// if an error occured, then use the passed in defaults,
|
|
// if set to 1, else don't set.
|
|
if ( (ERROR_SUCCESS != lRet) && (ERROR_NO_MORE_ITEMS != lRet))
|
|
{
|
|
if ( (-1 != dwLogonDefault) && (0 != dwLogonDefault))
|
|
{
|
|
fSetLogon = TRUE;
|
|
dwLogon = dwLogonDefault;
|
|
}
|
|
|
|
if ( (-1 != dwLogoffDefault) && (0 != dwLogoffDefault))
|
|
{
|
|
fSetLogoff = TRUE;
|
|
dwLogoff = dwLogoffDefault;
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
fSetLogon = TRUE;
|
|
fSetLogoff = TRUE;
|
|
}
|
|
|
|
// write out new flags even if errors occured. work thing that happens is
|
|
// we don't set up someones autosync automatically.
|
|
|
|
if (fSetLogon)
|
|
{
|
|
RegSetValueEx(hkeyAutoSync,TEXT("Logon"),NULL, REG_DWORD,
|
|
(LPBYTE) &(dwLogon), sizeof(dwLogon));
|
|
}
|
|
|
|
if (fSetLogoff)
|
|
{
|
|
RegSetValueEx(hkeyAutoSync,TEXT("Logoff"),NULL, REG_DWORD,
|
|
(LPBYTE) &(dwLogoff), sizeof(dwLogoff));
|
|
}
|
|
|
|
RegWriteTimeStamp(hkeyAutoSync);
|
|
|
|
CMutexRegistry.Leave();
|
|
}
|
|
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegUpdateUserIdleKey
|
|
|
|
Summary: given an Idle User Key makes sure it is in the latest
|
|
format and if not updates it.
|
|
|
|
Returns:
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
void RegUpdateUserIdleKey(HKEY hkeyUser,BOOL fForce)
|
|
{
|
|
DWORD dwUserIdleEnabled;
|
|
DWORD cbDataSize = sizeof(dwUserIdleEnabled);
|
|
DWORD dwIndex = 0;
|
|
TCHAR lpName[MAX_PATH];
|
|
DWORD cbName = MAX_PATH;
|
|
|
|
DWORD dwIdleEnabled = 0;
|
|
|
|
if (!fForce && (ERROR_SUCCESS == SHRegGetValue(hkeyUser,NULL,TEXT("IdleEnabled"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwUserIdleEnabled,
|
|
&cbDataSize)) )
|
|
{
|
|
// if can open Logon Key this is up to date.
|
|
return;
|
|
}
|
|
|
|
// need to enum connection names and update the toplevel information.
|
|
while ( ERROR_SUCCESS == RegEnumKey(hkeyUser,dwIndex,
|
|
lpName,cbName) )
|
|
{
|
|
LONG lRet;
|
|
HKEY hKeyConnection;
|
|
|
|
lRet = RegOpenKeyEx( hkeyUser,
|
|
lpName,
|
|
NULL,
|
|
KEY_READ,
|
|
&hKeyConnection );
|
|
|
|
if (ERROR_SUCCESS == lRet)
|
|
{
|
|
|
|
cbDataSize = sizeof(dwUserIdleEnabled);
|
|
|
|
if (ERROR_SUCCESS == SHRegGetValue(hKeyConnection,NULL,TEXT("IdleEnabled"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwUserIdleEnabled,
|
|
&cbDataSize) )
|
|
{
|
|
dwIdleEnabled |= dwUserIdleEnabled;
|
|
}
|
|
|
|
RegCloseKey(hKeyConnection);
|
|
}
|
|
|
|
dwIndex++;
|
|
}
|
|
|
|
// write out new flags even if errors occured. work thing that happens is
|
|
// we don't set up someones autosync automatically.
|
|
RegSetValueEx(hkeyUser,TEXT("IdleEnabled"),NULL, REG_DWORD,
|
|
(LPBYTE) &(dwIdleEnabled), sizeof(dwIdleEnabled));
|
|
RegWriteTimeStamp(hkeyUser);
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegUpdateIdleKeyValue
|
|
|
|
Summary: walks through the UserList Updating the Idle RegKey
|
|
|
|
Returns:
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
void RegUpdateIdleKeyValue(HKEY hkeyIdle,DWORD dwDefault)
|
|
{
|
|
DWORD dwIndex = 0;
|
|
WCHAR lpName[MAX_PATH];
|
|
DWORD dwIdleEnabled = 0;
|
|
LONG lRet = -1;
|
|
BOOL fSetDefault;
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
// need to walk the idle user key and set the top level information
|
|
// based on whther someone logon/logoff
|
|
|
|
fSetDefault = FALSE;
|
|
|
|
|
|
while ( ERROR_SUCCESS == (lRet = RegEnumKey(hkeyIdle,dwIndex,
|
|
lpName,ARRAYSIZE(lpName))) )
|
|
{
|
|
DWORD dwUserIdleEnabled;
|
|
DWORD cbDataSize = sizeof(dwUserIdleEnabled);
|
|
HKEY hKeyDomainUser;
|
|
|
|
lRet = RegOpenKeyEx( hkeyIdle,
|
|
lpName,
|
|
NULL,
|
|
KEY_READ,
|
|
&hKeyDomainUser );
|
|
|
|
if (ERROR_SUCCESS == lRet)
|
|
{
|
|
|
|
// if query fails don't consider this an error as far as
|
|
// setDefault goes.
|
|
if (ERROR_SUCCESS == (SHRegGetValue(hKeyDomainUser,NULL,TEXT("IdleEnabled"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwUserIdleEnabled,
|
|
&cbDataSize) ))
|
|
{
|
|
dwIdleEnabled |= dwUserIdleEnabled;
|
|
}
|
|
|
|
RegCloseKey(hKeyDomainUser);
|
|
}
|
|
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
break;
|
|
}
|
|
|
|
dwIndex++;
|
|
}
|
|
|
|
// if an error occured, then use the passed in defaults,
|
|
// if set to 1, else don't set.
|
|
if ( (ERROR_SUCCESS != lRet) && (ERROR_NO_MORE_ITEMS != lRet))
|
|
{
|
|
if ( (-1 != dwDefault) && (0 != dwDefault))
|
|
{
|
|
fSetDefault = TRUE;
|
|
dwIdleEnabled = dwDefault;
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
fSetDefault = TRUE;
|
|
}
|
|
|
|
|
|
// write out new flags even if errors occured. work thing that happens is
|
|
// we don't set up someones autosync automatically.
|
|
|
|
if (fSetDefault)
|
|
{
|
|
RegSetValueEx(hkeyIdle,TEXT("IdleEnabled"),NULL, REG_DWORD,
|
|
(LPBYTE) &(dwIdleEnabled), sizeof(dwIdleEnabled));
|
|
}
|
|
|
|
RegWriteTimeStamp(hkeyIdle);
|
|
CMutexRegistry.Leave();
|
|
}
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegSetAutoSyncSettings( LPCONNECTIONSETTINGS lpConnectionSettings,
|
|
int iNumConnections)
|
|
|
|
Summary: Sets the logon, logoff and prompt me first user selections.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
STDAPI_(BOOL) RegSetAutoSyncSettings(LPCONNECTIONSETTINGS lpConnectionSettings,
|
|
int iNumConnections,
|
|
CRasUI *pRas,
|
|
BOOL fCleanReg,
|
|
BOOL fSetMachineState,
|
|
BOOL fPerUser)
|
|
{
|
|
SCODE sc = S_OK;
|
|
HKEY hAutoSync;
|
|
HKEY hKeyUser;
|
|
HKEY hkeyConnection;
|
|
DWORD dwUserConfigured = 1;
|
|
DWORD dwLogonDefault = -1;
|
|
DWORD dwLogoffDefault = -1;
|
|
int i;
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
// make sure keys are converted to newest format
|
|
RegUpdateTopLevelKeys();
|
|
|
|
Assert(-1 != TRUE); // we rely on TRUE boolean being something other than -1
|
|
|
|
hAutoSync = RegGetSyncTypeKey(SYNCTYPE_AUTOSYNC,KEY_WRITE| KEY_READ,TRUE);
|
|
|
|
if (NULL == hAutoSync)
|
|
{
|
|
goto EH_Err;
|
|
}
|
|
|
|
hKeyUser = RegOpenUserKey(hAutoSync,KEY_WRITE| KEY_READ,TRUE,fCleanReg);
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
goto EH_Err2;
|
|
}
|
|
|
|
if (fPerUser)
|
|
{
|
|
|
|
smChkTo(EH_Err3,RegSetValueEx(hKeyUser,TEXT("UserConfigured"),NULL, REG_DWORD,
|
|
(LPBYTE) &dwUserConfigured,
|
|
sizeof(dwUserConfigured)));
|
|
}
|
|
else
|
|
{
|
|
DWORD cbDataSize = sizeof(dwUserConfigured);
|
|
|
|
//If this value isn't added yet,
|
|
if (ERROR_SUCCESS == SHRegGetValue(hKeyUser,NULL,TEXT("UserConfigured"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwUserConfigured,
|
|
&cbDataSize))
|
|
{
|
|
//if the user setup their configuration, we won't override on a
|
|
//subsequent handler registration.
|
|
if (dwUserConfigured)
|
|
{
|
|
RegCloseKey(hKeyUser);
|
|
RegCloseKey(hAutoSync);
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (i=0; i<iNumConnections; i++)
|
|
{
|
|
|
|
smChkTo(EH_Err3,RegCreateUserSubKey (hKeyUser,
|
|
lpConnectionSettings[i].pszConnectionName,
|
|
KEY_WRITE | KEY_READ,
|
|
&hkeyConnection));
|
|
|
|
|
|
if (-1 != lpConnectionSettings[i].dwLogon)
|
|
{
|
|
|
|
if (0 != lpConnectionSettings[i].dwLogon)
|
|
{
|
|
dwLogonDefault = lpConnectionSettings[i].dwLogon;
|
|
}
|
|
|
|
smChkTo(EH_Err4,RegSetValueEx(hkeyConnection,TEXT("Logon"),NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings[i].dwLogon),
|
|
sizeof(lpConnectionSettings[i].dwLogon)));
|
|
}
|
|
|
|
if (-1 != lpConnectionSettings[i].dwLogoff)
|
|
{
|
|
|
|
if (0 != lpConnectionSettings[i].dwLogoff)
|
|
{
|
|
dwLogoffDefault = lpConnectionSettings[i].dwLogoff;
|
|
}
|
|
|
|
smChkTo(EH_Err4,RegSetValueEx(hkeyConnection,TEXT("Logoff"),NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings[i].dwLogoff),
|
|
sizeof(lpConnectionSettings[i].dwLogoff)));
|
|
}
|
|
|
|
|
|
if (-1 != lpConnectionSettings[i].dwPromptMeFirst)
|
|
{
|
|
smChkTo(EH_Err4,RegSetValueEx(hkeyConnection,TEXT("PromptMeFirst"),NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings[i].dwPromptMeFirst),
|
|
sizeof(lpConnectionSettings[i].dwPromptMeFirst)));
|
|
}
|
|
|
|
RegCloseKey(hkeyConnection);
|
|
|
|
}
|
|
|
|
// update the toplevel User information
|
|
RegUpdateUserAutosyncKey(hKeyUser,TRUE /* fForce */);
|
|
|
|
// update the top-level key
|
|
RegUpdateAutoSyncKeyValue(hAutoSync,dwLogonDefault,dwLogoffDefault);
|
|
|
|
RegCloseKey(hKeyUser);
|
|
RegCloseKey(hAutoSync);
|
|
|
|
// update our global sens state based on Autosync and Registration
|
|
|
|
if (fSetMachineState)
|
|
{
|
|
RegRegisterForEvents(FALSE /* fUninstall */);
|
|
}
|
|
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
|
|
EH_Err4:
|
|
RegCloseKey(hkeyConnection);
|
|
EH_Err3:
|
|
RegCloseKey(hKeyUser);
|
|
EH_Err2:
|
|
RegCloseKey(hAutoSync);
|
|
EH_Err:
|
|
|
|
CMutexRegistry.Leave();
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
/****************************************************************************
|
|
|
|
Scheduled Sync Registry Functions
|
|
|
|
***************************************************************************F-F*/
|
|
//--------------------------------------------------------------------------------
|
|
//
|
|
// FUNCTION: RegGetSchedFriendlyName(LPCTSTR ptszScheduleGUIDName,
|
|
// LPTSTR ptstrFriendlyName,
|
|
// UINT cchFriendlyName)
|
|
//
|
|
// PURPOSE: Get the friendly name of this Schedule.
|
|
//
|
|
// History: 27-Feb-98 susia Created.
|
|
//
|
|
//--------------------------------------------------------------------------------
|
|
STDAPI_(BOOL) RegGetSchedFriendlyName(LPCTSTR ptszScheduleGUIDName,
|
|
LPTSTR ptstrFriendlyName,
|
|
UINT cchFriendlyName)
|
|
{
|
|
BOOL fResult = FALSE;
|
|
HKEY hKeyUser;
|
|
|
|
hKeyUser = RegGetCurrentUserKey(SYNCTYPE_SCHEDULED,KEY_READ,FALSE);
|
|
if (NULL != hKeyUser)
|
|
{
|
|
HKEY hkeySchedName;
|
|
if (NOERROR == RegOpenKeyEx (hKeyUser, ptszScheduleGUIDName, 0,KEY_READ,
|
|
&hkeySchedName))
|
|
{
|
|
|
|
DWORD cbFriendlyName = cchFriendlyName * sizeof(TCHAR);
|
|
|
|
if (ERROR_SUCCESS == SHRegGetValue(hkeySchedName,NULL,TEXT("FriendlyName"),SRRF_RT_REG_SZ | SRRF_NOEXPAND, NULL,
|
|
(LPBYTE) ptstrFriendlyName, &cbFriendlyName))
|
|
{
|
|
fResult = TRUE;
|
|
}
|
|
|
|
RegCloseKey(hkeySchedName);
|
|
}
|
|
}
|
|
|
|
|
|
RegCloseKey(hKeyUser);
|
|
return fResult;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------
|
|
//
|
|
// FUNCTION: RegSetSchedFriendlyName(LPCTSTR ptszScheduleGUIDName,
|
|
// LPCTSTR ptstrFriendlyName)
|
|
//
|
|
// PURPOSE: Set the friendly name of this Schedule.
|
|
//
|
|
// History: 27-Feb-98 susia Created.
|
|
//
|
|
//--------------------------------------------------------------------------------
|
|
STDAPI_(BOOL) RegSetSchedFriendlyName(LPCTSTR ptszScheduleGUIDName,
|
|
LPCTSTR ptstrFriendlyName)
|
|
|
|
{
|
|
SCODE sc;
|
|
HKEY hkeySchedName;
|
|
HKEY hKeyUser;
|
|
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
hKeyUser = RegGetCurrentUserKey(SYNCTYPE_SCHEDULED,KEY_WRITE | KEY_READ,TRUE);
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
goto EH_Err;
|
|
}
|
|
|
|
smChkTo(EH_Err3,RegCreateUserSubKey (hKeyUser, ptszScheduleGUIDName,
|
|
KEY_WRITE | KEY_READ,
|
|
&hkeySchedName));
|
|
|
|
|
|
smChkTo(EH_Err4,RegSetValueEx (hkeySchedName,TEXT("FriendlyName"),NULL,
|
|
REG_SZ,
|
|
(LPBYTE) ptstrFriendlyName,
|
|
(lstrlen(ptstrFriendlyName) + 1)*sizeof(TCHAR)));
|
|
|
|
RegCloseKey(hkeySchedName);
|
|
RegCloseKey(hKeyUser);
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
|
|
EH_Err4:
|
|
RegCloseKey(hkeySchedName);
|
|
EH_Err3:
|
|
RegCloseKey(hKeyUser);
|
|
EH_Err:
|
|
|
|
CMutexRegistry.Leave();
|
|
return FALSE;
|
|
|
|
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegGetSchedSyncSettings( LPCONNECTIONSETTINGS lpConnectionSettings,
|
|
TCHAR *pszSchedName)
|
|
|
|
Summary: Gets the MakeAutoConnection user selections.
|
|
If no selections on this connection, the default is FALSE
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
STDAPI_(BOOL) RegGetSchedSyncSettings(LPCONNECTIONSETTINGS lpConnectionSettings,
|
|
TCHAR *pszSchedName)
|
|
{
|
|
SCODE sc;
|
|
HKEY hkeySchedName,
|
|
hkeyConnection;
|
|
DWORD cbDataSize = sizeof(lpConnectionSettings->dwHidden);
|
|
HKEY hKeyUser;
|
|
|
|
// Set these first to default values, in case there are no current
|
|
// user preferences in the registry
|
|
lpConnectionSettings->dwConnType = SYNCSCHEDINFO_FLAGS_CONNECTION_LAN;
|
|
lpConnectionSettings->dwMakeConnection = FALSE;
|
|
lpConnectionSettings->dwHidden = FALSE;
|
|
lpConnectionSettings->dwReadOnly = FALSE;
|
|
|
|
hKeyUser = RegGetCurrentUserKey(SYNCTYPE_SCHEDULED,KEY_READ,FALSE);
|
|
if (NULL == hKeyUser)
|
|
{
|
|
goto EH_Err2;
|
|
}
|
|
|
|
|
|
smChkTo(EH_Err3,RegOpenKeyEx (hKeyUser,
|
|
pszSchedName,0,KEY_READ,
|
|
&hkeySchedName));
|
|
|
|
smChkTo(EH_Err4,SHRegGetValue(hkeySchedName,NULL,TEXT("ScheduleHidden"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->dwHidden),
|
|
&cbDataSize));
|
|
|
|
cbDataSize = sizeof(lpConnectionSettings->dwReadOnly);
|
|
|
|
smChkTo(EH_Err4,SHRegGetValue(hkeySchedName,NULL,TEXT("ScheduleReadOnly"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->dwReadOnly),
|
|
&cbDataSize));
|
|
|
|
smChkTo(EH_Err4,RegOpenKeyEx (hkeySchedName,
|
|
lpConnectionSettings->pszConnectionName,
|
|
0,KEY_READ,
|
|
&hkeyConnection));
|
|
|
|
cbDataSize = sizeof(lpConnectionSettings->dwMakeConnection);
|
|
|
|
smChkTo(EH_Err5,SHRegGetValue(hkeyConnection,NULL,TEXT("MakeAutoConnection"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->dwMakeConnection),
|
|
&cbDataSize));
|
|
|
|
cbDataSize = sizeof(lpConnectionSettings->dwConnType);
|
|
|
|
smChkTo(EH_Err5,SHRegGetValue(hkeyConnection,NULL,TEXT("Connection Type"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->dwConnType),
|
|
&cbDataSize));
|
|
|
|
RegCloseKey(hkeyConnection);
|
|
RegCloseKey(hkeySchedName);
|
|
RegCloseKey(hKeyUser);
|
|
return TRUE;
|
|
|
|
EH_Err5:
|
|
RegCloseKey(hkeyConnection);
|
|
EH_Err4:
|
|
RegCloseKey(hkeySchedName);
|
|
EH_Err3:
|
|
RegCloseKey(hKeyUser);
|
|
EH_Err2:
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegSetSchedSyncSettings( LPCONNECTIONSETTINGS lpConnectionSettings,
|
|
TCHAR *pszSchedName)
|
|
|
|
Summary: Sets the hidden and readonly schedule flags.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
STDAPI_(BOOL) RegSetSchedSyncSettings(LPCONNECTIONSETTINGS lpConnectionSettings,
|
|
TCHAR *pszSchedName)
|
|
{
|
|
BOOL fRetVal = FALSE;
|
|
HKEY hKeyUser,
|
|
hkeySchedName,
|
|
hkeyConnection;
|
|
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
hKeyUser = RegGetCurrentUserKey(SYNCTYPE_SCHEDULED,KEY_READ | KEY_WRITE,TRUE);
|
|
|
|
if (hKeyUser)
|
|
{
|
|
if (ERROR_SUCCESS == RegCreateUserSubKey (hKeyUser,
|
|
pszSchedName,
|
|
KEY_WRITE | KEY_READ,
|
|
&hkeySchedName))
|
|
{
|
|
if (ERROR_SUCCESS == RegSetValueEx(hkeySchedName,TEXT("ScheduleHidden"),NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings->dwHidden),
|
|
sizeof(lpConnectionSettings->dwHidden)))
|
|
{
|
|
if (ERROR_SUCCESS == RegSetValueEx(hkeySchedName,TEXT("ScheduleReadOnly"),NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings->dwReadOnly),
|
|
sizeof(lpConnectionSettings->dwReadOnly)))
|
|
{
|
|
if (ERROR_SUCCESS == RegCreateUserSubKey (hkeySchedName, lpConnectionSettings->pszConnectionName,
|
|
KEY_WRITE | KEY_READ, &hkeyConnection))
|
|
{
|
|
if (ERROR_SUCCESS == RegSetValueEx(hkeyConnection,TEXT("MakeAutoConnection"),NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings->dwMakeConnection),
|
|
sizeof(lpConnectionSettings->dwMakeConnection)))
|
|
{
|
|
if (ERROR_SUCCESS == RegSetValueEx(hkeyConnection,TEXT("Connection Type"),NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings->dwConnType),
|
|
sizeof(lpConnectionSettings->dwConnType)))
|
|
{
|
|
fRetVal = TRUE;
|
|
}
|
|
}
|
|
|
|
RegCloseKey(hkeyConnection);
|
|
}
|
|
}
|
|
}
|
|
RegCloseKey(hkeySchedName);
|
|
}
|
|
|
|
RegCloseKey(hKeyUser);
|
|
}
|
|
CMutexRegistry.Leave();
|
|
|
|
return fRetVal;
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
Function: RegGetSchedConnectionName(TCHAR *pszSchedName,
|
|
TCHAR *pszConnectionName,
|
|
DWORD cbConnectionName)
|
|
|
|
|
|
Summary: returns the Connection Name for the Scheduled Item.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegGetSchedConnectionName(TCHAR *pszSchedName,TCHAR *pszConnectionName,
|
|
DWORD cbConnectionName)
|
|
{
|
|
SCODE sc;
|
|
BOOL fResult = FALSE;
|
|
HKEY hKeyUser,hkeySchedName;
|
|
DWORD dwIndex = 0;
|
|
DWORD cb = cbConnectionName;
|
|
|
|
//First Set the connectionName to a default.
|
|
// for now we always assume a LAN card is present.
|
|
// if add support for connection manager should
|
|
// update this.
|
|
|
|
LoadString(g_hmodThisDll, IDS_LAN_CONNECTION, pszConnectionName, cbConnectionName);
|
|
|
|
hKeyUser = RegGetCurrentUserKey(SYNCTYPE_SCHEDULED,KEY_READ,FALSE);
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
goto EH_Err;
|
|
}
|
|
|
|
smChkTo(EH_Err3, RegOpenKeyEx (hKeyUser,pszSchedName,
|
|
0,KEY_READ, &hkeySchedName));
|
|
|
|
// next enumeration of keys is the names of the Schedules connection
|
|
// currently just have one so only get the first.
|
|
if ( ERROR_SUCCESS == RegEnumKeyEx(hkeySchedName,dwIndex,
|
|
pszConnectionName,&cb,NULL,NULL,NULL,NULL))
|
|
{
|
|
fResult = TRUE;
|
|
}
|
|
|
|
RegCloseKey(hkeySchedName);
|
|
RegCloseKey(hKeyUser);
|
|
|
|
return fResult;
|
|
|
|
|
|
EH_Err3:
|
|
RegCloseKey(hKeyUser);
|
|
EH_Err:
|
|
return FALSE;
|
|
|
|
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: BOOL RegSetSIDForSchedule( TCHAR *pszSchedName)
|
|
|
|
Summary: sets the SID for this schedule
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
STDAPI_(BOOL) RegSetSIDForSchedule(TCHAR *pszSchedName)
|
|
|
|
{
|
|
SCODE sc;
|
|
HKEY hkeySchedName;
|
|
TCHAR szSID[MAX_PATH];
|
|
DWORD dwType = REG_SZ;
|
|
|
|
|
|
if (!GetUserTextualSid(szSID, ARRAYSIZE(szSID)))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
HKEY hKeyUser = RegGetCurrentUserKey(SYNCTYPE_SCHEDULED,KEY_READ | KEY_WRITE,TRUE);
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
goto EH_Err2;
|
|
}
|
|
|
|
smChkTo(EH_Err3,RegCreateUserSubKey (hKeyUser, pszSchedName,
|
|
KEY_WRITE | KEY_READ,
|
|
&hkeySchedName));
|
|
|
|
|
|
smChkTo(EH_Err4,RegSetValueEx (hkeySchedName,TEXT("SID"),NULL,
|
|
dwType,
|
|
(LPBYTE) szSID,
|
|
(lstrlen(szSID) + 1)*sizeof(TCHAR)));
|
|
|
|
RegCloseKey(hkeySchedName);
|
|
RegCloseKey(hKeyUser);
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
|
|
EH_Err4:
|
|
RegCloseKey(hkeySchedName);
|
|
EH_Err3:
|
|
RegCloseKey(hKeyUser);
|
|
EH_Err2:
|
|
CMutexRegistry.Leave();
|
|
return FALSE;
|
|
|
|
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: BOOL RegGetSIDForSchedule( TCHAR *ptszTextualSidSched,
|
|
DWORD *pcbSizeSid, TCHAR *pszSchedName)
|
|
|
|
Summary: returns the SID for this schedule
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
STDAPI_(BOOL) RegGetSIDForSchedule(TCHAR *ptszTextualSidSched, DWORD cchTextualSidSched, TCHAR *pszSchedName)
|
|
{
|
|
SCODE sc;
|
|
HKEY hkeySchedName;
|
|
DWORD cbValue;
|
|
|
|
ptszTextualSidSched[0] = TEXT('\0');
|
|
HKEY hKeyUser = RegGetCurrentUserKey(SYNCTYPE_SCHEDULED,KEY_READ,FALSE);
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
goto EH_Err2;
|
|
}
|
|
|
|
smChkTo(EH_Err3, RegOpenKeyEx (hKeyUser,pszSchedName,
|
|
0,KEY_READ, &hkeySchedName));
|
|
|
|
|
|
cbValue = cchTextualSidSched * sizeof(*ptszTextualSidSched);
|
|
if (ERROR_SUCCESS != (sc = SHRegGetValue(hkeySchedName,NULL,TEXT("SID"),SRRF_RT_REG_SZ | SRRF_NOEXPAND, NULL,
|
|
(LPBYTE) ptszTextualSidSched, &cbValue)))
|
|
{
|
|
//handle migration from schedules without SIDs
|
|
// like from Beta to current builds
|
|
RegSetSIDForSchedule(pszSchedName);
|
|
|
|
cbValue = cchTextualSidSched * sizeof(*ptszTextualSidSched);
|
|
SHRegGetValue(hkeySchedName,NULL,TEXT("SID"),SRRF_RT_REG_SZ | SRRF_NOEXPAND, NULL,
|
|
(LPBYTE) ptszTextualSidSched, &cbValue);
|
|
}
|
|
RegCloseKey(hkeySchedName);
|
|
RegCloseKey(hKeyUser);
|
|
return TRUE;
|
|
|
|
|
|
EH_Err3:
|
|
RegCloseKey(hKeyUser);
|
|
EH_Err2:
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
BOOL RemoveScheduledJobFile(TCHAR *pszTaskName)
|
|
|
|
Summary: Remove the TaskScheduler .job file
|
|
|
|
Note: Try to use ITask->Delete first
|
|
Call only when TaskScheduler isn't present or if ITask->Delete failed
|
|
|
|
Returns: Returns TRUE always
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RemoveScheduledJobFile(TCHAR *pszTaskName)
|
|
{
|
|
SCODE sc;
|
|
TCHAR pszFullFileName[MAX_PATH+1];
|
|
TCHAR pszTaskFolderPath[MAX_PATH+1];
|
|
HKEY hkeyTaskSchedulerPath;
|
|
DWORD cbDataSize = sizeof(pszTaskFolderPath);
|
|
|
|
|
|
if (ERROR_SUCCESS == (sc = RegOpenKeyEx (HKEY_LOCAL_MACHINE,
|
|
TEXT("SOFTWARE\\Microsoft\\SchedulingAgent"),
|
|
NULL,KEY_READ,&hkeyTaskSchedulerPath)))
|
|
{
|
|
sc = SHRegGetValue(hkeyTaskSchedulerPath,NULL,TEXT("TasksFolder"),
|
|
SRRF_RT_REG_SZ | SRRF_NOEXPAND, NULL,
|
|
(LPBYTE) pszTaskFolderPath, &cbDataSize);
|
|
RegCloseKey(hkeyTaskSchedulerPath);
|
|
}
|
|
|
|
//If this get doesn't exist then bail.
|
|
if (ERROR_SUCCESS != sc)
|
|
{
|
|
return FALSE;
|
|
}
|
|
ExpandEnvironmentStrings(pszTaskFolderPath,pszFullFileName,MAX_PATH);
|
|
if (!PathAppend(pszFullFileName, pszTaskName))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
//if we fail this, ignore the error. We tried, there isn't much else we can do.
|
|
//So we have a turd file.
|
|
DeleteFile(pszFullFileName);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
BOOL RegRemoveScheduledTask(TCHAR *pszTaskName)
|
|
|
|
Summary: Remove the Scheduled task info from the registry.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegRemoveScheduledTask(TCHAR *pszTaskName)
|
|
{
|
|
HKEY hKeyUser;
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
hKeyUser = RegGetCurrentUserKey(SYNCTYPE_SCHEDULED,KEY_READ | KEY_WRITE,FALSE);
|
|
|
|
if (hKeyUser)
|
|
{
|
|
RegDeleteKeyNT(hKeyUser, pszTaskName);
|
|
RegCloseKey(hKeyUser);
|
|
}
|
|
|
|
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
BOOL RegUninstallSchedules()
|
|
|
|
Summary: Uninstall the scheduled tasks.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
STDAPI_(BOOL) RegUninstallSchedules()
|
|
{
|
|
SCODE sc;
|
|
ITaskScheduler *pITaskScheduler = NULL;
|
|
HKEY hkeySchedSync;
|
|
|
|
// Obtain a task scheduler class instance.
|
|
sc = CoCreateInstance(
|
|
CLSID_CTaskScheduler,
|
|
NULL,
|
|
CLSCTX_INPROC_SERVER,
|
|
IID_ITaskScheduler,
|
|
(VOID **)&pITaskScheduler);
|
|
|
|
if (NOERROR != sc)
|
|
{
|
|
pITaskScheduler = NULL;
|
|
}
|
|
|
|
//now go through and delete the schedules
|
|
|
|
hkeySchedSync = RegGetSyncTypeKey(SYNCTYPE_SCHEDULED,KEY_READ,FALSE);
|
|
|
|
if (hkeySchedSync)
|
|
{
|
|
int iUserCount = 0;
|
|
|
|
while (sc == ERROR_SUCCESS )
|
|
{
|
|
HKEY hkeyDomainUser;
|
|
TCHAR szDomainUser[MAX_KEY_LENGTH];
|
|
SCODE sc2;
|
|
DWORD cchDomainUser = ARRAYSIZE(szDomainUser);
|
|
|
|
sc = RegEnumKeyEx(hkeySchedSync,iUserCount,szDomainUser,&cchDomainUser,NULL,NULL,NULL,NULL);
|
|
iUserCount++;
|
|
|
|
if(sc == ERROR_NO_MORE_ITEMS)
|
|
{
|
|
break;
|
|
}
|
|
|
|
sc2 = RegOpenKeyEx (hkeySchedSync,szDomainUser,0,KEY_READ, &hkeyDomainUser);
|
|
|
|
if (ERROR_SUCCESS == sc2)
|
|
{
|
|
int iScheduleCount = 0;
|
|
|
|
while (sc2 == ERROR_SUCCESS )
|
|
{
|
|
TCHAR ptszScheduleGUIDName[MAX_KEY_LENGTH];
|
|
//Add 4 to ensure that we can hold the .job extension if necessary
|
|
WCHAR pwszScheduleGUIDName[MAX_SCHEDULENAMESIZE + 4];
|
|
DWORD cchScheduleGUIDName = ARRAYSIZE(ptszScheduleGUIDName);
|
|
|
|
sc2 = RegEnumKeyEx(hkeyDomainUser,iScheduleCount,ptszScheduleGUIDName,&cchScheduleGUIDName,NULL,NULL,NULL,NULL);
|
|
|
|
iScheduleCount++;
|
|
|
|
if(sc2 == ERROR_NO_MORE_ITEMS)
|
|
{
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
Assert(ARRAYSIZE(pwszScheduleGUIDName) >= ARRAYSIZE(ptszScheduleGUIDName));
|
|
StringCchCopy(pwszScheduleGUIDName, ARRAYSIZE(pwszScheduleGUIDName), ptszScheduleGUIDName);
|
|
|
|
if ((!pITaskScheduler) ||
|
|
FAILED(pITaskScheduler->Delete(pwszScheduleGUIDName)))
|
|
{
|
|
if (SUCCEEDED(StringCchCat(pwszScheduleGUIDName, ARRAYSIZE(pwszScheduleGUIDName), L".job")))
|
|
{
|
|
RemoveScheduledJobFile(pwszScheduleGUIDName);
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
RegCloseKey(hkeyDomainUser);
|
|
}
|
|
|
|
}
|
|
|
|
RegCloseKey(hkeySchedSync);
|
|
}
|
|
|
|
if (pITaskScheduler)
|
|
{
|
|
pITaskScheduler->Release();
|
|
}
|
|
|
|
RegDeleteKeyNT(HKEY_LOCAL_MACHINE, SCHEDSYNC_REGKEY);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/****************************************************************************
|
|
|
|
Idle Registry Functions
|
|
|
|
***************************************************************************F-F*/
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegGetIdleSyncSettings( LPCONNECTIONSETTINGS lpConnectionSettings)
|
|
|
|
Summary: Gets the Idle Specific settings.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegGetIdleSyncSettings(LPCONNECTIONSETTINGS lpConnectionSettings)
|
|
{
|
|
SCODE sc;
|
|
HKEY hkeyConnection;
|
|
DWORD cbDataSize;
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
// set up defaults
|
|
|
|
lpConnectionSettings->dwIdleEnabled = 0;
|
|
|
|
// following are really per user not per connection
|
|
lpConnectionSettings->ulIdleWaitMinutes = UL_DEFAULTIDLEWAITMINUTES;
|
|
lpConnectionSettings->ulIdleRetryMinutes = UL_DEFAULTIDLERETRYMINUTES;
|
|
lpConnectionSettings->ulDelayIdleShutDownTime = UL_DELAYIDLESHUTDOWNTIME;
|
|
lpConnectionSettings->dwRepeatSynchronization = UL_DEFAULTREPEATSYNCHRONIZATION;
|
|
lpConnectionSettings->dwRunOnBatteries = UL_DEFAULTFRUNONBATTERIES;
|
|
|
|
HKEY hKeyUser = RegGetCurrentUserKey(SYNCTYPE_IDLE,KEY_READ,FALSE);
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
goto EH_Err2;
|
|
}
|
|
|
|
|
|
cbDataSize = sizeof(lpConnectionSettings->ulIdleWaitMinutes);
|
|
|
|
// if got the Idle key open then fill in global settings
|
|
|
|
SHRegGetValue(hKeyUser,NULL,SZ_IDLEWAITAFTERIDLEMINUTESKEY,SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->ulIdleWaitMinutes),
|
|
&cbDataSize);
|
|
|
|
cbDataSize = sizeof(lpConnectionSettings->dwRepeatSynchronization);
|
|
SHRegGetValue(hKeyUser,NULL,SZ_IDLEREPEATESYNCHRONIZATIONKEY,SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->dwRepeatSynchronization),
|
|
&cbDataSize);
|
|
|
|
cbDataSize = sizeof(lpConnectionSettings->ulIdleRetryMinutes);
|
|
SHRegGetValue(hKeyUser,NULL,SZ_IDLERETRYMINUTESKEY,SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->ulIdleRetryMinutes),
|
|
&cbDataSize);
|
|
|
|
cbDataSize = sizeof(lpConnectionSettings->ulDelayIdleShutDownTime);
|
|
SHRegGetValue(hKeyUser,NULL,SZ_IDLEDELAYSHUTDOWNTIMEKEY,SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->ulDelayIdleShutDownTime),
|
|
&cbDataSize);
|
|
|
|
cbDataSize = sizeof(lpConnectionSettings->dwRunOnBatteries);
|
|
SHRegGetValue(hKeyUser,NULL,SZ_IDLERUNONBATTERIESKEY,SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->dwRunOnBatteries),
|
|
&cbDataSize);
|
|
|
|
smChkTo(EH_Err3,RegOpenKeyEx(hKeyUser, lpConnectionSettings->pszConnectionName,0,KEY_READ,
|
|
&hkeyConnection));
|
|
|
|
cbDataSize = sizeof(lpConnectionSettings->dwIdleEnabled);
|
|
SHRegGetValue(hkeyConnection,NULL,TEXT("IdleEnabled"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(lpConnectionSettings->dwIdleEnabled),
|
|
&cbDataSize);
|
|
|
|
RegCloseKey(hkeyConnection);
|
|
RegCloseKey(hKeyUser);
|
|
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
|
|
EH_Err3:
|
|
RegCloseKey(hKeyUser);
|
|
EH_Err2:
|
|
CMutexRegistry.Leave();
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegSetIdleSyncSettings(LPCONNECTIONSETTINGS lpConnectionSettings,
|
|
int iNumConnections,
|
|
CRasUI *pRas,
|
|
BOOL fCleanReg,
|
|
BOOL fPerUser)
|
|
|
|
Summary: Sets the Idle Information user selections.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
STDAPI_(BOOL) RegSetIdleSyncSettings(LPCONNECTIONSETTINGS lpConnectionSettings,
|
|
int iNumConnections,
|
|
CRasUI *pRas,
|
|
BOOL fCleanReg,
|
|
BOOL fPerUser)
|
|
{
|
|
HKEY hkeyIdleSync = NULL;
|
|
HKEY hKeyUser;;
|
|
HKEY hkeyConnection;
|
|
HRESULT hr;
|
|
ULONG ulWaitMinutes = UL_DEFAULTWAITMINUTES;
|
|
DWORD dwIdleEnabled;
|
|
BOOL fRunOnBatteries = UL_DEFAULTFRUNONBATTERIES;
|
|
int i;
|
|
DWORD cbDataSize;
|
|
DWORD dwUserConfigured;
|
|
DWORD dwTopLevelDefaultValue = -1;
|
|
|
|
RegUpdateTopLevelKeys(); // make sure top-level keys are latest version
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
Assert(-1 != TRUE); // we rely on TRUE boolean being something other than -1
|
|
|
|
hkeyIdleSync = RegGetSyncTypeKey(SYNCTYPE_IDLE,KEY_WRITE| KEY_READ,TRUE);
|
|
|
|
if (NULL == hkeyIdleSync)
|
|
{
|
|
goto EH_Err;
|
|
}
|
|
|
|
hKeyUser = RegOpenUserKey(hkeyIdleSync,KEY_WRITE| KEY_READ,TRUE,fCleanReg);
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
goto EH_Err2;
|
|
}
|
|
|
|
|
|
if (fPerUser)
|
|
{
|
|
|
|
dwUserConfigured = 1;
|
|
|
|
if (ERROR_SUCCESS != RegSetValueEx(hKeyUser,TEXT("UserConfigured"),NULL, REG_DWORD,
|
|
(LPBYTE) &dwUserConfigured,
|
|
sizeof(dwUserConfigured)))
|
|
{
|
|
goto EH_Err3;
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
cbDataSize = sizeof(dwUserConfigured);
|
|
|
|
//If this value isn't added yet,
|
|
if (ERROR_SUCCESS == SHRegGetValue(hKeyUser,NULL,TEXT("UserConfigured"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwUserConfigured,
|
|
&cbDataSize))
|
|
{
|
|
//if the user setup their configuration, we won't override on a
|
|
//subsequent handler registration.
|
|
if (dwUserConfigured)
|
|
{
|
|
RegCloseKey(hkeyIdleSync);
|
|
RegCloseKey(hKeyUser);
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (i=0; i<iNumConnections; i++)
|
|
{
|
|
|
|
if (ERROR_SUCCESS != RegCreateUserSubKey (hKeyUser,
|
|
lpConnectionSettings[i].pszConnectionName,
|
|
KEY_WRITE | KEY_READ,
|
|
&hkeyConnection))
|
|
{
|
|
goto EH_Err3;
|
|
}
|
|
|
|
|
|
hr = RegSetValueEx(hkeyConnection,TEXT("IdleEnabled"),NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings[i].dwIdleEnabled),
|
|
sizeof(lpConnectionSettings[i].dwIdleEnabled));
|
|
|
|
if (lpConnectionSettings[i].dwIdleEnabled)
|
|
{
|
|
dwTopLevelDefaultValue = lpConnectionSettings[i].dwIdleEnabled;
|
|
}
|
|
|
|
RegCloseKey(hkeyConnection);
|
|
|
|
}
|
|
// write out the global idle information for Retry minutes and DelayIdleShutDown.
|
|
// then call function to reg/unregister with TS.
|
|
|
|
|
|
Assert(hkeyIdleSync); // should have already returned if this failed.
|
|
|
|
if (iNumConnections) // make sure at least one connection
|
|
{
|
|
|
|
// ONLY UPDATE SETTINGS IF NOT -1;
|
|
|
|
if (-1 != lpConnectionSettings[0].ulIdleRetryMinutes)
|
|
{
|
|
hr = RegSetValueEx(hKeyUser,SZ_IDLERETRYMINUTESKEY,NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings[0].ulIdleRetryMinutes),
|
|
sizeof(lpConnectionSettings[0].ulIdleRetryMinutes));
|
|
}
|
|
|
|
if (-1 != lpConnectionSettings[0].ulDelayIdleShutDownTime)
|
|
{
|
|
hr = RegSetValueEx(hKeyUser,SZ_IDLEDELAYSHUTDOWNTIMEKEY,NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings[0].ulDelayIdleShutDownTime),
|
|
sizeof(lpConnectionSettings[0].ulDelayIdleShutDownTime));
|
|
}
|
|
|
|
if (-1 != lpConnectionSettings[0].dwRepeatSynchronization)
|
|
{
|
|
hr = RegSetValueEx(hKeyUser,SZ_IDLEREPEATESYNCHRONIZATIONKEY,NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings[0].dwRepeatSynchronization),
|
|
sizeof(lpConnectionSettings[0].dwRepeatSynchronization));
|
|
}
|
|
|
|
|
|
|
|
if (-1 != lpConnectionSettings[0].ulIdleWaitMinutes)
|
|
{
|
|
hr = RegSetValueEx(hKeyUser,SZ_IDLEWAITAFTERIDLEMINUTESKEY,NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings[0].ulIdleWaitMinutes),
|
|
sizeof(lpConnectionSettings[0].ulIdleWaitMinutes));
|
|
}
|
|
|
|
|
|
if (-1 != lpConnectionSettings[0].dwRunOnBatteries)
|
|
{
|
|
hr = RegSetValueEx(hKeyUser,SZ_IDLERUNONBATTERIESKEY,NULL, REG_DWORD,
|
|
(LPBYTE) &(lpConnectionSettings[0].dwRunOnBatteries),
|
|
sizeof(lpConnectionSettings[0].dwRunOnBatteries));
|
|
}
|
|
|
|
ulWaitMinutes = lpConnectionSettings[0].ulIdleWaitMinutes;
|
|
fRunOnBatteries = lpConnectionSettings[0].dwRunOnBatteries;
|
|
|
|
// if -1 is passed in for ulWait or fRun on Batteries we need to
|
|
// get these and set them up so they can be passed onto Task Schedule
|
|
if (-1 == ulWaitMinutes)
|
|
{
|
|
cbDataSize = sizeof(ulWaitMinutes);
|
|
|
|
if (!(ERROR_SUCCESS == SHRegGetValue(hKeyUser,NULL,SZ_IDLEWAITAFTERIDLEMINUTESKEY,SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &ulWaitMinutes,
|
|
&cbDataSize)) )
|
|
{
|
|
ulWaitMinutes = UL_DEFAULTIDLEWAITMINUTES;
|
|
}
|
|
}
|
|
|
|
if (-1 == fRunOnBatteries)
|
|
{
|
|
cbDataSize = sizeof(fRunOnBatteries);
|
|
|
|
if (!(ERROR_SUCCESS == SHRegGetValue(hKeyUser,NULL,SZ_IDLERUNONBATTERIESKEY,SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &fRunOnBatteries,
|
|
&cbDataSize)) )
|
|
{
|
|
fRunOnBatteries = UL_DEFAULTFRUNONBATTERIES;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
RegUpdateUserIdleKey(hKeyUser,TRUE /* fForce */); // set userlevel IdleFlags
|
|
|
|
// read in dwIdleEnabled key now that the UserKey is Updated.
|
|
cbDataSize = sizeof(dwIdleEnabled);
|
|
|
|
if (!(ERROR_SUCCESS == SHRegGetValue(hKeyUser,NULL,TEXT("IdleEnabled"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwIdleEnabled,
|
|
&cbDataSize)) )
|
|
{
|
|
AssertSz(0,"Unable to query User IdleEnabledKey");
|
|
dwIdleEnabled = FALSE;
|
|
}
|
|
|
|
RegCloseKey(hKeyUser);
|
|
|
|
// update the toplevel IdleSyncInfo
|
|
RegUpdateIdleKeyValue(hkeyIdleSync,dwTopLevelDefaultValue);
|
|
|
|
RegCloseKey(hkeyIdleSync);
|
|
|
|
CMutexRegistry.Leave();
|
|
|
|
RegRegisterForIdleTrigger(dwIdleEnabled,ulWaitMinutes,fRunOnBatteries);
|
|
|
|
return TRUE;
|
|
|
|
EH_Err3:
|
|
RegCloseKey(hKeyUser);
|
|
EH_Err2:
|
|
RegCloseKey(hkeyIdleSync);
|
|
EH_Err:
|
|
CMutexRegistry.Leave();
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: RegRegisterForIdleTrigger()
|
|
|
|
Summary: Sets or removes the Idle trigger.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegRegisterForIdleTrigger(BOOL fRegister,ULONG ulWaitMinutes,BOOL fRunOnBatteries)
|
|
{
|
|
HRESULT hr;
|
|
CSyncMgrSynchronize *pSyncMgrSynchronize;
|
|
LPSYNCSCHEDULEMGR pScheduleMgr;
|
|
|
|
// Review - Currently the mobsync dll is registered as apartment
|
|
// and this function can be called from Logon/Logoff which is FreeThreaded
|
|
// Correct fix is to change DLL to be registered as BOTH but until then
|
|
// just create the class directly.
|
|
|
|
pSyncMgrSynchronize = new CSyncMgrSynchronize;
|
|
hr = E_OUTOFMEMORY;
|
|
|
|
if (pSyncMgrSynchronize)
|
|
{
|
|
hr = pSyncMgrSynchronize->QueryInterface(IID_ISyncScheduleMgr,(void **) &pScheduleMgr);
|
|
pSyncMgrSynchronize->Release();
|
|
}
|
|
|
|
|
|
if (NOERROR != hr)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
if (fRegister)
|
|
{
|
|
ISyncSchedule *pSyncSchedule = NULL;
|
|
SYNCSCHEDULECOOKIE SyncScheduleCookie;
|
|
BOOL fNewlyCreated = FALSE;
|
|
|
|
SyncScheduleCookie = GUID_IDLESCHEDULE;
|
|
|
|
// if there isn't an existing schedule create one, else update
|
|
// the existing.
|
|
|
|
if (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) == (hr = pScheduleMgr->OpenSchedule(&SyncScheduleCookie,0,&pSyncSchedule)))
|
|
{
|
|
SyncScheduleCookie = GUID_IDLESCHEDULE;
|
|
// see if it is an exiting schedule.
|
|
|
|
hr = pScheduleMgr->CreateSchedule(L"Idle",0,&SyncScheduleCookie,&pSyncSchedule);
|
|
fNewlyCreated = TRUE;
|
|
}
|
|
|
|
// If created or found a schedule update the trigger settings.
|
|
if (NOERROR == hr)
|
|
{
|
|
ITaskTrigger *pTrigger;
|
|
|
|
pSyncSchedule->SetFlags(SYNCSCHEDINFO_FLAGS_READONLY | SYNCSCHEDINFO_FLAGS_HIDDEN);
|
|
|
|
if (NOERROR == pSyncSchedule->GetTrigger(&pTrigger))
|
|
{
|
|
TASK_TRIGGER trigger;
|
|
ITask *pTask;
|
|
|
|
trigger.cbTriggerSize = sizeof(trigger);
|
|
|
|
if (SUCCEEDED(pTrigger->GetTrigger(&trigger)))
|
|
{
|
|
DWORD dwFlags;
|
|
|
|
// need to set Idle, ULONG ulWaitMinutes,BOOL fRunOnBatteries
|
|
trigger.cbTriggerSize = sizeof(trigger);
|
|
trigger.TriggerType = TASK_EVENT_TRIGGER_ON_IDLE;
|
|
trigger.rgFlags = 0;
|
|
pTrigger->SetTrigger(&trigger);
|
|
|
|
if (SUCCEEDED(pSyncSchedule->GetITask(&pTask)))
|
|
{
|
|
// set up if run on battery.
|
|
if (SUCCEEDED(pTask->GetFlags(&dwFlags)))
|
|
{
|
|
dwFlags &= ~TASK_FLAG_DONT_START_IF_ON_BATTERIES;
|
|
dwFlags |= !fRunOnBatteries ? TASK_FLAG_DONT_START_IF_ON_BATTERIES : 0;
|
|
|
|
dwFlags |= TASK_FLAG_RUN_ONLY_IF_LOGGED_ON; // don't require password.
|
|
|
|
pTask->SetFlags(dwFlags);
|
|
}
|
|
|
|
// if this schedule was just created, get the current user name and reset
|
|
// account information password to NULL, If existing schedule don't change
|
|
// since user may have added a password for schedule to run while not logged on.
|
|
if (fNewlyCreated)
|
|
{
|
|
TCHAR szAccountName[MAX_DOMANDANDMACHINENAMESIZE];
|
|
WCHAR *pszAccountName = NULL;
|
|
|
|
// Review, this never returns an errorl
|
|
*szAccountName = WCHAR('\0');
|
|
GetDefaultDomainAndUserName(szAccountName,TEXT("\\"),ARRAYSIZE(szAccountName));
|
|
|
|
pszAccountName = szAccountName;
|
|
|
|
Assert(pszAccountName);
|
|
|
|
if (pszAccountName)
|
|
{
|
|
pTask->SetAccountInformation(pszAccountName,NULL);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
// set up the IdleWaitTime.
|
|
pTask->SetIdleWait((WORD) ulWaitMinutes,1);
|
|
|
|
// turn off the option to kill task after xxx minutes.
|
|
pTask->SetMaxRunTime(INFINITE);
|
|
|
|
pTask->Release();
|
|
}
|
|
|
|
pTrigger->Release();
|
|
|
|
}
|
|
|
|
pSyncSchedule->Save();
|
|
}
|
|
|
|
|
|
|
|
pSyncSchedule->Release();
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
SYNCSCHEDULECOOKIE SyncScheduleCookie = GUID_IDLESCHEDULE;
|
|
|
|
// see if there is an existing schedule and if so remove it.
|
|
pScheduleMgr->RemoveSchedule(&SyncScheduleCookie);
|
|
|
|
}
|
|
|
|
pScheduleMgr->Release();
|
|
|
|
// set the temporary sens flags according so it can start on an idle trigger.
|
|
// not an error to not be able to get and set key since sens will
|
|
// be running anyways eventually.
|
|
HKEY hkeyAutoSync;
|
|
DWORD dwFlags = 0;
|
|
DWORD cbDataSize = sizeof(dwFlags);
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
hkeyAutoSync = RegGetSyncTypeKey(SYNCTYPE_AUTOSYNC,KEY_WRITE | KEY_READ,TRUE);
|
|
if (NULL == hkeyAutoSync)
|
|
{
|
|
CMutexRegistry.Leave();
|
|
return FALSE;
|
|
}
|
|
|
|
if (ERROR_SUCCESS == SHRegGetValue(hkeyAutoSync,NULL,TEXT("Flags"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(dwFlags),&cbDataSize))
|
|
{
|
|
// Here we are setting Idle only so retail the other settings.
|
|
dwFlags &= ~AUTOSYNC_IDLE;
|
|
dwFlags |= (fRegister? AUTOSYNC_IDLE : 0);
|
|
|
|
RegSetValueEx(hkeyAutoSync,TEXT("Flags"),NULL, REG_DWORD,
|
|
(LPBYTE) &(dwFlags), sizeof(dwFlags));
|
|
}
|
|
RegCloseKey(hkeyAutoSync);
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
BOOL RegGetSyncSettings(DWORD dwSyncType,LPCONNECTIONSETTINGS lpConnectionSettings)
|
|
|
|
Summary: Get ConnectionSettings appropriate to the synctype..
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegGetSyncSettings(DWORD dwSyncType,LPCONNECTIONSETTINGS lpConnectionSettings)
|
|
{
|
|
|
|
switch(dwSyncType)
|
|
{
|
|
case SYNCTYPE_AUTOSYNC:
|
|
return RegGetAutoSyncSettings(lpConnectionSettings);
|
|
break;
|
|
case SYNCTYPE_IDLE:
|
|
return RegGetIdleSyncSettings(lpConnectionSettings);
|
|
break;
|
|
default:
|
|
AssertSz(0,"Unknown SyncType in RegGetSyncSettings");
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
Manual Sync Registry Functions
|
|
|
|
***************************************************************************F-F*/
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
BOOL RegRemoveManualSyncSettings(TCHAR *pszTaskName)
|
|
|
|
Summary: Remove the manual settings info from the registry.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegRemoveManualSyncSettings(TCHAR *pszConnectionName)
|
|
{
|
|
HKEY hkeyUser;
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
hkeyUser = RegGetCurrentUserKey(SYNCTYPE_MANUAL,KEY_WRITE | KEY_READ,FALSE);
|
|
|
|
if (hkeyUser)
|
|
{
|
|
RegDeleteKeyNT(hkeyUser, pszConnectionName);
|
|
RegCloseKey(hkeyUser);
|
|
}
|
|
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
BOOL RegWriteEvents(BOOL fWanLogon,BOOL fWanLogoff,BOOL fLanLogon,BOOL fLanLogoff)
|
|
|
|
Summary: Write out the Wan/Lan Logon/Logoff preferences fo SENS knows whether to invoke us.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
// Run key under HKLM
|
|
const WCHAR wszRunKey[] = TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Run");
|
|
const WCHAR wszRunKeyCommandLine[] = TEXT("%SystemRoot%\\system32\\mobsync.exe /logon");
|
|
|
|
|
|
STDAPI_(BOOL) RegWriteEvents(BOOL Logon,BOOL Logoff)
|
|
{
|
|
HRESULT hr;
|
|
HKEY hkeyAutoSync;
|
|
DWORD dwFlags = 0;
|
|
DWORD dwType = REG_DWORD;
|
|
DWORD cbDataSize = sizeof(DWORD);
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
hkeyAutoSync = RegGetSyncTypeKey(SYNCTYPE_AUTOSYNC,KEY_WRITE | KEY_READ,TRUE);
|
|
if (NULL == hkeyAutoSync)
|
|
{
|
|
|
|
CMutexRegistry.Leave();
|
|
return FALSE;
|
|
}
|
|
|
|
RegQueryValueEx(hkeyAutoSync,TEXT("Flags"),NULL, &dwType,
|
|
(LPBYTE) &(dwFlags),
|
|
&cbDataSize);
|
|
|
|
|
|
// Review, Shouldn't need to worry about schedule/idle once IsNetworkAlive
|
|
// is setup properly. Leave for now first time anyone sets idle or schedule
|
|
// stuck
|
|
|
|
|
|
// Here we are setting autosync only,
|
|
// so retain the registry settings for scheduled and idle.
|
|
|
|
dwFlags &= ~(AUTOSYNC_WAN_LOGON | AUTOSYNC_LAN_LOGON | AUTOSYNC_LOGONWITHRUNKEY
|
|
| AUTOSYNC_WAN_LOGOFF | AUTOSYNC_LAN_LOGOFF);
|
|
|
|
dwFlags |= (Logoff ? AUTOSYNC_WAN_LOGOFF : 0);
|
|
dwFlags |= (Logoff ? AUTOSYNC_LAN_LOGOFF : 0);
|
|
|
|
// Since now use Run key instead of SENS always set both Logon Flags
|
|
// that SENS looks for to do a CreateProcess on us to FALSE.
|
|
// Then set the AUTOSYNC_LOGONWITHRUNKEY key to true so sens still gets loaded.
|
|
|
|
dwFlags |= (Logon ? AUTOSYNC_LOGONWITHRUNKEY : 0);
|
|
|
|
hr = RegSetValueEx(hkeyAutoSync,TEXT("Flags"),NULL, REG_DWORD,
|
|
(LPBYTE) &(dwFlags), sizeof(DWORD));
|
|
|
|
RegCloseKey(hkeyAutoSync);
|
|
|
|
// now add /delete the run key appropriately.
|
|
|
|
|
|
HKEY hKeyRun;
|
|
|
|
// call private RegOpen since don't want to set security on RunKey
|
|
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE, wszRunKey, NULL, KEY_READ | KEY_WRITE, &hKeyRun))
|
|
{
|
|
if (Logon)
|
|
{
|
|
RegSetValueEx(hKeyRun, SZ_SYNCMGRNAME, 0, REG_EXPAND_SZ,
|
|
(BYTE *) wszRunKeyCommandLine,(lstrlen(wszRunKeyCommandLine) + 1)*sizeof(TCHAR));
|
|
}
|
|
else
|
|
{
|
|
RegDeleteValue(hKeyRun, SZ_SYNCMGRNAME);
|
|
}
|
|
|
|
|
|
RegCloseKey(hKeyRun);
|
|
}
|
|
else
|
|
{
|
|
// if can't open run key try calling SENS if that fails give up.
|
|
SyncMgrExecCmd_UpdateRunKey(Logon);
|
|
}
|
|
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
BOOL RegFixRunKey(BOOL fScheduled)
|
|
|
|
Summary: The original version of SyncMgr for WinMe/Win2000 wrote the
|
|
"run" value as "mobsync.exe /logon". Since this is not a fully-
|
|
qualified path to the mobsync.exe image, the system's search
|
|
path is utilized to locate the image. This can create an
|
|
opportunity for someone to build a 'trojan' mobsync.exe, place
|
|
it in the search path ahead of the real mobsync.exe and have
|
|
the 'trojan' code run whenever a synchronization is invoked.
|
|
To fix this, the path must be stored in the registry using
|
|
fully-qualified syntax.
|
|
|
|
i.e. "%SystemRoot%\System32\mobsync.exe /logon"
|
|
|
|
This function is called from DllRegisterServer to correct this
|
|
registry entry during setup.
|
|
|
|
Returns: Always returns TRUE.
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegFixRunKey(void)
|
|
{
|
|
HKEY hKeyRun;
|
|
|
|
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE,
|
|
wszRunKey,
|
|
NULL,
|
|
KEY_READ | KEY_WRITE,
|
|
&hKeyRun))
|
|
{
|
|
TCHAR szRunValue[MAX_PATH];
|
|
DWORD cbValue = sizeof(szRunValue);
|
|
if (ERROR_SUCCESS == SHRegGetValue(hKeyRun,NULL,SZ_SYNCMGRNAME, SRRF_RT_REG_SZ | SRRF_NOEXPAND, NULL,
|
|
(LPBYTE)szRunValue, &cbValue))
|
|
{
|
|
if (0 == lstrcmp(szRunValue, TEXT("mobsync.exe /logon")))
|
|
{
|
|
//
|
|
// Upgrade only if it's our original value.
|
|
//
|
|
RegSetValueEx(hKeyRun,
|
|
SZ_SYNCMGRNAME,
|
|
0,
|
|
REG_EXPAND_SZ,
|
|
(BYTE *)wszRunKeyCommandLine,
|
|
(lstrlen(wszRunKeyCommandLine) + 1) * sizeof(TCHAR));
|
|
}
|
|
}
|
|
RegCloseKey(hKeyRun);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
BOOL RegRegisterForScheduledTasks(BOOL fScheduled)
|
|
|
|
Summary: Register/unregister for scheduled tasks
|
|
so SENS knows whether to invoke us.
|
|
|
|
Returns: Returns TRUE if successful, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
|
|
STDAPI_(BOOL) RegRegisterForScheduledTasks(BOOL fScheduled)
|
|
{
|
|
HKEY hkeyAutoSync;
|
|
DWORD dwFlags = 0;
|
|
DWORD cbDataSize = sizeof(dwFlags);
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
hkeyAutoSync = RegGetSyncTypeKey(SYNCTYPE_AUTOSYNC,KEY_WRITE | KEY_READ,TRUE);
|
|
|
|
if (NULL == hkeyAutoSync)
|
|
{
|
|
CMutexRegistry.Leave();
|
|
return FALSE;
|
|
}
|
|
|
|
SHRegGetValue(hkeyAutoSync,NULL,TEXT("Flags"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &(dwFlags), &cbDataSize);
|
|
|
|
|
|
// Here we are setting schedsync only,
|
|
// so retain the registry settings for autosync and idle.
|
|
|
|
dwFlags &= AUTOSYNC_WAN_LOGON |
|
|
AUTOSYNC_WAN_LOGOFF |
|
|
AUTOSYNC_LAN_LOGON |
|
|
AUTOSYNC_LAN_LOGOFF |
|
|
AUTOSYNC_IDLE;
|
|
|
|
dwFlags |= (fScheduled? AUTOSYNC_SCHEDULED : 0);
|
|
|
|
RegSetValueEx(hkeyAutoSync,TEXT("Flags"),NULL, REG_DWORD,
|
|
(LPBYTE) &(dwFlags), sizeof(dwFlags));
|
|
|
|
RegCloseKey(hkeyAutoSync);
|
|
|
|
CMutexRegistry.Leave();
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: RegGetCombinedUserRegFlags, private
|
|
//
|
|
// Synopsis: Gets an or'ing together of user settings for setting up globals.
|
|
//
|
|
// Arguments: [dwSyncMgrRegisterFlags] - On Success gets set to flags
|
|
// on failure they are set to zero
|
|
//
|
|
// Returns: Appropriate status code
|
|
//
|
|
// Modifies:
|
|
//
|
|
// History: 24-Aug-98 rogerg Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
BOOL RegGetCombinedUserRegFlags(DWORD *pdwSyncMgrRegisterFlags)
|
|
{
|
|
HKEY hkey;
|
|
BOOL fResult = TRUE;
|
|
DWORD dw;
|
|
DWORD cb = sizeof(dw);
|
|
|
|
|
|
*pdwSyncMgrRegisterFlags = 0;
|
|
|
|
// update the AutoSync Key
|
|
hkey = RegGetSyncTypeKey(SYNCTYPE_AUTOSYNC,KEY_READ,FALSE);
|
|
|
|
if (hkey)
|
|
{
|
|
|
|
|
|
if (ERROR_SUCCESS == SHRegGetValue(hkey,NULL,TEXT("Logon"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dw, &cb) )
|
|
{
|
|
*pdwSyncMgrRegisterFlags |= dw ? SYNCMGRREGISTERFLAG_CONNECT : 0;
|
|
}
|
|
|
|
cb = sizeof(dw);
|
|
|
|
if (ERROR_SUCCESS == SHRegGetValue(hkey,NULL,TEXT("Logoff"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dw, &cb) )
|
|
{
|
|
*pdwSyncMgrRegisterFlags |= dw ? SYNCMGRREGISTERFLAG_PENDINGDISCONNECT : 0;
|
|
}
|
|
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
|
|
// update the Idle Key
|
|
hkey = RegGetSyncTypeKey(SYNCTYPE_IDLE,KEY_READ,FALSE);
|
|
|
|
if (hkey)
|
|
{
|
|
cb = sizeof(dw);
|
|
|
|
if (ERROR_SUCCESS == SHRegGetValue(hkey,NULL,TEXT("IdleEnabled"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dw, &cb) )
|
|
{
|
|
*pdwSyncMgrRegisterFlags |= dw ? SYNCMGRREGISTERFLAG_IDLE : 0;
|
|
}
|
|
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
return TRUE; // always return true but don't set flags on error.
|
|
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: RegGetCombinedHandlerRegFlags, private
|
|
//
|
|
// Synopsis: Gets an or'ing together of handler registration Keys
|
|
//
|
|
// Arguments: [dwSyncMgrRegisterFlags] - On Success gets set to flags
|
|
// on failure they are set to zero
|
|
// [ft] - On Success filed with timestamp
|
|
//
|
|
// Returns: Appropriate status code
|
|
//
|
|
// Modifies:
|
|
//
|
|
// History: 24-Aug-98 rogerg Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
BOOL RegGetCombinedHandlerRegFlags(DWORD *pdwSyncMgrRegisterFlags,FILETIME *pft)
|
|
{
|
|
HKEY hkey;
|
|
|
|
*pdwSyncMgrRegisterFlags = 0;
|
|
|
|
hkey = RegGetHandlerTopLevelKey(KEY_READ);
|
|
if (hkey)
|
|
{
|
|
DWORD dwRegistrationFlags;
|
|
DWORD cbDataSize = sizeof(dwRegistrationFlags);
|
|
|
|
if (ERROR_SUCCESS == SHRegGetValue(hkey,NULL,SZ_REGISTRATIONFLAGSKEY,SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwRegistrationFlags,&cbDataSize) )
|
|
{
|
|
*pdwSyncMgrRegisterFlags = dwRegistrationFlags;
|
|
}
|
|
|
|
RegGetTimeStamp(hkey,pft);
|
|
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
return TRUE; // always return true but don't set flags on error.
|
|
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: RegGetChangedHandlerFlags, private
|
|
//
|
|
// Synopsis: Gets an or'ing together of handler registration Keys
|
|
// that have changed since the given FILETIME
|
|
//
|
|
// Arguments: [pft] - Pointer to FileTime for Compare
|
|
// [pdwChangedFlags] - On Success filed with flags that channged
|
|
//
|
|
// Returns: TRUE if could gather flags.
|
|
//
|
|
// Modifies:
|
|
//
|
|
// History: 24-Aug-98 rogerg Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
BOOL RegGetChangedHandlerFlags(FILETIME *pft,DWORD *pdwHandlerChandedFlags)
|
|
{
|
|
HKEY hkeyHandler;
|
|
|
|
*pdwHandlerChandedFlags = 0;
|
|
|
|
hkeyHandler = RegGetHandlerTopLevelKey(KEY_READ);
|
|
|
|
if (hkeyHandler)
|
|
{
|
|
TCHAR lpName[MAX_PATH + 1];
|
|
DWORD dwRegistrationFlags = 0;
|
|
FILETIME ftHandlerReg;
|
|
DWORD dwIndex = 0;
|
|
LONG lRet;
|
|
HKEY hKeyClsid;
|
|
|
|
// enumerate the keys
|
|
while ( ERROR_SUCCESS == RegEnumKey(hkeyHandler,dwIndex,lpName,ARRAYSIZE(lpName)) )
|
|
{
|
|
lRet = RegOpenKeyEx( hkeyHandler,
|
|
lpName,
|
|
NULL,
|
|
KEY_READ,
|
|
&hKeyClsid );
|
|
|
|
|
|
if (ERROR_SUCCESS == lRet)
|
|
{
|
|
|
|
RegGetTimeStamp(hKeyClsid,&ftHandlerReg);
|
|
|
|
// handler reg is new time than our gvien time add it to the flags.
|
|
if (CompareFileTime(pft,&ftHandlerReg) < 0)
|
|
{
|
|
DWORD dwHandlerRegFlags;
|
|
DWORD cbDataSize = sizeof(dwHandlerRegFlags);
|
|
if (ERROR_SUCCESS == SHRegGetValue(hKeyClsid,NULL,SZ_REGISTRATIONFLAGSKEY,SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwHandlerRegFlags,&cbDataSize) )
|
|
{
|
|
dwRegistrationFlags |= dwHandlerRegFlags;
|
|
}
|
|
}
|
|
|
|
RegCloseKey(hKeyClsid);
|
|
}
|
|
|
|
dwIndex++;
|
|
}
|
|
|
|
|
|
*pdwHandlerChandedFlags = dwRegistrationFlags;
|
|
|
|
RegCloseKey(hkeyHandler);
|
|
}
|
|
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Member: RegRegisterForEvents, private
|
|
//
|
|
// Synopsis: Registers/UnRegisters for appropriate SENS and WinLogon Events.
|
|
// and any other per machine registration we need to do
|
|
//
|
|
// Arguments: [fUninstall] - set to true by uninstall to force us to unregister
|
|
// regardless of current machine state.
|
|
//
|
|
// Returns: Appropriate status code
|
|
//
|
|
// Modifies:
|
|
//
|
|
// History: 05-Nov-97 rogerg Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
// !!!Warning - Assumes toplevel key information is up to date.
|
|
|
|
STDAPI RegRegisterForEvents(BOOL fUninstall)
|
|
{
|
|
HRESULT hr = NOERROR;
|
|
BOOL fLogon = FALSE;
|
|
BOOL fLogoff = FALSE;
|
|
BOOL fIdle = FALSE;
|
|
#ifdef _SENS
|
|
IEventSystem *pEventSystem;
|
|
#endif // _SENS
|
|
CCriticalSection cCritSect(&g_DllCriticalSection,GetCurrentThreadId());
|
|
|
|
cCritSect.Enter();
|
|
|
|
if (!fUninstall)
|
|
{
|
|
FILETIME ftHandlerReg;
|
|
DWORD dwSyncMgrUsersRegisterFlags; // or'ing of all Users settings
|
|
DWORD dwSyncMgrHandlerRegisterFlags; // or'ing of all handler settings.
|
|
DWORD dwCombinedFlags; // or together user and handler.
|
|
|
|
// if not an uninstall determine the true machine state
|
|
// if Logon set for handler or user set or if handler
|
|
// wants an idle set.
|
|
// If Logoff set we register for Logoff.
|
|
|
|
RegGetCombinedUserRegFlags(&dwSyncMgrUsersRegisterFlags);
|
|
RegGetCombinedHandlerRegFlags(&dwSyncMgrHandlerRegisterFlags,&ftHandlerReg);
|
|
|
|
dwCombinedFlags = dwSyncMgrUsersRegisterFlags | dwSyncMgrHandlerRegisterFlags;
|
|
|
|
if ( (dwCombinedFlags & SYNCMGRREGISTERFLAG_CONNECT)
|
|
|| (dwSyncMgrHandlerRegisterFlags & SYNCMGRREGISTERFLAG_IDLE) )
|
|
{
|
|
fLogon = TRUE;
|
|
}
|
|
|
|
if ( (dwCombinedFlags & SYNCMGRREGISTERFLAG_PENDINGDISCONNECT) )
|
|
{
|
|
fLogoff = TRUE;
|
|
}
|
|
|
|
}
|
|
|
|
// update Registry entries for SENS to lookup
|
|
RegWriteEvents(fLogon,fLogoff);
|
|
|
|
#ifdef _SENS
|
|
|
|
|
|
// we were able to load ole automation so reg/unreg with the event system.
|
|
hr = CoCreateInstance(CLSID_CEventSystem,NULL,CLSCTX_SERVER,IID_IEventSystem,
|
|
(LPVOID *) &pEventSystem);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
IEventSubscription *pIEventSubscription;
|
|
WCHAR szGuid[GUID_SIZE+1];
|
|
BSTR bstrSubscriberID = NULL;
|
|
BSTR bstrPROGID_EventSubscription = NULL;
|
|
|
|
|
|
bstrPROGID_EventSubscription = SysAllocString(PROGID_EventSubscription);
|
|
|
|
StringFromGUID2(GUID_SENSSUBSCRIBER_SYNCMGRP,szGuid, ARRAYSIZE(szGuid));
|
|
bstrSubscriberID = SysAllocString(szGuid);
|
|
|
|
if (bstrSubscriberID && bstrPROGID_EventSubscription)
|
|
{
|
|
// register for RasConnect
|
|
hr = CoCreateInstance(
|
|
CLSID_CEventSubscription,
|
|
NULL,
|
|
CLSCTX_SERVER,
|
|
IID_IEventSubscription,
|
|
(LPVOID *) &pIEventSubscription
|
|
);
|
|
}
|
|
else
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
BSTR bstrPublisherID = NULL;
|
|
BSTR bstrSubscriptionID = NULL;
|
|
BSTR bstrSubscriptionName = NULL;
|
|
BSTR bstrSubscriberCLSID = NULL;
|
|
BSTR bstrEventID = NULL;
|
|
BSTR bstrEventClassID = NULL;
|
|
BSTR bstrIID = NULL;
|
|
|
|
// if there are any events, register with ens for messages.
|
|
if (fLogon)
|
|
{
|
|
|
|
StringFromGUID2(GUID_SENSLOGONSUBSCRIPTION_SYNCMGRP,szGuid, ARRAYSIZE(szGuid));
|
|
bstrSubscriptionID = SysAllocString(szGuid);
|
|
|
|
if (bstrSubscriptionID && SUCCEEDED(hr))
|
|
{
|
|
hr = pIEventSubscription->put_SubscriptionID(bstrSubscriptionID);
|
|
}
|
|
|
|
StringFromGUID2(CLSID_SyncMgrp,szGuid, ARRAYSIZE(szGuid));
|
|
bstrSubscriberCLSID = SysAllocString(szGuid);
|
|
|
|
if (bstrSubscriberCLSID && SUCCEEDED(hr))
|
|
{
|
|
hr = pIEventSubscription->put_SubscriberCLSID(bstrSubscriberCLSID);
|
|
}
|
|
|
|
StringFromGUID2(SENSGUID_PUBLISHER,szGuid, ARRAYSIZE(szGuid));
|
|
bstrPublisherID = SysAllocString(szGuid);
|
|
if (bstrPublisherID && SUCCEEDED(hr))
|
|
{
|
|
hr = pIEventSubscription->put_PublisherID(bstrPublisherID);
|
|
}
|
|
|
|
|
|
bstrSubscriptionName = SysAllocString(SZ_SYNCMGRNAME);
|
|
if (bstrSubscriptionName && SUCCEEDED(hr))
|
|
{
|
|
hr = pIEventSubscription->put_SubscriptionName(bstrSubscriptionName);
|
|
}
|
|
|
|
bstrEventID = SysAllocString(L"ConnectionMade");
|
|
if (bstrEventID && SUCCEEDED(hr))
|
|
{
|
|
hr = pIEventSubscription->put_MethodName(bstrEventID);
|
|
}
|
|
|
|
StringFromGUID2(SENSGUID_EVENTCLASS_NETWORK,szGuid,ARRAYSIZE(szGuid));
|
|
bstrEventClassID = SysAllocString(szGuid);
|
|
if (bstrEventClassID && SUCCEEDED(hr))
|
|
{
|
|
hr = pIEventSubscription->put_EventClassID(bstrEventClassID);
|
|
}
|
|
|
|
// set this up for roaming
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
// hr = pIEventSubscription->put_PerUser(TRUE); // don't register PerUser for Nw
|
|
}
|
|
|
|
StringFromGUID2(IID_ISensNetwork,szGuid,ARRAYSIZE(szGuid));
|
|
bstrIID = SysAllocString(szGuid);
|
|
if (bstrIID && SUCCEEDED(hr))
|
|
{
|
|
hr = pIEventSubscription->put_InterfaceID(bstrIID);
|
|
}
|
|
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = pEventSystem->Store(bstrPROGID_EventSubscription,pIEventSubscription);
|
|
}
|
|
|
|
if (bstrIID)
|
|
{
|
|
SysFreeString(bstrIID);
|
|
}
|
|
|
|
if (bstrPublisherID)
|
|
SysFreeString(bstrPublisherID);
|
|
|
|
if (bstrSubscriberCLSID)
|
|
SysFreeString(bstrSubscriberCLSID);
|
|
|
|
if (bstrEventClassID)
|
|
SysFreeString(bstrEventClassID);
|
|
|
|
if (bstrEventID)
|
|
SysFreeString(bstrEventID);
|
|
|
|
if (bstrSubscriptionID)
|
|
SysFreeString(bstrSubscriptionID);
|
|
|
|
if (bstrSubscriptionName)
|
|
SysFreeString(bstrSubscriptionName);
|
|
}
|
|
else // don't need to be registered, remove.
|
|
{
|
|
|
|
if (NOERROR == hr)
|
|
{
|
|
int errorIndex;
|
|
|
|
bstrSubscriptionID = SysAllocString(L"SubscriptionID={6295df30-35ee-11d1-8707-00C04FD93327}");
|
|
|
|
if (bstrSubscriptionID)
|
|
{
|
|
hr = pEventSystem->Remove(bstrPROGID_EventSubscription,bstrSubscriptionID /* QUERY */,&errorIndex);
|
|
SysFreeString(bstrSubscriptionID);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|
|
pIEventSubscription->Release();
|
|
|
|
}
|
|
|
|
if (bstrSubscriberID)
|
|
{
|
|
SysFreeString(bstrSubscriberID);
|
|
}
|
|
|
|
if (bstrPROGID_EventSubscription)
|
|
{
|
|
SysFreeString(bstrPROGID_EventSubscription);
|
|
}
|
|
|
|
pEventSystem->Release();
|
|
|
|
}
|
|
|
|
#endif // _SENS
|
|
|
|
cCritSect.Leave();
|
|
|
|
return hr;
|
|
}
|
|
|
|
// helper functions for handler registration
|
|
STDAPI_(BOOL) RegGetTimeStamp(HKEY hKey, FILETIME *pft)
|
|
{
|
|
FILETIME ft;
|
|
LONG lr;
|
|
DWORD cbSize = sizeof(ft);
|
|
|
|
Assert(pft);
|
|
|
|
lr = SHRegGetValue( hKey,NULL,
|
|
SZ_REGISTRATIONTIMESTAMPKEY,
|
|
SRRF_RT_REG_BINARY,
|
|
NULL,
|
|
(BYTE *)&ft,
|
|
&cbSize );
|
|
|
|
|
|
if ( lr == ERROR_SUCCESS )
|
|
{
|
|
Assert(cbSize == sizeof(FILETIME));
|
|
*pft = ft;
|
|
}
|
|
else
|
|
{
|
|
// set the filetime to way back when to
|
|
// any compares will just say older instead
|
|
// of having to check success code
|
|
(*pft).dwLowDateTime = 0;
|
|
(*pft).dwHighDateTime = 0;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
STDAPI_(BOOL) RegWriteTimeStamp(HKEY hkey)
|
|
{
|
|
SYSTEMTIME sysTime;
|
|
FILETIME ft;
|
|
LONG lr = -1;
|
|
|
|
GetSystemTime(&sysTime); // void can't check for errors
|
|
|
|
if (SystemTimeToFileTime(&sysTime,&ft) )
|
|
{
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
|
|
// write out the UpdateTime
|
|
lr = RegSetValueEx( hkey,
|
|
SZ_REGISTRATIONTIMESTAMPKEY,
|
|
NULL,
|
|
REG_BINARY,
|
|
(BYTE *)&ft,
|
|
sizeof(ft) );
|
|
|
|
CMutexRegistry.Leave();
|
|
|
|
}
|
|
return (ERROR_SUCCESS == lr) ? TRUE : FALSE;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: UpdateHandlerKeyInformation
|
|
//
|
|
// Synopsis: Updates the top-level handler key information
|
|
//
|
|
// Arguments:
|
|
//
|
|
// Returns: void
|
|
//
|
|
// Modifies: enumerates the handlers underneath the given key
|
|
// updating the registrationFlags which is an || or
|
|
// all registered handler flags and then updates the
|
|
// timestamp on this key
|
|
//
|
|
// History: 05-Nov-97 rogerg Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
void UpdateHandlerKeyInformation(HKEY hKeyHandler)
|
|
{
|
|
DWORD dwSyncMgrTopLevelRegisterFlags = 0;
|
|
DWORD dwIndex = 0;
|
|
TCHAR lpName[MAX_PATH];
|
|
|
|
while ( ERROR_SUCCESS == RegEnumKey(hKeyHandler,dwIndex,
|
|
lpName,ARRAYSIZE(lpName)) )
|
|
{
|
|
DWORD dwHandlerRegFlags;
|
|
DWORD cbDataSize = sizeof(dwHandlerRegFlags);
|
|
LONG lRet;
|
|
HKEY hKeyClsid;
|
|
|
|
lRet = RegOpenKeyEx( hKeyHandler,
|
|
lpName,
|
|
NULL,
|
|
KEY_READ,
|
|
&hKeyClsid );
|
|
|
|
if (ERROR_SUCCESS == lRet)
|
|
{
|
|
|
|
if (ERROR_SUCCESS == SHRegGetValue(hKeyClsid,NULL,SZ_REGISTRATIONFLAGSKEY,SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwHandlerRegFlags,
|
|
&cbDataSize) )
|
|
{
|
|
dwSyncMgrTopLevelRegisterFlags |= dwHandlerRegFlags;
|
|
}
|
|
|
|
RegCloseKey(hKeyClsid);
|
|
}
|
|
|
|
dwIndex++;
|
|
}
|
|
|
|
|
|
// not much we can do if RegFlags are messed up other than assert and mask out
|
|
Assert(dwSyncMgrTopLevelRegisterFlags <= SYNCMGRREGISTERFLAGS_MASK);
|
|
dwSyncMgrTopLevelRegisterFlags &= SYNCMGRREGISTERFLAGS_MASK;
|
|
|
|
// write out new flags even if errors occured. work thing that happens is
|
|
// we don't set up someones autosync automatically.
|
|
RegSetValueEx(hKeyHandler,SZ_REGISTRATIONFLAGSKEY,NULL, REG_DWORD,
|
|
(LPBYTE) &(dwSyncMgrTopLevelRegisterFlags), sizeof(dwSyncMgrTopLevelRegisterFlags));
|
|
|
|
RegWriteTimeStamp(hKeyHandler);
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: RegUpdateTopLevelKeys
|
|
//
|
|
// Synopsis: Looks at toplevel AutoSync,Idle, etc. keys and determines
|
|
// if they need to be updated and if so goes for it.
|
|
//
|
|
// Arguments:
|
|
//
|
|
// Returns: Appropriate status code
|
|
//
|
|
// Modifies: set pfFirstRegistration out param to true if this is
|
|
// the first handler that has registered so we can setup defaults.
|
|
//
|
|
// History: 24-Aug-98 rogerg Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDAPI_(void) RegUpdateTopLevelKeys()
|
|
{
|
|
HKEY hkey;
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
|
|
CMutexRegistry.Enter();
|
|
|
|
|
|
// update the AutoSync Key
|
|
hkey = RegGetSyncTypeKey(SYNCTYPE_AUTOSYNC,KEY_READ | KEY_WRITE,TRUE);
|
|
|
|
|
|
if (hkey)
|
|
{
|
|
DWORD dwUserLogonLogoff;
|
|
DWORD cbDataSize = sizeof(dwUserLogonLogoff);
|
|
|
|
// see if has a logon value and if it is either newly created or
|
|
// old format. Call Update to setthings up
|
|
if (ERROR_SUCCESS != SHRegGetValue(hkey,NULL,TEXT("Logon"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwUserLogonLogoff,
|
|
&cbDataSize) )
|
|
{
|
|
RegUpdateAutoSyncKeyValue(hkey,-1,-1);
|
|
}
|
|
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
|
|
// update the Idle Key
|
|
hkey = RegGetSyncTypeKey(SYNCTYPE_IDLE,KEY_READ | KEY_WRITE,TRUE);
|
|
|
|
if (hkey)
|
|
{
|
|
DWORD dwIdleEnabled;
|
|
DWORD cbDataSize = sizeof(dwIdleEnabled);
|
|
|
|
// see if has a Idle value and if it is either newly created or
|
|
// old format. Call Update to setthings up
|
|
if (ERROR_SUCCESS != SHRegGetValue(hkey,NULL,TEXT("IdleEnabled"),SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwIdleEnabled, &cbDataSize) )
|
|
{
|
|
RegUpdateIdleKeyValue(hkey,-1);
|
|
}
|
|
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
CMutexRegistry.Leave();
|
|
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: RegRegisterHandler
|
|
//
|
|
// Synopsis: Registers Handlers with SyncMgr.
|
|
//
|
|
// Arguments:
|
|
//
|
|
// Returns: Appropriate status code
|
|
//
|
|
// Modifies: set pfFirstRegistration out param to true if this is
|
|
// the first handler that has registered so we can setup defaults.
|
|
//
|
|
// History: 05-Nov-97 rogerg Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDAPI_(BOOL) RegRegisterHandler(REFCLSID rclsidHandler,
|
|
WCHAR const* pwszDescription,
|
|
DWORD dwSyncMgrRegisterFlags,
|
|
BOOL *pfFirstRegistration)
|
|
{
|
|
LONG lRet;
|
|
|
|
RegUpdateTopLevelKeys(); // make sure other top-level keys are up to date.
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
*pfFirstRegistration = FALSE;
|
|
HKEY hKeyHandler;
|
|
|
|
hKeyHandler = RegGetHandlerTopLevelKey(KEY_READ | KEY_WRITE);
|
|
|
|
if (NULL == hKeyHandler)
|
|
{
|
|
CMutexRegistry.Leave();
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// Check if this is the first handler being registerd
|
|
//
|
|
TCHAR szGuid[GUID_SIZE+1];
|
|
DWORD cchGuid = ARRAYSIZE(szGuid);
|
|
|
|
lRet = RegEnumKeyEx( hKeyHandler,
|
|
0,
|
|
szGuid,
|
|
&cchGuid,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL );
|
|
|
|
if ( lRet != ERROR_SUCCESS )
|
|
*pfFirstRegistration = TRUE;
|
|
|
|
//
|
|
// Convert guid and description to TCHAR
|
|
//
|
|
TCHAR *pszDesc;
|
|
BOOL fOk = FALSE;
|
|
|
|
StringFromGUID2( rclsidHandler, szGuid, ARRAYSIZE(szGuid));
|
|
pszDesc = (TCHAR *)pwszDescription;
|
|
|
|
|
|
// write out the registration flags. If fail go ahead
|
|
// and succed registration anyways.
|
|
if (hKeyHandler)
|
|
{
|
|
HKEY hKeyClsid;
|
|
|
|
hKeyClsid = RegGetHandlerKey(hKeyHandler,szGuid,KEY_WRITE | KEY_READ,TRUE);
|
|
|
|
if (hKeyClsid)
|
|
{
|
|
|
|
fOk = TRUE; // if make handle key say registered okay
|
|
|
|
if (pszDesc)
|
|
{
|
|
RegSetValueEx(hKeyClsid,NULL,NULL, REG_SZ,
|
|
(LPBYTE) pszDesc,
|
|
(lstrlen(pszDesc) +1)*sizeof(TCHAR));
|
|
}
|
|
|
|
RegSetValueEx(hKeyClsid,SZ_REGISTRATIONFLAGSKEY,NULL, REG_DWORD,
|
|
(LPBYTE) &(dwSyncMgrRegisterFlags), sizeof(dwSyncMgrRegisterFlags));
|
|
|
|
// update the TimeStamp on the handler clsid
|
|
|
|
RegWriteTimeStamp(hKeyClsid);
|
|
RegCloseKey( hKeyClsid );
|
|
|
|
// update the toplevel key
|
|
UpdateHandlerKeyInformation(hKeyHandler);
|
|
}
|
|
}
|
|
|
|
// update the user information.
|
|
RegSetUserDefaults();
|
|
RegRegisterForEvents(FALSE /* fUninstall */);
|
|
|
|
CMutexRegistry.Leave();
|
|
RegCloseKey(hKeyHandler);
|
|
|
|
return fOk;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: RegRegRemoveHandler
|
|
//
|
|
// Synopsis: UnRegisters Handlers with SyncMgr.
|
|
//
|
|
// Arguments:
|
|
//
|
|
// Returns: Appropriate status code
|
|
//
|
|
// Modifies: set pfAllHandlerUnRegistered out param to true if this is
|
|
// the last handler that needs to be unregistered before
|
|
// turning off our defaults..
|
|
//
|
|
// History: 05-Nov-97 rogerg Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDAPI_(BOOL) RegRegRemoveHandler(REFCLSID rclsidHandler)
|
|
{
|
|
HKEY hKeyHandler;
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
|
|
CMutexRegistry.Enter();
|
|
|
|
|
|
hKeyHandler = RegGetHandlerTopLevelKey(KEY_WRITE | KEY_READ);
|
|
|
|
if (NULL == hKeyHandler)
|
|
{
|
|
//
|
|
// Non-existent key, so nothing to remove
|
|
//
|
|
CMutexRegistry.Leave();
|
|
return TRUE;
|
|
}
|
|
|
|
TCHAR szGuid[GUID_SIZE+1];
|
|
|
|
StringFromGUID2( rclsidHandler, szGuid, ARRAYSIZE(szGuid) );
|
|
|
|
HKEY hKeyClsid;
|
|
|
|
hKeyClsid = RegGetHandlerKey(hKeyHandler,szGuid,KEY_WRITE | KEY_READ,FALSE);
|
|
|
|
if (hKeyClsid)
|
|
{
|
|
RegCloseKey( hKeyClsid );
|
|
RegDeleteKey( hKeyHandler, szGuid );
|
|
|
|
// update the toplevel key
|
|
UpdateHandlerKeyInformation(hKeyHandler);
|
|
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Non-existent key, so nothing to remove
|
|
//
|
|
}
|
|
|
|
|
|
RegRegisterForEvents(FALSE /* fUninstall */); // UPDATE EVENT REGISTRATION.
|
|
|
|
CMutexRegistry.Leave();
|
|
RegCloseKey(hKeyHandler);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: RegGetHandlerRegistrationInfo
|
|
//
|
|
// Synopsis: Gets Information of the specified handler.
|
|
//
|
|
// Arguments:
|
|
//
|
|
// Returns: Appropriate status code
|
|
//
|
|
// Modifies: pdwSyncMgrRegisterFlags
|
|
//
|
|
// History: 20-Aug-98 rogerg Created.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDAPI_(BOOL) RegGetHandlerRegistrationInfo(REFCLSID rclsidHandler,LPDWORD pdwSyncMgrRegisterFlags)
|
|
{
|
|
HKEY hKeyHandler;
|
|
|
|
*pdwSyncMgrRegisterFlags = 0;
|
|
|
|
|
|
hKeyHandler = RegGetHandlerTopLevelKey(KEY_READ);
|
|
|
|
if (NULL == hKeyHandler)
|
|
{
|
|
//
|
|
// Non-existent key
|
|
//
|
|
return FALSE;
|
|
}
|
|
TCHAR szGuid[GUID_SIZE+1];
|
|
|
|
StringFromGUID2( rclsidHandler, szGuid, ARRAYSIZE(szGuid));
|
|
|
|
HKEY hKeyClsid;
|
|
BOOL fResult = FALSE;
|
|
|
|
hKeyClsid = RegGetHandlerKey(hKeyHandler,szGuid,KEY_READ,FALSE);
|
|
|
|
if (hKeyClsid)
|
|
{
|
|
DWORD cbDataSize = sizeof(*pdwSyncMgrRegisterFlags);
|
|
LONG lRet;
|
|
|
|
lRet = SHRegGetValue(hKeyClsid,NULL,SZ_REGISTRATIONFLAGSKEY,SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) pdwSyncMgrRegisterFlags,
|
|
&cbDataSize);
|
|
RegCloseKey( hKeyClsid );
|
|
|
|
fResult = (ERROR_SUCCESS == lRet) ? TRUE : FALSE;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Non-existent key, so nothing to remove
|
|
//
|
|
|
|
}
|
|
|
|
RegCloseKey(hKeyHandler);
|
|
|
|
return fResult;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: RegSetUserDefaults
|
|
//
|
|
// Synopsis: Registers default values for auto and idle sync
|
|
//
|
|
// Setup based on Handler and UserPreferences
|
|
//
|
|
// History: 20-May-98 SitaramR Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDAPI_(void) RegSetUserDefaults()
|
|
{
|
|
HKEY hKeyUser = NULL;
|
|
FILETIME ftHandlerReg;
|
|
DWORD dwHandlerRegistrationFlags;
|
|
|
|
CMutex CMutexRegistry(NULL, FALSE,SZ_REGISTRYMUTEXNAME);
|
|
CMutexRegistry.Enter();
|
|
|
|
// get the combined handler registration toplevel flags and timeStamp
|
|
// to see if should bother enumerating the rest.
|
|
if (!RegGetCombinedHandlerRegFlags(&dwHandlerRegistrationFlags,&ftHandlerReg))
|
|
{
|
|
CMutexRegistry.Leave();
|
|
return;
|
|
}
|
|
|
|
if (0 != (dwHandlerRegistrationFlags &
|
|
(SYNCMGRREGISTERFLAG_CONNECT | SYNCMGRREGISTERFLAG_PENDINGDISCONNECT) ) )
|
|
{
|
|
|
|
// See if AutoSync key needs to be updated
|
|
hKeyUser = RegGetCurrentUserKey(SYNCTYPE_AUTOSYNC,KEY_WRITE | KEY_READ,TRUE);
|
|
|
|
if (hKeyUser)
|
|
{
|
|
FILETIME ftUserAutoSync;
|
|
|
|
// if got the User get the timestamp and see if it is older than the handlers
|
|
// If this is a new user filetime will be 0
|
|
|
|
RegGetTimeStamp(hKeyUser,&ftUserAutoSync);
|
|
|
|
if (CompareFileTime(&ftUserAutoSync,&ftHandlerReg) < 0)
|
|
{
|
|
DWORD dwHandlerChangedFlags;
|
|
|
|
// need to walk through handlers and update what we need to set based
|
|
// on each handlers timestamp since we don't want a handler that registered
|
|
// for idle to cause us to turn AutoSync back on and vis-a-versa
|
|
|
|
if (RegGetChangedHandlerFlags(&ftUserAutoSync,&dwHandlerChangedFlags))
|
|
{
|
|
BOOL fLogon = (dwHandlerChangedFlags & SYNCMGRREGISTERFLAG_CONNECT) ? TRUE : FALSE;
|
|
BOOL fLogoff = (dwHandlerChangedFlags & SYNCMGRREGISTERFLAG_PENDINGDISCONNECT) ? TRUE : FALSE;
|
|
|
|
RegSetAutoSyncDefaults(fLogon,fLogoff);
|
|
}
|
|
|
|
}
|
|
|
|
RegCloseKey(hKeyUser);
|
|
hKeyUser = NULL;
|
|
}
|
|
}
|
|
|
|
if (0 != (dwHandlerRegistrationFlags & SYNCMGRREGISTERFLAG_IDLE ) )
|
|
{
|
|
|
|
// now check for Idle same logic as above could probably combine
|
|
// into a function
|
|
// See if AutoSync key needs to be updated
|
|
hKeyUser = RegGetCurrentUserKey(SYNCTYPE_IDLE, KEY_WRITE | KEY_READ,TRUE);
|
|
|
|
|
|
if (hKeyUser)
|
|
{
|
|
FILETIME ftUserIdleSync;
|
|
|
|
// if got the User get the timestamp and see if it is older than the handlers
|
|
// If this is a new user filetime will be 0
|
|
|
|
RegGetTimeStamp(hKeyUser,&ftUserIdleSync);
|
|
|
|
if (CompareFileTime(&ftUserIdleSync,&ftHandlerReg) < 0)
|
|
{
|
|
DWORD dwHandlerChangedFlags;
|
|
|
|
// need to walk through handlers and update what we need to set based
|
|
// on each handlers timestamp since we don't want a handler that registered
|
|
// for AutoSync to cause us to turn Idle back on and vis-a-versa
|
|
|
|
if (RegGetChangedHandlerFlags(&ftUserIdleSync,&dwHandlerChangedFlags))
|
|
{
|
|
if (dwHandlerChangedFlags & SYNCMGRREGISTERFLAG_IDLE)
|
|
{
|
|
RegSetIdleSyncDefaults(TRUE);
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
RegCloseKey(hKeyUser);
|
|
hKeyUser = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
CMutexRegistry.Leave();
|
|
}
|
|
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: RegSetAutoSyncDefaults
|
|
//
|
|
// Synopsis: Registers default values for auto sync
|
|
//
|
|
// History: 20-May-98 SitaramR Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDAPI_(void) RegSetAutoSyncDefaults(BOOL fLogon,BOOL fLogoff)
|
|
{
|
|
CONNECTIONSETTINGS *pConnection = (LPCONNECTIONSETTINGS)
|
|
ALLOC(sizeof(*pConnection));
|
|
if ( pConnection == 0 )
|
|
return;
|
|
INT iRet = LoadString(g_hmodThisDll,
|
|
IDS_LAN_CONNECTION,
|
|
pConnection->pszConnectionName,
|
|
ARRAYSIZE(pConnection->pszConnectionName) );
|
|
Assert( iRet != 0 );
|
|
|
|
// -1 values are ignored by RegSetAutoSyncSettings.
|
|
// if not turning on leave the User Preferences alone,
|
|
pConnection->dwConnType = 0;
|
|
pConnection->dwLogon = fLogon ? TRUE : -1;
|
|
pConnection->dwLogoff = fLogoff ? TRUE : -1;
|
|
pConnection->dwPromptMeFirst = -1;
|
|
pConnection->dwMakeConnection = -1;
|
|
pConnection->dwIdleEnabled = -1;
|
|
|
|
// since this bases settings on what is already set no need to
|
|
// do a cleanreg or update the machine state
|
|
RegSetAutoSyncSettings(pConnection, 1, 0,
|
|
FALSE /* fCleanReg */,
|
|
FALSE /* fSetMachineState */,
|
|
FALSE /* fPerUser */);
|
|
|
|
FREE(pConnection);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: RegSetUserAutoSyncDefaults
|
|
//
|
|
// Synopsis: Registers user default values for auto sync
|
|
//
|
|
// History: 39-March-99 rogerg Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDAPI RegSetUserAutoSyncDefaults(DWORD dwSyncMgrRegisterMask,
|
|
DWORD dwSyncMgrRegisterFlags)
|
|
{
|
|
|
|
// if not changing either logon or logoff just return
|
|
if (!(dwSyncMgrRegisterMask & SYNCMGRREGISTERFLAG_CONNECT)
|
|
&& !(dwSyncMgrRegisterMask & SYNCMGRREGISTERFLAG_PENDINGDISCONNECT) )
|
|
{
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
CONNECTIONSETTINGS *pConnection = (LPCONNECTIONSETTINGS)
|
|
ALLOC(sizeof(*pConnection));
|
|
if ( pConnection == 0 )
|
|
return E_OUTOFMEMORY;
|
|
INT iRet = LoadString(g_hmodThisDll,
|
|
IDS_LAN_CONNECTION,
|
|
pConnection->pszConnectionName,
|
|
ARRAYSIZE(pConnection->pszConnectionName) );
|
|
Assert( iRet != 0 );
|
|
|
|
// -1 values are ignored by RegSetAutoSyncSettings.
|
|
// if not turning on leave the User Preferences alone,
|
|
pConnection->dwConnType = 0;
|
|
pConnection->dwLogon = -1;
|
|
pConnection->dwLogoff = -1;
|
|
pConnection->dwPromptMeFirst = -1;
|
|
pConnection->dwMakeConnection = -1;
|
|
pConnection->dwIdleEnabled = -1;
|
|
|
|
if (dwSyncMgrRegisterMask & SYNCMGRREGISTERFLAG_CONNECT)
|
|
{
|
|
pConnection->dwLogon = (dwSyncMgrRegisterFlags & SYNCMGRREGISTERFLAG_CONNECT)
|
|
? TRUE : FALSE;
|
|
}
|
|
|
|
if (dwSyncMgrRegisterMask & SYNCMGRREGISTERFLAG_PENDINGDISCONNECT)
|
|
{
|
|
pConnection->dwLogoff = (dwSyncMgrRegisterFlags & SYNCMGRREGISTERFLAG_PENDINGDISCONNECT)
|
|
? TRUE : FALSE;
|
|
}
|
|
|
|
// since this bases settings on what is already set no need to
|
|
// do a cleanreg or update the machine state
|
|
RegSetAutoSyncSettings(pConnection, 1, 0,
|
|
FALSE /* fCleanReg */,
|
|
TRUE /* fSetMachineState */,
|
|
TRUE /* fPerUser */);
|
|
|
|
FREE(pConnection);
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: RegSetIdleSyncDefaults
|
|
//
|
|
// Synopsis: Registers default values for idle sync
|
|
//
|
|
// History: 20-May-98 SitaramR Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDAPI_(void) RegSetIdleSyncDefaults(BOOL fIdle)
|
|
{
|
|
|
|
Assert(fIdle); // for now this should only be called when true;
|
|
|
|
if (!fIdle)
|
|
{
|
|
return;
|
|
}
|
|
|
|
CONNECTIONSETTINGS *pConnection = (LPCONNECTIONSETTINGS)
|
|
ALLOC(sizeof(*pConnection));
|
|
if ( pConnection == 0 )
|
|
return;
|
|
INT iRet = LoadString(g_hmodThisDll,
|
|
IDS_LAN_CONNECTION,
|
|
pConnection->pszConnectionName,
|
|
ARRAYSIZE(pConnection->pszConnectionName) );
|
|
Assert( iRet != 0 );
|
|
|
|
pConnection->dwConnType = 0;
|
|
pConnection->dwLogon = -1;
|
|
pConnection->dwLogoff = -1;
|
|
pConnection->dwPromptMeFirst = -1;
|
|
pConnection->dwMakeConnection = -1;
|
|
pConnection->dwIdleEnabled = TRUE;
|
|
|
|
// set all userLevel items to -1 so user gets the defaults if new
|
|
// but keep their settings if have already tweaked them.
|
|
pConnection->ulIdleRetryMinutes = -1;
|
|
pConnection->ulDelayIdleShutDownTime = -1;
|
|
pConnection->dwRepeatSynchronization = -1;
|
|
pConnection->ulIdleWaitMinutes = -1;
|
|
pConnection->dwRunOnBatteries = -1;
|
|
|
|
RegSetIdleSyncSettings(pConnection, 1, 0,
|
|
FALSE /* fCleanReg */,
|
|
FALSE /* fPerUser */);
|
|
|
|
FREE(pConnection);
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: RegSetIdleSyncDefaults
|
|
//
|
|
// Synopsis: Registers default values for idle sync
|
|
//
|
|
// History: 30-March-99 ROGERG Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDAPI RegSetUserIdleSyncDefaults(DWORD dwSyncMgrRegisterMask,
|
|
DWORD dwSyncMgrRegisterFlags)
|
|
{
|
|
|
|
// RegSetIdleSyncSettings doesn't handle idle enabled of -1 so only
|
|
// call if Idle actually is set in the flags, if not just return
|
|
|
|
if (!(dwSyncMgrRegisterMask & SYNCMGRREGISTERFLAG_IDLE))
|
|
{
|
|
return NOERROR;
|
|
}
|
|
|
|
CONNECTIONSETTINGS *pConnection = (LPCONNECTIONSETTINGS)
|
|
ALLOC(sizeof(*pConnection));
|
|
if ( pConnection == 0 )
|
|
return E_OUTOFMEMORY;
|
|
INT iRet = LoadString(g_hmodThisDll,
|
|
IDS_LAN_CONNECTION,
|
|
pConnection->pszConnectionName,
|
|
ARRAYSIZE(pConnection->pszConnectionName) );
|
|
Assert( iRet != 0 );
|
|
|
|
pConnection->dwConnType = 0;
|
|
pConnection->dwLogon = -1;
|
|
pConnection->dwLogoff = -1;
|
|
pConnection->dwPromptMeFirst = -1;
|
|
pConnection->dwMakeConnection = -1;
|
|
|
|
pConnection->dwIdleEnabled = (SYNCMGRREGISTERFLAG_IDLE & dwSyncMgrRegisterFlags)
|
|
? TRUE : FALSE;
|
|
|
|
// set all userLevel items to -1 so user gets the defaults if new
|
|
// but keep their settings if have already tweaked them.
|
|
pConnection->ulIdleRetryMinutes = -1;
|
|
pConnection->ulDelayIdleShutDownTime = -1;
|
|
pConnection->dwRepeatSynchronization = -1;
|
|
pConnection->ulIdleWaitMinutes = -1;
|
|
pConnection->dwRunOnBatteries = -1;
|
|
|
|
RegSetIdleSyncSettings(pConnection, 1, 0,
|
|
FALSE /* fCleanReg */,
|
|
TRUE /* fPerUser */);
|
|
|
|
|
|
FREE(pConnection);
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: RegGetUserRegisterFlags
|
|
//
|
|
// Synopsis: returns current registration flags for the User.
|
|
//
|
|
// History: 30-March-99 ROGERG Created
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
STDAPI RegGetUserRegisterFlags(LPDWORD pdwSyncMgrRegisterFlags)
|
|
{
|
|
CONNECTIONSETTINGS connectSettings;
|
|
|
|
*pdwSyncMgrRegisterFlags = 0;
|
|
|
|
INT iRet = LoadString(g_hmodThisDll,
|
|
IDS_LAN_CONNECTION,
|
|
connectSettings.pszConnectionName,
|
|
ARRAYSIZE(connectSettings.pszConnectionName) );
|
|
if (0 == iRet)
|
|
{
|
|
Assert( iRet != 0 );
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
RegGetSyncSettings(SYNCTYPE_AUTOSYNC,&connectSettings);
|
|
|
|
if (connectSettings.dwLogon)
|
|
{
|
|
*pdwSyncMgrRegisterFlags |= (SYNCMGRREGISTERFLAG_CONNECT);
|
|
}
|
|
|
|
if (connectSettings.dwLogoff)
|
|
{
|
|
*pdwSyncMgrRegisterFlags |= (SYNCMGRREGISTERFLAG_PENDINGDISCONNECT);
|
|
}
|
|
|
|
|
|
RegGetSyncSettings(SYNCTYPE_IDLE,&connectSettings);
|
|
|
|
if (connectSettings.dwIdleEnabled)
|
|
{
|
|
*pdwSyncMgrRegisterFlags |= (SYNCMGRREGISTERFLAG_IDLE);
|
|
}
|
|
|
|
|
|
|
|
return NOERROR;
|
|
}
|
|
|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
|
|
|
|
Function: BOOL RegSchedHandlerItemsChecked(TCHAR *pszHandlerName,
|
|
TCHAR *pszConnectionName,
|
|
TCHAR *pszScheduleName)
|
|
|
|
Summary: Determine if any items are checked on this handler for this schedule
|
|
|
|
Returns: Returns TRUE if one or more are checked, FALSE otherwise
|
|
|
|
------------------------------------------------------------------------F-F*/
|
|
BOOL RegSchedHandlerItemsChecked(TCHAR *pszHandlerName,
|
|
TCHAR *pszConnectionName,
|
|
TCHAR *pszScheduleName)
|
|
{
|
|
SCODE sc;
|
|
HKEY hKeyUser,
|
|
hkeySchedName,
|
|
hkeyConnection,
|
|
hkeyHandler,
|
|
hkeyItem;
|
|
DWORD cbName = MAX_PATH,
|
|
dwIndex = 0,
|
|
dwCheckState = 0;
|
|
|
|
BOOL fItemsChecked = FALSE;
|
|
TCHAR lpName[MAX_PATH + 1];
|
|
|
|
hKeyUser = RegGetCurrentUserKey(SYNCTYPE_SCHEDULED,KEY_READ,FALSE);
|
|
|
|
if (NULL == hKeyUser)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
smChkTo(EH_Err3,RegOpenKeyEx (hKeyUser,
|
|
pszScheduleName,0,KEY_READ,
|
|
&hkeySchedName));
|
|
|
|
smChkTo(EH_Err4,RegOpenKeyEx (hkeySchedName,
|
|
pszConnectionName,
|
|
0,KEY_READ,
|
|
&hkeyConnection));
|
|
|
|
smChkTo(EH_Err5,RegOpenKeyEx (hkeyConnection,
|
|
pszHandlerName,
|
|
0,KEY_READ,
|
|
&hkeyHandler));
|
|
// need to enum handler items.
|
|
while ( ERROR_SUCCESS == RegEnumKey(hkeyHandler,dwIndex,
|
|
lpName,cbName) )
|
|
{
|
|
LONG lRet;
|
|
|
|
lRet = RegOpenKeyEx( hkeyHandler,
|
|
lpName,
|
|
NULL,
|
|
KEY_READ,
|
|
&hkeyItem);
|
|
|
|
if (ERROR_SUCCESS == lRet)
|
|
{
|
|
DWORD cbDataSize = sizeof(dwCheckState);
|
|
SHRegGetValue(hkeyItem,NULL,TEXT("CheckState"), SRRF_RT_REG_DWORD, NULL,
|
|
(LPBYTE) &dwCheckState, &cbDataSize);
|
|
|
|
RegCloseKey(hkeyItem);
|
|
|
|
}
|
|
else
|
|
{
|
|
goto EH_Err5;
|
|
}
|
|
|
|
if (dwCheckState)
|
|
{
|
|
fItemsChecked = TRUE;
|
|
break;
|
|
}
|
|
dwIndex++;
|
|
}
|
|
|
|
RegCloseKey(hkeyHandler);
|
|
RegCloseKey(hkeyConnection);
|
|
RegCloseKey(hkeySchedName);
|
|
RegCloseKey(hKeyUser);
|
|
return fItemsChecked;
|
|
|
|
EH_Err5:
|
|
RegCloseKey(hkeyConnection);
|
|
EH_Err4:
|
|
RegCloseKey(hkeySchedName);
|
|
EH_Err3:
|
|
RegCloseKey(hKeyUser);
|
|
|
|
return fItemsChecked;
|
|
}
|
|
|