diff options
Diffstat (limited to 'src/libs/dutil/WixToolset.DUtil/acl2util.cpp')
| -rw-r--r-- | src/libs/dutil/WixToolset.DUtil/acl2util.cpp | 135 |
1 files changed, 135 insertions, 0 deletions
diff --git a/src/libs/dutil/WixToolset.DUtil/acl2util.cpp b/src/libs/dutil/WixToolset.DUtil/acl2util.cpp new file mode 100644 index 00000000..598f12e7 --- /dev/null +++ b/src/libs/dutil/WixToolset.DUtil/acl2util.cpp | |||
| @@ -0,0 +1,135 @@ | |||
| 1 | // Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. | ||
| 2 | |||
| 3 | #include "precomp.h" | ||
| 4 | |||
| 5 | // Exit macros | ||
| 6 | #define AclExitOnLastError(x, s, ...) ExitOnLastErrorSource(DUTIL_SOURCE_ACLUTIL, x, s, __VA_ARGS__) | ||
| 7 | #define AclExitOnLastErrorDebugTrace(x, s, ...) ExitOnLastErrorDebugTraceSource(DUTIL_SOURCE_ACLUTIL, x, s, __VA_ARGS__) | ||
| 8 | #define AclExitWithLastError(x, s, ...) ExitWithLastErrorSource(DUTIL_SOURCE_ACLUTIL, x, s, __VA_ARGS__) | ||
| 9 | #define AclExitOnFailure(x, s, ...) ExitOnFailureSource(DUTIL_SOURCE_ACLUTIL, x, s, __VA_ARGS__) | ||
| 10 | #define AclExitOnRootFailure(x, s, ...) ExitOnRootFailureSource(DUTIL_SOURCE_ACLUTIL, x, s, __VA_ARGS__) | ||
| 11 | #define AclExitOnFailureDebugTrace(x, s, ...) ExitOnFailureDebugTraceSource(DUTIL_SOURCE_ACLUTIL, x, s, __VA_ARGS__) | ||
| 12 | #define AclExitOnNull(p, x, e, s, ...) ExitOnNullSource(DUTIL_SOURCE_ACLUTIL, p, x, e, s, __VA_ARGS__) | ||
| 13 | #define AclExitOnNullWithLastError(p, x, s, ...) ExitOnNullWithLastErrorSource(DUTIL_SOURCE_ACLUTIL, p, x, s, __VA_ARGS__) | ||
| 14 | #define AclExitOnNullDebugTrace(p, x, e, s, ...) ExitOnNullDebugTraceSource(DUTIL_SOURCE_ACLUTIL, p, x, e, s, __VA_ARGS__) | ||
| 15 | #define AclExitOnInvalidHandleWithLastError(p, x, s, ...) ExitOnInvalidHandleWithLastErrorSource(DUTIL_SOURCE_ACLUTIL, p, x, s, __VA_ARGS__) | ||
| 16 | #define AclExitOnWin32Error(e, x, s, ...) ExitOnWin32ErrorSource(DUTIL_SOURCE_ACLUTIL, e, x, s, __VA_ARGS__) | ||
| 17 | #define AclExitOnGdipFailure(g, x, s, ...) ExitOnGdipFailureSource(DUTIL_SOURCE_ACLUTIL, g, x, s, __VA_ARGS__) | ||
| 18 | |||
| 19 | /******************************************************************** | ||
| 20 | AclCalculateServiceSidString - gets the SID string for the given service name | ||
| 21 | |||
| 22 | NOTE: psczSid should be freed with StrFree() | ||
| 23 | ********************************************************************/ | ||
| 24 | extern "C" HRESULT DAPI AclCalculateServiceSidString( | ||
| 25 | __in LPCWSTR wzServiceName, | ||
| 26 | __in SIZE_T cchServiceName, | ||
| 27 | __deref_out_z LPWSTR* psczSid | ||
| 28 | ) | ||
| 29 | { | ||
| 30 | // TODO: use undocumented RtlCreateServiceSid function? | ||
| 31 | // http://blogs.technet.com/b/voy/archive/2007/03/22/per-service-sid.aspx | ||
| 32 | // Assume little endian. | ||
| 33 | HRESULT hr = S_OK; | ||
| 34 | LPWSTR sczUpperServiceName = NULL; | ||
| 35 | DWORD cbHash = SHA1_HASH_LEN; | ||
| 36 | BYTE* pbHash = NULL; | ||
| 37 | |||
| 38 | Assert(psczSid); | ||
| 39 | |||
| 40 | if (0 == cchServiceName) | ||
| 41 | { | ||
| 42 | hr = ::StringCchLengthW(wzServiceName, STRSAFE_MAX_CCH, reinterpret_cast<size_t*>(&cchServiceName)); | ||
| 43 | AclExitOnFailure(hr, "Failed to get the length of the service name."); | ||
| 44 | } | ||
| 45 | |||
| 46 | hr = StrAllocStringToUpperInvariant(&sczUpperServiceName, wzServiceName, cchServiceName); | ||
| 47 | AclExitOnFailure(hr, "Failed to upper case the service name."); | ||
| 48 | |||
| 49 | pbHash = reinterpret_cast<BYTE*>(MemAlloc(cbHash, TRUE)); | ||
| 50 | AclExitOnNull(pbHash, hr, E_OUTOFMEMORY, "Failed to allocate hash byte array."); | ||
| 51 | |||
| 52 | hr = CrypHashBuffer(reinterpret_cast<BYTE*>(sczUpperServiceName), cchServiceName * sizeof(WCHAR), PROV_RSA_FULL, CALG_SHA1, pbHash, cbHash); | ||
| 53 | AclExitOnNull(pbHash, hr, E_OUTOFMEMORY, "Failed to hash the service name."); | ||
| 54 | |||
| 55 | hr = StrAllocFormatted(psczSid, L"S-1-5-80-%u-%u-%u-%u-%u", | ||
| 56 | MAKEDWORD(MAKEWORD(pbHash[0], pbHash[1]), MAKEWORD(pbHash[2], pbHash[3])), | ||
| 57 | MAKEDWORD(MAKEWORD(pbHash[4], pbHash[5]), MAKEWORD(pbHash[6], pbHash[7])), | ||
| 58 | MAKEDWORD(MAKEWORD(pbHash[8], pbHash[9]), MAKEWORD(pbHash[10], pbHash[11])), | ||
| 59 | MAKEDWORD(MAKEWORD(pbHash[12], pbHash[13]), MAKEWORD(pbHash[14], pbHash[15])), | ||
| 60 | MAKEDWORD(MAKEWORD(pbHash[16], pbHash[17]), MAKEWORD(pbHash[18], pbHash[19]))); | ||
| 61 | |||
| 62 | LExit: | ||
| 63 | ReleaseMem(pbHash); | ||
| 64 | ReleaseStr(sczUpperServiceName); | ||
| 65 | |||
| 66 | return hr; | ||
| 67 | } | ||
| 68 | |||
| 69 | |||
| 70 | /******************************************************************** | ||
| 71 | AclGetAccountSidStringEx - gets a string version of the account's SID | ||
| 72 | calculates a service's SID if lookup fails | ||
| 73 | |||
| 74 | NOTE: psczSid should be freed with StrFree() | ||
| 75 | ********************************************************************/ | ||
| 76 | extern "C" HRESULT DAPI AclGetAccountSidStringEx( | ||
| 77 | __in_z LPCWSTR wzSystem, | ||
| 78 | __in_z LPCWSTR wzAccount, | ||
| 79 | __deref_out_z LPWSTR* psczSid | ||
| 80 | ) | ||
| 81 | { | ||
| 82 | HRESULT hr = S_OK; | ||
| 83 | SIZE_T cchAccount = 0; | ||
| 84 | PSID psid = NULL; | ||
| 85 | LPWSTR pwz = NULL; | ||
| 86 | LPWSTR sczSid = NULL; | ||
| 87 | |||
| 88 | Assert(psczSid); | ||
| 89 | |||
| 90 | hr = AclGetAccountSid(wzSystem, wzAccount, &psid); | ||
| 91 | if (SUCCEEDED(hr)) | ||
| 92 | { | ||
| 93 | Assert(::IsValidSid(psid)); | ||
| 94 | |||
| 95 | if (!::ConvertSidToStringSidW(psid, &pwz)) | ||
| 96 | { | ||
| 97 | AclExitWithLastError(hr, "Failed to convert SID to string for Account: %ls", wzAccount); | ||
| 98 | } | ||
| 99 | |||
| 100 | hr = StrAllocString(psczSid, pwz, 0); | ||
| 101 | } | ||
| 102 | else | ||
| 103 | { | ||
| 104 | if (HRESULT_FROM_WIN32(ERROR_NONE_MAPPED) == hr) | ||
| 105 | { | ||
| 106 | HRESULT hrLength = ::StringCchLengthW(wzAccount, STRSAFE_MAX_CCH, reinterpret_cast<size_t*>(&cchAccount)); | ||
| 107 | AclExitOnFailure(hrLength, "Failed to get the length of the account name."); | ||
| 108 | |||
| 109 | if (11 < cchAccount && CSTR_EQUAL == CompareStringW(LOCALE_NEUTRAL, NORM_IGNORECASE, L"NT SERVICE\\", 11, wzAccount, 11)) | ||
| 110 | { | ||
| 111 | // If the service is not installed then LookupAccountName doesn't resolve the SID, but we can calculate it. | ||
| 112 | LPCWSTR wzServiceName = &wzAccount[11]; | ||
| 113 | hr = AclCalculateServiceSidString(wzServiceName, cchAccount - 11, &sczSid); | ||
| 114 | AclExitOnFailure(hr, "Failed to calculate the service SID for %ls", wzServiceName); | ||
| 115 | |||
| 116 | *psczSid = sczSid; | ||
| 117 | sczSid = NULL; | ||
| 118 | } | ||
| 119 | } | ||
| 120 | AclExitOnFailure(hr, "Failed to get SID for account: %ls", wzAccount); | ||
| 121 | } | ||
| 122 | |||
| 123 | LExit: | ||
| 124 | ReleaseStr(sczSid); | ||
| 125 | if (pwz) | ||
| 126 | { | ||
| 127 | ::LocalFree(pwz); | ||
| 128 | } | ||
| 129 | if (psid) | ||
| 130 | { | ||
| 131 | AclFreeSid(psid); | ||
| 132 | } | ||
| 133 | |||
| 134 | return hr; | ||
| 135 | } | ||
