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 + -
显示快捷键?