2005-12-11 15:16:03 +03:00
|
|
|
/* $NetBSD: opl.c,v 1.23 2005/12/11 12:21:28 christos Exp $ */
|
1998-08-18 01:16:09 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 1998 The NetBSD Foundation, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
1998-11-26 01:17:06 +03:00
|
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
2003-12-04 16:57:30 +03:00
|
|
|
* by Lennart Augustsson (augustss@NetBSD.org).
|
1998-08-18 01:16:09 +04:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
|
|
* must display the following acknowledgement:
|
|
|
|
* This product includes software developed by the NetBSD
|
|
|
|
* Foundation, Inc. and its contributors.
|
|
|
|
* 4. Neither the name of The NetBSD Foundation nor the names of its
|
|
|
|
* contributors may be used to endorse or promote products derived
|
|
|
|
* from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. 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 FOUNDATION 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The OPL3 (YMF262) manual can be found at
|
1999-05-23 20:07:04 +04:00
|
|
|
* ftp://ftp.yamahayst.com/Fax_Back_Doc/sound/YMF262.PDF
|
1998-08-18 01:16:09 +04:00
|
|
|
*/
|
|
|
|
|
2001-11-13 16:14:31 +03:00
|
|
|
#include <sys/cdefs.h>
|
2005-12-11 15:16:03 +03:00
|
|
|
__KERNEL_RCSID(0, "$NetBSD: opl.c,v 1.23 2005/12/11 12:21:28 christos Exp $");
|
2001-11-13 16:14:31 +03:00
|
|
|
|
1998-08-18 01:16:09 +04:00
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/systm.h>
|
|
|
|
#include <sys/errno.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <sys/syslog.h>
|
|
|
|
#include <sys/device.h>
|
|
|
|
#include <sys/select.h>
|
|
|
|
|
|
|
|
#include <machine/cpu.h>
|
|
|
|
#include <machine/bus.h>
|
|
|
|
|
|
|
|
#include <sys/audioio.h>
|
|
|
|
#include <sys/midiio.h>
|
|
|
|
#include <dev/audio_if.h>
|
|
|
|
|
|
|
|
#include <dev/midi_if.h>
|
|
|
|
#include <dev/midivar.h>
|
|
|
|
#include <dev/midisynvar.h>
|
|
|
|
|
|
|
|
#include <dev/ic/oplreg.h>
|
|
|
|
#include <dev/ic/oplvar.h>
|
|
|
|
|
|
|
|
#ifdef AUDIO_DEBUG
|
|
|
|
#define DPRINTF(x) if (opldebug) printf x
|
|
|
|
#define DPRINTFN(n,x) if (opldebug >= (n)) printf x
|
1998-08-23 02:54:11 +04:00
|
|
|
int opldebug = 0;
|
1998-08-18 01:16:09 +04:00
|
|
|
#else
|
|
|
|
#define DPRINTF(x)
|
|
|
|
#define DPRINTFN(n,x)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
struct real_voice {
|
|
|
|
u_int8_t voice_num;
|
|
|
|
u_int8_t voice_mode; /* 0=unavailable, 2=2 OP, 4=4 OP */
|
|
|
|
u_int8_t iooffs; /* I/O port (left or right side) */
|
|
|
|
u_int8_t op[4]; /* Operator offsets */
|
|
|
|
};
|
|
|
|
|
2001-01-18 23:28:15 +03:00
|
|
|
const struct opl_voice voicetab[] = {
|
1999-10-04 23:31:39 +04:00
|
|
|
/* No I/O offs OP1 OP2 OP3 OP4 */
|
|
|
|
/* --------------------------------------------- */
|
1998-08-26 17:08:10 +04:00
|
|
|
{ 0, OPL_L, {0x00, 0x03, 0x08, 0x0b}},
|
|
|
|
{ 1, OPL_L, {0x01, 0x04, 0x09, 0x0c}},
|
|
|
|
{ 2, OPL_L, {0x02, 0x05, 0x0a, 0x0d}},
|
|
|
|
|
|
|
|
{ 3, OPL_L, {0x08, 0x0b, 0x00, 0x00}},
|
|
|
|
{ 4, OPL_L, {0x09, 0x0c, 0x00, 0x00}},
|
|
|
|
{ 5, OPL_L, {0x0a, 0x0d, 0x00, 0x00}},
|
|
|
|
|
|
|
|
{ 6, OPL_L, {0x10, 0x13, 0x00, 0x00}},
|
|
|
|
{ 7, OPL_L, {0x11, 0x14, 0x00, 0x00}},
|
|
|
|
{ 8, OPL_L, {0x12, 0x15, 0x00, 0x00}},
|
|
|
|
|
|
|
|
{ 0, OPL_R, {0x00, 0x03, 0x08, 0x0b}},
|
|
|
|
{ 1, OPL_R, {0x01, 0x04, 0x09, 0x0c}},
|
|
|
|
{ 2, OPL_R, {0x02, 0x05, 0x0a, 0x0d}},
|
|
|
|
{ 3, OPL_R, {0x08, 0x0b, 0x00, 0x00}},
|
|
|
|
{ 4, OPL_R, {0x09, 0x0c, 0x00, 0x00}},
|
|
|
|
{ 5, OPL_R, {0x0a, 0x0d, 0x00, 0x00}},
|
|
|
|
|
|
|
|
{ 6, OPL_R, {0x10, 0x13, 0x00, 0x00}},
|
|
|
|
{ 7, OPL_R, {0x11, 0x14, 0x00, 0x00}},
|
|
|
|
{ 8, OPL_R, {0x12, 0x15, 0x00, 0x00}}
|
1998-08-18 01:16:09 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
static void opl_command(struct opl_softc *, int, int, int);
|
|
|
|
void opl_reset(struct opl_softc *);
|
|
|
|
void opl_freq_to_fnum (int freq, int *block, int *fnum);
|
|
|
|
|
2005-02-04 05:10:35 +03:00
|
|
|
int oplsyn_open(midisyn *ms, int);
|
|
|
|
void oplsyn_close(midisyn *);
|
|
|
|
void oplsyn_reset(void *);
|
|
|
|
void oplsyn_noteon(midisyn *, u_int32_t, u_int32_t, u_int32_t);
|
|
|
|
void oplsyn_noteoff(midisyn *, u_int32_t, u_int32_t, u_int32_t);
|
|
|
|
void oplsyn_keypressure(midisyn *, u_int32_t, u_int32_t, u_int32_t);
|
|
|
|
void oplsyn_ctlchange(midisyn *, u_int32_t, u_int32_t, u_int32_t);
|
|
|
|
void oplsyn_programchange(midisyn *, u_int32_t, u_int32_t);
|
|
|
|
void oplsyn_pitchbend(midisyn *, u_int32_t, u_int32_t, u_int32_t);
|
|
|
|
void oplsyn_loadpatch(midisyn *, struct sysex_info *, struct uio *);
|
|
|
|
|
|
|
|
|
|
|
|
void opl_set_op_reg(struct opl_softc *, int, int, int, u_char);
|
|
|
|
void opl_set_ch_reg(struct opl_softc *, int, int, u_char);
|
|
|
|
void opl_load_patch(struct opl_softc *, int);
|
|
|
|
u_int32_t opl_get_block_fnum(int freq);
|
|
|
|
int opl_calc_vol(int regbyte, int volume, int main_vol);
|
1998-08-18 01:16:09 +04:00
|
|
|
|
|
|
|
struct midisyn_methods opl3_midi = {
|
|
|
|
oplsyn_open,
|
|
|
|
oplsyn_close,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
oplsyn_noteon,
|
|
|
|
oplsyn_noteoff,
|
|
|
|
oplsyn_keypressure,
|
|
|
|
oplsyn_ctlchange,
|
1999-10-04 23:31:39 +04:00
|
|
|
oplsyn_programchange,
|
1998-08-18 01:16:09 +04:00
|
|
|
0,
|
|
|
|
oplsyn_pitchbend,
|
|
|
|
0
|
|
|
|
};
|
2003-02-09 12:28:21 +03:00
|
|
|
|
1998-08-18 01:16:09 +04:00
|
|
|
void
|
|
|
|
opl_attach(sc)
|
|
|
|
struct opl_softc *sc;
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!opl_find(sc)) {
|
|
|
|
printf("\nopl: find failed\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
sc->syn.mets = &opl3_midi;
|
2004-04-22 04:17:10 +04:00
|
|
|
snprintf(sc->syn.name, sizeof(sc->syn.name), "%sYamaha OPL%d",
|
|
|
|
sc->syn.name, sc->model);
|
1998-08-18 01:16:09 +04:00
|
|
|
sc->syn.data = sc;
|
|
|
|
sc->syn.nvoice = sc->model == OPL_2 ? OPL2_NVOICE : OPL3_NVOICE;
|
|
|
|
sc->syn.flags = MS_DOALLOC | MS_FREQXLATE;
|
1998-08-26 16:10:22 +04:00
|
|
|
midisyn_attach(&sc->mididev, &sc->syn);
|
2003-02-09 12:28:21 +03:00
|
|
|
|
1998-08-18 01:16:09 +04:00
|
|
|
/* Set up voice table */
|
|
|
|
for (i = 0; i < OPL3_NVOICE; i++)
|
|
|
|
sc->voices[i] = voicetab[i];
|
|
|
|
|
2001-11-04 09:44:31 +03:00
|
|
|
opl_reset(sc);
|
|
|
|
|
|
|
|
printf(": model OPL%d", sc->model);
|
|
|
|
|
2001-10-23 17:09:43 +04:00
|
|
|
/* Set up panpot */
|
2001-11-04 09:44:31 +03:00
|
|
|
sc->panl = OPL_VOICE_TO_LEFT;
|
|
|
|
sc->panr = OPL_VOICE_TO_RIGHT;
|
|
|
|
if (sc->model == OPL_3 &&
|
|
|
|
sc->mididev.dev.dv_cfdata->cf_flags & OPL_FLAGS_SWAP_LR) {
|
|
|
|
sc->panl = OPL_VOICE_TO_RIGHT;
|
|
|
|
sc->panr = OPL_VOICE_TO_LEFT;
|
|
|
|
printf(": LR swapped");
|
|
|
|
}
|
2001-10-23 17:09:43 +04:00
|
|
|
|
2001-11-04 09:44:31 +03:00
|
|
|
printf("\n");
|
1998-08-18 01:16:09 +04:00
|
|
|
|
2001-09-29 17:56:04 +04:00
|
|
|
sc->sc_mididev =
|
|
|
|
midi_attach_mi(&midisyn_hw_if, &sc->syn, &sc->mididev.dev);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
opl_detach(sc, flags)
|
|
|
|
struct opl_softc *sc;
|
|
|
|
int flags;
|
|
|
|
{
|
|
|
|
int rv = 0;
|
|
|
|
|
|
|
|
if (sc->sc_mididev != NULL)
|
|
|
|
rv = config_detach(sc->sc_mididev, flags);
|
|
|
|
|
|
|
|
return(rv);
|
1998-08-18 01:16:09 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
opl_command(sc, offs, addr, data)
|
|
|
|
struct opl_softc *sc;
|
|
|
|
int offs;
|
|
|
|
int addr, data;
|
|
|
|
{
|
2003-02-09 12:28:21 +03:00
|
|
|
DPRINTFN(4, ("opl_command: sc=%p, offs=%d addr=0x%02x data=0x%02x\n",
|
1998-08-18 01:16:09 +04:00
|
|
|
sc, offs, addr, data));
|
|
|
|
offs += sc->offs;
|
|
|
|
bus_space_write_1(sc->iot, sc->ioh, OPL_ADDR+offs, addr);
|
1998-08-26 16:10:22 +04:00
|
|
|
if (sc->model == OPL_2)
|
1998-08-18 01:16:09 +04:00
|
|
|
delay(10);
|
1998-08-26 16:10:22 +04:00
|
|
|
else
|
1998-09-13 08:41:34 +04:00
|
|
|
delay(6);
|
1998-08-18 01:16:09 +04:00
|
|
|
bus_space_write_1(sc->iot, sc->ioh, OPL_DATA+offs, data);
|
1998-08-26 16:10:22 +04:00
|
|
|
if (sc->model == OPL_2)
|
1998-08-18 01:16:09 +04:00
|
|
|
delay(30);
|
1998-08-26 16:10:22 +04:00
|
|
|
else
|
1998-09-13 08:41:34 +04:00
|
|
|
delay(6);
|
1998-08-18 01:16:09 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
opl_find(sc)
|
|
|
|
struct opl_softc *sc;
|
|
|
|
{
|
|
|
|
u_int8_t status1, status2;
|
|
|
|
|
1998-08-23 02:54:11 +04:00
|
|
|
DPRINTFN(2,("opl_find: ioh=0x%x\n", (int)sc->ioh));
|
2003-11-02 14:07:44 +03:00
|
|
|
sc->model = OPL_2; /* worst case assumption */
|
1998-08-18 01:16:09 +04:00
|
|
|
|
|
|
|
/* Reset timers 1 and 2 */
|
2003-02-09 12:28:21 +03:00
|
|
|
opl_command(sc, OPL_L, OPL_TIMER_CONTROL,
|
1998-08-18 01:16:09 +04:00
|
|
|
OPL_TIMER1_MASK | OPL_TIMER2_MASK);
|
|
|
|
/* Reset the IRQ of the FM chip */
|
1998-08-26 17:08:10 +04:00
|
|
|
opl_command(sc, OPL_L, OPL_TIMER_CONTROL, OPL_IRQ_RESET);
|
1998-08-18 01:16:09 +04:00
|
|
|
|
|
|
|
/* get status bits */
|
1998-08-26 17:08:10 +04:00
|
|
|
status1 = bus_space_read_1(sc->iot,sc->ioh,OPL_STATUS+OPL_L+sc->offs);
|
1998-08-18 01:16:09 +04:00
|
|
|
|
1998-08-26 17:08:10 +04:00
|
|
|
opl_command(sc, OPL_L, OPL_TIMER1, -2); /* wait 2 ticks */
|
|
|
|
opl_command(sc, OPL_L, OPL_TIMER_CONTROL, /* start timer1 */
|
1998-08-18 01:16:09 +04:00
|
|
|
OPL_TIMER1_START | OPL_TIMER2_MASK);
|
|
|
|
delay(1000); /* wait for timer to expire */
|
|
|
|
|
|
|
|
/* get status bits again */
|
1998-08-26 17:08:10 +04:00
|
|
|
status2 = bus_space_read_1(sc->iot,sc->ioh,OPL_STATUS+OPL_L+sc->offs);
|
1998-08-18 01:16:09 +04:00
|
|
|
|
2003-02-09 12:28:21 +03:00
|
|
|
opl_command(sc, OPL_L, OPL_TIMER_CONTROL,
|
1998-08-18 01:16:09 +04:00
|
|
|
OPL_TIMER1_MASK | OPL_TIMER2_MASK);
|
1998-08-26 17:08:10 +04:00
|
|
|
opl_command(sc, OPL_L, OPL_TIMER_CONTROL, OPL_IRQ_RESET);
|
1998-08-18 01:16:09 +04:00
|
|
|
|
1998-08-23 02:54:11 +04:00
|
|
|
DPRINTFN(2,("opl_find: %02x %02x\n", status1, status2));
|
1998-08-18 01:16:09 +04:00
|
|
|
|
|
|
|
if ((status1 & OPL_STATUS_MASK) != 0 ||
|
|
|
|
(status2 & OPL_STATUS_MASK) != (OPL_STATUS_IRQ | OPL_STATUS_FT1))
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
switch(status1) {
|
|
|
|
case 0x00:
|
|
|
|
case 0x0f:
|
|
|
|
sc->model = OPL_3;
|
|
|
|
break;
|
|
|
|
case 0x06:
|
|
|
|
sc->model = OPL_2;
|
|
|
|
break;
|
|
|
|
default:
|
1999-10-04 23:31:39 +04:00
|
|
|
return (0);
|
1998-08-18 01:16:09 +04:00
|
|
|
}
|
|
|
|
|
2003-02-09 12:28:21 +03:00
|
|
|
DPRINTFN(2,("opl_find: OPL%d at 0x%x detected\n",
|
1998-08-18 01:16:09 +04:00
|
|
|
sc->model, (int)sc->ioh));
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
|
2003-02-09 12:28:21 +03:00
|
|
|
void
|
1998-08-18 01:16:09 +04:00
|
|
|
opl_set_op_reg(sc, base, voice, op, value)
|
|
|
|
struct opl_softc *sc;
|
|
|
|
int base;
|
|
|
|
int voice;
|
|
|
|
int op;
|
|
|
|
u_char value;
|
|
|
|
{
|
|
|
|
struct opl_voice *v = &sc->voices[voice];
|
|
|
|
opl_command(sc, v->iooffs, base + v->op[op], value);
|
|
|
|
}
|
|
|
|
|
2003-02-09 12:28:21 +03:00
|
|
|
void
|
1998-08-18 01:16:09 +04:00
|
|
|
opl_set_ch_reg(sc, base, voice, value)
|
|
|
|
struct opl_softc *sc;
|
|
|
|
int base;
|
|
|
|
int voice;
|
|
|
|
u_char value;
|
|
|
|
{
|
|
|
|
struct opl_voice *v = &sc->voices[voice];
|
|
|
|
opl_command(sc, v->iooffs, base + v->voiceno, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-02-09 12:28:21 +03:00
|
|
|
void
|
1998-08-18 01:16:09 +04:00
|
|
|
opl_load_patch(sc, v)
|
|
|
|
struct opl_softc *sc;
|
|
|
|
int v;
|
|
|
|
{
|
2001-01-18 23:28:15 +03:00
|
|
|
const struct opl_operators *p = sc->voices[v].patch;
|
1998-08-18 01:16:09 +04:00
|
|
|
|
|
|
|
opl_set_op_reg(sc, OPL_AM_VIB, v, 0, p->ops[OO_CHARS+0]);
|
|
|
|
opl_set_op_reg(sc, OPL_AM_VIB, v, 1, p->ops[OO_CHARS+1]);
|
|
|
|
opl_set_op_reg(sc, OPL_KSL_LEVEL, v, 0, p->ops[OO_KSL_LEV+0]);
|
|
|
|
opl_set_op_reg(sc, OPL_KSL_LEVEL, v, 1, p->ops[OO_KSL_LEV+1]);
|
|
|
|
opl_set_op_reg(sc, OPL_ATTACK_DECAY, v, 0, p->ops[OO_ATT_DEC+0]);
|
|
|
|
opl_set_op_reg(sc, OPL_ATTACK_DECAY, v, 1, p->ops[OO_ATT_DEC+1]);
|
|
|
|
opl_set_op_reg(sc, OPL_SUSTAIN_RELEASE, v, 0, p->ops[OO_SUS_REL+0]);
|
|
|
|
opl_set_op_reg(sc, OPL_SUSTAIN_RELEASE, v, 1, p->ops[OO_SUS_REL+1]);
|
|
|
|
opl_set_op_reg(sc, OPL_WAVE_SELECT, v, 0, p->ops[OO_WAV_SEL+0]);
|
|
|
|
opl_set_op_reg(sc, OPL_WAVE_SELECT, v, 1, p->ops[OO_WAV_SEL+1]);
|
|
|
|
opl_set_ch_reg(sc, OPL_FEEDBACK_CONNECTION, v, p->ops[OO_FB_CONN]);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define OPL_FNUM_FAIL 0xffff
|
|
|
|
u_int32_t
|
|
|
|
opl_get_block_fnum(freq)
|
|
|
|
int freq;
|
|
|
|
{
|
|
|
|
u_int32_t f_num = freq / 3125;
|
|
|
|
u_int32_t block = 0;
|
|
|
|
|
1999-10-04 23:31:39 +04:00
|
|
|
while (f_num > 0x3ff && block < 8) {
|
1998-08-18 01:16:09 +04:00
|
|
|
block++;
|
|
|
|
f_num >>= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (block > 7)
|
|
|
|
return (OPL_FNUM_FAIL);
|
|
|
|
else
|
|
|
|
return ((block << 10) | f_num);
|
2003-02-09 12:28:21 +03:00
|
|
|
}
|
1998-08-18 01:16:09 +04:00
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
opl_reset(sc)
|
|
|
|
struct opl_softc *sc;
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 1; i <= OPL_MAXREG; i++)
|
1998-08-26 17:08:10 +04:00
|
|
|
opl_command(sc, OPL_L, OPL_KEYON_BLOCK + i, 0);
|
1998-08-18 01:16:09 +04:00
|
|
|
|
1998-08-26 17:08:10 +04:00
|
|
|
opl_command(sc, OPL_L, OPL_TEST, OPL_ENABLE_WAVE_SELECT);
|
|
|
|
opl_command(sc, OPL_L, OPL_PERCUSSION, 0);
|
1998-08-26 16:10:22 +04:00
|
|
|
if (sc->model == OPL_3) {
|
1998-08-26 17:08:10 +04:00
|
|
|
opl_command(sc, OPL_R, OPL_MODE, OPL3_ENABLE);
|
|
|
|
opl_command(sc, OPL_R,OPL_CONNECTION_SELECT,OPL_NOCONNECTION);
|
1998-08-26 16:10:22 +04:00
|
|
|
}
|
1998-08-18 01:16:09 +04:00
|
|
|
|
|
|
|
sc->volume = 64;
|
2003-02-09 12:28:21 +03:00
|
|
|
|
|
|
|
for (i = 0; i < MIDI_MAX_CHANS; i++)
|
|
|
|
sc->pan[i] = OPL_VOICE_TO_LEFT | OPL_VOICE_TO_RIGHT;
|
1998-08-18 01:16:09 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
oplsyn_open(ms, flags)
|
|
|
|
midisyn *ms;
|
|
|
|
int flags;
|
|
|
|
{
|
1998-08-26 16:10:22 +04:00
|
|
|
struct opl_softc *sc = ms->data;
|
|
|
|
|
1998-08-18 01:16:09 +04:00
|
|
|
DPRINTFN(2, ("oplsyn_open: %d\n", flags));
|
1998-08-26 16:10:22 +04:00
|
|
|
|
1999-10-05 07:29:22 +04:00
|
|
|
#ifndef AUDIO_NO_POWER_CTL
|
|
|
|
if (sc->powerctl)
|
|
|
|
sc->powerctl(sc->powerarg, 1);
|
|
|
|
#endif
|
1998-08-18 01:16:09 +04:00
|
|
|
opl_reset(ms->data);
|
1998-08-26 16:10:22 +04:00
|
|
|
if (sc->spkrctl)
|
|
|
|
sc->spkrctl(sc->spkrarg, 1);
|
1998-08-18 01:16:09 +04:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
oplsyn_close(ms)
|
|
|
|
midisyn *ms;
|
|
|
|
{
|
|
|
|
struct opl_softc *sc = ms->data;
|
|
|
|
|
|
|
|
DPRINTFN(2, ("oplsyn_close:\n"));
|
1998-09-13 08:41:34 +04:00
|
|
|
|
1998-08-18 01:16:09 +04:00
|
|
|
/*opl_reset(ms->data);*/
|
1998-08-26 16:10:22 +04:00
|
|
|
if (sc->spkrctl)
|
|
|
|
sc->spkrctl(sc->spkrarg, 0);
|
1999-10-05 07:29:22 +04:00
|
|
|
#ifndef AUDIO_NO_POWER_CTL
|
|
|
|
if (sc->powerctl)
|
|
|
|
sc->powerctl(sc->powerarg, 0);
|
|
|
|
#endif
|
1998-08-18 01:16:09 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
void
|
|
|
|
oplsyn_getinfo(addr, sd)
|
|
|
|
void *addr;
|
|
|
|
struct synth_dev *sd;
|
|
|
|
{
|
|
|
|
struct opl_softc *sc = addr;
|
|
|
|
|
|
|
|
sd->name = sc->model == OPL_2 ? "Yamaha OPL2" : "Yamaha OPL3";
|
|
|
|
sd->type = SYNTH_TYPE_FM;
|
2003-02-09 12:28:21 +03:00
|
|
|
sd->subtype = sc->model == OPL_2 ? SYNTH_SUB_FM_TYPE_ADLIB
|
1998-08-18 01:16:09 +04:00
|
|
|
: SYNTH_SUB_FM_TYPE_OPL3;
|
|
|
|
sd->capabilities = 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void
|
|
|
|
oplsyn_reset(addr)
|
|
|
|
void *addr;
|
|
|
|
{
|
|
|
|
struct opl_softc *sc = addr;
|
|
|
|
DPRINTFN(3, ("oplsyn_reset:\n"));
|
|
|
|
opl_reset(sc);
|
|
|
|
}
|
|
|
|
|
2001-01-18 23:28:15 +03:00
|
|
|
const int8_t opl_volume_table[128] =
|
1998-08-18 01:16:09 +04:00
|
|
|
{-64, -48, -40, -35, -32, -29, -27, -26,
|
|
|
|
-24, -23, -21, -20, -19, -18, -18, -17,
|
|
|
|
-16, -15, -15, -14, -13, -13, -12, -12,
|
|
|
|
-11, -11, -10, -10, -10, -9, -9, -8,
|
|
|
|
-8, -8, -7, -7, -7, -6, -6, -6,
|
|
|
|
-5, -5, -5, -5, -4, -4, -4, -4,
|
|
|
|
-3, -3, -3, -3, -2, -2, -2, -2,
|
|
|
|
-2, -1, -1, -1, -1, 0, 0, 0,
|
|
|
|
0, 0, 0, 1, 1, 1, 1, 1,
|
|
|
|
1, 2, 2, 2, 2, 2, 2, 2,
|
|
|
|
3, 3, 3, 3, 3, 3, 3, 4,
|
|
|
|
4, 4, 4, 4, 4, 4, 4, 5,
|
|
|
|
5, 5, 5, 5, 5, 5, 5, 5,
|
|
|
|
6, 6, 6, 6, 6, 6, 6, 6,
|
|
|
|
6, 7, 7, 7, 7, 7, 7, 7,
|
|
|
|
7, 7, 7, 8, 8, 8, 8, 8};
|
|
|
|
|
|
|
|
int
|
|
|
|
opl_calc_vol(regbyte, volume, mainvol)
|
2003-02-09 12:28:21 +03:00
|
|
|
int regbyte;
|
1998-08-18 01:16:09 +04:00
|
|
|
int volume;
|
|
|
|
int mainvol;
|
|
|
|
{
|
|
|
|
int level = ~regbyte & OPL_TOTAL_LEVEL_MASK;
|
|
|
|
|
|
|
|
if (mainvol > 127)
|
|
|
|
mainvol = 127;
|
|
|
|
|
|
|
|
volume = (volume * mainvol) / 127;
|
|
|
|
|
|
|
|
if (level)
|
|
|
|
level += opl_volume_table[volume];
|
|
|
|
|
|
|
|
if (level > OPL_TOTAL_LEVEL_MASK)
|
|
|
|
level = OPL_TOTAL_LEVEL_MASK;
|
|
|
|
if (level < 0)
|
|
|
|
level = 0;
|
|
|
|
|
|
|
|
return (~level & OPL_TOTAL_LEVEL_MASK);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
oplsyn_noteon(ms, voice, freq, vel)
|
|
|
|
midisyn *ms;
|
|
|
|
u_int32_t voice, freq, vel;
|
|
|
|
{
|
|
|
|
struct opl_softc *sc = ms->data;
|
|
|
|
struct opl_voice *v;
|
2001-01-18 23:28:15 +03:00
|
|
|
const struct opl_operators *p;
|
1998-08-18 01:16:09 +04:00
|
|
|
u_int32_t block_fnum;
|
|
|
|
int mult;
|
|
|
|
int c_mult, m_mult;
|
2001-10-23 17:09:43 +04:00
|
|
|
u_int32_t chan;
|
1998-08-26 17:08:10 +04:00
|
|
|
u_int8_t chars0, chars1, ksl0, ksl1, fbc;
|
1998-08-18 01:16:09 +04:00
|
|
|
u_int8_t r20m, r20c, r40m, r40c, rA0, rB0;
|
|
|
|
u_int8_t vol0, vol1;
|
|
|
|
|
2003-02-09 12:28:21 +03:00
|
|
|
DPRINTFN(3, ("oplsyn_noteon: %p %d %d\n", sc, voice,
|
1998-08-18 01:16:09 +04:00
|
|
|
MIDISYN_FREQ_TO_HZ(freq)));
|
|
|
|
|
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
if (voice < 0 || voice >= sc->syn.nvoice) {
|
|
|
|
printf("oplsyn_noteon: bad voice %d\n", voice);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
/* Turn off old note */
|
|
|
|
opl_set_op_reg(sc, OPL_KSL_LEVEL, voice, 0, 0xff);
|
|
|
|
opl_set_op_reg(sc, OPL_KSL_LEVEL, voice, 1, 0xff);
|
|
|
|
opl_set_ch_reg(sc, OPL_KEYON_BLOCK, voice, 0);
|
|
|
|
|
|
|
|
v = &sc->voices[voice];
|
2003-02-09 12:28:21 +03:00
|
|
|
|
2001-10-23 17:09:43 +04:00
|
|
|
chan = MS_GETCHAN(&ms->voices[voice]);
|
|
|
|
p = &opl2_instrs[ms->pgms[chan]];
|
1998-08-18 01:16:09 +04:00
|
|
|
v->patch = p;
|
|
|
|
opl_load_patch(sc, voice);
|
|
|
|
|
|
|
|
mult = 1;
|
|
|
|
for (;;) {
|
|
|
|
block_fnum = opl_get_block_fnum(freq / mult);
|
|
|
|
if (block_fnum != OPL_FNUM_FAIL)
|
|
|
|
break;
|
|
|
|
mult *= 2;
|
|
|
|
if (mult == 16)
|
|
|
|
mult = 15;
|
|
|
|
}
|
|
|
|
|
|
|
|
chars0 = p->ops[OO_CHARS+0];
|
|
|
|
chars1 = p->ops[OO_CHARS+1];
|
|
|
|
m_mult = (chars0 & OPL_MULTIPLE_MASK) * mult;
|
|
|
|
c_mult = (chars1 & OPL_MULTIPLE_MASK) * mult;
|
|
|
|
if ((block_fnum == OPL_FNUM_FAIL) || (m_mult > 15) || (c_mult > 15)) {
|
2000-08-13 02:24:26 +04:00
|
|
|
printf("oplsyn_noteon: frequency out of range %d\n",
|
1998-08-18 01:16:09 +04:00
|
|
|
MIDISYN_FREQ_TO_HZ(freq));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
r20m = (chars0 &~ OPL_MULTIPLE_MASK) | m_mult;
|
|
|
|
r20c = (chars1 &~ OPL_MULTIPLE_MASK) | c_mult;
|
|
|
|
|
|
|
|
/* 2 voice */
|
|
|
|
ksl0 = p->ops[OO_KSL_LEV+0];
|
|
|
|
ksl1 = p->ops[OO_KSL_LEV+1];
|
|
|
|
if (p->ops[OO_FB_CONN] & 0x01) {
|
|
|
|
vol0 = opl_calc_vol(ksl0, vel, sc->volume);
|
|
|
|
vol1 = opl_calc_vol(ksl1, vel, sc->volume);
|
|
|
|
} else {
|
|
|
|
vol0 = ksl0;
|
|
|
|
vol1 = opl_calc_vol(ksl1, vel, sc->volume);
|
|
|
|
}
|
|
|
|
r40m = (ksl0 & OPL_KSL_MASK) | vol0;
|
|
|
|
r40c = (ksl1 & OPL_KSL_MASK) | vol1;
|
|
|
|
|
|
|
|
rA0 = block_fnum & 0xFF;
|
|
|
|
rB0 = (block_fnum >> 8) | OPL_KEYON_BIT;
|
|
|
|
|
|
|
|
v->rB0 = rB0;
|
|
|
|
|
1998-08-26 17:08:10 +04:00
|
|
|
fbc = p->ops[OO_FB_CONN];
|
|
|
|
if (sc->model == OPL_3) {
|
|
|
|
fbc &= ~OPL_STEREO_BITS;
|
2001-10-23 17:09:43 +04:00
|
|
|
fbc |= sc->pan[chan];
|
1998-08-26 17:08:10 +04:00
|
|
|
}
|
|
|
|
opl_set_ch_reg(sc, OPL_FEEDBACK_CONNECTION, voice, fbc);
|
|
|
|
|
1998-08-18 01:16:09 +04:00
|
|
|
opl_set_op_reg(sc, OPL_AM_VIB, voice, 0, r20m);
|
|
|
|
opl_set_op_reg(sc, OPL_AM_VIB, voice, 1, r20c);
|
|
|
|
opl_set_op_reg(sc, OPL_KSL_LEVEL, voice, 0, r40m);
|
|
|
|
opl_set_op_reg(sc, OPL_KSL_LEVEL, voice, 1, r40c);
|
|
|
|
opl_set_ch_reg(sc, OPL_FNUM_LOW, voice, rA0);
|
|
|
|
opl_set_ch_reg(sc, OPL_KEYON_BLOCK, voice, rB0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
oplsyn_noteoff(ms, voice, note, vel)
|
|
|
|
midisyn *ms;
|
|
|
|
u_int32_t voice, note, vel;
|
|
|
|
{
|
|
|
|
struct opl_softc *sc = ms->data;
|
|
|
|
struct opl_voice *v;
|
|
|
|
|
2003-02-09 12:28:21 +03:00
|
|
|
DPRINTFN(3, ("oplsyn_noteoff: %p %d %d\n", sc, voice,
|
1998-08-18 01:16:09 +04:00
|
|
|
MIDISYN_FREQ_TO_HZ(note)));
|
|
|
|
|
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
if (voice < 0 || voice >= sc->syn.nvoice) {
|
|
|
|
printf("oplsyn_noteoff: bad voice %d\n", voice);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
v = &sc->voices[voice];
|
|
|
|
opl_set_ch_reg(sc, 0xB0, voice, v->rB0 & ~OPL_KEYON_BIT);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
oplsyn_keypressure(ms, voice, note, vel)
|
|
|
|
midisyn *ms;
|
|
|
|
u_int32_t voice, note, vel;
|
|
|
|
{
|
|
|
|
#ifdef AUDIO_DEBUG
|
|
|
|
struct opl_softc *sc = ms->data;
|
|
|
|
DPRINTFN(1, ("oplsyn_keypressure: %p %d\n", sc, note));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2001-10-23 17:09:43 +04:00
|
|
|
oplsyn_ctlchange(ms, chan, parm, w14)
|
1998-08-18 01:16:09 +04:00
|
|
|
midisyn *ms;
|
2001-10-23 17:09:43 +04:00
|
|
|
u_int32_t chan, parm, w14;
|
1998-08-18 01:16:09 +04:00
|
|
|
{
|
|
|
|
struct opl_softc *sc = ms->data;
|
2001-10-23 17:09:43 +04:00
|
|
|
|
2001-10-23 21:58:15 +04:00
|
|
|
DPRINTFN(1, ("oplsyn_ctlchange: %p %d\n", sc, chan));
|
2001-10-23 17:09:43 +04:00
|
|
|
switch (parm) {
|
|
|
|
case MIDI_CTRL_PAN_MSB:
|
|
|
|
sc->pan[chan] =
|
2001-11-04 09:44:31 +03:00
|
|
|
(w14 <= OPL_MIDI_CENTER_MAX ? sc->panl : 0) |
|
|
|
|
(w14 >= OPL_MIDI_CENTER_MIN ? sc->panr : 0);
|
2001-10-23 17:09:43 +04:00
|
|
|
break;
|
|
|
|
}
|
1998-08-18 01:16:09 +04:00
|
|
|
}
|
|
|
|
|
1999-10-04 23:31:39 +04:00
|
|
|
/* PROGRAM CHANGE midi event: */
|
|
|
|
void
|
|
|
|
oplsyn_programchange(ms, chan, prog)
|
|
|
|
midisyn *ms;
|
|
|
|
u_int32_t chan;
|
|
|
|
u_int32_t prog;
|
|
|
|
{
|
|
|
|
/* sanity checks */
|
|
|
|
if (chan >= MIDI_MAX_CHANS || prog >= OPL_NINSTR)
|
|
|
|
return;
|
|
|
|
|
|
|
|
ms->pgms[chan] = prog;
|
|
|
|
}
|
|
|
|
|
1998-08-18 01:16:09 +04:00
|
|
|
void
|
|
|
|
oplsyn_pitchbend(ms, voice, parm, x)
|
|
|
|
midisyn *ms;
|
|
|
|
u_int32_t voice, parm, x;
|
|
|
|
{
|
|
|
|
#ifdef AUDIO_DEBUG
|
|
|
|
struct opl_softc *sc = ms->data;
|
|
|
|
DPRINTFN(1, ("oplsyn_pitchbend: %p %d\n", sc, voice));
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
oplsyn_loadpatch(ms, sysex, uio)
|
|
|
|
midisyn *ms;
|
|
|
|
struct sysex_info *sysex;
|
|
|
|
struct uio *uio;
|
|
|
|
{
|
|
|
|
#if 0
|
|
|
|
struct opl_softc *sc = ms->data;
|
|
|
|
struct sbi_instrument ins;
|
|
|
|
|
|
|
|
DPRINTFN(1, ("oplsyn_loadpatch: %p\n", sc));
|
|
|
|
|
|
|
|
memcpy(&ins, sysex, sizeof *sysex);
|
|
|
|
if (uio->uio_resid >= sizeof ins - sizeof *sysex)
|
|
|
|
return EINVAL;
|
|
|
|
uiomove((char *)&ins + sizeof *sysex, sizeof ins - sizeof *sysex, uio);
|
|
|
|
/* XXX */
|
|
|
|
#endif
|
|
|
|
}
|