4781 lines
140 KiB
C++
4781 lines
140 KiB
C++
//+---------------------------------------------------------------------------
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1992 - 1996.
|
|
//
|
|
// File: security.cxx
|
|
//
|
|
// Contents:
|
|
//
|
|
// Classes:
|
|
//
|
|
// Functions: None.
|
|
//
|
|
// History: 15-May-96 MarkBl Created
|
|
// 26-Feb-01 JBenton Prefix Bug 160502 - using uninit memory
|
|
// 17-Apr-01 a-JyotiG Fixed Bug 367263 - Should not assign any privilege/right
|
|
// to system account.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
#include "..\pch\headers.hxx"
|
|
#include <modes.h> // found in private\inc\crypto
|
|
#include <ntsecapi.h>
|
|
#include <ntdsapi.h> // DsCrackNames
|
|
#include "resource.h"
|
|
#include "globals.hxx" // BUGBUG 254102
|
|
#include "sch_cls.hxx" // To implement AddAtJobWithHash
|
|
#include "authzi.h" // for auditing
|
|
#include <FolderSecurity.h>
|
|
#include "svc_core.hxx"
|
|
#include "security.hxx"
|
|
#include "auditing.hxx"
|
|
#include "misc.hxx"
|
|
|
|
//
|
|
// some prototypes for functions not in a header
|
|
//
|
|
BOOL IsThreadCallerAnAdmin(
|
|
HANDLE hThreadToken);
|
|
|
|
//
|
|
// global stuff
|
|
//
|
|
WCHAR gwszComputerName[MAX_COMPUTERNAME_LENGTH + 2] = L""; // this buffer must remain this size or it will break old credentials
|
|
LPWSTR gpwszComputerName = NULL;
|
|
DWORD gdwKeyElement = 0;
|
|
DWORD gccComputerName = MAX_COMPUTERNAME_LENGTH + 2;
|
|
POLICY_ACCOUNT_DOMAIN_INFO* gpDomainInfo = NULL;
|
|
DWORD gcbMachineSid = 0;
|
|
PSID gpMachineSid = NULL;
|
|
extern CStaticCritSec gcsSSCritSection;
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Helper function: ValidateRunAs
|
|
//
|
|
// Synopsis: Verify that password entered for Run As account is correct
|
|
// by actually trying to log on using the credentials
|
|
//
|
|
// *** Verification of NULL passwords is handled elsewhere ***
|
|
//
|
|
// Returns: bool
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
bool ValidateRunAs(
|
|
LPCWSTR pwszAccount,
|
|
LPCWSTR pwszDomain,
|
|
LPCWSTR pwszPassword)
|
|
{
|
|
// NOTE - don't zero out the password anywhere in here -- we still need it!
|
|
|
|
//
|
|
// copy to buffers we can manipulate
|
|
//
|
|
WCHAR wszDomain [MAX_DOMAINNAME + 1];
|
|
WCHAR wszAccount [MAX_USERNAME + 1];
|
|
|
|
//
|
|
// if the domain is present in the account name (SAM names), skip over it
|
|
//
|
|
WCHAR* pSlash = (WCHAR *) wcschr(pwszAccount, L'\\');
|
|
if (pSlash)
|
|
StringCchCopy(wszAccount, MAX_USERNAME + 1, pSlash + 1);
|
|
else
|
|
StringCchCopy(wszAccount, MAX_USERNAME + 1, pwszAccount);
|
|
|
|
StringCchCopy(wszDomain, MAX_DOMAINNAME + 1, pwszDomain);
|
|
|
|
//
|
|
// If the name was passed in as a UPN, convert it to a SAM name first.
|
|
// Treat the account name as a UPN if it lacks a \ and has an @.
|
|
// Otherwise, treat it as a SAM name.
|
|
//
|
|
if (wcschr(pwszAccount, L'\\') == NULL && wcschr(pwszAccount, L'@') != NULL)
|
|
{
|
|
LPWSTR pwszSamName;
|
|
DWORD dwErr = SchedUPNToAccountName(pwszAccount, &pwszSamName);
|
|
if (dwErr != NO_ERROR)
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
pSlash = wcschr(pwszSamName, L'\\');
|
|
schAssert(pSlash);
|
|
*pSlash = L'\0';
|
|
StringCchCopy(wszDomain, MAX_DOMAINNAME + 1, pwszSamName);
|
|
StringCchCopy(wszAccount, MAX_USERNAME + 1, pSlash + 1);
|
|
delete pwszSamName;
|
|
}
|
|
}
|
|
|
|
HANDLE hToken = NULL;
|
|
if (LogonUser(wszAccount,
|
|
wszDomain,
|
|
pwszPassword,
|
|
LOGON32_LOGON_NETWORK,
|
|
LOGON32_PROVIDER_DEFAULT,
|
|
&hToken))
|
|
{
|
|
CloseHandle(hToken);
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Helper function: NotifyLsaOfPasswordChange
|
|
//
|
|
// Synopsis: Notify LSA if the password has been changed for an account so
|
|
// that it can determine if any user sessions need to be refreshed.
|
|
//
|
|
// This code was stolen and modified from base\cluster\service\nm\setpass.c.
|
|
//
|
|
// Returns: ERROR_SUCCESS if successful, Win32 error code otherwise.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
DWORD NotifyLsaOfPasswordChange(
|
|
LPCWSTR pwszAccount,
|
|
LPCWSTR pwszDomain,
|
|
LPCWSTR pwszPassword)
|
|
{
|
|
DWORD ReturnStatus;
|
|
NTSTATUS Status;
|
|
NTSTATUS SubStatus;
|
|
LSA_STRING LsaStringBuf;
|
|
char* AuthPackage = MSV1_0_PACKAGE_NAME;
|
|
HANDLE LsaHandle = NULL;
|
|
ULONG PackageId;
|
|
|
|
PMSV1_0_CHANGEPASSWORD_REQUEST Request = NULL;
|
|
ULONG RequestSize;
|
|
PBYTE Where;
|
|
PVOID Response = NULL;
|
|
ULONG ResponseSize;
|
|
|
|
//
|
|
// Change password in LSA cache
|
|
//
|
|
Status = LsaConnectUntrusted(&LsaHandle);
|
|
|
|
if (Status != STATUS_SUCCESS)
|
|
{
|
|
ReturnStatus = LsaNtStatusToWinError(Status);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
RtlInitString(&LsaStringBuf, AuthPackage);
|
|
|
|
Status = LsaLookupAuthenticationPackage(
|
|
LsaHandle, // Handle
|
|
&LsaStringBuf, // MSV1_0 authentication package
|
|
&PackageId // output: authentication package identifier
|
|
);
|
|
|
|
if (Status != STATUS_SUCCESS)
|
|
{
|
|
ReturnStatus = LsaNtStatusToWinError(Status);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Prepare to call LsaCallAuthenticationPackage()
|
|
//
|
|
RequestSize = sizeof(MSV1_0_CHANGEPASSWORD_REQUEST) +
|
|
( ( wcslen(pwszAccount) +
|
|
wcslen(pwszDomain) +
|
|
wcslen(pwszPassword) + 3
|
|
) * sizeof(WCHAR)
|
|
);
|
|
|
|
Request = (PMSV1_0_CHANGEPASSWORD_REQUEST)
|
|
HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, RequestSize);
|
|
|
|
if (Request == NULL)
|
|
{
|
|
ReturnStatus = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto ErrorExit;
|
|
}
|
|
|
|
ULONG BuffSize = RequestSize;
|
|
|
|
Where = (PBYTE) (Request + 1);
|
|
BuffSize--;
|
|
Request->MessageType = MsV1_0ChangeCachedPassword;
|
|
StringCbCopy((LPWSTR) Where, BuffSize, pwszDomain );
|
|
RtlInitUnicodeString( &Request->DomainName, (wchar_t *) Where );
|
|
Where += Request->DomainName.MaximumLength;
|
|
BuffSize -= Request->DomainName.MaximumLength;
|
|
|
|
StringCbCopy((LPWSTR) Where, BuffSize , pwszAccount );
|
|
RtlInitUnicodeString( &Request->AccountName, (wchar_t *) Where );
|
|
Where += Request->AccountName.MaximumLength;
|
|
BuffSize -= Request->AccountName.MaximumLength;
|
|
|
|
StringCbCopy((LPWSTR) Where, BuffSize , pwszPassword );
|
|
RtlInitUnicodeString( &Request->NewPassword, (wchar_t *) Where );
|
|
Where += Request->NewPassword.MaximumLength;
|
|
|
|
Status = LsaCallAuthenticationPackage(
|
|
LsaHandle,
|
|
PackageId,
|
|
Request, // MSV1_0_CHANGEPASSWORD_REQUEST
|
|
RequestSize,
|
|
&Response,
|
|
&ResponseSize,
|
|
&SubStatus // Receives NSTATUS code indicating the
|
|
// completion status of the authentication
|
|
// package if ERROR_SUCCESS is returned.
|
|
);
|
|
|
|
|
|
if (Status != STATUS_SUCCESS)
|
|
{
|
|
ReturnStatus = LsaNtStatusToWinError(Status);
|
|
goto ErrorExit;
|
|
}
|
|
else if (LsaNtStatusToWinError(SubStatus) != ERROR_SUCCESS)
|
|
{
|
|
ReturnStatus = LsaNtStatusToWinError(SubStatus);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
ReturnStatus = ERROR_SUCCESS;
|
|
|
|
ErrorExit:
|
|
|
|
if (LsaHandle != NULL)
|
|
{
|
|
Status = LsaDeregisterLogonProcess(LsaHandle);
|
|
if (Status != STATUS_SUCCESS)
|
|
{
|
|
// ignore; could possibly log this
|
|
}
|
|
}
|
|
|
|
if (Request != NULL)
|
|
{
|
|
if (!HeapFree(GetProcessHeap(), 0, Request))
|
|
{
|
|
// ignore; could possibly log this
|
|
}
|
|
}
|
|
|
|
if (Response != NULL)
|
|
{
|
|
Status = LsaFreeReturnBuffer(Response);
|
|
if (Status != STATUS_SUCCESS)
|
|
{
|
|
// ignore; could possibly log this
|
|
}
|
|
}
|
|
|
|
return ReturnStatus;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// RPC: SASetAccountInformation
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments: [Handle] --
|
|
// [pwszJobName] -- Relative job name. eg: MyJob.job.
|
|
// [pwszAccount] --
|
|
// [pwszPassword] --
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
SASetAccountInformation(
|
|
SASEC_HANDLE Handle,
|
|
LPCWSTR pwszJobName,
|
|
LPCWSTR pwszAccount,
|
|
LPCWSTR pwszPassword,
|
|
DWORD dwJobFlags)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
// we're going to do the access check in two stages,
|
|
// first make sure that the principal is allowed to
|
|
// do any scheduling whatsoever - later on, we'll
|
|
// check permissions on the specific file in question
|
|
if (FAILED(hr = RPCFolderAccessCheck(g_TasksFolderInfo.ptszPath, FILE_WRITE_DATA, HandleImpersonation)))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// Check for invalid params (note that pwszPassword is allowed to be NULL)
|
|
//
|
|
if (pwszJobName == NULL || pwszAccount == NULL)
|
|
{
|
|
CHECK_HRESULT(E_INVALIDARG);
|
|
return(E_INVALIDARG);
|
|
}
|
|
|
|
//
|
|
// Disallow files outside the tasks folder
|
|
//
|
|
if (wcschr(pwszJobName, L'\\') || wcschr(pwszJobName, L'/'))
|
|
{
|
|
CHECK_HRESULT(E_INVALIDARG);
|
|
return(E_INVALIDARG);
|
|
}
|
|
|
|
//
|
|
// Append the job name to the local Task's folder path.
|
|
//
|
|
schAssert(g_TasksFolderInfo.ptszPath != NULL);
|
|
WCHAR wszJobPath[MAX_PATH + 1];
|
|
if ((wcslen(g_TasksFolderInfo.ptszPath) + 1 + wcslen(pwszJobName) + 1) > (MAX_PATH + 1))
|
|
{
|
|
CHECK_HRESULT(SCHED_E_CANNOT_OPEN_TASK);
|
|
return(SCHED_E_CANNOT_OPEN_TASK);
|
|
}
|
|
StringCchCopy(wszJobPath, MAX_PATH + 1, g_TasksFolderInfo.ptszPath);
|
|
StringCchCat(wszJobPath, MAX_PATH + 1, L"\\");
|
|
StringCchCat(wszJobPath, MAX_PATH + 1, pwszJobName);
|
|
|
|
//
|
|
// Get the account's SID and domain
|
|
//
|
|
PSID pAccountSid = NULL;
|
|
DWORD cbAccountSid = MAX_SID_SIZE;
|
|
DWORD ccDomain = MAX_DOMAINNAME + 1;
|
|
BYTE pbAccountSid[MAX_SID_SIZE];
|
|
WCHAR wszDomain[MAX_DOMAINNAME + 1] = L"";
|
|
|
|
HRESULT hrGetAccountSidAndDomain = GetAccountSidAndDomain(pwszAccount, pbAccountSid, cbAccountSid, wszDomain, ccDomain);
|
|
if (FAILED(hrGetAccountSidAndDomain))
|
|
{
|
|
// continue on -- we don't want to return yet on failure, because we don't want to reveal that
|
|
// the "run as" account is invalid if the caller shouldn't even be allowed to make this call;
|
|
}
|
|
else
|
|
{
|
|
pAccountSid = pbAccountSid;
|
|
}
|
|
|
|
//
|
|
// Impersonate the caller, open his token, then end impersonation so we aren't impersonated during Auditing
|
|
//
|
|
DWORD RpcStatus = RpcImpersonateClient(NULL);
|
|
if (RpcStatus != RPC_S_OK)
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(RpcStatus);
|
|
CHECK_HRESULT(hr);
|
|
return hr;
|
|
}
|
|
|
|
HANDLE hToken;
|
|
if (!OpenThreadToken(GetCurrentThread(),
|
|
TOKEN_QUERY, // Desired access.
|
|
TRUE, // Open as self.
|
|
&hToken))
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
goto Clean0;
|
|
}
|
|
|
|
if ((RpcStatus = RpcRevertToSelf()) != RPC_S_OK)
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(RpcStatus);
|
|
CHECK_HRESULT(hr);
|
|
goto Clean1;
|
|
}
|
|
|
|
//
|
|
// Now that we have the thread token, audit the job creation.
|
|
// We do this here regardless of whether the user gets access denied down below.
|
|
// However, we can only do this if we succeeded in looking up the "run as" account,
|
|
// as that information is needed for the audit logging.
|
|
//
|
|
if (SUCCEEDED(hrGetAccountSidAndDomain))
|
|
{
|
|
hr = AuditJob(hToken, pAccountSid, wszJobPath);
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("SASetAccountInformation: AuditJob", hr);
|
|
|
|
// let's just forget this happened, OK?
|
|
hr = S_OK;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Reimpersonate client
|
|
//
|
|
RpcStatus = RpcImpersonateClient(NULL);
|
|
if (RpcStatus != RPC_S_OK)
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(RpcStatus);
|
|
CHECK_HRESULT(hr);
|
|
goto Clean1;
|
|
}
|
|
|
|
//
|
|
// Check whether caller should even be allowed to make this call
|
|
//
|
|
if (FAILED(hr = FolderAccessCheck(wszJobPath, hToken, FILE_WRITE_DATA)))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto Clean1;
|
|
}
|
|
|
|
if (FAILED(hrGetAccountSidAndDomain))
|
|
{
|
|
//
|
|
// OK, caller passed the above access check, so reveal that the "run as" account is bad
|
|
//
|
|
hr = hrGetAccountSidAndDomain;
|
|
CHECK_HRESULT(hr);
|
|
goto Clean1;
|
|
}
|
|
|
|
//
|
|
// If the password is NULL, this task is meant to be run
|
|
// without prompting the user for credentials
|
|
//
|
|
if (pwszPassword == NULL)
|
|
{
|
|
DWORD dwError = NO_ERROR;
|
|
|
|
do // Not a loop. Error break out.
|
|
{
|
|
//
|
|
// If the caller has a restricted token (e.g., an ActiveX
|
|
// control), it's not allowed to use a NULL password.
|
|
//
|
|
if (IsTokenRestricted(hToken))
|
|
{
|
|
dwError = ERROR_ACCESS_DENIED;
|
|
schDebugOut((DEB_ERROR, "Restricted token tried to set NULL "
|
|
"password for %ws. Denying access.\n", pwszJobName));
|
|
break;
|
|
}
|
|
|
|
//
|
|
// To set credentials for the job, the caller must have write
|
|
// access to the job file.
|
|
//
|
|
HANDLE hFile;
|
|
hr = OpenFileWithRetry(wszJobPath, GENERIC_WRITE, FILE_SHARE_WRITE, &hFile);
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("SASetAccountInformation: caller's open of task file", hr);
|
|
break;
|
|
}
|
|
|
|
CloseHandle(hFile);
|
|
|
|
//
|
|
// Unless the task is being set to run as LocalSystem, a NULL
|
|
// password means that the task must be scheduled to run only
|
|
// if the user is logged on, so make sure that flag is set in
|
|
// that case
|
|
//
|
|
// An account name of "" signifies the local system account.
|
|
//
|
|
BOOL fIsAccountLocalSystem = (pwszAccount[0] == L'\0');
|
|
|
|
if (!fIsAccountLocalSystem
|
|
&&
|
|
!(dwJobFlags & TASK_FLAG_RUN_ONLY_IF_LOGGED_ON))
|
|
{
|
|
schDebugOut((DEB_ERROR, "SetAccountInformation with NULL "
|
|
"password is only supported for LocalSystem "
|
|
"account or for job with "
|
|
"TASK_FLAG_RUN_ONLY_IF_LOGGED_ON\n",
|
|
pwszJobName));
|
|
hr = SCHED_E_UNSUPPORTED_ACCOUNT_OPTION;
|
|
break;
|
|
}
|
|
|
|
//
|
|
// The caller must be either LocalSystem, an administrator or
|
|
// the user named in pwszAccount (the latter being the most
|
|
// common case. CODEWORK - rearrange to optimize for that case?)
|
|
//
|
|
|
|
BOOL fIsCallerLocalSystem;
|
|
SID LocalSystemSid = {SID_REVISION,
|
|
1,
|
|
SECURITY_NT_AUTHORITY,
|
|
SECURITY_LOCAL_SYSTEM_RID };
|
|
|
|
if (!CheckTokenMembership(hToken,
|
|
&LocalSystemSid,
|
|
&fIsCallerLocalSystem))
|
|
{
|
|
dwError = GetLastError();
|
|
ERR_OUT("CheckTokenMembership", dwError);
|
|
// translate this to E_UNEXPECTED?
|
|
break;
|
|
}
|
|
|
|
if (fIsCallerLocalSystem || IsThreadCallerAnAdmin(hToken))
|
|
{
|
|
//
|
|
// (success)
|
|
//
|
|
break;
|
|
}
|
|
|
|
if (fIsAccountLocalSystem)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED);
|
|
schDebugOut((DEB_ERROR, "Non-system, non-admin tried "
|
|
"to schedule task as LocalSystem\n"));
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Compare the caller's token with the account's SID
|
|
//
|
|
BOOL fIsCallerAccount;
|
|
if (!CheckTokenMembership(hToken,
|
|
pAccountSid,
|
|
&fIsCallerAccount))
|
|
{
|
|
dwError = GetLastError();
|
|
ERR_OUT("CheckTokenMembership", dwError);
|
|
// translate this to E_UNEXPECTED?
|
|
break;
|
|
}
|
|
|
|
if (! fIsCallerAccount)
|
|
{
|
|
schDebugOut((DEB_ERROR, "Caller is neither LocalSystem "
|
|
"nor admin nor the named account\n"));
|
|
hr = HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED);
|
|
}
|
|
|
|
//
|
|
// else success -- the caller is the named account
|
|
//
|
|
} while (0);
|
|
|
|
if (dwError != NO_ERROR)
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(dwError);
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
}
|
|
else
|
|
{
|
|
schDebugOut((DEB_TRACE, "Saving NULL password for %ws\n", pwszJobName));
|
|
}
|
|
|
|
// end of NULL password stuff
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Verify that the credentials entered actually work.
|
|
// This prevents someone from scheduling jobs for a valid account with an invalid password
|
|
// and causing the credential database to be updated with the bad password.
|
|
// It also prevents someone from creating lots of bogus jobs.
|
|
//
|
|
if (!ValidateRunAs(pwszAccount, wszDomain, pwszPassword))
|
|
{
|
|
hr = E_ACCESSDENIED;
|
|
CHECK_HRESULT(hr);
|
|
}
|
|
}
|
|
|
|
Clean1:
|
|
//
|
|
// Close the handle to the thread token
|
|
//
|
|
CloseHandle(hToken);
|
|
|
|
Clean0:
|
|
//
|
|
// End impersonation.
|
|
//
|
|
if ((RpcStatus = RpcRevertToSelf()) != RPC_S_OK)
|
|
{
|
|
ERR_OUT("RpcRevertToSelf", RpcStatus);
|
|
schAssert(!"RpcRevertToSelf failed");
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
//
|
|
// Write the credentials to the database
|
|
// If given a UPN, save "" for the domain and the entire UPN for the user.
|
|
// Treat the account name as a UPN if it lacks a \ and has an @.
|
|
// Otherwise, treat it as a SAM name.
|
|
//
|
|
BOOL fUpn = (wcschr(pwszAccount, L'\\') == NULL && wcschr(pwszAccount, L'@') != NULL);
|
|
|
|
//
|
|
// Retrieve the original creds and compare with the ones we're about to save
|
|
// in order to determine if just the password is being updated. If so, notify LSA.
|
|
// There's no need to do any of this for local system, and we also shouldn't do this
|
|
// if the job is flagged to run only if logged on, as the NULL password supplied in
|
|
// this case is not really the user's password. We can exclude both cases by testing
|
|
// for a non-NULL password as there is no other situation where a NULL password will
|
|
// be allowed. Blank passwords are legit, but they are non-NULL and therefore OK.
|
|
//
|
|
if (pwszPassword)
|
|
{
|
|
JOB_CREDENTIALS jc;
|
|
hr = GetAccountInformation(wszJobPath, &jc);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if ((lstrcmpiW(jc.wszAccount, fUpn ? pwszAccount : SkipDomainName(pwszAccount)) == 0) &&
|
|
(lstrcmpiW(jc.wszPassword, pwszPassword) != 0))
|
|
{
|
|
NotifyLsaOfPasswordChange(fUpn ? pwszAccount : SkipDomainName(pwszAccount),
|
|
fUpn ? L"" : wszDomain,
|
|
pwszPassword);
|
|
}
|
|
ZERO_PASSWORD(jc.wszPassword);
|
|
}
|
|
}
|
|
|
|
hr = SaveJobCredentials(
|
|
wszJobPath,
|
|
fUpn ? pwszAccount : SkipDomainName(pwszAccount),
|
|
fUpn ? L"" : wszDomain,
|
|
pwszPassword,
|
|
pAccountSid
|
|
);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: GetAccountSidAndDomain
|
|
//
|
|
// Synopsis: Gets the SID and Domain of an account.
|
|
// This was factored out of SASetAccountInformation() above, because this is a
|
|
// task that now needs to be performed in more than one place, and I did not
|
|
// wish to duplicate code.
|
|
//
|
|
// Arguments:
|
|
// IN LPCWSTR pwszAccount -- account to look up
|
|
// IN OUT PSID pAccountSid -- pointer to buffer to receive SID
|
|
// IN DWORD cbAccountSid -- size of buffer
|
|
// IN OUT LPWSTR pwszDomain -- pointer to buffer to receive domain
|
|
// IN DWORD ccDomain -- size of buffer
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
GetAccountSidAndDomain(
|
|
LPCWSTR pwszAccount,
|
|
PSID pAccountSid,
|
|
DWORD cbAccountSid,
|
|
LPWSTR pwszDomain,
|
|
DWORD ccDomain)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if (pwszAccount == NULL || pAccountSid == NULL || pwszDomain == NULL)
|
|
{
|
|
CHECK_HRESULT(E_INVALIDARG);
|
|
return(E_INVALIDARG);
|
|
}
|
|
|
|
//
|
|
// An account name of "" signifies the local system account.
|
|
//
|
|
BOOL fIsAccountLocalSystem = (pwszAccount[0] == L'\0');
|
|
|
|
//
|
|
// Get the account's SID
|
|
//
|
|
if (fIsAccountLocalSystem)
|
|
{
|
|
SID LocalSystemSid = {SID_REVISION,
|
|
1,
|
|
SECURITY_NT_AUTHORITY,
|
|
SECURITY_LOCAL_SYSTEM_RID };
|
|
|
|
if (!CopySid(cbAccountSid, pAccountSid, &LocalSystemSid))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
return hr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Treat the account name as a UPN if it lacks a \ and has an @.
|
|
// Otherwise, treat it as a SAM name.
|
|
//
|
|
BOOL fUpn = (wcschr(pwszAccount, L'\\') == NULL && wcschr(pwszAccount, L'@') != NULL);
|
|
schDebugOut((DEB_TRACE, "Name '%S' is a %s name\n", pwszAccount, fUpn ? "UPN" : "SAM"));
|
|
|
|
LPWSTR pwszSamName;
|
|
|
|
if (fUpn)
|
|
{
|
|
//
|
|
// Get the SAM name, so we can call LookupAccountName
|
|
//
|
|
DWORD dwErr = SchedUPNToAccountName(pwszAccount, &pwszSamName);
|
|
if (dwErr != NO_ERROR)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(dwErr);
|
|
CHECK_HRESULT(hr);
|
|
return hr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pwszSamName = (LPWSTR) pwszAccount;
|
|
}
|
|
|
|
DWORD ccDomain = MAX_DOMAINNAME + 1;
|
|
WCHAR wszDomain[MAX_DOMAINNAME + 1] = L"";
|
|
SID_NAME_USE snu;
|
|
|
|
if (!LookupAccountNameWrap(NULL,
|
|
pwszSamName,
|
|
pAccountSid,
|
|
&cbAccountSid,
|
|
pwszDomain,
|
|
&ccDomain,
|
|
&snu))
|
|
{
|
|
CHECK_HRESULT(_HRESULT_FROM_WIN32(GetLastError()));
|
|
hr = SCHED_E_ACCOUNT_NAME_NOT_FOUND;
|
|
}
|
|
|
|
if (fUpn)
|
|
{
|
|
delete pwszSamName;
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
schAssert(IsValidSid(pAccountSid));
|
|
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: GetNSAccountSid
|
|
//
|
|
// Synopsis: Gets the SID of the account set to be used with the Net Schedule API (AT command).
|
|
//
|
|
// Arguments:
|
|
// IN OUT PSID pAccountSid -- pointer to buffer to receive SID
|
|
// IN DWORD cbAccountSid -- size of buffer
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
HRESULT
|
|
GetNSAccountSid(
|
|
PSID pAccountSid,
|
|
DWORD cbAccountSid)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if (pAccountSid == NULL)
|
|
{
|
|
CHECK_HRESULT(E_INVALIDARG);
|
|
return(E_INVALIDARG);
|
|
}
|
|
|
|
//
|
|
// get the name of the AT service account
|
|
//
|
|
DWORD cchAccount = MAX_USERNAME + 1;
|
|
WCHAR wszAccount[MAX_USERNAME + 1];
|
|
hr = SAGetNSAccountInformation(NULL, cchAccount, wszAccount);
|
|
if (FAILED(hr))
|
|
return hr;
|
|
|
|
//
|
|
// Get the account's SID
|
|
//
|
|
DWORD ccDomain = MAX_DOMAINNAME + 1;
|
|
WCHAR wszDomain[MAX_DOMAINNAME + 1] = L"";
|
|
hr = GetAccountSidAndDomain(wszAccount, pAccountSid, cbAccountSid, wszDomain, ccDomain);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: SaveJobCredentials
|
|
//
|
|
// Synopsis: Writes the job credentials to the credential database
|
|
//
|
|
// Arguments:
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
SaveJobCredentials(
|
|
LPCWSTR pwszJobPath,
|
|
LPCWSTR pwszAccount,
|
|
LPCWSTR pwszDomain,
|
|
LPCWSTR pwszPassword,
|
|
PSID pAccountSid
|
|
)
|
|
{
|
|
BYTE rgbIdentity[HASH_DATA_SIZE];
|
|
BYTE rgbHashedAccountSid[HASH_DATA_SIZE] = { 0 };
|
|
RC2_KEY_INFO RC2KeyInfo;
|
|
HRESULT hr;
|
|
DWORD cbSAI;
|
|
DWORD cbSAC;
|
|
DWORD cbCredentialNew;
|
|
DWORD cbEncryptedData;
|
|
DWORD CredentialIndexNew, CredentialIndexPrev;
|
|
BYTE * pbEncryptedData;
|
|
BYTE * pbFoundIdentity;
|
|
BYTE * pbIdentitySet;
|
|
BYTE * pbCredentialNew = NULL;
|
|
BYTE * pbSAI = NULL;
|
|
BYTE * pbSAC = NULL;
|
|
HCRYPTPROV hCSP = NULL;
|
|
|
|
//
|
|
// Obtain a provider handle to the CSP (for use with Crypto API).
|
|
//
|
|
|
|
hr = GetCSPHandle(&hCSP);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
return(hr);
|
|
}
|
|
|
|
//
|
|
// Hash the job into a unique identity.
|
|
//
|
|
|
|
hr = HashJobIdentity(hCSP, pwszJobPath, rgbIdentity);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CloseCSPHandle(hCSP);
|
|
return(hr);
|
|
}
|
|
|
|
//
|
|
// Store a NULL password by flipping the last bit of the hash data.
|
|
//
|
|
|
|
if (pwszPassword == NULL)
|
|
{
|
|
LAST_HASH_BYTE(rgbIdentity) ^= 1;
|
|
}
|
|
|
|
//
|
|
// Guard SA security database access.
|
|
//
|
|
|
|
EnterCriticalSection(&gcsSSCritSection);
|
|
|
|
//
|
|
// Generate the encryption key & encrypt the account information passed.
|
|
//
|
|
|
|
hr = ComputeCredentialKey(hCSP, &RC2KeyInfo);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
hr = EncryptCredentials(RC2KeyInfo,
|
|
pwszAccount,
|
|
pwszDomain,
|
|
pwszPassword,
|
|
pAccountSid,
|
|
&cbEncryptedData,
|
|
&pbEncryptedData);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Read SAI & SAC databases.
|
|
//
|
|
|
|
hr = ReadSecurityDBase(&cbSAI, &pbSAI, &cbSAC, &pbSAC);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Check whether we will be in danger of exceeding the max secret size.
|
|
// We don't know at this time whether we'll be increasing the size of the
|
|
// secret as the data may already be present, but if it does need to be added,
|
|
// the calculation below will show if the size will be over the limit. If so,
|
|
// do a scavenge operation first as a precaution to remove all unused data,
|
|
// then reread the db.
|
|
//
|
|
if ((cbSAI + sizeof(DWORD) + HASH_DATA_SIZE) > MAX_SECRET_SIZE ||
|
|
(cbSAC + sizeof(DWORD) + HASH_DATA_SIZE + cbEncryptedData) > MAX_SECRET_SIZE)
|
|
{
|
|
if (pbSAI != NULL) LocalFree(pbSAI);
|
|
if (pbSAC != NULL) LocalFree(pbSAC);
|
|
pbSAI = pbSAC = NULL;
|
|
|
|
ScavengeSASecurityDBase();
|
|
|
|
hr = ReadSecurityDBase(&cbSAI, &pbSAI, &cbSAC, &pbSAC);
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Check if the identity exists in the SAI.
|
|
// (Note, SAIFindIdentity ignores the last bit of the hash data
|
|
// when searching for a match.)
|
|
//
|
|
|
|
hr = SAIFindIdentity(rgbIdentity,
|
|
cbSAI,
|
|
pbSAI,
|
|
&CredentialIndexPrev,
|
|
NULL,
|
|
&pbFoundIdentity,
|
|
NULL,
|
|
&pbIdentitySet);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Check if the caller-specified credentials already exist in the SAC.
|
|
// Ensure also, if the credentials exist, that the caller has access.
|
|
//
|
|
|
|
hr = CredentialLookupAndAccessCheck(hCSP,
|
|
pAccountSid,
|
|
cbSAC,
|
|
pbSAC,
|
|
&CredentialIndexNew,
|
|
rgbHashedAccountSid,
|
|
&cbCredentialNew,
|
|
&pbCredentialNew);
|
|
|
|
if (FAILED(hr) && hr != SCHED_E_ACCOUNT_INFORMATION_NOT_SET)
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
if (pbFoundIdentity == NULL)
|
|
{
|
|
//
|
|
// This job is new to the SAI. That is, there are no credentials
|
|
// associated with this job yet.
|
|
//
|
|
|
|
if (pbCredentialNew != NULL)
|
|
{
|
|
//
|
|
// If the credentials the caller specified already exist in the
|
|
// SAC, use them. Note, we've already established the caller
|
|
// has permission to use them.
|
|
//
|
|
// Insert the job identity into the SAI identity set associated
|
|
// with this credential.
|
|
//
|
|
|
|
hr = SAIIndexIdentity(cbSAI,
|
|
pbSAI,
|
|
CredentialIndexNew,
|
|
0,
|
|
NULL,
|
|
NULL,
|
|
&pbIdentitySet);
|
|
|
|
if (hr == S_FALSE)
|
|
{
|
|
//
|
|
// The SAC & SAI databases are out of sync.
|
|
// Should *never* occur. Logic on exit handles this.
|
|
//
|
|
|
|
ASSERT_SECURITY_DBASE_CORRUPT();
|
|
hr = SCHED_E_ACCOUNT_DBASE_CORRUPT;
|
|
goto ErrorExit;
|
|
}
|
|
else if (SUCCEEDED(hr))
|
|
{
|
|
hr = SAIInsertIdentity(rgbIdentity,
|
|
pbIdentitySet,
|
|
&cbSAI,
|
|
&pbSAI);
|
|
CHECK_HRESULT(hr);
|
|
|
|
if (SUCCEEDED(hr) && pwszPassword != NULL)
|
|
{
|
|
//
|
|
// Simply change of existing credentials (password change).
|
|
// If we're setting a NULL password, we're setting it for
|
|
// this job alone, and we don't need to touch the SAC.
|
|
// If we're setting a non-NULL password, we're setting it
|
|
// for all jobs in this account, and we need to update the
|
|
// SAC credential in-place.
|
|
//
|
|
|
|
hr = SACUpdateCredential(cbEncryptedData,
|
|
pbEncryptedData,
|
|
cbCredentialNew,
|
|
pbCredentialNew,
|
|
&cbSAC,
|
|
&pbSAC);
|
|
CHECK_HRESULT(hr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// The credentials didn't exist in the SAC.
|
|
//
|
|
// Append new credentials to the SAC & append the new job
|
|
// identity to the SAI. As a result, the identity will be
|
|
// associated with the new credentials.
|
|
//
|
|
|
|
hr = SACAddCredential(rgbHashedAccountSid,
|
|
cbEncryptedData,
|
|
pbEncryptedData,
|
|
&cbSAC,
|
|
&pbSAC);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
hr = SAIAddIdentity(rgbIdentity, &cbSAI, &pbSAI);
|
|
CHECK_HRESULT(hr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Account change for an existing job's credentials.
|
|
//
|
|
// Ensure the caller has permission to change account information.
|
|
// Do so by verifying caller access to the existing credentials.
|
|
//
|
|
|
|
DWORD cbCredentialPrev;
|
|
BYTE * pbCredentialPrev;
|
|
|
|
hr = SACIndexCredential(CredentialIndexPrev,
|
|
cbSAC,
|
|
pbSAC,
|
|
&cbCredentialPrev,
|
|
&pbCredentialPrev);
|
|
|
|
if (hr == S_FALSE)
|
|
{
|
|
//
|
|
// Credential not found? The SAC & SAI databases are out of sync.
|
|
// This should *never* occur. Logic on exit handles this.
|
|
//
|
|
|
|
ASSERT_SECURITY_DBASE_CORRUPT();
|
|
hr = SCHED_E_ACCOUNT_DBASE_CORRUPT;
|
|
goto ErrorExit;
|
|
}
|
|
else if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Only check the credentials if we're dealing with a non-NULL password
|
|
//
|
|
if (pwszPassword != NULL)
|
|
{
|
|
//
|
|
// pbCredentialPrev points to the start of the credential identity.
|
|
//
|
|
|
|
if (!CredentialAccessCheck(hCSP,
|
|
pbCredentialPrev))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED);
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
|
|
if ((pbCredentialNew != NULL) &&
|
|
(CredentialIndexPrev != CredentialIndexNew))
|
|
{
|
|
//
|
|
// The credentials the caller wishes to use already exist in the
|
|
// SAC, yet it differs from the previous.
|
|
//
|
|
// Remove the job identity from its existing SAI position
|
|
// (associated with the previous credentials) and relocate
|
|
// to be associated with the new credentials.
|
|
//
|
|
// SAIRemoveIdentity could result in removal of the associated
|
|
// credential, if this was the last identity associated with it.
|
|
// Save away the original SAC size to see if we must fix up the
|
|
// new credential index on remove.
|
|
//
|
|
|
|
DWORD cbSACOrg = cbSAC;
|
|
|
|
hr = SAIRemoveIdentity(pbFoundIdentity,
|
|
pbIdentitySet,
|
|
&cbSAI,
|
|
&pbSAI,
|
|
CredentialIndexPrev,
|
|
&cbSAC,
|
|
&pbSAC);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
if (cbSACOrg != cbSAC)
|
|
{
|
|
//
|
|
// The new credential index must be adjusted.
|
|
//
|
|
|
|
if (CredentialIndexNew > CredentialIndexPrev)
|
|
{
|
|
CredentialIndexNew--;
|
|
}
|
|
}
|
|
|
|
hr = SAIIndexIdentity(cbSAI,
|
|
pbSAI,
|
|
CredentialIndexNew,
|
|
0,
|
|
NULL,
|
|
NULL,
|
|
&pbIdentitySet); // [out] ptr.
|
|
|
|
if (hr == S_FALSE)
|
|
{
|
|
//
|
|
// The SAC & SAI databases are out of sync. This should
|
|
// *never* occur. Logic on exit handles this.
|
|
//
|
|
|
|
ASSERT_SECURITY_DBASE_CORRUPT();
|
|
hr = SCHED_E_ACCOUNT_DBASE_CORRUPT;
|
|
goto ErrorExit;
|
|
}
|
|
else if (SUCCEEDED(hr))
|
|
{
|
|
hr = SAIInsertIdentity(rgbIdentity,
|
|
pbIdentitySet,
|
|
&cbSAI,
|
|
&pbSAI);
|
|
CHECK_HRESULT(hr);
|
|
|
|
if (SUCCEEDED(hr) && pwszPassword != NULL)
|
|
{
|
|
//
|
|
// Update the existing credentials if the user has
|
|
// specified a non-NULL password.
|
|
//
|
|
// First, re-index the credential since the remove
|
|
// above may have altered SAC content.
|
|
//
|
|
|
|
hr = SACIndexCredential(CredentialIndexNew,
|
|
cbSAC,
|
|
pbSAC,
|
|
&cbCredentialNew,
|
|
&pbCredentialNew);
|
|
|
|
if (hr == S_FALSE)
|
|
{
|
|
//
|
|
// Something is terribly wrong. This should *never*
|
|
// occur. Logic on exit handles this.
|
|
//
|
|
|
|
ASSERT_SECURITY_DBASE_CORRUPT();
|
|
hr = SCHED_E_ACCOUNT_DBASE_CORRUPT;
|
|
goto ErrorExit;
|
|
}
|
|
else if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
hr = SACUpdateCredential(cbEncryptedData,
|
|
pbEncryptedData,
|
|
cbCredentialNew,
|
|
pbCredentialNew,
|
|
&cbSAC,
|
|
&pbSAC);
|
|
CHECK_HRESULT(hr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
else if (pbCredentialNew == NULL)
|
|
{
|
|
//
|
|
// The credentials the caller wishes to use do not exist in the
|
|
// SAC.
|
|
//
|
|
// Remove the job identity from its existing SAI position
|
|
// (associated with the previous credentials), then add both
|
|
// the new credentials and the identity to the SAC & SAI
|
|
// respectively. As a result, the identity will be associated
|
|
// with the new credentials.
|
|
//
|
|
|
|
//
|
|
// NB : This routine also removes the associated credential from
|
|
// the SAC if this was the last identity associated with it.
|
|
// Also, do not reference pbFoundIdentity & pbIdentitySet
|
|
// after this call, as they will be invalid.
|
|
//
|
|
|
|
hr = SAIRemoveIdentity(pbFoundIdentity,
|
|
pbIdentitySet,
|
|
&cbSAI,
|
|
&pbSAI,
|
|
CredentialIndexPrev,
|
|
&cbSAC,
|
|
&pbSAC);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Append the identity and the new credentials to the SAI and
|
|
// SAC respectively.
|
|
//
|
|
|
|
hr = SACAddCredential(rgbHashedAccountSid,
|
|
cbEncryptedData,
|
|
pbEncryptedData,
|
|
&cbSAC,
|
|
&pbSAC);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
hr = SAIAddIdentity(rgbIdentity, &cbSAI, &pbSAI);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Simply change of existing credentials (password change).
|
|
// If we're setting a NULL password, we're setting it for this job
|
|
// alone, and we don't need to touch the SAC. If we're setting a
|
|
// non-NULL password, we're setting it for all jobs in this
|
|
// account, and we need to update the SAC credential in-place.
|
|
//
|
|
|
|
if (pwszPassword != NULL)
|
|
{
|
|
hr = SACUpdateCredential(cbEncryptedData,
|
|
pbEncryptedData,
|
|
cbCredentialPrev,
|
|
pbCredentialPrev,
|
|
&cbSAC,
|
|
&pbSAC);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
|
|
//
|
|
// We also need to rewrite the SAI data, because if the password
|
|
// changed from NULL to non-NULL or vice versa, the last bit of
|
|
// the SAI data will have changed.
|
|
//
|
|
hr = SAIUpdateIdentity(rgbIdentity,
|
|
pbFoundIdentity,
|
|
cbSAI,
|
|
pbSAI);
|
|
|
|
CHECK_HRESULT(hr);
|
|
}
|
|
}
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = WriteSecurityDBase(cbSAI, pbSAI, cbSAC, pbSAC);
|
|
CHECK_HRESULT(hr);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
//
|
|
// Grant the account batch privilege.
|
|
// We could choose to ignore the return code here, since the
|
|
// privilege can still be granted later; but if we ignored it,
|
|
// a caller might never know that the call failed until it was
|
|
// time to run the job, which is not good behavior. (See
|
|
// bug 366582)
|
|
//
|
|
|
|
//Also we should not assign any privilege/right to system account. Refer to bug 367263
|
|
SID LocalSystemSid = { SID_REVISION,
|
|
1,
|
|
SECURITY_NT_AUTHORITY,
|
|
SECURITY_LOCAL_SYSTEM_RID };
|
|
|
|
if(!EqualSid(&LocalSystemSid,pAccountSid)) {
|
|
hr = GrantAccountBatchPrivilege(pAccountSid);
|
|
}
|
|
}
|
|
}
|
|
|
|
ErrorExit:
|
|
if (pbSAI != NULL) LocalFree(pbSAI);
|
|
if (pbSAC != NULL) LocalFree(pbSAC);
|
|
if (hCSP != NULL) CloseCSPHandle(hCSP);
|
|
|
|
//
|
|
// Log an error & rest the SA security dbases SAI & SAC if corruption
|
|
// is detected.
|
|
//
|
|
|
|
if (hr == SCHED_E_ACCOUNT_DBASE_CORRUPT)
|
|
{
|
|
//
|
|
// Log an error.
|
|
//
|
|
|
|
LogServiceError(IERR_SECURITY_DBASE_CORRUPTION, 0,
|
|
IDS_HELP_HINT_DBASE_CORRUPT);
|
|
|
|
//
|
|
// Reset SAI & SAC by writing four bytes of zeros into each.
|
|
// Ignore the return code. No recourse if this fails.
|
|
//
|
|
|
|
DWORD dwZero = 0;
|
|
WriteSecurityDBase(sizeof(dwZero), (BYTE *)&dwZero, sizeof(dwZero),
|
|
(BYTE *)&dwZero);
|
|
}
|
|
|
|
LeaveCriticalSection(&gcsSSCritSection);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// RPC: SASetNSAccountInformation
|
|
//
|
|
// Synopsis: Configure the NetSchedule account.
|
|
//
|
|
// Arguments: [Handle] -- Unused.
|
|
// [pwszAccount] -- Account name. If NULL, reset the credential
|
|
// information to zero.
|
|
// [pwszPassword] -- Account password.
|
|
//
|
|
// Returns: S_OK -- Operation successful.
|
|
// HRESULT -- Error.
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
SASetNSAccountInformation(
|
|
SASEC_HANDLE Handle,
|
|
LPCWSTR pwszAccount,
|
|
LPCWSTR pwszPassword)
|
|
{
|
|
|
|
HRESULT hr = S_OK;
|
|
RPC_STATUS RpcStatus;
|
|
|
|
//
|
|
// If not done so already, initialize the DWORD global data element to be
|
|
// used in generation of the encryption key. It's possible this hasn't
|
|
// been performed yet.
|
|
//
|
|
|
|
if (!gdwKeyElement)
|
|
{
|
|
//
|
|
// NB : This routine enters (and leaves) the gcsSSCritSection
|
|
// critical section.
|
|
//
|
|
|
|
SetMysteryDWORDValue();
|
|
}
|
|
|
|
//
|
|
// The RPC caller must be an administrator to perform this function.
|
|
//
|
|
// Impersonate the caller.
|
|
//
|
|
|
|
if ((RpcStatus = RpcImpersonateClient(NULL)) != RPC_S_OK)
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(RpcStatus);
|
|
CHECK_HRESULT(hr);
|
|
return(hr);
|
|
}
|
|
|
|
if (! IsThreadCallerAnAdmin(NULL))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED);
|
|
}
|
|
|
|
//
|
|
// End impersonation.
|
|
//
|
|
|
|
if ((RpcStatus = RpcRevertToSelf()) != RPC_S_OK)
|
|
{
|
|
//
|
|
// BUGBUG : What to do if the impersonation revert fails?
|
|
//
|
|
|
|
hr = _HRESULT_FROM_WIN32(RpcStatus);
|
|
CHECK_HRESULT(hr);
|
|
schAssert(!"Couldn't revert to self");
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
return(hr);
|
|
}
|
|
|
|
if (pwszPassword && wcslen(pwszPassword) > REAL_PWLEN)
|
|
return _HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
|
|
//
|
|
// Privilege level check above succeeded if we've gotten to this point.
|
|
//
|
|
// Retrieve the SID of the account name specified.
|
|
//
|
|
|
|
RC2_KEY_INFO RC2KeyInfo;
|
|
BYTE pbAccountSid[MAX_SID_SIZE];
|
|
PSID pAccountSid = NULL;
|
|
WCHAR wszDomain[MAX_DOMAINNAME + 1] = L"";
|
|
DWORD cbAccountSid = MAX_SID_SIZE;
|
|
DWORD ccDomain = MAX_DOMAINNAME + 1;
|
|
DWORD dwZero = 0;
|
|
DWORD cbEncryptedData = 0;
|
|
BYTE * pbEncryptedData = NULL;
|
|
SID_NAME_USE snu;
|
|
HCRYPTPROV hCSP = NULL;
|
|
|
|
if (pwszAccount != NULL)
|
|
{
|
|
if (!LookupAccountName(NULL,
|
|
pwszAccount,
|
|
pbAccountSid,
|
|
&cbAccountSid,
|
|
wszDomain,
|
|
&ccDomain,
|
|
&snu))
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
return(SCHED_E_ACCOUNT_NAME_NOT_FOUND);
|
|
}
|
|
|
|
pAccountSid = pbAccountSid;
|
|
pwszAccount = SkipDomainName(pwszAccount);
|
|
|
|
//
|
|
// Verify that the credentials entered actually work.
|
|
// Also note that for NetSchedule jobs, there is no TASK_FLAG_RUN_ONLY_IF_LOGGED_ON,
|
|
// so if a NULL password is entered that mean the password really is supposed to be NULL.
|
|
//
|
|
if (!ValidateRunAs(pwszAccount, wszDomain, pwszPassword))
|
|
{
|
|
hr = E_ACCESSDENIED;
|
|
CHECK_HRESULT(hr);
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// Retrieve the original creds and compare with the ones we're about to save
|
|
// in order to determine if just the password is being updated. If so, notify LSA.
|
|
//
|
|
JOB_CREDENTIALS jc;
|
|
hr = GetNSAccountInformation(&jc);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if ((lstrcmpiW(jc.wszAccount, pwszAccount) == 0) &&
|
|
(lstrcmpiW(jc.wszPassword, pwszPassword) != 0))
|
|
{
|
|
NotifyLsaOfPasswordChange(pwszAccount, wszDomain, pwszPassword);
|
|
}
|
|
ZERO_PASSWORD(jc.wszPassword);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Guard SA security database access.
|
|
//
|
|
|
|
EnterCriticalSection(&gcsSSCritSection);
|
|
|
|
if (pwszAccount == NULL)
|
|
{
|
|
//
|
|
// zero the cred info out to indicate LocalSystem
|
|
//
|
|
hr = WriteLsaData(sizeof(WSZ_SANSC), WSZ_SANSC, sizeof(dwZero),
|
|
(BYTE *)&dwZero);
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Obtain a provider handle to the CSP (for use with Crypto API).
|
|
//
|
|
|
|
hr = GetCSPHandle(&hCSP);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Generate the encryption key & encrypt the account information
|
|
// passed.
|
|
//
|
|
|
|
hr = ComputeCredentialKey(hCSP, &RC2KeyInfo);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
hr = EncryptCredentials(RC2KeyInfo,
|
|
pwszAccount,
|
|
wszDomain,
|
|
pwszPassword,
|
|
pAccountSid,
|
|
&cbEncryptedData,
|
|
&pbEncryptedData);
|
|
|
|
// Clear key content.
|
|
//
|
|
SecureZeroMemory(&RC2KeyInfo, sizeof(RC2KeyInfo));
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
hr = WriteLsaData(sizeof(WSZ_SANSC), WSZ_SANSC, cbEncryptedData,
|
|
pbEncryptedData);
|
|
|
|
delete [] pbEncryptedData;
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Grant the account batch privilege.
|
|
// We could choose to ignore the return code here, since the
|
|
// privilege can still be granted later; but if we ignored it,
|
|
// a caller might never know that the call failed until it was
|
|
// time to run the job, which is not good behavior. (See
|
|
// bug 366582)
|
|
//
|
|
if (pAccountSid != NULL)
|
|
{
|
|
hr = GrantAccountBatchPrivilege(pAccountSid);
|
|
}
|
|
|
|
ErrorExit:
|
|
LeaveCriticalSection(&gcsSSCritSection);
|
|
|
|
if (hCSP != NULL) CloseCSPHandle(hCSP);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// RPC: SAGetNSAccountInformation
|
|
//
|
|
// Synopsis: Retrieve the NetSchedule account name.
|
|
//
|
|
// Arguments: [Handle] --
|
|
// [ccBufferSize] --
|
|
// [wszBuffer] --
|
|
//
|
|
// Returns: S_OK -- Operation successful.
|
|
// S_FALSE -- No account specified.
|
|
// HRESULT -- Error.
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
SAGetNSAccountInformation(
|
|
SASEC_HANDLE Handle,
|
|
DWORD ccBufferSize,
|
|
WCHAR wszBuffer[])
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
//
|
|
// Verify that caller has permission before proceeding any further
|
|
//
|
|
schAssert(g_TasksFolderInfo.ptszPath != NULL);
|
|
if (FAILED(hr = RPCFolderAccessCheck(g_TasksFolderInfo.ptszPath, FILE_READ_DATA, HandleImpersonation)))
|
|
return hr;
|
|
|
|
//
|
|
// Check for invalid params
|
|
//
|
|
if (!wszBuffer)
|
|
{
|
|
CHECK_HRESULT(E_INVALIDARG);
|
|
return(E_INVALIDARG);
|
|
}
|
|
|
|
//
|
|
// Retrieve the NetSchedule credentials, but return only the account name.
|
|
//
|
|
JOB_CREDENTIALS jc;
|
|
hr = GetNSAccountInformation(&jc);
|
|
|
|
if (SUCCEEDED(hr) && hr != S_FALSE)
|
|
{
|
|
ZERO_PASSWORD(jc.wszPassword); // Not needed; NULL handled.
|
|
|
|
if (ccBufferSize > (jc.ccAccount + 1 + jc.ccDomain))
|
|
{
|
|
StringCchCopy(wszBuffer, ccBufferSize, jc.wszDomain);
|
|
StringCchCat(wszBuffer, ccBufferSize, L"\\");
|
|
StringCchCat(wszBuffer, ccBufferSize, jc.wszAccount);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Should *never* occur.
|
|
//
|
|
|
|
hr = HRESULT_FROM_WIN32(ERROR_INSUFFICIENT_BUFFER);
|
|
CHECK_HRESULT(hr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Note that LocalSystem accounts will be returned under the S_FALSE condition;
|
|
// set the buffer to the empty string to reflect this
|
|
//
|
|
if (S_FALSE == hr)
|
|
{
|
|
StringCchCopy(wszBuffer, ccBufferSize, L"");
|
|
}
|
|
}
|
|
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: GetNSAccountInformation
|
|
//
|
|
// Synopsis: Retrieve the NetSchedule account credentials.
|
|
//
|
|
// Arguments: [pjc] -- Returned credentials.
|
|
//
|
|
// Returns: S_OK -- Operation successful.
|
|
// S_FALSE -- No account specified.
|
|
// HRESULT -- Error.
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
GetNSAccountInformation(
|
|
PJOB_CREDENTIALS pjc)
|
|
{
|
|
RC2_KEY_INFO RC2KeyInfo;
|
|
DWORD cbEncryptedData = 0;
|
|
BYTE * pbEncryptedData = NULL;
|
|
HCRYPTPROV hCSP = NULL;
|
|
HRESULT hr;
|
|
|
|
//
|
|
// If not done so already, initialize the DWORD global data element to be
|
|
// used in generation of the encryption key. It's possible this hasn't
|
|
// been performed yet.
|
|
//
|
|
|
|
if (!gdwKeyElement)
|
|
{
|
|
//
|
|
// NB : This routine enters (and leaves) the gcsSSCritSection
|
|
// critical section.
|
|
//
|
|
|
|
SetMysteryDWORDValue();
|
|
}
|
|
|
|
//
|
|
// Guard SA security database access.
|
|
//
|
|
|
|
EnterCriticalSection(&gcsSSCritSection);
|
|
|
|
//
|
|
// Read SAI & SAC databases.
|
|
//
|
|
|
|
hr = ReadLsaData(sizeof(WSZ_SANSC), WSZ_SANSC, &cbEncryptedData,
|
|
&pbEncryptedData);
|
|
|
|
if (FAILED(hr) || hr == S_FALSE)
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
else if (cbEncryptedData <= sizeof(DWORD))
|
|
{
|
|
//
|
|
// The information was specified previously but has been reset since.
|
|
//
|
|
// NOTE: This will be the case if the value has been reset back to LocalSystem,
|
|
// as it merely stores a dword = 0x00000000 in that case
|
|
//
|
|
hr = S_FALSE;
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Obtain a provider handle to the CSP (for use with Crypto API).
|
|
//
|
|
|
|
hr = GetCSPHandle(&hCSP);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Generate key & decrypt the credentials.
|
|
//
|
|
|
|
hr = ComputeCredentialKey(hCSP, &RC2KeyInfo);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
// *** Important ***
|
|
//
|
|
// The encrypted credentials passed are decrypted *in-place*.
|
|
// The decrypted data must be zeroed immediately following decryption
|
|
// (even in a failure case).
|
|
//
|
|
|
|
hr = DecryptCredentials(RC2KeyInfo,
|
|
cbEncryptedData,
|
|
pbEncryptedData,
|
|
pjc);
|
|
|
|
// Don't leave the plain-text password on the heap.
|
|
//
|
|
SecureZeroMemory(pbEncryptedData, cbEncryptedData);
|
|
|
|
// Clear key content.
|
|
//
|
|
SecureZeroMemory(&RC2KeyInfo, sizeof(RC2KeyInfo));
|
|
}
|
|
|
|
ErrorExit:
|
|
LeaveCriticalSection(&gcsSSCritSection);
|
|
|
|
if (pbEncryptedData != NULL) LocalFree(pbEncryptedData);
|
|
|
|
if (hCSP != NULL) CloseCSPHandle(hCSP);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// RPC: SAGetAccountInformation
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments: [pwszJobName] -- Relative job name. eg: MyJob.job.
|
|
// [ccBufferSize] --
|
|
// [wszBuffer] --
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
SAGetAccountInformation(
|
|
SASEC_HANDLE Handle,
|
|
LPCWSTR pwszJobName,
|
|
DWORD ccBufferSize,
|
|
WCHAR wszBuffer[])
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
// we're going to do the access check in two stages,
|
|
// first make sure that the principal is allowed to
|
|
// do any scheduling whatsoever - later on, we'll
|
|
// check permissions on the specific file in question
|
|
if (FAILED(hr = RPCFolderAccessCheck(g_TasksFolderInfo.ptszPath, FILE_READ_DATA, HandleImpersonation)))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// Check for invalid params
|
|
//
|
|
if (pwszJobName == NULL || wszBuffer == NULL)
|
|
{
|
|
CHECK_HRESULT(E_INVALIDARG);
|
|
return(E_INVALIDARG);
|
|
}
|
|
|
|
//
|
|
// Disallow files outside the tasks folder
|
|
//
|
|
if (wcschr(pwszJobName, L'\\') || wcschr(pwszJobName, L'/'))
|
|
{
|
|
CHECK_HRESULT(E_INVALIDARG);
|
|
return(E_INVALIDARG);
|
|
}
|
|
|
|
//
|
|
// Append the job name to the local Task's folder path.
|
|
//
|
|
WCHAR wszJobPath[MAX_PATH + 1];
|
|
schAssert(g_TasksFolderInfo.ptszPath != NULL);
|
|
if ((wcslen(g_TasksFolderInfo.ptszPath) + 1 + wcslen(pwszJobName) + 1) > (MAX_PATH + 1))
|
|
{
|
|
CHECK_HRESULT(SCHED_E_CANNOT_OPEN_TASK);
|
|
return(SCHED_E_CANNOT_OPEN_TASK);
|
|
}
|
|
|
|
StringCchCopy(wszJobPath, MAX_PATH + 1, g_TasksFolderInfo.ptszPath);
|
|
StringCchCat(wszJobPath, MAX_PATH + 1, L"\\");
|
|
StringCchCat(wszJobPath, MAX_PATH + 1, pwszJobName);
|
|
|
|
//
|
|
// Verify that caller has permission before proceeding any further
|
|
//
|
|
if (FAILED(hr = RPCFolderAccessCheck(wszJobPath, FILE_READ_DATA, HandleImpersonation)))
|
|
return hr;
|
|
|
|
//
|
|
// Retrieve the job's credentials, but return only the account name.
|
|
//
|
|
JOB_CREDENTIALS jc;
|
|
hr = GetAccountInformation(wszJobPath, &jc);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
ZERO_PASSWORD(jc.wszPassword); // Not needed; NULL handled.
|
|
|
|
if (ccBufferSize > (jc.ccAccount + 1 + jc.ccDomain))
|
|
{
|
|
//
|
|
// If the job was scheduled to run in the LocalSystem account,
|
|
// Accountname is the empty string
|
|
//
|
|
if (jc.wszAccount[0] == L'\0')
|
|
{
|
|
wszBuffer[0] = L'\0';
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// If the account was supplied as a UPN, DomainName is
|
|
// the empty string
|
|
//
|
|
StringCchCopy(wszBuffer, ccBufferSize, jc.wszDomain);
|
|
if (wszBuffer[0] != L'\0')
|
|
{
|
|
StringCchCat(wszBuffer, ccBufferSize, L"\\");
|
|
}
|
|
StringCchCat(wszBuffer, ccBufferSize, jc.wszAccount);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Should *never* occur.
|
|
//
|
|
|
|
hr = HRESULT_FROM_WIN32(ERROR_INSUFFICIENT_BUFFER);
|
|
CHECK_HRESULT(hr);
|
|
}
|
|
}
|
|
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: GetAccountInformation
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments: [pwszJobPath] -- Fully qualified job path.
|
|
// eg: D:\NT\Tasks\MyJob.job.
|
|
// [pjc] --
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
GetAccountInformation(
|
|
LPCWSTR pwszJobPath,
|
|
PJOB_CREDENTIALS pjc)
|
|
{
|
|
BYTE rgbIdentity[HASH_DATA_SIZE];
|
|
HCRYPTPROV hCSP = NULL;
|
|
DWORD CredentialIndex;
|
|
DWORD cbSAI;
|
|
DWORD cbSAC;
|
|
DWORD cbCredential;
|
|
BYTE * pbCredential;
|
|
BYTE * pbSAI = NULL;
|
|
BYTE * pbSAC = NULL;
|
|
BOOL fIsPasswordNull = FALSE;
|
|
HRESULT hr;
|
|
|
|
//
|
|
// Obtain a provider handle to the CSP (for use with Crypto API).
|
|
//
|
|
|
|
hr = GetCSPHandle(&hCSP);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
return(hr);
|
|
}
|
|
|
|
//
|
|
// Hash the job into a unique identity.
|
|
// It will be used for credential lookup.
|
|
//
|
|
|
|
hr = HashJobIdentity(hCSP, pwszJobPath, rgbIdentity);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CloseCSPHandle(hCSP);
|
|
return(hr);
|
|
}
|
|
|
|
//
|
|
// Guard SA security database access.
|
|
//
|
|
|
|
EnterCriticalSection(&gcsSSCritSection);
|
|
|
|
//
|
|
// Read SAI & SAC databases.
|
|
//
|
|
|
|
hr = ReadSecurityDBase(&cbSAI, &pbSAI, &cbSAC, &pbSAC);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Does this identity exist in the LSA?
|
|
//
|
|
|
|
hr = SAIFindIdentity(rgbIdentity,
|
|
cbSAI,
|
|
pbSAI,
|
|
&CredentialIndex,
|
|
&fIsPasswordNull);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
else if (hr == S_OK) // Found it.
|
|
{
|
|
//
|
|
// Index the credential associated with the identity.
|
|
//
|
|
|
|
hr = SACIndexCredential(CredentialIndex,
|
|
cbSAC,
|
|
pbSAC,
|
|
&cbCredential,
|
|
&pbCredential);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
else if (hr == S_FALSE)
|
|
{
|
|
//
|
|
// Credential not found? The SAC & SAI databases are out of sync.
|
|
// This should *never* occur.
|
|
//
|
|
|
|
ASSERT_SECURITY_DBASE_CORRUPT();
|
|
hr = SCHED_E_ACCOUNT_DBASE_CORRUPT;
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Generate key & decrypt the credentials.
|
|
//
|
|
|
|
RC2_KEY_INFO RC2KeyInfo;
|
|
|
|
hr = ComputeCredentialKey(hCSP, &RC2KeyInfo);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
// *** Important ***
|
|
//
|
|
// The encrypted credentials passed are decrypted
|
|
// *in-place*. Therefore, SAC buffer content has been
|
|
// compromised; plus, the decrypted data must be zeroed
|
|
// immediately following decryption (even in a failure
|
|
// case).
|
|
//
|
|
// NB : The start of the credential refers to the
|
|
// credential identity. Skip over this to refer
|
|
// to the encrypted bits.
|
|
//
|
|
|
|
DWORD cbEncryptedData = cbCredential - HASH_DATA_SIZE;
|
|
BYTE * pbEncryptedData = pbCredential + HASH_DATA_SIZE;
|
|
|
|
hr = DecryptCredentials(RC2KeyInfo,
|
|
cbEncryptedData,
|
|
pbEncryptedData,
|
|
pjc);
|
|
|
|
CHECK_HRESULT(hr);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
// Don't leave the plain-text password on the heap.
|
|
//
|
|
SecureZeroMemory(pbEncryptedData, cbEncryptedData);
|
|
|
|
//
|
|
// If the SAI said this job has a null password, that
|
|
// overrides the password read from the SAC.
|
|
//
|
|
if (fIsPasswordNull)
|
|
{
|
|
pjc->fIsPasswordNull = TRUE;
|
|
SecureZeroMemory(pjc->wszPassword, sizeof pjc->wszPassword);
|
|
pjc->ccPassword = 0;
|
|
}
|
|
}
|
|
// Clear key content.
|
|
//
|
|
SecureZeroMemory(&RC2KeyInfo, sizeof(RC2KeyInfo));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = SCHED_E_ACCOUNT_INFORMATION_NOT_SET;
|
|
}
|
|
|
|
ErrorExit:
|
|
if (pbSAI != NULL) LocalFree(pbSAI);
|
|
if (pbSAC != NULL) LocalFree(pbSAC);
|
|
|
|
if (hCSP != NULL) CloseCSPHandle(hCSP);
|
|
|
|
//
|
|
// Log an error & rest the SA security dbases SAI & SAC
|
|
// if corruption is detected.
|
|
//
|
|
|
|
if (hr == SCHED_E_ACCOUNT_DBASE_CORRUPT)
|
|
{
|
|
//
|
|
// Log an error.
|
|
//
|
|
|
|
LogServiceError(IERR_SECURITY_DBASE_CORRUPTION, 0,
|
|
IDS_HELP_HINT_DBASE_CORRUPT);
|
|
|
|
//
|
|
// Reset SAI & SAC by writing four bytes of zeros into each.
|
|
// Ignore the return code. No recourse if this fails.
|
|
//
|
|
|
|
DWORD dwZero = 0;
|
|
WriteSecurityDBase(sizeof(dwZero), (BYTE *)&dwZero, sizeof(dwZero),
|
|
(BYTE *)&dwZero);
|
|
}
|
|
|
|
LeaveCriticalSection(&gcsSSCritSection);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: HashJobIdentity
|
|
//
|
|
// Synopsis: calculate a hash from several pieces of data specific to the job file
|
|
// that can help to uniquely identify the job and detect tampering
|
|
//
|
|
// Arguments: [hCSP] -- handle to cryptographic service provider
|
|
// [pwszFileName] -- job file name
|
|
// [rgbHash] -- hashed identity
|
|
// [dwHashMethod -- dword value indicating which hash method to use;
|
|
// Default if not specified is the latest method.
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: 11/09/2002 - it was discovered that the value retrieved for domain name
|
|
// (and possibly account name) may not always be the same case, thus causing
|
|
// different hashes to be produced even though the domain had not changed,
|
|
// and the file had not been touched. Always forcing the names to upper case
|
|
// prior to calculating the hash prevents such a change from affecting the
|
|
// hash. Removing the values from the hash calculation altogether also
|
|
// avoids the problem and prevents localization from having negative affects
|
|
// as well. A new parameter, dwHashMethod, has been introduced to allow different
|
|
// hashing methods to be employed to facilitate conversion of existing data.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
HashJobIdentity(
|
|
HCRYPTPROV hCSP,
|
|
LPCWSTR pwszFileName,
|
|
BYTE rgbHash[],
|
|
DWORD dwHashMethod /* = 1 */)
|
|
{
|
|
WCHAR wszApplication[MAX_PATH + 1] = L"";
|
|
WCHAR wszOwnerName[MAX_USERNAME + 1] = L"";
|
|
WCHAR wszOwnerDomain[MAX_DOMAINNAME + 1] = L"";
|
|
UUID JobID;
|
|
FILETIME ftCreationTime;
|
|
PSECURITY_DESCRIPTOR pOwnerSecDescr = NULL;
|
|
DWORD cbOwnerSid;
|
|
PSID pOwnerSid;
|
|
DWORD dwVolumeSerialNo;
|
|
HRESULT hr;
|
|
|
|
hr = GetFileInformation(pwszFileName,
|
|
&cbOwnerSid,
|
|
&pOwnerSid,
|
|
&pOwnerSecDescr,
|
|
&JobID,
|
|
MAX_USERNAME + 1,
|
|
MAX_DOMAINNAME + 1,
|
|
MAX_PATH + 1,
|
|
wszOwnerName,
|
|
wszOwnerDomain,
|
|
wszApplication,
|
|
&ftCreationTime,
|
|
&dwVolumeSerialNo);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
DWORD cbHash = HASH_DATA_SIZE;
|
|
BYTE * pbHash = rgbHash;
|
|
|
|
if (dwHashMethod == 0)
|
|
{
|
|
hr = MarshalData(hCSP,
|
|
NULL,
|
|
HashAndSign,
|
|
&cbHash,
|
|
&pbHash,
|
|
7,
|
|
cbOwnerSid,
|
|
pOwnerSid,
|
|
sizeof(JobID),
|
|
&JobID,
|
|
(wcslen(wszOwnerName) + 1) * sizeof(WCHAR),
|
|
wszOwnerName,
|
|
(wcslen(wszOwnerDomain) + 1) * sizeof(WCHAR),
|
|
wszOwnerDomain,
|
|
(wcslen(wszApplication) + 1) * sizeof(WCHAR),
|
|
wszApplication,
|
|
sizeof(ftCreationTime),
|
|
&ftCreationTime,
|
|
sizeof(dwVolumeSerialNo),
|
|
&dwVolumeSerialNo);
|
|
}
|
|
else /* if (dwHashMethod == 1) */
|
|
{
|
|
hr = MarshalData(hCSP,
|
|
NULL,
|
|
HashAndSign,
|
|
&cbHash,
|
|
&pbHash,
|
|
5,
|
|
cbOwnerSid,
|
|
pOwnerSid,
|
|
sizeof(JobID),
|
|
&JobID,
|
|
(wcslen(wszApplication) + 1) * sizeof(WCHAR),
|
|
wszApplication,
|
|
sizeof(ftCreationTime),
|
|
&ftCreationTime,
|
|
sizeof(dwVolumeSerialNo),
|
|
&dwVolumeSerialNo);
|
|
}
|
|
|
|
schAssert(pbHash == rgbHash);
|
|
}
|
|
|
|
// BUGBUG Is pOwnerSid leaked???
|
|
|
|
delete pOwnerSecDescr;
|
|
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: GrantAccountBatchPrivilege
|
|
//
|
|
// Synopsis: Grant the account batch privilege.
|
|
//
|
|
// Arguments: [pAccountSid] -- Account set.
|
|
//
|
|
// Arguments: None.
|
|
//
|
|
// Returns: HRESULTs
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
GrantAccountBatchPrivilege(PSID pAccountSid)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
LSA_OBJECT_ATTRIBUTES ObjectAttributes = {
|
|
sizeof(LSA_OBJECT_ATTRIBUTES),
|
|
NULL,
|
|
NULL,
|
|
0L,
|
|
NULL,
|
|
NULL
|
|
};
|
|
LSA_HANDLE hPolicy;
|
|
|
|
NTSTATUS Status = LsaOpenPolicy(NULL,
|
|
&ObjectAttributes,
|
|
POLICY_CREATE_ACCOUNT,
|
|
&hPolicy);
|
|
if (Status >= 0)
|
|
{
|
|
LSA_UNICODE_STRING PrivilegeString = {
|
|
sizeof(SE_BATCH_LOGON_NAME) - 2,
|
|
sizeof(SE_BATCH_LOGON_NAME),
|
|
SE_BATCH_LOGON_NAME,
|
|
};
|
|
|
|
Status = LsaAddAccountRights(hPolicy, pAccountSid, &PrivilegeString, 1);
|
|
if (Status < 0)
|
|
{
|
|
ERR_OUT("LsaAddAccountRights", Status);
|
|
}
|
|
|
|
LsaClose(hPolicy);
|
|
}
|
|
else
|
|
{
|
|
ERR_OUT("LsaOpenPolicy", Status);
|
|
}
|
|
|
|
if (Status < 0)
|
|
{
|
|
schAssert(!"Grant Batch Privilege failed, shouldn't have");
|
|
DWORD err = RtlNtStatusToDosError(Status);
|
|
hr = HRESULT_FROM_WIN32(err);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: MarshalData
|
|
//
|
|
// Synopsis: [hCSP] --
|
|
// [phHash] --
|
|
// [MarshalFunction] --
|
|
// [pcbSignature] --
|
|
// [ppbSignature] --
|
|
// [cArgs] --
|
|
// [...] --
|
|
//
|
|
// Arguments: None.
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
MarshalData(
|
|
HCRYPTPROV hCSP,
|
|
HCRYPTHASH * phHash,
|
|
MARSHAL_FUNCTION MarshalFunction,
|
|
DWORD * pcbSignature,
|
|
BYTE ** ppbSignature,
|
|
DWORD cArgs,
|
|
...)
|
|
{
|
|
#define COPYMEMORY(dest, src, size) { \
|
|
CopyMemory(*dest, src, size); \
|
|
*(BYTE **)dest += size; \
|
|
}
|
|
|
|
HCRYPTHASH hHash = NULL;
|
|
DWORD cbSignature = 0;
|
|
BYTE * pbSignature = NULL;
|
|
HRESULT hr = S_OK;
|
|
|
|
va_list pvarg;
|
|
|
|
va_start(pvarg, cArgs);
|
|
|
|
DWORD i, cbSize, cbData = 0;
|
|
|
|
for (i = cArgs; i--; )
|
|
{
|
|
cbData += va_arg(pvarg, DWORD);
|
|
va_arg(pvarg, BYTE *);
|
|
}
|
|
|
|
BYTE * pbData, * pb;
|
|
|
|
pbData = pb = new BYTE[cbData];
|
|
|
|
if (pbData == NULL)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
va_start(pvarg, cArgs);
|
|
|
|
for (i = cArgs; i--; )
|
|
{
|
|
cbSize = va_arg(pvarg, DWORD);
|
|
COPYMEMORY(&pb, va_arg(pvarg, BYTE *), cbSize);
|
|
}
|
|
|
|
if (MarshalFunction == Marshal)
|
|
{
|
|
//
|
|
// Done. Return marshal data in the signature return args.
|
|
//
|
|
|
|
*pcbSignature = cbData;
|
|
*ppbSignature = pbData;
|
|
va_end(pvarg);
|
|
return(S_OK);
|
|
}
|
|
|
|
//
|
|
// Acquire a handle to an MD5 hashing object. MD5 is the most secure
|
|
// hashing algorithm.
|
|
//
|
|
|
|
schAssert(hCSP != NULL);
|
|
|
|
#if DBG
|
|
//
|
|
// We must not be impersonating while calling the Crypto APIs.
|
|
// If we are, the key data will go in the wrong hives.
|
|
//
|
|
HANDLE hToken;
|
|
schAssert(!OpenThreadToken(GetCurrentThread(),
|
|
TOKEN_QUERY, // Desired access.
|
|
TRUE, // Open as self.
|
|
&hToken));
|
|
#endif
|
|
|
|
if (!CryptCreateHash(hCSP,
|
|
CALG_MD5, // Use MD5 hashing.
|
|
0, // MD5 is non-keyed.
|
|
0, // New key container.
|
|
&hHash)) // Returned handle.
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Hash and optionally sign the data. The hash is cached w/in the hash
|
|
// object and returned upon signing.
|
|
//
|
|
|
|
if (!CryptHashData(hHash,
|
|
pbData, // Hash data.
|
|
cbData, // Hash data size.
|
|
0)) // No special flags.
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
if (MarshalFunction == HashAndSign)
|
|
{
|
|
//
|
|
// First, determine necessary signature buffer size & allocate it.
|
|
//
|
|
|
|
if (!CryptSignHash(hHash,
|
|
AT_SIGNATURE, // Signature private key.
|
|
NULL, // No signature.
|
|
0, // Reserved.
|
|
NULL, // NULL return buffer.
|
|
&cbSignature)) // Returned size.
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Caller can supply a buffer to return the signed data only with
|
|
// the HashAndSign option. This is an optimization to reduce the
|
|
// number of memory allocations with known data sizes such as
|
|
// hashed data.
|
|
//
|
|
|
|
if (*pcbSignature)
|
|
{
|
|
if (*pcbSignature >= cbSignature)
|
|
{
|
|
//
|
|
// Caller supplied a buffer & the signed data will fit in it.
|
|
//
|
|
|
|
pbSignature = *ppbSignature;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Caller supplied buffer insufficient size.
|
|
// This is a developer error only.
|
|
//
|
|
|
|
hr = HRESULT_FROM_WIN32(ERROR_INSUFFICIENT_BUFFER);
|
|
schAssert(0 && "MarshalData insufficient buffer!");
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pbSignature = new BYTE[cbSignature];
|
|
|
|
if (pbSignature == NULL)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Perform the actual signing.
|
|
//
|
|
|
|
if (!CryptSignHash(hHash,
|
|
AT_SIGNATURE, // Signature private key.
|
|
NULL, // No signature.
|
|
0, // Reserved.
|
|
pbSignature, // Signature buffer.
|
|
&cbSignature)) // Buffer size.
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
*pcbSignature = cbSignature;
|
|
*ppbSignature = pbSignature;
|
|
}
|
|
|
|
if (phHash != NULL)
|
|
{
|
|
*phHash = hHash;
|
|
hHash = NULL;
|
|
}
|
|
|
|
ErrorExit:
|
|
delete pbData;
|
|
if (FAILED(hr))
|
|
{
|
|
//
|
|
// Caller may have supplied the signature data buffer in the
|
|
// HashAndSign option. If so, don't delete it.
|
|
//
|
|
|
|
if (pbSignature != *ppbSignature)
|
|
{
|
|
delete pbSignature;
|
|
}
|
|
}
|
|
if (hHash != NULL) CryptDestroyHash(hHash);
|
|
va_end(pvarg);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: HashSid
|
|
//
|
|
// Synopsis: [hCSP] --
|
|
// [pSid] --
|
|
// [rgbHash] --
|
|
//
|
|
// Arguments: None.
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
STATIC HRESULT
|
|
HashSid(
|
|
HCRYPTPROV hCSP,
|
|
PSID pSid,
|
|
BYTE rgbHash[])
|
|
{
|
|
DWORD rgdwSubAuthorities[SID_MAX_SUB_AUTHORITIES];
|
|
SID_IDENTIFIER_AUTHORITY * pAuthority;
|
|
|
|
//
|
|
// Validate the sid passed. This is important since the win32
|
|
// documentation for the sid-related api states the returns are
|
|
// undefined if the functions fail.
|
|
//
|
|
|
|
if (!IsValidSid(pSid))
|
|
{
|
|
CHECK_HRESULT(_HRESULT_FROM_WIN32(GetLastError()));
|
|
return(E_UNEXPECTED);
|
|
}
|
|
|
|
//
|
|
// Fetch the sid identifier authority.
|
|
// BUGBUG : I hate this. The doc states if these functions fail, the
|
|
// return value is undefined. How to determine failure?
|
|
//
|
|
|
|
pAuthority = GetSidIdentifierAuthority(pSid);
|
|
|
|
//
|
|
// Fetch all sid subauthorities. Copy them to a temporary buffer in
|
|
// preparation for hashing.
|
|
//
|
|
|
|
PUCHAR pcSubAuthorities = GetSidSubAuthorityCount(pSid);
|
|
|
|
UCHAR cSubAuthoritiesCopied = min(*pcSubAuthorities,
|
|
SID_MAX_SUB_AUTHORITIES);
|
|
|
|
for (UCHAR i = 0; i < cSubAuthoritiesCopied; i++)
|
|
{
|
|
rgdwSubAuthorities[i] = *GetSidSubAuthority(pSid, i);
|
|
}
|
|
|
|
DWORD cbHash = HASH_DATA_SIZE;
|
|
BYTE * pbHash = rgbHash;
|
|
|
|
HRESULT hr = MarshalData(hCSP,
|
|
NULL,
|
|
HashAndSign,
|
|
&cbHash,
|
|
&pbHash,
|
|
2,
|
|
sizeof(SID_IDENTIFIER_AUTHORITY),
|
|
pAuthority,
|
|
cSubAuthoritiesCopied * sizeof(DWORD),
|
|
rgdwSubAuthorities);
|
|
|
|
schAssert(pbHash == rgbHash);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: InitSS
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments: None.
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
InitSS(void)
|
|
{
|
|
LSA_OBJECT_ATTRIBUTES ObjectAttributes = {
|
|
sizeof(LSA_OBJECT_ATTRIBUTES),
|
|
NULL,
|
|
NULL,
|
|
0L,
|
|
NULL,
|
|
NULL
|
|
};
|
|
NTSTATUS Status;
|
|
HRESULT hr;
|
|
|
|
gccComputerName = sizeof(gwszComputerName) / sizeof(TCHAR);
|
|
|
|
if (!GetComputerName(gwszComputerName, &gccComputerName))
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// gwszComputerName will be munged. Save an unmunged copy in
|
|
// gpwszComputerName.
|
|
//
|
|
gpwszComputerName = new WCHAR[gccComputerName + 1];
|
|
if (gpwszComputerName == NULL)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
StringCchCopy(gpwszComputerName, gccComputerName + 1, gwszComputerName);
|
|
|
|
//
|
|
// gwszComputerName is used only for credential encryption. The
|
|
// computer might have been renamed since the credential database was
|
|
// created, so the credential database might have been encrypted using
|
|
// a different computer name than the present one. If a computer name
|
|
// is stored in the registry, use that one rather than the present name.
|
|
// If no name is stored in the registry, store the present one.
|
|
//
|
|
|
|
{
|
|
//
|
|
// Open the schedule agent key
|
|
//
|
|
HKEY hSchedKey;
|
|
long lErr = RegOpenKeyEx(HKEY_LOCAL_MACHINE, SCH_AGENT_KEY, 0,
|
|
KEY_QUERY_VALUE | KEY_SET_VALUE, &hSchedKey);
|
|
if (lErr != ERROR_SUCCESS)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(lErr);
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Get the saved computer name
|
|
//
|
|
WCHAR wszOldName[MAX_COMPUTERNAME_LENGTH + 2];
|
|
DWORD dwType;
|
|
DWORD cb = sizeof(wszOldName);
|
|
lErr = RegQueryValueEx(hSchedKey, SCH_OLDNAME_VALUE, NULL, &dwType,
|
|
(LPBYTE)wszOldName, &cb);
|
|
|
|
if (lErr != ERROR_SUCCESS || dwType != REG_SZ)
|
|
{
|
|
schDebugOut((DEB_ERROR, "InitSS: Couldn't read OldName: err %u, "
|
|
"type %u. Writing '%ws'\n",
|
|
lErr, dwType, gwszComputerName));
|
|
//
|
|
// Write the present computer name
|
|
//
|
|
lErr = RegSetValueEx(hSchedKey, SCH_OLDNAME_VALUE, NULL, REG_SZ,
|
|
(LPBYTE) gwszComputerName,
|
|
(gccComputerName + 1) * sizeof(WCHAR));
|
|
if (lErr != ERROR_SUCCESS)
|
|
{
|
|
schDebugOut((DEB_ERROR, "InitSS: Couldn't write OldName: err %u\n",
|
|
lErr));
|
|
}
|
|
}
|
|
else if (lstrcmpi(gwszComputerName, wszOldName) != 0)
|
|
{
|
|
//
|
|
// Use the stored name instead of the present name
|
|
//
|
|
schDebugOut((DEB_ERROR, "InitSS: Using OldName '%ws'\n", wszOldName));
|
|
StringCchCopy(gwszComputerName, MAX_COMPUTERNAME_LENGTH + 2, wszOldName);
|
|
gccComputerName = (cb / sizeof(WCHAR)) - 1;
|
|
}
|
|
|
|
//
|
|
// Close the key
|
|
//
|
|
RegCloseKey(hSchedKey);
|
|
}
|
|
|
|
|
|
LSA_HANDLE hPolicy;
|
|
|
|
if (!(LsaOpenPolicy(NULL,
|
|
&ObjectAttributes,
|
|
POLICY_VIEW_LOCAL_INFORMATION,
|
|
&hPolicy) >= 0))
|
|
{
|
|
hr = E_UNEXPECTED;
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
Status = LsaQueryInformationPolicy(hPolicy,
|
|
PolicyAccountDomainInformation,
|
|
(void **)&gpDomainInfo);
|
|
|
|
LsaClose(hPolicy);
|
|
|
|
if (!(Status >= 0))
|
|
{
|
|
hr = E_UNEXPECTED;
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
MungeComputerName(gccComputerName);
|
|
|
|
gpMachineSid = gpDomainInfo->DomainSid;
|
|
gcbMachineSid = GetLengthSid(gpDomainInfo->DomainSid);
|
|
|
|
DWORD dwRet = StartupAuditing();
|
|
return _HRESULT_FROM_WIN32(dwRet);
|
|
|
|
ErrorExit:
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: UninitSS
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments: None.
|
|
//
|
|
// Returns: None.
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
void
|
|
UninitSS(void)
|
|
{
|
|
ShutdownAuditing();
|
|
|
|
if (gpDomainInfo != NULL)
|
|
{
|
|
LsaFreeMemory(gpDomainInfo);
|
|
gpDomainInfo = NULL;
|
|
}
|
|
|
|
if (gpwszComputerName != NULL)
|
|
{
|
|
delete gpwszComputerName;
|
|
gpwszComputerName = NULL;
|
|
}
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: MungeComputerName
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments: [psidUser] --
|
|
// [ccAccountName] --
|
|
// [wszAccountName] --
|
|
// [wszAccountNameSize] --
|
|
//
|
|
// Returns: None.
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
STATIC void
|
|
MungeComputerName(DWORD ccComputerName)
|
|
{
|
|
WCHAR * pwszStart = gwszComputerName;
|
|
|
|
while (*pwszStart) pwszStart++;
|
|
|
|
gwszComputerName[MAX_COMPUTERNAME_LENGTH + 1] = L'\0';
|
|
|
|
//
|
|
// Set the character following the computername to a '+' or '-' depending
|
|
// on the value of ccAccountName (if the 2nd bit is set).
|
|
//
|
|
|
|
if ((ccComputerName - 1) & 0x00000001)
|
|
{
|
|
gwszComputerName[MAX_COMPUTERNAME_LENGTH] = L'+';
|
|
}
|
|
else
|
|
{
|
|
gwszComputerName[MAX_COMPUTERNAME_LENGTH] = L'-';
|
|
}
|
|
|
|
//
|
|
// Fill any intermediary buffer space with space characters. Note, no
|
|
// portion of the computername is overwritten.
|
|
//
|
|
// NB : The astute reader will notice the subtle difference in behavior
|
|
// if the computername should be of maximum length. In this case,
|
|
// the '+' or '-' character written above will be overwritten with
|
|
// a space.
|
|
//
|
|
|
|
WCHAR * pwszEnd = &gwszComputerName[MAX_COMPUTERNAME_LENGTH - 1];
|
|
|
|
if (pwszEnd > pwszStart)
|
|
{
|
|
while (pwszEnd != pwszStart)
|
|
{
|
|
*pwszEnd-- = L' ';
|
|
}
|
|
}
|
|
|
|
*pwszStart = L' ';
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: GetCSPHandle
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments: None.
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
GetCSPHandle(HCRYPTPROV * phCSP)
|
|
{
|
|
#if DBG
|
|
//
|
|
// We must not be impersonating while calling the Crypto APIs.
|
|
// If we are, the key data will go in the wrong hives.
|
|
//
|
|
HANDLE hToken;
|
|
schAssert(!OpenThreadToken(GetCurrentThread(),
|
|
TOKEN_QUERY, // Desired access.
|
|
TRUE, // Open as self.
|
|
&hToken));
|
|
#endif
|
|
|
|
HRESULT hr;
|
|
|
|
if (!CryptAcquireContext(phCSP, // Returned CSP handle.
|
|
g_tszSrvcName, // Default Key container.
|
|
// MSFT RSA Base Provider.
|
|
NULL, // Default user provider.
|
|
PROV_RSA_FULL, // Default provider type.
|
|
0)) // No special flags.
|
|
{
|
|
DWORD Status = GetLastError();
|
|
|
|
if (Status == NTE_KEYSET_ENTRY_BAD || Status == NTE_BAD_KEYSET)
|
|
{
|
|
//
|
|
// Delete the keyset and try again.
|
|
// Ignore this return code.
|
|
//
|
|
|
|
if (!CryptAcquireContext(phCSP,
|
|
g_tszSrvcName,
|
|
NULL,
|
|
PROV_RSA_FULL,
|
|
CRYPT_DELETEKEYSET))
|
|
{
|
|
ERR_OUT("CryptAcquireContext(delete)", GetLastError());
|
|
}
|
|
else
|
|
{
|
|
LogServiceError(IERR_SECURITY_KEYSET_CORRUPT, 0, IDS_HELP_HINT_DBASE_CORRUPT);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Print the error in debug builds, but otherwise ignore it.
|
|
//
|
|
ERR_OUT("CryptAcquireContext(open)", Status);
|
|
}
|
|
|
|
//
|
|
// Assume this is the first time this code has been run on this
|
|
// particular machine. Must create a new keyset & key.
|
|
//
|
|
|
|
if (!CryptAcquireContext(phCSP,
|
|
g_tszSrvcName,
|
|
NULL,
|
|
PROV_RSA_FULL,
|
|
CRYPT_NEWKEYSET)) // New keyset.
|
|
{
|
|
Status = GetLastError();
|
|
if (Status == NTE_EXISTS)
|
|
{
|
|
//
|
|
// Our assumption was wrong!
|
|
// Delete the keyset and try again.
|
|
// Ignore this return code.
|
|
//
|
|
if (!CryptAcquireContext(phCSP,
|
|
g_tszSrvcName,
|
|
NULL,
|
|
PROV_RSA_FULL,
|
|
CRYPT_DELETEKEYSET))
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
return(hr);
|
|
}
|
|
else
|
|
{
|
|
LogServiceError(IERR_SECURITY_KEYSET_CORRUPT, 0, IDS_HELP_HINT_DBASE_CORRUPT);
|
|
}
|
|
|
|
//
|
|
// Must now create a new keyset & key.
|
|
//
|
|
if (!CryptAcquireContext(phCSP,
|
|
g_tszSrvcName,
|
|
NULL,
|
|
PROV_RSA_FULL,
|
|
CRYPT_NEWKEYSET)) // New keyset.
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
return(hr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(Status);
|
|
CHECK_HRESULT(hr);
|
|
return(hr);
|
|
}
|
|
}
|
|
|
|
HCRYPTKEY hKey;
|
|
|
|
//
|
|
// The upper 16 bits of the 3rd parm to CryptGenKey specify the key
|
|
// size in bits. The size of the signature from CryptSignHash will
|
|
// be equal to the size of this key. Since we rely on the signature
|
|
// being a specific size, we must explicitly specify the key size.
|
|
//
|
|
if (!CryptGenKey(*phCSP,
|
|
AT_SIGNATURE, // Digital signature.
|
|
(HASH_DATA_SIZE * 8) << 16, // see above
|
|
&hKey ))
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
return(hr);
|
|
}
|
|
CryptDestroyKey(hKey); // No further use for
|
|
// the key.
|
|
}
|
|
|
|
return(S_OK);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: CloseCSPHandle
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments: None.
|
|
//
|
|
// Returns: None.
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
void
|
|
CloseCSPHandle(HCRYPTPROV hCSP)
|
|
{
|
|
CryptReleaseContext(hCSP, 0);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: ComputeCredentialKey
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments: [hCSP] --
|
|
// [pRC2KeyInfo] --
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
ComputeCredentialKey(HCRYPTPROV hCSP, RC2_KEY_INFO * pRC2KeyInfo)
|
|
{
|
|
BYTE rgbHash[HASH_DATA_SIZE];
|
|
HCRYPTHASH hHash = NULL;
|
|
DWORD cbHash = 0;
|
|
BYTE * pbHash = NULL;
|
|
HRESULT hr = S_OK;
|
|
DWORD i;
|
|
|
|
//
|
|
// Hash misc. global data.
|
|
//
|
|
// NB : MarshalData actually does nothing with the 3rd & 4th arguments
|
|
// with the Hash option.
|
|
//
|
|
|
|
hr = MarshalData(hCSP,
|
|
&hHash,
|
|
Hash,
|
|
&cbHash,
|
|
&pbHash,
|
|
2,
|
|
(gccComputerName & 0x00000001 ?
|
|
(MAX_COMPUTERNAME_LENGTH + 2) * sizeof(WCHAR) :
|
|
sizeof(DWORD)),
|
|
(gccComputerName & 0x00000001 ?
|
|
(BYTE *)gwszComputerName : (BYTE *)&gdwKeyElement),
|
|
gcbMachineSid,
|
|
gpMachineSid);
|
|
|
|
//
|
|
// Generate the key.
|
|
//
|
|
// NB : In place of CryptDeriveKey, statically generate the key. This
|
|
// is done to work around Crypto restrictions in France.
|
|
//
|
|
// Old:
|
|
//
|
|
// CryptDeriveKey(ghCSP, CALG_RC2, hHash, 0, &hKey);
|
|
//
|
|
// New:
|
|
//
|
|
|
|
cbHash = sizeof(rgbHash);
|
|
|
|
if (!CryptGetHashParam(hHash, HP_HASHVAL, rgbHash, &cbHash, 0))
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Clear RC2KeyInfo content.
|
|
//
|
|
schAssert(pRC2KeyInfo != NULL);
|
|
SecureZeroMemory(pRC2KeyInfo, sizeof(*pRC2KeyInfo));
|
|
|
|
//
|
|
// Set the upper eleven bytes to 0x00 because Derive key by default
|
|
// uses 11 bytes of 0x00 salt
|
|
//
|
|
SecureZeroMemory(rgbHash + 5, 11);
|
|
|
|
//
|
|
// Use the 5 bytes (40 bits) of the hash as a key.
|
|
//
|
|
RC2KeyEx(pRC2KeyInfo->rgwKeyTable, rgbHash, 16, 40);
|
|
|
|
ErrorExit:
|
|
if (hHash != NULL) CryptDestroyHash(hHash);
|
|
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: EncryptCredentials
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments: [RC2KeyInfo] --
|
|
// [pwszAccount] --
|
|
// [pwszDomain] --
|
|
// [pwszPassword] --
|
|
// [pSid] --
|
|
// [pcbEncryptedData] --
|
|
// [ppbEncryptedData] --
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
EncryptCredentials(
|
|
const RC2_KEY_INFO & RC2KeyInfo,
|
|
LPCWSTR pwszAccount,
|
|
LPCWSTR pwszDomain,
|
|
LPCWSTR pwszPassword,
|
|
PSID pSid,
|
|
DWORD * pcbEncryptedData,
|
|
BYTE ** ppbEncryptedData)
|
|
{
|
|
BYTE rgbBuf[RC2_BLOCKLEN];
|
|
WCHAR * pwszPasswordLocal;
|
|
DWORD cbAccount;
|
|
DWORD cbDomain;
|
|
DWORD cbPassword;
|
|
DWORD cbData = 0;
|
|
DWORD cbEncryptedData = 0;
|
|
DWORD cbPartial;
|
|
DWORD dwPadVal;
|
|
BYTE * pbData = NULL;
|
|
BYTE * pbEncryptedData = NULL;
|
|
HRESULT hr;
|
|
|
|
*pcbEncryptedData = 0;
|
|
*ppbEncryptedData = NULL;
|
|
|
|
if (pwszAccount == NULL || pwszDomain == NULL)
|
|
{
|
|
CHECK_HRESULT(E_INVALIDARG);
|
|
return(E_INVALIDARG);
|
|
}
|
|
|
|
if (pwszPassword == NULL)
|
|
{
|
|
//
|
|
// In the SAC, a NULL password is stored the same as a "" password.
|
|
// (The distinction is made per-job, in the SAI.)
|
|
//
|
|
pwszPasswordLocal = L"";
|
|
}
|
|
else
|
|
{
|
|
pwszPasswordLocal = (WCHAR *)pwszPassword;
|
|
}
|
|
|
|
cbAccount = wcslen(pwszAccount) * sizeof(WCHAR);
|
|
cbDomain = wcslen(pwszDomain) * sizeof(WCHAR);
|
|
cbPassword = wcslen(pwszPasswordLocal) * sizeof(WCHAR);
|
|
|
|
hr = MarshalData(NULL,
|
|
NULL,
|
|
Marshal,
|
|
&cbData,
|
|
&pbData,
|
|
6,
|
|
sizeof(cbAccount),
|
|
&cbAccount,
|
|
cbAccount,
|
|
pwszAccount,
|
|
sizeof(cbDomain),
|
|
&cbDomain,
|
|
cbDomain,
|
|
pwszDomain,
|
|
sizeof(cbPassword),
|
|
&cbPassword,
|
|
cbPassword,
|
|
pwszPasswordLocal);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
//
|
|
// NB : This code exists in place of a call to CryptEncrypt to
|
|
// work around France's Crypto API restrictions. Since
|
|
// CryptEncrypt cannot be called directly, the code from
|
|
// the API to accomplish cypher block encryption is duplicated
|
|
// here.
|
|
//
|
|
|
|
//
|
|
// Calculate the number of pad bytes necessary (must be a multiple)
|
|
// of RC2_BLOCKLEN). If already a multiple of blocklen, do a full
|
|
// block of pad.
|
|
//
|
|
|
|
cbPartial = (cbData % RC2_BLOCKLEN);
|
|
|
|
dwPadVal = RC2_BLOCKLEN - cbPartial;
|
|
|
|
cbEncryptedData = cbData + dwPadVal;
|
|
|
|
//
|
|
// Allocate a buffer for the encrypted data.
|
|
//
|
|
|
|
pbEncryptedData = new BYTE[cbEncryptedData];
|
|
|
|
if (pbEncryptedData == NULL)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
CopyMemory(pbEncryptedData, pbData, cbData);
|
|
|
|
if (dwPadVal)
|
|
{
|
|
//
|
|
// Fill the pad with a value equal to the length of the padding,
|
|
// so decrypt will know the length of the original data and as
|
|
// a simple integrity check.
|
|
//
|
|
|
|
memset(pbEncryptedData + cbData, (INT)dwPadVal, (size_t)dwPadVal);
|
|
}
|
|
|
|
//
|
|
// Perform the encryption - cypher block.
|
|
//
|
|
|
|
*pcbEncryptedData = cbEncryptedData;
|
|
*ppbEncryptedData = pbEncryptedData;
|
|
|
|
while (cbEncryptedData)
|
|
{
|
|
//
|
|
// Put the plaintext into a temporary buffer, then encrypt the
|
|
// data back into the allocated buffer.
|
|
//
|
|
|
|
CopyMemory(rgbBuf, pbEncryptedData, RC2_BLOCKLEN);
|
|
|
|
CBC(RC2,
|
|
RC2_BLOCKLEN,
|
|
pbEncryptedData,
|
|
rgbBuf,
|
|
(void *)RC2KeyInfo.rgwKeyTable,
|
|
ENCRYPT,
|
|
(BYTE *)RC2KeyInfo.rgbIV);
|
|
|
|
pbEncryptedData += RC2_BLOCKLEN;
|
|
cbEncryptedData -= RC2_BLOCKLEN;
|
|
}
|
|
}
|
|
|
|
pbEncryptedData = NULL; // For delete below.
|
|
|
|
ErrorExit:
|
|
delete pbData;
|
|
delete pbEncryptedData;
|
|
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: SkipDomainName
|
|
//
|
|
// Synopsis: Return the relative username if the username passed is in
|
|
// distinguished form. eg: return 'Joe' from 'DogFood\Joe'.
|
|
//
|
|
// Arguments: [pwszUserName] -- User name.
|
|
//
|
|
// Returns: Pointer index to/into pwszUserName.
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
LPWSTR
|
|
SkipDomainName(LPCWSTR pwszUserName)
|
|
{
|
|
LPWSTR pwsz = (LPWSTR)pwszUserName;
|
|
|
|
while (*pwsz && *pwsz != '\\')
|
|
{
|
|
pwsz++;
|
|
}
|
|
|
|
if (*pwsz == L'\\')
|
|
{
|
|
return(++pwsz);
|
|
}
|
|
|
|
return((LPWSTR)pwszUserName);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: DecryptCredentials
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments: [RC2KeyInfo] --
|
|
// [cbEncryptedData] --
|
|
// [pbEncryptedData] --
|
|
// [pjc] --
|
|
// [fDecryptInPlace] --
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
HRESULT
|
|
DecryptCredentials(
|
|
const RC2_KEY_INFO & RC2KeyInfo,
|
|
DWORD cbEncryptedData,
|
|
BYTE * pbEncryptedData,
|
|
PJOB_CREDENTIALS pjc,
|
|
BOOL fDecryptInPlace)
|
|
{
|
|
BYTE rgbBuf[RC2_BLOCKLEN];
|
|
DWORD cbDecryptedData = cbEncryptedData;
|
|
BYTE * pbDecryptedData;
|
|
DWORD BytePos;
|
|
DWORD dwPadVal;
|
|
DWORD i;
|
|
DWORD cbAccount, cbDomain, cbPassword;
|
|
BYTE * pbAccount, * pbDomain, * pbPassword;
|
|
BOOL fIsPasswordNull = FALSE;
|
|
BYTE * pb;
|
|
HRESULT hr = S_OK;
|
|
|
|
//
|
|
// The encrypted data length *must* be a multiple of RC2_BLOCKLEN.
|
|
//
|
|
|
|
if (cbEncryptedData % RC2_BLOCKLEN)
|
|
{
|
|
CHECK_HRESULT(E_UNEXPECTED);
|
|
return(E_UNEXPECTED);
|
|
}
|
|
|
|
//
|
|
// Decrypt overwrites the encrypted data with the decrypted data.
|
|
// If fDecryptInPlace is FALSE, allocate an additional buffer for
|
|
// the decrypted bits, so the encrypted data buffer will not be
|
|
// overwritten.
|
|
//
|
|
|
|
if (!fDecryptInPlace)
|
|
{
|
|
pbDecryptedData = new BYTE[cbEncryptedData];
|
|
|
|
if (pbDecryptedData == NULL)
|
|
{
|
|
CHECK_HRESULT(E_OUTOFMEMORY);
|
|
return(E_OUTOFMEMORY);
|
|
}
|
|
CopyMemory(pbDecryptedData, pbEncryptedData, cbEncryptedData);
|
|
}
|
|
else
|
|
{
|
|
pbDecryptedData = pbEncryptedData;
|
|
}
|
|
|
|
//
|
|
// NB : This code exists in place of a call to CryptDencrypt to
|
|
// work around France's Crypto API restrictions. Since
|
|
// CryptDecrypt cannot be called directly, the code from
|
|
// the API to accomplish cypher block decryption is duplicated
|
|
// here.
|
|
//
|
|
|
|
for (BytePos = 0; (BytePos + RC2_BLOCKLEN) <= cbEncryptedData;
|
|
BytePos += RC2_BLOCKLEN)
|
|
{
|
|
//
|
|
// Use a temporary buffer to store the encrypted data.
|
|
//
|
|
|
|
CopyMemory(rgbBuf, pbDecryptedData + BytePos, RC2_BLOCKLEN);
|
|
|
|
CBC(RC2,
|
|
RC2_BLOCKLEN,
|
|
pbDecryptedData + BytePos,
|
|
rgbBuf,
|
|
(void *)RC2KeyInfo.rgwKeyTable,
|
|
DECRYPT,
|
|
(BYTE *)RC2KeyInfo.rgbIV);
|
|
}
|
|
|
|
//
|
|
// Verify the padding and remove the pad size from the data length.
|
|
// NOTE: The padding is filled with a value equal to the length
|
|
// of the padding and we are guaranteed >= 1 byte of pad.
|
|
//
|
|
// NB : If the pad is wrong, the user's buffer is hosed, because
|
|
// we've decrypted into the user's buffer -- can we re-encrypt it?
|
|
//
|
|
|
|
dwPadVal = (DWORD)*(pbDecryptedData + cbEncryptedData - 1);
|
|
|
|
if (dwPadVal == 0 || dwPadVal > (DWORD) RC2_BLOCKLEN)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Make sure all the (rest of the) pad bytes are correct.
|
|
//
|
|
|
|
for (i = 1; i < dwPadVal; i++)
|
|
{
|
|
if (pbDecryptedData[cbEncryptedData - (i + 1)] != dwPadVal)
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
|
|
pb = pbDecryptedData;
|
|
|
|
//
|
|
// Have to do the following incantation since otherwise we'd likely
|
|
// fault on an unaligned fetch.
|
|
//
|
|
// Cache account name size & position.
|
|
//
|
|
|
|
CopyMemory(&cbAccount, pb, sizeof(cbAccount));
|
|
pbAccount = pb + sizeof(cbAccount);
|
|
pb = pbAccount + cbAccount;
|
|
|
|
if (((DWORD)(pb - pbDecryptedData) > cbDecryptedData) || // Check size.
|
|
(cbAccount > (MAX_USERNAME * sizeof(WCHAR))))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Cache domain name size & position.
|
|
//
|
|
|
|
CopyMemory(&cbDomain, pb, sizeof(cbDomain));
|
|
pbDomain = pb + sizeof(cbDomain);
|
|
pb = pbDomain + cbDomain;
|
|
|
|
if (((DWORD)(pb - pbDecryptedData) > cbDecryptedData) || // Check size.
|
|
(cbDomain > (MAX_DOMAINNAME * sizeof(WCHAR))))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Cache password size & position.
|
|
//
|
|
|
|
CopyMemory(&cbPassword, pb, sizeof(cbPassword));
|
|
pbPassword = pb + sizeof(cbPassword);
|
|
// In the IE 5 release of the Task Scheduler, a NULL password was denoted
|
|
// by a size of 0xFFFFFFFF in the SAC. The following check lets us read
|
|
// databases created by the IE 5 TS.
|
|
if (cbPassword == NULL_PASSWORD_SIZE)
|
|
{
|
|
fIsPasswordNull = TRUE;
|
|
cbPassword = 0;
|
|
}
|
|
pb = pbPassword + cbPassword;
|
|
|
|
if (((DWORD)(pb - pbDecryptedData) > cbDecryptedData) || // Check size.
|
|
(cbPassword > (MAX_PASSWORD * sizeof(WCHAR))))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_INVALID_DATA);
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Finally, copy the return data.
|
|
//
|
|
|
|
CopyMemory(pjc->wszAccount, pbAccount, cbAccount);
|
|
*(WCHAR *)(((BYTE *)pjc->wszAccount) + cbAccount) = L'\0';
|
|
pjc->ccAccount = cbAccount / sizeof(WCHAR);
|
|
|
|
CopyMemory(pjc->wszDomain, pbDomain, cbDomain);
|
|
*(WCHAR *)(((BYTE *)pjc->wszDomain) + cbDomain) = L'\0';
|
|
pjc->ccDomain = cbDomain / sizeof(WCHAR);
|
|
|
|
CopyMemory(pjc->wszPassword, pbPassword, cbPassword);
|
|
*(WCHAR *)(((BYTE *)pjc->wszPassword) + cbPassword) = L'\0';
|
|
pjc->ccPassword = cbPassword / sizeof(WCHAR);
|
|
|
|
pjc->fIsPasswordNull = fIsPasswordNull;
|
|
|
|
ErrorExit:
|
|
if (!fDecryptInPlace) delete pbDecryptedData;
|
|
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: CredentialLookupAndAccessCheck
|
|
//
|
|
// Synopsis:
|
|
//
|
|
// Arguments: [hCSP] --
|
|
// [pSid] --
|
|
// [cbSAC] --
|
|
// [pbSAC] --
|
|
// [pCredentialIndex] --
|
|
// [rgbHashedSid] --
|
|
// [pcbCredential] --
|
|
// [ppbCredential] --
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
STATIC HRESULT
|
|
CredentialLookupAndAccessCheck(
|
|
HCRYPTPROV hCSP,
|
|
PSID pSid,
|
|
DWORD cbSAC,
|
|
BYTE * pbSAC,
|
|
DWORD * pCredentialIndex,
|
|
BYTE rgbHashedSid[],
|
|
DWORD * pcbCredential,
|
|
BYTE ** ppbCredential)
|
|
{
|
|
HRESULT hr;
|
|
|
|
// Either pSid or rgbHashedSid must be specified.
|
|
//
|
|
schAssert(rgbHashedSid != NULL && (pSid != NULL || *rgbHashedSid));
|
|
|
|
if (pSid != NULL)
|
|
{
|
|
if (!IsValidSid(pSid))
|
|
{
|
|
CHECK_HRESULT(E_UNEXPECTED);
|
|
return(E_UNEXPECTED);
|
|
}
|
|
|
|
hr = HashSid(hCSP, pSid, rgbHashedSid);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
return(hr);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Find the credential in the SAC associated with the account sid. The
|
|
// hashed account sid is utilized as a SAC database key.
|
|
//
|
|
|
|
DWORD cbEncryptedData;
|
|
BYTE * pbEncryptedData;
|
|
|
|
hr = SACFindCredential(rgbHashedSid,
|
|
cbSAC,
|
|
pbSAC,
|
|
pCredentialIndex,
|
|
&cbEncryptedData,
|
|
&pbEncryptedData);
|
|
|
|
if (hr == S_OK)
|
|
{
|
|
//
|
|
// Found it. Does the caller have access to this credential?
|
|
//
|
|
|
|
BYTE * pbCredential = pbEncryptedData - HASH_DATA_SIZE;
|
|
|
|
if (CredentialAccessCheck(hCSP, pbCredential))
|
|
{
|
|
// Update out ptrs.
|
|
//
|
|
*ppbCredential = pbCredential;
|
|
CopyMemory(pcbCredential,
|
|
*ppbCredential - sizeof(*pcbCredential),
|
|
sizeof(*pcbCredential));
|
|
}
|
|
else
|
|
{
|
|
hr = HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED);
|
|
}
|
|
}
|
|
else if (hr == S_FALSE)
|
|
{
|
|
//
|
|
// Didn't find the credential.
|
|
//
|
|
|
|
hr = SCHED_E_ACCOUNT_INFORMATION_NOT_SET;
|
|
}
|
|
|
|
return(hr);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: CredentialAccessCheck
|
|
//
|
|
// Synopsis: Determine if the RPC client has access to the credential
|
|
// indicated.
|
|
//
|
|
// Arguments: [hCSP] -- CSP provider handle (for use with
|
|
// Crypto API).
|
|
// [pbCredentialIdentity] -- Credential identity.
|
|
//
|
|
// Returns: TRUE -- RPC client has permission to access this credential.
|
|
// FALSE -- RPC client doesn't have credential access or an
|
|
// unexpected error occurred.
|
|
//
|
|
// Notes: ** Important **
|
|
//
|
|
// Thread impersonation is performed in this routine via
|
|
// RpcImpersonateClient; therefore, it is assumed only RPC
|
|
// threads enter it.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
STATIC BOOL
|
|
CredentialAccessCheck(
|
|
HCRYPTPROV hCSP,
|
|
BYTE * pbCredentialIdentity)
|
|
{
|
|
RPC_STATUS RpcStatus;
|
|
|
|
//
|
|
// Impersonate the caller.
|
|
//
|
|
|
|
if ((RpcStatus = RpcImpersonateClient(NULL)) != RPC_S_OK)
|
|
{
|
|
CHECK_HRESULT(RpcStatus);
|
|
return(FALSE);
|
|
}
|
|
|
|
HANDLE hToken;
|
|
BOOL bRet;
|
|
|
|
if (!OpenThreadToken(GetCurrentThread(),
|
|
TOKEN_QUERY, // Desired access.
|
|
TRUE, // Open as self.
|
|
&hToken))
|
|
{
|
|
CHECK_HRESULT(_HRESULT_FROM_WIN32(GetLastError()));
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// End impersonation, but don't close the token yet.
|
|
// (We must not be impersonating when we call HashSid, which is
|
|
// called by MatchThreadCallerAgainstCredential.)
|
|
//
|
|
if ((RpcStatus = RpcRevertToSelf()) != RPC_S_OK)
|
|
{
|
|
ERR_OUT("RpcRevertToSelf", RpcStatus);
|
|
schAssert(!"RpcRevertToSelf failed");
|
|
}
|
|
|
|
//
|
|
// Does the thread caller's hashed SID match the credential identity.
|
|
// If so, the caller's account is the same as that specified in the
|
|
// credentials.
|
|
//
|
|
|
|
if (!(bRet = MatchThreadCallerAgainstCredential(hCSP,
|
|
hToken,
|
|
pbCredentialIdentity)))
|
|
{
|
|
//
|
|
// Nope. Thread caller account/credential account mismatch.
|
|
// Is the caller an administrator?
|
|
//
|
|
|
|
bRet = IsThreadCallerAnAdmin(hToken);
|
|
}
|
|
|
|
CloseHandle(hToken);
|
|
|
|
return(bRet);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: MatchThreadCallerAgainstCredential
|
|
//
|
|
// Synopsis: Hash the user SID of the thread indicated and compare it
|
|
// against the credential identity passed. A credential identity
|
|
// is the hashed SID of the associated account.
|
|
//
|
|
// Arguments: [hCSP] -- CSP provider handle (for use with
|
|
// Cryto API).
|
|
// [hThreadToken] -- Obtain the user SID from this
|
|
// thread.
|
|
// [pbCredentialIdentity] -- Matched credential identity.
|
|
//
|
|
// Returns: TRUE -- Match
|
|
// FALSE -- No match or an error occurred.
|
|
//
|
|
// Notes: None.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
STATIC BOOL
|
|
MatchThreadCallerAgainstCredential(
|
|
HCRYPTPROV hCSP,
|
|
HANDLE hThreadToken,
|
|
BYTE * pbCredentialIdentity)
|
|
{
|
|
BYTE rgbTokenInformation[USER_TOKEN_STACK_BUFFER_SIZE];
|
|
TOKEN_USER * pTokenUser = (TOKEN_USER *)rgbTokenInformation;
|
|
DWORD cbReturnLength;
|
|
DWORD Status = ERROR_SUCCESS;
|
|
|
|
if (!GetTokenInformation(hThreadToken,
|
|
TokenUser,
|
|
pTokenUser,
|
|
USER_TOKEN_STACK_BUFFER_SIZE,
|
|
&cbReturnLength))
|
|
{
|
|
//
|
|
// Buffer space should have been sufficient. Check if we goofed.
|
|
//
|
|
|
|
schAssert(GetLastError() != ERROR_INSUFFICIENT_BUFFER);
|
|
CHECK_HRESULT(_HRESULT_FROM_WIN32(GetLastError()));
|
|
return(FALSE);
|
|
}
|
|
|
|
//
|
|
// Hash the user's SID.
|
|
//
|
|
|
|
BYTE rgbHashedSid[HASH_DATA_SIZE] = { 0 };
|
|
|
|
if (SUCCEEDED(HashSid(hCSP, pTokenUser->User.Sid, rgbHashedSid)))
|
|
{
|
|
if (memcmp(pbCredentialIdentity, rgbHashedSid, HASH_DATA_SIZE) == 0)
|
|
{
|
|
return(TRUE);
|
|
}
|
|
else
|
|
{
|
|
CHECK_HRESULT(HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED));
|
|
}
|
|
}
|
|
|
|
return(FALSE);
|
|
}
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: ScavengeSASecurityDBase
|
|
//
|
|
// Synopsis: Enumerate the jobs folder and remove identities in the SAI
|
|
// for which no current jobs hash to. Note, SAC credentials
|
|
// are also removed if the removed identity was the last to be
|
|
// associated with it.
|
|
//
|
|
// Arguments: None.
|
|
//
|
|
// Returns: None.
|
|
//
|
|
// Notes: Should read of any job fail, for any reason, the scavenge
|
|
// task is abandoned. Reason is, if the removal process was
|
|
// to continue anyway, credentials might be removed for existent
|
|
// jobs.
|
|
//
|
|
// The service state is checked periodically as this could
|
|
// potentially be a lengthy routine time-wise. Bail as soon
|
|
// as service stop or service stop pending is detected.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
void
|
|
ScavengeSASecurityDBase(void)
|
|
{
|
|
TCHAR tszSearchPath[MAX_PATH + 1];
|
|
BYTE rgbIdentity[HASH_DATA_SIZE];
|
|
WIN32_FIND_DATA fd;
|
|
JOB_IDENTITY_SET * rgIdentitySet = NULL;
|
|
HRESULT hr = S_OK;
|
|
HANDLE hFileEnum;
|
|
DWORD dwZero = 0;
|
|
DWORD i, j;
|
|
DWORD iConcatenation;
|
|
DWORD dwRet;
|
|
DWORD dwSetCount = 0;
|
|
DWORD dwSetSubCount;
|
|
DWORD cbIdentitySetArraySize;
|
|
BYTE * pbSet;
|
|
BOOL fDirty = FALSE;
|
|
|
|
//
|
|
// Build the enumeration search path.
|
|
//
|
|
|
|
StringCchCopy(tszSearchPath, MAX_PATH + 1, g_TasksFolderInfo.ptszPath);
|
|
StringCchCat(tszSearchPath, MAX_PATH + 1, EXTENSION_WILDCARD TSZ_JOB);
|
|
|
|
//
|
|
// Initialize the enumeration.
|
|
//
|
|
|
|
if ((hFileEnum = FindFirstFile(tszSearchPath,
|
|
&fd)) == INVALID_HANDLE_VALUE)
|
|
{
|
|
//
|
|
// Either no jobs, or an error occurred.
|
|
//
|
|
|
|
dwRet = GetLastError();
|
|
|
|
if (dwRet == ERROR_FILE_NOT_FOUND)
|
|
{
|
|
EnterCriticalSection(&gcsSSCritSection);
|
|
|
|
//
|
|
// No files found. Reset SAI & SAC by writing four bytes of
|
|
// zeros into each.
|
|
//
|
|
hr = WriteSecurityDBase(sizeof(dwZero), (BYTE *)&dwZero,
|
|
sizeof(dwZero), (BYTE *)&dwZero);
|
|
CHECK_HRESULT(hr);
|
|
|
|
LeaveCriticalSection(&gcsSSCritSection);
|
|
}
|
|
else
|
|
{
|
|
CHECK_HRESULT(_HRESULT_FROM_WIN32(dwRet));
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
DWORD cbSAI;
|
|
DWORD cbSAC;
|
|
BYTE * pbSAI = NULL;
|
|
BYTE * pbSAC = NULL;
|
|
BYTE * pbSAIEnd;
|
|
BYTE * pb;
|
|
HCRYPTPROV hCSP = NULL;
|
|
|
|
//
|
|
// Check if the service is stopping.
|
|
//
|
|
|
|
if (IsServiceStopping())
|
|
{
|
|
return;
|
|
}
|
|
|
|
EnterCriticalSection(&gcsSSCritSection);
|
|
|
|
hr = ReadSecurityDBase(&cbSAI, &pbSAI, &cbSAC, &pbSAC);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
if (cbSAI <= SAI_HEADER_SIZE)
|
|
{
|
|
//
|
|
// Database empty.
|
|
//
|
|
|
|
hr = S_OK;
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Some background first. The SAI consists of an array of arrays. The
|
|
// first dimension represents the set of job identities per credential
|
|
// in the SAC. SAI/SAC indices are associative in this case. The set of
|
|
// job identities at SAI row[n] correspond to the credential at SAC
|
|
// row[n].
|
|
//
|
|
// We need to construct an SAI pending deletion data structure. It will
|
|
// consist of an array of JOB_IDENTITY_SET structures, in which each
|
|
// structure refers to an array of pointers to job identities in the
|
|
// SAI (literally indexing the SAI).
|
|
//
|
|
// Once the data structure is built and initialized, we'll enumerate the
|
|
// jobs in the local tasks folder. For each job found, the corresponding
|
|
// job identity pointer in the job identity set array will be set to NULL.
|
|
// Upon completion of the enumeration, the non-NULL job identity ptr
|
|
// entries within the job identity set array refer to non-existent jobs.
|
|
// The job identitites these entries refer to are removed from the SAI,
|
|
// and the associated credential in the SAC, if there are no longer
|
|
// entries in the SAI associated with it.
|
|
//
|
|
// First, allocate the array.
|
|
//
|
|
|
|
pb = pbSAI + USN_SIZE;
|
|
|
|
CopyMemory(&dwSetCount, pb, sizeof(dwSetCount));
|
|
pb += sizeof(dwSetCount);
|
|
|
|
cbIdentitySetArraySize = dwSetCount * sizeof(JOB_IDENTITY_SET);
|
|
|
|
rgIdentitySet = (JOB_IDENTITY_SET *)LocalAlloc(LMEM_FIXED,
|
|
cbIdentitySetArraySize);
|
|
|
|
if (rgIdentitySet == NULL)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
SecureZeroMemory(rgIdentitySet, cbIdentitySetArraySize);
|
|
|
|
pb = pbSAI + SAI_HEADER_SIZE;
|
|
pbSAIEnd = pbSAI + cbSAI;
|
|
|
|
//
|
|
// Check if the service is stopping.
|
|
//
|
|
|
|
if (IsServiceStopping())
|
|
{
|
|
hr = S_OK;
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Now allocate, intialize individual identity sets.
|
|
//
|
|
|
|
for (i = 0; i < dwSetCount; i++)
|
|
{
|
|
//
|
|
// Check boundary.
|
|
//
|
|
|
|
if ((pb + sizeof(dwSetSubCount)) > pbSAIEnd)
|
|
{
|
|
ASSERT_SECURITY_DBASE_CORRUPT();
|
|
hr = SCHED_E_ACCOUNT_DBASE_CORRUPT;
|
|
goto ErrorExit;
|
|
}
|
|
|
|
CopyMemory(&dwSetSubCount, pb, sizeof(dwSetSubCount));
|
|
pb += sizeof(dwSetSubCount);
|
|
|
|
BYTE ** rgpbIdentity = (BYTE **)LocalAlloc(
|
|
LMEM_FIXED,
|
|
sizeof(BYTE *) * dwSetSubCount);
|
|
|
|
if (rgpbIdentity == NULL)
|
|
{
|
|
hr = E_OUTOFMEMORY;
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
rgIdentitySet[i].pbSetStart = pb;
|
|
rgIdentitySet[i].dwSetSubCount = dwSetSubCount;
|
|
rgIdentitySet[i].rgpbIdentity = rgpbIdentity;
|
|
|
|
for (j = 0; j < dwSetSubCount; j++)
|
|
{
|
|
rgpbIdentity[j] = pb;
|
|
pb += HASH_DATA_SIZE;
|
|
|
|
if (pb > pbSAIEnd)
|
|
{
|
|
ASSERT_SECURITY_DBASE_CORRUPT();
|
|
hr = SCHED_E_ACCOUNT_DBASE_CORRUPT;
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Check if the service is stopping.
|
|
//
|
|
|
|
if (IsServiceStopping())
|
|
{
|
|
hr = S_OK;
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Enumerate job objects in the task's folder directory. Set
|
|
// corresponding job identity ptrs in the job identity set array to
|
|
// NULL for existent jobs.
|
|
//
|
|
|
|
//
|
|
// First, obtain a provider handle to the CSP (for use with Crypto API).
|
|
//
|
|
|
|
hr = GetCSPHandle(&hCSP);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Must concatenate the filename returned from the enumeration onto
|
|
// the folder path.
|
|
//
|
|
|
|
StringCchCopy(tszSearchPath, MAX_PATH + 1, g_TasksFolderInfo.ptszPath);
|
|
iConcatenation = lstrlenW(g_TasksFolderInfo.ptszPath);
|
|
tszSearchPath[iConcatenation++] = L'\\';
|
|
|
|
for (;;)
|
|
{
|
|
//
|
|
// Append the filename to the folder path.
|
|
//
|
|
|
|
tszSearchPath[iConcatenation] = L'\0';
|
|
StringCchCat(tszSearchPath, MAX_PATH + 1, fd.cFileName);
|
|
|
|
//
|
|
// Hash the job into a unique identity.
|
|
//
|
|
|
|
hr = HashJobIdentity(hCSP, tszSearchPath, rgbIdentity);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
//
|
|
// Must bail if the hash fails. If this is ignored, one, or more,
|
|
// identities may be removed for existent jobs - not good.
|
|
//
|
|
// TBD : Log error.
|
|
//
|
|
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Does an identity exist in the SAI for this job? If so, NULL out
|
|
// the corresponding entry in the job identity set array.
|
|
//
|
|
|
|
DWORD CredentialIndex;
|
|
BYTE * pbIdentity;
|
|
|
|
hr = SAIFindIdentity(rgbIdentity,
|
|
cbSAI,
|
|
pbSAI,
|
|
&CredentialIndex,
|
|
NULL,
|
|
&pbIdentity,
|
|
NULL,
|
|
&pbSet);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
|
|
if (pbIdentity != NULL)
|
|
{
|
|
for (i = 0; i < dwSetCount; i++)
|
|
{
|
|
for (j = 0; j < rgIdentitySet[i].dwSetSubCount; j++)
|
|
{
|
|
if (pbIdentity == rgIdentitySet[i].rgpbIdentity[j])
|
|
{
|
|
rgIdentitySet[i].rgpbIdentity[j] = NULL;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!FindNextFile(hFileEnum, &fd))
|
|
{
|
|
dwRet = GetLastError();
|
|
|
|
if (dwRet == ERROR_NO_MORE_FILES)
|
|
{
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
hr = _HRESULT_FROM_WIN32(GetLastError());
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Check if the service is stopping.
|
|
//
|
|
|
|
if (IsServiceStopping())
|
|
{
|
|
hr = S_OK;
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Non-NULL entries in the identity set array refer to job identities in
|
|
// the SAI to be removed. Mark them for removal.
|
|
//
|
|
|
|
for (i = 0; i < dwSetCount; i++)
|
|
{
|
|
if (rgIdentitySet[i].rgpbIdentity != NULL)
|
|
{
|
|
dwSetSubCount = rgIdentitySet[i].dwSetSubCount;
|
|
|
|
for (j = 0; j < dwSetSubCount; j++)
|
|
{
|
|
if (rgIdentitySet[i].rgpbIdentity[j] != NULL)
|
|
{
|
|
MARK_DELETED_ENTRY(rgIdentitySet[i].rgpbIdentity[j]);
|
|
rgIdentitySet[i].dwSetSubCount--;
|
|
fDirty = TRUE;
|
|
|
|
if (rgIdentitySet[i].dwSetSubCount == 0)
|
|
{
|
|
//
|
|
// Last identity in set. Mark associated SAC
|
|
// credential for removal also.
|
|
//
|
|
|
|
DWORD cbCredential;
|
|
BYTE * pbCredential;
|
|
|
|
hr = SACIndexCredential(i,
|
|
cbSAC,
|
|
pbSAC,
|
|
&cbCredential,
|
|
&pbCredential);
|
|
|
|
if (hr == S_FALSE)
|
|
{
|
|
//
|
|
// This should *never* happen. Consider the
|
|
// database corrupt if so.
|
|
//
|
|
|
|
ASSERT_SECURITY_DBASE_CORRUPT();
|
|
hr = SCHED_E_ACCOUNT_DBASE_CORRUPT;
|
|
goto ErrorExit;
|
|
}
|
|
else if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
goto ErrorExit;
|
|
}
|
|
else
|
|
{
|
|
MARK_DELETED_ENTRY(pbCredential);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Check if the service is stopping.
|
|
//
|
|
|
|
if (IsServiceStopping())
|
|
{
|
|
hr = S_OK;
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Removed entries marked for deletion.
|
|
//
|
|
|
|
if (fDirty)
|
|
{
|
|
hr = SAICoalesceDeletedEntries(&cbSAI, &pbSAI);
|
|
CHECK_HRESULT(hr);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = SACCoalesceDeletedEntries(&cbSAC, &pbSAC);
|
|
CHECK_HRESULT(hr);
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
goto ErrorExit;
|
|
}
|
|
|
|
//
|
|
// Finally, persist the changes made to the SAI & SAC.
|
|
//
|
|
hr = WriteSecurityDBase(cbSAI, pbSAI, cbSAC, pbSAC);
|
|
CHECK_HRESULT(hr);
|
|
}
|
|
|
|
ErrorExit:
|
|
//
|
|
// Deallocate data structures allocated above.
|
|
//
|
|
for (i = 0; i < dwSetCount; i++)
|
|
{
|
|
if (rgIdentitySet[i].rgpbIdentity != NULL)
|
|
{
|
|
LocalFree(rgIdentitySet[i].rgpbIdentity);
|
|
}
|
|
}
|
|
|
|
if (rgIdentitySet != NULL) LocalFree(rgIdentitySet);
|
|
if (pbSAI != NULL) LocalFree(pbSAI);
|
|
if (pbSAC != NULL) LocalFree(pbSAC);
|
|
|
|
if (hFileEnum != INVALID_HANDLE_VALUE) FindClose(hFileEnum);
|
|
|
|
if (hCSP != NULL) CloseCSPHandle(hCSP);
|
|
//
|
|
// Log an error & rest the SA security dbases SAI & SAC if corruption
|
|
// is detected.
|
|
//
|
|
|
|
if (hr == SCHED_E_ACCOUNT_DBASE_CORRUPT)
|
|
{
|
|
//
|
|
// Log an error.
|
|
//
|
|
|
|
LogServiceError(IERR_SECURITY_DBASE_CORRUPTION, 0,
|
|
IDS_HELP_HINT_DBASE_CORRUPT);
|
|
|
|
//
|
|
// Reset SAI & SAC by writing four bytes of zeros into each.
|
|
// Ignore the return code. No recourse if this fails.
|
|
//
|
|
DWORD dwZero = 0;
|
|
WriteSecurityDBase(sizeof(dwZero), (BYTE *)&dwZero, sizeof(dwZero),
|
|
(BYTE *)&dwZero);
|
|
}
|
|
|
|
LeaveCriticalSection(&gcsSSCritSection);
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: SchedUPNToAccountName
|
|
//
|
|
// Synopsis: Converts a UPN to an Account Name
|
|
//
|
|
// Arguments: lpUPN - The UPN
|
|
// ppAccountName - Pointer to the location to create/copy the account name
|
|
//
|
|
// Returns: NO_ERROR - Success (ppAccountName contains the converted UPN)
|
|
// Any other Win32 error - error at some stage of conversion
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
|
|
DWORD
|
|
SchedUPNToAccountName(
|
|
IN LPCWSTR lpUPN,
|
|
OUT LPWSTR *ppAccountName
|
|
)
|
|
{
|
|
DWORD dwError;
|
|
HANDLE hDS;
|
|
PDS_NAME_RESULT pdsResult;
|
|
|
|
schAssert(ppAccountName != NULL);
|
|
|
|
schDebugOut((DEB_TRACE, "SchedUPNToAccountName: Converting \"%ws\"\n", lpUPN));
|
|
|
|
//
|
|
// Get a binding handle to the DS
|
|
//
|
|
dwError = DsBind(NULL, NULL, &hDS);
|
|
|
|
if (dwError != NO_ERROR)
|
|
{
|
|
schDebugOut((DEB_ERROR, "SchedUPNToAccountName: DsBind failed %d\n", dwError));
|
|
return dwError;
|
|
}
|
|
|
|
dwError = DsCrackNames(hDS, // Handle to the DS
|
|
DS_NAME_NO_FLAGS, // No parsing flags
|
|
DS_USER_PRINCIPAL_NAME, // We have a UPN
|
|
DS_NT4_ACCOUNT_NAME, // We want Domain\User
|
|
1, // Number of names to crack
|
|
&lpUPN, // Array of name(s)
|
|
&pdsResult); // Filled in by API
|
|
|
|
if (dwError != NO_ERROR)
|
|
{
|
|
schDebugOut((DEB_ERROR, "SchedUPNToAccountName: DsCrackNames failed %d\n", dwError));
|
|
|
|
DsUnBind(&hDS);
|
|
return dwError;
|
|
}
|
|
|
|
schAssert(pdsResult->cItems == 1);
|
|
schAssert(pdsResult->rItems != NULL);
|
|
|
|
if (pdsResult->rItems[0].status == DS_NAME_ERROR_DOMAIN_ONLY)
|
|
{
|
|
//
|
|
// Couldn't crack the name but we got the name of
|
|
// the domain where it is -- let's try it
|
|
//
|
|
DsUnBind(&hDS);
|
|
|
|
schAssert(pdsResult->rItems[0].pDomain != NULL);
|
|
|
|
schDebugOut((DEB_TRACE, "Retrying DsBind on domain %ws\n", pdsResult->rItems[0].pDomain));
|
|
|
|
dwError = DsBind(NULL, pdsResult->rItems[0].pDomain, &hDS);
|
|
|
|
//
|
|
// Free up the structure holding the old info
|
|
//
|
|
DsFreeNameResult(pdsResult);
|
|
|
|
if (dwError != NO_ERROR)
|
|
{
|
|
schDebugOut((DEB_ERROR, "SchedUPNToAccountName: DsBind #2 failed %d\n", dwError));
|
|
return dwError;
|
|
}
|
|
|
|
dwError = DsCrackNames(hDS, // Handle to the DS
|
|
DS_NAME_NO_FLAGS, // No parsing flags
|
|
DS_USER_PRINCIPAL_NAME, // We have a UPN
|
|
DS_NT4_ACCOUNT_NAME, // We want Domain\User
|
|
1, // Number of names to crack
|
|
&lpUPN, // Array of name(s)
|
|
&pdsResult); // Filled in by API
|
|
|
|
if (dwError != NO_ERROR)
|
|
{
|
|
schDebugOut((DEB_ERROR, "SchedUPNToAccountName: DsCrackNames #2 failed %d\n", dwError));
|
|
|
|
DsUnBind(&hDS);
|
|
return dwError;
|
|
}
|
|
|
|
schAssert(pdsResult->cItems == 1);
|
|
schAssert(pdsResult->rItems != NULL);
|
|
}
|
|
|
|
if (pdsResult->rItems[0].status != DS_NAME_NO_ERROR)
|
|
{
|
|
schDebugOut((DEB_ERROR, "SchedUPNToAccountName: DsCrackNames failure (status %#x)\n", pdsResult->rItems[0].status));
|
|
|
|
//
|
|
// DS errors don't map to Win32 errors -- this is the best we can do
|
|
//
|
|
dwError = SCHED_E_ACCOUNT_NAME_NOT_FOUND;
|
|
}
|
|
else
|
|
{
|
|
schDebugOut((DEB_TRACE, "SchedUPNToAccountName: Got \"%ws\"\n",
|
|
pdsResult->rItems[0].pName));
|
|
|
|
size_t cchBuff = wcslen(pdsResult->rItems[0].pName) + 1;
|
|
*ppAccountName = new WCHAR[cchBuff];
|
|
|
|
if (*ppAccountName != NULL)
|
|
{
|
|
StringCchCopy(*ppAccountName, cchBuff, pdsResult->rItems[0].pName);
|
|
}
|
|
else
|
|
{
|
|
dwError = GetLastError();
|
|
schDebugOut((DEB_ERROR, "SchedUPNToAccountName: LocalAlloc failed %d\n", dwError));
|
|
}
|
|
}
|
|
|
|
DsUnBind(&hDS);
|
|
DsFreeNameResult(pdsResult);
|
|
return dwError;
|
|
}
|
|
|
|
|
|
//+---------------------------------------------------------------------------
|
|
//
|
|
// Function: LookupAccountNameWrap
|
|
//
|
|
// Synopsis: BUGBUG This is a workaround for bug 254102 - LookupAccountName
|
|
// doesn't work when the DC can't be reached, even for the
|
|
// currently logged-on user, and even though LookupAccountSid
|
|
// does work. Remove this function when that bug is fixed.
|
|
//
|
|
// Arguments: Same as LookupAccountName - but cbSid and cbReferencedDomainName
|
|
// are assumed to be large enough, and peUse is ignored.
|
|
//
|
|
// Returns: Same as LookupAccountName.
|
|
//
|
|
//----------------------------------------------------------------------------
|
|
BOOL
|
|
LookupAccountNameWrap(
|
|
LPCTSTR lpSystemName, // address of string for system name
|
|
LPCTSTR lpAccountName, // address of string for account name
|
|
PSID Sid, // address of security identifier
|
|
LPDWORD cbSid, // address of size of security identifier
|
|
LPTSTR ReferencedDomainName,
|
|
// address of string for referenced domain
|
|
LPDWORD cbReferencedDomainName,
|
|
// address of size of domain string
|
|
PSID_NAME_USE peUse // address of SID-type indicator
|
|
)
|
|
{
|
|
//
|
|
// See if the account name matches the account name we cached
|
|
//
|
|
|
|
EnterCriticalSection(gUserLogonInfo.CritSection);
|
|
|
|
if (gUserLogonInfo.DomainUserName != NULL &&
|
|
lstrcmpi(gUserLogonInfo.DomainUserName, lpAccountName) == 0)
|
|
{
|
|
//
|
|
// The names match. Return the cached SID.
|
|
//
|
|
schDebugOut((DEB_TRACE, "Using cached SID for user \"%ws\"\n", lpAccountName));
|
|
if (CopySid(*cbSid, Sid, gUserLogonInfo.Sid))
|
|
{
|
|
LeaveCriticalSection(gUserLogonInfo.CritSection);
|
|
|
|
//
|
|
// Copy the ReferencedDomainName from the account name
|
|
//
|
|
PCWCH pchSlash = wcschr(lpAccountName, L'\\');
|
|
schAssert(pchSlash != NULL);
|
|
DWORD DomainLen = (DWORD)(pchSlash - lpAccountName);
|
|
schAssert(DomainLen+1 <= *cbReferencedDomainName);
|
|
wcsncpy(ReferencedDomainName, lpAccountName, DomainLen);
|
|
ReferencedDomainName[DomainLen] = L'\0';
|
|
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
schAssert(0);
|
|
CHECK_HRESULT(HRESULT_FROM_WIN32(GetLastError()));
|
|
}
|
|
}
|
|
|
|
LeaveCriticalSection(gUserLogonInfo.CritSection);
|
|
|
|
return LookupAccountName(
|
|
lpSystemName,
|
|
lpAccountName,
|
|
Sid,
|
|
cbSid,
|
|
ReferencedDomainName,
|
|
cbReferencedDomainName,
|
|
peUse
|
|
);
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: ComputeJobSignature
|
|
//
|
|
// Synopsis: Creates a signature for the job file
|
|
//
|
|
// Arguments: [pwszFileName] - name of job file
|
|
// [pSignature] - block in which to store the signature. Must
|
|
// be at least SIGNATURE_SIZE bytes long.
|
|
// [dwHashMethod - dword value indicating which hash method to use;
|
|
// Default if not specified is the latest method.
|
|
//
|
|
// Returns: HRESULT
|
|
//
|
|
// Notes: The job must have been saved to disk before calling this
|
|
// function.
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
HRESULT
|
|
ComputeJobSignature(
|
|
LPCWSTR pwszFileName,
|
|
LPBYTE pbSignature,
|
|
DWORD dwHashMethod /* = 1 */
|
|
)
|
|
{
|
|
HCRYPTPROV hCSP;
|
|
|
|
HRESULT hr = GetCSPHandle(&hCSP);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = HashJobIdentity(hCSP, pwszFileName, pbSignature, dwHashMethod);
|
|
CloseCSPHandle(hCSP);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CJob::Sign
|
|
//
|
|
// Synopsis: Computes and sets the job's signature
|
|
//
|
|
// Arguments: None
|
|
//
|
|
// Notes: The job must have been written to disk before calling this method
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
HRESULT
|
|
CJob::Sign(
|
|
VOID
|
|
)
|
|
{
|
|
BYTE rgbSignature[SIGNATURE_SIZE];
|
|
HRESULT hr = ComputeJobSignature(m_ptszFileName, rgbSignature);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
return hr;
|
|
}
|
|
|
|
hr = _SetSignature(rgbSignature);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CJob::VerifySignature
|
|
//
|
|
// Synopsis: Compares the job file's hash to the one stored in the file
|
|
//
|
|
// Arguments: None
|
|
//
|
|
// Notes: The job must have been written to disk before calling this method
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
BOOL
|
|
CJob::VerifySignature(
|
|
DWORD dwHashMethod /* = 1 */
|
|
) const
|
|
{
|
|
if (m_pbSignature == NULL)
|
|
{
|
|
CHECK_HRESULT(SCHED_E_ACCOUNT_INFORMATION_NOT_SET);
|
|
return FALSE;
|
|
}
|
|
|
|
BYTE rgbSignature[SIGNATURE_SIZE];
|
|
HRESULT hr = ComputeJobSignature(m_ptszFileName, rgbSignature, dwHashMethod);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
CHECK_HRESULT(hr);
|
|
return FALSE;
|
|
}
|
|
|
|
if (memcmp(m_pbSignature, rgbSignature, SIGNATURE_SIZE) != 0)
|
|
{
|
|
CHECK_HRESULT(E_ACCESSDENIED);
|
|
return(FALSE);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::AddAtJobWithHash
|
|
//
|
|
// Synopsis: create a downlevel job
|
|
//
|
|
// Arguments: [At] - reference to an AT_INFO struct
|
|
// [pID] - returns the new ID (optional, can be NULL)
|
|
//
|
|
// Returns: HRESULTS
|
|
//
|
|
// Notes: This method is not exposed to external clients, thus it is not
|
|
// part of a public interface.
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CSchedule::AddAtJobWithHash(const AT_INFO &At, DWORD * pID)
|
|
{
|
|
TRACE(CSchedule, AddAtJob);
|
|
HRESULT hr = S_OK;
|
|
CJob *pJob;
|
|
WCHAR wszName[MAX_PATH + 1];
|
|
WCHAR wszID[SCH_SMBUF_LEN];
|
|
|
|
hr = AddAtJobCommon(At, pID, &pJob, wszName, MAX_PATH + 1, wszID);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("AddAtJobWithHash: AddAtJobCommon", hr);
|
|
return hr;
|
|
}
|
|
|
|
hr = AuditATJob(At, wszName);
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("AddAtJobWithHash: AuditATJob", hr);
|
|
}
|
|
|
|
//
|
|
// Now get a signature for the job file and add it to the job object
|
|
//
|
|
hr = pJob->Sign();
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("AddAtJobWithHash: Sign", hr);
|
|
pJob->Release();
|
|
return hr;
|
|
}
|
|
|
|
hr = pJob->SaveWithRetry(pJob->GetFileName(),
|
|
FALSE,
|
|
SAVEP_VARIABLE_LENGTH_DATA |
|
|
SAVEP_PRESERVE_NET_SCHEDULE);
|
|
|
|
//
|
|
// Free the job object.
|
|
//
|
|
pJob->Release();
|
|
|
|
//
|
|
// Return the new job's ID and increment the ID counter
|
|
//
|
|
if (pID != NULL)
|
|
{
|
|
*pID = m_dwNextID;
|
|
}
|
|
|
|
hr = IncrementAndSaveID();
|
|
|
|
return hr;
|
|
}
|