diff options
Diffstat (limited to 'drivers/input/misc/rotary_encoder.c')
| -rw-r--r-- | drivers/input/misc/rotary_encoder.c | 280 | 
1 files changed, 180 insertions, 100 deletions
diff --git a/drivers/input/misc/rotary_encoder.c b/drivers/input/misc/rotary_encoder.c index 1f8e0108962..93558a1c7f7 100644 --- a/drivers/input/misc/rotary_encoder.c +++ b/drivers/input/misc/rotary_encoder.c @@ -2,11 +2,12 @@   * rotary_encoder.c   *   * (c) 2009 Daniel Mack <daniel@caiaq.de> + * Copyright (C) 2011 Johan Hovold <jhovold@gmail.com>   *   * state machine code inspired by code from Tim Ruetz   *   * A generic driver for rotary encoders connected to GPIO lines. - * See file:Documentation/input/rotary_encoder.txt for more information + * See file:Documentation/input/rotary-encoder.txt for more information   *   * This program is free software; you can redistribute it and/or modify   * it under the terms of the GNU General Public License version 2 as @@ -15,7 +16,6 @@  #include <linux/kernel.h>  #include <linux/module.h> -#include <linux/init.h>  #include <linux/interrupt.h>  #include <linux/input.h>  #include <linux/device.h> @@ -23,12 +23,15 @@  #include <linux/gpio.h>  #include <linux/rotary_encoder.h>  #include <linux/slab.h> +#include <linux/of.h> +#include <linux/of_platform.h> +#include <linux/of_gpio.h>  #define DRV_NAME "rotary-encoder"  struct rotary_encoder {  	struct input_dev *input; -	struct rotary_encoder_platform_data *pdata; +	const struct rotary_encoder_platform_data *pdata;  	unsigned int axis;  	unsigned int pos; @@ -38,52 +41,66 @@ struct rotary_encoder {  	bool armed;  	unsigned char dir;	/* 0 - clockwise, 1 - CCW */ + +	char last_stable;  }; -static irqreturn_t rotary_encoder_irq(int irq, void *dev_id) +static int rotary_encoder_get_state(const struct rotary_encoder_platform_data *pdata)  { -	struct rotary_encoder *encoder = dev_id; -	struct rotary_encoder_platform_data *pdata = encoder->pdata;  	int a = !!gpio_get_value(pdata->gpio_a);  	int b = !!gpio_get_value(pdata->gpio_b); -	int state;  	a ^= pdata->inverted_a;  	b ^= pdata->inverted_b; -	state = (a << 1) | b; -	switch (state) { +	return ((a << 1) | b); +} -	case 0x0: -		if (!encoder->armed) -			break; +static void rotary_encoder_report_event(struct rotary_encoder *encoder) +{ +	const struct rotary_encoder_platform_data *pdata = encoder->pdata; -		if (pdata->relative_axis) { -			input_report_rel(encoder->input, pdata->axis, -					 encoder->dir ? -1 : 1); -		} else { -			unsigned int pos = encoder->pos; - -			if (encoder->dir) { -				/* turning counter-clockwise */ -				if (pdata->rollover) -					pos += pdata->steps; -				if (pos) -					pos--; -			} else { -				/* turning clockwise */ -				if (pdata->rollover || pos < pdata->steps) -					pos++; -			} +	if (pdata->relative_axis) { +		input_report_rel(encoder->input, +				 pdata->axis, encoder->dir ? -1 : 1); +	} else { +		unsigned int pos = encoder->pos; + +		if (encoder->dir) { +			/* turning counter-clockwise */  			if (pdata->rollover) -				pos %= pdata->steps; -			encoder->pos = pos; -			input_report_abs(encoder->input, pdata->axis, -					 encoder->pos); +				pos += pdata->steps; +			if (pos) +				pos--; +		} else { +			/* turning clockwise */ +			if (pdata->rollover || pos < pdata->steps) +				pos++;  		} -		input_sync(encoder->input); -		encoder->armed = false; +		if (pdata->rollover) +			pos %= pdata->steps; + +		encoder->pos = pos; +		input_report_abs(encoder->input, pdata->axis, encoder->pos); +	} + +	input_sync(encoder->input); +} + +static irqreturn_t rotary_encoder_irq(int irq, void *dev_id) +{ +	struct rotary_encoder *encoder = dev_id; +	int state; + +	state = rotary_encoder_get_state(encoder->pdata); + +	switch (state) { +	case 0x0: +		if (encoder->armed) { +			rotary_encoder_report_event(encoder); +			encoder->armed = false; +		}  		break;  	case 0x1: @@ -100,35 +117,113 @@ static irqreturn_t rotary_encoder_irq(int irq, void *dev_id)  	return IRQ_HANDLED;  } -static int __devinit rotary_encoder_probe(struct platform_device *pdev) +static irqreturn_t rotary_encoder_half_period_irq(int irq, void *dev_id) +{ +	struct rotary_encoder *encoder = dev_id; +	int state; + +	state = rotary_encoder_get_state(encoder->pdata); + +	switch (state) { +	case 0x00: +	case 0x03: +		if (state != encoder->last_stable) { +			rotary_encoder_report_event(encoder); +			encoder->last_stable = state; +		} +		break; + +	case 0x01: +	case 0x02: +		encoder->dir = (encoder->last_stable + state) & 0x01; +		break; +	} + +	return IRQ_HANDLED; +} + +#ifdef CONFIG_OF +static const struct of_device_id rotary_encoder_of_match[] = { +	{ .compatible = "rotary-encoder", }, +	{ }, +}; +MODULE_DEVICE_TABLE(of, rotary_encoder_of_match); + +static struct rotary_encoder_platform_data *rotary_encoder_parse_dt(struct device *dev)  { -	struct rotary_encoder_platform_data *pdata = pdev->dev.platform_data; +	const struct of_device_id *of_id = +				of_match_device(rotary_encoder_of_match, dev); +	struct device_node *np = dev->of_node; +	struct rotary_encoder_platform_data *pdata; +	enum of_gpio_flags flags; + +	if (!of_id || !np) +		return NULL; + +	pdata = kzalloc(sizeof(struct rotary_encoder_platform_data), +			GFP_KERNEL); +	if (!pdata) +		return ERR_PTR(-ENOMEM); + +	of_property_read_u32(np, "rotary-encoder,steps", &pdata->steps); +	of_property_read_u32(np, "linux,axis", &pdata->axis); + +	pdata->gpio_a = of_get_gpio_flags(np, 0, &flags); +	pdata->inverted_a = flags & OF_GPIO_ACTIVE_LOW; + +	pdata->gpio_b = of_get_gpio_flags(np, 1, &flags); +	pdata->inverted_b = flags & OF_GPIO_ACTIVE_LOW; + +	pdata->relative_axis = !!of_get_property(np, +					"rotary-encoder,relative-axis", NULL); +	pdata->rollover = !!of_get_property(np, +					"rotary-encoder,rollover", NULL); +	pdata->half_period = !!of_get_property(np, +					"rotary-encoder,half-period", NULL); + +	return pdata; +} +#else +static inline struct rotary_encoder_platform_data * +rotary_encoder_parse_dt(struct device *dev) +{ +	return NULL; +} +#endif + +static int rotary_encoder_probe(struct platform_device *pdev) +{ +	struct device *dev = &pdev->dev; +	const struct rotary_encoder_platform_data *pdata = dev_get_platdata(dev);  	struct rotary_encoder *encoder;  	struct input_dev *input; +	irq_handler_t handler;  	int err;  	if (!pdata) { -		dev_err(&pdev->dev, "missing platform data\n"); -		return -ENOENT; +		pdata = rotary_encoder_parse_dt(dev); +		if (IS_ERR(pdata)) +			return PTR_ERR(pdata); + +		if (!pdata) { +			dev_err(dev, "missing platform data\n"); +			return -EINVAL; +		}  	}  	encoder = kzalloc(sizeof(struct rotary_encoder), GFP_KERNEL);  	input = input_allocate_device();  	if (!encoder || !input) { -		dev_err(&pdev->dev, "failed to allocate memory for device\n");  		err = -ENOMEM;  		goto exit_free_mem;  	}  	encoder->input = input;  	encoder->pdata = pdata; -	encoder->irq_a = gpio_to_irq(pdata->gpio_a); -	encoder->irq_b = gpio_to_irq(pdata->gpio_b); -	/* create and register the input driver */  	input->name = pdev->name;  	input->id.bustype = BUS_HOST; -	input->dev.parent = &pdev->dev; +	input->dev.parent = dev;  	if (pdata->relative_axis) {  		input->evbit[0] = BIT_MASK(EV_REL); @@ -139,119 +234,104 @@ static int __devinit rotary_encoder_probe(struct platform_device *pdev)  				     pdata->axis, 0, pdata->steps, 0, 1);  	} -	err = input_register_device(input); -	if (err) { -		dev_err(&pdev->dev, "failed to register input device\n"); -		goto exit_free_mem; -	} -  	/* request the GPIOs */ -	err = gpio_request(pdata->gpio_a, DRV_NAME); -	if (err) { -		dev_err(&pdev->dev, "unable to request GPIO %d\n", -			pdata->gpio_a); -		goto exit_unregister_input; -	} - -	err = gpio_direction_input(pdata->gpio_a); +	err = gpio_request_one(pdata->gpio_a, GPIOF_IN, dev_name(dev));  	if (err) { -		dev_err(&pdev->dev, "unable to set GPIO %d for input\n", -			pdata->gpio_a); -		goto exit_unregister_input; +		dev_err(dev, "unable to request GPIO %d\n", pdata->gpio_a); +		goto exit_free_mem;  	} -	err = gpio_request(pdata->gpio_b, DRV_NAME); +	err = gpio_request_one(pdata->gpio_b, GPIOF_IN, dev_name(dev));  	if (err) { -		dev_err(&pdev->dev, "unable to request GPIO %d\n", -			pdata->gpio_b); +		dev_err(dev, "unable to request GPIO %d\n", pdata->gpio_b);  		goto exit_free_gpio_a;  	} -	err = gpio_direction_input(pdata->gpio_b); -	if (err) { -		dev_err(&pdev->dev, "unable to set GPIO %d for input\n", -			pdata->gpio_b); -		goto exit_free_gpio_a; -	} +	encoder->irq_a = gpio_to_irq(pdata->gpio_a); +	encoder->irq_b = gpio_to_irq(pdata->gpio_b);  	/* request the IRQs */ -	err = request_irq(encoder->irq_a, &rotary_encoder_irq, -			  IORESOURCE_IRQ_HIGHEDGE | IORESOURCE_IRQ_LOWEDGE, +	if (pdata->half_period) { +		handler = &rotary_encoder_half_period_irq; +		encoder->last_stable = rotary_encoder_get_state(pdata); +	} else { +		handler = &rotary_encoder_irq; +	} + +	err = request_irq(encoder->irq_a, handler, +			  IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING,  			  DRV_NAME, encoder);  	if (err) { -		dev_err(&pdev->dev, "unable to request IRQ %d\n", -			encoder->irq_a); +		dev_err(dev, "unable to request IRQ %d\n", encoder->irq_a);  		goto exit_free_gpio_b;  	} -	err = request_irq(encoder->irq_b, &rotary_encoder_irq, -			  IORESOURCE_IRQ_HIGHEDGE | IORESOURCE_IRQ_LOWEDGE, +	err = request_irq(encoder->irq_b, handler, +			  IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING,  			  DRV_NAME, encoder);  	if (err) { -		dev_err(&pdev->dev, "unable to request IRQ %d\n", -			encoder->irq_b); +		dev_err(dev, "unable to request IRQ %d\n", encoder->irq_b);  		goto exit_free_irq_a;  	} +	err = input_register_device(input); +	if (err) { +		dev_err(dev, "failed to register input device\n"); +		goto exit_free_irq_b; +	} +  	platform_set_drvdata(pdev, encoder);  	return 0; +exit_free_irq_b: +	free_irq(encoder->irq_b, encoder);  exit_free_irq_a:  	free_irq(encoder->irq_a, encoder);  exit_free_gpio_b:  	gpio_free(pdata->gpio_b);  exit_free_gpio_a:  	gpio_free(pdata->gpio_a); -exit_unregister_input: -	input_unregister_device(input); -	input = NULL; /* so we don't try to free it */  exit_free_mem:  	input_free_device(input);  	kfree(encoder); +	if (!dev_get_platdata(&pdev->dev)) +		kfree(pdata); +  	return err;  } -static int __devexit rotary_encoder_remove(struct platform_device *pdev) +static int rotary_encoder_remove(struct platform_device *pdev)  {  	struct rotary_encoder *encoder = platform_get_drvdata(pdev); -	struct rotary_encoder_platform_data *pdata = pdev->dev.platform_data; +	const struct rotary_encoder_platform_data *pdata = encoder->pdata;  	free_irq(encoder->irq_a, encoder);  	free_irq(encoder->irq_b, encoder);  	gpio_free(pdata->gpio_a);  	gpio_free(pdata->gpio_b); +  	input_unregister_device(encoder->input); -	platform_set_drvdata(pdev, NULL);  	kfree(encoder); +	if (!dev_get_platdata(&pdev->dev)) +		kfree(pdata); +  	return 0;  }  static struct platform_driver rotary_encoder_driver = {  	.probe		= rotary_encoder_probe, -	.remove		= __devexit_p(rotary_encoder_remove), +	.remove		= rotary_encoder_remove,  	.driver		= {  		.name	= DRV_NAME,  		.owner	= THIS_MODULE, +		.of_match_table = of_match_ptr(rotary_encoder_of_match),  	}  }; - -static int __init rotary_encoder_init(void) -{ -	return platform_driver_register(&rotary_encoder_driver); -} - -static void __exit rotary_encoder_exit(void) -{ -	platform_driver_unregister(&rotary_encoder_driver); -} - -module_init(rotary_encoder_init); -module_exit(rotary_encoder_exit); +module_platform_driver(rotary_encoder_driver);  MODULE_ALIAS("platform:" DRV_NAME);  MODULE_DESCRIPTION("GPIO rotary encoder driver"); -MODULE_AUTHOR("Daniel Mack <daniel@caiaq.de>"); +MODULE_AUTHOR("Daniel Mack <daniel@caiaq.de>, Johan Hovold");  MODULE_LICENSE("GPL v2"); -  | 
