Merge git://git.kernel.org/pub/scm/linux/kernel/git/jejb/scsi-misc-2.6
[pandora-kernel.git] / drivers / input / touchscreen / tsc2005.c
index bc7e2f9..cbf0ff3 100644 (file)
@@ -129,61 +129,76 @@ struct tsc2005 {
        int                     in_z1;
        int                     in_z2;
 
+       spinlock_t              lock;
        struct timer_list       penup_timer;
-       struct work_struct      penup_work;
 
        unsigned int            esd_timeout;
-       struct timer_list       esd_timer;
-       struct work_struct      esd_work;
+       struct delayed_work     esd_work;
+       unsigned long           last_valid_interrupt;
 
        unsigned int            x_plate_ohm;
 
-       bool                    disabled;
-       unsigned int            disable_depth;
-       unsigned int            pen_down;
+       bool                    opened;
+       bool                    suspended;
+
+       bool                    pen_down;
 
        void                    (*set_reset)(bool enable);
 };
 
-static void tsc2005_cmd(struct tsc2005 *ts, u8 cmd)
+static int tsc2005_cmd(struct tsc2005 *ts, u8 cmd)
 {
-       u8 tx;
+       u8 tx = TSC2005_CMD | TSC2005_CMD_12BIT | cmd;
+       struct spi_transfer xfer = {
+               .tx_buf         = &tx,
+               .len            = 1,
+               .bits_per_word  = 8,
+       };
        struct spi_message msg;
-       struct spi_transfer xfer = { 0 };
-
-       tx = TSC2005_CMD | TSC2005_CMD_12BIT | cmd;
-
-       xfer.tx_buf = &tx;
-       xfer.rx_buf = NULL;
-       xfer.len = 1;
-       xfer.bits_per_word = 8;
+       int error;
 
        spi_message_init(&msg);
        spi_message_add_tail(&xfer, &msg);
-       spi_sync(ts->spi, &msg);
+
+       error = spi_sync(ts->spi, &msg);
+       if (error) {
+               dev_err(&ts->spi->dev, "%s: failed, command: %x, error: %d\n",
+                       __func__, cmd, error);
+               return error;
+       }
+
+       return 0;
 }
 
-static void tsc2005_write(struct tsc2005 *ts, u8 reg, u16 value)
+static int tsc2005_write(struct tsc2005 *ts, u8 reg, u16 value)
 {
-       u32 tx;
+       u32 tx = ((reg | TSC2005_REG_PND0) << 16) | value;
+       struct spi_transfer xfer = {
+               .tx_buf         = &tx,
+               .len            = 4,
+               .bits_per_word  = 24,
+       };
        struct spi_message msg;
-       struct spi_transfer xfer = { 0 };
-
-       tx = (reg | TSC2005_REG_PND0) << 16;
-       tx |= value;
-
-       xfer.tx_buf = &tx;
-       xfer.rx_buf = NULL;
-       xfer.len = 4;
-       xfer.bits_per_word = 24;
+       int error;
 
        spi_message_init(&msg);
        spi_message_add_tail(&xfer, &msg);
-       spi_sync(ts->spi, &msg);
+
+       error = spi_sync(ts->spi, &msg);
+       if (error) {
+               dev_err(&ts->spi->dev,
+                       "%s: failed, register: %x, value: %x, error: %d\n",
+                       __func__, reg, value, error);
+               return error;
+       }
+
+       return 0;
 }
 
 static void tsc2005_setup_read(struct tsc2005_spi_rd *rd, u8 reg, bool last)
 {
+       memset(rd, 0, sizeof(*rd));
+
        rd->spi_tx                 = (reg | TSC2005_REG_READ) << 16;
        rd->spi_xfer.tx_buf        = &rd->spi_tx;
        rd->spi_xfer.rx_buf        = &rd->spi_rx;
@@ -192,17 +207,23 @@ static void tsc2005_setup_read(struct tsc2005_spi_rd *rd, u8 reg, bool last)
        rd->spi_xfer.cs_change     = !last;
 }
 
-static void tsc2005_read(struct tsc2005 *ts, u8 reg, u16 *value)
+static int tsc2005_read(struct tsc2005 *ts, u8 reg, u16 *value)
 {
+       struct tsc2005_spi_rd spi_rd;
        struct spi_message msg;
-       struct tsc2005_spi_rd spi_rd = { { 0 }, 0, 0 };
+       int error;
 
-       tsc2005_setup_read(&spi_rd, reg, 1);
+       tsc2005_setup_read(&spi_rd, reg, true);
 
        spi_message_init(&msg);
        spi_message_add_tail(&spi_rd.spi_xfer, &msg);
-       spi_sync(ts->spi, &msg);
+
+       error = spi_sync(ts->spi, &msg);
+       if (error)
+               return error;
+
        *value = spi_rd.spi_rx;
+       return 0;
 }
 
 static void tsc2005_update_pen_state(struct tsc2005 *ts,
@@ -214,13 +235,13 @@ static void tsc2005_update_pen_state(struct tsc2005 *ts,
                input_report_abs(ts->idev, ABS_PRESSURE, pressure);
                if (!ts->pen_down) {
                        input_report_key(ts->idev, BTN_TOUCH, !!pressure);
-                       ts->pen_down = 1;
+                       ts->pen_down = true;
                }
        } else {
                input_report_abs(ts->idev, ABS_PRESSURE, 0);
                if (ts->pen_down) {
                        input_report_key(ts->idev, BTN_TOUCH, 0);
-                       ts->pen_down = 0;
+                       ts->pen_down = false;
                }
        }
        input_sync(ts->idev);
@@ -228,33 +249,20 @@ static void tsc2005_update_pen_state(struct tsc2005 *ts,
                pressure);
 }
 
-static irqreturn_t tsc2005_irq_handler(int irq, void *dev_id)
-{
-       struct tsc2005 *ts = dev_id;
-
-       /* update the penup timer only if it's pending */
-       mod_timer_pending(&ts->penup_timer,
-                         jiffies + msecs_to_jiffies(TSC2005_PENUP_TIME_MS));
-
-       return IRQ_WAKE_THREAD;
-}
-
 static irqreturn_t tsc2005_irq_thread(int irq, void *_ts)
 {
        struct tsc2005 *ts = _ts;
+       unsigned long flags;
        unsigned int pressure;
-       u32 x;
-       u32 y;
-       u32 z1;
-       u32 z2;
-
-       mutex_lock(&ts->mutex);
+       u32 x, y;
+       u32 z1, z2;
+       int error;
 
-       if (unlikely(ts->disable_depth))
+       /* read the coordinates */
+       error = spi_sync(ts->spi, &ts->spi_read_msg);
+       if (unlikely(error))
                goto out;
 
-       /* read the coordinates */
-       spi_sync(ts->spi, &ts->spi_read_msg);
        x = ts->spi_x.spi_rx;
        y = ts->spi_y.spi_rx;
        z1 = ts->spi_z1.spi_rx;
@@ -264,66 +272,56 @@ static irqreturn_t tsc2005_irq_thread(int irq, void *_ts)
        if (unlikely(x > MAX_12BIT || y > MAX_12BIT))
                goto out;
 
-       /* skip coords if the pressure components are out of range */
+       /* Skip reading if the pressure components are out of range */
        if (unlikely(z1 == 0 || z2 > MAX_12BIT || z1 >= z2))
                goto out;
 
-       /* skip point if this is a pen down with the exact same values as
+       /*
+       * Skip point if this is a pen down with the exact same values as
        * the value before pen-up - that implies SPI fed us stale data
        */
        if (!ts->pen_down &&
-       ts->in_x == x &&
-       ts->in_y == y &&
-       ts->in_z1 == z1 &&
-       ts->in_z2 == z2)
+           ts->in_x == x && ts->in_y == y &&
+           ts->in_z1 == z1 && ts->in_z2 == z2) {
                goto out;
+       }
 
-       /* At this point we are happy we have a valid and useful reading.
-       * Remember it for later comparisons. We may now begin downsampling
-       */
+       /*
+        * At this point we are happy we have a valid and useful reading.
+        * Remember it for later comparisons. We may now begin downsampling.
+        */
        ts->in_x = x;
        ts->in_y = y;
        ts->in_z1 = z1;
        ts->in_z2 = z2;
 
-       /* compute touch pressure resistance using equation #1 */
+       /* Compute touch pressure resistance using equation #1 */
        pressure = x * (z2 - z1) / z1;
        pressure = pressure * ts->x_plate_ohm / 4096;
        if (unlikely(pressure > MAX_12BIT))
                goto out;
 
-       tsc2005_update_pen_state(ts, x, y, pressure);
+       spin_lock_irqsave(&ts->lock, flags);
 
-       /* set the penup timer */
+       tsc2005_update_pen_state(ts, x, y, pressure);
        mod_timer(&ts->penup_timer,
                  jiffies + msecs_to_jiffies(TSC2005_PENUP_TIME_MS));
 
-       if (!ts->esd_timeout)
-               goto out;
-
-       /* update the watchdog timer */
-       mod_timer(&ts->esd_timer,
-                 round_jiffies(jiffies + msecs_to_jiffies(ts->esd_timeout)));
+       spin_unlock_irqrestore(&ts->lock, flags);
 
+       ts->last_valid_interrupt = jiffies;
 out:
-       mutex_unlock(&ts->mutex);
        return IRQ_HANDLED;
 }
 
 static void tsc2005_penup_timer(unsigned long data)
 {
        struct tsc2005 *ts = (struct tsc2005 *)data;
+       unsigned long flags;
 
-       schedule_work(&ts->penup_work);
-}
-
-static void tsc2005_penup_work(struct work_struct *work)
-{
-       struct tsc2005 *ts = container_of(work, struct tsc2005, penup_work);
-
-       mutex_lock(&ts->mutex);
+       spin_lock_irqsave(&ts->lock, flags);
        tsc2005_update_pen_state(ts, 0, 0, 0);
-       mutex_unlock(&ts->mutex);
+       spin_unlock_irqrestore(&ts->lock, flags);
 }
 
 static void tsc2005_start_scan(struct tsc2005 *ts)
@@ -339,66 +337,32 @@ static void tsc2005_stop_scan(struct tsc2005 *ts)
        tsc2005_cmd(ts, TSC2005_CMD_STOP);
 }
 
-/* must be called with mutex held */
-static void tsc2005_disable(struct tsc2005 *ts)
+/* must be called with ts->mutex held */
+static void __tsc2005_disable(struct tsc2005 *ts)
 {
-       if (ts->disable_depth++ != 0)
-               return;
+       tsc2005_stop_scan(ts);
+
        disable_irq(ts->spi->irq);
-       if (ts->esd_timeout)
-               del_timer_sync(&ts->esd_timer);
        del_timer_sync(&ts->penup_timer);
-       tsc2005_stop_scan(ts);
-}
 
-/* must be called with mutex held */
-static void tsc2005_enable(struct tsc2005 *ts)
-{
-       if (--ts->disable_depth != 0)
-               return;
-       tsc2005_start_scan(ts);
-       enable_irq(ts->spi->irq);
-       if (!ts->esd_timeout)
-               return;
-       mod_timer(&ts->esd_timer,
-                 round_jiffies(jiffies + msecs_to_jiffies(ts->esd_timeout)));
-}
+       cancel_delayed_work_sync(&ts->esd_work);
 
-static ssize_t tsc2005_disable_show(struct device *dev,
-                                   struct device_attribute *attr, char *buf)
-{
-       struct spi_device *spi = to_spi_device(dev);
-       struct tsc2005 *ts = spi_get_drvdata(spi);
-
-       return sprintf(buf, "%u\n", ts->disabled);
+       enable_irq(ts->spi->irq);
 }
 
-static ssize_t tsc2005_disable_store(struct device *dev,
-                                    struct device_attribute *attr,
-                                    const char *buf, size_t count)
+/* must be called with ts->mutex held */
+static void __tsc2005_enable(struct tsc2005 *ts)
 {
-       struct spi_device *spi = to_spi_device(dev);
-       struct tsc2005 *ts = spi_get_drvdata(spi);
-       unsigned long res;
-       int i;
+       tsc2005_start_scan(ts);
 
-       if (strict_strtoul(buf, 10, &res) < 0)
-               return -EINVAL;
-       i = res ? 1 : 0;
+       if (ts->esd_timeout && ts->set_reset) {
+               ts->last_valid_interrupt = jiffies;
+               schedule_delayed_work(&ts->esd_work,
+                               round_jiffies_relative(
+                                       msecs_to_jiffies(ts->esd_timeout)));
+       }
 
-       mutex_lock(&ts->mutex);
-       if (i == ts->disabled)
-               goto out;
-       ts->disabled = i;
-       if (i)
-               tsc2005_disable(ts);
-       else
-               tsc2005_enable(ts);
-out:
-       mutex_unlock(&ts->mutex);
-       return count;
 }
-static DEVICE_ATTR(disable, 0664, tsc2005_disable_show, tsc2005_disable_store);
 
 static ssize_t tsc2005_selftest_show(struct device *dev,
                                     struct device_attribute *attr,
@@ -409,48 +373,79 @@ static ssize_t tsc2005_selftest_show(struct device *dev,
        u16 temp_high;
        u16 temp_high_orig;
        u16 temp_high_test;
-       unsigned int result;
+       bool success = true;
+       int error;
 
        mutex_lock(&ts->mutex);
 
        /*
         * Test TSC2005 communications via temp high register.
         */
-       tsc2005_disable(ts);
-       result = 1;
-       tsc2005_read(ts, TSC2005_REG_TEMP_HIGH, &temp_high_orig);
+       __tsc2005_disable(ts);
+
+       error = tsc2005_read(ts, TSC2005_REG_TEMP_HIGH, &temp_high_orig);
+       if (error) {
+               dev_warn(dev, "selftest failed: read error %d\n", error);
+               success = false;
+               goto out;
+       }
+
        temp_high_test = (temp_high_orig - 1) & MAX_12BIT;
-       tsc2005_write(ts, TSC2005_REG_TEMP_HIGH, temp_high_test);
-       tsc2005_read(ts, TSC2005_REG_TEMP_HIGH, &temp_high);
+
+       error = tsc2005_write(ts, TSC2005_REG_TEMP_HIGH, temp_high_test);
+       if (error) {
+               dev_warn(dev, "selftest failed: write error %d\n", error);
+               success = false;
+               goto out;
+       }
+
+       error = tsc2005_read(ts, TSC2005_REG_TEMP_HIGH, &temp_high);
+       if (error) {
+               dev_warn(dev, "selftest failed: read error %d after write\n",
+                        error);
+               success = false;
+               goto out;
+       }
+
        if (temp_high != temp_high_test) {
                dev_warn(dev, "selftest failed: %d != %d\n",
                         temp_high, temp_high_test);
-               result = 0;
+               success = false;
        }
 
        /* hardware reset */
-       ts->set_reset(0);
+       ts->set_reset(false);
        usleep_range(100, 500); /* only 10us required */
-       ts->set_reset(1);
-       tsc2005_enable(ts);
+       ts->set_reset(true);
+
+       if (!success)
+               goto out;
 
        /* test that the reset really happened */
-       tsc2005_read(ts, TSC2005_REG_TEMP_HIGH, &temp_high);
+       error = tsc2005_read(ts, TSC2005_REG_TEMP_HIGH, &temp_high);
+       if (error) {
+               dev_warn(dev, "selftest failed: read error %d after reset\n",
+                        error);
+               success = false;
+               goto out;
+       }
+
        if (temp_high != temp_high_orig) {
                dev_warn(dev, "selftest failed after reset: %d != %d\n",
                         temp_high, temp_high_orig);
-               result = 0;
+               success = false;
        }
 
+out:
+       __tsc2005_enable(ts);
        mutex_unlock(&ts->mutex);
 
-       return sprintf(buf, "%u\n", result);
+       return sprintf(buf, "%d\n", success);
 }
 
 static DEVICE_ATTR(selftest, S_IRUGO, tsc2005_selftest_show, NULL);
 
 static struct attribute *tsc2005_attrs[] = {
-       &dev_attr_disable.attr,
        &dev_attr_selftest.attr,
        NULL
 };
@@ -476,51 +471,96 @@ static const struct attribute_group tsc2005_attr_group = {
        .attrs          = tsc2005_attrs,
 };
 
-static void tsc2005_esd_timer(unsigned long data)
-{
-       struct tsc2005 *ts = (struct tsc2005 *)data;
-
-       schedule_work(&ts->esd_work);
-}
-
 static void tsc2005_esd_work(struct work_struct *work)
 {
-       struct tsc2005 *ts = container_of(work, struct tsc2005, esd_work);
+       struct tsc2005 *ts = container_of(work, struct tsc2005, esd_work.work);
+       int error;
        u16 r;
 
-       mutex_lock(&ts->mutex);
+       if (!mutex_trylock(&ts->mutex)) {
+               /*
+                * If the mutex is taken, it means that disable or enable is in
+                * progress. In that case just reschedule the work. If the work
+                * is not needed, it will be canceled by disable.
+                */
+               goto reschedule;
+       }
 
-       if (ts->disable_depth)
+       if (time_is_after_jiffies(ts->last_valid_interrupt +
+                                 msecs_to_jiffies(ts->esd_timeout)))
                goto out;
 
+       /* We should be able to read register without disabling interrupts. */
+       error = tsc2005_read(ts, TSC2005_REG_CFR0, &r);
+       if (!error &&
+           !((r ^ TSC2005_CFR0_INITVALUE) & TSC2005_CFR0_RW_MASK)) {
+               goto out;
+       }
+
        /*
-        * If we cannot read our known value from configuration register 0 then
-        * reset the controller as if from power-up and start scanning again.
+        * If we could not read our known value from configuration register 0
+        * then we should reset the controller as if from power-up and start
+        * scanning again.
         */
-       tsc2005_read(ts, TSC2005_REG_CFR0, &r);
-       if ((r ^ TSC2005_CFR0_INITVALUE) & TSC2005_CFR0_RW_MASK) {
-               dev_info(&ts->spi->dev, "TSC2005 not responding - resetting\n");
-               ts->set_reset(0);
-               tsc2005_update_pen_state(ts, 0, 0, 0);
-               usleep_range(100, 500); /* only 10us required */
-               ts->set_reset(1);
-               tsc2005_start_scan(ts);
-       }
+       dev_info(&ts->spi->dev, "TSC2005 not responding - resetting\n");
 
-       /* re-arm the watchdog */
-       mod_timer(&ts->esd_timer,
-                 round_jiffies(jiffies + msecs_to_jiffies(ts->esd_timeout)));
+       disable_irq(ts->spi->irq);
+       del_timer_sync(&ts->penup_timer);
+
+       tsc2005_update_pen_state(ts, 0, 0, 0);
+
+       ts->set_reset(false);
+       usleep_range(100, 500); /* only 10us required */
+       ts->set_reset(true);
+
+       enable_irq(ts->spi->irq);
+       tsc2005_start_scan(ts);
 
 out:
+       mutex_unlock(&ts->mutex);
+reschedule:
+       /* re-arm the watchdog */
+       schedule_delayed_work(&ts->esd_work,
+                             round_jiffies_relative(
+                                       msecs_to_jiffies(ts->esd_timeout)));
+}
+
+static int tsc2005_open(struct input_dev *input)
+{
+       struct tsc2005 *ts = input_get_drvdata(input);
+
+       mutex_lock(&ts->mutex);
+
+       if (!ts->suspended)
+               __tsc2005_enable(ts);
+
+       ts->opened = true;
+
+       mutex_unlock(&ts->mutex);
+
+       return 0;
+}
+
+static void tsc2005_close(struct input_dev *input)
+{
+       struct tsc2005 *ts = input_get_drvdata(input);
+
+       mutex_lock(&ts->mutex);
+
+       if (!ts->suspended)
+               __tsc2005_disable(ts);
+
+       ts->opened = false;
+
        mutex_unlock(&ts->mutex);
 }
 
 static void __devinit tsc2005_setup_spi_xfer(struct tsc2005 *ts)
 {
-       tsc2005_setup_read(&ts->spi_x, TSC2005_REG_X, 0);
-       tsc2005_setup_read(&ts->spi_y, TSC2005_REG_Y, 0);
-       tsc2005_setup_read(&ts->spi_z1, TSC2005_REG_Z1, 0);
-       tsc2005_setup_read(&ts->spi_z2, TSC2005_REG_Z2, 1);
+       tsc2005_setup_read(&ts->spi_x, TSC2005_REG_X, false);
+       tsc2005_setup_read(&ts->spi_y, TSC2005_REG_Y, false);
+       tsc2005_setup_read(&ts->spi_z1, TSC2005_REG_Z1, false);
+       tsc2005_setup_read(&ts->spi_z2, TSC2005_REG_Z2, true);
 
        spi_message_init(&ts->spi_read_msg);
        spi_message_add_tail(&ts->spi_x.spi_xfer, &ts->spi_read_msg);
@@ -580,11 +620,10 @@ static int __devinit tsc2005_probe(struct spi_device *spi)
 
        mutex_init(&ts->mutex);
 
+       spin_lock_init(&ts->lock);
        setup_timer(&ts->penup_timer, tsc2005_penup_timer, (unsigned long)ts);
-       INIT_WORK(&ts->penup_work, tsc2005_penup_work);
 
-       setup_timer(&ts->esd_timer, tsc2005_esd_timer, (unsigned long)ts);
-       INIT_WORK(&ts->esd_work, tsc2005_esd_work);
+       INIT_DELAYED_WORK(&ts->esd_work, tsc2005_esd_work);
 
        tsc2005_setup_spi_xfer(ts);
 
@@ -602,8 +641,15 @@ static int __devinit tsc2005_probe(struct spi_device *spi)
        input_set_abs_params(input_dev, ABS_Y, 0, max_y, fudge_y, 0);
        input_set_abs_params(input_dev, ABS_PRESSURE, 0, max_p, fudge_p, 0);
 
-       error = request_threaded_irq(spi->irq,
-                                    tsc2005_irq_handler, tsc2005_irq_thread,
+       input_dev->open = tsc2005_open;
+       input_dev->close = tsc2005_close;
+
+       input_set_drvdata(input_dev, ts);
+
+       /* Ensure the touchscreen is off */
+       tsc2005_stop_scan(ts);
+
+       error = request_threaded_irq(spi->irq, NULL, tsc2005_irq_thread,
                                     IRQF_TRIGGER_RISING, "tsc2005", ts);
        if (error) {
                dev_err(&spi->dev, "Failed to request irq, err: %d\n", error);
@@ -625,15 +671,7 @@ static int __devinit tsc2005_probe(struct spi_device *spi)
                goto err_remove_sysfs;
        }
 
-       tsc2005_start_scan(ts);
-
-       if (ts->esd_timeout && ts->set_reset) {
-               /* start the optional ESD watchdog */
-               mod_timer(&ts->esd_timer, round_jiffies(jiffies +
-                                       msecs_to_jiffies(ts->esd_timeout)));
-       }
-
-       set_irq_wake(spi->irq, 1);
+       irq_set_irq_wake(spi->irq, 1);
        return 0;
 
 err_remove_sysfs:
@@ -653,17 +691,6 @@ static int __devexit tsc2005_remove(struct spi_device *spi)
 
        sysfs_remove_group(&ts->spi->dev.kobj, &tsc2005_attr_group);
 
-       mutex_lock(&ts->mutex);
-       tsc2005_disable(ts);
-       mutex_unlock(&ts->mutex);
-
-       if (ts->esd_timeout)
-               del_timer_sync(&ts->esd_timer);
-       del_timer_sync(&ts->penup_timer);
-
-       flush_work(&ts->esd_work);
-       flush_work(&ts->penup_work);
-
        free_irq(ts->spi->irq, ts);
        input_unregister_device(ts->idev);
        kfree(ts);
@@ -679,7 +706,12 @@ static int tsc2005_suspend(struct device *dev)
        struct tsc2005 *ts = spi_get_drvdata(spi);
 
        mutex_lock(&ts->mutex);
-       tsc2005_disable(ts);
+
+       if (!ts->suspended && ts->opened)
+               __tsc2005_disable(ts);
+
+       ts->suspended = true;
+
        mutex_unlock(&ts->mutex);
 
        return 0;
@@ -691,7 +723,12 @@ static int tsc2005_resume(struct device *dev)
        struct tsc2005 *ts = spi_get_drvdata(spi);
 
        mutex_lock(&ts->mutex);
-       tsc2005_enable(ts);
+
+       if (ts->suspended && ts->opened)
+               __tsc2005_enable(ts);
+
+       ts->suspended = false;
+
        mutex_unlock(&ts->mutex);
 
        return 0;