2 * OSS compatible sequencer driver
6 * Copyright (C) 1998,99 Takashi Iwai <tiwai@suse.de>
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23 #include <sound/asoundef.h>
24 #include "seq_oss_midi.h"
25 #include "seq_oss_readq.h"
26 #include "seq_oss_timer.h"
27 #include "seq_oss_event.h"
28 #include <sound/seq_midi_event.h>
29 #include "../seq_lock.h"
30 #include <linux/init.h>
31 #include <linux/slab.h>
37 #define SNDRV_SEQ_OSS_MAX_MIDI_NAME 30
40 * definition of midi device record
43 int seq_device
; /* device number */
44 int client
; /* sequencer client number */
45 int port
; /* sequencer port number */
46 unsigned int flags
; /* port capability */
47 int opened
; /* flag for opening */
48 unsigned char name
[SNDRV_SEQ_OSS_MAX_MIDI_NAME
];
49 struct snd_midi_event
*coder
; /* MIDI event coder */
50 struct seq_oss_devinfo
*devinfo
; /* assigned OSSseq device */
51 snd_use_lock_t use_lock
;
58 static int max_midi_devs
;
59 static struct seq_oss_midi
*midi_devs
[SNDRV_SEQ_OSS_MAX_MIDI_DEVS
];
61 static DEFINE_SPINLOCK(register_lock
);
66 static struct seq_oss_midi
*get_mdev(int dev
);
67 static struct seq_oss_midi
*get_mididev(struct seq_oss_devinfo
*dp
, int dev
);
68 static int send_synth_event(struct seq_oss_devinfo
*dp
, struct snd_seq_event
*ev
, int dev
);
69 static int send_midi_event(struct seq_oss_devinfo
*dp
, struct snd_seq_event
*ev
, struct seq_oss_midi
*mdev
);
72 * look up the existing ports
73 * this looks a very exhausting job.
76 snd_seq_oss_midi_lookup_ports(int client
)
78 struct snd_seq_client_info
*clinfo
;
79 struct snd_seq_port_info
*pinfo
;
81 clinfo
= kzalloc(sizeof(*clinfo
), GFP_KERNEL
);
82 pinfo
= kzalloc(sizeof(*pinfo
), GFP_KERNEL
);
83 if (! clinfo
|| ! pinfo
) {
89 while (snd_seq_kernel_client_ctl(client
, SNDRV_SEQ_IOCTL_QUERY_NEXT_CLIENT
, clinfo
) == 0) {
90 if (clinfo
->client
== client
)
91 continue; /* ignore myself */
92 pinfo
->addr
.client
= clinfo
->client
;
93 pinfo
->addr
.port
= -1;
94 while (snd_seq_kernel_client_ctl(client
, SNDRV_SEQ_IOCTL_QUERY_NEXT_PORT
, pinfo
) == 0)
95 snd_seq_oss_midi_check_new_port(pinfo
);
105 static struct seq_oss_midi
*
108 struct seq_oss_midi
*mdev
;
111 spin_lock_irqsave(®ister_lock
, flags
);
112 mdev
= midi_devs
[dev
];
114 snd_use_lock_use(&mdev
->use_lock
);
115 spin_unlock_irqrestore(®ister_lock
, flags
);
120 * look for the identical slot
122 static struct seq_oss_midi
*
123 find_slot(int client
, int port
)
126 struct seq_oss_midi
*mdev
;
129 spin_lock_irqsave(®ister_lock
, flags
);
130 for (i
= 0; i
< max_midi_devs
; i
++) {
132 if (mdev
&& mdev
->client
== client
&& mdev
->port
== port
) {
134 snd_use_lock_use(&mdev
->use_lock
);
135 spin_unlock_irqrestore(®ister_lock
, flags
);
139 spin_unlock_irqrestore(®ister_lock
, flags
);
144 #define PERM_WRITE (SNDRV_SEQ_PORT_CAP_WRITE|SNDRV_SEQ_PORT_CAP_SUBS_WRITE)
145 #define PERM_READ (SNDRV_SEQ_PORT_CAP_READ|SNDRV_SEQ_PORT_CAP_SUBS_READ)
147 * register a new port if it doesn't exist yet
150 snd_seq_oss_midi_check_new_port(struct snd_seq_port_info
*pinfo
)
153 struct seq_oss_midi
*mdev
;
156 /* the port must include generic midi */
157 if (! (pinfo
->type
& SNDRV_SEQ_PORT_TYPE_MIDI_GENERIC
))
159 /* either read or write subscribable */
160 if ((pinfo
->capability
& PERM_WRITE
) != PERM_WRITE
&&
161 (pinfo
->capability
& PERM_READ
) != PERM_READ
)
165 * look for the identical slot
167 if ((mdev
= find_slot(pinfo
->addr
.client
, pinfo
->addr
.port
)) != NULL
) {
169 snd_use_lock_free(&mdev
->use_lock
);
174 * allocate midi info record
176 if ((mdev
= kzalloc(sizeof(*mdev
), GFP_KERNEL
)) == NULL
) {
177 pr_err("ALSA: seq_oss: can't malloc midi info\n");
181 /* copy the port information */
182 mdev
->client
= pinfo
->addr
.client
;
183 mdev
->port
= pinfo
->addr
.port
;
184 mdev
->flags
= pinfo
->capability
;
186 snd_use_lock_init(&mdev
->use_lock
);
188 /* copy and truncate the name of synth device */
189 strlcpy(mdev
->name
, pinfo
->name
, sizeof(mdev
->name
));
191 /* create MIDI coder */
192 if (snd_midi_event_new(MAX_MIDI_EVENT_BUF
, &mdev
->coder
) < 0) {
193 pr_err("ALSA: seq_oss: can't malloc midi coder\n");
197 /* OSS sequencer adds running status to all sequences */
198 snd_midi_event_no_status(mdev
->coder
, 1);
201 * look for en empty slot
203 spin_lock_irqsave(®ister_lock
, flags
);
204 for (i
= 0; i
< max_midi_devs
; i
++) {
205 if (midi_devs
[i
] == NULL
)
208 if (i
>= max_midi_devs
) {
209 if (max_midi_devs
>= SNDRV_SEQ_OSS_MAX_MIDI_DEVS
) {
210 spin_unlock_irqrestore(®ister_lock
, flags
);
211 snd_midi_event_free(mdev
->coder
);
217 mdev
->seq_device
= i
;
218 midi_devs
[mdev
->seq_device
] = mdev
;
219 spin_unlock_irqrestore(®ister_lock
, flags
);
225 * release the midi device if it was registered
228 snd_seq_oss_midi_check_exit_port(int client
, int port
)
230 struct seq_oss_midi
*mdev
;
234 if ((mdev
= find_slot(client
, port
)) != NULL
) {
235 spin_lock_irqsave(®ister_lock
, flags
);
236 midi_devs
[mdev
->seq_device
] = NULL
;
237 spin_unlock_irqrestore(®ister_lock
, flags
);
238 snd_use_lock_free(&mdev
->use_lock
);
239 snd_use_lock_sync(&mdev
->use_lock
);
241 snd_midi_event_free(mdev
->coder
);
244 spin_lock_irqsave(®ister_lock
, flags
);
245 for (index
= max_midi_devs
- 1; index
>= 0; index
--) {
246 if (midi_devs
[index
])
249 max_midi_devs
= index
+ 1;
250 spin_unlock_irqrestore(®ister_lock
, flags
);
256 * release the midi device if it was registered
259 snd_seq_oss_midi_clear_all(void)
262 struct seq_oss_midi
*mdev
;
265 spin_lock_irqsave(®ister_lock
, flags
);
266 for (i
= 0; i
< max_midi_devs
; i
++) {
267 if ((mdev
= midi_devs
[i
]) != NULL
) {
269 snd_midi_event_free(mdev
->coder
);
275 spin_unlock_irqrestore(®ister_lock
, flags
);
283 snd_seq_oss_midi_setup(struct seq_oss_devinfo
*dp
)
285 dp
->max_mididev
= max_midi_devs
;
289 * clean up midi tables
292 snd_seq_oss_midi_cleanup(struct seq_oss_devinfo
*dp
)
295 for (i
= 0; i
< dp
->max_mididev
; i
++)
296 snd_seq_oss_midi_close(dp
, i
);
302 * open all midi devices. ignore errors.
305 snd_seq_oss_midi_open_all(struct seq_oss_devinfo
*dp
, int file_mode
)
308 for (i
= 0; i
< dp
->max_mididev
; i
++)
309 snd_seq_oss_midi_open(dp
, i
, file_mode
);
314 * get the midi device information
316 static struct seq_oss_midi
*
317 get_mididev(struct seq_oss_devinfo
*dp
, int dev
)
319 if (dev
< 0 || dev
>= dp
->max_mididev
)
321 return get_mdev(dev
);
326 * open the midi device if not opened yet
329 snd_seq_oss_midi_open(struct seq_oss_devinfo
*dp
, int dev
, int fmode
)
332 struct seq_oss_midi
*mdev
;
333 struct snd_seq_port_subscribe subs
;
335 if ((mdev
= get_mididev(dp
, dev
)) == NULL
)
339 if (mdev
->opened
&& mdev
->devinfo
!= dp
) {
340 snd_use_lock_free(&mdev
->use_lock
);
345 if (is_write_mode(fmode
))
347 if (is_read_mode(fmode
))
351 snd_use_lock_free(&mdev
->use_lock
);
355 /* already opened? */
356 if ((mdev
->opened
& perm
) == perm
) {
357 snd_use_lock_free(&mdev
->use_lock
);
361 perm
&= ~mdev
->opened
;
363 memset(&subs
, 0, sizeof(subs
));
365 if (perm
& PERM_WRITE
) {
366 subs
.sender
= dp
->addr
;
367 subs
.dest
.client
= mdev
->client
;
368 subs
.dest
.port
= mdev
->port
;
369 if (snd_seq_kernel_client_ctl(dp
->cseq
, SNDRV_SEQ_IOCTL_SUBSCRIBE_PORT
, &subs
) >= 0)
370 mdev
->opened
|= PERM_WRITE
;
372 if (perm
& PERM_READ
) {
373 subs
.sender
.client
= mdev
->client
;
374 subs
.sender
.port
= mdev
->port
;
375 subs
.dest
= dp
->addr
;
376 subs
.flags
= SNDRV_SEQ_PORT_SUBS_TIMESTAMP
;
377 subs
.queue
= dp
->queue
; /* queue for timestamps */
378 if (snd_seq_kernel_client_ctl(dp
->cseq
, SNDRV_SEQ_IOCTL_SUBSCRIBE_PORT
, &subs
) >= 0)
379 mdev
->opened
|= PERM_READ
;
382 if (! mdev
->opened
) {
383 snd_use_lock_free(&mdev
->use_lock
);
388 snd_use_lock_free(&mdev
->use_lock
);
393 * close the midi device if already opened
396 snd_seq_oss_midi_close(struct seq_oss_devinfo
*dp
, int dev
)
398 struct seq_oss_midi
*mdev
;
399 struct snd_seq_port_subscribe subs
;
401 if ((mdev
= get_mididev(dp
, dev
)) == NULL
)
403 if (! mdev
->opened
|| mdev
->devinfo
!= dp
) {
404 snd_use_lock_free(&mdev
->use_lock
);
408 memset(&subs
, 0, sizeof(subs
));
409 if (mdev
->opened
& PERM_WRITE
) {
410 subs
.sender
= dp
->addr
;
411 subs
.dest
.client
= mdev
->client
;
412 subs
.dest
.port
= mdev
->port
;
413 snd_seq_kernel_client_ctl(dp
->cseq
, SNDRV_SEQ_IOCTL_UNSUBSCRIBE_PORT
, &subs
);
415 if (mdev
->opened
& PERM_READ
) {
416 subs
.sender
.client
= mdev
->client
;
417 subs
.sender
.port
= mdev
->port
;
418 subs
.dest
= dp
->addr
;
419 snd_seq_kernel_client_ctl(dp
->cseq
, SNDRV_SEQ_IOCTL_UNSUBSCRIBE_PORT
, &subs
);
423 mdev
->devinfo
= NULL
;
425 snd_use_lock_free(&mdev
->use_lock
);
430 * change seq capability flags to file mode flags
433 snd_seq_oss_midi_filemode(struct seq_oss_devinfo
*dp
, int dev
)
435 struct seq_oss_midi
*mdev
;
438 if ((mdev
= get_mididev(dp
, dev
)) == NULL
)
442 if (mdev
->opened
& PERM_WRITE
)
443 mode
|= SNDRV_SEQ_OSS_FILE_WRITE
;
444 if (mdev
->opened
& PERM_READ
)
445 mode
|= SNDRV_SEQ_OSS_FILE_READ
;
447 snd_use_lock_free(&mdev
->use_lock
);
452 * reset the midi device and close it:
453 * so far, only close the device.
456 snd_seq_oss_midi_reset(struct seq_oss_devinfo
*dp
, int dev
)
458 struct seq_oss_midi
*mdev
;
460 if ((mdev
= get_mididev(dp
, dev
)) == NULL
)
462 if (! mdev
->opened
) {
463 snd_use_lock_free(&mdev
->use_lock
);
467 if (mdev
->opened
& PERM_WRITE
) {
468 struct snd_seq_event ev
;
471 memset(&ev
, 0, sizeof(ev
));
472 ev
.dest
.client
= mdev
->client
;
473 ev
.dest
.port
= mdev
->port
;
474 ev
.queue
= dp
->queue
;
475 ev
.source
.port
= dp
->port
;
476 if (dp
->seq_mode
== SNDRV_SEQ_OSS_MODE_SYNTH
) {
477 ev
.type
= SNDRV_SEQ_EVENT_SENSING
;
478 snd_seq_oss_dispatch(dp
, &ev
, 0, 0);
480 for (c
= 0; c
< 16; c
++) {
481 ev
.type
= SNDRV_SEQ_EVENT_CONTROLLER
;
482 ev
.data
.control
.channel
= c
;
483 ev
.data
.control
.param
= MIDI_CTL_ALL_NOTES_OFF
;
484 snd_seq_oss_dispatch(dp
, &ev
, 0, 0);
485 if (dp
->seq_mode
== SNDRV_SEQ_OSS_MODE_MUSIC
) {
486 ev
.data
.control
.param
=
487 MIDI_CTL_RESET_CONTROLLERS
;
488 snd_seq_oss_dispatch(dp
, &ev
, 0, 0);
489 ev
.type
= SNDRV_SEQ_EVENT_PITCHBEND
;
490 ev
.data
.control
.value
= 0;
491 snd_seq_oss_dispatch(dp
, &ev
, 0, 0);
495 // snd_seq_oss_midi_close(dp, dev);
496 snd_use_lock_free(&mdev
->use_lock
);
501 * get client/port of the specified MIDI device
504 snd_seq_oss_midi_get_addr(struct seq_oss_devinfo
*dp
, int dev
, struct snd_seq_addr
*addr
)
506 struct seq_oss_midi
*mdev
;
508 if ((mdev
= get_mididev(dp
, dev
)) == NULL
)
510 addr
->client
= mdev
->client
;
511 addr
->port
= mdev
->port
;
512 snd_use_lock_free(&mdev
->use_lock
);
517 * input callback - this can be atomic
520 snd_seq_oss_midi_input(struct snd_seq_event
*ev
, int direct
, void *private_data
)
522 struct seq_oss_devinfo
*dp
= (struct seq_oss_devinfo
*)private_data
;
523 struct seq_oss_midi
*mdev
;
526 if (dp
->readq
== NULL
)
528 if ((mdev
= find_slot(ev
->source
.client
, ev
->source
.port
)) == NULL
)
530 if (! (mdev
->opened
& PERM_READ
)) {
531 snd_use_lock_free(&mdev
->use_lock
);
535 if (dp
->seq_mode
== SNDRV_SEQ_OSS_MODE_MUSIC
)
536 rc
= send_synth_event(dp
, ev
, mdev
->seq_device
);
538 rc
= send_midi_event(dp
, ev
, mdev
);
540 snd_use_lock_free(&mdev
->use_lock
);
545 * convert ALSA sequencer event to OSS synth event
548 send_synth_event(struct seq_oss_devinfo
*dp
, struct snd_seq_event
*ev
, int dev
)
552 memset(&ossev
, 0, sizeof(ossev
));
555 case SNDRV_SEQ_EVENT_NOTEON
:
556 ossev
.v
.cmd
= MIDI_NOTEON
; break;
557 case SNDRV_SEQ_EVENT_NOTEOFF
:
558 ossev
.v
.cmd
= MIDI_NOTEOFF
; break;
559 case SNDRV_SEQ_EVENT_KEYPRESS
:
560 ossev
.v
.cmd
= MIDI_KEY_PRESSURE
; break;
561 case SNDRV_SEQ_EVENT_CONTROLLER
:
562 ossev
.l
.cmd
= MIDI_CTL_CHANGE
; break;
563 case SNDRV_SEQ_EVENT_PGMCHANGE
:
564 ossev
.l
.cmd
= MIDI_PGM_CHANGE
; break;
565 case SNDRV_SEQ_EVENT_CHANPRESS
:
566 ossev
.l
.cmd
= MIDI_CHN_PRESSURE
; break;
567 case SNDRV_SEQ_EVENT_PITCHBEND
:
568 ossev
.l
.cmd
= MIDI_PITCH_BEND
; break;
570 return 0; /* not supported */
576 case SNDRV_SEQ_EVENT_NOTEON
:
577 case SNDRV_SEQ_EVENT_NOTEOFF
:
578 case SNDRV_SEQ_EVENT_KEYPRESS
:
579 ossev
.v
.code
= EV_CHN_VOICE
;
580 ossev
.v
.note
= ev
->data
.note
.note
;
581 ossev
.v
.parm
= ev
->data
.note
.velocity
;
582 ossev
.v
.chn
= ev
->data
.note
.channel
;
584 case SNDRV_SEQ_EVENT_CONTROLLER
:
585 case SNDRV_SEQ_EVENT_PGMCHANGE
:
586 case SNDRV_SEQ_EVENT_CHANPRESS
:
587 ossev
.l
.code
= EV_CHN_COMMON
;
588 ossev
.l
.p1
= ev
->data
.control
.param
;
589 ossev
.l
.val
= ev
->data
.control
.value
;
590 ossev
.l
.chn
= ev
->data
.control
.channel
;
592 case SNDRV_SEQ_EVENT_PITCHBEND
:
593 ossev
.l
.code
= EV_CHN_COMMON
;
594 ossev
.l
.val
= ev
->data
.control
.value
+ 8192;
595 ossev
.l
.chn
= ev
->data
.control
.channel
;
599 snd_seq_oss_readq_put_timestamp(dp
->readq
, ev
->time
.tick
, dp
->seq_mode
);
600 snd_seq_oss_readq_put_event(dp
->readq
, &ossev
);
606 * decode event and send MIDI bytes to read queue
609 send_midi_event(struct seq_oss_devinfo
*dp
, struct snd_seq_event
*ev
, struct seq_oss_midi
*mdev
)
614 snd_seq_oss_readq_put_timestamp(dp
->readq
, ev
->time
.tick
, dp
->seq_mode
);
615 if (!dp
->timer
->running
)
616 len
= snd_seq_oss_timer_start(dp
->timer
);
617 if (ev
->type
== SNDRV_SEQ_EVENT_SYSEX
) {
618 if ((ev
->flags
& SNDRV_SEQ_EVENT_LENGTH_MASK
) == SNDRV_SEQ_EVENT_LENGTH_VARIABLE
)
619 snd_seq_oss_readq_puts(dp
->readq
, mdev
->seq_device
,
620 ev
->data
.ext
.ptr
, ev
->data
.ext
.len
);
622 len
= snd_midi_event_decode(mdev
->coder
, msg
, sizeof(msg
), ev
);
624 snd_seq_oss_readq_puts(dp
->readq
, mdev
->seq_device
, msg
, len
);
633 * return 0 : enqueued
634 * non-zero : invalid - ignored
637 snd_seq_oss_midi_putc(struct seq_oss_devinfo
*dp
, int dev
, unsigned char c
, struct snd_seq_event
*ev
)
639 struct seq_oss_midi
*mdev
;
641 if ((mdev
= get_mididev(dp
, dev
)) == NULL
)
643 if (snd_midi_event_encode_byte(mdev
->coder
, c
, ev
) > 0) {
644 snd_seq_oss_fill_addr(dp
, ev
, mdev
->client
, mdev
->port
);
645 snd_use_lock_free(&mdev
->use_lock
);
648 snd_use_lock_free(&mdev
->use_lock
);
653 * create OSS compatible midi_info record
656 snd_seq_oss_midi_make_info(struct seq_oss_devinfo
*dp
, int dev
, struct midi_info
*inf
)
658 struct seq_oss_midi
*mdev
;
660 if ((mdev
= get_mididev(dp
, dev
)) == NULL
)
663 inf
->dev_type
= 0; /* FIXME: ?? */
664 inf
->capabilities
= 0; /* FIXME: ?? */
665 strlcpy(inf
->name
, mdev
->name
, sizeof(inf
->name
));
666 snd_use_lock_free(&mdev
->use_lock
);
671 #ifdef CONFIG_PROC_FS
678 val
&= PERM_READ
|PERM_WRITE
;
679 if (val
== (PERM_READ
|PERM_WRITE
))
681 else if (val
== PERM_READ
)
683 else if (val
== PERM_WRITE
)
690 snd_seq_oss_midi_info_read(struct snd_info_buffer
*buf
)
693 struct seq_oss_midi
*mdev
;
695 snd_iprintf(buf
, "\nNumber of MIDI devices: %d\n", max_midi_devs
);
696 for (i
= 0; i
< max_midi_devs
; i
++) {
697 snd_iprintf(buf
, "\nmidi %d: ", i
);
700 snd_iprintf(buf
, "*empty*\n");
703 snd_iprintf(buf
, "[%s] ALSA port %d:%d\n", mdev
->name
,
704 mdev
->client
, mdev
->port
);
705 snd_iprintf(buf
, " capability %s / opened %s\n",
706 capmode_str(mdev
->flags
),
707 capmode_str(mdev
->opened
));
708 snd_use_lock_free(&mdev
->use_lock
);
711 #endif /* CONFIG_PROC_FS */