ref: 0b7b60511d71e3c80deb8d1900e1ec778d964c0c
author: qwx <qwx@sciops.net>
date: Mon May 27 19:51:52 EDT 2019
initial import
--- /dev/null
+++ b/README
@@ -1,0 +1,6 @@
+based on opl3emu2 version 1.1.1 by Joel Yliluoma, itself based on fmopl.c
+version 0.72 from MAME, which is:
+Copyright (C) 2002,2003 Jarek Burczynski (bujar at mame dot net)
+Copyright (C) 1999,2000 Tatsuyuki Satoh , MultiArcadeMachineEmulator development
+
+this is a... work in progress. use at your own peril.
--- /dev/null
+++ b/dat.h
@@ -1,0 +1,443 @@
+/*
+ * The OPL-3 mode is switched on by writing 0x01, to the offset 5
+ * of the right side.
+ *
+ * Another special register at the right side is at offset 4. It contains
+ * a bit mask defining which voices are used as 4 OP voices.
+ *
+ * The percussive mode is implemented in the left side only.
+ *
+ * With the above exceptions the both sides can be operated independently.
+ *
+ * A 4 OP voice can be created by setting the corresponding
+ * bit at offset 4 of the right side.
+ *
+ * For example setting the rightmost bit (0x01) changes the
+ * first voice on the right side to the 4 OP mode. The fourth
+ * voice is made inaccessible.
+ *
+ * If a voice is set to the 2 OP mode, it works like 2 OP modes
+ * of the original YM3812 (AdLib). In addition the voice can
+ * be connected the left, right or both stereo channels. It can
+ * even be left unconnected. This works with 4 OP voices also.
+ *
+ * The stereo connection bits are located in the FEEDBACK_CONNECTION
+ * register of the voice (0xC0-0xC8). In 4 OP voices these bits are
+ * in the second half of the voice.
+ */
+
+/*
+ * Register numbers for the global registers
+ */
+
+#define TEST_REGISTER 0x01
+#define ENABLE_WAVE_SELECT 0x20
+
+#define TIMER1_REGISTER 0x02
+#define TIMER2_REGISTER 0x03
+#define TIMER_CONTROL_REGISTER 0x04 /* Left side */
+#define IRQ_RESET 0x80
+#define TIMER1_MASK 0x40
+#define TIMER2_MASK 0x20
+#define TIMER1_START 0x01
+#define TIMER2_START 0x02
+
+#define CONNECTION_SELECT_REGISTER 0x04 /* Right side */
+#define RIGHT_4OP_0 0x01
+#define RIGHT_4OP_1 0x02
+#define RIGHT_4OP_2 0x04
+#define LEFT_4OP_0 0x08
+#define LEFT_4OP_1 0x10
+#define LEFT_4OP_2 0x20
+
+#define OPL3_MODE_REGISTER 0x05 /* Right side */
+#define OPL3_ENABLE 0x01
+#define OPL4_ENABLE 0x02
+
+#define KBD_SPLIT_REGISTER 0x08 /* Left side */
+#define COMPOSITE_SINE_WAVE_MODE 0x80 /* Don't use with OPL-3? */
+#define KEYBOARD_SPLIT 0x40
+
+#define PERCOSSION_REGISTER 0xbd /* Left side only */
+#define TREMOLO_DEPTH 0x80
+#define VIBRATO_DEPTH 0x40
+#define PERCOSSION_ENABLE 0x20
+#define BASSDRUM_ON 0x10
+#define SNAREDRUM_ON 0x08
+#define TOMTOM_ON 0x04
+#define CYMBAL_ON 0x02
+#define HIHAT_ON 0x01
+
+/*
+ * Offsets to the register banks for operators. To get the
+ * register number just add the operator offset to the bank offset
+ *
+ * AM/VIB/EG/KSR/Multiple (0x20 to 0x35)
+ */
+enum{
+ AM_VIB = 0x20,
+ TREMOLO_ON = 0x80,
+ VIBRATO_ON = 0x40,
+ SUSTAIN_ON = 0x20,
+ KSR = 0x10, /* Key scaling rate */
+ MULTIPLE_MASK = 0x0f /* Frequency multiplier */
+};
+
+ /*
+ * KSL/Total level (0x40 to 0x55)
+ */
+#define KSL_LEVEL 0x40
+#define KSL_MASK 0xc0 /* Envelope scaling bits */
+#define TOTAL_LEVEL_MASK 0x3f /* Strength (volume) of OP */
+
+/*
+ * Attack / Decay rate (0x60 to 0x75)
+ */
+#define ATTACK_DECAY 0x60
+#define ATTACK_MASK 0xf0
+#define DECAY_MASK 0x0f
+
+/*
+ * Sustain level / Release rate (0x80 to 0x95)
+ */
+#define SUSTAIN_RELEASE 0x80
+#define SUSTAIN_MASK 0xf0
+#define RELEASE_MASK 0x0f
+
+/*
+ * Wave select (0xE0 to 0xF5)
+ */
+#define WAVE_SELECT 0xe0
+
+/*
+ * Offsets to the register banks for voices. Just add to the
+ * voice number to get the register number.
+ *
+ * F-Number low bits (0xA0 to 0xA8).
+ */
+#define FNUM_LOW 0xa0
+
+/*
+ * F-number high bits / Key on / Block (octave) (0xB0 to 0xB8)
+ */
+#define KEYON_BLOCK 0xb0
+#define KEYON_BIT 0x20
+#define BLOCKNUM_MASK 0x1c
+#define FNUM_HIGH_MASK 0x03
+
+/*
+ * Feedback / Connection (0xc0 to 0xc8)
+ *
+ * These registers have two new bits when the OPL-3 mode
+ * is selected. These bits controls connecting the voice
+ * to the stereo channels. For 4 OP voices this bit is
+ * defined in the second half of the voice (add 3 to the
+ * register offset).
+ *
+ * For 4 OP voices the connection bit is used in the
+ * both halves (gives 4 ways to connect the operators).
+ */
+#define FEEDBACK_CONNECTION 0xc0
+#define FEEDBACK_MASK 0x0e /* Valid just for 1st OP of a voice */
+#define CONNECTION_BIT 0x01
+/*
+ * In the 4 OP mode there is four possible configurations how the
+ * operators can be connected together (in 2 OP modes there is just
+ * AM or FM). The 4 OP connection mode is defined by the rightmost
+ * bit of the FEEDBACK_CONNECTION (0xC0-0xC8) on the both halves.
+ *
+ * First half Second half Mode
+ *
+ * +---+
+ * v |
+ * 0 0 >+-1-+--2--3--4-->
+ *
+ *
+ *
+ * +---+
+ * | |
+ * 0 1 >+-1-+--2-+
+ * |->
+ * >--3----4-+
+ *
+ * +---+
+ * | |
+ * 1 0 >+-1-+-----+
+ * |->
+ * >--2--3--4-+
+ *
+ * +---+
+ * | |
+ * 1 1 >+-1-+--+
+ * |
+ * >--2--3-+->
+ * |
+ * >--4----+
+ */
+#define STEREO_BITS 0x30 /* OPL-3 only */
+#define VOICE_TO_LEFT 0x10
+#define VOICE_TO_RIGHT 0x20
+
+enum{
+ MAXINST = 100, /* FIXME: what the fuck is this? */
+ MAXPATN = 100,
+ MAXCHN = 15 /* Remember that the percussion channel is reserved! */
+};
+
+typedef struct MusPlayerPos MusPlayerPos;
+typedef struct Pattern Pattern;
+typedef struct InternalHdr InternalHdr;
+typedef struct InternalSample InternalSample;
+typedef struct MusData MusData;
+
+struct MusPlayerPos{
+ int Ord;
+ int Pat;
+ int Row;
+ char *Pos;
+ int Wait;
+};
+
+struct Pattern{
+ ushort Len;
+ char *Ptr;
+};
+
+struct InternalHdr{
+ ushort OrdNum; /* Number of orders, is even */
+ /* If bit 8 (&256) is set, there exist no */
+ /* notes between 126..253 and the 7th bit */
+ /* of note value can be used to tell if */
+ /* there is instrument number. */
+ ushort InsNum; /* Number of instruments */
+ ushort PatNum; /* Number of patterns */
+ /* If bit 9 set, all instruments have */
+ /* the insname[] filled with AsciiZ name. */
+ ushort Ins2Num; /* Number of instruments - for checking! */
+ uchar InSpeed; /* Initial speed in 7 lowermost bits */
+ /* If highest bit (&128) set, no logaritmic */
+ /* adlib volume conversion needs to be done */
+ /* when playing with midi device. */
+ uchar InTempo; /* Initial tempo */
+ uchar ChanCount[]; /* ChanCount[0] present if PatNum bit 9 set */
+}; /* Size: 2+2+2+2+1+1 = 8+2 = 10 */
+
+struct InternalSample{
+ /* unused */
+ char FT; /*finetune value 1 */
+ /* signed seminotes. */
+ uchar D[11]; /*Identificators 11 */
+ /* - 12nd byte not needed. */
+ /* six high bits of D[8] */
+ /* define the midi volume */
+ /* scaling level */
+ /* - default is 63. */
+ /* D[9] bits 5..2 have */
+ /* the automatical SDx */
+ /* adjust value. */
+ /* - default is 0. */
+ /* D[9] bits 6-7 are free. */
+ char Volume; /*0..63 1 */
+ /* If bit 6 set, */
+ /* the instrument will be */
+ /* played simultaneously */
+ /* on two channels when */
+ /* playing with a midi device. */
+ /* To gain more volume. */
+ /* If bit 7 set, */
+ /* the finetune value */
+ /* affects also FM. */
+ /* (SMP->AME conversion) */
+ ushort C2Spd; /*8363 = normal..? 2 */
+ uchar GM; /*GM program number 1 */
+ uchar Bank; /*Bank number, 0=normal. 1 */
+ /*Highest bit (&128) is free. */
+ char insname[];
+ /*Only if PatNum&512 set */
+}; /* total:17 = 0x11 */
+
+struct MusData{
+ int PlayNext;
+ /* Set to 1 every time next line is played.
+ Use it to syncronize screen with music. */
+
+ long RowTimer;
+ /* You can delay the music with this.
+ Or fasten. Units: Rate */
+
+ char *Playing; /* Currently playing music from this buffer. */
+ int Paused; /* If paused, interrupts normally but does not play. *
+ * Read only. There's PauseS3M() for this. */
+ InternalHdr *Hdr;
+ /* FIXME: >127 for percussion; was set to MAXINST before, causing bad reads; is
+ this used correctly now? why is it 100? what the fuck is it for??? */
+ InternalSample *Instr[2*MAXINST+1];
+
+ int LinearMidiVol;
+ int GlobalVolume;
+
+ int Speed;
+ int Tempo;
+
+ /* Data from song */
+ char *Orders;
+ Pattern Patn[MAXPATN];
+
+ /* Player: For SBx */
+ MusPlayerPos posi;
+ MusPlayerPos saved;
+ int LoopCount;
+ int PendingSB0;
+
+ long CurVol[MAXCHN];
+ uchar NNum[MAXCHN];
+ uchar CurPuh[MAXCHN];
+ int NoteCut[MAXCHN];
+ int NoteDelay[MAXCHN];
+ uchar CurInst[MAXCHN];
+ uchar InstNum[MAXCHN]; /* For the instrument number optimizer */
+ uchar PatternDelay;
+
+ char FineVolSlide[MAXCHN];
+ char FineFrqSlide[MAXCHN];
+ char VolSlide[MAXCHN];
+ char FreqSlide[MAXCHN];
+ uchar VibDepth[MAXCHN];
+ uchar VibSpd [MAXCHN];
+ uchar VibPos [MAXCHN];
+ uchar ArpegSpd;
+ uchar Arpeggio[MAXCHN];
+ uchar ArpegCnt[MAXCHN];
+ uchar Retrig[MAXCHN];
+
+ char Active[MAXCHN];
+ char Doubles[MAXCHN];
+
+ uchar DxxDefault[MAXCHN];
+ uchar ExxDefault[MAXCHN];
+ uchar HxxDefault[MAXCHN];
+ uchar MxxDefault[MAXCHN];
+ uchar SxxDefault[MAXCHN];
+
+ ulong Herz[MAXCHN]; /* K�ytt�� m_opl */
+};
+#define FxxDefault ExxDefault /* They really are combined in ST3 */
+
+/* fmopl */
+enum{
+ OPL_SAMPLE_BITS = 16
+};
+
+typedef s16int OPLSAMPLE; //INT32
+typedef void (*OPL_TIMERHANDLER)(void *param, int timer, double interval_sec);
+typedef void (*OPL_IRQHANDLER)(void *param, int irq);
+typedef void (*OPL_UPDATEHANDLER)(void *param, int min_interval_us);
+typedef void (*OPL_PORTHANDLER_W)(void *param, uchar data);
+typedef uchar (*OPL_PORTHANDLER_R)(void *param);
+
+typedef struct OPL_SLOT OPL_SLOT;
+typedef struct OPL_CH OPL_CH;
+typedef struct FM_OPL FM_OPL;
+
+struct OPL_SLOT{
+ u32int ar; /* attack rate: AR<<2 */
+ u32int dr; /* decay rate: DR<<2 */
+ u32int rr; /* release rate:RR<<2 */
+ u8int KSR; /* key scale rate */
+ u8int ksl; /* keyscale level */
+ u8int ksr; /* key scale rate: kcode>>KSR */
+ u8int mul; /* multiple: mul_tab[ML] */
+
+ /* Phase Generator */
+ u32int Cnt; /* frequency counter */
+ u32int Incr; /* frequency counter step */
+ u8int FB; /* feedback shift value */
+ s32int *connect1; /* slot1 output pointer */
+ s32int op1_out[2]; /* slot1 output for feedback */
+ u8int CON; /* connection (algorithm) type */
+
+ /* Envelope Generator */
+ u8int eg_type; /* percussive/non-percussive mode */
+ u8int state; /* phase type */
+ u32int TL; /* total level: TL << 2 */
+ s32int TLL; /* adjusted now TL */
+ s32int volume; /* envelope counter */
+ u32int sl; /* sustain level: sl_tab[SL] */
+ u8int eg_sh_ar; /* (attack state) */
+ u8int eg_sel_ar; /* (attack state) */
+ u8int eg_sh_dr; /* (decay state) */
+ u8int eg_sel_dr; /* (decay state) */
+ u8int eg_sh_rr; /* (release state) */
+ u8int eg_sel_rr; /* (release state) */
+ u32int key; /* 0 = KEY OFF, >0 = KEY ON */
+
+ /* LFO */
+ u32int AMmask; /* LFO Amplitude Modulation enable mask */
+ u8int vib; /* LFO Phase Modulation enable flag (active high)*/
+
+ /* waveform select */
+ u16int wavetable;
+};
+struct OPL_CH{
+ OPL_SLOT SLOT[2];
+ /* phase generator state */
+ u32int block_fnum; /* block+fnum */
+ u32int fc; /* Freq. Increment base */
+ u32int ksl_base; /* KeyScaleLevel Base step */
+ u8int kcode; /* key code (for key scaling) */
+};
+struct FM_OPL{
+ /* FM channel slots */
+ OPL_CH P_CH[9]; /* OPL/OPL2 chips have 9 channels*/
+
+ u32int eg_cnt; /* global envelope generator counter */
+ u32int eg_timer; /* global envelope generator counter works at frequency = chipclock/72 */
+ u32int eg_timer_add; /* step of eg_timer */
+ u32int eg_timer_overflow; /* envelope generator timer overlfows every 1 sample (on real chip) */
+
+ u8int rhythm; /* Rhythm mode */
+
+ u32int fn_tab[1024]; /* fnumber->increment counter */
+
+ /* LFO */
+ u32int LFO_AM;
+ s32int LFO_PM;
+
+ u8int lfo_am_depth;
+ u8int lfo_pm_depth_range;
+ u32int lfo_am_cnt;
+ u32int lfo_am_inc;
+ u32int lfo_pm_cnt;
+ u32int lfo_pm_inc;
+
+ u32int noise_rng; /* 23 bit noise shift register */
+ u32int noise_p; /* current noise 'phase' */
+ u32int noise_f; /* current noise period */
+
+ u8int wavesel; /* waveform select enable flag */
+
+ u32int T[2]; /* timer counters */
+ u8int st[2]; /* timer enable */
+
+ /* external event callback handlers */
+ OPL_TIMERHANDLER timer_handler; /* TIMER handler */
+ void *TimerParam; /* TIMER parameter */
+ OPL_IRQHANDLER IRQHandler; /* IRQ handler */
+ void *IRQParam; /* IRQ parameter */
+ OPL_UPDATEHANDLER UpdateHandler;/* stream update handler */
+ void *UpdateParam; /* stream update parameter */
+
+ u8int type; /* chip type */
+ u8int address; /* address register */
+ u8int status; /* status flag */
+ u8int statusmask; /* status mask */
+ u8int mode; /* Reg.08 : CSM,notesel,etc. */
+
+ u32int clock; /* master clock (Hz) */
+ u32int rate; /* sampling rate (Hz) */
+ double freqbase; /* frequency base */
+ double TimerBase; /* Timer base time (==sampling time)*/
+
+ s32int phase_modulation; /* phase modulation input (SLOT 2) */
+ s32int output[1];
+};
--- /dev/null
+++ b/fmopl.c
@@ -1,0 +1,1676 @@
+#include <u.h>
+#include <libc.h>
+#include <stdio.h>
+#include "dat.h"
+#include "fns.h"
+
+/* output final shift */
+#define FINAL_SH (0)
+#define MAXOUT (+32767)
+#define MINOUT (-32768)
+
+
+#define FREQ_SH 16 /* 16.16 fixed point (frequency calculations) */
+#define EG_SH 16 /* 16.16 fixed point (EG timing) */
+#define LFO_SH 24 /* 8.24 fixed point (LFO calculations) */
+#define TIMER_SH 16 /* 16.16 fixed point (timers calculations) */
+
+#define FREQ_MASK ((1<<FREQ_SH)-1)
+
+/* envelope output entries */
+#define ENV_BITS 10
+#define ENV_LEN (1<<ENV_BITS)
+#define ENV_STEP (128.0/ENV_LEN)
+
+#define MAX_ATT_INDEX ((1<<(ENV_BITS-1))-1) /*511*/
+#define MIN_ATT_INDEX (0)
+
+/* sinwave entries */
+#define SIN_BITS 10
+#define SIN_LEN (1<<SIN_BITS)
+#define SIN_MASK (SIN_LEN-1)
+
+#define TL_RES_LEN (256) /* 8 bits addressing (real chip) */
+
+
+
+/* register number to channel number , slot offset */
+#define SLOT1 0
+#define SLOT2 1
+
+/* Envelope Generator phases */
+
+#define EG_ATT 4
+#define EG_DEC 3
+#define EG_SUS 2
+#define EG_REL 1
+#define EG_OFF 0
+
+
+int acc_calc(signed int value)
+{
+ if (value>=0)
+ {
+ if (value < 0x0200)
+ return (value & ~0);
+ if (value < 0x0400)
+ return (value & ~1);
+ if (value < 0x0800)
+ return (value & ~3);
+ if (value < 0x1000)
+ return (value & ~7);
+ if (value < 0x2000)
+ return (value & ~15);
+ if (value < 0x4000)
+ return (value & ~31);
+ return (value & ~63);
+ }
+ if (value > -0x0200)
+ return (~abs(value) & ~0);
+ if (value > -0x0400)
+ return (~abs(value) & ~1);
+ if (value > -0x0800)
+ return (~abs(value) & ~3);
+ if (value > -0x1000)
+ return (~abs(value) & ~7);
+ if (value > -0x2000)
+ return (~abs(value) & ~15);
+ if (value > -0x4000)
+ return (~abs(value) & ~31);
+ return (~abs(value) & ~63);
+}
+
+#define OPL_TYPE_WAVESEL 0x01 /* waveform select */
+/* ---------- Generic interface section ---------- */
+#define OPL_TYPE_YM3812 (OPL_TYPE_WAVESEL)
+
+/* mapping of register number (offset) to slot number used by the emulator */
+static int slot_array[32]=
+{
+ 0, 2, 4, 1, 3, 5,-1,-1,
+ 6, 8,10, 7, 9,11,-1,-1,
+ 12,14,16,13,15,17,-1,-1,
+ -1,-1,-1,-1,-1,-1,-1,-1
+};
+
+/* key scale level */
+/* table is 3dB/octave , DV converts this into 6dB/octave */
+/* 0.1875 is bit 0 weight of the envelope counter (volume) expressed in the 'decibel' scale */
+#define DV (0.1875/2.0)
+static u32int ksl_tab[8*16]=
+{
+ /* OCT 0 */
+ 0.000/DV, 0.000/DV, 0.000/DV, 0.000/DV,
+ 0.000/DV, 0.000/DV, 0.000/DV, 0.000/DV,
+ 0.000/DV, 0.000/DV, 0.000/DV, 0.000/DV,
+ 0.000/DV, 0.000/DV, 0.000/DV, 0.000/DV,
+ /* OCT 1 */
+ 0.000/DV, 0.000/DV, 0.000/DV, 0.000/DV,
+ 0.000/DV, 0.000/DV, 0.000/DV, 0.000/DV,
+ 0.000/DV, 0.750/DV, 1.125/DV, 1.500/DV,
+ 1.875/DV, 2.250/DV, 2.625/DV, 3.000/DV,
+ /* OCT 2 */
+ 0.000/DV, 0.000/DV, 0.000/DV, 0.000/DV,
+ 0.000/DV, 1.125/DV, 1.875/DV, 2.625/DV,
+ 3.000/DV, 3.750/DV, 4.125/DV, 4.500/DV,
+ 4.875/DV, 5.250/DV, 5.625/DV, 6.000/DV,
+ /* OCT 3 */
+ 0.000/DV, 0.000/DV, 0.000/DV, 1.875/DV,
+ 3.000/DV, 4.125/DV, 4.875/DV, 5.625/DV,
+ 6.000/DV, 6.750/DV, 7.125/DV, 7.500/DV,
+ 7.875/DV, 8.250/DV, 8.625/DV, 9.000/DV,
+ /* OCT 4 */
+ 0.000/DV, 0.000/DV, 3.000/DV, 4.875/DV,
+ 6.000/DV, 7.125/DV, 7.875/DV, 8.625/DV,
+ 9.000/DV, 9.750/DV,10.125/DV,10.500/DV,
+ 10.875/DV,11.250/DV,11.625/DV,12.000/DV,
+ /* OCT 5 */
+ 0.000/DV, 3.000/DV, 6.000/DV, 7.875/DV,
+ 9.000/DV,10.125/DV,10.875/DV,11.625/DV,
+ 12.000/DV,12.750/DV,13.125/DV,13.500/DV,
+ 13.875/DV,14.250/DV,14.625/DV,15.000/DV,
+ /* OCT 6 */
+ 0.000/DV, 6.000/DV, 9.000/DV,10.875/DV,
+ 12.000/DV,13.125/DV,13.875/DV,14.625/DV,
+ 15.000/DV,15.750/DV,16.125/DV,16.500/DV,
+ 16.875/DV,17.250/DV,17.625/DV,18.000/DV,
+ /* OCT 7 */
+ 0.000/DV, 9.000/DV,12.000/DV,13.875/DV,
+ 15.000/DV,16.125/DV,16.875/DV,17.625/DV,
+ 18.000/DV,18.750/DV,19.125/DV,19.500/DV,
+ 19.875/DV,20.250/DV,20.625/DV,21.000/DV
+};
+#undef DV
+
+/* 0 / 3.0 / 1.5 / 6.0 dB/OCT */
+static u32int ksl_shift[4] = { 31, 1, 2, 0 };
+
+
+/* sustain level table (3dB per step) */
+/* 0 - 15: 0, 3, 6, 9,12,15,18,21,24,27,30,33,36,39,42,93 (dB)*/
+#define SC(db) (u32int) ( db * (2.0/ENV_STEP) )
+static u32int sl_tab[16]={
+ SC( 0),SC( 1),SC( 2),SC(3 ),SC(4 ),SC(5 ),SC(6 ),SC( 7),
+ SC( 8),SC( 9),SC(10),SC(11),SC(12),SC(13),SC(14),SC(31)
+};
+#undef SC
+
+
+#define RATE_STEPS (8)
+static uchar eg_inc[15*RATE_STEPS]={
+/*cycle:0 1 2 3 4 5 6 7*/
+
+/* 0 */ 0,1, 0,1, 0,1, 0,1, /* rates 00..12 0 (increment by 0 or 1) */
+/* 1 */ 0,1, 0,1, 1,1, 0,1, /* rates 00..12 1 */
+/* 2 */ 0,1, 1,1, 0,1, 1,1, /* rates 00..12 2 */
+/* 3 */ 0,1, 1,1, 1,1, 1,1, /* rates 00..12 3 */
+
+/* 4 */ 1,1, 1,1, 1,1, 1,1, /* rate 13 0 (increment by 1) */
+/* 5 */ 1,1, 1,2, 1,1, 1,2, /* rate 13 1 */
+/* 6 */ 1,2, 1,2, 1,2, 1,2, /* rate 13 2 */
+/* 7 */ 1,2, 2,2, 1,2, 2,2, /* rate 13 3 */
+
+/* 8 */ 2,2, 2,2, 2,2, 2,2, /* rate 14 0 (increment by 2) */
+/* 9 */ 2,2, 2,4, 2,2, 2,4, /* rate 14 1 */
+/*10 */ 2,4, 2,4, 2,4, 2,4, /* rate 14 2 */
+/*11 */ 2,4, 4,4, 2,4, 4,4, /* rate 14 3 */
+
+/*12 */ 4,4, 4,4, 4,4, 4,4, /* rates 15 0, 15 1, 15 2, 15 3 (increment by 4) */
+/*13 */ 8,8, 8,8, 8,8, 8,8, /* rates 15 2, 15 3 for attack */
+/*14 */ 0,0, 0,0, 0,0, 0,0, /* infinity rates for attack and decay(s) */
+};
+
+
+#define O(a) (a*RATE_STEPS)
+
+/*note that there is no O(13) in this table - it's directly in the code */
+static uchar eg_rate_select[16+64+16]={ /* Envelope Generator rates (16 + 64 rates + 16 RKS) */
+/* 16 infinite time rates */
+O(14),O(14),O(14),O(14),O(14),O(14),O(14),O(14),
+O(14),O(14),O(14),O(14),O(14),O(14),O(14),O(14),
+
+/* rates 00-12 */
+O( 0),O( 1),O( 2),O( 3),
+O( 0),O( 1),O( 2),O( 3),
+O( 0),O( 1),O( 2),O( 3),
+O( 0),O( 1),O( 2),O( 3),
+O( 0),O( 1),O( 2),O( 3),
+O( 0),O( 1),O( 2),O( 3),
+O( 0),O( 1),O( 2),O( 3),
+O( 0),O( 1),O( 2),O( 3),
+O( 0),O( 1),O( 2),O( 3),
+O( 0),O( 1),O( 2),O( 3),
+O( 0),O( 1),O( 2),O( 3),
+O( 0),O( 1),O( 2),O( 3),
+O( 0),O( 1),O( 2),O( 3),
+
+/* rate 13 */
+O( 4),O( 5),O( 6),O( 7),
+
+/* rate 14 */
+O( 8),O( 9),O(10),O(11),
+
+/* rate 15 */
+O(12),O(12),O(12),O(12),
+
+/* 16 dummy rates (same as 15 3) */
+O(12),O(12),O(12),O(12),O(12),O(12),O(12),O(12),
+O(12),O(12),O(12),O(12),O(12),O(12),O(12),O(12),
+
+};
+#undef O
+
+/*rate 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15 */
+/*shift 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0, 0, 0, 0 */
+/*mask 4095, 2047, 1023, 511, 255, 127, 63, 31, 15, 7, 3, 1, 0, 0, 0, 0 */
+
+#define O(a) (a*1)
+static uchar eg_rate_shift[16+64+16]={ /* Envelope Generator counter shifts (16 + 64 rates + 16 RKS) */
+/* 16 infinite time rates */
+O(0),O(0),O(0),O(0),O(0),O(0),O(0),O(0),
+O(0),O(0),O(0),O(0),O(0),O(0),O(0),O(0),
+
+/* rates 00-12 */
+O(12),O(12),O(12),O(12),
+O(11),O(11),O(11),O(11),
+O(10),O(10),O(10),O(10),
+O( 9),O( 9),O( 9),O( 9),
+O( 8),O( 8),O( 8),O( 8),
+O( 7),O( 7),O( 7),O( 7),
+O( 6),O( 6),O( 6),O( 6),
+O( 5),O( 5),O( 5),O( 5),
+O( 4),O( 4),O( 4),O( 4),
+O( 3),O( 3),O( 3),O( 3),
+O( 2),O( 2),O( 2),O( 2),
+O( 1),O( 1),O( 1),O( 1),
+O( 0),O( 0),O( 0),O( 0),
+
+/* rate 13 */
+O( 0),O( 0),O( 0),O( 0),
+
+/* rate 14 */
+O( 0),O( 0),O( 0),O( 0),
+
+/* rate 15 */
+O( 0),O( 0),O( 0),O( 0),
+
+/* 16 dummy rates (same as 15 3) */
+O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),
+O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),O( 0),
+
+};
+#undef O
+
+
+/* multiple table */
+#define ML 2
+static u8int mul_tab[16]= {
+/* 1/2, 1, 2, 3, 4, 5, 6, 7, 8, 9,10,10,12,12,15,15 */
+ ML/2, 1*ML, 2*ML, 3*ML, 4*ML, 5*ML, 6*ML, 7*ML,
+ 8*ML, 9*ML,10*ML,10*ML,12*ML,12*ML,15*ML,15*ML
+};
+#undef ML
+
+/* TL_TAB_LEN is calculated as:
+* 12 - sinus amplitude bits (Y axis)
+* 2 - sinus sign bit (Y axis)
+* TL_RES_LEN - sinus resolution (X axis)
+*/
+#define TL_TAB_LEN (12*2*TL_RES_LEN)
+static signed int tl_tab[TL_TAB_LEN];
+
+#define ENV_QUIET (TL_TAB_LEN>>4)
+
+/* sin waveform table in 'decibel' scale */
+/* four waveforms on OPL2 type chips */
+static unsigned int sin_tab[SIN_LEN * 4];
+
+
+/* LFO Amplitude Modulation table (verified on real YM3812)
+ 27 output levels (triangle waveform); 1 level takes one of: 192, 256 or 448 samples
+
+ Length: 210 elements.
+
+ Each of the elements has to be repeated
+ exactly 64 times (on 64 consecutive samples).
+ The whole table takes: 64 * 210 = 13440 samples.
+
+ When AM = 1 data is used directly
+ When AM = 0 data is divided by 4 before being used (losing precision is important)
+*/
+
+#define LFO_AM_TAB_ELEMENTS 210U
+
+static u8int lfo_am_table[LFO_AM_TAB_ELEMENTS] = {
+0,0,0,0,0,0,0,
+1,1,1,1,
+2,2,2,2,
+3,3,3,3,
+4,4,4,4,
+5,5,5,5,
+6,6,6,6,
+7,7,7,7,
+8,8,8,8,
+9,9,9,9,
+10,10,10,10,
+11,11,11,11,
+12,12,12,12,
+13,13,13,13,
+14,14,14,14,
+15,15,15,15,
+16,16,16,16,
+17,17,17,17,
+18,18,18,18,
+19,19,19,19,
+20,20,20,20,
+21,21,21,21,
+22,22,22,22,
+23,23,23,23,
+24,24,24,24,
+25,25,25,25,
+26,26,26,
+25,25,25,25,
+24,24,24,24,
+23,23,23,23,
+22,22,22,22,
+21,21,21,21,
+20,20,20,20,
+19,19,19,19,
+18,18,18,18,
+17,17,17,17,
+16,16,16,16,
+15,15,15,15,
+14,14,14,14,
+13,13,13,13,
+12,12,12,12,
+11,11,11,11,
+10,10,10,10,
+9,9,9,9,
+8,8,8,8,
+7,7,7,7,
+6,6,6,6,
+5,5,5,5,
+4,4,4,4,
+3,3,3,3,
+2,2,2,2,
+1,1,1,1
+};
+
+/* LFO Phase Modulation table (verified on real YM3812) */
+static s8int lfo_pm_table[8*8*2] = {
+/* FNUM2/FNUM = 00 0xxxxxxx (0x0000) */
+0, 0, 0, 0, 0, 0, 0, 0, /*LFO PM depth = 0*/
+0, 0, 0, 0, 0, 0, 0, 0, /*LFO PM depth = 1*/
+
+/* FNUM2/FNUM = 00 1xxxxxxx (0x0080) */
+0, 0, 0, 0, 0, 0, 0, 0, /*LFO PM depth = 0*/
+1, 0, 0, 0,-1, 0, 0, 0, /*LFO PM depth = 1*/
+
+/* FNUM2/FNUM = 01 0xxxxxxx (0x0100) */
+1, 0, 0, 0,-1, 0, 0, 0, /*LFO PM depth = 0*/
+2, 1, 0,-1,-2,-1, 0, 1, /*LFO PM depth = 1*/
+
+/* FNUM2/FNUM = 01 1xxxxxxx (0x0180) */
+1, 0, 0, 0,-1, 0, 0, 0, /*LFO PM depth = 0*/
+3, 1, 0,-1,-3,-1, 0, 1, /*LFO PM depth = 1*/
+
+/* FNUM2/FNUM = 10 0xxxxxxx (0x0200) */
+2, 1, 0,-1,-2,-1, 0, 1, /*LFO PM depth = 0*/
+4, 2, 0,-2,-4,-2, 0, 2, /*LFO PM depth = 1*/
+
+/* FNUM2/FNUM = 10 1xxxxxxx (0x0280) */
+2, 1, 0,-1,-2,-1, 0, 1, /*LFO PM depth = 0*/
+5, 2, 0,-2,-5,-2, 0, 2, /*LFO PM depth = 1*/
+
+/* FNUM2/FNUM = 11 0xxxxxxx (0x0300) */
+3, 1, 0,-1,-3,-1, 0, 1, /*LFO PM depth = 0*/
+6, 3, 0,-3,-6,-3, 0, 3, /*LFO PM depth = 1*/
+
+/* FNUM2/FNUM = 11 1xxxxxxx (0x0380) */
+3, 1, 0,-1,-3,-1, 0, 1, /*LFO PM depth = 0*/
+7, 3, 0,-3,-7,-3, 0, 3 /*LFO PM depth = 1*/
+};
+
+
+/* lock level of common table */
+static int num_lock = 0;
+
+
+#define SLOT7_1 (&OPL->P_CH[7].SLOT[SLOT1])
+#define SLOT7_2 (&OPL->P_CH[7].SLOT[SLOT2])
+#define SLOT8_1 (&OPL->P_CH[8].SLOT[SLOT1])
+#define SLOT8_2 (&OPL->P_CH[8].SLOT[SLOT2])
+
+
+
+
+static int limit( int val, int max, int min ) {
+ if ( val > max )
+ val = max;
+ else if ( val < min )
+ val = min;
+
+ return val;
+}
+
+
+/* status set and IRQ handling */
+static void OPL_STATUS_SET(FM_OPL *OPL,int flag)
+{
+ /* set status flag */
+ OPL->status |= flag;
+ if(!(OPL->status & 0x80))
+ {
+ if(OPL->status & OPL->statusmask)
+ { /* IRQ on */
+ OPL->status |= 0x80;
+ /* callback user interrupt handler (IRQ is OFF to ON) */
+ if(OPL->IRQHandler) (OPL->IRQHandler)(OPL->IRQParam,1);
+ }
+ }
+}
+
+/* status reset and IRQ handling */
+static void OPL_STATUS_RESET(FM_OPL *OPL,int flag)
+{
+ /* reset status flag */
+ OPL->status &=~flag;
+ if((OPL->status & 0x80))
+ {
+ if (!(OPL->status & OPL->statusmask) )
+ {
+ OPL->status &= 0x7f;
+ /* callback user interrupt handler (IRQ is ON to OFF) */
+ if(OPL->IRQHandler) (OPL->IRQHandler)(OPL->IRQParam,0);
+ }
+ }
+}
+
+/* IRQ mask set */
+static void OPL_STATUSMASK_SET(FM_OPL *OPL,int flag)
+{
+ OPL->statusmask = flag;
+ /* IRQ handling check */
+ OPL_STATUS_SET(OPL,0);
+ OPL_STATUS_RESET(OPL,0);
+}
+
+
+/* advance LFO to next sample */
+static void advance_lfo(FM_OPL *OPL)
+{
+ u8int tmp;
+
+ /* LFO */
+ OPL->lfo_am_cnt += OPL->lfo_am_inc;
+ if (OPL->lfo_am_cnt >= ((u32int)LFO_AM_TAB_ELEMENTS<<LFO_SH) ) /* lfo_am_table is 210 elements long */
+ OPL->lfo_am_cnt -= ((u32int)LFO_AM_TAB_ELEMENTS<<LFO_SH);
+
+ tmp = lfo_am_table[ OPL->lfo_am_cnt >> LFO_SH ];
+
+ if (OPL->lfo_am_depth)
+ OPL->LFO_AM = tmp;
+ else
+ OPL->LFO_AM = tmp>>2;
+
+ OPL->lfo_pm_cnt += OPL->lfo_pm_inc;
+ OPL->LFO_PM = ((OPL->lfo_pm_cnt>>LFO_SH) & 7) | OPL->lfo_pm_depth_range;
+}
+
+/* advance to next sample */
+static void advance(FM_OPL *OPL)
+{
+ OPL_CH *CH;
+ OPL_SLOT *op;
+ int i;
+
+ OPL->eg_timer += OPL->eg_timer_add;
+
+ while (OPL->eg_timer >= OPL->eg_timer_overflow)
+ {
+ OPL->eg_timer -= OPL->eg_timer_overflow;
+
+ OPL->eg_cnt++;
+
+ for (i=0; i<9*2; i++)
+ {
+ CH = &OPL->P_CH[i/2];
+ op = &CH->SLOT[i&1];
+
+ /* Envelope Generator */
+ switch(op->state)
+ {
+ case EG_ATT: /* attack phase */
+ if ( !(OPL->eg_cnt & ((1<<op->eg_sh_ar)-1) ) )
+ {
+ /* we want sign extension here */
+ op->volume += (s32int)(~op->volume *
+ (eg_inc[op->eg_sel_ar + ((OPL->eg_cnt>>op->eg_sh_ar)&7)])
+ ) >>3;
+
+ if (op->volume <= MIN_ATT_INDEX)
+ {
+ op->volume = MIN_ATT_INDEX;
+ op->state = EG_DEC;
+ }
+
+ }
+ break;
+
+ case EG_DEC: /* decay phase */
+ if ( !(OPL->eg_cnt & ((1<<op->eg_sh_dr)-1) ) )
+ {
+ op->volume += eg_inc[op->eg_sel_dr + ((OPL->eg_cnt>>op->eg_sh_dr)&7)];
+
+ if ( op->volume >= op->sl )
+ op->state = EG_SUS;
+
+ }
+ break;
+
+ case EG_SUS: /* sustain phase */
+
+ /* this is important behaviour:
+ one can change percusive/non-percussive modes on the fly and
+ the chip will remain in sustain phase - verified on real YM3812 */
+
+ if(op->eg_type) /* non-percussive mode */
+ {
+ /* do nothing */
+ }
+ else /* percussive mode */
+ {
+ /* during sustain phase chip adds Release Rate (in percussive mode) */
+ if ( !(OPL->eg_cnt & ((1<<op->eg_sh_rr)-1) ) )
+ {
+ op->volume += eg_inc[op->eg_sel_rr + ((OPL->eg_cnt>>op->eg_sh_rr)&7)];
+
+ if ( op->volume >= MAX_ATT_INDEX )
+ op->volume = MAX_ATT_INDEX;
+ }
+ /* else do nothing in sustain phase */
+ }
+ break;
+
+ case EG_REL: /* release phase */
+ if ( !(OPL->eg_cnt & ((1<<op->eg_sh_rr)-1) ) )
+ {
+ op->volume += eg_inc[op->eg_sel_rr + ((OPL->eg_cnt>>op->eg_sh_rr)&7)];
+
+ if ( op->volume >= MAX_ATT_INDEX )
+ {
+ op->volume = MAX_ATT_INDEX;
+ op->state = EG_OFF;
+ }
+
+ }
+ break;
+
+ default:
+ break;
+ }
+ }
+ }
+
+ for (i=0; i<9*2; i++)
+ {
+ CH = &OPL->P_CH[i/2];
+ op = &CH->SLOT[i&1];
+
+ /* Phase Generator */
+ if(op->vib)
+ {
+ u8int block;
+ unsigned int block_fnum = CH->block_fnum;
+
+ unsigned int fnum_lfo = (block_fnum&0x0380) >> 7;
+
+ signed int lfo_fn_table_index_offset = lfo_pm_table[OPL->LFO_PM + 16*fnum_lfo ];
+
+ if (lfo_fn_table_index_offset) /* LFO phase modulation active */
+ {
+ block_fnum += lfo_fn_table_index_offset;
+ block = (block_fnum&0x1c00) >> 10;
+ op->Cnt += (OPL->fn_tab[block_fnum&0x03ff] >> (7-block)) * op->mul;
+ }
+ else /* LFO phase modulation = zero */
+ {
+ op->Cnt += op->Incr;
+ }
+ }
+ else /* LFO phase modulation disabled for this operator */
+ {
+ op->Cnt += op->Incr;
+ }
+ }
+
+ /* The Noise Generator of the YM3812 is 23-bit shift register.
+ * Period is equal to 2^23-2 samples.
+ * Register works at sampling frequency of the chip, so output
+ * can change on every sample.
+ *
+ * Output of the register and input to the bit 22 is:
+ * bit0 XOR bit14 XOR bit15 XOR bit22
+ *
+ * Simply use bit 22 as the noise output.
+ */
+
+ OPL->noise_p += OPL->noise_f;
+ i = OPL->noise_p >> FREQ_SH; /* number of events (shifts of the shift register) */
+ OPL->noise_p &= FREQ_MASK;
+ while (i)
+ {
+ /*
+ u32int j;
+ j = ( (OPL->noise_rng) ^ (OPL->noise_rng>>14) ^ (OPL->noise_rng>>15) ^ (OPL->noise_rng>>22) ) & 1;
+ OPL->noise_rng = (j<<22) | (OPL->noise_rng>>1);
+ */
+
+ /*
+ Instead of doing all the logic operations above, we
+ use a trick here (and use bit 0 as the noise output).
+ The difference is only that the noise bit changes one
+ step ahead. This doesn't matter since we don't know
+ what is real state of the noise_rng after the reset.
+ */
+
+ if (OPL->noise_rng & 1) OPL->noise_rng ^= 0x800302;
+ OPL->noise_rng >>= 1;
+
+ i--;
+ }
+}
+
+
+static signed int op_calc(u32int phase, unsigned int env, signed int pm, unsigned int wave_tab)
+{
+ u32int p;
+
+ p = (env<<4) + sin_tab[wave_tab + ((((signed int)((phase & ~FREQ_MASK) + (pm<<16))) >> FREQ_SH ) & SIN_MASK) ];
+
+ if (p >= TL_TAB_LEN)
+ return 0;
+ return tl_tab[p];
+}
+
+static signed int op_calc1(u32int phase, unsigned int env, signed int pm, unsigned int wave_tab)
+{
+ u32int p;
+
+ p = (env<<4) + sin_tab[wave_tab + ((((signed int)((phase & ~FREQ_MASK) + pm )) >> FREQ_SH ) & SIN_MASK) ];
+
+ if (p >= TL_TAB_LEN)
+ return 0;
+ return tl_tab[p];
+}
+
+
+#define volume_calc(OP) ((OP)->TLL + ((u32int)(OP)->volume) + (OPL->LFO_AM & (OP)->AMmask))
+
+/* calculate output */
+static void OPL_CALC_CH( FM_OPL *OPL, OPL_CH *CH )
+{
+ OPL_SLOT *SLOT;
+ unsigned int env;
+ signed int out;
+
+ OPL->phase_modulation = 0;
+
+ /* SLOT 1 */
+ SLOT = &CH->SLOT[SLOT1];
+ env = volume_calc(SLOT);
+ out = SLOT->op1_out[0] + SLOT->op1_out[1];
+ SLOT->op1_out[0] = SLOT->op1_out[1];
+ *SLOT->connect1 += SLOT->op1_out[0];
+ SLOT->op1_out[1] = 0;
+ if( env < ENV_QUIET )
+ {
+ if (!SLOT->FB)
+ out = 0;
+ SLOT->op1_out[1] = op_calc1(SLOT->Cnt, env, (out<<SLOT->FB), SLOT->wavetable );
+ }
+
+ /* SLOT 2 */
+ SLOT++;
+ env = volume_calc(SLOT);
+ if( env < ENV_QUIET )
+ OPL->output[0] += op_calc(SLOT->Cnt, env, OPL->phase_modulation, SLOT->wavetable);
+}
+
+/*
+ operators used in the rhythm sounds generation process:
+
+ Envelope Generator:
+
+channel operator register number Bass High Snare Tom Top
+/ slot number TL ARDR SLRR Wave Drum Hat Drum Tom Cymbal
+ 6 / 0 12 50 70 90 f0 +
+ 6 / 1 15 53 73 93 f3 +
+ 7 / 0 13 51 71 91 f1 +
+ 7 / 1 16 54 74 94 f4 +
+ 8 / 0 14 52 72 92 f2 +
+ 8 / 1 17 55 75 95 f5 +
+
+ Phase Generator:
+
+channel operator register number Bass High Snare Tom Top
+/ slot number MULTIPLE Drum Hat Drum Tom Cymbal
+ 6 / 0 12 30 +
+ 6 / 1 15 33 +
+ 7 / 0 13 31 + + +
+ 7 / 1 16 34 ----- n o t u s e d -----
+ 8 / 0 14 32 +
+ 8 / 1 17 35 + +
+
+channel operator register number Bass High Snare Tom Top
+number number BLK/FNUM2 FNUM Drum Hat Drum Tom Cymbal
+ 6 12,15 B6 A6 +
+
+ 7 13,16 B7 A7 + + +
+
+ 8 14,17 B8 A8 + + +
+
+*/
+
+/* calculate rhythm */
+
+static void OPL_CALC_RH( FM_OPL *OPL, OPL_CH *CH, unsigned int noise )
+{
+ OPL_SLOT *SLOT;
+ signed int out;
+ unsigned int env;
+
+
+ /* Bass Drum (verified on real YM3812):
+ - depends on the channel 6 'connect' register:
+ when connect = 0 it works the same as in normal (non-rhythm) mode (op1->op2->out)
+ when connect = 1 _only_ operator 2 is present on output (op2->out), operator 1 is ignored
+ - output sample always is multiplied by 2
+ */
+
+ OPL->phase_modulation = 0;
+ /* SLOT 1 */
+ SLOT = &CH[6].SLOT[SLOT1];
+ env = volume_calc(SLOT);
+
+ out = SLOT->op1_out[0] + SLOT->op1_out[1];
+ SLOT->op1_out[0] = SLOT->op1_out[1];
+
+ if (!SLOT->CON)
+ OPL->phase_modulation = SLOT->op1_out[0];
+ /* else ignore output of operator 1 */
+
+ SLOT->op1_out[1] = 0;
+ if( env < ENV_QUIET )
+ {
+ if (!SLOT->FB)
+ out = 0;
+ SLOT->op1_out[1] = op_calc1(SLOT->Cnt, env, (out<<SLOT->FB), SLOT->wavetable );
+ }
+
+ /* SLOT 2 */
+ SLOT++;
+ env = volume_calc(SLOT);
+ if( env < ENV_QUIET )
+ OPL->output[0] += op_calc(SLOT->Cnt, env, OPL->phase_modulation, SLOT->wavetable) * 2;
+
+
+ /* Phase generation is based on: */
+ /* HH (13) channel 7->slot 1 combined with channel 8->slot 2 (same combination as TOP CYMBAL but different output phases) */
+ /* SD (16) channel 7->slot 1 */
+ /* TOM (14) channel 8->slot 1 */
+ /* TOP (17) channel 7->slot 1 combined with channel 8->slot 2 (same combination as HIGH HAT but different output phases) */
+
+ /* Envelope generation based on: */
+ /* HH channel 7->slot1 */
+ /* SD channel 7->slot2 */
+ /* TOM channel 8->slot1 */
+ /* TOP channel 8->slot2 */
+
+
+ /* The following formulas can be well optimized.
+ I leave them in direct form for now (in case I've missed something).
+ */
+
+ /* High Hat (verified on real YM3812) */
+ env = volume_calc(SLOT7_1);
+ if( env < ENV_QUIET )
+ {
+ /* high hat phase generation:
+ phase = d0 or 234 (based on frequency only)
+ phase = 34 or 2d0 (based on noise)
+ */
+
+ /* base frequency derived from operator 1 in channel 7 */
+ unsigned char bit7 = ((SLOT7_1->Cnt>>FREQ_SH)>>7)&1;
+ unsigned char bit3 = ((SLOT7_1->Cnt>>FREQ_SH)>>3)&1;
+ unsigned char bit2 = ((SLOT7_1->Cnt>>FREQ_SH)>>2)&1;
+
+ unsigned char res1 = (bit2 ^ bit7) | bit3;
+
+ /* when res1 = 0 phase = 0x000 | 0xd0; */
+ /* when res1 = 1 phase = 0x200 | (0xd0>>2); */
+ u32int phase = res1 ? (0x200|(0xd0>>2)) : 0xd0;
+
+ /* enable gate based on frequency of operator 2 in channel 8 */
+ unsigned char bit5e= ((SLOT8_2->Cnt>>FREQ_SH)>>5)&1;
+ unsigned char bit3e= ((SLOT8_2->Cnt>>FREQ_SH)>>3)&1;
+
+ unsigned char res2 = (bit3e ^ bit5e);
+
+ /* when res2 = 0 pass the phase from calculation above (res1); */
+ /* when res2 = 1 phase = 0x200 | (0xd0>>2); */
+ if (res2)
+ phase = (0x200|(0xd0>>2));
+
+
+ /* when phase & 0x200 is set and noise=1 then phase = 0x200|0xd0 */
+ /* when phase & 0x200 is set and noise=0 then phase = 0x200|(0xd0>>2), ie no change */
+ if (phase&0x200)
+ {
+ if (noise)
+ phase = 0x200|0xd0;
+ }
+ else
+ /* when phase & 0x200 is clear and noise=1 then phase = 0xd0>>2 */
+ /* when phase & 0x200 is clear and noise=0 then phase = 0xd0, ie no change */
+ {
+ if (noise)
+ phase = 0xd0>>2;
+ }
+
+ OPL->output[0] += op_calc(phase<<FREQ_SH, env, 0, SLOT7_1->wavetable) * 2;
+ }
+
+ /* Snare Drum (verified on real YM3812) */
+ env = volume_calc(SLOT7_2);
+ if( env < ENV_QUIET )
+ {
+ /* base frequency derived from operator 1 in channel 7 */
+ unsigned char bit8 = ((SLOT7_1->Cnt>>FREQ_SH)>>8)&1;
+
+ /* when bit8 = 0 phase = 0x100; */
+ /* when bit8 = 1 phase = 0x200; */
+ u32int phase = bit8 ? 0x200 : 0x100;
+
+ /* Noise bit XOR'es phase by 0x100 */
+ /* when noisebit = 0 pass the phase from calculation above */
+ /* when noisebit = 1 phase ^= 0x100; */
+ /* in other words: phase ^= (noisebit<<8); */
+ if (noise)
+ phase ^= 0x100;
+
+ OPL->output[0] += op_calc(phase<<FREQ_SH, env, 0, SLOT7_2->wavetable) * 2;
+ }
+
+ /* Tom Tom (verified on real YM3812) */
+ env = volume_calc(SLOT8_1);
+ if( env < ENV_QUIET )
+ OPL->output[0] += op_calc(SLOT8_1->Cnt, env, 0, SLOT8_1->wavetable) * 2;
+
+ /* Top Cymbal (verified on real YM3812) */
+ env = volume_calc(SLOT8_2);
+ if( env < ENV_QUIET )
+ {
+ /* base frequency derived from operator 1 in channel 7 */
+ unsigned char bit7 = ((SLOT7_1->Cnt>>FREQ_SH)>>7)&1;
+ unsigned char bit3 = ((SLOT7_1->Cnt>>FREQ_SH)>>3)&1;
+ unsigned char bit2 = ((SLOT7_1->Cnt>>FREQ_SH)>>2)&1;
+
+ unsigned char res1 = (bit2 ^ bit7) | bit3;
+
+ /* when res1 = 0 phase = 0x000 | 0x100; */
+ /* when res1 = 1 phase = 0x200 | 0x100; */
+ u32int phase = res1 ? 0x300 : 0x100;
+
+ /* enable gate based on frequency of operator 2 in channel 8 */
+ unsigned char bit5e= ((SLOT8_2->Cnt>>FREQ_SH)>>5)&1;
+ unsigned char bit3e= ((SLOT8_2->Cnt>>FREQ_SH)>>3)&1;
+
+ unsigned char res2 = (bit3e ^ bit5e);
+ /* when res2 = 0 pass the phase from calculation above (res1); */
+ /* when res2 = 1 phase = 0x200 | 0x100; */
+ if (res2)
+ phase = 0x300;
+
+ OPL->output[0] += op_calc(phase<<FREQ_SH, env, 0, SLOT8_2->wavetable) * 2;
+ }
+}
+
+
+/* generic table initialize */
+static int init_tables(void)
+{
+ signed int i,x;
+ signed int n;
+ double o,m;
+
+
+ for (x=0; x<TL_RES_LEN; x++)
+ {
+ m = (1<<16) / pow(2, (x+1) * (ENV_STEP/4.0) / 8.0);
+ m = floor(m);
+
+ /* we never reach (1<<16) here due to the (x+1) */
+ /* result fits within 16 bits at maximum */
+
+ n = (int)m; /* 16 bits here */
+ n >>= 4; /* 12 bits here */
+ if (n&1) /* round to nearest */
+ n = (n>>1)+1;
+ else
+ n = n>>1;
+ /* 11 bits here (rounded) */
+ n <<= 1; /* 12 bits here (as in real chip) */
+ tl_tab[ x*2 + 0 ] = n;
+ tl_tab[ x*2 + 1 ] = -tl_tab[ x*2 + 0 ];
+
+ for (i=1; i<12; i++)
+ {
+ tl_tab[ x*2+0 + i*2*TL_RES_LEN ] = tl_tab[ x*2+0 ]>>i;
+ tl_tab[ x*2+1 + i*2*TL_RES_LEN ] = -tl_tab[ x*2+0 + i*2*TL_RES_LEN ];
+ }
+ }
+
+ for (i=0; i<SIN_LEN; i++)
+ {
+ /* non-standard sinus */
+ m = sin( ((i*2)+1) * PI / SIN_LEN ); /* checked against the real chip */
+
+ /* we never reach zero here due to ((i*2)+1) */
+
+ if (m>0.0)
+ o = 8*log(1.0/m)/log(2.0); /* convert to 'decibels' */
+ else
+ o = 8*log(-1.0/m)/log(2.0); /* convert to 'decibels' */
+
+ o = o / (ENV_STEP/4);
+
+ n = (int)(2.0*o);
+ if (n&1) /* round to nearest */
+ n = (n>>1)+1;
+ else
+ n = n>>1;
+
+ sin_tab[ i ] = n*2 + (m>=0.0? 0: 1 );
+ }
+
+ for (i=0; i<SIN_LEN; i++)
+ {
+ /* waveform 1: __ __ */
+ /* / \____/ \____*/
+ /* output only first half of the sinus waveform (positive one) */
+
+ if (i & (1<<(SIN_BITS-1)) )
+ sin_tab[1*SIN_LEN+i] = TL_TAB_LEN;
+ else
+ sin_tab[1*SIN_LEN+i] = sin_tab[i];
+
+ /* waveform 2: __ __ __ __ */
+ /* / \/ \/ \/ \*/
+ /* abs(sin) */
+
+ sin_tab[2*SIN_LEN+i] = sin_tab[i & (SIN_MASK>>1) ];
+
+ /* waveform 3: _ _ _ _ */
+ /* / |_/ |_/ |_/ |_*/
+ /* abs(output only first quarter of the sinus waveform) */
+
+ if (i & (1<<(SIN_BITS-2)) )
+ sin_tab[3*SIN_LEN+i] = TL_TAB_LEN;
+ else
+ sin_tab[3*SIN_LEN+i] = sin_tab[i & (SIN_MASK>>2)];
+ }
+
+ return 1;
+}
+
+static void OPL_initalize(FM_OPL *OPL)
+{
+ int i;
+
+ /* frequency base */
+ OPL->freqbase = (OPL->rate) ? ((double)OPL->clock / 72.0) / OPL->rate : 0;
+/*
+ OPL->rate = (double)OPL->clock / 72.0;
+ OPL->freqbase = 1.0;
+*/
+
+ /* Timer base time */
+ OPL->TimerBase = 1.0 / ((double)OPL->clock / 72.0);
+
+ /* make fnumber -> increment counter table */
+ for( i=0 ; i < 1024 ; i++ )
+ {
+ /* opn phase increment counter = 20bit */
+ OPL->fn_tab[i] = (u32int)( (double)i * 64 * OPL->freqbase * (1<<(FREQ_SH-10)) ); /* -10 because chip works with 10.10 fixed point, while we use 16.16 */
+ }
+
+ /* Amplitude modulation: 27 output levels (triangle waveform); 1 level takes one of: 192, 256 or 448 samples */
+ /* One entry from LFO_AM_TABLE lasts for 64 samples */
+ OPL->lfo_am_inc = (1.0 / 64.0 ) * (1<<LFO_SH) * OPL->freqbase;
+
+ /* Vibrato: 8 output levels (triangle waveform); 1 level takes 1024 samples */
+ OPL->lfo_pm_inc = (1.0 / 1024.0) * (1<<LFO_SH) * OPL->freqbase;
+
+ /* Noise generator: a step takes 1 sample */
+ OPL->noise_f = (1.0 / 1.0) * (1<<FREQ_SH) * OPL->freqbase;
+
+ OPL->eg_timer_add = (1<<EG_SH) * OPL->freqbase;
+ OPL->eg_timer_overflow = ( 1 ) * (1<<EG_SH);
+}
+
+static void FM_KEYON(OPL_SLOT *SLOT, u32int key_set)
+{
+ if( !SLOT->key )
+ {
+ /* restart Phase Generator */
+ SLOT->Cnt = 0;
+ /* phase -> Attack */
+ SLOT->state = EG_ATT;
+ }
+ SLOT->key |= key_set;
+}
+
+static void FM_KEYOFF(OPL_SLOT *SLOT, u32int key_clr)
+{
+ if( SLOT->key )
+ {
+ SLOT->key &= key_clr;
+
+ if( !SLOT->key )
+ {
+ /* phase -> Release */
+ if (SLOT->state>EG_REL)
+ SLOT->state = EG_REL;
+ }
+ }
+}
+
+/* update phase increment counter of operator (also update the EG rates if necessary) */
+static void CALC_FCSLOT(OPL_CH *CH,OPL_SLOT *SLOT)
+{
+ int ksr;
+
+ /* (frequency) phase increment counter */
+ SLOT->Incr = CH->fc * SLOT->mul;
+ ksr = CH->kcode >> SLOT->KSR;
+
+ if( SLOT->ksr != ksr )
+ {
+ SLOT->ksr = ksr;
+
+ /* calculate envelope generator rates */
+ if ((SLOT->ar + SLOT->ksr) < 16+62)
+ {
+ SLOT->eg_sh_ar = eg_rate_shift [SLOT->ar + SLOT->ksr ];
+ SLOT->eg_sel_ar = eg_rate_select[SLOT->ar + SLOT->ksr ];
+ }
+ else
+ {
+ SLOT->eg_sh_ar = 0;
+ SLOT->eg_sel_ar = 13*RATE_STEPS;
+ }
+ SLOT->eg_sh_dr = eg_rate_shift [SLOT->dr + SLOT->ksr ];
+ SLOT->eg_sel_dr = eg_rate_select[SLOT->dr + SLOT->ksr ];
+ SLOT->eg_sh_rr = eg_rate_shift [SLOT->rr + SLOT->ksr ];
+ SLOT->eg_sel_rr = eg_rate_select[SLOT->rr + SLOT->ksr ];
+ }
+}
+
+/* set multi,am,vib,EG-TYP,KSR,mul */
+static void set_mul(FM_OPL *OPL,int slot,int v)
+{
+ OPL_CH *CH = &OPL->P_CH[slot/2];
+ OPL_SLOT *SLOT = &CH->SLOT[slot&1];
+
+ SLOT->mul = mul_tab[v&0x0f];
+ SLOT->KSR = (v&0x10) ? 0 : 2;
+ SLOT->eg_type = (v&0x20);
+ SLOT->vib = (v&0x40);
+ SLOT->AMmask = (v&0x80) ? ~0 : 0;
+ CALC_FCSLOT(CH,SLOT);
+}
+
+/* set ksl & tl */
+static void set_ksl_tl(FM_OPL *OPL,int slot,int v)
+{
+ OPL_CH *CH = &OPL->P_CH[slot/2];
+ OPL_SLOT *SLOT = &CH->SLOT[slot&1];
+
+ SLOT->ksl = ksl_shift[v >> 6];
+ SLOT->TL = (v&0x3f)<<(ENV_BITS-1-7); /* 7 bits TL (bit 6 = always 0) */
+
+ SLOT->TLL = SLOT->TL + (CH->ksl_base>>SLOT->ksl);
+}
+
+/* set attack rate & decay rate */
+static void set_ar_dr(FM_OPL *OPL,int slot,int v)
+{
+ OPL_CH *CH = &OPL->P_CH[slot/2];
+ OPL_SLOT *SLOT = &CH->SLOT[slot&1];
+
+ SLOT->ar = (v>>4) ? 16 + ((v>>4) <<2) : 0;
+
+ if ((SLOT->ar + SLOT->ksr) < 16+62)
+ {
+ SLOT->eg_sh_ar = eg_rate_shift [SLOT->ar + SLOT->ksr ];
+ SLOT->eg_sel_ar = eg_rate_select[SLOT->ar + SLOT->ksr ];
+ }
+ else
+ {
+ SLOT->eg_sh_ar = 0;
+ SLOT->eg_sel_ar = 13*RATE_STEPS;
+ }
+
+ SLOT->dr = (v&0x0f)? 16 + ((v&0x0f)<<2) : 0;
+ SLOT->eg_sh_dr = eg_rate_shift [SLOT->dr + SLOT->ksr ];
+ SLOT->eg_sel_dr = eg_rate_select[SLOT->dr + SLOT->ksr ];
+}
+
+/* set sustain level & release rate */
+static void set_sl_rr(FM_OPL *OPL,int slot,int v)
+{
+ OPL_CH *CH = &OPL->P_CH[slot/2];
+ OPL_SLOT *SLOT = &CH->SLOT[slot&1];
+
+ SLOT->sl = sl_tab[ v>>4 ];
+
+ SLOT->rr = (v&0x0f)? 16 + ((v&0x0f)<<2) : 0;
+ SLOT->eg_sh_rr = eg_rate_shift [SLOT->rr + SLOT->ksr ];
+ SLOT->eg_sel_rr = eg_rate_select[SLOT->rr + SLOT->ksr ];
+}
+
+
+/* write a value v to register r on OPL chip */
+static void OPLWriteReg(FM_OPL *OPL, int r, int v)
+{
+ OPL_CH *CH;
+ int slot;
+ int block_fnum;
+
+
+ /* adjust bus to 8 bits */
+ r &= 0xff;
+ v &= 0xff;
+
+ switch(r&0xe0)
+ {
+ case 0x00: /* 00-1f:control */
+ switch(r&0x1f)
+ {
+ case 0x01: /* waveform select enable */
+ if(OPL->type&OPL_TYPE_WAVESEL)
+ {
+ OPL->wavesel = v&0x20;
+ /* do not change the waveform previously selected */
+ }
+ break;
+ case 0x02: /* Timer 1 */
+ OPL->T[0] = (256-v)*4;
+ break;
+ case 0x03: /* Timer 2 */
+ OPL->T[1] = (256-v)*16;
+ break;
+ case 0x04: /* IRQ clear / mask and Timer enable */
+ if(v&0x80)
+ { /* IRQ flag clear */
+ OPL_STATUS_RESET(OPL,0x7f-0x08); /* don't reset BFRDY flag or we will have to call deltat module to set the flag */
+ }
+ else
+ { /* set IRQ mask ,timer enable*/
+ u8int st1 = v&1;
+ u8int st2 = (v>>1)&1;
+
+ /* IRQRST,T1MSK,t2MSK,EOSMSK,BRMSK,x,ST2,ST1 */
+ OPL_STATUS_RESET(OPL, v & (0x78-0x08) );
+ OPL_STATUSMASK_SET(OPL, (~v) & 0x78 );
+
+ /* timer 2 */
+ if(OPL->st[1] != st2)
+ {
+ double period = st2 ? (OPL->TimerBase * OPL->T[1]) : 0;
+ OPL->st[1] = st2;
+ if (OPL->timer_handler) (OPL->timer_handler)(OPL->TimerParam,1,period);
+ }
+ /* timer 1 */
+ if(OPL->st[0] != st1)
+ {
+ double period = st1 ? (OPL->TimerBase * OPL->T[0]) : 0;
+ OPL->st[0] = st1;
+ if (OPL->timer_handler) (OPL->timer_handler)(OPL->TimerParam,0,period);
+ }
+ }
+ break;
+ case 0x08: /* MODE,DELTA-T control 2 : CSM,NOTESEL,x,x,smpl,da/ad,64k,rom */
+ OPL->mode = v;
+ break;
+ default:
+ fprint(2, "write to unknown register: %02x\n",r);
+ break;
+ }
+ break;
+ case 0x20: /* am ON, vib ON, ksr, eg_type, mul */
+ slot = slot_array[r&0x1f];
+ if(slot < 0) return;
+ set_mul(OPL,slot,v);
+ break;
+ case 0x40:
+ slot = slot_array[r&0x1f];
+ if(slot < 0) return;
+ set_ksl_tl(OPL,slot,v);
+ break;
+ case 0x60:
+ slot = slot_array[r&0x1f];
+ if(slot < 0) return;
+ set_ar_dr(OPL,slot,v);
+ break;
+ case 0x80:
+ slot = slot_array[r&0x1f];
+ if(slot < 0) return;
+ set_sl_rr(OPL,slot,v);
+ break;
+ case 0xa0:
+ if (r == 0xbd) /* am depth, vibrato depth, r,bd,sd,tom,tc,hh */
+ {
+ OPL->lfo_am_depth = v & 0x80;
+ OPL->lfo_pm_depth_range = (v&0x40) ? 8 : 0;
+
+ OPL->rhythm = v&0x3f;
+
+ if(OPL->rhythm&0x20)
+ {
+ /* BD key on/off */
+ if(v&0x10)
+ {
+ FM_KEYON (&OPL->P_CH[6].SLOT[SLOT1], 2);
+ FM_KEYON (&OPL->P_CH[6].SLOT[SLOT2], 2);
+ }
+ else
+ {
+ FM_KEYOFF(&OPL->P_CH[6].SLOT[SLOT1],~2);
+ FM_KEYOFF(&OPL->P_CH[6].SLOT[SLOT2],~2);
+ }
+ /* HH key on/off */
+ if(v&0x01) FM_KEYON (&OPL->P_CH[7].SLOT[SLOT1], 2);
+ else FM_KEYOFF(&OPL->P_CH[7].SLOT[SLOT1],~2);
+ /* SD key on/off */
+ if(v&0x08) FM_KEYON (&OPL->P_CH[7].SLOT[SLOT2], 2);
+ else FM_KEYOFF(&OPL->P_CH[7].SLOT[SLOT2],~2);
+ /* TOM key on/off */
+ if(v&0x04) FM_KEYON (&OPL->P_CH[8].SLOT[SLOT1], 2);
+ else FM_KEYOFF(&OPL->P_CH[8].SLOT[SLOT1],~2);
+ /* TOP-CY key on/off */
+ if(v&0x02) FM_KEYON (&OPL->P_CH[8].SLOT[SLOT2], 2);
+ else FM_KEYOFF(&OPL->P_CH[8].SLOT[SLOT2],~2);
+ }
+ else
+ {
+ /* BD key off */
+ FM_KEYOFF(&OPL->P_CH[6].SLOT[SLOT1],~2);
+ FM_KEYOFF(&OPL->P_CH[6].SLOT[SLOT2],~2);
+ /* HH key off */
+ FM_KEYOFF(&OPL->P_CH[7].SLOT[SLOT1],~2);
+ /* SD key off */
+ FM_KEYOFF(&OPL->P_CH[7].SLOT[SLOT2],~2);
+ /* TOM key off */
+ FM_KEYOFF(&OPL->P_CH[8].SLOT[SLOT1],~2);
+ /* TOP-CY off */
+ FM_KEYOFF(&OPL->P_CH[8].SLOT[SLOT2],~2);
+ }
+ return;
+ }
+ /* keyon,block,fnum */
+ if( (r&0x0f) > 8) return;
+ CH = &OPL->P_CH[r&0x0f];
+ if(!(r&0x10))
+ { /* a0-a8 */
+ block_fnum = (CH->block_fnum&0x1f00) | v;
+ }
+ else
+ { /* b0-b8 */
+ block_fnum = ((v&0x1f)<<8) | (CH->block_fnum&0xff);
+
+ if(v&0x20)
+ {
+ FM_KEYON (&CH->SLOT[SLOT1], 1);
+ FM_KEYON (&CH->SLOT[SLOT2], 1);
+ }
+ else
+ {
+ FM_KEYOFF(&CH->SLOT[SLOT1],~1);
+ FM_KEYOFF(&CH->SLOT[SLOT2],~1);
+ }
+ }
+ /* update */
+ if(CH->block_fnum != block_fnum)
+ {
+ u8int block = block_fnum >> 10;
+
+ CH->block_fnum = block_fnum;
+
+ CH->ksl_base = ksl_tab[block_fnum>>6];
+ CH->fc = OPL->fn_tab[block_fnum&0x03ff] >> (7-block);
+
+ /* BLK 2,1,0 bits -> bits 3,2,1 of kcode */
+ CH->kcode = (CH->block_fnum&0x1c00)>>9;
+
+ /* the info below is actually opposite to what is stated in the Manuals (verifed on real YM3812) */
+ /* if notesel == 0 -> lsb of kcode is bit 10 (MSB) of fnum */
+ /* if notesel == 1 -> lsb of kcode is bit 9 (MSB-1) of fnum */
+ if (OPL->mode&0x40)
+ CH->kcode |= (CH->block_fnum&0x100)>>8; /* notesel == 1 */
+ else
+ CH->kcode |= (CH->block_fnum&0x200)>>9; /* notesel == 0 */
+
+ /* refresh Total Level in both SLOTs of this channel */
+ CH->SLOT[SLOT1].TLL = CH->SLOT[SLOT1].TL + (CH->ksl_base>>CH->SLOT[SLOT1].ksl);
+ CH->SLOT[SLOT2].TLL = CH->SLOT[SLOT2].TL + (CH->ksl_base>>CH->SLOT[SLOT2].ksl);
+
+ /* refresh frequency counter in both SLOTs of this channel */
+ CALC_FCSLOT(CH,&CH->SLOT[SLOT1]);
+ CALC_FCSLOT(CH,&CH->SLOT[SLOT2]);
+ }
+ break;
+ case 0xc0:
+ /* FB,C */
+ if( (r&0x0f) > 8) return;
+ CH = &OPL->P_CH[r&0x0f];
+ CH->SLOT[SLOT1].FB = (v>>1)&7 ? ((v>>1)&7) + 7 : 0;
+ CH->SLOT[SLOT1].CON = v&1;
+ CH->SLOT[SLOT1].connect1 = CH->SLOT[SLOT1].CON ? &OPL->output[0] : &OPL->phase_modulation;
+ break;
+ case 0xe0: /* waveform select */
+ /* simply ignore write to the waveform select register if selecting not enabled in test register */
+ if(OPL->wavesel)
+ {
+ slot = slot_array[r&0x1f];
+ if(slot < 0) return;
+ CH = &OPL->P_CH[slot/2];
+
+ CH->SLOT[slot&1].wavetable = (v&0x03)*SIN_LEN;
+ }
+ break;
+ }
+}
+
+/* lock/unlock for common table */
+static int OPL_LockTable(void)
+{
+ num_lock++;
+ if(num_lock>1) return 0;
+
+ /* first time */
+
+ /* allocate total level table (128kb space) */
+ if( !init_tables() )
+ {
+ num_lock--;
+ return -1;
+ }
+ return 0;
+}
+
+static void OPL_UnLockTable(void)
+{
+ if(num_lock) num_lock--;
+ if(num_lock) return;
+}
+
+static void OPLResetChip(FM_OPL *OPL)
+{
+ int c,s;
+ int i;
+
+ OPL->eg_timer = 0;
+ OPL->eg_cnt = 0;
+
+ OPL->noise_rng = 1; /* noise shift register */
+ OPL->mode = 0; /* normal mode */
+ OPL_STATUS_RESET(OPL,0x7f);
+
+ /* reset with register write */
+ OPLWriteReg(OPL,0x01,0); /* wavesel disable */
+ OPLWriteReg(OPL,0x02,0); /* Timer1 */
+ OPLWriteReg(OPL,0x03,0); /* Timer2 */
+ OPLWriteReg(OPL,0x04,0); /* IRQ mask clear */
+ for(i = 0xff ; i >= 0x20 ; i-- ) OPLWriteReg(OPL,i,0);
+
+ /* reset operator parameters */
+ for( c = 0 ; c < 9 ; c++ )
+ {
+ OPL_CH *CH = &OPL->P_CH[c];
+ for(s = 0 ; s < 2 ; s++ )
+ {
+ /* wave table */
+ CH->SLOT[s].wavetable = 0;
+ CH->SLOT[s].state = EG_OFF;
+ CH->SLOT[s].volume = MAX_ATT_INDEX;
+ }
+ }
+}
+
+
+static void OPL_postload(FM_OPL *OPL)
+{
+ int slot, ch;
+
+ for( ch=0 ; ch < 9 ; ch++ )
+ {
+ OPL_CH *CH = &OPL->P_CH[ch];
+
+ /* Look up key scale level */
+ u32int block_fnum = CH->block_fnum;
+ CH->ksl_base = ksl_tab[block_fnum >> 6];
+ CH->fc = OPL->fn_tab[block_fnum & 0x03ff] >> (7 - (block_fnum >> 10));
+
+ for( slot=0 ; slot < 2 ; slot++ )
+ {
+ OPL_SLOT *SLOT = &CH->SLOT[slot];
+
+ /* Calculate key scale rate */
+ SLOT->ksr = CH->kcode >> SLOT->KSR;
+
+ /* Calculate attack, decay and release rates */
+ if ((SLOT->ar + SLOT->ksr) < 16+62)
+ {
+ SLOT->eg_sh_ar = eg_rate_shift [SLOT->ar + SLOT->ksr ];
+ SLOT->eg_sel_ar = eg_rate_select[SLOT->ar + SLOT->ksr ];
+ }
+ else
+ {
+ SLOT->eg_sh_ar = 0;
+ SLOT->eg_sel_ar = 13*RATE_STEPS;
+ }
+ SLOT->eg_sh_dr = eg_rate_shift [SLOT->dr + SLOT->ksr ];
+ SLOT->eg_sel_dr = eg_rate_select[SLOT->dr + SLOT->ksr ];
+ SLOT->eg_sh_rr = eg_rate_shift [SLOT->rr + SLOT->ksr ];
+ SLOT->eg_sel_rr = eg_rate_select[SLOT->rr + SLOT->ksr ];
+
+ /* Calculate phase increment */
+ SLOT->Incr = CH->fc * SLOT->mul;
+
+ /* Total level */
+ SLOT->TLL = SLOT->TL + (CH->ksl_base >> SLOT->ksl);
+
+ /* Connect output */
+ SLOT->connect1 = SLOT->CON ? &OPL->output[0] : &OPL->phase_modulation;
+ }
+ }
+}
+
+static FM_OPL *
+OPLCreate(u32int clock, u32int rate, int type)
+{
+ FM_OPL *p;
+
+ if(OPL_LockTable() == -1)
+ sysfatal("OPLCreate: locktable");
+
+ /* allocate memory block */
+ if((p = mallocz(sizeof *p, 1)) == nil)
+ sysfatal("mallocz: %r");
+ p->type = type;
+ p->clock = clock; /* Hz */
+ p->rate = rate;
+
+ /* init global tables */
+ OPL_initalize(p);
+
+ return p;
+}
+
+/* Destroy one of virtual YM3812 */
+static void OPLDestroy(FM_OPL *OPL)
+{
+ OPL_UnLockTable();
+ free(OPL);
+}
+
+/* Optional handlers */
+
+static void OPLSetTimerHandler(FM_OPL *OPL,OPL_TIMERHANDLER timer_handler,void *param)
+{
+ OPL->timer_handler = timer_handler;
+ OPL->TimerParam = param;
+}
+static void OPLSetIRQHandler(FM_OPL *OPL,OPL_IRQHANDLER IRQHandler,void *param)
+{
+ OPL->IRQHandler = IRQHandler;
+ OPL->IRQParam = param;
+}
+static void OPLSetUpdateHandler(FM_OPL *OPL,OPL_UPDATEHANDLER UpdateHandler,void *param)
+{
+ OPL->UpdateHandler = UpdateHandler;
+ OPL->UpdateParam = param;
+}
+
+static int OPLWrite(FM_OPL *OPL,int a,int v)
+{
+ if( !(a&1) )
+ { /* address port */
+ OPL->address = v & 0xff;
+ }
+ else
+ { /* data port */
+ if(OPL->UpdateHandler) OPL->UpdateHandler(OPL->UpdateParam,0);
+ OPLWriteReg(OPL,OPL->address,v);
+ }
+ return OPL->status>>7;
+}
+
+static unsigned char OPLRead(FM_OPL *OPL,int a)
+{
+ if( !(a&1) )
+ {
+ /* status port */
+
+ /* OPL and OPL2 */
+ return OPL->status & (OPL->statusmask|0x80);
+ }
+ return 0xff;
+}
+
+/* CSM Key Controll */
+static void CSMKeyControll(OPL_CH *CH)
+{
+ FM_KEYON (&CH->SLOT[SLOT1], 4);
+ FM_KEYON (&CH->SLOT[SLOT2], 4);
+
+ /* The key off should happen exactly one sample later - not implemented correctly yet */
+
+ FM_KEYOFF(&CH->SLOT[SLOT1], ~4);
+ FM_KEYOFF(&CH->SLOT[SLOT2], ~4);
+}
+
+
+static int OPLTimerOver(FM_OPL *OPL,int c)
+{
+ if( c )
+ { /* Timer B */
+ OPL_STATUS_SET(OPL,0x20);
+ }
+ else
+ { /* Timer A */
+ OPL_STATUS_SET(OPL,0x40);
+ /* CSM mode key,TL controll */
+ if( OPL->mode & 0x80 )
+ { /* CSM mode total level latch and auto key on */
+ int ch;
+ if(OPL->UpdateHandler) OPL->UpdateHandler(OPL->UpdateParam,0);
+ for(ch=0; ch<9; ch++)
+ CSMKeyControll( &OPL->P_CH[ch] );
+ }
+ }
+ /* reload timer */
+ if (OPL->timer_handler)
+ (OPL->timer_handler)(OPL->TimerParam,c,OPL->TimerBase * OPL->T[c]);
+ return OPL->status>>7;
+}
+
+
+FM_OPL *
+ym3812_init(u32int clock, u32int rate)
+{
+ FM_OPL *p;
+
+ if((p = OPLCreate(clock, rate, OPL_TYPE_YM3812)) == nil)
+ sysfatal("OPLCreate: %r"); /* FIXME */
+ ym3812_reset_chip(p);
+ return p;
+}
+
+/* FIXME: remove abstractions; this is now opl2 only */
+void
+ym3812_shutdown(FM_OPL *p)
+{
+ OPLDestroy(p);
+}
+
+void
+ym3812_reset_chip(FM_OPL *p)
+{
+ OPLResetChip(p);
+}
+
+int
+ym3812_write(FM_OPL *p, int a, int v)
+{
+ return OPLWrite(p, a, v);
+}
+
+uchar
+ym3812_read(FM_OPL *p, int a)
+{
+ /* YM3812 always returns bit2 and bit1 in HIGH state */
+ return OPLRead(p, a) | 0x06 ;
+}
+
+int
+ym3812_timer_over(FM_OPL *p, int c)
+{
+ return OPLTimerOver(p, c);
+}
+
+void
+ym3812_set_timer_handler(FM_OPL *p, OPL_TIMERHANDLER timer_handler, void *param)
+{
+ OPLSetTimerHandler(p, timer_handler, param);
+}
+
+void
+ym3812_set_irq_handler(FM_OPL *p,OPL_IRQHANDLER IRQHandler, void *param)
+{
+ OPLSetIRQHandler(p, IRQHandler, param);
+}
+
+void
+ym3812_set_update_handler(FM_OPL *p, OPL_UPDATEHANDLER UpdateHandler, void *param)
+{
+ OPLSetUpdateHandler(p, UpdateHandler, param);
+}
+
+void
+ym3812_update_one(FM_OPL *p, OPLSAMPLE *buf, int n)
+{
+ int i, lt;
+ u8int rhythm;
+
+ rhythm = p->rhythm & 0x20;
+ for(i=0; i < n; i++){
+ p->output[0] = 0;
+
+ advance_lfo(p);
+
+ /* FM part */
+ OPL_CALC_CH(p, &p->P_CH[0]);
+ OPL_CALC_CH(p, &p->P_CH[1]);
+ OPL_CALC_CH(p, &p->P_CH[2]);
+ OPL_CALC_CH(p, &p->P_CH[3]);
+ OPL_CALC_CH(p, &p->P_CH[4]);
+ OPL_CALC_CH(p, &p->P_CH[5]);
+
+ if(!rhythm)
+ {
+ OPL_CALC_CH(p, &p->P_CH[6]);
+ OPL_CALC_CH(p, &p->P_CH[7]);
+ OPL_CALC_CH(p, &p->P_CH[8]);
+ }
+ else /* Rhythm part */
+ {
+ OPL_CALC_RH(p, &p->P_CH[0], (p->noise_rng>>0)&1 );
+ }
+
+ lt = p->output[0];
+
+ lt >>= FINAL_SH;
+
+ /* limit check */
+ lt = limit( lt , MAXOUT, MINOUT );
+
+ /* store to sound buffer */
+ buf[i] = lt;
+
+ advance(p);
+ }
+}
--- /dev/null
+++ b/fns.h
@@ -1,0 +1,10 @@
+FM_OPL* ym3812_init(u32int, u32int);
+void ym3812_shutdown(FM_OPL *);
+void ym3812_reset_chip(FM_OPL *);
+int ym3812_write(FM_OPL *, int, int);
+uchar ym3812_read(FM_OPL *, int);
+int ym3812_timer_over(FM_OPL *, int);
+void ym3812_update_one(FM_OPL *, OPLSAMPLE *, int);
+void ym3812_set_timer_handler(FM_OPL *, OPL_TIMERHANDLER, void *);
+void ym3812_set_irq_handler(FM_OPL *, OPL_IRQHANDLER, void *);
+void ym3812_set_update_handler(FM_OPL *, OPL_UPDATEHANDLER, void *);
--- /dev/null
+++ b/mkfile
@@ -1,0 +1,8 @@
+</$objtype/mkfile
+
+BIN=$home/bin/$objtype
+TARG=omidi
+OFILES= omidi.$O fmopl.$O
+HFILES= dat.h fns.h
+
+</sys/src/cmd/mkone
--- /dev/null
+++ b/omidi.c
@@ -1,0 +1,1230 @@
+#include <u.h>
+#include <libc.h>
+#include "dat.h"
+#include "fns.h"
+
+enum{
+ NCHIPS = 1,
+ RATE = 44100
+};
+/* FIXME: 8192 bytes or shorts? */
+static short abuf[8192];
+static short *ap;
+static int afd = -1;
+
+/* FIXME: using multiple chips */
+/* FIXME: the only reason this exists and FM_OPL is defined in dat.h instead of being
+ * static in fmopl.c is to have multiple cards... and we don't support that; either
+ * implement it and it's cool and it should be that way, or nuke this */
+static FM_OPL *chip[NCHIPS];
+static int oplretval;
+static int oplregno;
+static double otm;
+
+enum{
+ MINDELAY = 1,
+ OPLBASE = 0x388
+};
+
+static int verbose;
+
+static struct MusData mdat;
+
+int ColorNums = -1;
+
+const long Period[12] =
+{
+ 907,960,1016,1076,
+ 1140,1208,1280,1356,
+ 1440,1524,1616,1712
+};
+
+static const char Portit[9] = {0,1,2, 8,9,10, 16,17,18};
+
+static signed char Pans[18] = {0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0};
+
+signed char Trig[18] = {0,0,0,0,0,0,0,0,0, 0,0,0,0,0,0,0,0,0};
+
+//#define SetBase(a) ((a)==2 ? (a) : 9)
+#define SetBase(a) (a)
+
+static uchar FMVol[65] = {
+ /* Generated similarly as LogTable */
+ 0,32,48,58,64,70,74,77,80,83,86,
+ 88,90,92,93,95,96,98,99,100,102,
+ 103,104,105,106,107,108,108,109,
+ 110,111,112,112,113,114,114,115,
+ 116,116,117,118,118,119,119,120,
+ 120,121,121,122,122,123,123,124,
+ 124,124,125,125,126,126,126,127,
+ 127,127,128,128
+};
+
+static uchar adl[] = {
+ 14,0,0,1,143,242,244,0,8,1,6,242,247,0,14,0,0,1,75,242,244,0,8,1,0,242,247,0,14,0,0,1,73,242,244,0,8,1,0,242,246,0,14,0,0,129,18,242,247,0,6,65,0,242,247,0,14,0,0,1,87,241,247,0,0,1,0,242,247,0,14,0,0,1,147,241,247,0,0,1,0,242,247,0,14,0,0,1,128,161,
+ 242,0,8,22,14,242,245,0,14,0,0,1,146,194,248,0,10,1,0,194,248,0,14,0,0,12,92,246,244,0,0,129,0,243,245,0,14,0,0,7,151,243,242,0,2,17,128,242,241,0,14,0,0,23,33,84,244,0,2,1,0,244,244,0,14,0,0,152,98,243,246,0,0,129,0,242,246,0,14,0,0,24,35,246,246,
+ 0,0,1,0,231,247,0,14,0,0,21,145,246,246,0,4,1,0,246,246,0,14,0,0,69,89,211,243,0,12,129,128,163,243,0,14,0,0,3,73,117,245,1,4,129,128,181,245,0,14,0,0,113,146,246,20,0,2,49,0,241,7,0,14,0,0,114,20,199,88,0,2,48,0,199,8,0,14,0,0,112,68,170,24,0,4,177,
+ 0,138,8,0,14,0,0,35,147,151,35,1,4,177,0,85,20,0,14,0,0,97,19,151,4,1,0,177,128,85,4,0,14,0,0,36,72,152,42,1,12,177,0,70,26,0,14,0,0,97,19,145,6,1,10,33,0,97,7,0,14,0,0,33,19,113,6,0,6,161,137,97,7,0,14,0,0,2,156,243,148,1,12,65,128,243,200,0,14,0,
+ 0,3,84,243,154,1,12,17,0,241,231,0,14,0,0,35,95,241,58,0,0,33,0,242,248,0,14,0,0,3,135,246,34,1,6,33,128,243,248,0,14,0,0,3,71,249,84,0,0,33,0,246,58,0,14,0,0,35,74,145,65,1,8,33,5,132,25,0,14,0,0,35,74,149,25,1,8,33,0,148,25,0,14,0,0,9,161,32,79,
+ 0,8,132,128,209,248,0,14,0,0,33,30,148,6,0,2,162,0,195,166,0,14,0,0,49,18,241,40,0,10,49,0,241,24,0,14,0,0,49,141,241,232,0,10,49,0,241,120,0,14,0,0,49,91,81,40,0,12,50,0,113,72,0,14,0,0,1,139,161,154,0,8,33,64,242,223,0,14,0,0,33,139,162,22,0,8,33,
+ 8,161,223,0,14,0,0,49,139,244,232,0,10,49,0,241,120,0,14,0,0,49,18,241,40,0,10,49,0,241,24,0,14,0,0,49,21,221,19,1,8,33,0,86,38,0,14,0,0,49,22,221,19,1,8,33,0,102,6,0,14,0,0,113,73,209,28,1,8,49,0,97,12,0,14,0,0,33,77,113,18,1,2,35,128,114,6,0,14,
+ 0,0,241,64,241,33,1,2,225,0,111,22,0,14,0,0,2,26,245,117,1,0,1,128,133,53,0,14,0,0,2,29,245,117,1,0,1,128,243,244,0,14,0,0,16,65,245,5,1,2,17,0,242,195,0,14,0,0,33,155,177,37,1,14,162,1,114,8,0,14,0,0,161,152,127,3,1,0,33,0,63,7,1,14,0,0,161,147,193,
+ 18,0,10,97,0,79,5,0,14,0,0,33,24,193,34,0,12,97,0,79,5,0,14,0,0,49,91,244,21,0,0,114,131,138,5,0,14,0,0,161,144,116,57,0,0,97,0,113,103,0,14,0,0,113,87,84,5,0,12,114,0,122,5,0,14,0,0,144,0,84,99,0,8,65,0,165,69,0,14,0,0,33,146,133,23,0,12,33,1,143,
+ 9,0,14,0,0,33,148,117,23,0,12,33,5,143,9,0,14,0,0,33,148,118,21,0,12,97,0,130,55,0,14,0,0,49,67,158,23,1,2,33,0,98,44,1,14,0,0,33,155,97,106,0,2,33,0,127,10,0,14,0,0,97,138,117,31,0,8,34,6,116,15,0,14,0,0,161,134,114,85,1,0,33,131,113,24,0,14,0,0,
+ 33,77,84,60,0,8,33,0,166,28,0,14,0,0,49,143,147,2,1,8,97,0,114,11,0,14,0,0,49,142,147,3,1,8,97,0,114,9,0,14,0,0,49,145,147,3,1,10,97,0,130,9,0,14,0,0,49,142,147,15,1,10,97,0,114,15,0,14,0,0,33,75,170,22,1,8,33,0,143,10,0,14,0,0,49,144,126,23,1,6,33,
+ 0,139,12,1,14,0,0,49,129,117,25,1,0,50,0,97,25,0,14,0,0,50,144,155,33,0,4,33,0,114,23,0,14,0,0,225,31,133,95,0,0,225,0,101,26,0,14,0,0,225,70,136,95,0,0,225,0,101,26,0,14,0,0,161,156,117,31,0,2,33,0,117,10,0,14,0,0,49,139,132,88,0,0,33,0,101,26,0,
+ 14,0,0,225,76,102,86,0,0,161,0,101,38,0,14,0,0,98,203,118,70,0,0,161,0,85,54,0,14,0,0,98,153,87,7,0,11,161,0,86,7,0,14,0,0,98,147,119,7,0,11,161,0,118,7,0,14,0,0,34,89,255,3,2,0,33,0,255,15,0,14,0,0,33,14,255,15,1,0,33,0,255,15,1,14,0,0,34,70,134,
+ 85,0,0,33,128,100,24,0,14,0,0,33,69,102,18,0,0,161,0,150,10,0,14,0,0,33,139,146,42,1,0,34,0,145,42,0,14,0,0,162,158,223,5,0,2,97,64,111,7,0,14,0,0,32,26,239,1,0,0,96,0,143,6,2,14,0,0,33,143,241,41,0,10,33,128,244,9,0,14,0,0,119,165,83,148,0,2,161,
+ 0,160,5,0,14,0,0,97,31,168,17,0,10,177,128,37,3,0,14,0,0,97,23,145,52,0,12,97,0,85,22,0,14,0,0,113,93,84,1,0,0,114,0,106,3,0,14,0,0,33,151,33,67,0,8,162,0,66,53,0,14,0,0,161,28,161,119,1,0,33,0,49,71,1,14,0,0,33,137,17,51,0,10,97,3,66,37,0,14,0,0,
+ 161,21,17,71,1,0,33,0,207,7,0,14,0,0,58,206,248,246,0,2,81,0,134,2,0,14,0,0,33,21,33,35,1,0,33,0,65,19,0,14,0,0,6,91,116,149,0,0,1,0,165,114,0,14,0,0,34,146,177,129,0,12,97,131,242,38,0,14,0,0,65,77,241,81,1,0,66,0,242,245,0,14,0,0,97,148,17,81,1,
+ 6,163,128,17,19,0,14,0,0,97,140,17,49,0,6,161,128,29,3,0,14,0,0,164,76,243,115,1,4,97,0,129,35,0,14,0,0,2,133,210,83,0,0,7,3,242,246,1,14,0,0,17,12,163,17,1,0,19,128,162,229,0,14,0,0,17,6,246,65,1,4,17,0,242,230,2,14,0,0,147,145,212,50,0,8,145,0,235,
+ 17,1,14,0,0,4,79,250,86,0,12,1,0,194,5,0,14,0,0,33,73,124,32,0,6,34,0,111,12,1,14,0,0,49,133,221,51,1,10,33,0,86,22,0,14,0,0,32,4,218,5,2,6,33,129,143,11,0,14,0,0,5,106,241,229,0,6,3,128,195,229,0,14,0,0,7,21,236,38,0,10,2,0,248,22,0,14,0,0,5,157,
+ 103,53,0,8,1,0,223,5,0,14,0,0,24,150,250,40,0,10,18,0,248,229,0,14,0,0,16,134,168,7,0,6,0,3,250,3,0,14,0,0,17,65,248,71,2,4,16,3,243,3,0,14,0,0,1,142,241,6,2,14,16,0,243,2,0,14,0,0,14,0,31,0,0,14,192,0,31,255,3,14,0,0,6,128,248,36,0,14,3,136,86,132,
+ 2,14,0,0,14,0,248,0,0,14,208,5,52,4,3,14,0,0,14,0,246,0,0,14,192,0,31,2,3,14,0,0,213,149,55,163,0,0,218,64,86,55,0,14,0,0,53,92,178,97,2,10,20,8,244,21,0,14,0,0,14,0,246,0,0,14,208,0,79,245,3,14,0,0,38,0,255,1,0,14,228,0,18,22,1,14,0,0,0,0,243,240,
+ 0,14,0,0,246,201,2,14,0,35,16,68,248,119,2,8,17,0,243,6,0,14,0,35,16,68,248,119,2,8,17,0,243,6,0,14,0,52,2,7,249,255,0,8,17,0,248,255,0,14,0,48,0,0,252,5,2,14,0,0,250,23,0,14,0,58,0,2,255,7,0,0,1,0,255,8,0,14,0,60,0,0,252,5,2,14,0,0,250,23,0,14,0,
+ 47,0,0,246,12,0,4,0,0,246,6,0,14,0,43,12,0,246,8,0,10,18,0,251,71,2,14,0,49,0,0,246,12,0,4,0,0,246,6,0,14,0,43,12,0,246,8,0,10,18,5,123,71,2,14,0,51,0,0,246,12,0,4,0,0,246,6,0,14,0,43,12,0,246,2,0,10,18,0,203,67,2,14,0,54,0,0,246,12,0,4,0,0,246,6,
+ 0,14,0,57,0,0,246,12,0,4,0,0,246,6,0,14,0,72,14,0,246,0,0,14,208,0,159,2,3,14,0,60,0,0,246,12,0,4,0,0,246,6,0,14,0,76,14,8,248,66,0,14,7,74,244,228,3,14,0,84,14,0,245,48,0,14,208,10,159,2,0,14,0,36,14,10,228,228,3,6,7,93,245,229,1,14,0,65,2,3,180,
+ 4,0,14,5,10,151,247,0,14,0,84,78,0,246,0,0,14,158,0,159,2,3,14,0,83,17,69,248,55,2,8,16,8,243,5,0,14,0,84,14,0,246,0,0,14,208,0,159,2,3,14,0,24,128,0,255,3,3,12,16,13,255,20,0,14,0,77,14,8,248,66,0,14,7,74,244,228,3,14,0,60,6,11,245,12,0,6,2,0,245,
+ 8,0,14,0,65,1,0,250,191,0,7,2,0,200,151,0,14,0,59,1,81,250,135,0,6,1,0,250,183,0,14,0,51,1,84,250,141,0,6,2,0,248,184,0,14,0,45,1,89,250,136,0,6,2,0,248,182,0,14,0,71,1,0,249,10,3,14,0,0,250,6,0,14,0,60,0,128,249,137,3,14,0,0,246,108,0,14,0,58,3,128,
+ 248,136,3,15,12,8,246,182,0,14,0,53,3,133,248,136,3,15,12,0,246,182,0,14,0,64,14,64,118,79,0,14,0,8,119,24,2,14,0,71,14,64,200,73,0,14,3,0,155,105,2,14,0,61,215,220,173,5,3,14,199,0,141,5,0,14,0,61,215,220,168,4,3,14,199,0,136,4,0,14,0,44,128,0,246,
+ 6,3,14,17,0,103,23,3,14,0,40,128,0,245,5,2,14,17,9,70,22,3,14,0,69,6,63,0,244,0,1,21,0,247,245,0,14,0,68,6,63,0,244,3,0,18,0,247,245,0,14,0,63,6,63,0,244,0,1,18,0,247,245,0,14,0,74,1,88,103,231,0,0,2,0,117,7,0,14,0,60,65,69,248,72,0,0,66,8,117,5,0,
+ 14,0,80,10,64,224,240,3,8,30,78,255,5,0,14,0,64,10,124,224,240,3,8,30,82,255,2,0,14,0,72,14,64,122,74,0,14,0,8,123,27,2,14,0,73,14,10,228,228,3,6,7,64,85,57,1,14,0,70,5,5,249,50,3,14,4,64,214,165,0,14,0,68,2,63,0,243,3,8,21,0,247,245,0,14,0,48,1,79,
+ 250,141,0,7,2,0,248,181,0,14,0,53,0,0,246,12,0,4,0,0,246,6,0
+};
+
+
+static void
+pcmout(int v)
+{
+ if(v < -32768)
+ v = -32768;
+ if(v > 32767)
+ v = 32767;
+ /* FIXME: portability (endianness) */
+ *ap++ = v;
+ /* FIXME: stereo shit */
+ *ap++ = v;
+
+ /* FIXME: write last samples before end */
+ if(ap >= abuf+nelem(abuf)){
+ write(afd, abuf, sizeof abuf);
+ ap = abuf;
+ }
+}
+
+static uint
+tadv(double length, double add)
+{
+ add += otm;
+ otm = fmod(add, length);
+
+ return add / length;
+}
+
+static void
+mix(uint usecs)
+{
+ int i;
+ uint n;
+ s16int *buf;
+
+ n = tadv((double)1.0 / RATE, (double)usecs / 1E6);
+ if((buf = mallocz(n * sizeof *buf, 1)) == nil)
+ sysfatal("mallocz: %r");
+
+ ym3812_update_one(chip[0], buf, n);
+ for(i = 0; i < n; i++)
+ //pcmout((double)buf[i] / 32768.0 * 10000.0); /* FIXME: why? */
+ pcmout(buf[i]);
+
+ free(buf);
+}
+
+static uchar
+inb(uint port)
+{
+ if(port >= 0x388 && port <= 0x38b)
+ return oplretval;
+ return 0;
+}
+
+static void
+outb(uint port, uchar v)
+{
+ uint ind;
+
+ if(port >= 0x388 && port <= 0x38b){
+ ind = port - 0x388;
+ ym3812_write(chip[0], ind, v);
+
+ if(ind & 1){
+ if(oplregno == 4){
+ if(v == 0x80)
+ oplretval = 0x02;
+ else if(v == 0x21)
+ oplretval = 0xc0;
+ }
+ }else
+ oplregno = v;
+ }
+}
+
+static void
+OPL_Byte(uchar Index, uchar Data)
+{
+ int a;
+
+ outb(OPLBASE, Index);
+ for(a=0; a<6; a++)
+ inb(OPLBASE);
+
+ outb(OPLBASE+1, Data);
+ for(a=0; a<35; a++)
+ inb(OPLBASE);
+}
+
+static void
+OPL_NoteOff(int c)
+{
+ Trig[c] = 0;
+
+ c = SetBase(c);
+ if(c<9)
+ {
+ int Ope = Portit[c];
+ /* KEYON_BLOCK+c seems to not work alone?? */
+ OPL_Byte(KEYON_BLOCK+c, 0);
+ OPL_Byte(KSL_LEVEL+ Ope, 0xFF);
+ OPL_Byte(KSL_LEVEL+3+Ope, 0xFF);
+ }
+}
+
+/* OPL_NoteOn changes the frequency on specified
+ channel and guarantees the key is on. (Doesn't
+ retrig, just turns the note on and sets freq.) */
+/* Could be used for pitch bending also. */
+static void OPL_NoteOn(int c, unsigned long Herz)
+{
+ int Oct;
+
+ Trig[c] = 127;
+
+ c = SetBase(c);
+ if(c >= 9)return;
+
+ for(Oct=0; Herz>0x1FF; Oct++)Herz >>= 1;
+
+/*
+ Bytes A0-B8 - Octave / F-Number / Key-On
+
+ 7 6 5 4 3 2 1 0
+ +-----+-----+-----+-----+-----+-----+-----+-----+
+ | F-Number (least significant byte) | (A0-A8)
+ +-----+-----+-----+-----+-----+-----+-----+-----+
+ | Unused | Key | Octave | F-Number | (B0-B8)
+ | | On | | most sig. |
+ +-----+-----+-----+-----+-----+-----+-----+-----+
+*/
+
+ OPL_Byte(0xA0+c, Herz&255); //F-Number low 8 bits
+ OPL_Byte(0xB0+c, 0x20 //Key on
+ | ((Herz>>8)&3) //F-number high 2 bits
+ | ((Oct&7)<<2)
+ );
+}
+
+static void
+OPL_Touch(int c, int Instru, ushort Vol)
+{
+ int Ope;
+ //int level;
+
+ c = SetBase(c);
+ if(c >= 9)return;
+
+ Ope = Portit[c];
+
+/*
+ Bytes 40-55 - Level Key Scaling / Total Level
+
+ 7 6 5 4 3 2 1 0
+ +-----+-----+-----+-----+-----+-----+-----+-----+
+ | Scaling | Total Level |
+ | Level | 24 12 6 3 1.5 .75 | <-- dB
+ +-----+-----+-----+-----+-----+-----+-----+-----+
+ bits 7-6 - causes output levels to decrease as the frequency
+ rises:
+ 00 - no change
+ 10 - 1.5 dB/8ve
+ 01 - 3 dB/8ve
+ 11 - 6 dB/8ve
+ bits 5-0 - controls the total output level of the operator.
+ all bits CLEAR is loudest; all bits SET is the
+ softest. Don't ask me why.
+*/
+/* if 1 */
+ OPL_Byte(KSL_LEVEL+ Ope, (mdat.Instr[Instru]->D[2]&KSL_MASK)
+ | (63 + (mdat.Instr[Instru]->D[2]&63) * Vol / 63 - Vol));
+ OPL_Byte(KSL_LEVEL+3+Ope, (mdat.Instr[Instru]->D[3]&KSL_MASK)
+ | (63 + (mdat.Instr[Instru]->D[3]&63) * Vol / 63 - Vol));
+/* else
+ level = (mdat.Instr[Instru]->D[2]&63) - (Vol*72-8);
+ if(level<0)level=0;
+ if(level>63)level=63;
+
+ OPL_Byte(KSL_LEVEL+ Ope, (mdat.Instr[Instru]->D[2]&KSL_MASK) | level);
+
+ level = (mdat.Instr[Instru]->D[3]&63) - (Vol*72-8);
+ if(level<0)level=0;
+ if(level>63)level=63;
+
+ OPL_Byte(KSL_LEVEL+3+Ope, (mdat.Instr[Instru]->D[3]&KSL_MASK) | level);
+ endif */
+}
+
+static void OPL_Pan(int c, uchar val)
+{
+ Pans[c] = val - 128;
+}
+
+static void OPL_Patch(int c, int Instru)
+{
+ int Ope;
+
+ c = SetBase(c);
+ if(c >= 9)return;
+
+ Ope = Portit[c];
+
+ if(Instru > sizeof(mdat.Instr)-1)
+ sysfatal("invalid instrument patch %ud", Instru); /* FIXME: see dat.h comments */
+
+ OPL_Byte(AM_VIB+ Ope, mdat.Instr[Instru]->D[0]);
+ OPL_Byte(ATTACK_DECAY+ Ope, mdat.Instr[Instru]->D[4]);
+ OPL_Byte(SUSTAIN_RELEASE+ Ope, mdat.Instr[Instru]->D[6]);
+ OPL_Byte(WAVE_SELECT+ Ope, mdat.Instr[Instru]->D[8]&3);// 6 high bits used elsewhere
+
+ OPL_Byte(AM_VIB+ 3+Ope, mdat.Instr[Instru]->D[1]);
+ OPL_Byte(ATTACK_DECAY+ 3+Ope, mdat.Instr[Instru]->D[5]);
+ OPL_Byte(SUSTAIN_RELEASE+3+Ope, mdat.Instr[Instru]->D[7]);
+ OPL_Byte(WAVE_SELECT+ 3+Ope, mdat.Instr[Instru]->D[9]&3);// 6 high bits used elsewhere
+
+ /* Panning... */
+ OPL_Byte(FEEDBACK_CONNECTION+c,
+ (mdat.Instr[Instru]->D[10] & ~STEREO_BITS)
+ | (Pans[c]<-32 ? VOICE_TO_LEFT
+ : Pans[c]>32 ? VOICE_TO_RIGHT
+ : (VOICE_TO_LEFT | VOICE_TO_RIGHT)
+ ) );
+}
+
+/* u32int, word and byte have been defined in adlib.h */
+
+static u32int ConvL(u8int *s)
+{
+ return (((u32int)s[0] << 24) | ((u32int)s[1] << 16) | ((u16int)s[2] << 8) | s[3]);
+}
+
+static u16int ConvI(u8int *s)
+{
+ return (s[0] << 8) | s[1];
+}
+
+static struct Midi
+{
+/* Fixed by ReadMIDI() */
+ int Fmt;
+ int TrackCount;
+ int DeltaTicks;
+ ulong *TrackLen;
+ u8int **Tracks;
+/* Used by play() */
+ u32int *Waiting, *sWaiting, *SWaiting;
+ u8int *Running, *sRunning, *SRunning;
+ ulong *Posi, *sPosi, *SPosi;
+ u32int Tempo;
+ u32int oldtempo;
+ u32int initempo;
+/* Per channel */
+ u8int Pan[16];
+ u8int Patch[16];
+ u8int MainVol[16];
+ u8int PitchSense[16];
+ int Bend[16];
+ int OldBend[16];
+ int Used[16][127]; /* contains references to adlib channels per note */
+} MIDI;
+
+#define snNoteOff 0x7fb1
+#define snNoteOn 0x7fb2
+#define snNoteModify 0x7fb3
+#define snNoteUpdate 0x7fb4
+
+enum{
+ MAXS3MCHAN = 9
+};
+
+typedef struct
+{
+/* Important - at offset 0 to be saved to file */
+ long Age;
+ u8int Note;
+ u8int Instru;
+ u8int Volume;
+ u8int cmd;
+ u8int info;
+ u8int KeyFlag; /* Required to diff aftertouch and noteon */
+ /* Byte to save space */
+/* Real volume(0..127) and main volume(0..127) */
+ int RealVol;
+ int MainVol;
+ int exp; /* vol = MainVol * RealVol/127 * exp/127 */
+ /* RealVol must be first non-saved */
+/* Less important */
+ int LastInstru; /* Needed by SuggestNewChan() */
+ int BendFlag;
+ u8int LastPan;
+/* To fasten forcement */
+ int uChn, uNote;
+} S3MChan;
+
+static S3MChan Chan[MAXCHN];
+static S3MChan PrevChan[MAXCHN];
+
+#define chansavesize ((int)((long)&Chan[0].RealVol - (long)&Chan[0].Age))
+
+static int InstruUsed[256];
+static int Forced=0;
+static const int tempochanged = -5;
+
+
+static void AnalyzeRow(void)
+{
+ int a;
+
+ for(a=0; a<MAXS3MCHAN; a++)
+ if(!Chan[a].Age)break;
+
+ if(a==MAXS3MCHAN)
+ return;
+
+ memcpy(PrevChan, Chan, sizeof PrevChan);
+
+ Forced=0;
+
+ for(a=0; a<MAXS3MCHAN; a++)
+ Chan[a].KeyFlag=0;
+}
+
+static void FixSpeed(float *Speed, int *Tempo)
+{
+ int a;
+ float tmp = 1.0;
+ *Speed = MIDI.Tempo * 4E-7 / MIDI.DeltaTicks;
+ *Tempo = 125;
+
+ for(a=0x40; a<=0xFF; a++)
+ {
+ double Tmp;
+ float n = a * *Speed;
+ n = modf(1.0/n, &Tmp);
+ if(n < tmp)
+ {
+ *Tempo = a;
+ tmp = n;
+ }
+ }
+ *Speed *= *Tempo;
+}
+
+static int MakeVolume(int vol)
+{
+ return FMVol[vol*64/127]*63/128;
+}
+
+static void Bendi(int chn, int a)
+{
+ int bc, nt;
+ long HZ1, HZ2, Herz;
+
+ bc = MIDI.Bend[chn];
+ nt = Chan[a].Note;
+
+ if(bc > MIDI.OldBend[chn])
+ Chan[a].BendFlag |= 1;
+ else if(bc < MIDI.OldBend[chn])
+ Chan[a].BendFlag |= 2;
+ else
+ Chan[a].BendFlag = 0;
+
+ MIDI.OldBend[chn] = bc;
+
+ for(; bc < 0; bc += 0x1000)
+ nt--;
+ for(; bc >= 0x1000; bc -= 0x1000)
+ nt++;
+
+ HZ1 = Period[(nt+1)%12] * (8363L << (nt+1)/12) / 44100U;
+ HZ2 = Period[(nt )%12] * (8363L << (nt )/12) / 44100U;
+
+ Herz = HZ2 + bc * (HZ1 - HZ2) / 0x1000;
+
+ OPL_NoteOn(a, Herz);
+}
+
+/* vole = RealVol*MainVol/127 */
+static int SuggestNewChan(int instru, int /*vole*/)
+{
+ int a, c=MAXS3MCHAN, f;
+ long b;
+
+ for(a=f=0; a<MAXS3MCHAN; a++)
+ if(Chan[a].LastInstru==instru)
+ f=1;
+
+ /* Arvostellaan channels */
+ for(b=a=0; a<MAXS3MCHAN; a++)
+ {
+ /* empty if channel is silent */
+ if(!Chan[a].Volume)
+ {
+ long d;
+ /* Pohjapisteet...
+ * Jos instru oli uusi, mieluiten sijoitetaan
+ * sellaiselle kanavalle, joka on pitk��n ollut hiljaa.
+ * Muuten sille, mill� se juuri �skenkin soi.
+ */
+ d = f?1:Chan[a].Age;
+
+ /* Jos kanavan edellinen instru oli joku
+ * soinniltaan hyvin lyhyt, pisteit� annetaan lis�� */
+ if(strchr("\x81\x82\x83\x84\x85\x86\x87\x88\x89"
+ "\x8B\x8D\x8E\x90\x94\x96\x9A\x9B\x9C"
+ "\x9D\xA4\xAA\xAB",
+ Chan[a].LastInstru))d += 2;
+ else
+ {
+ /* Jos oli pitk�sointinen percussion, *
+ * annetaan pisteit� i�n mukaan. */
+ if(Chan[a].LastInstru > 0x80)
+ d += Chan[a].Age*2;
+ }
+
+ /* Jos oli samaa instrua, pisteit� tulee paljon lis�� */
+ if(Chan[a].LastInstru == instru)d += 3;
+
+ //d = (d-1)*Chan[a].Age+1;
+ if(d > b)
+ {
+ b = d;
+ c = a;
+ }
+ }
+ }
+ return c;
+}
+
+/* vole = RealVol*MainVol/127 */
+static int ForceNewChan(int instru, int vole)
+{
+ int a, c;
+ long b=0;
+
+ vole *= 127;
+
+ Forced=1;
+ for(a=c=0; c<MAXS3MCHAN; c++)
+ if(Chan[c].Age
+ > b
+ + (((instru<128 && Chan[c].Instru>128)
+ || (vole > Chan[c].RealVol*Chan[c].MainVol)
+ ) ? 1:0
+ ) )
+ {
+ a=c;
+ b=Chan[c].Age;
+ }
+ return a;
+}
+
+/* Used twice by SetNote. This should be considered as a macro. */
+static void SubNoteOff(int a, int chn, int note)
+{
+ Chan[a].RealVol = 0;
+ Chan[a].MainVol = 0;
+ Chan[a].exp = 0x7f;
+
+ Chan[a].Age = 0;
+ Chan[a].Volume = 0;
+ Chan[a].BendFlag= 0;
+
+ MIDI.Used[chn][note] = 0;
+
+ if(Chan[a].Instru < 0x80)OPL_NoteOff(a);
+}
+
+static void SetNote(int chn, int note, int RealVol,int MainVol, int bend, int e)
+{
+ int a, vole;
+
+ //vole = RealVol*(MainVol*e)/127;
+ vole = RealVol*(MainVol)/127;
+
+ if(!vole && (bend==snNoteOn || bend==snNoteModify))bend=snNoteOff;
+ if(bend==snNoteOn && MIDI.Used[chn][note])bend=snNoteModify;
+
+ switch(bend)
+ {
+ /* snNoteOn:ssa note ei koskaan ole -1 */
+ case snNoteOn:
+ {
+ int p;
+
+ /* FIXME */
+ p = chn==9 ? 128+note-35 : MIDI.Patch[chn];
+
+ a = SuggestNewChan(p, vole);
+
+ if(a==MAXS3MCHAN)
+ {
+ a = ForceNewChan(p, vole);
+ MIDI.Used[Chan[a].uChn][Chan[a].uNote] = 0;
+ }
+
+ if(a < MAXS3MCHAN)
+ {
+ Chan[a].exp = e;
+ Chan[a].RealVol= RealVol;
+ Chan[a].MainVol= MainVol;
+ Chan[a].Note = chn==9 ? 60 : note;
+ Chan[a].Volume = MakeVolume(vole);
+ Chan[a].Age = 0;
+ Chan[a].Instru = p;
+
+ Chan[a].LastInstru = p;
+ Chan[a].KeyFlag= 1;
+ Chan[a].uChn = chn;
+ Chan[a].uNote = note;
+
+/*
+ if(MIDI.Bend[chn] && Chan[a].Volume)
+ {
+ int adlbend = MIDI.Bend[chn] * 127L / 8000;
+
+ //TODO: Fix this. It doesn't work at all.
+
+ Chan[a].cmd = 'N'-64;
+ Chan[a].info= adlbend+0x80; // To make it unsigned
+ }
+ else
+*/
+ if(MIDI.Pan[chn] != Chan[a].LastPan && Chan[a].Volume)
+ {
+ Chan[a].cmd = 'X'-64;
+ Chan[a].info= MIDI.Pan[chn]*2;
+ Chan[a].LastPan = MIDI.Pan[chn];
+ }
+ else
+ {
+ Chan[a].cmd = Chan[a].info = 0;
+ }
+
+ OPL_NoteOff(a);
+
+ OPL_Patch(a, Chan[a].Instru);
+ if(MIDI.Pan[chn] != 64)OPL_Pan(a, Chan[a].info);
+ OPL_Touch(a, Chan[a].Instru, Chan[a].Volume);
+
+ Bendi(chn, a);
+
+ MIDI.Used[chn][note] = a+1;
+ }
+ break;
+ }
+ /* snNoteOff:ssa note voi olla -1 */
+ case snNoteOff:
+ {
+ int b=note, c=note;
+ if(note < 0)b=0, c=127;
+
+ MIDI.Bend[chn] = 0; /* N�in vaikuttaisi olevan hyv� */
+
+ for(note=b; note<=c; note++)
+ {
+ a = MIDI.Used[chn][note];
+ if(a > 0)
+ SubNoteOff(a-1, chn, note);
+ }
+ break;
+ }
+ /* snNoteModify:ssa note ei koskaan ole -1 */
+ case snNoteModify:
+ a = MIDI.Used[chn][note]-1;
+ if(a != -1)
+ {
+ Chan[a].exp = e;
+ Chan[a].RealVol= RealVol;
+ Chan[a].MainVol= MainVol;
+
+ Chan[a].Volume = MakeVolume(vole);
+ Chan[a].Age = 0;
+
+ OPL_Touch(a, Chan[a].Instru, Chan[a].Volume);
+ }
+ break;
+ /* snNoteUpdate:ssa note on aina -1 */
+ case snNoteUpdate:
+ /* snNoteUpdatessa RealVol ei muutu, vain MainVol */
+ for(note=0; note<=127; note++)
+ {
+ a = MIDI.Used[chn][note]-1;
+
+ //vole = MainVol*(Chan[a].RealVol*Chan[a].exp)/127;
+ vole = MainVol*(Chan[a].RealVol)/127;
+
+ if(a >= 0 && Chan[a].Volume != MakeVolume(vole))
+ {
+ Chan[a].MainVol= MainVol;
+
+ if(!vole)
+ SubNoteOff(a, chn, note);
+ else
+ {
+ Chan[a].Volume = MakeVolume(vole);
+ Chan[a].Age = 0;
+ OPL_Touch(a, Chan[a].Instru, Chan[a].Volume);
+ }
+ }
+ }
+ break;
+ /* Bendiss� note on aina -1 */
+ default:
+ MIDI.Bend[chn] = bend;
+
+ for(note=0; note<=127; note++)
+ {
+ a = MIDI.Used[chn][note]-1;
+ if(a >= 0)
+ Bendi(chn, a);
+ }
+ }
+}
+
+static u32int GetVarLen(int Track, ulong *Posi)
+{
+ u32int d = 0;
+ for(;;)
+ {
+ u8int b = MIDI.Tracks[Track][(*Posi)++];
+ d = (d<<7) + (b&127);
+ if(b < 128)break;
+ }
+ return d;
+}
+
+/* NOTICE: This copies len-2 bytes */
+static char *Sana(int len, const unsigned char *buf)
+{
+ static char Buf[128];
+ char *s = Buf;
+ len -= 2;
+#define add(c) if(s<&Buf[sizeof(Buf)-1])*s++=(c)
+ while(len>0)
+ {
+ if(*buf<32||(*buf>=127&&*buf<160)){add('^');add(*buf + 64);}
+ else add(*buf);
+ buf++;
+ len--;
+ }
+#undef add
+ *s=0;
+ return Buf;
+}
+
+static void
+MPU_Byte(uchar)
+{
+
+}
+
+static void
+readmeta(int t, u8int *p, ulong n)
+{
+ u8int b;
+
+ b = *p;
+ p += 2;
+ //fprint(2, "meta %ux\n", b);
+ switch(b){
+ case 1:
+ case 2:
+ case 3:
+ case 4:
+ case 5:
+ case 6:
+ if(verbose)
+ fprint(2, "%s\n", Sana(n, p));
+ break;
+ case 0x2f:
+ MIDI.Posi[t] = MIDI.TrackLen[t];
+ if(verbose)
+ fprint(2, "end\n");
+ break;
+ case 0x51:
+ MIDI.Tempo = p[0]<<16 | p[1]<<8 | p[2];
+ if(verbose)
+ fprint(2, "tempo %ud\n", MIDI.Tempo);
+ break;
+ case 0: case 0x20: case 0x21: case 0x58: case 0x59: case 0x7f:
+ break;
+ default:
+ fprint(2, "unknown metaevent %ux\n", b);
+ }
+}
+
+static void
+Tavuja(int Track, u8int First, ulong posi, ulong Len)
+{
+ ulong a;
+
+ if(verbose){
+ fprint(2, "[%d]: %02ux ", Track, First);
+ for(a=0; a<Len; a++)
+ fprint(2, "%02ux", MIDI.Tracks[Track][a+posi]);
+ fprint(2, "\n");
+ }
+
+ /* FIXME: ignore sysex completely, do it like games/midi */
+ if(First == 0xff)
+ readmeta(Track, MIDI.Tracks[Track]+posi, Len);
+ else if(First < 0xf0){
+ //ulong a; // ← what the fuck
+ a=0;
+ if((First>>4) == 8) /* Note off, modify it a bit */
+ {
+ /* FIXME: */
+ MPU_Byte(First);
+ MPU_Byte(MIDI.Tracks[Track][posi]);
+ MPU_Byte(0);
+ Len -= 2;
+ }
+ else
+ MPU_Byte(First);
+ for(; a<Len; a++)MPU_Byte(MIDI.Tracks[Track][a+posi]);
+ //fprint(2, "ev %ux\n", First >> 4);
+ switch(First >> 4)
+ {
+ case 0x8: /* Note off */
+ SetNote(
+ First&15, /* chn */
+ MIDI.Tracks[Track][posi+0], /* note */
+ MIDI.Tracks[Track][posi+1], /* volume */
+ MIDI.MainVol[First&15], /* mainvol */
+ snNoteOff, 127);
+ break;
+ case 0x9: /* Note on */
+ SetNote(
+ First&15, /* chn */
+ MIDI.Tracks[Track][posi+0], /* note */
+ MIDI.Tracks[Track][posi+1], /* volume */
+ MIDI.MainVol[First&15], /* mainvol */
+ snNoteOn, Chan[First&15].exp);
+ break;
+ case 0xA: /* Key after-touch */
+ SetNote(
+ First&15, /* chn */
+ MIDI.Tracks[Track][posi+0], /* note */
+ MIDI.Tracks[Track][posi+1], /* volume */
+ MIDI.MainVol[First&15], /* mainvol */
+ snNoteModify, Chan[First&15].exp);
+ break;
+ case 0xC: /* Patch change */
+ MIDI.Patch[First&15] = MIDI.Tracks[Track][posi];
+ break;
+ case 0xD: /* Channel after-touch */
+ break;
+ case 0xE: /* Wheel - 0x2000 = no change */
+ a = MIDI.Tracks[Track][posi+0] |
+ MIDI.Tracks[Track][posi+1] << 7;
+ SetNote(First&15,
+ -1, 0,0,
+ /* FIXME: these are fucked up, see below */
+ //(int)((long)a*MIDI.PitchSense[First&15]/2 - 0x2000L), Chan[First&15].exp);
+ //(short)(((long)a - 0x2000L) *(double)MIDI.PitchSense[First&15]/8192.0), Chan[First&15].exp);
+ (short)((long)a - 0x2000L), Chan[First&15].exp);
+ break;
+ case 0xB: /* Controller change */
+ switch(MIDI.Tracks[Track][posi+0])
+ {
+ case 123: /* All notes off on channel */
+ SetNote(First&15, -1,0,0, snNoteOff, 127);
+ break;
+ case 121: /* Reset vibrato and bending */
+ MIDI.PitchSense[First&15] = 2;
+ SetNote(First&15, -1,0,0, 0, Chan[First&15].exp); /* last 0=bend 0 */
+ break;
+ case 7:
+ MIDI.MainVol[First&15] = MIDI.Tracks[Track][posi+1];
+
+ SetNote(First&15, -1, 0, MIDI.MainVol[First&15], snNoteUpdate, Chan[First&15].exp);
+ /* FIXME: why doesn't this work? */
+ //for(a = 0; a < 16; a++)
+ // SetNote(a, -1, 0, MIDI.MainVol[First&15], snNoteUpdate, Chan[First&15].exp);
+ break;
+ case 64:
+ if(verbose)
+ fprint(2, "ctl 64: sustain unsupported\n");
+ break;
+ case 1:
+ if(verbose)
+ fprint(2, "ctl 1: vibrato unsupported\n");
+ break;
+ case 91:
+ if(verbose)
+ fprint(2, "ctl 91 %ud: reverb depth unsupported\n", MIDI.Tracks[Track][posi+1]);
+ break;
+ case 93:
+ if(verbose)
+ fprint(2, "ctl 93 %ud: chorus depth unsupported\n", MIDI.Tracks[Track][posi+1]);
+ break;
+ case 0x06: /* Pitch bender sensitivity */
+ /* FIXME: the fuck is this? whatever it is,
+ * it's broken */
+ MIDI.PitchSense[First&15] = MIDI.Tracks[Track][posi+1];
+ break;
+ case 0x0a: /* Pan */
+ MIDI.Pan[First&15] = MIDI.Tracks[Track][posi+1];
+ break;
+ case 0:
+ /* FIXME: unimplemented: select bank */
+ if(verbose)
+ fprint(2, "ctl 0 %ud: bank select unsupported\n", MIDI.Tracks[Track][posi+1]);
+ break;
+ case 0x0b: /* FIXME: probably bullshizzles */
+ SetNote(First&15, -1,0,MIDI.MainVol[First&15], snNoteUpdate, MIDI.Tracks[Track][posi+1]);
+ break;
+ default:
+ if(verbose)
+ fprint(2, "unknown ctl %ud: %ux\n",
+ MIDI.Tracks[Track][posi+0],
+ MIDI.Tracks[Track][posi+1]);
+ }
+ break;
+ }
+ }
+}
+
+/* Return value: 0=ok, -1=user break */
+static int play(void)
+{
+ int a, NotFirst, Userbreak=0;
+ long Viivetta;
+
+ if((MIDI.Waiting = mallocz(MIDI.TrackCount * sizeof *MIDI.Waiting, 1)) == nil
+ || (MIDI.sWaiting = mallocz(MIDI.TrackCount * sizeof *MIDI.sWaiting, 1)) == nil
+ || (MIDI.SWaiting = mallocz(MIDI.TrackCount * sizeof *MIDI.SWaiting, 1)) == nil
+ || (MIDI.Posi = mallocz(MIDI.TrackCount * sizeof *MIDI.Posi, 1)) == nil
+ || (MIDI.sPosi = mallocz(MIDI.TrackCount * sizeof *MIDI.sPosi, 1)) == nil
+ || (MIDI.SPosi = mallocz(MIDI.TrackCount * sizeof *MIDI.SPosi, 1)) == nil
+ || (MIDI.Running = mallocz(MIDI.TrackCount * sizeof *MIDI.Running, 1)) == nil
+ || (MIDI.sRunning = mallocz(MIDI.TrackCount * sizeof *MIDI.sRunning, 1)) == nil
+ || (MIDI.SRunning = mallocz(MIDI.TrackCount * sizeof *MIDI.SRunning, 1)) == nil)
+ sysfatal("mallocz: %r");
+
+ for(a=0; a<MIDI.TrackCount; a++)
+ {
+ ulong c = 0;
+ MIDI.sWaiting[a]= GetVarLen(a, &c);
+ MIDI.sRunning[a]= 0;
+ MIDI.sPosi[a] = c;
+ }
+
+ for(a=0; a<16; a++)
+ {
+ MIDI.Pan[a] = 64; /* Middle */
+ MIDI.Patch[a] = 1; /* Piano */
+ MIDI.PitchSense[a] = 2; /* � seminotes */
+ MIDI.MainVol[a] = 127;
+ MIDI.Bend[a] = 0;
+ }
+
+ NotFirst = 0;
+//ReLoop:
+ for(a=0; a<MIDI.TrackCount; a++)
+ {
+ MIDI.Posi[a] = MIDI.sPosi[a];
+ MIDI.Waiting[a] = MIDI.sWaiting[a];
+ MIDI.Running[a] = MIDI.sRunning[a];
+ }
+
+ MIDI.Tempo = MIDI.initempo;
+
+ memset(Chan, 0, sizeof Chan);
+ for(a = 0; a < nelem(Chan); a++){
+ Chan[a].exp = 0x7f;
+ }
+
+ memset(MIDI.Used, 0, sizeof MIDI.Used);
+
+ Viivetta = 0;
+
+ for(;;)
+ {
+ int Fin, Act;
+
+ if(NotFirst)
+ {
+ long Lisa = MIDI.Tempo/MIDI.DeltaTicks;
+
+ /* tempo = microseconds per quarter note
+ * deltaticks = number of delta-time ticks per quarter note
+ *
+ * So, when tempo = 200000
+ * and deltaticks = 10,
+ * then 10 ticks have 200000 microseconds.
+ * 20 ticks have 400000 microseconds.
+ * When deltaticks = 5,
+ * then 10 ticks have 40000 microseconds.
+ */
+
+ Viivetta += Lisa;
+ if(Viivetta >= MINDELAY)
+ {
+ mix(Viivetta);
+ Viivetta = 0;
+ }
+ AnalyzeRow();
+ }
+ else
+ {
+ u32int b = 0xFFFFFFFFUL;
+ /* Find the smallest delay */
+ for(a=0; a<MIDI.TrackCount; a++)
+ if(MIDI.Waiting[a] < b)
+ b = MIDI.Waiting[a];
+ /* Elapse that delay from all tracks */
+ for(a=0; a<MIDI.TrackCount; a++)
+ MIDI.Waiting[a] -= b;
+ }
+
+ /* Let the notes on channels become older (Age++) only if *
+ * something happens. This way, we don't overflow the ages *
+ * too easily. */
+ for(Act=a=0; a<MAXS3MCHAN; a++)
+ if(MIDI.Waiting[a]<=1 && MIDI.Posi[a]<MIDI.TrackLen[a])
+ Act++;
+ for(a=0; a<MAXS3MCHAN; a++)
+ if(!Chan[a].Age||Act!=0)
+ Chan[a].Age++;
+
+ for(a=0; a<MIDI.TrackCount; ++a)
+ {
+ MIDI.SPosi[a] = MIDI.Posi[a];
+ MIDI.SWaiting[a] = MIDI.Waiting[a];
+ MIDI.SRunning[a] = MIDI.Running[a];
+ }
+ for(Fin=1, a=0; a<MIDI.TrackCount; a++)
+ {
+ if(MIDI.Waiting[a] > 0)MIDI.Waiting[a]--;
+
+ if(MIDI.Posi[a] < MIDI.TrackLen[a])Fin=0;
+
+ /* While this track has events that we should have handled */
+ while(MIDI.Waiting[a]<=0 && MIDI.Posi[a]<MIDI.TrackLen[a])
+ {
+ ulong pos;
+ u8int b = MIDI.Tracks[a][MIDI.Posi[a]];
+ if(b < 128)
+ b = MIDI.Running[a];
+ else
+ {
+ MIDI.Posi[a]++;
+ if(b < 0xF0)MIDI.Running[a] = b;
+ }
+
+ pos = MIDI.Posi[a];
+
+ //fprint(2, "b %ux\n", b);
+ if(b == 0xFF)
+ {
+ int ls=0;
+ ulong len;
+ u8int typi = MIDI.Tracks[a][MIDI.Posi[a]++];
+ len = (ulong)GetVarLen(a, &MIDI.Posi[a]);
+ if(typi == 6) /* marker */
+ if(!strncmp((char *)(MIDI.Tracks[a]+MIDI.Posi[a]),
+ "loopStart", len))
+ {
+ if(verbose)
+ fprint(2, "Found loopStart\n");
+ ls=1;
+ }
+ MIDI.Posi[a] += len;
+ if(ls)goto SaveLoopStart;
+ }
+ else if(b==0xF7 || b==0xF0)
+ {
+ MIDI.Posi[a] += (ulong)GetVarLen(a, &MIDI.Posi[a]);
+ }
+ else if(b == 0xF3)MIDI.Posi[a]++;
+ else if(b == 0xF2)MIDI.Posi[a]+=2;
+ else if(b>=0xC0 && b<=0xDF)MIDI.Posi[a]++;
+ else if(b>=0x80 && b<=0xEF)
+ {
+ MIDI.Posi[a]+=2;
+ if(b>=0x90 && b<=0x9F && !NotFirst)
+ {
+ int c;
+SaveLoopStart: NotFirst=1;
+ /* Save the starting position for looping */
+ for(c=0; c<MIDI.TrackCount; c++)
+ {
+ MIDI.sPosi[c] = MIDI.SPosi[c];
+ MIDI.sWaiting[c] = MIDI.SWaiting[c];
+ MIDI.sRunning[c] = MIDI.SRunning[c];
+ }
+ MIDI.initempo = MIDI.Tempo;
+ }
+ }
+
+ Tavuja(a, b, pos, MIDI.Posi[a]-pos);
+
+ if(MIDI.Posi[a] < MIDI.TrackLen[a])
+ MIDI.Waiting[a] += GetVarLen(a, &MIDI.Posi[a]);
+ }
+ }
+ if(Fin)
+ break;
+ }
+ write(afd, abuf, ap-abuf);
+ return Userbreak;
+}
+
+static void
+initinst(void)
+{
+ int m, i;
+ InternalSample *t, *p;
+
+ m = sizeof(adl) / 14;
+
+ if((t = calloc(m, sizeof *t)) == nil)
+ sysfatal("initinst: %r");
+ p = t;
+ /* FIXME: .D needs to be uchar(?) */
+ /* FIXME: use char adl[][14] rather than this */
+ /* FIXME: find out where the values are from */
+ for(i=0; i<m; i++){
+ p->D[0] = adl[i * 14 + 3];
+ p->D[1] = adl[i * 14 + 9];
+ p->D[2] = adl[i * 14 + 4];
+ p->D[3] = adl[i * 14 + 10];
+ p->D[4] = adl[i * 14 + 5];
+ p->D[5] = adl[i * 14 + 11];
+ p->D[6] = adl[i * 14 + 6];
+ p->D[7] = adl[i * 14 + 12];
+ p->D[8] = adl[i * 14 + 7] & 3;
+ p->D[9] = adl[i * 14 + 13] & 3;
+ p->D[10]= adl[i * 14 + 8];
+ mdat.Instr[i] = p++;
+ }
+}
+
+/* FIXME: get8/16/32 approach is better */
+/* FIXME: use bio.h */
+static void
+eread(int fd, void *buf, int n)
+{
+ if(readn(fd, buf, n) < 0)
+ sysfatal("read: %r");
+}
+
+static void
+readmid(char *mid)
+{
+ int i, fd;
+ uint n;
+ uchar id[4];
+
+ fd = 0;
+ if(mid != nil && (fd = open(mid, OREAD)) < 0)
+ sysfatal("open: %r");
+
+ /* FIXME: get8/16/32 better; Conv shit is also stupid */
+ /* FIXME: also, don't read file into memory but use bio? */
+ eread(fd, id, 4);
+ if(memcmp(id, "MThd", 4) != 0)
+ sysfatal("invalid header");
+ eread(fd, id, 4);
+ if(ConvL(id) != 6)
+ sysfatal("invalid midi file");
+
+ eread(fd, id, 2);
+ MIDI.Fmt = ConvI(id);
+ eread(fd, id, 2);
+ MIDI.TrackCount = ConvI(id);
+ eread(fd, id, 2);
+ MIDI.DeltaTicks = ConvI(id);
+
+ MIDI.TrackLen = calloc(MIDI.TrackCount, sizeof *MIDI.TrackLen);
+ if(MIDI.TrackLen == nil)
+ sysfatal("calloc len %ux %ux: %r", MIDI.TrackCount, sizeof *MIDI.TrackLen);
+ MIDI.Tracks = calloc(MIDI.TrackCount, sizeof *MIDI.Tracks);
+ if(MIDI.Tracks == nil)
+ sysfatal("calloc trs %ux %ux: %r", MIDI.TrackCount, sizeof *MIDI.Tracks);
+
+ for(i = 0; i < MIDI.TrackCount; i++){
+ eread(fd, id, 4);
+ if(memcmp(id, "MTrk", 4) != 0)
+ sysfatal("invalid track");
+
+ eread(fd, id, 4);
+ n = ConvL(id);
+ MIDI.TrackLen[i] = n;
+ if((MIDI.Tracks[i] = mallocz(n, 1)) == nil)
+ sysfatal("mallocz %ux: %r", n);
+
+ eread(fd, MIDI.Tracks[i], n);
+ }
+ close(fd);
+
+ MIDI.initempo = 150000;
+}
+
+static void
+usage(void)
+{
+ print("%s [-cd] [midfile]\n", argv0);
+ exits("usage");
+}
+
+void
+main(int argc, char **argv)
+{
+ int i;
+
+ ARGBEGIN{
+ case 'c':
+ afd = 1;
+ break;
+ case 'd':
+ verbose++;
+ break;
+ default:
+ usage();
+ }ARGEND
+
+ readmid(*argv);
+ initinst();
+ for(i = 0; i < NCHIPS; i++)
+ chip[i] = ym3812_init(1789772*2, RATE);
+
+ if(afd < 0 && (afd = open("/dev/audio", OWRITE)) < 0)
+ sysfatal("open: %r");
+ ap = abuf;
+ play();
+
+ for(i = 0; i < NCHIPS; i++)
+ ym3812_shutdown(chip[i]);
+ exits(nil);
+}