2 * Driver for simulating a mouse on GPIO lines.
4 * Copyright (C) 2007 Atmel Corporation
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 as
8 * published by the Free Software Foundation.
11 #include <linux/init.h>
12 #include <linux/module.h>
13 #include <linux/platform_device.h>
14 #include <linux/input-polldev.h>
15 #include <linux/gpio.h>
16 #include <linux/gpio_mouse.h>
20 * Timer function which is run every scan_ms ms when the device is opened.
21 * The dev input variable is set to the the input_dev pointer.
23 static void gpio_mouse_scan(struct input_polled_dev
*dev
)
25 struct gpio_mouse_platform_data
*gpio
= dev
->private;
26 struct input_dev
*input
= dev
->input
;
30 input_report_key(input
, BTN_LEFT
,
31 gpio_get_value(gpio
->bleft
) ^ gpio
->polarity
);
32 if (gpio
->bmiddle
>= 0)
33 input_report_key(input
, BTN_MIDDLE
,
34 gpio_get_value(gpio
->bmiddle
) ^ gpio
->polarity
);
35 if (gpio
->bright
>= 0)
36 input_report_key(input
, BTN_RIGHT
,
37 gpio_get_value(gpio
->bright
) ^ gpio
->polarity
);
39 x
= (gpio_get_value(gpio
->right
) ^ gpio
->polarity
)
40 - (gpio_get_value(gpio
->left
) ^ gpio
->polarity
);
41 y
= (gpio_get_value(gpio
->down
) ^ gpio
->polarity
)
42 - (gpio_get_value(gpio
->up
) ^ gpio
->polarity
);
44 input_report_rel(input
, REL_X
, x
);
45 input_report_rel(input
, REL_Y
, y
);
49 static int gpio_mouse_probe(struct platform_device
*pdev
)
51 struct gpio_mouse_platform_data
*pdata
= pdev
->dev
.platform_data
;
52 struct input_polled_dev
*input_poll
;
53 struct input_dev
*input
;
58 dev_err(&pdev
->dev
, "no platform data\n");
63 if (pdata
->scan_ms
< 0) {
64 dev_err(&pdev
->dev
, "invalid scan time\n");
69 for (i
= 0; i
< GPIO_MOUSE_PIN_MAX
; i
++) {
74 if (i
<= GPIO_MOUSE_PIN_RIGHT
) {
75 /* Mouse direction is required. */
77 "missing GPIO for directions\n");
82 if (i
== GPIO_MOUSE_PIN_BLEFT
)
83 dev_dbg(&pdev
->dev
, "no left button defined\n");
86 error
= gpio_request(pin
, "gpio_mouse");
88 dev_err(&pdev
->dev
, "fail %d pin (%d idx)\n",
93 gpio_direction_input(pin
);
97 input_poll
= input_allocate_polled_device();
99 dev_err(&pdev
->dev
, "not enough memory for input device\n");
104 platform_set_drvdata(pdev
, input_poll
);
106 /* set input-polldev handlers */
107 input_poll
->private = pdata
;
108 input_poll
->poll
= gpio_mouse_scan
;
109 input_poll
->poll_interval
= pdata
->scan_ms
;
111 input
= input_poll
->input
;
112 input
->name
= pdev
->name
;
113 input
->id
.bustype
= BUS_HOST
;
114 input
->dev
.parent
= &pdev
->dev
;
116 input_set_capability(input
, EV_REL
, REL_X
);
117 input_set_capability(input
, EV_REL
, REL_Y
);
118 if (pdata
->bleft
>= 0)
119 input_set_capability(input
, EV_KEY
, BTN_LEFT
);
120 if (pdata
->bmiddle
>= 0)
121 input_set_capability(input
, EV_KEY
, BTN_MIDDLE
);
122 if (pdata
->bright
>= 0)
123 input_set_capability(input
, EV_KEY
, BTN_RIGHT
);
125 error
= input_register_polled_device(input_poll
);
127 dev_err(&pdev
->dev
, "could not register input device\n");
128 goto out_free_polldev
;
131 dev_dbg(&pdev
->dev
, "%d ms scan time, buttons: %s%s%s\n",
133 pdata
->bleft
< 0 ? "" : "left ",
134 pdata
->bmiddle
< 0 ? "" : "middle ",
135 pdata
->bright
< 0 ? "" : "right");
140 input_free_polled_device(input_poll
);
144 pin
= pdata
->pins
[i
];
152 static int gpio_mouse_remove(struct platform_device
*pdev
)
154 struct input_polled_dev
*input
= platform_get_drvdata(pdev
);
155 struct gpio_mouse_platform_data
*pdata
= input
->private;
158 input_unregister_polled_device(input
);
159 input_free_polled_device(input
);
161 for (i
= 0; i
< GPIO_MOUSE_PIN_MAX
; i
++) {
162 pin
= pdata
->pins
[i
];
170 static struct platform_driver gpio_mouse_device_driver
= {
171 .probe
= gpio_mouse_probe
,
172 .remove
= gpio_mouse_remove
,
174 .name
= "gpio_mouse",
175 .owner
= THIS_MODULE
,
178 module_platform_driver(gpio_mouse_device_driver
);
180 MODULE_AUTHOR("Hans-Christian Egtvedt <egtvedt@samfundet.no>");
181 MODULE_DESCRIPTION("GPIO mouse driver");
182 MODULE_LICENSE("GPL");
183 MODULE_ALIAS("platform:gpio_mouse"); /* work with hotplug and coldplug */