* Felipe Balbi <me@xxxxxxxxxxxxxxx> [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 <felipe.balbi@xxxxxxxxx> > 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 <felipe.balbi@xxxxxxxxx> > --- > 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 majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html