/* * Interface Descriptor of wacom devices can be incomplete and * inconsistent so wacom_features table is used to store stylus * device's packet lengths, various maximum values, and tablet * resolution based on product ID's. * * For devices that contain 2 interfaces, wacom_features table is * inaccurate for the touch interface. Since the Interface Descriptor * for touch interfaces has pretty complete data, this function exists * to query tablet for this missing information instead of hard coding in * an additional table. * * A typical Interface Descriptor for a stylus will contain a * boot mouse application collection that is not of interest and this * function will ignore it. * * It also contains a digitizer application collection that also is not * of interest since any information it contains would be duplicate * of what is in wacom_features. Usually it defines a report of an array * of bytes that could be used as max length of the stylus packet returned. * If it happens to define a Digitizer-Stylus Physical Collection then * the X and Y logical values contain valid data but it is ignored. * * A typical Interface Descriptor for a touch interface will contain a * Digitizer-Finger Physical Collection which will define both logical * X/Y maximum as well as the physical size of tablet. Since touch * interfaces haven't supported pressure or distance, this is enough * information to override invalid values in the wacom_features table. * * 3rd gen Bamboo Touch no longer define a Digitizer-Finger Pysical * Collection. Instead they define a Logical Collection with a single * Logical Maximum for both X and Y. * * Intuos5 touch interface does not contain useful data. We deal with * this after returning from this function. */ static int wacom_parse_hid(struct usb_interface *intf, struct hid_descriptor *hid_desc, struct wacom_features *features) { struct usb_device *dev = interface_to_usbdev(intf); char limit = 0; /* result has to be defined as int for some devices */ int result = 0, touch_max = 0; int i = 0, usage = WCM_UNDEFINED, finger = 0, pen = 0; unsigned char *report; report = kzalloc(hid_desc->wDescriptorLength, GFP_KERNEL); if (!report) return -ENOMEM; /* retrive report descriptors */ do { result = usb_control_msg(dev, usb_rcvctrlpipe(dev, 0), USB_REQ_GET_DESCRIPTOR, USB_RECIP_INTERFACE | USB_DIR_IN, HID_DEVICET_REPORT << 8, intf->altsetting[0].desc.bInterfaceNumber, /* interface */ report, hid_desc->wDescriptorLength, 5000); /* 5 secs */ } while (result < 0 && limit++ < WAC_MSG_RETRIES); /* No need to parse the Descriptor. It isn't an error though */ if (result < 0) goto out; for (i = 0; i < hid_desc->wDescriptorLength; i++) { switch (report[i]) { case HID_USAGE_PAGE: switch (report[i + 1]) { case HID_USAGE_PAGE_DIGITIZER: usage = WCM_DIGITIZER; i++; break; case HID_USAGE_PAGE_DESKTOP: usage = WCM_DESKTOP; i++; break; } break; case HID_USAGE: switch (report[i + 1]) { case HID_USAGE_X: if (usage == WCM_DESKTOP) { if (finger) { features->device_type = BTN_TOOL_FINGER; /* touch device at least supports one touch point */ touch_max = 1; switch (features->type) { case TABLETPC2FG: features->pktlen = WACOM_PKGLEN_TPC2FG; break; case MTSCREEN: case WACOM_24HDT: features->pktlen = WACOM_PKGLEN_MTOUCH; break; case MTTPC: features->pktlen = WACOM_PKGLEN_MTTPC; break; case BAMBOO_PT: features->pktlen = WACOM_PKGLEN_BBTOUCH; break; default: features->pktlen = WACOM_PKGLEN_GRAPHIRE; break; } switch (features->type) { case BAMBOO_PT: features->x_phy = get_unaligned_le16(&report[i + 5]); features->x_max = get_unaligned_le16(&report[i + 8]); i += 15; break; case WACOM_24HDT: features->x_max = get_unaligned_le16(&report[i + 3]); features->x_phy = get_unaligned_le16(&report[i + 8]); features->unit = report[i - 1]; features->unitExpo = report[i - 3]; i += 12; break; default: features->x_max = get_unaligned_le16(&report[i + 3]); features->x_phy = get_unaligned_le16(&report[i + 6]); features->unit = report[i + 9]; features->unitExpo = report[i + 11]; i += 12; break; } } else if (pen) { /* penabled only accepts exact bytes of data */ if (features->type >= TABLETPC) features->pktlen = WACOM_PKGLEN_GRAPHIRE; features->device_type = BTN_TOOL_PEN; features->x_max = get_unaligned_le16(&report[i + 3]); i += 4; } } break; case HID_USAGE_Y: if (usage == WCM_DESKTOP) { if (finger) { switch (features->type) { case TABLETPC2FG: case MTSCREEN: case MTTPC: features->y_max = get_unaligned_le16(&report[i + 3]); features->y_phy = get_unaligned_le16(&report[i + 6]); i += 7; break; case WACOM_24HDT: features->y_max = get_unaligned_le16(&report[i + 3]); features->y_phy = get_unaligned_le16(&report[i - 2]); i += 7; break; case BAMBOO_PT: features->y_phy = get_unaligned_le16(&report[i + 3]); features->y_max = get_unaligned_le16(&report[i + 6]); i += 12; break; default: features->y_max = features->x_max; features->y_phy = get_unaligned_le16(&report[i + 3]); i += 4; break; } } else if (pen) { features->y_max = get_unaligned_le16(&report[i + 3]); i += 4; } } break; case HID_USAGE_FINGER: finger = 1; i++; break; /* * Requiring Stylus Usage will ignore boot mouse * X/Y values and some cases of invalid Digitizer X/Y * values commonly reported. */ case HID_USAGE_STYLUS: pen = 1; i++; break; case HID_USAGE_CONTACTMAX: /* leave touch_max as is if predefined */ if (!features->touch_max) wacom_retrieve_report_data(intf, features); i++; break; } break; case HID_COLLECTION_END: /* reset UsagePage and Finger */ finger = usage = 0; break; case HID_COLLECTION: i++; switch (report[i]) { case HID_COLLECTION_LOGICAL: i += wacom_parse_logical_collection(&report[i], features); break; } break; } } out: if (!features->touch_max && touch_max) features->touch_max = touch_max; result = 0; kfree(report); return result; }
static int wacom_parse_hid(struct usb_interface *intf, struct hid_descriptor *hid_desc, struct wacom_features *features) { struct usb_device *dev = interface_to_usbdev(intf); char limit = 0; /* */ int result = 0; int i = 0, usage = WCM_UNDEFINED, finger = 0, pen = 0; unsigned char *report; report = kzalloc(hid_desc->wDescriptorLength, GFP_KERNEL); if (!report) return -ENOMEM; /* */ do { result = usb_control_msg(dev, usb_rcvctrlpipe(dev, 0), USB_REQ_GET_DESCRIPTOR, USB_RECIP_INTERFACE | USB_DIR_IN, HID_DEVICET_REPORT << 8, intf->altsetting[0].desc.bInterfaceNumber, /* */ report, hid_desc->wDescriptorLength, 5000); /* */ } while (result < 0 && limit++ < WAC_MSG_RETRIES); /* */ if (result < 0) goto out; for (i = 0; i < hid_desc->wDescriptorLength; i++) { switch (report[i]) { case HID_USAGE_PAGE: switch (report[i + 1]) { case HID_USAGE_PAGE_DIGITIZER: usage = WCM_DIGITIZER; i++; break; case HID_USAGE_PAGE_DESKTOP: usage = WCM_DESKTOP; i++; break; } break; case HID_USAGE: switch (report[i + 1]) { case HID_USAGE_X: if (usage == WCM_DESKTOP) { if (finger) { features->device_type = BTN_TOOL_FINGER; if (features->type == TABLETPC2FG) { /* */ features->pktlen = WACOM_PKGLEN_TPC2FG; } if (features->type == BAMBOO_PT) { /* */ features->pktlen = WACOM_PKGLEN_BBTOUCH; features->x_phy = get_unaligned_le16(&report[i + 5]); features->x_max = get_unaligned_le16(&report[i + 8]); i += 15; } else { features->x_max = get_unaligned_le16(&report[i + 3]); features->x_phy = get_unaligned_le16(&report[i + 6]); features->unit = report[i + 9]; features->unitExpo = report[i + 11]; i += 12; } } else if (pen) { /* */ if (features->type == TABLETPC2FG) features->pktlen = WACOM_PKGLEN_GRAPHIRE; features->device_type = BTN_TOOL_PEN; features->x_max = get_unaligned_le16(&report[i + 3]); i += 4; } } break; case HID_USAGE_Y: if (usage == WCM_DESKTOP) { if (finger) { features->device_type = BTN_TOOL_FINGER; if (features->type == TABLETPC2FG) { /* */ features->pktlen = WACOM_PKGLEN_TPC2FG; features->y_max = get_unaligned_le16(&report[i + 3]); features->y_phy = get_unaligned_le16(&report[i + 6]); i += 7; } else if (features->type == BAMBOO_PT) { /* */ features->pktlen = WACOM_PKGLEN_BBTOUCH; features->y_phy = get_unaligned_le16(&report[i + 3]); features->y_max = get_unaligned_le16(&report[i + 6]); i += 12; } else { features->y_max = features->x_max; features->y_phy = get_unaligned_le16(&report[i + 3]); i += 4; } } else if (pen) { /* */ if (features->type == TABLETPC2FG) features->pktlen = WACOM_PKGLEN_GRAPHIRE; features->device_type = BTN_TOOL_PEN; features->y_max = get_unaligned_le16(&report[i + 3]); i += 4; } } break; case HID_USAGE_FINGER: finger = 1; i++; break; /* */ case HID_USAGE_STYLUS: pen = 1; i++; break; } break; case HID_COLLECTION_END: /* */ finger = usage = 0; break; case HID_COLLECTION: i++; switch (report[i]) { case HID_COLLECTION_LOGICAL: i += wacom_parse_logical_collection(&report[i], features); break; } break; } } out: result = 0; kfree(report); return result; }