VirtualBox

source: vbox/trunk/src/VBox/Devices/Input/UsbMouse.cpp@ 95644

Last change on this file since 95644 was 95276, checked in by vboxsync, 2 years ago

Doxygen.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 108.1 KB
Line 
1/* $Id: UsbMouse.cpp 95276 2022-06-14 11:20:21Z vboxsync $ */
2/** @file
3 * UsbMouse - USB Human Interface Device Emulation (Mouse).
4 */
5
6/*
7 * Copyright (C) 2007-2022 Oracle Corporation
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
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_USB_MOUSE
23#include <VBox/vmm/pdmusb.h>
24#include <VBox/log.h>
25#include <VBox/err.h>
26#include <iprt/assert.h>
27#include <iprt/critsect.h>
28#include <iprt/mem.h>
29#include <iprt/semaphore.h>
30#include <iprt/string.h>
31#include <iprt/uuid.h>
32#include "VBoxDD.h"
33
34
35/*********************************************************************************************************************************
36* Defined Constants And Macros *
37*********************************************************************************************************************************/
38/** @name USB HID string IDs
39 * @{ */
40#define USBHID_STR_ID_MANUFACTURER 1
41#define USBHID_STR_ID_PRODUCT_M 2
42#define USBHID_STR_ID_PRODUCT_T 3
43#define USBHID_STR_ID_PRODUCT_MT 4
44#define USBHID_STR_ID_PRODUCT_TP 5
45/** @} */
46
47/** @name USB HID specific descriptor types
48 * @{ */
49#define DT_IF_HID_DESCRIPTOR 0x21
50#define DT_IF_HID_REPORT 0x22
51/** @} */
52
53/** @name USB HID vendor and product IDs
54 * @{ */
55#define VBOX_USB_VENDOR 0x80EE
56#define USBHID_PID_MOUSE 0x0020
57#define USBHID_PID_TABLET 0x0021
58#define USBHID_PID_MT_TOUCHSCREEN 0x0022
59#define USBHID_PID_MT_TOUCHPAD 0x0023
60/** @} */
61
62
63/*********************************************************************************************************************************
64* Structures and Typedefs *
65*********************************************************************************************************************************/
66
67/**
68 * The USB HID request state.
69 */
70typedef enum USBHIDREQSTATE
71{
72 /** Invalid status. */
73 USBHIDREQSTATE_INVALID = 0,
74 /** Ready to receive a new read request. */
75 USBHIDREQSTATE_READY,
76 /** Have (more) data for the host. */
77 USBHIDREQSTATE_DATA_TO_HOST,
78 /** Waiting to supply status information to the host. */
79 USBHIDREQSTATE_STATUS,
80 /** The end of the valid states. */
81 USBHIDREQSTATE_END
82} USBHIDREQSTATE;
83
84/**
85 * The device reporting mode.
86 * @todo Use an interface instead of an enum and switches.
87 */
88typedef enum USBHIDMODE
89{
90 /** Relative. */
91 USBHIDMODE_RELATIVE = 0,
92 /** Absolute. */
93 USBHIDMODE_ABSOLUTE,
94 /** Multi-touch Touchscreen. */
95 USBHIDMODE_MT_ABSOLUTE,
96 /** Multi-touch Touchpad. */
97 USBHIDMODE_MT_RELATIVE,
98} USBHIDMODE;
99
100
101/**
102 * Endpoint status data.
103 */
104typedef struct USBHIDEP
105{
106 bool fHalted;
107} USBHIDEP;
108/** Pointer to the endpoint status. */
109typedef USBHIDEP *PUSBHIDEP;
110
111
112/**
113 * A URB queue.
114 */
115typedef struct USBHIDURBQUEUE
116{
117 /** The head pointer. */
118 PVUSBURB pHead;
119 /** Where to insert the next entry. */
120 PVUSBURB *ppTail;
121} USBHIDURBQUEUE;
122/** Pointer to a URB queue. */
123typedef USBHIDURBQUEUE *PUSBHIDURBQUEUE;
124/** Pointer to a const URB queue. */
125typedef USBHIDURBQUEUE const *PCUSBHIDURBQUEUE;
126
127
128/**
129 * Mouse movement accumulator.
130 */
131typedef struct USBHIDM_ACCUM
132{
133 union
134 {
135 struct
136 {
137 uint32_t fButtons;
138 int32_t dx;
139 int32_t dy;
140 int32_t dz;
141 } Relative;
142 struct
143 {
144 uint32_t fButtons;
145 int32_t dz;
146 int32_t dw;
147 uint32_t x;
148 uint32_t y;
149 } Absolute;
150 } u;
151} USBHIDM_ACCUM, *PUSBHIDM_ACCUM;
152
153#define MT_CONTACTS_PER_REPORT 5
154
155#define MT_CONTACT_MAX_COUNT 10
156#define TPAD_CONTACT_MAX_COUNT 5
157
158#define MT_CONTACT_F_IN_CONTACT 0x01
159#define MT_CONTACT_F_IN_RANGE 0x02
160#define MT_CONTACT_F_CONFIDENCE 0x04
161
162#define MT_CONTACT_S_ACTIVE 0x01 /* Contact must be reported to the guest. */
163#define MT_CONTACT_S_CANCELLED 0x02 /* Contact loss must be reported to the guest. */
164#define MT_CONTACT_S_REUSED 0x04 /* Report contact loss for the oldId and then new contact for the id. */
165#define MT_CONTACT_S_DIRTY 0x08 /* Temporary flag used to track already processed elements. */
166
167typedef struct MTCONTACT
168{
169 uint16_t x;
170 uint16_t y;
171 uint8_t id;
172 uint8_t flags;
173 uint8_t status;
174 uint8_t oldId; /* Valid only if MT_CONTACT_S_REUSED is set. */
175} MTCONTACT;
176
177
178/**
179 * The USB HID instance data.
180 */
181typedef struct USBHID
182{
183 /** Pointer back to the PDM USB Device instance structure. */
184 PPDMUSBINS pUsbIns;
185 /** Critical section protecting the device state. */
186 RTCRITSECT CritSect;
187
188 /** The current configuration.
189 * (0 - default, 1 - the one supported configuration, i.e configured.) */
190 uint8_t bConfigurationValue;
191 /** Endpoint 0 is the default control pipe, 1 is the dev->host interrupt one. */
192 USBHIDEP aEps[2];
193 /** The state of the HID (state machine).*/
194 USBHIDREQSTATE enmState;
195
196 /** Pointer movement accumulator. */
197 USBHIDM_ACCUM PtrDelta;
198
199 /** Pending to-host queue.
200 * The URBs waiting here are waiting for data to become available.
201 */
202 USBHIDURBQUEUE ToHostQueue;
203
204 /** Done queue
205 * The URBs stashed here are waiting to be reaped. */
206 USBHIDURBQUEUE DoneQueue;
207 /** Signalled when adding an URB to the done queue and fHaveDoneQueueWaiter
208 * is set. */
209 RTSEMEVENT hEvtDoneQueue;
210
211 /** Someone is waiting on the done queue. */
212 bool fHaveDoneQueueWaiter;
213 /** If device has pending changes. */
214 bool fHasPendingChanges;
215 /** Is this a relative, absolute or multi-touch pointing device? */
216 USBHIDMODE enmMode;
217 /** Tablet coordinate shift factor for old and broken operating systems. */
218 uint8_t u8CoordShift;
219
220 /**
221 * Mouse port - LUN#0.
222 *
223 * @implements PDMIBASE
224 * @implements PDMIMOUSEPORT
225 */
226 struct
227 {
228 /** The base interface for the mouse port. */
229 PDMIBASE IBase;
230 /** The mouse port base interface. */
231 PDMIMOUSEPORT IPort;
232
233 /** The base interface of the attached mouse driver. */
234 R3PTRTYPE(PPDMIBASE) pDrvBase;
235 /** The mouse interface of the attached mouse driver. */
236 R3PTRTYPE(PPDMIMOUSECONNECTOR) pDrv;
237 } Lun0;
238
239 MTCONTACT aCurrentContactState[MT_CONTACT_MAX_COUNT];
240 MTCONTACT aReportingContactState[MT_CONTACT_MAX_COUNT];
241 uint32_t u32LastTouchScanTime;
242 bool fTouchReporting;
243 bool fTouchStateUpdated;
244} USBHID;
245/** Pointer to the USB HID instance data. */
246typedef USBHID *PUSBHID;
247
248#pragma pack(1)
249/**
250 * The USB HID report structure for relative device.
251 */
252typedef struct USBHIDM_REPORT
253{
254 uint8_t fButtons;
255 int8_t dx;
256 int8_t dy;
257 int8_t dz;
258} USBHIDM_REPORT, *PUSBHIDM_REPORT;
259
260/**
261 * The USB HID report structure for absolute device.
262 */
263
264typedef struct USBHIDT_REPORT
265{
266 uint8_t fButtons;
267 int8_t dz;
268 int8_t dw;
269 uint8_t padding;
270 uint16_t x;
271 uint16_t y;
272} USBHIDT_REPORT, *PUSBHIDT_REPORT;
273
274/**
275 * The combined USB HID report union for relative and absolute
276 * devices.
277 */
278typedef union USBHIDTM_REPORT
279{
280 USBHIDM_REPORT m;
281 USBHIDT_REPORT t;
282} USBHIDTM_REPORT, *PUSBHIDTM_REPORT;
283
284/**
285 * The USB HID report structure for the multi-touch device.
286 */
287typedef struct USBHIDMT_REPORT
288{
289 uint8_t idReport;
290 uint8_t cContacts;
291 struct
292 {
293 uint8_t fContact;
294 uint8_t cContact;
295 uint16_t x;
296 uint16_t y;
297 } aContacts[MT_CONTACTS_PER_REPORT];
298 uint32_t u32ScanTime;
299} USBHIDMT_REPORT, *PUSBHIDMT_REPORT;
300
301typedef struct USBHIDMT_REPORT_POINTER
302{
303 uint8_t idReport;
304 uint8_t fButtons;
305 uint16_t x;
306 uint16_t y;
307} USBHIDMT_REPORT_POINTER;
308
309/**
310 * The USB HID report structure for the touchpad device.
311 * It is a superset of the multi-touch report.
312 */
313typedef struct USBHIDTP_REPORT
314{
315 USBHIDMT_REPORT mt;
316 uint8_t buttons; /* Required by Win10, not used. */
317} USBHIDTP_REPORT, *PUSBHIDTP_REPORT;
318
319typedef union USBHIDALL_REPORT
320{
321 USBHIDM_REPORT m;
322 USBHIDT_REPORT t;
323 USBHIDMT_REPORT mt;
324 USBHIDMT_REPORT_POINTER mp;
325 USBHIDTP_REPORT tp;
326} USBHIDALL_REPORT, *PUSBHIDALL_REPORT;
327
328#pragma pack()
329
330
331/*********************************************************************************************************************************
332* Global Variables *
333*********************************************************************************************************************************/
334static const PDMUSBDESCCACHESTRING g_aUsbHidStrings_en_US[] =
335{
336 { USBHID_STR_ID_MANUFACTURER, "VirtualBox" },
337 { USBHID_STR_ID_PRODUCT_M, "USB Mouse" },
338 { USBHID_STR_ID_PRODUCT_T, "USB Tablet" },
339 { USBHID_STR_ID_PRODUCT_MT, "USB Multi-Touch" },
340 { USBHID_STR_ID_PRODUCT_TP, "USB Touchpad" },
341};
342
343static const PDMUSBDESCCACHELANG g_aUsbHidLanguages[] =
344{
345 { 0x0409, RT_ELEMENTS(g_aUsbHidStrings_en_US), g_aUsbHidStrings_en_US }
346};
347
348static const VUSBDESCENDPOINTEX g_aUsbHidMEndpointDescs[] =
349{
350 {
351 {
352 /* .bLength = */ sizeof(VUSBDESCENDPOINT),
353 /* .bDescriptorType = */ VUSB_DT_ENDPOINT,
354 /* .bEndpointAddress = */ 0x81 /* ep=1, in */,
355 /* .bmAttributes = */ 3 /* interrupt */,
356 /* .wMaxPacketSize = */ 4,
357 /* .bInterval = */ 10,
358 },
359 /* .pvMore = */ NULL,
360 /* .pvClass = */ NULL,
361 /* .cbClass = */ 0
362 },
363};
364
365static const VUSBDESCENDPOINTEX g_aUsbHidTEndpointDescs[] =
366{
367 {
368 {
369 /* .bLength = */ sizeof(VUSBDESCENDPOINT),
370 /* .bDescriptorType = */ VUSB_DT_ENDPOINT,
371 /* .bEndpointAddress = */ 0x81 /* ep=1, in */,
372 /* .bmAttributes = */ 3 /* interrupt */,
373 /* .wMaxPacketSize = */ 8,
374 /* .bInterval = */ 10,
375 },
376 /* .pvMore = */ NULL,
377 /* .pvClass = */ NULL,
378 /* .cbClass = */ 0
379 },
380};
381
382static const VUSBDESCENDPOINTEX g_aUsbHidMTEndpointDescs[] =
383{
384 {
385 {
386 /* .bLength = */ sizeof(VUSBDESCENDPOINT),
387 /* .bDescriptorType = */ VUSB_DT_ENDPOINT,
388 /* .bEndpointAddress = */ 0x81 /* ep=1, in */,
389 /* .bmAttributes = */ 3 /* interrupt */,
390 /* .wMaxPacketSize = */ 64,
391 /* .bInterval = */ 10,
392 },
393 /* .pvMore = */ NULL,
394 /* .pvClass = */ NULL,
395 /* .cbClass = */ 0
396 },
397};
398
399static const VUSBDESCENDPOINTEX g_aUsbHidTPEndpointDescs[] =
400{
401 {
402 {
403 /* .bLength = */ sizeof(VUSBDESCENDPOINT),
404 /* .bDescriptorType = */ VUSB_DT_ENDPOINT,
405 /* .bEndpointAddress = */ 0x81 /* ep=1, in */,
406 /* .bmAttributes = */ 3 /* interrupt */,
407 /* .wMaxPacketSize = */ 64,
408 /* .bInterval = */ 10,
409 },
410 /* .pvMore = */ NULL,
411 /* .pvClass = */ NULL,
412 /* .cbClass = */ 0
413 },
414};
415
416/* HID report descriptor (mouse). */
417static const uint8_t g_UsbHidMReportDesc[] =
418{
419 /* Usage Page */ 0x05, 0x01, /* Generic Desktop */
420 /* Usage */ 0x09, 0x02, /* Mouse */
421 /* Collection */ 0xA1, 0x01, /* Application */
422 /* Usage */ 0x09, 0x01, /* Pointer */
423 /* Collection */ 0xA1, 0x00, /* Physical */
424 /* Usage Page */ 0x05, 0x09, /* Button */
425 /* Usage Minimum */ 0x19, 0x01, /* Button 1 */
426 /* Usage Maximum */ 0x29, 0x05, /* Button 5 */
427 /* Logical Minimum */ 0x15, 0x00, /* 0 */
428 /* Logical Maximum */ 0x25, 0x01, /* 1 */
429 /* Report Count */ 0x95, 0x05, /* 5 */
430 /* Report Size */ 0x75, 0x01, /* 1 */
431 /* Input */ 0x81, 0x02, /* Data, Value, Absolute, Bit field */
432 /* Report Count */ 0x95, 0x01, /* 1 */
433 /* Report Size */ 0x75, 0x03, /* 3 (padding bits) */
434 /* Input */ 0x81, 0x03, /* Constant, Value, Absolute, Bit field */
435 /* Usage Page */ 0x05, 0x01, /* Generic Desktop */
436 /* Usage */ 0x09, 0x30, /* X */
437 /* Usage */ 0x09, 0x31, /* Y */
438 /* Usage */ 0x09, 0x38, /* Z (wheel) */
439 /* Logical Minimum */ 0x15, 0x81, /* -127 */
440 /* Logical Maximum */ 0x25, 0x7F, /* +127 */
441 /* Report Size */ 0x75, 0x08, /* 8 */
442 /* Report Count */ 0x95, 0x03, /* 3 */
443 /* Input */ 0x81, 0x06, /* Data, Value, Relative, Bit field */
444 /* End Collection */ 0xC0,
445 /* End Collection */ 0xC0,
446};
447
448/* HID report descriptor (tablet). */
449/* NB: The layout is far from random. Having the buttons and Z axis grouped
450 * together avoids alignment issues. Also, if X/Y is reported first, followed
451 * by buttons/Z, Windows gets phantom Z movement. That is likely a bug in Windows
452 * as OS X shows no such problem. When X/Y is reported last, Windows behaves
453 * properly.
454 */
455
456static const uint8_t g_UsbHidTReportDesc[] =
457{
458 /* Usage Page */ 0x05, 0x01, /* Generic Desktop */
459 /* Usage */ 0x09, 0x02, /* Mouse */
460 /* Collection */ 0xA1, 0x01, /* Application */
461 /* Usage */ 0x09, 0x01, /* Pointer */
462 /* Collection */ 0xA1, 0x00, /* Physical */
463 /* Usage Page */ 0x05, 0x09, /* Button */
464 /* Usage Minimum */ 0x19, 0x01, /* Button 1 */
465 /* Usage Maximum */ 0x29, 0x05, /* Button 5 */
466 /* Logical Minimum */ 0x15, 0x00, /* 0 */
467 /* Logical Maximum */ 0x25, 0x01, /* 1 */
468 /* Report Count */ 0x95, 0x05, /* 5 */
469 /* Report Size */ 0x75, 0x01, /* 1 */
470 /* Input */ 0x81, 0x02, /* Data, Value, Absolute, Bit field */
471 /* Report Count */ 0x95, 0x01, /* 1 */
472 /* Report Size */ 0x75, 0x03, /* 3 (padding bits) */
473 /* Input */ 0x81, 0x03, /* Constant, Value, Absolute, Bit field */
474 /* Usage Page */ 0x05, 0x01, /* Generic Desktop */
475 /* Usage */ 0x09, 0x38, /* Z (wheel) */
476 /* Logical Minimum */ 0x15, 0x81, /* -127 */
477 /* Logical Maximum */ 0x25, 0x7F, /* +127 */
478 /* Report Size */ 0x75, 0x08, /* 8 */
479 /* Report Count */ 0x95, 0x01, /* 1 */
480 /* Input */ 0x81, 0x06, /* Data, Value, Relative, Bit field */
481 /* Usage Page */ 0x05, 0x0C, /* Consumer Devices */
482 /* Usage */ 0x0A, 0x38, 0x02,/* AC Pan (horizontal wheel) */
483 /* Report Count */ 0x95, 0x01, /* 1 */
484 /* Input */ 0x81, 0x06, /* Data, Value, Relative, Bit field */
485 /* Report Size */ 0x75, 0x08, /* 8 (padding byte) */
486 /* Report Count */ 0x95, 0x01, /* 1 */
487 /* Input */ 0x81, 0x03, /* Constant, Value, Absolute, Bit field */
488 /* Usage Page */ 0x05, 0x01, /* Generic Desktop */
489 /* Usage */ 0x09, 0x30, /* X */
490 /* Usage */ 0x09, 0x31, /* Y */
491 /* Logical Minimum */ 0x15, 0x00, /* 0 */
492 /* Logical Maximum */ 0x26, 0xFF,0x7F,/* 0x7fff */
493 /* Physical Minimum */ 0x35, 0x00, /* 0 */
494 /* Physical Maximum */ 0x46, 0xFF,0x7F,/* 0x7fff */
495 /* Report Size */ 0x75, 0x10, /* 16 */
496 /* Report Count */ 0x95, 0x02, /* 2 */
497 /* Input */ 0x81, 0x02, /* Data, Value, Absolute, Bit field */
498 /* End Collection */ 0xC0,
499 /* End Collection */ 0xC0,
500};
501
502/*
503 * Multi-touch device implementation based on "Windows Pointer Device Data Delivery Protocol"
504 * specification.
505 */
506
507#define REPORTID_TOUCH_POINTER 1
508#define REPORTID_TOUCH_EVENT 2
509#define REPORTID_TOUCH_MAX_COUNT 3
510#define REPORTID_TOUCH_QABLOB 4
511#define REPORTID_TOUCH_DEVCONFIG 5
512
513static const uint8_t g_UsbHidMTReportDesc[] =
514{
515/* Usage Page (Digitizer) */ 0x05, 0x0D,
516/* Usage (Touch Screen) */ 0x09, 0x04,
517/* Collection (Application) */ 0xA1, 0x01,
518/* Report ID */ 0x85, REPORTID_TOUCH_EVENT,
519/* Usage Page (Digitizer) */ 0x05, 0x0D,
520/* Usage (Contact count) */ 0x09, 0x54,
521/* Report Size (8) */ 0x75, 0x08,
522/* Logical Minimum (0) */ 0x15, 0x00,
523/* Logical Maximum (12) */ 0x25, 0x0C,
524/* Report Count (1) */ 0x95, 0x01,
525/* Input (Var) */ 0x81, 0x02,
526
527/* MT_CONTACTS_PER_REPORT structs u8TipSwitch, u8ContactIdentifier, u16X, u16Y */
528/* 1 of 5 */
529/* Usage (Finger) */ 0x09, 0x22,
530/* Collection (Logical) */ 0xA1, 0x02,
531/* Usage (Tip Switch) */ 0x09, 0x42,
532/* Logical Minimum (0) */ 0x15, 0x00,
533/* Logical Maximum (1) */ 0x25, 0x01,
534/* Report Size (1) */ 0x75, 0x01,
535/* Report Count (1) */ 0x95, 0x01,
536/* Input (Var) */ 0x81, 0x02,
537
538/* Usage (In Range) */ 0x09, 0x32,
539/* Logical Minimum (0) */ 0x15, 0x00,
540/* Logical Maximum (1) */ 0x25, 0x01,
541/* Report Size (1) */ 0x75, 0x01,
542/* Report Count (1) */ 0x95, 0x01,
543/* Input (Var) */ 0x81, 0x02,
544
545/* Report Count (6) */ 0x95, 0x06,
546/* Input (Cnst,Var) */ 0x81, 0x03,
547
548/* Report Size (8) */ 0x75, 0x08,
549/* Usage (Contact identifier) */ 0x09, 0x51,
550/* Report Count (1) */ 0x95, 0x01,
551/* Logical Minimum (0) */ 0x15, 0x00,
552/* Logical Maximum (32) */ 0x25, 0x20,
553/* Input (Var) */ 0x81, 0x02,
554
555/* Usage Page (Generic Desktop) */ 0x05, 0x01,
556/* Logical Maximum (32K) */ 0x26, 0xFF, 0x7F,
557/* Report Size (16) */ 0x75, 0x10,
558/* Usage (X) */ 0x09, 0x30,
559/* Input (Var) */ 0x81, 0x02,
560
561/* Usage (Y) */ 0x09, 0x31,
562/* Input (Var) */ 0x81, 0x02,
563/* End Collection */ 0xC0,
564/* 2 of 5 */
565/* Usage Page (Digitizer) */ 0x05, 0x0D,
566/* Usage (Finger) */ 0x09, 0x22,
567/* Collection (Logical) */ 0xA1, 0x02,
568/* Usage (Tip Switch) */ 0x09, 0x42,
569/* Logical Minimum (0) */ 0x15, 0x00,
570/* Logical Maximum (1) */ 0x25, 0x01,
571/* Report Size (1) */ 0x75, 0x01,
572/* Report Count (1) */ 0x95, 0x01,
573/* Input (Var) */ 0x81, 0x02,
574/* Usage (In Range) */ 0x09, 0x32,
575/* Logical Minimum (0) */ 0x15, 0x00,
576/* Logical Maximum (1) */ 0x25, 0x01,
577/* Report Size (1) */ 0x75, 0x01,
578/* Report Count (1) */ 0x95, 0x01,
579/* Input (Var) */ 0x81, 0x02,
580/* Report Count (6) */ 0x95, 0x06,
581/* Input (Cnst,Var) */ 0x81, 0x03,
582/* Report Size (8) */ 0x75, 0x08,
583/* Usage (Contact identifier) */ 0x09, 0x51,
584/* Report Count (1) */ 0x95, 0x01,
585/* Logical Minimum (0) */ 0x15, 0x00,
586/* Logical Maximum (32) */ 0x25, 0x20,
587/* Input (Var) */ 0x81, 0x02,
588/* Usage Page (Generic Desktop) */ 0x05, 0x01,
589/* Logical Maximum (32K) */ 0x26, 0xFF, 0x7F,
590/* Report Size (16) */ 0x75, 0x10,
591/* Usage (X) */ 0x09, 0x30,
592/* Input (Var) */ 0x81, 0x02,
593/* Usage (Y) */ 0x09, 0x31,
594/* Input (Var) */ 0x81, 0x02,
595/* End Collection */ 0xC0,
596/* 3 of 5 */
597/* Usage Page (Digitizer) */ 0x05, 0x0D,
598/* Usage (Finger) */ 0x09, 0x22,
599/* Collection (Logical) */ 0xA1, 0x02,
600/* Usage (Tip Switch) */ 0x09, 0x42,
601/* Logical Minimum (0) */ 0x15, 0x00,
602/* Logical Maximum (1) */ 0x25, 0x01,
603/* Report Size (1) */ 0x75, 0x01,
604/* Report Count (1) */ 0x95, 0x01,
605/* Input (Var) */ 0x81, 0x02,
606/* Usage (In Range) */ 0x09, 0x32,
607/* Logical Minimum (0) */ 0x15, 0x00,
608/* Logical Maximum (1) */ 0x25, 0x01,
609/* Report Size (1) */ 0x75, 0x01,
610/* Report Count (1) */ 0x95, 0x01,
611/* Input (Var) */ 0x81, 0x02,
612/* Report Count (6) */ 0x95, 0x06,
613/* Input (Cnst,Var) */ 0x81, 0x03,
614/* Report Size (8) */ 0x75, 0x08,
615/* Usage (Contact identifier) */ 0x09, 0x51,
616/* Report Count (1) */ 0x95, 0x01,
617/* Logical Minimum (0) */ 0x15, 0x00,
618/* Logical Maximum (32) */ 0x25, 0x20,
619/* Input (Var) */ 0x81, 0x02,
620/* Usage Page (Generic Desktop) */ 0x05, 0x01,
621/* Logical Maximum (32K) */ 0x26, 0xFF, 0x7F,
622/* Report Size (16) */ 0x75, 0x10,
623/* Usage (X) */ 0x09, 0x30,
624/* Input (Var) */ 0x81, 0x02,
625/* Usage (Y) */ 0x09, 0x31,
626/* Input (Var) */ 0x81, 0x02,
627/* End Collection */ 0xC0,
628/* 4 of 5 */
629/* Usage Page (Digitizer) */ 0x05, 0x0D,
630/* Usage (Finger) */ 0x09, 0x22,
631/* Collection (Logical) */ 0xA1, 0x02,
632/* Usage (Tip Switch) */ 0x09, 0x42,
633/* Logical Minimum (0) */ 0x15, 0x00,
634/* Logical Maximum (1) */ 0x25, 0x01,
635/* Report Size (1) */ 0x75, 0x01,
636/* Report Count (1) */ 0x95, 0x01,
637/* Input (Var) */ 0x81, 0x02,
638/* Usage (In Range) */ 0x09, 0x32,
639/* Logical Minimum (0) */ 0x15, 0x00,
640/* Logical Maximum (1) */ 0x25, 0x01,
641/* Report Size (1) */ 0x75, 0x01,
642/* Report Count (1) */ 0x95, 0x01,
643/* Input (Var) */ 0x81, 0x02,
644/* Report Count (6) */ 0x95, 0x06,
645/* Input (Cnst,Var) */ 0x81, 0x03,
646/* Report Size (8) */ 0x75, 0x08,
647/* Usage (Contact identifier) */ 0x09, 0x51,
648/* Report Count (1) */ 0x95, 0x01,
649/* Logical Minimum (0) */ 0x15, 0x00,
650/* Logical Maximum (32) */ 0x25, 0x20,
651/* Input (Var) */ 0x81, 0x02,
652/* Usage Page (Generic Desktop) */ 0x05, 0x01,
653/* Logical Maximum (32K) */ 0x26, 0xFF, 0x7F,
654/* Report Size (16) */ 0x75, 0x10,
655/* Usage (X) */ 0x09, 0x30,
656/* Input (Var) */ 0x81, 0x02,
657/* Usage (Y) */ 0x09, 0x31,
658/* Input (Var) */ 0x81, 0x02,
659/* End Collection */ 0xC0,
660/* 5 of 5 */
661/* Usage Page (Digitizer) */ 0x05, 0x0D,
662/* Usage (Finger) */ 0x09, 0x22,
663/* Collection (Logical) */ 0xA1, 0x02,
664/* Usage (Tip Switch) */ 0x09, 0x42,
665/* Logical Minimum (0) */ 0x15, 0x00,
666/* Logical Maximum (1) */ 0x25, 0x01,
667/* Report Size (1) */ 0x75, 0x01,
668/* Report Count (1) */ 0x95, 0x01,
669/* Input (Var) */ 0x81, 0x02,
670/* Usage (In Range) */ 0x09, 0x32,
671/* Logical Minimum (0) */ 0x15, 0x00,
672/* Logical Maximum (1) */ 0x25, 0x01,
673/* Report Size (1) */ 0x75, 0x01,
674/* Report Count (1) */ 0x95, 0x01,
675/* Input (Var) */ 0x81, 0x02,
676/* Report Count (6) */ 0x95, 0x06,
677/* Input (Cnst,Var) */ 0x81, 0x03,
678/* Report Size (8) */ 0x75, 0x08,
679/* Usage (Contact identifier) */ 0x09, 0x51,
680/* Report Count (1) */ 0x95, 0x01,
681/* Logical Minimum (0) */ 0x15, 0x00,
682/* Logical Maximum (32) */ 0x25, 0x20,
683/* Input (Var) */ 0x81, 0x02,
684/* Usage Page (Generic Desktop) */ 0x05, 0x01,
685/* Logical Maximum (32K) */ 0x26, 0xFF, 0x7F,
686/* Report Size (16) */ 0x75, 0x10,
687/* Usage (X) */ 0x09, 0x30,
688/* Input (Var) */ 0x81, 0x02,
689/* Usage (Y) */ 0x09, 0x31,
690/* Input (Var) */ 0x81, 0x02,
691/* End Collection */ 0xC0,
692
693/* Note: "Scan time" usage is required for all touch devices (in 100microseconds units). */
694/* Usage Page (Digitizer) */ 0x05, 0x0D,
695/* Logical Minimum (0) */ 0x17, 0x00, 0x00, 0x00, 0x00,
696/* Logical Maximum (2147483647) */ 0x27, 0xFF, 0xFF, 0xFF, 0x7F,
697/* Report Size (32) */ 0x75, 0x20,
698/* Report Count (1) */ 0x95, 0x01,
699/* Unit Exponent (0) */ 0x55, 0x00,
700/* Unit (None) */ 0x65, 0x00,
701/* Usage (Scan time) */ 0x09, 0x56,
702/* Input (Var) */ 0x81, 0x02,
703
704/* Report ID */ 0x85, REPORTID_TOUCH_MAX_COUNT,
705/* Usage (Contact count maximum) */ 0x09, 0x55,
706/* Usage (Device identifier) */ 0x09, 0x53,
707/* Report Size (8) */ 0x75, 0x08,
708/* Report Count (2) */ 0x95, 0x02,
709/* Logical Maximum (255) */ 0x26, 0xFF, 0x00,
710/* Feature (Var) */ 0xB1, 0x02,
711
712/* Usage Page (Vendor-Defined 1) */ 0x06, 0x00, 0xFF,
713/* Usage (QA blob) */ 0x09, 0xC5,
714/* Report ID */ 0x85, REPORTID_TOUCH_QABLOB,
715/* Logical Minimum (0) */ 0x15, 0x00,
716/* Logical Maximum (255) */ 0x26, 0xFF, 0x00,
717/* Report Size (8) */ 0x75, 0x08,
718/* Report Count (256) */ 0x96, 0x00, 0x01,
719/* Feature (Var) */ 0xB1, 0x02,
720/* End Collection */ 0xC0,
721
722/* Note: the pointer report is required by specification:
723 * "The report descriptor for a multiple input device must include at least
724 * one top-level collection for the primary device and a separate top-level
725 * collection for the mouse."
726 */
727/* Usage Page (Generic Desktop) */ 0x05, 0x01,
728/* Usage (Pointer) */ 0x09, 0x01,
729/* Collection (Application) */ 0xA1, 0x01,
730/* Report ID */ 0x85, REPORTID_TOUCH_POINTER,
731/* Usage (Pointer) */ 0x09, 0x01,
732/* Collection (Logical) */ 0xA1, 0x02,
733/* Usage Page (Button) */ 0x05, 0x09,
734/* Usage Minimum (Button 1) */ 0x19, 0x01,
735/* Usage Maximum (Button 2) */ 0x29, 0x02,
736/* Logical Minimum (0) */ 0x15, 0x00,
737/* Logical Maximum (1) */ 0x25, 0x01,
738/* Report Count (2) */ 0x95, 0x02,
739/* Report Size (1) */ 0x75, 0x01,
740/* Input (Var) */ 0x81, 0x02,
741/* Report Count (1) */ 0x95, 0x01,
742/* Report Size (6) */ 0x75, 0x06,
743/* Input (Cnst,Ary,Abs) */ 0x81, 0x01,
744/* Usage Page (Generic Desktop) */ 0x05, 0x01,
745/* Usage (X) */ 0x09, 0x30,
746/* Usage (Y) */ 0x09, 0x31,
747/* Logical Minimum (0) */ 0x16, 0x00, 0x00,
748/* Logical Maximum (32K) */ 0x26, 0xFF, 0x7F,
749/* Physical Minimum (0) */ 0x36, 0x00, 0x00,
750/* Physical Maximum (32K) */ 0x46, 0xFF, 0x7F,
751/* Unit (None) */ 0x66, 0x00, 0x00,
752/* Report Size (16) */ 0x75, 0x10,
753/* Report Count (2) */ 0x95, 0x02,
754/* Input (Var) */ 0x81, 0x02,
755/* End Collection */ 0xC0,
756/* End Collection */ 0xC0,
757
758/* Usage Page (Digitizer) */ 0x05, 0x0D,
759/* Usage (Device configuration) */ 0x09, 0x0E,
760/* Collection (Application) */ 0xA1, 0x01,
761/* Report ID */ 0x85, REPORTID_TOUCH_DEVCONFIG,
762/* Usage (Device settings) */ 0x09, 0x23,
763/* Collection (Logical) */ 0xA1, 0x02,
764/* Usage (Device mode) */ 0x09, 0x52,
765/* Usage (Device identifier) */ 0x09, 0x53,
766/* Logical Minimum (0) */ 0x15, 0x00,
767/* Logical Maximum (10) */ 0x25, 0x0A,
768/* Report Size (8) */ 0x75, 0x08,
769/* Report Count (2) */ 0x95, 0x02,
770/* Feature (Var) */ 0xB1, 0x02,
771/* End Collection */ 0xC0,
772/* End Collection */ 0xC0
773};
774
775
776#define TOUCHPAD_REPORT_FINGER_USAGE \
777/* Usage Page (Digitizer) */ 0x05, 0x0D, \
778/* Usage (Finger) */ 0x09, 0x22, \
779/* Collection (Logical) */ 0xA1, 0x02, \
780/* Usage (Tip Switch) */ 0x09, 0x42, \
781/* Logical Minimum (0) */ 0x15, 0x00, \
782/* Logical Maximum (1) */ 0x25, 0x01, \
783/* Report Size (1) */ 0x75, 0x01, \
784/* Report Count (1) */ 0x95, 0x01, \
785/* Input (Var) */ 0x81, 0x02, \
786 \
787/* Note: In Range not required */ \
788/* Report Count (1) */ 0x95, 0x01, \
789/* Input (Cnst,Var) */ 0x81, 0x03, \
790 \
791/* Usage (Confidence) */ 0x09, 0x47, \
792/* Logical Minimum (0) */ 0x15, 0x00, \
793/* Logical Maximum (1) */ 0x25, 0x01, \
794/* Report Size (1) */ 0x75, 0x01, \
795/* Report Count (1) */ 0x95, 0x01, \
796/* Input (Var) */ 0x81, 0x02, \
797 \
798/* Report Count (5) */ 0x95, 0x05, \
799/* Input (Cnst,Var) */ 0x81, 0x03, \
800 \
801/* Report Size (8) */ 0x75, 0x08, \
802/* Usage (Contact identifier) */ 0x09, 0x51, \
803/* Report Count (1) */ 0x95, 0x01, \
804/* Logical Minimum (0) */ 0x15, 0x00, \
805/* Logical Maximum (32) */ 0x25, 0x20, \
806/* Input (Var) */ 0x81, 0x02, \
807 \
808/* Usage Page (Generic Desktop) */ 0x05, 0x01, \
809/* Logical Minimum (0) */ 0x15, 0x00, \
810/* Logical Maximum (65535) */ 0x27, 0xFF, 0xFF, 0x00, 0x00, \
811/* Report Size (16) */ 0x75, 0x10, \
812/* Unit Exponent (-2) */ 0x55, 0x0e, \
813/* Unit (Eng Lin: Length (in)) */ 0x65, 0x13, \
814/* Usage (X) */ 0x09, 0x30, \
815/* Physical Minimum (0) */ 0x35, 0x00, \
816/* Physical Maximum (461) */ 0x46, 0xcd, 0x01, \
817/* Report Count (1) */ 0x95, 0x01, \
818/* Input (Var) */ 0x81, 0x02, \
819/* Usage (Y) */ 0x09, 0x31, \
820/* Physical Maximum (346) */ 0x46, 0x5a, 0x01, \
821/* Input (Var) */ 0x81, 0x02, \
822/* End Collection */ 0xC0,
823
824static const uint8_t g_UsbHidTPReportDesc[] =
825 {
826/* Usage Page (Digitizer) */ 0x05, 0x0D,
827/* Usage (Touch Pad) */ 0x09, 0x05,
828/* Collection (Application) */ 0xA1, 0x01,
829/* Report ID */ 0x85, REPORTID_TOUCH_EVENT,
830/* Usage Page (Digitizer) */ 0x05, 0x0D,
831/* Usage (Contact count) */ 0x09, 0x54,
832/* Report Size (8) */ 0x75, 0x08,
833/* Logical Minimum (0) */ 0x15, 0x00,
834/* Logical Maximum (12) */ 0x25, 0x0C,
835/* Report Count (1) */ 0x95, 0x01,
836/* Input (Var) */ 0x81, 0x02,
837
838/* MT_CONTACTS_PER_REPORT structs u8TipSwitch, u8ContactIdentifier, u16X, u16Y */
839TOUCHPAD_REPORT_FINGER_USAGE
840TOUCHPAD_REPORT_FINGER_USAGE
841TOUCHPAD_REPORT_FINGER_USAGE
842TOUCHPAD_REPORT_FINGER_USAGE
843TOUCHPAD_REPORT_FINGER_USAGE
844
845/* Note: "Scan time" usage is required for all touch devices (in 100microseconds units). */
846/* Usage Page (Digitizer) */ 0x05, 0x0D,
847/* Logical Minimum (0) */ 0x17, 0x00, 0x00, 0x00, 0x00,
848/* Logical Maximum (2147483647) */ 0x27, 0xFF, 0xFF, 0xFF, 0x7F,
849/* Report Size (32) */ 0x75, 0x20,
850/* Report Count (1) */ 0x95, 0x01,
851/* Unit Exponent (0) */ 0x55, 0x00,
852/* Unit (None) */ 0x65, 0x00,
853/* Usage (Scan time) */ 0x09, 0x56,
854/* Input (Var) */ 0x81, 0x02,
855
856/* Note: Button required by Windows 10 Precision Touchpad */
857/* Usage Page (Button) */ 0x05, 0x09,
858/* Usage (Button 1) */ 0x09, 0x01,
859/* Logical Maximum (1) */ 0x25, 0x01,
860/* Report Size (1) */ 0x75, 0x01,
861/* Report Count (1) */ 0x95, 0x01,
862/* Input (Var) */ 0x81, 0x02,
863/* Report Count (7) */ 0x95, 0x07,
864/* Input (Cnst,Var) */ 0x81, 0x03,
865
866/* Usage Page (Digitizer) */ 0x05, 0x0D,
867/* Report ID */ 0x85, REPORTID_TOUCH_MAX_COUNT,
868/* Usage (Contact count maximum) */ 0x09, 0x55,
869/* Usage (Device identifier) */ 0x09, 0x53,
870/* Report Size (8) */ 0x75, 0x08,
871/* Report Count (2) */ 0x95, 0x02,
872/* Logical Maximum (255) */ 0x26, 0xFF, 0x00,
873/* Feature (Var) */ 0xB1, 0x02,
874
875/* Usage Page (Vendor-Defined 1) */ 0x06, 0x00, 0xFF,
876/* Usage (QA blob) */ 0x09, 0xC5,
877/* Report ID */ 0x85, REPORTID_TOUCH_QABLOB,
878/* Logical Minimum (0) */ 0x15, 0x00,
879/* Logical Maximum (255) */ 0x26, 0xFF, 0x00,
880/* Report Size (8) */ 0x75, 0x08,
881/* Report Count (256) */ 0x96, 0x00, 0x01,
882/* Feature (Var) */ 0xB1, 0x02,
883/* End Collection */ 0xC0,
884
885/* Note: the pointer report is required by specification:
886 * "The report descriptor for a multiple input device must include at least
887 * one top-level collection for the primary device and a separate top-level
888 * collection for the mouse."
889 */
890/* Usage Page (Generic Desktop) */ 0x05, 0x01,
891/* Usage (Pointer) */ 0x09, 0x01,
892/* Collection (Application) */ 0xA1, 0x01,
893/* Report ID */ 0x85, REPORTID_TOUCH_POINTER,
894/* Usage (Pointer) */ 0x09, 0x01,
895/* Collection (Logical) */ 0xA1, 0x02,
896/* Usage Page (Button) */ 0x05, 0x09,
897/* Usage Minimum (Button 1) */ 0x19, 0x01,
898/* Usage Maximum (Button 2) */ 0x29, 0x02,
899/* Logical Minimum (0) */ 0x15, 0x00,
900/* Logical Maximum (1) */ 0x25, 0x01,
901/* Report Count (2) */ 0x95, 0x02,
902/* Report Size (1) */ 0x75, 0x01,
903/* Input (Var) */ 0x81, 0x02,
904/* Report Count (1) */ 0x95, 0x01,
905/* Report Size (6) */ 0x75, 0x06,
906/* Input (Cnst,Ary,Abs) */ 0x81, 0x01,
907/* Usage Page (Generic Desktop) */ 0x05, 0x01,
908/* Usage (X) */ 0x09, 0x30,
909/* Usage (Y) */ 0x09, 0x31,
910/* Logical Minimum (0) */ 0x16, 0x00, 0x00,
911/* Logical Maximum (32K) */ 0x26, 0xFF, 0x7F,
912/* Physical Minimum (0) */ 0x36, 0x00, 0x00,
913/* Physical Maximum (32K) */ 0x46, 0xFF, 0x7F,
914/* Unit (None) */ 0x66, 0x00, 0x00,
915/* Report Size (16) */ 0x75, 0x10,
916/* Report Count (2) */ 0x95, 0x02,
917/* Input (Var) */ 0x81, 0x02,
918/* End Collection */ 0xC0,
919/* End Collection */ 0xC0,
920
921/* Usage Page (Digitizer) */ 0x05, 0x0D,
922/* Usage (Device configuration) */ 0x09, 0x0E,
923/* Collection (Application) */ 0xA1, 0x01,
924/* Report ID */ 0x85, REPORTID_TOUCH_DEVCONFIG,
925/* Usage (Device settings) */ 0x09, 0x23,
926/* Collection (Logical) */ 0xA1, 0x02,
927/* Usage (Device mode) */ 0x09, 0x52,
928/* Usage (Device identifier) */ 0x09, 0x53,
929/* Logical Minimum (0) */ 0x15, 0x00,
930/* Logical Maximum (10) */ 0x25, 0x0A,
931/* Report Size (8) */ 0x75, 0x08,
932/* Report Count (2) */ 0x95, 0x02,
933/* Feature (Var) */ 0xB1, 0x02,
934/* End Collection */ 0xC0,
935/* End Collection */ 0xC0
936 };
937
938
939/** @todo Do these really have to all be duplicated three times? */
940/* Additional HID class interface descriptor. */
941static const uint8_t g_UsbHidMIfHidDesc[] =
942{
943 /* .bLength = */ 0x09,
944 /* .bDescriptorType = */ 0x21, /* HID */
945 /* .bcdHID = */ 0x10, 0x01, /* 1.1 */
946 /* .bCountryCode = */ 0,
947 /* .bNumDescriptors = */ 1,
948 /* .bDescriptorType = */ 0x22, /* Report */
949 /* .wDescriptorLength = */ sizeof(g_UsbHidMReportDesc), 0x00
950};
951
952/* Additional HID class interface descriptor. */
953static const uint8_t g_UsbHidTIfHidDesc[] =
954{
955 /* .bLength = */ 0x09,
956 /* .bDescriptorType = */ 0x21, /* HID */
957 /* .bcdHID = */ 0x10, 0x01, /* 1.1 */
958 /* .bCountryCode = */ 0,
959 /* .bNumDescriptors = */ 1,
960 /* .bDescriptorType = */ 0x22, /* Report */
961 /* .wDescriptorLength = */ sizeof(g_UsbHidTReportDesc), 0x00
962};
963
964/* Additional HID class interface descriptor. */
965static const uint8_t g_UsbHidMTIfHidDesc[] =
966{
967 /* .bLength = */ 0x09,
968 /* .bDescriptorType = */ 0x21, /* HID */
969 /* .bcdHID = */ 0x10, 0x02, /* 2.1 */
970 /* .bCountryCode = */ 0,
971 /* .bNumDescriptors = */ 1,
972 /* .bDescriptorType = */ 0x22, /* Report */
973 /* .wDescriptorLength = */ (uint8_t)(sizeof(g_UsbHidMTReportDesc) & 0xFF),
974 (uint8_t)((sizeof(g_UsbHidMTReportDesc) >> 8) & 0xFF)
975};
976
977/* Additional HID class interface descriptor. */
978static const uint8_t g_UsbHidTPIfHidDesc[] =
979{
980 /* .bLength = */ 0x09,
981 /* .bDescriptorType = */ 0x21, /* HID */
982 /* .bcdHID = */ 0x10, 0x02, /* 2.1 */
983 /* .bCountryCode = */ 0,
984 /* .bNumDescriptors = */ 1,
985 /* .bDescriptorType = */ 0x22, /* Report */
986 /* .wDescriptorLength = */ (uint8_t)(sizeof(g_UsbHidTPReportDesc) & 0xFF),
987 (uint8_t)((sizeof(g_UsbHidTPReportDesc) >> 8) & 0xFF)
988};
989
990static const VUSBDESCINTERFACEEX g_UsbHidMInterfaceDesc =
991{
992 {
993 /* .bLength = */ sizeof(VUSBDESCINTERFACE),
994 /* .bDescriptorType = */ VUSB_DT_INTERFACE,
995 /* .bInterfaceNumber = */ 0,
996 /* .bAlternateSetting = */ 0,
997 /* .bNumEndpoints = */ 1,
998 /* .bInterfaceClass = */ 3 /* HID */,
999 /* .bInterfaceSubClass = */ 1 /* Boot Interface */,
1000 /* .bInterfaceProtocol = */ 2 /* Mouse */,
1001 /* .iInterface = */ 0
1002 },
1003 /* .pvMore = */ NULL,
1004 /* .pvClass = */ &g_UsbHidMIfHidDesc,
1005 /* .cbClass = */ sizeof(g_UsbHidMIfHidDesc),
1006 &g_aUsbHidMEndpointDescs[0],
1007 /* .pIAD = */ NULL,
1008 /* .cbIAD = */ 0
1009};
1010
1011static const VUSBDESCINTERFACEEX g_UsbHidTInterfaceDesc =
1012{
1013 {
1014 /* .bLength = */ sizeof(VUSBDESCINTERFACE),
1015 /* .bDescriptorType = */ VUSB_DT_INTERFACE,
1016 /* .bInterfaceNumber = */ 0,
1017 /* .bAlternateSetting = */ 0,
1018 /* .bNumEndpoints = */ 1,
1019 /* .bInterfaceClass = */ 3 /* HID */,
1020 /* .bInterfaceSubClass = */ 0 /* No subclass - no boot interface. */,
1021 /* .bInterfaceProtocol = */ 0 /* No protocol - no boot interface. */,
1022 /* .iInterface = */ 0
1023 },
1024 /* .pvMore = */ NULL,
1025 /* .pvClass = */ &g_UsbHidTIfHidDesc,
1026 /* .cbClass = */ sizeof(g_UsbHidTIfHidDesc),
1027 &g_aUsbHidTEndpointDescs[0],
1028 /* .pIAD = */ NULL,
1029 /* .cbIAD = */ 0
1030};
1031
1032static const VUSBDESCINTERFACEEX g_UsbHidMTInterfaceDesc =
1033{
1034 {
1035 /* .bLength = */ sizeof(VUSBDESCINTERFACE),
1036 /* .bDescriptorType = */ VUSB_DT_INTERFACE,
1037 /* .bInterfaceNumber = */ 0,
1038 /* .bAlternateSetting = */ 0,
1039 /* .bNumEndpoints = */ 1,
1040 /* .bInterfaceClass = */ 3 /* HID */,
1041 /* .bInterfaceSubClass = */ 0 /* No subclass - no boot interface. */,
1042 /* .bInterfaceProtocol = */ 0 /* No protocol - no boot interface. */,
1043 /* .iInterface = */ 0
1044 },
1045 /* .pvMore = */ NULL,
1046 /* .pvClass = */ &g_UsbHidMTIfHidDesc,
1047 /* .cbClass = */ sizeof(g_UsbHidMTIfHidDesc),
1048 &g_aUsbHidMTEndpointDescs[0],
1049 /* .pIAD = */ NULL,
1050 /* .cbIAD = */ 0
1051};
1052
1053static const VUSBDESCINTERFACEEX g_UsbHidTPInterfaceDesc =
1054{
1055 {
1056 /* .bLength = */ sizeof(VUSBDESCINTERFACE),
1057 /* .bDescriptorType = */ VUSB_DT_INTERFACE,
1058 /* .bInterfaceNumber = */ 0,
1059 /* .bAlternateSetting = */ 0,
1060 /* .bNumEndpoints = */ 1,
1061 /* .bInterfaceClass = */ 3 /* HID */,
1062 /* .bInterfaceSubClass = */ 0 /* No subclass - no boot interface. */,
1063 /* .bInterfaceProtocol = */ 0 /* No protocol - no boot interface. */,
1064 /* .iInterface = */ 0
1065 },
1066 /* .pvMore = */ NULL,
1067 /* .pvClass = */ &g_UsbHidTPIfHidDesc,
1068 /* .cbClass = */ sizeof(g_UsbHidTPIfHidDesc),
1069 &g_aUsbHidTPEndpointDescs[0],
1070 /* .pIAD = */ NULL,
1071 /* .cbIAD = */ 0
1072};
1073
1074static const VUSBINTERFACE g_aUsbHidMInterfaces[] =
1075{
1076 { &g_UsbHidMInterfaceDesc, /* .cSettings = */ 1 },
1077};
1078
1079static const VUSBINTERFACE g_aUsbHidTInterfaces[] =
1080{
1081 { &g_UsbHidTInterfaceDesc, /* .cSettings = */ 1 },
1082};
1083
1084static const VUSBINTERFACE g_aUsbHidMTInterfaces[] =
1085{
1086 { &g_UsbHidMTInterfaceDesc, /* .cSettings = */ 1 },
1087};
1088
1089static const VUSBINTERFACE g_aUsbHidTPInterfaces[] =
1090{
1091 { &g_UsbHidTPInterfaceDesc, /* .cSettings = */ 1 },
1092};
1093
1094static const VUSBDESCCONFIGEX g_UsbHidMConfigDesc =
1095{
1096 {
1097 /* .bLength = */ sizeof(VUSBDESCCONFIG),
1098 /* .bDescriptorType = */ VUSB_DT_CONFIG,
1099 /* .wTotalLength = */ 0 /* recalculated on read */,
1100 /* .bNumInterfaces = */ RT_ELEMENTS(g_aUsbHidMInterfaces),
1101 /* .bConfigurationValue =*/ 1,
1102 /* .iConfiguration = */ 0,
1103 /* .bmAttributes = */ RT_BIT(7),
1104 /* .MaxPower = */ 50 /* 100mA */
1105 },
1106 NULL, /* pvMore */
1107 NULL, /* pvClass */
1108 0, /* cbClass */
1109 &g_aUsbHidMInterfaces[0],
1110 NULL /* pvOriginal */
1111};
1112
1113static const VUSBDESCCONFIGEX g_UsbHidTConfigDesc =
1114{
1115 {
1116 /* .bLength = */ sizeof(VUSBDESCCONFIG),
1117 /* .bDescriptorType = */ VUSB_DT_CONFIG,
1118 /* .wTotalLength = */ 0 /* recalculated on read */,
1119 /* .bNumInterfaces = */ RT_ELEMENTS(g_aUsbHidTInterfaces),
1120 /* .bConfigurationValue =*/ 1,
1121 /* .iConfiguration = */ 0,
1122 /* .bmAttributes = */ RT_BIT(7),
1123 /* .MaxPower = */ 50 /* 100mA */
1124 },
1125 NULL, /* pvMore */
1126 NULL, /* pvClass */
1127 0, /* cbClass */
1128 &g_aUsbHidTInterfaces[0],
1129 NULL /* pvOriginal */
1130};
1131
1132static const VUSBDESCCONFIGEX g_UsbHidMTConfigDesc =
1133{
1134 {
1135 /* .bLength = */ sizeof(VUSBDESCCONFIG),
1136 /* .bDescriptorType = */ VUSB_DT_CONFIG,
1137 /* .wTotalLength = */ 0 /* recalculated on read */,
1138 /* .bNumInterfaces = */ RT_ELEMENTS(g_aUsbHidMTInterfaces),
1139 /* .bConfigurationValue =*/ 1,
1140 /* .iConfiguration = */ 0,
1141 /* .bmAttributes = */ RT_BIT(7),
1142 /* .MaxPower = */ 50 /* 100mA */
1143 },
1144 NULL, /* pvMore */
1145 NULL, /* pvClass */
1146 0, /* cbClass */
1147 &g_aUsbHidMTInterfaces[0],
1148 NULL /* pvOriginal */
1149};
1150
1151static const VUSBDESCCONFIGEX g_UsbHidTPConfigDesc =
1152{
1153 {
1154 /* .bLength = */ sizeof(VUSBDESCCONFIG),
1155 /* .bDescriptorType = */ VUSB_DT_CONFIG,
1156 /* .wTotalLength = */ 0 /* recalculated on read */,
1157 /* .bNumInterfaces = */ RT_ELEMENTS(g_aUsbHidTPInterfaces),
1158 /* .bConfigurationValue =*/ 1,
1159 /* .iConfiguration = */ 0,
1160 /* .bmAttributes = */ RT_BIT(7),
1161 /* .MaxPower = */ 50 /* 100mA */
1162 },
1163 NULL, /* pvMore */
1164 NULL, /* pvClass */
1165 0, /* cbClass */
1166 &g_aUsbHidTPInterfaces[0],
1167 NULL /* pvOriginal */
1168};
1169
1170static const VUSBDESCDEVICE g_UsbHidMDeviceDesc =
1171{
1172 /* .bLength = */ sizeof(g_UsbHidMDeviceDesc),
1173 /* .bDescriptorType = */ VUSB_DT_DEVICE,
1174 /* .bcdUsb = */ 0x110, /* 1.1 */
1175 /* .bDeviceClass = */ 0 /* Class specified in the interface desc. */,
1176 /* .bDeviceSubClass = */ 0 /* Subclass specified in the interface desc. */,
1177 /* .bDeviceProtocol = */ 0 /* Protocol specified in the interface desc. */,
1178 /* .bMaxPacketSize0 = */ 8,
1179 /* .idVendor = */ VBOX_USB_VENDOR,
1180 /* .idProduct = */ USBHID_PID_MOUSE,
1181 /* .bcdDevice = */ 0x0100, /* 1.0 */
1182 /* .iManufacturer = */ USBHID_STR_ID_MANUFACTURER,
1183 /* .iProduct = */ USBHID_STR_ID_PRODUCT_M,
1184 /* .iSerialNumber = */ 0,
1185 /* .bNumConfigurations = */ 1
1186};
1187
1188static const VUSBDESCDEVICE g_UsbHidTDeviceDesc =
1189{
1190 /* .bLength = */ sizeof(g_UsbHidTDeviceDesc),
1191 /* .bDescriptorType = */ VUSB_DT_DEVICE,
1192 /* .bcdUsb = */ 0x110, /* 1.1 */
1193 /* .bDeviceClass = */ 0 /* Class specified in the interface desc. */,
1194 /* .bDeviceSubClass = */ 0 /* Subclass specified in the interface desc. */,
1195 /* .bDeviceProtocol = */ 0 /* Protocol specified in the interface desc. */,
1196 /* .bMaxPacketSize0 = */ 8,
1197 /* .idVendor = */ VBOX_USB_VENDOR,
1198 /* .idProduct = */ USBHID_PID_TABLET,
1199 /* .bcdDevice = */ 0x0100, /* 1.0 */
1200 /* .iManufacturer = */ USBHID_STR_ID_MANUFACTURER,
1201 /* .iProduct = */ USBHID_STR_ID_PRODUCT_T,
1202 /* .iSerialNumber = */ 0,
1203 /* .bNumConfigurations = */ 1
1204};
1205
1206static const VUSBDESCDEVICE g_UsbHidMTDeviceDesc =
1207{
1208 /* .bLength = */ sizeof(g_UsbHidMTDeviceDesc),
1209 /* .bDescriptorType = */ VUSB_DT_DEVICE,
1210 /* .bcdUsb = */ 0x110, /* 1.1 */
1211 /* .bDeviceClass = */ 0 /* Class specified in the interface desc. */,
1212 /* .bDeviceSubClass = */ 0 /* Subclass specified in the interface desc. */,
1213 /* .bDeviceProtocol = */ 0 /* Protocol specified in the interface desc. */,
1214 /* .bMaxPacketSize0 = */ 8,
1215 /* .idVendor = */ VBOX_USB_VENDOR,
1216 /* .idProduct = */ USBHID_PID_MT_TOUCHSCREEN,
1217 /* .bcdDevice = */ 0x0100, /* 1.0 */
1218 /* .iManufacturer = */ USBHID_STR_ID_MANUFACTURER,
1219 /* .iProduct = */ USBHID_STR_ID_PRODUCT_MT,
1220 /* .iSerialNumber = */ 0,
1221 /* .bNumConfigurations = */ 1
1222};
1223
1224static const VUSBDESCDEVICE g_UsbHidTPDeviceDesc =
1225{
1226 /* .bLength = */ sizeof(g_UsbHidTPDeviceDesc),
1227 /* .bDescriptorType = */ VUSB_DT_DEVICE,
1228 /* .bcdUsb = */ 0x110, /* 1.1 */
1229 /* .bDeviceClass = */ 0 /* Class specified in the interface desc. */,
1230 /* .bDeviceSubClass = */ 0 /* Subclass specified in the interface desc. */,
1231 /* .bDeviceProtocol = */ 0 /* Protocol specified in the interface desc. */,
1232 /* .bMaxPacketSize0 = */ 8,
1233 /* .idVendor = */ VBOX_USB_VENDOR,
1234 /* .idProduct = */ USBHID_PID_MT_TOUCHPAD,
1235 /* .bcdDevice = */ 0x0100, /* 1.0 */
1236 /* .iManufacturer = */ USBHID_STR_ID_MANUFACTURER,
1237 /* .iProduct = */ USBHID_STR_ID_PRODUCT_TP,
1238 /* .iSerialNumber = */ 0,
1239 /* .bNumConfigurations = */ 1
1240};
1241
1242
1243static const PDMUSBDESCCACHE g_UsbHidMDescCache =
1244{
1245 /* .pDevice = */ &g_UsbHidMDeviceDesc,
1246 /* .paConfigs = */ &g_UsbHidMConfigDesc,
1247 /* .paLanguages = */ g_aUsbHidLanguages,
1248 /* .cLanguages = */ RT_ELEMENTS(g_aUsbHidLanguages),
1249 /* .fUseCachedDescriptors = */ true,
1250 /* .fUseCachedStringsDescriptors = */ true
1251};
1252
1253static const PDMUSBDESCCACHE g_UsbHidTDescCache =
1254{
1255 /* .pDevice = */ &g_UsbHidTDeviceDesc,
1256 /* .paConfigs = */ &g_UsbHidTConfigDesc,
1257 /* .paLanguages = */ g_aUsbHidLanguages,
1258 /* .cLanguages = */ RT_ELEMENTS(g_aUsbHidLanguages),
1259 /* .fUseCachedDescriptors = */ true,
1260 /* .fUseCachedStringsDescriptors = */ true
1261};
1262
1263static const PDMUSBDESCCACHE g_UsbHidMTDescCache =
1264{
1265 /* .pDevice = */ &g_UsbHidMTDeviceDesc,
1266 /* .paConfigs = */ &g_UsbHidMTConfigDesc,
1267 /* .paLanguages = */ g_aUsbHidLanguages,
1268 /* .cLanguages = */ RT_ELEMENTS(g_aUsbHidLanguages),
1269 /* .fUseCachedDescriptors = */ true,
1270 /* .fUseCachedStringsDescriptors = */ true
1271};
1272
1273static const PDMUSBDESCCACHE g_UsbHidTPDescCache =
1274{
1275 /* .pDevice = */ &g_UsbHidTPDeviceDesc,
1276 /* .paConfigs = */ &g_UsbHidTPConfigDesc,
1277 /* .paLanguages = */ g_aUsbHidLanguages,
1278 /* .cLanguages = */ RT_ELEMENTS(g_aUsbHidLanguages),
1279 /* .fUseCachedDescriptors = */ true,
1280 /* .fUseCachedStringsDescriptors = */ true
1281};
1282
1283
1284/*********************************************************************************************************************************
1285* Internal Functions *
1286*********************************************************************************************************************************/
1287
1288/**
1289 * Initializes an URB queue.
1290 *
1291 * @param pQueue The URB queue.
1292 */
1293static void usbHidQueueInit(PUSBHIDURBQUEUE pQueue)
1294{
1295 pQueue->pHead = NULL;
1296 pQueue->ppTail = &pQueue->pHead;
1297}
1298
1299
1300
1301/**
1302 * Inserts an URB at the end of the queue.
1303 *
1304 * @param pQueue The URB queue.
1305 * @param pUrb The URB to insert.
1306 */
1307DECLINLINE(void) usbHidQueueAddTail(PUSBHIDURBQUEUE pQueue, PVUSBURB pUrb)
1308{
1309 pUrb->Dev.pNext = NULL;
1310 *pQueue->ppTail = pUrb;
1311 pQueue->ppTail = &pUrb->Dev.pNext;
1312}
1313
1314
1315/**
1316 * Unlinks the head of the queue and returns it.
1317 *
1318 * @returns The head entry.
1319 * @param pQueue The URB queue.
1320 */
1321DECLINLINE(PVUSBURB) usbHidQueueRemoveHead(PUSBHIDURBQUEUE pQueue)
1322{
1323 PVUSBURB pUrb = pQueue->pHead;
1324 if (pUrb)
1325 {
1326 PVUSBURB pNext = pUrb->Dev.pNext;
1327 pQueue->pHead = pNext;
1328 if (!pNext)
1329 pQueue->ppTail = &pQueue->pHead;
1330 else
1331 pUrb->Dev.pNext = NULL;
1332 }
1333 return pUrb;
1334}
1335
1336
1337/**
1338 * Removes an URB from anywhere in the queue.
1339 *
1340 * @returns true if found, false if not.
1341 * @param pQueue The URB queue.
1342 * @param pUrb The URB to remove.
1343 */
1344DECLINLINE(bool) usbHidQueueRemove(PUSBHIDURBQUEUE pQueue, PVUSBURB pUrb)
1345{
1346 PVUSBURB pCur = pQueue->pHead;
1347 if (pCur == pUrb)
1348 {
1349 pQueue->pHead = pUrb->Dev.pNext;
1350 if (!pUrb->Dev.pNext)
1351 pQueue->ppTail = &pQueue->pHead;
1352 }
1353 else
1354 {
1355 while (pCur)
1356 {
1357 if (pCur->Dev.pNext == pUrb)
1358 {
1359 pCur->Dev.pNext = pUrb->Dev.pNext;
1360 break;
1361 }
1362 pCur = pCur->Dev.pNext;
1363 }
1364 if (!pCur)
1365 return false;
1366 if (!pUrb->Dev.pNext)
1367 pQueue->ppTail = &pCur->Dev.pNext;
1368 }
1369 pUrb->Dev.pNext = NULL;
1370 return true;
1371}
1372
1373
1374#if 0 /* unused */
1375/**
1376 * Checks if the queue is empty or not.
1377 *
1378 * @returns true if it is, false if it isn't.
1379 * @param pQueue The URB queue.
1380 */
1381DECLINLINE(bool) usbHidQueueIsEmpty(PCUSBHIDURBQUEUE pQueue)
1382{
1383 return pQueue->pHead == NULL;
1384}
1385#endif /* unused */
1386
1387
1388/**
1389 * Links an URB into the done queue.
1390 *
1391 * @param pThis The HID instance.
1392 * @param pUrb The URB.
1393 */
1394static void usbHidLinkDone(PUSBHID pThis, PVUSBURB pUrb)
1395{
1396 usbHidQueueAddTail(&pThis->DoneQueue, pUrb);
1397
1398 if (pThis->fHaveDoneQueueWaiter)
1399 {
1400 int rc = RTSemEventSignal(pThis->hEvtDoneQueue);
1401 AssertRC(rc);
1402 }
1403}
1404
1405
1406
1407/**
1408 * Completes the URB with a stalled state, halting the pipe.
1409 */
1410static int usbHidCompleteStall(PUSBHID pThis, PUSBHIDEP pEp, PVUSBURB pUrb, const char *pszWhy)
1411{
1412 LogRelFlow(("usbHidCompleteStall/#%u: pUrb=%p:%s: %s\n",
1413 pThis->pUsbIns->iInstance, pUrb, pUrb->pszDesc, pszWhy));
1414
1415 pUrb->enmStatus = VUSBSTATUS_STALL;
1416
1417 /** @todo figure out if the stall is global or pipe-specific or both. */
1418 if (pEp)
1419 pEp->fHalted = true;
1420 else
1421 {
1422 pThis->aEps[0].fHalted = true;
1423 pThis->aEps[1].fHalted = true;
1424 }
1425
1426 usbHidLinkDone(pThis, pUrb);
1427 return VINF_SUCCESS;
1428}
1429
1430
1431/**
1432 * Completes the URB after device successfully processed it. Optionally copies data
1433 * into the URB. May still generate an error if the URB is not big enough.
1434 */
1435static int usbHidCompleteOk(PUSBHID pThis, PVUSBURB pUrb, const void *pSrc, size_t cbSrc)
1436{
1437 Log(("usbHidCompleteOk/#%u: pUrb=%p:%s (cbData=%#x) cbSrc=%#zx\n", pThis->pUsbIns->iInstance, pUrb, pUrb->pszDesc, pUrb->cbData, cbSrc));
1438
1439 pUrb->enmStatus = VUSBSTATUS_OK;
1440 size_t cbCopy = 0;
1441 size_t cbSetup = 0;
1442
1443 if (pSrc) /* Can be NULL if not copying anything. */
1444 {
1445 Assert(cbSrc);
1446 uint8_t *pDst = pUrb->abData;
1447
1448 /* Returned data is written after the setup message in control URBs. */
1449 if (pUrb->enmType == VUSBXFERTYPE_MSG)
1450 cbSetup = sizeof(VUSBSETUP);
1451
1452 Assert(pUrb->cbData >= cbSetup); /* Only triggers if URB is corrupted. */
1453
1454 if (pUrb->cbData > cbSetup)
1455 {
1456 /* There is at least one byte of room in the URB. */
1457 cbCopy = RT_MIN(pUrb->cbData - cbSetup, cbSrc);
1458 memcpy(pDst + cbSetup, pSrc, cbCopy);
1459 pUrb->cbData = (uint32_t)(cbCopy + cbSetup);
1460 Log(("Copied %zu bytes to pUrb->abData[%zu], source had %zu bytes\n", cbCopy, cbSetup, cbSrc));
1461 }
1462
1463 /* Need to check length differences. If cbSrc is less than what
1464 * the URB has space for, it'll be resolved as a short packet. But
1465 * if cbSrc is bigger, there is a real problem and the host needs
1466 * to see an overrun/babble error.
1467 */
1468 if (RT_UNLIKELY(cbSrc > cbCopy))
1469 pUrb->enmStatus = VUSBSTATUS_DATA_OVERRUN;
1470 }
1471 else
1472 Assert(cbSrc == 0); /* Make up your mind, caller! */
1473
1474 usbHidLinkDone(pThis, pUrb);
1475 return VINF_SUCCESS;
1476}
1477
1478
1479/**
1480 * Reset worker for usbHidUsbReset, usbHidUsbSetConfiguration and
1481 * usbHidHandleDefaultPipe.
1482 *
1483 * @returns VBox status code.
1484 * @param pThis The HID instance.
1485 * @param pUrb Set when usbHidHandleDefaultPipe is the
1486 * caller.
1487 * @param fSetConfig Set when usbHidUsbSetConfiguration is the
1488 * caller.
1489 */
1490static int usbHidResetWorker(PUSBHID pThis, PVUSBURB pUrb, bool fSetConfig)
1491{
1492 /*
1493 * Wait for the any command currently executing to complete before
1494 * resetting. (We cannot cancel its execution.) How we do this depends
1495 * on the reset method.
1496 */
1497
1498 /*
1499 * Reset the device state.
1500 */
1501 pThis->enmState = USBHIDREQSTATE_READY;
1502 pThis->fHasPendingChanges = false;
1503 pThis->fTouchStateUpdated = false;
1504
1505 for (unsigned i = 0; i < RT_ELEMENTS(pThis->aEps); i++)
1506 pThis->aEps[i].fHalted = false;
1507
1508 if (!pUrb && !fSetConfig) /* (only device reset) */
1509 pThis->bConfigurationValue = 0; /* default */
1510
1511 /*
1512 * Ditch all pending URBs.
1513 */
1514 PVUSBURB pCurUrb;
1515 while ((pCurUrb = usbHidQueueRemoveHead(&pThis->ToHostQueue)) != NULL)
1516 {
1517 pCurUrb->enmStatus = VUSBSTATUS_CRC;
1518 usbHidLinkDone(pThis, pCurUrb);
1519 }
1520
1521 if (pUrb)
1522 return usbHidCompleteOk(pThis, pUrb, NULL, 0);
1523 return VINF_SUCCESS;
1524}
1525
1526static int8_t clamp_i8(int32_t val)
1527{
1528 if (val > 127) {
1529 val = 127;
1530 } else if (val < -127) {
1531 val = -127;
1532 }
1533 return val;
1534}
1535
1536/**
1537 * Create a USB HID report report based on the currently accumulated data.
1538 */
1539static size_t usbHidFillReport(PUSBHIDTM_REPORT pReport,
1540 PUSBHIDM_ACCUM pAccumulated, USBHIDMODE enmMode)
1541{
1542 size_t cbCopy;
1543
1544 switch (enmMode)
1545 {
1546 case USBHIDMODE_ABSOLUTE:
1547 pReport->t.fButtons = pAccumulated->u.Absolute.fButtons;
1548 pReport->t.dz = clamp_i8(pAccumulated->u.Absolute.dz);
1549 pReport->t.dw = clamp_i8(pAccumulated->u.Absolute.dw);
1550 pReport->t.padding = 0;
1551 pReport->t.x = pAccumulated->u.Absolute.x;
1552 pReport->t.y = pAccumulated->u.Absolute.y;
1553
1554 cbCopy = sizeof(pReport->t);
1555 LogRel3(("Abs event, x=%d, y=%d, fButtons=%02x, report size %d\n",
1556 pReport->t.x, pReport->t.y, pReport->t.fButtons,
1557 cbCopy));
1558 break;
1559 case USBHIDMODE_RELATIVE:
1560 pReport->m.fButtons = pAccumulated->u.Relative.fButtons;
1561 pReport->m.dx = clamp_i8(pAccumulated->u.Relative.dx);
1562 pReport->m.dy = clamp_i8(pAccumulated->u.Relative.dy);
1563 pReport->m.dz = clamp_i8(pAccumulated->u.Relative.dz);
1564
1565 cbCopy = sizeof(pReport->m);
1566 LogRel3(("Rel event, dx=%d, dy=%d, dz=%d, fButtons=%02x, report size %d\n",
1567 pReport->m.dx, pReport->m.dy, pReport->m.dz,
1568 pReport->m.fButtons, cbCopy));
1569 break;
1570 default:
1571 AssertFailed(); /* Unexpected here. */
1572 cbCopy = 0;
1573 break;
1574 }
1575
1576 /* Clear the accumulated movement. */
1577 RT_ZERO(*pAccumulated);
1578
1579 return cbCopy;
1580}
1581
1582DECLINLINE(MTCONTACT *) usbHidFindMTContact(MTCONTACT *paContacts, size_t cContacts,
1583 uint8_t u8Mask, uint8_t u8Value)
1584{
1585 size_t i;
1586 for (i = 0; i < cContacts; i++)
1587 {
1588 if ((paContacts[i].status & u8Mask) == u8Value)
1589 {
1590 return &paContacts[i];
1591 }
1592 }
1593
1594 return NULL;
1595}
1596
1597static int usbHidSendMultiTouchReport(PUSBHID pThis, PVUSBURB pUrb)
1598{
1599 uint8_t i;
1600 MTCONTACT *pRepContact;
1601 MTCONTACT *pCurContact;
1602
1603 /* Number of contacts to be reported. In hybrid mode the first report contains
1604 * total number of contacts and subsequent reports contain 0.
1605 */
1606 uint8_t cContacts = 0;
1607
1608 uint8_t cMaxContacts = pThis->enmMode == USBHIDMODE_MT_RELATIVE ? TPAD_CONTACT_MAX_COUNT : MT_CONTACT_MAX_COUNT;
1609 size_t cbReport = pThis->enmMode == USBHIDMODE_MT_RELATIVE ? sizeof(USBHIDTP_REPORT) : sizeof(USBHIDMT_REPORT);
1610
1611 Assert(pThis->fHasPendingChanges);
1612
1613 if (!pThis->fTouchReporting)
1614 {
1615 pThis->fTouchReporting = true;
1616 pThis->fTouchStateUpdated = false;
1617
1618 /* Update the reporting state with the new current state.
1619 * Also mark all active contacts in reporting state as dirty,
1620 * that is they must be reported to the guest.
1621 */
1622 for (i = 0; i < cMaxContacts; i++)
1623 {
1624 pRepContact = &pThis->aReportingContactState[i];
1625 pCurContact = &pThis->aCurrentContactState[i];
1626
1627 if (pCurContact->status & MT_CONTACT_S_ACTIVE)
1628 {
1629 if (pCurContact->status & MT_CONTACT_S_REUSED)
1630 {
1631 pCurContact->status &= ~MT_CONTACT_S_REUSED;
1632
1633 /* Keep x,y. Will report lost contact at this point. */
1634 pRepContact->id = pCurContact->oldId;
1635 pRepContact->flags = 0;
1636 pRepContact->status = MT_CONTACT_S_REUSED;
1637 }
1638 else if (pThis->aCurrentContactState[i].status & MT_CONTACT_S_CANCELLED)
1639 {
1640 pCurContact->status &= ~(MT_CONTACT_S_CANCELLED | MT_CONTACT_S_ACTIVE);
1641
1642 /* Keep x,y. Will report lost contact at this point. */
1643 pRepContact->id = pCurContact->id;
1644 pRepContact->flags = 0;
1645 pRepContact->status = 0;
1646 }
1647 else
1648 {
1649 if (pCurContact->flags == 0)
1650 {
1651 pCurContact->status &= ~MT_CONTACT_S_ACTIVE; /* Contact disapeared. */
1652 }
1653
1654 pRepContact->x = pCurContact->x;
1655 pRepContact->y = pCurContact->y;
1656 pRepContact->id = pCurContact->id;
1657 pRepContact->flags = pCurContact->flags;
1658 pRepContact->status = 0;
1659 }
1660
1661 cContacts++;
1662
1663 pRepContact->status |= MT_CONTACT_S_DIRTY;
1664 }
1665 else
1666 {
1667 pRepContact->status = 0;
1668 }
1669 }
1670 }
1671
1672 /* Report current state. */
1673 USBHIDTP_REPORT r;
1674 USBHIDTP_REPORT *p = &r;
1675 RT_ZERO(*p);
1676
1677 p->mt.idReport = REPORTID_TOUCH_EVENT;
1678 p->mt.cContacts = cContacts;
1679 p->buttons = 0; /* Not currently used. */
1680
1681 uint8_t iReportedContact;
1682 for (iReportedContact = 0; iReportedContact < MT_CONTACTS_PER_REPORT; iReportedContact++)
1683 {
1684 /* Find the next not reported contact. */
1685 pRepContact = usbHidFindMTContact(pThis->aReportingContactState, RT_ELEMENTS(pThis->aReportingContactState),
1686 MT_CONTACT_S_DIRTY, MT_CONTACT_S_DIRTY);
1687
1688 if (!pRepContact)
1689 {
1690 LogRel3(("usbHid: no more touch contacts to report\n"));
1691 break;
1692 }
1693
1694 if (pRepContact->status & MT_CONTACT_S_REUSED)
1695 {
1696 /* Do not clear DIRTY flag for contacts which were reused.
1697 * Because two reports must be generated:
1698 * one for old contact off, and the second for new contact on.
1699 */
1700 pRepContact->status &= ~MT_CONTACT_S_REUSED;
1701 }
1702 else
1703 {
1704 pRepContact->status &= ~MT_CONTACT_S_DIRTY;
1705 }
1706
1707 p->mt.aContacts[iReportedContact].fContact = pRepContact->flags;
1708 p->mt.aContacts[iReportedContact].cContact = pRepContact->id;
1709 p->mt.aContacts[iReportedContact].x = pRepContact->x >> pThis->u8CoordShift;
1710 p->mt.aContacts[iReportedContact].y = pRepContact->y >> pThis->u8CoordShift;
1711
1712 if (pThis->enmMode == USBHIDMODE_MT_RELATIVE) {
1713 /** @todo Parse touch confidence in Qt frontend */
1714 p->mt.aContacts[iReportedContact].fContact |= MT_CONTACT_F_CONFIDENCE;
1715 }
1716 }
1717
1718 p->mt.u32ScanTime = pThis->u32LastTouchScanTime * 10;
1719
1720 Assert(iReportedContact > 0);
1721
1722 /* Reset TouchReporting if all contacts reported. */
1723 pRepContact = usbHidFindMTContact(pThis->aReportingContactState, RT_ELEMENTS(pThis->aReportingContactState),
1724 MT_CONTACT_S_DIRTY, MT_CONTACT_S_DIRTY);
1725
1726 if (!pRepContact)
1727 {
1728 LogRel3(("usbHid: all touch contacts reported\n"));
1729 pThis->fTouchReporting = false;
1730 pThis->fHasPendingChanges = pThis->fTouchStateUpdated;
1731 }
1732 else
1733 {
1734 pThis->fHasPendingChanges = true;
1735 }
1736
1737 LogRel3(("usbHid: reporting touch contact:\n%.*Rhxd\n", cbReport, p));
1738 return usbHidCompleteOk(pThis, pUrb, p, cbReport);
1739}
1740
1741
1742/**
1743 * Sends a state report to the host if there is a pending URB.
1744 */
1745static int usbHidSendReport(PUSBHID pThis)
1746{
1747 PVUSBURB pUrb = usbHidQueueRemoveHead(&pThis->ToHostQueue);
1748
1749 if (pThis->enmMode == USBHIDMODE_MT_ABSOLUTE || pThis->enmMode == USBHIDMODE_MT_RELATIVE)
1750 {
1751 /* These modes use a different reporting method and maintain fHasPendingChanges. */
1752 if (pUrb)
1753 return usbHidSendMultiTouchReport(pThis, pUrb);
1754 return VINF_SUCCESS;
1755 }
1756
1757 if (pUrb)
1758 {
1759 USBHIDTM_REPORT report;
1760 PUSBHIDTM_REPORT pReport = &report;
1761 size_t cbCopy;
1762
1763 cbCopy = usbHidFillReport(pReport, &pThis->PtrDelta, pThis->enmMode);
1764 pThis->fHasPendingChanges = false;
1765 return usbHidCompleteOk(pThis, pUrb, pReport, cbCopy);
1766 }
1767 else
1768 {
1769 LogRelFlow(("No available URB for USB mouse\n"));
1770 pThis->fHasPendingChanges = true;
1771 }
1772 return VINF_EOF;
1773}
1774
1775/**
1776 * @interface_method_impl{PDMIBASE,pfnQueryInterface}
1777 */
1778static DECLCALLBACK(void *) usbHidMouseQueryInterface(PPDMIBASE pInterface, const char *pszIID)
1779{
1780 PUSBHID pThis = RT_FROM_MEMBER(pInterface, USBHID, Lun0.IBase);
1781 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIBASE, &pThis->Lun0.IBase);
1782 PDMIBASE_RETURN_INTERFACE(pszIID, PDMIMOUSEPORT, &pThis->Lun0.IPort);
1783 return NULL;
1784}
1785
1786/**
1787 * @interface_method_impl{PDMIMOUSEPORT,pfnPutEvent}
1788 */
1789static DECLCALLBACK(int) usbHidMousePutEvent(PPDMIMOUSEPORT pInterface, int32_t dx, int32_t dy,
1790 int32_t dz, int32_t dw, uint32_t fButtons)
1791{
1792 RT_NOREF1(dw);
1793 PUSBHID pThis = RT_FROM_MEMBER(pInterface, USBHID, Lun0.IPort);
1794 RTCritSectEnter(&pThis->CritSect);
1795
1796 /* Accumulate movement - the events from the front end may arrive
1797 * at a much higher rate than USB can handle.
1798 */
1799 pThis->PtrDelta.u.Relative.fButtons = fButtons;
1800 pThis->PtrDelta.u.Relative.dx += dx;
1801 pThis->PtrDelta.u.Relative.dy += dy;
1802 pThis->PtrDelta.u.Relative.dz -= dz; /* Inverted! */
1803
1804 /* Send a report if possible. */
1805 usbHidSendReport(pThis);
1806
1807 RTCritSectLeave(&pThis->CritSect);
1808 return VINF_SUCCESS;
1809}
1810
1811/**
1812 * @interface_method_impl{PDMIMOUSEPORT,pfnPutEventAbs}
1813 */
1814static DECLCALLBACK(int) usbHidMousePutEventAbs(PPDMIMOUSEPORT pInterface,
1815 uint32_t x, uint32_t y,
1816 int32_t dz, int32_t dw,
1817 uint32_t fButtons)
1818{
1819 PUSBHID pThis = RT_FROM_MEMBER(pInterface, USBHID, Lun0.IPort);
1820 RTCritSectEnter(&pThis->CritSect);
1821
1822 Assert(pThis->enmMode == USBHIDMODE_ABSOLUTE);
1823
1824 /* Accumulate movement - the events from the front end may arrive
1825 * at a much higher rate than USB can handle. Probably not a real issue
1826 * when only the Z axis is relative (X/Y movement isn't technically
1827 * accumulated and only the last value is used).
1828 */
1829 pThis->PtrDelta.u.Absolute.fButtons = fButtons;
1830 pThis->PtrDelta.u.Absolute.x = x >> pThis->u8CoordShift;
1831 pThis->PtrDelta.u.Absolute.y = y >> pThis->u8CoordShift;
1832 pThis->PtrDelta.u.Absolute.dz -= dz; /* Inverted! */
1833 pThis->PtrDelta.u.Absolute.dw -= dw; /* Inverted! */
1834
1835 /* Send a report if possible. */
1836 usbHidSendReport(pThis);
1837
1838 RTCritSectLeave(&pThis->CritSect);
1839 return VINF_SUCCESS;
1840}
1841
1842/**
1843 * Worker for usbHidMousePutEventTouchScreen and
1844 * usbHidMousePutEventTouchPad.
1845 */
1846static DECLCALLBACK(int) usbHidMousePutEventMultiTouch(PUSBHID pThis,
1847 uint8_t cContacts,
1848 const uint64_t *pau64Contacts,
1849 uint32_t u32ScanTime)
1850{
1851 uint8_t i;
1852 uint8_t j;
1853
1854 /* Make a copy of new contacts */
1855 MTCONTACT *paNewContacts = (MTCONTACT *)RTMemTmpAlloc(sizeof(MTCONTACT) * cContacts);
1856 if (!paNewContacts)
1857 return VERR_NO_MEMORY;
1858
1859 for (i = 0; i < cContacts; i++)
1860 {
1861 uint32_t u32Lo = RT_LO_U32(pau64Contacts[i]);
1862 uint32_t u32Hi = RT_HI_U32(pau64Contacts[i]);
1863 paNewContacts[i].x = (uint16_t)u32Lo;
1864 paNewContacts[i].y = (uint16_t)(u32Lo >> 16);
1865 paNewContacts[i].id = RT_BYTE1(u32Hi);
1866 paNewContacts[i].flags = RT_BYTE2(u32Hi);
1867 paNewContacts[i].status = MT_CONTACT_S_DIRTY;
1868 paNewContacts[i].oldId = 0; /* Not used. */
1869
1870 if (pThis->enmMode == USBHIDMODE_MT_ABSOLUTE)
1871 {
1872 paNewContacts[i].flags &= MT_CONTACT_F_IN_CONTACT | MT_CONTACT_F_IN_RANGE;
1873 if (paNewContacts[i].flags & MT_CONTACT_F_IN_CONTACT)
1874 {
1875 paNewContacts[i].flags |= MT_CONTACT_F_IN_RANGE;
1876 }
1877 }
1878 else
1879 {
1880 Assert(pThis->enmMode == USBHIDMODE_MT_RELATIVE);
1881 paNewContacts[i].flags &= MT_CONTACT_F_IN_CONTACT;
1882 }
1883 }
1884
1885 MTCONTACT *pCurContact = NULL;
1886 MTCONTACT *pNewContact = NULL;
1887
1888 RTCritSectEnter(&pThis->CritSect);
1889
1890 /* Maintain a state of all current contacts.
1891 * Intr URBs will be completed according to the state.
1892 */
1893
1894 /* Mark all existing contacts as dirty. */
1895 for (i = 0; i < RT_ELEMENTS(pThis->aCurrentContactState); i++)
1896 pThis->aCurrentContactState[i].status |= MT_CONTACT_S_DIRTY;
1897
1898 /* Update existing contacts and mark new contacts. */
1899 for (i = 0; i < cContacts; i++)
1900 {
1901 pNewContact = &paNewContacts[i];
1902
1903 /* Find existing contact with the same id. */
1904 pCurContact = NULL;
1905 for (j = 0; j < RT_ELEMENTS(pThis->aCurrentContactState); j++)
1906 {
1907 if ( (pThis->aCurrentContactState[j].status & MT_CONTACT_S_ACTIVE) != 0
1908 && pThis->aCurrentContactState[j].id == pNewContact->id)
1909 {
1910 pCurContact = &pThis->aCurrentContactState[j];
1911 break;
1912 }
1913 }
1914
1915 if (pCurContact)
1916 {
1917 pNewContact->status &= ~MT_CONTACT_S_DIRTY;
1918
1919 pCurContact->x = pNewContact->x;
1920 pCurContact->y = pNewContact->y;
1921 if (pCurContact->flags == 0) /* Contact disappeared already. */
1922 {
1923 if ((pCurContact->status & MT_CONTACT_S_REUSED) == 0)
1924 {
1925 pCurContact->status |= MT_CONTACT_S_REUSED; /* Report to the guest that the contact not in touch. */
1926 pCurContact->oldId = pCurContact->id;
1927 }
1928 }
1929 pCurContact->flags = pNewContact->flags;
1930 pCurContact->status &= ~MT_CONTACT_S_DIRTY;
1931 }
1932 }
1933
1934 /* Append new contacts (the dirty one in the paNewContacts). */
1935 for (i = 0; i < cContacts; i++)
1936 {
1937 pNewContact = &paNewContacts[i];
1938
1939 if (pNewContact->status & MT_CONTACT_S_DIRTY)
1940 {
1941 /* It is a new contact, copy is to one of not ACTIVE or not updated existing contacts. */
1942 pCurContact = usbHidFindMTContact(pThis->aCurrentContactState, RT_ELEMENTS(pThis->aCurrentContactState),
1943 MT_CONTACT_S_ACTIVE, 0);
1944
1945 if (pCurContact)
1946 {
1947 *pCurContact = *pNewContact;
1948 pCurContact->status = MT_CONTACT_S_ACTIVE; /* Reset status. */
1949 }
1950 else
1951 {
1952 /* Dirty existing contacts can be reused. */
1953 pCurContact = usbHidFindMTContact(pThis->aCurrentContactState, RT_ELEMENTS(pThis->aCurrentContactState),
1954 MT_CONTACT_S_ACTIVE | MT_CONTACT_S_DIRTY,
1955 MT_CONTACT_S_ACTIVE | MT_CONTACT_S_DIRTY);
1956
1957 if (pCurContact)
1958 {
1959 pCurContact->x = pNewContact->x;
1960 pCurContact->y = pNewContact->y;
1961 if ((pCurContact->status & MT_CONTACT_S_REUSED) == 0)
1962 {
1963 pCurContact->status |= MT_CONTACT_S_REUSED; /* Report to the guest that the contact not in touch. */
1964 pCurContact->oldId = pCurContact->id;
1965 }
1966 pCurContact->flags = pNewContact->flags;
1967 pCurContact->status &= ~MT_CONTACT_S_DIRTY;
1968 }
1969 else
1970 {
1971 LogRel3(("usbHid: dropped new contact: %d,%d id %d flags %RX8 status %RX8 oldId %d\n",
1972 pNewContact->x,
1973 pNewContact->y,
1974 pNewContact->id,
1975 pNewContact->flags,
1976 pNewContact->status,
1977 pNewContact->oldId
1978 ));
1979 }
1980 }
1981 }
1982 }
1983
1984 /* Mark still dirty existing contacts as cancelled, because a new set of contacts does not include them. */
1985 for (i = 0; i < RT_ELEMENTS(pThis->aCurrentContactState); i++)
1986 {
1987 pCurContact = &pThis->aCurrentContactState[i];
1988 if (pCurContact->status & MT_CONTACT_S_DIRTY)
1989 {
1990 pCurContact->status |= MT_CONTACT_S_CANCELLED;
1991 pCurContact->status &= ~MT_CONTACT_S_DIRTY;
1992 }
1993 }
1994
1995 pThis->u32LastTouchScanTime = u32ScanTime;
1996
1997 LogRel3(("usbHid: scanTime (ms): %d\n", pThis->u32LastTouchScanTime));
1998 for (i = 0; i < RT_ELEMENTS(pThis->aCurrentContactState); i++)
1999 {
2000 LogRel3(("usbHid: contact state[%d]: %d,%d id %d flags %RX8 status %RX8 oldId %d\n",
2001 i,
2002 pThis->aCurrentContactState[i].x,
2003 pThis->aCurrentContactState[i].y,
2004 pThis->aCurrentContactState[i].id,
2005 pThis->aCurrentContactState[i].flags,
2006 pThis->aCurrentContactState[i].status,
2007 pThis->aCurrentContactState[i].oldId
2008 ));
2009 }
2010
2011 pThis->fTouchStateUpdated = true;
2012 pThis->fHasPendingChanges = true;
2013
2014 /* Send a report if possible. */
2015 usbHidSendReport(pThis);
2016
2017 RTCritSectLeave(&pThis->CritSect);
2018
2019 RTMemTmpFree(paNewContacts);
2020 return VINF_SUCCESS;
2021}
2022
2023/**
2024 * @interface_method_impl{PDMIMOUSEPORT,pfnPutEventTouchScreen}
2025 */
2026static DECLCALLBACK(int) usbHidMousePutEventTouchScreen(PPDMIMOUSEPORT pInterface,
2027 uint8_t cContacts,
2028 const uint64_t *pau64Contacts,
2029 uint32_t u32ScanTime)
2030{
2031 PUSBHID pThis = RT_FROM_MEMBER(pInterface, USBHID, Lun0.IPort);
2032
2033 Assert(pThis->enmMode == USBHIDMODE_MT_ABSOLUTE);
2034
2035 return usbHidMousePutEventMultiTouch(pThis, cContacts, pau64Contacts, u32ScanTime);
2036}
2037
2038/**
2039 * @interface_method_impl{PDMIMOUSEPORT,pfnPutEventTouchPad}
2040 */
2041static DECLCALLBACK(int) usbHidMousePutEventTouchPad(PPDMIMOUSEPORT pInterface,
2042 uint8_t cContacts,
2043 const uint64_t *pau64Contacts,
2044 uint32_t u32ScanTime)
2045{
2046 PUSBHID pThis = RT_FROM_MEMBER(pInterface, USBHID, Lun0.IPort);
2047
2048 Assert(pThis->enmMode == USBHIDMODE_MT_RELATIVE);
2049
2050 return usbHidMousePutEventMultiTouch(pThis, cContacts, pau64Contacts, u32ScanTime);
2051}
2052
2053/**
2054 * @interface_method_impl{PDMUSBREG,pfnUrbReap}
2055 */
2056static DECLCALLBACK(PVUSBURB) usbHidUrbReap(PPDMUSBINS pUsbIns, RTMSINTERVAL cMillies)
2057{
2058 PUSBHID pThis = PDMINS_2_DATA(pUsbIns, PUSBHID);
2059
2060 LogFlowFunc(("pUsbIns=%p cMillies=%u\n", pUsbIns, cMillies));
2061
2062 RTCritSectEnter(&pThis->CritSect);
2063
2064 PVUSBURB pUrb = usbHidQueueRemoveHead(&pThis->DoneQueue);
2065 if (!pUrb && cMillies)
2066 {
2067 /* Wait */
2068 pThis->fHaveDoneQueueWaiter = true;
2069 RTCritSectLeave(&pThis->CritSect);
2070
2071 RTSemEventWait(pThis->hEvtDoneQueue, cMillies);
2072
2073 RTCritSectEnter(&pThis->CritSect);
2074 pThis->fHaveDoneQueueWaiter = false;
2075
2076 pUrb = usbHidQueueRemoveHead(&pThis->DoneQueue);
2077 }
2078
2079 RTCritSectLeave(&pThis->CritSect);
2080
2081 if (pUrb)
2082 LogRelFlow(("usbHidUrbReap/#%u: pUrb=%p:%s\n", pUsbIns->iInstance, pUrb,
2083 pUrb->pszDesc));
2084 return pUrb;
2085}
2086
2087/**
2088 * @interface_method_impl{PDMUSBREG,pfnWakeup}
2089 */
2090static DECLCALLBACK(int) usbHidWakeup(PPDMUSBINS pUsbIns)
2091{
2092 PUSBHID pThis = PDMINS_2_DATA(pUsbIns, PUSBHID);
2093
2094 return RTSemEventSignal(pThis->hEvtDoneQueue);
2095}
2096
2097/**
2098 * @interface_method_impl{PDMUSBREG,pfnUrbCancel}
2099 */
2100static DECLCALLBACK(int) usbHidUrbCancel(PPDMUSBINS pUsbIns, PVUSBURB pUrb)
2101{
2102 PUSBHID pThis = PDMINS_2_DATA(pUsbIns, PUSBHID);
2103 LogRelFlow(("usbHidUrbCancel/#%u: pUrb=%p:%s\n", pUsbIns->iInstance, pUrb,
2104 pUrb->pszDesc));
2105 RTCritSectEnter(&pThis->CritSect);
2106
2107 /*
2108 * Remove the URB from the to-host queue and move it onto the done queue.
2109 */
2110 if (usbHidQueueRemove(&pThis->ToHostQueue, pUrb))
2111 usbHidLinkDone(pThis, pUrb);
2112
2113 RTCritSectLeave(&pThis->CritSect);
2114 return VINF_SUCCESS;
2115}
2116
2117
2118/**
2119 * Handles request sent to the inbound (device to host) interrupt pipe. This is
2120 * rather different from bulk requests because an interrupt read URB may complete
2121 * after arbitrarily long time.
2122 */
2123static int usbHidHandleIntrDevToHost(PUSBHID pThis, PUSBHIDEP pEp, PVUSBURB pUrb)
2124{
2125 /*
2126 * Stall the request if the pipe is halted.
2127 */
2128 if (RT_UNLIKELY(pEp->fHalted))
2129 return usbHidCompleteStall(pThis, NULL, pUrb, "Halted pipe");
2130
2131 /*
2132 * Deal with the URB according to the state.
2133 */
2134 switch (pThis->enmState)
2135 {
2136 /*
2137 * We've data left to transfer to the host.
2138 */
2139 case USBHIDREQSTATE_DATA_TO_HOST:
2140 {
2141 AssertFailed();
2142 LogRelFlow(("usbHidHandleIntrDevToHost: Entering STATUS\n"));
2143 return usbHidCompleteOk(pThis, pUrb, NULL, 0);
2144 }
2145
2146 /*
2147 * Status transfer.
2148 */
2149 case USBHIDREQSTATE_STATUS:
2150 {
2151 AssertFailed();
2152 LogRelFlow(("usbHidHandleIntrDevToHost: Entering READY\n"));
2153 pThis->enmState = USBHIDREQSTATE_READY;
2154 return usbHidCompleteOk(pThis, pUrb, NULL, 0);
2155 }
2156
2157 case USBHIDREQSTATE_READY:
2158 usbHidQueueAddTail(&pThis->ToHostQueue, pUrb);
2159 LogRelFlow(("usbHidHandleIntrDevToHost: Added %p:%s to the queue\n",
2160 pUrb, pUrb->pszDesc));
2161 /* If a report is pending, send it right away. */
2162 if (pThis->fHasPendingChanges)
2163 usbHidSendReport(pThis);
2164 return VINF_SUCCESS;
2165
2166 /*
2167 * Bad states, stall.
2168 */
2169 default:
2170 LogRelFlow(("usbHidHandleIntrDevToHost: enmState=%d cbData=%#x\n",
2171 pThis->enmState, pUrb->cbData));
2172 return usbHidCompleteStall(pThis, NULL, pUrb, "Really bad state (D2H)!");
2173 }
2174}
2175
2176#define GET_REPORT 0x01
2177#define GET_IDLE 0x02
2178#define GET_PROTOCOL 0x03
2179#define SET_REPORT 0x09
2180#define SET_IDLE 0x0A
2181#define SET_PROTOCOL 0x0B
2182
2183static uint8_t const g_abQASampleBlob[256 + 1] =
2184{
2185 REPORTID_TOUCH_QABLOB, /* Report Id. */
2186 0xfc, 0x28, 0xfe, 0x84, 0x40, 0xcb, 0x9a, 0x87,
2187 0x0d, 0xbe, 0x57, 0x3c, 0xb6, 0x70, 0x09, 0x88,
2188 0x07, 0x97, 0x2d, 0x2b, 0xe3, 0x38, 0x34, 0xb6,
2189 0x6c, 0xed, 0xb0, 0xf7, 0xe5, 0x9c, 0xf6, 0xc2,
2190 0x2e, 0x84, 0x1b, 0xe8, 0xb4, 0x51, 0x78, 0x43,
2191 0x1f, 0x28, 0x4b, 0x7c, 0x2d, 0x53, 0xaf, 0xfc,
2192 0x47, 0x70, 0x1b, 0x59, 0x6f, 0x74, 0x43, 0xc4,
2193 0xf3, 0x47, 0x18, 0x53, 0x1a, 0xa2, 0xa1, 0x71,
2194 0xc7, 0x95, 0x0e, 0x31, 0x55, 0x21, 0xd3, 0xb5,
2195 0x1e, 0xe9, 0x0c, 0xba, 0xec, 0xb8, 0x89, 0x19,
2196 0x3e, 0xb3, 0xaf, 0x75, 0x81, 0x9d, 0x53, 0xb9,
2197 0x41, 0x57, 0xf4, 0x6d, 0x39, 0x25, 0x29, 0x7c,
2198 0x87, 0xd9, 0xb4, 0x98, 0x45, 0x7d, 0xa7, 0x26,
2199 0x9c, 0x65, 0x3b, 0x85, 0x68, 0x89, 0xd7, 0x3b,
2200 0xbd, 0xff, 0x14, 0x67, 0xf2, 0x2b, 0xf0, 0x2a,
2201 0x41, 0x54, 0xf0, 0xfd, 0x2c, 0x66, 0x7c, 0xf8,
2202 0xc0, 0x8f, 0x33, 0x13, 0x03, 0xf1, 0xd3, 0xc1,
2203 0x0b, 0x89, 0xd9, 0x1b, 0x62, 0xcd, 0x51, 0xb7,
2204 0x80, 0xb8, 0xaf, 0x3a, 0x10, 0xc1, 0x8a, 0x5b,
2205 0xe8, 0x8a, 0x56, 0xf0, 0x8c, 0xaa, 0xfa, 0x35,
2206 0xe9, 0x42, 0xc4, 0xd8, 0x55, 0xc3, 0x38, 0xcc,
2207 0x2b, 0x53, 0x5c, 0x69, 0x52, 0xd5, 0xc8, 0x73,
2208 0x02, 0x38, 0x7c, 0x73, 0xb6, 0x41, 0xe7, 0xff,
2209 0x05, 0xd8, 0x2b, 0x79, 0x9a, 0xe2, 0x34, 0x60,
2210 0x8f, 0xa3, 0x32, 0x1f, 0x09, 0x78, 0x62, 0xbc,
2211 0x80, 0xe3, 0x0f, 0xbd, 0x65, 0x20, 0x08, 0x13,
2212 0xc1, 0xe2, 0xee, 0x53, 0x2d, 0x86, 0x7e, 0xa7,
2213 0x5a, 0xc5, 0xd3, 0x7d, 0x98, 0xbe, 0x31, 0x48,
2214 0x1f, 0xfb, 0xda, 0xaf, 0xa2, 0xa8, 0x6a, 0x89,
2215 0xd6, 0xbf, 0xf2, 0xd3, 0x32, 0x2a, 0x9a, 0xe4,
2216 0xcf, 0x17, 0xb7, 0xb8, 0xf4, 0xe1, 0x33, 0x08,
2217 0x24, 0x8b, 0xc4, 0x43, 0xa5, 0xe5, 0x24, 0xc2
2218};
2219
2220static int usbHidRequestClass(PUSBHID pThis, PUSBHIDEP pEp, PVUSBURB pUrb)
2221{
2222 PVUSBSETUP pSetup = (PVUSBSETUP)&pUrb->abData[0];
2223
2224 if ((pThis->enmMode != USBHIDMODE_MT_ABSOLUTE) && (pThis->enmMode != USBHIDMODE_MT_RELATIVE))
2225 {
2226 LogRelFlow(("usbHid: bmRequestType=%#x bRequest=%#x wValue=%#x wIndex=%#x wLength=%#x\n",
2227 pSetup->bmRequestType, pSetup->bRequest, pSetup->wValue,
2228 pSetup->wIndex, pSetup->wLength));
2229 return usbHidCompleteStall(pThis, pEp, pUrb, "Unsupported class req");
2230 }
2231
2232 int rc = VINF_SUCCESS;
2233
2234 switch (pSetup->bRequest)
2235 {
2236 case SET_REPORT:
2237 case GET_REPORT:
2238 {
2239 uint8_t u8ReportType = RT_HI_U8(pSetup->wValue);
2240 uint8_t u8ReportID = RT_LO_U8(pSetup->wValue);
2241 LogRelFlow(("usbHid: %s: type %d, ID %d, data\n%.*Rhxd\n",
2242 pSetup->bRequest == GET_REPORT? "GET_REPORT": "SET_REPORT",
2243 u8ReportType, u8ReportID,
2244 pUrb->cbData - sizeof(VUSBSETUP), &pUrb->abData[sizeof(VUSBSETUP)]));
2245 if (pSetup->bRequest == GET_REPORT)
2246 {
2247 uint8_t abData[sizeof(USBHIDALL_REPORT)];
2248 uint8_t *pData = (uint8_t *)&abData;
2249 uint32_t cbData = 0; /* 0 means that the report is unsupported. */
2250
2251 if (u8ReportType == 1 && u8ReportID == REPORTID_TOUCH_POINTER)
2252 {
2253 USBHIDMT_REPORT_POINTER *p = (USBHIDMT_REPORT_POINTER *)&abData;
2254 /* The actual state should be reported here. */
2255 p->idReport = REPORTID_TOUCH_POINTER;
2256 p->fButtons = 0;
2257 p->x = 0;
2258 p->y = 0;
2259 cbData = sizeof(USBHIDMT_REPORT_POINTER);
2260 }
2261 else if (u8ReportType == 1 && u8ReportID == REPORTID_TOUCH_EVENT)
2262 {
2263 switch (pThis->enmMode)
2264 {
2265 case USBHIDMODE_MT_ABSOLUTE:
2266 {
2267 USBHIDMT_REPORT *p = (USBHIDMT_REPORT *)&abData;
2268 /* The actual state should be reported here. */
2269 RT_ZERO(*p);
2270 p->idReport = REPORTID_TOUCH_EVENT;
2271 cbData = sizeof(USBHIDMT_REPORT);
2272 break;
2273 }
2274 case USBHIDMODE_MT_RELATIVE:
2275 {
2276 USBHIDTP_REPORT *p = (USBHIDTP_REPORT *)&abData;
2277 /* The actual state should be reported here. */
2278 RT_ZERO(*p);
2279 p->mt.idReport = REPORTID_TOUCH_EVENT;
2280 cbData = sizeof(USBHIDTP_REPORT);
2281 break;
2282 }
2283 default:
2284 AssertMsgFailed(("Invalid HID mode %d\n", pThis->enmMode));
2285 break;
2286 }
2287 }
2288 else if (u8ReportType == 3 && u8ReportID == REPORTID_TOUCH_MAX_COUNT)
2289 {
2290 uint8_t cMaxContacts = 0;
2291 switch (pThis->enmMode)
2292 {
2293 case USBHIDMODE_MT_ABSOLUTE:
2294 cMaxContacts = MT_CONTACT_MAX_COUNT;
2295 break;
2296 case USBHIDMODE_MT_RELATIVE:
2297 cMaxContacts = TPAD_CONTACT_MAX_COUNT;
2298 break;
2299 default:
2300 AssertMsgFailed(("Invalid HID mode %d\n", pThis->enmMode));
2301 break;
2302 }
2303 abData[0] = REPORTID_TOUCH_MAX_COUNT;
2304 abData[1] = cMaxContacts; /* Contact count maximum. */
2305 abData[2] = 0; /* Device identifier */
2306 cbData = 3;
2307 }
2308 else if (u8ReportType == 3 && u8ReportID == REPORTID_TOUCH_QABLOB)
2309 {
2310 pData = (uint8_t *)&g_abQASampleBlob;
2311 cbData = sizeof(g_abQASampleBlob);
2312 }
2313 else if (u8ReportType == 3 && u8ReportID == REPORTID_TOUCH_DEVCONFIG)
2314 {
2315 abData[0] = REPORTID_TOUCH_DEVCONFIG;
2316 abData[1] = 2; /* Device mode:
2317 * "HID touch device supporting contact
2318 * identifier and contact count maximum."
2319 */
2320 abData[2] = 0; /* Device identifier */
2321 cbData = 3;
2322 }
2323
2324 if (cbData > 0)
2325 {
2326 rc = usbHidCompleteOk(pThis, pUrb, pData, cbData);
2327 }
2328 else
2329 {
2330 rc = usbHidCompleteStall(pThis, pEp, pUrb, "Unsupported GET_REPORT MT");
2331 }
2332 }
2333 else
2334 {
2335 /* SET_REPORT */
2336 rc = usbHidCompleteOk(pThis, pUrb, NULL, 0);
2337 }
2338 } break;
2339 default:
2340 {
2341 LogRelFlow(("usbHid: bmRequestType=%#x bRequest=%#x wValue=%#x wIndex=%#x wLength=%#x\n",
2342 pSetup->bmRequestType, pSetup->bRequest, pSetup->wValue,
2343 pSetup->wIndex, pSetup->wLength));
2344 rc = usbHidCompleteStall(pThis, pEp, pUrb, "Unsupported class req MT");
2345 }
2346 }
2347
2348 return rc;
2349}
2350
2351/**
2352 * Handles request sent to the default control pipe.
2353 */
2354static int usbHidHandleDefaultPipe(PUSBHID pThis, PUSBHIDEP pEp, PVUSBURB pUrb)
2355{
2356 PVUSBSETUP pSetup = (PVUSBSETUP)&pUrb->abData[0];
2357 AssertReturn(pUrb->cbData >= sizeof(*pSetup), VERR_VUSB_FAILED_TO_QUEUE_URB);
2358
2359 if ((pSetup->bmRequestType & VUSB_REQ_MASK) == VUSB_REQ_STANDARD)
2360 {
2361 switch (pSetup->bRequest)
2362 {
2363 case VUSB_REQ_GET_DESCRIPTOR:
2364 {
2365 switch (pSetup->bmRequestType)
2366 {
2367 case VUSB_TO_DEVICE | VUSB_REQ_STANDARD | VUSB_DIR_TO_HOST:
2368 {
2369 switch (pSetup->wValue >> 8)
2370 {
2371 case VUSB_DT_STRING:
2372 LogRelFlow(("usbHid: GET_DESCRIPTOR DT_STRING wValue=%#x wIndex=%#x\n",
2373 pSetup->wValue, pSetup->wIndex));
2374 break;
2375 default:
2376 LogRelFlow(("usbHid: GET_DESCRIPTOR, huh? wValue=%#x wIndex=%#x\n",
2377 pSetup->wValue, pSetup->wIndex));
2378 break;
2379 }
2380 break;
2381 }
2382
2383 case VUSB_TO_INTERFACE | VUSB_REQ_STANDARD | VUSB_DIR_TO_HOST:
2384 {
2385 switch (pSetup->wValue >> 8)
2386 {
2387 uint32_t cbCopy;
2388 uint32_t cbDesc;
2389 const uint8_t *pDesc;
2390
2391 case DT_IF_HID_DESCRIPTOR:
2392 {
2393 switch (pThis->enmMode)
2394 {
2395 case USBHIDMODE_ABSOLUTE:
2396 cbDesc = sizeof(g_UsbHidTIfHidDesc);
2397 pDesc = (const uint8_t *)&g_UsbHidTIfHidDesc;
2398 break;
2399 case USBHIDMODE_RELATIVE:
2400 cbDesc = sizeof(g_UsbHidMIfHidDesc);
2401 pDesc = (const uint8_t *)&g_UsbHidMIfHidDesc;
2402 break;
2403 case USBHIDMODE_MT_ABSOLUTE:
2404 cbDesc = sizeof(g_UsbHidMTIfHidDesc);
2405 pDesc = (const uint8_t *)&g_UsbHidMTIfHidDesc;
2406 break;
2407 case USBHIDMODE_MT_RELATIVE:
2408 cbDesc = sizeof(g_UsbHidTPIfHidDesc);
2409 pDesc = (const uint8_t *)&g_UsbHidTPIfHidDesc;
2410 break;
2411 default:
2412 cbDesc = 0;
2413 pDesc = 0;
2414 break;
2415 }
2416 /* Returned data is written after the setup message. */
2417 cbCopy = RT_MIN(pSetup->wValue, cbDesc);
2418 LogRelFlow(("usbHidMouse: GET_DESCRIPTOR DT_IF_HID_DESCRIPTOR wValue=%#x wIndex=%#x cbCopy=%#x\n",
2419 pSetup->wValue, pSetup->wIndex,
2420 cbCopy));
2421 return usbHidCompleteOk(pThis, pUrb, pDesc, cbCopy);
2422 }
2423
2424 case DT_IF_HID_REPORT:
2425 {
2426 switch (pThis->enmMode)
2427 {
2428 case USBHIDMODE_ABSOLUTE:
2429 cbDesc = sizeof(g_UsbHidTReportDesc);
2430 pDesc = (const uint8_t *)&g_UsbHidTReportDesc;
2431 break;
2432 case USBHIDMODE_RELATIVE:
2433 cbDesc = sizeof(g_UsbHidMReportDesc);
2434 pDesc = (const uint8_t *)&g_UsbHidMReportDesc;
2435 break;
2436 case USBHIDMODE_MT_ABSOLUTE:
2437 cbDesc = sizeof(g_UsbHidMTReportDesc);
2438 pDesc = (const uint8_t *)&g_UsbHidMTReportDesc;
2439 break;
2440 case USBHIDMODE_MT_RELATIVE:
2441 cbDesc = sizeof(g_UsbHidTPReportDesc);
2442 pDesc = (const uint8_t *)&g_UsbHidTPReportDesc;
2443 break;
2444 default:
2445 cbDesc = 0;
2446 pDesc = 0;
2447 break;
2448 }
2449 /* Returned data is written after the setup message. */
2450 cbCopy = RT_MIN(pSetup->wLength, cbDesc);
2451 LogRelFlow(("usbHid: GET_DESCRIPTOR DT_IF_HID_REPORT wValue=%#x wIndex=%#x cbCopy=%#x\n",
2452 pSetup->wValue, pSetup->wIndex,
2453 cbCopy));
2454 return usbHidCompleteOk(pThis, pUrb, pDesc, cbCopy);
2455 }
2456
2457 default:
2458 LogRelFlow(("usbHid: GET_DESCRIPTOR, huh? wValue=%#x wIndex=%#x\n",
2459 pSetup->wValue, pSetup->wIndex));
2460 break;
2461 }
2462 break;
2463 }
2464
2465 default:
2466 LogRelFlow(("usbHid: Bad GET_DESCRIPTOR req: bmRequestType=%#x\n",
2467 pSetup->bmRequestType));
2468 return usbHidCompleteStall(pThis, pEp, pUrb, "Bad GET_DESCRIPTOR");
2469 }
2470 break;
2471 }
2472
2473 case VUSB_REQ_GET_STATUS:
2474 {
2475 uint16_t wRet = 0;
2476
2477 if (pSetup->wLength != 2)
2478 {
2479 LogRelFlow(("usbHid: Bad GET_STATUS req: wLength=%#x\n",
2480 pSetup->wLength));
2481 break;
2482 }
2483 Assert(pSetup->wValue == 0);
2484 switch (pSetup->bmRequestType)
2485 {
2486 case VUSB_TO_DEVICE | VUSB_REQ_STANDARD | VUSB_DIR_TO_HOST:
2487 {
2488 Assert(pSetup->wIndex == 0);
2489 LogRelFlow(("usbHid: GET_STATUS (device)\n"));
2490 wRet = 0; /* Not self-powered, no remote wakeup. */
2491 return usbHidCompleteOk(pThis, pUrb, &wRet, sizeof(wRet));
2492 }
2493
2494 case VUSB_TO_INTERFACE | VUSB_REQ_STANDARD | VUSB_DIR_TO_HOST:
2495 {
2496 if (pSetup->wIndex == 0)
2497 {
2498 return usbHidCompleteOk(pThis, pUrb, &wRet, sizeof(wRet));
2499 }
2500 LogRelFlow(("usbHid: GET_STATUS (interface) invalid, wIndex=%#x\n", pSetup->wIndex));
2501 break;
2502 }
2503
2504 case VUSB_TO_ENDPOINT | VUSB_REQ_STANDARD | VUSB_DIR_TO_HOST:
2505 {
2506 if (pSetup->wIndex < RT_ELEMENTS(pThis->aEps))
2507 {
2508 wRet = pThis->aEps[pSetup->wIndex].fHalted ? 1 : 0;
2509 return usbHidCompleteOk(pThis, pUrb, &wRet, sizeof(wRet));
2510 }
2511 LogRelFlow(("usbHid: GET_STATUS (endpoint) invalid, wIndex=%#x\n", pSetup->wIndex));
2512 break;
2513 }
2514
2515 default:
2516 LogRelFlow(("usbHid: Bad GET_STATUS req: bmRequestType=%#x\n",
2517 pSetup->bmRequestType));
2518 return usbHidCompleteStall(pThis, pEp, pUrb, "Bad GET_STATUS");
2519 }
2520 break;
2521 }
2522
2523 case VUSB_REQ_CLEAR_FEATURE:
2524 break;
2525 }
2526
2527 /** @todo implement this. */
2528 LogRelFlow(("usbHid: Implement standard request: bmRequestType=%#x bRequest=%#x wValue=%#x wIndex=%#x wLength=%#x\n",
2529 pSetup->bmRequestType, pSetup->bRequest, pSetup->wValue,
2530 pSetup->wIndex, pSetup->wLength));
2531
2532 usbHidCompleteStall(pThis, pEp, pUrb, "TODO: standard request stuff");
2533 }
2534 else if ((pSetup->bmRequestType & VUSB_REQ_MASK) == VUSB_REQ_CLASS)
2535 {
2536 /* Only VUSB_TO_INTERFACE is allowed. */
2537 if ((pSetup->bmRequestType & VUSB_RECIP_MASK) == VUSB_TO_INTERFACE)
2538 {
2539 return usbHidRequestClass(pThis, pEp, pUrb);
2540 }
2541
2542 LogRelFlow(("usbHid: invalid recipient of class req: bmRequestType=%#x bRequest=%#x wValue=%#x wIndex=%#x wLength=%#x\n",
2543 pSetup->bmRequestType, pSetup->bRequest, pSetup->wValue,
2544 pSetup->wIndex, pSetup->wLength));
2545 return usbHidCompleteStall(pThis, pEp, pUrb, "Invalid recip");
2546 }
2547 else
2548 {
2549 LogRelFlow(("usbHid: Unknown control msg: bmRequestType=%#x bRequest=%#x wValue=%#x wIndex=%#x wLength=%#x\n",
2550 pSetup->bmRequestType, pSetup->bRequest, pSetup->wValue,
2551 pSetup->wIndex, pSetup->wLength));
2552 return usbHidCompleteStall(pThis, pEp, pUrb, "Unknown control msg");
2553 }
2554
2555 return VINF_SUCCESS;
2556}
2557
2558
2559/**
2560 * @interface_method_impl{PDMUSBREG,pfnUrbQueue}
2561 */
2562static DECLCALLBACK(int) usbHidQueue(PPDMUSBINS pUsbIns, PVUSBURB pUrb)
2563{
2564 PUSBHID pThis = PDMINS_2_DATA(pUsbIns, PUSBHID);
2565 LogRelFlow(("usbHidQueue/#%u: pUrb=%p:%s EndPt=%#x\n", pUsbIns->iInstance,
2566 pUrb, pUrb->pszDesc, pUrb->EndPt));
2567 RTCritSectEnter(&pThis->CritSect);
2568
2569 /*
2570 * Parse on a per end-point basis.
2571 */
2572 int rc;
2573 switch (pUrb->EndPt)
2574 {
2575 case 0:
2576 rc = usbHidHandleDefaultPipe(pThis, &pThis->aEps[0], pUrb);
2577 break;
2578
2579 case 0x81:
2580 AssertFailed();
2581 RT_FALL_THRU();
2582 case 0x01:
2583 rc = usbHidHandleIntrDevToHost(pThis, &pThis->aEps[1], pUrb);
2584 break;
2585
2586 default:
2587 AssertMsgFailed(("EndPt=%d\n", pUrb->EndPt));
2588 rc = VERR_VUSB_FAILED_TO_QUEUE_URB;
2589 break;
2590 }
2591
2592 RTCritSectLeave(&pThis->CritSect);
2593 return rc;
2594}
2595
2596
2597/**
2598 * @interface_method_impl{PDMUSBREG,pfnUsbClearHaltedEndpoint}
2599 */
2600static DECLCALLBACK(int) usbHidUsbClearHaltedEndpoint(PPDMUSBINS pUsbIns, unsigned uEndpoint)
2601{
2602 PUSBHID pThis = PDMINS_2_DATA(pUsbIns, PUSBHID);
2603 LogRelFlow(("usbHidUsbClearHaltedEndpoint/#%u: uEndpoint=%#x\n",
2604 pUsbIns->iInstance, uEndpoint));
2605
2606 if ((uEndpoint & ~0x80) < RT_ELEMENTS(pThis->aEps))
2607 {
2608 RTCritSectEnter(&pThis->CritSect);
2609 pThis->aEps[(uEndpoint & ~0x80)].fHalted = false;
2610 RTCritSectLeave(&pThis->CritSect);
2611 }
2612
2613 return VINF_SUCCESS;
2614}
2615
2616
2617/**
2618 * @interface_method_impl{PDMUSBREG,pfnUsbSetInterface}
2619 */
2620static DECLCALLBACK(int) usbHidUsbSetInterface(PPDMUSBINS pUsbIns, uint8_t bInterfaceNumber, uint8_t bAlternateSetting)
2621{
2622 LogRelFlow(("usbHidUsbSetInterface/#%u: bInterfaceNumber=%u bAlternateSetting=%u\n",
2623 pUsbIns->iInstance, bInterfaceNumber, bAlternateSetting));
2624 Assert(bAlternateSetting == 0);
2625 return VINF_SUCCESS;
2626}
2627
2628
2629/**
2630 * @interface_method_impl{PDMUSBREG,pfnUsbSetConfiguration}
2631 */
2632static DECLCALLBACK(int) usbHidUsbSetConfiguration(PPDMUSBINS pUsbIns, uint8_t bConfigurationValue,
2633 const void *pvOldCfgDesc, const void *pvOldIfState, const void *pvNewCfgDesc)
2634{
2635 RT_NOREF3(pvOldCfgDesc, pvOldIfState, pvNewCfgDesc);
2636 PUSBHID pThis = PDMINS_2_DATA(pUsbIns, PUSBHID);
2637 LogRelFlow(("usbHidUsbSetConfiguration/#%u: bConfigurationValue=%u\n",
2638 pUsbIns->iInstance, bConfigurationValue));
2639 Assert(bConfigurationValue == 1);
2640 RTCritSectEnter(&pThis->CritSect);
2641
2642 /*
2643 * If the same config is applied more than once, it's a kind of reset.
2644 */
2645 if (pThis->bConfigurationValue == bConfigurationValue)
2646 usbHidResetWorker(pThis, NULL, true /*fSetConfig*/); /** @todo figure out the exact difference */
2647 pThis->bConfigurationValue = bConfigurationValue;
2648
2649 /*
2650 * Set received event type to absolute or relative.
2651 */
2652 pThis->Lun0.pDrv->pfnReportModes(pThis->Lun0.pDrv,
2653 pThis->enmMode == USBHIDMODE_RELATIVE,
2654 pThis->enmMode == USBHIDMODE_ABSOLUTE,
2655 pThis->enmMode == USBHIDMODE_MT_ABSOLUTE,
2656 pThis->enmMode == USBHIDMODE_MT_RELATIVE);
2657
2658 RTCritSectLeave(&pThis->CritSect);
2659 return VINF_SUCCESS;
2660}
2661
2662
2663/**
2664 * @interface_method_impl{PDMUSBREG,pfnUsbGetDescriptorCache}
2665 */
2666static DECLCALLBACK(PCPDMUSBDESCCACHE) usbHidUsbGetDescriptorCache(PPDMUSBINS pUsbIns)
2667{
2668 PUSBHID pThis = PDMINS_2_DATA(pUsbIns, PUSBHID);
2669 LogRelFlow(("usbHidUsbGetDescriptorCache/#%u:\n", pUsbIns->iInstance));
2670 switch (pThis->enmMode)
2671 {
2672 case USBHIDMODE_ABSOLUTE:
2673 return &g_UsbHidTDescCache;
2674 case USBHIDMODE_RELATIVE:
2675 return &g_UsbHidMDescCache;
2676 case USBHIDMODE_MT_ABSOLUTE:
2677 return &g_UsbHidMTDescCache;
2678 case USBHIDMODE_MT_RELATIVE:
2679 return &g_UsbHidTPDescCache;
2680 default:
2681 return NULL;
2682 }
2683}
2684
2685
2686/**
2687 * @interface_method_impl{PDMUSBREG,pfnUsbReset}
2688 */
2689static DECLCALLBACK(int) usbHidUsbReset(PPDMUSBINS pUsbIns, bool fResetOnLinux)
2690{
2691 RT_NOREF1(fResetOnLinux);
2692 PUSBHID pThis = PDMINS_2_DATA(pUsbIns, PUSBHID);
2693 LogRelFlow(("usbHidUsbReset/#%u:\n", pUsbIns->iInstance));
2694 RTCritSectEnter(&pThis->CritSect);
2695
2696 /* We can not handle any input until device is configured again. */
2697 pThis->Lun0.pDrv->pfnReportModes(pThis->Lun0.pDrv, false, false, false, false);
2698
2699 int rc = usbHidResetWorker(pThis, NULL, false /*fSetConfig*/);
2700
2701 RTCritSectLeave(&pThis->CritSect);
2702 return rc;
2703}
2704
2705
2706/**
2707 * @interface_method_impl{PDMUSBREG,pfnDestruct}
2708 */
2709static DECLCALLBACK(void) usbHidDestruct(PPDMUSBINS pUsbIns)
2710{
2711 PDMUSB_CHECK_VERSIONS_RETURN_VOID(pUsbIns);
2712 PUSBHID pThis = PDMINS_2_DATA(pUsbIns, PUSBHID);
2713 LogRelFlow(("usbHidDestruct/#%u:\n", pUsbIns->iInstance));
2714
2715 if (RTCritSectIsInitialized(&pThis->CritSect))
2716 {
2717 RTCritSectEnter(&pThis->CritSect);
2718 RTCritSectLeave(&pThis->CritSect);
2719 RTCritSectDelete(&pThis->CritSect);
2720 }
2721
2722 if (pThis->hEvtDoneQueue != NIL_RTSEMEVENT)
2723 {
2724 RTSemEventDestroy(pThis->hEvtDoneQueue);
2725 pThis->hEvtDoneQueue = NIL_RTSEMEVENT;
2726 }
2727}
2728
2729
2730/**
2731 * @interface_method_impl{PDMUSBREG,pfnConstruct}
2732 */
2733static DECLCALLBACK(int) usbHidConstruct(PPDMUSBINS pUsbIns, int iInstance, PCFGMNODE pCfg, PCFGMNODE pCfgGlobal)
2734{
2735 RT_NOREF1(pCfgGlobal);
2736 PDMUSB_CHECK_VERSIONS_RETURN(pUsbIns);
2737 PUSBHID pThis = PDMINS_2_DATA(pUsbIns, PUSBHID);
2738 PCPDMUSBHLP pHlp = pUsbIns->pHlpR3;
2739
2740 LogRelFlow(("usbHidConstruct/#%u:\n", iInstance));
2741
2742 /*
2743 * Perform the basic structure initialization first so the destructor
2744 * will not misbehave.
2745 */
2746 pThis->pUsbIns = pUsbIns;
2747 pThis->hEvtDoneQueue = NIL_RTSEMEVENT;
2748 usbHidQueueInit(&pThis->ToHostQueue);
2749 usbHidQueueInit(&pThis->DoneQueue);
2750
2751 int rc = RTCritSectInit(&pThis->CritSect);
2752 AssertRCReturn(rc, rc);
2753
2754 rc = RTSemEventCreate(&pThis->hEvtDoneQueue);
2755 AssertRCReturn(rc, rc);
2756
2757 /*
2758 * Validate and read the configuration.
2759 */
2760 rc = pHlp->pfnCFGMValidateConfig(pCfg, "/", "Mode|CoordShift", "Config", "UsbHid", iInstance);
2761 if (RT_FAILURE(rc))
2762 return rc;
2763 char szMode[64];
2764 rc = pHlp->pfnCFGMQueryStringDef(pCfg, "Mode", szMode, sizeof(szMode), "relative");
2765 if (RT_FAILURE(rc))
2766 return PDMUsbHlpVMSetError(pUsbIns, rc, RT_SRC_POS, N_("HID failed to query settings"));
2767 if (!RTStrCmp(szMode, "relative"))
2768 pThis->enmMode = USBHIDMODE_RELATIVE;
2769 else if (!RTStrCmp(szMode, "absolute"))
2770 pThis->enmMode = USBHIDMODE_ABSOLUTE;
2771 else if (!RTStrCmp(szMode, "multitouch"))
2772 pThis->enmMode = USBHIDMODE_MT_ABSOLUTE;
2773 else if (!RTStrCmp(szMode, "touchpad"))
2774 pThis->enmMode = USBHIDMODE_MT_RELATIVE;
2775 else
2776 return PDMUsbHlpVMSetError(pUsbIns, rc, RT_SRC_POS,
2777 N_("Invalid HID device mode"));
2778
2779 LogRelFlow(("usbHidConstruct/#%u: mode '%s'\n", iInstance, szMode));
2780
2781 pThis->Lun0.IBase.pfnQueryInterface = usbHidMouseQueryInterface;
2782 pThis->Lun0.IPort.pfnPutEvent = usbHidMousePutEvent;
2783 pThis->Lun0.IPort.pfnPutEventAbs = usbHidMousePutEventAbs;
2784 pThis->Lun0.IPort.pfnPutEventTouchScreen = usbHidMousePutEventTouchScreen;
2785 pThis->Lun0.IPort.pfnPutEventTouchPad = usbHidMousePutEventTouchPad;
2786
2787 /*
2788 * Attach the mouse driver.
2789 */
2790 rc = PDMUsbHlpDriverAttach(pUsbIns, 0 /*iLun*/, &pThis->Lun0.IBase, &pThis->Lun0.pDrvBase, "Mouse Port");
2791 if (RT_FAILURE(rc))
2792 return PDMUsbHlpVMSetError(pUsbIns, rc, RT_SRC_POS, N_("HID failed to attach mouse driver"));
2793
2794 pThis->Lun0.pDrv = PDMIBASE_QUERY_INTERFACE(pThis->Lun0.pDrvBase, PDMIMOUSECONNECTOR);
2795 if (!pThis->Lun0.pDrv)
2796 return PDMUsbHlpVMSetError(pUsbIns, VERR_PDM_MISSING_INTERFACE, RT_SRC_POS, N_("HID failed to query mouse interface"));
2797
2798 rc = pHlp->pfnCFGMQueryU8Def(pCfg, "CoordShift", &pThis->u8CoordShift, 1);
2799 if (RT_FAILURE(rc))
2800 return PDMUsbHlpVMSetError(pUsbIns, rc, RT_SRC_POS, N_("HID failed to query shift factor"));
2801
2802 return VINF_SUCCESS;
2803}
2804
2805
2806/**
2807 * The USB Human Interface Device (HID) Mouse registration record.
2808 */
2809const PDMUSBREG g_UsbHidMou =
2810{
2811 /* u32Version */
2812 PDM_USBREG_VERSION,
2813 /* szName */
2814 "HidMouse",
2815 /* pszDescription */
2816 "USB HID Mouse.",
2817 /* fFlags */
2818 0,
2819 /* cMaxInstances */
2820 ~0U,
2821 /* cbInstance */
2822 sizeof(USBHID),
2823 /* pfnConstruct */
2824 usbHidConstruct,
2825 /* pfnDestruct */
2826 usbHidDestruct,
2827 /* pfnVMInitComplete */
2828 NULL,
2829 /* pfnVMPowerOn */
2830 NULL,
2831 /* pfnVMReset */
2832 NULL,
2833 /* pfnVMSuspend */
2834 NULL,
2835 /* pfnVMResume */
2836 NULL,
2837 /* pfnVMPowerOff */
2838 NULL,
2839 /* pfnHotPlugged */
2840 NULL,
2841 /* pfnHotUnplugged */
2842 NULL,
2843 /* pfnDriverAttach */
2844 NULL,
2845 /* pfnDriverDetach */
2846 NULL,
2847 /* pfnQueryInterface */
2848 NULL,
2849 /* pfnUsbReset */
2850 usbHidUsbReset,
2851 /* pfnUsbGetDescriptorCache */
2852 usbHidUsbGetDescriptorCache,
2853 /* pfnUsbSetConfiguration */
2854 usbHidUsbSetConfiguration,
2855 /* pfnUsbSetInterface */
2856 usbHidUsbSetInterface,
2857 /* pfnUsbClearHaltedEndpoint */
2858 usbHidUsbClearHaltedEndpoint,
2859 /* pfnUrbNew */
2860 NULL/*usbHidUrbNew*/,
2861 /* pfnUrbQueue */
2862 usbHidQueue,
2863 /* pfnUrbCancel */
2864 usbHidUrbCancel,
2865 /* pfnUrbReap */
2866 usbHidUrbReap,
2867 /* pfnWakeup */
2868 usbHidWakeup,
2869 /* u32TheEnd */
2870 PDM_USBREG_VERSION
2871};
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