-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathkps2_driver.c
106 lines (94 loc) · 1.93 KB
/
kps2_driver.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
#include "kps_driver.h"
MODULE_LICENSE("GPL");
MODULE_AUTHOR("Lenart");
MODULE_DESCRIPTION("A simple kps2 module");
MODULE_VERSION("0.01");
static int irq = 1;
static unsigned int cookie = 0xdeadd00d;
Modifiers gmodifiers = None;
struct kps_data kps_data = {
.lock = __MUTEX_INITIALIZER(kps_data.lock),
.entries = LIST_HEAD_INIT(kps_data.entries),
.aLen = 0,
};
static void check_for_modifiers(unsigned char scancode)
{
switch (scancode) {
// presses
case 0x2A:
gmodifiers |= LShift;
break;
case 0x3A:
gmodifiers |= Capslock;
break;
case 0x1D:
gmodifiers |= LCtrl;
break;
case 0x38:
gmodifiers |= LAlt;
break;
case 0x36:
gmodifiers |= RShift;
break;
// releases
case 0xAA:
gmodifiers &= ~LShift;
break;
case 0xBA:
gmodifiers &= ~Capslock;
break;
case 0x9D:
gmodifiers &= ~LCtrl;
break;
case 0xB8:
gmodifiers &= ~LAlt;
break;
case 0xB6:
gmodifiers &= ~RShift;
break;
default:
break;
}
}
static irqreturn_t irq_handler(int ir, void* dev_id)
{
static unsigned char scancode;
scancode = inb(0x60);
check_for_modifiers(scancode);
if ((scancode >= 0x01 && scancode <= 0x5d)
|| (scancode >= 0x81 && scancode <= 0xDD))
{
if (new_node(scancode))
goto FREE_FATAL;
}
return IRQ_HANDLED;
FREE_FATAL:
synchronize_irq(irq);
free_irq(irq, &cookie);
deregister_misc_device();
return -1;
}
static int __init keyboard_ps2(void)
{
int ret;
printk(KERN_INFO "%s: init...\n", __func__);
ret = register_misc_device();
if (ret)
return ret;
ret = request_irq(irq, irq_handler, IRQF_SHARED, "Keyboard key", &cookie);
if (ret) {
pr_err("%s: failed to request irq %d\n", __func__, ret);
deregister_misc_device();
return ret;
}
return ret;
}
static void __exit mod_exit(void)
{
printk(KERN_INFO "%s: cleaning up...\n", __func__);
synchronize_irq(irq);
free_irq(irq, &cookie);
deregister_misc_device();
}
module_init(keyboard_ps2);
module_exit(mod_exit);