1 /*
2  * PROJECT:     ReactOS Spooler Router
3  * LICENSE:     GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+)
4  * PURPOSE:     Functions related to switching between security contexts
5  * COPYRIGHT:   Copyright 2015 Colin Finck (colin@reactos.org)
6  */
7 
8 #include "precomp.h"
9 
10 /**
11  * @see RevertToPrinterSelf
12  */
13 BOOL WINAPI
14 ImpersonatePrinterClient(HANDLE hToken)
15 {
16     DWORD cbReturned;
17     DWORD dwErrorCode;
18     TOKEN_TYPE Type;
19 
20     // Sanity check
21     if (!hToken)
22     {
23         dwErrorCode = ERROR_INVALID_HANDLE;
24         goto Cleanup;
25     }
26 
27     // Get the type of the supplied token.
28     if (!GetTokenInformation(hToken, TokenType, &Type, sizeof(TOKEN_TYPE), &cbReturned))
29     {
30         dwErrorCode = GetLastError();
31         ERR("GetTokenInformation failed with error %lu!\n", dwErrorCode);
32         goto Cleanup;
33     }
34 
35     // Check if this is an impersonation token and only set it as the thread token in this case.
36     // This is not always an impersonation token, see RevertToPrinterSelf.
37     if (Type == TokenImpersonation)
38     {
39         if (!SetThreadToken(NULL, hToken))
40         {
41             dwErrorCode = GetLastError();
42             ERR("SetThreadToken failed with error %lu!\n", dwErrorCode);
43             goto Cleanup;
44         }
45     }
46 
47 Cleanup:
48     if (hToken)
49         CloseHandle(hToken);
50 
51     SetLastError(dwErrorCode);
52     return (dwErrorCode == ERROR_SUCCESS);
53 }
54 
55 /**
56  * RevertToPrinterSelf reverts the security context from the current user's context back to the process context.
57  * As spoolss.dll is used by spoolsv.exe, this is usually the SYSTEM security context.
58  *
59  * Unlike the traditional ImpersonateClient and then RevertToSelf approach, we do it the other way round here,
60  * because spoolss.dll is delay-loaded by spoolsv.exe in the current user's context. Use RevertToPrinterSelf then to
61  * return to the SYSTEM context for specific tasks.
62  */
63 HANDLE WINAPI
64 RevertToPrinterSelf()
65 {
66     DWORD dwErrorCode;
67     HANDLE hReturnValue = NULL;
68     HANDLE hToken = NULL;
69 
70     // All spoolss code is usually called after impersonating the client. In this case, we can retrieve our current thread impersonation token using OpenThreadToken.
71     // But in rare occasions, spoolss code is also called from a higher-privileged thread that doesn't impersonate the client. Then we don't get an impersonation token.
72     // Anyway, we can't just return nothing in this case, because this is being treated as failure by the caller. So we return the token of the current process.
73     // This behaviour is verified with Windows!
74     if (OpenThreadToken(GetCurrentThread(), TOKEN_IMPERSONATE, TRUE, &hToken))
75     {
76         // Tell the thread to stop impersonating.
77         if (!SetThreadToken(NULL, NULL))
78         {
79             dwErrorCode = GetLastError();
80             ERR("SetThreadToken failed with error %lu!\n", dwErrorCode);
81             goto Cleanup;
82         }
83     }
84     else if (!OpenProcessToken(GetCurrentProcess(), TOKEN_QUERY, &hToken))
85     {
86         dwErrorCode = GetLastError();
87         ERR("OpenProcessToken failed with error %lu!\n", dwErrorCode);
88         goto Cleanup;
89     }
90 
91     // We were successful, return a token!
92     dwErrorCode = ERROR_SUCCESS;
93     hReturnValue = hToken;
94 
95     // Don't let the cleanup routine close this.
96     hToken = NULL;
97 
98 Cleanup:
99     if (hToken)
100         CloseHandle(hToken);
101 
102     SetLastError(dwErrorCode);
103     return hReturnValue;
104 }
105