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.
 
 
 
 
 
 

975 lines
22 KiB

/*++
Copyright (c) 2000 Microsoft Corporation
Module Name:
users.c
Abstract:
Creates user profiles and enumerates users
Author:
Jim Schmidt (jimschm) 15-May-2000
Revision History:
<alias> <date> <comments>
--*/
//
// Includes
//
#include "pch.h"
#include "ism.h"
#include "ismp.h"
#define DBG_ISMUSERS "IsmUsers"
//
// Strings
//
#define S_TEMP_HKCU TEXT("$HKCU$")
//
// Constants
//
// None
//
// Macros
//
// None
//
// Types
//
typedef BOOL (WINAPI GETDEFAULTUSERPROFILEDIRECTORY)(PTSTR ProfileDir, PDWORD Size);
typedef GETDEFAULTUSERPROFILEDIRECTORY * PGETDEFAULTUSERPROFILEDIRECTORY;
typedef BOOL (WINAPI GETPROFILESDIRECTORY)(PTSTR ProfileDir, PDWORD Size);
typedef GETPROFILESDIRECTORY * PGETPROFILESDIRECTORY;
typedef LONG (WINAPI REGOVERRIDEPREDEFKEY)(HKEY hKey, HKEY hNewHKey);
typedef REGOVERRIDEPREDEFKEY * PREGOVERRIDEPREDEFKEY;
typedef BOOL (WINAPI CONVERTSIDTOSTRINGSID)(PSID Sid, PTSTR *SidString);
typedef CONVERTSIDTOSTRINGSID * PCONVERTSIDTOSTRINGSID;
typedef BOOL (WINAPI CREATEUSERPROFILE)(
PSID Sid,
PCTSTR UserName,
PCTSTR UserHive,
PTSTR ProfileDir,
DWORD DirSize,
BOOL IsWin9xUpgrade
);
typedef CREATEUSERPROFILE * PCREATEUSERPROFILE;
typedef BOOL (WINAPI OLDCREATEUSERPROFILE)(
PSID Sid,
PCTSTR UserName,
PCTSTR UserHive,
PTSTR ProfileDir,
DWORD DirSize
);
typedef OLDCREATEUSERPROFILE * POLDCREATEUSERPROFILE;
typedef BOOL (WINAPI GETUSERPROFILEDIRECTORY)(HANDLE hToken, PTSTR lpProfileDir, PDWORD lpcchSize);
typedef GETUSERPROFILEDIRECTORY * PGETUSERPROFILEDIRECTORY;
typedef BOOL (WINAPI DELETEPROFILE)(PCTSTR lpSidString, PCTSTR lpProfilePath, PCTSTR lpComputerName);
typedef DELETEPROFILE * PDELETEPROFILE;
//
// Globals
//
PTEMPORARYPROFILE g_CurrentOverrideUser;
//
// Macro expansion list
//
// None
//
// Private function prototypes
//
// None
//
// Macro expansion definition
//
// None
//
// Code
//
HANDLE
pGetUserEnvLib (
VOID
)
{
static HANDLE lib;
if (lib) {
return lib;
}
lib = LoadLibrary (TEXT("userenv.dll"));
if (!lib) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_LOAD_USERENV));
}
return lib;
}
HANDLE
pGetAdvApi32Lib (
VOID
)
{
static HANDLE lib;
if (lib) {
return lib;
}
lib = LoadLibrary (TEXT("advapi32.dll"));
if (!lib) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_LOAD_ADVAPI32));
}
return lib;
}
BOOL
pOurGetDefaultUserProfileDirectory (
OUT PTSTR ProfileDir,
IN OUT PDWORD Size
)
{
HANDLE lib;
PGETDEFAULTUSERPROFILEDIRECTORY getDefaultUserProfileDirectory;
lib = pGetUserEnvLib();
if (!lib) {
return FALSE;
}
#ifdef UNICODE
getDefaultUserProfileDirectory = (PGETDEFAULTUSERPROFILEDIRECTORY) GetProcAddress (lib, "GetDefaultUserProfileDirectoryW");
#else
getDefaultUserProfileDirectory = (PGETDEFAULTUSERPROFILEDIRECTORY) GetProcAddress (lib, "GetDefaultUserProfileDirectoryA");
#endif
if (!getDefaultUserProfileDirectory) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_FIND_GETDEFAULTUSERPROFILEDIRECTORY));
return FALSE;
}
return getDefaultUserProfileDirectory (ProfileDir, Size);
}
BOOL
pOurGetProfilesDirectory (
OUT PTSTR ProfileDir,
IN OUT PDWORD Size
)
{
HANDLE lib;
PGETPROFILESDIRECTORY getProfilesDirectory;
lib = pGetUserEnvLib();
if (!lib) {
return FALSE;
}
#ifdef UNICODE
getProfilesDirectory = (PGETPROFILESDIRECTORY) GetProcAddress (lib, "GetProfilesDirectoryW");
#else
getProfilesDirectory = (PGETPROFILESDIRECTORY) GetProcAddress (lib, "GetProfilesDirectoryA");
#endif
if (!getProfilesDirectory) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_FIND_GETPROFILESDIRECTORY));
return FALSE;
}
return getProfilesDirectory (ProfileDir, Size);
}
LONG
pOurConvertSidToStringSid (
IN PSID Sid,
IN PTSTR *SidString
)
{
HANDLE lib;
PCONVERTSIDTOSTRINGSID convertSidToStringSid;
BOOL result = FALSE;
DWORD error;
lib = pGetAdvApi32Lib();
if (!lib) {
error = GetLastError();
if (error == ERROR_SUCCESS) {
SetLastError (ERROR_PROC_NOT_FOUND);
}
} else {
#ifdef UNICODE
convertSidToStringSid = (PCONVERTSIDTOSTRINGSID) GetProcAddress(lib, "ConvertSidToStringSidW");
#else
convertSidToStringSid = (PCONVERTSIDTOSTRINGSID) GetProcAddress(lib, "ConvertSidToStringSidA");
#endif
if (convertSidToStringSid) {
result = convertSidToStringSid (Sid, SidString);
}
}
return result;
}
BOOL
pOurGetUserProfileDirectory (
IN HANDLE Token,
IN PTSTR ProfileDir,
IN PDWORD ProfileDirSize
)
{
HANDLE lib;
PGETUSERPROFILEDIRECTORY getUserProfileDirectory;
BOOL result = FALSE;
DWORD error;
lib = pGetUserEnvLib();
if (!lib) {
error = GetLastError();
if (error == ERROR_SUCCESS) {
SetLastError (ERROR_PROC_NOT_FOUND);
}
} else {
#ifdef UNICODE
getUserProfileDirectory = (PGETUSERPROFILEDIRECTORY) GetProcAddress (lib, "GetUserProfileDirectoryW");
#else
getUserProfileDirectory = (PGETUSERPROFILEDIRECTORY) GetProcAddress (lib, "GetUserProfileDirectoryA");
#endif
if (getUserProfileDirectory) {
result = getUserProfileDirectory (Token, ProfileDir, ProfileDirSize);
}
}
return result;
}
BOOL
pOurDeleteProfile (
IN PCTSTR UserStringSid,
IN PCTSTR UserProfilePath,
IN PCTSTR ComputerName
)
{
HANDLE lib;
PDELETEPROFILE deleteProfile;
BOOL result = FALSE;
DWORD error;
lib = pGetUserEnvLib();
if (!lib) {
error = GetLastError();
if (error == ERROR_SUCCESS) {
SetLastError (ERROR_PROC_NOT_FOUND);
}
} else {
#ifdef UNICODE
deleteProfile = (PDELETEPROFILE) GetProcAddress (lib, "DeleteProfileW");
#else
deleteProfile = (PDELETEPROFILE) GetProcAddress (lib, "DeleteProfileA");
#endif
if (deleteProfile) {
result = deleteProfile (UserStringSid, UserProfilePath, ComputerName);
}
}
return result;
}
LONG
pOurRegOverridePredefKey (
IN HKEY hKey,
IN HKEY hNewHKey
)
{
HANDLE lib;
PREGOVERRIDEPREDEFKEY regOverridePredefKey;
LONG result;
lib = pGetAdvApi32Lib();
if (!lib) {
result = GetLastError();
if (result == ERROR_SUCCESS) {
result = ERROR_PROC_NOT_FOUND;
}
} else {
regOverridePredefKey = (PREGOVERRIDEPREDEFKEY) GetProcAddress (lib, "RegOverridePredefKey");
if (!regOverridePredefKey) {
result = GetLastError();
} else {
result = regOverridePredefKey (hKey, hNewHKey);
}
}
return result;
}
BOOL
pOurCreateUserProfile (
IN PSID Sid,
IN PCTSTR UserName,
IN PCTSTR UserHive,
OUT PTSTR ProfileDir,
IN DWORD DirSize
)
{
HANDLE lib;
PCREATEUSERPROFILE createUserProfile;
POLDCREATEUSERPROFILE oldCreateUserProfile;
MIG_OSVERSIONINFO versionInfo;
BOOL useNew = FALSE;
lib = pGetUserEnvLib();
if (!lib) {
return FALSE;
}
if (IsmGetOsVersionInfo (g_IsmCurrentPlatform, &versionInfo)) {
if ((versionInfo.OsMajorVersion > OSMAJOR_WINNT5) ||
(versionInfo.OsMajorVersion == OSMAJOR_WINNT5 &&
((versionInfo.OsMinorVersion > OSMINOR_WINNT51) ||
((versionInfo.OsMinorVersion == OSMINOR_WINNT51) &&
(versionInfo.OsBuildNumber >= 2464))))) {
useNew = TRUE;
}
}
if (useNew) {
#ifdef UNICODE
createUserProfile = (PCREATEUSERPROFILE) GetProcAddress (lib, (PCSTR) 154);
#else
createUserProfile = (PCREATEUSERPROFILE) GetProcAddress (lib, (PCSTR) 153);
#endif
if (!createUserProfile) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_FIND_CREATEUSERPROFILE));
return FALSE;
}
return createUserProfile (
Sid,
UserName,
UserHive,
ProfileDir,
DirSize,
FALSE
);
} else {
#ifdef UNICODE
oldCreateUserProfile = (POLDCREATEUSERPROFILE) GetProcAddress (lib, (PCSTR) 110);
#else
oldCreateUserProfile = (POLDCREATEUSERPROFILE) GetProcAddress (lib, (PCSTR) 109);
#endif
if (!oldCreateUserProfile) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_FIND_CREATEUSERPROFILE));
return FALSE;
}
return oldCreateUserProfile (
Sid,
UserName,
UserHive,
ProfileDir,
DirSize
);
}
}
BOOL
pCloneDefaultUserProfile (
IN PSID Sid,
IN PCTSTR UserName,
OUT PTSTR OutUserProfileRoot
)
{
TCHAR userProfile[MAX_TCHAR_PATH];
BOOL result = FALSE;
__try {
if (!pOurCreateUserProfile (
Sid,
UserName,
NULL,
userProfile,
ARRAYSIZE(userProfile)
)) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_CREATE_PROFILE, UserName));
__leave;
}
MYASSERT (OutUserProfileRoot);
StringCopy (OutUserProfileRoot, userProfile);
result = TRUE;
}
__finally {
if (result) {
LOG ((LOG_INFORMATION, (PCSTR) MSG_PROFILE_INFO, UserName, OutUserProfileRoot));
}
}
return result;
}
PTEMPORARYPROFILE
OpenTemporaryProfile (
IN PCTSTR UserName,
IN PCTSTR Domain
)
{
DWORD sidSize;
DWORD domainSize;
SID_NAME_USE use;
PTSTR domainBuffer = NULL;
PSID sidBuffer = NULL;
PTSTR sidString = NULL;
PTEMPORARYPROFILE result = NULL;
PCTSTR accountName = NULL;
TCHAR userProfileRoot[MAX_TCHAR_PATH];
PCTSTR hiveFile = NULL;
LONG rc;
HKEY key = NULL;
PMHANDLE allocPool;
BOOL b;
__try {
//
// Generate the account name
//
if (!UserName || !Domain) {
DEBUGMSG ((DBG_WHOOPS, "EstablishTemporaryProfile requires user and domain"));
__leave;
}
accountName = JoinPaths (Domain, UserName);
//
// Obtain the buffer sizes needed to obtain the user's SID
//
sidSize = 0;
domainSize = 0;
b = LookupAccountName (
NULL,
accountName,
NULL,
&sidSize,
NULL,
&domainSize,
&use
);
if (!b && GetLastError() != ERROR_INSUFFICIENT_BUFFER) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_FIND_ACCOUNT, accountName));
__leave;
}
//
// Allocate the buffers
//
domainBuffer = AllocText (domainSize);
sidBuffer = MemAllocUninit (sidSize);
if (!domainBuffer || !sidBuffer) {
__leave;
}
//
// Get the SID
//
b = LookupAccountName (
NULL,
accountName,
sidBuffer,
&sidSize,
domainBuffer,
&domainSize,
&use
);
if (!b) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_FIND_ACCOUNT_SID, accountName));
__leave;
}
if (use != SidTypeUser) {
SetLastError (ERROR_INVALID_ACCOUNT_NAME);
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_NOT_USER_ACCOUNT, accountName));
__leave;
}
//
// Copy the default profile
//
b = pCloneDefaultUserProfile (sidBuffer, UserName, userProfileRoot);
if (!b) {
__leave;
}
//
// convert SID into a string SID
//
if (!pOurConvertSidToStringSid (sidBuffer, &sidString) || !sidString) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CONVERT_SID_FAILURE));
__leave;
}
//
// Load the user's hive
//
RegUnLoadKey (HKEY_USERS, sidString);
hiveFile = JoinPaths (userProfileRoot, TEXT("ntuser.dat"));
rc = RegLoadKey (HKEY_USERS, sidString, hiveFile);
if (rc != ERROR_SUCCESS) {
SetLastError (rc);
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_LOAD_HIVE, hiveFile));
__leave;
}
//
// Make the hive the new HKCU
//
key = OpenRegKey (HKEY_USERS, sidString);
if (!key) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_MAP_HIVE, hiveFile));
__leave;
}
if (g_CurrentOverrideUser) {
pOurRegOverridePredefKey (HKEY_CURRENT_USER, NULL);
g_CurrentOverrideUser = NULL;
}
rc = pOurRegOverridePredefKey (HKEY_CURRENT_USER, key);
if (rc != ERROR_SUCCESS) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_REDIRECT_HIVE, hiveFile));
__leave;
}
//
// Prepare outbound handle
//
allocPool = PmCreateNamedPool ("TempProfile");
if (!allocPool) {
__leave;
}
result = (PTEMPORARYPROFILE) PmGetMemory (allocPool, sizeof (TEMPORARYPROFILE));
if (!result) {
__leave;
}
g_CurrentOverrideUser = result;
result->AllocPool = allocPool;
result->UserName = PmDuplicateString (allocPool, UserName);
result->DomainName = PmDuplicateString (allocPool, Domain);
result->AccountName = PmDuplicateString (allocPool, accountName);
result->UserProfileRoot = PmDuplicateString (allocPool, userProfileRoot);
result->MapKey = PmDuplicateString (allocPool, sidString);
result->UserStringSid = PmDuplicateString (allocPool, sidString);
result->UserHive = PmDuplicateString (allocPool, hiveFile);
result->UserSid = (PSID) PmDuplicateMemory (
allocPool,
sidBuffer,
GetLengthSid (sidBuffer)
);
}
__finally {
FreePathString (hiveFile);
FreePathString (accountName);
FreeText (domainBuffer);
if (sidBuffer) {
FreeAlloc (sidBuffer);
INVALID_POINTER (sidBuffer);
}
if (key) {
CloseRegKey (key);
}
if (!result) {
if (sidString) {
RegTerminateCache ();
RegUnLoadKey (HKEY_USERS, sidString);
}
pOurRegOverridePredefKey (HKEY_CURRENT_USER, NULL);
}
if (sidString) {
LocalFree (sidString);
}
}
return result;
}
BOOL
SelectTemporaryProfile (
IN PTEMPORARYPROFILE Profile
)
{
LONG rc;
HKEY key;
if (g_CurrentOverrideUser == Profile) {
return TRUE;
}
key = OpenRegKey (HKEY_LOCAL_MACHINE, Profile->MapKey);
if (!key) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_OPEN_USER_REGISTRY, Profile->UserName));
return FALSE;
}
if (g_CurrentOverrideUser) {
pOurRegOverridePredefKey (HKEY_CURRENT_USER, NULL);
g_CurrentOverrideUser = NULL;
}
rc = pOurRegOverridePredefKey (HKEY_CURRENT_USER, key);
CloseRegKey (key);
if (rc == ERROR_SUCCESS) {
g_CurrentOverrideUser = Profile;
return TRUE;
}
return FALSE;
}
BOOL
CloseTemporaryProfile (
IN PTEMPORARYPROFILE Profile,
IN BOOL MakeProfilePermanent
)
{
BOOL result = TRUE;
LONG rc;
DWORD error;
MIG_OSVERSIONINFO osVersionInfo;
if (g_CurrentOverrideUser == Profile) {
pOurRegOverridePredefKey (HKEY_CURRENT_USER, NULL);
g_CurrentOverrideUser = NULL;
}
RegTerminateCache ();
rc = RegUnLoadKey (HKEY_USERS, Profile->MapKey);
DEBUGMSG_IF ((
rc != ERROR_SUCCESS,
DBG_WHOOPS,
"Can't unload mapped hive: rc=%u; check for registry handle leaks",
rc
));
if (MakeProfilePermanent) {
if (!pOurCreateUserProfile (
Profile->UserSid,
Profile->UserName,
Profile->UserHive,
NULL,
0
)) {
// on Win2k it is known that this will fail with error ERROR_SHARING_VIOLATION
// but the hive will actually be OK. So, if this is Win2k
// and the error is ERROR_SHARING_VIOLATION we'll just consider a success.
result = FALSE;
error = GetLastError ();
if (IsmGetOsVersionInfo (PLATFORM_DESTINATION, &osVersionInfo)) {
if ((osVersionInfo.OsType == OSTYPE_WINDOWSNT) &&
(osVersionInfo.OsMajorVersion == OSMAJOR_WINNT5) &&
(osVersionInfo.OsMinorVersion == OSMINOR_GOLD) &&
(error == ERROR_SHARING_VIOLATION)
) {
result = TRUE;
}
}
if (!result) {
SetLastError (error);
}
}
}
if (result) {
PmDestroyPool (Profile->AllocPool);
INVALID_POINTER (Profile);
}
return result;
}
BOOL
MapUserProfile (
IN PCTSTR UserStringSid,
IN PCTSTR UserProfilePath
)
{
PCTSTR hiveFile = NULL;
LONG rc;
HKEY key;
//
// Unload UserStringSid if loaded
//
RegUnLoadKey (HKEY_USERS, UserStringSid);
hiveFile = JoinPaths (UserProfilePath, TEXT("ntuser.dat"));
rc = RegLoadKey (HKEY_USERS, UserStringSid, hiveFile);
if (rc != ERROR_SUCCESS) {
SetLastError (rc);
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_LOAD_HIVE, hiveFile));
FreePathString (hiveFile);
return FALSE;
}
//
// Make the hive the new HKCU
//
key = OpenRegKey (HKEY_USERS, UserStringSid);
if (!key) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_MAP_HIVE, hiveFile));
RegUnLoadKey (HKEY_USERS, UserStringSid);
FreePathString (hiveFile);
return FALSE;
}
rc = pOurRegOverridePredefKey (HKEY_CURRENT_USER, key);
if (rc != ERROR_SUCCESS) {
LOG ((LOG_MODULE_ERROR, (PCSTR) MSG_CANT_REDIRECT_HIVE, hiveFile));
CloseRegKey (key);
RegTerminateCache ();
RegUnLoadKey (HKEY_USERS, UserStringSid);
FreePathString (hiveFile);
return FALSE;
}
CloseRegKey (key);
FreePathString (hiveFile);
return TRUE;
}
BOOL
UnmapUserProfile (
IN PCTSTR UserStringSid
)
{
LONG rc;
pOurRegOverridePredefKey (HKEY_CURRENT_USER, NULL);
RegTerminateCache ();
rc = RegUnLoadKey (HKEY_USERS, UserStringSid);
DEBUGMSG_IF ((
rc != ERROR_SUCCESS,
DBG_WHOOPS,
"Can't unmap user profile: rc=%u; check for registry handle leaks",
rc
));
return TRUE;
}
BOOL
DeleteUserProfile (
IN PCTSTR UserStringSid,
IN PCTSTR UserProfilePath
)
{
RegTerminateCache ();
RegUnLoadKey (HKEY_USERS, UserStringSid);
return pOurDeleteProfile (UserStringSid, UserProfilePath, NULL);
}
PCURRENT_USER_DATA
GetCurrentUserData (
VOID
)
{
PCURRENT_USER_DATA result = NULL;
HANDLE token;
DWORD bytesRequired;
PTOKEN_USER tokenUser;
PMHANDLE allocPool;
PTSTR sidString = NULL;
TCHAR userName[256];
DWORD nameSize;
TCHAR userDomain[256];
DWORD domainSize;
SID_NAME_USE dontCare;
//
// Open the process token.
//
if (!OpenProcessToken (GetCurrentProcess(), TOKEN_QUERY, &token)) {
return FALSE;
}
bytesRequired = 0;
if (GetTokenInformation (token, TokenUser, NULL, 0, &bytesRequired)) {
return FALSE;
}
if (GetLastError () != ERROR_INSUFFICIENT_BUFFER) {
return FALSE;
}
tokenUser = (PTOKEN_USER) MemAllocUninit (bytesRequired);
if (!GetTokenInformation (token, TokenUser, tokenUser, bytesRequired, &bytesRequired)) {
FreeAlloc (tokenUser);
return FALSE;
}
nameSize = ARRAYSIZE (userName);
domainSize = ARRAYSIZE (userDomain);
ZeroMemory (userName, nameSize);
ZeroMemory (userDomain, domainSize);
LookupAccountSid (
NULL,
tokenUser->User.Sid,
userName,
&nameSize,
userDomain,
&domainSize,
&dontCare
);
allocPool = PmCreateNamedPool ("CurrentUser");
if (!allocPool) {
FreeAlloc (tokenUser);
return FALSE;
}
PmDisableTracking (allocPool);
result = (PCURRENT_USER_DATA) PmGetMemory (allocPool, sizeof (CURRENT_USER_DATA));
if (!result) {
FreeAlloc (tokenUser);
return FALSE;
}
result->AllocPool = allocPool;
result->UserName = PmDuplicateString (result->AllocPool, userName);
result->UserDomain = PmDuplicateString (result->AllocPool, userDomain);
if (!pOurConvertSidToStringSid (tokenUser->User.Sid, &sidString) || !sidString) {
PmDestroyPool (allocPool);
FreeAlloc (tokenUser);
return FALSE;
}
result->UserStringSid = PmDuplicateString (allocPool, sidString);
LocalFree (sidString);
FreeAlloc (tokenUser);
// now just get the current user profile path
bytesRequired = MAX_TCHAR_PATH;
result->UserProfilePath = PmGetMemory (allocPool, bytesRequired);
if (!pOurGetUserProfileDirectory (token, (PTSTR)result->UserProfilePath, &bytesRequired)) {
result->UserProfilePath = PmGetMemory (allocPool, bytesRequired);
if (!pOurGetUserProfileDirectory (token, (PTSTR)result->UserProfilePath, &bytesRequired)) {
PmDestroyPool (allocPool);
return FALSE;
}
}
return result;
}
VOID
FreeCurrentUserData (
IN PCURRENT_USER_DATA CurrentUserData
)
{
PmDestroyPool (CurrentUserData->AllocPool);
}
PCTSTR
IsmGetCurrentSidString (
VOID
)
{
if (!g_CurrentOverrideUser) {
return NULL;
} else {
return PmDuplicateString (g_IsmPool, g_CurrentOverrideUser->UserStringSid);
}
}