VirtualBox

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

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

VBoxMMR: Switch to using Windows TSMF API hooking for redirection

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 26.5 KB
Line 
1/* $Id: VBoxTray.cpp 42003 2012-07-03 14:57:45Z vboxsync $ */
2/** @file
3 * VBoxTray - Guest Additions Tray Application
4 */
5
6/*
7 * Copyright (C) 2006-2011 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
19/*******************************************************************************
20* Header Files *
21*******************************************************************************/
22#include "VBoxTray.h"
23#include "VBoxTrayMsg.h"
24#include "VBoxHelpers.h"
25#include "VBoxSeamless.h"
26#include "VBoxClipboard.h"
27#include "VBoxDisplay.h"
28#include "VBoxRestore.h"
29#include "VBoxVRDP.h"
30#include "VBoxHostVersion.h"
31#include "VBoxSharedFolders.h"
32#include "VBoxIPC.h"
33#include "VBoxLA.h"
34#include "VBoxMMR.h"
35#include <VBoxHook.h>
36#include "resource.h"
37#include <malloc.h>
38#include <VBoxGuestInternal.h>
39
40#include <sddl.h>
41
42#include <iprt/buildconfig.h>
43
44
45/*******************************************************************************
46* Internal Functions *
47*******************************************************************************/
48static int vboxTrayCreateTrayIcon(void);
49static LRESULT CALLBACK vboxToolWndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam);
50
51/* Global message handler prototypes. */
52static int vboxTrayGlMsgTaskbarCreated(WPARAM lParam, LPARAM wParam);
53/*static int vboxTrayGlMsgShowBalloonMsg(WPARAM lParam, LPARAM wParam);*/
54
55
56/*******************************************************************************
57* Global Variables *
58*******************************************************************************/
59HANDLE ghVBoxDriver;
60HANDLE ghStopSem;
61HANDLE ghSeamlessNotifyEvent = 0;
62SERVICE_STATUS gVBoxServiceStatus;
63SERVICE_STATUS_HANDLE gVBoxServiceStatusHandle;
64HINSTANCE ghInstance;
65HWND ghwndToolWindow;
66NOTIFYICONDATA gNotifyIconData;
67DWORD gMajorVersion;
68
69
70/* The service table. */
71static VBOXSERVICEINFO vboxServiceTable[] =
72{
73 {
74 "Display",
75 VBoxDisplayInit,
76 VBoxDisplayThread,
77 VBoxDisplayDestroy
78 },
79 {
80 "Shared Clipboard",
81 VBoxClipboardInit,
82 VBoxClipboardThread,
83 VBoxClipboardDestroy
84 },
85 {
86 "Seamless Windows",
87 VBoxSeamlessInit,
88 VBoxSeamlessThread,
89 VBoxSeamlessDestroy
90 },
91#ifdef VBOX_WITH_VRDP_SESSION_HANDLING
92 {
93 "Restore",
94 VBoxRestoreInit,
95 VBoxRestoreThread,
96 VBoxRestoreDestroy
97 },
98#endif
99 {
100 "VRDP",
101 VBoxVRDPInit,
102 VBoxVRDPThread,
103 VBoxVRDPDestroy
104 },
105 {
106 "IPC",
107 VBoxIPCInit,
108 VBoxIPCThread,
109 VBoxIPCDestroy
110 },
111 {
112 "Location Awareness",
113 VBoxLAInit,
114 VBoxLAThread,
115 VBoxLADestroy
116 },
117#ifdef VBOX_WITH_MMR
118 {
119 "Multimedia Redirection",
120 VBoxMMRInit,
121 VBoxMMRThread,
122 VBoxMMRDestroy
123 },
124#endif
125 {
126 NULL
127 }
128};
129
130/* The global message table. */
131static VBOXGLOBALMESSAGE s_vboxGlobalMessageTable[] =
132{
133 /* Windows specific stuff. */
134 {
135 "TaskbarCreated",
136 vboxTrayGlMsgTaskbarCreated
137 },
138
139 /* VBoxTray specific stuff. */
140 /** @todo Add new messages here! */
141
142 {
143 NULL
144 }
145};
146
147/**
148 * Gets called whenever the Windows main taskbar
149 * get (re-)created. Nice to install our tray icon.
150 *
151 * @return IPRT status code.
152 * @param wParam
153 * @param lParam
154 */
155static int vboxTrayGlMsgTaskbarCreated(WPARAM wParam, LPARAM lParam)
156{
157 return vboxTrayCreateTrayIcon();
158}
159
160static int vboxTrayCreateTrayIcon(void)
161{
162 HICON hIcon = LoadIcon(ghInstance, MAKEINTRESOURCE(IDI_VIRTUALBOX));
163 if (hIcon == NULL)
164 {
165 DWORD dwErr = GetLastError();
166 Log(("VBoxTray: Could not load tray icon, error %08X\n", dwErr));
167 return RTErrConvertFromWin32(dwErr);
168 }
169
170 /* Prepare the system tray icon. */
171 RT_ZERO(gNotifyIconData);
172 gNotifyIconData.cbSize = NOTIFYICONDATA_V1_SIZE; // sizeof(NOTIFYICONDATA);
173 gNotifyIconData.hWnd = ghwndToolWindow;
174 gNotifyIconData.uID = ID_TRAYICON;
175 gNotifyIconData.uFlags = NIF_ICON | NIF_MESSAGE | NIF_TIP;
176 gNotifyIconData.uCallbackMessage = WM_VBOXTRAY_TRAY_ICON;
177 gNotifyIconData.hIcon = hIcon;
178
179 sprintf(gNotifyIconData.szTip, "%s Guest Additions %d.%d.%dr%d",
180 VBOX_PRODUCT, VBOX_VERSION_MAJOR, VBOX_VERSION_MINOR, VBOX_VERSION_BUILD, VBOX_SVN_REV);
181
182 int rc = VINF_SUCCESS;
183 if (!Shell_NotifyIcon(NIM_ADD, &gNotifyIconData))
184 {
185 DWORD dwErr = GetLastError();
186 Log(("VBoxTray: Could not create tray icon, error = %08X\n", dwErr));
187 rc = RTErrConvertFromWin32(dwErr);
188 RT_ZERO(gNotifyIconData);
189 }
190
191 if (hIcon)
192 DestroyIcon(hIcon);
193 return rc;
194}
195
196static void vboxTrayRemoveTrayIcon()
197{
198 if (gNotifyIconData.cbSize > 0)
199 {
200 /* Remove the system tray icon and refresh system tray. */
201 Shell_NotifyIcon(NIM_DELETE, &gNotifyIconData);
202 HWND hTrayWnd = FindWindow("Shell_TrayWnd", NULL); /* We assume we only have one tray atm. */
203 if (hTrayWnd)
204 {
205 HWND hTrayNotifyWnd = FindWindowEx(hTrayWnd, 0, "TrayNotifyWnd", NULL);
206 if (hTrayNotifyWnd)
207 SendMessage(hTrayNotifyWnd, WM_PAINT, 0, NULL);
208 }
209 RT_ZERO(gNotifyIconData);
210 }
211}
212
213static int vboxTrayStartServices(VBOXSERVICEENV *pEnv, VBOXSERVICEINFO *pTable)
214{
215 Log(("VBoxTray: Starting services ...\n"));
216
217 pEnv->hStopEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
218
219 if (!pEnv->hStopEvent)
220 {
221 /* Could not create event. */
222 return VERR_NOT_SUPPORTED;
223 }
224
225 while (pTable->pszName)
226 {
227 Log(("VBoxTray: Starting %s ...\n", pTable->pszName));
228
229 int rc = VINF_SUCCESS;
230
231 bool fStartThread = false;
232
233 pTable->hThread = (HANDLE)0;
234 pTable->pInstance = NULL;
235 pTable->fStarted = false;
236
237 if (pTable->pfnInit)
238 rc = pTable->pfnInit (pEnv, &pTable->pInstance, &fStartThread);
239
240 if (RT_FAILURE(rc))
241 {
242 Log(("VBoxTray: Failed to initialize rc = %Rrc\n", rc));
243 }
244 else
245 {
246 if (pTable->pfnThread && fStartThread)
247 {
248 unsigned threadid;
249 pTable->hThread = (HANDLE)_beginthreadex(NULL, /* security */
250 0, /* stacksize */
251 pTable->pfnThread,
252 pTable->pInstance,
253 0, /* initflag */
254 &threadid);
255
256 if (pTable->hThread == (HANDLE)(0))
257 rc = VERR_NOT_SUPPORTED;
258 }
259
260 if (RT_SUCCESS(rc))
261 pTable->fStarted = true;
262 else
263 {
264 Log(("VBoxTray: Failed to start the thread\n"));
265 if (pTable->pfnDestroy)
266 pTable->pfnDestroy(pEnv, pTable->pInstance);
267 }
268 }
269
270 /* Advance to next table element. */
271 pTable++;
272 }
273
274 return VINF_SUCCESS;
275}
276
277static void vboxTrayStopServices(VBOXSERVICEENV *pEnv, VBOXSERVICEINFO *pTable)
278{
279 if (!pEnv->hStopEvent)
280 return;
281
282 /* Signal to all threads. */
283 SetEvent(pEnv->hStopEvent);
284
285 while (pTable->pszName)
286 {
287 if (pTable->fStarted)
288 {
289 if (pTable->pfnThread)
290 {
291 /* There is a thread, wait for termination. */
292 WaitForSingleObject(pTable->hThread, INFINITE);
293
294 CloseHandle(pTable->hThread);
295 pTable->hThread = 0;
296 }
297
298 if (pTable->pfnDestroy)
299 pTable->pfnDestroy (pEnv, pTable->pInstance);
300 pTable->fStarted = false;
301 }
302
303 /* Advance to next table element. */
304 pTable++;
305 }
306
307 CloseHandle(pEnv->hStopEvent);
308}
309
310static int vboxTrayRegisterGlobalMessages(PVBOXGLOBALMESSAGE pTable)
311{
312 int rc = VINF_SUCCESS;
313 if (pTable == NULL) /* No table to register? Skip. */
314 return rc;
315 while ( pTable->pszName
316 && RT_SUCCESS(rc))
317 {
318 /* Register global accessible window messages. */
319 pTable->uMsgID = RegisterWindowMessage(TEXT(pTable->pszName));
320 if (!pTable->uMsgID)
321 {
322 DWORD dwErr = GetLastError();
323 Log(("VBoxTray: Registering global message \"%s\" failed, error = %08X\n", dwErr));
324 rc = RTErrConvertFromWin32(dwErr);
325 }
326
327 /* Advance to next table element. */
328 pTable++;
329 }
330 return rc;
331}
332
333static bool vboxTrayHandleGlobalMessages(PVBOXGLOBALMESSAGE pTable, UINT uMsg,
334 WPARAM wParam, LPARAM lParam)
335{
336 if (pTable == NULL)
337 return false;
338 while (pTable && pTable->pszName)
339 {
340 if (pTable->uMsgID == uMsg)
341 {
342 if (pTable->pfnHandler)
343 pTable->pfnHandler(wParam, lParam);
344 return true;
345 }
346
347 /* Advance to next table element. */
348 pTable++;
349 }
350 return false;
351}
352
353static int vboxTrayOpenBaseDriver(void)
354{
355 /* Open VBox guest driver. */
356 DWORD dwErr = ERROR_SUCCESS;
357 ghVBoxDriver = CreateFile(VBOXGUEST_DEVICE_NAME,
358 GENERIC_READ | GENERIC_WRITE,
359 FILE_SHARE_READ | FILE_SHARE_WRITE,
360 NULL,
361 OPEN_EXISTING,
362 FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED,
363 NULL);
364 if (ghVBoxDriver == INVALID_HANDLE_VALUE)
365 {
366 dwErr = GetLastError();
367 LogRel(("VBoxTray: Could not open VirtualBox Guest Additions driver! Please install / start it first! Error = %08X\n", dwErr));
368 }
369 return RTErrConvertFromWin32(dwErr);
370}
371
372static void vboxTrayCloseBaseDriver(void)
373{
374 if (ghVBoxDriver)
375 {
376 CloseHandle(ghVBoxDriver);
377 ghVBoxDriver = NULL;
378 }
379}
380
381static void vboxTrayDestroyToolWindow(void)
382{
383 if (ghwndToolWindow)
384 {
385 Log(("VBoxTray: Destroying tool window ...\n"));
386
387 /* Destroy the tool window. */
388 DestroyWindow(ghwndToolWindow);
389 ghwndToolWindow = NULL;
390
391 UnregisterClass("VBoxTrayToolWndClass", ghInstance);
392 }
393}
394
395static int vboxTrayCreateToolWindow(void)
396{
397 DWORD dwErr = ERROR_SUCCESS;
398
399 /* Create a custom window class. */
400 WNDCLASS windowClass = {0};
401 windowClass.style = CS_NOCLOSE;
402 windowClass.lpfnWndProc = (WNDPROC)vboxToolWndProc;
403 windowClass.hInstance = ghInstance;
404 windowClass.hCursor = LoadCursor(NULL, IDC_ARROW);
405 windowClass.lpszClassName = "VBoxTrayToolWndClass";
406 if (!RegisterClass(&windowClass))
407 {
408 dwErr = GetLastError();
409 Log(("VBoxTray: Registering invisible tool window failed, error = %08X\n", dwErr));
410 }
411 else
412 {
413 /*
414 * Create our (invisible) tool window.
415 * Note: The window name ("VBoxTrayToolWnd") and class ("VBoxTrayToolWndClass") is
416 * needed for posting globally registered messages to VBoxTray and must not be
417 * changed! Otherwise things get broken!
418 *
419 */
420 ghwndToolWindow = CreateWindowEx(WS_EX_TOOLWINDOW | WS_EX_TRANSPARENT | WS_EX_TOPMOST,
421 "VBoxTrayToolWndClass", "VBoxTrayToolWnd",
422 WS_POPUPWINDOW,
423 -200, -200, 100, 100, NULL, NULL, ghInstance, NULL);
424 if (!ghwndToolWindow)
425 {
426 dwErr = GetLastError();
427 Log(("VBoxTray: Creating invisible tool window failed, error = %08X\n", dwErr));
428 }
429 else
430 {
431 /* Reload the cursor(s). */
432 hlpReloadCursor();
433
434 Log(("VBoxTray: Invisible tool window handle = %p\n", ghwndToolWindow));
435 }
436 }
437
438 if (dwErr != ERROR_SUCCESS)
439 vboxTrayDestroyToolWindow();
440 return RTErrConvertFromWin32(dwErr);
441}
442
443static int vboxTraySetupSeamless(void)
444{
445 OSVERSIONINFO info;
446 gMajorVersion = 5; /* Default to Windows XP. */
447 info.dwOSVersionInfoSize = sizeof(info);
448 if (GetVersionEx(&info))
449 {
450 Log(("VBoxTray: Windows version %ld.%ld\n", info.dwMajorVersion, info.dwMinorVersion));
451 gMajorVersion = info.dwMajorVersion;
452 }
453
454 /* We need to setup a security descriptor to allow other processes modify access to the seamless notification event semaphore. */
455 SECURITY_ATTRIBUTES SecAttr;
456 DWORD dwErr = ERROR_SUCCESS;
457 char secDesc[SECURITY_DESCRIPTOR_MIN_LENGTH];
458 BOOL fRC;
459
460 SecAttr.nLength = sizeof(SecAttr);
461 SecAttr.bInheritHandle = FALSE;
462 SecAttr.lpSecurityDescriptor = &secDesc;
463 InitializeSecurityDescriptor(SecAttr.lpSecurityDescriptor, SECURITY_DESCRIPTOR_REVISION);
464 fRC = SetSecurityDescriptorDacl(SecAttr.lpSecurityDescriptor, TRUE, 0, FALSE);
465 if (!fRC)
466 {
467 dwErr = GetLastError();
468 Log(("VBoxTray: SetSecurityDescriptorDacl failed with last error = %08X\n", dwErr));
469 }
470 else
471 {
472 /* For Vista and up we need to change the integrity of the security descriptor, too. */
473 if (gMajorVersion >= 6)
474 {
475 BOOL (WINAPI * pfnConvertStringSecurityDescriptorToSecurityDescriptorA)(LPCSTR StringSecurityDescriptor, DWORD StringSDRevision, PSECURITY_DESCRIPTOR *SecurityDescriptor, PULONG SecurityDescriptorSize);
476
477 HMODULE hModule = LoadLibrary("ADVAPI32.DLL");
478 if (!hModule)
479 {
480 dwErr = GetLastError();
481 Log(("VBoxTray: Loading module ADVAPI32.DLL failed with last error = %08X\n", dwErr));
482 }
483 else
484 {
485 PSECURITY_DESCRIPTOR pSD;
486 PACL pSacl = NULL;
487 BOOL fSaclPresent = FALSE;
488 BOOL fSaclDefaulted = FALSE;
489
490 *(uintptr_t *)&pfnConvertStringSecurityDescriptorToSecurityDescriptorA = (uintptr_t)GetProcAddress(hModule, "ConvertStringSecurityDescriptorToSecurityDescriptorA");
491
492 Log(("VBoxTray: pfnConvertStringSecurityDescriptorToSecurityDescriptorA = %x\n", pfnConvertStringSecurityDescriptorToSecurityDescriptorA));
493 if (pfnConvertStringSecurityDescriptorToSecurityDescriptorA)
494 {
495 fRC = pfnConvertStringSecurityDescriptorToSecurityDescriptorA("S:(ML;;NW;;;LW)", /* this means "low integrity" */
496 SDDL_REVISION_1, &pSD, NULL);
497 if (!fRC)
498 {
499 dwErr = GetLastError();
500 Log(("VBoxTray: ConvertStringSecurityDescriptorToSecurityDescriptorA failed with last error = %08X\n", dwErr));
501 }
502 else
503 {
504 fRC = GetSecurityDescriptorSacl(pSD, &fSaclPresent, &pSacl, &fSaclDefaulted);
505 if (!fRC)
506 {
507 dwErr = GetLastError();
508 Log(("VBoxTray: GetSecurityDescriptorSacl failed with last error = %08X\n", dwErr));
509 }
510 else
511 {
512 fRC = SetSecurityDescriptorSacl(SecAttr.lpSecurityDescriptor, TRUE, pSacl, FALSE);
513 if (!fRC)
514 {
515 dwErr = GetLastError();
516 Log(("VBoxTray: SetSecurityDescriptorSacl failed with last error = %08X\n", dwErr));
517 }
518 }
519 }
520 }
521 }
522 }
523
524 if ( dwErr == ERROR_SUCCESS
525 && gMajorVersion >= 5) /* Only for W2K and up ... */
526 {
527 ghSeamlessNotifyEvent = CreateEvent(&SecAttr, FALSE, FALSE, VBOXHOOK_GLOBAL_EVENT_NAME);
528 if (ghSeamlessNotifyEvent == NULL)
529 {
530 dwErr = GetLastError();
531 Log(("VBoxTray: CreateEvent for Seamless failed, last error = %08X\n", dwErr));
532 }
533 }
534 }
535 return RTErrConvertFromWin32(dwErr);
536}
537
538static void vboxTrayShutdownSeamless(void)
539{
540 if (ghSeamlessNotifyEvent)
541 {
542 CloseHandle(ghSeamlessNotifyEvent);
543 ghSeamlessNotifyEvent = NULL;
544 }
545}
546
547static int vboxTrayServiceMain(void)
548{
549 int rc = VINF_SUCCESS;
550 Log(("VBoxTray: Entering vboxTrayServiceMain\n"));
551
552 ghStopSem = CreateEvent(NULL, TRUE, FALSE, NULL);
553 if (ghStopSem == NULL)
554 {
555 rc = RTErrConvertFromWin32(GetLastError());
556 Log(("VBoxTray: CreateEvent for stopping VBoxTray failed, rc=%Rrc\n", rc));
557 }
558 else
559 {
560 /*
561 * Start services listed in the vboxServiceTable.
562 */
563 VBOXSERVICEENV svcEnv;
564 svcEnv.hInstance = ghInstance;
565 svcEnv.hDriver = ghVBoxDriver;
566
567 /* Initializes disp-if to default (XPDM) mode. */
568 VBoxDispIfInit(&svcEnv.dispIf); /* Cannot fail atm. */
569 #ifdef VBOX_WITH_WDDM
570 /*
571 * For now the display mode will be adjusted to WDDM mode if needed
572 * on display service initialization when it detects the display driver type.
573 */
574 #endif
575
576 /* Finally start all the built-in services! */
577 rc = vboxTrayStartServices(&svcEnv, vboxServiceTable);
578 if (RT_FAILURE(rc))
579 {
580 /* Terminate service if something went wrong. */
581 vboxTrayStopServices(&svcEnv, vboxServiceTable);
582 }
583 else
584 {
585 rc = vboxTrayCreateTrayIcon();
586 if ( RT_SUCCESS(rc)
587 && gMajorVersion >= 5) /* Only for W2K and up ... */
588 {
589 /* We're ready to create the tooltip balloon.
590 Check in 10 seconds (@todo make seconds configurable) ... */
591 SetTimer(ghwndToolWindow,
592 TIMERID_VBOXTRAY_CHECK_HOSTVERSION,
593 10 * 1000, /* 10 seconds */
594 NULL /* No timerproc */);
595 }
596
597 if (RT_SUCCESS(rc))
598 {
599 /* Do the Shared Folders auto-mounting stuff. */
600 rc = VBoxSharedFoldersAutoMount();
601 if (RT_SUCCESS(rc))
602 {
603 /* Report the host that we're up and running! */
604 hlpReportStatus(VBoxGuestFacilityStatus_Active);
605 }
606 }
607
608 if (RT_SUCCESS(rc))
609 {
610 /* Boost thread priority to make sure we wake up early for seamless window notifications
611 * (not sure if it actually makes any difference though). */
612 SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_HIGHEST);
613
614 /*
615 * Main execution loop
616 * Wait for the stop semaphore to be posted or a window event to arrive
617 */
618
619 DWORD dwEventCount = 2;
620 HANDLE hWaitEvent[2] = { ghStopSem, ghSeamlessNotifyEvent };
621
622 if (0 == ghSeamlessNotifyEvent) /* If seamless mode is not active / supported, reduce event array count. */
623 dwEventCount = 1;
624
625 Log(("VBoxTray: Number of events to wait in main loop: %ld\n", dwEventCount));
626 while (true)
627 {
628 DWORD waitResult = MsgWaitForMultipleObjectsEx(dwEventCount, hWaitEvent, 500, QS_ALLINPUT, 0);
629 waitResult = waitResult - WAIT_OBJECT_0;
630
631 /* Only enable for message debugging, lots of traffic! */
632 //Log(("VBoxTray: Wait result = %ld\n", waitResult));
633
634 if (waitResult == 0)
635 {
636 Log(("VBoxTray: Event 'Exit' triggered\n"));
637 /* exit */
638 break;
639 }
640 else if ( waitResult == 1
641 && ghSeamlessNotifyEvent != 0) /* Only jump in, if seamless is active! */
642 {
643 Log(("VBoxTray: Event 'Seamless' triggered\n"));
644
645 /* seamless window notification */
646 VBoxSeamlessCheckWindows();
647 }
648 else
649 {
650 /* timeout or a window message, handle it */
651 MSG msg;
652 while (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
653 {
654 Log(("VBoxTray: msg %p\n", msg.message));
655 if (msg.message == WM_QUIT)
656 {
657 Log(("VBoxTray: WM_QUIT!\n"));
658 SetEvent(ghStopSem);
659 continue;
660 }
661 TranslateMessage(&msg);
662 DispatchMessage(&msg);
663 }
664 }
665 }
666 Log(("VBoxTray: Returned from main loop, exiting ...\n"));
667 }
668 Log(("VBoxTray: Waiting for services to stop ...\n"));
669 vboxTrayStopServices(&svcEnv, vboxServiceTable);
670 } /* Services started */
671 CloseHandle(ghStopSem);
672 } /* Stop event created */
673
674 vboxTrayRemoveTrayIcon();
675
676 Log(("VBoxTray: Leaving vboxTrayServiceMain with rc=%Rrc\n", rc));
677 return rc;
678}
679
680/**
681 * Main function
682 */
683int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
684{
685 /* Do not use a global namespace ("Global\\") for mutex name here, will blow up NT4 compatibility! */
686 HANDLE hMutexAppRunning = CreateMutex(NULL, FALSE, "VBoxTray");
687 if ( hMutexAppRunning != NULL
688 && GetLastError() == ERROR_ALREADY_EXISTS)
689 {
690 /* Close the mutex for this application instance. */
691 CloseHandle (hMutexAppRunning);
692 hMutexAppRunning = NULL;
693 return 0;
694 }
695
696 LogRel(("VBoxTray: %s r%s\n", RTBldCfgVersion(), RTBldCfgRevisionStr()));
697
698 int rc = RTR3InitExeNoArguments(0);
699 if (RT_SUCCESS(rc))
700 {
701 rc = VbglR3Init();
702 if (RT_SUCCESS(rc))
703 rc = vboxTrayOpenBaseDriver();
704 }
705
706 if (RT_SUCCESS(rc))
707 {
708 /* Save instance handle. */
709 ghInstance = hInstance;
710
711 hlpReportStatus(VBoxGuestFacilityStatus_Init);
712 rc = vboxTrayCreateToolWindow();
713 if (RT_SUCCESS(rc))
714 {
715 rc = vboxTraySetupSeamless();
716 if (RT_SUCCESS(rc))
717 {
718 Log(("VBoxTray: Init successful\n"));
719 rc = vboxTrayServiceMain();
720 if (RT_SUCCESS(rc))
721 hlpReportStatus(VBoxGuestFacilityStatus_Terminating);
722 vboxTrayShutdownSeamless();
723 }
724 vboxTrayDestroyToolWindow();
725 }
726 if (RT_SUCCESS(rc))
727 hlpReportStatus(VBoxGuestFacilityStatus_Terminated);
728 }
729
730 if (RT_FAILURE(rc))
731 {
732 LogRel(("VBoxTray: Error while starting, rc=%Rrc\n", rc));
733 hlpReportStatus(VBoxGuestFacilityStatus_Failed);
734 }
735 LogRel(("VBoxTray: Ended\n"));
736 vboxTrayCloseBaseDriver();
737
738 /* Release instance mutex. */
739 if (hMutexAppRunning != NULL)
740 {
741 CloseHandle(hMutexAppRunning);
742 hMutexAppRunning = NULL;
743 }
744
745 VbglR3Term();
746 return RT_SUCCESS(rc) ? 0 : 1;
747}
748
749/**
750 * Window procedure for our tool window
751 */
752static LRESULT CALLBACK vboxToolWndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
753{
754 switch (uMsg)
755 {
756 case WM_CREATE:
757 {
758 Log(("VBoxTray: Tool window created\n"));
759
760 int rc = vboxTrayRegisterGlobalMessages(&s_vboxGlobalMessageTable[0]);
761 if (RT_FAILURE(rc))
762 Log(("VBoxTray: Error registering global window messages, rc=%Rrc\n", rc));
763 return 0;
764 }
765
766 case WM_CLOSE:
767 return 0;
768
769 case WM_DESTROY:
770 Log(("VBoxTray: Tool window destroyed\n"));
771 KillTimer(ghwndToolWindow, TIMERID_VBOXTRAY_CHECK_HOSTVERSION);
772 return 0;
773
774 case WM_TIMER:
775 switch (wParam)
776 {
777 case TIMERID_VBOXTRAY_CHECK_HOSTVERSION:
778 if (RT_SUCCESS(VBoxCheckHostVersion()))
779 {
780 /* After successful run we don't need to check again. */
781 KillTimer(ghwndToolWindow, TIMERID_VBOXTRAY_CHECK_HOSTVERSION);
782 }
783 return 0;
784
785 default:
786 break;
787 }
788 break; /* Make sure other timers get processed the usual way! */
789
790 case WM_VBOXTRAY_TRAY_ICON:
791 switch (lParam)
792 {
793 case WM_LBUTTONDBLCLK:
794 break;
795
796 case WM_RBUTTONDOWN:
797 break;
798 }
799 return 0;
800
801 case WM_VBOX_INSTALL_SEAMLESS_HOOK:
802 VBoxSeamlessInstallHook();
803 return 0;
804
805 case WM_VBOX_REMOVE_SEAMLESS_HOOK:
806 VBoxSeamlessRemoveHook();
807 return 0;
808
809 case WM_VBOX_SEAMLESS_UPDATE:
810 VBoxSeamlessCheckWindows();
811 return 0;
812
813 case WM_VBOXTRAY_VM_RESTORED:
814 VBoxRestoreSession();
815 return 0;
816
817 case WM_VBOXTRAY_VRDP_CHECK:
818 VBoxRestoreCheckVRDP();
819 return 0;
820
821 default:
822
823 /* Handle all globally registered window messages. */
824 if (vboxTrayHandleGlobalMessages(&s_vboxGlobalMessageTable[0], uMsg,
825 wParam, lParam))
826 {
827 return 0; /* We handled the message. @todo Add return value!*/
828 }
829 break; /* We did not handle the message, dispatch to DefWndProc. */
830 }
831
832 /* Only if message was *not* handled by our switch above, dispatch
833 * to DefWindowProc. */
834 return DefWindowProc(hWnd, uMsg, wParam, lParam);
835}
836
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