2005-09-08 02:19:48 -06:00
|
|
|
/*
|
2006-04-19 15:36:40 -06:00
|
|
|
* Apple USB Touchpad (for post-February 2005 PowerBooks and MacBooks) driver
|
2005-09-08 02:19:48 -06:00
|
|
|
*
|
|
|
|
* 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)
|
2005-12-20 22:50:23 -07:00
|
|
|
* Copyright (C) 2005 Michael Hanselmann (linux-kernel@hansmi.ch)
|
2006-04-19 15:36:40 -06:00
|
|
|
* Copyright (C) 2006 Nicolas Boichat (nicolas@boichat.ch)
|
2005-09-08 02:19:48 -06:00
|
|
|
*
|
|
|
|
* 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>
|
2006-06-13 11:04:34 -06:00
|
|
|
#include <linux/usb/input.h>
|
2005-09-08 02:19:48 -06:00
|
|
|
|
|
|
|
/* Apple has powerbooks which have the keyboard with different Product IDs */
|
|
|
|
#define APPLE_VENDOR_ID 0x05AC
|
|
|
|
|
2005-12-20 22:50:23 -07:00
|
|
|
/* These names come from Info.plist in AppleUSBTrackpad.kext */
|
2006-11-16 23:06:25 -07:00
|
|
|
#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
|
2005-12-20 22:50:23 -07:00
|
|
|
|
2006-04-19 15:36:40 -06:00
|
|
|
/* MacBook devices */
|
2006-11-16 23:06:25 -07:00
|
|
|
#define GEYSER3_ANSI_PRODUCT_ID 0x0217
|
|
|
|
#define GEYSER3_ISO_PRODUCT_ID 0x0218
|
|
|
|
#define GEYSER3_JIS_PRODUCT_ID 0x0219
|
2006-04-19 15:36:40 -06:00
|
|
|
|
2006-11-16 23:06:13 -07:00
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
|
2005-09-08 02:19:48 -06:00
|
|
|
#define ATP_DEVICE(prod) \
|
2005-09-15 01:01:47 -06:00
|
|
|
.match_flags = USB_DEVICE_ID_MATCH_DEVICE | \
|
2005-09-08 02:19:48 -06:00
|
|
|
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 [] = {
|
2006-11-16 23:06:25 -07:00
|
|
|
{ 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) },
|
2005-12-20 22:50:23 -07:00
|
|
|
|
|
|
|
/* PowerBooks Oct 2005 */
|
|
|
|
{ ATP_DEVICE(GEYSER_ANSI_PRODUCT_ID) },
|
|
|
|
{ ATP_DEVICE(GEYSER_ISO_PRODUCT_ID) },
|
|
|
|
{ ATP_DEVICE(GEYSER_JIS_PRODUCT_ID) },
|
|
|
|
|
2006-11-16 23:06:13 -07:00
|
|
|
/* Core Duo MacBook & MacBook Pro */
|
2006-04-19 15:36:40 -06:00
|
|
|
{ ATP_DEVICE(GEYSER3_ANSI_PRODUCT_ID) },
|
|
|
|
{ ATP_DEVICE(GEYSER3_ISO_PRODUCT_ID) },
|
|
|
|
{ ATP_DEVICE(GEYSER3_JIS_PRODUCT_ID) },
|
|
|
|
|
2006-11-16 23:06:13 -07:00
|
|
|
/* Core2 Duo MacBook & MacBook Pro */
|
|
|
|
{ ATP_DEVICE(GEYSER4_ANSI_PRODUCT_ID) },
|
|
|
|
{ ATP_DEVICE(GEYSER4_ISO_PRODUCT_ID) },
|
|
|
|
{ ATP_DEVICE(GEYSER4_JIS_PRODUCT_ID) },
|
|
|
|
|
2005-12-20 22:50:23 -07:00
|
|
|
/* Terminating entry */
|
|
|
|
{ }
|
2005-09-08 02:19:48 -06:00
|
|
|
};
|
|
|
|
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:
|
2005-09-15 01:01:47 -06:00
|
|
|
* 0 <= x < 960 on 12" and 15" Powerbooks
|
|
|
|
* 0 <= x < 1600 on 17" Powerbooks
|
|
|
|
* 0 <= y < 646
|
2005-09-08 02:19:48 -06:00
|
|
|
*/
|
|
|
|
#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
|
|
|
|
|
2006-11-16 23:06:13 -07:00
|
|
|
/* MacBook Pro (Geyser 3 & 4) initialization constants */
|
2006-04-19 15:36:40 -06:00
|
|
|
#define ATP_GEYSER3_MODE_READ_REQUEST_ID 1
|
|
|
|
#define ATP_GEYSER3_MODE_WRITE_REQUEST_ID 9
|
|
|
|
#define ATP_GEYSER3_MODE_REQUEST_VALUE 0x300
|
|
|
|
#define ATP_GEYSER3_MODE_REQUEST_INDEX 0
|
|
|
|
#define ATP_GEYSER3_MODE_VENDOR_VALUE 0x04
|
|
|
|
|
2005-09-08 02:19:48 -06:00
|
|
|
/* Structure to hold all of our device specific stuff */
|
|
|
|
struct atp {
|
2005-09-15 01:01:47 -06:00
|
|
|
char phys[64];
|
2005-09-08 02:19:48 -06:00
|
|
|
struct usb_device * udev; /* usb device */
|
|
|
|
struct urb * urb; /* usb request block */
|
|
|
|
signed char * data; /* transferred data */
|
|
|
|
int open; /* non-zero if opened */
|
2005-09-15 01:01:47 -06:00
|
|
|
struct input_dev *input; /* input dev */
|
2005-09-08 02:19:48 -06:00
|
|
|
int valid; /* are the sensors valid ? */
|
|
|
|
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];
|
2005-12-20 22:50:23 -07:00
|
|
|
int overflowwarn; /* overflow warning printed? */
|
|
|
|
int datalen; /* size of an USB urb transfer */
|
2007-07-19 22:29:32 -06:00
|
|
|
int idlecount; /* number of empty packets */
|
|
|
|
struct work_struct work;
|
2005-09-08 02:19:48 -06:00
|
|
|
};
|
|
|
|
|
|
|
|
#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++) \
|
2005-09-15 01:01:47 -06:00
|
|
|
printk(" %02x", tab[i]); \
|
|
|
|
printk("\n"); \
|
2005-09-08 02:19:48 -06:00
|
|
|
}
|
|
|
|
|
2005-09-15 01:01:47 -06:00
|
|
|
#define dprintk(format, a...) \
|
2005-09-08 02:19:48 -06:00
|
|
|
do { \
|
|
|
|
if (debug) printk(format, ##a); \
|
|
|
|
} while (0)
|
|
|
|
|
2005-12-20 22:50:23 -07:00
|
|
|
MODULE_AUTHOR("Johannes Berg, Stelian Pop, Frank Arnold, Michael Hanselmann");
|
2005-09-08 02:19:48 -06:00
|
|
|
MODULE_DESCRIPTION("Apple PowerBooks USB touchpad driver");
|
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
|
2006-11-16 23:05:58 -07:00
|
|
|
/*
|
|
|
|
* 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");
|
|
|
|
|
2005-09-08 02:19:48 -06:00
|
|
|
static int debug = 1;
|
|
|
|
module_param(debug, int, 0644);
|
|
|
|
MODULE_PARM_DESC(debug, "Activate debugging output");
|
|
|
|
|
2005-12-20 22:50:23 -07:00
|
|
|
/* Checks if the device a Geyser 2 (ANSI, ISO, JIS) */
|
|
|
|
static inline int atp_is_geyser_2(struct atp *dev)
|
|
|
|
{
|
2006-04-19 15:36:40 -06:00
|
|
|
u16 productId = le16_to_cpu(dev->udev->descriptor.idProduct);
|
2005-12-20 22:50:23 -07:00
|
|
|
|
|
|
|
return (productId == GEYSER_ANSI_PRODUCT_ID) ||
|
|
|
|
(productId == GEYSER_ISO_PRODUCT_ID) ||
|
|
|
|
(productId == GEYSER_JIS_PRODUCT_ID);
|
|
|
|
}
|
|
|
|
|
2006-04-19 15:36:40 -06:00
|
|
|
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) ||
|
2006-11-16 23:06:13 -07:00
|
|
|
(productId == GEYSER3_JIS_PRODUCT_ID) ||
|
|
|
|
(productId == GEYSER4_ANSI_PRODUCT_ID) ||
|
|
|
|
(productId == GEYSER4_ISO_PRODUCT_ID) ||
|
|
|
|
(productId == GEYSER4_JIS_PRODUCT_ID);
|
2006-04-19 15:36:40 -06:00
|
|
|
}
|
|
|
|
|
2007-07-19 22:29:32 -06:00
|
|
|
/*
|
|
|
|
* By default Geyser 3 device sends 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_geyser3_init(struct usb_device *udev)
|
|
|
|
{
|
|
|
|
char data[8];
|
|
|
|
int size;
|
|
|
|
|
|
|
|
size = usb_control_msg(udev, usb_rcvctrlpipe(udev, 0),
|
|
|
|
ATP_GEYSER3_MODE_READ_REQUEST_ID,
|
|
|
|
USB_DIR_IN | USB_TYPE_CLASS | USB_RECIP_INTERFACE,
|
|
|
|
ATP_GEYSER3_MODE_REQUEST_VALUE,
|
|
|
|
ATP_GEYSER3_MODE_REQUEST_INDEX, &data, 8, 5000);
|
|
|
|
|
|
|
|
if (size != 8) {
|
|
|
|
err("Could not do mode read request from device"
|
|
|
|
" (Geyser 3 mode)");
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Apply the mode switch */
|
|
|
|
data[0] = ATP_GEYSER3_MODE_VENDOR_VALUE;
|
|
|
|
|
|
|
|
size = usb_control_msg(udev, usb_sndctrlpipe(udev, 0),
|
|
|
|
ATP_GEYSER3_MODE_WRITE_REQUEST_ID,
|
|
|
|
USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE,
|
|
|
|
ATP_GEYSER3_MODE_REQUEST_VALUE,
|
|
|
|
ATP_GEYSER3_MODE_REQUEST_INDEX, &data, 8, 5000);
|
|
|
|
|
|
|
|
if (size != 8) {
|
|
|
|
err("Could not do mode write request to device"
|
|
|
|
" (Geyser 3 mode)");
|
|
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reinitialise the device if it's a geyser 3 */
|
|
|
|
static void atp_reinit(struct work_struct *work)
|
|
|
|
{
|
|
|
|
struct atp *dev = container_of(work, struct atp, work);
|
|
|
|
struct usb_device *udev = dev->udev;
|
|
|
|
|
|
|
|
dev->idlecount = 0;
|
|
|
|
atp_geyser3_init(udev);
|
|
|
|
}
|
|
|
|
|
2005-09-08 02:19:48 -06:00
|
|
|
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;
|
2006-11-16 23:05:58 -07:00
|
|
|
int is_increasing = 0;
|
2005-09-08 02:19:48 -06:00
|
|
|
|
|
|
|
*fingers = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < nb_sensors; i++) {
|
2006-11-16 23:05:58 -07:00
|
|
|
if (xy_sensors[i] < threshold) {
|
|
|
|
if (is_increasing)
|
|
|
|
is_increasing = 0;
|
|
|
|
|
2005-09-08 02:19:48 -06:00
|
|
|
continue;
|
2006-11-16 23:05:58 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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])) {
|
2005-09-08 02:19:48 -06:00
|
|
|
(*fingers)++;
|
2006-11-16 23:05:58 -07:00
|
|
|
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);
|
2005-09-08 02:19:48 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 07:55:46 -06:00
|
|
|
static void atp_complete(struct urb* urb)
|
2005-09-08 02:19:48 -06:00
|
|
|
{
|
|
|
|
int x, y, x_z, y_z, x_f, y_f;
|
2005-12-20 22:50:23 -07:00
|
|
|
int retval, i, j;
|
2007-09-24 22:06:25 -06:00
|
|
|
int key;
|
2005-09-08 02:19:48 -06:00
|
|
|
struct atp *dev = urb->context;
|
|
|
|
|
|
|
|
switch (urb->status) {
|
|
|
|
case 0:
|
|
|
|
/* success */
|
|
|
|
break;
|
2005-12-20 22:50:23 -07:00
|
|
|
case -EOVERFLOW:
|
|
|
|
if(!dev->overflowwarn) {
|
|
|
|
printk("appletouch: OVERFLOW with data "
|
|
|
|
"length %d, actual length is %d\n",
|
|
|
|
dev->datalen, dev->urb->actual_length);
|
|
|
|
dev->overflowwarn = 1;
|
|
|
|
}
|
2005-09-08 02:19:48 -06:00
|
|
|
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 */
|
2005-12-20 22:50:23 -07:00
|
|
|
if (dev->urb->actual_length != dev->datalen) {
|
2006-04-19 15:36:40 -06:00
|
|
|
dprintk("appletouch: incomplete data package"
|
|
|
|
" (first byte: %d, length: %d).\n",
|
|
|
|
dev->data[0], dev->urb->actual_length);
|
2005-09-08 02:19:48 -06:00
|
|
|
goto exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* reorder the sensors values */
|
2006-04-19 15:36:40 -06:00
|
|
|
if (atp_is_geyser_3(dev)) {
|
|
|
|
memset(dev->xy_cur, 0, sizeof(dev->xy_cur));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The values are laid out like this:
|
|
|
|
* -, Y1, Y2, -, Y3, Y4, -, ..., -, X1, X2, -, X3, X4, ...
|
|
|
|
* '-' is an unused value.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* read X values */
|
|
|
|
for (i = 0, j = 19; i < 20; i += 2, j += 3) {
|
|
|
|
dev->xy_cur[i] = dev->data[j + 1];
|
|
|
|
dev->xy_cur[i + 1] = dev->data[j + 2];
|
|
|
|
}
|
|
|
|
/* read Y values */
|
|
|
|
for (i = 0, j = 1; i < 9; i += 2, j += 3) {
|
|
|
|
dev->xy_cur[ATP_XSENSORS + i] = dev->data[j + 1];
|
|
|
|
dev->xy_cur[ATP_XSENSORS + i + 1] = dev->data[j + 2];
|
|
|
|
}
|
|
|
|
} else if (atp_is_geyser_2(dev)) {
|
2005-12-20 22:50:23 -07:00
|
|
|
memset(dev->xy_cur, 0, sizeof(dev->xy_cur));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The values are laid out like this:
|
|
|
|
* Y1, Y2, -, Y3, Y4, -, ..., X1, X2, -, X3, X4, -, ...
|
|
|
|
* '-' is an unused value.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* read X values */
|
|
|
|
for (i = 0, j = 19; i < 20; i += 2, j += 3) {
|
|
|
|
dev->xy_cur[i] = dev->data[j];
|
|
|
|
dev->xy_cur[i + 1] = dev->data[j + 1];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* read Y values */
|
|
|
|
for (i = 0, j = 1; i < 9; i += 2, j += 3) {
|
|
|
|
dev->xy_cur[ATP_XSENSORS + i] = dev->data[j];
|
|
|
|
dev->xy_cur[ATP_XSENSORS + i + 1] = dev->data[j + 1];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for (i = 0; i < 8; i++) {
|
|
|
|
/* X values */
|
|
|
|
dev->xy_cur[i ] = dev->data[5 * i + 2];
|
|
|
|
dev->xy_cur[i + 8] = dev->data[5 * i + 4];
|
|
|
|
dev->xy_cur[i + 16] = dev->data[5 * i + 42];
|
|
|
|
if (i < 2)
|
|
|
|
dev->xy_cur[i + 24] = dev->data[5 * i + 44];
|
|
|
|
|
|
|
|
/* Y values */
|
|
|
|
dev->xy_cur[i + 26] = dev->data[5 * i + 1];
|
|
|
|
dev->xy_cur[i + 34] = dev->data[5 * i + 3];
|
|
|
|
}
|
2005-09-08 02:19:48 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
dbg_dump("sample", dev->xy_cur);
|
|
|
|
|
|
|
|
if (!dev->valid) {
|
|
|
|
/* first sample */
|
|
|
|
dev->valid = 1;
|
|
|
|
dev->x_old = dev->y_old = -1;
|
|
|
|
memcpy(dev->xy_old, dev->xy_cur, sizeof(dev->xy_old));
|
|
|
|
|
2006-04-19 15:36:40 -06:00
|
|
|
if (atp_is_geyser_3(dev)) /* No 17" Macbooks (yet) */
|
|
|
|
goto exit;
|
|
|
|
|
2005-12-20 22:50:23 -07:00
|
|
|
/* 17" Powerbooks have extra X sensors */
|
|
|
|
for (i = (atp_is_geyser_2(dev)?15:16); i < ATP_XSENSORS; i++) {
|
|
|
|
if (!dev->xy_cur[i]) continue;
|
|
|
|
|
|
|
|
printk("appletouch: 17\" model detected.\n");
|
|
|
|
if(atp_is_geyser_2(dev))
|
|
|
|
input_set_abs_params(dev->input, ABS_X, 0,
|
|
|
|
(20 - 1) *
|
|
|
|
ATP_XFACT - 1,
|
|
|
|
ATP_FUZZ, 0);
|
|
|
|
else
|
2005-09-15 01:01:47 -06:00
|
|
|
input_set_abs_params(dev->input, ABS_X, 0,
|
|
|
|
(ATP_XSENSORS - 1) *
|
2005-09-08 02:19:48 -06:00
|
|
|
ATP_XFACT - 1,
|
|
|
|
ATP_FUZZ, 0);
|
2005-12-20 22:50:23 -07:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2005-09-08 02:19:48 -06:00
|
|
|
|
|
|
|
goto exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < ATP_XSENSORS + ATP_YSENSORS; i++) {
|
|
|
|
/* accumulate the change */
|
|
|
|
signed char change = dev->xy_old[i] - dev->xy_cur[i];
|
|
|
|
dev->xy_acc[i] -= change;
|
|
|
|
|
|
|
|
/* prevent down drifting */
|
|
|
|
if (dev->xy_acc[i] < 0)
|
|
|
|
dev->xy_acc[i] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(dev->xy_old, dev->xy_cur, sizeof(dev->xy_old));
|
|
|
|
|
|
|
|
dbg_dump("accumulator", dev->xy_acc);
|
|
|
|
|
|
|
|
x = atp_calculate_abs(dev->xy_acc, ATP_XSENSORS,
|
|
|
|
ATP_XFACT, &x_z, &x_f);
|
|
|
|
y = atp_calculate_abs(dev->xy_acc + ATP_XSENSORS, ATP_YSENSORS,
|
|
|
|
ATP_YFACT, &y_z, &y_f);
|
2007-09-24 22:06:25 -06:00
|
|
|
key = dev->data[dev->datalen - 1] & 1;
|
2005-09-08 02:19:48 -06:00
|
|
|
|
|
|
|
if (x && y) {
|
|
|
|
if (dev->x_old != -1) {
|
|
|
|
x = (dev->x_old * 3 + x) >> 2;
|
|
|
|
y = (dev->y_old * 3 + y) >> 2;
|
|
|
|
dev->x_old = x;
|
|
|
|
dev->y_old = y;
|
|
|
|
|
|
|
|
if (debug > 1)
|
|
|
|
printk("appletouch: X: %3d Y: %3d "
|
|
|
|
"Xz: %3d Yz: %3d\n",
|
|
|
|
x, y, x_z, y_z);
|
|
|
|
|
2005-09-15 01:01:47 -06:00
|
|
|
input_report_key(dev->input, BTN_TOUCH, 1);
|
|
|
|
input_report_abs(dev->input, ABS_X, x);
|
|
|
|
input_report_abs(dev->input, ABS_Y, y);
|
|
|
|
input_report_abs(dev->input, ABS_PRESSURE,
|
2005-09-08 02:19:48 -06:00
|
|
|
min(ATP_PRESSURE, x_z + y_z));
|
2005-09-15 01:01:47 -06:00
|
|
|
atp_report_fingers(dev->input, max(x_f, y_f));
|
2005-09-08 02:19:48 -06:00
|
|
|
}
|
|
|
|
dev->x_old = x;
|
|
|
|
dev->y_old = y;
|
2007-07-19 22:29:32 -06:00
|
|
|
|
|
|
|
} else if (!x && !y) {
|
2005-09-08 02:19:48 -06:00
|
|
|
|
|
|
|
dev->x_old = dev->y_old = -1;
|
2005-09-15 01:01:47 -06:00
|
|
|
input_report_key(dev->input, BTN_TOUCH, 0);
|
|
|
|
input_report_abs(dev->input, ABS_PRESSURE, 0);
|
|
|
|
atp_report_fingers(dev->input, 0);
|
2005-09-08 02:19:48 -06:00
|
|
|
|
|
|
|
/* reset the accumulator on release */
|
|
|
|
memset(dev->xy_acc, 0, sizeof(dev->xy_acc));
|
2007-10-12 22:31:15 -06:00
|
|
|
}
|
|
|
|
|
2007-10-21 22:59:59 -06:00
|
|
|
input_report_key(dev->input, BTN_LEFT, key);
|
|
|
|
input_sync(dev->input);
|
|
|
|
|
|
|
|
/* Many Geysers will continue to send packets continually after
|
2007-10-12 22:31:15 -06:00
|
|
|
the first touch unless reinitialised. Do so if it's been
|
|
|
|
idle for a while in order to avoid waking the kernel up
|
|
|
|
several hundred times a second */
|
2005-09-08 02:19:48 -06:00
|
|
|
|
2007-10-21 22:59:59 -06:00
|
|
|
if (!x && !y && !key) {
|
|
|
|
dev->idlecount++;
|
|
|
|
if (dev->idlecount == 10) {
|
|
|
|
dev->valid = 0;
|
|
|
|
schedule_work(&dev->work);
|
2007-07-19 22:29:32 -06:00
|
|
|
}
|
2007-10-21 22:59:59 -06:00
|
|
|
} else
|
|
|
|
dev->idlecount = 0;
|
2005-09-08 02:19:48 -06:00
|
|
|
|
|
|
|
exit:
|
|
|
|
retval = usb_submit_urb(dev->urb, GFP_ATOMIC);
|
|
|
|
if (retval) {
|
|
|
|
err("%s - usb_submit_urb failed with result %d",
|
|
|
|
__FUNCTION__, retval);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int atp_open(struct input_dev *input)
|
|
|
|
{
|
2007-04-11 23:34:39 -06:00
|
|
|
struct atp *dev = input_get_drvdata(input);
|
2005-09-08 02:19:48 -06:00
|
|
|
|
|
|
|
if (usb_submit_urb(dev->urb, GFP_ATOMIC))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
dev->open = 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void atp_close(struct input_dev *input)
|
|
|
|
{
|
2007-04-11 23:34:39 -06:00
|
|
|
struct atp *dev = input_get_drvdata(input);
|
2005-09-08 02:19:48 -06:00
|
|
|
|
|
|
|
usb_kill_urb(dev->urb);
|
2007-07-19 22:29:32 -06:00
|
|
|
cancel_work_sync(&dev->work);
|
2005-09-08 02:19:48 -06:00
|
|
|
dev->open = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int atp_probe(struct usb_interface *iface, const struct usb_device_id *id)
|
|
|
|
{
|
2005-09-15 01:01:47 -06:00
|
|
|
struct atp *dev;
|
|
|
|
struct input_dev *input_dev;
|
|
|
|
struct usb_device *udev = interface_to_usbdev(iface);
|
2005-09-08 02:19:48 -06:00
|
|
|
struct usb_host_interface *iface_desc;
|
|
|
|
struct usb_endpoint_descriptor *endpoint;
|
|
|
|
int int_in_endpointAddr = 0;
|
2007-04-11 23:33:39 -06:00
|
|
|
int i, error = -ENOMEM;
|
2005-09-08 02:19:48 -06:00
|
|
|
|
|
|
|
/* set up the endpoint information */
|
|
|
|
/* use only the first interrupt-in endpoint */
|
|
|
|
iface_desc = iface->cur_altsetting;
|
|
|
|
for (i = 0; i < iface_desc->desc.bNumEndpoints; i++) {
|
|
|
|
endpoint = &iface_desc->endpoint[i].desc;
|
2006-09-27 12:58:53 -06:00
|
|
|
if (!int_in_endpointAddr && usb_endpoint_is_int_in(endpoint)) {
|
2005-09-08 02:19:48 -06:00
|
|
|
/* we found an interrupt in endpoint */
|
|
|
|
int_in_endpointAddr = endpoint->bEndpointAddress;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!int_in_endpointAddr) {
|
|
|
|
err("Could not find int-in endpoint");
|
2005-09-15 01:01:47 -06:00
|
|
|
return -EIO;
|
2005-09-08 02:19:48 -06:00
|
|
|
}
|
|
|
|
|
2005-09-15 01:01:47 -06:00
|
|
|
/* allocate memory for our device state and initialize it */
|
|
|
|
dev = kzalloc(sizeof(struct atp), GFP_KERNEL);
|
|
|
|
input_dev = input_allocate_device();
|
|
|
|
if (!dev || !input_dev) {
|
|
|
|
err("Out of memory");
|
|
|
|
goto err_free_devs;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev->udev = udev;
|
|
|
|
dev->input = input_dev;
|
2005-12-20 22:50:23 -07:00
|
|
|
dev->overflowwarn = 0;
|
2006-04-19 15:36:40 -06:00
|
|
|
if (atp_is_geyser_3(dev))
|
|
|
|
dev->datalen = 64;
|
|
|
|
else if (atp_is_geyser_2(dev))
|
|
|
|
dev->datalen = 64;
|
|
|
|
else
|
|
|
|
dev->datalen = 81;
|
|
|
|
|
|
|
|
if (atp_is_geyser_3(dev)) {
|
2007-07-19 22:29:32 -06:00
|
|
|
/* switch to raw sensor mode */
|
|
|
|
if (atp_geyser3_init(udev))
|
2006-04-19 15:36:40 -06:00
|
|
|
goto err_free_devs;
|
|
|
|
|
|
|
|
printk("appletouch Geyser 3 inited.\n");
|
|
|
|
}
|
2005-09-08 02:19:48 -06:00
|
|
|
|
|
|
|
dev->urb = usb_alloc_urb(0, GFP_KERNEL);
|
2007-04-11 23:33:39 -06:00
|
|
|
if (!dev->urb)
|
2005-09-15 01:01:47 -06:00
|
|
|
goto err_free_devs;
|
|
|
|
|
2005-12-20 22:50:23 -07:00
|
|
|
dev->data = usb_buffer_alloc(dev->udev, dev->datalen, GFP_KERNEL,
|
2005-09-08 02:19:48 -06:00
|
|
|
&dev->urb->transfer_dma);
|
2007-04-11 23:33:39 -06:00
|
|
|
if (!dev->data)
|
2005-09-15 01:01:47 -06:00
|
|
|
goto err_free_urb;
|
|
|
|
|
|
|
|
usb_fill_int_urb(dev->urb, udev,
|
|
|
|
usb_rcvintpipe(udev, int_in_endpointAddr),
|
2005-12-20 22:50:23 -07:00
|
|
|
dev->data, dev->datalen, atp_complete, dev, 1);
|
2005-09-08 02:19:48 -06:00
|
|
|
|
2005-09-15 01:01:47 -06:00
|
|
|
usb_make_path(udev, dev->phys, sizeof(dev->phys));
|
|
|
|
strlcat(dev->phys, "/input0", sizeof(dev->phys));
|
|
|
|
|
|
|
|
input_dev->name = "appletouch";
|
|
|
|
input_dev->phys = dev->phys;
|
|
|
|
usb_to_input_id(dev->udev, &input_dev->id);
|
2007-04-11 23:35:03 -06:00
|
|
|
input_dev->dev.parent = &iface->dev;
|
2005-09-08 02:19:48 -06:00
|
|
|
|
2007-04-11 23:34:39 -06:00
|
|
|
input_set_drvdata(input_dev, dev);
|
|
|
|
|
2005-09-15 01:01:47 -06:00
|
|
|
input_dev->open = atp_open;
|
|
|
|
input_dev->close = atp_close;
|
2005-09-08 02:19:48 -06:00
|
|
|
|
2005-09-15 01:01:47 -06:00
|
|
|
set_bit(EV_ABS, input_dev->evbit);
|
2005-09-08 02:19:48 -06:00
|
|
|
|
2006-04-19 15:36:40 -06:00
|
|
|
if (atp_is_geyser_3(dev)) {
|
|
|
|
/*
|
|
|
|
* MacBook have 20 X sensors, 10 Y sensors
|
|
|
|
*/
|
|
|
|
input_set_abs_params(input_dev, ABS_X, 0,
|
|
|
|
((20 - 1) * ATP_XFACT) - 1, ATP_FUZZ, 0);
|
|
|
|
input_set_abs_params(input_dev, ABS_Y, 0,
|
|
|
|
((10 - 1) * ATP_YFACT) - 1, ATP_FUZZ, 0);
|
|
|
|
} else if (atp_is_geyser_2(dev)) {
|
2005-12-20 22:50:23 -07:00
|
|
|
/*
|
|
|
|
* Oct 2005 15" PowerBooks have 15 X sensors, 17" are detected
|
|
|
|
* later.
|
|
|
|
*/
|
|
|
|
input_set_abs_params(input_dev, ABS_X, 0,
|
|
|
|
((15 - 1) * ATP_XFACT) - 1, ATP_FUZZ, 0);
|
|
|
|
input_set_abs_params(input_dev, ABS_Y, 0,
|
|
|
|
((9 - 1) * ATP_YFACT) - 1, ATP_FUZZ, 0);
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* 12" and 15" Powerbooks only have 16 x sensors,
|
|
|
|
* 17" models are detected later.
|
|
|
|
*/
|
|
|
|
input_set_abs_params(input_dev, ABS_X, 0,
|
|
|
|
(16 - 1) * ATP_XFACT - 1, ATP_FUZZ, 0);
|
|
|
|
input_set_abs_params(input_dev, ABS_Y, 0,
|
|
|
|
(ATP_YSENSORS - 1) * ATP_YFACT - 1, ATP_FUZZ, 0);
|
|
|
|
}
|
2005-09-15 01:01:47 -06:00
|
|
|
input_set_abs_params(input_dev, ABS_PRESSURE, 0, ATP_PRESSURE, 0, 0);
|
2005-09-08 02:19:48 -06:00
|
|
|
|
2005-09-15 01:01:47 -06:00
|
|
|
set_bit(EV_KEY, input_dev->evbit);
|
|
|
|
set_bit(BTN_TOUCH, input_dev->keybit);
|
|
|
|
set_bit(BTN_TOOL_FINGER, input_dev->keybit);
|
|
|
|
set_bit(BTN_TOOL_DOUBLETAP, input_dev->keybit);
|
|
|
|
set_bit(BTN_TOOL_TRIPLETAP, input_dev->keybit);
|
|
|
|
set_bit(BTN_LEFT, input_dev->keybit);
|
2005-09-08 02:19:48 -06:00
|
|
|
|
2007-04-11 23:33:39 -06:00
|
|
|
error = input_register_device(dev->input);
|
|
|
|
if (error)
|
|
|
|
goto err_free_buffer;
|
2005-09-08 02:19:48 -06:00
|
|
|
|
2005-09-15 01:01:47 -06:00
|
|
|
/* save our data pointer in this interface device */
|
|
|
|
usb_set_intfdata(iface, dev);
|
2005-09-08 02:19:48 -06:00
|
|
|
|
2007-07-19 22:29:32 -06:00
|
|
|
INIT_WORK(&dev->work, atp_reinit);
|
|
|
|
|
2005-09-08 02:19:48 -06:00
|
|
|
return 0;
|
|
|
|
|
2007-04-11 23:33:39 -06:00
|
|
|
err_free_buffer:
|
|
|
|
usb_buffer_free(dev->udev, dev->datalen,
|
|
|
|
dev->data, dev->urb->transfer_dma);
|
2005-09-15 01:01:47 -06:00
|
|
|
err_free_urb:
|
2005-09-08 02:19:48 -06:00
|
|
|
usb_free_urb(dev->urb);
|
2005-09-15 01:01:47 -06:00
|
|
|
err_free_devs:
|
2005-09-08 02:19:48 -06:00
|
|
|
usb_set_intfdata(iface, NULL);
|
|
|
|
kfree(dev);
|
2005-09-15 01:01:47 -06:00
|
|
|
input_free_device(input_dev);
|
2007-04-11 23:33:39 -06:00
|
|
|
return error;
|
2005-09-08 02:19:48 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
static void atp_disconnect(struct usb_interface *iface)
|
|
|
|
{
|
|
|
|
struct atp *dev = usb_get_intfdata(iface);
|
|
|
|
|
|
|
|
usb_set_intfdata(iface, NULL);
|
|
|
|
if (dev) {
|
|
|
|
usb_kill_urb(dev->urb);
|
2005-09-15 01:01:47 -06:00
|
|
|
input_unregister_device(dev->input);
|
2005-12-20 22:50:23 -07:00
|
|
|
usb_buffer_free(dev->udev, dev->datalen,
|
2005-09-08 02:19:48 -06:00
|
|
|
dev->data, dev->urb->transfer_dma);
|
2006-07-19 07:39:46 -06:00
|
|
|
usb_free_urb(dev->urb);
|
2005-09-08 02:19:48 -06:00
|
|
|
kfree(dev);
|
|
|
|
}
|
|
|
|
printk(KERN_INFO "input: appletouch disconnected\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
static int atp_suspend(struct usb_interface *iface, pm_message_t message)
|
|
|
|
{
|
|
|
|
struct atp *dev = usb_get_intfdata(iface);
|
2007-07-19 22:29:32 -06:00
|
|
|
|
2005-09-08 02:19:48 -06:00
|
|
|
usb_kill_urb(dev->urb);
|
|
|
|
dev->valid = 0;
|
2007-07-19 22:29:32 -06:00
|
|
|
|
2005-09-08 02:19:48 -06:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int atp_resume(struct usb_interface *iface)
|
|
|
|
{
|
|
|
|
struct atp *dev = usb_get_intfdata(iface);
|
2007-07-19 22:29:32 -06:00
|
|
|
|
2005-09-08 02:19:48 -06:00
|
|
|
if (dev->open && usb_submit_urb(dev->urb, GFP_ATOMIC))
|
|
|
|
return -EIO;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct usb_driver atp_driver = {
|
|
|
|
.name = "appletouch",
|
|
|
|
.probe = atp_probe,
|
|
|
|
.disconnect = atp_disconnect,
|
|
|
|
.suspend = atp_suspend,
|
|
|
|
.resume = atp_resume,
|
|
|
|
.id_table = atp_table,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int __init atp_init(void)
|
|
|
|
{
|
|
|
|
return usb_register(&atp_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __exit atp_exit(void)
|
|
|
|
{
|
|
|
|
usb_deregister(&atp_driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
module_init(atp_init);
|
|
|
|
module_exit(atp_exit);
|