VirtualBox

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

Last change on this file since 39561 was 38648, checked in by vboxsync, 13 years ago

Additions/common and X11: more VBOXGUEST_IOCTL_SET_MOUSE_STATUS adjustments

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 17.6 KB
Line 
1/** @file
2 * VirtualBox X11 Additions graphics driver utility functions
3 */
4
5/*
6 * Copyright (C) 2006-2007 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
31#include "vboxvideo.h"
32
33#define VBOX_MAX_CURSOR_WIDTH 64
34#define VBOX_MAX_CURSOR_HEIGHT 64
35
36/**************************************************************************
37* Debugging functions and macros *
38**************************************************************************/
39
40/* #define DEBUG_POINTER */
41
42#ifdef DEBUG
43# define PUT_PIXEL(c) ErrorF ("%c", c)
44#else /* DEBUG_VIDEO not defined */
45# define PUT_PIXEL(c) do { } while(0)
46#endif /* DEBUG_VIDEO not defined */
47
48/** Macro to printf an error message and return from a function */
49#define RETERROR(scrnIndex, RetVal, ...) \
50 do \
51 { \
52 xf86DrvMsg(scrnIndex, X_ERROR, __VA_ARGS__); \
53 return RetVal; \
54 } \
55 while (0)
56
57/** Structure to pass cursor image data between realise_cursor() and
58 * load_cursor_image(). The members match the parameters to
59 * @a VBoxHGSMIUpdatePointerShape(). */
60struct vboxCursorImage
61{
62 uint32_t fFlags;
63 uint32_t cHotX;
64 uint32_t cHotY;
65 uint32_t cWidth;
66 uint32_t cHeight;
67 uint8_t *pPixels;
68 uint32_t cbLength;
69};
70
71#ifdef DEBUG_POINTER
72static void
73vbox_show_shape(unsigned short w, unsigned short h, CARD32 bg, unsigned char *image)
74{
75 size_t x, y;
76 unsigned short pitch;
77 CARD32 *color;
78 unsigned char *mask;
79 size_t sizeMask;
80
81 image += sizeof(struct vboxCursorImage);
82 mask = image;
83 pitch = (w + 7) / 8;
84 sizeMask = (pitch * h + 3) & ~3;
85 color = (CARD32 *)(image + sizeMask);
86
87 TRACE_ENTRY();
88 for (y = 0; y < h; ++y, mask += pitch, color += w)
89 {
90 for (x = 0; x < w; ++x)
91 {
92 if (mask[x / 8] & (1 << (7 - (x % 8))))
93 ErrorF (" ");
94 else
95 {
96 CARD32 c = color[x];
97 if (c == bg)
98 ErrorF("Y");
99 else
100 ErrorF("X");
101 }
102 }
103 ErrorF("\n");
104 }
105}
106#endif
107
108/**************************************************************************
109* Helper functions and macros *
110**************************************************************************/
111
112/* This is called by the X server every time it loads a new cursor to see
113 * whether our "cursor hardware" can handle the cursor. This provides us with
114 * a mechanism (the only one!) to switch back from a software to a hardware
115 * cursor. */
116static Bool
117vbox_host_uses_hwcursor(ScrnInfoPtr pScrn)
118{
119 Bool rc = TRUE;
120 uint32_t fFeatures = 0;
121 VBOXPtr pVBox = pScrn->driverPrivate;
122
123 /* We may want to force the use of a software cursor. Currently this is
124 * needed if the guest uses a large virtual resolution, as in this case
125 * the host and guest tend to disagree about the pointer location. */
126 if (pVBox->forceSWCursor)
127 rc = FALSE;
128 /* Query information about mouse integration from the host. */
129 if (rc) {
130 int vrc = VbglR3GetMouseStatus(&fFeatures, NULL, NULL);
131 if (RT_FAILURE(vrc)) {
132 xf86DrvMsg(pScrn->scrnIndex, X_ERROR,
133 "Unable to determine whether the virtual machine supports mouse pointer integration - request initialization failed with return code %d\n", vrc);
134 rc = FALSE;
135 }
136 }
137 /* If we got the information from the host then make sure the host wants
138 * to draw the pointer. */
139 if (rc)
140 {
141 if ( (fFeatures & VMMDEV_MOUSE_GUEST_CAN_ABSOLUTE)
142#if GET_ABI_MAJOR(ABI_VIDEODRV_VERSION) >= 5
143 /* As of this version (server 1.6) all major Linux releases
144 * are known to handle USB tablets correctly. */
145 || (fFeatures & VMMDEV_MOUSE_HOST_HAS_ABS_DEV)
146#endif
147 )
148 /* Assume this will never be unloaded as long as the X session is
149 * running. */
150 pVBox->guestCanAbsolute = TRUE;
151 if ( (fFeatures & VMMDEV_MOUSE_HOST_CANNOT_HWPOINTER)
152 || !pVBox->guestCanAbsolute
153 || !(fFeatures & VMMDEV_MOUSE_HOST_WANTS_ABSOLUTE)
154 )
155 rc = FALSE;
156 }
157 return rc;
158}
159
160/**************************************************************************
161* Main functions *
162**************************************************************************/
163
164void
165vbox_close(ScrnInfoPtr pScrn, VBOXPtr pVBox)
166{
167 TRACE_ENTRY();
168
169 xf86DestroyCursorInfoRec(pVBox->pCurs);
170 pVBox->pCurs = NULL;
171 TRACE_EXIT();
172}
173
174Bool
175vbox_init(int scrnIndex, VBOXPtr pVBox)
176{
177 Bool rc = TRUE;
178 int vrc;
179 uint32_t fMouseFeatures = 0;
180
181 TRACE_ENTRY();
182 vrc = VbglR3Init();
183 if (RT_FAILURE(vrc))
184 {
185 xf86DrvMsg(scrnIndex, X_ERROR,
186 "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",
187 vrc);
188 rc = FALSE;
189 }
190 pVBox->useDevice = rc;
191 return rc;
192}
193
194static void
195vbox_vmm_hide_cursor(ScrnInfoPtr pScrn, VBOXPtr pVBox)
196{
197 int rc;
198
199 rc = VBoxHGSMIUpdatePointerShape(&pVBox->guestCtx, 0, 0, 0, 0, 0, NULL, 0);
200 if (RT_FAILURE(rc))
201 {
202 xf86DrvMsg(pScrn->scrnIndex, X_ERROR, "Could not hide the virtual mouse pointer, VBox error %d.\n", rc);
203 /* Play safe, and disable the hardware cursor until the next mode
204 * switch, since obviously something happened that we didn't
205 * anticipate. */
206 pVBox->forceSWCursor = TRUE;
207 }
208}
209
210static void
211vbox_vmm_show_cursor(ScrnInfoPtr pScrn, VBOXPtr pVBox)
212{
213 int rc;
214
215 if (!vbox_host_uses_hwcursor(pScrn))
216 return;
217 rc = VBoxHGSMIUpdatePointerShape(&pVBox->guestCtx, VBOX_MOUSE_POINTER_VISIBLE,
218 0, 0, 0, 0, NULL, 0);
219 if (RT_FAILURE(rc)) {
220 xf86DrvMsg(pScrn->scrnIndex, X_ERROR, "Could not unhide the virtual mouse pointer.\n");
221 /* Play safe, and disable the hardware cursor until the next mode
222 * switch, since obviously something happened that we didn't
223 * anticipate. */
224 pVBox->forceSWCursor = TRUE;
225 }
226}
227
228static void
229vbox_vmm_load_cursor_image(ScrnInfoPtr pScrn, VBOXPtr pVBox,
230 unsigned char *pvImage)
231{
232 int rc;
233 struct vboxCursorImage *pImage;
234 pImage = (struct vboxCursorImage *)pvImage;
235
236#ifdef DEBUG_POINTER
237 vbox_show_shape(pImage->cWidth, pImage->cHeight, 0, pvImage);
238#endif
239
240 rc = VBoxHGSMIUpdatePointerShape(&pVBox->guestCtx, pImage->fFlags,
241 pImage->cHotX, pImage->cHotY, pImage->cWidth, pImage->cHeight,
242 pImage->pPixels, pImage->cbLength);
243 if (RT_FAILURE(rc)) {
244 xf86DrvMsg(pScrn->scrnIndex, X_ERROR, "Unable to set the virtual mouse pointer image.\n");
245 /* Play safe, and disable the hardware cursor until the next mode
246 * switch, since obviously something happened that we didn't
247 * anticipate. */
248 pVBox->forceSWCursor = TRUE;
249 }
250}
251
252static void
253vbox_set_cursor_colors(ScrnInfoPtr pScrn, int bg, int fg)
254{
255 NOREF(pScrn);
256 NOREF(bg);
257 NOREF(fg);
258 /* ErrorF("vbox_set_cursor_colors NOT IMPLEMENTED\n"); */
259}
260
261
262static void
263vbox_set_cursor_position(ScrnInfoPtr pScrn, int x, int y)
264{
265 /* Nothing to do here, as we are telling the guest where the mouse is,
266 * not vice versa. */
267 NOREF(pScrn);
268 NOREF(x);
269 NOREF(y);
270}
271
272static void
273vbox_hide_cursor(ScrnInfoPtr pScrn)
274{
275 VBOXPtr pVBox = pScrn->driverPrivate;
276
277 vbox_vmm_hide_cursor(pScrn, pVBox);
278}
279
280static void
281vbox_show_cursor(ScrnInfoPtr pScrn)
282{
283 VBOXPtr pVBox = pScrn->driverPrivate;
284
285 vbox_vmm_show_cursor(pScrn, pVBox);
286}
287
288static void
289vbox_load_cursor_image(ScrnInfoPtr pScrn, unsigned char *image)
290{
291 VBOXPtr pVBox = pScrn->driverPrivate;
292
293 vbox_vmm_load_cursor_image(pScrn, pVBox, image);
294}
295
296static Bool
297vbox_use_hw_cursor(ScreenPtr pScreen, CursorPtr pCurs)
298{
299 ScrnInfoPtr pScrn = xf86Screens[pScreen->myNum];
300 return vbox_host_uses_hwcursor(pScrn);
301}
302
303static unsigned char
304color_to_byte(unsigned c)
305{
306 return (c >> 8) & 0xff;
307}
308
309static unsigned char *
310vbox_realize_cursor(xf86CursorInfoPtr infoPtr, CursorPtr pCurs)
311{
312 VBOXPtr pVBox;
313 CursorBitsPtr bitsp;
314 unsigned short w, h, x, y;
315 unsigned char *c, *p, *pm, *ps, *m;
316 size_t sizeRequest, sizeRgba, sizeMask, srcPitch, dstPitch;
317 CARD32 fc, bc, *cp;
318 int rc, scrnIndex = infoPtr->pScrn->scrnIndex;
319 struct vboxCursorImage *pImage;
320
321 pVBox = infoPtr->pScrn->driverPrivate;
322 bitsp = pCurs->bits;
323 w = bitsp->width;
324 h = bitsp->height;
325
326 if (!w || !h || w > VBOX_MAX_CURSOR_WIDTH || h > VBOX_MAX_CURSOR_HEIGHT)
327 RETERROR(scrnIndex, NULL,
328 "Error invalid cursor dimensions %dx%d\n", w, h);
329
330 if ((bitsp->xhot > w) || (bitsp->yhot > h))
331 RETERROR(scrnIndex, NULL,
332 "Error invalid cursor hotspot location %dx%d (max %dx%d)\n",
333 bitsp->xhot, bitsp->yhot, w, h);
334
335 srcPitch = PixmapBytePad (bitsp->width, 1);
336 dstPitch = (w + 7) / 8;
337 sizeMask = ((dstPitch * h) + 3) & (size_t) ~3;
338 sizeRgba = w * h * 4;
339 sizeRequest = sizeMask + sizeRgba + sizeof(*pImage);
340
341 p = c = calloc (1, sizeRequest);
342 if (!c)
343 RETERROR(scrnIndex, NULL,
344 "Error failed to alloc %lu bytes for cursor\n",
345 (unsigned long) sizeRequest);
346
347 pImage = (struct vboxCursorImage *)p;
348 pImage->pPixels = m = p + sizeof(*pImage);
349 cp = (CARD32 *)(m + sizeMask);
350
351 TRACE_LOG ("w=%d h=%d sm=%d sr=%d p=%d\n",
352 w, h, (int) sizeMask, (int) sizeRgba, (int) dstPitch);
353 TRACE_LOG ("m=%p c=%p cp=%p\n", m, c, (void *)cp);
354
355 fc = color_to_byte (pCurs->foreBlue)
356 | (color_to_byte (pCurs->foreGreen) << 8)
357 | (color_to_byte (pCurs->foreRed) << 16);
358
359 bc = color_to_byte (pCurs->backBlue)
360 | (color_to_byte (pCurs->backGreen) << 8)
361 | (color_to_byte (pCurs->backRed) << 16);
362
363 /*
364 * Convert the Xorg source/mask bits to the and/xor bits VBox needs.
365 * Xorg:
366 * The mask is a bitmap indicating which parts of the cursor are
367 * transparent and which parts are drawn. The source is a bitmap
368 * indicating which parts of the non-transparent portion of the
369 * the cursor should be painted in the foreground color and which
370 * should be painted in the background color. By default, set bits
371 * indicate the opaque part of the mask bitmap and clear bits
372 * indicate the transparent part.
373 * VBox:
374 * The color data is the XOR mask. The AND mask bits determine
375 * which pixels of the color data (XOR mask) will replace (overwrite)
376 * the screen pixels (AND mask bit = 0) and which ones will be XORed
377 * with existing screen pixels (AND mask bit = 1).
378 * For example when you have the AND mask all 0, then you see the
379 * correct mouse pointer image surrounded by black square.
380 */
381 for (pm = bitsp->mask, ps = bitsp->source, y = 0;
382 y < h;
383 ++y, pm += srcPitch, ps += srcPitch, m += dstPitch)
384 {
385 for (x = 0; x < w; ++x)
386 {
387 if (pm[x / 8] & (1 << (x % 8)))
388 {
389 /* opaque, leave AND mask bit at 0 */
390 if (ps[x / 8] & (1 << (x % 8)))
391 {
392 *cp++ = fc;
393 PUT_PIXEL('X');
394 }
395 else
396 {
397 *cp++ = bc;
398 PUT_PIXEL('*');
399 }
400 }
401 else
402 {
403 /* transparent, set AND mask bit */
404 m[x / 8] |= 1 << (7 - (x % 8));
405 /* don't change the screen pixel */
406 *cp++ = 0;
407 PUT_PIXEL(' ');
408 }
409 }
410 PUT_PIXEL('\n');
411 }
412
413 pImage->cWidth = w;
414 pImage->cHeight = h;
415 pImage->cHotX = bitsp->xhot;
416 pImage->cHotY = bitsp->yhot;
417 pImage->fFlags = VBOX_MOUSE_POINTER_VISIBLE | VBOX_MOUSE_POINTER_SHAPE;
418 pImage->cbLength = sizeRequest - sizeof(*pImage);
419
420#ifdef DEBUG_POINTER
421 ErrorF("shape = %p\n", p);
422 vbox_show_shape(w, h, bc, c);
423#endif
424
425 return p;
426}
427
428#ifdef ARGB_CURSOR
429static Bool
430vbox_use_hw_cursor_argb(ScreenPtr pScreen, CursorPtr pCurs)
431{
432 ScrnInfoPtr pScrn = xf86Screens[pScreen->myNum];
433 Bool rc = TRUE;
434
435 if (!vbox_host_uses_hwcursor(pScrn))
436 rc = FALSE;
437 if ( rc
438 && ( (pCurs->bits->height > VBOX_MAX_CURSOR_HEIGHT)
439 || (pCurs->bits->width > VBOX_MAX_CURSOR_WIDTH)
440 || (pScrn->bitsPerPixel <= 8)
441 )
442 )
443 rc = FALSE;
444#ifndef VBOXVIDEO_13
445 /* Evil hack - we use this as another way of poking the driver to update
446 * our list of video modes. */
447 vboxWriteHostModes(pScrn, pScrn->currentMode);
448#endif
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 rc = 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