VirtualBox

source: vbox/trunk/src/VBox/Additions/x11/xclient/seamless-x11.cpp@ 7539

Last change on this file since 7539 was 7449, checked in by vboxsync, 17 years ago

Additions/x11: added flow logging statements to the VBoxClient code

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 13.9 KB
Line 
1/** @file
2 *
3 * Seamless mode:
4 * Linux guest.
5 */
6
7/*
8 * Copyright (C) 2008 innotek GmbH
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19/*****************************************************************************
20* Header files *
21*****************************************************************************/
22
23#include <iprt/err.h>
24#include <iprt/assert.h>
25#include <VBox/log.h>
26#include <VBox/VBoxGuest.h>
27
28#include "seamless-guest.h"
29
30#include <X11/Xatom.h>
31#include <X11/Xmu/WinUtil.h>
32
33/* This is defined wrong in my X11 header files! */
34#define VBoxShapeNotify 64
35
36/*****************************************************************************
37* Static functions *
38*****************************************************************************/
39
40static unsigned char *XXGetProperty (Display *aDpy, Window aWnd, Atom aPropType,
41 const char *aPropName, unsigned long *nItems)
42{
43 LogFlowFunc(("\n"));
44 Atom propNameAtom = XInternAtom (aDpy, aPropName,
45 True /* only_if_exists */);
46 if (propNameAtom == None)
47 {
48 return NULL;
49 }
50
51 Atom actTypeAtom = None;
52 int actFmt = 0;
53 unsigned long nBytesAfter = 0;
54 unsigned char *propVal = 0;
55 int rc = XGetWindowProperty (aDpy, aWnd, propNameAtom,
56 0, LONG_MAX, False /* delete */,
57 aPropType, &actTypeAtom, &actFmt,
58 nItems, &nBytesAfter, &propVal);
59 if (rc != Success)
60 return NULL;
61
62 LogFlowFunc(("returning\n"));
63 return propVal;
64}
65
66/**
67 * Initialise the guest and ensure that it is capable of handling seamless mode
68 *
69 * @returns true if it can handle seamless, false otherwise
70 */
71int VBoxGuestSeamlessX11::init(VBoxGuestSeamlessObserver *pObserver)
72{
73 int rc = VINF_SUCCESS;
74
75 LogFlowThisFunc(("\n"));
76 if (0 != mObserver) /* Assertion */
77 {
78 LogRel(("VBoxClient: ERROR: attempt to initialise seamless guest object twice!\n"));
79 return VERR_INTERNAL_ERROR;
80 }
81 if (!mDisplay.init())
82 {
83 LogRel(("VBoxClient: seamless guest object failed to acquire a connection to the display.\n"));
84 return VERR_ACCESS_DENIED;
85 }
86 mObserver = pObserver;
87 LogFlowThisFunc(("returning %Rrc\n", rc));
88 return rc;
89}
90
91/**
92 * Read information about currently visible windows in the guest and subscribe to X11
93 * events about changes to this information.
94 *
95 * @note This class does not contain its own event thread, so an external thread must
96 * call nextEvent() for as long as events are wished.
97 * @todo This function should switch the guest to fullscreen mode.
98 */
99int VBoxGuestSeamlessX11::start(void)
100{
101 int rc = VINF_SUCCESS;
102 /** Dummy values for XShapeQueryExtension */
103 int error, event;
104
105 LogFlowThisFunc(("\n"));
106 mSupportsShape = XShapeQueryExtension(mDisplay, &event, &error);
107 mEnabled = true;
108 monitorClientList();
109 rebuildWindowTree();
110 LogFlowThisFunc(("returning %Rrc\n", rc));
111 return rc;
112}
113
114/** Stop reporting seamless events to the host. Free information about guest windows
115 and stop requesting updates. */
116void VBoxGuestSeamlessX11::stop(void)
117{
118 LogFlowThisFunc(("\n"));
119 mEnabled = false;
120 unmonitorClientList();
121 freeWindowTree();
122 LogFlowThisFunc(("returning\n"));
123}
124
125void VBoxGuestSeamlessX11::monitorClientList(void)
126{
127 LogFlowThisFunc(("called\n"));
128 XSelectInput(mDisplay, DefaultRootWindow(mDisplay.get()), SubstructureNotifyMask);
129}
130
131void VBoxGuestSeamlessX11::unmonitorClientList(void)
132{
133 LogFlowThisFunc(("called\n"));
134 XSelectInput(mDisplay, DefaultRootWindow(mDisplay.get()), 0);
135}
136
137/**
138 * Recreate the table of toplevel windows of clients on the default root window of the
139 * X server.
140 */
141void VBoxGuestSeamlessX11::rebuildWindowTree(void)
142{
143 LogFlowThisFunc(("called\n"));
144 freeWindowTree();
145 addClients(DefaultRootWindow(mDisplay.get()));
146}
147
148
149/**
150 * Look at the list of children of a virtual root window and add them to the list of clients
151 * if they belong to a client which is not a virtual root.
152 *
153 * @param hRoot the virtual root window to be examined
154 */
155void VBoxGuestSeamlessX11::addClients(const Window hRoot)
156{
157 /** Unused out parameters of XQueryTree */
158 Window hRealRoot, hParent;
159 /** The list of children of the root supplied, raw pointer */
160 Window *phChildrenRaw;
161 /** The list of children of the root supplied, auto-pointer */
162 VBoxGuestX11Pointer<Window> phChildren;
163 /** The number of children of the root supplied */
164 unsigned cChildren;
165
166 LogFlowThisFunc(("\n"));
167 if (!XQueryTree(mDisplay.get(), hRoot, &hRealRoot, &hParent, &phChildrenRaw, &cChildren))
168 return;
169 phChildren = phChildrenRaw;
170 for (unsigned i = 0; i < cChildren; ++i)
171 addClientWindow(phChildren.get()[i]);
172 LogFlowThisFunc(("returning\n"));
173}
174
175
176void VBoxGuestSeamlessX11::addClientWindow(const Window hWin)
177{
178 LogFlowThisFunc(("\n"));
179 XWindowAttributes winAttrib;
180 bool fAddWin = true;
181 char *pszWinName = NULL;
182 Window hClient = XmuClientWindow(mDisplay, hWin);
183
184 if (isVirtualRoot(hClient))
185 fAddWin = false;
186 if (fAddWin && !XGetWindowAttributes(mDisplay, hWin, &winAttrib))
187 {
188 LogRelFunc(("VBoxClient: Failed to get the window attributes for window %d\n", hWin));
189 fAddWin = false;
190 }
191 if (fAddWin && (winAttrib.map_state == IsUnmapped))
192 fAddWin = false;
193 if (fAddWin && (XFetchName(mDisplay, hClient, &pszWinName) != 0) && (pszWinName != NULL))
194 XFree(pszWinName);
195 else
196 /* kwin sometimes creates temporary fullscreen windows with no name. */
197 fAddWin = false;
198 if (fAddWin)
199 {
200 VBoxGuestX11Pointer<XRectangle> rects;
201 int cRects = 0, iOrdering;
202 bool hasShape = false;
203
204 if (mSupportsShape)
205 {
206 XShapeSelectInput(mDisplay, hWin, ShapeNotifyMask);
207 rects = XShapeGetRectangles(mDisplay, hWin, ShapeBounding, &cRects, &iOrdering);
208 if (0 == rects.get())
209 cRects = 0;
210 else
211 {
212 if ( (cRects > 1)
213 || (rects.get()[0].x != 0)
214 || (rects.get()[0].y != 0)
215 || (rects.get()[0].width != winAttrib.width)
216 || (rects.get()[0].height != winAttrib.height)
217 )
218 hasShape = true;
219 }
220 }
221 mGuestWindows.addWindow(hWin, hasShape, winAttrib.x, winAttrib.y,
222 winAttrib.width, winAttrib.height, cRects, rects);
223 }
224 LogFlowThisFunc(("returning\n"));
225}
226
227
228/**
229 * Checks whether a window is a virtual root.
230 * @returns true if it is, false otherwise
231 * @param hWin the window to be examined
232 */
233bool VBoxGuestSeamlessX11::isVirtualRoot(Window hWin)
234{
235 unsigned char *windowTypeRaw;
236 VBoxGuestX11Pointer<Atom> windowType;
237 unsigned long ulCount;
238 bool rc = false;
239
240 LogFlowThisFunc(("\n"));
241 windowTypeRaw = XXGetProperty(mDisplay, hWin, XA_ATOM, WM_TYPE_PROP, &ulCount);
242 if (windowTypeRaw != NULL)
243 {
244 windowType = reinterpret_cast<Atom *>(windowTypeRaw);
245 if ( (ulCount != 0)
246 && (*windowType == XInternAtom(mDisplay, WM_TYPE_DESKTOP_PROP, True)))
247 rc = true;
248 }
249 LogFlowThisFunc(("returning %s\n", rc ? "true" : "false"));
250 return rc;
251}
252
253
254/**
255 * Free all information in the tree of visible windows
256 */
257void VBoxGuestSeamlessX11::freeWindowTree(void)
258{
259 /* We use post-increment in the operation to prevent the iterator from being invalidated. */
260 LogFlowThisFunc(("\n"));
261 for (VBoxGuestWindowList::iterator it = mGuestWindows.begin(); it != mGuestWindows.end();
262 mGuestWindows.removeWindow(it++))
263 {
264 XShapeSelectInput(mDisplay, it->first, 0);
265 }
266 LogFlowThisFunc(("returning\n"));
267}
268
269
270/**
271 * Waits for a position or shape-related event from guest windows
272 *
273 * @note Called from the guest event thread.
274 */
275void VBoxGuestSeamlessX11::nextEvent(void)
276{
277 XEvent event;
278
279 LogFlowThisFunc(("\n"));
280 /* Start by sending information about the current window setup to the host. We do this
281 here because we want to send all such information from a single thread. */
282 mObserver->notify();
283 XNextEvent(mDisplay, &event);
284 switch (event.type)
285 {
286 case ConfigureNotify:
287 doConfigureEvent(&event.xconfigure);
288 break;
289 case MapNotify:
290 doMapEvent(&event.xmap);
291 break;
292 case VBoxShapeNotify: /* This is defined wrong in my X11 header files! */
293 doShapeEvent(reinterpret_cast<XShapeEvent *>(&event));
294 break;
295 case UnmapNotify:
296 doUnmapEvent(&event.xunmap);
297 break;
298 default:
299 break;
300 }
301 LogFlowThisFunc(("returning\n"));
302}
303
304/**
305 * Handle a configuration event in the seamless event thread by setting the new position.
306 *
307 * @param event the X11 event structure
308 */
309void VBoxGuestSeamlessX11::doConfigureEvent(const XConfigureEvent *event)
310{
311 LogFlowThisFunc(("\n"));
312 VBoxGuestWindowList::iterator iter;
313
314 iter = mGuestWindows.find(event->window);
315 if (iter != mGuestWindows.end())
316 {
317 XWindowAttributes winAttrib;
318
319 if (XGetWindowAttributes(mDisplay, event->window, &winAttrib))
320 {
321 iter->second->mX = winAttrib.x;
322 iter->second->mY = winAttrib.y;
323 iter->second->mWidth = winAttrib.width;
324 iter->second->mHeight = winAttrib.height;
325 }
326 }
327 LogFlowThisFunc(("returning\n"));
328}
329
330/**
331 * Handle a map event in the seamless event thread.
332 *
333 * @param event the X11 event structure
334 */
335void VBoxGuestSeamlessX11::doMapEvent(const XMapEvent *event)
336{
337 LogFlowThisFunc(("\n"));
338 VBoxGuestWindowList::iterator iter;
339
340 iter = mGuestWindows.find(event->window);
341 if (mGuestWindows.end() == iter)
342 {
343 addClientWindow(event->window);
344 }
345 LogFlowThisFunc(("returning\n"));
346}
347
348
349/**
350 * Handle a window shape change event in the seamless event thread.
351 *
352 * @param event the X11 event structure
353 */
354void VBoxGuestSeamlessX11::doShapeEvent(const XShapeEvent *event)
355{
356 LogFlowThisFunc(("\n"));
357 VBoxGuestWindowList::iterator iter;
358
359 iter = mGuestWindows.find(event->window);
360 if (iter != mGuestWindows.end())
361 {
362 VBoxGuestX11Pointer<XRectangle> rects;
363 int cRects = 0, iOrdering;
364
365 rects = XShapeGetRectangles(mDisplay, event->window, ShapeBounding, &cRects, &iOrdering);
366 iter->second->mhasShape = true;
367 iter->second->mcRects = cRects;
368 iter->second->mapRects = rects;
369 }
370 LogFlowThisFunc(("returning\n"));
371}
372
373/**
374 * Handle an unmap event in the seamless event thread.
375 *
376 * @param event the X11 event structure
377 */
378void VBoxGuestSeamlessX11::doUnmapEvent(const XUnmapEvent *event)
379{
380 LogFlowThisFunc(("\n"));
381 VBoxGuestWindowList::iterator iter;
382
383 iter = mGuestWindows.find(event->window);
384 if (mGuestWindows.end() != iter)
385 {
386 mGuestWindows.removeWindow(iter);
387 }
388 LogFlowThisFunc(("returning\n"));
389}
390
391/**
392 * Sends an updated list of visible rectangles to the host
393 */
394std::auto_ptr<std::vector<RTRECT> > VBoxGuestSeamlessX11::getRects(void)
395{
396 LogFlowThisFunc(("\n"));
397 unsigned cRects = 0;
398 std::auto_ptr<std::vector<RTRECT> > apRects(new std::vector<RTRECT>);
399
400 if (0 != mcRects)
401 {
402 apRects.get()->reserve(mcRects * 2);
403 }
404 for (VBoxGuestWindowList::iterator it = mGuestWindows.begin();
405 it != mGuestWindows.end(); ++it)
406 {
407 if (it->second->mhasShape)
408 {
409 for (int i = 0; i < it->second->mcRects; ++i)
410 {
411 RTRECT rect;
412 rect.xLeft = it->second->mX
413 + it->second->mapRects.get()[i].x;
414 rect.yBottom = it->second->mY
415 + it->second->mapRects.get()[i].y
416 + it->second->mapRects.get()[i].height;
417 rect.xRight = it->second->mX
418 + it->second->mapRects.get()[i].x
419 + it->second->mapRects.get()[i].width;
420 rect.yTop = it->second->mY
421 + it->second->mapRects.get()[i].y;
422 apRects.get()->push_back(rect);
423 }
424 cRects += it->second->mcRects;
425 }
426 else
427 {
428 RTRECT rect;
429 rect.xLeft = it->second->mX;
430 rect.yBottom = it->second->mY
431 + it->second->mHeight;
432 rect.xRight = it->second->mX
433 + it->second->mWidth;
434 rect.yTop = it->second->mY;
435 apRects.get()->push_back(rect);
436 ++cRects;
437 }
438 }
439 mcRects = cRects;
440 LogFlowThisFunc(("returning\n"));
441 return apRects;
442}
443
444/**
445 * Send a client event to wake up the X11 seamless event loop prior to stopping it.
446 *
447 * @note This function should only be called from the host event thread.
448 */
449bool VBoxGuestSeamlessX11::interruptEvent(void)
450{
451 bool rc = false;
452
453 LogFlowThisFunc(("\n"));
454 /* Message contents set to zero. */
455 XClientMessageEvent clientMessage = { ClientMessage, 0, 0, 0, 0, 0, 8 };
456
457 if (0 != XSendEvent(mDisplay, DefaultRootWindow(mDisplay.get()), false, PropertyChangeMask,
458 reinterpret_cast<XEvent *>(&clientMessage)))
459 {
460 XFlush(mDisplay);
461 rc = true;
462 }
463 LogFlowThisFunc(("returning %s\n", rc ? "true" : "false"));
464 return rc;
465}
Note: See TracBrowser for help on using the repository browser.

© 2025 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette