VirtualBox

source: vbox/trunk/src/VBox/Devices/Input/DrvMouseQueue.cpp@ 25127

Last change on this file since 25127 was 22810, checked in by vboxsync, 15 years ago

FE/Qt, Devices/Input, Main, FE/*: upgrade mouse device to an MS IntelliMouse Explorer (five buttons and tilt wheel)

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 10.5 KB
Line 
1/** @file
2 *
3 * VBox input devices:
4 * Mouse queue driver
5 */
6
7/*
8 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22
23
24/*******************************************************************************
25* Header Files *
26*******************************************************************************/
27#define LOG_GROUP LOG_GROUP_DRV_MOUSE_QUEUE
28#include <VBox/pdmdrv.h>
29#include <iprt/assert.h>
30
31#include "Builtins.h"
32
33
34
35/*******************************************************************************
36* Structures and Typedefs *
37*******************************************************************************/
38/**
39 * Mouse queue driver instance data.
40 */
41typedef struct DRVMOUSEQUEUE
42{
43 /** Pointer to the driver instance structure. */
44 PPDMDRVINS pDrvIns;
45 /** Pointer to the mouse port interface of the driver/device above us. */
46 PPDMIMOUSEPORT pUpPort;
47 /** Pointer to the mouse port interface of the driver/device below us. */
48 PPDMIMOUSECONNECTOR pDownConnector;
49 /** Our mouse connector interface. */
50 PDMIMOUSECONNECTOR Connector;
51 /** Our mouse port interface. */
52 PDMIMOUSEPORT Port;
53 /** The queue handle. */
54 PPDMQUEUE pQueue;
55 /** Discard input when this flag is set.
56 * We only accept input when the VM is running. */
57 bool fInactive;
58} DRVMOUSEQUEUE, *PDRVMOUSEQUEUE;
59
60
61/**
62 * Mouse queue item.
63 */
64typedef struct DRVMOUSEQUEUEITEM
65{
66 /** The core part owned by the queue manager. */
67 PDMQUEUEITEMCORE Core;
68 int32_t i32DeltaX;
69 int32_t i32DeltaY;
70 int32_t i32DeltaZ;
71 int32_t i32DeltaW;
72 uint32_t fButtonStates;
73} DRVMOUSEQUEUEITEM, *PDRVMOUSEQUEUEITEM;
74
75
76
77/* -=-=-=-=- IBase -=-=-=-=- */
78
79/**
80 * Queries an interface to the driver.
81 *
82 * @returns Pointer to interface.
83 * @returns NULL if the interface was not supported by the driver.
84 * @param pInterface Pointer to this interface structure.
85 * @param enmInterface The requested interface identification.
86 */
87static DECLCALLBACK(void *) drvMouseQueueQueryInterface(PPDMIBASE pInterface, PDMINTERFACE enmInterface)
88{
89 PPDMDRVINS pDrvIns = PDMIBASE_2_PDMDRV(pInterface);
90 PDRVMOUSEQUEUE pDrv = PDMINS_2_DATA(pDrvIns, PDRVMOUSEQUEUE);
91 switch (enmInterface)
92 {
93 case PDMINTERFACE_BASE:
94 return &pDrvIns->IBase;
95 case PDMINTERFACE_MOUSE_PORT:
96 return &pDrv->Port;
97 case PDMINTERFACE_MOUSE_CONNECTOR:
98 return &pDrv->Connector;
99 default:
100 return NULL;
101 }
102}
103
104
105/* -=-=-=-=- IMousePort -=-=-=-=- */
106
107/** Converts a pointer to DRVMOUSEQUEUE::Port to a DRVMOUSEQUEUE pointer. */
108#define IMOUSEPORT_2_DRVMOUSEQUEUE(pInterface) ( (PDRVMOUSEQUEUE)((char *)(pInterface) - RT_OFFSETOF(DRVMOUSEQUEUE, Port)) )
109
110
111/**
112 * Queues a mouse event.
113 * Because of the event queueing the EMT context requirement is lifted.
114 *
115 * @returns VBox status code.
116 * @param pInterface Pointer to interface structure.
117 * @param i32DeltaX The X delta.
118 * @param i32DeltaY The Y delta.
119 * @param i32DeltaZ The Z delta.
120 * @param fButtonStates The button states.
121 * @thread Any thread.
122 */
123static DECLCALLBACK(int) drvMouseQueuePutEvent(PPDMIMOUSEPORT pInterface, int32_t i32DeltaX, int32_t i32DeltaY, int32_t i32DeltaZ, int32_t i32DeltaW, uint32_t fButtonStates)
124{
125 PDRVMOUSEQUEUE pDrv = IMOUSEPORT_2_DRVMOUSEQUEUE(pInterface);
126 if (pDrv->fInactive)
127 return VINF_SUCCESS;
128
129 PDRVMOUSEQUEUEITEM pItem = (PDRVMOUSEQUEUEITEM)PDMQueueAlloc(pDrv->pQueue);
130 if (pItem)
131 {
132 pItem->i32DeltaX = i32DeltaX;
133 pItem->i32DeltaY = i32DeltaY;
134 pItem->i32DeltaZ = i32DeltaZ;
135 pItem->i32DeltaW = i32DeltaW;
136 pItem->fButtonStates = fButtonStates;
137 PDMQueueInsert(pDrv->pQueue, &pItem->Core);
138 return VINF_SUCCESS;
139 }
140 return VERR_PDM_NO_QUEUE_ITEMS;
141}
142
143
144/* -=-=-=-=- queue -=-=-=-=- */
145
146/**
147 * Queue callback for processing a queued item.
148 *
149 * @returns Success indicator.
150 * If false the item will not be removed and the flushing will stop.
151 * @param pDrvIns The driver instance.
152 * @param pItemCore Pointer to the queue item to process.
153 */
154static DECLCALLBACK(bool) drvMouseQueueConsumer(PPDMDRVINS pDrvIns, PPDMQUEUEITEMCORE pItemCore)
155{
156 PDRVMOUSEQUEUE pThis = PDMINS_2_DATA(pDrvIns, PDRVMOUSEQUEUE);
157 PDRVMOUSEQUEUEITEM pItem = (PDRVMOUSEQUEUEITEM)pItemCore;
158 int rc = pThis->pUpPort->pfnPutEvent(pThis->pUpPort, pItem->i32DeltaX, pItem->i32DeltaY, pItem->i32DeltaZ, pItem->i32DeltaW, pItem->fButtonStates);
159 return RT_SUCCESS(rc);
160}
161
162
163/* -=-=-=-=- driver interface -=-=-=-=- */
164
165/**
166 * Power On notification.
167 *
168 * @returns VBox status.
169 * @param pDrvIns The drive instance data.
170 */
171static DECLCALLBACK(void) drvMouseQueuePowerOn(PPDMDRVINS pDrvIns)
172{
173 PDRVMOUSEQUEUE pThis = PDMINS_2_DATA(pDrvIns, PDRVMOUSEQUEUE);
174 pThis->fInactive = false;
175}
176
177
178/**
179 * Reset notification.
180 *
181 * @returns VBox status.
182 * @param pDrvIns The drive instance data.
183 */
184static DECLCALLBACK(void) drvMouseQueueReset(PPDMDRVINS pDrvIns)
185{
186 //PDRVKBDQUEUE pThis = PDMINS_2_DATA(pDrvIns, PDRVKBDQUEUE);
187 /** @todo purge the queue on reset. */
188}
189
190
191/**
192 * Suspend notification.
193 *
194 * @returns VBox status.
195 * @param pDrvIns The drive instance data.
196 */
197static DECLCALLBACK(void) drvMouseQueueSuspend(PPDMDRVINS pDrvIns)
198{
199 PDRVMOUSEQUEUE pThis = PDMINS_2_DATA(pDrvIns, PDRVMOUSEQUEUE);
200 pThis->fInactive = true;
201}
202
203
204/**
205 * Resume notification.
206 *
207 * @returns VBox status.
208 * @param pDrvIns The drive instance data.
209 */
210static DECLCALLBACK(void) drvMouseQueueResume(PPDMDRVINS pDrvIns)
211{
212 PDRVMOUSEQUEUE pThis = PDMINS_2_DATA(pDrvIns, PDRVMOUSEQUEUE);
213 pThis->fInactive = false;
214}
215
216
217/**
218 * Power Off notification.
219 *
220 * @param pDrvIns The drive instance data.
221 */
222static DECLCALLBACK(void) drvMouseQueuePowerOff(PPDMDRVINS pDrvIns)
223{
224 PDRVMOUSEQUEUE pThis = PDMINS_2_DATA(pDrvIns, PDRVMOUSEQUEUE);
225 pThis->fInactive = true;
226}
227
228
229/**
230 * Construct a mouse driver instance.
231 *
232 * @copydoc FNPDMDRVCONSTRUCT
233 */
234static DECLCALLBACK(int) drvMouseQueueConstruct(PPDMDRVINS pDrvIns, PCFGMNODE pCfgHandle, uint32_t fFlags)
235{
236 PDRVMOUSEQUEUE pDrv = PDMINS_2_DATA(pDrvIns, PDRVMOUSEQUEUE);
237 LogFlow(("drvMouseQueueConstruct: iInstance=%d\n", pDrvIns->iInstance));
238
239 /*
240 * Validate configuration.
241 */
242 if (!CFGMR3AreValuesValid(pCfgHandle, "QueueSize\0Interval\0"))
243 return VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES;
244
245 /*
246 * Init basic data members and interfaces.
247 */
248 pDrv->fInactive = true;
249 /* IBase. */
250 pDrvIns->IBase.pfnQueryInterface = drvMouseQueueQueryInterface;
251 /* IMousePort. */
252 pDrv->Port.pfnPutEvent = drvMouseQueuePutEvent;
253
254 /*
255 * Get the IMousePort interface of the above driver/device.
256 */
257 pDrv->pUpPort = (PPDMIMOUSEPORT)pDrvIns->pUpBase->pfnQueryInterface(pDrvIns->pUpBase, PDMINTERFACE_MOUSE_PORT);
258 if (!pDrv->pUpPort)
259 {
260 AssertMsgFailed(("Configuration error: No mouse port interface above!\n"));
261 return VERR_PDM_MISSING_INTERFACE_ABOVE;
262 }
263
264 /*
265 * Attach driver below and query it's connector interface.
266 */
267 PPDMIBASE pDownBase;
268 int rc = PDMDrvHlpAttach(pDrvIns, fFlags, &pDownBase);
269 if (RT_FAILURE(rc))
270 {
271 AssertMsgFailed(("Failed to attach driver below us! rc=%Rra\n", rc));
272 return rc;
273 }
274 pDrv->pDownConnector = (PPDMIMOUSECONNECTOR)pDownBase->pfnQueryInterface(pDownBase, PDMINTERFACE_MOUSE_CONNECTOR);
275 if (!pDrv->pDownConnector)
276 {
277 AssertMsgFailed(("Configuration error: No mouse connector interface below!\n"));
278 return VERR_PDM_MISSING_INTERFACE_BELOW;
279 }
280
281 /*
282 * Create the queue.
283 */
284 uint32_t cMilliesInterval = 0;
285 rc = CFGMR3QueryU32(pCfgHandle, "Interval", &cMilliesInterval);
286 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
287 cMilliesInterval = 0;
288 else if (RT_FAILURE(rc))
289 {
290 AssertMsgFailed(("Configuration error: 32-bit \"Interval\" -> rc=%Rrc\n", rc));
291 return rc;
292 }
293
294 uint32_t cItems = 0;
295 rc = CFGMR3QueryU32(pCfgHandle, "QueueSize", &cItems);
296 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
297 cItems = 128;
298 else if (RT_FAILURE(rc))
299 {
300 AssertMsgFailed(("Configuration error: 32-bit \"QueueSize\" -> rc=%Rrc\n", rc));
301 return rc;
302 }
303
304 rc = PDMDrvHlpPDMQueueCreate(pDrvIns, sizeof(DRVMOUSEQUEUEITEM), cItems, cMilliesInterval, drvMouseQueueConsumer, "Mouse", &pDrv->pQueue);
305 if (RT_FAILURE(rc))
306 {
307 AssertMsgFailed(("Failed to create driver: cItems=%d cMilliesInterval=%d rc=%Rrc\n", cItems, cMilliesInterval, rc));
308 return rc;
309 }
310
311 return VINF_SUCCESS;
312}
313
314
315/**
316 * Mouse queue driver registration record.
317 */
318const PDMDRVREG g_DrvMouseQueue =
319{
320 /* u32Version */
321 PDM_DRVREG_VERSION,
322 /* szDriverName */
323 "MouseQueue",
324 /* pszDescription */
325 "Mouse queue driver to plug in between the key source and the device to do queueing and inter-thread transport.",
326 /* fFlags */
327 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
328 /* fClass. */
329 PDM_DRVREG_CLASS_MOUSE,
330 /* cMaxInstances */
331 ~0,
332 /* cbInstance */
333 sizeof(DRVMOUSEQUEUE),
334 /* pfnConstruct */
335 drvMouseQueueConstruct,
336 /* pfnDestruct */
337 NULL,
338 /* pfnIOCtl */
339 NULL,
340 /* pfnPowerOn */
341 drvMouseQueuePowerOn,
342 /* pfnReset */
343 drvMouseQueueReset,
344 /* pfnSuspend */
345 drvMouseQueueSuspend,
346 /* pfnResume */
347 drvMouseQueueResume,
348 /* pfnAttach */
349 NULL,
350 /* pfnDetach */
351 NULL,
352 /* pfnPowerOff */
353 drvMouseQueuePowerOff,
354 /* pfnSoftReset */
355 NULL,
356 /* u32EndVersion */
357 PDM_DRVREG_VERSION
358};
359
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