freebsd-nq/sys/dev/sound/midi/midisynth.c
Seigo Tanimura 995231304d - Provide toggles to show debug messages. Set new sysctl variables
hw.midi.debug and hw.midi.seq.debug to 1 to enable debug log.

- Make debug messages human-frendly.

- Implement /dev/music.

- Add a timer engine required by /dev/music.

- Fix nonblocking I/O.

- Fix the numbering of midi and synth devices.
2002-01-04 01:13:49 +00:00

533 lines
11 KiB
C

/*
* Copyright by Hannu Savolainen 1993
*
* Redistribution and use in source and binary forms, with or
* without modification, are permitted provided that the following
* conditions are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following
* disclaimer in the documentation and/or other materials provided
* with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS
* IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* AUTHOR OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
* $FreeBSD$
*
*/
/*
* This is the interface for a sequencer to interact a midi driver.
* This interface translates the sequencer operations to the corresponding
* midi messages, and vice versa.
*/
#include <dev/sound/midi/midi.h>
#define TYPEDRANGE(type, x, lower, upper) \
{ \
type tl, tu; \
tl = (lower); \
tu = (upper); \
if (x < tl) { \
x = tl; \
} else if(x > tu) { \
x = tu; \
} \
}
/*
* These functions goes into midisynthdev_op_desc.
*/
static mdsy_killnote_t synth_killnote;
static mdsy_setinstr_t synth_setinstr;
static mdsy_startnote_t synth_startnote;
static mdsy_reset_t synth_reset;
static mdsy_hwcontrol_t synth_hwcontrol;
static mdsy_loadpatch_t synth_loadpatch;
static mdsy_panning_t synth_panning;
static mdsy_aftertouch_t synth_aftertouch;
static mdsy_controller_t synth_controller;
static mdsy_patchmgr_t synth_patchmgr;
static mdsy_bender_t synth_bender;
static mdsy_allocvoice_t synth_allocvoice;
static mdsy_setupvoice_t synth_setupvoice;
static mdsy_sendsysex_t synth_sendsysex;
static mdsy_prefixcmd_t synth_prefixcmd;
static mdsy_volumemethod_t synth_volumemethod;
static mdsy_readraw_t synth_readraw;
static mdsy_writeraw_t synth_writeraw;
/*
* This is the synthdev_info for a midi interface device.
* You may have to replace a few of functions for an internal
* synthesizer.
*/
synthdev_info midisynth_op_desc = {
synth_killnote,
synth_setinstr,
synth_startnote,
synth_reset,
synth_hwcontrol,
synth_loadpatch,
synth_panning,
synth_aftertouch,
synth_controller,
synth_patchmgr,
synth_bender,
synth_allocvoice,
synth_setupvoice,
synth_sendsysex,
synth_prefixcmd,
synth_volumemethod,
synth_readraw,
synth_writeraw,
};
/* The following functions are local. */
static int synth_leavesysex(mididev_info *md);
/*
* Here are the main functions to interact to the midi sequencer.
* These are called from the sequencer functions in sequencer.c.
*/
static int
synth_killnote(mididev_info *md, int chn, int note, int vel)
{
int unit, lenw;
synthdev_info *sd;
u_char c[3];
unit = md->unit;
sd = &md->synth;
if (note < 0 || note > 127 || chn < 0 || chn > 15)
return (EINVAL);
TYPEDRANGE(int, vel, 0, 127);
if (synth_leavesysex(md) == EAGAIN)
return (EAGAIN);
if (vel == 64) {
c[0] = 0x90 | (chn & 0x0f); /* Note on. */
c[1] = (u_char)note;
c[2] = 0;
} else {
c[0] = 0x80 | (chn & 0x0f); /* Note off. */
c[1] = (u_char)note;
c[2] = (u_char)vel;
}
if (synth_prefixcmd(md, c[0]))
return (0);
return (md->synth.writeraw(md, c, 3, &lenw, 1));
}
static int
synth_setinstr(mididev_info *md, int chn, int instr)
{
int unit, lenw;
synthdev_info *sd;
u_char c[2];
unit = md->unit;
sd = &md->synth;
if (instr < 0 || instr > 127 || chn < 0 || chn > 15)
return (EINVAL);
if (synth_leavesysex(md) == EAGAIN)
return (EAGAIN);
c[0] = 0xc0 | (chn & 0x0f); /* Progamme change. */
c[1] = (u_char)instr;
return (md->synth.writeraw(md, c, 3, &lenw, 1));
}
static int
synth_startnote(mididev_info *md, int chn, int note, int vel)
{
int unit, lenw;
synthdev_info *sd;
u_char c[3];
unit = md->unit;
sd = &md->synth;
if (note < 0 || note > 127 || chn < 0 || chn > 15)
return (EINVAL);
TYPEDRANGE(int, vel, 0, 127);
if (synth_leavesysex(md) == EAGAIN)
return (EAGAIN);
c[0] = 0x90 | (chn & 0x0f); /* Note on. */
c[1] = (u_char)note;
c[2] = (u_char)vel;
if (synth_prefixcmd(md, c[0]))
return (0);
return (md->synth.writeraw(md, c, 3, &lenw, 1));
}
static int
synth_reset(mididev_info *md)
{
synth_leavesysex(md);
return (0);
}
static int
synth_hwcontrol(mididev_info *md, u_char *event)
{
/* NOP. */
return (0);
}
static int
synth_loadpatch(mididev_info *md, int format, struct uio *buf, int offs, int count, int pmgr_flag)
{
struct sysex_info sysex;
synthdev_info *sd;
int unit, i, eox_seen, first_byte, left, src_offs, hdr_size, lenw;
u_char c[count];
unit = md->unit;
sd = &md->synth;
eox_seen = 0;
first_byte = 1;
hdr_size = offsetof(struct sysex_info, data);
if (synth_leavesysex(md) == EAGAIN)
return (EAGAIN);
if (synth_prefixcmd(md, 0xf0))
return (0);
if (format != SYSEX_PATCH) {
printf("synth_loadpatch: patch format 0x%x is invalid.\n", format);
return (EINVAL);
}
if (count < hdr_size) {
printf("synth_loadpatch: patch header is too short.\n");
return (EINVAL);
}
count -= hdr_size;
/* Copy the patch data. */
if (uiomove((caddr_t)&((char *)&sysex)[offs], hdr_size - offs, buf))
printf("synth_loadpatch: memory mangled?\n");
if (count < sysex.len) {
sysex.len = (long)count;
printf("synth_loadpatch: sysex record of %d bytes is too long, adjusted to %d bytes.\n", (int)sysex.len, count);
}
left = sysex.len;
src_offs = 0;
for (i = 0 ; i < left ; i++) {
uiomove((caddr_t)&c[i], 1, buf);
eox_seen = i > 0 && (c[i] & 0x80) != 0;
if (eox_seen && c[i] != 0xf7)
c[i] = 0xf7;
if (i == 0 && c[i] != 0x80) {
printf("synth_loadpatch: sysex does not begin with the status.\n");
return (EINVAL);
}
if (!first_byte && (c[i] & 0x80) != 0) {
md->synth.writeraw(md, c, i + 1, &lenw, 0);
return (0);
}
first_byte = 0;
}
if (!eox_seen) {
c[0] = 0xf7;
md->synth.writeraw(md, c, 1, &lenw, 0);
}
return (0);
}
static int
synth_panning(mididev_info *md, int chn, int pan)
{
/* NOP. */
return (0);
}
static int
synth_aftertouch(mididev_info *md, int chn, int press)
{
int unit, lenw;
synthdev_info *sd;
u_char c[2];
unit = md->unit;
sd = &md->synth;
if (press < 0 || press > 127 || chn < 0 || chn > 15)
return (EINVAL);
if (synth_leavesysex(md) == EAGAIN)
return (EAGAIN);
c[0] = 0xd0 | (chn & 0x0f); /* Channel Pressure. */
c[1] = (u_char)press;
if (synth_prefixcmd(md, c[0]))
return (0);
return (md->synth.writeraw(md, c, 2, &lenw, 1));
}
static int
synth_controller(mididev_info *md, int chn, int ctrlnum, int val)
{
int unit, lenw;
synthdev_info *sd;
u_char c[3];
unit = md->unit;
sd = &md->synth;
if (ctrlnum < 1 || ctrlnum > 127 || chn < 0 || chn > 15)
return (EINVAL);
if (synth_leavesysex(md) == EAGAIN)
return (EAGAIN);
c[0] = 0xb0 | (chn & 0x0f); /* Control Message. */
c[1] = (u_char)ctrlnum;
if (synth_prefixcmd(md, c[0]))
return (0);
return (md->synth.writeraw(md, c, 3, &lenw, 1));
}
static int
synth_patchmgr(mididev_info *md, struct patmgr_info *rec)
{
return (EINVAL);
}
static int
synth_bender(mididev_info *md, int chn, int val)
{
int unit, lenw;
synthdev_info *sd;
u_char c[3];
unit = md->unit;
sd = &md->synth;
if (val < 0 || val > 16383 || chn < 0 || chn > 15)
return (EINVAL);
if (synth_leavesysex(md) == EAGAIN)
return (EAGAIN);
c[0] = 0xe0 | (chn & 0x0f); /* Pitch bend. */
c[1] = (u_char)val & 0x7f;
c[2] = (u_char)(val >> 7) & 0x7f;
if (synth_prefixcmd(md, c[0]))
return (0);
return (md->synth.writeraw(md, c, 3, &lenw, 1));
}
static int
synth_allocvoice(mididev_info *md, int chn, int note, struct voice_alloc_info *alloc)
{
/* NOP. */
return (0);
}
static int
synth_setupvoice(mididev_info *md, int voice, int chn)
{
/* NOP. */
return (0);
}
static int
synth_sendsysex(mididev_info *md, u_char *sysex, int len)
{
int unit, i, lenw;
synthdev_info *sd;
u_char c[len];
unit = md->unit;
sd = &md->synth;
mtx_lock(&sd->status_mtx);
for (i = 0 ; i < len ; i++) {
switch (sysex[i]) {
case 0xf0:
/* Sysex begins. */
if (synth_prefixcmd(md, 0xf0)) {
mtx_unlock(&sd->status_mtx);
return (0);
}
sd->sysex_state = 1;
break;
case 0xf7:
/* Sysex ends. */
if (!sd->sysex_state) {
mtx_unlock(&sd->status_mtx);
return (0);
}
sd->sysex_state = 0;
break;
default:
if (!sd->sysex_state) {
mtx_unlock(&sd->status_mtx);
return (0);
}
if ((sysex[i] & 0x80) != 0) {
/* A status in a sysex? */
sysex[i] = 0xf7;
sd->sysex_state = 0;
}
break;
}
c[i] = sysex[i];
if (!sd->sysex_state)
break;
}
mtx_unlock(&sd->status_mtx);
return (md->synth.writeraw(md, c, i, &lenw, 1));
}
static int
synth_prefixcmd(mididev_info *md, int status)
{
/* NOP. */
return (0);
}
static int
synth_volumemethod(mididev_info *md, int mode)
{
/* NOP. */
return (0);
}
static int
synth_readraw(mididev_info *md, u_char *buf, int len, int *lenr, int nonblock)
{
int unit, ret;
if (md == NULL)
return (ENXIO);
if (lenr == NULL)
return (EINVAL);
*lenr = 0;
unit = md->unit;
if ((md->fflags & FREAD) == 0) {
MIDI_DEBUG(printf("synth_readraw: unit %d is not for reading.\n", unit));
return (EIO);
}
mtx_lock(&md->flagqueue_mtx);
/* Begin recording. */
if ((md->flags & MIDI_F_READING) == 0)
md->callback(md, MIDI_CB_START | MIDI_CB_RD);
if (nonblock) {
/* Have we got enough data to read? */
if (md->midi_dbuf_in.rl < len) {
mtx_unlock(&md->flagqueue_mtx);
return (EAGAIN);
}
}
ret = midibuf_seqread(&md->midi_dbuf_in, buf, len, lenr,
md->callback, md, MIDI_CB_START | MIDI_CB_RD,
&md->flagqueue_mtx);
mtx_unlock(&md->flagqueue_mtx);
return (ret);
}
static int
synth_writeraw(mididev_info *md, u_char *buf, int len, int *lenw, int nonblock)
{
int unit, ret;
if (md == NULL)
return (ENXIO);
if (lenw == NULL)
return (EINVAL);
*lenw = 0;
unit = md->unit;
if ((md->fflags & FWRITE) == 0) {
MIDI_DEBUG(printf("synth_writeraw: unit %d is not for writing.\n", unit));
return (EIO);
}
/* For nonblocking, have we got enough space to write? */
mtx_lock(&md->flagqueue_mtx);
if (nonblock && md->midi_dbuf_out.fl < len) {
/* Begin playing. */
md->callback(md, MIDI_CB_START | MIDI_CB_WR);
mtx_unlock(&md->flagqueue_mtx);
return (EAGAIN);
}
ret = midibuf_seqwrite(&md->midi_dbuf_out, buf, len, lenw,
md->callback, md, MIDI_CB_START | MIDI_CB_WR,
&md->flagqueue_mtx);
if (ret == 0)
/* Begin playing. */
md->callback(md, MIDI_CB_START | MIDI_CB_WR);
mtx_unlock(&md->flagqueue_mtx);
return (ret);
}
/*
* The functions below here are the libraries for the above ones.
*/
static int
synth_leavesysex(mididev_info *md)
{
int unit, lenw;
synthdev_info *sd;
u_char c;
unit = md->unit;
sd = &md->synth;
mtx_lock(&sd->status_mtx);
if (!sd->sysex_state) {
mtx_unlock(&sd->status_mtx);
return (0);
}
sd->sysex_state = 0;
mtx_unlock(&sd->status_mtx);
c = 0xf7;
return (md->synth.writeraw(md, &c, sizeof(c), &lenw, 1));
}