On Tue, Feb 12, 2013 at 11:05:12AM +0100, Samuel Ortiz wrote:
> Adding Dmitry to the thread, for the input parts.

Looks reasonable, however the patch is against older version of the
driver and most likely will not apply anymore...

Thanks.

> 
> On Thu, Dec 20, 2012 at 09:57:19AM +0100, Jean-Nicolas Graux wrote:
> > Provides support for 1801 variant of stmpe gpio port expanders.
> > This chip has 18 gpios configurable as GPI, GPO, keypad matrix,
> > special key or dedicated key function.
> > 
> > Note that special/dedicated key function is not supported yet.
> > 
> > Signed-off-by: Jean-Nicolas Graux <jean-nicolas.gr...@stericsson.com>
> > ---
> >  drivers/gpio/gpio-stmpe.c             |   52 +++++--
> >  drivers/input/keyboard/stmpe-keypad.c |  251 
> > +++++++++++++++++++++++++++------
> >  drivers/mfd/Kconfig                   |    1 +
> >  drivers/mfd/stmpe-i2c.c               |    1 +
> >  drivers/mfd/stmpe.c                   |   97 ++++++++++++-
> >  drivers/mfd/stmpe.h                   |   49 +++++++
> >  include/linux/mfd/stmpe.h             |    3 +
> >  7 files changed, 392 insertions(+), 62 deletions(-)
> > 
> > diff --git a/drivers/gpio/gpio-stmpe.c b/drivers/gpio/gpio-stmpe.c
> > index dce3472..662c415 100644
> > --- a/drivers/gpio/gpio-stmpe.c
> > +++ b/drivers/gpio/gpio-stmpe.c
> > @@ -42,14 +42,25 @@ static inline struct stmpe_gpio *to_stmpe_gpio(struct 
> > gpio_chip *chip)
> >     return container_of(chip, struct stmpe_gpio, chip);
> >  }
> >  
> > +static inline u8 stmpe_gpio_reg(struct stmpe *stmpe, u8 index, int offset)
> > +{
> > +   u8 reg;
> > +   if (stmpe->partnum == STMPE1801)
> > +           reg = stmpe->regs[index] + offset;
> > +   else
> > +           reg = stmpe->regs[index] - offset;
> > +   return reg;
> > +}
> > +
> >  static int stmpe_gpio_get(struct gpio_chip *chip, unsigned offset)
> >  {
> >     struct stmpe_gpio *stmpe_gpio = to_stmpe_gpio(chip);
> >     struct stmpe *stmpe = stmpe_gpio->stmpe;
> > -   u8 reg = stmpe->regs[STMPE_IDX_GPMR_LSB] - (offset / 8);
> > -   u8 mask = 1 << (offset % 8);
> >     int ret;
> > +   u8 reg, mask;
> >  
> > +   reg = stmpe_gpio_reg(stmpe, STMPE_IDX_GPMR_LSB, offset / 8);
> > +   mask = 1 << (offset % 8);
> >     ret = stmpe_reg_read(stmpe, reg);
> >     if (ret < 0)
> >             return ret;
> > @@ -62,8 +73,10 @@ static void stmpe_gpio_set(struct gpio_chip *chip, 
> > unsigned offset, int val)
> >     struct stmpe_gpio *stmpe_gpio = to_stmpe_gpio(chip);
> >     struct stmpe *stmpe = stmpe_gpio->stmpe;
> >     int which = val ? STMPE_IDX_GPSR_LSB : STMPE_IDX_GPCR_LSB;
> > -   u8 reg = stmpe->regs[which] - (offset / 8);
> > -   u8 mask = 1 << (offset % 8);
> > +   u8 reg, mask;
> > +
> > +   reg = stmpe_gpio_reg(stmpe, which, offset / 8);
> > +   mask = 1 << (offset % 8);
> >  
> >     /*
> >      * Some variants have single register for gpio set/clear functionality.
> > @@ -80,8 +93,10 @@ static int stmpe_gpio_direction_output(struct gpio_chip 
> > *chip,
> >  {
> >     struct stmpe_gpio *stmpe_gpio = to_stmpe_gpio(chip);
> >     struct stmpe *stmpe = stmpe_gpio->stmpe;
> > -   u8 reg = stmpe->regs[STMPE_IDX_GPDR_LSB] - (offset / 8);
> > -   u8 mask = 1 << (offset % 8);
> > +   u8 reg, mask;
> > +
> > +   reg = stmpe_gpio_reg(stmpe, STMPE_IDX_GPDR_LSB, offset / 8);
> > +   mask = 1 << (offset % 8);
> >  
> >     stmpe_gpio_set(chip, offset, val);
> >  
> > @@ -93,8 +108,10 @@ static int stmpe_gpio_direction_input(struct gpio_chip 
> > *chip,
> >  {
> >     struct stmpe_gpio *stmpe_gpio = to_stmpe_gpio(chip);
> >     struct stmpe *stmpe = stmpe_gpio->stmpe;
> > -   u8 reg = stmpe->regs[STMPE_IDX_GPDR_LSB] - (offset / 8);
> > -   u8 mask = 1 << (offset % 8);
> > +   u8 reg, mask;
> > +
> > +   reg = stmpe_gpio_reg(stmpe, STMPE_IDX_GPDR_LSB, offset / 8);
> > +   mask = 1 << (offset % 8);
> >  
> >     return stmpe_set_bits(stmpe, reg, mask, 0);
> >  }
> > @@ -174,6 +191,7 @@ static void stmpe_gpio_irq_sync_unlock(struct irq_data 
> > *d)
> >             [REG_IE]        = STMPE_IDX_IEGPIOR_LSB,
> >     };
> >     int i, j;
> > +   u8 reg;
> >  
> >     for (i = 0; i < CACHE_NR_REGS; i++) {
> >             /* STMPE801 doesn't have RE and FE registers */
> > @@ -189,7 +207,8 @@ static void stmpe_gpio_irq_sync_unlock(struct irq_data 
> > *d)
> >                             continue;
> >  
> >                     stmpe_gpio->oldregs[i][j] = new;
> > -                   stmpe_reg_write(stmpe, stmpe->regs[regmap[i]] - j, new);
> > +                   reg = stmpe_gpio_reg(stmpe, regmap[i], j);
> > +                   stmpe_reg_write(stmpe, reg, new);
> >             }
> >     }
> >  
> > @@ -229,18 +248,20 @@ static irqreturn_t stmpe_gpio_irq(int irq, void *dev)
> >  {
> >     struct stmpe_gpio *stmpe_gpio = dev;
> >     struct stmpe *stmpe = stmpe_gpio->stmpe;
> > -   u8 statmsbreg = stmpe->regs[STMPE_IDX_ISGPIOR_MSB];
> >     int num_banks = DIV_ROUND_UP(stmpe->num_gpios, 8);
> >     u8 status[num_banks];
> >     int ret;
> >     int i;
> > +   bool lsb = stmpe->partnum == STMPE1801;
> > +   u8 statmsbreg = lsb ? stmpe->regs[STMPE_IDX_ISGPIOR_LSB] :
> > +                   stmpe->regs[STMPE_IDX_ISGPIOR_MSB];
> >  
> >     ret = stmpe_block_read(stmpe, statmsbreg, num_banks, status);
> >     if (ret < 0)
> >             return IRQ_NONE;
> >  
> >     for (i = 0; i < num_banks; i++) {
> > -           int bank = num_banks - i - 1;
> > +           int bank = lsb ? i : num_banks - i - 1;
> >             unsigned int enabled = stmpe_gpio->regs[REG_IE][bank];
> >             unsigned int stat = status[i];
> >  
> > @@ -258,10 +279,11 @@ static irqreturn_t stmpe_gpio_irq(int irq, void *dev)
> >  
> >             stmpe_reg_write(stmpe, statmsbreg + i, status[i]);
> >  
> > -           /* Edge detect register is not present on 801 */
> > -           if (stmpe->partnum != STMPE801)
> > -                   stmpe_reg_write(stmpe, stmpe->regs[STMPE_IDX_GPEDR_MSB]
> > -                                   + i, status[i]);
> > +           /* Edge detect register is not present on 801 and 1801 */
> > +           if (stmpe->partnum != STMPE801 && stmpe->partnum != STMPE1801)
> > +                   stmpe_reg_write(stmpe,
> > +                           stmpe->regs[STMPE_IDX_GPEDR_MSB] + i,
> > +                           status[i]);
> >     }
> >  
> >     return IRQ_HANDLED;
> > diff --git a/drivers/input/keyboard/stmpe-keypad.c 
> > b/drivers/input/keyboard/stmpe-keypad.c
> > index 8923352..2b081b7 100644
> > --- a/drivers/input/keyboard/stmpe-keypad.c
> > +++ b/drivers/input/keyboard/stmpe-keypad.c
> > @@ -14,41 +14,114 @@
> >  #include <linux/input/matrix_keypad.h>
> >  #include <linux/mfd/stmpe.h>
> >  
> > -/* These are at the same addresses in all STMPE variants */
> > -#define STMPE_KPC_COL                      0x60
> > -#define STMPE_KPC_ROW_MSB          0x61
> > -#define STMPE_KPC_ROW_LSB          0x62
> > -#define STMPE_KPC_CTRL_MSB         0x63
> > -#define STMPE_KPC_CTRL_LSB         0x64
> > -#define STMPE_KPC_COMBI_KEY_0              0x65
> > -#define STMPE_KPC_COMBI_KEY_1              0x66
> > -#define STMPE_KPC_COMBI_KEY_2              0x67
> > -#define STMPE_KPC_DATA_BYTE0               0x68
> > -#define STMPE_KPC_DATA_BYTE1               0x69
> > -#define STMPE_KPC_DATA_BYTE2               0x6a
> > -#define STMPE_KPC_DATA_BYTE3               0x6b
> > -#define STMPE_KPC_DATA_BYTE4               0x6c
> > +/* These are at the same addresses in most of STMPE variants */
> > +#define STMPE_REG_KPC_COL          0x60
> > +#define STMPE_REG_KPC_ROW_MSB              0x61
> > +#define STMPE_REG_KPC_ROW_LSB              0x62
> > +#define STMPE_REG_KPC_CTRL_MSB             0x63
> > +#define STMPE_REG_KPC_CTRL_LSB             0x64
> > +#define STMPE_REG_KPC_COMBI_KEY_0  0x65
> > +#define STMPE_REG_KPC_COMBI_KEY_1  0x66
> > +#define STMPE_REG_KPC_COMBI_KEY_2  0x67
> > +#define STMPE_REG_KPC_DATA_BYTE0   0x68
> > +#define STMPE_REG_KPC_DATA_BYTE1   0x69
> > +#define STMPE_REG_KPC_DATA_BYTE2   0x6a
> > +#define STMPE_REG_KPC_DATA_BYTE3   0x6b
> > +#define STMPE_REG_KPC_DATA_BYTE4   0x6c
> >  
> >  #define STMPE_KPC_CTRL_LSB_SCAN            (0x1 << 0)
> >  #define STMPE_KPC_CTRL_LSB_DEBOUNCE        (0x7f << 1)
> >  #define STMPE_KPC_CTRL_MSB_SCAN_COUNT      (0xf << 4)
> >  
> > +/* STMPE1801 */
> > +#define STMPE1801_REG_KPC_ROW              0x30
> > +#define STMPE1801_REG_KPC_COL_LOW  0x31
> > +#define STMPE1801_REG_KPC_COL_HIGH 0x32
> > +#define STMPE1801_REG_KPC_CTRL_LOW 0x33
> > +#define STMPE1801_REG_KPC_CTRL_MID 0x34
> > +#define STMPE1801_REG_KPC_CTRL_HIGH        0x35
> > +#define STMPE1801_REG_KPC_CMD              0x36
> > +#define STMPE1801_REG_KPC_COMBI_KEY_0      0x37
> > +#define STMPE1801_REG_KPC_COMBI_KEY_1      0x38
> > +#define STMPE1801_REG_KPC_COMBI_KEY_2      0x39
> > +#define STMPE1801_REG_KPC_DATA_BYTE0       0x3a
> > +#define STMPE1801_REG_KPC_DATA_BYTE1       0x3b
> > +#define STMPE1801_REG_KPC_DATA_BYTE2       0x3c
> > +#define STMPE1801_REG_KPC_DATA_BYTE3       0x3d
> > +#define STMPE1801_REG_KPC_DATA_BYTE4       0x3e
> > +
> > +#define STMPE1801_MSK_KPC_SCAN_COUNT       (0xf << 4)
> > +#define STMPE1801_MSK_KPC_DEBOUNCE (0x3f << 2)
> > +#define STMPE1801_MSK_KPC_CMD_SCAN (0x1 << 0)
> > +
> >  #define STMPE_KPC_ROW_MSB_ROWS             0xff
> >  
> >  #define STMPE_KPC_DATA_UP          (0x1 << 7)
> > -#define STMPE_KPC_DATA_ROW         (0xf << 3)
> > -#define STMPE_KPC_DATA_COL         (0x7 << 0)
> > +
> >  #define STMPE_KPC_DATA_NOKEY_MASK  0x78
> >  
> >  #define STMPE_KEYPAD_MAX_DEBOUNCE  127
> >  #define STMPE_KEYPAD_MAX_SCAN_COUNT        15
> >  
> >  #define STMPE_KEYPAD_MAX_ROWS              8
> > -#define STMPE_KEYPAD_MAX_COLS              8
> > -#define STMPE_KEYPAD_ROW_SHIFT             3
> > +#define STMPE_KEYPAD_MAX_COLS              10
> > +
> >  #define STMPE_KEYPAD_KEYMAP_SIZE   \
> >     (STMPE_KEYPAD_MAX_ROWS * STMPE_KEYPAD_MAX_COLS)
> >  
> > +enum {
> > +   STMPE_IDX_KPC_COL_LSB,
> > +   STMPE_IDX_KPC_COL_MSB,
> > +   STMPE_IDX_KPC_ROW_LSB,
> > +   STMPE_IDX_KPC_ROW_MSB,
> > +   STMPE_IDX_KPC_CTRL_LSB,
> > +   STMPE_IDX_KPC_CTRL_MID,
> > +   STMPE_IDX_KPC_CTRL_MSB,
> > +   STMPE_IDX_KPC_CMD,
> > +   STMPE_IDX_KPC_COMBI_KEY_0,
> > +   STMPE_IDX_KPC_COMBI_KEY_1,
> > +   STMPE_IDX_KPC_COMBI_KEY_2,
> > +   STMPE_IDX_KPC_DATA_BYTE0,
> > +   STMPE_IDX_KPC_DATA_BYTE1,
> > +   STMPE_IDX_KPC_DATA_BYTE2,
> > +   STMPE_IDX_KPC_DATA_BYTE3,
> > +   STMPE_IDX_KPC_DATA_BYTE4,
> > +};
> > +
> > +static const u8 stmpe_default_regs[] = {
> > +   [STMPE_IDX_KPC_COL_LSB]         = STMPE_REG_KPC_COL,
> > +   [STMPE_IDX_KPC_ROW_LSB]         = STMPE_REG_KPC_ROW_LSB,
> > +   [STMPE_IDX_KPC_ROW_MSB]         = STMPE_REG_KPC_ROW_MSB,
> > +   [STMPE_IDX_KPC_CTRL_LSB]        = STMPE_REG_KPC_CTRL_LSB,
> > +   [STMPE_IDX_KPC_CTRL_MSB]        = STMPE_REG_KPC_CTRL_MSB,
> > +   [STMPE_IDX_KPC_COMBI_KEY_0]     = STMPE_REG_KPC_COMBI_KEY_0,
> > +   [STMPE_IDX_KPC_COMBI_KEY_1]     = STMPE_REG_KPC_COMBI_KEY_1,
> > +   [STMPE_IDX_KPC_COMBI_KEY_2]     = STMPE_REG_KPC_COMBI_KEY_2,
> > +   [STMPE_IDX_KPC_DATA_BYTE0]      = STMPE_REG_KPC_DATA_BYTE0,
> > +   [STMPE_IDX_KPC_DATA_BYTE1]      = STMPE_REG_KPC_DATA_BYTE1,
> > +   [STMPE_IDX_KPC_DATA_BYTE2]      = STMPE_REG_KPC_DATA_BYTE2,
> > +   [STMPE_IDX_KPC_DATA_BYTE3]      = STMPE_REG_KPC_DATA_BYTE3,
> > +   [STMPE_IDX_KPC_DATA_BYTE4]      = STMPE_REG_KPC_DATA_BYTE4,
> > +};
> > +
> > +static const u8 stmpe_1801_regs[] = {
> > +   [STMPE_IDX_KPC_COL_LSB]         = STMPE1801_REG_KPC_COL_LOW,
> > +   [STMPE_IDX_KPC_COL_MSB]         = STMPE1801_REG_KPC_COL_HIGH,
> > +   [STMPE_IDX_KPC_ROW_LSB]         = STMPE1801_REG_KPC_ROW,
> > +   [STMPE_IDX_KPC_CTRL_LSB]        = STMPE1801_REG_KPC_CTRL_LOW,
> > +   [STMPE_IDX_KPC_CTRL_MID]        = STMPE1801_REG_KPC_CTRL_MID,
> > +   [STMPE_IDX_KPC_CTRL_MSB]        = STMPE1801_REG_KPC_CTRL_HIGH,
> > +   [STMPE_IDX_KPC_CMD]             = STMPE1801_REG_KPC_CMD,
> > +   [STMPE_IDX_KPC_COMBI_KEY_0]     = STMPE1801_REG_KPC_COMBI_KEY_0,
> > +   [STMPE_IDX_KPC_COMBI_KEY_1]     = STMPE1801_REG_KPC_COMBI_KEY_1,
> > +   [STMPE_IDX_KPC_COMBI_KEY_2]     = STMPE1801_REG_KPC_COMBI_KEY_2,
> > +   [STMPE_IDX_KPC_DATA_BYTE0]      = STMPE1801_REG_KPC_DATA_BYTE0,
> > +   [STMPE_IDX_KPC_DATA_BYTE1]      = STMPE1801_REG_KPC_DATA_BYTE1,
> > +   [STMPE_IDX_KPC_DATA_BYTE2]      = STMPE1801_REG_KPC_DATA_BYTE2,
> > +   [STMPE_IDX_KPC_DATA_BYTE3]      = STMPE1801_REG_KPC_DATA_BYTE3,
> > +   [STMPE_IDX_KPC_DATA_BYTE4]      = STMPE1801_REG_KPC_DATA_BYTE4,
> > +};
> > +
> >  /**
> >   * struct stmpe_keypad_variant - model-specific attributes
> >   * @auto_increment: whether the KPC_DATA_BYTE register address
> > @@ -57,6 +130,10 @@
> >   * @num_normal_data: number of normal keys' data bytes
> >   * @max_cols: maximum number of columns supported
> >   * @max_rows: maximum number of rows supported
> > + * @row_mask: mask used to get row number in KPC_DATA_BYTEx registers
> > + * @col_mask: mask used to get column number in KPC_DATA_BYTEx registers
> > + * @row_shift: shift used to get row number in KPC_DATA_BYTEx registers
> > + * @col_shift: shift used to get column number in KPC_DATA_BYTEx registers
> >   * @col_gpios: bitmask of gpios which can be used for columns
> >   * @row_gpios: bitmask of gpios which can be used for rows
> >   */
> > @@ -66,8 +143,13 @@ struct stmpe_keypad_variant {
> >     int             num_normal_data;
> >     int             max_cols;
> >     int             max_rows;
> > +   unsigned int    row_mask;
> > +   unsigned int    col_mask;
> > +   unsigned char   row_shift;
> > +   unsigned char   col_shift;
> >     unsigned int    col_gpios;
> >     unsigned int    row_gpios;
> > +   const u8        *regs;
> >  };
> >  
> >  static const struct stmpe_keypad_variant stmpe_keypad_variants[] = {
> > @@ -77,8 +159,27 @@ static const struct stmpe_keypad_variant 
> > stmpe_keypad_variants[] = {
> >             .num_normal_data        = 3,
> >             .max_cols               = 8,
> >             .max_rows               = 8,
> > +           .row_mask               = (0xf << 3),
> > +           .row_shift              = 3,
> > +           .col_mask               = (0x7 << 0),
> > +           .col_shift              = 0,
> >             .col_gpios              = 0x000ff,      /* GPIO 0 - 7 */
> >             .row_gpios              = 0x0ff00,      /* GPIO 8 - 15 */
> > +           .regs                   = stmpe_default_regs,
> > +   },
> > +   [STMPE1801] = {
> > +           .auto_increment         = true,
> > +           .num_data               = 5,
> > +           .num_normal_data        = 3,
> > +           .max_cols               = 10,
> > +           .max_rows               = 8,
> > +           .row_mask               = (0x7 << 0),
> > +           .row_shift              = 0,
> > +           .col_mask               = (0xf << 3),
> > +           .col_shift              = 3,
> > +           .col_gpios              = 0x3ff00,      /* GPIO 8 - 17 */
> > +           .row_gpios              = 0x000ff,      /* GPIO 0 - 7 */
> > +           .regs                   = stmpe_1801_regs,
> >     },
> >     [STMPE2401] = {
> >             .auto_increment         = false,
> > @@ -86,8 +187,13 @@ static const struct stmpe_keypad_variant 
> > stmpe_keypad_variants[] = {
> >             .num_normal_data        = 2,
> >             .max_cols               = 8,
> >             .max_rows               = 12,
> > +           .row_mask               = (0xf << 3),
> > +           .row_shift              = 3,
> > +           .col_mask               = (0x7 << 0),
> > +           .col_shift              = 0,
> >             .col_gpios              = 0x0000ff,     /* GPIO 0 - 7*/
> >             .row_gpios              = 0x1fef00,     /* GPIO 8-14, 16-20 */
> > +           .regs                   = stmpe_default_regs,
> >     },
> >     [STMPE2403] = {
> >             .auto_increment         = true,
> > @@ -95,8 +201,13 @@ static const struct stmpe_keypad_variant 
> > stmpe_keypad_variants[] = {
> >             .num_normal_data        = 3,
> >             .max_cols               = 8,
> >             .max_rows               = 12,
> > +           .row_mask               = (0xf << 3),
> > +           .row_shift              = 3,
> > +           .col_mask               = (0x7 << 0),
> > +           .col_shift              = 0,
> >             .col_gpios              = 0x0000ff,     /* GPIO 0 - 7*/
> >             .row_gpios              = 0x1fef00,     /* GPIO 8-14, 16-20 */
> > +           .regs                   = stmpe_default_regs,
> >     },
> >  };
> >  
> > @@ -108,6 +219,7 @@ struct stmpe_keypad {
> >  
> >     unsigned int rows;
> >     unsigned int cols;
> > +   unsigned char scan_code_row_shift;
> >     bool enable;
> >  
> >     unsigned short keymap[STMPE_KEYPAD_KEYMAP_SIZE];
> > @@ -162,11 +274,13 @@ static int stmpe_keypad_read_data(struct stmpe_keypad 
> > *keypad, u8 *data)
> >     int i;
> >  
> >     if (variant->auto_increment)
> > -           return stmpe_block_read(stmpe, STMPE_KPC_DATA_BYTE0,
> > -                                   variant->num_data, data);
> > +           return stmpe_block_read(stmpe,
> > +                           variant->regs[STMPE_IDX_KPC_DATA_BYTE0],
> > +                           variant->num_data, data);
> >  
> >     for (i = 0; i < variant->num_data; i++) {
> > -           ret = stmpe_reg_read(stmpe, STMPE_KPC_DATA_BYTE0 + i);
> > +           ret = stmpe_reg_read(stmpe,
> > +                           variant->regs[STMPE_IDX_KPC_DATA_BYTE0] + i);
> >             if (ret < 0)
> >                     return ret;
> >  
> > @@ -191,9 +305,10 @@ static irqreturn_t stmpe_keypad_irq(int irq, void *dev)
> >  
> >     for (i = 0; i < variant->num_normal_data; i++) {
> >             u8 data = fifo[i];
> > -           int row = (data & STMPE_KPC_DATA_ROW) >> 3;
> > -           int col = data & STMPE_KPC_DATA_COL;
> > -           int code = MATRIX_SCAN_CODE(row, col, STMPE_KEYPAD_ROW_SHIFT);
> > +           int row = (data & variant->row_mask) >> variant->row_shift;
> > +           int col = (data & variant->col_mask) >> variant->col_shift;
> > +           int code = MATRIX_SCAN_CODE(row, col,
> > +                           keypad->scan_code_row_shift);
> >             bool up = data & STMPE_KPC_DATA_UP;
> >  
> >             if ((data & STMPE_KPC_DATA_NOKEY_MASK)
> > @@ -270,33 +385,69 @@ static int __devinit stmpe_keypad_chip_init(struct 
> > stmpe_keypad *keypad)
> >     if (ret < 0)
> >             return ret;
> >  
> > -   ret = stmpe_reg_write(stmpe, STMPE_KPC_COL, keypad->cols);
> > +   ret = stmpe_reg_write(stmpe, variant->regs[STMPE_IDX_KPC_COL_LSB],
> > +                   keypad->cols);
> >     if (ret < 0)
> >             return ret;
> >  
> > -   ret = stmpe_reg_write(stmpe, STMPE_KPC_ROW_LSB, keypad->rows);
> > +   if (stmpe->partnum == STMPE1801 && variant->max_cols > 8) {
> > +           ret = stmpe_set_bits(stmpe,
> > +                           variant->regs[STMPE_IDX_KPC_COL_MSB],
> > +                           0x3,
> > +                           keypad->cols >> 8);
> > +           if (ret < 0)
> > +                   return ret;
> > +   }
> > +
> > +   ret = stmpe_reg_write(stmpe, variant->regs[STMPE_IDX_KPC_ROW_LSB],
> > +                   keypad->rows);
> >     if (ret < 0)
> >             return ret;
> >  
> >     if (variant->max_rows > 8) {
> > -           ret = stmpe_set_bits(stmpe, STMPE_KPC_ROW_MSB,
> > -                                STMPE_KPC_ROW_MSB_ROWS,
> > -                                keypad->rows >> 8);
> > +           ret = stmpe_set_bits(stmpe,
> > +                           variant->regs[STMPE_IDX_KPC_ROW_MSB],
> > +                           STMPE_KPC_ROW_MSB_ROWS,
> > +                           keypad->rows >> 8);
> >             if (ret < 0)
> >                     return ret;
> >     }
> >  
> > -   ret = stmpe_set_bits(stmpe, STMPE_KPC_CTRL_MSB,
> > -                        STMPE_KPC_CTRL_MSB_SCAN_COUNT,
> > -                        plat->scan_count << 4);
> > -   if (ret < 0)
> > -           return ret;
> > +   if (stmpe->partnum == STMPE1801) {
> > +           ret = stmpe_set_bits(stmpe,
> > +                           variant->regs[STMPE_IDX_KPC_CTRL_LSB],
> > +                           STMPE1801_MSK_KPC_SCAN_COUNT,
> > +                           plat->scan_count << 4);
> > +           if (ret < 0)
> > +                   return ret;
> > +
> > +           ret = stmpe_set_bits(stmpe,
> > +                           variant->regs[STMPE_IDX_KPC_CTRL_MID],
> > +                           STMPE1801_MSK_KPC_DEBOUNCE,
> > +                           (plat->debounce_ms << 1));
> > +           if (ret < 0)
> > +                   return ret;
> > +
> > +           return  stmpe_set_bits(stmpe,
> > +                           variant->regs[STMPE_IDX_KPC_CMD],
> > +                           STMPE1801_MSK_KPC_CMD_SCAN,
> > +                           STMPE1801_MSK_KPC_CMD_SCAN);
> > +   } else {
> > +           ret = stmpe_set_bits(stmpe,
> > +                           variant->regs[STMPE_IDX_KPC_CTRL_MSB],
> > +                           STMPE_KPC_CTRL_MSB_SCAN_COUNT,
> > +                           plat->scan_count << 4);
> > +           if (ret < 0)
> > +                   return ret;
> > +
> > +           return stmpe_set_bits(stmpe,
> > +                           variant->regs[STMPE_IDX_KPC_CTRL_LSB],
> > +                           STMPE_KPC_CTRL_LSB_SCAN |
> > +                           STMPE_KPC_CTRL_LSB_DEBOUNCE,
> > +                           STMPE_KPC_CTRL_LSB_SCAN |
> > +                           (plat->debounce_ms << 1));
> > +   }
> >  
> > -   return stmpe_set_bits(stmpe, STMPE_KPC_CTRL_LSB,
> > -                         STMPE_KPC_CTRL_LSB_SCAN |
> > -                         STMPE_KPC_CTRL_LSB_DEBOUNCE,
> > -                         STMPE_KPC_CTRL_LSB_SCAN |
> > -                         (plat->debounce_ms << 1));
> >  }
> >  
> >  static int __devinit stmpe_keypad_probe(struct platform_device *pdev)
> > @@ -341,8 +492,21 @@ static int __devinit stmpe_keypad_probe(struct 
> > platform_device *pdev)
> >     input->keycodesize = sizeof(keypad->keymap[0]);
> >     input->keycodemax = ARRAY_SIZE(keypad->keymap);
> >  
> > -   matrix_keypad_build_keymap(plat->keymap_data, STMPE_KEYPAD_ROW_SHIFT,
> > -                              input->keycode, input->keybit);
> > +   keypad->stmpe = stmpe;
> > +   keypad->plat = plat;
> > +   keypad->input = input;
> > +   keypad->variant = &stmpe_keypad_variants[stmpe->partnum];
> > +
> > +   /*
> > +    * compute keypad->scan_code_row_shift by figuring out
> > +    * how many bits are needed to encode keypad->variant->max_cols
> > +    */
> > +   keypad->scan_code_row_shift =
> > +                   get_count_order(keypad->variant->max_cols);
> > +
> > +   matrix_keypad_build_keymap(plat->keymap_data,
> > +                   keypad->scan_code_row_shift,
> > +                   input->keycode, input->keybit);
> >  
> >     for (i = 0; i < plat->keymap_data->keymap_size; i++) {
> >             unsigned int key = plat->keymap_data->keymap[i];
> > @@ -351,11 +515,6 @@ static int __devinit stmpe_keypad_probe(struct 
> > platform_device *pdev)
> >             keypad->rows |= 1 << KEY_ROW(key);
> >     }
> >  
> > -   keypad->stmpe = stmpe;
> > -   keypad->plat = plat;
> > -   keypad->input = input;
> > -   keypad->variant = &stmpe_keypad_variants[stmpe->partnum];
> > -
> >     ret = stmpe_keypad_chip_init(keypad);
> >     if (ret < 0)
> >             goto out_freeinput;
> > diff --git a/drivers/mfd/Kconfig b/drivers/mfd/Kconfig
> > index acf35bf..e3713c8 100644
> > --- a/drivers/mfd/Kconfig
> > +++ b/drivers/mfd/Kconfig
> > @@ -293,6 +293,7 @@ config MFD_STMPE
> >  
> >             STMPE811: GPIO, Touchscreen
> >             STMPE1601: GPIO, Keypad
> > +           STMPE1801: GPIO, Keypad
> >             STMPE2401: GPIO, Keypad
> >             STMPE2403: GPIO, Keypad
> >  
> > diff --git a/drivers/mfd/stmpe-i2c.c b/drivers/mfd/stmpe-i2c.c
> > index 373f423..a2135f8 100644
> > --- a/drivers/mfd/stmpe-i2c.c
> > +++ b/drivers/mfd/stmpe-i2c.c
> > @@ -75,6 +75,7 @@ static const struct i2c_device_id stmpe_i2c_id[] = {
> >     { "stmpe801", STMPE801 },
> >     { "stmpe811", STMPE811 },
> >     { "stmpe1601", STMPE1601 },
> > +   { "stmpe1801", STMPE1801 },
> >     { "stmpe2401", STMPE2401 },
> >     { "stmpe2403", STMPE2403 },
> >     { }
> > diff --git a/drivers/mfd/stmpe.c b/drivers/mfd/stmpe.c
> > index 6b8f941..902038d 100644
> > --- a/drivers/mfd/stmpe.c
> > +++ b/drivers/mfd/stmpe.c
> > @@ -15,6 +15,7 @@
> >  #include <linux/pm.h>
> >  #include <linux/slab.h>
> >  #include <linux/mfd/core.h>
> > +#include <linux/delay.h>
> >  #include "stmpe.h"
> >  
> >  static int __stmpe_enable(struct stmpe *stmpe, unsigned int blocks)
> > @@ -643,6 +644,88 @@ static struct stmpe_variant_info stmpe1601 = {
> >  };
> >  
> >  /*
> > + * STMPE1801
> > + */
> > +static const u8 stmpe1801_regs[] = {
> > +   [STMPE_IDX_CHIP_ID]     = STMPE1801_REG_CHIP_ID,
> > +   [STMPE_IDX_ICR_LSB]     = STMPE1801_REG_INT_CTRL_LOW,
> > +   [STMPE_IDX_IER_LSB]     = STMPE1801_REG_INT_EN_MASK_LOW,
> > +   [STMPE_IDX_ISR_LSB]     = STMPE1801_REG_INT_STA_LOW,
> > +   [STMPE_IDX_GPMR_LSB]    = STMPE1801_REG_GPIO_MP_LOW,
> > +   [STMPE_IDX_GPSR_LSB]    = STMPE1801_REG_GPIO_SET_LOW,
> > +   [STMPE_IDX_GPCR_LSB]    = STMPE1801_REG_GPIO_CLR_LOW,
> > +   [STMPE_IDX_GPDR_LSB]    = STMPE1801_REG_GPIO_SET_DIR_LOW,
> > +   [STMPE_IDX_GPRER_LSB]   = STMPE1801_REG_GPIO_RE_LOW,
> > +   [STMPE_IDX_GPFER_LSB]   = STMPE1801_REG_GPIO_FE_LOW,
> > +   [STMPE_IDX_IEGPIOR_LSB] = STMPE1801_REG_INT_EN_GPIO_MASK_LOW,
> > +   [STMPE_IDX_ISGPIOR_LSB] = STMPE1801_REG_INT_STA_GPIO_LOW,
> > +};
> > +
> > +static struct stmpe_variant_block stmpe1801_blocks[] = {
> > +   {
> > +           .cell   = &stmpe_gpio_cell,
> > +           .irq    = STMPE1801_IRQ_GPIOC,
> > +           .block  = STMPE_BLOCK_GPIO,
> > +   },
> > +   {
> > +           .cell   = &stmpe_keypad_cell,
> > +           .irq    = STMPE1801_IRQ_KEYPAD,
> > +           .block  = STMPE_BLOCK_KEYPAD,
> > +   },
> > +};
> > +
> > +static int stmpe1801_enable(struct stmpe *stmpe, unsigned int blocks,
> > +                       bool enable)
> > +{
> > +   unsigned int mask = 0;
> > +   if (blocks & STMPE_BLOCK_GPIO)
> > +           mask |= STMPE1801_MSK_INT_EN_GPIO;
> > +
> > +   if (blocks & STMPE_BLOCK_KEYPAD)
> > +           mask |= STMPE1801_MSK_INT_EN_KPC;
> > +
> > +   return __stmpe_set_bits(stmpe, STMPE1801_REG_INT_EN_MASK_LOW, mask,
> > +                           enable ? mask : 0);
> > +}
> > +
> > +static int stmpe1801_reset(struct stmpe *stmpe)
> > +{
> > +   unsigned long timeout;
> > +   int ret = 0;
> > +
> > +   ret = __stmpe_set_bits(stmpe, STMPE1801_REG_SYS_CTRL,
> > +           STMPE1801_MSK_SYS_CTRL_RESET, STMPE1801_MSK_SYS_CTRL_RESET);
> > +   if (ret < 0)
> > +           return ret;
> > +
> > +   timeout = jiffies + msecs_to_jiffies(100);
> > +   while (time_before(jiffies, timeout)) {
> > +           ret = __stmpe_reg_read(stmpe, STMPE1801_REG_SYS_CTRL);
> > +           if (ret < 0)
> > +                   return ret;
> > +           if (!(ret & STMPE1801_MSK_SYS_CTRL_RESET))
> > +                   return 0;
> > +           usleep_range(100, 200);
> > +   };
> > +   return -EIO;
> > +}
> > +
> > +static struct stmpe_variant_info stmpe1801 = {
> > +   .name           = "stmpe1801",
> > +   .id_val         = STMPE1801_ID,
> > +   .id_mask        = 0xfff0,
> > +   .num_gpios      = 18,
> > +   .af_bits        = 0,
> > +   .regs           = stmpe1801_regs,
> > +   .blocks         = stmpe1801_blocks,
> > +   .num_blocks     = ARRAY_SIZE(stmpe1801_blocks),
> > +   .num_irqs       = STMPE1801_NR_INTERNAL_IRQS,
> > +   .enable         = stmpe1801_enable,
> > +   /* stmpe1801 do not have any gpio alternate function */
> > +   .get_altfunc    = NULL,
> > +};
> > +
> > +/*
> >   * STMPE24XX
> >   */
> >  
> > @@ -740,6 +823,7 @@ static struct stmpe_variant_info 
> > *stmpe_variant_info[STMPE_NBR_PARTS] = {
> >     [STMPE801]      = &stmpe801,
> >     [STMPE811]      = &stmpe811,
> >     [STMPE1601]     = &stmpe1601,
> > +   [STMPE1801]     = &stmpe1801,
> >     [STMPE2401]     = &stmpe2401,
> >     [STMPE2403]     = &stmpe2403,
> >  };
> > @@ -759,7 +843,7 @@ static irqreturn_t stmpe_irq(int irq, void *data)
> >     struct stmpe *stmpe = data;
> >     struct stmpe_variant_info *variant = stmpe->variant;
> >     int num = DIV_ROUND_UP(variant->num_irqs, 8);
> > -   u8 israddr = stmpe->regs[STMPE_IDX_ISR_MSB];
> > +   u8 israddr;
> >     u8 isr[num];
> >     int ret;
> >     int i;
> > @@ -769,6 +853,11 @@ static irqreturn_t stmpe_irq(int irq, void *data)
> >             return IRQ_HANDLED;
> >     }
> >  
> > +   if (variant->id_val == STMPE1801_ID)
> > +           israddr = stmpe->regs[STMPE_IDX_ISR_LSB];
> > +   else
> > +           israddr = stmpe->regs[STMPE_IDX_ISR_MSB];
> > +
> >     ret = stmpe_block_read(stmpe, israddr, num, isr);
> >     if (ret < 0)
> >             return IRQ_NONE;
> > @@ -936,6 +1025,12 @@ static int __devinit stmpe_chip_init(struct stmpe 
> > *stmpe)
> >     if (ret)
> >             return ret;
> >  
> > +   if (id == STMPE1801_ID) {
> > +           ret =  stmpe1801_reset(stmpe);
> > +           if (ret < 0)
> > +                   return ret;
> > +   }
> > +
> >     if (stmpe->irq >= 0) {
> >             if (id == STMPE801_ID)
> >                     icr = STMPE801_REG_SYS_CTRL_INT_EN;
> > diff --git a/drivers/mfd/stmpe.h b/drivers/mfd/stmpe.h
> > index 7b8e13f..ff2b09b 100644
> > --- a/drivers/mfd/stmpe.h
> > +++ b/drivers/mfd/stmpe.h
> > @@ -199,6 +199,55 @@ int stmpe_remove(struct stmpe *stmpe);
> >  #define STPME1601_AUTOSLEEP_ENABLE         (1 << 3)
> >  
> >  /*
> > + * STMPE1801
> > + */
> > +#define STMPE1801_ID                       0xc110
> > +#define STMPE1801_NR_INTERNAL_IRQS 5
> > +#define STMPE1801_IRQ_KEYPAD_COMBI 4
> > +#define STMPE1801_IRQ_GPIOC                3
> > +#define STMPE1801_IRQ_KEYPAD_OVER  2
> > +#define STMPE1801_IRQ_KEYPAD               1
> > +#define STMPE1801_IRQ_WAKEUP               0
> > +
> > +#define STMPE1801_REG_CHIP_ID                      0x00
> > +#define STMPE1801_REG_SYS_CTRL                     0x02
> > +#define STMPE1801_REG_INT_CTRL_LOW         0x04
> > +#define STMPE1801_REG_INT_EN_MASK_LOW              0x06
> > +#define STMPE1801_REG_INT_STA_LOW          0x08
> > +#define STMPE1801_REG_INT_EN_GPIO_MASK_LOW 0x0A
> > +#define STMPE1801_REG_INT_EN_GPIO_MASK_MID 0x0B
> > +#define STMPE1801_REG_INT_EN_GPIO_MASK_HIGH        0x0C
> > +#define STMPE1801_REG_INT_STA_GPIO_LOW             0x0D
> > +#define STMPE1801_REG_INT_STA_GPIO_MID             0x0E
> > +#define STMPE1801_REG_INT_STA_GPIO_HIGH            0x0F
> > +#define STMPE1801_REG_GPIO_SET_LOW         0x10
> > +#define STMPE1801_REG_GPIO_SET_MID         0x11
> > +#define STMPE1801_REG_GPIO_SET_HIGH                0x12
> > +#define STMPE1801_REG_GPIO_CLR_LOW         0x13
> > +#define STMPE1801_REG_GPIO_CLR_MID         0x14
> > +#define STMPE1801_REG_GPIO_CLR_HIGH                0x15
> > +#define STMPE1801_REG_GPIO_MP_LOW          0x16
> > +#define STMPE1801_REG_GPIO_MP_MID          0x17
> > +#define STMPE1801_REG_GPIO_MP_HIGH         0x18
> > +#define STMPE1801_REG_GPIO_SET_DIR_LOW             0x19
> > +#define STMPE1801_REG_GPIO_SET_DIR_MID             0x1A
> > +#define STMPE1801_REG_GPIO_SET_DIR_HIGH            0x1B
> > +#define STMPE1801_REG_GPIO_RE_LOW          0x1C
> > +#define STMPE1801_REG_GPIO_RE_MID          0x1D
> > +#define STMPE1801_REG_GPIO_RE_HIGH         0x1E
> > +#define STMPE1801_REG_GPIO_FE_LOW          0x1F
> > +#define STMPE1801_REG_GPIO_FE_MID          0x20
> > +#define STMPE1801_REG_GPIO_FE_HIGH         0x21
> > +#define STMPE1801_REG_GPIO_PULL_UP_LOW             0x22
> > +#define STMPE1801_REG_GPIO_PULL_UP_MID             0x23
> > +#define STMPE1801_REG_GPIO_PULL_UP_HIGH            0x24
> > +
> > +#define STMPE1801_MSK_SYS_CTRL_RESET               (1 << 7)
> > +
> > +#define STMPE1801_MSK_INT_EN_KPC           (1 << 1)
> > +#define STMPE1801_MSK_INT_EN_GPIO          (1 << 3)
> > +
> > +/*
> >   * STMPE24xx
> >   */
> >  
> > diff --git a/include/linux/mfd/stmpe.h b/include/linux/mfd/stmpe.h
> > index f8d5b4d..5b87ec4 100644
> > --- a/include/linux/mfd/stmpe.h
> > +++ b/include/linux/mfd/stmpe.h
> > @@ -26,6 +26,7 @@ enum stmpe_partnum {
> >     STMPE801,
> >     STMPE811,
> >     STMPE1601,
> > +   STMPE1801,
> >     STMPE2401,
> >     STMPE2403,
> >     STMPE_NBR_PARTS
> > @@ -39,6 +40,7 @@ enum {
> >     STMPE_IDX_CHIP_ID,
> >     STMPE_IDX_ICR_LSB,
> >     STMPE_IDX_IER_LSB,
> > +   STMPE_IDX_ISR_LSB,
> >     STMPE_IDX_ISR_MSB,
> >     STMPE_IDX_GPMR_LSB,
> >     STMPE_IDX_GPSR_LSB,
> > @@ -49,6 +51,7 @@ enum {
> >     STMPE_IDX_GPFER_LSB,
> >     STMPE_IDX_GPAFR_U_MSB,
> >     STMPE_IDX_IEGPIOR_LSB,
> > +   STMPE_IDX_ISGPIOR_LSB,
> >     STMPE_IDX_ISGPIOR_MSB,
> >     STMPE_IDX_MAX,
> >  };
> > -- 
> > 1.7.10
> > 
> 
> -- 
> Intel Open Source Technology Centre
> http://oss.intel.com/
> ---------------------------------------------------------------------
> Intel Corporation SAS (French simplified joint stock company)
> Registered headquarters: "Les Montalets"- 2, rue de Paris, 
> 92196 Meudon Cedex, France
> Registration Number:  302 456 199 R.C.S. NANTERRE
> Capital: 4,572,000 Euros
> 
> This e-mail and any attachments may contain confidential material for
> the sole use of the intended recipient(s). Any review or distribution
> by others is strictly prohibited. If you are not the intended
> recipient, please contact the sender and delete all copies.
> 

-- 
Dmitry
--
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