VirtualBox

source: vbox/trunk/src/VBox/Frontends/VBoxSDL/VBoxSDL.cpp@ 26951

Last change on this file since 26951 was 26782, checked in by vboxsync, 15 years ago

Main, Frontends: added support for virtual pointing devices with no relative reporting and cleaned up the VMMDev/mouse device absolute reporting interaction

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 173.1 KB
Line 
1/** @file
2 * VBox frontends: VBoxSDL (simple frontend based on SDL):
3 * Main code
4 */
5
6/*
7 * Copyright (C) 2006-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* Header Files *
24*******************************************************************************/
25#define LOG_GROUP LOG_GROUP_GUI
26
27#include <VBox/com/com.h>
28#include <VBox/com/string.h>
29#include <VBox/com/Guid.h>
30#include <VBox/com/array.h>
31#include <VBox/com/ErrorInfo.h>
32#include <VBox/com/errorprint.h>
33
34#include <VBox/com/EventQueue.h>
35#include <VBox/com/VirtualBox.h>
36
37using namespace com;
38
39#if defined (VBOXSDL_WITH_X11)
40# include <X11/Xlib.h>
41# include <X11/cursorfont.h> /* for XC_left_ptr */
42# if !defined (VBOX_WITHOUT_XCURSOR)
43# include <X11/Xcursor/Xcursor.h>
44# endif
45# include <unistd.h>
46#endif
47
48#ifndef RT_OS_DARWIN
49#include <SDL_syswm.h> /* for SDL_GetWMInfo() */
50#endif
51
52#include "VBoxSDL.h"
53#include "Framebuffer.h"
54#include "Helper.h"
55
56#include <VBox/types.h>
57#include <VBox/err.h>
58#include <VBox/param.h>
59#include <VBox/log.h>
60#include <VBox/version.h>
61#include <VBox/VBoxVideo.h>
62
63#include <iprt/alloca.h>
64#include <iprt/asm.h>
65#include <iprt/assert.h>
66#include <iprt/env.h>
67#include <iprt/file.h>
68#include <iprt/ldr.h>
69#include <iprt/initterm.h>
70#include <iprt/path.h>
71#include <iprt/process.h>
72#include <iprt/semaphore.h>
73#include <iprt/string.h>
74#include <iprt/stream.h>
75#include <iprt/uuid.h>
76
77#include <signal.h>
78
79#include <vector>
80#include <list>
81
82/* Xlib would re-define our enums */
83#undef True
84#undef False
85
86/*******************************************************************************
87* Defined Constants And Macros *
88*******************************************************************************/
89#ifdef VBOX_SECURELABEL
90/** extra data key for the secure label */
91#define VBOXSDL_SECURELABEL_EXTRADATA "VBoxSDL/SecureLabel"
92/** label area height in pixels */
93#define SECURE_LABEL_HEIGHT 20
94#endif
95
96/** Enables the rawr[0|3], patm, and casm options. */
97#define VBOXSDL_ADVANCED_OPTIONS
98
99/*******************************************************************************
100* Structures and Typedefs *
101*******************************************************************************/
102/** Pointer shape change event data strucure */
103struct PointerShapeChangeData
104{
105 PointerShapeChangeData (BOOL aVisible, BOOL aAlpha, ULONG aXHot, ULONG aYHot,
106 ULONG aWidth, ULONG aHeight, const uint8_t *aShape)
107 : visible (aVisible), alpha (aAlpha), xHot (aXHot), yHot (aYHot),
108 width (aWidth), height (aHeight), shape (NULL)
109 {
110 // make a copy of the shape
111 if (aShape)
112 {
113 uint32_t shapeSize = ((((aWidth + 7) / 8) * aHeight + 3) & ~3) + aWidth * 4 * aHeight;
114 shape = new uint8_t [shapeSize];
115 if (shape)
116 memcpy ((void *) shape, (void *) aShape, shapeSize);
117 }
118 }
119
120 ~PointerShapeChangeData()
121 {
122 if (shape) delete[] shape;
123 }
124
125 const BOOL visible;
126 const BOOL alpha;
127 const ULONG xHot;
128 const ULONG yHot;
129 const ULONG width;
130 const ULONG height;
131 const uint8_t *shape;
132};
133
134enum TitlebarMode
135{
136 TITLEBAR_NORMAL = 1,
137 TITLEBAR_STARTUP = 2,
138 TITLEBAR_SAVE = 3,
139 TITLEBAR_SNAPSHOT = 4
140};
141
142/*******************************************************************************
143* Internal Functions *
144*******************************************************************************/
145static bool UseAbsoluteMouse(void);
146static void ResetKeys(void);
147static void ProcessKey(SDL_KeyboardEvent *ev);
148static void InputGrabStart(void);
149static void InputGrabEnd(void);
150static void SendMouseEvent(VBoxSDLFB *fb, int dz, int button, int down);
151static void UpdateTitlebar(TitlebarMode mode, uint32_t u32User = 0);
152static void SetPointerShape(const PointerShapeChangeData *data);
153static void HandleGuestCapsChanged(void);
154static int HandleHostKey(const SDL_KeyboardEvent *pEv);
155static Uint32 StartupTimer(Uint32 interval, void *param);
156static Uint32 ResizeTimer(Uint32 interval, void *param);
157static Uint32 QuitTimer(Uint32 interval, void *param);
158static int WaitSDLEvent(SDL_Event *event);
159static void SetFullscreen(bool enable);
160#ifdef VBOX_WITH_SDL13
161static VBoxSDLFB * getFbFromWinId(SDL_WindowID id);
162#endif
163
164
165/*******************************************************************************
166* Global Variables *
167*******************************************************************************/
168#if defined (DEBUG_dmik)
169// my mini kbd doesn't have RCTRL...
170static int gHostKeyMod = KMOD_RSHIFT;
171static int gHostKeySym1 = SDLK_RSHIFT;
172static int gHostKeySym2 = SDLK_UNKNOWN;
173#else
174static int gHostKeyMod = KMOD_RCTRL;
175static int gHostKeySym1 = SDLK_RCTRL;
176static int gHostKeySym2 = SDLK_UNKNOWN;
177#endif
178static const char *gHostKeyDisabledCombinations = "";
179static const char *gpszPidFile;
180static BOOL gfGrabbed = FALSE;
181static BOOL gfGrabOnMouseClick = TRUE;
182static BOOL gfFullscreenResize = FALSE;
183static BOOL gfIgnoreNextResize = FALSE;
184static BOOL gfAllowFullscreenToggle = TRUE;
185static BOOL gfAbsoluteMouseHost = FALSE;
186static BOOL gfAbsoluteMouseGuest = FALSE;
187static BOOL gfRelativeMouseGuest = TRUE;
188static BOOL gfGuestNeedsHostCursor = FALSE;
189static BOOL gfOffCursorActive = FALSE;
190static BOOL gfGuestNumLockPressed = FALSE;
191static BOOL gfGuestCapsLockPressed = FALSE;
192static BOOL gfGuestScrollLockPressed = FALSE;
193static BOOL gfACPITerm = FALSE;
194static BOOL gfXCursorEnabled = FALSE;
195static int gcGuestNumLockAdaptions = 2;
196static int gcGuestCapsLockAdaptions = 2;
197static uint32_t gmGuestNormalXRes;
198static uint32_t gmGuestNormalYRes;
199
200/** modifier keypress status (scancode as index) */
201static uint8_t gaModifiersState[256];
202
203static ComPtr<IMachine> gMachine;
204static ComPtr<IConsole> gConsole;
205static ComPtr<IMachineDebugger> gMachineDebugger;
206static ComPtr<IKeyboard> gKeyboard;
207static ComPtr<IMouse> gMouse;
208static ComPtr<IDisplay> gDisplay;
209static ComPtr<IVRDPServer> gVrdpServer;
210static ComPtr<IProgress> gProgress;
211
212static ULONG gcMonitors = 1;
213static VBoxSDLFB *gpFramebuffer[64];
214static SDL_Cursor *gpDefaultCursor = NULL;
215#ifdef VBOXSDL_WITH_X11
216static Cursor gpDefaultOrigX11Cursor;
217#ifdef RT_OS_LINUX
218static BOOL guseEvdevKeymap = FALSE;
219#endif
220#endif
221static SDL_Cursor *gpCustomCursor = NULL;
222#ifndef VBOX_WITH_SDL13
223static WMcursor *gpCustomOrigWMcursor = NULL;
224#endif
225static SDL_Cursor *gpOffCursor = NULL;
226static SDL_TimerID gSdlResizeTimer = NULL;
227static SDL_TimerID gSdlQuitTimer = NULL;
228
229#if defined(VBOXSDL_WITH_X11) && !defined(VBOX_WITH_SDL13)
230static SDL_SysWMinfo gSdlInfo;
231#endif
232
233#ifdef VBOX_SECURELABEL
234#ifdef RT_OS_WINDOWS
235#define LIBSDL_TTF_NAME "SDL_ttf"
236#else
237#define LIBSDL_TTF_NAME "libSDL_ttf-2.0.so.0"
238#endif
239RTLDRMOD gLibrarySDL_ttf = NIL_RTLDRMOD;
240#endif
241
242static RTSEMEVENT g_EventSemSDLEvents;
243static volatile int32_t g_cNotifyUpdateEventsPending;
244
245/**
246 * Callback handler for VirtualBox events
247 */
248class VBoxSDLCallback :
249 VBOX_SCRIPTABLE_IMPL(IVirtualBoxCallback)
250{
251public:
252 VBoxSDLCallback()
253 {
254#if defined (RT_OS_WINDOWS)
255 refcnt = 0;
256#endif
257 }
258
259 virtual ~VBoxSDLCallback()
260 {
261 }
262
263#ifdef RT_OS_WINDOWS
264 STDMETHOD_(ULONG, AddRef)()
265 {
266 return ::InterlockedIncrement(&refcnt);
267 }
268 STDMETHOD_(ULONG, Release)()
269 {
270 long cnt = ::InterlockedDecrement(&refcnt);
271 if (cnt == 0)
272 delete this;
273 return cnt;
274 }
275#endif
276 VBOX_SCRIPTABLE_DISPATCH_IMPL(IVirtualBoxCallback)
277
278 NS_DECL_ISUPPORTS
279
280 STDMETHOD(OnMachineStateChange)(IN_BSTR machineId, MachineState_T state)
281 {
282 return S_OK;
283 }
284
285 STDMETHOD(OnMachineDataChange)(IN_BSTR machineId)
286 {
287 return S_OK;
288 }
289
290 STDMETHOD(OnExtraDataCanChange)(IN_BSTR machineId, IN_BSTR key, IN_BSTR value,
291 BSTR *error, BOOL *changeAllowed)
292 {
293 /* we never disagree */
294 if (!changeAllowed)
295 return E_INVALIDARG;
296 *changeAllowed = TRUE;
297 return S_OK;
298 }
299
300 STDMETHOD(OnExtraDataChange)(IN_BSTR machineId, IN_BSTR key, IN_BSTR value)
301 {
302#ifdef VBOX_SECURELABEL
303 Assert(key);
304 if (gMachine)
305 {
306 /*
307 * check if we're interested in the message
308 */
309 Bstr ourGuid;
310 gMachine->COMGETTER(Id)(ourGuid.asOutParam());
311 if (ourGuid == machineId)
312 {
313 Bstr keyString = key;
314 if (keyString && keyString == VBOXSDL_SECURELABEL_EXTRADATA)
315 {
316 /*
317 * Notify SDL thread of the string update
318 */
319 SDL_Event event = {0};
320 event.type = SDL_USEREVENT;
321 event.user.type = SDL_USER_EVENT_SECURELABEL_UPDATE;
322 PushSDLEventForSure(&event);
323 }
324 }
325 }
326#endif /* VBOX_SECURELABEL */
327 return S_OK;
328 }
329
330 STDMETHOD(OnMediumRegistered)(IN_BSTR mediaId, DeviceType_T mediaType,
331 BOOL registered)
332 {
333 NOREF (mediaId);
334 NOREF (mediaType);
335 NOREF (registered);
336 return S_OK;
337 }
338
339 STDMETHOD(OnMachineRegistered)(IN_BSTR machineId, BOOL registered)
340 {
341 return S_OK;
342 }
343
344 STDMETHOD(OnSessionStateChange)(IN_BSTR machineId, SessionState_T state)
345 {
346 return S_OK;
347 }
348
349 STDMETHOD(OnSnapshotTaken) (IN_BSTR aMachineId, IN_BSTR aSnapshotId)
350 {
351 return S_OK;
352 }
353
354 STDMETHOD(OnSnapshotDiscarded) (IN_BSTR aMachineId, IN_BSTR aSnapshotId)
355 {
356 return S_OK;
357 }
358
359 STDMETHOD(OnSnapshotChange) (IN_BSTR aMachineId, IN_BSTR aSnapshotId)
360 {
361 return S_OK;
362 }
363
364 STDMETHOD(OnGuestPropertyChange)(IN_BSTR machineId, IN_BSTR key, IN_BSTR value, IN_BSTR flags)
365 {
366 return S_OK;
367 }
368
369private:
370#ifdef RT_OS_WINDOWS
371 long refcnt;
372#endif
373
374};
375
376/**
377 * Callback handler for machine events
378 */
379class VBoxSDLConsoleCallback :
380 VBOX_SCRIPTABLE_IMPL(IConsoleCallback)
381{
382public:
383 VBoxSDLConsoleCallback() : m_fIgnorePowerOffEvents(false)
384 {
385#if defined (RT_OS_WINDOWS)
386 refcnt = 0;
387#endif
388 }
389
390 virtual ~VBoxSDLConsoleCallback()
391 {
392 }
393
394#ifdef RT_OS_WINDOWS
395 STDMETHOD_(ULONG, AddRef)()
396 {
397 return ::InterlockedIncrement(&refcnt);
398 }
399 STDMETHOD_(ULONG, Release)()
400 {
401 long cnt = ::InterlockedDecrement(&refcnt);
402 if (cnt == 0)
403 delete this;
404 return cnt;
405 }
406#endif
407 VBOX_SCRIPTABLE_DISPATCH_IMPL(IConsoleCallback)
408
409 NS_DECL_ISUPPORTS
410
411 STDMETHOD(OnMousePointerShapeChange) (BOOL visible, BOOL alpha, ULONG xHot, ULONG yHot,
412 ULONG width, ULONG height, BYTE *shape)
413 {
414 PointerShapeChangeData *data;
415 data = new PointerShapeChangeData (visible, alpha, xHot, yHot, width, height,
416 shape);
417 Assert (data);
418 if (!data)
419 return E_FAIL;
420
421 SDL_Event event = {0};
422 event.type = SDL_USEREVENT;
423 event.user.type = SDL_USER_EVENT_POINTER_CHANGE;
424 event.user.data1 = data;
425
426 int rc = PushSDLEventForSure (&event);
427 if (rc)
428 delete data;
429
430 return S_OK;
431 }
432
433 STDMETHOD(OnMouseCapabilityChange)(BOOL supportsAbsolute, BOOL supportsRelative, BOOL needsHostCursor)
434 {
435 LogFlow(("OnMouseCapabilityChange: supportsAbsolute = %d, supportsRelative = %d, needsHostCursor = %d\n",
436 supportsAbsolute, supportsRelative, needsHostCursor));
437 gfAbsoluteMouseGuest = supportsAbsolute;
438 gfRelativeMouseGuest = supportsRelative;
439 gfGuestNeedsHostCursor = needsHostCursor;
440
441 SDL_Event event = {0};
442 event.type = SDL_USEREVENT;
443 event.user.type = SDL_USER_EVENT_GUEST_CAP_CHANGED;
444
445 PushSDLEventForSure (&event);
446 return S_OK;
447 }
448
449 STDMETHOD(OnKeyboardLedsChange)(BOOL fNumLock, BOOL fCapsLock, BOOL fScrollLock)
450 {
451 /* Don't bother the guest with NumLock scancodes if he doesn't set the NumLock LED */
452 if (gfGuestNumLockPressed != fNumLock)
453 gcGuestNumLockAdaptions = 2;
454 if (gfGuestCapsLockPressed != fCapsLock)
455 gcGuestCapsLockAdaptions = 2;
456 gfGuestNumLockPressed = fNumLock;
457 gfGuestCapsLockPressed = fCapsLock;
458 gfGuestScrollLockPressed = fScrollLock;
459 return S_OK;
460 }
461
462 STDMETHOD(OnStateChange)(MachineState_T machineState)
463 {
464 LogFlow(("OnStateChange: machineState = %d (%s)\n", machineState, GetStateName(machineState)));
465 SDL_Event event = {0};
466
467 if ( machineState == MachineState_Aborted
468 || machineState == MachineState_Teleported
469 || (machineState == MachineState_Saved && !m_fIgnorePowerOffEvents)
470 || (machineState == MachineState_PoweredOff && !m_fIgnorePowerOffEvents)
471 )
472 {
473 /*
474 * We have to inform the SDL thread that the application has be terminated
475 */
476 event.type = SDL_USEREVENT;
477 event.user.type = SDL_USER_EVENT_TERMINATE;
478 event.user.code = machineState == MachineState_Aborted
479 ? VBOXSDL_TERM_ABEND
480 : VBOXSDL_TERM_NORMAL;
481 }
482 else
483 {
484 /*
485 * Inform the SDL thread to refresh the titlebar
486 */
487 event.type = SDL_USEREVENT;
488 event.user.type = SDL_USER_EVENT_UPDATE_TITLEBAR;
489 }
490
491 PushSDLEventForSure(&event);
492 return S_OK;
493 }
494
495 STDMETHOD(OnAdditionsStateChange)()
496 {
497 return S_OK;
498 }
499
500 STDMETHOD(OnNetworkAdapterChange) (INetworkAdapter *aNetworkAdapter)
501 {
502 return S_OK;
503 }
504
505 STDMETHOD(OnSerialPortChange) (ISerialPort *aSerialPort)
506 {
507 return S_OK;
508 }
509
510 STDMETHOD(OnParallelPortChange) (IParallelPort *aParallelPort)
511 {
512 return S_OK;
513 }
514
515 STDMETHOD(OnVRDPServerChange)()
516 {
517 return S_OK;
518 }
519
520 STDMETHOD(OnRemoteDisplayInfoChange)()
521 {
522 return S_OK;
523 }
524
525 STDMETHOD(OnUSBControllerChange)()
526 {
527 return S_OK;
528 }
529
530 STDMETHOD(OnUSBDeviceStateChange) (IUSBDevice *aDevice, BOOL aAttached,
531 IVirtualBoxErrorInfo *aError)
532 {
533 return S_OK;
534 }
535
536 STDMETHOD(OnSharedFolderChange) (Scope_T aScope)
537 {
538 return S_OK;
539 }
540
541 STDMETHOD(OnStorageControllerChange) ()
542 {
543 return S_OK;
544 }
545
546 STDMETHOD(OnMediumChange)(IMediumAttachment * /*aMediumAttachment*/)
547 {
548 return S_OK;
549 }
550
551 STDMETHOD(OnCPUChange)(ULONG /*aCPU*/, BOOL /* aRemove */)
552 {
553 return S_OK;
554 }
555
556 STDMETHOD(OnRuntimeError)(BOOL fFatal, IN_BSTR id, IN_BSTR message)
557 {
558 MachineState_T machineState;
559 gMachine->COMGETTER(State)(&machineState);
560 const char *pszType;
561 bool fPaused = machineState == MachineState_Paused;
562 if (fFatal)
563 pszType = "FATAL ERROR";
564 else if (machineState == MachineState_Paused)
565 pszType = "Non-fatal ERROR";
566 else
567 pszType = "WARNING";
568 RTPrintf("\n%s: ** %lS **\n%lS\n%s\n", pszType, id, message,
569 fPaused ? "The VM was paused. Continue with HostKey + P after you solved the problem.\n" : "");
570 return S_OK;
571 }
572
573 STDMETHOD(OnCanShowWindow)(BOOL *canShow)
574 {
575 if (!canShow)
576 return E_POINTER;
577#ifdef RT_OS_DARWIN
578 /* SDL feature not available on Quartz */
579 *canShow = TRUE;
580#else
581 SDL_SysWMinfo info;
582 SDL_VERSION(&info.version);
583 *canShow = !!SDL_GetWMInfo(&info);
584#endif
585 return S_OK;
586 }
587
588 STDMETHOD(OnShowWindow) (ULONG64 *winId)
589 {
590#ifndef RT_OS_DARWIN
591 SDL_SysWMinfo info;
592 SDL_VERSION(&info.version);
593 if (SDL_GetWMInfo(&info))
594 {
595#if defined (VBOXSDL_WITH_X11)
596 *winId = (ULONG64) info.info.x11.wmwindow;
597#elif defined (RT_OS_WINDOWS)
598 *winId = (ULONG64) info.window;
599#else
600 AssertFailed();
601 return E_FAIL;
602#endif
603 return S_OK;
604 }
605#endif /* !RT_OS_DARWIN */
606 AssertFailed();
607 return E_FAIL;
608 }
609
610 static const char *GetStateName(MachineState_T machineState)
611 {
612 switch (machineState)
613 {
614 case MachineState_Null: return "<null>";
615 case MachineState_PoweredOff: return "PoweredOff";
616 case MachineState_Saved: return "Saved";
617 case MachineState_Teleported: return "Teleported";
618 case MachineState_Aborted: return "Aborted";
619 case MachineState_Running: return "Running";
620 case MachineState_Teleporting: return "Teleporting";
621 case MachineState_LiveSnapshotting: return "LiveSnapshotting";
622 case MachineState_Paused: return "Paused";
623 case MachineState_Stuck: return "GuruMeditation";
624 case MachineState_Starting: return "Starting";
625 case MachineState_Stopping: return "Stopping";
626 case MachineState_Saving: return "Saving";
627 case MachineState_Restoring: return "Restoring";
628 case MachineState_TeleportingPausedVM: return "TeleportingPausedVM";
629 case MachineState_TeleportingIn: return "TeleportingIn";
630 case MachineState_RestoringSnapshot: return "RestoringSnapshot";
631 case MachineState_DeletingSnapshot: return "DeletingSnapshot";
632 case MachineState_SettingUp: return "SettingUp";
633 default: return "no idea";
634 }
635 }
636
637 void ignorePowerOffEvents(bool fIgnore)
638 {
639 m_fIgnorePowerOffEvents = fIgnore;
640 }
641
642private:
643#ifdef RT_OS_WINDOWS
644 long refcnt;
645#endif
646 bool m_fIgnorePowerOffEvents;
647};
648
649#ifdef VBOX_WITH_XPCOM
650NS_DECL_CLASSINFO(VBoxSDLCallback)
651NS_IMPL_ISUPPORTS1_CI(VBoxSDLCallback, IVirtualBoxCallback)
652NS_DECL_CLASSINFO(VBoxSDLConsoleCallback)
653NS_IMPL_ISUPPORTS1_CI(VBoxSDLConsoleCallback, IConsoleCallback)
654#endif /* VBOX_WITH_XPCOM */
655
656static void show_usage()
657{
658 RTPrintf("Usage:\n"
659 " --startvm <uuid|name> Virtual machine to start, either UUID or name\n"
660 " --hda <file> Set temporary first hard disk to file\n"
661 " --fda <file> Set temporary first floppy disk to file\n"
662 " --cdrom <file> Set temporary CDROM/DVD to file/device ('none' to unmount)\n"
663 " --boot <a|c|d|n> Set temporary boot device (a = floppy, c = 1st HD, d = DVD, n = network)\n"
664 " --memory <size> Set temporary memory size in megabytes\n"
665 " --vram <size> Set temporary size of video memory in megabytes\n"
666 " --fullscreen Start VM in fullscreen mode\n"
667 " --fullscreenresize Resize the guest on fullscreen\n"
668 " --fixedmode <w> <h> <bpp> Use a fixed SDL video mode with given width, height and bits per pixel\n"
669 " --nofstoggle Forbid switching to/from fullscreen mode\n"
670 " --noresize Make the SDL frame non resizable\n"
671 " --nohostkey Disable all hostkey combinations\n"
672 " --nohostkeys ... Disable specific hostkey combinations, see below for valid keys\n"
673 " --nograbonclick Disable mouse/keyboard grabbing on mouse click w/o additions\n"
674 " --detecthostkey Get the hostkey identifier and modifier state\n"
675 " --hostkey <key> {<key2>} <mod> Set the host key to the values obtained using --detecthostkey\n"
676 " --termacpi Send an ACPI power button event when closing the window\n"
677#if defined(RT_OS_LINUX)
678 " --evdevkeymap Use evdev keycode map\n"
679#endif
680#ifdef VBOX_WITH_VRDP
681 " --vrdp <ports> Listen for VRDP connections on one of specified ports (default if not specified)\n"
682#endif
683 " --discardstate Discard saved state (if present) and revert to last snapshot (if present)\n"
684#ifdef VBOX_SECURELABEL
685 " --securelabel Display a secure VM label at the top of the screen\n"
686 " --seclabelfnt TrueType (.ttf) font file for secure session label\n"
687 " --seclabelsiz Font point size for secure session label (default 12)\n"
688 " --seclabelofs Font offset within the secure label (default 0)\n"
689 " --seclabelfgcol <rgb> Secure label text color RGB value in 6 digit hexadecimal (eg: FFFF00)\n"
690 " --seclabelbgcol <rgb> Secure label background color RGB value in 6 digit hexadecimal (eg: FF0000)\n"
691#endif
692#ifdef VBOXSDL_ADVANCED_OPTIONS
693 " --[no]rawr0 Enable or disable raw ring 3\n"
694 " --[no]rawr3 Enable or disable raw ring 0\n"
695 " --[no]patm Enable or disable PATM\n"
696 " --[no]csam Enable or disable CSAM\n"
697 " --[no]hwvirtex Permit or deny the usage of VT-x/AMD-V\n"
698#endif
699 "\n"
700 " --convertSettings Allow to auto-convert settings files\n"
701 " --convertSettingsBackup Allow to auto-convert settings files\n"
702 " but create backup copies before\n"
703 " --convertSettingsIgnore Allow to auto-convert settings files\n"
704 " but don't explicitly save the results\n"
705 "\n"
706 "Key bindings:\n"
707 " <hostkey> + f Switch to full screen / restore to previous view\n"
708 " h Press ACPI power button\n"
709 " n Take a snapshot and continue execution\n"
710 " p Pause / resume execution\n"
711 " q Power off\n"
712 " r VM reset\n"
713 " s Save state and power off\n"
714 " <del> Send <ctrl><alt><del>\n"
715 " <F1>...<F12> Send <ctrl><alt><Fx>\n"
716#if defined(DEBUG) || defined(VBOX_WITH_STATISTICS)
717 "\n"
718 "Further key bindings useful for debugging:\n"
719 " LCtrl + Alt + F12 Reset statistics counter\n"
720 " LCtrl + Alt + F11 Dump statistics to logfile\n"
721 " Alt + F12 Toggle R0 recompiler\n"
722 " Alt + F11 Toggle R3 recompiler\n"
723 " Alt + F10 Toggle PATM\n"
724 " Alt + F9 Toggle CSAM\n"
725 " Alt + F8 Toggle single step mode\n"
726 " LCtrl/RCtrl + F12 Toggle logger\n"
727 " F12 Write log marker to logfile\n"
728#endif
729 "\n");
730}
731
732static void PrintError(const char *pszName, CBSTR pwszDescr, CBSTR pwszComponent=NULL)
733{
734 const char *pszFile, *pszFunc, *pszStat;
735 char pszBuffer[1024];
736 com::ErrorInfo info;
737
738 RTStrPrintf(pszBuffer, sizeof(pszBuffer), "%lS", pwszDescr);
739
740 RTPrintf("\n%s! Error info:\n", pszName);
741 if ( (pszFile = strstr(pszBuffer, "At '"))
742 && (pszFunc = strstr(pszBuffer, ") in "))
743 && (pszStat = strstr(pszBuffer, "VBox status code: ")))
744 RTPrintf(" %.*s %.*s\n In%.*s %s",
745 pszFile-pszBuffer, pszBuffer,
746 pszFunc-pszFile+1, pszFile,
747 pszStat-pszFunc-4, pszFunc+4,
748 pszStat);
749 else
750 RTPrintf("%s\n", pszBuffer);
751
752 if (pwszComponent)
753 RTPrintf("(component %lS).\n", pwszComponent);
754
755 RTPrintf("\n");
756}
757
758#ifdef VBOXSDL_WITH_X11
759/**
760 * Custom signal handler. Currently it is only used to release modifier
761 * keys when receiving the USR1 signal. When switching VTs, we might not
762 * get release events for Ctrl-Alt and in case a savestate is performed
763 * on the new VT, the VM will be saved with modifier keys stuck. This is
764 * annoying enough for introducing this hack.
765 */
766void signal_handler_SIGUSR1(int sig, siginfo_t *info, void *secret)
767{
768 /* only SIGUSR1 is interesting */
769 if (sig == SIGUSR1)
770 {
771 /* just release the modifiers */
772 ResetKeys();
773 }
774}
775
776/**
777 * Custom signal handler for catching exit events.
778 */
779void signal_handler_SIGINT(int sig)
780{
781 if (gpszPidFile)
782 RTFileDelete(gpszPidFile);
783 signal(SIGINT, SIG_DFL);
784 signal(SIGQUIT, SIG_DFL);
785 signal(SIGSEGV, SIG_DFL);
786 kill(getpid(), sig);
787}
788#endif /* VBOXSDL_WITH_X11 */
789
790/** entry point */
791extern "C"
792DECLEXPORT(int) TrustedMain(int argc, char **argv, char **envp)
793{
794#ifdef VBOXSDL_WITH_X11
795 /*
796 * Lock keys on SDL behave different from normal keys: A KeyPress event is generated
797 * if the lock mode gets active and a keyRelease event is genereated if the lock mode
798 * gets inactive, that is KeyPress and KeyRelease are sent when pressing the lock key
799 * to change the mode. The current lock mode is reflected in SDL_GetModState().
800 *
801 * Debian patched libSDL to make the lock keys behave like normal keys generating a
802 * KeyPress/KeyRelease event if the lock key was pressed/released. But the lock status
803 * is not reflected in the mod status anymore. We disable the Debian-specific extension
804 * to ensure a defined environment and work around the missing KeyPress/KeyRelease
805 * events in ProcessKeys().
806 */
807 RTEnvSet("SDL_DISABLE_LOCK_KEYS", "1");
808#endif
809
810 /*
811 * the hostkey detection mode is unrelated to VM processing, so handle it before
812 * we initialize anything COM related
813 */
814 if (argc == 2 && ( !strcmp(argv[1], "-detecthostkey")
815 || !strcmp(argv[1], "--detecthostkey")))
816 {
817 int rc = SDL_InitSubSystem(SDL_INIT_VIDEO | SDL_INIT_TIMER | SDL_INIT_NOPARACHUTE);
818 if (rc != 0)
819 {
820 RTPrintf("Error: SDL_InitSubSystem failed with message '%s'\n", SDL_GetError());
821 return 1;
822 }
823 /* we need a video window for the keyboard stuff to work */
824 if (!SDL_SetVideoMode(640, 480, 16, SDL_SWSURFACE))
825 {
826 RTPrintf("Error: could not set SDL video mode\n");
827 return 1;
828 }
829
830 RTPrintf("Please hit one or two function key(s) to get the --hostkey value...\n");
831
832 SDL_Event event1;
833 while (SDL_WaitEvent(&event1))
834 {
835 if (event1.type == SDL_KEYDOWN)
836 {
837 SDL_Event event2;
838 unsigned mod = SDL_GetModState() & ~(KMOD_MODE | KMOD_NUM | KMOD_RESERVED);
839 while (SDL_WaitEvent(&event2))
840 {
841 if (event2.type == SDL_KEYDOWN || event2.type == SDL_KEYUP)
842 {
843 /* pressed additional host key */
844 RTPrintf("--hostkey %d", event1.key.keysym.sym);
845 if (event2.type == SDL_KEYDOWN)
846 {
847 RTPrintf(" %d", event2.key.keysym.sym);
848 RTPrintf(" %d\n", SDL_GetModState() & ~(KMOD_MODE | KMOD_NUM | KMOD_RESERVED));
849 }
850 else
851 {
852 RTPrintf(" %d\n", mod);
853 }
854 /* we're done */
855 break;
856 }
857 }
858 /* we're down */
859 break;
860 }
861 }
862 SDL_Quit();
863 return 1;
864 }
865
866 HRESULT rc;
867 int vrc;
868 Guid uuidVM;
869 char *vmName = NULL;
870 DeviceType_T bootDevice = DeviceType_Null;
871 uint32_t memorySize = 0;
872 uint32_t vramSize = 0;
873 VBoxSDLCallback *cbVBoxImpl = NULL;
874 /* wrapper around above object */
875 ComPtr<IVirtualBoxCallback> callback;
876 VBoxSDLConsoleCallback *cbConsoleImpl = NULL;
877 ComPtr<IConsoleCallback> consoleCallback;
878
879 bool fFullscreen = false;
880 bool fResizable = true;
881#ifdef USE_XPCOM_QUEUE_THREAD
882 bool fXPCOMEventThreadSignaled = false;
883#endif
884 char *hdaFile = NULL;
885 char *cdromFile = NULL;
886 char *fdaFile = NULL;
887#ifdef VBOX_WITH_VRDP
888 const char *portVRDP = NULL;
889#endif
890 bool fDiscardState = false;
891#ifdef VBOX_SECURELABEL
892 BOOL fSecureLabel = false;
893 uint32_t secureLabelPointSize = 12;
894 uint32_t secureLabelFontOffs = 0;
895 char *secureLabelFontFile = NULL;
896 uint32_t secureLabelColorFG = 0x0000FF00;
897 uint32_t secureLabelColorBG = 0x00FFFF00;
898#endif
899#ifdef VBOXSDL_ADVANCED_OPTIONS
900 unsigned fRawR0 = ~0U;
901 unsigned fRawR3 = ~0U;
902 unsigned fPATM = ~0U;
903 unsigned fCSAM = ~0U;
904 unsigned fHWVirt = ~0U;
905 uint32_t u32WarpDrive = 0;
906#endif
907#ifdef VBOX_WIN32_UI
908 bool fWin32UI = true;
909 uint64_t winId = 0;
910#endif
911 bool fShowSDLConfig = false;
912 uint32_t fixedWidth = ~(uint32_t)0;
913 uint32_t fixedHeight = ~(uint32_t)0;
914 uint32_t fixedBPP = ~(uint32_t)0;
915 uint32_t uResizeWidth = ~(uint32_t)0;
916 uint32_t uResizeHeight = ~(uint32_t)0;
917
918 /* The damned GOTOs forces this to be up here - totally out of place. */
919 /*
920 * Host key handling.
921 *
922 * The golden rule is that host-key combinations should not be seen
923 * by the guest. For instance a CAD should not have any extra RCtrl down
924 * and RCtrl up around itself. Nor should a resume be followed by a Ctrl-P
925 * that could encourage applications to start printing.
926 *
927 * We must not confuse the hostkey processing into any release sequences
928 * either, the host key is supposed to be explicitly pressing one key.
929 *
930 * Quick state diagram:
931 *
932 * host key down alone
933 * (Normal) ---------------
934 * ^ ^ |
935 * | | v host combination key down
936 * | | (Host key down) ----------------
937 * | | host key up v | |
938 * | |-------------- | other key down v host combination key down
939 * | | (host key used) -------------
940 * | | | ^ |
941 * | (not host key)-- | |---------------
942 * | | | | |
943 * | | ---- other |
944 * | modifiers = 0 v v
945 * -----------------------------------------------
946 */
947 enum HKEYSTATE
948 {
949 /** The initial and most common state, pass keystrokes to the guest.
950 * Next state: HKEYSTATE_DOWN
951 * Prev state: Any */
952 HKEYSTATE_NORMAL = 1,
953 /** The first host key was pressed down
954 */
955 HKEYSTATE_DOWN_1ST,
956 /** The second host key was pressed down (if gHostKeySym2 != SDLK_UNKNOWN)
957 */
958 HKEYSTATE_DOWN_2ND,
959 /** The host key has been pressed down.
960 * Prev state: HKEYSTATE_NORMAL
961 * Next state: HKEYSTATE_NORMAL - host key up, capture toggle.
962 * Next state: HKEYSTATE_USED - host key combination down.
963 * Next state: HKEYSTATE_NOT_IT - non-host key combination down.
964 */
965 HKEYSTATE_DOWN,
966 /** A host key combination was pressed.
967 * Prev state: HKEYSTATE_DOWN
968 * Next state: HKEYSTATE_NORMAL - when modifiers are all 0
969 */
970 HKEYSTATE_USED,
971 /** A non-host key combination was attempted. Send hostkey down to the
972 * guest and continue until all modifiers have been released.
973 * Prev state: HKEYSTATE_DOWN
974 * Next state: HKEYSTATE_NORMAL - when modifiers are all 0
975 */
976 HKEYSTATE_NOT_IT
977 } enmHKeyState = HKEYSTATE_NORMAL;
978 /** The host key down event which we have been hiding from the guest.
979 * Used when going from HKEYSTATE_DOWN to HKEYSTATE_NOT_IT. */
980 SDL_Event EvHKeyDown1;
981 SDL_Event EvHKeyDown2;
982
983 LogFlow(("SDL GUI started\n"));
984 RTPrintf(VBOX_PRODUCT " SDL GUI version %s\n"
985 "(C) 2005-" VBOX_C_YEAR " " VBOX_VENDOR "\n"
986 "All rights reserved.\n\n",
987 VBOX_VERSION_STRING);
988
989 // less than one parameter is not possible
990 if (argc < 2)
991 {
992 show_usage();
993 return 1;
994 }
995
996 // command line argument parsing stuff
997 for (int curArg = 1; curArg < argc; curArg++)
998 {
999 if ( !strcmp(argv[curArg], "--vm")
1000 || !strcmp(argv[curArg], "-vm")
1001 || !strcmp(argv[curArg], "--startvm")
1002 || !strcmp(argv[curArg], "-startvm")
1003 || !strcmp(argv[curArg], "-s")
1004 )
1005 {
1006 if (++curArg >= argc)
1007 {
1008 RTPrintf("Error: VM not specified (UUID or name)!\n");
1009 return 1;
1010 }
1011 // first check if a UUID was supplied
1012 if (RT_FAILURE(RTUuidFromStr(uuidVM.ptr(), argv[curArg])))
1013 {
1014 LogFlow(("invalid UUID format, assuming it's a VM name\n"));
1015 vmName = argv[curArg];
1016 }
1017 }
1018 else if ( !strcmp(argv[curArg], "--comment")
1019 || !strcmp(argv[curArg], "-comment"))
1020 {
1021 if (++curArg >= argc)
1022 {
1023 RTPrintf("Error: missing argument for comment!\n");
1024 return 1;
1025 }
1026 }
1027 else if ( !strcmp(argv[curArg], "--boot")
1028 || !strcmp(argv[curArg], "-boot"))
1029 {
1030 if (++curArg >= argc)
1031 {
1032 RTPrintf("Error: missing argument for boot drive!\n");
1033 return 1;
1034 }
1035 switch (argv[curArg][0])
1036 {
1037 case 'a':
1038 {
1039 bootDevice = DeviceType_Floppy;
1040 break;
1041 }
1042
1043 case 'c':
1044 {
1045 bootDevice = DeviceType_HardDisk;
1046 break;
1047 }
1048
1049 case 'd':
1050 {
1051 bootDevice = DeviceType_DVD;
1052 break;
1053 }
1054
1055 case 'n':
1056 {
1057 bootDevice = DeviceType_Network;
1058 break;
1059 }
1060
1061 default:
1062 {
1063 RTPrintf("Error: wrong argument for boot drive!\n");
1064 return 1;
1065 }
1066 }
1067 }
1068 else if ( !strcmp(argv[curArg], "--memory")
1069 || !strcmp(argv[curArg], "-memory")
1070 || !strcmp(argv[curArg], "-m"))
1071 {
1072 if (++curArg >= argc)
1073 {
1074 RTPrintf("Error: missing argument for memory size!\n");
1075 return 1;
1076 }
1077 memorySize = atoi(argv[curArg]);
1078 }
1079 else if ( !strcmp(argv[curArg], "--vram")
1080 || !strcmp(argv[curArg], "-vram"))
1081 {
1082 if (++curArg >= argc)
1083 {
1084 RTPrintf("Error: missing argument for vram size!\n");
1085 return 1;
1086 }
1087 vramSize = atoi(argv[curArg]);
1088 }
1089 else if ( !strcmp(argv[curArg], "--fullscreen")
1090 || !strcmp(argv[curArg], "-fullscreen"))
1091 {
1092 fFullscreen = true;
1093 }
1094 else if ( !strcmp(argv[curArg], "--fullscreenresize")
1095 || !strcmp(argv[curArg], "-fullscreenresize"))
1096 {
1097 gfFullscreenResize = true;
1098#ifdef VBOXSDL_WITH_X11
1099 RTEnvSet("SDL_VIDEO_X11_VIDMODE", "0");
1100#endif
1101 }
1102 else if ( !strcmp(argv[curArg], "--fixedmode")
1103 || !strcmp(argv[curArg], "-fixedmode"))
1104 {
1105 /* three parameters follow */
1106 if (curArg + 3 >= argc)
1107 {
1108 RTPrintf("Error: missing arguments for fixed video mode!\n");
1109 return 1;
1110 }
1111 fixedWidth = atoi(argv[++curArg]);
1112 fixedHeight = atoi(argv[++curArg]);
1113 fixedBPP = atoi(argv[++curArg]);
1114 }
1115 else if ( !strcmp(argv[curArg], "--nofstoggle")
1116 || !strcmp(argv[curArg], "-nofstoggle"))
1117 {
1118 gfAllowFullscreenToggle = FALSE;
1119 }
1120 else if ( !strcmp(argv[curArg], "--noresize")
1121 || !strcmp(argv[curArg], "-noresize"))
1122 {
1123 fResizable = false;
1124 }
1125 else if ( !strcmp(argv[curArg], "--nohostkey")
1126 || !strcmp(argv[curArg], "-nohostkey"))
1127 {
1128 gHostKeyMod = 0;
1129 gHostKeySym1 = 0;
1130 }
1131 else if ( !strcmp(argv[curArg], "--nohostkeys")
1132 || !strcmp(argv[curArg], "-nohostkeys"))
1133 {
1134 if (++curArg >= argc)
1135 {
1136 RTPrintf("Error: missing a string of disabled hostkey combinations\n");
1137 return 1;
1138 }
1139 gHostKeyDisabledCombinations = argv[curArg];
1140 size_t cch = strlen(gHostKeyDisabledCombinations);
1141 for (size_t i = 0; i < cch; i++)
1142 {
1143 if (!strchr("fhnpqrs", gHostKeyDisabledCombinations[i]))
1144 {
1145 RTPrintf("Error: <hostkey> + '%c' is not a valid combination\n",
1146 gHostKeyDisabledCombinations[i]);
1147 return 1;
1148 }
1149 }
1150 }
1151 else if ( !strcmp(argv[curArg], "--nograbonclick")
1152 || !strcmp(argv[curArg], "-nograbonclick"))
1153 {
1154 gfGrabOnMouseClick = FALSE;
1155 }
1156 else if ( !strcmp(argv[curArg], "--termacpi")
1157 || !strcmp(argv[curArg], "-termacpi"))
1158 {
1159 gfACPITerm = TRUE;
1160 }
1161 else if ( !strcmp(argv[curArg], "--pidfile")
1162 || !strcmp(argv[curArg], "-pidfile"))
1163 {
1164 if (++curArg >= argc)
1165 {
1166 RTPrintf("Error: missing file name for --pidfile!\n");
1167 return 1;
1168 }
1169 gpszPidFile = argv[curArg];
1170 }
1171 else if ( !strcmp(argv[curArg], "--hda")
1172 || !strcmp(argv[curArg], "-hda"))
1173 {
1174 if (++curArg >= argc)
1175 {
1176 RTPrintf("Error: missing file name for first hard disk!\n");
1177 return 1;
1178 }
1179 /* resolve it. */
1180 if (RTPathExists(argv[curArg]))
1181 hdaFile = RTPathRealDup(argv[curArg]);
1182 if (!hdaFile)
1183 {
1184 RTPrintf("Error: The path to the specified harddisk, '%s', could not be resolved.\n", argv[curArg]);
1185 return 1;
1186 }
1187 }
1188 else if ( !strcmp(argv[curArg], "--fda")
1189 || !strcmp(argv[curArg], "-fda"))
1190 {
1191 if (++curArg >= argc)
1192 {
1193 RTPrintf("Error: missing file/device name for first floppy disk!\n");
1194 return 1;
1195 }
1196 /* resolve it. */
1197 if (RTPathExists(argv[curArg]))
1198 fdaFile = RTPathRealDup(argv[curArg]);
1199 if (!fdaFile)
1200 {
1201 RTPrintf("Error: The path to the specified floppy disk, '%s', could not be resolved.\n", argv[curArg]);
1202 return 1;
1203 }
1204 }
1205 else if ( !strcmp(argv[curArg], "--cdrom")
1206 || !strcmp(argv[curArg], "-cdrom"))
1207 {
1208 if (++curArg >= argc)
1209 {
1210 RTPrintf("Error: missing file/device name for cdrom!\n");
1211 return 1;
1212 }
1213 /* resolve it. */
1214 if (RTPathExists(argv[curArg]))
1215 cdromFile = RTPathRealDup(argv[curArg]);
1216 if (!cdromFile)
1217 {
1218 RTPrintf("Error: The path to the specified cdrom, '%s', could not be resolved.\n", argv[curArg]);
1219 return 1;
1220 }
1221 }
1222#if defined(RT_OS_LINUX) && defined(VBOXSDL_WITH_X11)
1223 else if ( !strcmp(argv[curArg], "--evdevkeymap")
1224 || !strcmp(argv[curArg], "-evdevkeymap"))
1225 {
1226 guseEvdevKeymap = TRUE;
1227 }
1228#endif /* RT_OS_LINUX */
1229#ifdef VBOX_WITH_VRDP
1230 else if ( !strcmp(argv[curArg], "--vrdp")
1231 || !strcmp(argv[curArg], "-vrdp"))
1232 {
1233 // start with the standard VRDP port
1234 portVRDP = "0";
1235
1236 // is there another argument
1237 if (argc > (curArg + 1))
1238 {
1239 curArg++;
1240 portVRDP = argv[curArg];
1241 LogFlow(("Using non standard VRDP port %s\n", portVRDP));
1242 }
1243 }
1244#endif /* VBOX_WITH_VRDP */
1245 else if ( !strcmp(argv[curArg], "--discardstate")
1246 || !strcmp(argv[curArg], "-discardstate"))
1247 {
1248 fDiscardState = true;
1249 }
1250#ifdef VBOX_SECURELABEL
1251 else if ( !strcmp(argv[curArg], "--securelabel")
1252 || !strcmp(argv[curArg], "-securelabel"))
1253 {
1254 fSecureLabel = true;
1255 LogFlow(("Secure labelling turned on\n"));
1256 }
1257 else if ( !strcmp(argv[curArg], "--seclabelfnt")
1258 || !strcmp(argv[curArg], "-seclabelfnt"))
1259 {
1260 if (++curArg >= argc)
1261 {
1262 RTPrintf("Error: missing font file name for secure label!\n");
1263 return 1;
1264 }
1265 secureLabelFontFile = argv[curArg];
1266 }
1267 else if ( !strcmp(argv[curArg], "--seclabelsiz")
1268 || !strcmp(argv[curArg], "-seclabelsiz"))
1269 {
1270 if (++curArg >= argc)
1271 {
1272 RTPrintf("Error: missing font point size for secure label!\n");
1273 return 1;
1274 }
1275 secureLabelPointSize = atoi(argv[curArg]);
1276 }
1277 else if ( !strcmp(argv[curArg], "--seclabelofs")
1278 || !strcmp(argv[curArg], "-seclabelofs"))
1279 {
1280 if (++curArg >= argc)
1281 {
1282 RTPrintf("Error: missing font pixel offset for secure label!\n");
1283 return 1;
1284 }
1285 secureLabelFontOffs = atoi(argv[curArg]);
1286 }
1287 else if ( !strcmp(argv[curArg], "--seclabelfgcol")
1288 || !strcmp(argv[curArg], "-seclabelfgcol"))
1289 {
1290 if (++curArg >= argc)
1291 {
1292 RTPrintf("Error: missing text color value for secure label!\n");
1293 return 1;
1294 }
1295 sscanf(argv[curArg], "%X", &secureLabelColorFG);
1296 }
1297 else if ( !strcmp(argv[curArg], "--seclabelbgcol")
1298 || !strcmp(argv[curArg], "-seclabelbgcol"))
1299 {
1300 if (++curArg >= argc)
1301 {
1302 RTPrintf("Error: missing background color value for secure label!\n");
1303 return 1;
1304 }
1305 sscanf(argv[curArg], "%X", &secureLabelColorBG);
1306 }
1307#endif
1308#ifdef VBOXSDL_ADVANCED_OPTIONS
1309 else if ( !strcmp(argv[curArg], "--rawr0")
1310 || !strcmp(argv[curArg], "-rawr0"))
1311 fRawR0 = true;
1312 else if ( !strcmp(argv[curArg], "--norawr0")
1313 || !strcmp(argv[curArg], "-norawr0"))
1314 fRawR0 = false;
1315 else if ( !strcmp(argv[curArg], "--rawr3")
1316 || !strcmp(argv[curArg], "-rawr3"))
1317 fRawR3 = true;
1318 else if ( !strcmp(argv[curArg], "--norawr3")
1319 || !strcmp(argv[curArg], "-norawr3"))
1320 fRawR3 = false;
1321 else if ( !strcmp(argv[curArg], "--patm")
1322 || !strcmp(argv[curArg], "-patm"))
1323 fPATM = true;
1324 else if ( !strcmp(argv[curArg], "--nopatm")
1325 || !strcmp(argv[curArg], "-nopatm"))
1326 fPATM = false;
1327 else if ( !strcmp(argv[curArg], "--csam")
1328 || !strcmp(argv[curArg], "-csam"))
1329 fCSAM = true;
1330 else if ( !strcmp(argv[curArg], "--nocsam")
1331 || !strcmp(argv[curArg], "-nocsam"))
1332 fCSAM = false;
1333 else if ( !strcmp(argv[curArg], "--hwvirtex")
1334 || !strcmp(argv[curArg], "-hwvirtex"))
1335 fHWVirt = true;
1336 else if ( !strcmp(argv[curArg], "--nohwvirtex")
1337 || !strcmp(argv[curArg], "-nohwvirtex"))
1338 fHWVirt = false;
1339 else if ( !strcmp(argv[curArg], "--warpdrive")
1340 || !strcmp(argv[curArg], "-warpdrive"))
1341 {
1342 if (++curArg >= argc)
1343 {
1344 RTPrintf("Error: missing the rate value for the --warpdrive option!\n");
1345 return 1;
1346 }
1347 u32WarpDrive = RTStrToUInt32(argv[curArg]);
1348 if (u32WarpDrive < 2 || u32WarpDrive > 20000)
1349 {
1350 RTPrintf("Error: the warp drive rate is restricted to [2..20000]. (%d)\n", u32WarpDrive);
1351 return 1;
1352 }
1353 }
1354#endif /* VBOXSDL_ADVANCED_OPTIONS */
1355#ifdef VBOX_WIN32_UI
1356 else if ( !strcmp(argv[curArg], "--win32ui")
1357 || !strcmp(argv[curArg], "-win32ui"))
1358 fWin32UI = true;
1359#endif
1360 else if ( !strcmp(argv[curArg], "--showsdlconfig")
1361 || !strcmp(argv[curArg], "-showsdlconfig"))
1362 fShowSDLConfig = true;
1363 else if ( !strcmp(argv[curArg], "--hostkey")
1364 || !strcmp(argv[curArg], "-hostkey"))
1365 {
1366 if (++curArg + 1 >= argc)
1367 {
1368 RTPrintf("Error: not enough arguments for host keys!\n");
1369 return 1;
1370 }
1371 gHostKeySym1 = atoi(argv[curArg++]);
1372 if (curArg + 1 < argc && (argv[curArg+1][0] == '0' || atoi(argv[curArg+1]) > 0))
1373 {
1374 /* two-key sequence as host key specified */
1375 gHostKeySym2 = atoi(argv[curArg++]);
1376 }
1377 gHostKeyMod = atoi(argv[curArg]);
1378 }
1379 /* just show the help screen */
1380 else
1381 {
1382 if ( strcmp(argv[curArg], "-h")
1383 && strcmp(argv[curArg], "-help")
1384 && strcmp(argv[curArg], "--help"))
1385 RTPrintf("Error: unrecognized switch '%s'\n", argv[curArg]);
1386 show_usage();
1387 return 1;
1388 }
1389 }
1390
1391 rc = com::Initialize();
1392 if (FAILED(rc))
1393 {
1394 RTPrintf("Error: COM initialization failed, rc = 0x%x!\n", rc);
1395 return 1;
1396 }
1397
1398 /* NOTE: do not convert the following scope to a "do {} while (0);", as
1399 * this would make it all too tempting to use "break;" incorrectly - it
1400 * would skip over the cleanup. */
1401 {
1402 // scopes all the stuff till shutdown
1403 ////////////////////////////////////////////////////////////////////////////
1404
1405 ComPtr <IVirtualBox> virtualBox;
1406 ComPtr <ISession> session;
1407 bool sessionOpened = false;
1408 EventQueue* eventQ = com::EventQueue::getMainEventQueue();
1409 const CLSID sessionID = CLSID_Session;
1410
1411 rc = virtualBox.createLocalObject(CLSID_VirtualBox);
1412 if (FAILED(rc))
1413 {
1414 com::ErrorInfo info;
1415 if (info.isFullAvailable())
1416 PrintError("Failed to create VirtualBox object",
1417 info.getText().raw(), info.getComponent().raw());
1418 else
1419 RTPrintf("Failed to create VirtualBox object! No error information available (rc = 0x%x).\n", rc);
1420 goto leave;
1421 }
1422 rc = session.createInprocObject(sessionID);
1423 if (FAILED(rc))
1424 {
1425 RTPrintf("Failed to create session object, rc = 0x%x!\n", rc);
1426 goto leave;
1427 }
1428
1429 /*
1430 * Do we have a name but no UUID?
1431 */
1432 if (vmName && uuidVM.isEmpty())
1433 {
1434 ComPtr<IMachine> aMachine;
1435 Bstr bstrVMName = vmName;
1436 rc = virtualBox->FindMachine(bstrVMName, aMachine.asOutParam());
1437 if ((rc == S_OK) && aMachine)
1438 {
1439 Bstr id;
1440 aMachine->COMGETTER(Id)(id.asOutParam());
1441 uuidVM = Guid(id);
1442 }
1443 else
1444 {
1445 RTPrintf("Error: machine with the given ID not found!\n");
1446 goto leave;
1447 }
1448 }
1449 else if (uuidVM.isEmpty())
1450 {
1451 RTPrintf("Error: no machine specified!\n");
1452 goto leave;
1453 }
1454
1455 /* create SDL event semaphore */
1456 vrc = RTSemEventCreate(&g_EventSemSDLEvents);
1457 AssertReleaseRC(vrc);
1458
1459 rc = virtualBox->OpenSession(session, uuidVM.toUtf16());
1460 if (FAILED(rc))
1461 {
1462 com::ErrorInfo info;
1463 if (info.isFullAvailable())
1464 PrintError("Could not open VirtualBox session",
1465 info.getText().raw(), info.getComponent().raw());
1466 goto leave;
1467 }
1468 if (!session)
1469 {
1470 RTPrintf("Could not open VirtualBox session!\n");
1471 goto leave;
1472 }
1473 sessionOpened = true;
1474 // get the VM we're dealing with
1475 session->COMGETTER(Machine)(gMachine.asOutParam());
1476 if (!gMachine)
1477 {
1478 com::ErrorInfo info;
1479 if (info.isFullAvailable())
1480 PrintError("Cannot start VM!",
1481 info.getText().raw(), info.getComponent().raw());
1482 else
1483 RTPrintf("Error: given machine not found!\n");
1484 goto leave;
1485 }
1486 // get the VM console
1487 session->COMGETTER(Console)(gConsole.asOutParam());
1488 if (!gConsole)
1489 {
1490 RTPrintf("Given console not found!\n");
1491 goto leave;
1492 }
1493
1494 /*
1495 * Are we supposed to use a different hard disk file?
1496 */
1497 if (hdaFile)
1498 {
1499 /*
1500 * Strategy: iterate through all registered hard disk
1501 * and see if one of them points to the same file. If
1502 * so, assign it. If not, register a new image and assign
1503 * it to the VM.
1504 */
1505 Bstr hdaFileBstr = hdaFile;
1506 ComPtr<IMedium> hardDisk;
1507 virtualBox->FindHardDisk(hdaFileBstr, hardDisk.asOutParam());
1508 if (!hardDisk)
1509 {
1510 /* we've not found the image */
1511 RTPrintf("Adding hard disk '%S'...\n", hdaFile);
1512 virtualBox->OpenHardDisk(hdaFileBstr, AccessMode_ReadWrite, false, Bstr(""), false, Bstr(""), hardDisk.asOutParam());
1513 }
1514 /* do we have the right image now? */
1515 if (hardDisk)
1516 {
1517 /*
1518 * Go and attach it!
1519 */
1520 Bstr uuidHD;
1521 Bstr storageCtlName;
1522 hardDisk->COMGETTER(Id)(uuidHD.asOutParam());
1523
1524 /* get the first IDE controller to attach the harddisk to
1525 * and if there is none, add one temporarily
1526 */
1527 {
1528 ComPtr<IStorageController> storageCtl;
1529 com::SafeIfaceArray<IStorageController> aStorageControllers;
1530 CHECK_ERROR (gMachine, COMGETTER(StorageControllers)(ComSafeArrayAsOutParam(aStorageControllers)));
1531 for (size_t i = 0; i < aStorageControllers.size(); ++ i)
1532 {
1533 StorageBus_T storageBus = StorageBus_Null;
1534
1535 CHECK_ERROR (aStorageControllers[i], COMGETTER(Bus)(&storageBus));
1536 if (storageBus == StorageBus_IDE)
1537 {
1538 storageCtl = aStorageControllers[i];
1539 break;
1540 }
1541 }
1542
1543 if (storageCtl)
1544 {
1545 CHECK_ERROR (storageCtl, COMGETTER(Name)(storageCtlName.asOutParam()));
1546 gMachine->DetachDevice(storageCtlName, 0, 0);
1547 }
1548 else
1549 {
1550 storageCtlName = "IDE Controller";
1551 CHECK_ERROR (gMachine, AddStorageController(storageCtlName,
1552 StorageBus_IDE,
1553 storageCtl.asOutParam()));
1554 }
1555 }
1556
1557 gMachine->AttachDevice(storageCtlName, 0, 0, DeviceType_HardDisk, uuidHD);
1558 /// @todo why is this attachment saved?
1559 }
1560 else
1561 {
1562 RTPrintf("Error: failed to mount the specified hard disk image!\n");
1563 goto leave;
1564 }
1565 }
1566
1567 /*
1568 * Mount a floppy if requested.
1569 */
1570 if (fdaFile)
1571 do
1572 {
1573 ComPtr<IMedium> floppyMedium;
1574
1575 /* unmount? */
1576 if (!strcmp(fdaFile, "none"))
1577 {
1578 /* nothing to do, NULL object will cause unmount */
1579 }
1580 else
1581 {
1582 Bstr medium = fdaFile;
1583
1584 /* Assume it's a host drive name */
1585 ComPtr <IHost> host;
1586 CHECK_ERROR_BREAK(virtualBox, COMGETTER(Host)(host.asOutParam()));
1587 rc = host->FindHostFloppyDrive(medium, floppyMedium.asOutParam());
1588 if (FAILED(rc))
1589 {
1590 /* try to find an existing one */
1591 rc = virtualBox->FindFloppyImage(medium, floppyMedium.asOutParam());
1592 if (FAILED(rc))
1593 {
1594 /* try to add to the list */
1595 RTPrintf("Adding floppy image '%S'...\n", fdaFile);
1596 CHECK_ERROR_BREAK(virtualBox, OpenFloppyImage(medium, Bstr(),
1597 floppyMedium.asOutParam()));
1598 }
1599 }
1600 }
1601
1602 Bstr id;
1603 Bstr storageCtlName;
1604 floppyMedium->COMGETTER(Id)(id.asOutParam());
1605
1606 /* get the first floppy controller to attach the floppy to
1607 * and if there is none, add one temporarily
1608 */
1609 {
1610 ComPtr<IStorageController> storageCtl;
1611 com::SafeIfaceArray<IStorageController> aStorageControllers;
1612 CHECK_ERROR (gMachine, COMGETTER(StorageControllers)(ComSafeArrayAsOutParam(aStorageControllers)));
1613 for (size_t i = 0; i < aStorageControllers.size(); ++ i)
1614 {
1615 StorageBus_T storageBus = StorageBus_Null;
1616
1617 CHECK_ERROR (aStorageControllers[i], COMGETTER(Bus)(&storageBus));
1618 if (storageBus == StorageBus_Floppy)
1619 {
1620 storageCtl = aStorageControllers[i];
1621 break;
1622 }
1623 }
1624
1625 if (storageCtl)
1626 {
1627 ComPtr<IMediumAttachment> floppyAttachment;
1628
1629 CHECK_ERROR (storageCtl, COMGETTER(Name)(storageCtlName.asOutParam()));
1630 rc = gMachine->GetMediumAttachment(storageCtlName, 0, 0, floppyAttachment.asOutParam());
1631 if (FAILED(rc))
1632 CHECK_ERROR (gMachine, AttachDevice(storageCtlName, 0, 0,
1633 DeviceType_Floppy, NULL));
1634 }
1635 else
1636 {
1637 storageCtlName = "Floppy Controller";
1638 CHECK_ERROR (gMachine, AddStorageController(storageCtlName,
1639 StorageBus_Floppy,
1640 storageCtl.asOutParam()));
1641 CHECK_ERROR (gMachine, AttachDevice(storageCtlName, 0, 0,
1642 DeviceType_Floppy, NULL));
1643 }
1644 }
1645
1646 CHECK_ERROR (gMachine, MountMedium(storageCtlName, 0, 0, id, FALSE /* aForce */));
1647 }
1648 while (0);
1649 if (FAILED(rc))
1650 goto leave;
1651
1652 /*
1653 * Mount a CD-ROM if requested.
1654 */
1655 if (cdromFile)
1656 do
1657 {
1658 ComPtr<IMedium> dvdMedium;
1659
1660 /* unmount? */
1661 if (!strcmp(cdromFile, "none"))
1662 {
1663 /* nothing to do, NULL object will cause unmount */
1664 }
1665 else
1666 {
1667 Bstr medium = cdromFile;
1668
1669 /* Assume it's a host drive name */
1670 ComPtr <IHost> host;
1671 CHECK_ERROR_BREAK(virtualBox, COMGETTER(Host)(host.asOutParam()));
1672 rc = host->FindHostDVDDrive(medium,dvdMedium.asOutParam());
1673 if (FAILED(rc))
1674 {
1675 /* try to find an existing one */
1676 rc = virtualBox->FindDVDImage(medium, dvdMedium.asOutParam());
1677 if (FAILED(rc))
1678 {
1679 /* try to add to the list */
1680 RTPrintf("Adding ISO image '%S'...\n", cdromFile);
1681 CHECK_ERROR_BREAK(virtualBox, OpenDVDImage(medium, Bstr(),
1682 dvdMedium.asOutParam()));
1683 }
1684 }
1685 }
1686
1687 Bstr id;
1688 Bstr storageCtlName;
1689 dvdMedium->COMGETTER(Id)(id.asOutParam());
1690
1691 /* get the first IDE controller to attach the DVD Drive to
1692 * and if there is none, add one temporarily
1693 */
1694 {
1695 ComPtr<IStorageController> storageCtl;
1696 com::SafeIfaceArray<IStorageController> aStorageControllers;
1697 CHECK_ERROR (gMachine, COMGETTER(StorageControllers)(ComSafeArrayAsOutParam(aStorageControllers)));
1698 for (size_t i = 0; i < aStorageControllers.size(); ++ i)
1699 {
1700 StorageBus_T storageBus = StorageBus_Null;
1701
1702 CHECK_ERROR (aStorageControllers[i], COMGETTER(Bus)(&storageBus));
1703 if (storageBus == StorageBus_IDE)
1704 {
1705 storageCtl = aStorageControllers[i];
1706 break;
1707 }
1708 }
1709
1710 if (storageCtl)
1711 {
1712 ComPtr<IMediumAttachment> dvdAttachment;
1713
1714 CHECK_ERROR (storageCtl, COMGETTER(Name)(storageCtlName.asOutParam()));
1715 gMachine->DetachDevice(storageCtlName, 1, 0);
1716 CHECK_ERROR (gMachine, AttachDevice(storageCtlName, 1, 0,
1717 DeviceType_DVD, NULL));
1718 }
1719 else
1720 {
1721 storageCtlName = "IDE Controller";
1722 CHECK_ERROR (gMachine, AddStorageController(storageCtlName,
1723 StorageBus_IDE,
1724 storageCtl.asOutParam()));
1725 CHECK_ERROR (gMachine, AttachDevice(storageCtlName, 1, 0,
1726 DeviceType_DVD, NULL));
1727 }
1728 }
1729
1730 CHECK_ERROR(gMachine, MountMedium(storageCtlName, 1, 0, id, FALSE /*aForce */));
1731 }
1732 while (0);
1733 if (FAILED(rc))
1734 goto leave;
1735
1736 if (fDiscardState)
1737 {
1738 /*
1739 * If the machine is currently saved,
1740 * discard the saved state first.
1741 */
1742 MachineState_T machineState;
1743 gMachine->COMGETTER(State)(&machineState);
1744 if (machineState == MachineState_Saved)
1745 {
1746 CHECK_ERROR(gConsole, ForgetSavedState(true));
1747 }
1748 /*
1749 * If there are snapshots, discard the current state,
1750 * i.e. revert to the last snapshot.
1751 */
1752 ULONG cSnapshots;
1753 gMachine->COMGETTER(SnapshotCount)(&cSnapshots);
1754 if (cSnapshots)
1755 {
1756 gProgress = NULL;
1757
1758 ComPtr<ISnapshot> pCurrentSnapshot;
1759 CHECK_ERROR(gMachine, COMGETTER(CurrentSnapshot)(pCurrentSnapshot.asOutParam()));
1760 if (FAILED(rc))
1761 goto leave;
1762
1763 CHECK_ERROR(gConsole, RestoreSnapshot(pCurrentSnapshot, gProgress.asOutParam()));
1764 rc = gProgress->WaitForCompletion(-1);
1765 }
1766 }
1767
1768 // get the machine debugger (does not have to be there)
1769 gConsole->COMGETTER(Debugger)(gMachineDebugger.asOutParam());
1770 if (gMachineDebugger)
1771 {
1772 Log(("Machine debugger available!\n"));
1773 }
1774 gConsole->COMGETTER(Display)(gDisplay.asOutParam());
1775 if (!gDisplay)
1776 {
1777 RTPrintf("Error: could not get display object!\n");
1778 goto leave;
1779 }
1780
1781 // set the boot drive
1782 if (bootDevice != DeviceType_Null)
1783 {
1784 rc = gMachine->SetBootOrder(1, bootDevice);
1785 if (rc != S_OK)
1786 {
1787 RTPrintf("Error: could not set boot device, using default.\n");
1788 }
1789 }
1790
1791 // set the memory size if not default
1792 if (memorySize)
1793 {
1794 rc = gMachine->COMSETTER(MemorySize)(memorySize);
1795 if (rc != S_OK)
1796 {
1797 ULONG ramSize = 0;
1798 gMachine->COMGETTER(MemorySize)(&ramSize);
1799 RTPrintf("Error: could not set memory size, using current setting of %d MBytes\n", ramSize);
1800 }
1801 }
1802
1803 if (vramSize)
1804 {
1805 rc = gMachine->COMSETTER(VRAMSize)(vramSize);
1806 if (rc != S_OK)
1807 {
1808 gMachine->COMGETTER(VRAMSize)((ULONG*)&vramSize);
1809 RTPrintf("Error: could not set VRAM size, using current setting of %d MBytes\n", vramSize);
1810 }
1811 }
1812
1813 // we're always able to process absolute mouse events and we prefer that
1814 gfAbsoluteMouseHost = TRUE;
1815
1816#ifdef VBOX_WIN32_UI
1817 if (fWin32UI)
1818 {
1819 /* initialize the Win32 user interface inside which SDL will be embedded */
1820 if (initUI(fResizable, winId))
1821 return 1;
1822 }
1823#endif
1824
1825 /* static initialization of the SDL stuff */
1826 if (!VBoxSDLFB::init(fShowSDLConfig))
1827 goto leave;
1828
1829 gMachine->COMGETTER(MonitorCount)(&gcMonitors);
1830 if (gcMonitors > 64)
1831 gcMonitors = 64;
1832
1833 for (unsigned i = 0; i < gcMonitors; i++)
1834 {
1835 // create our SDL framebuffer instance
1836 gpFramebuffer[i] = new VBoxSDLFB(i, fFullscreen, fResizable, fShowSDLConfig, false,
1837 fixedWidth, fixedHeight, fixedBPP);
1838
1839 if (!gpFramebuffer[i])
1840 {
1841 RTPrintf("Error: could not create framebuffer object!\n");
1842 goto leave;
1843 }
1844 }
1845
1846#ifdef VBOX_WIN32_UI
1847 gpFramebuffer[0]->setWinId(winId);
1848#endif
1849
1850 for (unsigned i = 0; i < gcMonitors; i++)
1851 {
1852 if (!gpFramebuffer[i]->initialized())
1853 goto leave;
1854 gpFramebuffer[i]->AddRef();
1855 if (fFullscreen)
1856 SetFullscreen(true);
1857 }
1858
1859#ifdef VBOX_SECURELABEL
1860 if (fSecureLabel)
1861 {
1862 if (!secureLabelFontFile)
1863 {
1864 RTPrintf("Error: no font file specified for secure label!\n");
1865 goto leave;
1866 }
1867 /* load the SDL_ttf library and get the required imports */
1868 vrc = RTLdrLoad(LIBSDL_TTF_NAME, &gLibrarySDL_ttf);
1869 if (RT_SUCCESS(vrc))
1870 vrc = RTLdrGetSymbol(gLibrarySDL_ttf, "TTF_Init", (void**)&pTTF_Init);
1871 if (RT_SUCCESS(vrc))
1872 vrc = RTLdrGetSymbol(gLibrarySDL_ttf, "TTF_OpenFont", (void**)&pTTF_OpenFont);
1873 if (RT_SUCCESS(vrc))
1874 vrc = RTLdrGetSymbol(gLibrarySDL_ttf, "TTF_RenderUTF8_Solid", (void**)&pTTF_RenderUTF8_Solid);
1875 if (RT_SUCCESS(vrc))
1876 {
1877 /* silently ignore errors here */
1878 vrc = RTLdrGetSymbol(gLibrarySDL_ttf, "TTF_RenderUTF8_Blended", (void**)&pTTF_RenderUTF8_Blended);
1879 if (RT_FAILURE(vrc))
1880 pTTF_RenderUTF8_Blended = NULL;
1881 vrc = VINF_SUCCESS;
1882 }
1883 if (RT_SUCCESS(vrc))
1884 vrc = RTLdrGetSymbol(gLibrarySDL_ttf, "TTF_CloseFont", (void**)&pTTF_CloseFont);
1885 if (RT_SUCCESS(vrc))
1886 vrc = RTLdrGetSymbol(gLibrarySDL_ttf, "TTF_Quit", (void**)&pTTF_Quit);
1887 if (RT_SUCCESS(vrc))
1888 vrc = gpFramebuffer[0]->initSecureLabel(SECURE_LABEL_HEIGHT, secureLabelFontFile, secureLabelPointSize, secureLabelFontOffs);
1889 if (RT_FAILURE(vrc))
1890 {
1891 RTPrintf("Error: could not initialize secure labeling: rc = %Rrc\n", vrc);
1892 goto leave;
1893 }
1894 Bstr key = VBOXSDL_SECURELABEL_EXTRADATA;
1895 Bstr label;
1896 gMachine->GetExtraData(key, label.asOutParam());
1897 Utf8Str labelUtf8 = label;
1898 /*
1899 * Now update the label
1900 */
1901 gpFramebuffer[0]->setSecureLabelColor(secureLabelColorFG, secureLabelColorBG);
1902 gpFramebuffer[0]->setSecureLabelText(labelUtf8.raw());
1903 }
1904#endif
1905
1906#ifdef VBOXSDL_WITH_X11
1907 /* NOTE1: We still want Ctrl-C to work, so we undo the SDL redirections.
1908 * NOTE2: We have to remove the PidFile if this file exists. */
1909 signal(SIGINT, signal_handler_SIGINT);
1910 signal(SIGQUIT, signal_handler_SIGINT);
1911 signal(SIGSEGV, signal_handler_SIGINT);
1912#endif
1913
1914
1915 for (ULONG i = 0; i < gcMonitors; i++)
1916 {
1917 // register our framebuffer
1918 rc = gDisplay->SetFramebuffer(i, gpFramebuffer[i]);
1919 if (FAILED(rc))
1920 {
1921 RTPrintf("Error: could not register framebuffer object!\n");
1922 goto leave;
1923 }
1924 IFramebuffer *dummyFb;
1925 LONG xOrigin, yOrigin;
1926 rc = gDisplay->GetFramebuffer(i, &dummyFb, &xOrigin, &yOrigin);
1927 gpFramebuffer[i]->setOrigin(xOrigin, yOrigin);
1928 }
1929
1930 // register a callback for global events
1931 cbVBoxImpl = new VBoxSDLCallback();
1932 rc = createCallbackWrapper((IVirtualBoxCallback*)cbVBoxImpl, callback.asOutParam());
1933 if (FAILED(rc))
1934 goto leave;
1935 virtualBox->RegisterCallback(callback);
1936
1937 // register a callback for machine events
1938 cbConsoleImpl = new VBoxSDLConsoleCallback();
1939 rc = createCallbackWrapper((IConsoleCallback*)cbConsoleImpl, consoleCallback.asOutParam());
1940 if (FAILED(rc))
1941 goto leave;
1942 gConsole->RegisterCallback(consoleCallback);
1943 // until we've tried to to start the VM, ignore power off events
1944 cbConsoleImpl->ignorePowerOffEvents(true);
1945
1946#ifdef VBOX_WITH_VRDP
1947 if (portVRDP)
1948 {
1949 rc = gMachine->COMGETTER(VRDPServer)(gVrdpServer.asOutParam());
1950 AssertMsg((rc == S_OK) && gVrdpServer, ("Could not get VRDP Server! rc = 0x%x\n", rc));
1951 if (gVrdpServer)
1952 {
1953 // has a non standard VRDP port been requested?
1954 if (portVRDP > 0)
1955 {
1956 Bstr bstr = portVRDP;
1957 rc = gVrdpServer->COMSETTER(Ports)(bstr);
1958 if (rc != S_OK)
1959 {
1960 RTPrintf("Error: could not set VRDP port! rc = 0x%x\n", rc);
1961 goto leave;
1962 }
1963 }
1964 // now enable VRDP
1965 rc = gVrdpServer->COMSETTER(Enabled)(TRUE);
1966 if (rc != S_OK)
1967 {
1968 RTPrintf("Error: could not enable VRDP server! rc = 0x%x\n", rc);
1969 goto leave;
1970 }
1971 }
1972 }
1973#endif
1974
1975 rc = E_FAIL;
1976#ifdef VBOXSDL_ADVANCED_OPTIONS
1977 if (fRawR0 != ~0U)
1978 {
1979 if (!gMachineDebugger)
1980 {
1981 RTPrintf("Error: No debugger object; -%srawr0 cannot be executed!\n", fRawR0 ? "" : "no");
1982 goto leave;
1983 }
1984 gMachineDebugger->COMSETTER(RecompileSupervisor)(!fRawR0);
1985 }
1986 if (fRawR3 != ~0U)
1987 {
1988 if (!gMachineDebugger)
1989 {
1990 RTPrintf("Error: No debugger object; -%srawr3 cannot be executed!\n", fRawR0 ? "" : "no");
1991 goto leave;
1992 }
1993 gMachineDebugger->COMSETTER(RecompileUser)(!fRawR3);
1994 }
1995 if (fPATM != ~0U)
1996 {
1997 if (!gMachineDebugger)
1998 {
1999 RTPrintf("Error: No debugger object; -%spatm cannot be executed!\n", fRawR0 ? "" : "no");
2000 goto leave;
2001 }
2002 gMachineDebugger->COMSETTER(PATMEnabled)(fPATM);
2003 }
2004 if (fCSAM != ~0U)
2005 {
2006 if (!gMachineDebugger)
2007 {
2008 RTPrintf("Error: No debugger object; -%scsam cannot be executed!\n", fRawR0 ? "" : "no");
2009 goto leave;
2010 }
2011 gMachineDebugger->COMSETTER(CSAMEnabled)(fCSAM);
2012 }
2013 if (fHWVirt != ~0U)
2014 {
2015 gMachine->SetHWVirtExProperty(HWVirtExPropertyType_Enabled, fHWVirt);
2016 }
2017 if (u32WarpDrive != 0)
2018 {
2019 if (!gMachineDebugger)
2020 {
2021 RTPrintf("Error: No debugger object; --warpdrive %d cannot be executed!\n", u32WarpDrive);
2022 goto leave;
2023 }
2024 gMachineDebugger->COMSETTER(VirtualTimeRate)(u32WarpDrive);
2025 }
2026#endif /* VBOXSDL_ADVANCED_OPTIONS */
2027
2028 /* start with something in the titlebar */
2029 UpdateTitlebar(TITLEBAR_NORMAL);
2030
2031 /* memorize the default cursor */
2032 gpDefaultCursor = SDL_GetCursor();
2033
2034#if !defined(VBOX_WITH_SDL13)
2035# if defined(VBOXSDL_WITH_X11)
2036 /* Get Window Manager info. We only need the X11 display. */
2037 SDL_VERSION(&gSdlInfo.version);
2038 if (!SDL_GetWMInfo(&gSdlInfo))
2039 RTPrintf("Error: could not get SDL Window Manager info -- no Xcursor support!\n");
2040 else
2041 gfXCursorEnabled = TRUE;
2042
2043# if !defined(VBOX_WITHOUT_XCURSOR)
2044 /* SDL uses its own (plain) default cursor. Use the left arrow cursor instead which might look
2045 * much better if a mouse cursor theme is installed. */
2046 if (gfXCursorEnabled)
2047 {
2048 gpDefaultOrigX11Cursor = *(Cursor*)gpDefaultCursor->wm_cursor;
2049 *(Cursor*)gpDefaultCursor->wm_cursor = XCreateFontCursor(gSdlInfo.info.x11.display, XC_left_ptr);
2050 SDL_SetCursor(gpDefaultCursor);
2051 }
2052# endif
2053# endif /* VBOXSDL_WITH_X11 */
2054
2055 /* create a fake empty cursor */
2056 {
2057 uint8_t cursorData[1] = {0};
2058 gpCustomCursor = SDL_CreateCursor(cursorData, cursorData, 8, 1, 0, 0);
2059 gpCustomOrigWMcursor = gpCustomCursor->wm_cursor;
2060 gpCustomCursor->wm_cursor = NULL;
2061 }
2062#endif /* !VBOX_WITH_SDL13 */
2063
2064 /*
2065 * Register our user signal handler.
2066 */
2067#ifdef VBOXSDL_WITH_X11
2068 struct sigaction sa;
2069 sa.sa_sigaction = signal_handler_SIGUSR1;
2070 sigemptyset (&sa.sa_mask);
2071 sa.sa_flags = SA_RESTART | SA_SIGINFO;
2072 sigaction (SIGUSR1, &sa, NULL);
2073#endif /* VBOXSDL_WITH_X11 */
2074
2075 /*
2076 * Start the VM execution thread. This has to be done
2077 * asynchronously as powering up can take some time
2078 * (accessing devices such as the host DVD drive). In
2079 * the meantime, we have to service the SDL event loop.
2080 */
2081 SDL_Event event;
2082
2083 LogFlow(("Powering up the VM...\n"));
2084 rc = gConsole->PowerUp(gProgress.asOutParam());
2085 if (rc != S_OK)
2086 {
2087 com::ErrorInfo info(gConsole);
2088 if (info.isBasicAvailable())
2089 PrintError("Failed to power up VM", info.getText().raw());
2090 else
2091 RTPrintf("Error: failed to power up VM! No error text available.\n");
2092 goto leave;
2093 }
2094
2095#ifdef USE_XPCOM_QUEUE_THREAD
2096 /*
2097 * Before we starting to do stuff, we have to launch the XPCOM
2098 * event queue thread. It will wait for events and send messages
2099 * to the SDL thread. After having done this, we should fairly
2100 * quickly start to process the SDL event queue as an XPCOM
2101 * event storm might arrive. Stupid SDL has a ridiculously small
2102 * event queue buffer!
2103 */
2104 startXPCOMEventQueueThread(eventQ->getSelectFD());
2105#endif /* USE_XPCOM_QUEUE_THREAD */
2106
2107 /* termination flag */
2108 bool fTerminateDuringStartup;
2109 fTerminateDuringStartup = false;
2110
2111 LogRel(("VBoxSDL: NUM lock initially %s, CAPS lock initially %s\n",
2112 !!(SDL_GetModState() & KMOD_NUM) ? "ON" : "OFF",
2113 !!(SDL_GetModState() & KMOD_CAPS) ? "ON" : "OFF"));
2114
2115 /* start regular timer so we don't starve in the event loop */
2116 SDL_TimerID sdlTimer;
2117 sdlTimer = SDL_AddTimer(100, StartupTimer, NULL);
2118
2119 /* loop until the powerup processing is done */
2120 MachineState_T machineState;
2121 do
2122 {
2123 rc = gMachine->COMGETTER(State)(&machineState);
2124 if ( rc == S_OK
2125 && ( machineState == MachineState_Starting
2126 || machineState == MachineState_Restoring
2127 || machineState == MachineState_TeleportingIn
2128 )
2129 )
2130 {
2131 /*
2132 * wait for the next event. This is uncritical as
2133 * power up guarantees to change the machine state
2134 * to either running or aborted and a machine state
2135 * change will send us an event. However, we have to
2136 * service the XPCOM event queue!
2137 */
2138#ifdef USE_XPCOM_QUEUE_THREAD
2139 if (!fXPCOMEventThreadSignaled)
2140 {
2141 signalXPCOMEventQueueThread();
2142 fXPCOMEventThreadSignaled = true;
2143 }
2144#endif
2145 /*
2146 * Wait for SDL events.
2147 */
2148 if (WaitSDLEvent(&event))
2149 {
2150 switch (event.type)
2151 {
2152 /*
2153 * Timer event. Used to have the titlebar updated.
2154 */
2155 case SDL_USER_EVENT_TIMER:
2156 {
2157 /*
2158 * Update the title bar.
2159 */
2160 UpdateTitlebar(TITLEBAR_STARTUP);
2161 break;
2162 }
2163
2164 /*
2165 * User specific resize event.
2166 */
2167 case SDL_USER_EVENT_RESIZE:
2168 {
2169 LogFlow(("SDL_USER_EVENT_RESIZE\n"));
2170 IFramebuffer *dummyFb;
2171 LONG xOrigin, yOrigin;
2172 gpFramebuffer[event.user.code]->resizeGuest();
2173 /* update xOrigin, yOrigin -> mouse */
2174 rc = gDisplay->GetFramebuffer(event.user.code, &dummyFb, &xOrigin, &yOrigin);
2175 gpFramebuffer[event.user.code]->setOrigin(xOrigin, yOrigin);
2176 /* notify the display that the resize has been completed */
2177 gDisplay->ResizeCompleted(event.user.code);
2178 break;
2179 }
2180
2181#ifdef USE_XPCOM_QUEUE_THREAD
2182 /*
2183 * User specific XPCOM event queue event
2184 */
2185 case SDL_USER_EVENT_XPCOM_EVENTQUEUE:
2186 {
2187 LogFlow(("SDL_USER_EVENT_XPCOM_EVENTQUEUE: processing XPCOM event queue...\n"));
2188 eventQ->processEventQueue(0);
2189 signalXPCOMEventQueueThread();
2190 break;
2191 }
2192#endif /* USE_XPCOM_QUEUE_THREAD */
2193
2194 /*
2195 * Termination event from the on state change callback.
2196 */
2197 case SDL_USER_EVENT_TERMINATE:
2198 {
2199 if (event.user.code != VBOXSDL_TERM_NORMAL)
2200 {
2201 com::ProgressErrorInfo info(gProgress);
2202 if (info.isBasicAvailable())
2203 PrintError("Failed to power up VM", info.getText().raw());
2204 else
2205 RTPrintf("Error: failed to power up VM! No error text available.\n");
2206 }
2207 fTerminateDuringStartup = true;
2208 break;
2209 }
2210
2211 default:
2212 {
2213 LogBird(("VBoxSDL: Unknown SDL event %d (pre)\n", event.type));
2214 break;
2215 }
2216 }
2217
2218 }
2219 }
2220 eventQ->processEventQueue(0);
2221 } while ( rc == S_OK
2222 && ( machineState == MachineState_Starting
2223 || machineState == MachineState_Restoring
2224 || machineState == MachineState_TeleportingIn
2225 )
2226 );
2227
2228 /* kill the timer again */
2229 SDL_RemoveTimer(sdlTimer);
2230 sdlTimer = 0;
2231
2232 /* are we supposed to terminate the process? */
2233 if (fTerminateDuringStartup)
2234 goto leave;
2235
2236 /* did the power up succeed? */
2237 if (machineState != MachineState_Running)
2238 {
2239 com::ProgressErrorInfo info(gProgress);
2240 if (info.isBasicAvailable())
2241 PrintError("Failed to power up VM", info.getText().raw());
2242 else
2243 RTPrintf("Error: failed to power up VM! No error text available (rc = 0x%x state = %d)\n", rc, machineState);
2244 goto leave;
2245 }
2246
2247 // accept power off events from now on because we're running
2248 // note that there's a possible race condition here...
2249 cbConsoleImpl->ignorePowerOffEvents(false);
2250
2251 rc = gConsole->COMGETTER(Keyboard)(gKeyboard.asOutParam());
2252 if (!gKeyboard)
2253 {
2254 RTPrintf("Error: could not get keyboard object!\n");
2255 goto leave;
2256 }
2257 gConsole->COMGETTER(Mouse)(gMouse.asOutParam());
2258 if (!gMouse)
2259 {
2260 RTPrintf("Error: could not get mouse object!\n");
2261 goto leave;
2262 }
2263
2264 UpdateTitlebar(TITLEBAR_NORMAL);
2265
2266 /*
2267 * Enable keyboard repeats
2268 */
2269 SDL_EnableKeyRepeat(SDL_DEFAULT_REPEAT_DELAY, SDL_DEFAULT_REPEAT_INTERVAL);
2270
2271 /*
2272 * Create PID file.
2273 */
2274 if (gpszPidFile)
2275 {
2276 char szBuf[32];
2277 const char *pcszLf = "\n";
2278 RTFILE PidFile;
2279 RTFileOpen(&PidFile, gpszPidFile, RTFILE_O_WRITE | RTFILE_O_CREATE_REPLACE | RTFILE_O_DENY_NONE);
2280 RTStrFormatNumber(szBuf, RTProcSelf(), 10, 0, 0, 0);
2281 RTFileWrite(PidFile, szBuf, strlen(szBuf), NULL);
2282 RTFileWrite(PidFile, pcszLf, strlen(pcszLf), NULL);
2283 RTFileClose(PidFile);
2284 }
2285
2286 /*
2287 * Main event loop
2288 */
2289#ifdef USE_XPCOM_QUEUE_THREAD
2290 if (!fXPCOMEventThreadSignaled)
2291 {
2292 signalXPCOMEventQueueThread();
2293 }
2294#endif
2295 LogFlow(("VBoxSDL: Entering big event loop\n"));
2296 while (WaitSDLEvent(&event))
2297 {
2298 switch (event.type)
2299 {
2300 /*
2301 * The screen needs to be repainted.
2302 */
2303#ifdef VBOX_WITH_SDL13
2304 case SDL_WINDOWEVENT:
2305 {
2306 switch (event.window.event)
2307 {
2308 case SDL_WINDOWEVENT_EXPOSED:
2309 {
2310 VBoxSDLFB *fb = getFbFromWinId(event.window.windowID);
2311 if (fb)
2312 fb->repaint();
2313 break;
2314 }
2315 case SDL_WINDOWEVENT_FOCUS_GAINED:
2316 {
2317 break;
2318 }
2319 default:
2320 break;
2321 }
2322 }
2323#else
2324 case SDL_VIDEOEXPOSE:
2325 {
2326 gpFramebuffer[0]->repaint();
2327 break;
2328 }
2329#endif
2330
2331 /*
2332 * Keyboard events.
2333 */
2334 case SDL_KEYDOWN:
2335 case SDL_KEYUP:
2336 {
2337 SDLKey ksym = event.key.keysym.sym;
2338
2339 switch (enmHKeyState)
2340 {
2341 case HKEYSTATE_NORMAL:
2342 {
2343 if ( event.type == SDL_KEYDOWN
2344 && ksym != SDLK_UNKNOWN
2345 && (ksym == gHostKeySym1 || ksym == gHostKeySym2))
2346 {
2347 EvHKeyDown1 = event;
2348 enmHKeyState = ksym == gHostKeySym1 ? HKEYSTATE_DOWN_1ST
2349 : HKEYSTATE_DOWN_2ND;
2350 break;
2351 }
2352 ProcessKey(&event.key);
2353 break;
2354 }
2355
2356 case HKEYSTATE_DOWN_1ST:
2357 case HKEYSTATE_DOWN_2ND:
2358 {
2359 if (gHostKeySym2 != SDLK_UNKNOWN)
2360 {
2361 if ( event.type == SDL_KEYDOWN
2362 && ksym != SDLK_UNKNOWN
2363 && ( (enmHKeyState == HKEYSTATE_DOWN_1ST && ksym == gHostKeySym2)
2364 || (enmHKeyState == HKEYSTATE_DOWN_2ND && ksym == gHostKeySym1)))
2365 {
2366 EvHKeyDown2 = event;
2367 enmHKeyState = HKEYSTATE_DOWN;
2368 break;
2369 }
2370 enmHKeyState = event.type == SDL_KEYUP ? HKEYSTATE_NORMAL
2371 : HKEYSTATE_NOT_IT;
2372 ProcessKey(&EvHKeyDown1.key);
2373 ProcessKey(&event.key);
2374 break;
2375 }
2376 /* fall through if no two-key sequence is used */
2377 }
2378
2379 case HKEYSTATE_DOWN:
2380 {
2381 if (event.type == SDL_KEYDOWN)
2382 {
2383 /* potential host key combination, try execute it */
2384 int irc = HandleHostKey(&event.key);
2385 if (irc == VINF_SUCCESS)
2386 {
2387 enmHKeyState = HKEYSTATE_USED;
2388 break;
2389 }
2390 if (RT_SUCCESS(irc))
2391 goto leave;
2392 }
2393 else /* SDL_KEYUP */
2394 {
2395 if ( ksym != SDLK_UNKNOWN
2396 && (ksym == gHostKeySym1 || ksym == gHostKeySym2))
2397 {
2398 /* toggle grabbing state */
2399 if (!gfGrabbed)
2400 InputGrabStart();
2401 else
2402 InputGrabEnd();
2403
2404 /* SDL doesn't always reset the keystates, correct it */
2405 ResetKeys();
2406 enmHKeyState = HKEYSTATE_NORMAL;
2407 break;
2408 }
2409 }
2410
2411 /* not host key */
2412 enmHKeyState = HKEYSTATE_NOT_IT;
2413 ProcessKey(&EvHKeyDown1.key);
2414 if (gHostKeySym2 != SDLK_UNKNOWN)
2415 ProcessKey(&EvHKeyDown2.key);
2416 ProcessKey(&event.key);
2417 break;
2418 }
2419
2420 case HKEYSTATE_USED:
2421 {
2422 if ((SDL_GetModState() & ~(KMOD_MODE | KMOD_NUM | KMOD_RESERVED)) == 0)
2423 enmHKeyState = HKEYSTATE_NORMAL;
2424 if (event.type == SDL_KEYDOWN)
2425 {
2426 int irc = HandleHostKey(&event.key);
2427 if (RT_SUCCESS(irc) && irc != VINF_SUCCESS)
2428 goto leave;
2429 }
2430 break;
2431 }
2432
2433 default:
2434 AssertMsgFailed(("enmHKeyState=%d\n", enmHKeyState));
2435 /* fall thru */
2436 case HKEYSTATE_NOT_IT:
2437 {
2438 if ((SDL_GetModState() & ~(KMOD_MODE | KMOD_NUM | KMOD_RESERVED)) == 0)
2439 enmHKeyState = HKEYSTATE_NORMAL;
2440 ProcessKey(&event.key);
2441 break;
2442 }
2443 } /* state switch */
2444 break;
2445 }
2446
2447 /*
2448 * The window was closed.
2449 */
2450 case SDL_QUIT:
2451 {
2452 if (!gfACPITerm || gSdlQuitTimer)
2453 goto leave;
2454 if (gConsole)
2455 gConsole->PowerButton();
2456 gSdlQuitTimer = SDL_AddTimer(1000, QuitTimer, NULL);
2457 break;
2458 }
2459
2460 /*
2461 * The mouse has moved
2462 */
2463 case SDL_MOUSEMOTION:
2464 {
2465 if (gfGrabbed || UseAbsoluteMouse())
2466 {
2467 VBoxSDLFB *fb;
2468#ifdef VBOX_WITH_SDL13
2469 fb = getFbFromWinId(event.motion.windowID);
2470#else
2471 fb = gpFramebuffer[0];
2472#endif
2473 SendMouseEvent(fb, 0, 0, 0);
2474 }
2475 break;
2476 }
2477
2478 /*
2479 * A mouse button has been clicked or released.
2480 */
2481 case SDL_MOUSEBUTTONDOWN:
2482 case SDL_MOUSEBUTTONUP:
2483 {
2484 SDL_MouseButtonEvent *bev = &event.button;
2485 /* don't grab on mouse click if we have guest additions */
2486 if (!gfGrabbed && !UseAbsoluteMouse() && gfGrabOnMouseClick)
2487 {
2488 if (event.type == SDL_MOUSEBUTTONDOWN && (bev->state & SDL_BUTTON_LMASK))
2489 {
2490 /* start grabbing all events */
2491 InputGrabStart();
2492 }
2493 }
2494 else if (gfGrabbed || UseAbsoluteMouse())
2495 {
2496 int dz = bev->button == SDL_BUTTON_WHEELUP
2497 ? -1
2498 : bev->button == SDL_BUTTON_WHEELDOWN
2499 ? +1
2500 : 0;
2501
2502 /* end host key combination (CTRL+MouseButton) */
2503 switch (enmHKeyState)
2504 {
2505 case HKEYSTATE_DOWN_1ST:
2506 case HKEYSTATE_DOWN_2ND:
2507 enmHKeyState = HKEYSTATE_NOT_IT;
2508 ProcessKey(&EvHKeyDown1.key);
2509 /* ugly hack: small delay to ensure that the key event is
2510 * actually handled _prior_ to the mouse click event */
2511 RTThreadSleep(20);
2512 break;
2513 case HKEYSTATE_DOWN:
2514 enmHKeyState = HKEYSTATE_NOT_IT;
2515 ProcessKey(&EvHKeyDown1.key);
2516 if (gHostKeySym2 != SDLK_UNKNOWN)
2517 ProcessKey(&EvHKeyDown2.key);
2518 /* ugly hack: small delay to ensure that the key event is
2519 * actually handled _prior_ to the mouse click event */
2520 RTThreadSleep(20);
2521 break;
2522 default:
2523 break;
2524 }
2525
2526 VBoxSDLFB *fb;
2527#ifdef VBOX_WITH_SDL13
2528 fb = getFbFromWinId(event.button.windowID);
2529#else
2530 fb = gpFramebuffer[0];
2531#endif
2532 SendMouseEvent(fb, dz, event.type == SDL_MOUSEBUTTONDOWN, bev->button);
2533 }
2534 break;
2535 }
2536
2537 /*
2538 * The window has gained or lost focus.
2539 */
2540 case SDL_ACTIVEEVENT:
2541 {
2542 /*
2543 * There is a strange behaviour in SDL when running without a window
2544 * manager: When SDL_WM_GrabInput(SDL_GRAB_ON) is called we receive two
2545 * consecutive events SDL_ACTIVEEVENTs (input lost, input gained).
2546 * Asking SDL_GetAppState() seems the better choice.
2547 */
2548 if (gfGrabbed && (SDL_GetAppState() & SDL_APPINPUTFOCUS) == 0)
2549 {
2550 /*
2551 * another window has stolen the (keyboard) input focus
2552 */
2553 InputGrabEnd();
2554 }
2555 break;
2556 }
2557
2558 /*
2559 * The SDL window was resized
2560 */
2561 case SDL_VIDEORESIZE:
2562 {
2563 if (gDisplay)
2564 {
2565 if (gfIgnoreNextResize)
2566 {
2567 gfIgnoreNextResize = FALSE;
2568 break;
2569 }
2570 uResizeWidth = event.resize.w;
2571#ifdef VBOX_SECURELABEL
2572 if (fSecureLabel)
2573 uResizeHeight = RT_MAX(0, event.resize.h - SECURE_LABEL_HEIGHT);
2574 else
2575#endif
2576 uResizeHeight = event.resize.h;
2577 if (gSdlResizeTimer)
2578 SDL_RemoveTimer(gSdlResizeTimer);
2579 gSdlResizeTimer = SDL_AddTimer(300, ResizeTimer, NULL);
2580 }
2581 break;
2582 }
2583
2584 /*
2585 * User specific update event.
2586 */
2587 /** @todo use a common user event handler so that SDL_PeepEvents() won't
2588 * possibly remove other events in the queue!
2589 */
2590 case SDL_USER_EVENT_UPDATERECT:
2591 {
2592 /*
2593 * Decode event parameters.
2594 */
2595 ASMAtomicDecS32(&g_cNotifyUpdateEventsPending);
2596 #define DECODEX(event) (int)((intptr_t)(event).user.data1 >> 16)
2597 #define DECODEY(event) (int)((intptr_t)(event).user.data1 & 0xFFFF)
2598 #define DECODEW(event) (int)((intptr_t)(event).user.data2 >> 16)
2599 #define DECODEH(event) (int)((intptr_t)(event).user.data2 & 0xFFFF)
2600 int x = DECODEX(event);
2601 int y = DECODEY(event);
2602 int w = DECODEW(event);
2603 int h = DECODEH(event);
2604 LogFlow(("SDL_USER_EVENT_UPDATERECT: x = %d, y = %d, w = %d, h = %d\n",
2605 x, y, w, h));
2606
2607 Assert(gpFramebuffer[event.user.code]);
2608 gpFramebuffer[event.user.code]->update(x, y, w, h, true /* fGuestRelative */);
2609
2610 #undef DECODEX
2611 #undef DECODEY
2612 #undef DECODEW
2613 #undef DECODEH
2614 break;
2615 }
2616
2617 /*
2618 * User event: Window resize done
2619 */
2620 case SDL_USER_EVENT_WINDOW_RESIZE_DONE:
2621 {
2622 /**
2623 * @todo This is a workaround for synchronization problems between EMT and the
2624 * SDL main thread. It can happen that the SDL thread already starts a
2625 * new resize operation while the EMT is still busy with the old one
2626 * leading to a deadlock. Therefore we call SetVideoModeHint only once
2627 * when the mouse button was released.
2628 */
2629 /* communicate the resize event to the guest */
2630 gDisplay->SetVideoModeHint(uResizeWidth, uResizeHeight, 0, 0);
2631 break;
2632
2633 }
2634
2635 /*
2636 * User specific resize event.
2637 */
2638 case SDL_USER_EVENT_RESIZE:
2639 {
2640 LogFlow(("SDL_USER_EVENT_RESIZE\n"));
2641 IFramebuffer *dummyFb;
2642 LONG xOrigin, yOrigin;
2643 gpFramebuffer[event.user.code]->resizeGuest();
2644 /* update xOrigin, yOrigin -> mouse */
2645 rc = gDisplay->GetFramebuffer(event.user.code, &dummyFb, &xOrigin, &yOrigin);
2646 gpFramebuffer[event.user.code]->setOrigin(xOrigin, yOrigin);
2647 /* notify the display that the resize has been completed */
2648 gDisplay->ResizeCompleted(event.user.code);
2649 break;
2650 }
2651
2652#ifdef USE_XPCOM_QUEUE_THREAD
2653 /*
2654 * User specific XPCOM event queue event
2655 */
2656 case SDL_USER_EVENT_XPCOM_EVENTQUEUE:
2657 {
2658 LogFlow(("SDL_USER_EVENT_XPCOM_EVENTQUEUE: processing XPCOM event queue...\n"));
2659 eventQ->processEventQueue(0);
2660 signalXPCOMEventQueueThread();
2661 break;
2662 }
2663#endif /* USE_XPCOM_QUEUE_THREAD */
2664
2665 /*
2666 * User specific update title bar notification event
2667 */
2668 case SDL_USER_EVENT_UPDATE_TITLEBAR:
2669 {
2670 UpdateTitlebar(TITLEBAR_NORMAL);
2671 break;
2672 }
2673
2674 /*
2675 * User specific termination event
2676 */
2677 case SDL_USER_EVENT_TERMINATE:
2678 {
2679 if (event.user.code != VBOXSDL_TERM_NORMAL)
2680 RTPrintf("Error: VM terminated abnormally!\n");
2681 goto leave;
2682 }
2683
2684#ifdef VBOX_SECURELABEL
2685 /*
2686 * User specific secure label update event
2687 */
2688 case SDL_USER_EVENT_SECURELABEL_UPDATE:
2689 {
2690 /*
2691 * Query the new label text
2692 */
2693 Bstr key = VBOXSDL_SECURELABEL_EXTRADATA;
2694 Bstr label;
2695 gMachine->GetExtraData(key, label.asOutParam());
2696 Utf8Str labelUtf8 = label;
2697 /*
2698 * Now update the label
2699 */
2700 gpFramebuffer[0]->setSecureLabelText(labelUtf8.raw());
2701 break;
2702 }
2703#endif /* VBOX_SECURELABEL */
2704
2705 /*
2706 * User specific pointer shape change event
2707 */
2708 case SDL_USER_EVENT_POINTER_CHANGE:
2709 {
2710 PointerShapeChangeData *data = (PointerShapeChangeData *) event.user.data1;
2711 SetPointerShape (data);
2712 delete data;
2713 break;
2714 }
2715
2716 /*
2717 * User specific guest capabilities changed
2718 */
2719 case SDL_USER_EVENT_GUEST_CAP_CHANGED:
2720 {
2721 HandleGuestCapsChanged();
2722 break;
2723 }
2724
2725 default:
2726 {
2727 LogBird(("unknown SDL event %d\n", event.type));
2728 break;
2729 }
2730 }
2731 }
2732
2733leave:
2734 if (gpszPidFile)
2735 RTFileDelete(gpszPidFile);
2736
2737 LogFlow(("leaving...\n"));
2738#if defined(VBOX_WITH_XPCOM) && !defined(RT_OS_DARWIN) && !defined(RT_OS_OS2)
2739 /* make sure the XPCOM event queue thread doesn't do anything harmful */
2740 terminateXPCOMQueueThread();
2741#endif /* VBOX_WITH_XPCOM */
2742
2743#ifdef VBOX_WITH_VRDP
2744 if (gVrdpServer)
2745 rc = gVrdpServer->COMSETTER(Enabled)(FALSE);
2746#endif
2747
2748 /*
2749 * Get the machine state.
2750 */
2751 if (gMachine)
2752 gMachine->COMGETTER(State)(&machineState);
2753 else
2754 machineState = MachineState_Aborted;
2755
2756 /*
2757 * Turn off the VM if it's running
2758 */
2759 if ( gConsole
2760 && ( machineState == MachineState_Running
2761 || machineState == MachineState_Teleporting
2762 || machineState == MachineState_LiveSnapshotting
2763 /** @todo power off paused VMs too? */
2764 )
2765 )
2766 do
2767 {
2768 cbConsoleImpl->ignorePowerOffEvents(true);
2769 ComPtr<IProgress> progress;
2770 CHECK_ERROR_BREAK(gConsole, PowerDown(progress.asOutParam()));
2771 CHECK_ERROR_BREAK(progress, WaitForCompletion(-1));
2772 BOOL completed;
2773 CHECK_ERROR_BREAK(progress, COMGETTER(Completed)(&completed));
2774 ASSERT (completed);
2775 LONG hrc;
2776 CHECK_ERROR_BREAK(progress, COMGETTER(ResultCode)(&hrc));
2777 if (FAILED(hrc))
2778 {
2779 com::ErrorInfo info;
2780 if (info.isFullAvailable())
2781 PrintError("Failed to power down VM",
2782 info.getText().raw(), info.getComponent().raw());
2783 else
2784 RTPrintf("Failed to power down virtual machine! No error information available (rc = 0x%x).\n", hrc);
2785 break;
2786 }
2787 } while (0);
2788
2789 /*
2790 * Now we discard all settings so that our changes will
2791 * not be flushed to the permanent configuration
2792 */
2793 if ( gMachine
2794 && machineState != MachineState_Saved)
2795 {
2796 rc = gMachine->DiscardSettings();
2797 AssertComRC(rc);
2798 }
2799
2800 /* close the session */
2801 if (sessionOpened)
2802 {
2803 rc = session->Close();
2804 AssertComRC(rc);
2805 }
2806
2807#ifndef VBOX_WITH_SDL13
2808 /* restore the default cursor and free the custom one if any */
2809 if (gpDefaultCursor)
2810 {
2811# ifdef VBOXSDL_WITH_X11
2812 Cursor pDefaultTempX11Cursor = 0;
2813 if (gfXCursorEnabled)
2814 {
2815 pDefaultTempX11Cursor = *(Cursor*)gpDefaultCursor->wm_cursor;
2816 *(Cursor*)gpDefaultCursor->wm_cursor = gpDefaultOrigX11Cursor;
2817 }
2818# endif /* VBOXSDL_WITH_X11 */
2819 SDL_SetCursor(gpDefaultCursor);
2820# if defined(VBOXSDL_WITH_X11) && !defined(VBOX_WITHOUT_XCURSOR)
2821 if (gfXCursorEnabled)
2822 XFreeCursor(gSdlInfo.info.x11.display, pDefaultTempX11Cursor);
2823# endif /* VBOXSDL_WITH_X11 && !VBOX_WITHOUT_XCURSOR */
2824 }
2825
2826 if (gpCustomCursor)
2827 {
2828 WMcursor *pCustomTempWMCursor = gpCustomCursor->wm_cursor;
2829 gpCustomCursor->wm_cursor = gpCustomOrigWMcursor;
2830 SDL_FreeCursor(gpCustomCursor);
2831 if (pCustomTempWMCursor)
2832 {
2833# if defined (RT_OS_WINDOWS)
2834 ::DestroyCursor(*(HCURSOR *) pCustomTempWMCursor);
2835# elif defined (VBOXSDL_WITH_X11) && !defined (VBOX_WITHOUT_XCURSOR)
2836 if (gfXCursorEnabled)
2837 XFreeCursor(gSdlInfo.info.x11.display, *(Cursor *) pCustomTempWMCursor);
2838# endif /* VBOXSDL_WITH_X11 && !VBOX_WITHOUT_XCURSOR */
2839 free(pCustomTempWMCursor);
2840 }
2841 }
2842#endif
2843
2844 LogFlow(("Releasing mouse, keyboard, vrdpserver, display, console...\n"));
2845 if (gDisplay)
2846 {
2847 for (unsigned i = 0; i < gcMonitors; i++)
2848 gDisplay->SetFramebuffer(i, NULL);
2849 }
2850 gMouse = NULL;
2851 gKeyboard = NULL;
2852 gVrdpServer = NULL;
2853 gDisplay = NULL;
2854 gConsole = NULL;
2855 gMachineDebugger = NULL;
2856 gProgress = NULL;
2857 // we can only uninitialize SDL here because it is not threadsafe
2858
2859 for (unsigned i = 0; i < gcMonitors; i++)
2860 {
2861 if (gpFramebuffer[i])
2862 {
2863 LogFlow(("Releasing framebuffer...\n"));
2864 gpFramebuffer[i]->Release();
2865 gpFramebuffer[i] = NULL;
2866 }
2867 }
2868
2869 VBoxSDLFB::uninit();
2870
2871#ifdef VBOX_SECURELABEL
2872 /* must do this after destructing the framebuffer */
2873 if (gLibrarySDL_ttf)
2874 RTLdrClose(gLibrarySDL_ttf);
2875#endif
2876
2877 /* VirtualBox callback unregistration. */
2878 if (!virtualBox.isNull() && !callback.isNull())
2879 virtualBox->UnregisterCallback(callback);
2880
2881 LogFlow(("Releasing machine, session...\n"));
2882 gMachine = NULL;
2883 session = NULL;
2884 LogFlow(("Releasing VirtualBox object...\n"));
2885 virtualBox = NULL;
2886
2887 // end "all-stuff" scope
2888 ////////////////////////////////////////////////////////////////////////////
2889 }
2890
2891 /* Must be before com::Shutdown() */
2892 callback.setNull();
2893 consoleCallback.setNull();
2894
2895 LogFlow(("Uninitializing COM...\n"));
2896 com::Shutdown();
2897
2898 LogFlow(("Returning from main()!\n"));
2899 RTLogFlush(NULL);
2900 return FAILED(rc) ? 1 : 0;
2901}
2902
2903
2904#ifndef VBOX_WITH_HARDENING
2905/**
2906 * Main entry point
2907 */
2908int main(int argc, char **argv)
2909{
2910 /*
2911 * Before we do *anything*, we initialize the runtime.
2912 */
2913 int rcRT = RTR3InitAndSUPLib();
2914 if (RT_FAILURE(rcRT))
2915 {
2916 RTPrintf("Error: RTR3Init failed rcRC=%d\n", rcRT);
2917 return 1;
2918 }
2919 return TrustedMain(argc, argv, NULL);
2920}
2921#endif /* !VBOX_WITH_HARDENING */
2922
2923
2924/**
2925 * Returns whether the absolute mouse is in use, i.e. both host
2926 * and guest have opted to enable it.
2927 *
2928 * @returns bool Flag whether the absolute mouse is in use
2929 */
2930static bool UseAbsoluteMouse(void)
2931{
2932 return (gfAbsoluteMouseHost && gfAbsoluteMouseGuest);
2933}
2934
2935#if defined(RT_OS_DARWIN) || defined(RT_OS_OS2)
2936/**
2937 * Fallback keycode conversion using SDL symbols.
2938 *
2939 * This is used to catch keycodes that's missing from the translation table.
2940 *
2941 * @returns XT scancode
2942 * @param ev SDL scancode
2943 */
2944static uint16_t Keyevent2KeycodeFallback(const SDL_KeyboardEvent *ev)
2945{
2946 const SDLKey sym = ev->keysym.sym;
2947 Log(("SDL key event: sym=%d scancode=%#x unicode=%#x\n",
2948 sym, ev->keysym.scancode, ev->keysym.unicode));
2949 switch (sym)
2950 { /* set 1 scan code */
2951 case SDLK_ESCAPE: return 0x01;
2952 case SDLK_EXCLAIM:
2953 case SDLK_1: return 0x02;
2954 case SDLK_AT:
2955 case SDLK_2: return 0x03;
2956 case SDLK_HASH:
2957 case SDLK_3: return 0x04;
2958 case SDLK_DOLLAR:
2959 case SDLK_4: return 0x05;
2960 /* % */
2961 case SDLK_5: return 0x06;
2962 case SDLK_CARET:
2963 case SDLK_6: return 0x07;
2964 case SDLK_AMPERSAND:
2965 case SDLK_7: return 0x08;
2966 case SDLK_ASTERISK:
2967 case SDLK_8: return 0x09;
2968 case SDLK_LEFTPAREN:
2969 case SDLK_9: return 0x0a;
2970 case SDLK_RIGHTPAREN:
2971 case SDLK_0: return 0x0b;
2972 case SDLK_UNDERSCORE:
2973 case SDLK_MINUS: return 0x0c;
2974 case SDLK_EQUALS:
2975 case SDLK_PLUS: return 0x0d;
2976 case SDLK_BACKSPACE: return 0x0e;
2977 case SDLK_TAB: return 0x0f;
2978 case SDLK_q: return 0x10;
2979 case SDLK_w: return 0x11;
2980 case SDLK_e: return 0x12;
2981 case SDLK_r: return 0x13;
2982 case SDLK_t: return 0x14;
2983 case SDLK_y: return 0x15;
2984 case SDLK_u: return 0x16;
2985 case SDLK_i: return 0x17;
2986 case SDLK_o: return 0x18;
2987 case SDLK_p: return 0x19;
2988 case SDLK_LEFTBRACKET: return 0x1a;
2989 case SDLK_RIGHTBRACKET: return 0x1b;
2990 case SDLK_RETURN: return 0x1c;
2991 case SDLK_KP_ENTER: return 0x1c | 0x100;
2992 case SDLK_LCTRL: return 0x1d;
2993 case SDLK_RCTRL: return 0x1d | 0x100;
2994 case SDLK_a: return 0x1e;
2995 case SDLK_s: return 0x1f;
2996 case SDLK_d: return 0x20;
2997 case SDLK_f: return 0x21;
2998 case SDLK_g: return 0x22;
2999 case SDLK_h: return 0x23;
3000 case SDLK_j: return 0x24;
3001 case SDLK_k: return 0x25;
3002 case SDLK_l: return 0x26;
3003 case SDLK_COLON:
3004 case SDLK_SEMICOLON: return 0x27;
3005 case SDLK_QUOTEDBL:
3006 case SDLK_QUOTE: return 0x28;
3007 case SDLK_BACKQUOTE: return 0x29;
3008 case SDLK_LSHIFT: return 0x2a;
3009 case SDLK_BACKSLASH: return 0x2b;
3010 case SDLK_z: return 0x2c;
3011 case SDLK_x: return 0x2d;
3012 case SDLK_c: return 0x2e;
3013 case SDLK_v: return 0x2f;
3014 case SDLK_b: return 0x30;
3015 case SDLK_n: return 0x31;
3016 case SDLK_m: return 0x32;
3017 case SDLK_LESS:
3018 case SDLK_COMMA: return 0x33;
3019 case SDLK_GREATER:
3020 case SDLK_PERIOD: return 0x34;
3021 case SDLK_KP_DIVIDE: /*??*/
3022 case SDLK_QUESTION:
3023 case SDLK_SLASH: return 0x35;
3024 case SDLK_RSHIFT: return 0x36;
3025 case SDLK_KP_MULTIPLY:
3026 case SDLK_PRINT: return 0x37; /* fixme */
3027 case SDLK_LALT: return 0x38;
3028 case SDLK_MODE: /* alt gr*/
3029 case SDLK_RALT: return 0x38 | 0x100;
3030 case SDLK_SPACE: return 0x39;
3031 case SDLK_CAPSLOCK: return 0x3a;
3032 case SDLK_F1: return 0x3b;
3033 case SDLK_F2: return 0x3c;
3034 case SDLK_F3: return 0x3d;
3035 case SDLK_F4: return 0x3e;
3036 case SDLK_F5: return 0x3f;
3037 case SDLK_F6: return 0x40;
3038 case SDLK_F7: return 0x41;
3039 case SDLK_F8: return 0x42;
3040 case SDLK_F9: return 0x43;
3041 case SDLK_F10: return 0x44;
3042 case SDLK_PAUSE: return 0x45; /* not right */
3043 case SDLK_NUMLOCK: return 0x45;
3044 case SDLK_SCROLLOCK: return 0x46;
3045 case SDLK_KP7: return 0x47;
3046 case SDLK_HOME: return 0x47 | 0x100;
3047 case SDLK_KP8: return 0x48;
3048 case SDLK_UP: return 0x48 | 0x100;
3049 case SDLK_KP9: return 0x49;
3050 case SDLK_PAGEUP: return 0x49 | 0x100;
3051 case SDLK_KP_MINUS: return 0x4a;
3052 case SDLK_KP4: return 0x4b;
3053 case SDLK_LEFT: return 0x4b | 0x100;
3054 case SDLK_KP5: return 0x4c;
3055 case SDLK_KP6: return 0x4d;
3056 case SDLK_RIGHT: return 0x4d | 0x100;
3057 case SDLK_KP_PLUS: return 0x4e;
3058 case SDLK_KP1: return 0x4f;
3059 case SDLK_END: return 0x4f | 0x100;
3060 case SDLK_KP2: return 0x50;
3061 case SDLK_DOWN: return 0x50 | 0x100;
3062 case SDLK_KP3: return 0x51;
3063 case SDLK_PAGEDOWN: return 0x51 | 0x100;
3064 case SDLK_KP0: return 0x52;
3065 case SDLK_INSERT: return 0x52 | 0x100;
3066 case SDLK_KP_PERIOD: return 0x53;
3067 case SDLK_DELETE: return 0x53 | 0x100;
3068 case SDLK_SYSREQ: return 0x54;
3069 case SDLK_F11: return 0x57;
3070 case SDLK_F12: return 0x58;
3071 case SDLK_F13: return 0x5b;
3072 case SDLK_LMETA:
3073 case SDLK_LSUPER: return 0x5b | 0x100;
3074 case SDLK_F14: return 0x5c;
3075 case SDLK_RMETA:
3076 case SDLK_RSUPER: return 0x5c | 0x100;
3077 case SDLK_F15: return 0x5d;
3078 case SDLK_MENU: return 0x5d | 0x100;
3079#if 0
3080 case SDLK_CLEAR: return 0x;
3081 case SDLK_KP_EQUALS: return 0x;
3082 case SDLK_COMPOSE: return 0x;
3083 case SDLK_HELP: return 0x;
3084 case SDLK_BREAK: return 0x;
3085 case SDLK_POWER: return 0x;
3086 case SDLK_EURO: return 0x;
3087 case SDLK_UNDO: return 0x;
3088#endif
3089 default:
3090 Log(("Unhandled sdl key event: sym=%d scancode=%#x unicode=%#x\n",
3091 ev->keysym.sym, ev->keysym.scancode, ev->keysym.unicode));
3092 return 0;
3093 }
3094}
3095#endif /* RT_OS_DARWIN */
3096
3097/**
3098 * Converts an SDL keyboard eventcode to a XT scancode.
3099 *
3100 * @returns XT scancode
3101 * @param ev SDL scancode
3102 */
3103static uint16_t Keyevent2Keycode(const SDL_KeyboardEvent *ev)
3104{
3105 // start with the scancode determined by SDL
3106 int keycode = ev->keysym.scancode;
3107
3108#ifdef VBOXSDL_WITH_X11
3109# ifdef VBOX_WITH_SDL13
3110
3111 switch (ev->keysym.sym)
3112 {
3113 case SDLK_ESCAPE: return 0x01;
3114 case SDLK_EXCLAIM:
3115 case SDLK_1: return 0x02;
3116 case SDLK_AT:
3117 case SDLK_2: return 0x03;
3118 case SDLK_HASH:
3119 case SDLK_3: return 0x04;
3120 case SDLK_DOLLAR:
3121 case SDLK_4: return 0x05;
3122 /* % */
3123 case SDLK_5: return 0x06;
3124 case SDLK_CARET:
3125 case SDLK_6: return 0x07;
3126 case SDLK_AMPERSAND:
3127 case SDLK_7: return 0x08;
3128 case SDLK_ASTERISK:
3129 case SDLK_8: return 0x09;
3130 case SDLK_LEFTPAREN:
3131 case SDLK_9: return 0x0a;
3132 case SDLK_RIGHTPAREN:
3133 case SDLK_0: return 0x0b;
3134 case SDLK_UNDERSCORE:
3135 case SDLK_MINUS: return 0x0c;
3136 case SDLK_PLUS: return 0x0d;
3137 case SDLK_BACKSPACE: return 0x0e;
3138 case SDLK_TAB: return 0x0f;
3139 case SDLK_q: return 0x10;
3140 case SDLK_w: return 0x11;
3141 case SDLK_e: return 0x12;
3142 case SDLK_r: return 0x13;
3143 case SDLK_t: return 0x14;
3144 case SDLK_y: return 0x15;
3145 case SDLK_u: return 0x16;
3146 case SDLK_i: return 0x17;
3147 case SDLK_o: return 0x18;
3148 case SDLK_p: return 0x19;
3149 case SDLK_RETURN: return 0x1c;
3150 case SDLK_KP_ENTER: return 0x1c | 0x100;
3151 case SDLK_LCTRL: return 0x1d;
3152 case SDLK_RCTRL: return 0x1d | 0x100;
3153 case SDLK_a: return 0x1e;
3154 case SDLK_s: return 0x1f;
3155 case SDLK_d: return 0x20;
3156 case SDLK_f: return 0x21;
3157 case SDLK_g: return 0x22;
3158 case SDLK_h: return 0x23;
3159 case SDLK_j: return 0x24;
3160 case SDLK_k: return 0x25;
3161 case SDLK_l: return 0x26;
3162 case SDLK_COLON: return 0x27;
3163 case SDLK_QUOTEDBL:
3164 case SDLK_QUOTE: return 0x28;
3165 case SDLK_BACKQUOTE: return 0x29;
3166 case SDLK_LSHIFT: return 0x2a;
3167 case SDLK_z: return 0x2c;
3168 case SDLK_x: return 0x2d;
3169 case SDLK_c: return 0x2e;
3170 case SDLK_v: return 0x2f;
3171 case SDLK_b: return 0x30;
3172 case SDLK_n: return 0x31;
3173 case SDLK_m: return 0x32;
3174 case SDLK_LESS: return 0x33;
3175 case SDLK_GREATER: return 0x34;
3176 case SDLK_KP_DIVIDE: /*??*/
3177 case SDLK_QUESTION: return 0x35;
3178 case SDLK_RSHIFT: return 0x36;
3179 case SDLK_KP_MULTIPLY:
3180 case SDLK_PRINT: return 0x37; /* fixme */
3181 case SDLK_LALT: return 0x38;
3182 case SDLK_MODE: /* alt gr*/
3183 case SDLK_RALT: return 0x38 | 0x100;
3184 case SDLK_SPACE: return 0x39;
3185 case SDLK_CAPSLOCK: return 0x3a;
3186 case SDLK_F1: return 0x3b;
3187 case SDLK_F2: return 0x3c;
3188 case SDLK_F3: return 0x3d;
3189 case SDLK_F4: return 0x3e;
3190 case SDLK_F5: return 0x3f;
3191 case SDLK_F6: return 0x40;
3192 case SDLK_F7: return 0x41;
3193 case SDLK_F8: return 0x42;
3194 case SDLK_F9: return 0x43;
3195 case SDLK_F10: return 0x44;
3196 case SDLK_PAUSE: return 0x45; /* not right */
3197 case SDLK_NUMLOCK: return 0x45;
3198 case SDLK_SCROLLOCK: return 0x46;
3199 case SDLK_KP7: return 0x47;
3200 case SDLK_HOME: return 0x47 | 0x100;
3201 case SDLK_KP8: return 0x48;
3202 case SDLK_UP: return 0x48 | 0x100;
3203 case SDLK_KP9: return 0x49;
3204 case SDLK_PAGEUP: return 0x49 | 0x100;
3205 case SDLK_KP_MINUS: return 0x4a;
3206 case SDLK_KP4: return 0x4b;
3207 case SDLK_LEFT: return 0x4b | 0x100;
3208 case SDLK_KP5: return 0x4c;
3209 case SDLK_KP6: return 0x4d;
3210 case SDLK_RIGHT: return 0x4d | 0x100;
3211 case SDLK_KP_PLUS: return 0x4e;
3212 case SDLK_KP1: return 0x4f;
3213 case SDLK_END: return 0x4f | 0x100;
3214 case SDLK_KP2: return 0x50;
3215 case SDLK_DOWN: return 0x50 | 0x100;
3216 case SDLK_KP3: return 0x51;
3217 case SDLK_PAGEDOWN: return 0x51 | 0x100;
3218 case SDLK_KP0: return 0x52;
3219 case SDLK_INSERT: return 0x52 | 0x100;
3220 case SDLK_KP_PERIOD: return 0x53;
3221 case SDLK_DELETE: return 0x53 | 0x100;
3222 case SDLK_SYSREQ: return 0x54;
3223 case SDLK_F11: return 0x57;
3224 case SDLK_F12: return 0x58;
3225 case SDLK_F13: return 0x5b;
3226 case SDLK_F14: return 0x5c;
3227 case SDLK_F15: return 0x5d;
3228 case SDLK_MENU: return 0x5d | 0x100;
3229 default:
3230 return 0;
3231 }
3232 // workaround for SDL keyboard translation issues on Linux
3233 // keycodes > 0x100 are sent as 0xe0 keycode
3234 // Note that these are the keycodes used by XFree86/X.org
3235 // servers on a Linux host, and will almost certainly not
3236 // work on other hosts or on other servers on Linux hosts.
3237 // For a more general approach, see the Wine code in the GUI.
3238
3239# else
3240 static const uint16_t x_keycode_to_pc_keycode[61] =
3241 {
3242 0x47|0x100, /* 97 Home */
3243 0x48|0x100, /* 98 Up */
3244 0x49|0x100, /* 99 PgUp */
3245 0x4b|0x100, /* 100 Left */
3246 0x4c, /* 101 KP-5 */
3247 0x4d|0x100, /* 102 Right */
3248 0x4f|0x100, /* 103 End */
3249 0x50|0x100, /* 104 Down */
3250 0x51|0x100, /* 105 PgDn */
3251 0x52|0x100, /* 106 Ins */
3252 0x53|0x100, /* 107 Del */
3253 0x1c|0x100, /* 108 Enter */
3254 0x1d|0x100, /* 109 Ctrl-R */
3255 0x0, /* 110 Pause */
3256 0x37|0x100, /* 111 Print */
3257 0x35|0x100, /* 112 Divide */
3258 0x38|0x100, /* 113 Alt-R */
3259 0x46|0x100, /* 114 Break */
3260 0x5b|0x100, /* 115 Win Left */
3261 0x5c|0x100, /* 116 Win Right */
3262 0x5d|0x100, /* 117 Win Menu */
3263 0x0, /* 118 */
3264 0x0, /* 119 */
3265 0x0, /* 120 */
3266 0xf1, /* 121 Korean Hangul to Latin?? */
3267 0xf2, /* 122 Korean Hangul to Hanja?? */
3268 0x0, /* 123 */
3269 0x0, /* 124 */
3270 0x0, /* 125 */
3271 0x0, /* 126 */
3272 0x0, /* 127 */
3273 0x0, /* 128 */
3274 0x79, /* 129 Japanese Henkan */
3275 0x0, /* 130 */
3276 0x7b, /* 131 Japanese Muhenkan */
3277 0x0, /* 132 */
3278 0x7d, /* 133 Japanese Yen */
3279 0x7e, /* 134 Brazilian keypad */
3280 0x0, /* 135 */
3281 0x47, /* 136 KP_7 */
3282 0x48, /* 137 KP_8 */
3283 0x49, /* 138 KP_9 */
3284 0x4b, /* 139 KP_4 */
3285 0x4c, /* 140 KP_5 */
3286 0x4d, /* 141 KP_6 */
3287 0x4f, /* 142 KP_1 */
3288 0x50, /* 143 KP_2 */
3289 0x51, /* 144 KP_3 */
3290 0x52, /* 145 KP_0 */
3291 0x53, /* 146 KP_. */
3292 0x47, /* 147 KP_HOME */
3293 0x48, /* 148 KP_UP */
3294 0x49, /* 149 KP_PgUp */
3295 0x4b, /* 150 KP_Left */
3296 0x4c, /* 151 KP_ */
3297 0x4d, /* 152 KP_Right */
3298 0x4f, /* 153 KP_End */
3299 0x50, /* 154 KP_Down */
3300 0x51, /* 155 KP_PgDn */
3301 0x52, /* 156 KP_Ins */
3302 0x53, /* 157 KP_Del */
3303 };
3304
3305 // workaround for SDL keyboard translation issues on EVDEV
3306 // keycodes > 0x100 are sent as 0xe0 keycode
3307 // these values are simply pulled from x_keycode_to_pc_keycode
3308 // not a whole lot of testing of the 'weird' values has taken
3309 // place (I don't own a Japanese or Korean keyboard)
3310 static const uint16_t evdev_keycode_to_pc_keycode[61] =
3311 {
3312 0x0, /* 97 EVDEV - RO ("Internet" Keyboards) */
3313 0x0, /* 98 EVDEV - KATA (Katakana) */
3314 0x0, /* 99 EVDEV - HIRA (Hiragana) */
3315 0x79, /* 100 EVDEV - HENK (Henkan) */
3316 0x70, /* 101 EVDEV - HKTG (Hiragana/Katakana toggle) */
3317 0x7b, /* 102 EVDEV - MUHE (Muhenkan) */
3318 0x0, /* 103 EVDEV - JPCM (KPJPComma) */
3319 0x1c|0x100, /* 104 EVDEV - KPEN */
3320 0x1d|0x100, /* 105 EVDEV - RCTL */
3321 0x35|0x100, /* 106 EVDEV - KPDV */
3322 0x37|0x100, /* 107 EVDEV - PRSC ***FIXME*** */
3323 0x38|0x100, /* 108 EVDEV - RALT */
3324 0x0, /* 109 EVDEV - LNFD ("Internet" Keyboards) */
3325 0x47|0x100, /* 110 EVDEV - HOME ***FIXME*** */
3326 0x48|0x100, /* 111 EVDEV - UP */
3327 0x49|0x100, /* 112 EVDEV - PGUP */
3328 0x4b|0x100, /* 113 EVDEV - LEFT */
3329 0x4d|0x100, /* 114 EVDEV - RGHT */
3330 0x4f|0x100, /* 115 EVDEV - END */
3331 0x50|0x100, /* 116 EVDEV - DOWN */
3332 0x51|0x100, /* 117 EVDEV - PGDN */
3333 0x52|0x100, /* 118 EVDEV - INS */
3334 0x53|0x100, /* 119 EVDEV - DELE */
3335 0x0, /* 120 EVDEV - I120 ("Internet" Keyboards) */
3336 //121-124 Solaris Compatibilty Stuff
3337 0x0, /* 121 EVDEV - MUTE */
3338 0x0, /* 122 EVDEV - VOL- */
3339 0x0, /* 123 EVDEV - VOL+ */
3340 0x0, /* 124 EVDEV - POWR */
3341 0x0, /* 125 EVDEV - KPEQ */
3342 0x0, /* 126 EVDEV - I126 ("Internet" Keyboards) */
3343 0x0, /* 127 EVDEV - PAUS */
3344 0x0, /* 128 EVDEV - ???? */
3345 0x0, /* 129 EVDEV - I129 ("Internet" Keyboards) */
3346 0xf1, /* 130 EVDEV - HNGL (Korean Hangul Latin toggle) */
3347 0xf2, /* 131 EVDEV - HJCV (Korean Hangul Hanja toggle) */
3348 0x7d, /* 132 EVDEV - AE13 (Yen) */
3349 0x5b|0x100, /* 133 EVDEV - LWIN */
3350 0x5c|0x100, /* 134 EVDEV - RWIN */
3351 0x5d|0x100, /* 135 EVDEV - MENU */
3352 //136-146 Solaris Stuff
3353 0x0, /* 136 EVDEV - STOP */
3354 0x0, /* 137 EVDEV - AGAI */
3355 0x0, /* 138 EVDEV - PROP */
3356 0x0, /* 139 EVDEV - UNDO */
3357 0x0, /* 140 EVDEV - FRNT */
3358 0x0, /* 141 EVDEV - COPY */
3359 0x0, /* 142 EVDEV - OPEN */
3360 0x0, /* 143 EVDEV - PAST */
3361 0x0, /* 144 EVDEV - FIND */
3362 0x0, /* 145 EVDEV - CUT */
3363 0x0, /* 146 EVDEV - HELP */
3364 //Extended Keys ("Internet" Keyboards)
3365 0x0, /* 147 EVDEV - I147 */
3366 0x0, /* 148 EVDEV - I148 */
3367 0x0, /* 149 EVDEV - I149 */
3368 0x0, /* 150 EVDEV - I150 */
3369 0x0, /* 151 EVDEV - I151 */
3370 0x0, /* 152 EVDEV - I152 */
3371 0x0, /* 153 EVDEV - I153 */
3372 0x0, /* 154 EVDEV - I154 */
3373 0x0, /* 155 EVDEV - I156 */
3374 0x0, /* 156 EVDEV - I157 */
3375 0x0, /* 157 EVDEV - I158 */
3376 };
3377
3378 if (keycode < 9)
3379 {
3380 keycode = 0;
3381 }
3382 else if (keycode < 97)
3383 {
3384 // just an offset (Xorg MIN_KEYCODE)
3385 keycode -= 8;
3386 }
3387# ifdef RT_OS_LINUX
3388 else if (keycode < 158 && guseEvdevKeymap)
3389 {
3390 // apply EVDEV conversion table
3391 keycode = evdev_keycode_to_pc_keycode[keycode - 97];
3392 }
3393# endif
3394 else if (keycode < 158)
3395 {
3396 // apply conversion table
3397 keycode = x_keycode_to_pc_keycode[keycode - 97];
3398 }
3399 else if (keycode == 208)
3400 {
3401 // Japanese Hiragana to Katakana
3402 keycode = 0x70;
3403 }
3404 else if (keycode == 211)
3405 {
3406 // Japanese backslash/underscore and Brazilian backslash/question mark
3407 keycode = 0x73;
3408 }
3409 else
3410 {
3411 keycode = 0;
3412 }
3413# endif
3414#elif defined(RT_OS_DARWIN)
3415 /* This is derived partially from SDL_QuartzKeys.h and partially from testing. */
3416 static const uint16_t s_aMacToSet1[] =
3417 {
3418 /* set-1 SDL_QuartzKeys.h */
3419 0x1e, /* QZ_a 0x00 */
3420 0x1f, /* QZ_s 0x01 */
3421 0x20, /* QZ_d 0x02 */
3422 0x21, /* QZ_f 0x03 */
3423 0x23, /* QZ_h 0x04 */
3424 0x22, /* QZ_g 0x05 */
3425 0x2c, /* QZ_z 0x06 */
3426 0x2d, /* QZ_x 0x07 */
3427 0x2e, /* QZ_c 0x08 */
3428 0x2f, /* QZ_v 0x09 */
3429 0x56, /* between lshift and z. 'INT 1'? */
3430 0x30, /* QZ_b 0x0B */
3431 0x10, /* QZ_q 0x0C */
3432 0x11, /* QZ_w 0x0D */
3433 0x12, /* QZ_e 0x0E */
3434 0x13, /* QZ_r 0x0F */
3435 0x15, /* QZ_y 0x10 */
3436 0x14, /* QZ_t 0x11 */
3437 0x02, /* QZ_1 0x12 */
3438 0x03, /* QZ_2 0x13 */
3439 0x04, /* QZ_3 0x14 */
3440 0x05, /* QZ_4 0x15 */
3441 0x07, /* QZ_6 0x16 */
3442 0x06, /* QZ_5 0x17 */
3443 0x0d, /* QZ_EQUALS 0x18 */
3444 0x0a, /* QZ_9 0x19 */
3445 0x08, /* QZ_7 0x1A */
3446 0x0c, /* QZ_MINUS 0x1B */
3447 0x09, /* QZ_8 0x1C */
3448 0x0b, /* QZ_0 0x1D */
3449 0x1b, /* QZ_RIGHTBRACKET 0x1E */
3450 0x18, /* QZ_o 0x1F */
3451 0x16, /* QZ_u 0x20 */
3452 0x1a, /* QZ_LEFTBRACKET 0x21 */
3453 0x17, /* QZ_i 0x22 */
3454 0x19, /* QZ_p 0x23 */
3455 0x1c, /* QZ_RETURN 0x24 */
3456 0x26, /* QZ_l 0x25 */
3457 0x24, /* QZ_j 0x26 */
3458 0x28, /* QZ_QUOTE 0x27 */
3459 0x25, /* QZ_k 0x28 */
3460 0x27, /* QZ_SEMICOLON 0x29 */
3461 0x2b, /* QZ_BACKSLASH 0x2A */
3462 0x33, /* QZ_COMMA 0x2B */
3463 0x35, /* QZ_SLASH 0x2C */
3464 0x31, /* QZ_n 0x2D */
3465 0x32, /* QZ_m 0x2E */
3466 0x34, /* QZ_PERIOD 0x2F */
3467 0x0f, /* QZ_TAB 0x30 */
3468 0x39, /* QZ_SPACE 0x31 */
3469 0x29, /* QZ_BACKQUOTE 0x32 */
3470 0x0e, /* QZ_BACKSPACE 0x33 */
3471 0x9c, /* QZ_IBOOK_ENTER 0x34 */
3472 0x01, /* QZ_ESCAPE 0x35 */
3473 0x5c|0x100, /* QZ_RMETA 0x36 */
3474 0x5b|0x100, /* QZ_LMETA 0x37 */
3475 0x2a, /* QZ_LSHIFT 0x38 */
3476 0x3a, /* QZ_CAPSLOCK 0x39 */
3477 0x38, /* QZ_LALT 0x3A */
3478 0x1d, /* QZ_LCTRL 0x3B */
3479 0x36, /* QZ_RSHIFT 0x3C */
3480 0x38|0x100, /* QZ_RALT 0x3D */
3481 0x1d|0x100, /* QZ_RCTRL 0x3E */
3482 0, /* */
3483 0, /* */
3484 0x53, /* QZ_KP_PERIOD 0x41 */
3485 0, /* */
3486 0x37, /* QZ_KP_MULTIPLY 0x43 */
3487 0, /* */
3488 0x4e, /* QZ_KP_PLUS 0x45 */
3489 0, /* */
3490 0x45, /* QZ_NUMLOCK 0x47 */
3491 0, /* */
3492 0, /* */
3493 0, /* */
3494 0x35|0x100, /* QZ_KP_DIVIDE 0x4B */
3495 0x1c|0x100, /* QZ_KP_ENTER 0x4C */
3496 0, /* */
3497 0x4a, /* QZ_KP_MINUS 0x4E */
3498 0, /* */
3499 0, /* */
3500 0x0d/*?*/, /* QZ_KP_EQUALS 0x51 */
3501 0x52, /* QZ_KP0 0x52 */
3502 0x4f, /* QZ_KP1 0x53 */
3503 0x50, /* QZ_KP2 0x54 */
3504 0x51, /* QZ_KP3 0x55 */
3505 0x4b, /* QZ_KP4 0x56 */
3506 0x4c, /* QZ_KP5 0x57 */
3507 0x4d, /* QZ_KP6 0x58 */
3508 0x47, /* QZ_KP7 0x59 */
3509 0, /* */
3510 0x48, /* QZ_KP8 0x5B */
3511 0x49, /* QZ_KP9 0x5C */
3512 0, /* */
3513 0, /* */
3514 0, /* */
3515 0x3f, /* QZ_F5 0x60 */
3516 0x40, /* QZ_F6 0x61 */
3517 0x41, /* QZ_F7 0x62 */
3518 0x3d, /* QZ_F3 0x63 */
3519 0x42, /* QZ_F8 0x64 */
3520 0x43, /* QZ_F9 0x65 */
3521 0, /* */
3522 0x57, /* QZ_F11 0x67 */
3523 0, /* */
3524 0x37|0x100, /* QZ_PRINT / F13 0x69 */
3525 0x63, /* QZ_F16 0x6A */
3526 0x46, /* QZ_SCROLLOCK 0x6B */
3527 0, /* */
3528 0x44, /* QZ_F10 0x6D */
3529 0x5d|0x100, /* */
3530 0x58, /* QZ_F12 0x6F */
3531 0, /* */
3532 0/* 0xe1,0x1d,0x45*/, /* QZ_PAUSE 0x71 */
3533 0x52|0x100, /* QZ_INSERT / HELP 0x72 */
3534 0x47|0x100, /* QZ_HOME 0x73 */
3535 0x49|0x100, /* QZ_PAGEUP 0x74 */
3536 0x53|0x100, /* QZ_DELETE 0x75 */
3537 0x3e, /* QZ_F4 0x76 */
3538 0x4f|0x100, /* QZ_END 0x77 */
3539 0x3c, /* QZ_F2 0x78 */
3540 0x51|0x100, /* QZ_PAGEDOWN 0x79 */
3541 0x3b, /* QZ_F1 0x7A */
3542 0x4b|0x100, /* QZ_LEFT 0x7B */
3543 0x4d|0x100, /* QZ_RIGHT 0x7C */
3544 0x50|0x100, /* QZ_DOWN 0x7D */
3545 0x48|0x100, /* QZ_UP 0x7E */
3546 0x5e|0x100, /* QZ_POWER 0x7F */ /* have different break key! */
3547 };
3548
3549 if (keycode == 0)
3550 {
3551 /* This could be a modifier or it could be 'a'. */
3552 switch (ev->keysym.sym)
3553 {
3554 case SDLK_LSHIFT: keycode = 0x2a; break;
3555 case SDLK_RSHIFT: keycode = 0x36; break;
3556 case SDLK_LCTRL: keycode = 0x1d; break;
3557 case SDLK_RCTRL: keycode = 0x1d | 0x100; break;
3558 case SDLK_LALT: keycode = 0x38; break;
3559 case SDLK_MODE: /* alt gr */
3560 case SDLK_RALT: keycode = 0x38 | 0x100; break;
3561 case SDLK_RMETA:
3562 case SDLK_RSUPER: keycode = 0x5c | 0x100; break;
3563 case SDLK_LMETA:
3564 case SDLK_LSUPER: keycode = 0x5b | 0x100; break;
3565 /* Sssumes normal key. */
3566 default: keycode = s_aMacToSet1[keycode]; break;
3567 }
3568 }
3569 else
3570 {
3571 if ((unsigned)keycode < RT_ELEMENTS(s_aMacToSet1))
3572 keycode = s_aMacToSet1[keycode];
3573 else
3574 keycode = 0;
3575 if (!keycode)
3576 {
3577#ifdef DEBUG_bird
3578 RTPrintf("Untranslated: keycode=%#x (%d)\n", keycode, keycode);
3579#endif
3580 keycode = Keyevent2KeycodeFallback(ev);
3581 }
3582 }
3583#ifdef DEBUG_bird
3584 RTPrintf("scancode=%#x -> %#x\n", ev->keysym.scancode, keycode);
3585#endif
3586
3587#elif RT_OS_OS2
3588 keycode = Keyevent2KeycodeFallback(ev);
3589#endif /* RT_OS_DARWIN */
3590 return keycode;
3591}
3592
3593/**
3594 * Releases any modifier keys that are currently in pressed state.
3595 */
3596static void ResetKeys(void)
3597{
3598 int i;
3599
3600 if (!gKeyboard)
3601 return;
3602
3603 for(i = 0; i < 256; i++)
3604 {
3605 if (gaModifiersState[i])
3606 {
3607 if (i & 0x80)
3608 gKeyboard->PutScancode(0xe0);
3609 gKeyboard->PutScancode(i | 0x80);
3610 gaModifiersState[i] = 0;
3611 }
3612 }
3613}
3614
3615/**
3616 * Keyboard event handler.
3617 *
3618 * @param ev SDL keyboard event.
3619 */
3620static void ProcessKey(SDL_KeyboardEvent *ev)
3621{
3622#if (defined(DEBUG) || defined(VBOX_WITH_STATISTICS)) && !defined(VBOX_WITH_SDL13)
3623 if (gMachineDebugger && ev->type == SDL_KEYDOWN)
3624 {
3625 // first handle the debugger hotkeys
3626 uint8_t *keystate = SDL_GetKeyState(NULL);
3627#if 0
3628 // CTRL+ALT+Fn is not free on Linux hosts with Xorg ..
3629 if (keystate[SDLK_LALT] && !keystate[SDLK_LCTRL])
3630#else
3631 if (keystate[SDLK_LALT] && keystate[SDLK_LCTRL])
3632#endif
3633 {
3634 switch (ev->keysym.sym)
3635 {
3636 // pressing CTRL+ALT+F11 dumps the statistics counter
3637 case SDLK_F12:
3638 RTPrintf("ResetStats\n"); /* Visual feedback in console window */
3639 gMachineDebugger->ResetStats(NULL);
3640 break;
3641 // pressing CTRL+ALT+F12 resets all statistics counter
3642 case SDLK_F11:
3643 gMachineDebugger->DumpStats(NULL);
3644 RTPrintf("DumpStats\n"); /* Vistual feedback in console window */
3645 break;
3646 default:
3647 break;
3648 }
3649 }
3650#if 1
3651 else if (keystate[SDLK_LALT] && !keystate[SDLK_LCTRL])
3652 {
3653 switch (ev->keysym.sym)
3654 {
3655 // pressing Alt-F12 toggles the supervisor recompiler
3656 case SDLK_F12:
3657 {
3658 BOOL recompileSupervisor;
3659 gMachineDebugger->COMGETTER(RecompileSupervisor)(&recompileSupervisor);
3660 gMachineDebugger->COMSETTER(RecompileSupervisor)(!recompileSupervisor);
3661 break;
3662 }
3663 // pressing Alt-F11 toggles the user recompiler
3664 case SDLK_F11:
3665 {
3666 BOOL recompileUser;
3667 gMachineDebugger->COMGETTER(RecompileUser)(&recompileUser);
3668 gMachineDebugger->COMSETTER(RecompileUser)(!recompileUser);
3669 break;
3670 }
3671 // pressing Alt-F10 toggles the patch manager
3672 case SDLK_F10:
3673 {
3674 BOOL patmEnabled;
3675 gMachineDebugger->COMGETTER(PATMEnabled)(&patmEnabled);
3676 gMachineDebugger->COMSETTER(PATMEnabled)(!patmEnabled);
3677 break;
3678 }
3679 // pressing Alt-F9 toggles CSAM
3680 case SDLK_F9:
3681 {
3682 BOOL csamEnabled;
3683 gMachineDebugger->COMGETTER(CSAMEnabled)(&csamEnabled);
3684 gMachineDebugger->COMSETTER(CSAMEnabled)(!csamEnabled);
3685 break;
3686 }
3687 // pressing Alt-F8 toggles singlestepping mode
3688 case SDLK_F8:
3689 {
3690 BOOL singlestepEnabled;
3691 gMachineDebugger->COMGETTER(Singlestep)(&singlestepEnabled);
3692 gMachineDebugger->COMSETTER(Singlestep)(!singlestepEnabled);
3693 break;
3694 }
3695 default:
3696 break;
3697 }
3698 }
3699#endif
3700 // pressing Ctrl-F12 toggles the logger
3701 else if ((keystate[SDLK_RCTRL] || keystate[SDLK_LCTRL]) && ev->keysym.sym == SDLK_F12)
3702 {
3703 BOOL logEnabled = TRUE;
3704 gMachineDebugger->COMGETTER(LogEnabled)(&logEnabled);
3705 gMachineDebugger->COMSETTER(LogEnabled)(!logEnabled);
3706#ifdef DEBUG_bird
3707 return;
3708#endif
3709 }
3710 // pressing F12 sets a logmark
3711 else if (ev->keysym.sym == SDLK_F12)
3712 {
3713 RTLogPrintf("****** LOGGING MARK ******\n");
3714 RTLogFlush(NULL);
3715 }
3716 // now update the titlebar flags
3717 UpdateTitlebar(TITLEBAR_NORMAL);
3718 }
3719#endif // DEBUG || VBOX_WITH_STATISTICS
3720
3721 // the pause key is the weirdest, needs special handling
3722 if (ev->keysym.sym == SDLK_PAUSE)
3723 {
3724 int v = 0;
3725 if (ev->type == SDL_KEYUP)
3726 v |= 0x80;
3727 gKeyboard->PutScancode(0xe1);
3728 gKeyboard->PutScancode(0x1d | v);
3729 gKeyboard->PutScancode(0x45 | v);
3730 return;
3731 }
3732
3733 /*
3734 * Perform SDL key event to scancode conversion
3735 */
3736 int keycode = Keyevent2Keycode(ev);
3737
3738 switch(keycode)
3739 {
3740 case 0x00:
3741 {
3742 /* sent when leaving window: reset the modifiers state */
3743 ResetKeys();
3744 return;
3745 }
3746
3747 case 0x2a: /* Left Shift */
3748 case 0x36: /* Right Shift */
3749 case 0x1d: /* Left CTRL */
3750 case 0x1d|0x100: /* Right CTRL */
3751 case 0x38: /* Left ALT */
3752 case 0x38|0x100: /* Right ALT */
3753 {
3754 if (ev->type == SDL_KEYUP)
3755 gaModifiersState[keycode & ~0x100] = 0;
3756 else
3757 gaModifiersState[keycode & ~0x100] = 1;
3758 break;
3759 }
3760
3761 case 0x45: /* Num Lock */
3762 case 0x3a: /* Caps Lock */
3763 {
3764 /*
3765 * SDL generates a KEYDOWN event if the lock key is active and a KEYUP event
3766 * if the lock key is inactive. See SDL_DISABLE_LOCK_KEYS.
3767 */
3768 if (ev->type == SDL_KEYDOWN || ev->type == SDL_KEYUP)
3769 {
3770 gKeyboard->PutScancode(keycode);
3771 gKeyboard->PutScancode(keycode | 0x80);
3772 }
3773 return;
3774 }
3775 }
3776
3777 if (ev->type != SDL_KEYDOWN)
3778 {
3779 /*
3780 * Some keyboards (e.g. the one of mine T60) don't send a NumLock scan code on every
3781 * press of the key. Both the guest and the host should agree on the NumLock state.
3782 * If they differ, we try to alter the guest NumLock state by sending the NumLock key
3783 * scancode. We will get a feedback through the KBD_CMD_SET_LEDS command if the guest
3784 * tries to set/clear the NumLock LED. If a (silly) guest doesn't change the LED, don't
3785 * bother him with NumLock scancodes. At least our BIOS, Linux and Windows handle the
3786 * NumLock LED well.
3787 */
3788 if ( gcGuestNumLockAdaptions
3789 && (gfGuestNumLockPressed ^ !!(SDL_GetModState() & KMOD_NUM)))
3790 {
3791 gcGuestNumLockAdaptions--;
3792 gKeyboard->PutScancode(0x45);
3793 gKeyboard->PutScancode(0x45 | 0x80);
3794 }
3795 if ( gcGuestCapsLockAdaptions
3796 && (gfGuestCapsLockPressed ^ !!(SDL_GetModState() & KMOD_CAPS)))
3797 {
3798 gcGuestCapsLockAdaptions--;
3799 gKeyboard->PutScancode(0x3a);
3800 gKeyboard->PutScancode(0x3a | 0x80);
3801 }
3802 }
3803
3804 /*
3805 * Now we send the event. Apply extended and release prefixes.
3806 */
3807 if (keycode & 0x100)
3808 gKeyboard->PutScancode(0xe0);
3809
3810 gKeyboard->PutScancode(ev->type == SDL_KEYUP ? (keycode & 0x7f) | 0x80
3811 : (keycode & 0x7f));
3812}
3813
3814#ifdef RT_OS_DARWIN
3815#include <Carbon/Carbon.h>
3816RT_C_DECLS_BEGIN
3817/* Private interface in 10.3 and later. */
3818typedef int CGSConnection;
3819typedef enum
3820{
3821 kCGSGlobalHotKeyEnable = 0,
3822 kCGSGlobalHotKeyDisable,
3823 kCGSGlobalHotKeyInvalid = -1 /* bird */
3824} CGSGlobalHotKeyOperatingMode;
3825extern CGSConnection _CGSDefaultConnection(void);
3826extern CGError CGSGetGlobalHotKeyOperatingMode(CGSConnection Connection, CGSGlobalHotKeyOperatingMode *enmMode);
3827extern CGError CGSSetGlobalHotKeyOperatingMode(CGSConnection Connection, CGSGlobalHotKeyOperatingMode enmMode);
3828RT_C_DECLS_END
3829
3830/** Keeping track of whether we disabled the hotkeys or not. */
3831static bool g_fHotKeysDisabled = false;
3832/** Whether we've connected or not. */
3833static bool g_fConnectedToCGS = false;
3834/** Cached connection. */
3835static CGSConnection g_CGSConnection;
3836
3837/**
3838 * Disables or enabled global hot keys.
3839 */
3840static void DisableGlobalHotKeys(bool fDisable)
3841{
3842 if (!g_fConnectedToCGS)
3843 {
3844 g_CGSConnection = _CGSDefaultConnection();
3845 g_fConnectedToCGS = true;
3846 }
3847
3848 /* get current mode. */
3849 CGSGlobalHotKeyOperatingMode enmMode = kCGSGlobalHotKeyInvalid;
3850 CGSGetGlobalHotKeyOperatingMode(g_CGSConnection, &enmMode);
3851
3852 /* calc new mode. */
3853 if (fDisable)
3854 {
3855 if (enmMode != kCGSGlobalHotKeyEnable)
3856 return;
3857 enmMode = kCGSGlobalHotKeyDisable;
3858 }
3859 else
3860 {
3861 if ( enmMode != kCGSGlobalHotKeyDisable
3862 /*|| !g_fHotKeysDisabled*/)
3863 return;
3864 enmMode = kCGSGlobalHotKeyEnable;
3865 }
3866
3867 /* try set it and check the actual result. */
3868 CGSSetGlobalHotKeyOperatingMode(g_CGSConnection, enmMode);
3869 CGSGlobalHotKeyOperatingMode enmNewMode = kCGSGlobalHotKeyInvalid;
3870 CGSGetGlobalHotKeyOperatingMode(g_CGSConnection, &enmNewMode);
3871 if (enmNewMode == enmMode)
3872 g_fHotKeysDisabled = enmMode == kCGSGlobalHotKeyDisable;
3873}
3874#endif /* RT_OS_DARWIN */
3875
3876/**
3877 * Start grabbing the mouse.
3878 */
3879static void InputGrabStart(void)
3880{
3881#ifdef RT_OS_DARWIN
3882 DisableGlobalHotKeys(true);
3883#endif
3884 if (!gfGuestNeedsHostCursor && gfRelativeMouseGuest)
3885 SDL_ShowCursor(SDL_DISABLE);
3886 SDL_WM_GrabInput(SDL_GRAB_ON);
3887 // dummy read to avoid moving the mouse
3888 SDL_GetRelativeMouseState(
3889#ifdef VBOX_WITH_SDL13
3890 0,
3891#endif
3892 NULL, NULL);
3893 gfGrabbed = TRUE;
3894 UpdateTitlebar(TITLEBAR_NORMAL);
3895}
3896
3897/**
3898 * End mouse grabbing.
3899 */
3900static void InputGrabEnd(void)
3901{
3902 SDL_WM_GrabInput(SDL_GRAB_OFF);
3903 if (!gfGuestNeedsHostCursor && gfRelativeMouseGuest)
3904 SDL_ShowCursor(SDL_ENABLE);
3905#ifdef RT_OS_DARWIN
3906 DisableGlobalHotKeys(false);
3907#endif
3908 gfGrabbed = FALSE;
3909 UpdateTitlebar(TITLEBAR_NORMAL);
3910}
3911
3912/**
3913 * Query mouse position and button state from SDL and send to the VM
3914 *
3915 * @param dz Relative mouse wheel movement
3916 */
3917static void SendMouseEvent(VBoxSDLFB *fb, int dz, int down, int button)
3918{
3919 int x, y, state, buttons;
3920 bool abs;
3921
3922#ifdef VBOX_WITH_SDL13
3923 if (!fb)
3924 {
3925 SDL_GetMouseState(0, &x, &y);
3926 RTPrintf("MouseEvent: Cannot find fb mouse = %d,%d\n", x, y);
3927 return;
3928 }
3929#else
3930 AssertRelease(fb != NULL);
3931#endif
3932
3933 /*
3934 * If supported and we're not in grabbed mode, we'll use the absolute mouse.
3935 * If we are in grabbed mode and the guest is not able to draw the mouse cursor
3936 * itself, or can't handle relative reporting, we have to use absolute
3937 * coordinates, otherwise the host cursor and
3938 * the coordinates the guest thinks the mouse is at could get out-of-sync. From
3939 * the SDL mailing list:
3940 *
3941 * "The event processing is usually asynchronous and so somewhat delayed, and
3942 * SDL_GetMouseState is returning the immediate mouse state. So at the time you
3943 * call SDL_GetMouseState, the "button" is already up."
3944 */
3945 abs = (UseAbsoluteMouse() && !gfGrabbed)
3946 || gfGuestNeedsHostCursor
3947 || !gfRelativeMouseGuest;
3948
3949 /* only used if abs == TRUE */
3950 int xOrigin = fb->getOriginX();
3951 int yOrigin = fb->getOriginY();
3952 int xMin = fb->getXOffset() + xOrigin;
3953 int yMin = fb->getYOffset() + yOrigin;
3954 int xMax = xMin + (int)fb->getGuestXRes();
3955 int yMax = yMin + (int)fb->getGuestYRes();
3956
3957 state = abs ? SDL_GetMouseState(
3958#ifdef VBOX_WITH_SDL13
3959 0,
3960#endif
3961 &x, &y)
3962 : SDL_GetRelativeMouseState(
3963#ifdef VBOX_WITH_SDL13
3964 0,
3965#endif
3966 &x, &y);
3967
3968 /*
3969 * process buttons
3970 */
3971 buttons = 0;
3972 if (state & SDL_BUTTON(SDL_BUTTON_LEFT))
3973 buttons |= MouseButtonState_LeftButton;
3974 if (state & SDL_BUTTON(SDL_BUTTON_RIGHT))
3975 buttons |= MouseButtonState_RightButton;
3976 if (state & SDL_BUTTON(SDL_BUTTON_MIDDLE))
3977 buttons |= MouseButtonState_MiddleButton;
3978
3979 if (abs)
3980 {
3981 x += xOrigin;
3982 y += yOrigin;
3983
3984 /*
3985 * Check if the mouse event is inside the guest area. This solves the
3986 * following problem: Some guests switch off the VBox hardware mouse
3987 * cursor and draw the mouse cursor itself instead. Moving the mouse
3988 * outside the guest area then leads to annoying mouse hangs if we
3989 * don't pass mouse motion events into the guest.
3990 */
3991 if (x < xMin || y < yMin || x > xMax || y > yMax)
3992 {
3993 /*
3994 * Cursor outside of valid guest area (outside window or in secure
3995 * label area. Don't allow any mouse button press.
3996 */
3997 button = 0;
3998
3999 /*
4000 * Release any pressed button.
4001 */
4002#if 0
4003 /* disabled on customers request */
4004 buttons &= ~(MouseButtonState_LeftButton |
4005 MouseButtonState_MiddleButton |
4006 MouseButtonState_RightButton);
4007#endif
4008
4009 /*
4010 * Prevent negative coordinates.
4011 */
4012 if (x < xMin) x = xMin;
4013 if (x > xMax) x = xMax;
4014 if (y < yMin) y = yMin;
4015 if (y > yMax) y = yMax;
4016
4017 if (!gpOffCursor)
4018 {
4019 gpOffCursor = SDL_GetCursor(); /* Cursor image */
4020 gfOffCursorActive = SDL_ShowCursor(-1); /* enabled / disabled */
4021 SDL_SetCursor(gpDefaultCursor);
4022 SDL_ShowCursor (SDL_ENABLE);
4023 }
4024 }
4025 else
4026 {
4027 if (gpOffCursor)
4028 {
4029 /*
4030 * We just entered the valid guest area. Restore the guest mouse
4031 * cursor.
4032 */
4033 SDL_SetCursor(gpOffCursor);
4034 SDL_ShowCursor(gfOffCursorActive ? SDL_ENABLE : SDL_DISABLE);
4035 gpOffCursor = NULL;
4036 }
4037 }
4038 }
4039
4040 /*
4041 * Button was pressed but that press is not reflected in the button state?
4042 */
4043 if (down && !(state & SDL_BUTTON(button)))
4044 {
4045 /*
4046 * It can happen that a mouse up event follows a mouse down event immediately
4047 * and we see the events when the bit in the button state is already cleared
4048 * again. In that case we simulate the mouse down event.
4049 */
4050 int tmp_button = 0;
4051 switch (button)
4052 {
4053 case SDL_BUTTON_LEFT: tmp_button = MouseButtonState_LeftButton; break;
4054 case SDL_BUTTON_MIDDLE: tmp_button = MouseButtonState_MiddleButton; break;
4055 case SDL_BUTTON_RIGHT: tmp_button = MouseButtonState_RightButton; break;
4056 }
4057
4058 if (abs)
4059 {
4060 /**
4061 * @todo
4062 * PutMouseEventAbsolute() expects x and y starting from 1,1.
4063 * should we do the increment internally in PutMouseEventAbsolute()
4064 * or state it in PutMouseEventAbsolute() docs?
4065 */
4066 gMouse->PutMouseEventAbsolute(x + 1 - xMin + xOrigin,
4067 y + 1 - yMin + yOrigin,
4068 dz, 0 /* horizontal scroll wheel */,
4069 buttons | tmp_button);
4070 }
4071 else
4072 {
4073 gMouse->PutMouseEvent(0, 0, dz,
4074 0 /* horizontal scroll wheel */,
4075 buttons | tmp_button);
4076 }
4077 }
4078
4079 // now send the mouse event
4080 if (abs)
4081 {
4082 /**
4083 * @todo
4084 * PutMouseEventAbsolute() expects x and y starting from 1,1.
4085 * should we do the increment internally in PutMouseEventAbsolute()
4086 * or state it in PutMouseEventAbsolute() docs?
4087 */
4088 gMouse->PutMouseEventAbsolute(x + 1 - xMin + xOrigin,
4089 y + 1 - yMin + yOrigin,
4090 dz, 0 /* Horizontal wheel */, buttons);
4091 }
4092 else
4093 {
4094 gMouse->PutMouseEvent(x, y, dz, 0 /* Horizontal wheel */, buttons);
4095 }
4096}
4097
4098/**
4099 * Resets the VM
4100 */
4101void ResetVM(void)
4102{
4103 if (gConsole)
4104 gConsole->Reset();
4105}
4106
4107/**
4108 * Initiates a saved state and updates the titlebar with progress information
4109 */
4110void SaveState(void)
4111{
4112 ResetKeys();
4113 RTThreadYield();
4114 if (gfGrabbed)
4115 InputGrabEnd();
4116 RTThreadYield();
4117 UpdateTitlebar(TITLEBAR_SAVE);
4118 gProgress = NULL;
4119 HRESULT rc = gConsole->SaveState(gProgress.asOutParam());
4120 if (FAILED(S_OK))
4121 {
4122 RTPrintf("Error saving state! rc = 0x%x\n", rc);
4123 return;
4124 }
4125 Assert(gProgress);
4126
4127 /*
4128 * Wait for the operation to be completed and work
4129 * the title bar in the mean while.
4130 */
4131 ULONG cPercent = 0;
4132#ifndef RT_OS_DARWIN /* don't break the other guys yet. */
4133 for (;;)
4134 {
4135 BOOL fCompleted = false;
4136 rc = gProgress->COMGETTER(Completed)(&fCompleted);
4137 if (FAILED(rc) || fCompleted)
4138 break;
4139 ULONG cPercentNow;
4140 rc = gProgress->COMGETTER(Percent)(&cPercentNow);
4141 if (FAILED(rc))
4142 break;
4143 if (cPercentNow != cPercent)
4144 {
4145 UpdateTitlebar(TITLEBAR_SAVE, cPercent);
4146 cPercent = cPercentNow;
4147 }
4148
4149 /* wait */
4150 rc = gProgress->WaitForCompletion(100);
4151 if (FAILED(rc))
4152 break;
4153 /// @todo process gui events.
4154 }
4155
4156#else /* new loop which processes GUI events while saving. */
4157
4158 /* start regular timer so we don't starve in the event loop */
4159 SDL_TimerID sdlTimer;
4160 sdlTimer = SDL_AddTimer(100, StartupTimer, NULL);
4161
4162 for (;;)
4163 {
4164 /*
4165 * Check for completion.
4166 */
4167 BOOL fCompleted = false;
4168 rc = gProgress->COMGETTER(Completed)(&fCompleted);
4169 if (FAILED(rc) || fCompleted)
4170 break;
4171 ULONG cPercentNow;
4172 rc = gProgress->COMGETTER(Percent)(&cPercentNow);
4173 if (FAILED(rc))
4174 break;
4175 if (cPercentNow != cPercent)
4176 {
4177 UpdateTitlebar(TITLEBAR_SAVE, cPercent);
4178 cPercent = cPercentNow;
4179 }
4180
4181 /*
4182 * Wait for and process GUI a event.
4183 * This is necessary for XPCOM IPC and for updating the
4184 * title bar on the Mac.
4185 */
4186 SDL_Event event;
4187 if (WaitSDLEvent(&event))
4188 {
4189 switch (event.type)
4190 {
4191 /*
4192 * Timer event preventing us from getting stuck.
4193 */
4194 case SDL_USER_EVENT_TIMER:
4195 break;
4196
4197#ifdef USE_XPCOM_QUEUE_THREAD
4198 /*
4199 * User specific XPCOM event queue event
4200 */
4201 case SDL_USER_EVENT_XPCOM_EVENTQUEUE:
4202 {
4203 LogFlow(("SDL_USER_EVENT_XPCOM_EVENTQUEUE: processing XPCOM event queue...\n"));
4204 eventQ->ProcessPendingEvents();
4205 signalXPCOMEventQueueThread();
4206 break;
4207 }
4208#endif /* USE_XPCOM_QUEUE_THREAD */
4209
4210
4211 /*
4212 * Ignore all other events.
4213 */
4214 case SDL_USER_EVENT_RESIZE:
4215 case SDL_USER_EVENT_TERMINATE:
4216 default:
4217 break;
4218 }
4219 }
4220 }
4221
4222 /* kill the timer */
4223 SDL_RemoveTimer(sdlTimer);
4224 sdlTimer = 0;
4225
4226#endif /* RT_OS_DARWIN */
4227
4228 /*
4229 * What's the result of the operation?
4230 */
4231 LONG lrc;
4232 rc = gProgress->COMGETTER(ResultCode)(&lrc);
4233 if (FAILED(rc))
4234 lrc = ~0;
4235 if (!lrc)
4236 {
4237 UpdateTitlebar(TITLEBAR_SAVE, 100);
4238 RTThreadYield();
4239 RTPrintf("Saved the state successfully.\n");
4240 }
4241 else
4242 RTPrintf("Error saving state, lrc=%d (%#x)\n", lrc, lrc);
4243}
4244
4245/**
4246 * Build the titlebar string
4247 */
4248static void UpdateTitlebar(TitlebarMode mode, uint32_t u32User)
4249{
4250 static char szTitle[1024] = {0};
4251
4252 /* back up current title */
4253 char szPrevTitle[1024];
4254 strcpy(szPrevTitle, szTitle);
4255
4256 Bstr name;
4257 gMachine->COMGETTER(Name)(name.asOutParam());
4258
4259 RTStrPrintf(szTitle, sizeof(szTitle), "%s - " VBOX_PRODUCT,
4260 name ? Utf8Str(name).raw() : "<noname>");
4261
4262 /* which mode are we in? */
4263 switch (mode)
4264 {
4265 case TITLEBAR_NORMAL:
4266 {
4267 MachineState_T machineState;
4268 gMachine->COMGETTER(State)(&machineState);
4269 if (machineState == MachineState_Paused)
4270 RTStrPrintf(szTitle + strlen(szTitle), sizeof(szTitle) - strlen(szTitle), " - [Paused]");
4271
4272 if (gfGrabbed)
4273 RTStrPrintf(szTitle + strlen(szTitle), sizeof(szTitle) - strlen(szTitle), " - [Input captured]");
4274
4275#if defined(DEBUG) || defined(VBOX_WITH_STATISTICS)
4276 // do we have a debugger interface
4277 if (gMachineDebugger)
4278 {
4279 // query the machine state
4280 BOOL recompileSupervisor = FALSE;
4281 BOOL recompileUser = FALSE;
4282 BOOL patmEnabled = FALSE;
4283 BOOL csamEnabled = FALSE;
4284 BOOL singlestepEnabled = FALSE;
4285 BOOL logEnabled = FALSE;
4286 BOOL hwVirtEnabled = FALSE;
4287 ULONG virtualTimeRate = 100;
4288 gMachineDebugger->COMGETTER(RecompileSupervisor)(&recompileSupervisor);
4289 gMachineDebugger->COMGETTER(RecompileUser)(&recompileUser);
4290 gMachineDebugger->COMGETTER(PATMEnabled)(&patmEnabled);
4291 gMachineDebugger->COMGETTER(CSAMEnabled)(&csamEnabled);
4292 gMachineDebugger->COMGETTER(LogEnabled)(&logEnabled);
4293 gMachineDebugger->COMGETTER(Singlestep)(&singlestepEnabled);
4294 gMachineDebugger->COMGETTER(HWVirtExEnabled)(&hwVirtEnabled);
4295 gMachineDebugger->COMGETTER(VirtualTimeRate)(&virtualTimeRate);
4296 RTStrPrintf(szTitle + strlen(szTitle), sizeof(szTitle) - strlen(szTitle),
4297 " [STEP=%d CS=%d PAT=%d RR0=%d RR3=%d LOG=%d HWVirt=%d",
4298 singlestepEnabled == TRUE, csamEnabled == TRUE, patmEnabled == TRUE,
4299 recompileSupervisor == FALSE, recompileUser == FALSE,
4300 logEnabled == TRUE, hwVirtEnabled == TRUE);
4301 char *psz = strchr(szTitle, '\0');
4302 if (virtualTimeRate != 100)
4303 RTStrPrintf(psz, &szTitle[sizeof(szTitle)] - psz, " WD=%d%%]", virtualTimeRate);
4304 else
4305 RTStrPrintf(psz, &szTitle[sizeof(szTitle)] - psz, "]");
4306 }
4307#endif /* DEBUG || VBOX_WITH_STATISTICS */
4308 break;
4309 }
4310
4311 case TITLEBAR_STARTUP:
4312 {
4313 /*
4314 * Format it.
4315 */
4316 MachineState_T machineState;
4317 gMachine->COMGETTER(State)(&machineState);
4318 if (machineState == MachineState_Starting)
4319 RTStrPrintf(szTitle + strlen(szTitle), sizeof(szTitle) - strlen(szTitle),
4320 " - Starting...");
4321 else if (machineState == MachineState_Restoring)
4322 {
4323 ULONG cPercentNow;
4324 HRESULT rc = gProgress->COMGETTER(Percent)(&cPercentNow);
4325 if (SUCCEEDED(rc))
4326 RTStrPrintf(szTitle + strlen(szTitle), sizeof(szTitle) - strlen(szTitle),
4327 " - Restoring %d%%...", (int)cPercentNow);
4328 else
4329 RTStrPrintf(szTitle + strlen(szTitle), sizeof(szTitle) - strlen(szTitle),
4330 " - Restoring...");
4331 }
4332 else if (machineState == MachineState_TeleportingIn)
4333 {
4334 ULONG cPercentNow;
4335 HRESULT rc = gProgress->COMGETTER(Percent)(&cPercentNow);
4336 if (SUCCEEDED(rc))
4337 RTStrPrintf(szTitle + strlen(szTitle), sizeof(szTitle) - strlen(szTitle),
4338 " - Teleporting %d%%...", (int)cPercentNow);
4339 else
4340 RTStrPrintf(szTitle + strlen(szTitle), sizeof(szTitle) - strlen(szTitle),
4341 " - Teleporting...");
4342 }
4343 /* ignore other states, we could already be in running or aborted state */
4344 break;
4345 }
4346
4347 case TITLEBAR_SAVE:
4348 {
4349 AssertMsg(u32User <= 100, ("%d\n", u32User));
4350 RTStrPrintf(szTitle + strlen(szTitle), sizeof(szTitle) - strlen(szTitle),
4351 " - Saving %d%%...", u32User);
4352 break;
4353 }
4354
4355 case TITLEBAR_SNAPSHOT:
4356 {
4357 AssertMsg(u32User <= 100, ("%d\n", u32User));
4358 RTStrPrintf(szTitle + strlen(szTitle), sizeof(szTitle) - strlen(szTitle),
4359 " - Taking snapshot %d%%...", u32User);
4360 break;
4361 }
4362
4363 default:
4364 RTPrintf("Error: Invalid title bar mode %d!\n", mode);
4365 return;
4366 }
4367
4368 /*
4369 * Don't update if it didn't change.
4370 */
4371 if (!strcmp(szTitle, szPrevTitle))
4372 return;
4373
4374 /*
4375 * Set the new title
4376 */
4377#ifdef VBOX_WIN32_UI
4378 setUITitle(szTitle);
4379#else
4380 SDL_WM_SetCaption(szTitle, VBOX_PRODUCT);
4381#endif
4382}
4383
4384#if 0
4385static void vbox_show_shape (unsigned short w, unsigned short h,
4386 uint32_t bg, const uint8_t *image)
4387{
4388 size_t x, y;
4389 unsigned short pitch;
4390 const uint32_t *color;
4391 const uint8_t *mask;
4392 size_t size_mask;
4393
4394 mask = image;
4395 pitch = (w + 7) / 8;
4396 size_mask = (pitch * h + 3) & ~3;
4397
4398 color = (const uint32_t *) (image + size_mask);
4399
4400 printf ("show_shape %dx%d pitch %d size mask %d\n",
4401 w, h, pitch, size_mask);
4402 for (y = 0; y < h; ++y, mask += pitch, color += w)
4403 {
4404 for (x = 0; x < w; ++x) {
4405 if (mask[x / 8] & (1 << (7 - (x % 8))))
4406 printf (" ");
4407 else
4408 {
4409 uint32_t c = color[x];
4410 if (c == bg)
4411 printf ("Y");
4412 else
4413 printf ("X");
4414 }
4415 }
4416 printf ("\n");
4417 }
4418}
4419#endif
4420
4421/**
4422 * Sets the pointer shape according to parameters.
4423 * Must be called only from the main SDL thread.
4424 */
4425static void SetPointerShape (const PointerShapeChangeData *data)
4426{
4427 /*
4428 * don't allow to change the pointer shape if we are outside the valid
4429 * guest area. In that case set standard mouse pointer is set and should
4430 * not get overridden.
4431 */
4432 if (gpOffCursor)
4433 return;
4434
4435 if (data->shape)
4436 {
4437 bool ok = false;
4438
4439 uint32_t andMaskSize = (data->width + 7) / 8 * data->height;
4440 uint32_t srcShapePtrScan = data->width * 4;
4441
4442 const uint8_t *srcAndMaskPtr = data->shape;
4443 const uint8_t *srcShapePtr = data->shape + ((andMaskSize + 3) & ~3);
4444
4445#if 0
4446 /* pointer debugging code */
4447 // vbox_show_shape(data->width, data->height, 0, data->shape);
4448 uint32_t shapeSize = ((((data->width + 7) / 8) * data->height + 3) & ~3) + data->width * 4 * data->height;
4449 printf("visible: %d\n", data->visible);
4450 printf("width = %d\n", data->width);
4451 printf("height = %d\n", data->height);
4452 printf("alpha = %d\n", data->alpha);
4453 printf("xhot = %d\n", data->xHot);
4454 printf("yhot = %d\n", data->yHot);
4455 printf("uint8_t pointerdata[] = { ");
4456 for (uint32_t i = 0; i < shapeSize; i++)
4457 {
4458 printf("0x%x, ", data->shape[i]);
4459 }
4460 printf("};\n");
4461#endif
4462
4463#if defined (RT_OS_WINDOWS)
4464
4465 BITMAPV5HEADER bi;
4466 HBITMAP hBitmap;
4467 void *lpBits;
4468 HCURSOR hAlphaCursor = NULL;
4469
4470 ::ZeroMemory (&bi, sizeof (BITMAPV5HEADER));
4471 bi.bV5Size = sizeof (BITMAPV5HEADER);
4472 bi.bV5Width = data->width;
4473 bi.bV5Height = - (LONG) data->height;
4474 bi.bV5Planes = 1;
4475 bi.bV5BitCount = 32;
4476 bi.bV5Compression = BI_BITFIELDS;
4477 // specifiy a supported 32 BPP alpha format for Windows XP
4478 bi.bV5RedMask = 0x00FF0000;
4479 bi.bV5GreenMask = 0x0000FF00;
4480 bi.bV5BlueMask = 0x000000FF;
4481 if (data->alpha)
4482 bi.bV5AlphaMask = 0xFF000000;
4483 else
4484 bi.bV5AlphaMask = 0;
4485
4486 HDC hdc = ::GetDC (NULL);
4487
4488 // create the DIB section with an alpha channel
4489 hBitmap = ::CreateDIBSection (hdc, (BITMAPINFO *) &bi, DIB_RGB_COLORS,
4490 (void **) &lpBits, NULL, (DWORD) 0);
4491
4492 ::ReleaseDC (NULL, hdc);
4493
4494 HBITMAP hMonoBitmap = NULL;
4495 if (data->alpha)
4496 {
4497 // create an empty mask bitmap
4498 hMonoBitmap = ::CreateBitmap (data->width, data->height, 1, 1, NULL);
4499 }
4500 else
4501 {
4502 /* Word aligned AND mask. Will be allocated and created if necessary. */
4503 uint8_t *pu8AndMaskWordAligned = NULL;
4504
4505 /* Width in bytes of the original AND mask scan line. */
4506 uint32_t cbAndMaskScan = (data->width + 7) / 8;
4507
4508 if (cbAndMaskScan & 1)
4509 {
4510 /* Original AND mask is not word aligned. */
4511
4512 /* Allocate memory for aligned AND mask. */
4513 pu8AndMaskWordAligned = (uint8_t *)RTMemTmpAllocZ ((cbAndMaskScan + 1) * data->height);
4514
4515 Assert(pu8AndMaskWordAligned);
4516
4517 if (pu8AndMaskWordAligned)
4518 {
4519 /* According to MSDN the padding bits must be 0.
4520 * Compute the bit mask to set padding bits to 0 in the last byte of original AND mask.
4521 */
4522 uint32_t u32PaddingBits = cbAndMaskScan * 8 - data->width;
4523 Assert(u32PaddingBits < 8);
4524 uint8_t u8LastBytesPaddingMask = (uint8_t)(0xFF << u32PaddingBits);
4525
4526 Log(("u8LastBytesPaddingMask = %02X, aligned w = %d, width = %d, cbAndMaskScan = %d\n",
4527 u8LastBytesPaddingMask, (cbAndMaskScan + 1) * 8, data->width, cbAndMaskScan));
4528
4529 uint8_t *src = (uint8_t *)srcAndMaskPtr;
4530 uint8_t *dst = pu8AndMaskWordAligned;
4531
4532 unsigned i;
4533 for (i = 0; i < data->height; i++)
4534 {
4535 memcpy (dst, src, cbAndMaskScan);
4536
4537 dst[cbAndMaskScan - 1] &= u8LastBytesPaddingMask;
4538
4539 src += cbAndMaskScan;
4540 dst += cbAndMaskScan + 1;
4541 }
4542 }
4543 }
4544
4545 // create the AND mask bitmap
4546 hMonoBitmap = ::CreateBitmap (data->width, data->height, 1, 1,
4547 pu8AndMaskWordAligned? pu8AndMaskWordAligned: srcAndMaskPtr);
4548
4549 if (pu8AndMaskWordAligned)
4550 {
4551 RTMemTmpFree (pu8AndMaskWordAligned);
4552 }
4553 }
4554
4555 Assert (hBitmap);
4556 Assert (hMonoBitmap);
4557 if (hBitmap && hMonoBitmap)
4558 {
4559 DWORD *dstShapePtr = (DWORD *) lpBits;
4560
4561 for (uint32_t y = 0; y < data->height; y ++)
4562 {
4563 memcpy (dstShapePtr, srcShapePtr, srcShapePtrScan);
4564 srcShapePtr += srcShapePtrScan;
4565 dstShapePtr += data->width;
4566 }
4567
4568 ICONINFO ii;
4569 ii.fIcon = FALSE;
4570 ii.xHotspot = data->xHot;
4571 ii.yHotspot = data->yHot;
4572 ii.hbmMask = hMonoBitmap;
4573 ii.hbmColor = hBitmap;
4574
4575 hAlphaCursor = ::CreateIconIndirect (&ii);
4576 Assert (hAlphaCursor);
4577 if (hAlphaCursor)
4578 {
4579 // here we do a dirty trick by substituting a Window Manager's
4580 // cursor handle with the handle we created
4581
4582 WMcursor *pCustomTempWMCursor = gpCustomCursor->wm_cursor;
4583
4584 // see SDL12/src/video/wincommon/SDL_sysmouse.c
4585 void *wm_cursor = malloc (sizeof (HCURSOR) + sizeof (uint8_t *) * 2);
4586 *(HCURSOR *) wm_cursor = hAlphaCursor;
4587
4588 gpCustomCursor->wm_cursor = (WMcursor *) wm_cursor;
4589 SDL_SetCursor (gpCustomCursor);
4590 SDL_ShowCursor (SDL_ENABLE);
4591
4592 if (pCustomTempWMCursor)
4593 {
4594 ::DestroyCursor (* (HCURSOR *) pCustomTempWMCursor);
4595 free (pCustomTempWMCursor);
4596 }
4597
4598 ok = true;
4599 }
4600 }
4601
4602 if (hMonoBitmap)
4603 ::DeleteObject (hMonoBitmap);
4604 if (hBitmap)
4605 ::DeleteObject (hBitmap);
4606
4607#elif defined (VBOXSDL_WITH_X11) && !defined (VBOX_WITHOUT_XCURSOR)
4608
4609 if (gfXCursorEnabled)
4610 {
4611 XcursorImage *img = XcursorImageCreate (data->width, data->height);
4612 Assert (img);
4613 if (img)
4614 {
4615 img->xhot = data->xHot;
4616 img->yhot = data->yHot;
4617
4618 XcursorPixel *dstShapePtr = img->pixels;
4619
4620 for (uint32_t y = 0; y < data->height; y ++)
4621 {
4622 memcpy (dstShapePtr, srcShapePtr, srcShapePtrScan);
4623
4624 if (!data->alpha)
4625 {
4626 // convert AND mask to the alpha channel
4627 uint8_t byte = 0;
4628 for (uint32_t x = 0; x < data->width; x ++)
4629 {
4630 if (!(x % 8))
4631 byte = *(srcAndMaskPtr ++);
4632 else
4633 byte <<= 1;
4634
4635 if (byte & 0x80)
4636 {
4637 // Linux doesn't support inverted pixels (XOR ops,
4638 // to be exact) in cursor shapes, so we detect such
4639 // pixels and always replace them with black ones to
4640 // make them visible at least over light colors
4641 if (dstShapePtr [x] & 0x00FFFFFF)
4642 dstShapePtr [x] = 0xFF000000;
4643 else
4644 dstShapePtr [x] = 0x00000000;
4645 }
4646 else
4647 dstShapePtr [x] |= 0xFF000000;
4648 }
4649 }
4650
4651 srcShapePtr += srcShapePtrScan;
4652 dstShapePtr += data->width;
4653 }
4654
4655#ifndef VBOX_WITH_SDL13
4656 Cursor cur = XcursorImageLoadCursor (gSdlInfo.info.x11.display, img);
4657 Assert (cur);
4658 if (cur)
4659 {
4660 // here we do a dirty trick by substituting a Window Manager's
4661 // cursor handle with the handle we created
4662
4663 WMcursor *pCustomTempWMCursor = gpCustomCursor->wm_cursor;
4664
4665 // see SDL12/src/video/x11/SDL_x11mouse.c
4666 void *wm_cursor = malloc (sizeof (Cursor));
4667 *(Cursor *) wm_cursor = cur;
4668
4669 gpCustomCursor->wm_cursor = (WMcursor *) wm_cursor;
4670 SDL_SetCursor (gpCustomCursor);
4671 SDL_ShowCursor (SDL_ENABLE);
4672
4673 if (pCustomTempWMCursor)
4674 {
4675 XFreeCursor (gSdlInfo.info.x11.display, *(Cursor *) pCustomTempWMCursor);
4676 free (pCustomTempWMCursor);
4677 }
4678
4679 ok = true;
4680 }
4681#endif
4682 }
4683 XcursorImageDestroy (img);
4684 }
4685
4686#endif /* VBOXSDL_WITH_X11 && !VBOX_WITHOUT_XCURSOR */
4687
4688 if (!ok)
4689 {
4690 SDL_SetCursor (gpDefaultCursor);
4691 SDL_ShowCursor (SDL_ENABLE);
4692 }
4693 }
4694 else
4695 {
4696 if (data->visible)
4697 SDL_ShowCursor (SDL_ENABLE);
4698 else if (gfAbsoluteMouseGuest)
4699 /* Don't disable the cursor if the guest additions are not active (anymore) */
4700 SDL_ShowCursor (SDL_DISABLE);
4701 }
4702}
4703
4704/**
4705 * Handle changed mouse capabilities
4706 */
4707static void HandleGuestCapsChanged(void)
4708{
4709 if (!gfAbsoluteMouseGuest)
4710 {
4711 // Cursor could be overwritten by the guest tools
4712 SDL_SetCursor(gpDefaultCursor);
4713 SDL_ShowCursor (SDL_ENABLE);
4714 gpOffCursor = NULL;
4715 }
4716 if (gMouse && UseAbsoluteMouse())
4717 {
4718 // Actually switch to absolute coordinates
4719 if (gfGrabbed)
4720 InputGrabEnd();
4721 gMouse->PutMouseEventAbsolute(-1, -1, 0, 0, 0);
4722 }
4723}
4724
4725/**
4726 * Handles a host key down event
4727 */
4728static int HandleHostKey(const SDL_KeyboardEvent *pEv)
4729{
4730 /*
4731 * Revalidate the host key modifier
4732 */
4733 if ((SDL_GetModState() & ~(KMOD_MODE | KMOD_NUM | KMOD_RESERVED)) != gHostKeyMod)
4734 return VERR_NOT_SUPPORTED;
4735
4736 /*
4737 * What was pressed?
4738 */
4739 switch (pEv->keysym.sym)
4740 {
4741 /* Control-Alt-Delete */
4742 case SDLK_DELETE:
4743 {
4744 gKeyboard->PutCAD();
4745 break;
4746 }
4747
4748 /*
4749 * Fullscreen / Windowed toggle.
4750 */
4751 case SDLK_f:
4752 {
4753 if ( strchr(gHostKeyDisabledCombinations, 'f')
4754 || !gfAllowFullscreenToggle)
4755 return VERR_NOT_SUPPORTED;
4756
4757 /*
4758 * We have to pause/resume the machine during this
4759 * process because there might be a short moment
4760 * without a valid framebuffer
4761 */
4762 MachineState_T machineState;
4763 gMachine->COMGETTER(State)(&machineState);
4764 bool fPauseIt = machineState == MachineState_Running
4765 || machineState == MachineState_Teleporting
4766 || machineState == MachineState_LiveSnapshotting;
4767 if (fPauseIt)
4768 gConsole->Pause();
4769 SetFullscreen(!gpFramebuffer[0]->getFullscreen());
4770 if (fPauseIt)
4771 gConsole->Resume();
4772
4773 /*
4774 * We have switched from/to fullscreen, so request a full
4775 * screen repaint, just to be sure.
4776 */
4777 gDisplay->InvalidateAndUpdate();
4778 break;
4779 }
4780
4781 /*
4782 * Pause / Resume toggle.
4783 */
4784 case SDLK_p:
4785 {
4786 if (strchr(gHostKeyDisabledCombinations, 'p'))
4787 return VERR_NOT_SUPPORTED;
4788
4789 MachineState_T machineState;
4790 gMachine->COMGETTER(State)(&machineState);
4791 if ( machineState == MachineState_Running
4792 || machineState == MachineState_Teleporting
4793 || machineState == MachineState_LiveSnapshotting
4794 )
4795 {
4796 if (gfGrabbed)
4797 InputGrabEnd();
4798 gConsole->Pause();
4799 }
4800 else if (machineState == MachineState_Paused)
4801 {
4802 gConsole->Resume();
4803 }
4804 UpdateTitlebar(TITLEBAR_NORMAL);
4805 break;
4806 }
4807
4808 /*
4809 * Reset the VM
4810 */
4811 case SDLK_r:
4812 {
4813 if (strchr(gHostKeyDisabledCombinations, 'r'))
4814 return VERR_NOT_SUPPORTED;
4815
4816 ResetVM();
4817 break;
4818 }
4819
4820 /*
4821 * Terminate the VM
4822 */
4823 case SDLK_q:
4824 {
4825 if (strchr(gHostKeyDisabledCombinations, 'q'))
4826 return VERR_NOT_SUPPORTED;
4827
4828 return VINF_EM_TERMINATE;
4829 }
4830
4831 /*
4832 * Save the machine's state and exit
4833 */
4834 case SDLK_s:
4835 {
4836 if (strchr(gHostKeyDisabledCombinations, 's'))
4837 return VERR_NOT_SUPPORTED;
4838
4839 SaveState();
4840 return VINF_EM_TERMINATE;
4841 }
4842
4843 case SDLK_h:
4844 {
4845 if (strchr(gHostKeyDisabledCombinations, 'h'))
4846 return VERR_NOT_SUPPORTED;
4847
4848 if (gConsole)
4849 gConsole->PowerButton();
4850 break;
4851 }
4852
4853 /*
4854 * Perform an online snapshot. Continue operation.
4855 */
4856 case SDLK_n:
4857 {
4858 if (strchr(gHostKeyDisabledCombinations, 'n'))
4859 return VERR_NOT_SUPPORTED;
4860
4861 RTThreadYield();
4862 ULONG cSnapshots = 0;
4863 gMachine->COMGETTER(SnapshotCount)(&cSnapshots);
4864 char pszSnapshotName[20];
4865 RTStrPrintf(pszSnapshotName, sizeof(pszSnapshotName), "Snapshot %d", cSnapshots + 1);
4866 gProgress = NULL;
4867 HRESULT rc;
4868 CHECK_ERROR(gConsole, TakeSnapshot(Bstr(pszSnapshotName), Bstr("Taken by VBoxSDL"),
4869 gProgress.asOutParam()));
4870 if (FAILED(rc))
4871 {
4872 RTPrintf("Error taking snapshot! rc = 0x%x\n", rc);
4873 /* continue operation */
4874 return VINF_SUCCESS;
4875 }
4876 /*
4877 * Wait for the operation to be completed and work
4878 * the title bar in the mean while.
4879 */
4880 ULONG cPercent = 0;
4881 for (;;)
4882 {
4883 BOOL fCompleted = false;
4884 rc = gProgress->COMGETTER(Completed)(&fCompleted);
4885 if (FAILED(rc) || fCompleted)
4886 break;
4887 ULONG cPercentNow;
4888 rc = gProgress->COMGETTER(Percent)(&cPercentNow);
4889 if (FAILED(rc))
4890 break;
4891 if (cPercentNow != cPercent)
4892 {
4893 UpdateTitlebar(TITLEBAR_SNAPSHOT, cPercent);
4894 cPercent = cPercentNow;
4895 }
4896
4897 /* wait */
4898 rc = gProgress->WaitForCompletion(100);
4899 if (FAILED(rc))
4900 break;
4901 /// @todo process gui events.
4902 }
4903
4904 /* continue operation */
4905 return VINF_SUCCESS;
4906 }
4907
4908 case SDLK_F1: case SDLK_F2: case SDLK_F3:
4909 case SDLK_F4: case SDLK_F5: case SDLK_F6:
4910 case SDLK_F7: case SDLK_F8: case SDLK_F9:
4911 case SDLK_F10: case SDLK_F11: case SDLK_F12:
4912 {
4913 // /* send Ctrl-Alt-Fx to guest */
4914 com::SafeArray<LONG> keys(6);
4915
4916 keys[0] = 0x1d; // Ctrl down
4917 keys[1] = 0x38; // Alt down
4918 keys[2] = Keyevent2Keycode(pEv); // Fx down
4919 keys[3] = keys[2] + 0x80; // Fx up
4920 keys[4] = 0xb8; // Alt up
4921 keys[5] = 0x9d; // Ctrl up
4922
4923 gKeyboard->PutScancodes(ComSafeArrayAsInParam(keys), NULL);
4924 return VINF_SUCCESS;
4925 }
4926
4927 /*
4928 * Not a host key combination.
4929 * Indicate this by returning false.
4930 */
4931 default:
4932 return VERR_NOT_SUPPORTED;
4933 }
4934
4935 return VINF_SUCCESS;
4936}
4937
4938/**
4939 * Timer callback function for startup processing
4940 */
4941static Uint32 StartupTimer(Uint32 interval, void *param)
4942{
4943 /* post message so we can do something in the startup loop */
4944 SDL_Event event = {0};
4945 event.type = SDL_USEREVENT;
4946 event.user.type = SDL_USER_EVENT_TIMER;
4947 SDL_PushEvent(&event);
4948 RTSemEventSignal(g_EventSemSDLEvents);
4949 return interval;
4950}
4951
4952/**
4953 * Timer callback function to check if resizing is finished
4954 */
4955static Uint32 ResizeTimer(Uint32 interval, void *param)
4956{
4957 /* post message so the window is actually resized */
4958 SDL_Event event = {0};
4959 event.type = SDL_USEREVENT;
4960 event.user.type = SDL_USER_EVENT_WINDOW_RESIZE_DONE;
4961 PushSDLEventForSure(&event);
4962 /* one-shot */
4963 return 0;
4964}
4965
4966/**
4967 * Timer callback function to check if an ACPI power button event was handled by the guest.
4968 */
4969static Uint32 QuitTimer(Uint32 interval, void *param)
4970{
4971 BOOL fHandled = FALSE;
4972
4973 gSdlQuitTimer = NULL;
4974 if (gConsole)
4975 {
4976 int rc = gConsole->GetPowerButtonHandled(&fHandled);
4977 LogRel(("QuitTimer: rc=%d handled=%d\n", rc, fHandled));
4978 if (RT_FAILURE(rc) || !fHandled)
4979 {
4980 /* event was not handled, power down the guest */
4981 gfACPITerm = FALSE;
4982 SDL_Event event = {0};
4983 event.type = SDL_QUIT;
4984 PushSDLEventForSure(&event);
4985 }
4986 }
4987 /* one-shot */
4988 return 0;
4989}
4990
4991/**
4992 * Wait for the next SDL event. Don't use SDL_WaitEvent since this function
4993 * calls SDL_Delay(10) if the event queue is empty.
4994 */
4995static int WaitSDLEvent(SDL_Event *event)
4996{
4997 for (;;)
4998 {
4999 int rc = SDL_PollEvent (event);
5000 if (rc == 1)
5001 {
5002#ifdef USE_XPCOM_QUEUE_THREAD
5003 if (event->type == SDL_USER_EVENT_XPCOM_EVENTQUEUE)
5004 consumedXPCOMUserEvent();
5005#endif
5006 return 1;
5007 }
5008 /* Immediately wake up if new SDL events are available. This does not
5009 * work for internal SDL events. Don't wait more than 10ms. */
5010 RTSemEventWait(g_EventSemSDLEvents, 10);
5011 }
5012}
5013
5014/**
5015 * Ensure that an SDL event is really enqueued. Try multiple times if necessary.
5016 */
5017int PushSDLEventForSure(SDL_Event *event)
5018{
5019 int ntries = 10;
5020 for (; ntries > 0; ntries--)
5021 {
5022 int rc = SDL_PushEvent(event);
5023 RTSemEventSignal(g_EventSemSDLEvents);
5024#ifdef VBOX_WITH_SDL13
5025 if (rc == 1)
5026#else
5027 if (rc == 0)
5028#endif
5029 return 0;
5030 Log(("PushSDLEventForSure: waiting for 2ms (rc = %d)\n", rc));
5031 RTThreadSleep(2);
5032 }
5033 LogRel(("WARNING: Failed to enqueue SDL event %d.%d!\n",
5034 event->type, event->type == SDL_USEREVENT ? event->user.type : 0));
5035 return -1;
5036}
5037
5038#ifdef VBOXSDL_WITH_X11
5039/**
5040 * Special SDL_PushEvent function for NotifyUpdate events. These events may occur in bursts
5041 * so make sure they don't flood the SDL event queue.
5042 */
5043void PushNotifyUpdateEvent(SDL_Event *event)
5044{
5045 int rc = SDL_PushEvent(event);
5046#ifdef VBOX_WITH_SDL13
5047 bool fSuccess = (rc == 1);
5048#else
5049 bool fSuccess = (rc == 0);
5050#endif
5051
5052 RTSemEventSignal(g_EventSemSDLEvents);
5053 AssertMsg(fSuccess, ("SDL_PushEvent returned SDL error\n"));
5054 /* A global counter is faster than SDL_PeepEvents() */
5055 if (fSuccess)
5056 ASMAtomicIncS32(&g_cNotifyUpdateEventsPending);
5057 /* In order to not flood the SDL event queue, yield the CPU or (if there are already many
5058 * events queued) even sleep */
5059 if (g_cNotifyUpdateEventsPending > 96)
5060 {
5061 /* Too many NotifyUpdate events, sleep for a small amount to give the main thread time
5062 * to handle these events. The SDL queue can hold up to 128 events. */
5063 Log(("PushNotifyUpdateEvent: Sleep 1ms\n"));
5064 RTThreadSleep(1);
5065 }
5066 else
5067 RTThreadYield();
5068}
5069#endif /* VBOXSDL_WITH_X11 */
5070
5071/**
5072 *
5073 */
5074static void SetFullscreen(bool enable)
5075{
5076 if (enable == gpFramebuffer[0]->getFullscreen())
5077 return;
5078
5079 if (!gfFullscreenResize)
5080 {
5081 /*
5082 * The old/default way: SDL will resize the host to fit the guest screen resolution.
5083 */
5084 gpFramebuffer[0]->setFullscreen(enable);
5085 }
5086 else
5087 {
5088 /*
5089 * The alternate way: Switch to fullscreen with the host screen resolution and adapt
5090 * the guest screen resolution to the host window geometry.
5091 */
5092 uint32_t NewWidth = 0, NewHeight = 0;
5093 if (enable)
5094 {
5095 /* switch to fullscreen */
5096 gmGuestNormalXRes = gpFramebuffer[0]->getGuestXRes();
5097 gmGuestNormalYRes = gpFramebuffer[0]->getGuestYRes();
5098 gpFramebuffer[0]->getFullscreenGeometry(&NewWidth, &NewHeight);
5099 }
5100 else
5101 {
5102 /* switch back to saved geometry */
5103 NewWidth = gmGuestNormalXRes;
5104 NewHeight = gmGuestNormalYRes;
5105 }
5106 if (NewWidth != 0 && NewHeight != 0)
5107 {
5108 gpFramebuffer[0]->setFullscreen(enable);
5109 gfIgnoreNextResize = TRUE;
5110 gDisplay->SetVideoModeHint(NewWidth, NewHeight, 0, 0);
5111 }
5112 }
5113}
5114
5115#ifdef VBOX_WITH_SDL13
5116static VBoxSDLFB * getFbFromWinId(SDL_WindowID id)
5117{
5118 for (unsigned i = 0; i < gcMonitors; i++)
5119 if (gpFramebuffer[i]->hasWindow(id))
5120 return gpFramebuffer[i];
5121
5122 return NULL;
5123}
5124#endif
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