Subversion Repositories shark

Rev

Rev 523 | Go to most recent revision | Details | Last modification | View Log | RSS feed

Rev Author Line No. Line
494 giacomo 1
/*
2
 * $Id: serio.c,v 1.1 2004-03-08 18:47:41 giacomo Exp $
3
 *
4
 *  Copyright (c) 1999-2001 Vojtech Pavlik
5
 */
6
 
7
/*
8
 *  The Serio abstraction module
9
 */
10
 
11
/*
12
 * This program is free software; you can redistribute it and/or modify
13
 * it under the terms of the GNU General Public License as published by
14
 * the Free Software Foundation; either version 2 of the License, or
15
 * (at your option) any later version.
16
 *
17
 * This program is distributed in the hope that it will be useful,
18
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
20
 * GNU General Public License for more details.
21
 *
22
 * You should have received a copy of the GNU General Public License
23
 * along with this program; if not, write to the Free Software
24
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
25
 *
26
 * Should you need to contact me, the author, you can do so either by
27
 * e-mail - mail your message to <vojtech@ucw.cz>, or by paper mail:
28
 * Vojtech Pavlik, Simunkova 1594, Prague 8, 182 00 Czech Republic
29
 *
30
 * Changes:
31
 * 20 Jul. 2003    Daniele Bellucci <bellucda@tiscali.it>
32
 *                 Minor cleanups.
33
 */
34
 
35
#include <linuxcomp.h>
36
 
37
#include <linux/stddef.h>
38
#include <linux/module.h>
39
#include <linux/serio.h>
40
#include <linux/errno.h>
41
#include <linux/wait.h>
42
#include <linux/completion.h>
43
#include <linux/sched.h>
44
#include <linux/smp_lock.h>
45
#include <linux/suspend.h>
46
#include <linux/slab.h>
47
 
48
MODULE_AUTHOR("Vojtech Pavlik <vojtech@ucw.cz>");
49
MODULE_DESCRIPTION("Serio abstraction core");
50
MODULE_LICENSE("GPL");
51
 
52
EXPORT_SYMBOL(serio_interrupt);
53
EXPORT_SYMBOL(serio_register_port);
54
EXPORT_SYMBOL(serio_register_slave_port);
55
EXPORT_SYMBOL(serio_unregister_port);
56
EXPORT_SYMBOL(serio_unregister_slave_port);
57
EXPORT_SYMBOL(serio_register_device);
58
EXPORT_SYMBOL(serio_unregister_device);
59
EXPORT_SYMBOL(serio_open);
60
EXPORT_SYMBOL(serio_close);
61
EXPORT_SYMBOL(serio_rescan);
62
 
63
struct serio_event {
64
        int type;
65
        struct serio *serio;
66
        struct list_head node;
67
};
68
 
69
static DECLARE_MUTEX(serio_sem);
70
static LIST_HEAD(serio_list);
71
static LIST_HEAD(serio_dev_list);
72
static LIST_HEAD(serio_event_list);
73
static int serio_pid;
74
 
75
static void serio_find_dev(struct serio *serio)
76
{
77
        struct serio_dev *dev;
78
 
79
        list_for_each_entry(dev, &serio_dev_list, node) {
80
                if (serio->dev)
81
                        break;
82
                if (dev->connect)
83
                        dev->connect(serio, dev);
84
        }
85
}
86
 
87
#define SERIO_RESCAN    1
88
 
89
static DECLARE_WAIT_QUEUE_HEAD(serio_wait);
90
static DECLARE_COMPLETION(serio_exited);
91
 
92
void serio_handle_events(void)
93
{
94
        struct list_head *node, *next;
95
        struct serio_event *event;
96
 
97
        list_for_each_safe(node, next, &serio_event_list) {
98
                event = container_of(node, struct serio_event, node);  
99
 
100
                switch (event->type) {
101
                        case SERIO_RESCAN :
102
                                //!!!down(&serio_sem);
103
                                if (event->serio->dev && event->serio->dev->disconnect)
104
                                        event->serio->dev->disconnect(event->serio);
105
                                serio_find_dev(event->serio);
106
                                //!!!up(&serio_sem);
107
                                break;
108
                        default:
109
                                break;
110
                }
111
                list_del_init(node);
112
                kfree(event);
113
        }
114
}
115
 
116
static int serio_thread(void *nothing)
117
{
118
        /*lock_kernel();
119
        daemonize("kseriod");
120
        allow_signal(SIGTERM);
121
 
122
        do {
123
                serio_handle_events();
124
                wait_event_interruptible(serio_wait, !list_empty(&serio_event_list));
125
                if (current->flags & PF_FREEZE)
126
                        refrigerator(PF_IOTHREAD);
127
        } while (!signal_pending(current));
128
 
129
        printk(KERN_DEBUG "serio: kseriod exiting\n");
130
 
131
        unlock_kernel();
132
        complete_and_exit(&serio_exited, 0);*/
133
        return 1;
134
}
135
 
136
void serio_rescan(struct serio *serio)
137
{
138
        struct serio_event *event;
139
 
140
        if (!(event = kmalloc(sizeof(struct serio_event), GFP_ATOMIC)))
141
                return;
142
 
143
        event->type = SERIO_RESCAN;
144
        event->serio = serio;
145
 
146
        list_add_tail(&event->node, &serio_event_list);
147
        //!!!wake_up(&serio_wait);
148
}
149
 
150
irqreturn_t serio_interrupt(struct serio *serio,
151
                unsigned char data, unsigned int flags, struct pt_regs *regs)
152
{
153
        irqreturn_t ret = IRQ_NONE;
154
 
155
        if (serio->dev && serio->dev->interrupt) {
156
                ret = serio->dev->interrupt(serio, data, flags, regs);
157
        } else {
158
                if (!flags) {
159
                        serio_rescan(serio);
160
                        ret = IRQ_HANDLED;
161
                }
162
        }
163
        return ret;
164
}
165
 
166
void serio_register_port(struct serio *serio)
167
{
168
        //!!!down(&serio_sem);
169
        list_add_tail(&serio->node, &serio_list);
170
        serio_find_dev(serio);
171
        //!!!up(&serio_sem);
172
}
173
 
174
/*
175
 * Same as serio_register_port but does not try to acquire serio_sem.
176
 * Should be used when registering a serio from other input device's
177
 * connect() function.
178
 */
179
void serio_register_slave_port(struct serio *serio)
180
{
181
        list_add_tail(&serio->node, &serio_list);
182
        serio_find_dev(serio);
183
}
184
 
185
void serio_unregister_port(struct serio *serio)
186
{
187
        //!!!down(&serio_sem);
188
        list_del_init(&serio->node);
189
        if (serio->dev && serio->dev->disconnect)
190
                serio->dev->disconnect(serio);
191
        //!!!up(&serio_sem);
192
}
193
 
194
/*
195
 * Same as serio_unregister_port but does not try to acquire serio_sem.
196
 * Should be used when unregistering a serio from other input device's
197
 * disconnect() function.
198
 */
199
void serio_unregister_slave_port(struct serio *serio)
200
{
201
        list_del_init(&serio->node);
202
        if (serio->dev && serio->dev->disconnect)
203
                serio->dev->disconnect(serio);
204
}
205
 
206
void serio_register_device(struct serio_dev *dev)
207
{
208
        struct serio *serio;
209
        //!!!down(&serio_sem);
210
        list_add_tail(&dev->node, &serio_dev_list);
211
        list_for_each_entry(serio, &serio_list, node)
212
                if (!serio->dev && dev->connect)
213
                        dev->connect(serio, dev);
214
        //!!!up(&serio_sem);
215
}
216
 
217
void serio_unregister_device(struct serio_dev *dev)
218
{
219
        struct serio *serio;
220
 
221
        //!!!down(&serio_sem);
222
        list_del_init(&dev->node);
223
 
224
        list_for_each_entry(serio, &serio_list, node) {
225
                if (serio->dev == dev && dev->disconnect)
226
                        dev->disconnect(serio);
227
                serio_find_dev(serio);
228
        }
229
        //!!!up(&serio_sem);
230
}
231
 
232
/* called from serio_dev->connect/disconnect methods under serio_sem */
233
int serio_open(struct serio *serio, struct serio_dev *dev)
234
{
235
        serio->dev = dev;
236
        if (serio->open(serio)) {
237
                serio->dev = NULL;
238
                return -1;
239
        }
240
        return 0;
241
}
242
 
243
/* called from serio_dev->connect/disconnect methods under serio_sem */
244
void serio_close(struct serio *serio)
245
{
246
        serio->close(serio);
247
        serio->dev = NULL;
248
}
249
 
250
/*static*/ int __init serio_init(void)
251
{
252
        int pid = 1;
253
 
254
        //!!!pid = kernel_thread(serio_thread, NULL, CLONE_KERNEL);
255
 
256
        if (!pid) {
257
                printk(KERN_WARNING "serio: Failed to start kseriod\n");
258
                return -1;
259
        }
260
 
261
        serio_pid = pid;
262
 
263
        return 0;
264
}
265
 
266
/*static*/ void __exit serio_exit(void)
267
{
268
        //!!!kill_proc(serio_pid, SIGTERM, 1);
269
        wait_for_completion(&serio_exited);
270
}
271
 
272
module_init(serio_init);
273
module_exit(serio_exit);