Hi Dmitry , On Wed, May 9, 2012 at 10:48 AM, Dmitry Torokhov <dmitry.torokhov@xxxxxxxxx> wrote: > Ho Sourav, > > On Thu, Apr 26, 2012 at 11:24:37AM +0530, Sourav Poddar wrote: >> >> -config KEYBOARD_OMAP4 >> - tristate "TI OMAP4 keypad support" >> +config KEYBOARD_OMAP4+ > > I think this works purely by accident - '+' sign getting dropped by > parser... > >> @@ -139,16 +192,33 @@ static int omap4_keypad_open(struct input_dev *input) >> >> disable_irq(keypad_data->irq); >> >> - __raw_writel(OMAP4_VAL_FUNCTIONALCFG, >> - keypad_data->base + OMAP4_KBD_CTRL); >> - __raw_writel(OMAP4_VAL_DEBOUNCINGTIME, >> - keypad_data->base + OMAP4_KBD_DEBOUNCINGTIME); >> - __raw_writel(OMAP4_VAL_IRQDISABLE, >> - keypad_data->base + OMAP4_KBD_IRQSTATUS); >> - __raw_writel(OMAP4_DEF_IRQENABLE_EVENTEN | OMAP4_DEF_IRQENABLE_LONGKEY, >> - keypad_data->base + OMAP4_KBD_IRQENABLE); >> - __raw_writel(OMAP4_DEF_WUP_EVENT_ENA | OMAP4_DEF_WUP_LONG_KEY_ENA, >> - keypad_data->base + OMAP4_KBD_WAKEUPENABLE); >> + keypad_data->revision = kbd_read_revision(keypad_data, >> + OMAP4_KBD_REVISION); >> + switch (keypad_data->revision) { >> + case 1: >> + keypad_data->irqstatus = OMAP4_KBD_IRQSTATUS + 0x0c; >> + keypad_data->irqenable = OMAP4_KBD_IRQENABLE + 0x0c; >> + keypad_data->reg_offset = 0x10; >> + break; > > This should be done in probe(). > Dont we then require "pm_runtime_put_sync" in probe, since we are trying to read the keypad revision register.? > Could you please tell me if the version of the patch below works for > you? > I just quickly tested this patch and saw the following kernel carsh. [ 1.571838] Bad mode in data abort handler detected [ 1.576934] Internal error: Oops - bad mode: 0 [#1] SMP ARM [ 1.582763] Modules linked in: [ 1.585937] CPU: 0 Not tainted (3.4.0-rc5-dirty #11) [ 1.591522] PC is at __key.13633+0x3f39df40/0x8 [ 1.596252] LR is at omap4_keypad_probe+0x1cc/0x3dc [ 1.601348] pc : [<ffff0018>] lr : [<c0466ffc>] psr: 40000092 [ 1.601348] sp : df831e68 ip : fc31c000 fp : 00000040 [ 1.613311] r10: c2460000 r9 : 00000080 r8 : 00000003 [ 1.618774] r7 : de36e880 r6 : df966008 r5 : df965480 r4 : c2422bc0 [ 1.625579] r3 : 60000013 r2 : 00000000 r1 : 00000000 r0 : 0000001e [ 1.632385] Flags: nZcv IRQs off FIQs on Mode IRQ_32 ISA ARM Segment kernel [ 1.640106] Control: 10c53c7d Table: 8000404a DAC: 00000017 [ 1.646118] Process swapper/0 (pid: 1, stack limit = 0xdf8302f8) [ 1.652374] Stack: (0xdf831e68 to 0xdf832000) [ 1.656921] 1e60: 0000001e 00000000 00000000 60000013 c2422bc0 df965480 [ 1.665466] 1e80: df966008 de36e880 00000003 00000080 c2460000 00000040 fc31c000 df831e68 [ 1.674011] 1ea0: c0466ffc ffff0018 40000092 ffffffff 00000000 c008b744 df967590 00000098 [ 1.682556] 1ec0: df96603c df966008 c0c49dd8 df96603c c06f2a00 0000008e c06687f0 c0643fcc [ 1.691070] 1ee0: 00000000 c02b8748 c02b8730 c02b72c4 df966008 c06f2a00 df96603c 00000000 [ 1.699615] 1f00: 0000008e c02b74d8 c06f2a00 df831f18 c02b7444 c02b5b2c df8402a8 df968e90 [ 1.708160] 1f20: 00000000 c06f2a00 c06e5e28 de395bc0 00000000 c02b625c c0575768 c024d028 [ 1.716705] 1f40: df830000 c06f2a00 00000000 00000000 0000008e c06687f0 c0643fcc c02b7ac4 [ 1.725219] 1f60: df830000 00000007 c06fdc00 00000000 0000008e c0008630 00000001 c1058170 [ 1.733764] 1f80: c05f8500 c0643fcc 00000001 60000013 00000001 c05716e0 00000006 00000006 [ 1.742309] 1fa0: 60000013 c064dbb0 00000007 c06fdc00 c061e20c 0000008e c06687f0 c064dbb8 [ 1.750854] 1fc0: 00000000 c061e374 00000006 00000006 c061e20c 00000000 00000000 c061e28c [ 1.759399] 1fe0: c00152b8 00000013 00000000 00000000 00000000 c00152b8 ffffffff 00000000 [ 1.767944] [<c0466ffc>] (omap4_keypad_probe+0x1cc/0x3dc) from [<df831e68>] (0xdf831e68) [ 1.776397] Code: e59ff410 ea0000bb ea00009a ea0000fa (ea000078) [ 1.782775] ------------[ cut here ]------------ [ 1.787628] WARNING: at arch/arm/mach-omap2/omap_l3_noc.c:113 l3_interrupt_handler+0x17c/0x1b4() [ 1.796783] L3 custom error: MASTER:MPU TARGET:L4CFG [ 1.801971] Modules linked in: [ 1.805175] [<c001bba4>] (unwind_backtrace+0x0/0xf4) from [<c003f9f0>] (warn_slowpath_common+0x4c/0x64) [ 1.815002] [<c003f9f0>] (warn_slowpath_common+0x4c/0x64) from [<c003fa9c>] (warn_slowpath_fmt+0x30/0x40) [ 1.825012] [<c003fa9c>] (warn_slowpath_fmt+0x30/0x40) from [<c00341ac>] (l3_interrupt_handler+0x17c/0x1b4) [ 1.835205] [<c00341ac>] (l3_interrupt_handler+0x17c/0x1b4) from [<c009d88c>] (handle_irq_event_percpu+0x64/0x24c) [ 1.846008] [<c009d88c>] (handle_irq_event_percpu+0x64/0x24c) from [<c009dab0>] (handle_irq_event+0x3c/0x5c) [ 1.856292] [<c009dab0>] (handle_irq_event+0x3c/0x5c) from [<c00a05f4>] (handle_fasteoi_irq+0x98/0x13c) [ 1.866088] [<c00a05f4>] (handle_fasteoi_irq+0x98/0x13c) from [<c009d330>] (generic_handle_irq+0x34/0x44) [ 1.876098] [<c009d330>] (generic_handle_irq+0x34/0x44) from [<c00151cc>] (handle_IRQ+0x4c/0xac) [ 1.885284] [<c00151cc>] (handle_IRQ+0x4c/0xac) from [<c0008480>] (gic_handle_irq+0x2c/0x60) [ 1.894073] [<c0008480>] (gic_handle_irq+0x2c/0x60) from [<c04713a4>] (__irq_svc+0x44/0x60) [ 1.902801] Exception stack(0xdf831ce8 to 0xdf831d30) [ 1.908081] 1ce0: c06df3c8 0000000a 00000001 00000000 00000282 00000000 [ 1.916625] 1d00: df830000 c066e080 00000000 c070ffc0 ffff001a ffff001c df831d10 df831d30 [ 1.925170] 1d20: c0047390 c0047394 60000013 ffffffff [ 1.930450] [<c04713a4>] (__irq_svc+0x44/0x60) from [<c0047394>] (__do_softirq+0x64/0x214) [ 1.939086] [<c0047394>] (__do_softirq+0x64/0x214) from [<c0047720>] (irq_exit+0x90/0x98) [ 1.947631] [<c0047720>] (irq_exit+0x90/0x98) from [<c00151d0>] (handle_IRQ+0x50/0xac) [ 1.955902] [<c00151d0>] (handle_IRQ+0x50/0xac) from [<c0008480>] (gic_handle_irq+0x2c/0x60) [ 1.964691] [<c0008480>] (gic_handle_irq+0x2c/0x60) from [<c04713a4>] (__irq_svc+0x44/0x60) [ 1.973419] Exception stack(0xdf831da0 to 0xdf831de8) [ 1.978668] 1da0: c06df3c8 00000000 df831dd8 00000000 c068ffa0 df831e68 00000001 00000001 [ 1.987213] 1dc0: df831e2f 00000000 ffff001a ffff001c ffffffff df831de8 c0471138 c047113c [ 1.995758] 1de0: 60000013 ffffffff [ 1.999420] [<c04713a4>] (__irq_svc+0x44/0x60) from [<c047113c>] (_raw_spin_unlock_irq+0x28/0x2c) [ 2.008697] [<c047113c>] (_raw_spin_unlock_irq+0x28/0x2c) from [<c001857c>] (die+0xe0/0x324) [ 2.017486] [<c001857c>] (die+0xe0/0x324) from [<c00187fc>] (bad_mode+0x3c/0x60) [ 2.025207] [<c00187fc>] (bad_mode+0x3c/0x60) from [<c0466ffc>] (omap4_keypad_probe+0x1cc/0x3dc) [ 2.034393] [<c0466ffc>] (omap4_keypad_probe+0x1cc/0x3dc) from [<df831e68>] (0xdf831e68) [ 2.042877] ---[ end trace 2dead3f602d0a977 ]--- [ 2.047760] ---[ end trace 2dead3f602d0a978 ]--- [ 2.052642] Kernel panic - not syncing: Attempted to kill init! exitcode=0x0000000b Looking more into this. ~Sourav > Thanks. > > -- > Dmitry > > > Input: omap-keypad - dynamically handle register offsets > > From: G, Manjunath Kondaiah <manjugk@xxxxxx> > > Keypad controller register offsets are different for omap4 > and omap5. Handle these offsets through static mapping and > assign these mappings during run time. > > Tested on omap4430 sdp with 3.4-rc3. > Tested on omap5430evm with 3.1-custom kernel. > > Signed-off-by: Felipe Balbi <balbi@xxxxxx> > Signed-off-by: G, Manjunath Kondaiah <manjugk@xxxxxx> > Signed-off-by: Sourav Poddar <sourav.poddar@xxxxxx> > Signed-off-by: Dmitry Torokhov <dtor@xxxxxxx> > --- > > drivers/input/keyboard/Kconfig | 4 + > drivers/input/keyboard/omap4-keypad.c | 102 +++++++++++++++++++++++++-------- > 2 files changed, 80 insertions(+), 26 deletions(-) > > > diff --git a/drivers/input/keyboard/Kconfig b/drivers/input/keyboard/Kconfig > index 20a3753..84ee155 100644 > --- a/drivers/input/keyboard/Kconfig > +++ b/drivers/input/keyboard/Kconfig > @@ -531,9 +531,9 @@ config KEYBOARD_OMAP > module will be called omap-keypad. > > config KEYBOARD_OMAP4 > - tristate "TI OMAP4 keypad support" > + tristate "TI OMAP4+ keypad support" > help > - Say Y here if you want to use the OMAP4 keypad. > + Say Y here if you want to use the OMAP4+ keypad. > > To compile this driver as a module, choose M here: the > module will be called omap4-keypad. > diff --git a/drivers/input/keyboard/omap4-keypad.c b/drivers/input/keyboard/omap4-keypad.c > index e809ac0..ddd5c9e 100644 > --- a/drivers/input/keyboard/omap4-keypad.c > +++ b/drivers/input/keyboard/omap4-keypad.c > @@ -68,19 +68,52 @@ > > #define OMAP4_MASK_IRQSTATUSDISABLE 0xFFFF > > +enum { > + KBD_REVISION_OMAP4 = 0, > + KBD_REVISION_OMAP5, > +}; > + > struct omap4_keypad { > struct input_dev *input; > > void __iomem *base; > - int irq; > + unsigned int irq; > > unsigned int rows; > unsigned int cols; > + u32 reg_offset; > + u32 irqreg_offset; > unsigned int row_shift; > unsigned char key_state[8]; > unsigned short keymap[]; > }; > > +static int kbd_readl(struct omap4_keypad *keypad_data, u32 offset) > +{ > + return __raw_readl(keypad_data->base + > + keypad_data->reg_offset + offset); > +} > + > +static void kbd_writel(struct omap4_keypad *keypad_data, u32 offset, u32 value) > +{ > + __raw_writel(value, > + keypad_data->base + keypad_data->reg_offset + offset); > +} > + > +static int kbd_read_irqreg(struct omap4_keypad *keypad_data, u32 offset) > +{ > + return __raw_readl(keypad_data->base + > + keypad_data->irqreg_offset + offset); > +} > + > +static void kbd_write_irqreg(struct omap4_keypad *keypad_data, > + u32 offset, u32 value) > +{ > + __raw_writel(value, > + keypad_data->base + keypad_data->irqreg_offset + offset); > +} > + > + > /* Interrupt handler */ > static irqreturn_t omap4_keypad_interrupt(int irq, void *dev_id) > { > @@ -91,12 +124,11 @@ static irqreturn_t omap4_keypad_interrupt(int irq, void *dev_id) > u32 *new_state = (u32 *) key_state; > > /* Disable interrupts */ > - __raw_writel(OMAP4_VAL_IRQDISABLE, > - keypad_data->base + OMAP4_KBD_IRQENABLE); > + kbd_write_irqreg(keypad_data, OMAP4_KBD_IRQENABLE, > + OMAP4_VAL_IRQDISABLE); > > - *new_state = __raw_readl(keypad_data->base + OMAP4_KBD_FULLCODE31_0); > - *(new_state + 1) = __raw_readl(keypad_data->base > - + OMAP4_KBD_FULLCODE63_32); > + *new_state = kbd_readl(keypad_data, OMAP4_KBD_FULLCODE31_0); > + *(new_state + 1) = kbd_readl(keypad_data, OMAP4_KBD_FULLCODE63_32); > > for (row = 0; row < keypad_data->rows; row++) { > changed = key_state[row] ^ keypad_data->key_state[row]; > @@ -121,12 +153,13 @@ static irqreturn_t omap4_keypad_interrupt(int irq, void *dev_id) > sizeof(keypad_data->key_state)); > > /* clear pending interrupts */ > - __raw_writel(__raw_readl(keypad_data->base + OMAP4_KBD_IRQSTATUS), > - keypad_data->base + OMAP4_KBD_IRQSTATUS); > + kbd_write_irqreg(keypad_data, OMAP4_KBD_IRQSTATUS, > + kbd_read_irqreg(keypad_data, OMAP4_KBD_IRQSTATUS)); > > /* enable interrupts */ > - __raw_writel(OMAP4_DEF_IRQENABLE_EVENTEN | OMAP4_DEF_IRQENABLE_LONGKEY, > - keypad_data->base + OMAP4_KBD_IRQENABLE); > + kbd_write_irqreg(keypad_data, OMAP4_KBD_IRQENABLE, > + OMAP4_DEF_IRQENABLE_EVENTEN | > + OMAP4_DEF_IRQENABLE_LONGKEY); > > return IRQ_HANDLED; > } > @@ -139,16 +172,17 @@ static int omap4_keypad_open(struct input_dev *input) > > disable_irq(keypad_data->irq); > > - __raw_writel(OMAP4_VAL_FUNCTIONALCFG, > - keypad_data->base + OMAP4_KBD_CTRL); > - __raw_writel(OMAP4_VAL_DEBOUNCINGTIME, > - keypad_data->base + OMAP4_KBD_DEBOUNCINGTIME); > - __raw_writel(OMAP4_VAL_IRQDISABLE, > - keypad_data->base + OMAP4_KBD_IRQSTATUS); > - __raw_writel(OMAP4_DEF_IRQENABLE_EVENTEN | OMAP4_DEF_IRQENABLE_LONGKEY, > - keypad_data->base + OMAP4_KBD_IRQENABLE); > - __raw_writel(OMAP4_DEF_WUP_EVENT_ENA | OMAP4_DEF_WUP_LONG_KEY_ENA, > - keypad_data->base + OMAP4_KBD_WAKEUPENABLE); > + kbd_writel(keypad_data, OMAP4_KBD_CTRL, > + OMAP4_VAL_FUNCTIONALCFG); > + kbd_writel(keypad_data, OMAP4_KBD_DEBOUNCINGTIME, > + OMAP4_VAL_DEBOUNCINGTIME); > + kbd_write_irqreg(keypad_data, OMAP4_KBD_IRQSTATUS, > + OMAP4_VAL_IRQDISABLE); > + kbd_write_irqreg(keypad_data, OMAP4_KBD_IRQENABLE, > + OMAP4_DEF_IRQENABLE_EVENTEN | > + OMAP4_DEF_IRQENABLE_LONGKEY); > + kbd_writel(keypad_data, OMAP4_KBD_WAKEUPENABLE, > + OMAP4_DEF_WUP_EVENT_ENA | OMAP4_DEF_WUP_LONG_KEY_ENA); > > enable_irq(keypad_data->irq); > > @@ -162,12 +196,12 @@ static void omap4_keypad_close(struct input_dev *input) > disable_irq(keypad_data->irq); > > /* Disable interrupts */ > - __raw_writel(OMAP4_VAL_IRQDISABLE, > - keypad_data->base + OMAP4_KBD_IRQENABLE); > + kbd_write_irqreg(keypad_data, OMAP4_KBD_IRQENABLE, > + OMAP4_VAL_IRQDISABLE); > > /* clear pending interrupts */ > - __raw_writel(__raw_readl(keypad_data->base + OMAP4_KBD_IRQSTATUS), > - keypad_data->base + OMAP4_KBD_IRQSTATUS); > + kbd_write_irqreg(keypad_data, OMAP4_KBD_IRQSTATUS, > + kbd_read_irqreg(keypad_data, OMAP4_KBD_IRQSTATUS)); > > enable_irq(keypad_data->irq); > > @@ -182,6 +216,7 @@ static int __devinit omap4_keypad_probe(struct platform_device *pdev) > struct resource *res; > resource_size_t size; > unsigned int row_shift, max_keys; > + int rev; > int irq; > int error; > > @@ -241,6 +276,25 @@ static int __devinit omap4_keypad_probe(struct platform_device *pdev) > keypad_data->rows = pdata->rows; > keypad_data->cols = pdata->cols; > > + rev = __raw_readl(keypad_data->base + OMAP4_KBD_REVISION); > + rev &= 0x03 << 30; > + rev >>= 30; > + switch (rev) { > + case KBD_REVISION_OMAP4: > + keypad_data->reg_offset = 0x00; > + keypad_data->irqreg_offset = 0x00; > + break; > + case KBD_REVISION_OMAP5: > + keypad_data->reg_offset = 0x10; > + keypad_data->irqreg_offset = 0x0c; > + break; > + default: > + dev_err(&pdev->dev, > + "Keypad reports unsupported revision %d", rev); > + error = -EINVAL; > + goto err_unmap; > + } > + > /* input device allocation */ > keypad_data->input = input_dev = input_allocate_device(); > if (!input_dev) { -- 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