2020-09-30 17:12:29 +02:00

1558 lines
43 KiB
C

/*
* Copyright (c) 1992 Microsoft Corporation
*/
/*
* definition of interface functions to the adlib midi device type.
*
* These functions are called from midi.c when the kernel driver
* has decreed that this is an adlib-compatible device.
*
* Geraint Davies, Dec 92
*/
#include <windows.h>
#include <mmsystem.h>
#include <mmddk.h>
#include "driver.h"
#include "adlib.h"
/*
* overview
*
* The FM synthesis chip consists of 18 operator cells or 'slots'. Each slot
* can produce a sine wave modified by a number of parameters such
* as frequency, output level and envelope shape (attack/decay/sustain
* release). Slots are arranged in pairs, with one slot modulating
* the sine wave of another to produce the harmonics desired for
* a given instrument sound. Thus one pair of slots is a 'voice' and can
* play one note at a time.
*
* In percussive mode (which we always use), there are 6 melodic voices
* available, and one voice for the base drum. The remaining four slots
* are used singly rather than in pairs to produce four further percussive
* voices. The 6 melodic voices will be changed to any given timbre
* as needed. The five percussive voices are fixed to particular instrument
* timbres: bass drum, snare, tom-tom, hi-hat and cymbal.
*
* To play a note, we first find a free voice of the appropriate type. If
* there are none free, we use the oldest busy one. We then set the
* parameters for both operator slots from the patch table - this table gives
* parameter settings for the different instrument timbres available.
* We adjust the output level and frequency depending on the
* note played and the velocity it was played with, and then switch on the
* note.
*/
/* --- typedefs ------------------------------------------------- */
#define NUMVOICES (11) // number of voices we have
#define NUMMELODIC (6) // number of melodic voices
#define NUMPERCUSSIVE (5) // number of percussive voices
#define MAXPATCH 180 // nr of patches (including drums)
#define MAXVOLUME 0x7f
#define NUMLOCPARAM 14 // number of loc params per slot
#define FIRSTDRUMNOTE 35
#define LASTDRUMNOTE 81
#define NUMDRUMNOTES (LASTDRUMNOTE - FIRSTDRUMNOTE + 1)
#define MAX_PITCH 0x3fff // maximum pitch bend value
#define MID_PITCH 0x2000 // mid pitch bend value (no shift)
#define PITCHRANGE 2 // total bend range +- 2 semitones
#define NR_STEP_PITCH 25 // steps per half-tone for pitch bend
#define MID_C 60 // MIDI standard mid C
#define CHIP_MID_C 48 // sound chip mid C
/*
* to write to the device, we write a SYNTH_DATA port,data pair
* to the kernel driver.
*/
#define SndOutput(p,d) MidiSendFM(p, d)
/****************************************************************************
*
* definitions of sound chip parameters
*/
// parameters of each voice
#define prmKsl 0 // key scale level (KSL) - level controller
#define prmMulti 1 // frequency multiplier (MULTI) - oscillator
#define prmFeedBack 2 // modulation feedback (FB) - oscillator
#define prmAttack 3 // attack rate (AR) - envelope generator
#define prmSustain 4 // sustain level (SL) - envelope generator
#define prmStaining 5 // sustaining sound (SS) - envelope generator
#define prmDecay 6 // decay rate (DR) - envelope generator
#define prmRelease 7 // release rate (RR) - envelope generator
#define prmLevel 8 // output level (OL) - level controller
#define prmAm 9 // amplitude vibrato (AM) - level controller
#define prmVib 10 // frequency vibrator (VIB) - oscillator
#define prmKsr 11 // envelope scaling (KSR) - envelope generator
#define prmFm 12 // fm=0, additive=1 (FM) (operator 0 only)
#define prmWaveSel 13 // wave select
// global parameters
#define prmAmDepth 14
#define prmVibDepth 15
#define prmNoteSel 16
#define prmPercussion 17
// percussive voice numbers (0-5 are melodic voices, 2 op):
#define BD 6 // bass drum (2 op)
#define SD 7 // snare drum (1 op)
#define TOM 8 // tom-tom (1 op)
#define CYMB 9 // cymbal (1 op)
#define HIHAT 10 // hi-hat (1 op)
// In percussive mode, the last 4 voices (SD TOM HH CYMB) are created
// using melodic voices 7 & 8. A noise generator uses channels 7 & 8
// frequency information for creating rhythm instruments. Best results
// are obtained by setting TOM two octaves below mid C and SD 7 half-tones
// above TOM. In this implementation, only the TOM pitch may vary, with the
// SD always set 7 half-tones above.
#define TOM_PITCH 24 // best frequency, in range of 0 to 95
#define TOM_TO_SD 7 // 7 half-tones between voice 7 & 8
#define SD_PITCH (TOM_PITCH + TOM_TO_SD)
/****************************************************************************
*
* bank file support
*
***************************************************************************/
#define BANK_SIG_LEN 6
#define BANK_FILLER_SIZE 8
typedef BYTE *HPBYTE;
typedef BYTE NEAR * NPBYTE;
typedef WORD NEAR * NPWORD;
// instrument bank file header
typedef struct {
char majorVersion;
char minorVersion;
char sig[BANK_SIG_LEN]; // signature: "ADLIB-"
WORD nrDefined; // number of list entries used
WORD nrEntry; // number of total entries in list
long offsetIndex; // offset of start of list of names
long offsetTimbre; // offset of start of data
char filler[BANK_FILLER_SIZE]; // must be zero
} BANKHDR, NEAR *NPBANKHDR, FAR *LPBANKHDR;
// all the parameters for one slot
typedef struct {
BYTE ksl; // KSL
BYTE freqMult; // MULTI
BYTE feedBack; // FB
BYTE attack; // AR
BYTE sustLevel; // SL
BYTE sustain; // SS
BYTE decay; // DR
BYTE release; // RR
BYTE output; // OL
BYTE am; // AM
BYTE vib; // VIB
BYTE ksr; // KSR
BYTE fm; // FM
} OPERATOR, NEAR *NPOPERATOR, FAR *LPOPERATOR;
// definition of a particular instrument sound or timbre -
// one of these per patch
typedef struct {
BYTE mode; // 0 = melodic, 1 = percussive
BYTE percVoice; // if mode == 1, voice number to be used
OPERATOR op0; // params for slot 0
OPERATOR op1; // params for slot 1
BYTE wave0; // waveform for operator 0
BYTE wave1; // waveform for operator 1
} TIMBRE, NEAR *NPTIMBRE, FAR *LPTIMBRE;
typedef struct drumpatch_tag {
BYTE patch; // the patch to use
BYTE note; // the note to play
} DRUMPATCH;
// format of drumkit.dat file
typedef struct drumfilepatch_tag {
BYTE key; // the key to map
BYTE patch; // the patch to use
BYTE note; // the note to play
} DRUMFILEPATCH, *PDRUMFILEPATCH;
typedef struct _VOICE {
BYTE alloc; // is voice allocated?
BYTE note; // note that is currently being played
BYTE channel; // channel that it is being played on
BYTE volume; // current volume setting of voice
DWORD dwTimeStamp; // time voice was allocated
} VOICE;
#define GetLocPrm(slot_num, prm) ((WORD)paramSlot[slot_num][prm])
/* --- module data ---------------------------------------------- */
// this table gives the offset of each slot within the chip.
BYTE offsetSlot[] = {
0, 1, 2, 3, 4, 5,
8, 9, 10, 11, 12, 13,
16, 17, 18, 19, 20, 21
};
static BYTE percMasks[] = {
0x10, 0x08, 0x04, 0x02, 0x01 };
// voice number associated with each slot (melodic mode only)
static BYTE voiceSlot[] = {
0, 1, 2,
0, 1, 2,
3, 4, 5,
3, 4, 5,
6, 7, 8,
6, 7, 8,
};
// slot numbers for percussive voices (0 indicates that there is only one slot)
static BYTE slotPerc[][2] = {
{12, 15}, // Bass Drum
{16, 0}, // SD
{14, 0}, // TOM
{17, 0}, // TOP-CYM
{13, 0} // HH
};
// slot numbers as a function of the voice and the operator (melodic only)
static BYTE slotVoice[][2] = {
{0, 3}, // voice 0
{1, 4}, // 1
{2, 5}, // 2
{6, 9}, // 3
{7, 10}, // 4
{8, 11}, // 5
{12, 15}, // 6
{13, 16}, // 7
{14, 17} // 8
};
// this table indicates if the slot is a modulator (0) or a carrier (1).
BYTE operSlot[] = {
0, 0, 0, // 1 2 3
1, 1, 1, // 4 5 6
0, 0, 0, // 7 8 9
1, 1, 1, // 10 11 12
0, 0, 0, // 13 14 15
1, 1, 1, // 16 17 18
};
// this array adjusts the octave of a note for certain patches.
static char patchKeyOffset[] = {
0, -12, 12, 0, 0, 12, -12, 0, 0, -24, // 0 - 9
0, 0, 0, 0, 0, 0, 0, 0, -12, 0, // 10 - 19
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 20 - 29
0, 0, 12, 12, 12, 0, 0, 12, 12, 0, // 30 - 39
-12, -12, 0, 12, -12, -12, 0, 12, 0, 0, // 40 - 49
-12, 0, 0, 0, 12, 12, 0, 0, 12, 0, // 50 - 59
0, 0, 12, 0, 0, 0, 12, 12, 0, 12, // 60 - 69
0, 0, -12, 0, -12, -12, 0, 0, -12, -12, // 70 - 79
0, 0, 0, 0, 0, -12, -19, 0, 0, -12, // 80 - 89
0, 0, 0, 0, 0, 0, -31, -12, 0, 12, // 90 - 99
12, 12, 12, 0, 12, 0, 12, 0, 0, 0, // 100 - 109
0, 12, 0, 0, 0, 0, 12, 12, 12, 0, // 110 - 119
0, 0, 0, 0, -24, -36, 0, 0}; // 120 - 127
static BYTE loudervol[128] = {
0, 0, 65, 65, 66, 66, 67, 67, // 0 - 7
68, 68, 69, 69, 70, 70, 71, 71, // 8 - 15
72, 72, 73, 73, 74, 74, 75, 75, // 16 - 23
76, 76, 77, 77, 78, 78, 79, 79, // 24 - 31
80, 80, 81, 81, 82, 82, 83, 83, // 32 - 39
84, 84, 85, 85, 86, 86, 87, 87, // 40 - 47
88, 88, 89, 89, 90, 90, 91, 91, // 48 - 55
92, 92, 93, 93, 94, 94, 95, 95, // 56 - 63
96, 96, 97, 97, 98, 98, 99, 99, // 64 - 71
100, 100, 101, 101, 102, 102, 103, 103, // 72 - 79
104, 104, 105, 105, 106, 106, 107, 107, // 80 - 87
108, 108, 109, 109, 110, 110, 111, 111, // 88 - 95
112, 112, 113, 113, 114, 114, 115, 115, // 96 - 103
116, 116, 117, 117, 118, 118, 119, 119, // 104 - 111
120, 120, 121, 121, 122, 122, 123, 123, // 112 - 119
124, 124, 125, 125, 126, 126, 127, 127}; // 120 - 127
/*
* attenuation setting for each slot - combination
* of the channel attenuation for this channel, and the
* velocity (converted to attenuation). Combine with
* global attenuation and timbre output attenuation before
* writing to device.
*/
WORD slotAtten[18]; // vol-control attenuation of slots
WORD wSynthAtten; // overall volume setting
BYTE gbChanAtten[NUMCHANNELS]; // attenuation for each channel
VOICE voices[11]; // 9 voices if melodic mode or 11 if percussive
BYTE voiceKeyOn[11]; // keyOn bit for each voice (implicit 0 init)
BYTE paramSlot[18][NUMLOCPARAM]; // all the parameters of slots...
BYTE percBits; // control bits of percussive voices
WORD fNumNotes[NR_STEP_PITCH][12];
PWORD fNumFreqPtr[11]; // lines of fNumNotes table (one per voice)
int halfToneOffset[11]; // one per voice
BYTE notePitch[11]; // pitch value for each voice (implicit 0 init)
// patches - parameters for different instrument timbres
TIMBRE patches[MAXPATCH]; // patch data
DRUMPATCH drumpatch[NUMDRUMNOTES]; // drum kit data
DWORD dwAge = 0; // voice relative age
/* --- internal functions --------------------------------------- */
/* -- initialisation --- */
/****************************************************************************
* @doc INTERNAL
*
* @api int | LoadPatches | Reads the patch set from the BANK resource and
* builds the <p patches> array.
*
* @rdesc Returns the number of patches loaded, or 0 if an error occurs.
***************************************************************************/
static BYTE PatchRes[] = {
#include "adlib.dat"
};
int LoadPatches(void)
{
#ifdef WIN16
HANDLE hResInfo;
HANDLE hResData;
#endif
LPSTR lpRes;
int iPatches;
DWORD dwOffset;
DWORD dwResSize;
LPTIMBRE lpBankTimbre;
LPTIMBRE lpPatchTimbre;
LPBANKHDR lpBankHdr;
// find resource and get its size
#ifdef WIN16
hResInfo = FindResource(ghInstance, MAKEINTRESOURCE(DEFAULTBANK), MAKEINTRESOURCE(RT_BANK));
if (!hResInfo) {
D1(("Default bank resource not found"));
return 0;
}
dwResSize = (DWORD)SizeofResource(ghInstance, hResInfo);
// load and lock resource
hResData = LoadResource(ghInstance, hResInfo);
if (!hResData) {
D1(("Bank resource not loaded"));
return 0;
}
lpRes = LockResource(hResData);
if (!lpRes) {
D1(("Bank resource not locked"));
return 0;
}
#else
lpRes = PatchRes;
dwResSize = sizeof(PatchRes);
#endif
// read the bank resource, loading patches as we find them
D2(("loading patches"));
lpBankHdr = (LPBANKHDR)lpRes;
dwOffset = lpBankHdr->offsetTimbre; // point to first one
for (iPatches = 0; iPatches < MAXPATCH; iPatches++) {
lpBankTimbre = (LPTIMBRE)(lpRes + dwOffset);
lpPatchTimbre = &patches[iPatches];
*lpPatchTimbre = *lpBankTimbre;
dwOffset += sizeof(TIMBRE);
if (dwOffset + sizeof(TIMBRE) > dwResSize) {
D1(("Attempt to read past end of bank resource"));
break;
}
}
#ifdef WIN16
UnlockResource(hResData);
FreeResource(hResData);
#endif
return iPatches;
}
/****************************************************************************
* @doc INTERNAL
*
* @api int | LoadDrumPatches | Reads the drum kit patch set from the
* DRUMKIT resource and builds the <p drumpatch> array.
*
* @comm Each entry of the <t drumpatch> array (representing a key number
* from the "drum patch") consists of a patch number and note number
* from some other patch.
*
* @rdesc Returns the number of patches loaded, or 0 if an error occurs.
***************************************************************************/
static BYTE DrumRes[] = {
#include "drumkit.dat"
};
int LoadDrumPatches(void)
{
#ifdef WIN16
HANDLE hResInfo;
HANDLE hResData;
#endif
LPSTR lpRes;
int iPatches;
int key;
DWORD dwOffset;
DWORD dwResSize;
PDRUMFILEPATCH lpResPatch;
#ifdef WIN16
// find resource and get its size
hResInfo = FindResource(ghInstance, MAKEINTRESOURCE(DEFAULTDRUMKIT), MAKEINTRESOURCE(RT_DRUMKIT));
if (!hResInfo) {
D1(("Default drum resource not found"));
return 0;
}
dwResSize = (DWORD)SizeofResource(ghInstance, hResInfo);
// load and lock resource
hResData = LoadResource(ghInstance, hResInfo);
if (!hResData) {
D1(("Drum resource not loaded"));
return 0;
}
lpRes = LockResource(hResData);
if (!lpRes) {
D1(("Drum resource not locked"));
return 0;
}
#else
lpRes = DrumRes;
dwResSize = sizeof(DrumRes);
#endif
// read the drum resource, loading patches as we find them
D2(("reading drum data"));
dwOffset = 0;
for (iPatches = 0; iPatches < NUMDRUMNOTES; iPatches++) {
lpResPatch = (PDRUMFILEPATCH)(lpRes + dwOffset);
key = lpResPatch->key;
if ((key >= FIRSTDRUMNOTE) && (key <= LASTDRUMNOTE)) {
drumpatch[key - FIRSTDRUMNOTE].patch = lpResPatch->patch;
drumpatch[key - FIRSTDRUMNOTE].note = lpResPatch->note;
}
else {
D1(("Drum patch key out of range"));
}
dwOffset += sizeof(DRUMFILEPATCH);
if (dwOffset > dwResSize) {
D1(("Attempt to read past end of drum resource"));
break;
}
}
#ifdef WIN16
UnlockResource(hResData);
FreeResource(hResData);
#endif
return iPatches;
}
/****************************************************************************
* @doc INTERNAL
*
* @api long | CalcPremFNum | Calculates some magic number that is used in
* setting the values in the <p fNumNotes> table.
*
* @parm int | numDeltaDemiTon | Numerator (-100 to +100).
*
* @parm int | denDeltaDemiTon | Denominator (1 to 100).
*
* @comm If the numerator (numDeltaDemiTon) is positive, the frequency is
* increased; if negative, it is decreased. The function calculates:
* f8 = Fb(1 + 0.06 num /den) (where Fb = 26044 * 2 / 25)
* fNum8 = f8 * 65536 * 72 / 3.58e6
*
* @rdesc Returns fNum8, which is the binary value of the frequency 260.44 (C)
* shifted by +/- <p numdeltaDemiTon> / <p denDeltaDemiTon> * 8.
***************************************************************************/
long NEAR CalcPremFNum(int numDeltaDemiTon, int denDeltaDemiTon)
{
long f8;
long fNum8;
long d100;
d100 = denDeltaDemiTon * 100;
f8 = (d100 + 6 * numDeltaDemiTon) * (26044L * 2L);
f8 /= d100 * 25;
fNum8 = f8 * 16384;
fNum8 *= 9L;
fNum8 /= 179L * 625L;
return fNum8;
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | SetFNum | Initializes a line in the frequency table with
* shifted frequency values. The values are shifted a fraction (num/den)
* of a half-tone.
*
* @parm NPWORD | fNumVec | The line from the frequency table.
*
* @parm int | num | Numerator.
*
* @parm int | den | Denominator.
*
* @xref CalcPremFNum
*
* @rdesc There is no return value.
***************************************************************************/
void NEAR SetFNum(NPWORD fNumVec, int num, int den)
{
int i;
long val;
*fNumVec++ = (WORD)((4 + (val = CalcPremFNum(num, den))) >> 3);
for (i = 1; i < 12; i++) {
val *= 106;
*fNumVec++ = (WORD)((4 + (val /= 100)) >> 3);
}
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | InitFNums | Initializes all lines of the frequency table
* (the <p fNumNotes> array). Each line represents 12 half-tones shifted
* by (n / NR_STEP_PITCH), where 'n' is the line number and ranges from
* 1 to NR_STEP_PITCH.
*
* @rdesc There is no return value.
***************************************************************************/
void NEAR InitFNums(void)
{
WORD i;
WORD num; // numerator
WORD numStep; // step value for numerator
WORD row; // row in the frequency table
// calculate each row in the fNumNotes table
numStep = 100 / NR_STEP_PITCH;
for (num = row = 0; row < NR_STEP_PITCH; row++, num += numStep)
SetFNum(fNumNotes[row], num, 100);
// fNumFreqPtr has an element for each voice, pointing to the
// appropriate row in the fNumNotes table. They're all initialized
// to the first row, which represents no pitch shift.
for (i = 0; i < 11; i++) {
fNumFreqPtr[i] = fNumNotes[0];
halfToneOffset[i] = 0;
}
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | SoundChut | Sets the frequency of voice <p voice> to 0 Hz.
*
* @parm BYTE | voice | Specifies which voice to set.
*
* @rdesc There is no return value.
***************************************************************************/
void NEAR SoundChut(BYTE voice)
{
SndOutput((BYTE)(0xA0 | voice), 0);
SndOutput((BYTE)(0xB0 | voice), 0);
}
/* --- write parameters to chip ------------------------*/
/*
* switch chip to rhythm (percussive) mode, set the amplitude and
* vibrato depth and switch on any percussion voices that are on
*/
void SndSAmVibRhythm(void)
{
// we always set amdepth, vibdepth to 0 and perc mode on
// t1 = (BYTE)(amDepth ? 0x80 : 0);
// t1 |= vibDepth ? 0x40 : 0;
// t1 |= (percussionmode ? 0x20 : 0);
SndOutput((BYTE)0xBD, (BYTE)(0x20|percBits));
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | SndSNoteSel | Sets the NoteSel value.
*
* @rdesc There is no return value.
***************************************************************************/
void SndSNoteSel(BOOL bNoteSel)
{
SndOutput(0x08, (BYTE)(bNoteSel ? 64 : 0));
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | SndSKslLevel | Sets the KSL and LEVEL values.
*
* @parm BYTE | slot | Specifies which slot to set.
*
* @rdesc There is no return value.
***************************************************************************/
void SndSKslLevel(BYTE slot)
{
WORD t1;
t1 = GetLocPrm(slot, prmLevel) & 0x3f;
t1 += slotAtten[slot];
t1 = min (t1, 0x3f);
t1 |= GetLocPrm(slot, prmKsl) << 6;
SndOutput((BYTE)(0x40 | offsetSlot[slot]), (BYTE) t1);
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | SndSAVEK | Sets the AM, VIB, EG-TYP (sustaining), KSR, and
* MULTI values.
*
* @parm BYTE | slot | Specifies which slot to set.
*
* @rdesc There is no return value.
***************************************************************************/
void SndSAVEK(BYTE slot)
{
BYTE t1;
t1 = (BYTE)(GetLocPrm(slot, prmAm) ? 0x80 : 0);
t1 += GetLocPrm(slot, prmVib) ? 0x40 : 0;
t1 += GetLocPrm(slot, prmStaining) ? 0x20 : 0;
t1 += GetLocPrm(slot, prmKsr) ? 0x10 : 0;
t1 += GetLocPrm(slot, prmMulti) & 0xf;
SndOutput((BYTE)(0x20 | offsetSlot[slot]), t1);
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | SndSFeedFm | Sets the FEEDBACK and FM (connection) values.
* Applicable only to operator 0 for melodic voices.
*
* @parm BYTE | slot | Specifies which slot to set.
*
* @rdesc There is no return value.
***************************************************************************/
void SndSFeedFm(BYTE slot)
{
BYTE t1;
if (operSlot[slot])
return;
t1 = (BYTE)(GetLocPrm(slot, prmFeedBack) << 1);
t1 |= GetLocPrm(slot, prmFm) ? 0 : 1;
SndOutput((BYTE)(0xC0 | voiceSlot[slot]), t1);
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | SndSAttDecay | Sets the ATTACK and DECAY values.
*
* @parm BYTE | slot | Specifies which slot to set.
*
* @rdesc There is no return value.
***************************************************************************/
void SndSAttDecay(BYTE slot)
{
BYTE t1;
t1 = (BYTE)(GetLocPrm(slot, prmAttack) << 4);
t1 |= GetLocPrm(slot, prmDecay) & 0xf;
SndOutput((BYTE)(0x60 | offsetSlot[slot]), t1);
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | SndSSusRelease | Sets the SUSTAIN and RELEASE values.
*
* @parm BYTE | slot | Specifies which slot to set.
*
* @rdesc There is no return value.
***************************************************************************/
void SndSSusRelease(BYTE slot)
{
BYTE t1;
t1 = (BYTE)(GetLocPrm(slot, prmSustain) << 4);
t1 |= GetLocPrm(slot, prmRelease) & 0xf;
SndOutput((BYTE)(0x80 | offsetSlot[slot]), t1);
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | SndWaveSelect | Sets the wave-select parameter.
*
* @parm BYTE | slot | Specifies which slot to set.
*
* @rdesc There is no return value.
***************************************************************************/
void SndWaveSelect(BYTE slot)
{
BYTE wave;
wave = (BYTE)(GetLocPrm(slot, prmWaveSel) & 0x03);
SndOutput((BYTE)(0xE0 | offsetSlot[slot]), wave);
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | SndSetAllPrm | Transfers all the parameters from slot <p slot>
* to the chip.
*
* @parm BYTE | slot | Specifies which slot to set.
*
* @rdesc There is no return value.
***************************************************************************/
void SndSetAllPrm(BYTE slot)
{
/* global am-depth and vib-depth settings */
SndSAmVibRhythm();
/* note sel is always false */
SndSNoteSel(FALSE);
/* slot-specific parameters */
/* initialise volume to minimum to avoid clicks if the note is
* off but still playing (during decay phase).
*
* only applies to carrier slots.
*/
if (operSlot[slot]) {
/* its a carrier slot */
slotAtten[slot] = 0x3f; // max attenuation
}
SndSKslLevel(slot);
SndSFeedFm(slot);
SndSAttDecay(slot);
SndSSusRelease(slot);
SndSAVEK(slot);
SndWaveSelect(slot);
}
/* --- setting slot parameters ------------------ */
/****************************************************************************
* @doc INTERNAL
*
* @api void | SetSlotParam | Sets the 14 parameters (13 in <p param>,
* 1 in <p waveSel>) of slot <p slot>. Updates both the parameter array
* and the chip.
*
* @parm BYTE | slot | Specifies which slot to set.
*
* @parm NPBYTE | param | Pointer to the new parameter array.
*
* @parm BYTE | waveSel | The new waveSel value.
*
* @rdesc There is no return value.
***************************************************************************/
void SetSlotParam(BYTE slot, NPOPERATOR pOper, BYTE waveSel)
{
LPBYTE ptr;
ptr = &paramSlot[slot][0];
*ptr++ = pOper->ksl;
*ptr++ = pOper->freqMult;
*ptr++ = pOper->feedBack;
*ptr++ = pOper->attack;
*ptr++ = pOper->sustLevel;
*ptr++ = pOper->sustain;
*ptr++ = pOper->decay;
*ptr++ = pOper->release;
*ptr++ = pOper->output;
*ptr++ = pOper->am;
*ptr++ = pOper->vib;
*ptr++ = pOper->ksr;
*ptr++ = pOper->fm;
*ptr = waveSel & 0x3;
// set default volume settings
slotAtten[slot] = 0;
SndSetAllPrm(slot);
}
/*
* set this voice up to the correct parameters for a timbre
* copy the parameters to the slot array and write them to the
* chip
*
*/
void SetVoiceTimbre(BYTE voice, NPTIMBRE pTimbre)
{
if (voice < BD) { // melodic only
SetSlotParam(slotVoice[voice][0], &pTimbre->op0, pTimbre->wave0);
SetSlotParam(slotVoice[voice][1], &pTimbre->op1, pTimbre->wave1);
}
else if (voice == BD) { // bass drum
SetSlotParam(slotPerc[0][0], &pTimbre->op0, pTimbre->wave0);
SetSlotParam(slotPerc[0][1], &pTimbre->op1, pTimbre->wave1);
} else { // percussion, 1 slot
SetSlotParam(slotPerc[voice - BD][0], &pTimbre->op0, pTimbre->wave0);
}
}
/* --- frequency calculation -------------------- */
/* convert the given pitch (0 .. 95) into a frequency
* and write to the chip.
*
* this will turn the note on if keyon is true.
*/
VOID SetFreq(BYTE voice, BYTE pitch, BYTE keyOn)
{
WORD FNum;
BYTE t1;
// remember the keyon and pitch of the voice
voiceKeyOn[voice] = keyOn;
notePitch[voice] = pitch;
pitch += halfToneOffset[voice];
if (pitch > 95)
pitch = 95;
// get the FNum for the voice
FNum = * (fNumFreqPtr[voice] + pitch % 12);
// output the FNum, KeyOn and Block values
SndOutput((BYTE)(0xA0 | voice), (BYTE)FNum); // FNum bits 0 - 7 (D0 - D7)
t1 = (BYTE)(keyOn ? 32 : 0); // Key On (D5)
t1 += (pitch / 12 << 2); // Block (D2 - D4)
t1 += (0x3 & (FNum >> 8)); // FNum bits 8 - 9 (D0 - D1)
SndOutput((BYTE)(0xB0 | voice), t1);
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | ChangePitch | This routine sets the <t halfToneOffset[]> and
* <t fNumFreqPtr[]> arrays. These two global variables are used to
* determine the frequency variation to use when a note is played.
*
* @parm BYTE | voice | Specifies which voice to use.
*
* @parm WORD | pitchBend | Specifies the pitch bend value (0 to 0x3fff,
* where 0x2000 is exact tuning).
*
* @rdesc There is no return value.
***************************************************************************/
void ChangePitch(BYTE voice, WORD pitchBend)
{
int t1;
int t2;
int delta;
int pitchrange;
/* pitch bend range 0-3fff is +-PITCHRANGE semitones. We move
* NR_STEP_PITCH steps per semitone.
* so the bend range is +- (PITCHRANGE * NR_STEP_PITCH) steps.
*/
pitchrange = PITCHRANGE * NR_STEP_PITCH;
t1 = (int)(((long)((int)pitchBend - MID_PITCH) * pitchrange) / MID_PITCH);
if (t1 < 0)
{
t2 = NR_STEP_PITCH - 1 - t1;
halfToneOffset[voice] = -(t2 / NR_STEP_PITCH);
delta = (t2 - NR_STEP_PITCH + 1) % NR_STEP_PITCH;
if (delta) {
delta = NR_STEP_PITCH - delta;
}
}
else
{
halfToneOffset[voice] = t1 / NR_STEP_PITCH;
delta = t1 % NR_STEP_PITCH;
}
fNumFreqPtr[voice] = fNumNotes[delta];
}
/****************************************************************************
* @doc INTERNAL
*
* @api void | SetVoicePitch | Changes the pitch value of a voice. Does
* not affect the percussive voices, except for the bass drum. The change
* takes place immediately.
*
* @parm BYTE | voice | Specifies which voice to set.
*
* @parm WORD | pitchBend | Specifies the new pitch bend value (0 to 0x3fff,
* where 0x2000 == exact tuning).
*
* @comm The variation in pitch is a function of the previous call to
* <f SetPitchRange> and the value of <p pitchBend>. A value of 0 means
* -half-tone * pitchRangeStep, 0x2000 means no variation (exact pitch) and
* 0x3fff means +half-tone * pitchRangeStep.
*
* @rdesc There is no return value.
***************************************************************************/
void SetVoicePitch(BYTE voice, WORD pitchBend)
{
if (voice <= BD) { // melodic and bass drum voices
if (pitchBend > MAX_PITCH) {
pitchBend = MAX_PITCH;
}
ChangePitch(voice, pitchBend);
SetFreq(voice, notePitch[voice], voiceKeyOn[voice]);
}
}
/* --- volume calculation ------------------------ */
/*
* set the attenuation for a slot (combine the channel attenuation
* setting with the key velocity).
*/
VOID SetVoiceAtten(BYTE voice, BYTE bChannel, BYTE bVelocity)
{
BYTE bAtten;
BYTE slot;
PBYTE slots;
// scale up the volume since the patch maps are too quiet
//bVelocity = loudervol[bVelocity];
/*
* set channel attenuation
*/
bAtten = gbVelocityAtten[bVelocity >> 2] + gbChanAtten[bChannel];
/*
* add on any global (volume-setting) attenuation
*/
bAtten += wSynthAtten;
/* save this for each non-modifier slot */
if (voice <= BD) { // melodic voice
slots = slotVoice[voice];
slotAtten[slots[1]] = bAtten;
SndSKslLevel(slots[1]);
if (!GetLocPrm(slots[0], prmFm)) {
// additive synthesis: set volume of first slot too
slotAtten[slots[0]] = bAtten;
SndSKslLevel(slots[0]);
}
}
else { // percussive voice
slot = slotPerc[voice - BD][0];
slotAtten[slot] = bAtten;
SndSKslLevel(slot);
}
}
/* adjust each slot attenuation to allow for a global volume
* change - note that we only need to change the
* carrier, not the modifier slots.
*
* change for channel bChannel, or all channels if bChannel is
* 0xff
*/
VOID ChangeAtten(BYTE bChannel, int iChangeAtten)
{
BYTE voice;
BYTE slot;
PBYTE slots;
/* find voices using this channel */
for (voice = 0; voice < NUMVOICES; voice++) {
if ((voices[voice].channel == bChannel) || (bChannel == 0xff)) {
if (voice <= BD) {
/* melodic voice */
slots = slotVoice[voice];
slotAtten[slots[1]] += iChangeAtten;
SndSKslLevel(slots[1]);
if (!GetLocPrm(slots[0], prmFm)) {
// additive synthesis: set volume of first slot too
slotAtten[slots[0]] += iChangeAtten;
SndSKslLevel(slots[0]);
}
} else {
slot = slotPerc[voice - BD][0];
slotAtten[slot] += iChangeAtten;
SndSKslLevel(slot);
}
}
}
}
/* --- note on/off ------------------------------- */
/* switch off the note a given voice is playing */
void NoteOff(BYTE voice)
{
if (voice < BD) {
/* melodic voice */
SetFreq(voice, notePitch[voice], 0);
} else {
percBits &= ~percMasks[voice-BD];
SndSAmVibRhythm();
}
}
/* switch on a voice at a given note (0 - 127, where 60 is middle c) */
void NoteOn(BYTE voice, BYTE bNote)
{
BYTE bPitch;
/* convert note to chip pitch */
if (bNote < (MID_C - CHIP_MID_C)) {
bPitch = 0;
} else {
bPitch = bNote - (MID_C - CHIP_MID_C);
}
if (voice < BD) {
/* melodic voice */
/* set frequency and start note */
SetFreq(voice, bPitch, 1);
} else {
/*
* nb we don't change the pitch of some percussion instruments.
*
* also note that for percussive instruments (including BD),
* the note-on setting should always be 0. You switch the percussion
* on by writing to the AmVibRhythm register.
*/
if (voice == BD) {
SetFreq(voice, bPitch, 0);
} else if (voice == TOM) {
/*
* for best sounds, we do change the TOM freq, but we always keep
* the SD 7 semi-tones above TOM.
*/
SetFreq(TOM, bPitch, 0);
SetFreq(SD, (BYTE)(bPitch + TOM_TO_SD), 0);
}
/* other instruments never change */
percBits |= percMasks[voice - BD];
SndSAmVibRhythm();
}
}
/* -- voice allocation -------- */
/*
* find the voice that is currently playing a given channel/note pair
* (if any)
*/
BYTE
FindVoice(BYTE bNote, BYTE bChannel)
{
BYTE i;
for (i = 0; i < (BYTE)NUMVOICES; i++) {
if ((voices[i].alloc) && (voices[i].note == bNote)
&& (voices[i].channel == bChannel)) {
voices[i].dwTimeStamp = dwAge++;
return(i);
}
}
/* no voice is playing this */
return(0xff);
}
/*
* mark a voice as unused
*/
VOID FreeVoice(voice)
{
voices[voice].alloc = 0;
}
/*
* GetNewVoice - allocate a voice to play this note. if no voices are
* free, re-use the note with the oldest timestamp.
*/
BYTE GetNewVoice(BYTE patch, BYTE note, BYTE channel)
{
BYTE i;
BYTE voice;
BYTE bVoiceToUse, bVoiceSame, bVoiceOldest;
DWORD dwOldestTime = dwAge + 1; // init to past current "time"
DWORD dwOldestSame = dwAge + 1;
DWORD dwOldestOff = dwAge + 1;
if (patches[patch].mode) { // it's a percussive patch
voice = patches[patch].percVoice; // use fixed percussion voice
voices[voice].alloc = TRUE;
voices[voice].note = note;
voices[voice].channel = channel;
voices[voice].dwTimeStamp = MAKELONG(patch, 0);
SetVoiceTimbre(voice, &patches[patch]);
return voice;
}
bVoiceToUse = bVoiceSame = bVoiceOldest = 0xff;
// find a free melodic voice to use
for (i = 0; i < (BYTE)NUMMELODIC; i++) { // it's a melodic patch
if (!voices[i].alloc) {
if (voices[i].dwTimeStamp < dwOldestOff) {
bVoiceToUse = i;
dwOldestOff = voices[i].dwTimeStamp;
}
} else if (voices[i].channel == channel) {
if (voices[i].dwTimeStamp < dwOldestSame) {
dwOldestSame = voices[i].dwTimeStamp;
bVoiceSame = i;
}
} else if (voices[i].dwTimeStamp < dwOldestTime) {
dwOldestTime = voices[i].dwTimeStamp;
bVoiceOldest = i; // remember oldest one to steal
}
}
// choose a free voice if we have found one. If not, choose the
// oldest voice of the same channel. if none, choose the oldest voice.
if (bVoiceToUse == 0xff) {
if (bVoiceSame != 0xff) {
bVoiceToUse = bVoiceSame;
} else {
bVoiceToUse = bVoiceOldest;
}
}
if (voices[bVoiceToUse].alloc) { // if we stole it, turn it off
NoteOff(bVoiceToUse);
}
voices[bVoiceToUse].alloc = 1;
voices[bVoiceToUse].note = note;
voices[bVoiceToUse].channel = channel;
voices[bVoiceToUse].dwTimeStamp = dwAge++;
SetVoiceTimbre(bVoiceToUse, &patches[patch]);
return bVoiceToUse;
}
/* --- externally-called functions ------------------------------ */
/*
* Adlib_NoteOn - This turns a note on. (Including drums, with
* a patch # of the drum Note + 128)
*
* inputs
* BYTE bPatch - MIDI patch number
* BYTE bNote - MIDI note number
* BYTE bChannel - MIDI channel #
* BYTE bVelocity - Velocity #
* short iBend - current pitch bend from -32768, to 32767
* returns
* none
*/
VOID NEAR PASCAL Adlib_NoteOn (BYTE bPatch,
BYTE bNote, BYTE bChannel, BYTE bVelocity,
short iBend)
{
BYTE voice;
WORD wBend;
if (bVelocity == 0) { // 0 velocity means note off
Adlib_NoteOff(bPatch, bNote, bChannel);
return;
}
// octave registration for melodic patches
if (bPatch < 128) {
bNote += patchKeyOffset[bPatch];
if ((bNote < 0) || (bNote > 127)) {
bNote -= patchKeyOffset[bPatch];
}
}
if (bPatch >= 128) {
/*
* it's a percussion note
*/
bNote = bPatch - 128;
if ((bNote < FIRSTDRUMNOTE) || (bNote > LASTDRUMNOTE)) {
return;
}
/* use the drum patch table to map the note to a given
* TIMBRE/note pair
*/
bPatch = drumpatch[bNote - FIRSTDRUMNOTE].patch;
bNote = drumpatch[bNote - FIRSTDRUMNOTE].note;
/* each drum patch plays on one specific voice.
* find that voice
*/
voice = patches[bPatch].percVoice;
/* switch note off if playing */
if (voices[voice].alloc) {
NoteOff(voice);
}
/* call GetNewVoice to set the voice params and timestamp
* even if we found it.
*/
voice = GetNewVoice(bPatch, bNote, bChannel);
} else {
/* switch note off if it's playing */
if ( (voice = FindVoice(bNote, bChannel)) != 0xFF ) {
NoteOff(voice);
} else {
voice = GetNewVoice(bPatch, bNote, bChannel);
}
}
/* convert the velocity to an attenuation setting, and
* write that to the device
*/
SetVoiceAtten(voice, bChannel, bVelocity);
/*
* apply pitch bend. note that we are passed a pitch bend in the
* range 8000-7fff, but our code assumes 0-3fff, so we convert here.
*/
wBend = (((WORD)iBend + 0x8000) >> 2) & 0x3fff;
SetVoicePitch(voice, wBend);
// play the note
NoteOn(voice, bNote);
}
/* Adlib_NoteOff - This turns a note off. (Including drums,
* with a patch # of the drum note + 128)
*
* inputs
* BYTE bPatch - MIDI patch #
* BYTE bNote - MIDI note number
* BYTE bChannel - MIDI channel #
* returns
* none
*/
VOID FAR PASCAL Adlib_NoteOff (BYTE bPatch,
BYTE bNote, BYTE bChannel)
{
BYTE bVoice;
if (bPatch > 127) {
/* drum note. These all use a fixed voice */
if ((bNote < FIRSTDRUMNOTE) || (bNote > LASTDRUMNOTE)) {
return;
}
/* use the drum patch table to map the note to a given
* TIMBRE/note pair
*/
bPatch = drumpatch[bNote - FIRSTDRUMNOTE].patch;
bNote = drumpatch[bNote - FIRSTDRUMNOTE].note;
bVoice = patches[bPatch].percVoice;
/* switch note off if playing our patch */
if (LOWORD(voices[bVoice].dwTimeStamp) == bPatch) {
NoteOff(bVoice);
}
} else {
bVoice = FindVoice(bNote, bChannel);
if (bVoice != 0xff) {
if (voices[bVoice].note) {
NoteOff(bVoice);
FreeVoice(bVoice);
}
}
}
}
/* Adlib_AllNotesOff - turn off all notes
*
* inputs - none
* returns - none
*/
VOID Adlib_AllNotesOff(void)
{
BYTE i;
for (i = 0; i < NUMVOICES; i++) {
NoteOff(i);
}
}
/* Adlib_NewVolume - This should be called if a volume level
* has changed. This will adjust the levels of all the playing
* voices.
*
* inputs
* WORD wLeft - left attenuation (1.5 db units)
* WORD wRight - right attenuation (ignore if mono)
* returns
* none
*/
VOID FAR PASCAL Adlib_NewVolume (WORD wLeft, WORD wRight)
{
/* ignore the right attenuation since this is a mono device */
int iChange;
wLeft = min(wLeft, wRight) << 1;
iChange = wLeft - wSynthAtten;
wSynthAtten = wLeft;
/* change attenuation for all channels */
ChangeAtten(0xff, iChange);
}
/* Adlib_ChannelVolume - set the volume level for an individual channel.
*
* inputs
* BYTE bChannel - channel number to change
* WORD wAtten - attenuation in 1.5 db units
*
* returns
* none
*/
VOID FAR PASCAL Adlib_ChannelVolume(BYTE bChannel, WORD wAtten)
{
int iChange;
iChange = wAtten - gbChanAtten[bChannel];
gbChanAtten[bChannel] = (BYTE)wAtten;
/* change attenuation for this channel */
ChangeAtten(bChannel, iChange);
}
/* Adlib_SetPan - set the left-right pan position.
*
* inputs
* BYTE bChannel - channel number to alter
* BYTE bPan - 0 for left, 127 for right or somewhere in the middle.
*
* returns - none
*
* does nothing - this is a mono device
*/
VOID FAR PASCAL Adlib_SetPan(BYTE bChannel, BYTE bPan)
{
/* do nothing */
}
/* Adlib_PitchBend - This pitch bends a channel.
*
* inputs
* BYTE bChannel - channel
* short iBend - Values from -32768 to 32767, being
* -2 to +2 half steps
* returns
* none
*/
VOID NEAR PASCAL Adlib_PitchBend (BYTE bChannel, short iBend)
{
BYTE i;
WORD w;
/* note that our code expects 0 - 0x3fff not 0x8000 - 0x7fff */
w = (((WORD) iBend + 0x8000) >> 2) & 0x3fff;
for (i = 0; i < NUMVOICES; i++) {
if ((voices[i].alloc) && (voices[i].channel == bChannel)) {
SetVoicePitch(i, w);
}
}
}
/* Adlib_BoardInit - initialise board and load patches as necessary.
*
* inputs - none
* returns - 0 for success or the error code
*/
WORD Adlib_BoardInit(void)
{
BYTE i;
wSynthAtten = 0;
/* build the freq table */
InitFNums();
/* silence and free all voices */
for (i = 0; i <= 8; i++) {
SoundChut(i);
FreeVoice(i);
}
/* switch to percussive mode and set fixed frequencies */
SetFreq(TOM, TOM_PITCH, 0);
SetFreq(SD, SD_PITCH, 0);
percBits = 0;
SndSAmVibRhythm();
/* init all slots to sine-wave */
for (i= 0; i < 18; i++) {
SndOutput((BYTE)(0xE0 | offsetSlot[i]), 0);
}
/* enable wave-form selection */
SndOutput(1, 0x20);
LoadPatches();
LoadDrumPatches();
// don't initialise - the data is static and will thus
// be initialised to 0 at load time. no other change should be made
// since the mci sequencer will not re-send channel volume change
// messages.
//
// /* init all channels to loudest */
// for (i = 0; i < NUMCHANNELS; i++) {
// gbChanAtten[i] = 4;
// }
return(0);
}
/*
* Adlib_BoardReset - silence the board and set all voices off.
*/
VOID Adlib_BoardReset(void)
{
BYTE i;
/* silence and free all voices */
for (i = 0; i <= 8; i++) {
SoundChut(i);
FreeVoice(i);
}
/* switch to percussive mode and set fixed frequencies */
SetFreq(TOM, TOM_PITCH, 0);
SetFreq(SD, SD_PITCH, 0);
percBits = 0;
SndSAmVibRhythm();
}