diff options
Diffstat (limited to 'src/dutil/acl2util.cpp')
-rw-r--r-- | src/dutil/acl2util.cpp | 121 |
1 files changed, 121 insertions, 0 deletions
diff --git a/src/dutil/acl2util.cpp b/src/dutil/acl2util.cpp new file mode 100644 index 00000000..2261abe3 --- /dev/null +++ b/src/dutil/acl2util.cpp | |||
@@ -0,0 +1,121 @@ | |||
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 | /******************************************************************** | ||
6 | AclCalculateServiceSidString - gets the SID string for the given service name | ||
7 | |||
8 | NOTE: psczSid should be freed with StrFree() | ||
9 | ********************************************************************/ | ||
10 | extern "C" HRESULT DAPI AclCalculateServiceSidString( | ||
11 | __in LPCWSTR wzServiceName, | ||
12 | __in int cchServiceName, | ||
13 | __deref_out_z LPWSTR* psczSid | ||
14 | ) | ||
15 | { | ||
16 | // TODO: use undocumented RtlCreateServiceSid function? | ||
17 | // http://blogs.technet.com/b/voy/archive/2007/03/22/per-service-sid.aspx | ||
18 | // Assume little endian. | ||
19 | HRESULT hr = S_OK; | ||
20 | LPWSTR sczUpperServiceName = NULL; | ||
21 | DWORD cbHash = SHA1_HASH_LEN; | ||
22 | BYTE* pbHash = NULL; | ||
23 | |||
24 | Assert(psczSid); | ||
25 | |||
26 | if (0 == cchServiceName) | ||
27 | { | ||
28 | hr = ::StringCchLengthW(wzServiceName, INT_MAX, reinterpret_cast<size_t*>(&cchServiceName)); | ||
29 | ExitOnFailure(hr, "Failed to get the length of the service name."); | ||
30 | } | ||
31 | |||
32 | hr = StrAllocStringToUpperInvariant(&sczUpperServiceName, wzServiceName, cchServiceName); | ||
33 | ExitOnFailure(hr, "Failed to upper case the service name."); | ||
34 | |||
35 | pbHash = reinterpret_cast<BYTE*>(MemAlloc(cbHash, TRUE)); | ||
36 | ExitOnNull(pbHash, hr, E_OUTOFMEMORY, "Failed to allocate hash byte array."); | ||
37 | |||
38 | hr = CrypHashBuffer(reinterpret_cast<BYTE*>(sczUpperServiceName), cchServiceName * 2, PROV_RSA_FULL, CALG_SHA1, pbHash, cbHash); | ||
39 | ExitOnNull(pbHash, hr, E_OUTOFMEMORY, "Failed to hash the service name."); | ||
40 | |||
41 | hr = StrAllocFormatted(psczSid, L"S-1-5-80-%u-%u-%u-%u-%u", | ||
42 | MAKEDWORD(MAKEWORD(pbHash[0], pbHash[1]), MAKEWORD(pbHash[2], pbHash[3])), | ||
43 | MAKEDWORD(MAKEWORD(pbHash[4], pbHash[5]), MAKEWORD(pbHash[6], pbHash[7])), | ||
44 | MAKEDWORD(MAKEWORD(pbHash[8], pbHash[9]), MAKEWORD(pbHash[10], pbHash[11])), | ||
45 | MAKEDWORD(MAKEWORD(pbHash[12], pbHash[13]), MAKEWORD(pbHash[14], pbHash[15])), | ||
46 | MAKEDWORD(MAKEWORD(pbHash[16], pbHash[17]), MAKEWORD(pbHash[18], pbHash[19]))); | ||
47 | |||
48 | LExit: | ||
49 | ReleaseMem(pbHash); | ||
50 | ReleaseStr(sczUpperServiceName); | ||
51 | |||
52 | return hr; | ||
53 | } | ||
54 | |||
55 | |||
56 | /******************************************************************** | ||
57 | AclGetAccountSidStringEx - gets a string version of the account's SID | ||
58 | calculates a service's SID if lookup fails | ||
59 | |||
60 | NOTE: psczSid should be freed with StrFree() | ||
61 | ********************************************************************/ | ||
62 | extern "C" HRESULT DAPI AclGetAccountSidStringEx( | ||
63 | __in_z LPCWSTR wzSystem, | ||
64 | __in_z LPCWSTR wzAccount, | ||
65 | __deref_out_z LPWSTR* psczSid | ||
66 | ) | ||
67 | { | ||
68 | HRESULT hr = S_OK; | ||
69 | int cchAccount = 0; | ||
70 | PSID psid = NULL; | ||
71 | LPWSTR pwz = NULL; | ||
72 | LPWSTR sczSid = NULL; | ||
73 | |||
74 | Assert(psczSid); | ||
75 | |||
76 | hr = AclGetAccountSid(wzSystem, wzAccount, &psid); | ||
77 | if (SUCCEEDED(hr)) | ||
78 | { | ||
79 | Assert(::IsValidSid(psid)); | ||
80 | |||
81 | if (!::ConvertSidToStringSidW(psid, &pwz)) | ||
82 | { | ||
83 | ExitWithLastError(hr, "Failed to convert SID to string for Account: %ls", wzAccount); | ||
84 | } | ||
85 | |||
86 | hr = StrAllocString(psczSid, pwz, 0); | ||
87 | } | ||
88 | else | ||
89 | { | ||
90 | if (HRESULT_FROM_WIN32(ERROR_NONE_MAPPED) == hr) | ||
91 | { | ||
92 | HRESULT hrLength = ::StringCchLengthW(wzAccount, INT_MAX, reinterpret_cast<size_t*>(&cchAccount)); | ||
93 | ExitOnFailure(hrLength, "Failed to get the length of the account name."); | ||
94 | |||
95 | if (11 < cchAccount && CSTR_EQUAL == CompareStringW(LOCALE_NEUTRAL, NORM_IGNORECASE, L"NT SERVICE\\", 11, wzAccount, 11)) | ||
96 | { | ||
97 | // If the service is not installed then LookupAccountName doesn't resolve the SID, but we can calculate it. | ||
98 | LPCWSTR wzServiceName = &wzAccount[11]; | ||
99 | hr = AclCalculateServiceSidString(wzServiceName, cchAccount - 11, &sczSid); | ||
100 | ExitOnFailure(hr, "Failed to calculate the service SID for %ls", wzServiceName); | ||
101 | |||
102 | *psczSid = sczSid; | ||
103 | sczSid = NULL; | ||
104 | } | ||
105 | } | ||
106 | ExitOnFailure(hr, "Failed to get SID for account: %ls", wzAccount); | ||
107 | } | ||
108 | |||
109 | LExit: | ||
110 | ReleaseStr(sczSid); | ||
111 | if (pwz) | ||
112 | { | ||
113 | ::LocalFree(pwz); | ||
114 | } | ||
115 | if (psid) | ||
116 | { | ||
117 | AclFreeSid(psid); | ||
118 | } | ||
119 | |||
120 | return hr; | ||
121 | } | ||