Re: Git tree updated to v2.6.26-rc3, all header files moved

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



* 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, &reg, KEYP_ISR1, 1);
> +	ret = twl4030_kpread(kp, TWL4030_MODULE_KEYPAD, &reg, 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

[Index of Archives]     [Linux Arm (vger)]     [ARM Kernel]     [ARM MSM]     [Linux Tegra]     [Linux WPAN Networking]     [Linux Wireless Networking]     [Maemo Users]     [Linux USB Devel]     [Video for Linux]     [Linux Audio Users]     [Yosemite Trails]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux