-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathvmouse.c
111 lines (89 loc) · 2.62 KB
/
vmouse.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/input.h>
#include <linux/input/mt.h>
#include <linux/spinlock.h>
#include <linux/device.h>
#include <linux/slab.h>
#include "vinput.h"
#define VINPUT_MTS "vmouse"
static int vinput_vmouse_init(struct vinput *vinput)
{
int *buttons = kmalloc(sizeof(int), GFP_KERNEL);
__set_bit(EV_REL, vinput->input->evbit);
__set_bit(REL_X, vinput->input->relbit);
__set_bit(REL_Y, vinput->input->relbit);
__set_bit(REL_WHEEL, vinput->input->relbit);
__set_bit(EV_KEY, vinput->input->evbit);
__set_bit(BTN_LEFT, vinput->input->keybit);
__set_bit(BTN_RIGHT, vinput->input->keybit);
__set_bit(BTN_MIDDLE, vinput->input->keybit);
*buttons = 0;
vinput->priv_data = buttons;
return input_register_device(vinput->input);
}
static int vinput_vmouse_kill(struct vinput *vinput)
{
int *buttons = vinput->priv_data;
kfree(buttons);
return 0;
}
static int vinput_vmouse_read(struct vinput *vinput, char *buff, int len)
{
return len;
}
#define VBUTTON_LEFT 0
#define VBUTTON_RIGHT 1
#define VBUTTON_MIDDLE 2
static int vinput_vmouse_send(struct vinput *vinput, char *buff, int len)
{
int ret;
int x, y, wheel;
int buttons;
int *state = vinput->priv_data;
ret = sscanf(buff, "%d,%d,%d,%d", &x, &y, &wheel, &buttons);
if (ret != 4) {
dev_warn(&vinput->dev, "Invalid input format: x,y,wheel,buttons\n");
len = -EINVAL;
} else {
if (x)
input_report_rel(vinput->input, REL_X, x);
if (y)
input_report_rel(vinput->input, REL_Y, y);
if (wheel)
input_report_rel(vinput->input, REL_WHEEL, wheel);
if ((*state | buttons) & (0x1 << VBUTTON_LEFT))
input_report_key(vinput->input, BTN_LEFT, 1 & (buttons >> VBUTTON_LEFT));
else if ((*state | buttons) & (0x1 << VBUTTON_RIGHT))
input_report_key(vinput->input, BTN_RIGHT, 1 & (buttons >> VBUTTON_RIGHT));
else if ((*state | buttons) & (0x1 << VBUTTON_MIDDLE))
input_report_key(vinput->input, BTN_MIDDLE, 1 & (buttons >> VBUTTON_MIDDLE));
*state = buttons;
input_sync(vinput->input);
}
return len;
}
static struct vinput_ops vmouse_ops = {
.init = vinput_vmouse_init,
.kill = vinput_vmouse_kill,
.send = vinput_vmouse_send,
.read = vinput_vmouse_read,
};
static struct vinput_device vmouse_dev = {
.name = VINPUT_MTS,
.ops = &vmouse_ops,
};
static int __init vmouse_init(void)
{
return vinput_register(&vmouse_dev);
}
static void __exit vmouse_end(void)
{
vinput_unregister(&vmouse_dev);
}
module_init(vmouse_init);
module_exit(vmouse_end);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("Tristan Lelong <[email protected]>");
MODULE_DESCRIPTION("emulate mouse input events thru /dev/vinput");