This source file includes following definitions.
- drain_midi_queue
- midi_input_intr
- midi_output_intr
- midi_poll
- MIDIbuf_open
- MIDIbuf_release
- MIDIbuf_write
- MIDIbuf_read
- MIDIbuf_ioctl
- MIDIbuf_select
- MIDIbuf_init
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 #include <linux/config.h>
30
31
32 #include "sound_config.h"
33
34 #if defined(CONFIG_MIDI)
35
36
37
38
39
40 #define MAX_QUEUE_SIZE 4000
41
42 static wait_handle *midi_sleeper[MAX_MIDI_DEV] =
43 {NULL};
44 static volatile struct snd_wait midi_sleep_flag[MAX_MIDI_DEV] =
45 {
46 {0}};
47 static wait_handle *input_sleeper[MAX_MIDI_DEV] =
48 {NULL};
49 static volatile struct snd_wait input_sleep_flag[MAX_MIDI_DEV] =
50 {
51 {0}};
52
53 struct midi_buf
54 {
55 int len, head, tail;
56 unsigned char queue[MAX_QUEUE_SIZE];
57 };
58
59 struct midi_parms
60 {
61 int prech_timeout;
62
63
64 };
65
66 static struct midi_buf *midi_out_buf[MAX_MIDI_DEV] =
67 {NULL};
68 static struct midi_buf *midi_in_buf[MAX_MIDI_DEV] =
69 {NULL};
70 static struct midi_parms parms[MAX_MIDI_DEV];
71
72 static void midi_poll (unsigned long dummy);
73
74
75 static struct timer_list poll_timer =
76 {NULL, NULL, 0, 0, midi_poll};
77 static volatile int open_devs = 0;
78
79 #define DATA_AVAIL(q) (q->len)
80 #define SPACE_AVAIL(q) (MAX_QUEUE_SIZE - q->len)
81
82 #define QUEUE_BYTE(q, data) \
83 if (SPACE_AVAIL(q)) \
84 { \
85 unsigned long flags; \
86 save_flags(flags);cli(); \
87 q->queue[q->tail] = (data); \
88 q->len++; q->tail = (q->tail+1) % MAX_QUEUE_SIZE; \
89 restore_flags(flags); \
90 }
91
92 #define REMOVE_BYTE(q, data) \
93 if (DATA_AVAIL(q)) \
94 { \
95 unsigned long flags; \
96 save_flags(flags);cli(); \
97 data = q->queue[q->head]; \
98 q->len--; q->head = (q->head+1) % MAX_QUEUE_SIZE; \
99 restore_flags(flags); \
100 }
101
102 void
103 drain_midi_queue (int dev)
104 {
105
106
107
108
109
110 if (midi_devs[dev]->buffer_status != NULL)
111 while (!current_got_fatal_signal () &&
112 midi_devs[dev]->buffer_status (dev))
113
114 {
115 unsigned long tl;
116
117 if (HZ / 10)
118 current_set_timeout (tl = jiffies + (HZ / 10));
119 else
120 tl = (unsigned long) -1;
121 midi_sleep_flag[dev].mode = WK_SLEEP;
122 module_interruptible_sleep_on (&midi_sleeper[dev]);
123 if (!(midi_sleep_flag[dev].mode & WK_WAKEUP))
124 {
125 if (jiffies >= tl)
126 midi_sleep_flag[dev].mode |= WK_TIMEOUT;
127 }
128 midi_sleep_flag[dev].mode &= ~WK_SLEEP;
129 };
130 }
131
132 static void
133 midi_input_intr (int dev, unsigned char data)
134 {
135 if (midi_in_buf[dev] == NULL)
136 return;
137
138 if (data == 0xfe)
139
140
141 return;
142
143
144
145 if (SPACE_AVAIL (midi_in_buf[dev]))
146 {
147 QUEUE_BYTE (midi_in_buf[dev], data);
148 if ((input_sleep_flag[dev].mode & WK_SLEEP))
149 {
150 input_sleep_flag[dev].mode = WK_WAKEUP;
151 module_wake_up (&input_sleeper[dev]);
152 };
153 }
154
155 }
156
157 static void
158 midi_output_intr (int dev)
159 {
160
161
162
163 }
164
165 static void
166 midi_poll (unsigned long dummy)
167 {
168 unsigned long flags;
169 int dev;
170
171 save_flags (flags);
172 cli ();
173 if (open_devs)
174 {
175 for (dev = 0; dev < num_midis; dev++)
176 if (midi_out_buf[dev] != NULL)
177 {
178 while (DATA_AVAIL (midi_out_buf[dev]) &&
179 midi_devs[dev]->putc (dev,
180 midi_out_buf[dev]->queue[midi_out_buf[dev]->head]))
181 {
182 midi_out_buf[dev]->head = (midi_out_buf[dev]->head + 1) % MAX_QUEUE_SIZE;
183 midi_out_buf[dev]->len--;
184 }
185
186 if (DATA_AVAIL (midi_out_buf[dev]) < 100 &&
187 (midi_sleep_flag[dev].mode & WK_SLEEP))
188 {
189 midi_sleep_flag[dev].mode = WK_WAKEUP;
190 module_wake_up (&midi_sleeper[dev]);
191 };
192 }
193
194 {
195 poll_timer.expires = (1) + jiffies;
196 add_timer (&poll_timer);
197 };
198
199
200 }
201 restore_flags (flags);
202 }
203
204 int
205 MIDIbuf_open (int dev, struct fileinfo *file)
206 {
207 int mode, err;
208
209 dev = dev >> 4;
210 mode = file->mode & O_ACCMODE;
211
212 if (num_midis > MAX_MIDI_DEV)
213 {
214 printk ("Sound: FATAL ERROR: Too many midi interfaces\n");
215 num_midis = MAX_MIDI_DEV;
216 }
217
218 if (dev < 0 || dev >= num_midis)
219 {
220 printk ("Sound: Nonexistent MIDI interface %d\n", dev);
221 return -ENXIO;
222 }
223
224
225
226
227
228 if ((err = midi_devs[dev]->open (dev, mode,
229 midi_input_intr, midi_output_intr)) < 0)
230 {
231 return err;
232 }
233
234 parms[dev].prech_timeout = 0;
235
236 midi_in_buf[dev] = (struct midi_buf *) kmalloc (sizeof (struct midi_buf), GFP_KERNEL);
237
238 if (midi_in_buf[dev] == NULL)
239 {
240 printk ("midi: Can't allocate buffer\n");
241 midi_devs[dev]->close (dev);
242 return -EIO;
243 }
244 midi_in_buf[dev]->len = midi_in_buf[dev]->head = midi_in_buf[dev]->tail = 0;
245
246 midi_out_buf[dev] = (struct midi_buf *) kmalloc (sizeof (struct midi_buf), GFP_KERNEL);
247
248 if (midi_out_buf[dev] == NULL)
249 {
250 printk ("midi: Can't allocate buffer\n");
251 midi_devs[dev]->close (dev);
252 kfree (midi_in_buf[dev]);
253 midi_in_buf[dev] = NULL;
254 return -EIO;
255 }
256 midi_out_buf[dev]->len = midi_out_buf[dev]->head = midi_out_buf[dev]->tail = 0;
257 open_devs++;
258
259 midi_sleep_flag[dev].mode = WK_NONE;
260 input_sleep_flag[dev].mode = WK_NONE;
261
262 if (open_devs < 2)
263 {
264 ;
265
266 {
267 poll_timer.expires = (1) + jiffies;
268 add_timer (&poll_timer);
269 };
270 }
271
272 return err;
273 }
274
275 void
276 MIDIbuf_release (int dev, struct fileinfo *file)
277 {
278 int mode;
279 unsigned long flags;
280
281 dev = dev >> 4;
282 mode = file->mode & O_ACCMODE;
283
284 if (dev < 0 || dev >= num_midis)
285 return;
286
287 save_flags (flags);
288 cli ();
289
290
291
292
293
294 if (mode != OPEN_READ)
295 {
296 midi_devs[dev]->putc (dev, 0xfe);
297
298
299
300
301 while (!current_got_fatal_signal () &&
302 DATA_AVAIL (midi_out_buf[dev]))
303
304 {
305 unsigned long tl;
306
307 if (0)
308 current_set_timeout (tl = jiffies + (0));
309 else
310 tl = (unsigned long) -1;
311 midi_sleep_flag[dev].mode = WK_SLEEP;
312 module_interruptible_sleep_on (&midi_sleeper[dev]);
313 if (!(midi_sleep_flag[dev].mode & WK_WAKEUP))
314 {
315 if (jiffies >= tl)
316 midi_sleep_flag[dev].mode |= WK_TIMEOUT;
317 }
318 midi_sleep_flag[dev].mode &= ~WK_SLEEP;
319 };
320
321
322
323 drain_midi_queue (dev);
324
325
326 }
327
328 restore_flags (flags);
329
330 midi_devs[dev]->close (dev);
331
332 kfree (midi_in_buf[dev]);
333 kfree (midi_out_buf[dev]);
334 midi_in_buf[dev] = NULL;
335 midi_out_buf[dev] = NULL;
336 if (open_devs < 2)
337 del_timer (&poll_timer);;
338 open_devs--;
339 }
340
341 int
342 MIDIbuf_write (int dev, struct fileinfo *file, const char *buf, int count)
343 {
344 unsigned long flags;
345 int c, n, i;
346 unsigned char tmp_data;
347
348 dev = dev >> 4;
349
350 if (!count)
351 return 0;
352
353 save_flags (flags);
354 cli ();
355
356 c = 0;
357
358 while (c < count)
359 {
360 n = SPACE_AVAIL (midi_out_buf[dev]);
361
362 if (n == 0)
363
364
365 {
366
367 {
368 unsigned long tl;
369
370 if (0)
371 current_set_timeout (tl = jiffies + (0));
372 else
373 tl = (unsigned long) -1;
374 midi_sleep_flag[dev].mode = WK_SLEEP;
375 module_interruptible_sleep_on (&midi_sleeper[dev]);
376 if (!(midi_sleep_flag[dev].mode & WK_WAKEUP))
377 {
378 if (jiffies >= tl)
379 midi_sleep_flag[dev].mode |= WK_TIMEOUT;
380 }
381 midi_sleep_flag[dev].mode &= ~WK_SLEEP;
382 };
383 if (current_got_fatal_signal ())
384 {
385 restore_flags (flags);
386 return -EINTR;
387 }
388
389 n = SPACE_AVAIL (midi_out_buf[dev]);
390 }
391
392 if (n > (count - c))
393 n = count - c;
394
395 for (i = 0; i < n; i++)
396 {
397 memcpy_fromfs ((char *) &tmp_data, &((buf)[c]), 1);
398 QUEUE_BYTE (midi_out_buf[dev], tmp_data);
399 c++;
400 }
401 }
402
403 restore_flags (flags);
404
405 return c;
406 }
407
408
409 int
410 MIDIbuf_read (int dev, struct fileinfo *file, char *buf, int count)
411 {
412 int n, c = 0;
413 unsigned long flags;
414 unsigned char tmp_data;
415
416 dev = dev >> 4;
417
418 save_flags (flags);
419 cli ();
420
421 if (!DATA_AVAIL (midi_in_buf[dev]))
422
423
424 {
425
426 {
427 unsigned long tl;
428
429 if (parms[dev].prech_timeout)
430 current_set_timeout (tl = jiffies + (parms[dev].prech_timeout));
431 else
432 tl = (unsigned long) -1;
433 input_sleep_flag[dev].mode = WK_SLEEP;
434 module_interruptible_sleep_on (&input_sleeper[dev]);
435 if (!(input_sleep_flag[dev].mode & WK_WAKEUP))
436 {
437 if (jiffies >= tl)
438 input_sleep_flag[dev].mode |= WK_TIMEOUT;
439 }
440 input_sleep_flag[dev].mode &= ~WK_SLEEP;
441 };
442 if (current_got_fatal_signal ())
443 c = -EINTR;
444
445
446 }
447
448 if (c == 0 && DATA_AVAIL (midi_in_buf[dev]))
449
450
451 {
452 n = DATA_AVAIL (midi_in_buf[dev]);
453 if (n > count)
454 n = count;
455 c = 0;
456
457 while (c < n)
458 {
459 REMOVE_BYTE (midi_in_buf[dev], tmp_data);
460 memcpy_tofs (&((buf)[c]), (char *) &tmp_data, 1);
461 c++;
462 }
463 }
464
465 restore_flags (flags);
466
467 return c;
468 }
469
470 int
471 MIDIbuf_ioctl (int dev, struct fileinfo *file,
472 unsigned int cmd, caddr_t arg)
473 {
474 int val;
475
476 dev = dev >> 4;
477
478 if (((cmd >> 8) & 0xff) == 'C')
479 {
480 if (midi_devs[dev]->coproc)
481 return midi_devs[dev]->coproc->ioctl (midi_devs[dev]->coproc->devc, cmd, arg, 0);
482 else
483 printk ("/dev/midi%d: No coprocessor for this device\n", dev);
484
485 return -ENXIO;
486 }
487 else
488 switch (cmd)
489 {
490
491 case SNDCTL_MIDI_PRETIME:
492 val = (int) get_fs_long ((long *) arg);
493 if (val < 0)
494 val = 0;
495
496 val = (HZ * val) / 10;
497 parms[dev].prech_timeout = val;
498 return snd_ioctl_return ((int *) arg, val);
499 break;
500
501 default:
502 return midi_devs[dev]->ioctl (dev, cmd, arg);
503 }
504 }
505
506 int
507 MIDIbuf_select (int dev, struct fileinfo *file, int sel_type, select_table_handle * wait)
508 {
509 dev = dev >> 4;
510
511 switch (sel_type)
512 {
513 case SEL_IN:
514 if (!DATA_AVAIL (midi_in_buf[dev]))
515 {
516
517 input_sleep_flag[dev].mode = WK_SLEEP;
518 module_select_wait (&input_sleeper[dev], wait);
519 return 0;
520 }
521 return 1;
522 break;
523
524 case SEL_OUT:
525 if (SPACE_AVAIL (midi_out_buf[dev]))
526 {
527
528 midi_sleep_flag[dev].mode = WK_SLEEP;
529 module_select_wait (&midi_sleeper[dev], wait);
530 return 0;
531 }
532 return 1;
533 break;
534
535 case SEL_EX:
536 return 0;
537 }
538
539 return 0;
540 }
541
542
543 long
544 MIDIbuf_init (long mem_start)
545 {
546 return mem_start;
547 }
548
549 #endif