VirtualBox

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

Last change on this file since 50035 was 46625, checked in by vboxsync, 12 years ago

unify casing

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