VirtualBox

source: vbox/trunk/src/VBox/Additions/WINNT/VBoxTray/VBoxLA.cpp@ 40866

Last change on this file since 40866 was 40866, checked in by vboxsync, 13 years ago

VBoxTray: configuration for guest property monitor

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 36.6 KB
Line 
1/* $Id: VBoxLA.cpp 40866 2012-04-11 09:36:34Z vboxsync $ */
2/** @file
3 * VBoxLA - VBox Location Awareness notifications.
4 */
5
6/*
7 * Copyright (C) 2012 Oracle Corporation
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
18// #define LOG_ENABLED
19
20#define _WIN32_WINNT 0x0501
21#include <windows.h>
22
23#include "VBoxTray.h"
24#include "VBoxLA.h"
25
26#include <iprt/assert.h>
27#include <iprt/alloc.h>
28#include <iprt/list.h>
29
30#define LALOG(a) do { if (gCtx.fLogEnabled) LogRel(a); } while(0)
31#define LALOGFORCE(a) do { LogRel(a); } while(0)
32
33#define REG_KEY_LEN 1024
34#define MAX_CLIENT_NAME_CHARS 1024
35
36#define LA_DO_NOTHING 0
37#define LA_DO_ATTACH 1
38#define LA_DO_DETACH 2
39#define LA_DO_DETACH_AND_ATTACH 3
40#define LA_DO_ATTACH_AND_DETACH 4
41
42struct VBOXLACONTEXT
43{
44 const VBOXSERVICEENV *pEnv;
45
46 bool fLogEnabled;
47 bool fDetachOnDisconnect;
48
49 uint32_t u32GuestPropHandle; /* The client identifier of the guest property system. */
50
51 RTLISTANCHOR listAttachActions;
52 RTLISTANCHOR listDetachActions;
53
54 uint64_t u64LastQuery; /* The timestamp of the last query of the properties. */
55
56 uint32_t u32Action; /* Which action to do: LA_DO_*. */
57 uint32_t u32PrevAction; /* Which action were done last time. */
58
59 struct /* Information about the client, which properties are monitored. */
60 {
61 uint32_t u32ClientId; /* The RDP client identifier. 0 if none. */
62
63 uint32_t u32LastAttach;
64 uint64_t u64LastAttachTimestamp;
65
66 char *pszLastName;
67 uint64_t u64LastNameTimestamp;
68
69 char *pszPropName; /* The actual Client/%ID%/Name property name with client id. */
70 char *pszPropAttach; /* The actual Client/%ID%/Attach property name with client id. */
71
72 char *pszPropWaitPattern; /* Which properties are monitored. */
73 } activeClient;
74
75 HMODULE hModuleKernel32;
76
77 BOOL (WINAPI * pfnProcessIdToSessionId)(DWORD dwProcessId, DWORD *pSessionId);
78};
79
80typedef struct ACTIONENTRY
81{
82 RTLISTNODE nodeActionEntry;
83 uint32_t u32Index;
84 WCHAR wszCommandLine[1];
85} ACTIONENTRY;
86
87
88static VBOXLACONTEXT gCtx = {0};
89
90static const char *g_pszPropActiveClient = "/VirtualBox/HostInfo/VRDP/ActiveClient";
91
92static const char *g_pszPropNameTemplate = "/VirtualBox/HostInfo/VRDP/Client/%u/Name";
93static const char *g_pszPropAttachTemplate = "/VirtualBox/HostInfo/VRDP/Client/%u/Attach";
94
95static const char *g_pszVolatileEnvironment = "Volatile Environment";
96
97static const WCHAR *g_pwszUTCINFOClientName = L"UTCINFO_CLIENTNAME";
98static const WCHAR *g_pwszClientName = L"CLIENTNAME";
99
100#ifdef RT_ARCH_AMD64
101const WCHAR *g_pwszRegKeyDisconnectActions = L"Software\\Wow6432Node\\Oracle\\Sun Ray\\ClientInfoAgent\\DisconnectActions";
102const WCHAR *g_pwszRegKeyReconnectActions = L"Software\\Wow6432Node\\Oracle\\Sun Ray\\ClientInfoAgent\\ReconnectActions";
103#else
104const WCHAR *g_pwszRegKeyDisconnectActions = L"Software\\Oracle\\Sun Ray\\ClientInfoAgent\\DisconnectActions";
105const WCHAR *g_pwszRegKeyReconnectActions = L"Software\\Oracle\\Sun Ray\\ClientInfoAgent\\ReconnectActions";
106#endif /* !RT_ARCH_AMD64 */
107
108const char g_szCommandPrefix[] = "Command";
109
110static BOOL laGetRegistryDWORD(WCHAR *pwszRegKey, WCHAR *pwszName, DWORD *pdwValue)
111{
112 LONG lErr;
113
114 HKEY hKey;
115 lErr = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
116 pwszRegKey,
117 0,
118 KEY_QUERY_VALUE,
119 &hKey);
120
121 if (lErr != ERROR_SUCCESS)
122 {
123 LALOGFORCE(("LA: RegOpenKeyExW: failed [%ls]\n",
124 pwszRegKey));
125 return FALSE;
126 }
127
128 DWORD nRegData = sizeof(DWORD);
129 DWORD dwType = 0;
130 lErr = RegQueryValueExW(hKey,
131 pwszName,
132 NULL,
133 &dwType,
134 (BYTE *)pdwValue,
135 &nRegData);
136
137 if (lErr != ERROR_SUCCESS)
138 {
139 LALOGFORCE(("LA: RegQueryValueExW: failed [%ls/%ls]\n",
140 pwszRegKey, pwszName));
141 RegCloseKey(hKey);
142 return FALSE;
143 }
144
145 if (nRegData != sizeof(DWORD))
146 {
147 LALOGFORCE(("LA: buffer overflow reg %d, [%ls]\n",
148 nRegData, pwszRegKey));
149 RegCloseKey(hKey);
150 return FALSE;
151 }
152
153 if (dwType != REG_DWORD)
154 {
155 LALOGFORCE(("LA: wrong type %d, [%ls/%ls]\n",
156 dwType, pwszRegKey, pwszName));
157 RegCloseKey(hKey);
158 return FALSE;
159 }
160
161 RegCloseKey(hKey);
162
163 if (lErr != ERROR_SUCCESS)
164 {
165 return FALSE;
166 }
167
168 return TRUE;
169}
170
171static void ActionExecutorDeleteActions(RTLISTANCHOR *listActions)
172{
173 ACTIONENTRY *pIter = NULL;
174 ACTIONENTRY *pIterNext = NULL;
175 RTListForEachSafe(listActions, pIter, pIterNext, ACTIONENTRY, nodeActionEntry)
176 {
177 RTListNodeRemove(&pIter->nodeActionEntry);
178 RTMemFree(pIter);
179 }
180}
181
182static BOOL ActionExecutorEnumerateRegistryKey(const WCHAR *pwszRegKey,
183 RTLISTANCHOR *listActions)
184{
185 BOOL bRet = TRUE;
186 HKEY hKey;
187 DWORD dwErr;
188
189 dwErr = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
190 pwszRegKey,
191 0,
192 KEY_QUERY_VALUE,
193 &hKey);
194
195 if (dwErr != ERROR_SUCCESS)
196 {
197 LALOG(("LA: Can't open registry key [%ls], error %d\n",
198 pwszRegKey, dwErr));
199 return FALSE;
200 }
201
202 DWORD dwIndex = 0;
203
204 for (;;)
205 {
206 DWORD dwRet;
207
208 WCHAR wszValueName[256];
209 DWORD cchValueName = RT_ELEMENTS(wszValueName);
210 DWORD type;
211 BYTE abData[1024];
212 DWORD cbData = sizeof(abData);
213
214 dwRet = RegEnumValueW(hKey,
215 dwIndex++,
216 wszValueName,
217 &cchValueName,
218 NULL,
219 &type,
220 abData,
221 &cbData);
222
223 if (dwRet == ERROR_NO_MORE_ITEMS)
224 {
225 LALOG(("LA: Enumeration exhausted\n"));
226 bRet = TRUE;
227 break;
228 }
229 else if (dwRet != ERROR_SUCCESS)
230 {
231 LALOG(("LA: Enumeration failed, error %d\n",
232 dwRet));
233 bRet = FALSE;
234 break;
235 }
236
237 if ((type != REG_SZ) && (type != REG_EXPAND_SZ))
238 {
239 LALOG(("LA: skipped type %d\n",
240 type));
241 continue;
242 }
243
244 char szName[256];
245 char *pszName = &szName[0];
246 int rc = RTUtf16ToUtf8Ex(wszValueName,
247 RT_ELEMENTS(wszValueName),
248 &pszName, sizeof(szName), NULL);
249 if (RT_FAILURE(rc))
250 {
251 LALOG(("LA: RTUtf16ToUtf8Ex for [%ls] rc %Rrc\n",
252 wszValueName, rc));
253 continue;
254 }
255
256 /* Check if the name starts with "Command" */
257 if (RTStrNICmp(szName, g_szCommandPrefix, RT_ELEMENTS(g_szCommandPrefix) - 1) != 0)
258 {
259 LALOG(("LA: skipped prefix %s\n",
260 szName));
261 continue;
262 }
263
264 char *pszIndex = &szName[RT_ELEMENTS(g_szCommandPrefix) - 1];
265
266 uint32_t nIndex = RTStrToUInt32(pszIndex);
267 if (nIndex == 0)
268 {
269 LALOG(("LA: skipped index %s\n",
270 szName));
271 continue;
272 }
273
274 /* Allocate with terminating nul after data. */
275 ACTIONENTRY *pEntry = (ACTIONENTRY *)RTMemAlloc(sizeof(ACTIONENTRY) + cbData);
276 if (!pEntry)
277 {
278 LALOG(("LA: RTMemAlloc failed\n"));
279 bRet = FALSE;
280 break;
281 }
282
283 RT_ZERO(pEntry->nodeActionEntry);
284 pEntry->u32Index = nIndex;
285 memcpy(pEntry->wszCommandLine, abData, cbData);
286 pEntry->wszCommandLine[cbData / sizeof(WCHAR)] = 0;
287
288 /* Insert the new entry to the list. Sort by index. */
289 if (RTListIsEmpty(listActions))
290 {
291 RTListAppend(listActions, &pEntry->nodeActionEntry);
292 }
293 else
294 {
295 bool fAdded = false;
296 ACTIONENTRY *pIter = NULL;
297 RTListForEach(listActions, pIter, ACTIONENTRY, nodeActionEntry)
298 {
299 if (pIter->u32Index > nIndex)
300 {
301 RTListNodeInsertBefore(&pIter->nodeActionEntry, &pEntry->nodeActionEntry);
302 fAdded = true;
303 break;
304 }
305 }
306 if (!fAdded)
307 {
308 RTListAppend(listActions, &pEntry->nodeActionEntry);
309 }
310 }
311
312 LALOG(("LA: added %d %ls\n",
313 pEntry->u32Index, pEntry->wszCommandLine));
314 }
315
316 RegCloseKey(hKey);
317
318#ifdef LOG_ENABLED
319 ACTIONENTRY *pIter = NULL;
320 RTListForEach(listActions, pIter, ACTIONENTRY, nodeActionEntry)
321 {
322 LALOG(("LA: [%u]: [%ls]\n",
323 pIter->u32Index, pIter->wszCommandLine));
324 }
325#endif
326
327 if (!bRet)
328 {
329 ActionExecutorDeleteActions(listActions);
330 }
331
332 LALOG(("LA: action enum %d\n",
333 bRet));
334
335 return bRet;
336}
337
338static void ActionExecutorExecuteActions(RTLISTANCHOR *listActions)
339{
340 LALOG(("LA: ExecuteActions\n"));
341
342 ACTIONENTRY *pIter = NULL;
343 RTListForEach(listActions, pIter, ACTIONENTRY, nodeActionEntry)
344 {
345 LALOG(("LA: [%u]: [%ls]\n",
346 pIter->u32Index, pIter->wszCommandLine));
347
348 STARTUPINFOW si;
349 PROCESS_INFORMATION pi;
350
351 GetStartupInfoW(&si);
352
353 if (!CreateProcessW(NULL, // lpApplicationName
354 pIter->wszCommandLine, // lpCommandLine
355 NULL, // lpProcessAttributes
356 NULL, // lpThreadAttributes
357 FALSE, // bInheritHandles
358 0, // dwCreationFlags
359 NULL, // lpEnvironment
360 NULL, // lpCurrentDirectory
361 &si, // lpStartupInfo
362 &pi)) // lpProcessInformation
363 {
364 LALOG(("LA: Executing [%ls] failed, error %d\n",
365 pIter->wszCommandLine, GetLastError()));
366 }
367 else
368 {
369 LALOG(("LA: Executing [%ls] succeeded\n",
370 pIter->wszCommandLine));
371
372 /* Don't care about waiting on the new process, so close these. */
373 CloseHandle(pi.hProcess);
374 CloseHandle(pi.hThread);
375 }
376 }
377
378 LALOG(("LA: ExecuteActions leave\n"));
379}
380
381static BOOL GetVolatileEnvironmentKey(WCHAR *pwszRegKey, DWORD cbRegKey)
382{
383 BOOL fFound = FALSE;
384
385 DWORD nSessionID;
386 LONG lErr;
387 HKEY hKey;
388 char szRegKey[REG_KEY_LEN];
389
390 /* Attempt to open HKCU\Volatile Environment\<session ID> first. */
391 if ( gCtx.pfnProcessIdToSessionId != NULL
392 && gCtx.pfnProcessIdToSessionId(GetCurrentProcessId(), &nSessionID))
393 {
394 RTStrPrintf(szRegKey, sizeof(szRegKey),
395 "%s\\%d",
396 g_pszVolatileEnvironment, nSessionID);
397
398 lErr = RegOpenKeyExA(HKEY_CURRENT_USER,
399 szRegKey,
400 0,
401 KEY_SET_VALUE,
402 &hKey);
403
404 if (lErr == ERROR_SUCCESS)
405 {
406 RegCloseKey(hKey);
407 fFound = TRUE;
408 }
409 }
410
411 if (!fFound)
412 {
413 /* Fall back to HKCU\Volatile Environment. */
414 RTStrPrintf(szRegKey, sizeof(szRegKey),
415 "%s",
416 g_pszVolatileEnvironment);
417
418 lErr = RegOpenKeyExA(HKEY_CURRENT_USER,
419 szRegKey,
420 0,
421 KEY_SET_VALUE,
422 &hKey);
423
424 if (lErr == ERROR_SUCCESS)
425 {
426 RegCloseKey(hKey);
427 fFound = TRUE;
428 }
429 }
430
431 if (fFound)
432 {
433 LALOG(("LA: GetVolatileEnvironmentKey: [%s]\n", szRegKey));
434
435 /* Convert szRegKey to Utf16 string. */
436 PRTUTF16 putf16Unicode = pwszRegKey;
437 size_t cchUnicode = cbRegKey / sizeof(WCHAR);
438
439 int rc = RTStrToUtf16Ex(szRegKey, RTSTR_MAX,
440 &putf16Unicode, cchUnicode, NULL);
441 if (RT_FAILURE(rc))
442 {
443 LALOG(("LA: RTStrToUtf16Ex failed %Rrc\n", rc));
444 fFound = FALSE;
445 }
446 else
447 {
448 LALOG(("LA: unicode [%ls]\n", putf16Unicode));
449 }
450 }
451 else
452 {
453 LALOG(("LA: GetVolatileEnvironmentKey: not found\n"));
454 }
455
456 return fFound;
457}
458
459static BOOL GetUtcInfoClientName(WCHAR *pwszClientName, DWORD cbClientName)
460{
461 LONG lErr;
462
463 WCHAR wszRegKey[REG_KEY_LEN];
464 if (!GetVolatileEnvironmentKey(wszRegKey, sizeof(wszRegKey)))
465 {
466 return FALSE;
467 }
468
469 HKEY hKey;
470 lErr = RegOpenKeyExW(HKEY_CURRENT_USER,
471 wszRegKey,
472 0,
473 KEY_QUERY_VALUE,
474 &hKey);
475
476 if (lErr != ERROR_SUCCESS)
477 {
478 LALOG(("LA: RegOpenKeyExW: failed [%ls]\n",
479 wszRegKey));
480 return FALSE;
481 }
482
483 DWORD nRegData;
484 DWORD dwType;
485 lErr = RegQueryValueExW(hKey,
486 g_pwszUTCINFOClientName,
487 NULL,
488 &dwType,
489 NULL,
490 &nRegData);
491
492 if (lErr != ERROR_SUCCESS)
493 {
494 LALOG(("LA: RegQueryValueExW: failed [%ls]\n",
495 wszRegKey));
496 RegCloseKey(hKey);
497 return FALSE;
498 }
499
500 if (nRegData >= cbClientName)
501 {
502 LALOG(("LA: buffer overflow reg %d, buffer %d, [%ls]\n",
503 nRegData, cbClientName, wszRegKey));
504 RegCloseKey(hKey);
505 return FALSE;
506 }
507
508 if (dwType != REG_SZ)
509 {
510 LALOG(("LA: wrong type %d, [%ls]\n",
511 dwType, wszRegKey));
512 RegCloseKey(hKey);
513 return FALSE;
514 }
515
516 ZeroMemory(pwszClientName, cbClientName);
517
518 lErr = RegQueryValueExW(hKey,
519 g_pwszUTCINFOClientName,
520 NULL,
521 NULL,
522 (BYTE *)pwszClientName,
523 &nRegData);
524
525 RegCloseKey(hKey);
526
527 if (lErr != ERROR_SUCCESS)
528 {
529 return FALSE;
530 }
531
532 return TRUE;
533}
534
535static BOOL SetClientName(const WCHAR *pwszClientName)
536{
537 LONG lErr;
538
539 WCHAR wszRegKey[REG_KEY_LEN];
540 if (!GetVolatileEnvironmentKey(wszRegKey, sizeof(wszRegKey)))
541 {
542 return FALSE;
543 }
544
545 HKEY hKey;
546 lErr = RegOpenKeyExW(HKEY_CURRENT_USER,
547 wszRegKey,
548 0,
549 KEY_SET_VALUE,
550 &hKey);
551
552 if (lErr != ERROR_SUCCESS)
553 {
554 return FALSE;
555 }
556
557 DWORD nClientName = (lstrlenW(pwszClientName) + 1) * sizeof(WCHAR);
558 lErr = RegSetValueExW(hKey,
559 g_pwszClientName,
560 0,
561 REG_SZ,
562 (BYTE*)pwszClientName,
563 nClientName);
564
565 RegCloseKey(hKey);
566
567 if (lErr != ERROR_SUCCESS)
568 {
569 return FALSE;
570 }
571
572 return TRUE;
573}
574
575static void laBroadcastSettingChange(void)
576{
577 DWORD_PTR dwResult;
578
579 if (SendMessageTimeoutA(HWND_BROADCAST,
580 WM_SETTINGCHANGE,
581 NULL,
582 (LPARAM)"Environment",
583 SMTO_ABORTIFHUNG,
584 5000,
585 &dwResult) == 0)
586 {
587 LALOG(("LA: SendMessageTimeout failed, error %d\n", GetLastError()));
588 }
589}
590
591static void laUpdateClientName(VBOXLACONTEXT *pCtx)
592{
593 WCHAR wszUtcInfoClientName[MAX_CLIENT_NAME_CHARS];
594
595 if (GetUtcInfoClientName(wszUtcInfoClientName, sizeof(wszUtcInfoClientName)))
596 {
597 if (SetClientName(wszUtcInfoClientName))
598 {
599 laBroadcastSettingChange();
600 }
601 }
602}
603
604static void laOnClientName(const char *pszClientName)
605{
606 /* pszClientName is UTF8, make an Unicode copy for registry. */
607 PRTUTF16 putf16UnicodeName = NULL;
608 size_t cchUnicodeName = 0;
609 int rc = RTStrToUtf16Ex(pszClientName, MAX_CLIENT_NAME_CHARS,
610 &putf16UnicodeName, 0, &cchUnicodeName);
611
612 if (RT_FAILURE(rc))
613 {
614 LALOG(("LA: RTStrToUniEx failed %Rrc\n", rc));
615 return;
616 }
617
618 LALOG(("LA: unicode %d chars [%ls]\n", cchUnicodeName, putf16UnicodeName));
619
620 /*
621 * Write the client name to:
622 * HKCU\Volatile Environment\UTCINFO_CLIENTNAME or
623 * HKCU\Volatile Environment\<SessionID>\UTCINFO_CLIENTNAME
624 * depending on whether this is a Terminal Services or desktop session
625 * respectively.
626 */
627 WCHAR wszRegKey[REG_KEY_LEN];
628 if (!GetVolatileEnvironmentKey(wszRegKey, sizeof(wszRegKey)))
629 {
630 LALOG(("LA: Failed to get 'Volatile Environment' registry key\n"));
631 RTUtf16Free(putf16UnicodeName);
632 return;
633 }
634
635 /* Now write the client name under the appropriate key. */
636 LONG lRet;
637 HKEY hKey;
638
639 lRet = RegOpenKeyExW(HKEY_CURRENT_USER,
640 wszRegKey,
641 0,
642 KEY_SET_VALUE,
643 &hKey);
644
645 if (lRet != ERROR_SUCCESS)
646 {
647 LALOG(("LA: Failed to open key [%ls], error %lu\n",
648 wszRegKey, lRet));
649 RTUtf16Free(putf16UnicodeName);
650 return;
651 }
652
653 DWORD nDataLength = (DWORD)((cchUnicodeName + 1) * sizeof(WCHAR));
654 lRet = RegSetValueExW(hKey,
655 g_pwszUTCINFOClientName,
656 0,
657 REG_SZ,
658 (BYTE *)putf16UnicodeName,
659 nDataLength);
660
661 if (lRet != ERROR_SUCCESS)
662 {
663 LALOG(("LA: RegSetValueExW failed error %lu\n", lRet));
664 }
665
666 RegCloseKey(hKey);
667
668 laBroadcastSettingChange();
669
670 /* Also, write the client name to the environment of this process, as it
671 * doesn't listen for WM_SETTINGCHANGE messages.
672 */
673 SetEnvironmentVariableW(g_pwszUTCINFOClientName, putf16UnicodeName);
674
675 RTUtf16Free(putf16UnicodeName);
676}
677
678static void laDoAttach(VBOXLACONTEXT *pCtx)
679{
680 LALOG(("LA: laDoAttach\n"));
681
682 /* Hardcoded action. */
683 laUpdateClientName(pCtx);
684
685 /* Process configured actions. */
686 ActionExecutorExecuteActions(&pCtx->listAttachActions);
687}
688
689static void laDoDetach(VBOXLACONTEXT *pCtx)
690{
691 LALOG(("LA: laDoDetach\n"));
692
693 /* Process configured actions. */
694 ActionExecutorExecuteActions(&pCtx->listDetachActions);
695}
696
697static int laGetProperty(uint32_t u32GuestPropHandle, const char *pszName, uint64_t *pu64Timestamp, char **ppszValue)
698{
699 int rc = VINF_SUCCESS;
700
701 /* The buffer for storing the data and its initial size. We leave a bit
702 * of space here in case the maximum values are raised.
703 */
704 uint32_t cbBuf = 1024;
705 void *pvBuf = NULL;
706
707 /* Because there is a race condition between our reading the size of a
708 * property and the guest updating it, we loop a few times here and
709 * hope. Actually this should never go wrong, as we are generous
710 * enough with buffer space.
711 */
712 unsigned i;
713 for (i = 0; i < 3; ++i)
714 {
715 void *pvTmpBuf = RTMemRealloc(pvBuf, cbBuf);
716 if (pvTmpBuf == NULL)
717 {
718 rc = VERR_NO_MEMORY;
719 break;
720 }
721
722 pvBuf = pvTmpBuf;
723
724 rc = VbglR3GuestPropRead(u32GuestPropHandle, pszName, pvBuf, cbBuf,
725 NULL, pu64Timestamp, NULL,
726 &cbBuf);
727 if (rc != VERR_BUFFER_OVERFLOW)
728 {
729 break;
730 }
731
732 cbBuf += 1024;
733 }
734
735 if (RT_SUCCESS(rc))
736 {
737 LALOG(("LA: laGetProperty: [%s]\n"
738 " value: [%s]\n"
739 " timestamp: %lld ns\n",
740 pszName, (char *)pvBuf, *pu64Timestamp));
741
742 *ppszValue = (char *)pvBuf;
743 }
744 else if (rc == VERR_NOT_FOUND)
745 {
746 LALOG(("LA: laGetProperty: not found [%s]\n", pszName));
747 }
748 else
749 {
750 LALOG(("LA: Failed to retrieve the property value, error %Rrc\n", rc));
751 }
752
753 return rc;
754}
755
756static int laWaitProperties(uint32_t u32GuestPropHandle,
757 const char *pszPatterns,
758 uint64_t u64LastTimestamp,
759 uint64_t *pu64Timestamp,
760 uint32_t u32Timeout)
761{
762 int rc = VINF_SUCCESS;
763
764 /* The buffer for storing the data and its initial size. We leave a bit
765 * of space here in case the maximum values are raised.
766 */
767 void *pvBuf = NULL;
768 uint32_t cbBuf = 4096;
769
770 /* Because there is a race condition between our reading the size of a
771 * property and the guest updating it, we loop a few times here and
772 * hope. Actually this should never go wrong, as we are generous
773 * enough with buffer space.
774 */
775 unsigned i;
776 for (i = 0; i < 3; ++i)
777 {
778 void *pvTmpBuf = RTMemRealloc(pvBuf, cbBuf);
779 if (NULL == pvTmpBuf)
780 {
781 rc = VERR_NO_MEMORY;
782 break;
783 }
784
785 pvBuf = pvTmpBuf;
786
787 rc = VbglR3GuestPropWait(u32GuestPropHandle, pszPatterns, pvBuf, cbBuf,
788 u64LastTimestamp, u32Timeout,
789 NULL /* ppszName */,
790 NULL /* ppszValue */,
791 pu64Timestamp,
792 NULL /* ppszFlags */,
793 &cbBuf);
794
795 if (rc != VERR_BUFFER_OVERFLOW)
796 {
797 break;
798 }
799
800 cbBuf += 1024;
801 }
802
803 RTMemFree(pvBuf);
804
805 return rc;
806}
807
808static int laGetUint32(uint32_t u32GuestPropHandle, const char *pszName, uint64_t *pu64Timestamp, uint32_t *pu32Value)
809{
810 uint64_t u64Timestamp = 0;
811 char *pszValue = NULL;
812
813 int rc = laGetProperty(u32GuestPropHandle,
814 pszName,
815 &u64Timestamp,
816 &pszValue);
817 if (RT_SUCCESS(rc))
818 {
819 if (pszValue && *pszValue)
820 {
821 uint32_t u32 = 0;
822 rc = RTStrToUInt32Full(pszValue, 10, &u32);
823
824 if (RT_SUCCESS(rc))
825 {
826 *pu64Timestamp = u64Timestamp;
827 *pu32Value = u32;
828 }
829 }
830 else
831 {
832 rc = VERR_NOT_SUPPORTED;
833 }
834 }
835
836 if (pszValue)
837 {
838 RTMemFree(pszValue);
839 }
840
841 LALOG(("LA: laGetUint32: rc = %Rrc, [%s]\n",
842 rc, pszName));
843
844 return rc;
845}
846
847static int laGetString(uint32_t u32GuestPropHandle, const char *pszName, uint64_t *pu64Timestamp, char **ppszValue)
848{
849 int rc = laGetProperty(u32GuestPropHandle,
850 pszName,
851 pu64Timestamp,
852 ppszValue);
853
854 LALOG(("LA: laGetString: rc = %Rrc, [%s]\n",
855 rc, pszName));
856
857 return rc;
858}
859
860static int laGetActiveClient(VBOXLACONTEXT *pCtx, uint64_t *pu64Timestamp, uint32_t *pu32Value)
861{
862 int rc = laGetUint32(pCtx->u32GuestPropHandle,
863 g_pszPropActiveClient,
864 pu64Timestamp,
865 pu32Value);
866
867 LALOG(("LA: laGetActiveClient: rc %Rrc, %d, %lld\n", rc, *pu32Value, *pu64Timestamp));
868
869 return rc;
870}
871
872static int laUpdateCurrentState(VBOXLACONTEXT *pCtx, uint32_t u32ActiveClientId, uint64_t u64ActiveClientTS)
873{
874 /* Prepare the current state for the active client.
875 * If u32ActiveClientId is 0, then there is no connected clients.
876 */
877 LALOG(("LA: laUpdateCurrentState: %u %lld\n",
878 u32ActiveClientId, u64ActiveClientTS));
879
880 int rc = VINF_SUCCESS;
881
882 int l;
883
884 pCtx->activeClient.u32LastAttach = ~0;
885 pCtx->activeClient.u64LastAttachTimestamp = u64ActiveClientTS;
886
887 if (pCtx->activeClient.pszLastName)
888 {
889 RTMemFree(pCtx->activeClient.pszLastName);
890 }
891 pCtx->activeClient.pszLastName = NULL;
892 pCtx->activeClient.u64LastNameTimestamp = u64ActiveClientTS;
893
894 if (pCtx->activeClient.pszPropName)
895 {
896 RTMemFree(pCtx->activeClient.pszPropName);
897 pCtx->activeClient.pszPropName = NULL;
898 }
899 if (u32ActiveClientId != 0)
900 {
901 l = RTStrAPrintf(&pCtx->activeClient.pszPropName,
902 g_pszPropNameTemplate,
903 u32ActiveClientId);
904 if (l == -1)
905 {
906 pCtx->activeClient.pszPropName = NULL;
907 rc = VERR_NO_MEMORY;
908 }
909 }
910
911 if (RT_SUCCESS(rc))
912 {
913 if (pCtx->activeClient.pszPropAttach)
914 {
915 RTMemFree(pCtx->activeClient.pszPropAttach);
916 pCtx->activeClient.pszPropAttach = NULL;
917 }
918 if (u32ActiveClientId != 0)
919 {
920 l = RTStrAPrintf(&pCtx->activeClient.pszPropAttach,
921 g_pszPropAttachTemplate,
922 u32ActiveClientId);
923 if (l == -1)
924 {
925 pCtx->activeClient.pszPropAttach = NULL;
926 rc = VERR_NO_MEMORY;
927 }
928 }
929 }
930
931 if (RT_SUCCESS(rc))
932 {
933 if (pCtx->activeClient.pszPropWaitPattern)
934 {
935 RTMemFree(pCtx->activeClient.pszPropWaitPattern);
936 pCtx->activeClient.pszPropWaitPattern = NULL;
937 }
938 if (u32ActiveClientId != 0)
939 {
940 l = RTStrAPrintf(&pCtx->activeClient.pszPropWaitPattern,
941 "%s|%s",
942 pCtx->activeClient.pszPropName,
943 pCtx->activeClient.pszPropAttach);
944 if (l == -1)
945 {
946 pCtx->activeClient.pszPropWaitPattern = NULL;
947 rc = VERR_NO_MEMORY;
948 }
949 }
950 }
951
952 if (RT_SUCCESS(rc))
953 {
954 pCtx->activeClient.u32ClientId = u32ActiveClientId;
955 }
956 else
957 {
958 pCtx->activeClient.u32ClientId = 0;
959 }
960
961 LALOG(("LA: laUpdateCurrentState rc = %Rrc\n",
962 rc));
963
964 return rc;
965}
966
967static int laWait(VBOXLACONTEXT *pCtx, uint64_t *pu64Timestamp, uint32_t u32Timeout)
968{
969 LALOG(("LA: laWait [%s]\n",
970 pCtx->activeClient.pszPropWaitPattern));
971
972 int rc = laWaitProperties(pCtx->u32GuestPropHandle,
973 pCtx->activeClient.pszPropWaitPattern,
974 pCtx->u64LastQuery,
975 pu64Timestamp,
976 u32Timeout);
977
978 LALOG(("LA: laWait rc %Rrc\n",
979 rc));
980
981 return rc;
982}
983
984static void laProcessName(VBOXLACONTEXT *pCtx)
985{
986 /* Check if the name was changed. */
987 /* Get the name string and check if it was changed since last time.
988 * Write the name to the registry if changed.
989 */
990 uint64_t u64Timestamp = 0;
991 char *pszName = NULL;
992
993 int rc = laGetString(pCtx->u32GuestPropHandle,
994 pCtx->activeClient.pszPropName,
995 &u64Timestamp,
996 &pszName);
997 if (RT_SUCCESS(rc))
998 {
999 LALOG(("LA: laProcessName: read [%s], at %lld\n",
1000 pszName, u64Timestamp));
1001
1002 if (u64Timestamp != pCtx->activeClient.u64LastNameTimestamp)
1003 {
1004 laOnClientName(pszName);
1005
1006 pCtx->activeClient.u64LastNameTimestamp = u64Timestamp;
1007 }
1008
1009 }
1010
1011 if (pszName)
1012 {
1013 RTMemFree(pszName);
1014 }
1015}
1016
1017static void laProcessAttach(VBOXLACONTEXT *pCtx)
1018{
1019 /* Check if the attach was changed. */
1020 pCtx->u32Action = LA_DO_NOTHING;
1021
1022 uint64_t u64Timestamp = 0;
1023 uint32_t u32Attach = ~0;
1024
1025 int rc = laGetUint32(pCtx->u32GuestPropHandle,
1026 pCtx->activeClient.pszPropAttach,
1027 &u64Timestamp,
1028 &u32Attach);
1029
1030 if (RT_SUCCESS(rc))
1031 {
1032 LALOG(("LA: laProcessAttach: read %d, at %lld\n",
1033 u32Attach, u64Timestamp));
1034
1035 if (u64Timestamp != pCtx->activeClient.u64LastAttachTimestamp)
1036 {
1037 if (u32Attach != pCtx->activeClient.u32LastAttach)
1038 {
1039 LALOG(("LA: laProcessAttach: changed\n"));
1040
1041 /* Just do the last action. */
1042 pCtx->u32Action = u32Attach?
1043 LA_DO_ATTACH:
1044 LA_DO_DETACH;
1045
1046 pCtx->activeClient.u32LastAttach = u32Attach;
1047 }
1048 else
1049 {
1050 LALOG(("LA: laProcessAttach: same\n"));
1051
1052 /* The property has changed but the value is the same,
1053 * which means that it was changed and restored.
1054 */
1055 pCtx->u32Action = u32Attach?
1056 LA_DO_DETACH_AND_ATTACH:
1057 LA_DO_ATTACH_AND_DETACH;
1058 }
1059
1060 pCtx->activeClient.u64LastAttachTimestamp = u64Timestamp;
1061 }
1062
1063 }
1064
1065 LALOG(("LA: laProcessAttach: action %d\n",
1066 pCtx->u32Action));
1067}
1068
1069static void laDoActions(VBOXLACONTEXT *pCtx)
1070{
1071 /* Check if the attach was changed.
1072 *
1073 * Caller assumes that this function will filter double actions.
1074 * That is two or more LA_DO_ATTACH will do just one LA_DO_ATTACH.
1075 */
1076 LALOG(("LA: laDoActions: action %d, prev %d\n",
1077 pCtx->u32Action, pCtx->u32PrevAction));
1078
1079 switch(pCtx->u32Action)
1080 {
1081 case LA_DO_ATTACH:
1082 {
1083 if (pCtx->u32PrevAction != LA_DO_ATTACH)
1084 {
1085 pCtx->u32PrevAction = LA_DO_ATTACH;
1086 laDoAttach(pCtx);
1087 }
1088 } break;
1089
1090 case LA_DO_DETACH:
1091 {
1092 if (pCtx->u32PrevAction != LA_DO_DETACH)
1093 {
1094 pCtx->u32PrevAction = LA_DO_DETACH;
1095 laDoDetach(pCtx);
1096 }
1097 } break;
1098
1099 case LA_DO_DETACH_AND_ATTACH:
1100 {
1101 if (pCtx->u32PrevAction != LA_DO_DETACH)
1102 {
1103 pCtx->u32PrevAction = LA_DO_DETACH;
1104 laDoDetach(pCtx);
1105 }
1106 pCtx->u32PrevAction = LA_DO_ATTACH;
1107 laDoAttach(pCtx);
1108 } break;
1109
1110 case LA_DO_ATTACH_AND_DETACH:
1111 {
1112 if (pCtx->u32PrevAction != LA_DO_ATTACH)
1113 {
1114 pCtx->u32PrevAction = LA_DO_ATTACH;
1115 laDoAttach(pCtx);
1116 }
1117 pCtx->u32PrevAction = LA_DO_DETACH;
1118 laDoDetach(pCtx);
1119 } break;
1120
1121 case LA_DO_NOTHING:
1122 default:
1123 break;
1124 }
1125
1126 pCtx->u32Action = LA_DO_NOTHING;
1127
1128 LALOG(("LA: laDoActions: leave\n"));
1129}
1130
1131int VBoxLAInit(const VBOXSERVICEENV *pEnv, void **ppInstance, bool *pfStartThread)
1132{
1133 gCtx.pEnv = pEnv;
1134
1135 DWORD dwValue = 0;
1136 if ( laGetRegistryDWORD(L"SOFTWARE\\Oracle\\VirtualBox Guest Additions", L"VBoxTrayLog", &dwValue)
1137 && (dwValue & 0x10) != 0)
1138 {
1139 gCtx.fLogEnabled = true;
1140 }
1141 else
1142 {
1143 gCtx.fLogEnabled = false;
1144 }
1145
1146 LALOG(("VBoxTray: VBoxLAInit\n"));
1147
1148 dwValue = 0;
1149 if ( laGetRegistryDWORD(L"SOFTWARE\\Oracle\\VirtualBox Guest Additions", L"VBoxTrayLA", &dwValue)
1150 && (dwValue & 0x02) != 0)
1151 {
1152 gCtx.fDetachOnDisconnect = true;
1153 }
1154 else
1155 {
1156 gCtx.fDetachOnDisconnect = false;
1157 }
1158
1159 LALOG(("VBoxTray: VBoxLAInit: VBoxTrayLA %x\n", dwValue));
1160
1161 int rc = VbglR3GuestPropConnect(&gCtx.u32GuestPropHandle);
1162 if (RT_FAILURE(rc))
1163 {
1164 return rc;
1165 }
1166
1167 RTListInit(&gCtx.listAttachActions);
1168 RTListInit(&gCtx.listDetachActions);
1169
1170 RT_ZERO(gCtx.activeClient);
1171
1172 gCtx.hModuleKernel32 = LoadLibrary("KERNEL32");
1173
1174 if (gCtx.hModuleKernel32)
1175 {
1176 *(uintptr_t *)&gCtx.pfnProcessIdToSessionId = (uintptr_t)GetProcAddress(gCtx.hModuleKernel32, "ProcessIdToSessionId");
1177 }
1178 else
1179 {
1180 gCtx.pfnProcessIdToSessionId = NULL;
1181 }
1182 *pfStartThread = true;
1183 *ppInstance = &gCtx;
1184 return VINF_SUCCESS;
1185}
1186
1187
1188void VBoxLADestroy(const VBOXSERVICEENV *pEnv, void *pInstance)
1189{
1190 LALOG(("VBoxTray: VBoxLADestroy\n"));
1191
1192 VBOXLACONTEXT *pCtx = (VBOXLACONTEXT *)pInstance;
1193
1194 if (pCtx->u32GuestPropHandle != 0)
1195 {
1196 VbglR3GuestPropDisconnect(pCtx->u32GuestPropHandle);
1197 }
1198
1199 ActionExecutorDeleteActions(&pCtx->listAttachActions);
1200 ActionExecutorDeleteActions(&pCtx->listDetachActions);
1201
1202 if (pCtx->hModuleKernel32)
1203 {
1204 FreeLibrary(pCtx->hModuleKernel32);
1205 pCtx->pfnProcessIdToSessionId = NULL;
1206 }
1207 pCtx->hModuleKernel32 = NULL;
1208}
1209
1210/*
1211 * Thread function to wait for and process property changes
1212 */
1213unsigned __stdcall VBoxLAThread(void *pInstance)
1214{
1215 VBOXLACONTEXT *pCtx = (VBOXLACONTEXT *)pInstance;
1216
1217 LALOG(("VBoxTray: VBoxLAThread: Started.\n"));
1218
1219 /*
1220 * On name change event (/VirtualBox/HostInfo/VRDP/Client/%ID%/Name)
1221 * Store the name in the registry (HKCU\Volatile Environment\UTCINFO_CLIENTNAME).
1222 * On a client attach event (/VirtualBox/HostInfo/VRDP/Client/%ID%/Attach -> 1):
1223 * Execute ReconnectActions
1224 * On a client detach event (/VirtualBox/HostInfo/VRDP/Client/%ID%/Attach -> 0):
1225 * Execute DisconnectActions
1226 *
1227 * The active connected client id is /VirtualBox/HostInfo/VRDP/ActiveClientClient.
1228 */
1229
1230 if (!ActionExecutorEnumerateRegistryKey(g_pwszRegKeyReconnectActions, &gCtx.listAttachActions))
1231 {
1232 LALOG(("LA: Can't enumerate registry key %ls\n", g_pwszRegKeyReconnectActions));
1233 }
1234 if (!ActionExecutorEnumerateRegistryKey(g_pwszRegKeyDisconnectActions, &gCtx.listDetachActions))
1235 {
1236 LALOG(("LA: Can't enumerate registry key %ls\n", g_pwszRegKeyDisconnectActions));
1237 }
1238
1239 /* A non zero timestamp in the past. */
1240 pCtx->u64LastQuery = 1;
1241 /* Start at Detached state. */
1242 pCtx->u32PrevAction = LA_DO_DETACH;
1243
1244 for (;;)
1245 {
1246 /* Query current ActiveClient.
1247 * if it differs from the current active client
1248 * rebuild the context;
1249 * wait with timeout for properties change since the active client was changed;
1250 * if 'Name' was changed
1251 * update the name;
1252 * if 'Attach' was changed
1253 * do respective actions.
1254 * remember the query timestamp;
1255 */
1256 uint64_t u64Timestamp = 0;
1257 uint32_t u32ActiveClientId = 0;
1258 int rc = laGetActiveClient(pCtx, &u64Timestamp, &u32ActiveClientId);
1259
1260 if (RT_SUCCESS(rc))
1261 {
1262 bool fClientIdChanged = pCtx->activeClient.u32ClientId != u32ActiveClientId;
1263
1264 if (fClientIdChanged)
1265 {
1266 rc = laUpdateCurrentState(pCtx, u32ActiveClientId, u64Timestamp);
1267 }
1268
1269 if (RT_SUCCESS(rc))
1270 {
1271 if (pCtx->activeClient.u32ClientId != 0)
1272 {
1273 rc = laWait(pCtx, &u64Timestamp, 1000);
1274
1275 if (RT_SUCCESS(rc))
1276 {
1277 laProcessAttach(pCtx);
1278
1279 laProcessName(pCtx);
1280
1281 laDoActions(pCtx);
1282
1283 pCtx->u64LastQuery = u64Timestamp;
1284 }
1285 }
1286 else
1287 {
1288 /* If the client has been disconnected, do the detach actions. */
1289 if ( pCtx->fDetachOnDisconnect
1290 && fClientIdChanged)
1291 {
1292 LALOG(("LA: client disconnected\n"));
1293
1294 /* laDoActions will prevent a repeated detach action. So if there
1295 * was a detach already, then this detach will be ignored.
1296 */
1297 pCtx->u32Action = LA_DO_DETACH;
1298
1299 laDoActions(pCtx);
1300
1301 pCtx->u64LastQuery = u64Timestamp;
1302 }
1303 }
1304 }
1305 }
1306
1307 /* Check if it is time to exit.
1308 * If the code above failed, wait a bit until repeating to avoid a loop.
1309 * Otherwise just check if the stop event was signalled.
1310 */
1311 uint32_t u32Wait;
1312 if ( rc == VERR_NOT_FOUND
1313 || pCtx->activeClient.u32ClientId == 0)
1314 {
1315 /* No connections, wait longer. */
1316 u32Wait = 5000;
1317 }
1318 else if (RT_FAILURE(rc))
1319 {
1320 u32Wait = 1000;
1321 }
1322 else
1323 {
1324 u32Wait = 0;
1325 }
1326 if (WaitForSingleObject(pCtx->pEnv->hStopEvent, u32Wait) == WAIT_OBJECT_0)
1327 {
1328 break;
1329 }
1330 }
1331
1332 LALOG(("VBoxTray: VBoxLAThread: Finished.\n"));
1333 return 0;
1334}
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