VirtualBox

source: vbox/trunk/src/VBox/Main/MouseImpl.cpp@ 27124

Last change on this file since 27124 was 27061, checked in by vboxsync, 15 years ago

Main/MouseImpl: edge case: do not use VMMDev for mouse reporting if we do not have a relative device

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 20.0 KB
Line 
1/* $Id: MouseImpl.cpp 27061 2010-03-04 23:37:52Z vboxsync $ */
2/** @file
3 * VirtualBox COM class implementation
4 */
5
6/*
7 * Copyright (C) 2006-2008 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 "MouseImpl.h"
23#include "DisplayImpl.h"
24#include "VMMDev.h"
25
26#include "AutoCaller.h"
27#include "Logging.h"
28
29#include <VBox/pdmdrv.h>
30#include <iprt/asm.h>
31#include <VBox/VMMDev.h>
32
33/** @name Mouse device capabilities bitfield
34 * @{ */
35enum
36{
37 /** The mouse device can do relative reporting */
38 MOUSE_DEVCAP_RELATIVE = 1,
39 /** The mouse device can do absolute reporting */
40 MOUSE_DEVCAP_ABSOLUTE = 2
41};
42/** @} */
43
44/**
45 * Mouse driver instance data.
46 */
47struct DRVMAINMOUSE
48{
49 /** Pointer to the mouse object. */
50 Mouse *pMouse;
51 /** Pointer to the driver instance structure. */
52 PPDMDRVINS pDrvIns;
53 /** Pointer to the mouse port interface of the driver/device above us. */
54 PPDMIMOUSEPORT pUpPort;
55 /** Our mouse connector interface. */
56 PDMIMOUSECONNECTOR IConnector;
57 /** The capabilities of this device. */
58 uint32_t u32DevCaps;
59};
60
61
62// constructor / destructor
63/////////////////////////////////////////////////////////////////////////////
64
65DEFINE_EMPTY_CTOR_DTOR (Mouse)
66
67HRESULT Mouse::FinalConstruct()
68{
69 RT_ZERO(mpDrv);
70 fVMMDevCanAbs = false;
71 fVMMDevNeedsHostCursor = false;
72 mLastAbsX = 0x8000;
73 mLastAbsY = 0x8000;
74 mLastButtons = 0;
75 return S_OK;
76}
77
78void Mouse::FinalRelease()
79{
80 uninit();
81}
82
83// public methods only for internal purposes
84/////////////////////////////////////////////////////////////////////////////
85
86/**
87 * Initializes the mouse object.
88 *
89 * @returns COM result indicator
90 * @param parent handle of our parent object
91 */
92HRESULT Mouse::init (Console *parent)
93{
94 LogFlowThisFunc(("\n"));
95
96 ComAssertRet(parent, E_INVALIDARG);
97
98 /* Enclose the state transition NotReady->InInit->Ready */
99 AutoInitSpan autoInitSpan(this);
100 AssertReturn(autoInitSpan.isOk(), E_FAIL);
101
102 unconst(mParent) = parent;
103
104#ifdef RT_OS_L4
105 /* L4 console has no own mouse cursor */
106 uHostCaps = VMMDEV_MOUSE_HOST_CANNOT_HWPOINTER;
107#else
108 uHostCaps = 0;
109#endif
110
111 /* Confirm a successful initialization */
112 autoInitSpan.setSucceeded();
113
114 return S_OK;
115}
116
117/**
118 * Uninitializes the instance and sets the ready flag to FALSE.
119 * Called either from FinalRelease() or by the parent when it gets destroyed.
120 */
121void Mouse::uninit()
122{
123 LogFlowThisFunc(("\n"));
124
125 /* Enclose the state transition Ready->InUninit->NotReady */
126 AutoUninitSpan autoUninitSpan(this);
127 if (autoUninitSpan.uninitDone())
128 return;
129
130 for (unsigned i = 0; i < MOUSE_MAX_DEVICES; ++i)
131 {
132 if (mpDrv[i])
133 mpDrv[i]->pMouse = NULL;
134 mpDrv[i] = NULL;
135 }
136
137#ifdef VBOXBFE_WITHOUT_COM
138 mParent = NULL;
139#else
140 unconst(mParent).setNull();
141#endif
142}
143
144
145// IMouse properties
146/////////////////////////////////////////////////////////////////////////////
147
148int Mouse::getVMMDevMouseCaps(uint32_t *pfCaps)
149{
150 AssertPtrReturn(pfCaps, E_POINTER);
151 VMMDev *pVMMDev = mParent->getVMMDev();
152 ComAssertRet(pVMMDev, E_FAIL);
153 PPDMIVMMDEVPORT pVMMDevPort = pVMMDev->getVMMDevPort();
154 ComAssertRet(pVMMDevPort, E_FAIL);
155
156 int rc = pVMMDevPort->pfnQueryMouseCapabilities(pVMMDevPort, pfCaps);
157 return RT_SUCCESS(rc) ? S_OK : E_FAIL;
158}
159
160int Mouse::setVMMDevMouseCaps(uint32_t fCaps)
161{
162 VMMDev *pVMMDev = mParent->getVMMDev();
163 ComAssertRet(pVMMDev, E_FAIL);
164 PPDMIVMMDEVPORT pVMMDevPort = pVMMDev->getVMMDevPort();
165 ComAssertRet(pVMMDevPort, E_FAIL);
166
167 int rc = pVMMDevPort->pfnSetMouseCapabilities(pVMMDevPort, fCaps);
168 return RT_SUCCESS(rc) ? S_OK : E_FAIL;
169}
170
171/**
172 * Returns whether the current setup can accept absolute mouse
173 * events.
174 *
175 * @returns COM status code
176 * @param absoluteSupported address of result variable
177 */
178STDMETHODIMP Mouse::COMGETTER(AbsoluteSupported) (BOOL *absoluteSupported)
179{
180 if (!absoluteSupported)
181 return E_POINTER;
182
183 AutoCaller autoCaller(this);
184 if (FAILED(autoCaller.rc())) return autoCaller.rc();
185
186 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
187 bool fAbs = false;
188
189 if (fVMMDevCanAbs)
190 fAbs = TRUE;
191
192 for (unsigned i = 0; i < MOUSE_MAX_DEVICES; ++i)
193 if (mpDrv[i] && (mpDrv[i]->u32DevCaps & MOUSE_DEVCAP_ABSOLUTE))
194 fAbs = TRUE;
195
196 *absoluteSupported = fAbs;
197 return S_OK;
198}
199
200/**
201 * Returns whether the current setup can accept relative mouse
202 * events.
203 *
204 * @returns COM status code
205 * @param relativeSupported address of result variable
206 */
207STDMETHODIMP Mouse::COMGETTER(RelativeSupported) (BOOL *relativeSupported)
208{
209 if (!relativeSupported)
210 return E_POINTER;
211
212 AutoCaller autoCaller(this);
213 if (FAILED(autoCaller.rc())) return autoCaller.rc();
214
215 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
216 bool fRel = false;
217
218 for (unsigned i = 0; i < MOUSE_MAX_DEVICES; ++i)
219 if (mpDrv[i] && (mpDrv[i]->u32DevCaps & MOUSE_DEVCAP_RELATIVE))
220 fRel = TRUE;
221
222 *relativeSupported = fRel;
223 return S_OK;
224}
225
226/**
227 * Returns whether the guest can currently draw the mouse cursor itself.
228 *
229 * @returns COM status code
230 * @param pfNeedsHostCursor address of result variable
231 */
232STDMETHODIMP Mouse::COMGETTER(NeedsHostCursor) (BOOL *pfNeedsHostCursor)
233{
234 if (!pfNeedsHostCursor)
235 return E_POINTER;
236
237 AutoCaller autoCaller(this);
238 if (FAILED(autoCaller.rc())) return autoCaller.rc();
239
240 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
241
242 *pfNeedsHostCursor = fVMMDevNeedsHostCursor;
243 return S_OK;
244}
245
246// IMouse methods
247/////////////////////////////////////////////////////////////////////////////
248
249static uint32_t mouseButtonsToPDM(LONG buttonState)
250{
251 uint32_t fButtons = 0;
252 if (buttonState & MouseButtonState_LeftButton)
253 fButtons |= PDMIMOUSEPORT_BUTTON_LEFT;
254 if (buttonState & MouseButtonState_RightButton)
255 fButtons |= PDMIMOUSEPORT_BUTTON_RIGHT;
256 if (buttonState & MouseButtonState_MiddleButton)
257 fButtons |= PDMIMOUSEPORT_BUTTON_MIDDLE;
258 if (buttonState & MouseButtonState_XButton1)
259 fButtons |= PDMIMOUSEPORT_BUTTON_X1;
260 if (buttonState & MouseButtonState_XButton2)
261 fButtons |= PDMIMOUSEPORT_BUTTON_X2;
262 return fButtons;
263}
264
265
266/**
267 * Send a relative event to the mouse device.
268 *
269 * @returns COM status code
270 */
271int Mouse::reportRelEventToMouseDev(int32_t dx, int32_t dy, int32_t dz,
272 int32_t dw, uint32_t fButtons)
273{
274 if (dx || dy || dz || dw || fButtons != mLastButtons)
275 {
276 PPDMIMOUSEPORT pUpPort = NULL;
277 for (unsigned i = 0; !pUpPort && i < MOUSE_MAX_DEVICES; ++i)
278 {
279 if (mpDrv[i] && (mpDrv[i]->u32DevCaps & MOUSE_DEVCAP_RELATIVE))
280 pUpPort = mpDrv[i]->pUpPort;
281 }
282 if (!pUpPort)
283 return S_OK;
284
285 int vrc = pUpPort->pfnPutEvent(pUpPort, dx, dy, dz, dw, fButtons);
286
287 if (RT_FAILURE(vrc))
288 setError(VBOX_E_IPRT_ERROR,
289 tr("Could not send the mouse event to the virtual mouse (%Rrc)"),
290 vrc);
291 AssertRCReturn(vrc, VBOX_E_IPRT_ERROR);
292 }
293 return S_OK;
294}
295
296
297/**
298 * Send an absolute position event to the mouse device.
299 *
300 * @returns COM status code
301 */
302int Mouse::reportAbsEventToMouseDev(uint32_t mouseXAbs, uint32_t mouseYAbs,
303 int32_t dz, int32_t dw, uint32_t fButtons)
304{
305 if ( mouseXAbs != mLastAbsX || mouseYAbs != mLastAbsY
306 || dz || dw || fButtons != mLastButtons)
307 {
308 PPDMIMOUSEPORT pUpPort = NULL;
309 for (unsigned i = 0; !pUpPort && i < MOUSE_MAX_DEVICES; ++i)
310 {
311 if (mpDrv[i] && (mpDrv[i]->u32DevCaps & MOUSE_DEVCAP_ABSOLUTE))
312 pUpPort = mpDrv[i]->pUpPort;
313 }
314 if (!pUpPort)
315 return S_OK;
316
317 int vrc = pUpPort->pfnPutEventAbs(pUpPort, mouseXAbs, mouseYAbs, dz,
318 dw, fButtons);
319 if (RT_FAILURE(vrc))
320 setError(VBOX_E_IPRT_ERROR,
321 tr("Could not send the mouse event to the virtual mouse (%Rrc)"),
322 vrc);
323 AssertRCReturn(vrc, VBOX_E_IPRT_ERROR);
324 }
325 return S_OK;
326}
327
328
329/**
330 * Send an absolute position event to the VMM device.
331 *
332 * @returns COM status code
333 */
334int Mouse::reportAbsEventToVMMDev(uint32_t mouseXAbs, uint32_t mouseYAbs)
335{
336 VMMDev *pVMMDev = mParent->getVMMDev();
337 ComAssertRet(pVMMDev, E_FAIL);
338 PPDMIVMMDEVPORT pVMMDevPort = pVMMDev->getVMMDevPort();
339 ComAssertRet(pVMMDevPort, E_FAIL);
340
341 if (mouseXAbs != mLastAbsX || mouseYAbs != mLastAbsY)
342 {
343 int vrc = pVMMDevPort->pfnSetAbsoluteMouse(pVMMDevPort,
344 mouseXAbs, mouseYAbs);
345 if (RT_FAILURE(vrc))
346 setError(VBOX_E_IPRT_ERROR,
347 tr("Could not send the mouse event to the virtual mouse (%Rrc)"),
348 vrc);
349 AssertRCReturn(vrc, VBOX_E_IPRT_ERROR);
350 }
351 return S_OK;
352}
353
354/**
355 * Send a mouse event.
356 *
357 * @returns COM status code
358 * @param dx X movement
359 * @param dy Y movement
360 * @param dz Z movement
361 * @param buttonState The mouse button state
362 */
363STDMETHODIMP Mouse::PutMouseEvent(LONG dx, LONG dy, LONG dz, LONG dw, LONG buttonState)
364{
365 HRESULT rc = S_OK;
366
367 AutoCaller autoCaller(this);
368 if (FAILED(autoCaller.rc())) return autoCaller.rc();
369
370 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
371
372 LogRel3(("%s: dx=%d, dy=%d, dz=%d, dw=%d\n", __PRETTY_FUNCTION__,
373 dx, dy, dz, dw));
374 /*
375 * This method being called implies that the host no
376 * longer wants to use absolute coordinates. If the VMM
377 * device isn't aware of that yet, tell it.
378 */
379 if (uHostCaps & VMMDEV_MOUSE_HOST_CAN_ABSOLUTE)
380 {
381 uHostCaps &= ~VMMDEV_MOUSE_HOST_CAN_ABSOLUTE;
382 setVMMDevMouseCaps(uHostCaps);
383 }
384
385 uint32_t fButtons = mouseButtonsToPDM(buttonState);
386 rc = reportRelEventToMouseDev(dx, dy, dz, dw, fButtons);
387 if (SUCCEEDED(rc))
388 mLastButtons = fButtons;
389
390 return rc;
391}
392
393/**
394 * Convert an X value in screen co-ordinates to a value from 0 to 0xffff
395 *
396 * @returns COM status value
397 */
398int Mouse::convertDisplayWidth(LONG x, uint32_t *pcX)
399{
400 AssertPtrReturn(pcX, E_POINTER);
401 Display *pDisplay = mParent->getDisplay();
402 ComAssertRet(pDisplay, E_FAIL);
403
404 ULONG displayWidth;
405 int rc = pDisplay->COMGETTER(Width)(&displayWidth);
406 ComAssertComRCRet(rc, rc);
407
408 *pcX = displayWidth ? (x * 0xFFFF) / displayWidth: 0;
409 return S_OK;
410}
411
412/**
413 * Convert a Y value in screen co-ordinates to a value from 0 to 0xffff
414 *
415 * @returns COM status value
416 */
417int Mouse::convertDisplayHeight(LONG y, uint32_t *pcY)
418{
419 AssertPtrReturn(pcY, E_POINTER);
420 Display *pDisplay = mParent->getDisplay();
421 ComAssertRet(pDisplay, E_FAIL);
422
423 ULONG displayHeight;
424 int rc = pDisplay->COMGETTER(Height)(&displayHeight);
425 ComAssertComRCRet(rc, rc);
426
427 *pcY = displayHeight ? (y * 0xFFFF) / displayHeight: 0;
428 return S_OK;
429}
430
431
432/**
433 * Send an absolute mouse event to the VM. This only works
434 * when the required guest support has been installed.
435 *
436 * @returns COM status code
437 * @param x X position (pixel)
438 * @param y Y position (pixel)
439 * @param dz Z movement
440 * @param buttonState The mouse button state
441 */
442STDMETHODIMP Mouse::PutMouseEventAbsolute(LONG x, LONG y, LONG dz, LONG dw,
443 LONG buttonState)
444{
445 AutoCaller autoCaller(this);
446 if (FAILED(autoCaller.rc())) return autoCaller.rc();
447
448 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
449
450 LogRel3(("%s: x=%d, y=%d, dz=%d, dw=%d, buttonState=0x%x\n",
451 __PRETTY_FUNCTION__, x, y, dz, dw, buttonState));
452
453 uint32_t mouseXAbs;
454 HRESULT rc = convertDisplayWidth(x, &mouseXAbs);
455 ComAssertComRCRet(rc, rc);
456 /**
457 * @todo multi-monitor Windows guests expect this to be unbounded.
458 * Understand the issues involved and fix for the rest.
459 */
460 /* if (mouseXAbs > 0xffff)
461 mouseXAbs = mLastAbsX; */
462
463 uint32_t mouseYAbs;
464 rc = convertDisplayHeight(y, &mouseYAbs);
465 ComAssertComRCRet(rc, rc);
466 /* if (mouseYAbs > 0xffff)
467 mouseYAbs = mLastAbsY; */
468
469 uint32_t fButtons = mouseButtonsToPDM(buttonState);
470
471 uint32_t mouseCaps;
472 rc = getVMMDevMouseCaps(&mouseCaps);
473 ComAssertComRCRet(rc, rc);
474
475 /*
476 * This method being called implies that the host wants
477 * to use absolute coordinates. If the VMM device isn't
478 * aware of that yet, tell it.
479 */
480 if (!(mouseCaps & VMMDEV_MOUSE_HOST_CAN_ABSOLUTE))
481 {
482 uHostCaps |= VMMDEV_MOUSE_HOST_CAN_ABSOLUTE;
483 setVMMDevMouseCaps(uHostCaps);
484 }
485
486 if (fVMMDevCanAbs)
487 {
488 /*
489 * Send the absolute mouse position to the VMM device.
490 */
491 rc = reportAbsEventToVMMDev(mouseXAbs, mouseYAbs);
492 /* We may need to send an additional event for button information or
493 * to wake up older guests to the changed absolute co-ordinates. If
494 * the event is a pure wake up one, we make sure it contains some
495 * (possibly phony) event data to make sure it isn't just discarded
496 * on the way. */
497 bool fNeedsJiggle = !(mouseCaps & VMMDEV_MOUSE_GUEST_USES_VMMDEV);
498 if (fNeedsJiggle || fButtons != mLastButtons || dz || dw)
499 {
500 rc = reportRelEventToMouseDev(fNeedsJiggle ? 1 : 0, 0, dz, dw,
501 fButtons);
502 ComAssertComRCRet(rc, rc);
503 }
504 }
505 else
506 rc = reportAbsEventToMouseDev(mouseXAbs, mouseYAbs, dz, dw, fButtons);
507
508 ComAssertComRCRet(rc, rc);
509
510 mLastAbsX = mouseXAbs;
511 mLastAbsY = mouseYAbs;
512
513 mLastButtons = fButtons;
514 return rc;
515}
516
517// private methods
518/////////////////////////////////////////////////////////////////////////////
519
520
521void Mouse::sendMouseCapsNotifications(void)
522{
523 bool fAbsDev = false;
524 bool fRelDev = false;
525 uint32_t u32MouseCaps;
526 for (unsigned i = 0; i < MOUSE_MAX_DEVICES; ++i)
527 if (mpDrv[i])
528 {
529 if (mpDrv[i]->u32DevCaps & MOUSE_DEVCAP_ABSOLUTE)
530 fAbsDev = true;
531 if (mpDrv[i]->u32DevCaps & MOUSE_DEVCAP_RELATIVE)
532 fRelDev = true;
533 }
534 if (RT_SUCCESS(getVMMDevMouseCaps(&u32MouseCaps)))
535 fVMMDevCanAbs = (u32MouseCaps & VMMDEV_MOUSE_GUEST_CAN_ABSOLUTE)
536 && fRelDev;
537 else
538 fVMMDevCanAbs = false;
539 mParent->onMouseCapabilityChange(fAbsDev || fVMMDevCanAbs, fRelDev,
540 fVMMDevNeedsHostCursor);
541 /** @todo if this gets called during device initialisation we get an
542 * error due to VMMDev not being initialised yet. This is currently
543 * worked around in DevPS2 (which doesn't need the capability anyway of
544 * course), but it seems to me that Main should be initialised before
545 * the devices are. */
546 if (fAbsDev && !(uHostCaps & VMMDEV_MOUSE_HOST_HAS_ABS_DEV))
547 uHostCaps |= VMMDEV_MOUSE_HOST_HAS_ABS_DEV;
548 if (!fAbsDev && (uHostCaps & VMMDEV_MOUSE_HOST_HAS_ABS_DEV))
549 uHostCaps &= ~VMMDEV_MOUSE_HOST_HAS_ABS_DEV;
550 setVMMDevMouseCaps(uHostCaps);
551}
552
553
554/**
555 * @interface_method_impl{PDMIMOUSECONNECTOR,pfnReportModes}
556 */
557DECLCALLBACK(void) Mouse::mouseReportModes(PPDMIMOUSECONNECTOR pInterface, bool fRel, bool fAbs)
558{
559 PDRVMAINMOUSE pDrv = RT_FROM_MEMBER(pInterface, DRVMAINMOUSE, IConnector);
560 if (fRel)
561 pDrv->u32DevCaps |= MOUSE_DEVCAP_RELATIVE;
562 else
563 pDrv->u32DevCaps &= ~MOUSE_DEVCAP_RELATIVE;
564 if (fAbs)
565 pDrv->u32DevCaps |= MOUSE_DEVCAP_ABSOLUTE;
566 else
567 pDrv->u32DevCaps &= ~MOUSE_DEVCAP_ABSOLUTE;
568
569 pDrv->pMouse->sendMouseCapsNotifications();
570}
571
572
573/**
574 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
575 */
576DECLCALLBACK(void *) Mouse::drvQueryInterface(PPDMIBASE pInterface, const char *pszIID)
577{
578 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
579 PDRVMAINMOUSE pDrv = PDMINS_2_DATA(pDrvIns, PDRVMAINMOUSE);
580
581 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pDrvIns->IBase);
582 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMOUSECONNECTOR, &pDrv->IConnector);
583 return NULL;
584}
585
586
587/**
588 * Destruct a mouse driver instance.
589 *
590 * @returns VBox status.
591 * @param pDrvIns The driver instance data.
592 */
593DECLCALLBACK(void) Mouse::drvDestruct(PPDMDRVINS pDrvIns)
594{
595 PDRVMAINMOUSE pData = PDMINS_2_DATA(pDrvIns, PDRVMAINMOUSE);
596 LogFlow(("Mouse::drvDestruct: iInstance=%d\n", pDrvIns->iInstance));
597 PDMDRV_CHECK_VERSIONS_RETURN_VOID(pDrvIns);
598
599 if (pData->pMouse)
600 {
601 AutoWriteLock mouseLock(pData->pMouse COMMA_LOCKVAL_SRC_POS);
602 RT_ZERO(pData->pMouse->mpDrv);
603 }
604}
605
606
607/**
608 * Construct a mouse driver instance.
609 *
610 * @copydoc FNPDMDRVCONSTRUCT
611 */
612DECLCALLBACK(int) Mouse::drvConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfg, uint32_t fFlags)
613{
614 PDRVMAINMOUSE pData = PDMINS_2_DATA(pDrvIns, PDRVMAINMOUSE);
615 LogFlow(("drvMainMouse_Construct: iInstance=%d\n", pDrvIns->iInstance));
616 PDMDRV_CHECK_VERSIONS_RETURN(pDrvIns);
617
618 /*
619 * Validate configuration.
620 */
621 if (!CFGMR3AreValuesValid(pCfg, "Object\0"))
622 return VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES;
623 AssertMsgReturn(PDMDrvHlpNoAttach(pDrvIns) == VERR_PDM_NO_ATTACHED_DRIVER,
624 ("Configuration error: Not possible to attach anything to this driver!\n"),
625 VERR_PDM_DRVINS_NO_ATTACH);
626
627 /*
628 * IBase.
629 */
630 pDrvIns->IBase.pfnQueryInterface = Mouse::drvQueryInterface;
631
632 pData->IConnector.pfnReportModes = Mouse::mouseReportModes;
633
634 /*
635 * Get the IMousePort interface of the above driver/device.
636 */
637 pData->pUpPort = (PPDMIMOUSEPORT)pDrvIns->pUpBase->pfnQueryInterface(pDrvIns->pUpBase, PDMIMOUSEPORT_IID);
638 if (!pData->pUpPort)
639 {
640 AssertMsgFailed(("Configuration error: No mouse port interface above!\n"));
641 return VERR_PDM_MISSING_INTERFACE_ABOVE;
642 }
643
644 /*
645 * Get the Mouse object pointer and update the mpDrv member.
646 */
647 void *pv;
648 int rc = CFGMR3QueryPtr(pCfg, "Object", &pv);
649 if (RT_FAILURE(rc))
650 {
651 AssertMsgFailed(("Configuration error: No/bad \"Object\" value! rc=%Rrc\n", rc));
652 return rc;
653 }
654 pData->pMouse = (Mouse *)pv; /** @todo Check this cast! */
655 unsigned cDev;
656 for (cDev = 0; cDev < MOUSE_MAX_DEVICES; ++cDev)
657 if (!pData->pMouse->mpDrv[cDev])
658 {
659 pData->pMouse->mpDrv[cDev] = pData;
660 break;
661 }
662 if (cDev == MOUSE_MAX_DEVICES)
663 return VERR_NO_MORE_HANDLES;
664
665 return VINF_SUCCESS;
666}
667
668
669/**
670 * Main mouse driver registration record.
671 */
672const PDMDRVREG Mouse::DrvReg =
673{
674 /* u32Version */
675 PDM_DRVREG_VERSION,
676 /* szName */
677 "MainMouse",
678 /* szRCMod */
679 "",
680 /* szR0Mod */
681 "",
682 /* pszDescription */
683 "Main mouse driver (Main as in the API).",
684 /* fFlags */
685 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
686 /* fClass. */
687 PDM_DRVREG_CLASS_MOUSE,
688 /* cMaxInstances */
689 ~0,
690 /* cbInstance */
691 sizeof(DRVMAINMOUSE),
692 /* pfnConstruct */
693 Mouse::drvConstruct,
694 /* pfnDestruct */
695 Mouse::drvDestruct,
696 /* pfnRelocate */
697 NULL,
698 /* pfnIOCtl */
699 NULL,
700 /* pfnPowerOn */
701 NULL,
702 /* pfnReset */
703 NULL,
704 /* pfnSuspend */
705 NULL,
706 /* pfnResume */
707 NULL,
708 /* pfnAttach */
709 NULL,
710 /* pfnDetach */
711 NULL,
712 /* pfnPowerOff */
713 NULL,
714 /* pfnSoftReset */
715 NULL,
716 /* u32EndVersion */
717 PDM_DRVREG_VERSION
718};
719/* vi: set tabstop=4 shiftwidth=4 expandtab: */
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