cog/Frameworks/playptmod/playptmod/playptmod.c

3246 lines
90 KiB
C

/*
** PLAYPTMOD v1.35 - 7th of June 2017 - http://16-bits.org
** =========================================================
** This is the foobar2000 version, with added code by kode54
**
** Changelog from 1.31:
** - Make period variables unsigned, to simulate a period overflow quirk
** - Properly clamp periods<113 to 113 in PT mode
** - Removed a deprecated arpeggio volume quirk
** - Rewrote the mixer rate function to not use LUTs (WTF was I thinking?)
**
** Changelog from 1.3:
** - Fixed an unchecked memcpy when testing for ADPCM samples in possible STK files
**
** Changelog from 1.27:
** - Added support for FEST modules (.MOD with "FEST" tag instead of "M.K.")
** - Added the one-shot loop quirk for PT MODs
** - When setting mixSource, check loopLen>2 *or* loopStart>0 (yes, like PT)
** - Using 9xx on a >64kB sample should result in killing the voice (PT only)
**
** Changelog from 1.26:
** - Only loop module if speed is zero after fully processing an entire row
**
** Changelog from 1.25:
** - Invert Loop (EFx) was inaccurate
**
** Changelog from 1.24:
** - Sample swaps are now only handled for PT MODs
** - Handle sample swapping even during note delay (EDx)
** - "non-zero tick effects" should be handled in all ticks during pattern delay (EDx)
**
** Changelog from 1.23:
** - Handle CD81 .MOD/.OCT (fixes "MOD.checkered_subgliep" and other modules)
** - For PT MODs, instantly swap to new sample volume during a note delay (EDx)
** - Tweaked the volume scale value for the mixer, according to number of channels
** - Changed scaling of 4-bit panning (proper x*17 instead of x*16)
** - Code cleanup
**
** Changelog from 1.22:
** - Do sample swaps even if the new sample has a length of 0 (fixes "MOD.lsd ninja")
**
** Changelog from 1.21:
** - Only do portamentos if the voice has a period set (fixes first row of "MOD.cry of doom 3")
**
** Changelog from 1.20:
** - The EEx+Dxx quirk should also be handled for EEx+Bxx
**
** Changelog from 1.16:
** - The sample swap quirk had a bug (discovered by Wasp of PowerLine)
** - 3xx/5xy was still wrong in PT mode (fixes "MOD.lite teknalogy" by tEiS)
**
** Changelog from 1.15b:
** - Glissando (3xx/5xy) should not continue after its slide was done,
** but this only applies to MODs playing in ProTracker mode.
**
** Changelog from 1.15a:
** - Added a hack to find out what 8xx pan is used (7-bit/8-bit)
**
** Changelog from 1.10d:
** - Removed obsolete IFF sample handling (FT2/PT didn't have it)
** - Added two Ultimate Soundtracker sample loading hacks
** - The Invert Loop (EFx) table had a wrong value
** - Invert Loop (EFx) routines changed to be 100% accurate
** - Fixed a sample swap bug ("MOD.Ecstatic Guidance" by emax)
** - Proper zero'ing+init of channels on stop/play
** - Code cleanup (+ possible mixer speedup)
** - Much more that was changed long ago, without incrementing
** the version number. Let's do that from now on!
**
** Thanks to aciddose/adejr for the LED filter routines.
**
** Note: There's a lot of weird behavior in the code to
** "emulate" the weird stuff ProTracker on the Amiga does.
** If you see something fishy in the code, it's probably
** supposed to be like that. Please don't change it, you're
** literally asking for hours of hard debugging.
**
*/
#define _USE_MATH_DEFINES /* visual studio */
#include "playptmod.h"
#include "resampler.h"
#include <stdio.h>
#include <string.h> /* memcpy() */
#include <stdlib.h> /* malloc(), calloc(), free() */
#include <limits.h> /* INT_MAX, INT_MIN */
#include <math.h> /* floorf(), sinf() */
#define CLAMP(x, low, high) (((x) > (high)) ? (high) : (((x) < (low)) ? (low) : (x)))
#define DENORMAL_OFFSET 1e-10f
#define PT_MIN_PERIOD 78
#define PT_MAX_PERIOD 937
#define MAX_CHANNELS 32
#define MOD_ROWS 64
#define MOD_SAMPLES 31
#ifndef true
#define true 1
#define false 0
#endif
enum
{
FLAG_NOTE = 1,
FLAG_SAMPLE = 2,
FLAG_NEWSAMPLE = 4,
TEMPFLAG_START = 1,
TEMPFLAG_DELAY = 2,
TEMPFLAG_NEW_SAMPLE = 4
};
enum
{
soundBufferSize = 2048 * 4
};
typedef struct modnote
{
unsigned char sample;
unsigned char command;
unsigned char param;
unsigned short period;
} modnote_t;
typedef struct
{
unsigned char orderCount;
unsigned char patternCount;
unsigned char rowCount;
unsigned char restartPos;
unsigned char clippedRestartPos;
unsigned char order[128];
unsigned char pan[MAX_CHANNELS];
unsigned char ticks;
unsigned char format;
unsigned char channelCount;
short tempo;
short initBPM;
int moduleSize;
int totalSampleSize;
} MODULE_HEADER;
typedef struct
{
unsigned char fineTune;
unsigned char volume;
int loopStart;
int loopLength;
int length;
int reallength;
int tmpLoopStart;
int offset;
unsigned char attribute;
} MODULE_SAMPLE;
typedef struct
{
char patternLoopRow;
char patternLoopCounter;
signed char volume;
unsigned char sample;
unsigned char command;
unsigned char param;
unsigned char flags;
unsigned char tempFlags;
unsigned char tempFlagsBackup;
unsigned char fineTune;
unsigned char tremoloPos;
unsigned char vibratoPos;
unsigned char tremoloControl;
unsigned char tremoloSpeed;
unsigned char tremoloDepth;
unsigned char vibratoControl;
unsigned char vibratoSpeed;
unsigned char vibratoDepth;
unsigned char glissandoControl;
unsigned char glissandoSpeed;
unsigned char invertLoopDelay;
unsigned char invertLoopSpeed;
unsigned char chanIndex;
unsigned char toneportdirec;
unsigned short period;
unsigned short tempPeriod;
unsigned short wantedperiod;
int noNote;
int invertLoopOffset;
int offset;
int offsetTemp;
int offsetBugNotAdded;
signed char *invertLoopPtr;
signed char *invertLoopStart;
int invertLoopLength;
} mod_channel;
typedef struct
{
char moduleLoaded;
signed char *sampleData;
signed char *originalSampleData;
MODULE_HEADER head;
MODULE_SAMPLE samples[31];
modnote_t *patterns[256];
mod_channel channels[MAX_CHANNELS];
} MODULE;
typedef struct paula_filter_state
{
float LED[4];
} Filter;
typedef struct paula_filter_coefficients
{
float LED;
float LEDFb;
} FilterC;
typedef struct voice_data
{
const signed char *newData;
const signed char *data;
char swapSampleFlag;
char loopFlag;
int length;
int loopBegin;
int loopEnd;
char newLoopFlag;
int newLength;
int newLoopBegin;
int newLoopEnd;
int index;
int loopQuirk;
int vol;
int panL;
int panR;
int step;
int newStep;
float rate;
int interpolating;
int mute;
} Voice;
typedef struct
{
unsigned long length;
unsigned long remain;
const unsigned char *buf;
const unsigned char *t_buf;
} BUF;
typedef struct
{
int numChans;
char pattBreakBugPos;
char pattBreakFlag;
char pattDelayFlag;
char forceEffectsOff;
signed char tempVolume;
unsigned char modRow;
unsigned char modSpeed;
unsigned short modBPM;
unsigned char modTick;
unsigned char modPattern;
unsigned char modOrder;
unsigned char tempFlags;
unsigned char PBreakPosition;
unsigned char PattDelayTime;
unsigned char PattDelayTime2;
unsigned char PosJumpAssert;
unsigned char PBreakFlag;
unsigned short tempPeriod;
int tempoTimerVal;
char moduleLoaded;
char modulePlaying;
char useLEDFilter;
unsigned short soundBufferSize;
unsigned int soundFrequency;
char soundBuffers;
unsigned char *sinusTable;
int minPeriod;
int maxPeriod;
int loopCounter;
int sampleCounter;
int samplesPerTick;
int vBlankTiming;
int sevenBitPanning;
Voice v[MAX_CHANNELS];
Filter filter;
FilterC filterC;
float *mixBufferL;
float *mixBufferR;
void * blep[MAX_CHANNELS];
void * blepVol[MAX_CHANNELS];
unsigned int orderPlayed[256];
MODULE *source;
} player;
static const unsigned char invertLoopSpeeds[16] =
{
0x00, 0x05, 0x06, 0x07, 0x08, 0x0A, 0x0B, 0x0D,
0x10, 0x13, 0x16, 0x1A, 0x20, 0x2B, 0x40, 0x80
};
/* 16 fintunes, 3 octaves, 1 pad each octave, 14 paddings on end */
static const unsigned short rawAmigaPeriods[(16 * ((12 * 3) + 1)) + 14] =
{
856,808,762,720,678,640,604,570,538,508,480,453,
428,404,381,360,339,320,302,285,269,254,240,226,
214,202,190,180,170,160,151,143,135,127,120,113,0,
850,802,757,715,674,637,601,567,535,505,477,450,
425,401,379,357,337,318,300,284,268,253,239,225,
213,201,189,179,169,159,150,142,134,126,119,113,0,
844,796,752,709,670,632,597,563,532,502,474,447,
422,398,376,355,335,316,298,282,266,251,237,224,
211,199,188,177,167,158,149,141,133,125,118,112,0,
838,791,746,704,665,628,592,559,528,498,470,444,
419,395,373,352,332,314,296,280,264,249,235,222,
209,198,187,176,166,157,148,140,132,125,118,111,0,
832,785,741,699,660,623,588,555,524,495,467,441,
416,392,370,350,330,312,294,278,262,247,233,220,
208,196,185,175,165,156,147,139,131,124,117,110,0,
826,779,736,694,655,619,584,551,520,491,463,437,
413,390,368,347,328,309,292,276,260,245,232,219,
206,195,184,174,164,155,146,138,130,123,116,109,0,
820,774,730,689,651,614,580,547,516,487,460,434,
410,387,365,345,325,307,290,274,258,244,230,217,
205,193,183,172,163,154,145,137,129,122,115,109,0,
814,768,725,684,646,610,575,543,513,484,457,431,
407,384,363,342,323,305,288,272,256,242,228,216,
204,192,181,171,161,152,144,136,128,121,114,108,0,
907,856,808,762,720,678,640,604,570,538,508,480,
453,428,404,381,360,339,320,302,285,269,254,240,
226,214,202,190,180,170,160,151,143,135,127,120,0,
900,850,802,757,715,675,636,601,567,535,505,477,
450,425,401,379,357,337,318,300,284,268,253,238,
225,212,200,189,179,169,159,150,142,134,126,119,0,
894,844,796,752,709,670,632,597,563,532,502,474,
447,422,398,376,355,335,316,298,282,266,251,237,
223,211,199,188,177,167,158,149,141,133,125,118,0,
887,838,791,746,704,665,628,592,559,528,498,470,
444,419,395,373,352,332,314,296,280,264,249,235,
222,209,198,187,176,166,157,148,140,132,125,118,0,
881,832,785,741,699,660,623,588,555,524,494,467,
441,416,392,370,350,330,312,294,278,262,247,233,
220,208,196,185,175,165,156,147,139,131,123,117,0,
875,826,779,736,694,655,619,584,551,520,491,463,
437,413,390,368,347,328,309,292,276,260,245,232,
219,206,195,184,174,164,155,146,138,130,123,116,0,
868,820,774,730,689,651,614,580,547,516,487,460,
434,410,387,365,345,325,307,290,274,258,244,230,
217,205,193,183,172,163,154,145,137,129,122,115,0,
862,814,768,725,684,646,610,575,543,513,484,457,
431,407,384,363,342,323,305,288,272,256,242,228,
216,203,192,181,171,161,152,144,136,128,121,114,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0 /* 14 extra zeroes (padding) */
};
/* 16 fintunes, 7 octaves, 1 pad each octave, 14 paddings on end */
static unsigned short extendedRawPeriods[(16 * ((12 * 7) + 1)) + 14];
static const unsigned short npertab[12 * 7] = /* 7 octaves, C-3 .. B-9 */
{
0x06B0,0x0650,0x05F4,0x05A0,0x054C,0x0500,0x04B8,0x0474,0x0434,0x03F8,0x03C0,0x038A,
0x0358,0x0328,0x02FA,0x02D0,0x02A6,0x0280,0x025C,0x023A,0x021A,0x01FC,0x01E0,0x01C5,
0x01AC,0x0194,0x017D,0x0168,0x0153,0x0140,0x012E,0x011D,0x010D,0x00FE,0x00F0,0x00E2,
0x00D6,0x00CA,0x00BE,0x00B4,0x00AA,0x00A0,0x0097,0x008F,0x0087,0x007F,0x0078,0x0071,
0x006B,0x0065,0x005F,0x005A,0x0055,0x0050,0x004B,0x0047,0x0043,0x003F,0x003C,0x0038,
0x0035,0x0032,0x002F,0x002D,0x002A,0x0028,0x0025,0x0023,0x0021,0x001F,0x001E,0x001C,
0x001B,0x0019,0x0018,0x0016,0x0015,0x0014,0x0013,0x0012,0x0011,0x0010,0x000F,0x000E
};
static const short finetune[16] =
{
8363, 8413, 8463, 8529, 8581, 8651, 8723, 8757,
7895, 7941, 7985, 8046, 8107, 8169, 8232, 8280
};
static float calcRcCoeff(float sampleRate, float cutOffFreq)
{
if (cutOffFreq >= (sampleRate / 2.0f))
return (1.0f);
return (2.0f * (float)(M_PI) * cutOffFreq / sampleRate);
}
static BUF *bufopen(const unsigned char *bufToCopy, unsigned long bufferSize)
{
BUF *b;
b = (BUF *)malloc(sizeof (BUF));
b->t_buf = bufToCopy;
b->buf = bufToCopy;
b->length = bufferSize;
b->remain = bufferSize;
return (b);
}
static void bufclose(BUF *_SrcBuf)
{
if (_SrcBuf != NULL)
free(_SrcBuf);
}
static unsigned long buftell(BUF *_SrcBuf)
{
if (_SrcBuf->buf > _SrcBuf->t_buf)
return (_SrcBuf->buf - _SrcBuf->t_buf);
else
return (_SrcBuf->t_buf - _SrcBuf->buf);
}
static void bufread(void *_DstBuf, size_t _ElementSize, size_t _Count, BUF *_SrcBuf)
{
_Count *= _ElementSize;
if (_Count > _SrcBuf->remain)
_Count = _SrcBuf->remain;
_SrcBuf->remain -= _Count;
memcpy(_DstBuf, _SrcBuf->buf, _Count);
_SrcBuf->buf += _Count;
}
static void bufseek(BUF *_SrcBuf, long _Offset, int _Origin)
{
if (_SrcBuf->buf)
{
switch (_Origin)
{
case SEEK_SET: _SrcBuf->buf = _SrcBuf->t_buf + _Offset; break;
case SEEK_CUR: _SrcBuf->buf += _Offset; break;
case SEEK_END: _SrcBuf->buf = _SrcBuf->t_buf + _SrcBuf->length + _Offset; break;
default: break;
}
_Offset = _SrcBuf->buf - _SrcBuf->t_buf;
_SrcBuf->remain = (unsigned int)(_Offset) > _SrcBuf->length ? 0 : _SrcBuf->length - _Offset;
}
}
static inline int periodToNote(player *p, char finetune, unsigned short period)
{
unsigned char i;
unsigned char maxNotes;
short *tablePointer;
if (p->minPeriod == PT_MIN_PERIOD)
{
maxNotes = 12 * 3;
tablePointer = (unsigned short *)&rawAmigaPeriods[finetune * ((12 * 3) + 1)];
}
else
{
maxNotes = 12 * 7;
tablePointer = (unsigned short *)&extendedRawPeriods[finetune * ((12 * 7) + 1)];
}
for (i = 0; i < maxNotes; ++i)
{
if (period >= tablePointer[i])
break;
}
return (i);
}
static void mixerSwapChSource(player *p, int ch, const signed char *src, int length, int loopStart, int loopLength, int step)
{
Voice *v;
// If you swap sample in-realtime in PT
// without putting a period, the current
// sample will play through first (either
// >len or >loop_len), then when that is
// reached you change to the new sample you
// put earlier (if new sample is looped)
v = &p->v[ch];
v->loopQuirk = false;
v->swapSampleFlag = true;
v->newData = src;
v->newLoopFlag = (loopStart + loopLength) > (2 * step);
v->newLength = length;
v->newLoopBegin = loopStart;
v->newLoopEnd = loopStart + loopLength;
v->newStep = step;
v->interpolating = 1;
// if the mixer was already shut down earlier after a non-loop swap,
// force swap again, but only if the new sample has loop enabled (ONLY!)
if ((v->data == NULL) && v->newLoopFlag)
{
v->loopBegin = v->newLoopBegin;
v->loopEnd = v->newLoopEnd;
v->loopFlag = v->newLoopFlag;
v->data = v->newData;
v->length = v->newLength;
v->step = v->newStep;
/* we need to wrap here for safety reasons */
while (v->index >= v->loopEnd)
v->index = v->loopBegin + (v->index - v->loopEnd);
}
}
static void mixerSetChSource(player *p, int ch, const signed char *src, int length, int loopStart, int loopLength, int offset, int step)
{
Voice *v;
v = &p->v[ch];
v->loopQuirk = false;
v->swapSampleFlag = false;
v->data = src;
v->index = offset;
v->length = length;
v->loopFlag = (loopStart + loopLength) > (2 * step);
v->loopBegin = loopStart;
v->loopEnd = loopStart + loopLength;
v->step = step;
v->interpolating = 1;
resampler_clear(p->blep[ch]);
resampler_clear(p->blepVol[ch]);
/* Check external 9xx usage (Set Sample Offset) */
if (v->loopFlag)
{
if (offset >= v->loopEnd)
v->index = v->loopBegin;
}
else
{
if (offset >= v->length)
v->data = NULL;
}
/* PT specific quirks */
if (p->minPeriod == PT_MIN_PERIOD)
{
/* One-shot loops */
if ((loopLength > 2) && (loopStart == 0))
{
v->loopQuirk = v->loopEnd;
v->loopEnd = v->length;
}
/* 9xx on >64kB samples = kill voice */
if ((length > 65534) && (offset > 0))
{
v->loopQuirk = false;
v->data = NULL;
}
}
}
// adejr: these sin/cos approximations both use a 0..1
// parameter range and have 'normalized' (1/2 = 0db) coefficients
// The coefficients are for LERP(x, x * x, 0.224) * sqrt(2)
// max_error is minimized with 0.224 = 0.0013012886
static inline float sinApx(float x)
{
x = x * (2.0f - x);
return (x * 1.09742972f + x * x * 0.31678383f);
}
static inline float cosApx(float x)
{
x = (1.0f - x) * (1.0f + x);
return (x * 1.09742972f + x * x * 0.31678383f);
}
static void mixerSetChPan(player *p, int ch, int pan)
{
float newpan;
if (pan == 255) pan = 256; /* 100% R pan fix for 8-bit pan */
newpan = pan * (1.0f / 256.0f);
p->v[ch].panL = (int)(256.0f * cosApx(newpan));
p->v[ch].panR = (int)(256.0f * sinApx(newpan));
}
static void mixerSetChVol(player *p, int ch, int vol)
{
p->v[ch].vol = vol;
}
static void mixerCutChannels(player *p)
{
int i;
memset(p->v, 0, sizeof (p->v));
for (i = 0; i < MAX_CHANNELS; ++i)
{
resampler_clear(p->blep[i]);
resampler_clear(p->blepVol[i]);
}
memset(&p->filter, 0, sizeof (p->filter));
if (p->source)
{
for (i = 0; i < MAX_CHANNELS; ++i)
mixerSetChPan(p, i, p->source->head.pan[i]);
}
else
{
for (i = 0; i < MAX_CHANNELS; ++i)
mixerSetChPan(p, i, (i + 1) & 2 ? 160 : 96);
}
}
static void mixerSetChRate(player *p, int ch, unsigned short period)
{
if (period == 0)
{
p->v[ch].rate = 0.0f;
return;
}
if (p->minPeriod == PT_MIN_PERIOD)
{
/* in PT mode, wrap period around 113 (Paula behavior) */
if (period < 113)
period = 113;
}
p->v[ch].rate = (3546895.0f / period) / p->soundFrequency;
}
static void outputAudio(player *p, int *target, int numSamples)
{
signed short tempSample;
signed int i_smp;
int *out;
int step;
int tempVolume;
int delta;
int interpolating;
unsigned int i;
unsigned int j;
float L;
float R;
float t_vol;
float t_smp;
float downscale;
Voice *v;
void *bSmp;
void *bVol;
memset(p->mixBufferL, 0, numSamples * sizeof (float));
memset(p->mixBufferR, 0, numSamples * sizeof (float));
for (i = 0; i < p->source->head.channelCount; ++i)
{
j = 0;
v = &p->v[i];
bSmp = p->blep[i];
bVol = p->blepVol[i];
if (v->data && v->rate)
{
step = v->step;
interpolating = v->interpolating;
resampler_set_rate(bSmp, v->rate);
resampler_set_rate(bVol, v->rate);
for (j = 0; j < numSamples;)
{
tempVolume = (v->data && !v->mute ? v->vol : 0);
while (interpolating > 0 && (resampler_get_free_count(bSmp) ||
(!resampler_get_sample_count(bSmp) &&
!resampler_get_sample_count(bVol))))
{
tempSample = (v->data ? (step == 2 ? (v->data[v->index] + v->data[v->index + 1] * 0x100) : v->data[v->index] * 0x100) : 0);
resampler_write_sample_fixed(bSmp, tempSample, 1);
resampler_write_sample_fixed(bVol, tempVolume, 1);
if (v->data)
{
v->index += step;
if (v->loopFlag)
{
if (v->index >= v->loopEnd)
{
if (v->swapSampleFlag)
{
v->swapSampleFlag = false;
if (!v->newLoopFlag)
{
interpolating = -resampler_get_padding_size();
break;
}
v->loopBegin = v->newLoopBegin;
v->loopEnd = v->newLoopEnd;
v->loopFlag = v->newLoopFlag;
v->data = v->newData;
v->length = v->newLength;
v->step = v->newStep;
v->index = v->loopBegin;
}
else
{
v->index = v->loopBegin;
if (v->loopQuirk)
{
v->loopEnd = v->loopQuirk;
v->loopQuirk = false;
}
}
}
}
else if (v->index >= v->length)
{
if (v->swapSampleFlag)
{
v->swapSampleFlag = false;
if (!v->newLoopFlag)
{
interpolating = -resampler_get_padding_size();
break;
}
v->loopBegin = v->newLoopBegin;
v->loopEnd = v->newLoopEnd;
v->loopFlag = v->newLoopFlag;
v->data = v->newData;
v->length = v->newLength;
v->step = v->newStep;
v->index = v->loopBegin;
}
else
{
interpolating = -resampler_get_padding_size();
break;
}
}
}
}
while (interpolating < 0 && (resampler_get_free_count(bSmp) ||
(!resampler_get_sample_count(bSmp) &&
!resampler_get_sample_count(bVol))))
{
resampler_write_sample_fixed(bSmp, 0, 1);
resampler_write_sample_fixed(bVol, 0, 1);
++interpolating;
}
v->interpolating = interpolating;
while (j < numSamples && resampler_get_sample_count(bSmp))
{
t_vol = resampler_get_sample_float(bVol);
t_smp = resampler_get_sample_float(bSmp);
resampler_remove_sample(bVol, 0);
resampler_remove_sample(bSmp, 1);
t_smp *= t_vol;
i_smp = (signed int)t_smp;
p->mixBufferL[j] += i_smp * v->panL;
p->mixBufferR[j] += i_smp * v->panR;
j++;
}
if (!interpolating && !resampler_get_sample_count(bSmp))
{
v->data = NULL;
break;
}
}
}
}
if (p->numChans <= 4) downscale = 1.0f / 194.0f;
else if (p->numChans <= 16) downscale = 1.0f / 234.0f;
else if (p->numChans <= 32) downscale = 1.0f / 300.0f;
out = target;
for (i = 0; i < numSamples; ++i)
{
L = p->mixBufferL[i];
R = p->mixBufferR[i];
if (p->useLEDFilter)
{
p->filter.LED[0] += (p->filterC.LED * (L - p->filter.LED[0])
+ p->filterC.LEDFb * (p->filter.LED[0] - p->filter.LED[1]) + DENORMAL_OFFSET);
p->filter.LED[1] += (p->filterC.LED * (p->filter.LED[0] - p->filter.LED[1]) + DENORMAL_OFFSET);
p->filter.LED[2] += (p->filterC.LED * (R - p->filter.LED[2])
+ p->filterC.LEDFb * (p->filter.LED[2] - p->filter.LED[3]) + DENORMAL_OFFSET);
p->filter.LED[3] += (p->filterC.LED * (p->filter.LED[2] - p->filter.LED[3]) + DENORMAL_OFFSET);
L = p->filter.LED[1];
R = p->filter.LED[3];
}
L *= downscale;
R *= downscale;
L = CLAMP(L, INT_MIN, INT_MAX);
R = CLAMP(R, INT_MIN, INT_MAX);
if (out)
{
*out++ = (signed int)(L);
*out++ = (signed int)(R);
}
}
}
static unsigned short bufGetWordBigEndian(BUF *in)
{
unsigned char bytes[2];
bufread(bytes, 1, 2, in);
return ((bytes[0] << 8) | bytes[1]);
}
static unsigned short bufGetWordLittleEndian(BUF *in)
{
unsigned char bytes[2];
bufread(bytes, 1, 2, in);
return ((bytes[1] << 8) | bytes[0]);
}
static unsigned int bufGetDwordLittleEndian(BUF *in)
{
unsigned char bytes[4];
bufread(bytes, 1, 4, in);
return ((bytes[3] << 24) | (bytes[2] << 16) | (bytes[1] << 8) | bytes[0]);
}
static int playptmod_LoadMTM(player *p, BUF *fmodule)
{
int i, j, k;
unsigned int trackCount, commentLength;
unsigned char sampleCount;
unsigned long tracksOffset, sequencesOffset, commentOffset;
unsigned int totalSampleSize = 0, sampleOffset = 0;
modnote_t *note = NULL;
bufseek(fmodule, 24, SEEK_SET);
trackCount = bufGetWordLittleEndian(fmodule);
bufread(&p->source->head.patternCount, 1, 1, fmodule); p->source->head.patternCount++;
bufread(&p->source->head.orderCount, 1, 1, fmodule); p->source->head.orderCount++;
commentLength = bufGetWordLittleEndian(fmodule);
bufread(&sampleCount, 1, 1, fmodule);
bufseek(fmodule, 1, SEEK_CUR);
bufread(&p->source->head.rowCount, 1, 1, fmodule);
bufread(&p->source->head.channelCount, 1, 1, fmodule);
if (!trackCount || !sampleCount || !p->source->head.rowCount || p->source->head.rowCount > 64 || !p->source->head.channelCount || p->source->head.channelCount > 32)
return (false);
bufread(&p->source->head.pan, 1, 32, fmodule);
for (i = 0; i < 32; ++i)
{
if (p->source->head.pan[i] <= 15)
p->source->head.pan[i] <<= 4;
else
p->source->head.pan[i] = 128;
}
for (i = 0; i < sampleCount; ++i)
{
MODULE_SAMPLE * s = &p->source->samples[i];
bufseek(fmodule, 22, SEEK_CUR);
s->length = bufGetDwordLittleEndian(fmodule);
s->loopStart = bufGetDwordLittleEndian(fmodule);
s->loopLength = bufGetDwordLittleEndian(fmodule) - s->loopStart;
if (s->loopLength < 2)
s->loopLength = 2;
bufread(&s->fineTune, 1, 1, fmodule);
s->fineTune = s->fineTune & 0x0F;
bufread(&s->volume, 1, 1, fmodule);
bufread(&s->attribute, 1, 1, fmodule);
totalSampleSize += s->length;
}
bufread(&p->source->head.order, 1, 128, fmodule);
tracksOffset = fmodule->length - fmodule->remain;
sequencesOffset = tracksOffset + 192 * trackCount;
commentOffset = sequencesOffset + 64 * p->source->head.patternCount;
for (i = 0; i < p->source->head.patternCount; ++i)
{
note = p->source->patterns[i] = (modnote_t *)calloc(1, sizeof (modnote_t) * p->source->head.rowCount * p->source->head.channelCount);
if (!note)
{
for (j = 0; j < i; ++j)
{
if (p->source->patterns[j])
{
free(p->source->patterns[j]);
p->source->patterns[j] = NULL;
}
}
return 0;
}
for (j = 0; j < p->source->head.channelCount; ++j)
{
int trackNumber;
bufseek(fmodule, sequencesOffset + 64 * i + 2 * j, SEEK_SET);
trackNumber = bufGetWordLittleEndian(fmodule);
if (trackNumber--)
{
bufseek(fmodule, tracksOffset + 192 * trackNumber, SEEK_SET);
for (k = 0; k < p->source->head.rowCount; ++k)
{
unsigned char buf[3];
bufread(buf, 1, 3, fmodule);
if (buf[0] || buf[1] || buf[2])
{
note[k * p->source->head.channelCount + j].period = (buf[0] / 4) ? extendedRawPeriods[buf[0] / 4] : 0;
note[k * p->source->head.channelCount + j].sample = ((buf[0] << 4) | (buf[1] >> 4)) & 0x3F;
note[k * p->source->head.channelCount + j].command = buf[1] & 0x0F;
note[k * p->source->head.channelCount + j].param = buf[2];
if (note[k * p->source->head.channelCount + j].command == 0x0F && note[k * p->source->head.channelCount + j].param == 0x00)
note[k * p->source->head.channelCount + j].command = 0;
}
}
}
}
}
p->source->sampleData = (signed char *)malloc(totalSampleSize);
if (!p->source->sampleData)
{
for (i = 0; i < 256; ++i)
{
if (p->source->patterns[i] != NULL)
{
free(p->source->patterns[i]);
p->source->patterns[i] = NULL;
}
}
return (false);
}
bufseek(fmodule, commentOffset + commentLength, SEEK_SET);
for (i = 0; i < sampleCount; ++i)
{
MODULE_SAMPLE * s = &p->source->samples[i];
s->offset = sampleOffset;
bufread(&p->source->sampleData[sampleOffset], 1, s->length, fmodule);
if (!(s->attribute & 1))
{
for (j = (int)sampleOffset; (unsigned int)j < sampleOffset + s->length; ++j)
p->source->sampleData[(unsigned int)j] ^= 0x80;
}
sampleOffset += s->length;
}
p->source->originalSampleData = (signed char *)malloc(totalSampleSize);
if (p->source->originalSampleData == NULL)
{
free(p->source->sampleData);
p->source->sampleData = NULL;
for (i = 0; i < 256; ++i)
{
if (p->source->patterns[i] != NULL)
{
free(p->source->patterns[i]);
p->source->patterns[i] = NULL;
}
}
return (false);
}
memcpy(p->source->originalSampleData, p->source->sampleData, totalSampleSize);
p->source->head.totalSampleSize = totalSampleSize;
p->useLEDFilter = false;
p->moduleLoaded = true;
p->minPeriod = 14;
p->maxPeriod = 1712;
p->source->head.format = FORMAT_MTM;
p->source->head.initBPM = 125;
p->sevenBitPanning = false;
return (true);
}
static void checkModType(MODULE_HEADER *h, player *p, const char *buf)
{
if (!strncmp(buf, "M.K.", 4))
{
h->format = FORMAT_MK; /* ProTracker v1.x */
p->numChans = h->channelCount = 4;
/* Normal period range */
p->minPeriod = PT_MIN_PERIOD;
p->maxPeriod = PT_MAX_PERIOD;
return;
}
else if (!strncmp(buf, "M!K!", 4))
{
h->format = FORMAT_MK2; /* ProTracker v2.x (if >64 patterns) */
p->numChans = h->channelCount = 4;
/* Normal period range */
p->minPeriod = PT_MIN_PERIOD;
p->maxPeriod = PT_MAX_PERIOD;
return;
}
else if (!strncmp(buf, "FLT4", 4))
{
h->format = FORMAT_FLT4; /* StarTrekker (4 channel MODs only) */
p->numChans = h->channelCount = 4;
/* Normal period range */
p->minPeriod = PT_MIN_PERIOD;
p->maxPeriod = PT_MAX_PERIOD;
return;
}
else if (!strncmp(buf, "FLT8", 4))
{
h->format = FORMAT_FLT8;
p->numChans = h->channelCount = 8;
/* Normal period range */
p->minPeriod = PT_MIN_PERIOD;
p->maxPeriod = PT_MAX_PERIOD;
return;
}
else if (!strncmp(buf, "CD81", 4))
{
h->format = FORMAT_NNCH; /* Octalyzer (Atari STE/Falcon) */
p->numChans = h->channelCount = 8;
/* Normal period range (yes, CD81 has no extended periods) */
p->minPeriod = PT_MIN_PERIOD;
p->maxPeriod = PT_MAX_PERIOD;
return;
}
else if (!strncmp(buf + 1, "CHN", 3) && buf[0] >= '1' && buf[0] <= '9')
{
h->format = FORMAT_NCHN; /* FastTracker II (1-9 channel MODs) */
p->numChans = h->channelCount = buf[0] - '0';
/* Extended period range */
p->minPeriod = 14;
p->maxPeriod = 1712;
return;
}
else if (!strncmp(buf + 2, "CH", 2) && buf[0] >= '1' && buf[0] <= '3' && buf[1] >= '0' && buf[1] <= '9')
{
h->format = FORMAT_NNCH; /* FastTracker II (10-32 channel MODs) */
p->numChans = h->channelCount = (buf[0] - '0') * 10 + (buf[1] - '0');
if (h->channelCount > 32)
{
h->format = FORMAT_UNKNOWN;
h->channelCount = 4;
}
/* Extended period range */
p->minPeriod = 14;
p->maxPeriod = 1712;
return;
}
else if (!strncmp(buf, "16CN", 4))
{
h->format = FORMAT_16CN;
p->numChans = h->channelCount = 16;
/* Extended period range */
p->minPeriod = 14;
p->maxPeriod = 1712;
return;
}
else if (!strncmp(buf, "32CN", 4))
{
h->format = FORMAT_32CN;
p->numChans = h->channelCount = 32;
/* Extended period range */
p->minPeriod = 14;
p->maxPeriod = 1712;
return;
}
else if (!strncmp(buf, "N.T.", 4))
{
h->format = FORMAT_MK; /* NoiseTracker 1.0, almost same as ProTracker v1.x (?) */
p->numChans = h->channelCount = 4;
/* Normal period range */
p->minPeriod = PT_MIN_PERIOD;
p->maxPeriod = PT_MAX_PERIOD;
return;
}
else if (!strncmp(buf, "FEST", 4))
{
h->format = FORMAT_FEST; /* NoiseTracker 1.0, special ones (from music disk?) */
p->numChans = h->channelCount = 4;
/* Normal period range */
p->minPeriod = PT_MIN_PERIOD;
p->maxPeriod = PT_MAX_PERIOD;
return;
}
h->format = FORMAT_UNKNOWN; /* May be The Ultimate SoundTracker, 15 samples */
p->numChans = h->channelCount = 4;
/* Normal period range */
p->minPeriod = PT_MIN_PERIOD;
p->maxPeriod = PT_MAX_PERIOD;
}
int playptmod_LoadMem(void *_p, const unsigned char *buf, unsigned long bufLength)
{
player *p = (player *)_p;
unsigned char bytes[4];
char modSig[4];
char *smpDat8;
char tempSample[131070];
int i;
int j;
int pattern;
int row;
int channel;
int sampleOffset;
int mightBeSTK;
int lateVerSTKFlag;
int numSamples;
int tmp;
int leftPanning;
int extendedPanning;
unsigned long tempOffset;
modnote_t *note;
MODULE_SAMPLE *s;
BUF *fmodule;
sampleOffset = 0;
lateVerSTKFlag = false;
mightBeSTK = false;
p->source = (MODULE *)calloc(1, sizeof (MODULE));
if (p->source == NULL)
return (false);
fmodule = bufopen(buf, bufLength);
if (fmodule == NULL)
{
free(p->source);
return (false);
}
if (bufLength <= 1624)
{
free(p->source);
bufclose(fmodule);
return (false);
}
bufread(modSig, 1, 3, fmodule);
if (!strncmp(modSig, "MTM", 3))
{
i = playptmod_LoadMTM(p, fmodule);
bufclose(fmodule);
return (i);
}
bufseek(fmodule, 0x0438, SEEK_SET);
bufread(modSig, 1, 4, fmodule);
checkModType(&p->source->head, p, modSig);
if (p->source->head.format == FORMAT_UNKNOWN)
mightBeSTK = true;
bufseek(fmodule, 20, SEEK_SET);
for (i = 0; i < MOD_SAMPLES; ++i)
{
s = &p->source->samples[i];
if (mightBeSTK && (i > 14))
{
s->loopLength = 2;
}
else
{
bufseek(fmodule, 22, SEEK_CUR);
s->length = bufGetWordBigEndian(fmodule) * 2;
if (s->length > 9999)
lateVerSTKFlag = true; /* Only used if mightBeSTK is set */
tmp = 0;
bufread(&tmp, 1, 1, fmodule); /* xxx: is this big-endian safe? */
if (p->source->head.format == FORMAT_FEST)
s->fineTune = (-tmp & 0x1F) / 2; /* One more bit of precision, + inverted */
else
s->fineTune = tmp & 0x0F;
bufread(&s->volume, 1, 1, fmodule);
if (s->volume > 64)
s->volume = 64;
if (mightBeSTK)
s->loopStart = bufGetWordBigEndian(fmodule);
else
s->loopStart = bufGetWordBigEndian(fmodule) * 2;
s->loopLength = bufGetWordBigEndian(fmodule) * 2;
if (s->loopLength < 2)
s->loopLength = 2;
/* fix for poorly converted STK->PTMOD modules */
if (!mightBeSTK && ((s->loopStart + s->loopLength) > s->length))
{
if (((s->loopStart / 2) + s->loopLength) <= s->length)
{
s->loopStart /= 2;
}
else
{
s->loopStart = 0;
s->loopLength = 2;
}
}
if (mightBeSTK)
{
if (s->loopLength > 2)
{
tmp = s->loopStart;
s->length -= s->loopStart;
s->loopStart = 0;
s->tmpLoopStart = tmp;
}
/* No finetune in STK/UST */
s->fineTune = 0;
}
s->attribute = 0;
}
}
/* STK 2.5 had loopStart in words, not bytes. Convert if late version STK */
for (i = 0; i < 15; ++i)
{
if (mightBeSTK && lateVerSTKFlag)
{
s = &p->source->samples[i];
if (s->loopStart > 2)
{
s->length -= s->tmpLoopStart;
s->tmpLoopStart *= 2;
}
}
}
bufread(&p->source->head.orderCount, 1, 1, fmodule);
if (p->source->head.orderCount == 0)
{
free(p->source);
bufclose(fmodule);
return (false);
}
/* this fixes MOD.beatwave and others */
if (p->source->head.orderCount > 128)
p->source->head.orderCount = 128;
bufread(&p->source->head.restartPos, 1, 1, fmodule);
if (mightBeSTK && ((p->source->head.restartPos == 0) || (p->source->head.restartPos > 220)))
{
free(p->source);
bufclose(fmodule);
return (false);
}
p->source->head.initBPM = 125;
if (mightBeSTK)
{
p->source->head.format = FORMAT_STK;
p->source->head.clippedRestartPos = 0;
if (p->source->head.restartPos == 120)
{
p->source->head.restartPos = 125;
}
else
{
if (p->source->head.restartPos > 239)
p->source->head.restartPos = 239;
p->source->head.initBPM = (short)(1773447 / ((240 - p->source->head.restartPos) * 122));
}
}
else if (p->source->head.restartPos >= p->source->head.orderCount)
{
p->source->head.clippedRestartPos = 0;
}
else
{
p->source->head.clippedRestartPos = p->source->head.restartPos;
}
for (i = 0; i < 128; ++i)
{
bufread(&p->source->head.order[i], 1, 1, fmodule);
if (p->source->head.order[i] > p->source->head.patternCount)
p->source->head.patternCount = p->source->head.order[i];
}
p->source->head.patternCount++;
if (p->source->head.format != FORMAT_STK)
bufseek(fmodule, 4, SEEK_CUR);
for (pattern = 0; pattern < p->source->head.patternCount; ++pattern)
{
p->source->patterns[pattern] = (modnote_t *)calloc(64 * p->source->head.channelCount, sizeof (modnote_t));
if (p->source->patterns[pattern] == NULL)
{
for (i = 0; i < pattern; ++i)
{
if (p->source->patterns[i] != NULL)
{
free(p->source->patterns[i]);
p->source->patterns[i] = NULL;
}
}
bufclose(fmodule);
free(p->source);
return (false);
}
}
leftPanning = false;
extendedPanning = false;
for (pattern = 0; pattern < p->source->head.patternCount; ++pattern)
{
note = p->source->patterns[pattern];
if (p->source->head.format == FORMAT_FLT8)
{
for (row = 0; row < 64; ++row)
{
for (channel = 0; channel < 8; ++channel)
{
unsigned char bytes[4];
if (channel == 0 && row > 0) bufseek(fmodule, -1024, SEEK_CUR);
else if (channel == 4) bufseek(fmodule, 1024 - 4 * 4, SEEK_CUR);
bufread(bytes, 1, 4, fmodule);
note->period = ((bytes[0] & 0x0F) << 8) | bytes[1];
if (note->period != 0) /* FLT8 is 113..856 only */
note->period = CLAMP(note->period, 113, 856);
note->sample = (bytes[0] & 0xF0) | (bytes[2] >> 4);
note->command = bytes[2] & 0x0F;
note->param = bytes[3];
note++;
}
}
}
else
{
for (row = 0; row < 64; ++row)
{
for (channel = 0; channel < p->source->head.channelCount; ++channel)
{
bufread(bytes, 1, 4, fmodule);
note->period = ((bytes[0] & 0x0F) << 8) | bytes[1];
if (note->period != 0)
{
if ((unsigned)(note->period - 113) > (856 - 113))
{
p->minPeriod = 14;
p->maxPeriod = 1712;
}
note->period = CLAMP(note->period, p->minPeriod, p->maxPeriod);
}
note->sample = (bytes[0] & 0xF0) | (bytes[2] >> 4);
note->command = bytes[2] & 0x0F;
note->param = bytes[3];
/* 7-bit/8-bit styled pan hack, from OpenMPT */
if (note->command == 0x08)
{
if (note->param < 0x80)
leftPanning = true;
// Saga_Musix says: 8F instead of 80 is not a typo but
// required for a few mods which have 7-bit panning
// with slightly too big values
if ((note->param > 0x8F) && (note->param != 0xA4))
extendedPanning = true;
}
if (p->source->head.format == FORMAT_FEST)
{
/* Any Dxx == D00 in FEST modules */
if (note->command == 0x0D)
note->param = 0x00;
}
else if (mightBeSTK)
{
// Convert STK effects to PT effects
// TODO: Add tracker checking, as there
// are many more trackers with different
// effect/param designs :(
if (!lateVerSTKFlag)
{
/* Arpeggio */
if (note->command == 0x01)
{
note->command = 0x00;
}
/* Pitch slide */
else if (note->command == 0x02)
{
if (note->param & 0xF0)
{
note->command = 0x02;
note->param >>= 4;
}
else if (note->param & 0x0F)
{
note->command = 0x01;
}
}
}
/* Volume slide/pattern break */
if (note->command == 0x0D)
{
if (note->param == 0)
note->command = 0x0D;
else
note->command = 0x0A;
}
}
note++;
}
}
}
}
p->sevenBitPanning = false;
if (leftPanning && !extendedPanning)
p->sevenBitPanning = true;
tempOffset = buftell(fmodule);
sampleOffset = 0;
numSamples = (p->source->head.format == FORMAT_STK) ? 15 : 31;
for (i = 0; i < numSamples; ++i)
{
s = &p->source->samples[i];
s->offset = sampleOffset;
j = (s->length + 1) / 2 + 5 + 16;
if (j > s->length)
j = s->length;
if (j < 0 || j > 131072)
{
bufclose(fmodule);
for (pattern = 0; pattern < 256; ++i)
{
if (p->source->patterns[pattern] != NULL)
{
free(p->source->patterns[pattern]);
p->source->patterns[pattern] = NULL;
}
}
free(p->source);
return (false);
}
bufread(tempSample, 1, j, fmodule);
smpDat8 = tempSample;
if (j > 5 + 16 && memcmp(smpDat8, "ADPCM", 5) == 0)
s->reallength = j;
else
s->reallength = s->length;
sampleOffset += s->length;
p->source->head.totalSampleSize += s->length;
}
p->source->sampleData = (signed char *)malloc(p->source->head.totalSampleSize);
if (p->source->sampleData == NULL)
{
bufclose(fmodule);
for (pattern = 0; pattern < 256; ++i)
{
if (p->source->patterns[pattern] != NULL)
{
free(p->source->patterns[pattern]);
p->source->patterns[pattern] = NULL;
}
}
free(p->source);
return (false);
}
bufseek(fmodule, tempOffset, SEEK_SET);
numSamples = (p->source->head.format == FORMAT_STK) ? 15 : 31;
for (i = 0; i < numSamples; ++i)
{
s = &p->source->samples[i];
if (s->reallength < s->length)
{
const signed char *compressionTable = (const signed char *)tempSample + 5;
const unsigned char *adpcmData = (const unsigned char *)tempSample + 5 + 16;
int delta = 0;
bufread(tempSample, 1, s->reallength, fmodule);
for ( j = 0; j < s->length; ++j )
{
delta += compressionTable[(*adpcmData) & 0x0F];
p->source->sampleData[s->offset + j] = delta;
if (++j >= s->length)
break;
delta += compressionTable[(*adpcmData) >> 4];
p->source->sampleData[s->offset + j] = delta;
++adpcmData;
}
}
else if (mightBeSTK && (s->loopLength > 2))
{
for (j = 0; j < s->tmpLoopStart; ++j)
bufseek(fmodule, 1, SEEK_CUR);
bufread(&p->source->sampleData[s->offset], 1, s->length - s->loopStart, fmodule);
}
else
{
bufread(&p->source->sampleData[s->offset], 1, s->length, fmodule);
}
}
p->source->originalSampleData = (signed char *)malloc(p->source->head.totalSampleSize);
if (p->source->originalSampleData == NULL)
{
bufclose(fmodule);
free(p->source->sampleData);
for (pattern = 0; pattern < 256; ++i)
{
if (p->source->patterns[pattern] != NULL)
{
free(p->source->patterns[pattern]);
p->source->patterns[pattern] = NULL;
}
}
free(p->source);
return (false);
}
memcpy(p->source->originalSampleData, p->source->sampleData, p->source->head.totalSampleSize);
bufclose(fmodule);
p->source->head.rowCount = MOD_ROWS;
for (i = 0; i < MAX_CHANNELS; ++i)
p->source->head.pan[i] = ((i + 1) & 2) ? 160 : 96;
p->useLEDFilter = false;
p->moduleLoaded = true;
return (true);
}
int playptmod_Load(void *_p, const char *filename)
{
int i;
unsigned char *buffer;
unsigned long fileSize;
FILE *fileModule;
player *p = (player *)_p;
if (!p->moduleLoaded)
{
fileModule = fopen(filename, "rb");
if (fileModule == NULL)
return (false);
fseek(fileModule, 0, SEEK_END);
fileSize = ftell(fileModule);
fseek(fileModule, 0, SEEK_SET);
buffer = (unsigned char *)malloc(fileSize);
if (buffer == NULL)
{
fclose(fileModule);
return (false);
}
fread(buffer, 1, fileSize, fileModule);
fclose(fileModule);
i = playptmod_LoadMem(_p, buffer, fileSize);
free(buffer);
return (i);
}
return (false);
}
int playptmod_GetFormat(void *p)
{
return ((player *)p)->source->head.format;
}
static void fxArpeggio(player *p, mod_channel *ch);
static void fxPortamentoSlideUp(player *p, mod_channel *ch);
static void fxPortamentoSlideDown(player *p, mod_channel *ch);
static void fxGlissando(player *p, mod_channel *ch);
static void fxVibrato(player *p, mod_channel *ch);
static void fxGlissandoVolumeSlide(player *p, mod_channel *ch);
static void fxVibratoVolumeSlide(player *p, mod_channel *ch);
static void fxTremolo(player *p, mod_channel *ch);
static void fxNotInUse(player *p, mod_channel *ch);
static void fxSampleOffset(player *p, mod_channel *ch);
static void fxSampleOffset_FT2(player *p, mod_channel *ch);
static void fxVolumeSlide(player *p, mod_channel *ch);
static void fxPositionJump(player *p, mod_channel *ch);
static void fxSetVolume(player *p, mod_channel *ch);
static void fxPatternBreak(player *p, mod_channel *ch);
static void fxExtended(player *p, mod_channel *ch);
static void fxSetTempo(player *p, mod_channel *ch);
static void efxSetLEDFilter(player *p, mod_channel *ch);
static void efxFinePortamentoSlideUp(player *p, mod_channel *ch);
static void efxFinePortamentoSlideDown(player *p, mod_channel *ch);
static void efxGlissandoControl(player *p, mod_channel *ch);
static void efxVibratoControl(player *p, mod_channel *ch);
static void efxSetFineTune(player *p, mod_channel *ch);
static void efxPatternLoop(player *p, mod_channel *ch);
static void efxTremoloControl(player *p, mod_channel *ch);
static void efxKarplusStrong(player *p, mod_channel *ch);
static void efxRetrigNote(player *p, mod_channel *ch);
static void efxFineVolumeSlideUp(player *p, mod_channel *ch);
static void efxFineVolumeSlideDown(player *p, mod_channel *ch);
static void efxNoteCut(player *p, mod_channel *ch);
static void efxNoteDelay(player *p, mod_channel *ch);
static void efxPatternDelay(player *p, mod_channel *ch);
static void efxInvertLoop(player *p, mod_channel *ch);
static void fxExtended_FT2(player *p, mod_channel *ch);
static void fxPan(player *p, mod_channel *ch);
static void efxPan(player *p, mod_channel *ch);
typedef void (*effect_routine)(player *p, mod_channel *);
static effect_routine fxRoutines[16] =
{
fxArpeggio,
fxPortamentoSlideUp,
fxPortamentoSlideDown,
fxGlissando,
fxVibrato,
fxGlissandoVolumeSlide,
fxVibratoVolumeSlide,
fxTremolo,
fxNotInUse,
fxSampleOffset,
fxVolumeSlide,
fxPositionJump,
fxSetVolume,
fxPatternBreak,
fxExtended,
fxSetTempo
};
static effect_routine fxRoutines_FT2[16] =
{
fxArpeggio,
fxPortamentoSlideUp,
fxPortamentoSlideDown,
fxGlissando,
fxVibrato,
fxGlissandoVolumeSlide,
fxVibratoVolumeSlide,
fxTremolo,
fxPan,
fxSampleOffset_FT2,
fxVolumeSlide,
fxPositionJump,
fxSetVolume,
fxPatternBreak,
fxExtended_FT2,
fxSetTempo
};
static effect_routine efxRoutines[16] =
{
efxSetLEDFilter,
efxFinePortamentoSlideUp,
efxFinePortamentoSlideDown,
efxGlissandoControl,
efxVibratoControl,
efxSetFineTune,
efxPatternLoop,
efxTremoloControl,
efxKarplusStrong,
efxRetrigNote,
efxFineVolumeSlideUp,
efxFineVolumeSlideDown,
efxNoteCut,
efxNoteDelay,
efxPatternDelay,
efxInvertLoop
};
static effect_routine efxRoutines_FT2[16] =
{
fxNotInUse,
efxFinePortamentoSlideUp,
efxFinePortamentoSlideDown,
efxGlissandoControl,
efxVibratoControl,
efxSetFineTune,
efxPatternLoop,
efxTremoloControl,
efxPan,
efxRetrigNote,
efxFineVolumeSlideUp,
efxFineVolumeSlideDown,
efxNoteCut,
efxNoteDelay,
efxPatternDelay,
fxNotInUse
};
static void processInvertLoop(player *p, mod_channel *ch)
{
if (ch->invertLoopSpeed > 0)
{
ch->invertLoopDelay += invertLoopSpeeds[ch->invertLoopSpeed];
if (ch->invertLoopDelay >= 128)
{
ch->invertLoopDelay = 0;
if (ch->invertLoopPtr != NULL) /* SAFETY BUG FIX */
{
if (++ch->invertLoopPtr >= (ch->invertLoopStart + ch->invertLoopLength))
ch->invertLoopPtr = ch->invertLoopStart;
*ch->invertLoopPtr = -1 - *ch->invertLoopPtr;
}
}
}
}
static void efxSetLEDFilter(player *p, mod_channel *ch)
{
if (p->modTick == 0)
p->useLEDFilter = !(ch->param & 1);
}
static void efxFinePortamentoSlideUp(player *p, mod_channel *ch)
{
if (p->modTick == 0)
{
if (p->tempPeriod > 0)
{
ch->period -= (ch->param & 0x0F);
if (p->minPeriod == PT_MIN_PERIOD)
{
if ((ch->period & 0x0FFF) < 113)
{
ch->period &= 0xF000;
ch->period |= 113;
}
p->tempPeriod = ch->period & 0x0FFF;
}
else
{
if (ch->period < p->minPeriod)
ch->period = p->minPeriod;
p->tempPeriod = ch->period;
}
}
}
}
static void efxFinePortamentoSlideDown(player *p, mod_channel *ch)
{
if (p->modTick == 0)
{
if (p->tempPeriod > 0)
{
ch->period += (ch->param & 0x0F);
if (p->minPeriod == PT_MIN_PERIOD)
{
if ((ch->period & 0x0FFF) > 856)
{
ch->period &= 0xF000;
ch->period |= 856;
}
p->tempPeriod = ch->period & 0x0FFF;
}
else
{
if (ch->period > p->maxPeriod)
ch->period = p->maxPeriod;
p->tempPeriod = ch->period;
}
}
}
}
static void efxGlissandoControl(player *p, mod_channel *ch)
{
if (p->modTick == 0)
ch->glissandoControl = ch->param & 0x0F;
}
static void efxVibratoControl(player *p, mod_channel *ch)
{
if (p->modTick == 0)
ch->vibratoControl = ch->param & 0x0F;
}
static void efxSetFineTune(player *p, mod_channel *ch)
{
if (p->modTick == 0)
ch->fineTune = ch->param & 0x0F;
}
static void efxPatternLoop(player *p, mod_channel *ch)
{
unsigned char tempParam;
if (p->modTick == 0)
{
tempParam = ch->param & 0x0F;
if (tempParam == 0)
{
ch->patternLoopRow = p->modRow;
return;
}
if (ch->patternLoopCounter == 0)
{
ch->patternLoopCounter = tempParam;
}
else
{
ch->patternLoopCounter--;
if (ch->patternLoopCounter == 0)
return;
}
p->PBreakPosition = ch->patternLoopRow;
p->PBreakFlag = true;
}
}
static void efxTremoloControl(player *p, mod_channel *ch)
{
if (p->modTick == 0)
ch->tremoloControl = ch->param & 0x0F;
}
static void efxKarplusStrong(player *p, mod_channel *ch)
{
// WTF !
// KarplusStrong sucks, since some MODs
// used E8x for other effects instead.
(void)ch;
}
static void efxRetrigNote(player *p, mod_channel *ch)
{
unsigned char retrigTick;
retrigTick = ch->param & 0x0F;
if (retrigTick > 0)
{
if ((p->modTick % retrigTick) == 0)
p->tempFlags |= TEMPFLAG_START;
}
}
static void efxFineVolumeSlideUp(player *p, mod_channel *ch)
{
if (p->modTick == 0)
{
ch->volume += (ch->param & 0x0F);
if (ch->volume > 64)
ch->volume = 64;
p->tempVolume = ch->volume;
}
}
static void efxFineVolumeSlideDown(player *p, mod_channel *ch)
{
if (p->modTick == 0)
{
ch->volume -= (ch->param & 0x0F);
if (ch->volume < 0)
ch->volume = 0;
p->tempVolume = ch->volume;
}
}
static void efxNoteCut(player *p, mod_channel *ch)
{
if (p->modTick == (ch->param & 0x0F))
{
ch->volume = 0;
p->tempVolume = 0;
}
}
static void efxNoteDelay(player *p, mod_channel *ch)
{
unsigned char delayTick;
delayTick = ch->param & 0x0F;
if (p->modTick == 0)
ch->tempFlagsBackup = p->tempFlags;
if (p->modTick < delayTick)
p->tempFlags = TEMPFLAG_DELAY;
else if (p->modTick == delayTick)
p->tempFlags = ch->tempFlagsBackup;
}
static void efxPatternDelay(player *p, mod_channel *ch)
{
if (p->modTick == 0)
{
if (p->PattDelayTime2 == 0)
{
p->pattDelayFlag = true;
p->PattDelayTime = (ch->param & 0x0F) + 1;
}
}
}
static void efxInvertLoop(player *p, mod_channel *ch)
{
if (p->modTick == 0)
{
ch->invertLoopSpeed = ch->param & 0x0F;
if (ch->invertLoopSpeed > 0)
processInvertLoop(p, ch);
}
}
/* -- this is only for PT mode ---------------------------- */
static void setupGlissando(mod_channel *ch, unsigned short period)
{
unsigned char i;
const unsigned short *tablePointer;
tablePointer = &rawAmigaPeriods[ch->fineTune * ((12 * 3) + 1)];
i = 0;
while (true)
{
if (period >= tablePointer[i])
break;
if (++i >= ((12 * 3) + 1))
{
i = (12 * 3) - 1;
break;
}
}
if ((ch->fineTune & 8) && i) i--; /* yes, PT does this */
ch->wantedperiod = tablePointer[i];
ch->toneportdirec = 0;
if (ch->period == ch->wantedperiod)
ch->wantedperiod = 0; /* don't do any more slides */
else if (ch->period > ch->wantedperiod)
ch->toneportdirec = 1;
}
/* -------------------------------------------------------- */
static void processGlissando(player *p, mod_channel *ch)
{
unsigned char i;
signed char l;
signed char m;
signed char h;
const unsigned short *tablePointer;
if (p->tempPeriod == 0)
{
ch->wantedperiod = 0; /* don't do any more sliding */
return;
}
/* different routine for PT mode */
if (p->minPeriod == PT_MIN_PERIOD)
{
if (ch->wantedperiod > 0)
{
if (ch->toneportdirec == 0)
{
/* downwards */
ch->period += ch->glissandoSpeed;
if ((signed short)ch->period >= ch->wantedperiod)
{
ch->period = ch->wantedperiod;
ch->wantedperiod = 0;
}
}
else
{
/* upwards */
ch->period -= ch->glissandoSpeed;
if ((signed short)ch->period <= ch->wantedperiod)
{
ch->period = ch->wantedperiod;
ch->wantedperiod = 0;
}
}
if (ch->glissandoControl == 0)
{
p->tempPeriod = ch->period;
}
else
{
tablePointer = &rawAmigaPeriods[ch->fineTune * ((12 * 3) + 1)];
i = 0;
while (true)
{
if (ch->period >= tablePointer[i])
break;
if (++i >= 37)
{
i = 35;
break;
}
}
p->tempPeriod = tablePointer[i];
}
}
}
else
{
if (ch->period < ch->tempPeriod)
{
ch->period += ch->glissandoSpeed;
if ((signed short)ch->period > ch->tempPeriod)
ch->period = ch->tempPeriod;
}
else
{
ch->period -= ch->glissandoSpeed;
if ((signed short)ch->period < ch->tempPeriod)
ch->period = ch->tempPeriod;
}
if (ch->glissandoControl == 0)
{
p->tempPeriod = ch->period;
}
else
{
l = 0;
h = 83;
tablePointer = (unsigned short *)&extendedRawPeriods[ch->fineTune * ((12 * 7) + 1)];
while (h >= l)
{
m = (h + l) / 2;
if (tablePointer[m] == ch->period)
{
p->tempPeriod = tablePointer[m];
break;
}
else if (tablePointer[m] > ch->period)
{
l = m + 1;
}
else
{
h = m - 1;
}
}
}
}
}
static void processVibrato(player *p, mod_channel *ch)
{
unsigned char vibratoTemp;
int vibratoData;
int applyVibrato;
applyVibrato = 1;
if ((p->minPeriod == PT_MIN_PERIOD) && ((p->modTick == 0) && (p->PattDelayTime2 == 0))) /* PT/NT/UST/STK */
applyVibrato = 0;
if (applyVibrato)
{
if (p->tempPeriod > 0)
{
vibratoTemp = (ch->vibratoPos >> 2) & 0x1F;
switch (ch->vibratoControl & 3)
{
case 0:
vibratoData = p->sinusTable[vibratoTemp];
break;
case 1:
{
if (ch->vibratoPos < 128)
vibratoData = vibratoTemp << 3;
else
vibratoData = 255 - (vibratoTemp << 3);
}
break;
default:
vibratoData = 255;
break;
}
vibratoData = (vibratoData * ch->vibratoDepth) >> 7;
if (ch->vibratoPos < 128)
{
p->tempPeriod += (short)vibratoData;
if (p->tempPeriod > p->maxPeriod)
p->tempPeriod = p->maxPeriod;
}
else
{
p->tempPeriod -= (short)vibratoData;
if (p->tempPeriod < p->minPeriod)
p->tempPeriod = p->minPeriod;
}
}
}
if (p->modTick > 0)
ch->vibratoPos += (ch->vibratoSpeed << 2);
}
static void processTremolo(player *p, mod_channel *ch)
{
unsigned char tremoloTemp;
int tremoloData;
int applyTremolo;
applyTremolo = 1;
if ((p->minPeriod == PT_MIN_PERIOD) && ((p->modTick == 0) && (p->PattDelayTime2 == 0))) /* PT/NT/UST/STK */
applyTremolo = 0;
if (applyTremolo)
{
if (p->tempVolume > 0)
{
tremoloTemp = (ch->tremoloPos >> 2) & 0x1F;
switch (ch->tremoloControl & 3)
{
case 0:
tremoloData = p->sinusTable[tremoloTemp];
break;
case 1:
{
if (ch->vibratoPos < 128)
tremoloData = tremoloTemp << 3;
else
tremoloData = 255 - (tremoloTemp << 3);
}
break;
default:
tremoloData = 255;
break;
}
tremoloData = (tremoloData * ch->tremoloDepth) >> 6;
if (ch->tremoloPos < 128)
{
p->tempVolume += (signed char)tremoloData;
if (p->tempVolume > 64)
p->tempVolume = 64;
}
else
{
p->tempVolume -= (signed char)tremoloData;
if (p->tempVolume < 0)
p->tempVolume = 0;
}
}
}
if (p->modTick > 0)
ch->tremoloPos += (ch->tremoloSpeed << 2);
}
static void fxArpeggio(player *p, mod_channel *ch)
{
signed char l;
signed char m;
signed char h;
signed char noteToAdd;
signed char arpeggioTick;
unsigned char i;
unsigned short *tablePointer;
noteToAdd = 0;
arpeggioTick = p->modTick % 3;
if (arpeggioTick == 0)
{
p->tempPeriod = ch->period;
return;
}
else if (arpeggioTick == 1)
{
noteToAdd = ch->param >> 4;
}
else if (arpeggioTick == 2)
{
noteToAdd = ch->param & 0x0F;
}
if (p->minPeriod == PT_MIN_PERIOD) /* PT/NT/UST/STK */
{
tablePointer = (unsigned short *)&rawAmigaPeriods[ch->fineTune * ((12 * 3) + 1)];
for (i = 0; i < (12 * 3); ++i)
{
if (ch->period >= tablePointer[i])
{
p->tempPeriod = tablePointer[i + noteToAdd];
break;
}
}
}
else
{
l = 0;
h = (12 * 7) - 1;
tablePointer = (unsigned short *)&extendedRawPeriods[ch->fineTune * ((12 * 7) + 1)];
while (h >= l)
{
m = (h + l) / 2;
if (tablePointer[m] == ch->period)
{
p->tempPeriod = tablePointer[m + noteToAdd];
break;
}
else if (tablePointer[m] > ch->period)
{
l = m + 1;
}
else
{
h = m - 1;
}
}
}
}
static void fxPortamentoSlideUp(player *p, mod_channel *ch)
{
if ((p->modTick > 0) || (p->PattDelayTime2 > 0)) /* all ticks while EDx (weird) */
{
if (p->tempPeriod > 0)
{
ch->period -= ch->param;
if (p->minPeriod == PT_MIN_PERIOD)
{
if ((ch->period & 0x0FFF) < 113)
{
ch->period &= 0xF000;
ch->period |= 113;
}
p->tempPeriod = ch->period & 0x0FFF;
}
else
{
if (ch->period < p->minPeriod)
ch->period = p->minPeriod;
p->tempPeriod = ch->period;
}
}
}
}
static void fxPortamentoSlideDown(player *p, mod_channel *ch)
{
if ((p->modTick > 0) || (p->PattDelayTime2 > 0)) /* all ticks while EDx (weird) */
{
if (p->tempPeriod > 0)
{
ch->period += ch->param;
if (p->minPeriod == PT_MIN_PERIOD)
{
if ((ch->period & 0x0FFF) > 856)
{
ch->period &= 0xF000;
ch->period |= 856;
}
p->tempPeriod = ch->period & 0x0FFF;
}
else
{
if (ch->period > p->maxPeriod)
ch->period = p->maxPeriod;
p->tempPeriod = ch->period;
}
}
}
}
static void fxGlissando(player *p, mod_channel *ch)
{
if ((p->modTick > 0) || (p->PattDelayTime2 > 0)) /* all ticks while EDx (weird) */
{
if (ch->param != 0)
{
ch->glissandoSpeed = ch->param;
ch->param = 0;
}
processGlissando(p, ch);
}
}
static void fxVibrato(player *p, mod_channel *ch)
{
unsigned char hiNybble;
unsigned char loNybble;
if (p->modTick == 0)
{
hiNybble = ch->param >> 4;
loNybble = ch->param & 0x0F;
if (hiNybble != 0)
ch->vibratoSpeed = hiNybble;
if (loNybble != 0)
ch->vibratoDepth = loNybble;
}
processVibrato(p, ch);
}
static void fxGlissandoVolumeSlide(player *p, mod_channel *ch)
{
if ((p->modTick > 0) || (p->PattDelayTime2 > 0)) /* all ticks while EDx (weird) */
{
processGlissando(p, ch);
fxVolumeSlide(p, ch);
}
}
static void fxVibratoVolumeSlide(player *p, mod_channel *ch)
{
if ((p->modTick > 0) || (p->PattDelayTime2 > 0)) /* all ticks while EDx (weird) */
{
processVibrato(p, ch);
fxVolumeSlide(p, ch);
}
}
static void fxTremolo(player *p, mod_channel *ch)
{
unsigned char hiNybble;
unsigned char loNybble;
if (p->modTick == 0)
{
hiNybble = ch->param >> 4;
loNybble = ch->param & 0x0F;
if (hiNybble > 0)
ch->tremoloSpeed = hiNybble;
if (loNybble > 0)
ch->tremoloDepth = loNybble;
}
processTremolo(p, ch);
}
static void fxNotInUse(player *p, mod_channel *ch)
{
(void)p;
(void)ch;
}
static void fxSampleOffset(player *p, mod_channel *ch)
{
if (p->modTick == 0)
{
if (ch->param > 0)
ch->offsetTemp = (unsigned short)(ch->param) * 256;
ch->offset += ch->offsetTemp;
if (!ch->noNote)
ch->offsetBugNotAdded = false;
}
}
static void fxSampleOffset_FT2(player *p, mod_channel *ch)
{
if (p->modTick == 0)
{
if (ch->param > 0)
ch->offset = (unsigned short)(ch->param) * 256;
ch->offsetBugNotAdded = true;
}
}
static void fxVolumeSlide(player *p, mod_channel *ch)
{
unsigned char hiNybble;
unsigned char loNybble;
if ((p->modTick > 0) || (p->PattDelayTime2 > 0)) /* all ticks while EDx (weird) */
{
hiNybble = ch->param >> 4;
loNybble = ch->param & 0x0F;
if (hiNybble == 0)
{
ch->volume -= loNybble;
if (ch->volume < 0)
ch->volume = 0;
p->tempVolume = ch->volume;
}
else
{
ch->volume += hiNybble;
if (ch->volume > 64)
ch->volume = 64;
p->tempVolume = ch->volume;
}
}
}
static void fxPositionJump(player *p, mod_channel *ch)
{
if (p->modTick == 0)
{
p->modOrder = ch->param - 1;
p->PBreakPosition = 0;
p->PosJumpAssert = true;
p->pattBreakBugPos = 0;
p->pattBreakFlag = true;
}
}
static void fxSetVolume(player *p, mod_channel *ch)
{
if (p->modTick == 0)
{
if (ch->param > 64)
ch->param = 64;
ch->volume = ch->param;
p->tempVolume = ch->param;
}
}
static void fxPatternBreak(player *p, mod_channel *ch)
{
unsigned char pos;
if (p->modTick == 0)
{
pos = ((ch->param >> 4) * 10) + (ch->param & 0x0F);
if (pos > 63)
p->PBreakPosition = 0;
else
p->PBreakPosition = pos;
p->pattBreakBugPos = p->PBreakPosition;
p->pattBreakFlag = true;
p->PosJumpAssert = true;
}
}
static void fxExtended(player *p, mod_channel *ch)
{
efxRoutines[ch->param >> 4](p, ch);
}
static void fxExtended_FT2(player *p, mod_channel *ch)
{
efxRoutines_FT2[ch->param >> 4](p, ch);
}
static void modSetSpeed(player *p, unsigned char speed)
{
p->modSpeed = speed;
}
static void modSetTempo(player *p, unsigned short bpm)
{
p->modBPM = bpm;
p->samplesPerTick = p->tempoTimerVal / bpm;
}
static void fxSetTempo(player *p, mod_channel *ch)
{
if (p->modTick == 0)
{
if ((ch->param < 32) || p->vBlankTiming)
modSetSpeed(p, ch->param);
else
modSetTempo(p, ch->param);
}
}
static void processEffects(player *p, mod_channel *ch)
{
if (p->modTick > 0)
processInvertLoop(p, ch);
if ((!ch->command && !ch->param) == 0)
{
switch (p->source->head.format)
{
case FORMAT_NCHN:
case FORMAT_NNCH:
case FORMAT_16CN:
case FORMAT_32CN:
fxRoutines_FT2[ch->command](p, ch);
break;
default:
fxRoutines[ch->command](p, ch);
break;
}
}
}
static void fxPan(player *p, mod_channel *ch)
{
if (p->modTick == 0)
{
if (p->sevenBitPanning)
{
if (ch->param != 0xA4) /* don't do 0xA4 surround yet */
mixerSetChPan(p, ch->chanIndex, ch->param * 2); /* 7-bit .MOD pan */
}
else
{
mixerSetChPan(p, ch->chanIndex, ch->param); /* 8-bit .MOD pan */
}
}
}
static void efxPan(player *p, mod_channel *ch)
{
if (p->modTick == 0)
mixerSetChPan(p, ch->chanIndex, (ch->param & 0x0F) * 17);
}
void playptmod_Stop(void *_p)
{
player *p = (player *)_p;
int i;
mixerCutChannels(p);
p->modulePlaying = false;
memset(p->source->channels, 0, sizeof (mod_channel) * MAX_CHANNELS);
for (i = 0; i < MAX_CHANNELS; ++i)
{
p->source->channels[i].chanIndex = (char)i;
p->source->channels[i].noNote = true;
p->source->channels[i].offsetBugNotAdded = true;
}
p->tempFlags = 0;
p->pattBreakBugPos = -1;
p->pattBreakFlag = false;
p->pattDelayFlag = false;
p->forceEffectsOff = false;
p->PattDelayTime = 0;
p->PattDelayTime2 = 0;
p->PBreakPosition = 0;
p->PosJumpAssert = false;
}
static void fetchPatternData(player *p, mod_channel *ch)
{
int tempNote;
modnote_t *note;
note = &p->source->patterns[p->modPattern][(p->modRow * p->source->head.channelCount) + ch->chanIndex];
if ((note->sample > 0) && (note->sample <= 32))
{
if (ch->sample != note->sample)
ch->flags |= FLAG_NEWSAMPLE;
ch->sample = note->sample;
ch->flags |= FLAG_SAMPLE;
ch->fineTune = p->source->samples[ch->sample - 1].fineTune;
}
ch->command = note->command;
ch->param = note->param;
if ((note->period & 0x0FFF) > 0)
{
if (ch->command == 0x0E)
{
if ((ch->param >> 4) == 0x05)
ch->fineTune = ch->param & 0x0F;
}
tempNote = periodToNote(p, 0, note->period & 0x0FFF);
if ((p->minPeriod == PT_MIN_PERIOD) && (tempNote == (12 * 3))) /* PT/NT/STK/UST only */
{
ch->noNote = true;
mixerSetChSource(p, ch->chanIndex, NULL, 0, 0, 0, 0, 0);
}
else
{
ch->noNote = false;
ch->tempPeriod = (p->minPeriod == PT_MIN_PERIOD) ? rawAmigaPeriods[(ch->fineTune * ((12 * 3) + 1)) + tempNote] : extendedRawPeriods[(ch->fineTune * ((12 * 7) + 1)) + tempNote];
ch->flags |= FLAG_NOTE;
}
/* do a slightly different path for 3xx/5xy in PT mode */
if (p->minPeriod == PT_MIN_PERIOD)
{
if ((ch->command == 0x03) || (ch->command == 0x05))
setupGlissando(ch, note->period);
}
}
else
{
ch->noNote = true;
}
}
static void processChannel(player *p, mod_channel *ch)
{
MODULE_SAMPLE *s;
p->tempFlags = 0;
if (p->modTick == 0)
{
if (p->PattDelayTime2 == 0)
fetchPatternData(p, ch);
if (ch->flags & FLAG_NOTE)
{
ch->flags &= ~FLAG_NOTE;
if ((ch->command != 0x03) && (ch->command != 0x05))
{
ch->period = ch->tempPeriod;
if (ch->sample > 0)
p->tempFlags |= TEMPFLAG_START;
}
ch->tempFlagsBackup = 0;
if ((ch->vibratoControl & 4) == 0)
ch->vibratoPos = 0;
if ((ch->tremoloControl & 4) == 0)
ch->tremoloPos = 0;
}
if (ch->flags & FLAG_SAMPLE)
{
ch->flags &= ~FLAG_SAMPLE;
if (ch->sample > 0)
{
s = &p->source->samples[ch->sample - 1];
ch->volume = s->volume;
ch->invertLoopPtr = &p->source->sampleData[s->offset + s->loopStart];
ch->invertLoopStart = ch->invertLoopPtr;
ch->invertLoopLength = s->loopLength;
if ((ch->command != 0x03) && (ch->command != 0x05))
{
ch->offset = 0;
ch->offsetBugNotAdded = true;
}
if (ch->flags & FLAG_NEWSAMPLE)
{
ch->flags &= ~FLAG_NEWSAMPLE;
/* sample swapping (PT only) */
if ((ch->sample > 0) && (p->minPeriod == PT_MIN_PERIOD))
{
s = &p->source->samples[ch->sample - 1];
mixerSwapChSource(p, ch->chanIndex, p->source->sampleData + s->offset, s->length, s->loopStart, s->loopLength, (s->attribute & 1) ? 2 : 1);
}
}
}
}
}
p->tempPeriod = ch->period;
p->tempVolume = ch->volume;
if (!p->forceEffectsOff) /* EEx + Dxx/Bxx quirk simulation (FT2 had this bug too) */
processEffects(p, ch);
/* EDx quirk for PT: Change to new sample vol immediately */
if (p->minPeriod == PT_MIN_PERIOD)
mixerSetChVol(p, ch->chanIndex, p->tempVolume);
if (!(p->tempFlags & TEMPFLAG_DELAY))
{
if (p->tempFlags & TEMPFLAG_START)
{
if (ch->sample > 0)
{
s = &p->source->samples[ch->sample - 1];
if (s->length > 0)
{
if (ch->offset > 0)
{
mixerSetChSource(p, ch->chanIndex, p->source->sampleData + s->offset, s->length, s->loopStart, s->loopLength, ch->offset, (s->attribute & 1) ? 2 : 1);
if (p->minPeriod == PT_MIN_PERIOD) /* PT/NT/STK/UST bug only */
{
if (!ch->offsetBugNotAdded)
{
ch->offset += ch->offsetTemp; /* emulates add sample offset bug (fixes some quirky MODs) */
if (ch->offset > 0xFFFF)
ch->offset = 0xFFFF;
ch->offsetBugNotAdded = true;
}
}
}
else
{
mixerSetChSource(p, ch->chanIndex, p->source->sampleData + s->offset, s->length, s->loopStart, s->loopLength, 0, (s->attribute & 1) ? 2 : 1);
}
}
else
{
mixerSetChSource(p, ch->chanIndex, NULL, 0, 0, 0, 0, 0);
}
}
}
/* EDx quirk (set volume here instead for non-PT MODs) */
if (p->minPeriod != PT_MIN_PERIOD)
mixerSetChVol(p, ch->chanIndex, p->tempVolume);
mixerSetChRate(p, ch->chanIndex, p->tempPeriod);
}
}
static void nextPosition(player *p)
{
p->modRow = p->PBreakPosition;
p->PBreakPosition = 0;
p->PosJumpAssert = false;
p->modOrder++;
if (p->modOrder >= p->source->head.orderCount)
p->modOrder = p->source->head.clippedRestartPos;
p->modPattern = p->source->head.order[p->modOrder];
if (p->modRow == 0)
p->loopCounter = p->orderPlayed[p->modOrder]++;
}
static void processTick(player *p)
{
int i;
/* EEx + Dxx/Bxx quirk simulation (FT2 had this bug too) */
if (p->modTick == 0)
{
if (p->forceEffectsOff)
{
if (p->modRow != p->pattBreakBugPos)
{
p->forceEffectsOff = false;
p->pattBreakBugPos = -1;
}
}
}
for (i = 0; i < p->source->head.channelCount; ++i)
processChannel(p, p->source->channels + i);
/* EEx + Dxx/Bxx quirk simulation (FT2 had this bug too) */
if (p->modTick == 0)
{
if (p->pattBreakFlag && p->pattDelayFlag)
p->forceEffectsOff = true;
}
/* Only process speed 0 effect after processing entire row */
if (p->modSpeed == 0)
{
/* Bit of a hack, will alert code below of a full repeat */
p->modOrder = p->source->head.clippedRestartPos;
modSetSpeed(p, 6);
p->modTick = 6; /* cause instant repeat */
p->PBreakPosition = 0;
p->PosJumpAssert = true;
}
p->modTick++;
if (p->modTick >= p->modSpeed)
{
p->modTick = 0;
p->pattBreakFlag = false;
p->pattDelayFlag = false;
p->modRow++;
if (p->PattDelayTime > 0)
{
p->PattDelayTime2 = p->PattDelayTime;
p->PattDelayTime = 0;
}
if (p->PattDelayTime2 > 0)
{
p->PattDelayTime2--;
if (p->PattDelayTime2 > 0)
p->modRow--;
}
if (p->PBreakFlag)
{
p->PBreakFlag = false;
p->modRow = p->PBreakPosition;
p->PBreakPosition = 0;
}
if ((p->modRow == p->source->head.rowCount) || p->PosJumpAssert)
nextPosition(p);
}
}
void playptmod_Render(void *_p, int *target, int length)
{
player *p = (player *)_p;
if (p->modulePlaying)
{
static const int soundBufferSamples = soundBufferSize / 4;
while (length)
{
int tempSamples = CLAMP(length, 0, soundBufferSamples);
if (p->sampleCounter)
{
tempSamples = CLAMP(tempSamples, 0, p->sampleCounter);
if (target)
{
outputAudio(p, target, tempSamples);
target += tempSamples * 2;
}
p->sampleCounter -= tempSamples;
length -= tempSamples;
}
else
{
processTick(p);
p->sampleCounter = p->samplesPerTick;
}
}
}
}
void playptmod_Render16(void *_p, short *target, int length)
{
player *p = (player *)_p;
int tempBuffer[512];
int *temp;
unsigned int i;
temp = target ? tempBuffer : NULL;
while (length)
{
int tempSamples = CLAMP(length, 0, 256);
playptmod_Render(p, temp, tempSamples);
length -= tempSamples;
if (target)
{
for (i = 0; i < tempSamples * 2; ++i)
target[i] = (short)CLAMP(tempBuffer[i] / 256, -32768, 32767);
target += (tempSamples * 2);
}
}
}
void *playptmod_Create(int samplingFrequency)
{
int i, j;
player *p = (player *)calloc(1, sizeof (player));
resampler_init();
p->tempoTimerVal = (samplingFrequency * 125) / 50;
p->sinusTable = (unsigned char *)malloc(32);
for (i = 0; i < 32; ++i)
p->sinusTable[i] = (unsigned char)floorf(255.0f * sinf(((float)i * 3.141592f) / 32.0f));
/* generate extended period table */
for (j = 0; j < 16; ++j)
for (i = 0; i < ((12 * 7) + 1); ++i)
extendedRawPeriods[(j * ((12 * 7) + 1)) + i] = (i == (12 * 7)) ? 0 : (npertab[i] * 8363 / finetune[j]);
/* add padding (zeroes) */
for (i = 0; i < 14; ++i)
extendedRawPeriods[(16 * ((12 * 7) + 1)) + i] = 0;
p->soundFrequency = samplingFrequency;
p->mixBufferL = (float *)malloc(soundBufferSize * sizeof (float));
p->mixBufferR = (float *)malloc(soundBufferSize * sizeof (float));
p->filterC.LED = calcRcCoeff((float)samplingFrequency, 3090.0f);
p->filterC.LEDFb = 0.125f + 0.125f / (1.0f - p->filterC.LED);
p->useLEDFilter = false;
for (i = 0; i < MAX_CHANNELS; ++i)
{
p->blep[i] = resampler_create();
p->blepVol[i] = resampler_create();
resampler_set_quality(p->blep[i], RESAMPLER_QUALITY_BLEP);
resampler_set_quality(p->blepVol[i], RESAMPLER_QUALITY_BLEP);
}
mixerCutChannels(p);
return (p);
}
void playptmod_Config(void *_p, int option, int value)
{
player *p = (player *)_p;
switch (option)
{
case PTMOD_OPTION_VSYNC_TIMING:
p->vBlankTiming = value ? true : false;
break;
}
}
void playptmod_Play(void *_p, unsigned int startOrder)
{
player *p = (player *)_p;
int i;
if (!p->modulePlaying && p->moduleLoaded)
{
mixerCutChannels(p);
memset(p->source->channels, 0, sizeof (mod_channel) * MAX_CHANNELS);
for (i = 0; i < MAX_CHANNELS; ++i)
{
p->source->channels[i].chanIndex = (char)i;
p->source->channels[i].noNote = true;
p->source->channels[i].offsetBugNotAdded = true;
}
p->sampleCounter = 0;
modSetTempo(p, p->source->head.initBPM);
modSetSpeed(p, 6);
p->modOrder = startOrder;
p->modPattern = p->source->head.order[startOrder];
p->modRow = 0;
p->modTick = 0;
p->tempFlags = 0;
p->tempPeriod = 0;
p->modTick = 0;
p->PBreakPosition = 0;
p->PosJumpAssert = false;
p->pattBreakBugPos = -1;
p->pattBreakFlag = false;
p->pattDelayFlag = false;
p->forceEffectsOff = false;
p->PattDelayTime = 0;
p->PattDelayTime2 = 0;
p->PBreakFlag = false;
memcpy(p->source->sampleData, p->source->originalSampleData, p->source->head.totalSampleSize);
memset(p->orderPlayed, 0, sizeof (p->orderPlayed));
p->loopCounter = 0;
p->orderPlayed[startOrder] = 1;
p->modulePlaying = true;
}
}
void playptmod_Free(void *_p)
{
player *p = (player *)_p;
int i;
if (p->moduleLoaded)
{
p->modulePlaying = false;
if (p->source != NULL)
{
for (i = 0; i < 256; ++i)
{
if (p->source->patterns[i] != NULL)
{
free(p->source->patterns[i]);
p->source->patterns[i] = NULL;
}
}
if (p->source->originalSampleData != NULL)
{
free(p->source->originalSampleData);
p->source->originalSampleData = NULL;
}
if (p->source->sampleData != NULL)
{
free(p->source->sampleData);
p->source->sampleData = NULL;
}
free(p->source);
p->source = NULL;
}
p->moduleLoaded = false;
}
if (p->mixBufferL != NULL)
{
free(p->mixBufferL);
p->mixBufferL = NULL;
}
if (p->mixBufferR != NULL)
{
free(p->mixBufferR);
p->mixBufferR = NULL;
}
if (p->sinusTable != NULL)
{
free(p->sinusTable);
p->sinusTable = NULL;
}
for (i = 0; i < MAX_CHANNELS; ++i)
{
resampler_delete(p->blep[i]);
resampler_delete(p->blepVol[i]);
}
free(p);
}
unsigned int playptmod_LoopCounter(void *_p)
{
player *p = (player *)_p;
return p->loopCounter;
}
void playptmod_GetInfo(void *_p, playptmod_info *i)
{
int n, c;
player *p = (player *)_p;
int order = p->modOrder;
int row = p->modRow;
int pattern = p->modPattern;
if ((p->modRow >= p->source->head.rowCount) || p->PosJumpAssert)
{
order++;
if (order >= p->source->head.orderCount)
order = p->source->head.clippedRestartPos;
row = p->PBreakPosition;
pattern = p->source->head.order[order];
}
i->order = order;
i->pattern = pattern;
i->row = row;
i->speed = p->modSpeed;
i->tempo = p->modBPM;
for (c = 0, n = 0; n < p->source->head.channelCount; ++n)
{
if (p->v[n].data)
c++;
}
i->channelsPlaying = c;
}
void playptmod_Mute(void *_p, int channel, int mute)
{
player *p = (player *)_p;
p->v[channel].mute = mute;
}
/* END OF FILE */