* Felipe Balbi <[EMAIL PROTECTED]> [080814 17:12]:
> On Thu, Aug 14, 2008 at 04:35:41PM +0300, Tony Lindgren wrote:
> > Thanks, pushing to l-o. Your twl patch also needs to be updated
> > for the include path.
>
> Here it is
Thanks, pushing today.
Tony
> thanks
>
> --
> balbi
> From bc91d54477fef30befd382a538526b8a963aee8d Mon Sep 17 00:00:00 2001
> From: Felipe Balbi <[EMAIL PROTECTED]>
> Date: Thu, 14 Aug 2008 17:06:00 +0300
> Subject: [PATCH] input: keypad: General fixes to omap-twl4030keypad.c
>
> The following patch fixes some problems in T2 keypad
> driver.
>
> Basically we're passing irq number via platform_data,
> moving globals to a structure and fixing a problem
> while iterating over the keymap.
>
> It might be that we still have a few locking issues
> that might be solved on a later version of this same
> patch.
>
> Signed-off-by: Felipe Balbi <[EMAIL PROTECTED]>
> ---
> arch/arm/mach-omap2/board-2430sdp.c | 5 +-
> arch/arm/mach-omap2/board-3430sdp.c | 5 +-
> arch/arm/plat-omap/include/mach/keypad.h | 1 +
> drivers/input/keyboard/omap-twl4030keypad.c | 204
> ++++++++++++++++-----------
> 4 files changed, 129 insertions(+), 86 deletions(-)
>
> diff --git a/arch/arm/mach-omap2/board-2430sdp.c
> b/arch/arm/mach-omap2/board-2430sdp.c
> index cb38fc2..64e76e8 100644
> --- a/arch/arm/mach-omap2/board-2430sdp.c
> +++ b/arch/arm/mach-omap2/board-2430sdp.c
> @@ -176,9 +176,10 @@ static int sdp2430_keymap[] = {
> static struct omap_kp_platform_data sdp2430_kp_data = {
> .rows = 5,
> .cols = 6,
> - .keymap = sdp2430_keymap,
> - .keymapsize = ARRAY_SIZE(sdp2430_keymap),
> + .keymap = sdp2430_keymap,
> + .keymapsize = ARRAY_SIZE(sdp2430_keymap),
> .rep = 1,
> + .irq = TWL4030_MODIRQ_KEYPAD,
> };
>
> static struct platform_device sdp2430_kp_device = {
> diff --git a/arch/arm/mach-omap2/board-3430sdp.c
> b/arch/arm/mach-omap2/board-3430sdp.c
> index 637f1c8..d27158e 100644
> --- a/arch/arm/mach-omap2/board-3430sdp.c
> +++ b/arch/arm/mach-omap2/board-3430sdp.c
> @@ -113,9 +113,10 @@ static int sdp3430_keymap[] = {
> static struct omap_kp_platform_data sdp3430_kp_data = {
> .rows = 5,
> .cols = 6,
> - .keymap = sdp3430_keymap,
> - .keymapsize = ARRAY_SIZE(sdp3430_keymap),
> + .keymap = sdp3430_keymap,
> + .keymapsize = ARRAY_SIZE(sdp3430_keymap),
> .rep = 1,
> + .irq = TWL4030_MODIRQ_KEYPAD,
> };
>
> static struct platform_device sdp3430_kp_device = {
> diff --git a/arch/arm/plat-omap/include/mach/keypad.h
> b/arch/arm/plat-omap/include/mach/keypad.h
> index 232923a..ba1c95c 100644
> --- a/arch/arm/plat-omap/include/mach/keypad.h
> +++ b/arch/arm/plat-omap/include/mach/keypad.h
> @@ -14,6 +14,7 @@ struct omap_kp_platform_data {
> int rows;
> int cols;
> int *keymap;
> + int irq;
> unsigned int keymapsize;
> unsigned int rep:1;
> unsigned long delay;
> diff --git a/drivers/input/keyboard/omap-twl4030keypad.c
> b/drivers/input/keyboard/omap-twl4030keypad.c
> index 5dbb80f..3893d63 100644
> --- a/drivers/input/keyboard/omap-twl4030keypad.c
> +++ b/drivers/input/keyboard/omap-twl4030keypad.c
> @@ -31,6 +31,7 @@
> #include <linux/types.h>
> #include <linux/input.h>
> #include <linux/kernel.h>
> +#include <linux/mutex.h>
> #include <linux/delay.h>
> #include <linux/bitops.h>
> #include <linux/platform_device.h>
> @@ -47,52 +48,65 @@
> #define KEYNUM_MASK 0x00FFFFFF
>
> /* Global variables */
> -static int *keymap;
> -static u16 kp_state[MAX_ROWS];
> -static int n_rows, n_cols;
>
> -static struct device *dbg_dev;
> -static struct input_dev *omap_twl4030kp;
> +struct omap_keypad {
> + int *keymap;
> + unsigned int keymapsize;
> + u16 kp_state[MAX_ROWS];
> + int n_rows;
> + int n_cols;
> + int irq;
>
> -static int twl4030_kpread(u32 module, u8 *data, u32 reg, u8 num_bytes)
> + struct device *dbg_dev;
> + struct input_dev *omap_twl4030kp;
> +
> + /* sync read/write */
> + struct mutex mutex;
> +};
> +
> +static int twl4030_kpread(struct omap_keypad *kp,
> + u32 module, u8 *data, u32 reg, u8 num_bytes)
> {
> int ret;
>
> ret = twl4030_i2c_read(module, data, reg, num_bytes);
> if (ret < 0) {
> - dev_warn(dbg_dev, "Couldn't read TWL4030: %X - ret %d[%x]\n",
> + dev_warn(kp->dbg_dev,
> + "Couldn't read TWL4030: %X - ret %d[%x]\n",
> reg, ret, ret);
> return ret;
> }
> return ret;
> }
>
> -static int twl4030_kpwrite_u8(u32 module, u8 data, u32 reg)
> +static int twl4030_kpwrite_u8(struct omap_keypad *kp,
> + u32 module, u8 data, u32 reg)
> {
> int ret;
>
> ret = twl4030_i2c_write_u8(module, data, reg);
> if (ret < 0) {
> - dev_warn(dbg_dev, "Could not write TWL4030: %X - ret %d[%x]\n",
> + dev_warn(kp->dbg_dev,
> + "Could not write TWL4030: %X - ret %d[%x]\n",
> reg, ret, ret);
> return ret;
> }
> return ret;
> }
>
> -static int omap_kp_find_key(int col, int row)
> +static int omap_kp_find_key(struct omap_keypad *kp, int col, int row)
> {
> int i, rc;
>
> rc = KEY(col, row, 0);
> - for (i = 0; keymap[i] != 0; i++)
> - if ((keymap[i] & ROWCOL_MASK) == rc)
> - return keymap[i] & KEYNUM_MASK;
> + for (i = 0; i < kp->keymapsize; i++)
> + if ((kp->keymap[i] & ROWCOL_MASK) == rc)
> + return kp->keymap[i] & KEYNUM_MASK;
>
> return -EINVAL;
> }
>
> -static inline u16 omap_kp_col_xlate(u8 col)
> +static inline u16 omap_kp_col_xlate(struct omap_keypad *kp, u8 col)
> {
> /* If all bits in a row are active for all coloumns then
> * we have that row line connected to gnd. Mark this
> @@ -100,30 +114,30 @@ static inline u16 omap_kp_col_xlate(u8 col)
> * one higher than the size of the matrix).
> */
> if (col == 0xFF)
> - return (1 << n_cols);
> + return 1 << kp->n_cols;
> else
> - return col & ((1 << n_cols) - 1);
> + return col & ((1 << kp->n_cols) - 1);
> }
>
> -static int omap_kp_read_kp_matrix_state(u16 *state)
> +static int omap_kp_read_kp_matrix_state(struct omap_keypad *kp, u16 *state)
> {
> u8 new_state[MAX_ROWS];
> int row;
> - int ret = twl4030_kpread(TWL4030_MODULE_KEYPAD,
> - new_state, KEYP_FULL_CODE_7_0, n_rows);
> + int ret = twl4030_kpread(kp, TWL4030_MODULE_KEYPAD,
> + new_state, KEYP_FULL_CODE_7_0, kp->n_rows);
> if (ret >= 0) {
> - for (row = 0; row < n_rows; row++)
> - state[row] = omap_kp_col_xlate(new_state[row]);
> + for (row = 0; row < kp->n_rows; row++)
> + state[row] = omap_kp_col_xlate(kp, new_state[row]);
> }
> return ret;
> }
>
> -static int omap_kp_is_in_ghost_state(u16 *key_state)
> +static int omap_kp_is_in_ghost_state(struct omap_keypad *kp, u16 *key_state)
> {
> int i;
> u16 check = 0;
>
> - for (i = 0; i < n_rows; i++) {
> + for (i = 0; i < kp->n_rows; i++) {
> u16 col = key_state[i];
>
> if ((col & check) && hweight16(col) > 1)
> @@ -134,7 +148,7 @@ static int omap_kp_is_in_ghost_state(u16 *key_state)
> return 0;
> }
>
> -static void twl4030_kp_scan(int release_all)
> +static void twl4030_kp_scan(struct omap_keypad *kp, int release_all)
> {
> u16 new_state[MAX_ROWS];
> int col, row;
> @@ -143,60 +157,66 @@ static void twl4030_kp_scan(int release_all)
> memset(new_state, 0, sizeof(new_state));
> else {
> /* check for any changes */
> - int ret = omap_kp_read_kp_matrix_state(new_state);
> + int ret = omap_kp_read_kp_matrix_state(kp, new_state);
> if (ret < 0) /* panic ... */
> return;
>
> - if (omap_kp_is_in_ghost_state(new_state))
> + if (omap_kp_is_in_ghost_state(kp, new_state))
> return;
> }
>
> + mutex_lock(&kp->mutex);
> +
> /* check for changes and print those */
> - for (row = 0; row < n_rows; row++) {
> - int changed = new_state[row] ^ kp_state[row];
> + for (row = 0; row < kp->n_rows; row++) {
> + int changed = new_state[row] ^ kp->kp_state[row];
>
> if (!changed)
> continue;
>
> - for (col = 0; col < n_cols; col++) {
> + for (col = 0; col < kp->n_cols + 1; col++) {
> int key;
>
> if (!(changed & (1 << col)))
> continue;
>
> - dev_dbg(dbg_dev, "key [%d:%d] %s\n", row, col,
> + dev_dbg(kp->dbg_dev, "key [%d:%d] %s\n", row, col,
> (new_state[row] & (1 << col)) ?
> "press" : "release");
>
> - key = omap_kp_find_key(col, row);
> + key = omap_kp_find_key(kp, col, row);
> if (key < 0)
> - dev_warn(dbg_dev, "Spurious key event %d-%d\n",
> + dev_warn(kp->dbg_dev,
> + "Spurious key event %d-%d\n",
> col, row);
> else
> - input_report_key(omap_twl4030kp, key,
> + input_report_key(kp->omap_twl4030kp, key,
> new_state[row] & (1 << col));
> }
> - kp_state[row] = new_state[row];
> + kp->kp_state[row] = new_state[row];
> }
> +
> + mutex_unlock(&kp->mutex);
> }
>
> /*
> * Keypad interrupt handler
> */
> -static irqreturn_t do_kp_irq(int irq, void *dev_id)
> +static irqreturn_t do_kp_irq(int irq, void *_kp)
> {
> + struct omap_keypad *kp = _kp;
> u8 reg;
> int ret;
>
> /* Read & Clear TWL4030 pending interrupt */
> - ret = twl4030_kpread(TWL4030_MODULE_KEYPAD, ®, KEYP_ISR1, 1);
> + ret = twl4030_kpread(kp, TWL4030_MODULE_KEYPAD, ®, KEYP_ISR1, 1);
>
> /* Release all keys if I2C has gone bad or
> * the KEYP has gone to idle state */
> if ((ret >= 0) && (reg & KEYP_IMR1_KP))
> - twl4030_kp_scan(0);
> + twl4030_kp_scan(kp, 0);
> else
> - twl4030_kp_scan(1);
> + twl4030_kp_scan(kp, 1);
>
> return IRQ_HANDLED;
> }
> @@ -210,92 +230,108 @@ static int __init omap_kp_probe(struct platform_device
> *pdev)
> u8 reg;
> int i;
> int ret = 0;
> + struct omap_keypad *kp;
> struct omap_kp_platform_data *pdata = pdev->dev.platform_data;
>
> - /* Get the debug Device */
> - dbg_dev = &(pdev->dev);
> + kp = kzalloc(sizeof(*kp), GFP_KERNEL);
> + if (!kp)
> + return -ENOMEM;
>
> if (!pdata->rows || !pdata->cols || !pdata->keymap) {
> - dev_err(dbg_dev, "No rows, cols or keymap from pdata\n");
> + dev_err(kp->dbg_dev, "No rows, cols or keymap from pdata\n");
> + kfree(kp);
> return -EINVAL;
> }
>
> - omap_twl4030kp = input_allocate_device();
> - if (omap_twl4030kp == NULL)
> + dev_set_drvdata(&pdev->dev, kp);
> +
> + /* Get the debug Device */
> + kp->dbg_dev = &pdev->dev;
> +
> + kp->omap_twl4030kp = input_allocate_device();
> + if (!kp->omap_twl4030kp) {
> + kfree(kp);
> return -ENOMEM;
> + }
>
> - keymap = pdata->keymap;
> - n_rows = pdata->rows;
> - n_cols = pdata->cols;
> + mutex_init(&kp->mutex);
> +
> + kp->keymap = pdata->keymap;
> + kp->keymapsize = pdata->keymapsize;
> + kp->n_rows = pdata->rows;
> + kp->n_cols = pdata->cols;
> + kp->irq = pdata->irq;
>
> /* setup input device */
> - set_bit(EV_KEY, omap_twl4030kp->evbit);
> + set_bit(EV_KEY, kp->omap_twl4030kp->evbit);
>
> /* Enable auto repeat feature of Linux input subsystem */
> if (pdata->rep)
> - set_bit(EV_REP, omap_twl4030kp->evbit);
> + set_bit(EV_REP, kp->omap_twl4030kp->evbit);
>
> - for (i = 0; keymap[i] != 0; i++)
> - set_bit(keymap[i] & KEYNUM_MASK, omap_twl4030kp->keybit);
> + for (i = 0; i < kp->keymapsize; i++)
> + set_bit(kp->keymap[i] & KEYNUM_MASK,
> + kp->omap_twl4030kp->keybit);
>
> - omap_twl4030kp->name = "omap_twl4030keypad";
> - omap_twl4030kp->phys = "omap_twl4030keypad/input0";
> - omap_twl4030kp->dev.parent = &pdev->dev;
> + kp->omap_twl4030kp->name = "omap_twl4030keypad";
> + kp->omap_twl4030kp->phys = "omap_twl4030keypad/input0";
> + kp->omap_twl4030kp->dev.parent = &pdev->dev;
>
> - omap_twl4030kp->id.bustype = BUS_HOST;
> - omap_twl4030kp->id.vendor = 0x0001;
> - omap_twl4030kp->id.product = 0x0001;
> - omap_twl4030kp->id.version = 0x0003;
> + kp->omap_twl4030kp->id.bustype = BUS_HOST;
> + kp->omap_twl4030kp->id.vendor = 0x0001;
> + kp->omap_twl4030kp->id.product = 0x0001;
> + kp->omap_twl4030kp->id.version = 0x0003;
>
> - omap_twl4030kp->keycode = keymap;
> - omap_twl4030kp->keycodesize = sizeof(unsigned int);
> - omap_twl4030kp->keycodemax = pdata->keymapsize;
> + kp->omap_twl4030kp->keycode = kp->keymap;
> + kp->omap_twl4030kp->keycodesize = sizeof(unsigned int);
> + kp->omap_twl4030kp->keycodemax = kp->keymapsize;
>
> - ret = input_register_device(omap_twl4030kp);
> + ret = input_register_device(kp->omap_twl4030kp);
> if (ret < 0) {
> - dev_err(dbg_dev, "Unable to register twl4030 keypad device\n");
> + dev_err(kp->dbg_dev,
> + "Unable to register twl4030 keypad device\n");
> goto err2;
> }
>
> /* Disable auto-repeat */
> reg = KEYP_CTRL_NOAUTORPT;
> - ret = twl4030_kpwrite_u8(TWL4030_MODULE_KEYPAD, reg, KEYP_CTRL);
> + ret = twl4030_kpwrite_u8(kp, TWL4030_MODULE_KEYPAD, reg, KEYP_CTRL);
> if (ret < 0)
> goto err3;
>
> /* Enable TO rising and KP rising and falling edge detection */
> reg = KEYP_EDR_KP_BOTH | KEYP_EDR_TO_RISING;
> - ret = twl4030_kpwrite_u8(TWL4030_MODULE_KEYPAD, reg, KEYP_EDR);
> + ret = twl4030_kpwrite_u8(kp, TWL4030_MODULE_KEYPAD, reg, KEYP_EDR);
> if (ret < 0)
> goto err3;
>
> /* Set PTV prescaler Field */
> reg = (PTV_PRESCALER << KEYP_LK_PTV_PTV_SHIFT);
> - ret = twl4030_kpwrite_u8(TWL4030_MODULE_KEYPAD, reg, KEYP_LK_PTV);
> + ret = twl4030_kpwrite_u8(kp, TWL4030_MODULE_KEYPAD, reg, KEYP_LK_PTV);
> if (ret < 0)
> goto err3;
>
> /* Set key debounce time to 20 ms */
> i = KEYP_PERIOD_US(20000, PTV_PRESCALER);
> - ret = twl4030_kpwrite_u8(TWL4030_MODULE_KEYPAD, i, KEYP_DEB);
> + ret = twl4030_kpwrite_u8(kp, TWL4030_MODULE_KEYPAD, i, KEYP_DEB);
> if (ret < 0)
> goto err3;
>
> /* Set timeout period to 100 ms */
> i = KEYP_PERIOD_US(200000, PTV_PRESCALER);
> - ret = twl4030_kpwrite_u8(TWL4030_MODULE_KEYPAD,
> + ret = twl4030_kpwrite_u8(kp, TWL4030_MODULE_KEYPAD,
> (i & 0xFF), KEYP_TIMEOUT_L);
> if (ret < 0)
> goto err3;
>
> - ret = twl4030_kpwrite_u8(TWL4030_MODULE_KEYPAD,
> + ret = twl4030_kpwrite_u8(kp, TWL4030_MODULE_KEYPAD,
> (i >> 8), KEYP_TIMEOUT_H);
> if (ret < 0)
> goto err3;
>
> /* Enable Clear-on-Read */
> reg = KEYP_SIH_CTRL_COR | KEYP_SIH_CTRL_PEND_DIS;
> - ret = twl4030_kpwrite_u8(TWL4030_MODULE_KEYPAD,
> + ret = twl4030_kpwrite_u8(kp, TWL4030_MODULE_KEYPAD,
> reg, KEYP_SIH_CTRL);
> if (ret < 0)
> goto err3;
> @@ -304,50 +340,54 @@ static int __init omap_kp_probe(struct platform_device
> *pdev)
> * This ISR will always execute in kernel thread context because of
> * the need to access the TWL4030 over the I2C bus.
> */
> - ret = request_irq(TWL4030_MODIRQ_KEYPAD, do_kp_irq,
> - IRQF_DISABLED, "TWL4030 Keypad", omap_twl4030kp);
> + ret = request_irq(kp->irq, do_kp_irq, IRQF_DISABLED,
> + "TWL4030 Keypad", kp);
> if (ret < 0) {
> - dev_info(dbg_dev, "request_irq failed for irq no=%d\n",
> - TWL4030_MODIRQ_KEYPAD);
> + dev_info(kp->dbg_dev, "request_irq failed for irq no=%d\n",
> + kp->irq);
> goto err3;
> } else {
> /* Enable KP and TO interrupts now. */
> reg = ~(KEYP_IMR1_KP | KEYP_IMR1_TO);
> - ret = twl4030_kpwrite_u8(TWL4030_MODULE_KEYPAD,
> + ret = twl4030_kpwrite_u8(kp, TWL4030_MODULE_KEYPAD,
> reg, KEYP_IMR1);
> if (ret < 0)
> goto err5;
> }
>
> - ret = omap_kp_read_kp_matrix_state(kp_state);
> + ret = omap_kp_read_kp_matrix_state(kp, kp->kp_state);
> if (ret < 0)
> goto err4;
>
> return ret;
> err5:
> /* mask all events - we don't care about the result */
> - (void) twl4030_kpwrite_u8(TWL4030_MODULE_KEYPAD, 0xff, KEYP_IMR1);
> + (void) twl4030_kpwrite_u8(kp, TWL4030_MODULE_KEYPAD, 0xff, KEYP_IMR1);
> err4:
> - free_irq(TWL4030_MODIRQ_KEYPAD, NULL);
> + free_irq(kp->irq, NULL);
> err3:
> - input_unregister_device(omap_twl4030kp);
> + input_unregister_device(kp->omap_twl4030kp);
> err2:
> - input_free_device(omap_twl4030kp);
> + input_free_device(kp->omap_twl4030kp);
> +
> return -ENODEV;
> }
>
> static int omap_kp_remove(struct platform_device *pdev)
> {
> - free_irq(TWL4030_MODIRQ_KEYPAD, NULL);
> + struct omap_keypad *kp = dev_get_drvdata(&pdev->dev);
> +
> + free_irq(kp->irq, kp);
> + input_unregister_device(kp->omap_twl4030kp);
> + kfree(kp);
>
> - input_unregister_device(omap_twl4030kp);
> return 0;
> }
>
>
> static struct platform_driver omap_kp_driver = {
> .probe = omap_kp_probe,
> - .remove = omap_kp_remove,
> + .remove = __devexit_p(omap_kp_remove),
> .driver = {
> .name = "omap_twl4030keypad",
> .owner = THIS_MODULE,
> --
> 1.6.0.rc3.6.ga0653
>
--
To unsubscribe from this list: send the line "unsubscribe linux-omap" in
the body of a message to [EMAIL PROTECTED]
More majordomo info at http://vger.kernel.org/majordomo-info.html