Hi Sebastian,

On Wed, Jul 15, 2015 at 02:13:26PM +0200, Sebastian Reichel wrote:

>       /* read the coordinates */
> -     error = spi_sync(ts->spi, &ts->spi_read_msg);
> +     error = regmap_bulk_read(ts->regmap, TSC2005_REG_X, &tsdata,
> +                              TSC2005_DATA_REGS);
>       if (unlikely(error))
>               goto out;
>  
> -     x = ts->spi_x.spi_rx;
> -     y = ts->spi_y.spi_rx;
> -     z1 = ts->spi_z1.spi_rx;
> -     z2 = ts->spi_z2.spi_rx;
> -
>       /* validate position */
> -     if (unlikely(x > MAX_12BIT || y > MAX_12BIT))
> +     if (unlikely(tsdata.x > MAX_12BIT || tsdata.y > MAX_12BIT))
>               goto out;
>  
>       /* Skip reading if the pressure components are out of range */
> -     if (unlikely(z1 == 0 || z2 > MAX_12BIT || z1 >= z2))
> +     if (unlikely(tsdata.z1 == 0 || tsdata.z2 > MAX_12BIT))
> +             goto out;
> +     if (unlikely(tsdata.z1 >= tsdata.z2))
>               goto out;

I guess that means that tsc2005 is (and was) endian-broken. We can't use
the data off the wire directly...  So I'll drop bucnh of the changes in
this function so we can convert to CPU endianness before using.

I also got:

  CC [M]  drivers/input/touchscreen/tsc2005.o
  Building modules, stage 2.
Kernel: arch/x86/boot/bzImage is ready  (#1383)
  MODPOST 1403 modules
ERROR: "devm_regmap_init_spi" [drivers/input/touchscreen/tsc2005.ko]
undefined!
make[1]: *** [__modpost] Error 1
make: *** [modules] Error 2

I guess we need "select REGMAP_SPI".

Thanks.

-- 
Dmitry


Input: tsc2005 - convert to regmap

From: Sebastian Reichel <s...@kernel.org>

Convert driver, so that it uses regmap instead of directly using
spi_transfer for all register accesses.

Signed-off-by: Sebastian Reichel <s...@kernel.org>
Signed-off-by: Dmitry Torokhov <dmitry.torok...@gmail.com>
---
 drivers/input/touchscreen/Kconfig   |    9 +-
 drivers/input/touchscreen/tsc2005.c |  149 ++++++++++++-----------------------
 2 files changed, 55 insertions(+), 103 deletions(-)

diff --git a/drivers/input/touchscreen/Kconfig 
b/drivers/input/touchscreen/Kconfig
index 9b3da52..c1acbbc 100644
--- a/drivers/input/touchscreen/Kconfig
+++ b/drivers/input/touchscreen/Kconfig
@@ -915,10 +915,11 @@ config TOUCHSCREEN_TSC_SERIO
          module will be called tsc40.
 
 config TOUCHSCREEN_TSC2005
-        tristate "TSC2005 based touchscreens"
-        depends on SPI_MASTER
-        help
-          Say Y here if you have a TSC2005 based touchscreen.
+       tristate "TSC2005 based touchscreens"
+       depends on SPI_MASTER
+       select REGMAP_SPI
+       help
+         Say Y here if you have a TSC2005 based touchscreen.
 
          If unsure, say N.
 
diff --git a/drivers/input/touchscreen/tsc2005.c 
b/drivers/input/touchscreen/tsc2005.c
index 2a27a1f..a04248e 100644
--- a/drivers/input/touchscreen/tsc2005.c
+++ b/drivers/input/touchscreen/tsc2005.c
@@ -34,6 +34,7 @@
 #include <linux/spi/spi.h>
 #include <linux/spi/tsc2005.h>
 #include <linux/regulator/consumer.h>
+#include <linux/regmap.h>
 
 /*
  * The touchscreen interface operates as follows:
@@ -120,20 +121,36 @@
 #define TSC2005_SPI_MAX_SPEED_HZ       10000000
 #define TSC2005_PENUP_TIME_MS          40
 
-struct tsc2005_spi_rd {
-       struct spi_transfer     spi_xfer;
-       u32                     spi_tx;
-       u32                     spi_rx;
+static const struct regmap_range tsc2005_writable_ranges[] = {
+       regmap_reg_range(TSC2005_REG_AUX_HIGH, TSC2005_REG_CFR2),
 };
 
+static const struct regmap_access_table tsc2005_writable_table = {
+       .yes_ranges = tsc2005_writable_ranges,
+       .n_yes_ranges = ARRAY_SIZE(tsc2005_writable_ranges),
+};
+
+static struct regmap_config tsc2005_regmap_config = {
+       .reg_bits = 8,
+       .val_bits = 16,
+       .reg_stride = 0x08,
+       .max_register = 0x78,
+       .read_flag_mask = TSC2005_REG_READ,
+       .wr_table = &tsc2005_writable_table,
+       .use_single_rw = true,
+};
+
+struct tsc2005_data {
+       u16 x;
+       u16 y;
+       u16 z1;
+       u16 z2;
+} __packed;
+#define TSC2005_DATA_REGS 4
+
 struct tsc2005 {
        struct spi_device       *spi;
-
-       struct spi_message      spi_read_msg;
-       struct tsc2005_spi_rd   spi_x;
-       struct tsc2005_spi_rd   spi_y;
-       struct tsc2005_spi_rd   spi_z1;
-       struct tsc2005_spi_rd   spi_z2;
+       struct regmap           *regmap;
 
        struct input_dev        *idev;
        char                    phys[32];
@@ -190,62 +207,6 @@ static int tsc2005_cmd(struct tsc2005 *ts, u8 cmd)
        return 0;
 }
 
-static int tsc2005_write(struct tsc2005 *ts, u8 reg, u16 value)
-{
-       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;
-       int error;
-
-       spi_message_init(&msg);
-       spi_message_add_tail(&xfer, &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;
-       rd->spi_xfer.len           = 4;
-       rd->spi_xfer.bits_per_word = 24;
-       rd->spi_xfer.cs_change     = !last;
-}
-
-static int tsc2005_read(struct tsc2005 *ts, u8 reg, u16 *value)
-{
-       struct tsc2005_spi_rd spi_rd;
-       struct spi_message msg;
-       int error;
-
-       tsc2005_setup_read(&spi_rd, reg, true);
-
-       spi_message_init(&msg);
-       spi_message_add_tail(&spi_rd.spi_xfer, &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,
                                     int x, int y, int pressure)
 {
@@ -276,17 +237,19 @@ static irqreturn_t tsc2005_irq_thread(int irq, void *_ts)
        unsigned int pressure;
        u32 x, y;
        u32 z1, z2;
+       struct tsc2005_data tsdata;
        int error;
 
        /* read the coordinates */
-       error = spi_sync(ts->spi, &ts->spi_read_msg);
+       error = regmap_bulk_read(ts->regmap, TSC2005_REG_X, &tsdata,
+                                TSC2005_DATA_REGS);
        if (unlikely(error))
                goto out;
 
-       x = ts->spi_x.spi_rx;
-       y = ts->spi_y.spi_rx;
-       z1 = ts->spi_z1.spi_rx;
-       z2 = ts->spi_z2.spi_rx;
+       x = tsdata.x;
+       y = tsdata.y;
+       z1 = tsdata.z1;
+       z2 = tsdata.z2;
 
        /* validate position */
        if (unlikely(x > MAX_12BIT || y > MAX_12BIT))
@@ -346,9 +309,9 @@ static void tsc2005_penup_timer(unsigned long data)
 
 static void tsc2005_start_scan(struct tsc2005 *ts)
 {
-       tsc2005_write(ts, TSC2005_REG_CFR0, TSC2005_CFR0_INITVALUE);
-       tsc2005_write(ts, TSC2005_REG_CFR1, TSC2005_CFR1_INITVALUE);
-       tsc2005_write(ts, TSC2005_REG_CFR2, TSC2005_CFR2_INITVALUE);
+       regmap_write(ts->regmap, TSC2005_REG_CFR0, TSC2005_CFR0_INITVALUE);
+       regmap_write(ts->regmap, TSC2005_REG_CFR1, TSC2005_CFR1_INITVALUE);
+       regmap_write(ts->regmap, TSC2005_REG_CFR2, TSC2005_CFR2_INITVALUE);
        tsc2005_cmd(ts, TSC2005_CMD_NORMAL);
 }
 
@@ -398,9 +361,9 @@ static ssize_t tsc2005_selftest_show(struct device *dev,
 {
        struct spi_device *spi = to_spi_device(dev);
        struct tsc2005 *ts = spi_get_drvdata(spi);
-       u16 temp_high;
-       u16 temp_high_orig;
-       u16 temp_high_test;
+       unsigned int temp_high;
+       unsigned int temp_high_orig;
+       unsigned int temp_high_test;
        bool success = true;
        int error;
 
@@ -411,7 +374,7 @@ static ssize_t tsc2005_selftest_show(struct device *dev,
         */
        __tsc2005_disable(ts);
 
-       error = tsc2005_read(ts, TSC2005_REG_TEMP_HIGH, &temp_high_orig);
+       error = regmap_read(ts->regmap, TSC2005_REG_TEMP_HIGH, &temp_high_orig);
        if (error) {
                dev_warn(dev, "selftest failed: read error %d\n", error);
                success = false;
@@ -420,14 +383,14 @@ static ssize_t tsc2005_selftest_show(struct device *dev,
 
        temp_high_test = (temp_high_orig - 1) & MAX_12BIT;
 
-       error = tsc2005_write(ts, TSC2005_REG_TEMP_HIGH, temp_high_test);
+       error = regmap_write(ts->regmap, 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);
+       error = regmap_read(ts->regmap, TSC2005_REG_TEMP_HIGH, &temp_high);
        if (error) {
                dev_warn(dev, "selftest failed: read error %d after write\n",
                         error);
@@ -450,7 +413,7 @@ static ssize_t tsc2005_selftest_show(struct device *dev,
                goto out;
 
        /* test that the reset really happened */
-       error = tsc2005_read(ts, TSC2005_REG_TEMP_HIGH, &temp_high);
+       error = regmap_read(ts->regmap, TSC2005_REG_TEMP_HIGH, &temp_high);
        if (error) {
                dev_warn(dev, "selftest failed: read error %d after reset\n",
                         error);
@@ -503,7 +466,7 @@ static void tsc2005_esd_work(struct work_struct *work)
 {
        struct tsc2005 *ts = container_of(work, struct tsc2005, esd_work.work);
        int error;
-       u16 r;
+       unsigned int r;
 
        if (!mutex_trylock(&ts->mutex)) {
                /*
@@ -519,7 +482,7 @@ static void tsc2005_esd_work(struct work_struct *work)
                goto out;
 
        /* We should be able to read register without disabling interrupts. */
-       error = tsc2005_read(ts, TSC2005_REG_CFR0, &r);
+       error = regmap_read(ts->regmap, TSC2005_REG_CFR0, &r);
        if (!error &&
            !((r ^ TSC2005_CFR0_INITVALUE) & TSC2005_CFR0_RW_MASK)) {
                goto out;
@@ -583,20 +546,6 @@ static void tsc2005_close(struct input_dev *input)
        mutex_unlock(&ts->mutex);
 }
 
-static void tsc2005_setup_spi_xfer(struct tsc2005 *ts)
-{
-       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);
-       spi_message_add_tail(&ts->spi_y.spi_xfer, &ts->spi_read_msg);
-       spi_message_add_tail(&ts->spi_z1.spi_xfer, &ts->spi_read_msg);
-       spi_message_add_tail(&ts->spi_z2.spi_xfer, &ts->spi_read_msg);
-}
-
 static int tsc2005_probe(struct spi_device *spi)
 {
        const struct tsc2005_platform_data *pdata = dev_get_platdata(&spi->dev);
@@ -661,6 +610,10 @@ static int tsc2005_probe(struct spi_device *spi)
        ts->spi = spi;
        ts->idev = input_dev;
 
+       ts->regmap = devm_regmap_init_spi(spi, &tsc2005_regmap_config);
+       if (IS_ERR(ts->regmap))
+               return PTR_ERR(ts->regmap);
+
        ts->x_plate_ohm = x_plate_ohm;
        ts->esd_timeout = esd_timeout;
 
@@ -700,8 +653,6 @@ static int tsc2005_probe(struct spi_device *spi)
 
        INIT_DELAYED_WORK(&ts->esd_work, tsc2005_esd_work);
 
-       tsc2005_setup_spi_xfer(ts);
-
        snprintf(ts->phys, sizeof(ts->phys),
                 "%s/input-ts", dev_name(&spi->dev));
 
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to