VirtualBox

source: vbox/trunk/src/VBox/Additions/x11/vboxvideo/pointer.c@ 51374

Last change on this file since 51374 was 51242, checked in by vboxsync, 11 years ago

Additions/x11/vboxvideo: change size hint logic to take hints from a root window property and to notice RandR GetScreenInfo requests.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 17.5 KB
Line 
1/** @file
2 * VirtualBox X11 Additions graphics driver utility functions
3 */
4
5/*
6 * Copyright (C) 2006-2012 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 */
16
17#include <VBox/VMMDev.h>
18#include <VBox/VBoxGuestLib.h>
19
20#ifndef PCIACCESS
21# include <xf86Pci.h>
22# include <Pci.h>
23#endif
24
25#include "xf86.h"
26#define NEED_XF86_TYPES
27#include <iprt/string.h>
28#include "compiler.h"
29#include "cursorstr.h"
30#include "servermd.h"
31
32#include "vboxvideo.h"
33
34#ifdef XORG_7X
35# include <stdlib.h>
36#endif
37
38#define VBOX_MAX_CURSOR_WIDTH 64
39#define VBOX_MAX_CURSOR_HEIGHT 64
40
41/**************************************************************************
42* Debugging functions and macros *
43**************************************************************************/
44
45/* #define DEBUG_POINTER */
46
47#ifdef DEBUG
48# define PUT_PIXEL(c) ErrorF ("%c", c)
49#else /* DEBUG_VIDEO not defined */
50# define PUT_PIXEL(c) do { } while(0)
51#endif /* DEBUG_VIDEO not defined */
52
53/** Macro to printf an error message and return from a function */
54#define RETERROR(scrnIndex, RetVal, ...) \
55 do \
56 { \
57 xf86DrvMsg(scrnIndex, X_ERROR, __VA_ARGS__); \
58 return RetVal; \
59 } \
60 while (0)
61
62/** Structure to pass cursor image data between realise_cursor() and
63 * load_cursor_image(). The members match the parameters to
64 * @a VBoxHGSMIUpdatePointerShape(). */
65struct vboxCursorImage
66{
67 uint32_t fFlags;
68 uint32_t cHotX;
69 uint32_t cHotY;
70 uint32_t cWidth;
71 uint32_t cHeight;
72 uint8_t *pPixels;
73 uint32_t cbLength;
74};
75
76#ifdef DEBUG_POINTER
77static void
78vbox_show_shape(unsigned short w, unsigned short h, CARD32 bg, unsigned char *image)
79{
80 size_t x, y;
81 unsigned short pitch;
82 CARD32 *color;
83 unsigned char *mask;
84 size_t sizeMask;
85
86 image += sizeof(struct vboxCursorImage);
87 mask = image;
88 pitch = (w + 7) / 8;
89 sizeMask = (pitch * h + 3) & ~3;
90 color = (CARD32 *)(image + sizeMask);
91
92 TRACE_ENTRY();
93 for (y = 0; y < h; ++y, mask += pitch, color += w)
94 {
95 for (x = 0; x < w; ++x)
96 {
97 if (mask[x / 8] & (1 << (7 - (x % 8))))
98 ErrorF (" ");
99 else
100 {
101 CARD32 c = color[x];
102 if (c == bg)
103 ErrorF("Y");
104 else
105 ErrorF("X");
106 }
107 }
108 ErrorF("\n");
109 }
110}
111#endif
112
113/**************************************************************************
114* Helper functions and macros *
115**************************************************************************/
116
117/* This is called by the X server every time it loads a new cursor to see
118 * whether our "cursor hardware" can handle the cursor. This provides us with
119 * a mechanism (the only one!) to switch back from a software to a hardware
120 * cursor. */
121static Bool
122vbox_host_uses_hwcursor(ScrnInfoPtr pScrn)
123{
124 Bool rc = TRUE;
125 uint32_t fFeatures = 0;
126 VBOXPtr pVBox = pScrn->driverPrivate;
127
128 /* We may want to force the use of a software cursor. Currently this is
129 * needed if the guest uses a large virtual resolution, as in this case
130 * the host and guest tend to disagree about the pointer location. */
131 if (pVBox->forceSWCursor)
132 rc = FALSE;
133 /* Query information about mouse integration from the host. */
134 if (rc) {
135 int vrc = VbglR3GetMouseStatus(&fFeatures, NULL, NULL);
136 if (RT_FAILURE(vrc)) {
137 xf86DrvMsg(pScrn->scrnIndex, X_ERROR,
138 "Unable to determine whether the virtual machine supports mouse pointer integration - request initialization failed with return code %d\n", vrc);
139 rc = FALSE;
140 }
141 }
142 /* If we got the information from the host then make sure the host wants
143 * to draw the pointer. */
144 if (rc)
145 {
146 if ( (fFeatures & VMMDEV_MOUSE_GUEST_CAN_ABSOLUTE)
147#if GET_ABI_MAJOR(ABI_VIDEODRV_VERSION) >= 5
148 /* As of this version (server 1.6) all major Linux releases
149 * are known to handle USB tablets correctly. */
150 || (fFeatures & VMMDEV_MOUSE_HOST_HAS_ABS_DEV)
151#endif
152 )
153 /* Assume this will never be unloaded as long as the X session is
154 * running. */
155 pVBox->guestCanAbsolute = TRUE;
156 if ( (fFeatures & VMMDEV_MOUSE_HOST_CANNOT_HWPOINTER)
157 || !pVBox->guestCanAbsolute
158 || !(fFeatures & VMMDEV_MOUSE_HOST_WANTS_ABSOLUTE)
159 )
160 rc = FALSE;
161 }
162 return rc;
163}
164
165/**************************************************************************
166* Main functions *
167**************************************************************************/
168
169void
170vbox_close(ScrnInfoPtr pScrn, VBOXPtr pVBox)
171{
172 TRACE_ENTRY();
173
174 xf86DestroyCursorInfoRec(pVBox->pCurs);
175 pVBox->pCurs = NULL;
176 TRACE_EXIT();
177}
178
179Bool
180vbox_init(int scrnIndex, VBOXPtr pVBox)
181{
182 Bool rc = TRUE;
183 int vrc;
184 uint32_t fMouseFeatures = 0;
185
186 TRACE_ENTRY();
187 vrc = VbglR3Init();
188 if (RT_FAILURE(vrc))
189 {
190 xf86DrvMsg(scrnIndex, X_ERROR,
191 "Failed to initialize the VirtualBox device (rc=%d) - make sure that the VirtualBox guest additions are properly installed. If you are not sure, try reinstalling them. The X Window graphics drivers will run in compatibility mode.\n",
192 vrc);
193 rc = FALSE;
194 }
195 pVBox->useDevice = rc;
196 return rc;
197}
198
199static void
200vbox_vmm_hide_cursor(ScrnInfoPtr pScrn, VBOXPtr pVBox)
201{
202 int rc;
203
204 rc = VBoxHGSMIUpdatePointerShape(&pVBox->guestCtx, 0, 0, 0, 0, 0, NULL, 0);
205 if (RT_FAILURE(rc))
206 {
207 xf86DrvMsg(pScrn->scrnIndex, X_ERROR, "Could not hide the virtual mouse pointer, VBox error %d.\n", rc);
208 /* Play safe, and disable the hardware cursor until the next mode
209 * switch, since obviously something happened that we didn't
210 * anticipate. */
211 pVBox->forceSWCursor = TRUE;
212 }
213}
214
215static void
216vbox_vmm_show_cursor(ScrnInfoPtr pScrn, VBOXPtr pVBox)
217{
218 int rc;
219
220 if (!vbox_host_uses_hwcursor(pScrn))
221 return;
222 rc = VBoxHGSMIUpdatePointerShape(&pVBox->guestCtx, VBOX_MOUSE_POINTER_VISIBLE,
223 0, 0, 0, 0, NULL, 0);
224 if (RT_FAILURE(rc)) {
225 xf86DrvMsg(pScrn->scrnIndex, X_ERROR, "Could not unhide the virtual mouse pointer.\n");
226 /* Play safe, and disable the hardware cursor until the next mode
227 * switch, since obviously something happened that we didn't
228 * anticipate. */
229 pVBox->forceSWCursor = TRUE;
230 }
231}
232
233static void
234vbox_vmm_load_cursor_image(ScrnInfoPtr pScrn, VBOXPtr pVBox,
235 unsigned char *pvImage)
236{
237 int rc;
238 struct vboxCursorImage *pImage;
239 pImage = (struct vboxCursorImage *)pvImage;
240
241#ifdef DEBUG_POINTER
242 vbox_show_shape(pImage->cWidth, pImage->cHeight, 0, pvImage);
243#endif
244
245 rc = VBoxHGSMIUpdatePointerShape(&pVBox->guestCtx, pImage->fFlags,
246 pImage->cHotX, pImage->cHotY, pImage->cWidth, pImage->cHeight,
247 pImage->pPixels, pImage->cbLength);
248 if (RT_FAILURE(rc)) {
249 xf86DrvMsg(pScrn->scrnIndex, X_ERROR, "Unable to set the virtual mouse pointer image.\n");
250 /* Play safe, and disable the hardware cursor until the next mode
251 * switch, since obviously something happened that we didn't
252 * anticipate. */
253 pVBox->forceSWCursor = TRUE;
254 }
255}
256
257static void
258vbox_set_cursor_colors(ScrnInfoPtr pScrn, int bg, int fg)
259{
260 NOREF(pScrn);
261 NOREF(bg);
262 NOREF(fg);
263 /* ErrorF("vbox_set_cursor_colors NOT IMPLEMENTED\n"); */
264}
265
266
267static void
268vbox_set_cursor_position(ScrnInfoPtr pScrn, int x, int y)
269{
270 /* Nothing to do here, as we are telling the guest where the mouse is,
271 * not vice versa. */
272 NOREF(pScrn);
273 NOREF(x);
274 NOREF(y);
275}
276
277static void
278vbox_hide_cursor(ScrnInfoPtr pScrn)
279{
280 VBOXPtr pVBox = pScrn->driverPrivate;
281
282 vbox_vmm_hide_cursor(pScrn, pVBox);
283}
284
285static void
286vbox_show_cursor(ScrnInfoPtr pScrn)
287{
288 VBOXPtr pVBox = pScrn->driverPrivate;
289
290 vbox_vmm_show_cursor(pScrn, pVBox);
291}
292
293static void
294vbox_load_cursor_image(ScrnInfoPtr pScrn, unsigned char *image)
295{
296 VBOXPtr pVBox = pScrn->driverPrivate;
297
298 vbox_vmm_load_cursor_image(pScrn, pVBox, image);
299}
300
301static Bool
302vbox_use_hw_cursor(ScreenPtr pScreen, CursorPtr pCurs)
303{
304 ScrnInfoPtr pScrn = xf86Screens[pScreen->myNum];
305 return vbox_host_uses_hwcursor(pScrn);
306}
307
308static unsigned char
309color_to_byte(unsigned c)
310{
311 return (c >> 8) & 0xff;
312}
313
314static unsigned char *
315vbox_realize_cursor(xf86CursorInfoPtr infoPtr, CursorPtr pCurs)
316{
317 VBOXPtr pVBox;
318 CursorBitsPtr bitsp;
319 unsigned short w, h, x, y;
320 unsigned char *c, *p, *pm, *ps, *m;
321 size_t sizeRequest, sizeRgba, sizeMask, srcPitch, dstPitch;
322 CARD32 fc, bc, *cp;
323 int rc, scrnIndex = infoPtr->pScrn->scrnIndex;
324 struct vboxCursorImage *pImage;
325
326 pVBox = infoPtr->pScrn->driverPrivate;
327 bitsp = pCurs->bits;
328 w = bitsp->width;
329 h = bitsp->height;
330
331 if (!w || !h || w > VBOX_MAX_CURSOR_WIDTH || h > VBOX_MAX_CURSOR_HEIGHT)
332 RETERROR(scrnIndex, NULL,
333 "Error invalid cursor dimensions %dx%d\n", w, h);
334
335 if ((bitsp->xhot > w) || (bitsp->yhot > h))
336 RETERROR(scrnIndex, NULL,
337 "Error invalid cursor hotspot location %dx%d (max %dx%d)\n",
338 bitsp->xhot, bitsp->yhot, w, h);
339
340 srcPitch = PixmapBytePad (bitsp->width, 1);
341 dstPitch = (w + 7) / 8;
342 sizeMask = ((dstPitch * h) + 3) & (size_t) ~3;
343 sizeRgba = w * h * 4;
344 sizeRequest = sizeMask + sizeRgba + sizeof(*pImage);
345
346 p = c = calloc (1, sizeRequest);
347 if (!c)
348 RETERROR(scrnIndex, NULL,
349 "Error failed to alloc %lu bytes for cursor\n",
350 (unsigned long) sizeRequest);
351
352 pImage = (struct vboxCursorImage *)p;
353 pImage->pPixels = m = p + sizeof(*pImage);
354 cp = (CARD32 *)(m + sizeMask);
355
356 TRACE_LOG ("w=%d h=%d sm=%d sr=%d p=%d\n",
357 w, h, (int) sizeMask, (int) sizeRgba, (int) dstPitch);
358 TRACE_LOG ("m=%p c=%p cp=%p\n", m, c, (void *)cp);
359
360 fc = color_to_byte (pCurs->foreBlue)
361 | (color_to_byte (pCurs->foreGreen) << 8)
362 | (color_to_byte (pCurs->foreRed) << 16);
363
364 bc = color_to_byte (pCurs->backBlue)
365 | (color_to_byte (pCurs->backGreen) << 8)
366 | (color_to_byte (pCurs->backRed) << 16);
367
368 /*
369 * Convert the Xorg source/mask bits to the and/xor bits VBox needs.
370 * Xorg:
371 * The mask is a bitmap indicating which parts of the cursor are
372 * transparent and which parts are drawn. The source is a bitmap
373 * indicating which parts of the non-transparent portion of the
374 * the cursor should be painted in the foreground color and which
375 * should be painted in the background color. By default, set bits
376 * indicate the opaque part of the mask bitmap and clear bits
377 * indicate the transparent part.
378 * VBox:
379 * The color data is the XOR mask. The AND mask bits determine
380 * which pixels of the color data (XOR mask) will replace (overwrite)
381 * the screen pixels (AND mask bit = 0) and which ones will be XORed
382 * with existing screen pixels (AND mask bit = 1).
383 * For example when you have the AND mask all 0, then you see the
384 * correct mouse pointer image surrounded by black square.
385 */
386 for (pm = bitsp->mask, ps = bitsp->source, y = 0;
387 y < h;
388 ++y, pm += srcPitch, ps += srcPitch, m += dstPitch)
389 {
390 for (x = 0; x < w; ++x)
391 {
392 if (pm[x / 8] & (1 << (x % 8)))
393 {
394 /* opaque, leave AND mask bit at 0 */
395 if (ps[x / 8] & (1 << (x % 8)))
396 {
397 *cp++ = fc;
398 PUT_PIXEL('X');
399 }
400 else
401 {
402 *cp++ = bc;
403 PUT_PIXEL('*');
404 }
405 }
406 else
407 {
408 /* transparent, set AND mask bit */
409 m[x / 8] |= 1 << (7 - (x % 8));
410 /* don't change the screen pixel */
411 *cp++ = 0;
412 PUT_PIXEL(' ');
413 }
414 }
415 PUT_PIXEL('\n');
416 }
417
418 pImage->cWidth = w;
419 pImage->cHeight = h;
420 pImage->cHotX = bitsp->xhot;
421 pImage->cHotY = bitsp->yhot;
422 pImage->fFlags = VBOX_MOUSE_POINTER_VISIBLE | VBOX_MOUSE_POINTER_SHAPE;
423 pImage->cbLength = sizeRequest - sizeof(*pImage);
424
425#ifdef DEBUG_POINTER
426 ErrorF("shape = %p\n", p);
427 vbox_show_shape(w, h, bc, c);
428#endif
429
430 return p;
431}
432
433#ifdef ARGB_CURSOR
434static Bool
435vbox_use_hw_cursor_argb(ScreenPtr pScreen, CursorPtr pCurs)
436{
437 ScrnInfoPtr pScrn = xf86Screens[pScreen->myNum];
438 Bool rc = TRUE;
439
440 if (!vbox_host_uses_hwcursor(pScrn))
441 rc = FALSE;
442 if ( rc
443 && ( (pCurs->bits->height > VBOX_MAX_CURSOR_HEIGHT)
444 || (pCurs->bits->width > VBOX_MAX_CURSOR_WIDTH)
445 || (pScrn->bitsPerPixel <= 8)
446 )
447 )
448 rc = FALSE;
449 return rc;
450}
451
452
453static void
454vbox_load_cursor_argb(ScrnInfoPtr pScrn, CursorPtr pCurs)
455{
456 VBOXPtr pVBox;
457 VMMDevReqMousePointer *reqp;
458 CursorBitsPtr bitsp;
459 unsigned short w, h;
460 unsigned short cx, cy;
461 unsigned char *pm;
462 CARD32 *pc;
463 size_t sizeData, sizeMask;
464 CARD8 *p;
465 int scrnIndex;
466 uint32_t fFlags = VBOX_MOUSE_POINTER_VISIBLE | VBOX_MOUSE_POINTER_SHAPE
467 | VBOX_MOUSE_POINTER_ALPHA;
468 int rc;
469
470 pVBox = pScrn->driverPrivate;
471 bitsp = pCurs->bits;
472 w = bitsp->width;
473 h = bitsp->height;
474 scrnIndex = pScrn->scrnIndex;
475
476 /* Mask must be generated for alpha cursors, that is required by VBox. */
477 /* note: (michael) the next struct must be 32bit aligned. */
478 sizeMask = ((w + 7) / 8 * h + 3) & ~3;
479
480 if (!w || !h || w > VBOX_MAX_CURSOR_WIDTH || h > VBOX_MAX_CURSOR_HEIGHT)
481 RETERROR(scrnIndex, ,
482 "Error invalid cursor dimensions %dx%d\n", w, h);
483
484 if ((bitsp->xhot > w) || (bitsp->yhot > h))
485 RETERROR(scrnIndex, ,
486 "Error invalid cursor hotspot location %dx%d (max %dx%d)\n",
487 bitsp->xhot, bitsp->yhot, w, h);
488
489 sizeData = w * h * 4 + sizeMask;
490 p = calloc(1, sizeData);
491 if (!p)
492 RETERROR(scrnIndex, ,
493 "Error failed to alloc %lu bytes for cursor\n",
494 (unsigned long)sizeData);
495
496 memcpy(p + sizeMask, bitsp->argb, w * h * 4);
497
498 /* Emulate the AND mask. */
499 pm = p;
500 pc = bitsp->argb;
501
502 /* Init AND mask to 1 */
503 memset(pm, 0xFF, sizeMask);
504
505 /*
506 * The additions driver must provide the AND mask for alpha cursors. The host frontend
507 * which can handle alpha channel, will ignore the AND mask and draw an alpha cursor.
508 * But if the host does not support ARGB, then it simply uses the AND mask and the color
509 * data to draw a normal color cursor.
510 */
511 for (cy = 0; cy < h; cy++)
512 {
513 unsigned char bitmask = 0x80;
514
515 for (cx = 0; cx < w; cx++, bitmask >>= 1)
516 {
517 if (bitmask == 0)
518 bitmask = 0x80;
519
520 if (pc[cx] >= 0xF0000000)
521 pm[cx / 8] &= ~bitmask;
522 }
523
524 /* Point to next source and dest scans */
525 pc += w;
526 pm += (w + 7) / 8;
527 }
528
529 VBoxHGSMIUpdatePointerShape(&pVBox->guestCtx, fFlags, bitsp->xhot,
530 bitsp->yhot, w, h, p, sizeData);
531 free(p);
532}
533#endif
534
535Bool
536vbox_cursor_init(ScreenPtr pScreen)
537{
538 ScrnInfoPtr pScrn = xf86Screens[pScreen->myNum];
539 VBOXPtr pVBox = pScrn->driverPrivate;
540 xf86CursorInfoPtr pCurs = NULL;
541 Bool rc = TRUE;
542
543 TRACE_ENTRY();
544 if (!pVBox->fHaveHGSMI)
545 return FALSE;
546 pVBox->pCurs = pCurs = xf86CreateCursorInfoRec();
547 if (!pCurs) {
548 xf86DrvMsg(pScrn->scrnIndex, X_ERROR,
549 "Failed to create X Window cursor information structures for virtual mouse.\n");
550 rc = FALSE;
551 }
552 if (rc) {
553 pCurs->MaxWidth = VBOX_MAX_CURSOR_WIDTH;
554 pCurs->MaxHeight = VBOX_MAX_CURSOR_HEIGHT;
555 pCurs->Flags = HARDWARE_CURSOR_TRUECOLOR_AT_8BPP
556 | HARDWARE_CURSOR_SOURCE_MASK_INTERLEAVE_1
557 | HARDWARE_CURSOR_BIT_ORDER_MSBFIRST;
558
559 pCurs->SetCursorColors = vbox_set_cursor_colors;
560 pCurs->SetCursorPosition = vbox_set_cursor_position;
561 pCurs->LoadCursorImage = vbox_load_cursor_image;
562 pCurs->HideCursor = vbox_hide_cursor;
563 pCurs->ShowCursor = vbox_show_cursor;
564 pCurs->UseHWCursor = vbox_use_hw_cursor;
565 pCurs->RealizeCursor = vbox_realize_cursor;
566
567#ifdef ARGB_CURSOR
568 pCurs->UseHWCursorARGB = vbox_use_hw_cursor_argb;
569 pCurs->LoadCursorARGB = vbox_load_cursor_argb;
570#endif
571
572 /* Hide the host cursor before we initialise if we wish to use a
573 * software cursor. */
574 if (pVBox->forceSWCursor)
575 vbox_vmm_hide_cursor(pScrn, pVBox);
576 rc = xf86InitCursor(pScreen, pCurs);
577 }
578 if (!rc)
579 xf86DrvMsg(pScrn->scrnIndex, X_ERROR,
580 "Failed to enable mouse pointer integration.\n");
581 if (!rc && (pCurs != NULL))
582 xf86DestroyCursorInfoRec(pCurs);
583 return rc;
584}
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