Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

733 lines
24 KiB

//*************************************************************
//
// Copyright (c) Microsoft Corporation 1998
// All rights reserved
//
// fdeploy.cxx
//
//*************************************************************
#include "fdeploy.hxx"
#include "rsopdbg.h"
#define ABORT_IF_NECESSARY if (*pbAbort) \
{ \
Status = ERROR_REQUEST_ABORTED; \
goto ProcessGPOCleanup; \
}
#define FLUSH_AND_ABORT_IF_NECESSARY if (*pbAbort) \
{ \
Status = ERROR_REQUEST_ABORTED; \
goto ProcessGPOFlush; \
}
CDebug dbgRsop( L"Software\\Microsoft\\Windows NT\\CurrentVersion\\winlogon",
L"RsopDebugLevel",
L"gpdas.log",
L"gpdas.bak",
TRUE );
//status callback function
PFNSTATUSMESSAGECALLBACK gpStatusCallback;
//saved per user per machine settings
CSavedSettings gSavedSettings[(int)EndRedirectable];
//if CSC is enabled or not
BOOL g_bCSCEnabled;
// Used for LoadString.
HINSTANCE ghDllInstance = 0;
HINSTANCE ghFileDeployment = 0;
WCHAR gwszStatus[12];
WCHAR gwszNumber[20];
// User info.
CUsrInfo gUserInfo;
const WCHAR * gwszUserName = NULL;
//+--------------------------------------------------------------------------
//
// Function: ReinitGlobals
//
// Synopsis: Reinitializes the global variables that should not carry
// over to the next run of folder redirection.
//
// Arguments: none.
//
// Returns: nothing.
//
// History: 12/17/2000 RahulTh created
//
// Notes: static members of classes and other global variables are
// initialized only when the dll is loaded. So if this dll
// stays loaded across logons, then the fact that the globals
// have been initialized based on a previous logon can cause
// problems. Therefore, this function is used to reinitialize
// the globals.
//
//---------------------------------------------------------------------------
void ReinitGlobals (void)
{
DWORD i;
// First reset the static members of various classes.
CSavedSettings::ResetStaticMembers();
// Now reset members of various global objects.
gUserInfo.ResetMembers();
for (i = 0; i < (DWORD) EndRedirectable; i++)
{
gSavedSettings[i].ResetMembers();
gPolicyResultant[i].ResetMembers();
gAddedPolicyResultant[i].ResetMembers();
gDeletedPolicyResultant[i].ResetMembers();
}
return;
}
extern "C" DWORD WINAPI
ProcessGroupPolicyEx (
DWORD dwFlags,
HANDLE hUserToken,
HKEY hKeyRoot,
PGROUP_POLICY_OBJECT pDeletedGPOList,
PGROUP_POLICY_OBJECT pChangedGPOList,
ASYNCCOMPLETIONHANDLE pHandle,
BOOL* pbAbort,
PFNSTATUSMESSAGECALLBACK pStatusCallback,
IN IWbemServices *pWbemServices,
HRESULT *phrRsopStatus )
{
// Reinitialize all globals that should not get carried over from
// a previous run of folder redirection. This is necessary just in
// case this dll is not unloaded by userenv after each run. Also, we should
// do this before any other processing is done to ensure correct behavior.
ReinitGlobals();
*phrRsopStatus = S_OK;
CRsopContext DiagnosticModeContext( pWbemServices, phrRsopStatus, FDEPLOYEXTENSIONGUID );
return ProcessGroupPolicyInternal (
dwFlags,
hUserToken,
hKeyRoot,
pDeletedGPOList,
pChangedGPOList,
pHandle,
pbAbort,
pStatusCallback,
&DiagnosticModeContext );
}
extern "C" DWORD WINAPI
GenerateGroupPolicy (
IN DWORD dwFlags,
IN BOOL *pbAbort,
IN WCHAR *pwszSite,
IN PRSOP_TARGET pComputerTarget,
IN PRSOP_TARGET pUserTarget )
{
DWORD Status;
// Reinitialize all globals that should not get carried over from
// a previous run of folder redirection. This is necessary just in
// case this dll is not unloaded by userenv after each run. Also, we should
// do this before any other processing is done to ensure correct behavior.
ReinitGlobals();
CRsopContext PlanningModeContext( pUserTarget, FDEPLOYEXTENSIONGUID );
Status = ERROR_SUCCESS;
//
// There is no machine policy, only user --
// process only user policy
//
if ( pUserTarget && pUserTarget->pGPOList )
{
gUserInfo.SetPlanningModeContext( &PlanningModeContext );
Status = ProcessGroupPolicyInternal(
dwFlags,
NULL,
NULL,
NULL,
pUserTarget->pGPOList,
NULL,
pbAbort,
NULL,
&PlanningModeContext);
}
return Status;
}
DWORD ProcessGroupPolicyInternal (
DWORD dwFlags,
HANDLE hUserToken,
HKEY hKeyRoot,
PGROUP_POLICY_OBJECT pDeletedGPOList,
PGROUP_POLICY_OBJECT pChangedGPOList,
ASYNCCOMPLETIONHANDLE pHandle,
BOOL* pbAbort,
PFNSTATUSMESSAGECALLBACK pStatusCallback,
CRsopContext* pRsopContext )
{
BOOL bStatus;
DWORD Status = ERROR_SUCCESS;
DWORD RedirStatus;
CFileDB CurrentDB;
DWORD i;
PGROUP_POLICY_OBJECT pCurrGPO = NULL;
HANDLE hDupToken = NULL;
BOOL fUpdateMyPicsLinks = FALSE;
BOOL fPlanningMode;
BOOL fWriteRsopLog = FALSE;
BOOL bForcedRefresh = FALSE;
gpStatusCallback = pStatusCallback;
fPlanningMode = pRsopContext->IsPlanningModeEnabled();
//
// Even though this extension has indicated its preference for not
// handling machine policies, the admin. might still override these
// preferences through policy. Since, this extension is not designed to
// handled this situation, we need to explicitly check for these cases and
// quit at this point.
//
if (dwFlags & GPO_INFO_FLAG_MACHINE)
{
DebugMsg((DM_VERBOSE, IDS_INVALID_FLAGS));
Status = ERROR_INVALID_FLAGS;
goto ProcessGPOCleanup;
}
//some basic initializations first.
InitDebugSupport();
if ( dwFlags & GPO_INFO_FLAG_VERBOSE )
gDebugLevel |= DL_VERBOSE | DL_EVENTLOG;
gpEvents = new CEvents();
if (!gpEvents)
{
DebugMsg((DM_VERBOSE, IDS_INIT_FAILED));
Status = ERROR_OUTOFMEMORY;
goto ProcessGPOCleanup;
}
gpEvents->Init();
gpEvents->Reference();
DebugMsg((DM_VERBOSE, IDS_PROCESSGPO));
DebugMsg((DM_VERBOSE, IDS_GPO_FLAGS, dwFlags));
ConditionalBreakIntoDebugger();
if ( ! fPlanningMode )
{
bStatus = DuplicateToken (hUserToken,
SecurityImpersonation,
&hDupToken);
if (!bStatus)
{
Status = GetLastError();
gpEvents->Report (EVENT_FDEPLOY_INIT_FAILED, 0);
goto ProcessGPOCleanup;
}
//impersonate the logged on user,
bStatus = ImpersonateLoggedOnUser( hDupToken );
//bail out if impersonation fails
if (!bStatus)
{
gpEvents->Report (EVENT_FDEPLOY_INIT_FAILED, 0);
Status = GetLastError();
goto ProcessGPOCleanup;
}
g_bCSCEnabled = CSCIsCSCEnabled ();
//try to get set ownership privileges. These will be required
//when we copy over ownership information
GetSetOwnerPrivileges (hDupToken);
//get the user name -- this is used for tracking name changes.
gwszUserName = gUserInfo.GetUserName (Status);
if (ERROR_SUCCESS != Status)
{
DebugMsg ((DM_VERBOSE, IDS_GETNAME_FAILED, Status));
gpEvents->Report (EVENT_FDEPLOY_INIT_FAILED, 0);
goto ProcessGPOCleanup;
}
}
//load the localized folder names and relative paths. (also see
//notes before the function LoadLocalizedNames()
Status = LoadLocalizedFolderNames ();
if (ERROR_SUCCESS != Status)
{
gpEvents->Report (EVENT_FDEPLOY_INIT_FAILED, 0);
goto ProcessGPOCleanup;
}
//now initialize those values for the CFileDB object that are going to
//be the same across all the policies
if (ERROR_SUCCESS != (Status = CurrentDB.Initialize(hDupToken, hKeyRoot, pRsopContext )))
{
gpEvents->Report (EVENT_FDEPLOY_INIT_FAILED, 0);
goto ProcessGPOCleanup;
}
if ( ! fPlanningMode )
{
//now load the per user per machine settings saved during the last logon
for (i = 0; i < (DWORD) EndRedirectable; i++)
{
Status = gSavedSettings[i].Load (&CurrentDB);
if (ERROR_SUCCESS != Status)
{
gpEvents->Report (EVENT_FDEPLOY_INIT_FAILED, 0);
goto ProcessGPOCleanup;
}
}
}
//now if the GPO_NOCHANGES flag has been specified, make sure that it is
//okay not to do any processing
bStatus = TRUE;
if ( (dwFlags & GPO_INFO_FLAG_NOCHANGES) &&
!fPlanningMode &&
!( dwFlags & GPO_INFO_FLAG_LOGRSOP_TRANSITION ) )
{
for (bStatus = FALSE, i = 0;
i < (DWORD)EndRedirectable && (!bStatus);
i++)
{
bStatus = fPlanningMode ? TRUE : gSavedSettings[i].NeedsProcessing();
}
}
if (!bStatus) //we are in good shape. No processing is required.
{
Status = ERROR_SUCCESS;
DebugMsg ((DM_VERBOSE, IDS_NOCHANGES));
goto ProcessGPOCleanup;
}
else
{
if ((dwFlags & GPO_INFO_FLAG_BACKGROUND) ||
(dwFlags & GPO_INFO_FLAG_ASYNC_FOREGROUND))
{
//
// Log an event only in the async. foreground case. In all other
// cases just output a debug message. Note: The Background flag
// will be set even in the async. foreground case. So we must
// explicitly make this check against the async. foreground
// flag.
//
if (dwFlags & GPO_INFO_FLAG_ASYNC_FOREGROUND)
{
gpEvents->Report (EVENT_FDEPLOY_POLICY_DELAYED, 0);
}
else
{
DebugMsg ((DM_VERBOSE, IDS_POLICY_DELAYED));
}
Status = ERROR_SYNC_FOREGROUND_REFRESH_REQUIRED;
goto ProcessGPOCleanup;
}
if ( ! fPlanningMode &&
(dwFlags & GPO_INFO_FLAG_NOCHANGES))
{
// a user name or homedir change has occured and no gpo changes occurred,
// so in order to perform RSoP logging, we will need to get our own
// RSoP namespace since the GP engine does not give us the namespace
// if no changes occurred -- we note this below
bForcedRefresh = TRUE;
fWriteRsopLog = TRUE;
}
}
//
// If we have changes or we are in planning mode, enable logging
//
if ( pChangedGPOList || pDeletedGPOList || fPlanningMode )
{
fWriteRsopLog = TRUE;
}
if ( fWriteRsopLog )
{
//
// If RSoP logging should occur in logging mode, initialize
// the rsop context -- this is not necessary in planning mode
//
if ( pRsopContext->IsDiagnosticModeEnabled() )
{
PSID pUserSid;
pUserSid = gpEvents->UserSid();
if ( pUserSid )
{
// This call requires elevated privileges to succeed.
RevertToSelf();
(void) pRsopContext->InitializeContext( pUserSid );
// Re-impersonate the logged on user,
bStatus = ImpersonateLoggedOnUser( hDupToken );
// Bail out if impersonation fails
if (!bStatus)
{
gpEvents->Report (EVENT_FDEPLOY_INIT_FAILED, 0);
Status = GetLastError();
goto ProcessGPOCleanup;
}
}
else
{
pRsopContext->DisableRsop( ERROR_OUTOFMEMORY );
}
}
CurrentDB.InitRsop( pRsopContext, bForcedRefresh );
}
//first process any user name changes.
for (i = 0; i < (DWORD) EndRedirectable; i++)
{
Status = gSavedSettings[i].HandleUserNameChange(
&CurrentDB,
&(gPolicyResultant[i]));
if (ERROR_SUCCESS != Status)
{
// Failure events will be reported while processing the name change
goto ProcessGPOCleanup;
}
}
//first process the deleted GPOs
//we need to do this in reverse, because removed policies are sent
//in reverse, that is the closest policy is sent last. So if you have
//two policies that redirect the same folder and both of them are removed
//simultaneously, then, the code gets the wrong value for the redirection
//destination of the resultant set of removed policies and therefore
//assumes that someone else must have modified it and therefore leaves it
//alone
if (pDeletedGPOList)
{
//go all the way to the end
for (pCurrGPO = pDeletedGPOList; pCurrGPO->pNext; pCurrGPO = pCurrGPO->pNext)
;
//now pCurrGPO points to the last policy in the removed list, so go at
//it in the reverse order.
for (; pCurrGPO; pCurrGPO = pCurrGPO->pPrev)
{
DebugMsg((DM_VERBOSE, IDS_GPO_NAME, pCurrGPO->szGPOName));
DebugMsg((DM_VERBOSE, IDS_GPO_FILESYSPATH, pCurrGPO->lpFileSysPath));
DebugMsg((DM_VERBOSE, IDS_GPO_DSPATH, pCurrGPO->lpDSPath));
DebugMsg((DM_VERBOSE, IDS_GPO_DISPLAYNAME, pCurrGPO->lpDisplayName));
//if we are unable to process even a single policy, we must abort
//immediately otherwise we may end up with an incorrect resultant
//policy.
if (ERROR_SUCCESS != (Status = CurrentDB.Process (pCurrGPO, TRUE)))
goto ProcessGPOCleanup;
ABORT_IF_NECESSARY
}
}
//update the descendants
gDeletedPolicyResultant[(int) MyPics].UpdateDescendant ();
//now process the other GPOs
for (pCurrGPO = pChangedGPOList; pCurrGPO; pCurrGPO = pCurrGPO->pNext)
{
DebugMsg((DM_VERBOSE, IDS_GPO_NAME, pCurrGPO->szGPOName));
DebugMsg((DM_VERBOSE, IDS_GPO_FILESYSPATH, pCurrGPO->lpFileSysPath));
DebugMsg((DM_VERBOSE, IDS_GPO_DSPATH, pCurrGPO->lpDSPath));
DebugMsg((DM_VERBOSE, IDS_GPO_DISPLAYNAME, pCurrGPO->lpDisplayName));
//if we are unable to process even a single policy, we must abort
//immediately otherwise we may end up with an incorrect resultant
//policy.
if (ERROR_SUCCESS != (Status = CurrentDB.Process (pCurrGPO, FALSE)))
{
goto ProcessGPOCleanup;
}
ABORT_IF_NECESSARY
}
if ( fPlanningMode )
{
goto ProcessGPOCleanup;
}
//now update the My Pics data. UpdateDescendant will derive the settings for
//My Pics from My Docs if it is set to derive its settings from My Docs.
gAddedPolicyResultant[(int) MyPics].UpdateDescendant ();
//now merge the deleted policy resultant and added policy resultants
for (i = 0; i < (DWORD) EndRedirectable; i++)
{
gPolicyResultant[i] = gDeletedPolicyResultant[i];
gPolicyResultant[i] = gAddedPolicyResultant[i];
//check to see if any group membership change has caused a policy to
//be effectively removed for this user.
gPolicyResultant[i].ComputeEffectivePolicyRemoval (pDeletedGPOList,
pChangedGPOList,
&CurrentDB);
}
//do the final redirection
//we ignore errors that might occur in redirection
//so that redirection of other folders is not hampered.
//however, if there is a failure, we save that information
//so that we can inform the group policy engine
if (ERROR_SUCCESS == Status)
{
for (int i = 0; i < (int)EndRedirectable; i++)
{
RedirStatus= gPolicyResultant[i].Redirect(hDupToken, hKeyRoot,
&CurrentDB);
if ((ERROR_SUCCESS != RedirStatus) && (ERROR_SUCCESS == Status))
Status = RedirStatus;
FLUSH_AND_ABORT_IF_NECESSARY //abort if necessary, but first, flush the shell's special folder cache
///as we may have redirected some folders already.
}
//update shell links to MyPics within MyDocuments if policy specified
//the location of at least one of MyDocs and MyPics and succeeded in
//redirection. For additional details see comments at the beginning of
//the function UpdateMyPicsShellLinks.
if (
(
(!(gPolicyResultant[(int)MyDocs].GetFlags() & REDIR_DONT_CARE)) &&
(ERROR_SUCCESS == gPolicyResultant[(int)MyDocs].GetRedirStatus())
)
||
(
(!(gPolicyResultant[(int)MyPics].GetFlags() & REDIR_DONT_CARE)) &&
(ERROR_SUCCESS == gPolicyResultant[(int)MyPics].GetRedirStatus())
)
)
{
fUpdateMyPicsLinks = TRUE;
//note:we do not invoke the shell link update function here since
//we need to flush the shell special folder cache or we may not
//get the true current location of MyDocs or MyPics. Therefore,
//the function is actually invoked below.
}
}
else
{
DebugMsg((DM_VERBOSE, IDS_PROCESSREDIRECTS, Status));
}
//
// Do not try to update the link (shell shortcut) in planning mode since
// planning mode takes no real action, just records results
//
if ( fPlanningMode )
{
fUpdateMyPicsLinks = FALSE;
}
//flush the shell's special folder cache. we may have successfully redirected
//some folders by the time we reach here. So it is always a good idea to let
//the shell know about it.
ProcessGPOFlush:
if (fUpdateMyPicsLinks)
UpdateMyPicsShellLinks(hDupToken,
gPolicyResultant[(int)MyPics].GetLocalizedName());
ProcessGPOCleanup: //don't leave any turds behind.
if ( (ERROR_SUCCESS == Status) && !fPlanningMode )
{
//we have successfully applied all the policies, so remove any cached
//ini files for removed policies.
//any errors in deletion are ignored.
DeleteCachedConfigFiles (pDeletedGPOList, &CurrentDB);
}
//we are done, so we stop running as the user
if ( ! fPlanningMode )
{
RevertToSelf();
}
// In logging (aka diagnostic) mode, we need to ensure that
// we reset the saved namespace before logging so that in the
// no changes case where we need to log ( i.e. username / homedir change ),
// we only log rsop data if RSoP was enabled at the last change
if ( fWriteRsopLog )
{
(void) pRsopContext->DeleteSavedNameSpace();
if ( pRsopContext->IsRsopEnabled() )
{
HRESULT hrLog;
hrLog = CurrentDB.WriteRsopLog();
if ( SUCCEEDED( hrLog ) )
{
(void) pRsopContext->SaveNameSpace();
}
}
}
if ( ! fPlanningMode )
{
if (hDupToken)
CloseHandle (hDupToken);
}
if (gpEvents)
gpEvents->Release();
//restore the status message
DisplayStatusMessage (IDS_DEFAULT_CALLBACK);
// CPolicyDatabase::FreeDatabase();
return Status;
}
extern "C" BOOL WINAPI DllMain(HINSTANCE hInstance, DWORD dwReason, LPVOID lpReserved)
{
switch (dwReason)
{
case DLL_PROCESS_ATTACH :
ghDllInstance = hInstance;
DisableThreadLibraryCalls(hInstance);
break;
case DLL_PROCESS_DETACH :
break;
}
return TRUE;
}
/////////////////////////////////////////////////////////////////////////////
// DllRegisterServer - Adds entries to the system registry
STDAPI DllRegisterServer(void)
{
DWORD dwDisp;
LONG lResult;
HKEY hKey;
TCHAR EventFile[] = TEXT("%SystemRoot%\\System32\\fdeploy.dll");
TCHAR ParamFile[] = TEXT("%SystemRoot%\\System32\\kernel32.dll");
WCHAR EventSources[] = TEXT("(Folder Redirection,Application)\0" );
DWORD dwTypes = 0x7;
DWORD dwSet = 1;
DWORD dwReset = 0;
//register the dll as an extension of the policy engine
lResult = RegCreateKeyEx (HKEY_LOCAL_MACHINE,
TEXT("Software\\Microsoft\\Windows NT\\CurrentVersion\\Winlogon\\GPExtensions\\{25537BA6-77A8-11D2-9B6C-0000F8080861}"),
0, NULL, REG_OPTION_NON_VOLATILE, KEY_WRITE,
NULL, &hKey, &dwDisp);
if (lResult != ERROR_SUCCESS)
{
return SELFREG_E_CLASS;
}
RegSetValueEx (hKey, NULL, 0, REG_SZ, (LPBYTE) TEXT("Folder Redirection"),
(lstrlen (TEXT("Folder Redirection")) + 1) * sizeof (TCHAR));
RegSetValueEx (hKey, TEXT("ProcessGroupPolicyEx"), 0, REG_SZ, (LPBYTE)TEXT("ProcessGroupPolicyEx"),
(lstrlen(TEXT("ProcessGroupPolicyEx")) + 1) * sizeof(TCHAR));
RegSetValueEx (hKey, TEXT("DllName"), 0, REG_EXPAND_SZ, (LPBYTE)TEXT("fdeploy.dll"),
(lstrlen(TEXT("fdeploy.dll")) + 1) * sizeof(TCHAR));
RegSetValueEx (hKey, TEXT("NoMachinePolicy"), 0, REG_DWORD,
(LPBYTE)&dwSet, sizeof (dwSet));
RegSetValueEx (hKey, TEXT("NoSlowLink"), 0, REG_DWORD,
(LPBYTE)&dwSet, sizeof (dwSet));
RegSetValueEx (hKey, TEXT("PerUserLocalSettings"), 0, REG_DWORD,
(LPBYTE)&dwSet, sizeof (dwSet));
//we want the folder redirection extension to get loaded each time.
RegSetValueEx (hKey, TEXT("NoGPOListChanges"), 0, REG_DWORD,
(LPBYTE)&dwReset, sizeof (dwReset));
//
// New perf. stuff. We also want to get called in the background and
// async. foreground case.
//
RegSetValueEx (hKey, TEXT("NoBackgroundPolicy"), 0, REG_DWORD,
(LPBYTE)&dwReset, sizeof (dwReset));
RegSetValueEx (hKey, TEXT("GenerateGroupPolicy"), 0, REG_SZ, (LPBYTE)TEXT("GenerateGroupPolicy"),
(lstrlen (TEXT("GenerateGroupPolicy")) + 1) * sizeof(TCHAR));
// Need to register event sources for RSoP
RegSetValueEx (hKey, TEXT("EventSources"), 0, REG_MULTI_SZ, (LPBYTE)EventSources,
sizeof(EventSources) );
RegCloseKey (hKey);
//register the dll as a source for event log messages
lResult = RegCreateKeyEx (HKEY_LOCAL_MACHINE,
TEXT("SYSTEM\\CurrentControlSet\\Services\\EventLog\\Application\\Folder Redirection"),
0, NULL, REG_OPTION_NON_VOLATILE, KEY_WRITE,
NULL, &hKey, &dwDisp);
if (lResult != ERROR_SUCCESS)
{
return SELFREG_E_CLASS;
}
RegSetValueEx (hKey, TEXT("EventMessageFile"), 0, REG_EXPAND_SZ, (LPBYTE) EventFile,
(lstrlen(EventFile) + 1) * sizeof (TCHAR));
RegSetValueEx (hKey, TEXT("ParameterMessageFile"), 0, REG_EXPAND_SZ, (LPBYTE) ParamFile,
(lstrlen(ParamFile) + 1) * sizeof (TCHAR));
RegSetValueEx (hKey, TEXT("TypesSupported"), 0, REG_DWORD, (LPBYTE) &dwTypes,
sizeof(DWORD));
RegCloseKey (hKey);
return S_OK;
}
/////////////////////////////////////////////////////////////////////////////
// DllUnregisterServer - Removes entries from the system registry
STDAPI DllUnregisterServer(void)
{
RegDelnode (HKEY_LOCAL_MACHINE,
TEXT("Software\\Microsoft\\Windows NT\\CurrentVersion\\Winlogon\\GPExtensions\\{25537BA6-77A8-11D2-9B6C-0000F8080861}"));
RegDelnode (HKEY_LOCAL_MACHINE,
TEXT("SYSTEM\\CurrentControlSet\\Services\\EventLog\\Application\\Folder Redirection"));
return S_OK;
}