VirtualBox

source: vbox/trunk/src/VBox/Additions/x11/xclient/displaychange-x11.cpp@ 10466

Last change on this file since 10466 was 8695, checked in by vboxsync, 17 years ago

X11 Additions: improved termination of VBoxClient

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 4.7 KB
Line 
1/** @file
2 *
3 * Guest client: display auto-resize.
4 */
5
6/*
7 * Copyright (C) 2006-2007 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#include <VBox/log.h>
23#include <VBox/VBoxGuest.h>
24#include <iprt/assert.h>
25
26/** @todo this should probably be replaced by something IPRT */
27/* For system() and WEXITSTATUS() */
28#include <stdlib.h>
29#include <sys/types.h>
30#include <sys/wait.h>
31#include <errno.h>
32
33#include "displaychange.h"
34
35int VBoxGuestDisplayChangeThreadX11::init(void)
36{
37 int rc = VINF_SUCCESS, rcSystem, rcErrno;
38
39 LogFlowThisFunc(("\n"));
40 rcSystem = system("VBoxRandR --test");
41 if (-1 == rcSystem)
42 {
43 rcErrno = errno;
44 rc = RTErrConvertFromErrno(rcErrno);
45 }
46 if (RT_SUCCESS(rc))
47 {
48 if (0 != WEXITSTATUS(rcSystem))
49 rc = VERR_NOT_SUPPORTED;
50 }
51 if (RT_SUCCESS(rc))
52 rc = VbglR3CtlFilterMask(VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST, 0);
53 if (RT_SUCCESS(rc))
54 mInit = true;
55 LogFlowThisFunc(("returning %Rrc\n", rc));
56 return rc;
57}
58
59void VBoxGuestDisplayChangeThreadX11::uninit(void)
60{
61 LogFlowThisFunc(("\n"));
62 VbglR3CtlFilterMask(0, VMMDEV_EVENT_DISPLAY_CHANGE_REQUEST);
63 mInit = false;
64 LogFlowThisFunc(("returning\n"));
65}
66
67/**
68 * Display change request monitor thread function.
69 * Before entering the loop, we re-read the last request
70 * received, and if the first one received inside the
71 * loop is identical we ignore it, because it is probably
72 * stale.
73 */
74int VBoxGuestDisplayChangeThreadX11::threadFunction(VBoxGuestThread *pThread)
75{
76 mThread = pThread;
77 LogFlowThisFunc(("\n"));
78 uint32_t cx0 = 0, cy0 = 0, cBits0 = 0, iDisplay0 = 0;
79 int rc = VbglR3GetLastDisplayChangeRequest(&cx0, &cy0, &cBits0, &iDisplay0);
80 while (!mThread->isStopping())
81 {
82 uint32_t cx = 0, cy = 0, cBits = 0, iDisplay = 0;
83 rc = VbglR3DisplayChangeWaitEvent(&cx, &cy, &cBits, &iDisplay);
84 /* Ignore the request if it is stale */
85 if ((cx != cx0) || (cy != cy0))
86 {
87 /* If we are not stopping, sleep for a bit to avoid using up too
88 much CPU while retrying. */
89 if (RT_FAILURE(rc))
90 {
91 if (!mThread->isStopping())
92 mThread->yield();
93 }
94 else
95 system("VBoxRandR");
96 }
97 /* We do not want to ignore any further requests. */
98 cx0 = 0;
99 cy0 = 0;
100 }
101 LogFlowThisFunc(("returning VINF_SUCCESS\n"));
102 return VINF_SUCCESS;
103}
104
105/**
106 * Send a signal to the thread function that it should exit
107 */
108void VBoxGuestDisplayChangeThreadX11::stop(void)
109{
110 /**
111 * @todo is this reasonable? If the thread is in the event loop then the cancelEvent()
112 * will cause it to exit. If it enters or exits the event loop it will also
113 * notice that we wish it to exit. And if it is somewhere in-between, the
114 * yield() should give it time to get to one of places mentioned above.
115 */
116 LogFlowThisFunc(("\n"));
117 for (int i = 0; (i < 5) && mThread->isRunning(); ++i)
118 {
119 VbglR3InterruptEventWaits();;
120 mThread->yield();
121 }
122 LogFlowThisFunc(("returning\n"));
123}
124
125int VBoxGuestDisplayChangeMonitor::init(void)
126{
127 int rc = VINF_SUCCESS;
128
129 LogFlowThisFunc(("\n"));
130 if (mInit)
131 return VINF_SUCCESS;
132 rc = mThreadFunction.init();
133 if (RT_FAILURE(rc))
134 Log(("VBoxClient: failed to initialise the display change thread, rc=%Rrc (VBoxGuestDisplayChangeMonitor::init)\n", rc));
135 if (RT_SUCCESS(rc))
136 {
137 rc = mThread.start();
138 if (RT_FAILURE(rc))
139 Log(("VBoxClient: failed to start the display change thread, rc=%Rrc (VBoxGuestDisplayChangeMonitor::init)\n", rc));
140 }
141 if (RT_SUCCESS(rc))
142 mInit = true;
143 LogFlowThisFunc(("returning %Rrc\n, rc"));
144 return rc;
145}
146
147void VBoxGuestDisplayChangeMonitor::uninit(unsigned cMillies /* = RT_INDEFINITE_WAIT */)
148{
149 LogFlowThisFunc(("\n"));
150 if (mInit)
151 {
152 if (mThread.stop(cMillies, 0))
153 mThreadFunction.uninit();
154 }
155 LogFlowThisFunc(("returning\n"));
156}
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