I managed to get this working on my 15" PowerBook, but the USB id for my
Keyboard/Trackpad is 0x0214 as opposed to the 0x0215 you have in the
patch. Are you going to add 0x0214 (and any others?) to this patch
before sending it off?

Also, I found that the patch didn't apply cleanly on my kernel
2.6.15-rc5 kernel. I think many of the line numbers were out, so I ended
up patching the file manually. 

cheers,

-Andy 


On Sun, 2005-12-04 at 23:42 +0100, Michael Hanselmann wrote:
> On Fri, Dec 02, 2005 at 03:28:31PM +0100, Stelian Pop wrote:
> > Is this version really working well on the new Powerbooks ? From what
> > I've seen in this thread there are still issues and it's still a work
> > in progress, so it may be too early to integrate the changes in the
> > kernel.
> 
> It works fine for the 15" PowerBooks (Oct 2005) and has been tested by
> at least three people. 17" should work too, but I can't test that until
> later this week.
> 
> > +#if defined(CONFIG_RELAYFS_FS) || defined(CONFIG_RELAYFS_FS_MODULE)
> > +#include <linux/relayfs_fs.h>
> > +#endif
> 
> > While the relayfs code is ok for debugging, I'm wondering if it should
> > be left in the final version at all.
> 
> It doesn't hurt as it's not much code, it makes updating the driver for
> newer devices easier and it's only enabled if requested. Because of
> that, I would leave it in. However, I can split it into a separate patch
> if wanted.
> 
> > +       int                     overflowwarn;   /* overflow warning 
> > printed? */
> 
> > I would use a static variable in the case -OVERFLOW: block here.
> 
> As there could be, due to whatever reason, multiple devices using the
> same driver, one could overflow while another doesn't. Therefore, it
> should be device specific.
> 
> > +               dev->xy_cur[i++] = dev->data[19];
> > +               dev->xy_cur[i++] = dev->data[20];
> > +               dev->xy_cur[i++] = dev->data[22];
> > +               dev->xy_cur[i++] = dev->data[23];
> 
> > There is obviously a pattern here:
> 
> >     for (i = 0; i < 15; i++)
> >             dev->xy_cur[i] = dev->data[ 19 + (i * 3) / 2 ]
> 
> It's not that easy. This code wouldn't work as it should.
> 
> I wrote a working loop into the patch below, but as the code's called
> about 1'000 times a second I unrolled the loop myself. A friend of mine
> calculated that the code would take at least twice the time to run when
> written using a loop.
> 
> > I'm wondering if the same formula doesn't apply for more X and Y
> > sensors (like 16 X and 16 Y sensors on the old Powerbooks, 26 for the
> > 17" models)
> 
> It does and is addressed in the new patch below.
> 
> > What is the point in doing this since the dbg_dump is called a few lines
> > later ?
> 
> Those were some leftovers from my debugging and would have been deleted
> before submitting. I just didn't want to take everything out if someone
> else wants to do some extensive debugging or so.
> 
> Here's an updated patch including support for the 17" PowerBooks (Oct
> 2005). The informations about the 17" one are from Alex Harper.
> 
> ---
> --- linux-2.6.15-rc5/drivers/usb/input/appletouch.c.orig      2005-12-04 
> 20:25:21.000000000 +0100
> +++ linux-2.6.15-rc5/drivers/usb/input/appletouch.c   2005-12-04 
> 23:37:29.000000000 +0100
> @@ -6,9 +6,19 @@
>   * Copyright (C) 2005      Stelian Pop ([EMAIL PROTECTED])
>   * Copyright (C) 2005      Frank Arnold ([EMAIL PROTECTED])
>   * Copyright (C) 2005      Peter Osterlund ([EMAIL PROTECTED])
> + * Copyright (C) 2005      Parag Warudkar ([EMAIL PROTECTED])
> + * Copyright (C) 2005      Michael Hanselmann ([EMAIL PROTECTED])
>   *
>   * Thanks to Alex Harper <[EMAIL PROTECTED]> for his inputs.
>   *
> + * Nov 2005 - Parag Warudkar 
> + *  o Added ability to export data via relayfs
> + *
> + * Nov 2005 - Michael Hanselmann
> + *  o Compile relayfs support only if enabled in the kernel
> + *  o Enable relayfs only if requested by the user
> + *  o Added support for new October 2005 PowerBooks (Geyser 2)
> + *
>   * 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
> @@ -35,8 +45,13 @@
>  #include <linux/input.h>
>  #include <linux/usb_input.h>
>  
> +#if defined(CONFIG_RELAYFS_FS) || defined(CONFIG_RELAYFS_FS_MODULE)
> +#include <linux/relayfs_fs.h>
> +#endif
> +
>  /* Apple has powerbooks which have the keyboard with different Product IDs */
>  #define APPLE_VENDOR_ID              0x05AC
> +#define GEYSER_2_PRODUCT_ID  0x0215
>  
>  #define ATP_DEVICE(prod)                                     \
>       .match_flags = USB_DEVICE_ID_MATCH_DEVICE |             \
> @@ -51,14 +66,17 @@
>  static struct usb_device_id atp_table [] = {
>       { ATP_DEVICE(0x020E) },
>       { ATP_DEVICE(0x020F) },
> +     { ATP_DEVICE(GEYSER_2_PRODUCT_ID) },    /* PowerBooks Oct 2005 */
>       { ATP_DEVICE(0x030A) },
>       { ATP_DEVICE(0x030B) },
>       { }                                     /* Terminating entry */
>  };
>  MODULE_DEVICE_TABLE (usb, atp_table);
>  
> -/* size of a USB urb transfer */
> -#define ATP_DATASIZE 81
> +#if defined(CONFIG_RELAYFS_FS) || defined(CONFIG_RELAYFS_FS_MODULE)
> +struct rchan* rch = NULL;
> +struct rchan_callbacks* rcb = NULL;
> +#endif
>  
>  /*
>   * number of sensors. Note that only 16 instead of 26 X (horizontal)
> @@ -73,6 +91,7 @@
>  
>  /* 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
> @@ -108,6 +127,8 @@
>       signed char             xy_old[ATP_XSENSORS + ATP_YSENSORS];
>                                               /* accumulated sensors */
>       int                     xy_acc[ATP_XSENSORS + ATP_YSENSORS];
> +     int                     overflowwarn;   /* overflow warning printed? */
> +     int                     datalen;        /* size of an USB urb transfer 
> */
>  };
>  
>  #define dbg_dump(msg, tab) \
> @@ -124,7 +145,11 @@
>               if (debug) printk(format, ##a);                         \
>       } while (0)
>  
> -MODULE_AUTHOR("Johannes Berg, Stelian Pop, Frank Arnold");
> +/* Checks if the device a Geyser 2 */
> +#define IS_GEYSER_2(dev) \
> +     (le16_to_cpu(dev->udev->descriptor.idProduct) == GEYSER_2_PRODUCT_ID)
> +
> +MODULE_AUTHOR("Johannes Berg, Stelian Pop, Frank Arnold, Parag Warudkar, 
> Michael Hanselmann");
>  MODULE_DESCRIPTION("Apple PowerBooks USB touchpad driver");
>  MODULE_LICENSE("GPL");
>  
> @@ -132,6 +157,10 @@
>  module_param(debug, int, 0644);
>  MODULE_PARM_DESC(debug, "Activate debugging output");
>  
> +static int relayfs = 0;
> +module_param(relayfs, int, 0644);
> +MODULE_PARM_DESC(relayfs, "Activate relayfs support");
> +
>  static int atp_calculate_abs(int *xy_sensors, int nb_sensors, int fact,
>                            int *z, int *fingers)
>  {
> @@ -175,6 +204,13 @@
>       case 0:
>               /* success */
>               break;
> +     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;
> +             }
>       case -ECONNRESET:
>       case -ENOENT:
>       case -ESHUTDOWN:
> @@ -189,23 +225,83 @@
>       }
>  
>       /* drop incomplete datasets */
> -     if (dev->urb->actual_length != ATP_DATASIZE) {
> +     if (dev->urb->actual_length != dev->datalen) {
>               dprintk("appletouch: incomplete data package.\n");
>               goto exit;
>       }
>  
> +#if defined(CONFIG_RELAYFS_FS) || defined(CONFIG_RELAYFS_FS_MODULE)
> +     if (relayfs && dev->data) {
> +             relay_write(rch, dev->data, dev->urb->actual_length);
> +     }
> +#endif
> +
>       /* reorder the sensors values */
> -     for (i = 0; i < 8; i++) {
> +     if (IS_GEYSER_2(dev)) {
> +             memset(dev->xy_cur, 0, sizeof(dev->xy_cur));
> +
> +             /*
> +              * The values are laid out like this:
> +              * Y1, Y2, -, Y3, Y4, -, ...
> +              * '-' is an unused value.
> +              *
> +              * The logic in a loop:
> +              * 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];
> +              * }
> +              *
> +              * This code is called about 1'000 times per second for each
> +              * interrupt from the touchpad. Therefore it should be as fast
> +              * as possible. Writing the following code in a loop would take
> +              * about twice the time to run if not more.
> +              */
> +
>               /* 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];
> +             dev->xy_cur[0] = dev->data[19];
> +             dev->xy_cur[1] = dev->data[20];
> +             dev->xy_cur[2] = dev->data[22];
> +             dev->xy_cur[3] = dev->data[23];
> +             dev->xy_cur[4] = dev->data[25];
> +             dev->xy_cur[5] = dev->data[26];
> +             dev->xy_cur[6] = dev->data[28];
> +             dev->xy_cur[7] = dev->data[29];
> +             dev->xy_cur[8] = dev->data[31];
> +             dev->xy_cur[9] = dev->data[32];
> +             dev->xy_cur[10] = dev->data[34];
> +             dev->xy_cur[11] = dev->data[35];
> +             dev->xy_cur[12] = dev->data[37];
> +             dev->xy_cur[13] = dev->data[38];
> +             dev->xy_cur[14] = dev->data[40];
> +             dev->xy_cur[15] = dev->data[41];
> +             dev->xy_cur[16] = dev->data[43];
> +             dev->xy_cur[17] = dev->data[44];
> +             dev->xy_cur[18] = dev->data[46];
> +             dev->xy_cur[19] = dev->data[47];
>  
>               /* Y values */
> -             dev->xy_cur[i + 26] = dev->data[5 * i +  1];
> -             dev->xy_cur[i + 34] = dev->data[5 * i +  3];
> +             dev->xy_cur[ATP_XSENSORS + 0] = dev->data[1];
> +             dev->xy_cur[ATP_XSENSORS + 1] = dev->data[2];
> +             dev->xy_cur[ATP_XSENSORS + 2] = dev->data[4];
> +             dev->xy_cur[ATP_XSENSORS + 3] = dev->data[5];
> +             dev->xy_cur[ATP_XSENSORS + 4] = dev->data[7];
> +             dev->xy_cur[ATP_XSENSORS + 5] = dev->data[8];
> +             dev->xy_cur[ATP_XSENSORS + 6] = dev->data[10];
> +             dev->xy_cur[ATP_XSENSORS + 7] = dev->data[11];
> +             dev->xy_cur[ATP_XSENSORS + 8] = dev->data[13];
> +     } 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];
> +             }
>       }
>  
>       dbg_dump("sample", dev->xy_cur);
> @@ -216,16 +312,23 @@
>               dev->x_old = dev->y_old = -1;
>               memcpy(dev->xy_old, dev->xy_cur, sizeof(dev->xy_old));
>  
> -             /* 17" Powerbooks have 10 extra X sensors */
> -             for (i = 16; i < ATP_XSENSORS; i++)
> -                     if (dev->xy_cur[i]) {
> -                             printk("appletouch: 17\" model detected.\n");
> +             /* 17" Powerbooks have extra X sensors */
> +             for (i = (IS_GEYSER_2(dev)?15:16); i < ATP_XSENSORS; i++) {
> +                     if (!dev->xy_cur[i]) continue;
> +
> +                     printk("appletouch: 17\" model detected.\n");
> +                     if(IS_GEYSER_2(dev))
> +                             input_set_abs_params(dev->input, ABS_X, 0,
> +                                                  (20 - 1) *
> +                                                  ATP_XFACT - 1,
> +                                                  ATP_FUZZ, 0);
> +                     else
>                               input_set_abs_params(dev->input, ABS_X, 0,
>                                                    (ATP_XSENSORS - 1) *
>                                                    ATP_XFACT - 1,
>                                                    ATP_FUZZ, 0);
> -                             break;
> -                     }
> +                     break;
> +             }
>  
>               goto exit;
>       }
> @@ -282,7 +385,8 @@
>               memset(dev->xy_acc, 0, sizeof(dev->xy_acc));
>       }
>  
> -     input_report_key(dev->input, BTN_LEFT, !!dev->data[80]);
> +     input_report_key(dev->input, BTN_LEFT,
> +                      !!dev->data[dev->datalen - 1]);
>  
>       input_sync(dev->input);
>  
> @@ -323,7 +427,6 @@
>       int int_in_endpointAddr = 0;
>       int i, retval = -ENOMEM;
>  
> -
>       /* set up the endpoint information */
>       /* use only the first interrupt-in endpoint */
>       iface_desc = iface->cur_altsetting;
> @@ -353,6 +456,8 @@
>  
>       dev->udev = udev;
>       dev->input = input_dev;
> +     dev->overflowwarn = 0;
> +     dev->datalen = (IS_GEYSER_2(dev)?64:81);
>  
>       dev->urb = usb_alloc_urb(0, GFP_KERNEL);
>       if (!dev->urb) {
> @@ -360,7 +465,7 @@
>               goto err_free_devs;
>       }
>  
> -     dev->data = usb_buffer_alloc(dev->udev, ATP_DATASIZE, GFP_KERNEL,
> +     dev->data = usb_buffer_alloc(dev->udev, dev->datalen, GFP_KERNEL,
>                                    &dev->urb->transfer_dma);
>       if (!dev->data) {
>               retval = -ENOMEM;
> @@ -369,7 +474,7 @@
>  
>       usb_fill_int_urb(dev->urb, udev,
>                        usb_rcvintpipe(udev, int_in_endpointAddr),
> -                      dev->data, ATP_DATASIZE, atp_complete, dev, 1);
> +                      dev->data, dev->datalen, atp_complete, dev, 1);
>  
>       usb_make_path(udev, dev->phys, sizeof(dev->phys));
>       strlcat(dev->phys, "/input0", sizeof(dev->phys));
> @@ -385,14 +490,25 @@
>  
>       set_bit(EV_ABS, input_dev->evbit);
>  
> -     /*
> -      * 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);
> +     if (IS_GEYSER_2(dev)) {
> +             /*
> +              * 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);
> +     }
>       input_set_abs_params(input_dev, ABS_PRESSURE, 0, ATP_PRESSURE, 0, 0);
>  
>       set_bit(EV_KEY, input_dev->evbit);
> @@ -427,7 +543,7 @@
>               usb_kill_urb(dev->urb);
>               input_unregister_device(dev->input);
>               usb_free_urb(dev->urb);
> -             usb_buffer_free(dev->udev, ATP_DATASIZE,
> +             usb_buffer_free(dev->udev, dev->datalen,
>                               dev->data, dev->urb->transfer_dma);
>               kfree(dev);
>       }
> @@ -463,11 +579,30 @@
>  
>  static int __init atp_init(void)
>  {
> +#if defined(CONFIG_RELAYFS_FS) || defined(CONFIG_RELAYFS_FS_MODULE)
> +     if (relayfs) {
> +             rcb = kmalloc(sizeof(struct rchan_callbacks), GFP_KERNEL);
> +             rcb->subbuf_start = NULL;
> +             rcb->buf_mapped = NULL;
> +             rcb->buf_unmapped = NULL;
> +             rch = relay_open("atpdata", NULL, 256, 256, NULL);
> +             if (!rch) return -ENOMEM;
> +             printk("appletouch: Relayfs enabled.\n");
> +     } else {
> +             printk("appletouch: Relayfs disabled.\n");
> +     }
> +#endif
>       return usb_register(&atp_driver);
>  }
>  
>  static void __exit atp_exit(void)
>  {
> +#if defined(CONFIG_RELAYFS_FS) || defined(CONFIG_RELAYFS_FS_MODULE)
> +     if (relayfs) {
> +             relay_close(rch);
> +             kfree(rcb);
> +     }
> +#endif
>       usb_deregister(&atp_driver);
>  }
>  
> 
> 


-- 
To UNSUBSCRIBE, email to [EMAIL PROTECTED]
with a subject of "unsubscribe". Trouble? Contact [EMAIL PROTECTED]

Reply via email to