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 +- drivers/input/keyboard/omap-twl4030keypad.c | 204 ++++++++++++++++----------- include/asm-arm/arch-omap/keypad.h | 1 + 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 b2f8b9c..3270ea6 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 ee4ec18..94ad839 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/drivers/input/keyboard/omap-twl4030keypad.c b/drivers/input/keyboard/omap-twl4030keypad.c index 68c5b8e..05c7cf8 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, diff --git a/include/asm-arm/arch-omap/keypad.h b/include/asm-arm/arch-omap/keypad.h index b7f8307..ac87f37 100644 --- a/include/asm-arm/arch-omap/keypad.h +++ b/include/asm-arm/arch-omap/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; -- 1.6.0.rc1.71.gfba5 -- 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