mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-20 23:36:38 +00:00
ccf0b720ba
Original commit message from CVS: * gst/nsf/Makefile.am: * gst/nsf/fds_snd.c: * gst/nsf/mmc5_snd.c: * gst/nsf/nsf.c: * gst/nsf/types.h: * gst/nsf/vrc7_snd.c: * gst/nsf/vrcvisnd.c: * gst/nsf/memguard.c: * gst/nsf/memguard.h: Remove memguard again and apply hopefully all previously dropped local patches. Should be really better than the old version now.
390 lines
13 KiB
C
390 lines
13 KiB
C
/*
|
|
** Nofrendo (c) 1998-2000 Matthew Conte (matt@conte.com)
|
|
**
|
|
**
|
|
** This program is free software; you can redistribute it and/or
|
|
** modify it under the terms of version 2 of the GNU Library General
|
|
** Public License as published by the Free Software Foundation.
|
|
**
|
|
** This program is distributed in the hope that it will be useful,
|
|
** but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
** Library General Public License for more details. To obtain a
|
|
** copy of the GNU Library General Public License, write to the Free
|
|
** Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
**
|
|
** Any permitted reproduction of these routines, in whole or in part,
|
|
** must bear this legend.
|
|
**
|
|
**
|
|
** vrc7_snd.c
|
|
**
|
|
** VRCVII sound hardware emulation
|
|
** Thanks to Charles MacDonald (cgfm2@hooked.net) for donating code.
|
|
** $Id$
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include "types.h"
|
|
#include "vrc7_snd.h"
|
|
#include "fmopl.h"
|
|
|
|
|
|
static int buflen;
|
|
static int16 *buffer;
|
|
|
|
#define OPL_WRITE(opl, r, d) \
|
|
{ \
|
|
OPLWrite((opl)->ym3812, 0, (r)); \
|
|
OPLWrite((opl)->ym3812, 1, (d)); \
|
|
}
|
|
|
|
static vrc7_t vrc7;
|
|
|
|
/* Fixed instrument settings, from MAME's YM2413 emulation */
|
|
/* This might need some tweaking... */
|
|
unsigned char table[16][11] = {
|
|
/* 20 23 40 43 60 63 80 83 E0 E3 C0 */
|
|
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
|
|
|
|
/* MAME */
|
|
{0x01, 0x22, 0x23, 0x07, 0xF0, 0xF0, 0x07, 0x18, 0x00, 0x00, 0x00}, /* Violin */
|
|
{0x23, 0x01, 0x68, 0x05, 0xF2, 0x74, 0x6C, 0x89, 0x00, 0x00, 0x00}, /* Acoustic Guitar(steel) */
|
|
{0x13, 0x11, 0x25, 0x00, 0xD2, 0xB2, 0xF4, 0xF4, 0x00, 0x00, 0x00}, /* Acoustic Grand */
|
|
{0x22, 0x21, 0x1B, 0x05, 0xC0, 0xA1, 0x18, 0x08, 0x00, 0x00, 0x00}, /* Flute */
|
|
{0x22, 0x21, 0x2C, 0x03, 0xD2, 0xA1, 0x18, 0x57, 0x00, 0x00, 0x00}, /* Clarinet */
|
|
{0x01, 0x22, 0xBA, 0x01, 0xF1, 0xF1, 0x1E, 0x04, 0x00, 0x00, 0x00}, /* Oboe */
|
|
{0x21, 0x21, 0x28, 0x06, 0xF1, 0xF1, 0x6B, 0x3E, 0x00, 0x00, 0x00}, /* Trumpet */
|
|
{0x27, 0x21, 0x60, 0x00, 0xF0, 0xF0, 0x0D, 0x0F, 0x00, 0x00, 0x00}, /* Church Organ */
|
|
{0x20, 0x21, 0x2B, 0x06, 0x85, 0xF1, 0x6D, 0x89, 0x00, 0x00, 0x00}, /* French Horn */
|
|
{0x01, 0x21, 0xBF, 0x02, 0x53, 0x62, 0x5F, 0xAE, 0x01, 0x00, 0x00}, /* Synth Voice */
|
|
{0x23, 0x21, 0x70, 0x07, 0xD4, 0xA3, 0x4E, 0x64, 0x01, 0x00, 0x00}, /* Harpsichord */
|
|
{0x2B, 0x21, 0xA4, 0x07, 0xF6, 0x93, 0x5C, 0x4D, 0x00, 0x00, 0x00}, /* Vibraphone */
|
|
{0x21, 0x23, 0xAD, 0x07, 0x77, 0xF1, 0x18, 0x37, 0x00, 0x00, 0x00}, /* Synth Bass 1 */
|
|
{0x21, 0x21, 0x2A, 0x03, 0xF3, 0xE2, 0x29, 0x46, 0x00, 0x00, 0x00}, /* Acoustic Bass */
|
|
{0x21, 0x23, 0x37, 0x03, 0xF3, 0xE2, 0x29, 0x46, 0x00, 0x00, 0x00}, /* Electric Guitar(clean) */
|
|
|
|
|
|
#if 0
|
|
/* Horton, try 1 */
|
|
{0x05, 0x03, 0x10, 0x06, 0x74, 0xA1, 0x13, 0xF4, 0x00, 0x00, 0x00},
|
|
{0x05, 0x01, 0x16, 0x00, 0xF9, 0xA2, 0x15, 0xF5, 0x00, 0x00, 0x00},
|
|
{0x01, 0x41, 0x11, 0x00, 0xA0, 0xA0, 0x83, 0x95, 0x00, 0x00, 0x00},
|
|
{0x01, 0x41, 0x17, 0x00, 0x60, 0xF0, 0x83, 0x95, 0x00, 0x00, 0x00},
|
|
{0x24, 0x41, 0x1F, 0x00, 0x50, 0xB0, 0x94, 0x94, 0x00, 0x00, 0x00},
|
|
{0x05, 0x01, 0x0B, 0x04, 0x65, 0xA0, 0x54, 0x95, 0x00, 0x00, 0x00},
|
|
{0x11, 0x41, 0x0E, 0x04, 0x70, 0xC7, 0x13, 0x10, 0x00, 0x00, 0x00},
|
|
{0x02, 0x44, 0x16, 0x06, 0xE0, 0xE0, 0x31, 0x35, 0x00, 0x00, 0x00},
|
|
{0x48, 0x22, 0x22, 0x07, 0x50, 0xA1, 0xA5, 0xF4, 0x00, 0x00, 0x00},
|
|
{0x05, 0xA1, 0x18, 0x00, 0xA2, 0xA2, 0xF5, 0xF5, 0x00, 0x00, 0x00},
|
|
{0x07, 0x81, 0x2B, 0x05, 0xA5, 0xA5, 0x03, 0x03, 0x00, 0x00, 0x00},
|
|
{0x01, 0x41, 0x08, 0x08, 0xA0, 0xA0, 0x83, 0x95, 0x00, 0x00, 0x00},
|
|
{0x21, 0x61, 0x12, 0x00, 0x93, 0x92, 0x74, 0x75, 0x00, 0x00, 0x00},
|
|
{0x21, 0x62, 0x21, 0x00, 0x84, 0x85, 0x34, 0x15, 0x00, 0x00, 0x00},
|
|
{0x21, 0x62, 0x0E, 0x00, 0xA1, 0xA0, 0x34, 0x15, 0x00, 0x00, 0x00},
|
|
#endif
|
|
|
|
#if 0
|
|
/* Horton try 2 */
|
|
{0x31, 0x22, 0x23, 0x07, 0xF0, 0xF0, 0xE8, 0xF7, 0x00, 0x00, 0x00},
|
|
{0x03, 0x31, 0x68, 0x05, 0xF2, 0x74, 0x79, 0x9C, 0x00, 0x00, 0x00},
|
|
{0x01, 0x51, 0x72, 0x04, 0xF1, 0xD3, 0x9D, 0x8B, 0x00, 0x00, 0x00},
|
|
{0x22, 0x61, 0x1B, 0x05, 0xC0, 0xA1, 0xF8, 0xE8, 0x00, 0x00, 0x00},
|
|
{0x22, 0x61, 0x2C, 0x03, 0xD2, 0xA1, 0xA7, 0xE8, 0x00, 0x00, 0x00},
|
|
{0x31, 0x22, 0xFA, 0x01, 0xF1, 0xF1, 0xF4, 0xEE, 0x00, 0x00, 0x00},
|
|
{0x21, 0x61, 0x28, 0x06, 0xF1, 0xF1, 0xCE, 0x9B, 0x00, 0x00, 0x00},
|
|
{0x27, 0x61, 0x60, 0x00, 0xF0, 0xF0, 0xFF, 0xFD, 0x00, 0x00, 0x00},
|
|
{0x60, 0x21, 0x2B, 0x06, 0x85, 0xF1, 0x79, 0x9D, 0x00, 0x00, 0x00},
|
|
{0x31, 0xA1, 0xFF, 0x0A, 0x53, 0x62, 0x5E, 0xAF, 0x00, 0x00, 0x00},
|
|
{0x03, 0xA1, 0x70, 0x0F, 0xD4, 0xA3, 0x94, 0xBE, 0x00, 0x00, 0x00},
|
|
{0x2B, 0x61, 0xE4, 0x07, 0xF6, 0x93, 0xBD, 0xAC, 0x00, 0x00, 0x00},
|
|
{0x21, 0x63, 0xED, 0x07, 0x77, 0xF1, 0xC7, 0xE8, 0x00, 0x00, 0x00},
|
|
{0x21, 0x61, 0x2A, 0x03, 0xF3, 0xE2, 0xB6, 0xD9, 0x00, 0x00, 0x00},
|
|
{0x21, 0x63, 0x37, 0x03, 0xF3, 0xE2, 0xB6, 0xD9, 0x00, 0x00, 0x00},
|
|
#endif
|
|
};
|
|
|
|
static void
|
|
vrc7_reset (void)
|
|
{
|
|
int n;
|
|
|
|
/* Point to current VRC7 context */
|
|
vrc7_t *opll = &vrc7;
|
|
|
|
/* Clear all YM3812 registers */
|
|
for (n = 0; n < 0x100; n++)
|
|
OPL_WRITE (opll, n, 0x00);
|
|
|
|
/* Turn off rhythm mode and key-on bits */
|
|
OPL_WRITE (opll, 0xBD, 0xC0);
|
|
|
|
/* Enable waveform select */
|
|
OPL_WRITE (opll, 0x01, 0x20);
|
|
}
|
|
|
|
static void
|
|
vrc7_init (void)
|
|
{
|
|
vrc7.ym3812 =
|
|
OPLCreate (OPL_TYPE_YM3812, 3579545, apu_getcontext ()->sample_rate);
|
|
ASSERT (vrc7.ym3812);
|
|
buflen = apu_getcontext ()->num_samples;
|
|
buffer = malloc (buflen * 2);
|
|
ASSERT (buffer);
|
|
vrc7_reset ();
|
|
}
|
|
|
|
static void
|
|
vrc7_shutdown (void)
|
|
{
|
|
vrc7_reset ();
|
|
OPLDestroy (vrc7.ym3812);
|
|
free (buffer);
|
|
}
|
|
|
|
/* channel (0-9), instrument (0-F), volume (0-3F, YM3812 format) */
|
|
static void
|
|
load_instrument (uint8 ch, uint8 inst, uint8 vol)
|
|
{
|
|
/* Point to current VRC7 context */
|
|
vrc7_t *opll = &vrc7;
|
|
|
|
/* Point to fixed instrument or user table */
|
|
uint8 *param = (inst == 0) ? &opll->user[0] : &table[inst][0];
|
|
|
|
/* Maps channels to operator registers */
|
|
uint8 ch2op[] = { 0, 1, 2, 8, 9, 10, 16, 17, 18 };
|
|
|
|
/* Make operator offset from requested channel */
|
|
uint8 op = ch2op[ch];
|
|
|
|
/* Store volume level */
|
|
opll->channel[ch].volume = (vol & 0x3F);
|
|
|
|
/* Store instrument number */
|
|
opll->channel[ch].instrument = (inst & 0x0F);
|
|
|
|
/* Update instrument settings, except frequency registers */
|
|
OPL_WRITE (opll, 0x20 + op, param[0]);
|
|
OPL_WRITE (opll, 0x23 + op, param[1]);
|
|
OPL_WRITE (opll, 0x40 + op, param[2]);
|
|
OPL_WRITE (opll, 0x43 + op, (param[3] & 0xC0) | opll->channel[ch].volume);
|
|
OPL_WRITE (opll, 0x60 + op, param[4]);
|
|
OPL_WRITE (opll, 0x63 + op, param[5]);
|
|
OPL_WRITE (opll, 0x80 + op, param[6]);
|
|
OPL_WRITE (opll, 0x83 + op, param[7]);
|
|
OPL_WRITE (opll, 0xE0 + op, param[8]);
|
|
OPL_WRITE (opll, 0xE3 + op, param[9]);
|
|
OPL_WRITE (opll, 0xC0 + ch, param[10]);
|
|
}
|
|
|
|
static void
|
|
vrc7_write (uint32 address, uint8 data)
|
|
{
|
|
/* Point to current VRC7 context */
|
|
vrc7_t *opll = &vrc7;
|
|
|
|
if (address & 0x0020) { /* data port */
|
|
/* Store register data */
|
|
opll->reg[opll->latch] = data;
|
|
|
|
switch (opll->latch & 0x30) {
|
|
case 0x00: /* User instrument registers */
|
|
switch (opll->latch & 0x0F) {
|
|
case 0x00: /* Misc. ctrl. (modulator) */
|
|
case 0x01: /* Misc. ctrl. (carrier) */
|
|
case 0x02: /* Key scale level and total level (modulator) */
|
|
case 0x04: /* Attack / Decay (modulator) */
|
|
case 0x05: /* Attack / Decay (carrier) */
|
|
case 0x06: /* Sustain / Release (modulator) */
|
|
case 0x07: /* Sustain / Release (carrier) */
|
|
opll->user[(opll->latch & 0x07)] = data;
|
|
break;
|
|
|
|
case 0x03: /* Key scale level, carrier/modulator waveform, feedback */
|
|
|
|
/* Key scale level (carrier) */
|
|
/* Don't touch the total level (channel volume) */
|
|
opll->user[3] = (opll->user[3] & 0x3F) | (data & 0xC0);
|
|
|
|
/* Waveform select for the modulator */
|
|
opll->user[8] = (data >> 3) & 1;
|
|
|
|
/* Waveform select for the carrier */
|
|
opll->user[9] = (data >> 4) & 1;
|
|
|
|
/* Store feedback level in YM3812 format */
|
|
opll->user[10] = ((data & 0x07) << 1) & 0x0E;
|
|
break;
|
|
}
|
|
|
|
/* If the user instrument registers were accessed, then
|
|
go through each channel and update the ones that were
|
|
currently using the user instrument. We can skip the
|
|
last three channels in rhythm mode since they can
|
|
only use percussion sounds anyways. */
|
|
if (opll->latch <= 0x05) {
|
|
uint8 x;
|
|
|
|
for (x = 0; x < 6; x++)
|
|
if (opll->channel[x].instrument == 0x00)
|
|
load_instrument (x, 0x00, opll->channel[x].volume);
|
|
}
|
|
break;
|
|
|
|
case 0x10: /* Channel Frequency (LSB) */
|
|
case 0x20: /* Channel Frequency (MSB) + key-on and sustain control */
|
|
{
|
|
uint8 block;
|
|
uint16 frequency;
|
|
uint8 ch = (opll->latch & 0x0F);
|
|
|
|
/* Ensure proper channel range */
|
|
if (ch > 0x05)
|
|
break;
|
|
|
|
/* Get VRC7 channel frequency */
|
|
frequency =
|
|
((opll->reg[0x10 + ch] & 0xFF) | ((opll->reg[0x20 +
|
|
ch] & 0x01) << 8));
|
|
|
|
/* Scale 9 bit frequency to 10 bits */
|
|
frequency = (frequency << 1) & 0x1FFF;
|
|
|
|
/* Get VRC7 block */
|
|
block = (opll->reg[0x20 + ch] >> 1) & 7;
|
|
|
|
/* Add in block */
|
|
frequency |= (block << 10);
|
|
|
|
/* Add key-on flag */
|
|
if (opll->reg[0x20 + ch] & 0x10)
|
|
frequency |= 0x2000;
|
|
|
|
/* Save current frequency/block/key-on setting */
|
|
opll->channel[ch].frequency = (frequency & 0x3FFF);
|
|
|
|
/* Write changes to YM3812 */
|
|
OPL_WRITE (opll, 0xA0 + ch, (opll->channel[ch].frequency >> 0) & 0xFF);
|
|
OPL_WRITE (opll, 0xB0 + ch, (opll->channel[ch].frequency >> 8) & 0xFF);
|
|
}
|
|
break;
|
|
|
|
case 0x30: /* Channel Volume Level and Instrument Select */
|
|
|
|
/* Ensure proper channel range */
|
|
if (opll->latch > 0x35)
|
|
break;
|
|
|
|
{
|
|
uint8 ch = (opll->latch & 0x0F);
|
|
uint8 inst = (data >> 4) & 0x0F;
|
|
uint8 vol = (data & 0x0F) << 2;
|
|
|
|
load_instrument (ch, inst, vol);
|
|
}
|
|
|
|
break;
|
|
}
|
|
} else { /* Register latch */
|
|
|
|
opll->latch = (data & 0x3F);
|
|
}
|
|
}
|
|
|
|
static int32
|
|
vrc7_process (void)
|
|
{
|
|
static int sample = 0;
|
|
|
|
/* update a large chunk at once */
|
|
if (sample >= buflen) {
|
|
sample -= buflen;
|
|
YM3812UpdateOne (vrc7.ym3812, buffer, buflen);
|
|
}
|
|
|
|
return (int32) ((int16 *) buffer)[sample++];
|
|
}
|
|
|
|
static apu_memwrite vrc7_memwrite[] = {
|
|
{0x9010, 0x9010, vrc7_write},
|
|
{0x9030, 0x9030, vrc7_write},
|
|
{(uint32) - 1, (uint32) - 1, NULL}
|
|
};
|
|
|
|
apuext_t vrc7_ext = {
|
|
vrc7_init,
|
|
vrc7_shutdown,
|
|
vrc7_reset,
|
|
vrc7_process,
|
|
NULL, /* no reads */
|
|
vrc7_memwrite
|
|
};
|
|
|
|
/*
|
|
** $Log$
|
|
** Revision 1.4 2008/03/26 07:40:56 slomo
|
|
** * gst/nsf/Makefile.am:
|
|
** * gst/nsf/fds_snd.c:
|
|
** * gst/nsf/mmc5_snd.c:
|
|
** * gst/nsf/nsf.c:
|
|
** * gst/nsf/types.h:
|
|
** * gst/nsf/vrc7_snd.c:
|
|
** * gst/nsf/vrcvisnd.c:
|
|
** * gst/nsf/memguard.c:
|
|
** * gst/nsf/memguard.h:
|
|
** Remove memguard again and apply hopefully all previously dropped
|
|
** local patches. Should be really better than the old version now.
|
|
**
|
|
** Revision 1.3 2008-03-25 15:56:12 slomo
|
|
** Patch by: Andreas Henriksson <andreas at fatal dot set>
|
|
** * gst/nsf/Makefile.am:
|
|
** * gst/nsf/dis6502.h:
|
|
** * gst/nsf/fds_snd.c:
|
|
** * gst/nsf/fds_snd.h:
|
|
** * gst/nsf/fmopl.c:
|
|
** * gst/nsf/fmopl.h:
|
|
** * gst/nsf/gstnsf.c:
|
|
** * gst/nsf/log.c:
|
|
** * gst/nsf/log.h:
|
|
** * gst/nsf/memguard.c:
|
|
** * gst/nsf/memguard.h:
|
|
** * gst/nsf/mmc5_snd.c:
|
|
** * gst/nsf/mmc5_snd.h:
|
|
** * gst/nsf/nes6502.c:
|
|
** * gst/nsf/nes6502.h:
|
|
** * gst/nsf/nes_apu.c:
|
|
** * gst/nsf/nes_apu.h:
|
|
** * gst/nsf/nsf.c:
|
|
** * gst/nsf/nsf.h:
|
|
** * gst/nsf/osd.h:
|
|
** * gst/nsf/types.h:
|
|
** * gst/nsf/vrc7_snd.c:
|
|
** * gst/nsf/vrc7_snd.h:
|
|
** * gst/nsf/vrcvisnd.c:
|
|
** * gst/nsf/vrcvisnd.h:
|
|
** Update our internal nosefart to nosefart-2.7-mls to fix segfaults
|
|
** on some files. Fixes bug #498237.
|
|
** Remove some // comments, fix some compiler warnings and use pow()
|
|
** instead of a slow, selfmade implementation.
|
|
**
|
|
** Revision 1.1 2003/04/08 20:53:01 ben
|
|
** Adding more files...
|
|
**
|
|
** Revision 1.5 2000/07/04 04:51:02 matt
|
|
** made data types stricter
|
|
**
|
|
** Revision 1.4 2000/07/03 02:18:53 matt
|
|
** much better external module exporting
|
|
**
|
|
** Revision 1.3 2000/06/20 20:45:09 matt
|
|
** minor cleanups
|
|
**
|
|
** Revision 1.2 2000/06/20 04:06:16 matt
|
|
** migrated external sound definition to apu module
|
|
**
|
|
** Revision 1.1 2000/06/20 00:06:47 matt
|
|
** initial revision
|
|
**
|
|
*/
|