VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxService/VBoxServiceVMInfo-win.cpp@ 22586

Last change on this file since 22586 was 21323, checked in by vboxsync, 16 years ago

VBoxService/common: Differ between crOGL amd64 and x86 DLLs.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 18.9 KB
Line 
1/* $Id: VBoxServiceVMInfo-win.cpp 21323 2009-07-07 12:38:19Z vboxsync $ */
2/** @file
3 * VBoxVMInfo-win - Virtual machine (guest) information for the host.
4 */
5
6/*
7 * Copyright (C) 2009 Sun Microsystems, Inc.
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#include <windows.h>
27#include <Ntsecapi.h>
28#include <wtsapi32.h> /* For WTS* calls. */
29#include <psapi.h> /* EnumProcesses. */
30
31#include <iprt/assert.h>
32#include <iprt/mem.h>
33#include <iprt/thread.h>
34#include <iprt/string.h>
35#include <iprt/semaphore.h>
36#include <iprt/system.h>
37#include <iprt/time.h>
38#include <VBox/VBoxGuestLib.h>
39#include "VBoxServiceInternal.h"
40#include "VBoxServiceUtils.h"
41
42
43/*******************************************************************************
44* Global Variables *
45*******************************************************************************/
46#ifndef TARGET_NT4
47 /** Function prototypes for dynamic loading. */
48 extern fnWTSGetActiveConsoleSessionId g_pfnWTSGetActiveConsoleSessionId;
49 /** The vminfo interval (millseconds). */
50 uint32_t g_VMInfoLoggedInUsersCount = 0;
51#endif
52
53
54#ifndef TARGET_NT4
55/* Function GetLUIDsFromProcesses() written by Stefan Kuhr. */
56DWORD VboxServiceVMInfoWinGetLUIDsFromProcesses(PLUID *ppLuid)
57{
58 DWORD dwSize, dwSize2, dwIndex ;
59 LPDWORD lpdwPIDs ;
60 DWORD dwLastError = ERROR_SUCCESS;
61
62 if (!ppLuid)
63 {
64 SetLastError(ERROR_INVALID_PARAMETER);
65 return 0L;
66 }
67
68 /* Call the PSAPI function EnumProcesses to get all of the
69 ProcID's currently in the system.
70 NOTE: In the documentation, the third parameter of
71 EnumProcesses is named cbNeeded, which implies that you
72 can call the function once to find out how much space to
73 allocate for a buffer and again to fill the buffer.
74 This is not the case. The cbNeeded parameter returns
75 the number of PIDs returned, so if your buffer size is
76 zero cbNeeded returns zero.
77 NOTE: The "HeapAlloc" loop here ensures that we
78 actually allocate a buffer large enough for all the
79 PIDs in the system. */
80 dwSize2 = 256 * sizeof(DWORD);
81
82 lpdwPIDs = NULL;
83 do
84 {
85 if (lpdwPIDs)
86 {
87 HeapFree(GetProcessHeap(), 0, lpdwPIDs) ;
88 dwSize2 *= 2;
89 }
90 lpdwPIDs = (unsigned long *)HeapAlloc(GetProcessHeap(), 0, dwSize2);
91 if (lpdwPIDs == NULL)
92 return 0L; // Last error will be that of HeapAlloc
93
94 if (!EnumProcesses( lpdwPIDs, dwSize2, &dwSize))
95 {
96 DWORD dw = GetLastError();
97 HeapFree(GetProcessHeap(), 0, lpdwPIDs);
98 SetLastError(dw);
99 return 0L;
100 }
101 }
102 while (dwSize == dwSize2);
103
104 /* At this point we have an array of the PIDs at the
105 time of the last EnumProcesses invocation. We will
106 allocate an array of LUIDs passed back via the out
107 param ppLuid of exactly the number of PIDs. We will
108 only fill the first n values of this array, with n
109 being the number of unique LUIDs found in these PIDs. */
110
111 /* How many ProcIDs did we get? */
112 dwSize /= sizeof(DWORD);
113 dwSize2 = 0L; /* Our return value of found luids. */
114
115 *ppLuid = (LUID *)LocalAlloc(LPTR, dwSize*sizeof(LUID));
116 if (!(*ppLuid))
117 {
118 dwLastError = GetLastError();
119 goto CLEANUP;
120 }
121 for (dwIndex = 0; dwIndex < dwSize; dwIndex++)
122 {
123 (*ppLuid)[dwIndex].LowPart =0L;
124 (*ppLuid)[dwIndex].HighPart=0;
125
126 /* Open the process (if we can... security does not
127 permit every process in the system). */
128 HANDLE hProcess = OpenProcess(PROCESS_QUERY_INFORMATION,FALSE, lpdwPIDs[dwIndex]);
129 if ( hProcess != NULL )
130 {
131 HANDLE hAccessToken;
132 if (OpenProcessToken(hProcess, TOKEN_QUERY, &hAccessToken))
133 {
134 TOKEN_STATISTICS ts;
135 DWORD dwSize;
136 if (GetTokenInformation(hAccessToken, TokenStatistics, &ts, sizeof ts, &dwSize))
137 {
138 DWORD dwTmp = 0L;
139 BOOL bFound = FALSE;
140 for (;dwTmp<dwSize2 && !bFound;dwTmp++)
141 bFound = (*ppLuid)[dwTmp].HighPart == ts.AuthenticationId.HighPart &&
142 (*ppLuid)[dwTmp].LowPart == ts.AuthenticationId.LowPart;
143
144 if (!bFound)
145 (*ppLuid)[dwSize2++] = ts.AuthenticationId;
146 }
147 CloseHandle(hAccessToken);
148 }
149
150 CloseHandle(hProcess);
151 }
152
153 /* We don't really care if OpenProcess or OpenProcessToken fail or succeed, because
154 there are quite a number of system processes we cannot open anyway, not even as SYSTEM. */
155 }
156
157 CLEANUP:
158
159 if (lpdwPIDs)
160 HeapFree(GetProcessHeap(), 0, lpdwPIDs);
161
162 if (ERROR_SUCCESS !=dwLastError)
163 SetLastError(dwLastError);
164
165 return dwSize2;
166}
167
168BOOL VboxServiceVMInfoWinIsLoggedIn(VBOXSERVICEVMINFOUSER* a_pUserInfo,
169 PLUID a_pSession,
170 PLUID a_pLuid,
171 DWORD a_dwNumOfProcLUIDs)
172{
173 BOOL bLoggedIn = FALSE;
174 BOOL bFoundUser = FALSE;
175 PSECURITY_LOGON_SESSION_DATA sessionData = NULL;
176 NTSTATUS r = 0;
177 WCHAR *usBuffer = NULL;
178 int iLength = 0;
179
180 if (!a_pSession)
181 return FALSE;
182
183 r = LsaGetLogonSessionData (a_pSession, &sessionData);
184 if (r != STATUS_SUCCESS)
185 {
186 VBoxServiceError("LsaGetLogonSessionData failed, LSA error %lu\n", LsaNtStatusToWinError(r));
187
188 if (sessionData)
189 LsaFreeReturnBuffer(sessionData);
190
191 return FALSE;
192 }
193
194 if (!sessionData)
195 {
196 VBoxServiceError("Invalid logon session data.\n");
197 return FALSE;
198 }
199
200 VBoxServiceVerbose(3, "Users: Session data: Name = %ls, Len = %d, SID = %s, LogonID = %d,%d\n",
201 (sessionData->UserName).Buffer, (sessionData->UserName).Length, (sessionData->Sid != NULL) ? "1" : "0", sessionData->LogonId.HighPart, sessionData->LogonId.LowPart);
202
203 if ((sessionData->UserName.Buffer != NULL) &&
204 (sessionData->Sid != NULL) &&
205 (sessionData->LogonId.LowPart != 0))
206 {
207 /* Get the user name. */
208 usBuffer = (sessionData->UserName).Buffer;
209 iLength = (sessionData->UserName).Length;
210 if (iLength > sizeof(a_pUserInfo->szUser) - sizeof(TCHAR)) /* -sizeof(TCHAR) because we have to add the terminating null char at the end later. */
211 {
212 VBoxServiceVerbose(0, "User name too long (%d bytes) for buffer! Name will be truncated.\n", iLength);
213 iLength = sizeof(a_pUserInfo->szUser) - sizeof(TCHAR);
214 }
215 wcsncpy (a_pUserInfo->szUser, usBuffer, iLength);
216 wcscat (a_pUserInfo->szUser, L""); /* Add terminating null char. */
217
218 /* Get authentication package. */
219 usBuffer = (sessionData->AuthenticationPackage).Buffer;
220 iLength = (sessionData->AuthenticationPackage).Length;
221 if (iLength > sizeof(a_pUserInfo->szAuthenticationPackage) - sizeof(TCHAR)) /* -sizeof(TCHAR) because we have to add the terminating null char at the end later. */
222 {
223 VBoxServiceVerbose(0, "Authentication pkg name too long (%d bytes) for buffer! Name will be truncated.\n", iLength);
224 iLength = sizeof(a_pUserInfo->szAuthenticationPackage) - sizeof(TCHAR);
225 }
226 wcsncpy (a_pUserInfo->szAuthenticationPackage, usBuffer, iLength);
227 wcscat (a_pUserInfo->szAuthenticationPackage, L""); /* Add terminating null char. */
228
229 /* Get logon domain. */
230 usBuffer = (sessionData->LogonDomain).Buffer;
231 iLength = (sessionData->LogonDomain).Length;
232 if (iLength > sizeof(a_pUserInfo->szLogonDomain) - sizeof(TCHAR)) /* -sizeof(TCHAR) because we have to add the terminating null char at the end later. */
233 {
234 VBoxServiceVerbose(0, "Logon domain name too long (%d bytes) for buffer! Name will be truncated.\n", iLength);
235 iLength = sizeof(a_pUserInfo->szLogonDomain) - sizeof(TCHAR);
236 }
237 wcsncpy (a_pUserInfo->szLogonDomain, usBuffer, iLength);
238 wcscat (a_pUserInfo->szLogonDomain, L""); /* Add terminating null char. */
239
240 /* Only handle users which can login interactively or logged in remotely over native RDP. */
241 if ( (((SECURITY_LOGON_TYPE)sessionData->LogonType == Interactive)
242 || ((SECURITY_LOGON_TYPE)sessionData->LogonType == RemoteInteractive))
243 && (sessionData->Sid != NULL))
244 {
245 TCHAR szOwnerName [_MAX_PATH] = { 0 };
246 DWORD dwOwnerNameSize = _MAX_PATH;
247
248 TCHAR szDomainName [_MAX_PATH] = { 0 };
249 DWORD dwDomainNameSize = _MAX_PATH;
250
251 SID_NAME_USE ownerType;
252
253 if (LookupAccountSid(NULL,
254 sessionData->Sid,
255 szOwnerName,
256 &dwOwnerNameSize,
257 szDomainName,
258 &dwDomainNameSize,
259 &ownerType))
260 {
261 VBoxServiceVerbose(3, "Account User=%ls, Session=%ld, LUID=%ld,%ld, AuthPkg=%ls, Domain=%ls\n",
262 a_pUserInfo->szUser, sessionData->Session, sessionData->LogonId.HighPart, sessionData->LogonId.LowPart, a_pUserInfo->szAuthenticationPackage, a_pUserInfo->szLogonDomain);
263
264 /* The session ID increments/decrements on Vista often! So don't compare
265 the session data SID with the current SID here. */
266 DWORD dwActiveSession = 0;
267 if (g_pfnWTSGetActiveConsoleSessionId != NULL) /* Check terminal session ID. */
268 dwActiveSession = g_pfnWTSGetActiveConsoleSessionId();
269
270 /*VBoxServiceVerbose(3, ("Users: Current active session ID: %ld\n", dwActiveSession));*/
271
272 if (SidTypeUser == ownerType)
273 {
274 LPWSTR pBuffer = NULL;
275 DWORD dwBytesRet = 0;
276 int iState = 0;
277
278 if (WTSQuerySessionInformation( /* Detect RDP sessions as well. */
279 WTS_CURRENT_SERVER_HANDLE,
280 WTS_CURRENT_SESSION,
281 WTSConnectState,
282 &pBuffer,
283 &dwBytesRet))
284 {
285 /*VBoxServiceVerbose(3, ("Users: WTSQuerySessionInformation returned %ld bytes, p=%p, state=%d\n", dwBytesRet, pBuffer, pBuffer != NULL ? (INT)*pBuffer : -1));*/
286 if(dwBytesRet)
287 iState = *pBuffer;
288
289 if ( (iState == WTSActive) /* User logged on to WinStation. */
290 || (iState == WTSShadow) /* Shadowing another WinStation. */
291 || (iState == WTSDisconnected)) /* WinStation logged on without client. */
292 {
293 /** @todo On Vista and W2K, always "old" user name are still
294 * there. Filter out the old one! */
295 VBoxServiceVerbose(3, "Users: Account User=%ls is logged in via TCS/RDP. State=%d\n", a_pUserInfo->szUser, iState);
296 bFoundUser = TRUE;
297 }
298 }
299 else
300 {
301 /* Terminal services don't run (for example in W2K, nothing to worry about ...). */
302 /* ... or is on Vista fast user switching page! */
303 bFoundUser = TRUE;
304 }
305
306 if (pBuffer)
307 WTSFreeMemory(pBuffer);
308
309 /* A user logged in, but it could be a stale/orphaned logon session. */
310 BOOL bFoundInLUIDs = FALSE;
311 for (DWORD dwIndex = 0; dwIndex < a_dwNumOfProcLUIDs; dwIndex++)
312 {
313 if ( (a_pLuid[dwIndex].HighPart == sessionData->LogonId.HighPart)
314 && (a_pLuid[dwIndex].LowPart == sessionData->LogonId.LowPart))
315 {
316 bLoggedIn = TRUE;
317 VBoxServiceVerbose(3, "User \"%ls\" is logged in!\n", a_pUserInfo->szUser);
318 break;
319 }
320 }
321 }
322 }
323 }
324 }
325
326 LsaFreeReturnBuffer(sessionData);
327 return bLoggedIn;
328}
329
330#endif /* TARGET_NT4 */
331
332int VboxServiceWinGetAddsVersion(uint32_t uiClientID)
333{
334 char szInstDir[_MAX_PATH] = {0};
335 char szRev[_MAX_PATH] = {0};
336 char szVer[_MAX_PATH] = {0};
337
338 HKEY hKey = NULL;
339 long rc = 0;
340 DWORD dwSize = 0;
341 DWORD dwType = 0;
342
343 VBoxServiceVerbose(3, "Guest Additions version lookup: Looking up ...\n");
344
345 /* Check the new path first. */
346 rc = RegOpenKeyExA (HKEY_LOCAL_MACHINE, "SOFTWARE\\Sun\\VirtualBox Guest Additions", 0, KEY_READ, &hKey);
347 if (rc != ERROR_SUCCESS)
348 {
349 VBoxServiceVerbose(3, "Guest Additions version lookup: Looking in xVM key ...\n");
350
351 /* New path does not exist, check the old one ... */
352 rc = RegOpenKeyExA (HKEY_LOCAL_MACHINE, "SOFTWARE\\Sun\\xVM VirtualBox Guest Additions", 0, KEY_READ, &hKey);
353 if (rc != ERROR_SUCCESS)
354 {
355 /* Nothing seems to exist, print some warning. */
356 VBoxServiceError("Failed to open registry key (guest additions)! Error: %d\n", rc);
357 return 1;
358 }
359 }
360
361 VBoxServiceVerbose(3, "Guest Additions version lookup: Key: 0x%p, RC: %d\n", hKey, rc);
362
363 /* Installation directory. */
364 dwSize = sizeof(szInstDir);
365 rc = RegQueryValueExA (hKey, "InstallDir", NULL, &dwType, (BYTE*)(LPCTSTR)szInstDir, &dwSize);
366 if ((rc != ERROR_SUCCESS) && (rc != ERROR_FILE_NOT_FOUND))
367 {
368 RegCloseKey (hKey);
369 VBoxServiceError("Failed to query registry key (install directory)! Error: %d\n", rc);
370 return 1;
371 }
372
373 /* Flip slashes. */
374 for (char* pszTmp = &szInstDir[0]; *pszTmp; ++pszTmp)
375 if (*pszTmp == '\\')
376 *pszTmp = '/';
377
378 /* Revision. */
379 dwSize = sizeof(szRev);
380 rc = RegQueryValueExA (hKey, "Revision", NULL, &dwType, (BYTE*)(LPCTSTR)szRev, &dwSize);
381 if ((rc != ERROR_SUCCESS) && (rc != ERROR_FILE_NOT_FOUND))
382 {
383 RegCloseKey (hKey);
384 VBoxServiceError("Failed to query registry key (revision)! Error: %d\n", rc);
385 return 1;
386 }
387
388 /* Version. */
389 dwSize = sizeof(szVer);
390 rc = RegQueryValueExA (hKey, "Version", NULL, &dwType, (BYTE*)(LPCTSTR)szVer, &dwSize);
391 if ((rc != ERROR_SUCCESS) && (rc != ERROR_FILE_NOT_FOUND))
392 {
393 RegCloseKey (hKey);
394 VBoxServiceError("Failed to query registry key (version)! Error: %u\n", rc);
395 return 1;
396 }
397
398 /* Write information to host. */
399 VboxServiceWriteProp(uiClientID, "GuestAdd/InstallDir", szInstDir);
400 VboxServiceWriteProp(uiClientID, "GuestAdd/Revision", szRev);
401 VboxServiceWriteProp(uiClientID, "GuestAdd/Version", szVer);
402
403 RegCloseKey (hKey);
404
405 return VINF_SUCCESS;
406}
407
408int VboxServiceWinGetComponentVersions(uint32_t uiClientID)
409{
410 int rc;
411 char szVer[_MAX_PATH] = {0};
412 char szPropPath[_MAX_PATH] = {0};
413 TCHAR szSysDir[_MAX_PATH] = {0};
414 TCHAR szWinDir[_MAX_PATH] = {0};
415 TCHAR szDriversDir[_MAX_PATH + 32] = {0};
416
417 GetSystemDirectory(szSysDir, _MAX_PATH);
418 GetWindowsDirectory(szWinDir, _MAX_PATH);
419 swprintf(szDriversDir, (_MAX_PATH + 32), TEXT("%s\\drivers"), szSysDir);
420#ifdef RT_ARCH_AMD64
421 TCHAR szSysWowDir[_MAX_PATH + 32] = {0};
422 swprintf(szSysWowDir, (_MAX_PATH + 32), TEXT("%s\\SysWow64"), szWinDir);
423#endif
424
425 /* The file information table. */
426#ifndef TARGET_NT4
427 VBOXSERVICEVMINFOFILE vboxFileInfoTable[] =
428 {
429 { szSysDir, TEXT("VBoxControl.exe"), },
430 { szSysDir, TEXT("VBoxHook.dll"), },
431 { szSysDir, TEXT("VBoxDisp.dll"), },
432 { szSysDir, TEXT("VBoxMRXNP.dll"), },
433 { szSysDir, TEXT("VBoxService.exe"), },
434 { szSysDir, TEXT("VBoxTray.exe"), },
435 { szSysDir, TEXT("VBoxGINA.dll"), },
436
437 /* On 64-bit we don't yet have the OpenGL DLLs in native format.
438 So just enumerate the 32-bit files in the SYSWOW directory. */
439 #ifdef RT_ARCH_AMD64
440 { szSysWowDir, TEXT("VBoxOGLarrayspu.dll"), },
441 { szSysWowDir, TEXT("VBoxOGLcrutil.dll"), },
442 { szSysWowDir, TEXT("VBoxOGLerrorspu.dll"), },
443 { szSysWowDir, TEXT("VBoxOGLpackspu.dll"), },
444 { szSysWowDir, TEXT("VBoxOGLpassthroughspu.dll"), },
445 { szSysWowDir, TEXT("VBoxOGLfeedbackspu.dll"), },
446 { szSysWowDir, TEXT("VBoxOGL.dll"), },
447 #else
448 { szSysDir, TEXT("VBoxOGLarrayspu.dll"), },
449 { szSysDir, TEXT("VBoxOGLcrutil.dll"), },
450 { szSysDir, TEXT("VBoxOGLerrorspu.dll"), },
451 { szSysDir, TEXT("VBoxOGLpackspu.dll"), },
452 { szSysDir, TEXT("VBoxOGLpassthroughspu.dll"), },
453 { szSysDir, TEXT("VBoxOGLfeedbackspu.dll"), },
454 { szSysDir, TEXT("VBoxOGL.dll"), },
455 #endif
456
457 { szDriversDir, TEXT("VBoxGuest.sys"), },
458 { szDriversDir, TEXT("VBoxMouse.sys"), },
459 { szDriversDir, TEXT("VBoxSF.sys"), },
460 { szDriversDir, TEXT("VBoxVideo.sys"), },
461
462 {
463 NULL
464 }
465 };
466#else /* File lookup for NT4. */
467 VBOXSERVICEVMINFOFILE vboxFileInfoTable[] =
468 {
469 { szSysDir, TEXT("VBoxControl.exe"), },
470 { szSysDir, TEXT("VBoxHook.dll"), },
471 { szSysDir, TEXT("VBoxDisp.dll"), },
472 { szSysDir, TEXT("VBoxService.exe"), },
473 { szSysDir, TEXT("VBoxTray.exe"), },
474
475 { szDriversDir, TEXT("VBoxGuestNT.sys"), },
476 { szDriversDir, TEXT("VBoxMouseNT.sys"), },
477 { szDriversDir, TEXT("VBoxVideo.sys"), },
478
479 {
480 NULL
481 }
482 };
483#endif
484
485 PVBOXSERVICEVMINFOFILE pTable = vboxFileInfoTable;
486 Assert(pTable);
487 while (pTable->pszFileName)
488 {
489 rc = VboxServiceGetFileVersionString(pTable->pszFilePath, pTable->pszFileName, szVer, sizeof(szVer));
490 RTStrPrintf(szPropPath, sizeof(szPropPath), "GuestAdd/Components/%ls", pTable->pszFileName);
491 VboxServiceWriteProp(uiClientID, szPropPath, szVer);
492 pTable++;
493 }
494
495 return VINF_SUCCESS;
496}
497
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette