Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

778 lines
17 KiB

  1. /*++
  2. Copyright (c) 1991-1992 Microsoft Corporation
  3. Module Name:
  4. wsutil.c
  5. Abstract:
  6. This module contains miscellaneous utility routines used by the
  7. Workstation service.
  8. Author:
  9. Rita Wong (ritaw) 01-Mar-1991
  10. Revision History:
  11. --*/
  12. #include "wsutil.h"
  13. //-------------------------------------------------------------------//
  14. // //
  15. // Local function prototypes //
  16. // //
  17. //-------------------------------------------------------------------//
  18. STATIC
  19. NET_API_STATUS
  20. WsGrowTable(
  21. IN PUSERS_OBJECT Users
  22. );
  23. //-------------------------------------------------------------------//
  24. // //
  25. // Global variables //
  26. // //
  27. //-------------------------------------------------------------------//
  28. //
  29. // Debug trace flag for selecting which trace statements to output
  30. //
  31. #if DBG
  32. DWORD WorkstationTrace = 0;
  33. #endif // DBG
  34. NET_API_STATUS
  35. WsInitializeUsersObject(
  36. IN PUSERS_OBJECT Users
  37. )
  38. /*++
  39. Routine Description:
  40. This function allocates the table of users, and initializes the resource
  41. to serialize access to this table.
  42. Arguments:
  43. Users - Supplies a pointer to the users object.
  44. Return Value:
  45. NET_API_STATUS - NERR_Success or reason for failure.
  46. --*/
  47. {
  48. //
  49. // Allocate the users table memory so that it can be grown (reallocated)
  50. // as more entries are needed.
  51. //
  52. if ((Users->TableMemory = (HANDLE) LocalAlloc(
  53. LMEM_ZEROINIT | LMEM_MOVEABLE,
  54. INITIAL_USER_COUNT * sizeof(PER_USER_ENTRY)
  55. )) == NULL) {
  56. return GetLastError();
  57. }
  58. Users->TableSize = INITIAL_USER_COUNT;
  59. //
  60. // Keep the memory from moving by locking it to a specific location in
  61. // virtual memory. When it is necessary to grow this table, which may
  62. // result in the virtual memory being relocated, it will be unlocked.
  63. //
  64. if ((Users->Table = (PPER_USER_ENTRY)
  65. LocalLock(Users->TableMemory)) == NULL) {
  66. return GetLastError();
  67. }
  68. //
  69. // Initialize the resource for the users table.
  70. //
  71. try {
  72. RtlInitializeResource(&Users->TableResource);
  73. } except(EXCEPTION_EXECUTE_HANDLER) {
  74. return RtlNtStatusToDosError(GetExceptionCode());
  75. }
  76. return NERR_Success;
  77. }
  78. VOID
  79. WsDestroyUsersObject(
  80. IN PUSERS_OBJECT Users
  81. )
  82. /*++
  83. Routine Description:
  84. This function free the table allocated for logged on users, and deletes
  85. the resource used to serialize access to this table.
  86. Arguments:
  87. Users - Supplies a pointer to the users object.
  88. Return Value:
  89. None.
  90. --*/
  91. {
  92. //
  93. // Unlock the memory holding the table to allow us to free it.
  94. //
  95. LocalUnlock(Users->TableMemory);
  96. (void) LocalFree(Users->TableMemory);
  97. RtlDeleteResource(&(Users->TableResource));
  98. }
  99. NET_API_STATUS
  100. WsGetUserEntry(
  101. IN PUSERS_OBJECT Users,
  102. IN PLUID LogonId,
  103. OUT PULONG Index,
  104. IN BOOL IsAdd
  105. )
  106. /*++
  107. Routine Description:
  108. This function searches the table of user entries for one that matches the
  109. specified LogonId, and returns the index to the entry found. If none is
  110. found, an error is returned if IsAdd is FALSE. If IsAdd is TRUE a new
  111. entry in the users table is created for the user and the index to this
  112. new entry is returned.
  113. WARNING: This function assumes that the users table resource has been
  114. claimed.
  115. Arguments:
  116. Users - Supplies a pointer to the users object.
  117. LogonId - Supplies the pointer to the current user's Logon Id.
  118. Index - Returns the index to the users table of entry belonging to the
  119. current user.
  120. IsAdd - Supplies flag to indicate whether to add a new entry for the
  121. current user if none is found.
  122. Return Value:
  123. NET_API_STATUS - NERR_Success or reason for failure.
  124. --*/
  125. {
  126. NET_API_STATUS status;
  127. DWORD i;
  128. ULONG FreeEntryIndex = MAXULONG;
  129. for (i = 0; i < Users->TableSize; i++) {
  130. //
  131. // If the LogonId matches the entry in the UsersTable, we've found the
  132. // correct user entry.
  133. //
  134. if (RtlEqualLuid(LogonId, &Users->Table[i].LogonId)) {
  135. *Index = i;
  136. return NERR_Success;
  137. }
  138. else if (FreeEntryIndex == MAXULONG && Users->Table[i].List == NULL) {
  139. //
  140. // Save away first unused entry in table.
  141. //
  142. FreeEntryIndex = i;
  143. }
  144. }
  145. if (! IsAdd) {
  146. //
  147. // Current user is not found in users table and we are told not to
  148. // create a new entry
  149. //
  150. return NERR_UserNotFound;
  151. }
  152. //
  153. // Could not find an empty entry in the UsersTable, need to grow
  154. //
  155. if (FreeEntryIndex == MAXULONG) {
  156. if ((status = WsGrowTable(Users)) != NERR_Success) {
  157. return status;
  158. }
  159. FreeEntryIndex = i;
  160. }
  161. //
  162. // Create a new entry for current user
  163. //
  164. RtlCopyLuid(&Users->Table[FreeEntryIndex].LogonId, LogonId);
  165. *Index = FreeEntryIndex;
  166. return NERR_Success;
  167. }
  168. STATIC
  169. NET_API_STATUS
  170. WsGrowTable(
  171. IN PUSERS_OBJECT Users
  172. )
  173. /*++
  174. Routine Description:
  175. This function grows the users table to accomodate more users.
  176. WARNING: This function assumes that the users table resource has been
  177. claimed.
  178. Arguments:
  179. Users - Supplies a pointer to the users object.
  180. Return Value:
  181. NET_API_STATUS - NERR_Success or reason for failure.
  182. --*/
  183. {
  184. //
  185. // Unlock the Use Table virtual memory so that Win32 can move it
  186. // around to find a larger piece of contiguous virtual memory if
  187. // necessary.
  188. //
  189. LocalUnlock(Users->TableMemory);
  190. //
  191. // Grow users table
  192. //
  193. Users->TableMemory = LocalReAlloc(
  194. Users->TableMemory,
  195. (Users->TableSize + GROW_USER_COUNT)
  196. * sizeof(PER_USER_ENTRY),
  197. LMEM_ZEROINIT | LMEM_MOVEABLE
  198. );
  199. if (Users->TableMemory == NULL) {
  200. return GetLastError();
  201. }
  202. //
  203. // Update new size of Use Table
  204. //
  205. Users->TableSize += GROW_USER_COUNT;
  206. //
  207. // Lock Use Table virtual memory so that it cannot be moved
  208. //
  209. if ((Users->Table = (PPER_USER_ENTRY)
  210. LocalLock(Users->TableMemory)) == NULL) {
  211. return GetLastError();
  212. }
  213. return NERR_Success;
  214. }
  215. NET_API_STATUS
  216. WsMapStatus(
  217. IN NTSTATUS NtStatus
  218. )
  219. /*++
  220. Routine Description:
  221. This function takes an NT status code and maps it to the appropriate
  222. error code expected from calling a LAN Man API.
  223. Arguments:
  224. NtStatus - Supplies the NT status.
  225. Return Value:
  226. Returns the appropriate LAN Man error code for the NT status.
  227. --*/
  228. {
  229. //
  230. // A small optimization for the most common case.
  231. //
  232. if (NtStatus == STATUS_SUCCESS) {
  233. return NERR_Success;
  234. }
  235. switch (NtStatus) {
  236. case STATUS_OBJECT_NAME_COLLISION:
  237. return ERROR_ALREADY_ASSIGNED;
  238. case STATUS_OBJECT_NAME_NOT_FOUND:
  239. return NERR_UseNotFound;
  240. case STATUS_IMAGE_ALREADY_LOADED:
  241. case STATUS_REDIRECTOR_STARTED:
  242. return ERROR_SERVICE_ALREADY_RUNNING;
  243. case STATUS_REDIRECTOR_HAS_OPEN_HANDLES:
  244. return ERROR_REDIRECTOR_HAS_OPEN_HANDLES;
  245. default:
  246. return NetpNtStatusToApiStatus(NtStatus);
  247. }
  248. }
  249. int
  250. WsCompareString(
  251. IN LPTSTR String1,
  252. IN DWORD Length1,
  253. IN LPTSTR String2,
  254. IN DWORD Length2
  255. )
  256. /*++
  257. Routine Description:
  258. This function compares two strings based on their lengths. The return
  259. value indicates if the strings are equal or String1 is less than String2
  260. or String1 is greater than String2.
  261. This function is a modified version of RtlCompareString.
  262. Arguments:
  263. String1 - Supplies the pointer to the first string.
  264. Length1 - Supplies the length of String1 in characters.
  265. String2 - Supplies the pointer to the second string.
  266. Length2 - Supplies the length of String2 in characters.
  267. Return Value:
  268. Signed value that gives the results of the comparison:
  269. 0 - String1 equals String2
  270. < 0 - String1 less than String2
  271. > 0 - String1 greater than String2
  272. --*/
  273. {
  274. TCHAR Char1, Char2;
  275. int CharDiff;
  276. while (Length1 && Length2) {
  277. Char1 = *String1++;
  278. Char2 = *String2++;
  279. if ((CharDiff = (Char1 - Char2)) != 0) {
  280. return CharDiff;
  281. }
  282. Length1--;
  283. Length2--;
  284. }
  285. return Length1 - Length2;
  286. }
  287. int
  288. WsCompareStringU(
  289. IN LPWSTR String1,
  290. IN DWORD Length1,
  291. IN LPTSTR String2,
  292. IN DWORD Length2
  293. )
  294. {
  295. UNICODE_STRING S1;
  296. UNICODE_STRING S2;
  297. int rValue;
  298. S1.Length =
  299. S1.MaximumLength = (USHORT) (Length1 * sizeof(WCHAR));
  300. S1.Buffer = String1;
  301. S2.Length =
  302. S2.MaximumLength = (USHORT) (Length2 * sizeof(WCHAR));
  303. S2.Buffer = String2;
  304. rValue = RtlCompareUnicodeString(&S1, &S2, TRUE);
  305. return(rValue);
  306. }
  307. BOOL
  308. WsCopyStringToBuffer(
  309. IN PUNICODE_STRING SourceString,
  310. IN LPBYTE FixedPortion,
  311. IN OUT LPTSTR *EndOfVariableData,
  312. OUT LPTSTR *DestinationStringPointer
  313. )
  314. /*++
  315. Routine Description:
  316. This function converts the unicode source string to ANSI string (if
  317. we haven't flipped the unicode switch yet) and calls
  318. NetpCopyStringToBuffer.
  319. Arguments:
  320. SourceString - Supplies a pointer to the source string to copy into the
  321. output buffer. If String is null then a pointer to a zero terminator
  322. is inserted into output buffer.
  323. FixedDataEnd - Supplies a pointer to just after the end of the last
  324. fixed structure in the buffer.
  325. EndOfVariableData - Supplies an address to a pointer to just after the
  326. last position in the output buffer that variable data can occupy.
  327. Returns a pointer to the string written in the output buffer.
  328. DestinationStringPointer - Supplies a pointer to the place in the fixed
  329. portion of the output buffer where a pointer to the variable data
  330. should be written.
  331. Return Value:
  332. Returns TRUE if string fits into output buffer, FALSE otherwise.
  333. --*/
  334. {
  335. if (! NetpCopyStringToBuffer(
  336. SourceString->Buffer,
  337. SourceString->Length / sizeof(WCHAR),
  338. FixedPortion,
  339. EndOfVariableData,
  340. DestinationStringPointer
  341. )) {
  342. return FALSE;
  343. }
  344. return TRUE;
  345. }
  346. NET_API_STATUS
  347. WsImpersonateClient(
  348. VOID
  349. )
  350. /*++
  351. Routine Description:
  352. This function calls RpcImpersonateClient to impersonate the current caller
  353. of an API.
  354. Arguments:
  355. None.
  356. Return Value:
  357. NET_API_STATUS - NERR_Success or reason for failure.
  358. --*/
  359. {
  360. NET_API_STATUS status;
  361. if ((status = RpcImpersonateClient(NULL)) != NERR_Success) {
  362. NetpKdPrint(("[Wksta] Fail to impersonate client 0x%x\n", status));
  363. }
  364. return status;
  365. }
  366. NET_API_STATUS
  367. WsRevertToSelf(
  368. VOID
  369. )
  370. /*++
  371. Routine Description:
  372. This function calls RpcRevertToSelf to undo an impersonation.
  373. Arguments:
  374. None.
  375. Return Value:
  376. NET_API_STATUS - NERR_Success or reason for failure.
  377. --*/
  378. {
  379. NET_API_STATUS status;
  380. if (( status = RpcRevertToSelf()) != NERR_Success) {
  381. NetpKdPrint(("[Wksta] Fail to revert to self 0x%x\n", status));
  382. NetpAssert(FALSE);
  383. }
  384. return status;
  385. }
  386. NET_API_STATUS
  387. WsImpersonateAndGetLogonId(
  388. OUT PLUID LogonId
  389. )
  390. /*++
  391. Routine Description:
  392. This function gets the logon id of the current thread.
  393. Arguments:
  394. LogonId - Returns the logon id of the current process.
  395. Return Value:
  396. NET_API_STATUS - NERR_Success or reason for failure.
  397. --*/
  398. {
  399. NET_API_STATUS status;
  400. NTSTATUS ntstatus;
  401. HANDLE CurrentThreadToken;
  402. TOKEN_STATISTICS TokenStats;
  403. ULONG ReturnLength;
  404. if ((status = WsImpersonateClient()) != NERR_Success) {
  405. return status;
  406. }
  407. ntstatus = NtOpenThreadToken(
  408. NtCurrentThread(),
  409. TOKEN_QUERY,
  410. TRUE, // Use workstation service's security
  411. // context to open thread token
  412. &CurrentThreadToken
  413. );
  414. status = NetpNtStatusToApiStatus(ntstatus);
  415. if (! NT_SUCCESS(ntstatus)) {
  416. NetpKdPrint(("[Wksta] Cannot open the current thread token %08lx\n",
  417. ntstatus));
  418. goto RevertToSelf;
  419. }
  420. //
  421. // Get the logon id of the current thread
  422. //
  423. ntstatus = NtQueryInformationToken(
  424. CurrentThreadToken,
  425. TokenStatistics,
  426. (PVOID) &TokenStats,
  427. sizeof(TokenStats),
  428. &ReturnLength
  429. );
  430. status = NetpNtStatusToApiStatus(ntstatus);
  431. if (! NT_SUCCESS(ntstatus)) {
  432. NetpKdPrint(("[Wksta] Cannot query current thread's token %08lx\n",
  433. ntstatus));
  434. NtClose(CurrentThreadToken);
  435. goto RevertToSelf;
  436. }
  437. RtlCopyLuid(LogonId, &TokenStats.AuthenticationId);
  438. NtClose(CurrentThreadToken);
  439. RevertToSelf:
  440. WsRevertToSelf();
  441. return status;
  442. }
  443. NET_API_STATUS
  444. WsOpenDestinationMailslot(
  445. IN LPWSTR TargetName,
  446. IN LPWSTR MailslotName,
  447. OUT PHANDLE MailslotHandle
  448. )
  449. /*++
  450. Routine Description:
  451. This function combines the target domain or computer name and the mailslot
  452. name to form the destination mailslot name. It then opens this destination
  453. mailslot and returns a handle to it.
  454. Arguments:
  455. TargetName - Supplies the name of a domain or computer which we want to
  456. target when sending a mailslot message.
  457. MailslotName - Supplies the name of the mailslot.
  458. MailslotHandle - Returns the handle to the destination mailslot of
  459. \\TargetName\MailslotName.
  460. Return Value:
  461. NET_API_STATUS - NERR_Success or reason for failure.
  462. --*/
  463. {
  464. NET_API_STATUS status = NERR_Success;
  465. LPWSTR DestinationMailslot;
  466. if ((DestinationMailslot = (LPWSTR) LocalAlloc(
  467. LMEM_ZEROINIT,
  468. (UINT) (wcslen(TargetName) +
  469. wcslen(MailslotName) +
  470. 3) * sizeof(WCHAR)
  471. )) == NULL) {
  472. return GetLastError();
  473. }
  474. wcscpy(DestinationMailslot, L"\\\\");
  475. wcscat(DestinationMailslot, TargetName);
  476. wcscat(DestinationMailslot, MailslotName);
  477. if ((*MailslotHandle = (HANDLE) CreateFileW(
  478. DestinationMailslot,
  479. GENERIC_WRITE,
  480. FILE_SHARE_WRITE | FILE_SHARE_READ,
  481. (LPSECURITY_ATTRIBUTES) NULL,
  482. OPEN_EXISTING,
  483. FILE_ATTRIBUTE_NORMAL,
  484. NULL
  485. )) == INVALID_HANDLE_VALUE) {
  486. status = GetLastError();
  487. NetpKdPrint(("[Wksta] Error opening mailslot %s %lu",
  488. DestinationMailslot, status));
  489. }
  490. (void) LocalFree(DestinationMailslot);
  491. return status;
  492. }
  493. NET_API_STATUS
  494. WsImpersonateAndGetSessionId(
  495. OUT PULONG pSessionId
  496. )
  497. /*++
  498. Routine Description:
  499. This function gets the session id of the current thread.
  500. Arguments:
  501. pSessionId - Returns the session id of the current process.
  502. Return Value:
  503. NET_API_STATUS - NERR_Success or reason for failure.
  504. --*/
  505. {
  506. NET_API_STATUS status;
  507. NTSTATUS ntstatus;
  508. HANDLE CurrentThreadToken;
  509. ULONG SessionId;
  510. ULONG ReturnLength;
  511. if ((status = WsImpersonateClient()) != NERR_Success) {
  512. return status;
  513. }
  514. ntstatus = NtOpenThreadToken(
  515. NtCurrentThread(),
  516. TOKEN_QUERY,
  517. TRUE, // Use workstation service's security
  518. // context to open thread token
  519. &CurrentThreadToken
  520. );
  521. status = NetpNtStatusToApiStatus(ntstatus);
  522. if (! NT_SUCCESS(ntstatus)) {
  523. NetpKdPrint(("[Wksta] Cannot open the current thread token %08lx\n",
  524. ntstatus));
  525. goto RevertToSelf;
  526. }
  527. //
  528. // Get the session id of the current thread
  529. //
  530. ntstatus = NtQueryInformationToken(
  531. CurrentThreadToken,
  532. TokenSessionId,
  533. &SessionId,
  534. sizeof(ULONG),
  535. &ReturnLength
  536. );
  537. status = NetpNtStatusToApiStatus(ntstatus);
  538. if (! NT_SUCCESS(ntstatus)) {
  539. NetpKdPrint(("[Wksta] Cannot query current thread's token %08lx\n",
  540. ntstatus));
  541. NtClose(CurrentThreadToken);
  542. goto RevertToSelf;
  543. }
  544. NtClose(CurrentThreadToken);
  545. *pSessionId = SessionId;
  546. RevertToSelf:
  547. WsRevertToSelf();
  548. return status;
  549. }