// -------------------------------------------------------------------------- // Module Name: ThemeManagerSessionData.cpp // // Copyright (c) 2000, Microsoft Corporation // // This file contains a class that implements information the encapsulates a // client TS session for the theme server. // // History: 2000-10-10 vtan created // 2000-11-29 vtan moved to separate file // -------------------------------------------------------------------------- #include "StandardHeader.h" #include "ThemeManagerSessionData.h" #include #include #include "SingleThreadedExecution.h" #include "StatusCode.h" #include "ThemeManagerAPIRequest.h" #include "TokenInformation.h" // -------------------------------------------------------------------------- // CThemeManagerSessionData::s_pAPIConnection // // Purpose: Static member variables. // // History: 2000-12-02 vtan created // -------------------------------------------------------------------------- CAPIConnection* CThemeManagerSessionData::s_pAPIConnection = NULL; // -------------------------------------------------------------------------- // CThemeManagerSessionData::CThemeManagerSessionData // // Arguments: pAPIConnection = CAPIConnection for port access control. // dwSessionID = Session ID. // // Returns: // // Purpose: Constructor for CThemeManagerSessionData. // // History: 2000-11-17 vtan created // -------------------------------------------------------------------------- CThemeManagerSessionData::CThemeManagerSessionData (DWORD dwSessionID) : _dwSessionID(dwSessionID), _pvThemeLoaderData(NULL), _hToken(NULL), _hProcessClient(NULL), _hWait(NULL) { } // -------------------------------------------------------------------------- // CThemeManagerSessionData::~CThemeManagerSessionData // // Arguments: // // Returns: // // Purpose: Destructor for CThemeManagerSessionData. // // History: 2000-11-17 vtan created // -------------------------------------------------------------------------- CThemeManagerSessionData::~CThemeManagerSessionData (void) { ASSERTMSG(_hWait == NULL, "Wait not executed or removed in CThemeManagerSessionData::~CThemeManagerSessionData"); ASSERTMSG(_hProcessClient == NULL, "_hProcessClient not closed in CThemeManagerSessionData::~CThemeManagerSessionData"); TSTATUS(UserLogoff()); if (_pvThemeLoaderData != NULL) { SessionFree(_pvThemeLoaderData); _pvThemeLoaderData = NULL; } } // -------------------------------------------------------------------------- // CThemeManagerSessionData::GetData // // Arguments: // // Returns: void* // // Purpose: Returns the internal data blob allocated by SessionCreate. // // History: 2000-11-17 vtan created // -------------------------------------------------------------------------- void* CThemeManagerSessionData::GetData (void) const { return(_pvThemeLoaderData); } // -------------------------------------------------------------------------- // CThemeManagerSessionData::EqualSessionID // // Arguments: dwSessionID // // Returns: bool // // Purpose: Returns whether the given session ID matches this session // data. // // History: 2000-11-30 vtan created // -------------------------------------------------------------------------- bool CThemeManagerSessionData::EqualSessionID (DWORD dwSessionID) const { return(dwSessionID == _dwSessionID); } // -------------------------------------------------------------------------- // CThemeManagerSessionData::Allocate // // Arguments: hProcessClient = Handle to the client process. // // Returns: NTSTATUS // // Purpose: Allocates a data blob via SessionCreate which also keeps a // handle to the client process that initiated the session. This // is always winlogon in the client session ID. // // History: 2000-11-17 vtan created // -------------------------------------------------------------------------- NTSTATUS CThemeManagerSessionData::Allocate (HANDLE hProcessClient, DWORD dwServerChangeNumber, void *pfnRegister, void *pfnUnregister, void *pfnClearStockObjects, DWORD dwStackSizeReserve, DWORD dwStackSizeCommit) { NTSTATUS status; if (DuplicateHandle(GetCurrentProcess(), hProcessClient, GetCurrentProcess(), &_hProcessClient, SYNCHRONIZE, FALSE, 0) != FALSE) { ASSERTMSG(_hWait == NULL, "_hWait already exists in CThemeManagerSessionData::Allocate"); AddRef(); if (RegisterWaitForSingleObject(&_hWait, _hProcessClient, CB_SessionTermination, this, INFINITE, WT_EXECUTEDEFAULT | WT_EXECUTEONLYONCE) != FALSE) { _pvThemeLoaderData = SessionAllocate(hProcessClient, dwServerChangeNumber, pfnRegister, pfnUnregister, pfnClearStockObjects, dwStackSizeReserve, dwStackSizeCommit); if (_pvThemeLoaderData != NULL) { status = STATUS_SUCCESS; } else { status = STATUS_NO_MEMORY; } } else { status = CStatusCode::StatusCodeOfLastError(); } if (!NT_SUCCESS(status)) { HANDLE hWait; // In the case of failure grab the _hWait and try to unregister it. // If the unregister fails then the callback is already executing // and there's little we can to stop it. This means that the winlogon // for the client session died between the time we entered this function // and registered the wait and now. If the unregister worked then then // callback hasn't executed so just release the resources. hWait = InterlockedExchangePointer(&_hWait, NULL); if (hWait != NULL) { if (UnregisterWait(hWait) != FALSE) { Release(); } ReleaseHandle(_hProcessClient); if (_pvThemeLoaderData != NULL) { SessionFree(_pvThemeLoaderData); _pvThemeLoaderData = NULL; } } } } else { status = CStatusCode::StatusCodeOfLastError(); } return(status); } // -------------------------------------------------------------------------- // CThemeManagerSessionData::Cleanup // // Arguments: // // Returns: NTSTATUS // // Purpose: Used to unregister the wait on the client process. This is // necessary to prevent the callback from occurring after the // service has been shut down which will cause access to a static // member variable that is NULL'd out. // // History: 2001-01-05 vtan created // -------------------------------------------------------------------------- NTSTATUS CThemeManagerSessionData::Cleanup (void) { HANDLE hWait; hWait = InterlockedExchangePointer(&_hWait, NULL); if (hWait != NULL) { if (UnregisterWait(hWait) != FALSE) { Release(); } ReleaseHandle(_hProcessClient); } return(STATUS_SUCCESS); } // -------------------------------------------------------------------------- // CThemeManagerSessionData::UserLogon // // Arguments: hToken = Handle to the token of the user logging on. // // Returns: NTSTATUS // // Purpose: Saves a copy of the token for use at log off. Allows access // to the theme port to the logon SID of the token. // // History: 2000-11-17 vtan created // -------------------------------------------------------------------------- NTSTATUS CThemeManagerSessionData::UserLogon (HANDLE hToken) { NTSTATUS status; if (_hToken == NULL) { if (DuplicateHandle(GetCurrentProcess(), hToken, GetCurrentProcess(), &_hToken, 0, FALSE, DUPLICATE_SAME_ACCESS) != FALSE) { PSID pSIDLogon; CTokenInformation token(hToken); pSIDLogon = token.GetLogonSID(); if (pSIDLogon != NULL) { if (s_pAPIConnection != NULL) { status = s_pAPIConnection->AddAccess(pSIDLogon, PORT_CONNECT); } else { status = STATUS_SUCCESS; } } else { status = STATUS_INVALID_PARAMETER; } } else { status = CStatusCode::StatusCodeOfLastError(); } } else { status = STATUS_SUCCESS; } return(status); } // -------------------------------------------------------------------------- // CThemeManagerSessionData::UserLogoff // // Arguments: // // Returns: NTSTATUS // // Purpose: Remove access to the theme port for the user being logged off. // // History: 2000-11-17 vtan created // -------------------------------------------------------------------------- NTSTATUS CThemeManagerSessionData::UserLogoff (void) { NTSTATUS status; if (_hToken != NULL) { PSID pSIDLogon; CTokenInformation token(_hToken); pSIDLogon = token.GetLogonSID(); if (pSIDLogon != NULL) { if (s_pAPIConnection != NULL) { status = s_pAPIConnection->RemoveAccess(pSIDLogon); } else { status = STATUS_SUCCESS; } } else { status = STATUS_INVALID_PARAMETER; } ReleaseHandle(_hToken); } else { status = STATUS_SUCCESS; } return(status); } // -------------------------------------------------------------------------- // CThemeManagerSessionData::SetAPIConnection // // Arguments: // // Returns: // // Purpose: Sets the static CAPIConnection for port access changes. // // History: 2000-12-02 vtan created // -------------------------------------------------------------------------- void CThemeManagerSessionData::SetAPIConnection (CAPIConnection *pAPIConnection) { pAPIConnection->AddRef(); s_pAPIConnection = pAPIConnection; } // -------------------------------------------------------------------------- // CThemeManagerSessionData::ReleaseAPIConnection // // Arguments: // // Returns: // // Purpose: Releases the static CAPIConnection for port access changes. // // History: 2000-12-02 vtan created // -------------------------------------------------------------------------- void CThemeManagerSessionData::ReleaseAPIConnection (void) { s_pAPIConnection->Release(); s_pAPIConnection = NULL; } // -------------------------------------------------------------------------- // CThemeManagerSessionData::SessionTermination // // Arguments: // // Returns: // // Purpose: Callback on winlogon process termination for a given session. // We clean up the session specific data blob when this happens. // This allows the process handles on winlogon to be released. // If this isn't done then a zombie lives and the session is // never reclaimed. // // History: 2000-12-09 vtan created // -------------------------------------------------------------------------- void CThemeManagerSessionData::SessionTermination (void) { HANDLE hWait; hWait = InterlockedExchangePointer(&_hWait, NULL); if (hWait != NULL) { (BOOL)UnregisterWait(hWait); ReleaseHandle(_hProcessClient); } CThemeManagerAPIRequest::SessionDestroy(_dwSessionID); Release(); } // -------------------------------------------------------------------------- // CThemeManagerSessionData::CB_SessionTermination // // Arguments: pParameter = This object. // TimerOrWaitFired = Not used. // // Returns: // // Purpose: Callback stub to member function. // // History: 2000-12-09 vtan created // -------------------------------------------------------------------------- void CALLBACK CThemeManagerSessionData::CB_SessionTermination (void *pParameter, BOOLEAN TimerOrWaitFired) { UNREFERENCED_PARAMETER(TimerOrWaitFired); static_cast(pParameter)->SessionTermination(); }