VirtualBox

source: vbox/trunk/src/VBox/Additions/x11/xgraphics/vboxutils.c@ 14959

Last change on this file since 14959 was 13835, checked in by vboxsync, 16 years ago

s/VBOX_SUCCESS/RT_SUCCESS/g s/VBOX_FAILURE/RT_FAILURE/g - VBOX_SUCCESS and VBOX_FAILURE have *NOT* been retired (because old habbits die hard) just sligtly deprecated.

  • Property svn:eol-style set to native
File size: 33.6 KB
Line 
1/** @file
2 * VirtualBox X11 Additions graphics driver utility functions
3 */
4
5/*
6 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
17 * Clara, CA 95054 USA or visit http://www.sun.com if you need
18 * additional information or have any questions.
19 */
20
21#include <VBox/VBoxGuest.h>
22#include <VBox/VBoxDev.h>
23
24#ifndef PCIACCESS
25# include <xf86Pci.h>
26# include <Pci.h>
27#endif
28
29#include "xf86.h"
30#define NEED_XF86_TYPES
31#ifdef NO_ANSIC
32# include <string.h>
33#else
34# include "xf86_ansic.h"
35#endif
36#include "compiler.h"
37#include "cursorstr.h"
38
39#include "vboxvideo.h"
40
41#define VBOX_MAX_CURSOR_WIDTH 64
42#define VBOX_MAX_CURSOR_HEIGHT 64
43
44/**************************************************************************
45* Debugging functions and macros *
46**************************************************************************/
47
48/* #define DEBUG_POINTER */
49
50#ifdef DEBUG_michael
51# define DEBUG_VIDEO
52#endif
53
54#define BOOL_STR(a) ((a) ? "TRUE" : "FALSE")
55
56#ifdef DEBUG_VIDEO
57# define TRACE_ENTRY() \
58do { \
59 xf86Msg(X_INFO, __PRETTY_FUNCTION__); \
60 xf86Msg(X_INFO, ": entering\n"); \
61} while(0)
62# define TRACE_EXIT() \
63do { \
64 xf86Msg(X_INFO, __PRETTY_FUNCTION__); \
65 xf86Msg(X_INFO, ": leaving\n"); \
66} while(0)
67# define TRACE_LOG(...) \
68do { \
69 xf86Msg(X_INFO, __PRETTY_FUNCTION__); \
70 xf86Msg(X_INFO, __VA_ARGS__); \
71} while(0)
72# define TRACE_LINE() do \
73 { \
74 ErrorF ("%s: line %d\n", __FUNCTION__, __LINE__); \
75 } while(0)
76# define PUT_PIXEL(c) ErrorF ("%c", c)
77#else /* DEBUG_VIDEO not defined */
78# define PUT_PIXEL(c) do { } while(0)
79# define TRACE_ENTRY() do { } while(0)
80# define TRACE_EXIT() do { } while(0)
81# define TRACE_LOG(...) do { } while(0)
82# define TRACE_LINE() do { } while(0)
83#endif /* DEBUG_VIDEO not defined */
84
85/** Macro to printf an error message and return from a function */
86#define RETERROR(scrnIndex, RetVal, ...) \
87 do \
88 { \
89 xf86DrvMsg(scrnIndex, X_ERROR, __VA_ARGS__); \
90 return RetVal; \
91 } \
92 while (0)
93
94#ifdef DEBUG_POINTER
95static void
96vbox_show_shape(unsigned short w, unsigned short h, CARD32 bg, unsigned char *image)
97{
98 size_t x, y;
99 unsigned short pitch;
100 CARD32 *color;
101 unsigned char *mask;
102 size_t sizeMask;
103
104 image += offsetof(VMMDevReqMousePointer, pointerData);
105 mask = image;
106 pitch = (w + 7) / 8;
107 sizeMask = (pitch * h + 3) & ~3;
108 color = (CARD32 *)(image + sizeMask);
109
110 TRACE_ENTRY();
111 for (y = 0; y < h; ++y, mask += pitch, color += w)
112 {
113 for (x = 0; x < w; ++x)
114 {
115 if (mask[x / 8] & (1 << (7 - (x % 8))))
116 ErrorF (" ");
117 else
118 {
119 CARD32 c = color[x];
120 if (c == bg)
121 ErrorF("Y");
122 else
123 ErrorF("X");
124 }
125 }
126 ErrorF("\n");
127 }
128}
129#endif
130
131/**************************************************************************
132* Helper functions and macros *
133**************************************************************************/
134
135static Bool
136vbox_host_uses_hwcursor(ScrnInfoPtr pScrn)
137{
138 Bool rc = FALSE;
139 uint32_t fFeatures = 0;
140
141 TRACE_ENTRY();
142 int vrc = VbglR3GetMouseStatus(&fFeatures, NULL, NULL);
143 if (RT_FAILURE(vrc))
144 xf86DrvMsg(pScrn->scrnIndex, X_ERROR,
145 "Unable to determine whether the virtual machine supports mouse pointer integration - request initialization failed with return code %d\n", vrc);
146 else
147 {
148 if ( !(fFeatures & VBOXGUEST_MOUSE_HOST_CANNOT_HWPOINTER)
149 && (fFeatures & VBOXGUEST_MOUSE_GUEST_CAN_ABSOLUTE)
150 && (fFeatures & VBOXGUEST_MOUSE_HOST_CAN_ABSOLUTE)
151 )
152 rc = TRUE;
153 }
154 TRACE_LOG("rc=%s\n", BOOL_STR(rc));
155 return rc;
156}
157
158/**
159 * This function checks whether the X server agrees with us about whether
160 * to use a host or a guest-drawn cursor and gives it a nudge if it doesn't.
161 * Disabling and re-enabling framebuffer access was one of the few
162 * reliable (although not particularly nice) methods I could find to
163 * force the server to recheck whether to use a hardware or a software
164 * cursor.
165 */
166static void
167vboxRecheckHWCursor(ScrnInfoPtr pScrn)
168{
169 int vrc;
170 uint32_t fFeatures;
171 VBOXPtr pVBox = pScrn->driverPrivate;
172
173 TRACE_ENTRY();
174 /* Check whether we are using the hardware cursor or not, and whether this
175 has changed since the last time we checked. */
176 vrc = VbglR3GetMouseStatus(&fFeatures, NULL, NULL);
177 if (!!(fFeatures & VBOXGUEST_MOUSE_HOST_CAN_ABSOLUTE) != pVBox->usingHWCursor)
178 {
179 pVBox->usingHWCursor = !!(fFeatures & VBOXGUEST_MOUSE_HOST_CAN_ABSOLUTE);
180 /* This triggers a cursor image reload */
181 if (pVBox->accessEnabled)
182 {
183 pScrn->EnableDisableFBAccess(pScrn->scrnIndex, FALSE);
184 pScrn->EnableDisableFBAccess(pScrn->scrnIndex, TRUE);
185 }
186 }
187 TRACE_EXIT();
188}
189
190/**
191 * Macro to disable VBVA extensions and return, for use when an
192 * unexplained error occurs.
193 */
194#define DISABLE_VBVA_AND_RETURN(pScrn, ...) \
195 do \
196 { \
197 xf86DrvMsg(pScrn->scrnIndex, X_ERROR, __VA_ARGS__); \
198 vboxDisableVbva(pScrn); \
199 pVBox->useVbva = FALSE; \
200 return; \
201 } \
202 while (0)
203
204/**************************************************************************
205* Main functions *
206**************************************************************************/
207
208void
209vbox_close(ScrnInfoPtr pScrn, VBOXPtr pVBox)
210{
211 TRACE_ENTRY();
212
213 xfree (pVBox->reqp);
214 pVBox->reqp = NULL;
215 TRACE_EXIT();
216}
217
218/**
219 * Callback function called by the X server to tell us about dirty
220 * rectangles in the video buffer.
221 *
222 * @param pScreen pointer to the information structure for the current
223 * screen
224 * @param iRects Number of dirty rectangles to update
225 * @param aRects Array of structures containing the coordinates of the
226 * rectangles
227 */
228static void
229vboxHandleDirtyRect(ScrnInfoPtr pScrn, int iRects, BoxPtr aRects)
230{
231 VBVACMDHDR cmdHdr;
232 VBOXPtr pVBox;
233 VBVARECORD *pRecord;
234 VBVAMEMORY *pMem;
235 CARD32 indexRecordNext;
236 CARD32 off32Data;
237 CARD32 off32Free;
238 INT32 i32Diff;
239 CARD32 cbHwBufferAvail;
240 int scrnIndex;
241 int i;
242
243 pVBox = pScrn->driverPrivate;
244 TRACE_ENTRY();
245 if (pVBox->useVbva == FALSE)
246 return;
247 pMem = pVBox->pVbvaMemory;
248 /* Just return quietly if VBVA is not currently active. */
249 if ((pMem->fu32ModeFlags & VBVA_F_MODE_ENABLED) == 0)
250 return;
251 scrnIndex = pScrn->scrnIndex;
252
253 for (i = 0; i < iRects; i++)
254 {
255 cmdHdr.x = (int16_t)aRects[i].x1;
256 cmdHdr.y = (int16_t)aRects[i].y1;
257 cmdHdr.w = (uint16_t)(aRects[i].x2 - aRects[i].x1);
258 cmdHdr.h = (uint16_t)(aRects[i].y2 - aRects[i].y1);
259
260 /* Get the active record and move the pointer along */
261 indexRecordNext = (pMem->indexRecordFree + 1) % VBVA_MAX_RECORDS;
262 if (indexRecordNext == pMem->indexRecordFirst)
263 {
264 /* All slots in the records queue are used. */
265 if (VbglR3VideoAccelFlush() < 0)
266 DISABLE_VBVA_AND_RETURN(pScrn,
267 "Unable to clear the VirtualBox graphics acceleration queue "
268 "- the request to the virtual machine failed. Switching to "
269 "unaccelerated mode.\n");
270 }
271 if (indexRecordNext == pMem->indexRecordFirst)
272 DISABLE_VBVA_AND_RETURN(pScrn,
273 "Failed to clear the VirtualBox graphics acceleration queue. "
274 "Switching to unaccelerated mode.\n");
275 pRecord = &pMem->aRecords[pMem->indexRecordFree];
276 /* Mark the record as being updated. */
277 pRecord->cbRecord = VBVA_F_RECORD_PARTIAL;
278 pMem->indexRecordFree = indexRecordNext;
279 /* Compute how many bytes we have in the ring buffer. */
280 off32Free = pMem->off32Free;
281 off32Data = pMem->off32Data;
282 /* Free is writing position. Data is reading position.
283 * Data == Free means buffer is free.
284 * There must be always gap between free and data when data
285 * are in the buffer.
286 * Guest only changes free, host only changes data.
287 */
288 i32Diff = off32Data - off32Free;
289 cbHwBufferAvail = i32Diff > 0 ? i32Diff : VBVA_RING_BUFFER_SIZE + i32Diff;
290 if (cbHwBufferAvail <= VBVA_RING_BUFFER_THRESHOLD)
291 {
292 if (VbglR3VideoAccelFlush() < 0)
293 DISABLE_VBVA_AND_RETURN(pScrn,
294 "Unable to clear the VirtualBox graphics acceleration queue "
295 "- the request to the virtual machine failed. Switching to "
296 "unaccelerated mode.\n");
297 /* Calculate the free space again. */
298 off32Free = pMem->off32Free;
299 off32Data = pMem->off32Data;
300 i32Diff = off32Data - off32Free;
301 cbHwBufferAvail = i32Diff > 0? i32Diff:
302 VBVA_RING_BUFFER_SIZE + i32Diff;
303 if (cbHwBufferAvail <= VBVA_RING_BUFFER_THRESHOLD)
304 DISABLE_VBVA_AND_RETURN(pScrn,
305 "No space left in the VirtualBox graphics acceleration command buffer, "
306 "despite clearing the queue. Switching to unaccelerated mode.\n");
307 }
308 /* Now copy the data into the buffer */
309 if (off32Free + sizeof(cmdHdr) < VBVA_RING_BUFFER_SIZE)
310 {
311 memcpy(&pMem->au8RingBuffer[off32Free], &cmdHdr, sizeof(cmdHdr));
312 pMem->off32Free = pMem->off32Free + sizeof(cmdHdr);
313 }
314 else
315 {
316 CARD32 u32First = VBVA_RING_BUFFER_SIZE - off32Free;
317 /* The following is impressively ugly! */
318 CARD8 *pu8Second = (CARD8 *)&cmdHdr + u32First;
319 CARD32 u32Second = sizeof(cmdHdr) - u32First;
320 memcpy(&pMem->au8RingBuffer[off32Free], &cmdHdr, u32First);
321 if (u32Second)
322 memcpy(&pMem->au8RingBuffer[0], pu8Second, u32Second);
323 pMem->off32Free = u32Second;
324 }
325 pRecord->cbRecord += sizeof(cmdHdr);
326 /* Mark the record completed. */
327 pRecord->cbRecord &= ~VBVA_F_RECORD_PARTIAL;
328 }
329}
330
331#ifdef PCIACCESS
332/* As of X.org server 1.5, we are using the pciaccess library functions to
333 * access PCI. This structure describes our VMM device. */
334/** Structure describing the VMM device */
335static const struct pci_id_match vboxVMMDevID =
336{ VMMDEV_VENDORID, VMMDEV_DEVICEID, PCI_MATCH_ANY, PCI_MATCH_ANY,
337 0, 0, 0 };
338#endif
339
340/**
341 * Initialise VirtualBox's accelerated video extensions.
342 *
343 * @returns TRUE on success, FALSE on failure
344 */
345static Bool
346vboxInitVbva(int scrnIndex, ScreenPtr pScreen, VBOXPtr pVBox)
347{
348#ifdef PCIACCESS
349 struct pci_device_iterator *devIter = NULL;
350
351 TRACE_ENTRY();
352 pVBox->vmmDevInfo = NULL;
353 devIter = pci_id_match_iterator_create(&vboxVMMDevID);
354 if (devIter)
355 {
356 pVBox->vmmDevInfo = pci_device_next(devIter);
357 pci_iterator_destroy(devIter);
358 }
359 if (pVBox->vmmDevInfo)
360 {
361 if (pci_device_probe(pVBox->vmmDevInfo) != 0)
362 {
363 xf86DrvMsg (scrnIndex, X_ERROR,
364 "Failed to probe VMM device (vendor=%04x, devid=%04x)\n",
365 pVBox->vmmDevInfo->vendor_id,
366 pVBox->vmmDevInfo->device_id);
367 }
368 else
369 {
370 if (pci_device_map_range(pVBox->vmmDevInfo,
371 pVBox->vmmDevInfo->regions[1].base_addr,
372 pVBox->vmmDevInfo->regions[1].size,
373 PCI_DEV_MAP_FLAG_WRITABLE,
374 (void **)&pVBox->pVMMDevMemory) != 0)
375 xf86DrvMsg (scrnIndex, X_ERROR,
376 "Failed to map VMM device range\n");
377 }
378 }
379#else
380 PCITAG pciTag;
381 ADDRESS pciAddress;
382
383 TRACE_ENTRY();
384 /* Locate the device. It should already have been enabled by
385 the kernel driver. */
386 pciTag = pciFindFirst((unsigned) VMMDEV_DEVICEID << 16 | VMMDEV_VENDORID,
387 (CARD32) ~0);
388 if (pciTag == PCI_NOT_FOUND)
389 {
390 xf86DrvMsg(scrnIndex, X_ERROR,
391 "Could not find the VirtualBox base device on the PCI bus.\n");
392 return FALSE;
393 }
394 /* Read the address and size of the second I/O region. */
395 pciAddress = pciReadLong(pciTag, PCI_MAP_REG_START + 4);
396 if (pciAddress == 0 || pciAddress == (CARD32) ~0)
397 RETERROR(scrnIndex, FALSE,
398 "The VirtualBox base device contains an invalid memory address.\n");
399 if (PCI_MAP_IS64BITMEM(pciAddress))
400 RETERROR(scrnIndex, FALSE,
401 "The VirtualBox base device has a 64bit mapping address. "
402 "This is currently not supported.\n");
403 /* Map it. We hardcode the size as X does not export the
404 function needed to determine it. */
405 pVBox->pVMMDevMemory = xf86MapPciMem(scrnIndex, 0, pciTag, pciAddress,
406 sizeof(VMMDevMemory));
407#endif
408 if (pVBox->pVMMDevMemory == NULL)
409 {
410 xf86DrvMsg(scrnIndex, X_ERROR,
411 "Failed to map VirtualBox video extension memory.\n");
412 return FALSE;
413 }
414 pVBox->pVbvaMemory = &pVBox->pVMMDevMemory->vbvaMemory;
415 /* Set up the dirty rectangle handler. Since this seems to be a
416 delicate operation, and removing it doubly so, this will
417 remain in place whether it is needed or not, and will simply
418 return if VBVA is not active. I assume that it will be active
419 most of the time. */
420 if (ShadowFBInit2(pScreen, NULL, vboxHandleDirtyRect) != TRUE)
421 {
422 xf86DrvMsg(scrnIndex, X_ERROR,
423 "Unable to install dirty rectangle handler for VirtualBox graphics acceleration.\n");
424 return FALSE;
425 }
426 return TRUE;
427}
428
429Bool
430vbox_init(int scrnIndex, VBOXPtr pVBox)
431{
432 Bool rc = TRUE;
433 int vrc;
434
435 TRACE_ENTRY();
436 pVBox->useVbva = FALSE;
437 vrc = VbglR3Init();
438 if (RT_FAILURE(vrc))
439 {
440 xf86DrvMsg(scrnIndex, X_ERROR,
441 "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",
442 vrc);
443 rc = FALSE;
444 }
445 pVBox->useDevice = rc;
446 return rc;
447}
448
449Bool
450vbox_open(ScrnInfoPtr pScrn, ScreenPtr pScreen, VBOXPtr pVBox)
451{
452 int rc;
453 void *p;
454 size_t size;
455 int scrnIndex = pScrn->scrnIndex;
456
457 TRACE_ENTRY();
458
459 if (!pVBox->useDevice)
460 return FALSE;
461
462 if (pVBox->reqp)
463 {
464 /* still open, just re-enable VBVA after CloseScreen was called */
465 pVBox->useVbva = vboxInitVbva(scrnIndex, pScreen, pVBox);
466 return TRUE;
467 }
468
469 size = vmmdevGetRequestSize(VMMDevReq_SetPointerShape);
470 p = xcalloc(1, size);
471 if (p)
472 {
473 rc = vmmdevInitRequest(p, VMMDevReq_SetPointerShape);
474 if (RT_SUCCESS(rc))
475 {
476 pVBox->reqp = p;
477 pVBox->pCurs = NULL;
478 pVBox->pointerHeaderSize = size;
479 pVBox->pointerOffscreen = FALSE;
480 pVBox->useVbva = vboxInitVbva(scrnIndex, pScreen, pVBox);
481 return TRUE;
482 }
483 xf86DrvMsg(scrnIndex, X_ERROR, "Could not init VMM request: rc = %d\n", rc);
484 xfree(p);
485 }
486 xf86DrvMsg(scrnIndex, X_ERROR, "Could not allocate %lu bytes for VMM request\n", (unsigned long)size);
487 return FALSE;
488}
489
490Bool
491vbox_device_available(VBOXPtr pVBox)
492{
493 return pVBox->useDevice;
494}
495
496static void
497vbox_vmm_hide_cursor(ScrnInfoPtr pScrn, VBOXPtr pVBox)
498{
499 int rc;
500
501 TRACE_ENTRY();
502 pVBox->reqp->fFlags = 0;
503 rc = VbglR3SetPointerShapeReq(pVBox->reqp);
504 if (RT_FAILURE(rc))
505 xf86DrvMsg(pScrn->scrnIndex, X_ERROR, "Could not hide the virtual mouse pointer.\n");
506}
507
508static void
509vbox_vmm_show_cursor(ScrnInfoPtr pScrn, VBOXPtr pVBox)
510{
511 int rc;
512
513 TRACE_ENTRY();
514 pVBox->reqp->fFlags = VBOX_MOUSE_POINTER_VISIBLE;
515 rc = VbglR3SetPointerShapeReq(pVBox->reqp);
516 if (RT_FAILURE(rc))
517 xf86DrvMsg(pScrn->scrnIndex, X_ERROR, "Could not unhide the virtual mouse pointer.\n");
518}
519
520static void
521vbox_vmm_load_cursor_image(ScrnInfoPtr pScrn, VBOXPtr pVBox,
522 unsigned char *image)
523{
524 int rc;
525 VMMDevReqMousePointer *reqp;
526 reqp = (VMMDevReqMousePointer *)image;
527
528 TRACE_LOG("w=%d h=%d size=%d\n", reqp->width, reqp->height, reqp->header.size);
529#ifdef DEBUG_POINTER
530 vbox_show_shape(reqp->width, reqp->height, 0, image);
531#endif
532
533 rc = VbglR3SetPointerShapeReq(reqp);
534 if (RT_FAILURE(rc))
535 xf86DrvMsg(pScrn->scrnIndex, X_ERROR, "Unable to set the virtual mouse pointer image.\n");
536}
537
538static void
539vbox_set_cursor_colors(ScrnInfoPtr pScrn, int bg, int fg)
540{
541 TRACE_ENTRY();
542
543 NOREF(pScrn);
544 NOREF(bg);
545 NOREF(fg);
546 /* ErrorF("vbox_set_cursor_colors NOT IMPLEMENTED\n"); */
547}
548
549/**
550 * This function is called to set the position of the hardware cursor.
551 * Since we already know the position (exactly where the host pointer is),
552 * we only use this function to poll for whether we need to switch from a
553 * hardware to a software cursor (that is, whether the user has disabled
554 * pointer integration). Sadly we this doesn't work the other way round,
555 * as the server updates the software cursor itself without calling us.
556 */
557static void
558vbox_set_cursor_position(ScrnInfoPtr pScrn, int x, int y)
559{
560 vboxRecheckHWCursor(pScrn);
561 /* don't disable the mouse cursor if we go out of our visible area
562 * since the mouse cursor is drawn by the host anyway */
563#if 0
564 if ( (x < 0 || x > pScrn->pScreen->width)
565 || (y < 0 || y > pScrn->pScreen->height))
566 {
567 if (!pVBox->pointerOffscreen)
568 {
569 pVBox->pointerOffscreen = TRUE;
570 vbox_vmm_hide_cursor(pScrn, pVBox);
571 }
572 }
573 else
574 {
575 if (pVBox->pointerOffscreen)
576 {
577 pVBox->pointerOffscreen = FALSE;
578 vbox_vmm_show_cursor(pScrn, pVBox);
579 }
580 }
581#endif
582}
583
584static void
585vbox_hide_cursor(ScrnInfoPtr pScrn)
586{
587 VBOXPtr pVBox = pScrn->driverPrivate;
588
589 TRACE_ENTRY();
590
591 vbox_vmm_hide_cursor(pScrn, pVBox);
592}
593
594static void
595vbox_show_cursor(ScrnInfoPtr pScrn)
596{
597 VBOXPtr pVBox = pScrn->driverPrivate;
598
599 TRACE_ENTRY();
600
601 vbox_vmm_show_cursor(pScrn, pVBox);
602}
603
604static void
605vbox_load_cursor_image(ScrnInfoPtr pScrn, unsigned char *image)
606{
607 VBOXPtr pVBox = pScrn->driverPrivate;
608
609 TRACE_ENTRY();
610
611 vbox_vmm_load_cursor_image(pScrn, pVBox, image);
612}
613
614static Bool
615vbox_use_hw_cursor(ScreenPtr pScreen, CursorPtr pCurs)
616{
617 ScrnInfoPtr pScrn = xf86Screens[pScreen->myNum];
618 return vbox_host_uses_hwcursor(pScrn);
619}
620
621static unsigned char
622color_to_byte(unsigned c)
623{
624 return (c >> 8) & 0xff;
625}
626
627static unsigned char *
628vbox_realize_cursor(xf86CursorInfoPtr infoPtr, CursorPtr pCurs)
629{
630 VBOXPtr pVBox;
631 CursorBitsPtr bitsp;
632 unsigned short w, h, x, y;
633 unsigned char *c, *p, *pm, *ps, *m;
634 size_t sizeRequest, sizeRgba, sizeMask, srcPitch, dstPitch;
635 CARD32 fc, bc, *cp;
636 int rc, scrnIndex = infoPtr->pScrn->scrnIndex;
637 VMMDevReqMousePointer *reqp;
638
639 TRACE_ENTRY();
640 pVBox = infoPtr->pScrn->driverPrivate;
641 bitsp = pCurs->bits;
642 w = bitsp->width;
643 h = bitsp->height;
644
645 if (!w || !h || w > VBOX_MAX_CURSOR_WIDTH || h > VBOX_MAX_CURSOR_HEIGHT)
646 RETERROR(scrnIndex, NULL,
647 "Error invalid cursor dimensions %dx%d\n", w, h);
648
649 if ((bitsp->xhot > w) || (bitsp->yhot > h))
650 RETERROR(scrnIndex, NULL,
651 "Error invalid cursor hotspot location %dx%d (max %dx%d)\n",
652 bitsp->xhot, bitsp->yhot, w, h);
653
654 srcPitch = PixmapBytePad (bitsp->width, 1);
655 dstPitch = (w + 7) / 8;
656 sizeMask = ((dstPitch * h) + 3) & (size_t) ~3;
657 sizeRgba = w * h * 4;
658 pVBox->pointerSize = sizeMask + sizeRgba;
659 sizeRequest = pVBox->pointerSize + pVBox->pointerHeaderSize;
660
661 p = c = xcalloc (1, sizeRequest);
662 if (!c)
663 RETERROR(scrnIndex, NULL,
664 "Error failed to alloc %lu bytes for cursor\n",
665 (unsigned long) sizeRequest);
666
667 rc = vmmdevInitRequest((VMMDevRequestHeader *)p, VMMDevReq_SetPointerShape);
668 if (RT_FAILURE(rc))
669 {
670 xf86DrvMsg(scrnIndex, X_ERROR, "Could not init VMM request: rc = %d\n", rc);
671 xfree(p);
672 return NULL;
673 }
674
675 m = p + offsetof(VMMDevReqMousePointer, pointerData);
676 cp = (CARD32 *)(m + sizeMask);
677
678 TRACE_LOG ("w=%d h=%d sm=%d sr=%d p=%d\n",
679 w, h, (int) sizeMask, (int) sizeRgba, (int) dstPitch);
680 TRACE_LOG ("m=%p c=%p cp=%p\n", m, c, (void *)cp);
681
682 fc = color_to_byte (pCurs->foreBlue)
683 | (color_to_byte (pCurs->foreGreen) << 8)
684 | (color_to_byte (pCurs->foreRed) << 16);
685
686 bc = color_to_byte (pCurs->backBlue)
687 | (color_to_byte (pCurs->backGreen) << 8)
688 | (color_to_byte (pCurs->backRed) << 16);
689
690 /*
691 * Convert the Xorg source/mask bits to the and/xor bits VBox needs.
692 * Xorg:
693 * The mask is a bitmap indicating which parts of the cursor are
694 * transparent and which parts are drawn. The source is a bitmap
695 * indicating which parts of the non-transparent portion of the
696 * the cursor should be painted in the foreground color and which
697 * should be painted in the background color. By default, set bits
698 * indicate the opaque part of the mask bitmap and clear bits
699 * indicate the transparent part.
700 * VBox:
701 * The color data is the XOR mask. The AND mask bits determine
702 * which pixels of the color data (XOR mask) will replace (overwrite)
703 * the screen pixels (AND mask bit = 0) and which ones will be XORed
704 * with existing screen pixels (AND mask bit = 1).
705 * For example when you have the AND mask all 0, then you see the
706 * correct mouse pointer image surrounded by black square.
707 */
708 for (pm = bitsp->mask, ps = bitsp->source, y = 0;
709 y < h;
710 ++y, pm += srcPitch, ps += srcPitch, m += dstPitch)
711 {
712 for (x = 0; x < w; ++x)
713 {
714 if (pm[x / 8] & (1 << (x % 8)))
715 {
716 /* opaque, leave AND mask bit at 0 */
717 if (ps[x / 8] & (1 << (x % 8)))
718 {
719 *cp++ = fc;
720 PUT_PIXEL('X');
721 }
722 else
723 {
724 *cp++ = bc;
725 PUT_PIXEL('*');
726 }
727 }
728 else
729 {
730 /* transparent, set AND mask bit */
731 m[x / 8] |= 1 << (7 - (x % 8));
732 /* don't change the screen pixel */
733 *cp++ = 0;
734 PUT_PIXEL(' ');
735 }
736 }
737 PUT_PIXEL('\n');
738 }
739
740 reqp = (VMMDevReqMousePointer *)p;
741 reqp->width = w;
742 reqp->height = h;
743 reqp->xHot = bitsp->xhot;
744 reqp->yHot = bitsp->yhot;
745 reqp->fFlags = VBOX_MOUSE_POINTER_SHAPE;
746 reqp->header.size = sizeRequest;
747
748#ifdef DEBUG_POINTER
749 ErrorF("shape = %p\n", p);
750 vbox_show_shape(w, h, bc, c);
751#endif
752
753 return p;
754}
755
756#ifdef ARGB_CURSOR
757static Bool
758vbox_use_hw_cursor_argb(ScreenPtr pScreen, CursorPtr pCurs)
759{
760 ScrnInfoPtr pScrn = xf86Screens[pScreen->myNum];
761 Bool rc = TRUE;
762
763 TRACE_ENTRY();
764 if (!vbox_host_uses_hwcursor(pScrn))
765 rc = FALSE;
766 if ( rc
767 && ( (pCurs->bits->height > VBOX_MAX_CURSOR_HEIGHT)
768 || (pCurs->bits->width > VBOX_MAX_CURSOR_WIDTH)
769 || (pScrn->bitsPerPixel <= 8)
770 )
771 )
772 rc = FALSE;
773 TRACE_LOG("rc=%s\n", BOOL_STR(rc));
774 return rc;
775}
776
777
778static void
779vbox_load_cursor_argb(ScrnInfoPtr pScrn, CursorPtr pCurs)
780{
781 VBOXPtr pVBox;
782 VMMDevReqMousePointer *reqp;
783 CursorBitsPtr bitsp;
784 unsigned short w, h;
785 unsigned short cx, cy;
786 unsigned char *pm;
787 CARD32 *pc;
788 size_t sizeRequest, sizeMask;
789 CARD8 *p;
790 int scrnIndex;
791
792 TRACE_ENTRY();
793 pVBox = pScrn->driverPrivate;
794 bitsp = pCurs->bits;
795 w = bitsp->width;
796 h = bitsp->height;
797 scrnIndex = pScrn->scrnIndex;
798
799 /* Mask must be generated for alpha cursors, that is required by VBox. */
800 /* note: (michael) the next struct must be 32bit aligned. */
801 sizeMask = ((w + 7) / 8 * h + 3) & ~3;
802
803 if (!w || !h || w > VBOX_MAX_CURSOR_WIDTH || h > VBOX_MAX_CURSOR_HEIGHT)
804 RETERROR(scrnIndex, ,
805 "Error invalid cursor dimensions %dx%d\n", w, h);
806
807 if ((bitsp->xhot > w) || (bitsp->yhot > h))
808 RETERROR(scrnIndex, ,
809 "Error invalid cursor hotspot location %dx%d (max %dx%d)\n",
810 bitsp->xhot, bitsp->yhot, w, h);
811
812 pVBox->pointerSize = w * h * 4 + sizeMask;
813 sizeRequest = pVBox->pointerSize + pVBox->pointerHeaderSize;
814 p = xcalloc(1, sizeRequest);
815 if (!p)
816 RETERROR(scrnIndex, ,
817 "Error failed to alloc %lu bytes for cursor\n",
818 (unsigned long)sizeRequest);
819
820 reqp = (VMMDevReqMousePointer *)p;
821 *reqp = *pVBox->reqp;
822 reqp->width = w;
823 reqp->height = h;
824 reqp->xHot = bitsp->xhot;
825 reqp->yHot = bitsp->yhot;
826 reqp->fFlags = VBOX_MOUSE_POINTER_SHAPE | VBOX_MOUSE_POINTER_ALPHA;
827 reqp->header.size = sizeRequest;
828
829 memcpy(p + offsetof(VMMDevReqMousePointer, pointerData) + sizeMask, bitsp->argb, w * h * 4);
830
831 /* Emulate the AND mask. */
832 pm = p + offsetof(VMMDevReqMousePointer, pointerData);
833 pc = bitsp->argb;
834
835 /* Init AND mask to 1 */
836 memset(pm, 0xFF, sizeMask);
837
838 /*
839 * The additions driver must provide the AND mask for alpha cursors. The host frontend
840 * which can handle alpha channel, will ignore the AND mask and draw an alpha cursor.
841 * But if the host does not support ARGB, then it simply uses the AND mask and the color
842 * data to draw a normal color cursor.
843 */
844 for (cy = 0; cy < h; cy++)
845 {
846 unsigned char bitmask = 0x80;
847
848 for (cx = 0; cx < w; cx++, bitmask >>= 1)
849 {
850 if (bitmask == 0)
851 bitmask = 0x80;
852
853 if (pc[cx] >= 0xF0000000)
854 pm[cx / 8] &= ~bitmask;
855 }
856
857 /* Point to next source and dest scans */
858 pc += w;
859 pm += (w + 7) / 8;
860 }
861
862 VbglR3SetPointerShapeReq(reqp);
863 xfree(p);
864}
865#endif
866
867Bool
868vbox_cursor_init(ScreenPtr pScreen)
869{
870 ScrnInfoPtr pScrn = xf86Screens[pScreen->myNum];
871 VBOXPtr pVBox = pScrn->driverPrivate;
872 xf86CursorInfoPtr pCurs;
873 Bool rc;
874
875 TRACE_ENTRY();
876 if (!pVBox->useDevice)
877 return FALSE;
878 /* Initially assume we are using a hardware cursor, but this is
879 updated every time the mouse moves anyway. */
880 pVBox->usingHWCursor = TRUE;
881 pVBox->pCurs = pCurs = xf86CreateCursorInfoRec();
882 if (!pCurs)
883 RETERROR(pScrn->scrnIndex, FALSE,
884 "Failed to create X Window cursor information structures for virtual mouse.\n");
885
886 pCurs->MaxWidth = VBOX_MAX_CURSOR_WIDTH;
887 pCurs->MaxHeight = VBOX_MAX_CURSOR_HEIGHT;
888 pCurs->Flags = HARDWARE_CURSOR_TRUECOLOR_AT_8BPP
889 | HARDWARE_CURSOR_SOURCE_MASK_INTERLEAVE_1
890 | HARDWARE_CURSOR_BIT_ORDER_MSBFIRST;
891
892 pCurs->SetCursorColors = vbox_set_cursor_colors;
893 pCurs->SetCursorPosition = vbox_set_cursor_position;
894 pCurs->LoadCursorImage = vbox_load_cursor_image;
895 pCurs->HideCursor = vbox_hide_cursor;
896 pCurs->ShowCursor = vbox_show_cursor;
897 pCurs->UseHWCursor = vbox_use_hw_cursor;
898 pCurs->RealizeCursor = vbox_realize_cursor;
899
900#ifdef ARGB_CURSOR
901 pCurs->UseHWCursorARGB = vbox_use_hw_cursor_argb;
902 pCurs->LoadCursorARGB = vbox_load_cursor_argb;
903#endif
904
905 rc = xf86InitCursor(pScreen, pCurs);
906 if (rc == TRUE)
907 return TRUE;
908 RETERROR(pScrn->scrnIndex, FALSE, "Failed to enable mouse pointer integration.\n");
909}
910
911/**
912 * Inform VBox that we will supply it with dirty rectangle information
913 * and install the dirty rectangle handler.
914 *
915 * @returns TRUE for success, FALSE for failure
916 * @param pScrn Pointer to a structure describing the X screen in use
917 */
918Bool
919vboxEnableVbva(ScrnInfoPtr pScrn)
920{
921 int rc;
922 int scrnIndex = pScrn->scrnIndex;
923 VBOXPtr pVBox = pScrn->driverPrivate;
924
925 TRACE_ENTRY();
926 if (pVBox->useVbva != TRUE)
927 return FALSE;
928 rc = VbglR3VideoAccelEnable(true);
929 if (RT_FAILURE(rc))
930 {
931 /* Request not accepted - disable for old hosts. */
932 xf86DrvMsg(scrnIndex, X_ERROR,
933 "Unable to activate VirtualBox graphics acceleration "
934 "- the request to the virtual machine failed. "
935 "You may be running an old version of VirtualBox.\n");
936 pVBox->useVbva = FALSE;
937 VbglR3VideoAccelEnable(false);
938 return FALSE;
939 }
940 return TRUE;
941}
942
943/**
944 * Inform VBox that we will stop supplying it with dirty rectangle
945 * information. This function is intended to be called when an X
946 * virtual terminal is disabled, or the X server is terminated.
947 *
948 * @returns TRUE for success, FALSE for failure
949 * @param pScrn Pointer to a structure describing the X screen in use
950 */
951Bool
952vboxDisableVbva(ScrnInfoPtr pScrn)
953{
954 int rc;
955 int scrnIndex = pScrn->scrnIndex;
956 VBOXPtr pVBox = pScrn->driverPrivate;
957
958 TRACE_ENTRY();
959 if (pVBox->useVbva != TRUE) /* Ths function should not have been called */
960 return FALSE;
961 rc = VbglR3VideoAccelEnable(false);
962 if (RT_FAILURE(rc))
963 {
964 xf86DrvMsg(scrnIndex, X_ERROR,
965 "Unable to disable VirtualBox graphics acceleration "
966 "- the request to the virtual machine failed.\n");
967 }
968 else
969 memset(pVBox->pVbvaMemory, 0, sizeof(VBVAMEMORY));
970 return TRUE;
971}
972
973/**
974 * Inform VBox that we are aware of advanced graphics functions
975 * (i.e. dynamic resizing, seamless).
976 *
977 * @returns TRUE for success, FALSE for failure
978 */
979Bool
980vboxEnableGraphicsCap(VBOXPtr pVBox)
981{
982 TRACE_ENTRY();
983 if (!pVBox->useDevice)
984 return FALSE;
985 return RT_SUCCESS(VbglR3SetGuestCaps(VMMDEV_GUEST_SUPPORTS_GRAPHICS, 0));
986}
987
988/**
989 * Inform VBox that we are no longer aware of advanced graphics functions
990 * (i.e. dynamic resizing, seamless).
991 *
992 * @returns TRUE for success, FALSE for failure
993 */
994Bool
995vboxDisableGraphicsCap(VBOXPtr pVBox)
996{
997 TRACE_ENTRY();
998 if (!pVBox->useDevice)
999 return FALSE;
1000 return RT_SUCCESS(VbglR3SetGuestCaps(0, VMMDEV_GUEST_SUPPORTS_GRAPHICS));
1001}
1002
1003/**
1004 * Query the last display change request.
1005 *
1006 * @returns boolean success indicator.
1007 * @param pScrn Pointer to the X screen info structure.
1008 * @param pcx Where to store the horizontal pixel resolution (0 = do not change).
1009 * @param pcy Where to store the vertical pixel resolution (0 = do not change).
1010 * @param pcBits Where to store the bits per pixel (0 = do not change).
1011 * @param iDisplay Where to store the display number the request was for - 0 for the
1012 * primary display, 1 for the first secondary, etc.
1013 */
1014Bool
1015vboxGetDisplayChangeRequest(ScrnInfoPtr pScrn, uint32_t *pcx, uint32_t *pcy,
1016 uint32_t *pcBits, uint32_t *piDisplay)
1017{
1018 VBOXPtr pVBox = pScrn->driverPrivate;
1019 TRACE_ENTRY();
1020 if (!pVBox->useDevice)
1021 return FALSE;
1022 int rc = VbglR3GetLastDisplayChangeRequest(pcx, pcy, pcBits, piDisplay);
1023 if (RT_SUCCESS(rc))
1024 return TRUE;
1025 xf86DrvMsg(pScrn->scrnIndex, X_ERROR, "Failed to obtain the last resolution requested by the guest, rc=%d.\n", rc);
1026 return FALSE;
1027}
1028
1029
1030/**
1031 * Query the host as to whether it likes a specific video mode.
1032 *
1033 * @returns the result of the query
1034 * @param cx the width of the mode being queried
1035 * @param cy the height of the mode being queried
1036 * @param cBits the bpp of the mode being queried
1037 */
1038Bool
1039vboxHostLikesVideoMode(ScrnInfoPtr pScrn, uint32_t cx, uint32_t cy, uint32_t cBits)
1040{
1041 VBOXPtr pVBox = pScrn->driverPrivate;
1042 TRACE_ENTRY();
1043 if (!pVBox->useDevice)
1044 return TRUE; /* If we can't ask the host then we like everything. */
1045 return VbglR3HostLikesVideoMode(cx, cy, cBits);
1046}
1047
1048/**
1049 * Save video mode parameters to the registry.
1050 *
1051 * @returns iprt status value
1052 * @param pszName the name to save the mode parameters under
1053 * @param cx mode width
1054 * @param cy mode height
1055 * @param cBits bits per pixel for the mode
1056 */
1057Bool
1058vboxSaveVideoMode(ScrnInfoPtr pScrn, uint32_t cx, uint32_t cy, uint32_t cBits)
1059{
1060 VBOXPtr pVBox = pScrn->driverPrivate;
1061 TRACE_ENTRY();
1062 if (!pVBox->useDevice)
1063 return FALSE;
1064 return RT_SUCCESS(VbglR3SaveVideoMode("SavedMode", cx, cy, cBits));
1065}
1066
1067/**
1068 * Retrieve video mode parameters from the registry.
1069 *
1070 * @returns iprt status value
1071 * @param pszName the name under which the mode parameters are saved
1072 * @param pcx where to store the mode width
1073 * @param pcy where to store the mode height
1074 * @param pcBits where to store the bits per pixel for the mode
1075 */
1076Bool
1077vboxRetrieveVideoMode(ScrnInfoPtr pScrn, uint32_t *pcx, uint32_t *pcy, uint32_t *pcBits)
1078{
1079 VBOXPtr pVBox = pScrn->driverPrivate;
1080 TRACE_ENTRY();
1081 if (!pVBox->useDevice)
1082 return FALSE;
1083 int rc = VbglR3RetrieveVideoMode("SavedMode", pcx, pcy, pcBits);
1084 if (RT_SUCCESS(rc))
1085 TRACE_LOG("Retrieved a video mode of %dx%dx%d\n", *pcx, *pcy, *pcBits);
1086 else
1087 TRACE_LOG("Failed to retrieve video mode, error %d\n", rc);
1088 return (RT_SUCCESS(rc));
1089}
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