From: Hans-Christian Egtvedt <[EMAIL PROTECTED]>

This patch adds support for simulating a mouse using GPIO lines.

The driver needs a platform_data struct to be defined and registered with the
appropriate platform_device.

The driver has been tested on AT32AP7000 microprocessor using the ATSTK1000
development board.

Signed-off-by: Hans-Christian Egtvedt <[EMAIL PROTECTED]>
---
 drivers/input/mouse/Kconfig      |   15 ++
 drivers/input/mouse/Makefile     |    1 +
 drivers/input/mouse/gpio_mouse.c |  276 ++++++++++++++++++++++++++++++++++++++
 include/linux/gpio_mouse.h       |   53 +++++++
 4 files changed, 345 insertions(+), 0 deletions(-)

diff --git a/drivers/input/mouse/Kconfig b/drivers/input/mouse/Kconfig
index 35d998c..a3c7057 100644
--- a/drivers/input/mouse/Kconfig
+++ b/drivers/input/mouse/Kconfig
@@ -125,4 +125,19 @@ config MOUSE_HIL
        help
          Say Y here to support HIL pointers.
 
+config MOUSE_GPIO
+       tristate "GPIO mouse"
+       depends on GENERIC_GPIO
+       help
+         This driver simulates a mouse on GPIO lines of various CPUs (and some
+         other chips).
+
+         Say Y here if your device has buttons or a simple joystick connected
+         directly to GPIO lines. Your board-specific setup logic must also
+         provide a platform device and platform data saying which GPIOs are
+         used.
+
+         To compile this driver as a module, choose M here: the
+         module will be called gpio_mouse.
+
 endif
diff --git a/drivers/input/mouse/Makefile b/drivers/input/mouse/Makefile
index 21a1de6..5d0fc43 100644
--- a/drivers/input/mouse/Makefile
+++ b/drivers/input/mouse/Makefile
@@ -13,5 +13,6 @@ obj-$(CONFIG_MOUSE_PS2)               += psmouse.o
 obj-$(CONFIG_MOUSE_SERIAL)     += sermouse.o
 obj-$(CONFIG_MOUSE_HIL)                += hil_ptr.o
 obj-$(CONFIG_MOUSE_VSXXXAA)    += vsxxxaa.o
+obj-$(CONFIG_MOUSE_GPIO)       += gpio_mouse.o
 
 psmouse-objs  := psmouse-base.o alps.o logips2pp.o synaptics.o lifebook.o 
trackpoint.o
diff --git a/drivers/input/mouse/gpio_mouse.c b/drivers/input/mouse/gpio_mouse.c
new file mode 100644
index 0000000..3f36b73
--- /dev/null
+++ b/drivers/input/mouse/gpio_mouse.c
@@ -0,0 +1,276 @@
+/*
+ * Driver for simulating a mouse on GPIO lines.
+ *
+ * Copyright (C) 2007 Atmel Corporation
+ *
+ * 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
+ * published by the Free Software Foundation.
+ */
+
+/*#define DEBUG*/
+
+#include <linux/init.h>
+#include <linux/version.h>
+#include <linux/module.h>
+#include <linux/timer.h>
+#include <linux/jiffies.h>
+#include <linux/platform_device.h>
+#include <linux/input.h>
+#include <linux/gpio_mouse.h>
+
+#include <asm/gpio.h>
+
+/*
+ * Timer function which is run every scan_ms ms when the device is opened. The
+ * dev input varaible is set to the the input_dev pointer.
+ */
+static void gpio_mouse_scan(unsigned long _dev)
+{
+       struct input_dev *input = (struct input_dev *)_dev;
+       struct gpio_mouse_platform_data *gpio = input->private;
+       int x, y;
+
+       if (gpio->bleft >= 0)
+               input_report_key(input, BTN_LEFT,
+                               gpio_get_value(gpio->bleft) ^ gpio->polarity);
+       if (gpio->bmiddle >= 0)
+               input_report_key(input, BTN_MIDDLE,
+                               gpio_get_value(gpio->bmiddle) ^ gpio->polarity);
+       if (gpio->bright >= 0)
+               input_report_key(input, BTN_RIGHT,
+                               gpio_get_value(gpio->bright) ^ gpio->polarity);
+
+       x = (gpio_get_value(gpio->right) ^ gpio->polarity)
+               - (gpio_get_value(gpio->left) ^ gpio->polarity);
+       y = (gpio_get_value(gpio->down) ^ gpio->polarity)
+               - (gpio_get_value(gpio->up) ^ gpio->polarity);
+
+       input_report_rel(input, REL_X, x);
+       input_report_rel(input, REL_Y, y);
+       input_sync(input);
+
+       mod_timer(&gpio->timer, jiffies + msecs_to_jiffies(gpio->scan_ms));
+}
+
+/* Only start the timer when the device is actually in use, i.e. opened */
+static int gpio_mouse_open(struct input_dev *input)
+{
+       struct gpio_mouse_platform_data *gpio = input->private;
+       gpio->timer.expires = jiffies + msecs_to_jiffies(gpio->scan_ms);
+       gpio->timer.data = (unsigned long)input;
+       add_timer(&gpio->timer);
+       return 0;
+}
+
+static void gpio_mouse_close(struct input_dev *input)
+{
+       struct gpio_mouse_platform_data *gpio = input->private;
+       del_timer_sync(&gpio->timer);
+}
+
+static int __init gpio_mouse_probe(struct platform_device *pdev)
+{
+       struct gpio_mouse_platform_data *pdata = pdev->dev.platform_data;
+       struct input_dev *input;
+       int ret;
+
+       if (!pdata) {
+               dev_dbg(&pdev->dev, "no platform data\n");
+               ret = -ENXIO;
+               goto out;
+       }
+
+       if (pdata->up < 0 || pdata->down < 0 || pdata->right < 0
+                       || pdata->left < 0) {
+               dev_dbg(&pdev->dev, "missing GPIO for directions\n");
+               ret = -EINVAL;
+               goto out_gpio;
+       }
+       if (pdata->scan_ms < 0) {
+               dev_dbg(&pdev->dev, "invalid scan time\n");
+               ret = -EINVAL;
+               goto out_gpio;
+       }
+
+       /* Mouse direction, required */
+       ret = gpio_request(pdata->up, "gpio_mouse_up");
+       if (ret) {
+               dev_dbg(&pdev->dev, "fail up pin\n");
+               goto out_gpio;
+       }
+       ret = gpio_request(pdata->down, "gpio_mouse_down");
+       if (ret) {
+               dev_dbg(&pdev->dev, "fail down pin\n");
+               goto out_gpio;
+       }
+       ret = gpio_request(pdata->left, "gpio_mouse_left");
+       if (ret) {
+               dev_dbg(&pdev->dev, "fail left pin\n");
+               goto out_gpio;
+       }
+       ret = gpio_request(pdata->right, "gpio_mouse_right");
+       if (ret) {
+               dev_dbg(&pdev->dev, "fail right pin\n");
+               goto out_gpio;
+       }
+
+       gpio_direction_input(pdata->up);
+       gpio_direction_input(pdata->down);
+       gpio_direction_input(pdata->left);
+       gpio_direction_input(pdata->right);
+
+       /* Mouse buttons, not required, but should at least have bleft */
+       if (pdata->bleft >= 0) {
+               ret = gpio_request(pdata->bleft, "gpio_mouse_bleft");
+               if (ret) {
+                       dev_dbg(&pdev->dev, "fail bleft pin\n");
+                       goto out_gpio;
+               }
+               gpio_direction_input(pdata->bleft);
+       } else {
+               dev_dbg(&pdev->dev, "no left button defined\n");
+       }
+       if (pdata->bmiddle >= 0) {
+               ret = gpio_request(pdata->bmiddle, "gpio_mouse_bmiddle");
+               if (ret) {
+                       dev_dbg(&pdev->dev, "fail bmiddle pin\n");
+                       goto out_gpio;
+               }
+               gpio_direction_input(pdata->bmiddle);
+       }
+       if (pdata->bright >= 0) {
+               ret = gpio_request(pdata->bright, "gpio_mouse_bright");
+               if (ret) {
+                       dev_dbg(&pdev->dev, "fail bright pin\n");
+                       goto out_gpio;
+               }
+               gpio_direction_input(pdata->bright);
+       }
+
+       input = input_allocate_device();
+       if (!input) {
+               dev_dbg(&pdev->dev, "not enough memory for input device\n");
+               ret = -ENOMEM;
+               goto out_gpio;
+       }
+
+       platform_set_drvdata(pdev, input);
+
+       input->name = pdev->name;
+       input->cdev.dev = &pdev->dev;
+       input->private = pdata;
+
+       /*
+        * Revisit: is bustype, vendor, product and version needed to
+        * input->id? And if they should be present, what values should they
+        * have?
+        */
+
+       input->evbit[0] = BIT(EV_REL);
+       if (pdata->bleft >= 0 || pdata->bmiddle >= 0 || pdata->bright >= 0) {
+               input->evbit[0] |= BIT(EV_KEY);
+
+               if (pdata->bleft >= 0)
+                       input->keybit[LONG(BTN_MOUSE)] = BIT(BTN_LEFT);
+               if (pdata->bmiddle >= 0)
+                       input->keybit[LONG(BTN_MOUSE)] |= BIT(BTN_MIDDLE);
+               if (pdata->bright >= 0)
+                       input->keybit[LONG(BTN_MOUSE)] |= BIT(BTN_RIGHT);
+       }
+       input->relbit[0] = BIT(REL_X) | BIT(REL_Y);
+
+       input->open = gpio_mouse_open;
+       input->close = gpio_mouse_close;
+
+       /* init the scan timer */
+       pdata->timer.function = gpio_mouse_scan;
+       pdata->timer.base = &boot_tvec_bases;
+
+       ret = input_register_device(input);
+       if (ret) {
+               dev_dbg(&pdev->dev, "could not register input device\n");
+               goto out_reg_dev;
+       }
+
+       dev_dbg(&pdev->dev, "%d ms scan time, buttons: %s%s%s\n",
+                       pdata->scan_ms,
+                       pdata->bleft < 0 ? "" : "left ",
+                       pdata->bmiddle < 0 ? "" : "middle ",
+                       pdata->bright < 0 ? "" : "right");
+
+       return 0;
+
+out_reg_dev:
+       input_free_device(input);
+       platform_set_drvdata(pdev, NULL);
+out_gpio:
+       if (pdata->up >= 0)
+               gpio_free(pdata->up);
+       if (pdata->down >= 0)
+               gpio_free(pdata->down);
+       if (pdata->left >= 0)
+               gpio_free(pdata->left);
+       if (pdata->right >= 0)
+               gpio_free(pdata->right);
+       if (pdata->bleft >= 0)
+               gpio_free(pdata->bleft);
+       if (pdata->bmiddle >= 0)
+               gpio_free(pdata->bmiddle);
+       if (pdata->bright >= 0)
+               gpio_free(pdata->bright);
+out:
+       return ret;
+}
+
+static int __exit gpio_mouse_remove(struct platform_device *pdev)
+{
+       struct gpio_mouse_platform_data *pdata = pdev->dev.platform_data;
+       struct input_dev *input = platform_get_drvdata(pdev);
+
+       if (input)
+               input_unregister_device(input);
+
+       if (pdata->up >= 0)
+               gpio_free(pdata->up);
+       if (pdata->down >= 0)
+               gpio_free(pdata->down);
+       if (pdata->left >= 0)
+               gpio_free(pdata->left);
+       if (pdata->right >= 0)
+               gpio_free(pdata->right);
+       if (pdata->bleft >= 0)
+               gpio_free(pdata->bleft);
+       if (pdata->bmiddle >= 0)
+               gpio_free(pdata->bmiddle);
+       if (pdata->bright >= 0)
+               gpio_free(pdata->bright);
+
+       platform_set_drvdata(pdev, NULL);
+
+       return 0;
+}
+
+struct platform_driver gpio_mouse_device_driver = {
+       .remove         = __exit_p(gpio_mouse_remove),
+       .driver         = {
+               .name   = "gpio_mouse",
+       }
+};
+
+static int __init gpio_mouse_init(void)
+{
+       return platform_driver_probe(&gpio_mouse_device_driver,
+                       gpio_mouse_probe);
+}
+module_init(gpio_mouse_init);
+
+static void __exit gpio_mouse_exit(void)
+{
+       platform_driver_unregister(&gpio_mouse_device_driver);
+}
+module_exit(gpio_mouse_exit);
+
+MODULE_AUTHOR("Hans-Christian Egtvedt <[EMAIL PROTECTED]>");
+MODULE_DESCRIPTION("GPIO mouse driver");
+MODULE_LICENSE("GPL");
diff --git a/include/linux/gpio_mouse.h b/include/linux/gpio_mouse.h
new file mode 100644
index 0000000..b63d43f
--- /dev/null
+++ b/include/linux/gpio_mouse.h
@@ -0,0 +1,53 @@
+/*
+ * Driver for simulating a mouse on GPIO lines.
+ *
+ * Copyright (C) 2007 Atmel Corporation
+ *
+ * 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
+ * published by the Free Software Foundation.
+ */
+
+#ifndef _GPIO_MOUSE_H
+#define _GPIO_MOUSE_H
+
+#include <linux/timer.h>
+
+#define GPIO_MOUSE_POLARITY_ACT_HIGH   0x00
+#define GPIO_MOUSE_POLARITY_ACT_LOW    0x01
+
+/**
+ * struct GPIO_mouse_platform_data
+ * @scan_ms: integer in ms specifying the scan periode.
+ * @polarity: Pin polarity, active high or low.
+ * @up: GPIO line for up value.
+ * @down: GPIO line for down value.
+ * @left: GPIO line for left value.
+ * @right: GPIO line for right value.
+ * @bleft: GPIO line for left button.
+ * @bright: GPIO line for right button.
+ * @bmiddle: GPIO line for middle button.
+ * @timer: placeholder for struct timer_list initialized in the driver.
+ *
+ * This struct must be added to the platform_device in the board code. It is
+ * used by the gpio_mouse driver to setup GPIO lines, calculate mouse movement
+ * and have a reference to the timer used for scanning.
+ */
+struct gpio_mouse_platform_data {
+       int scan_ms;
+       int polarity;
+
+       int up;
+       int down;
+       int left;
+       int right;
+
+       int bleft;
+       int bmiddle;
+       int bright;
+
+       /* private */
+       struct timer_list timer;
+};
+
+#endif /* _GPIO_MOUSE_H */
-- 
1.4.4.2

Reply via email to