appletouch.c
来自「linux 内核源代码」· C语言 代码 · 共 777 行 · 第 1/2 页
C
777 行
/* * Apple USB Touchpad (for post-February 2005 PowerBooks and MacBooks) driver * * Copyright (C) 2001-2004 Greg Kroah-Hartman (greg@kroah.com) * Copyright (C) 2005 Johannes Berg (johannes@sipsolutions.net) * Copyright (C) 2005 Stelian Pop (stelian@popies.net) * Copyright (C) 2005 Frank Arnold (frank@scirocco-5v-turbo.de) * Copyright (C) 2005 Peter Osterlund (petero2@telia.com) * Copyright (C) 2005 Michael Hanselmann (linux-kernel@hansmi.ch) * Copyright (C) 2006 Nicolas Boichat (nicolas@boichat.ch) * * Thanks to Alex Harper <basilisk@foobox.net> for his inputs. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * */#include <linux/kernel.h>#include <linux/errno.h>#include <linux/init.h>#include <linux/slab.h>#include <linux/module.h>#include <linux/usb/input.h>/* Apple has powerbooks which have the keyboard with different Product IDs */#define APPLE_VENDOR_ID 0x05AC/* These names come from Info.plist in AppleUSBTrackpad.kext */#define FOUNTAIN_ANSI_PRODUCT_ID 0x020E#define FOUNTAIN_ISO_PRODUCT_ID 0x020F#define FOUNTAIN_TP_ONLY_PRODUCT_ID 0x030A#define GEYSER1_TP_ONLY_PRODUCT_ID 0x030B#define GEYSER_ANSI_PRODUCT_ID 0x0214#define GEYSER_ISO_PRODUCT_ID 0x0215#define GEYSER_JIS_PRODUCT_ID 0x0216/* MacBook devices */#define GEYSER3_ANSI_PRODUCT_ID 0x0217#define GEYSER3_ISO_PRODUCT_ID 0x0218#define GEYSER3_JIS_PRODUCT_ID 0x0219/* * Geyser IV: same as Geyser III according to Info.plist in AppleUSBTrackpad.kext * -> same IOClass (AppleUSBGrIIITrackpad), same acceleration tables */#define GEYSER4_ANSI_PRODUCT_ID 0x021A#define GEYSER4_ISO_PRODUCT_ID 0x021B#define GEYSER4_JIS_PRODUCT_ID 0x021C#define ATP_DEVICE(prod) \ .match_flags = USB_DEVICE_ID_MATCH_DEVICE | \ USB_DEVICE_ID_MATCH_INT_CLASS | \ USB_DEVICE_ID_MATCH_INT_PROTOCOL, \ .idVendor = APPLE_VENDOR_ID, \ .idProduct = (prod), \ .bInterfaceClass = 0x03, \ .bInterfaceProtocol = 0x02/* table of devices that work with this driver */static struct usb_device_id atp_table [] = { { ATP_DEVICE(FOUNTAIN_ANSI_PRODUCT_ID) }, { ATP_DEVICE(FOUNTAIN_ISO_PRODUCT_ID) }, { ATP_DEVICE(FOUNTAIN_TP_ONLY_PRODUCT_ID) }, { ATP_DEVICE(GEYSER1_TP_ONLY_PRODUCT_ID) }, /* PowerBooks Oct 2005 */ { ATP_DEVICE(GEYSER_ANSI_PRODUCT_ID) }, { ATP_DEVICE(GEYSER_ISO_PRODUCT_ID) }, { ATP_DEVICE(GEYSER_JIS_PRODUCT_ID) }, /* Core Duo MacBook & MacBook Pro */ { ATP_DEVICE(GEYSER3_ANSI_PRODUCT_ID) }, { ATP_DEVICE(GEYSER3_ISO_PRODUCT_ID) }, { ATP_DEVICE(GEYSER3_JIS_PRODUCT_ID) }, /* Core2 Duo MacBook & MacBook Pro */ { ATP_DEVICE(GEYSER4_ANSI_PRODUCT_ID) }, { ATP_DEVICE(GEYSER4_ISO_PRODUCT_ID) }, { ATP_DEVICE(GEYSER4_JIS_PRODUCT_ID) }, /* Terminating entry */ { }};MODULE_DEVICE_TABLE (usb, atp_table);/* * number of sensors. Note that only 16 instead of 26 X (horizontal) * sensors exist on 12" and 15" PowerBooks. All models have 16 Y * (vertical) sensors. */#define ATP_XSENSORS 26#define ATP_YSENSORS 16/* amount of fuzz this touchpad generates */#define ATP_FUZZ 16/* maximum pressure this driver will report */#define ATP_PRESSURE 300/* * multiplication factor for the X and Y coordinates. * We try to keep the touchpad aspect ratio while still doing only simple * arithmetics. * The factors below give coordinates like: * 0 <= x < 960 on 12" and 15" Powerbooks * 0 <= x < 1600 on 17" Powerbooks * 0 <= y < 646 */#define ATP_XFACT 64#define ATP_YFACT 43/* * Threshold for the touchpad sensors. Any change less than ATP_THRESHOLD is * ignored. */#define ATP_THRESHOLD 5/* Geyser initialization constants */#define ATP_GEYSER_MODE_READ_REQUEST_ID 1#define ATP_GEYSER_MODE_WRITE_REQUEST_ID 9#define ATP_GEYSER_MODE_REQUEST_VALUE 0x300#define ATP_GEYSER_MODE_REQUEST_INDEX 0#define ATP_GEYSER_MODE_VENDOR_VALUE 0x04/* Structure to hold all of our device specific stuff */struct atp { char phys[64]; struct usb_device * udev; /* usb device */ struct urb * urb; /* usb request block */ signed char * data; /* transferred data */ struct input_dev * input; /* input dev */ unsigned char open; /* non-zero if opened */ unsigned char valid; /* are the sensors valid ? */ unsigned char size_detect_done; unsigned char overflowwarn; /* overflow warning printed? */ int x_old; /* last reported x/y, */ int y_old; /* used for smoothing */ /* current value of the sensors */ signed char xy_cur[ATP_XSENSORS + ATP_YSENSORS]; /* last value of the sensors */ signed char xy_old[ATP_XSENSORS + ATP_YSENSORS]; /* accumulated sensors */ int xy_acc[ATP_XSENSORS + ATP_YSENSORS]; int datalen; /* size of an USB urb transfer */ int idlecount; /* number of empty packets */ struct work_struct work;};#define dbg_dump(msg, tab) \ if (debug > 1) { \ int i; \ printk("appletouch: %s %lld", msg, (long long)jiffies); \ for (i = 0; i < ATP_XSENSORS + ATP_YSENSORS; i++) \ printk(" %02x", tab[i]); \ printk("\n"); \ }#define dprintk(format, a...) \ do { \ if (debug) printk(KERN_DEBUG format, ##a); \ } while (0)MODULE_AUTHOR("Johannes Berg, Stelian Pop, Frank Arnold, Michael Hanselmann");MODULE_DESCRIPTION("Apple PowerBooks USB touchpad driver");MODULE_LICENSE("GPL");/* * Make the threshold a module parameter */static int threshold = ATP_THRESHOLD;module_param(threshold, int, 0644);MODULE_PARM_DESC(threshold, "Discards any change in data from a sensor (trackpad has hundreds of these sensors) less than this value");static int debug = 1;module_param(debug, int, 0644);MODULE_PARM_DESC(debug, "Activate debugging output");static inline int atp_is_fountain(struct atp *dev){ u16 productId = le16_to_cpu(dev->udev->descriptor.idProduct); return productId == FOUNTAIN_ANSI_PRODUCT_ID || productId == FOUNTAIN_ISO_PRODUCT_ID || productId == FOUNTAIN_TP_ONLY_PRODUCT_ID;}/* Checks if the device a Geyser 2 (ANSI, ISO, JIS) */static inline int atp_is_geyser_2(struct atp *dev){ u16 productId = le16_to_cpu(dev->udev->descriptor.idProduct); return (productId == GEYSER_ANSI_PRODUCT_ID) || (productId == GEYSER_ISO_PRODUCT_ID) || (productId == GEYSER_JIS_PRODUCT_ID);}static inline int atp_is_geyser_3(struct atp *dev){ u16 productId = le16_to_cpu(dev->udev->descriptor.idProduct); return (productId == GEYSER3_ANSI_PRODUCT_ID) || (productId == GEYSER3_ISO_PRODUCT_ID) || (productId == GEYSER3_JIS_PRODUCT_ID) || (productId == GEYSER4_ANSI_PRODUCT_ID) || (productId == GEYSER4_ISO_PRODUCT_ID) || (productId == GEYSER4_JIS_PRODUCT_ID);}/* * By default newer Geyser devices send standard USB HID mouse * packets (Report ID 2). This code changes device mode, so it * sends raw sensor reports (Report ID 5). */static int atp_geyser_init(struct usb_device *udev){ char data[8]; int size; size = usb_control_msg(udev, usb_rcvctrlpipe(udev, 0), ATP_GEYSER_MODE_READ_REQUEST_ID, USB_DIR_IN | USB_TYPE_CLASS | USB_RECIP_INTERFACE, ATP_GEYSER_MODE_REQUEST_VALUE, ATP_GEYSER_MODE_REQUEST_INDEX, &data, 8, 5000); if (size != 8) { err("Could not do mode read request from device" " (Geyser Raw mode)"); return -EIO; } /* Apply the mode switch */ data[0] = ATP_GEYSER_MODE_VENDOR_VALUE; size = usb_control_msg(udev, usb_sndctrlpipe(udev, 0), ATP_GEYSER_MODE_WRITE_REQUEST_ID, USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE, ATP_GEYSER_MODE_REQUEST_VALUE, ATP_GEYSER_MODE_REQUEST_INDEX, &data, 8, 5000); if (size != 8) { err("Could not do mode write request to device" " (Geyser Raw mode)"); return -EIO; } return 0;}/* * Reinitialise the device. This usually stops stream of empty packets * coming from it. */static void atp_reinit(struct work_struct *work){ struct atp *dev = container_of(work, struct atp, work); struct usb_device *udev = dev->udev; int retval; dev->idlecount = 0; atp_geyser_init(udev); retval = usb_submit_urb(dev->urb, GFP_ATOMIC); if (retval) { err("%s - usb_submit_urb failed with result %d", __FUNCTION__, retval); }}static int atp_calculate_abs(int *xy_sensors, int nb_sensors, int fact, int *z, int *fingers){ int i; /* values to calculate mean */ int pcum = 0, psum = 0; int is_increasing = 0; *fingers = 0; for (i = 0; i < nb_sensors; i++) { if (xy_sensors[i] < threshold) { if (is_increasing) is_increasing = 0; continue; } /* * Makes the finger detection more versatile. For example, * two fingers with no gap will be detected. Also, my * tests show it less likely to have intermittent loss * of multiple finger readings while moving around (scrolling). * * Changes the multiple finger detection to counting humps on * sensors (transitions from nonincreasing to increasing) * instead of counting transitions from low sensors (no * finger reading) to high sensors (finger above * sensor) * * - Jason Parekh <jasonparekh@gmail.com> */ if (i < 1 || (!is_increasing && xy_sensors[i - 1] < xy_sensors[i])) { (*fingers)++; is_increasing = 1; } else if (i > 0 && xy_sensors[i - 1] >= xy_sensors[i]) { is_increasing = 0; } /* * Subtracts threshold so a high sensor that just passes the threshold * won't skew the calculated absolute coordinate. Fixes an issue * where slowly moving the mouse would occassionaly jump a number of * pixels (let me restate--slowly moving the mouse makes this issue * most apparent). */ pcum += (xy_sensors[i] - threshold) * i; psum += (xy_sensors[i] - threshold); } if (psum > 0) { *z = psum; return pcum * fact / psum; } return 0;}static inline void atp_report_fingers(struct input_dev *input, int fingers){ input_report_key(input, BTN_TOOL_FINGER, fingers == 1); input_report_key(input, BTN_TOOL_DOUBLETAP, fingers == 2); input_report_key(input, BTN_TOOL_TRIPLETAP, fingers > 2);}static void atp_complete(struct urb* urb){ int x, y, x_z, y_z, x_f, y_f; int retval, i, j; int key; struct atp *dev = urb->context; switch (urb->status) { case 0: /* success */ break; case -EOVERFLOW: if(!dev->overflowwarn) { printk(KERN_WARNING "appletouch: OVERFLOW with data " "length %d, actual length is %d\n", dev->datalen, dev->urb->actual_length); dev->overflowwarn = 1; } case -ECONNRESET: case -ENOENT: case -ESHUTDOWN: /* This urb is terminated, clean up */ dbg("%s - urb shutting down with status: %d", __FUNCTION__, urb->status); return; default: dbg("%s - nonzero urb status received: %d", __FUNCTION__, urb->status); goto exit; } /* drop incomplete datasets */ if (dev->urb->actual_length != dev->datalen) { dprintk("appletouch: incomplete data package" " (first byte: %d, length: %d).\n", dev->data[0], dev->urb->actual_length); goto exit; } /* reorder the sensors values */ if (atp_is_geyser_3(dev)) { memset(dev->xy_cur, 0, sizeof(dev->xy_cur));
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?